1 //===--- Diagnostics.h - 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 /// \file 11 /// \brief Diagnostics class to manage error messages. 12 /// 13 //===----------------------------------------------------------------------===// 14 15 #ifndef LLVM_CLANG_AST_MATCHERS_DYNAMIC_DIAGNOSTICS_H 16 #define LLVM_CLANG_AST_MATCHERS_DYNAMIC_DIAGNOSTICS_H 17 18 #include <string> 19 #include <vector> 20 21 #include "clang/ASTMatchers/Dynamic/VariantValue.h" 22 #include "clang/Basic/LLVM.h" 23 #include "llvm/ADT/ArrayRef.h" 24 #include "llvm/ADT/StringRef.h" 25 #include "llvm/ADT/Twine.h" 26 #include "llvm/Support/raw_ostream.h" 27 28 namespace clang { 29 namespace ast_matchers { 30 namespace dynamic { 31 32 struct SourceLocation { 33 SourceLocation() : Line(), Column() {} 34 unsigned Line; 35 unsigned Column; 36 }; 37 38 struct SourceRange { 39 SourceLocation Start; 40 SourceLocation End; 41 }; 42 43 /// \brief A VariantValue instance annotated with its parser context. 44 struct ParserValue { 45 ParserValue() : Text(), Range(), Value() {} 46 StringRef Text; 47 SourceRange Range; 48 VariantValue Value; 49 }; 50 51 /// \brief Helper class to manage error messages. 52 class Diagnostics { 53 public: 54 /// \brief Parser context types. 55 enum ContextType { 56 CT_MatcherArg = 0, 57 CT_MatcherConstruct = 1 58 }; 59 60 /// \brief All errors from the system. 61 enum ErrorType { 62 ET_None = 0, 63 64 ET_RegistryNotFound = 1, 65 ET_RegistryWrongArgCount = 2, 66 ET_RegistryWrongArgType = 3, 67 ET_RegistryNotBindable = 4, 68 ET_RegistryAmbiguousOverload = 5, 69 70 ET_ParserStringError = 100, 71 ET_ParserNoOpenParen = 101, 72 ET_ParserNoCloseParen = 102, 73 ET_ParserNoComma = 103, 74 ET_ParserNoCode = 104, 75 ET_ParserNotAMatcher = 105, 76 ET_ParserInvalidToken = 106, 77 ET_ParserMalformedBindExpr = 107, 78 ET_ParserTrailingCode = 108, 79 ET_ParserUnsignedError = 109, 80 ET_ParserOverloadedType = 110 81 }; 82 83 /// \brief Helper stream class. 84 class ArgStream { 85 public: 86 ArgStream(std::vector<std::string> *Out) : Out(Out) {} 87 template <class T> ArgStream &operator<<(const T &Arg) { 88 return operator<<(Twine(Arg)); 89 } 90 ArgStream &operator<<(const Twine &Arg); 91 92 private: 93 std::vector<std::string> *Out; 94 }; 95 96 /// \brief Class defining a parser context. 97 /// 98 /// Used by the parser to specify (possibly recursive) contexts where the 99 /// parsing/construction can fail. Any error triggered within a context will 100 /// keep information about the context chain. 101 /// This class should be used as a RAII instance in the stack. 102 struct Context { 103 public: 104 /// \brief About to call the constructor for a matcher. 105 enum ConstructMatcherEnum { ConstructMatcher }; 106 Context(ConstructMatcherEnum, Diagnostics *Error, StringRef MatcherName, 107 const SourceRange &MatcherRange); 108 /// \brief About to recurse into parsing one argument for a matcher. 109 enum MatcherArgEnum { MatcherArg }; 110 Context(MatcherArgEnum, Diagnostics *Error, StringRef MatcherName, 111 const SourceRange &MatcherRange, unsigned ArgNumber); 112 ~Context(); 113 114 private: 115 Diagnostics *const Error; 116 }; 117 118 /// \brief Context for overloaded matcher construction. 119 /// 120 /// This context will take care of merging all errors that happen within it 121 /// as "candidate" overloads for the same matcher. 122 struct OverloadContext { 123 public: 124 OverloadContext(Diagnostics* Error); 125 ~OverloadContext(); 126 127 /// \brief Revert all errors that happened within this context. 128 void revertErrors(); 129 130 private: 131 Diagnostics *const Error; 132 unsigned BeginIndex; 133 }; 134 135 /// \brief Add an error to the diagnostics. 136 /// 137 /// All the context information will be kept on the error message. 138 /// \return a helper class to allow the caller to pass the arguments for the 139 /// error message, using the << operator. 140 ArgStream addError(const SourceRange &Range, ErrorType Error); 141 142 /// \brief Information stored for one frame of the context. 143 struct ContextFrame { 144 ContextType Type; 145 SourceRange Range; 146 std::vector<std::string> Args; 147 }; 148 149 /// \brief Information stored for each error found. 150 struct ErrorContent { 151 std::vector<ContextFrame> ContextStack; 152 struct Message { 153 SourceRange Range; 154 ErrorType Type; 155 std::vector<std::string> Args; 156 }; 157 std::vector<Message> Messages; 158 }; 159 ArrayRef<ErrorContent> errors() const { return Errors; } 160 161 /// \brief Returns a simple string representation of each error. 162 /// 163 /// Each error only shows the error message without any context. 164 void printToStream(llvm::raw_ostream &OS) const; 165 std::string toString() const; 166 167 /// \brief Returns the full string representation of each error. 168 /// 169 /// Each error message contains the full context. 170 void printToStreamFull(llvm::raw_ostream &OS) const; 171 std::string toStringFull() const; 172 173 private: 174 /// \brief Helper function used by the constructors of ContextFrame. 175 ArgStream pushContextFrame(ContextType Type, SourceRange Range); 176 177 std::vector<ContextFrame> ContextStack; 178 std::vector<ErrorContent> Errors; 179 }; 180 181 } // namespace dynamic 182 } // namespace ast_matchers 183 } // namespace clang 184 185 #endif // LLVM_CLANG_AST_MATCHERS_DYNAMIC_DIAGNOSTICS_H 186