Diagnostics.cpp revision 259701
1//===--- Diagnostics.cpp - Helper class for error diagnostics -----*- C++ -*-===//
2//
3//                     The LLVM Compiler Infrastructure
4//
5// This file is distributed under the University of Illinois Open Source
6// License. See LICENSE.TXT for details.
7//
8//===----------------------------------------------------------------------===//
9
10#include "clang/ASTMatchers/Dynamic/Diagnostics.h"
11
12namespace clang {
13namespace ast_matchers {
14namespace dynamic {
15
16Diagnostics::ArgStream Diagnostics::pushContextFrame(ContextType Type,
17                                                     SourceRange Range) {
18  ContextStack.push_back(ContextFrame());
19  ContextFrame& data = ContextStack.back();
20  data.Type = Type;
21  data.Range = Range;
22  return ArgStream(&data.Args);
23}
24
25Diagnostics::Context::Context(ConstructMatcherEnum, Diagnostics *Error,
26                              StringRef MatcherName,
27                              const SourceRange &MatcherRange)
28    : Error(Error) {
29  Error->pushContextFrame(CT_MatcherConstruct, MatcherRange) << MatcherName;
30}
31
32Diagnostics::Context::Context(MatcherArgEnum, Diagnostics *Error,
33                              StringRef MatcherName,
34                              const SourceRange &MatcherRange,
35                              unsigned ArgNumber)
36    : Error(Error) {
37  Error->pushContextFrame(CT_MatcherArg, MatcherRange) << ArgNumber
38                                                       << MatcherName;
39}
40
41Diagnostics::Context::~Context() { Error->ContextStack.pop_back(); }
42
43Diagnostics::OverloadContext::OverloadContext(Diagnostics *Error)
44    : Error(Error), BeginIndex(Error->Errors.size()) {}
45
46Diagnostics::OverloadContext::~OverloadContext() {
47  // Merge all errors that happened while in this context.
48  if (BeginIndex < Error->Errors.size()) {
49    Diagnostics::ErrorContent &Dest = Error->Errors[BeginIndex];
50    for (size_t i = BeginIndex + 1, e = Error->Errors.size(); i < e; ++i) {
51      Dest.Messages.push_back(Error->Errors[i].Messages[0]);
52    }
53    Error->Errors.resize(BeginIndex + 1);
54  }
55}
56
57void Diagnostics::OverloadContext::revertErrors() {
58  // Revert the errors.
59  Error->Errors.resize(BeginIndex);
60}
61
62Diagnostics::ArgStream &Diagnostics::ArgStream::operator<<(const Twine &Arg) {
63  Out->push_back(Arg.str());
64  return *this;
65}
66
67Diagnostics::ArgStream Diagnostics::addError(const SourceRange &Range,
68                                             ErrorType Error) {
69  Errors.push_back(ErrorContent());
70  ErrorContent &Last = Errors.back();
71  Last.ContextStack = ContextStack;
72  Last.Messages.push_back(ErrorContent::Message());
73  Last.Messages.back().Range = Range;
74  Last.Messages.back().Type = Error;
75  return ArgStream(&Last.Messages.back().Args);
76}
77
78StringRef contextTypeToFormatString(Diagnostics::ContextType Type) {
79  switch (Type) {
80    case Diagnostics::CT_MatcherConstruct:
81      return "Error building matcher $0.";
82    case Diagnostics::CT_MatcherArg:
83      return "Error parsing argument $0 for matcher $1.";
84  }
85  llvm_unreachable("Unknown ContextType value.");
86}
87
88StringRef errorTypeToFormatString(Diagnostics::ErrorType Type) {
89  switch (Type) {
90  case Diagnostics::ET_RegistryNotFound:
91    return "Matcher not found: $0";
92  case Diagnostics::ET_RegistryWrongArgCount:
93    return "Incorrect argument count. (Expected = $0) != (Actual = $1)";
94  case Diagnostics::ET_RegistryWrongArgType:
95    return "Incorrect type for arg $0. (Expected = $1) != (Actual = $2)";
96  case Diagnostics::ET_RegistryNotBindable:
97    return "Matcher does not support binding.";
98  case Diagnostics::ET_RegistryAmbiguousOverload:
99    // TODO: Add type info about the overload error.
100    return "Ambiguous matcher overload.";
101
102  case Diagnostics::ET_ParserStringError:
103    return "Error parsing string token: <$0>";
104  case Diagnostics::ET_ParserNoOpenParen:
105    return "Error parsing matcher. Found token <$0> while looking for '('.";
106  case Diagnostics::ET_ParserNoCloseParen:
107    return "Error parsing matcher. Found end-of-code while looking for ')'.";
108  case Diagnostics::ET_ParserNoComma:
109    return "Error parsing matcher. Found token <$0> while looking for ','.";
110  case Diagnostics::ET_ParserNoCode:
111    return "End of code found while looking for token.";
112  case Diagnostics::ET_ParserNotAMatcher:
113    return "Input value is not a matcher expression.";
114  case Diagnostics::ET_ParserInvalidToken:
115    return "Invalid token <$0> found when looking for a value.";
116  case Diagnostics::ET_ParserMalformedBindExpr:
117    return "Malformed bind() expression.";
118  case Diagnostics::ET_ParserTrailingCode:
119    return "Expected end of code.";
120  case Diagnostics::ET_ParserUnsignedError:
121    return "Error parsing unsigned token: <$0>";
122  case Diagnostics::ET_ParserOverloadedType:
123    return "Input value has unresolved overloaded type: $0";
124
125  case Diagnostics::ET_None:
126    return "<N/A>";
127  }
128  llvm_unreachable("Unknown ErrorType value.");
129}
130
131void formatErrorString(StringRef FormatString, ArrayRef<std::string> Args,
132                       llvm::raw_ostream &OS) {
133  while (!FormatString.empty()) {
134    std::pair<StringRef, StringRef> Pieces = FormatString.split("$");
135    OS << Pieces.first.str();
136    if (Pieces.second.empty()) break;
137
138    const char Next = Pieces.second.front();
139    FormatString = Pieces.second.drop_front();
140    if (Next >= '0' && Next <= '9') {
141      const unsigned Index = Next - '0';
142      if (Index < Args.size()) {
143        OS << Args[Index];
144      } else {
145        OS << "<Argument_Not_Provided>";
146      }
147    }
148  }
149}
150
151static void maybeAddLineAndColumn(const SourceRange &Range,
152                                  llvm::raw_ostream &OS) {
153  if (Range.Start.Line > 0 && Range.Start.Column > 0) {
154    OS << Range.Start.Line << ":" << Range.Start.Column << ": ";
155  }
156}
157
158static void printContextFrameToStream(const Diagnostics::ContextFrame &Frame,
159                                      llvm::raw_ostream &OS) {
160  maybeAddLineAndColumn(Frame.Range, OS);
161  formatErrorString(contextTypeToFormatString(Frame.Type), Frame.Args, OS);
162}
163
164static void
165printMessageToStream(const Diagnostics::ErrorContent::Message &Message,
166                     const Twine Prefix, llvm::raw_ostream &OS) {
167  maybeAddLineAndColumn(Message.Range, OS);
168  OS << Prefix;
169  formatErrorString(errorTypeToFormatString(Message.Type), Message.Args, OS);
170}
171
172static void printErrorContentToStream(const Diagnostics::ErrorContent &Content,
173                                      llvm::raw_ostream &OS) {
174  if (Content.Messages.size() == 1) {
175    printMessageToStream(Content.Messages[0], "", OS);
176  } else {
177    for (size_t i = 0, e = Content.Messages.size(); i != e; ++i) {
178      if (i != 0) OS << "\n";
179      printMessageToStream(Content.Messages[i],
180                           "Candidate " + Twine(i + 1) + ": ", OS);
181    }
182  }
183}
184
185void Diagnostics::printToStream(llvm::raw_ostream &OS) const {
186  for (size_t i = 0, e = Errors.size(); i != e; ++i) {
187    if (i != 0) OS << "\n";
188    printErrorContentToStream(Errors[i], OS);
189  }
190}
191
192std::string Diagnostics::toString() const {
193  std::string S;
194  llvm::raw_string_ostream OS(S);
195  printToStream(OS);
196  return OS.str();
197}
198
199void Diagnostics::printToStreamFull(llvm::raw_ostream &OS) const {
200  for (size_t i = 0, e = Errors.size(); i != e; ++i) {
201    if (i != 0) OS << "\n";
202    const ErrorContent &Error = Errors[i];
203    for (size_t i = 0, e = Error.ContextStack.size(); i != e; ++i) {
204      printContextFrameToStream(Error.ContextStack[i], OS);
205      OS << "\n";
206    }
207    printErrorContentToStream(Error, OS);
208  }
209}
210
211std::string Diagnostics::toStringFull() const {
212  std::string S;
213  llvm::raw_string_ostream OS(S);
214  printToStreamFull(OS);
215  return OS.str();
216}
217
218}  // namespace dynamic
219}  // namespace ast_matchers
220}  // namespace clang
221