1 //===- MCCodeView.h - Machine Code CodeView support -------------*- 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 // Holds state from .cv_file and .cv_loc directives for later emission.
11 //
12 //===----------------------------------------------------------------------===//
13 
14 #ifndef LLVM_MC_MCCODEVIEW_H
15 #define LLVM_MC_MCCODEVIEW_H
16 
17 #include "llvm/ADT/StringMap.h"
18 #include "llvm/ADT/StringRef.h"
19 #include "llvm/MC/MCFragment.h"
20 #include "llvm/MC/MCObjectStreamer.h"
21 #include <map>
22 #include <vector>
23 
24 namespace llvm {
25 class MCContext;
26 class MCObjectStreamer;
27 class MCStreamer;
28 class CodeViewContext;
29 
30 /// Instances of this class represent the information from a
31 /// .cv_loc directive.
32 class MCCVLoc {
33   uint32_t FunctionId;
34   uint32_t FileNum;
35   uint32_t Line;
36   uint16_t Column;
37   uint16_t PrologueEnd : 1;
38   uint16_t IsStmt : 1;
39 
40 private: // CodeViewContext manages these
41   friend class CodeViewContext;
MCCVLoc(unsigned functionid,unsigned fileNum,unsigned line,unsigned column,bool prologueend,bool isstmt)42   MCCVLoc(unsigned functionid, unsigned fileNum, unsigned line, unsigned column,
43           bool prologueend, bool isstmt)
44       : FunctionId(functionid), FileNum(fileNum), Line(line), Column(column),
45         PrologueEnd(prologueend), IsStmt(isstmt) {}
46 
47   // Allow the default copy constructor and assignment operator to be used
48   // for an MCCVLoc object.
49 
50 public:
getFunctionId()51   unsigned getFunctionId() const { return FunctionId; }
52 
53   /// Get the FileNum of this MCCVLoc.
getFileNum()54   unsigned getFileNum() const { return FileNum; }
55 
56   /// Get the Line of this MCCVLoc.
getLine()57   unsigned getLine() const { return Line; }
58 
59   /// Get the Column of this MCCVLoc.
getColumn()60   unsigned getColumn() const { return Column; }
61 
isPrologueEnd()62   bool isPrologueEnd() const { return PrologueEnd; }
isStmt()63   bool isStmt() const { return IsStmt; }
64 
setFunctionId(unsigned FID)65   void setFunctionId(unsigned FID) { FunctionId = FID; }
66 
67   /// Set the FileNum of this MCCVLoc.
setFileNum(unsigned fileNum)68   void setFileNum(unsigned fileNum) { FileNum = fileNum; }
69 
70   /// Set the Line of this MCCVLoc.
setLine(unsigned line)71   void setLine(unsigned line) { Line = line; }
72 
73   /// Set the Column of this MCCVLoc.
setColumn(unsigned column)74   void setColumn(unsigned column) {
75     assert(column <= UINT16_MAX);
76     Column = column;
77   }
78 
setPrologueEnd(bool PE)79   void setPrologueEnd(bool PE) { PrologueEnd = PE; }
setIsStmt(bool IS)80   void setIsStmt(bool IS) { IsStmt = IS; }
81 };
82 
83 /// Instances of this class represent the line information for
84 /// the CodeView line table entries.  Which is created after a machine
85 /// instruction is assembled and uses an address from a temporary label
86 /// created at the current address in the current section and the info from
87 /// the last .cv_loc directive seen as stored in the context.
88 class MCCVLineEntry : public MCCVLoc {
89   const MCSymbol *Label;
90 
91 private:
92   // Allow the default copy constructor and assignment operator to be used
93   // for an MCCVLineEntry object.
94 
95 public:
96   // Constructor to create an MCCVLineEntry given a symbol and the dwarf loc.
MCCVLineEntry(const MCSymbol * Label,const MCCVLoc loc)97   MCCVLineEntry(const MCSymbol *Label, const MCCVLoc loc)
98       : MCCVLoc(loc), Label(Label) {}
99 
getLabel()100   const MCSymbol *getLabel() const { return Label; }
101 
102   // This is called when an instruction is assembled into the specified
103   // section and if there is information from the last .cv_loc directive that
104   // has yet to have a line entry made for it is made.
105   static void Make(MCObjectStreamer *MCOS);
106 };
107 
108 /// Information describing a function or inlined call site introduced by
109 /// .cv_func_id or .cv_inline_site_id. Accumulates information from .cv_loc
110 /// directives used with this function's id or the id of an inlined call site
111 /// within this function or inlined call site.
112 struct MCCVFunctionInfo {
113   /// If this represents an inlined call site, then ParentFuncIdPlusOne will be
114   /// the parent function id plus one. If this represents a normal function,
115   /// then there is no parent, and ParentFuncIdPlusOne will be FunctionSentinel.
116   /// If this struct is an unallocated slot in the function info vector, then
117   /// ParentFuncIdPlusOne will be zero.
118   unsigned ParentFuncIdPlusOne = 0;
119 
120   enum : unsigned { FunctionSentinel = ~0U };
121 
122   struct LineInfo {
123     unsigned File;
124     unsigned Line;
125     unsigned Col;
126   };
127 
128   LineInfo InlinedAt;
129 
130   /// The section of the first .cv_loc directive used for this function, or null
131   /// if none has been seen yet.
132   MCSection *Section = nullptr;
133 
134   /// Map from inlined call site id to the inlined at location to use for that
135   /// call site. Call chains are collapsed, so for the call chain 'f -> g -> h',
136   /// the InlinedAtMap of 'f' will contain entries for 'g' and 'h' that both
137   /// list the line info for the 'g' call site.
138   DenseMap<unsigned, LineInfo> InlinedAtMap;
139 
140   /// Returns true if this is function info has not yet been used in a
141   /// .cv_func_id or .cv_inline_site_id directive.
isUnallocatedFunctionInfoMCCVFunctionInfo142   bool isUnallocatedFunctionInfo() const { return ParentFuncIdPlusOne == 0; }
143 
144   /// Returns true if this represents an inlined call site, meaning
145   /// ParentFuncIdPlusOne is neither zero nor ~0U.
isInlinedCallSiteMCCVFunctionInfo146   bool isInlinedCallSite() const {
147     return !isUnallocatedFunctionInfo() &&
148            ParentFuncIdPlusOne != FunctionSentinel;
149   }
150 
getParentFuncIdMCCVFunctionInfo151   unsigned getParentFuncId() const {
152     assert(isInlinedCallSite());
153     return ParentFuncIdPlusOne - 1;
154   }
155 };
156 
157 /// Holds state from .cv_file and .cv_loc directives for later emission.
158 class CodeViewContext {
159 public:
160   CodeViewContext();
161   ~CodeViewContext();
162 
163   bool isValidFileNumber(unsigned FileNumber) const;
164   bool addFile(MCStreamer &OS, unsigned FileNumber, StringRef Filename,
165                ArrayRef<uint8_t> ChecksumBytes, uint8_t ChecksumKind);
166 
167   /// Records the function id of a normal function. Returns false if the
168   /// function id has already been used, and true otherwise.
169   bool recordFunctionId(unsigned FuncId);
170 
171   /// Records the function id of an inlined call site. Records the "inlined at"
172   /// location info of the call site, including what function or inlined call
173   /// site it was inlined into. Returns false if the function id has already
174   /// been used, and true otherwise.
175   bool recordInlinedCallSiteId(unsigned FuncId, unsigned IAFunc,
176                                unsigned IAFile, unsigned IALine,
177                                unsigned IACol);
178 
179   /// Retreive the function info if this is a valid function id, or nullptr.
180   MCCVFunctionInfo *getCVFunctionInfo(unsigned FuncId);
181 
182   /// Saves the information from the currently parsed .cv_loc directive
183   /// and sets CVLocSeen.  When the next instruction is assembled an entry
184   /// in the line number table with this information and the address of the
185   /// instruction will be created.
setCurrentCVLoc(unsigned FunctionId,unsigned FileNo,unsigned Line,unsigned Column,bool PrologueEnd,bool IsStmt)186   void setCurrentCVLoc(unsigned FunctionId, unsigned FileNo, unsigned Line,
187                        unsigned Column, bool PrologueEnd, bool IsStmt) {
188     CurrentCVLoc.setFunctionId(FunctionId);
189     CurrentCVLoc.setFileNum(FileNo);
190     CurrentCVLoc.setLine(Line);
191     CurrentCVLoc.setColumn(Column);
192     CurrentCVLoc.setPrologueEnd(PrologueEnd);
193     CurrentCVLoc.setIsStmt(IsStmt);
194     CVLocSeen = true;
195   }
196 
getCVLocSeen()197   bool getCVLocSeen() { return CVLocSeen; }
clearCVLocSeen()198   void clearCVLocSeen() { CVLocSeen = false; }
199 
getCurrentCVLoc()200   const MCCVLoc &getCurrentCVLoc() { return CurrentCVLoc; }
201 
202   bool isValidCVFileNumber(unsigned FileNumber);
203 
204   /// Add a line entry.
205   void addLineEntry(const MCCVLineEntry &LineEntry);
206 
207   std::vector<MCCVLineEntry> getFunctionLineEntries(unsigned FuncId);
208 
209   std::pair<size_t, size_t> getLineExtent(unsigned FuncId);
210 
211   ArrayRef<MCCVLineEntry> getLinesForExtent(size_t L, size_t R);
212 
213   /// Emits a line table substream.
214   void emitLineTableForFunction(MCObjectStreamer &OS, unsigned FuncId,
215                                 const MCSymbol *FuncBegin,
216                                 const MCSymbol *FuncEnd);
217 
218   void emitInlineLineTableForFunction(MCObjectStreamer &OS,
219                                       unsigned PrimaryFunctionId,
220                                       unsigned SourceFileId,
221                                       unsigned SourceLineNum,
222                                       const MCSymbol *FnStartSym,
223                                       const MCSymbol *FnEndSym);
224 
225   /// Encodes the binary annotations once we have a layout.
226   void encodeInlineLineTable(MCAsmLayout &Layout,
227                              MCCVInlineLineTableFragment &F);
228 
229   void
230   emitDefRange(MCObjectStreamer &OS,
231                ArrayRef<std::pair<const MCSymbol *, const MCSymbol *>> Ranges,
232                StringRef FixedSizePortion);
233 
234   void encodeDefRange(MCAsmLayout &Layout, MCCVDefRangeFragment &F);
235 
236   /// Emits the string table substream.
237   void emitStringTable(MCObjectStreamer &OS);
238 
239   /// Emits the file checksum substream.
240   void emitFileChecksums(MCObjectStreamer &OS);
241 
242   /// Emits the offset into the checksum table of the given file number.
243   void emitFileChecksumOffset(MCObjectStreamer &OS, unsigned FileNo);
244 
245   /// Add something to the string table.  Returns the final string as well as
246   /// offset into the string table.
247   std::pair<StringRef, unsigned> addToStringTable(StringRef S);
248 
249 private:
250   /// The current CodeView line information from the last .cv_loc directive.
251   MCCVLoc CurrentCVLoc = MCCVLoc(0, 0, 0, 0, false, true);
252   bool CVLocSeen = false;
253 
254   /// Map from string to string table offset.
255   StringMap<unsigned> StringTable;
256 
257   /// The fragment that ultimately holds our strings.
258   MCDataFragment *StrTabFragment = nullptr;
259   bool InsertedStrTabFragment = false;
260 
261   MCDataFragment *getStringTableFragment();
262 
263   /// Get a string table offset.
264   unsigned getStringTableOffset(StringRef S);
265 
266   struct FileInfo {
267     unsigned StringTableOffset;
268 
269     // Indicates if this FileInfo corresponds to an actual file, or hasn't been
270     // set yet.
271     bool Assigned = false;
272 
273     uint8_t ChecksumKind;
274 
275     ArrayRef<uint8_t> Checksum;
276 
277     // Checksum offset stored as a symbol because it might be requested
278     // before it has been calculated, so a fixup may be needed.
279     MCSymbol *ChecksumTableOffset;
280   };
281 
282   /// Array storing added file information.
283   SmallVector<FileInfo, 4> Files;
284 
285   /// The offset of the first and last .cv_loc directive for a given function
286   /// id.
287   std::map<unsigned, std::pair<size_t, size_t>> MCCVLineStartStop;
288 
289   /// A collection of MCCVLineEntry for each section.
290   std::vector<MCCVLineEntry> MCCVLines;
291 
292   /// All known functions and inlined call sites, indexed by function id.
293   std::vector<MCCVFunctionInfo> Functions;
294 
295   /// Indicate whether we have already laid out the checksum table addresses or
296   /// not.
297   bool ChecksumOffsetsAssigned = false;
298 };
299 
300 } // end namespace llvm
301 #endif
302