1 //===- SourceMgr.h - Manager for Source Buffers & 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 // This file declares the SMDiagnostic and SourceMgr classes. This 11 // provides a simple substrate for diagnostics, #include handling, and other low 12 // level things for simple parsers. 13 // 14 //===----------------------------------------------------------------------===// 15 16 #ifndef LLVM_SUPPORT_SOURCEMGR_H 17 #define LLVM_SUPPORT_SOURCEMGR_H 18 19 #include "llvm/ADT/ArrayRef.h" 20 #include "llvm/ADT/StringRef.h" 21 #include "llvm/ADT/Twine.h" 22 #include "llvm/Support/SMLoc.h" 23 #include <string> 24 25 namespace llvm { 26 class MemoryBuffer; 27 class SourceMgr; 28 class SMDiagnostic; 29 class SMFixIt; 30 class Twine; 31 class raw_ostream; 32 33 /// SourceMgr - This owns the files read by a parser, handles include stacks, 34 /// and handles diagnostic wrangling. 35 class SourceMgr { 36 public: 37 enum DiagKind { 38 DK_Error, 39 DK_Warning, 40 DK_Note 41 }; 42 43 /// DiagHandlerTy - Clients that want to handle their own diagnostics in a 44 /// custom way can register a function pointer+context as a diagnostic 45 /// handler. It gets called each time PrintMessage is invoked. 46 typedef void (*DiagHandlerTy)(const SMDiagnostic &, void *Context); 47 private: 48 struct SrcBuffer { 49 /// Buffer - The memory buffer for the file. 50 MemoryBuffer *Buffer; 51 52 /// IncludeLoc - This is the location of the parent include, or null if at 53 /// the top level. 54 SMLoc IncludeLoc; 55 }; 56 57 /// Buffers - This is all of the buffers that we are reading from. 58 std::vector<SrcBuffer> Buffers; 59 60 // IncludeDirectories - This is the list of directories we should search for 61 // include files in. 62 std::vector<std::string> IncludeDirectories; 63 64 /// LineNoCache - This is a cache for line number queries, its implementation 65 /// is really private to SourceMgr.cpp. 66 mutable void *LineNoCache; 67 68 DiagHandlerTy DiagHandler; 69 void *DiagContext; 70 71 SourceMgr(const SourceMgr&) LLVM_DELETED_FUNCTION; 72 void operator=(const SourceMgr&) LLVM_DELETED_FUNCTION; 73 public: SourceMgr()74 SourceMgr() : LineNoCache(0), DiagHandler(0), DiagContext(0) {} 75 ~SourceMgr(); 76 setIncludeDirs(const std::vector<std::string> & Dirs)77 void setIncludeDirs(const std::vector<std::string> &Dirs) { 78 IncludeDirectories = Dirs; 79 } 80 81 /// setDiagHandler - Specify a diagnostic handler to be invoked every time 82 /// PrintMessage is called. Ctx is passed into the handler when it is invoked. 83 void setDiagHandler(DiagHandlerTy DH, void *Ctx = 0) { 84 DiagHandler = DH; 85 DiagContext = Ctx; 86 } 87 getDiagHandler()88 DiagHandlerTy getDiagHandler() const { return DiagHandler; } getDiagContext()89 void *getDiagContext() const { return DiagContext; } 90 getBufferInfo(unsigned i)91 const SrcBuffer &getBufferInfo(unsigned i) const { 92 assert(i < Buffers.size() && "Invalid Buffer ID!"); 93 return Buffers[i]; 94 } 95 getMemoryBuffer(unsigned i)96 const MemoryBuffer *getMemoryBuffer(unsigned i) const { 97 assert(i < Buffers.size() && "Invalid Buffer ID!"); 98 return Buffers[i].Buffer; 99 } 100 getNumBuffers()101 unsigned getNumBuffers() const { 102 return Buffers.size(); 103 } 104 getParentIncludeLoc(unsigned i)105 SMLoc getParentIncludeLoc(unsigned i) const { 106 assert(i < Buffers.size() && "Invalid Buffer ID!"); 107 return Buffers[i].IncludeLoc; 108 } 109 110 /// AddNewSourceBuffer - Add a new source buffer to this source manager. This 111 /// takes ownership of the memory buffer. AddNewSourceBuffer(MemoryBuffer * F,SMLoc IncludeLoc)112 unsigned AddNewSourceBuffer(MemoryBuffer *F, SMLoc IncludeLoc) { 113 SrcBuffer NB; 114 NB.Buffer = F; 115 NB.IncludeLoc = IncludeLoc; 116 Buffers.push_back(NB); 117 return Buffers.size()-1; 118 } 119 120 /// AddIncludeFile - Search for a file with the specified name in the current 121 /// directory or in one of the IncludeDirs. If no file is found, this returns 122 /// ~0, otherwise it returns the buffer ID of the stacked file. 123 /// The full path to the included file can be found in IncludedFile. 124 unsigned AddIncludeFile(const std::string &Filename, SMLoc IncludeLoc, 125 std::string &IncludedFile); 126 127 /// FindBufferContainingLoc - Return the ID of the buffer containing the 128 /// specified location, returning -1 if not found. 129 int FindBufferContainingLoc(SMLoc Loc) const; 130 131 /// FindLineNumber - Find the line number for the specified location in the 132 /// specified file. This is not a fast method. 133 unsigned FindLineNumber(SMLoc Loc, int BufferID = -1) const { 134 return getLineAndColumn(Loc, BufferID).first; 135 } 136 137 /// getLineAndColumn - Find the line and column number for the specified 138 /// location in the specified file. This is not a fast method. 139 std::pair<unsigned, unsigned> 140 getLineAndColumn(SMLoc Loc, int BufferID = -1) const; 141 142 /// PrintMessage - Emit a message about the specified location with the 143 /// specified string. 144 /// 145 /// @param ShowColors - Display colored messages if output is a terminal and 146 /// the default error handler is used. 147 void PrintMessage(SMLoc Loc, DiagKind Kind, const Twine &Msg, 148 ArrayRef<SMRange> Ranges = ArrayRef<SMRange>(), 149 ArrayRef<SMFixIt> FixIts = ArrayRef<SMFixIt>(), 150 bool ShowColors = true) const; 151 152 153 /// GetMessage - Return an SMDiagnostic at the specified location with the 154 /// specified string. 155 /// 156 /// @param Msg If non-null, the kind of message (e.g., "error") which is 157 /// prefixed to the message. 158 SMDiagnostic GetMessage(SMLoc Loc, DiagKind Kind, const Twine &Msg, 159 ArrayRef<SMRange> Ranges = ArrayRef<SMRange>(), 160 ArrayRef<SMFixIt> FixIts = ArrayRef<SMFixIt>()) const; 161 162 /// PrintIncludeStack - Prints the names of included files and the line of the 163 /// file they were included from. A diagnostic handler can use this before 164 /// printing its custom formatted message. 165 /// 166 /// @param IncludeLoc - The line of the include. 167 /// @param OS the raw_ostream to print on. 168 void PrintIncludeStack(SMLoc IncludeLoc, raw_ostream &OS) const; 169 }; 170 171 172 /// Represents a single fixit, a replacement of one range of text with another. 173 class SMFixIt { 174 SMRange Range; 175 176 std::string Text; 177 178 public: 179 // FIXME: Twine.str() is not very efficient. SMFixIt(SMLoc Loc,const Twine & Insertion)180 SMFixIt(SMLoc Loc, const Twine &Insertion) 181 : Range(Loc, Loc), Text(Insertion.str()) { 182 assert(Loc.isValid()); 183 } 184 185 // FIXME: Twine.str() is not very efficient. SMFixIt(SMRange R,const Twine & Replacement)186 SMFixIt(SMRange R, const Twine &Replacement) 187 : Range(R), Text(Replacement.str()) { 188 assert(R.isValid()); 189 } 190 getText()191 StringRef getText() const { return Text; } getRange()192 SMRange getRange() const { return Range; } 193 194 bool operator<(const SMFixIt &Other) const { 195 if (Range.Start.getPointer() != Other.Range.Start.getPointer()) 196 return Range.Start.getPointer() < Other.Range.Start.getPointer(); 197 if (Range.End.getPointer() != Other.Range.End.getPointer()) 198 return Range.End.getPointer() < Other.Range.End.getPointer(); 199 return Text < Other.Text; 200 } 201 }; 202 203 204 /// SMDiagnostic - Instances of this class encapsulate one diagnostic report, 205 /// allowing printing to a raw_ostream as a caret diagnostic. 206 class SMDiagnostic { 207 const SourceMgr *SM; 208 SMLoc Loc; 209 std::string Filename; 210 int LineNo, ColumnNo; 211 SourceMgr::DiagKind Kind; 212 std::string Message, LineContents; 213 std::vector<std::pair<unsigned, unsigned> > Ranges; 214 SmallVector<SMFixIt, 4> FixIts; 215 216 public: 217 // Null diagnostic. SMDiagnostic()218 SMDiagnostic() 219 : SM(0), LineNo(0), ColumnNo(0), Kind(SourceMgr::DK_Error) {} 220 // Diagnostic with no location (e.g. file not found, command line arg error). SMDiagnostic(StringRef filename,SourceMgr::DiagKind Knd,StringRef Msg)221 SMDiagnostic(StringRef filename, SourceMgr::DiagKind Knd, StringRef Msg) 222 : SM(0), Filename(filename), LineNo(-1), ColumnNo(-1), Kind(Knd), 223 Message(Msg) {} 224 225 // Diagnostic with a location. 226 SMDiagnostic(const SourceMgr &sm, SMLoc L, StringRef FN, 227 int Line, int Col, SourceMgr::DiagKind Kind, 228 StringRef Msg, StringRef LineStr, 229 ArrayRef<std::pair<unsigned,unsigned> > Ranges, 230 ArrayRef<SMFixIt> FixIts = ArrayRef<SMFixIt>()); 231 getSourceMgr()232 const SourceMgr *getSourceMgr() const { return SM; } getLoc()233 SMLoc getLoc() const { return Loc; } getFilename()234 StringRef getFilename() const { return Filename; } getLineNo()235 int getLineNo() const { return LineNo; } getColumnNo()236 int getColumnNo() const { return ColumnNo; } getKind()237 SourceMgr::DiagKind getKind() const { return Kind; } getMessage()238 StringRef getMessage() const { return Message; } getLineContents()239 StringRef getLineContents() const { return LineContents; } getRanges()240 ArrayRef<std::pair<unsigned, unsigned> > getRanges() const { 241 return Ranges; 242 } 243 addFixIt(const SMFixIt & Hint)244 void addFixIt(const SMFixIt &Hint) { 245 FixIts.push_back(Hint); 246 } 247 getFixIts()248 ArrayRef<SMFixIt> getFixIts() const { 249 return FixIts; 250 } 251 252 void print(const char *ProgName, raw_ostream &S, 253 bool ShowColors = true) const; 254 }; 255 256 } // end llvm namespace 257 258 #endif 259