Eugene Zelenko | fb69e66 | 2017-06-06 22:22:41 +0000 | [diff] [blame] | 1 | //===- llvm/lib/CodeGen/AsmPrinter/CodeViewDebug.h --------------*- C++ -*-===// |
Timur Iskhodzhanov | f166f6c | 2014-01-30 01:39:17 +0000 | [diff] [blame] | 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 | // |
Reid Kleckner | 70f5bc9 | 2016-01-14 19:25:04 +0000 | [diff] [blame] | 10 | // This file contains support for writing Microsoft CodeView debug info. |
Timur Iskhodzhanov | f166f6c | 2014-01-30 01:39:17 +0000 | [diff] [blame] | 11 | // |
| 12 | //===----------------------------------------------------------------------===// |
| 13 | |
Reid Kleckner | 70f5bc9 | 2016-01-14 19:25:04 +0000 | [diff] [blame] | 14 | #ifndef LLVM_LIB_CODEGEN_ASMPRINTER_CODEVIEWDEBUG_H |
| 15 | #define LLVM_LIB_CODEGEN_ASMPRINTER_CODEVIEWDEBUG_H |
Timur Iskhodzhanov | f166f6c | 2014-01-30 01:39:17 +0000 | [diff] [blame] | 16 | |
Eugene Zelenko | fb69e66 | 2017-06-06 22:22:41 +0000 | [diff] [blame] | 17 | #include "DbgValueHistoryCalculator.h" |
Reid Kleckner | f9c275f | 2016-02-10 20:55:49 +0000 | [diff] [blame] | 18 | #include "DebugHandlerBase.h" |
Eugene Zelenko | fb69e66 | 2017-06-06 22:22:41 +0000 | [diff] [blame] | 19 | #include "llvm/ADT/ArrayRef.h" |
Timur Iskhodzhanov | f166f6c | 2014-01-30 01:39:17 +0000 | [diff] [blame] | 20 | #include "llvm/ADT/DenseMap.h" |
Eugene Zelenko | fb69e66 | 2017-06-06 22:22:41 +0000 | [diff] [blame] | 21 | #include "llvm/ADT/DenseSet.h" |
| 22 | #include "llvm/ADT/MapVector.h" |
| 23 | #include "llvm/ADT/SetVector.h" |
| 24 | #include "llvm/ADT/SmallVector.h" |
| 25 | #include "llvm/DebugInfo/CodeView/CodeView.h" |
Zachary Turner | 048f8f9 | 2017-12-13 22:33:58 +0000 | [diff] [blame] | 26 | #include "llvm/DebugInfo/CodeView/GlobalTypeTableBuilder.h" |
Reid Kleckner | f3b9ba4 | 2016-01-29 18:16:43 +0000 | [diff] [blame] | 27 | #include "llvm/DebugInfo/CodeView/TypeIndex.h" |
Chandler Carruth | 9205140 | 2014-03-05 10:30:38 +0000 | [diff] [blame] | 28 | #include "llvm/IR/DebugLoc.h" |
Eugene Zelenko | fb69e66 | 2017-06-06 22:22:41 +0000 | [diff] [blame] | 29 | #include "llvm/Support/Allocator.h" |
| 30 | #include "llvm/Support/Compiler.h" |
| 31 | #include <cstdint> |
| 32 | #include <map> |
| 33 | #include <string> |
| 34 | #include <tuple> |
| 35 | #include <unordered_map> |
| 36 | #include <utility> |
| 37 | #include <vector> |
Timur Iskhodzhanov | f166f6c | 2014-01-30 01:39:17 +0000 | [diff] [blame] | 38 | |
| 39 | namespace llvm { |
Reid Kleckner | f9c275f | 2016-02-10 20:55:49 +0000 | [diff] [blame] | 40 | |
Amjad Aboud | 76c9eb9 | 2016-06-18 10:25:07 +0000 | [diff] [blame] | 41 | struct ClassInfo; |
Eugene Zelenko | fb69e66 | 2017-06-06 22:22:41 +0000 | [diff] [blame] | 42 | class StringRef; |
| 43 | class AsmPrinter; |
| 44 | class Function; |
| 45 | class GlobalVariable; |
| 46 | class MCSectionCOFF; |
| 47 | class MCStreamer; |
| 48 | class MCSymbol; |
| 49 | class MachineFunction; |
Reid Kleckner | f9c275f | 2016-02-10 20:55:49 +0000 | [diff] [blame] | 50 | |
Adrian Prantl | 5f8f34e4 | 2018-05-01 15:54:18 +0000 | [diff] [blame^] | 51 | /// Collects and handles line tables information in a CodeView format. |
Reid Kleckner | f9c275f | 2016-02-10 20:55:49 +0000 | [diff] [blame] | 52 | class LLVM_LIBRARY_VISIBILITY CodeViewDebug : public DebugHandlerBase { |
Reid Kleckner | dac21b4 | 2016-02-03 21:15:48 +0000 | [diff] [blame] | 53 | MCStreamer &OS; |
Eugene Zelenko | fb69e66 | 2017-06-06 22:22:41 +0000 | [diff] [blame] | 54 | BumpPtrAllocator Allocator; |
Zachary Turner | 048f8f9 | 2017-12-13 22:33:58 +0000 | [diff] [blame] | 55 | codeview::GlobalTypeTableBuilder TypeTable; |
Reid Kleckner | f9c275f | 2016-02-10 20:55:49 +0000 | [diff] [blame] | 56 | |
Reid Kleckner | 876330d | 2016-02-12 21:48:30 +0000 | [diff] [blame] | 57 | /// Represents the most general definition range. |
| 58 | struct LocalVarDefRange { |
| 59 | /// Indicates that variable data is stored in memory relative to the |
| 60 | /// specified register. |
| 61 | int InMemory : 1; |
| 62 | |
| 63 | /// Offset of variable data in memory. |
| 64 | int DataOffset : 31; |
| 65 | |
Reid Kleckner | 2b3e642 | 2016-10-05 21:21:33 +0000 | [diff] [blame] | 66 | /// Non-zero if this is a piece of an aggregate. |
| 67 | uint16_t IsSubfield : 1; |
| 68 | |
| 69 | /// Offset into aggregate. |
| 70 | uint16_t StructOffset : 15; |
Reid Kleckner | 876330d | 2016-02-12 21:48:30 +0000 | [diff] [blame] | 71 | |
| 72 | /// Register containing the data or the register base of the memory |
| 73 | /// location containing the data. |
| 74 | uint16_t CVRegister; |
| 75 | |
| 76 | /// Compares all location fields. This includes all fields except the label |
| 77 | /// ranges. |
| 78 | bool isDifferentLocation(LocalVarDefRange &O) { |
| 79 | return InMemory != O.InMemory || DataOffset != O.DataOffset || |
Reid Kleckner | 2b3e642 | 2016-10-05 21:21:33 +0000 | [diff] [blame] | 80 | IsSubfield != O.IsSubfield || StructOffset != O.StructOffset || |
| 81 | CVRegister != O.CVRegister; |
Reid Kleckner | 876330d | 2016-02-12 21:48:30 +0000 | [diff] [blame] | 82 | } |
| 83 | |
| 84 | SmallVector<std::pair<const MCSymbol *, const MCSymbol *>, 1> Ranges; |
| 85 | }; |
| 86 | |
| 87 | static LocalVarDefRange createDefRangeMem(uint16_t CVRegister, int Offset); |
Reid Kleckner | 2b3e642 | 2016-10-05 21:21:33 +0000 | [diff] [blame] | 88 | static LocalVarDefRange createDefRangeGeneral(uint16_t CVRegister, |
| 89 | bool InMemory, int Offset, |
| 90 | bool IsSubfield, |
| 91 | uint16_t StructOffset); |
Reid Kleckner | 876330d | 2016-02-12 21:48:30 +0000 | [diff] [blame] | 92 | |
Reid Kleckner | f9c275f | 2016-02-10 20:55:49 +0000 | [diff] [blame] | 93 | /// Similar to DbgVariable in DwarfDebug, but not dwarf-specific. |
| 94 | struct LocalVariable { |
| 95 | const DILocalVariable *DIVar = nullptr; |
Reid Kleckner | 876330d | 2016-02-12 21:48:30 +0000 | [diff] [blame] | 96 | SmallVector<LocalVarDefRange, 1> DefRanges; |
Reid Kleckner | 08f5fd5 | 2017-08-31 15:56:49 +0000 | [diff] [blame] | 97 | bool UseReferenceType = false; |
Reid Kleckner | f9c275f | 2016-02-10 20:55:49 +0000 | [diff] [blame] | 98 | }; |
Timur Iskhodzhanov | f166f6c | 2014-01-30 01:39:17 +0000 | [diff] [blame] | 99 | |
Reid Kleckner | f3b9ba4 | 2016-01-29 18:16:43 +0000 | [diff] [blame] | 100 | struct InlineSite { |
Reid Kleckner | f9c275f | 2016-02-10 20:55:49 +0000 | [diff] [blame] | 101 | SmallVector<LocalVariable, 1> InlinedLocals; |
| 102 | SmallVector<const DILocation *, 1> ChildSites; |
Reid Kleckner | f3b9ba4 | 2016-01-29 18:16:43 +0000 | [diff] [blame] | 103 | const DISubprogram *Inlinee = nullptr; |
Reid Kleckner | c29b4f0 | 2016-07-14 23:47:15 +0000 | [diff] [blame] | 104 | |
| 105 | /// The ID of the inline site or function used with .cv_loc. Not a type |
| 106 | /// index. |
Reid Kleckner | f3b9ba4 | 2016-01-29 18:16:43 +0000 | [diff] [blame] | 107 | unsigned SiteFuncId = 0; |
| 108 | }; |
| 109 | |
Reid Kleckner | 5a791ee | 2018-03-15 21:24:04 +0000 | [diff] [blame] | 110 | // Combines information from DILexicalBlock and LexicalScope. |
| 111 | struct LexicalBlock { |
| 112 | SmallVector<LocalVariable, 1> Locals; |
| 113 | SmallVector<LexicalBlock *, 1> Children; |
| 114 | const MCSymbol *Begin; |
| 115 | const MCSymbol *End; |
| 116 | StringRef Name; |
| 117 | }; |
| 118 | |
Timur Iskhodzhanov | f166f6c | 2014-01-30 01:39:17 +0000 | [diff] [blame] | 119 | // For each function, store a vector of labels to its instructions, as well as |
| 120 | // to the end of the function. |
| 121 | struct FunctionInfo { |
Reid Kleckner | 55baeef | 2018-03-15 21:12:21 +0000 | [diff] [blame] | 122 | FunctionInfo() = default; |
| 123 | |
| 124 | // Uncopyable. |
| 125 | FunctionInfo(const FunctionInfo &FI) = delete; |
| 126 | |
Reid Kleckner | f3b9ba4 | 2016-01-29 18:16:43 +0000 | [diff] [blame] | 127 | /// Map from inlined call site to inlined instructions and child inlined |
| 128 | /// call sites. Listed in program order. |
Reid Kleckner | f9c275f | 2016-02-10 20:55:49 +0000 | [diff] [blame] | 129 | std::unordered_map<const DILocation *, InlineSite> InlineSites; |
| 130 | |
| 131 | /// Ordered list of top-level inlined call sites. |
| 132 | SmallVector<const DILocation *, 1> ChildSites; |
| 133 | |
| 134 | SmallVector<LocalVariable, 1> Locals; |
Reid Kleckner | f3b9ba4 | 2016-01-29 18:16:43 +0000 | [diff] [blame] | 135 | |
Reid Kleckner | 5a791ee | 2018-03-15 21:24:04 +0000 | [diff] [blame] | 136 | std::unordered_map<const DILexicalBlockBase*, LexicalBlock> LexicalBlocks; |
| 137 | |
| 138 | // Lexical blocks containing local variables. |
| 139 | SmallVector<LexicalBlock *, 1> ChildBlocks; |
| 140 | |
Reid Kleckner | e33c94f | 2017-09-05 20:14:58 +0000 | [diff] [blame] | 141 | std::vector<std::pair<MCSymbol *, MDNode *>> Annotations; |
| 142 | |
Reid Kleckner | 1fcd610 | 2016-02-02 17:41:18 +0000 | [diff] [blame] | 143 | const MCSymbol *Begin = nullptr; |
| 144 | const MCSymbol *End = nullptr; |
Reid Kleckner | 2214ed8 | 2016-01-29 00:49:42 +0000 | [diff] [blame] | 145 | unsigned FuncId = 0; |
Reid Kleckner | f3b9ba4 | 2016-01-29 18:16:43 +0000 | [diff] [blame] | 146 | unsigned LastFileId = 0; |
Reid Kleckner | 2214ed8 | 2016-01-29 00:49:42 +0000 | [diff] [blame] | 147 | bool HaveLineInfo = false; |
Reid Kleckner | 9533af4 | 2016-01-16 00:09:09 +0000 | [diff] [blame] | 148 | }; |
Eugene Zelenko | fb69e66 | 2017-06-06 22:22:41 +0000 | [diff] [blame] | 149 | FunctionInfo *CurFn = nullptr; |
Timur Iskhodzhanov | f166f6c | 2014-01-30 01:39:17 +0000 | [diff] [blame] | 150 | |
Reid Kleckner | 5a791ee | 2018-03-15 21:24:04 +0000 | [diff] [blame] | 151 | // Map used to seperate variables according to the lexical scope they belong |
| 152 | // in. This is populated by recordLocalVariable() before |
| 153 | // collectLexicalBlocks() separates the variables between the FunctionInfo |
| 154 | // and LexicalBlocks. |
| 155 | DenseMap<const LexicalScope *, SmallVector<LocalVariable, 1>> ScopeVariables; |
| 156 | |
Reid Kleckner | 5d122f8 | 2016-05-25 23:16:12 +0000 | [diff] [blame] | 157 | /// The set of comdat .debug$S sections that we've seen so far. Each section |
| 158 | /// must start with a magic version number that must only be emitted once. |
| 159 | /// This set tracks which sections we've already opened. |
| 160 | DenseSet<MCSectionCOFF *> ComdatDebugSections; |
| 161 | |
| 162 | /// Switch to the appropriate .debug$S section for GVSym. If GVSym, the symbol |
| 163 | /// of an emitted global value, is in a comdat COFF section, this will switch |
| 164 | /// to a new .debug$S section in that comdat. This method ensures that the |
| 165 | /// section starts with the magic version number on first use. If GVSym is |
| 166 | /// null, uses the main .debug$S section. |
| 167 | void switchToDebugSectionForSymbol(const MCSymbol *GVSym); |
| 168 | |
Reid Kleckner | fbd7787 | 2016-03-18 18:54:32 +0000 | [diff] [blame] | 169 | /// The next available function index for use with our .cv_* directives. Not |
| 170 | /// to be confused with type indices for LF_FUNC_ID records. |
Reid Kleckner | 2214ed8 | 2016-01-29 00:49:42 +0000 | [diff] [blame] | 171 | unsigned NextFuncId = 0; |
Timur Iskhodzhanov | f166f6c | 2014-01-30 01:39:17 +0000 | [diff] [blame] | 172 | |
Reid Kleckner | 876330d | 2016-02-12 21:48:30 +0000 | [diff] [blame] | 173 | InlineSite &getInlineSite(const DILocation *InlinedAt, |
| 174 | const DISubprogram *Inlinee); |
Reid Kleckner | f3b9ba4 | 2016-01-29 18:16:43 +0000 | [diff] [blame] | 175 | |
David Majnemer | 75c3ebf | 2016-06-02 17:13:53 +0000 | [diff] [blame] | 176 | codeview::TypeIndex getFuncIdForSubprogram(const DISubprogram *SP); |
Reid Kleckner | 2280f93 | 2016-05-23 20:23:46 +0000 | [diff] [blame] | 177 | |
Bob Haarman | 223303c | 2017-08-29 20:59:25 +0000 | [diff] [blame] | 178 | void calculateRanges(LocalVariable &Var, |
| 179 | const DbgValueHistoryMap::InstrRanges &Ranges); |
| 180 | |
Reid Kleckner | 1fcd610 | 2016-02-02 17:41:18 +0000 | [diff] [blame] | 181 | static void collectInlineSiteChildren(SmallVectorImpl<unsigned> &Children, |
| 182 | const FunctionInfo &FI, |
| 183 | const InlineSite &Site); |
| 184 | |
Reid Kleckner | 2214ed8 | 2016-01-29 00:49:42 +0000 | [diff] [blame] | 185 | /// Remember some debug info about each function. Keep it in a stable order to |
| 186 | /// emit at the end of the TU. |
Reid Kleckner | 55baeef | 2018-03-15 21:12:21 +0000 | [diff] [blame] | 187 | MapVector<const Function *, std::unique_ptr<FunctionInfo>> FnDebugInfo; |
Timur Iskhodzhanov | f166f6c | 2014-01-30 01:39:17 +0000 | [diff] [blame] | 188 | |
Reid Kleckner | bc6f52d | 2017-10-31 21:52:15 +0000 | [diff] [blame] | 189 | /// Map from full file path to .cv_file id. Full paths are built from DIFiles |
| 190 | /// and are stored in FileToFilepathMap; |
| 191 | DenseMap<StringRef, unsigned> FileIdMap; |
Timur Iskhodzhanov | f166f6c | 2014-01-30 01:39:17 +0000 | [diff] [blame] | 192 | |
Reid Kleckner | fbd7787 | 2016-03-18 18:54:32 +0000 | [diff] [blame] | 193 | /// All inlined subprograms in the order they should be emitted. |
Reid Kleckner | 2280f93 | 2016-05-23 20:23:46 +0000 | [diff] [blame] | 194 | SmallSetVector<const DISubprogram *, 4> InlinedSubprograms; |
Reid Kleckner | f3b9ba4 | 2016-01-29 18:16:43 +0000 | [diff] [blame] | 195 | |
Amjad Aboud | 76c9eb9 | 2016-06-18 10:25:07 +0000 | [diff] [blame] | 196 | /// Map from a pair of DI metadata nodes and its DI type (or scope) that can |
| 197 | /// be nullptr, to CodeView type indices. Primarily indexed by |
| 198 | /// {DIType*, DIType*} and {DISubprogram*, DIType*}. |
| 199 | /// |
| 200 | /// The second entry in the key is needed for methods as DISubroutineType |
| 201 | /// representing static method type are shared with non-method function type. |
| 202 | DenseMap<std::pair<const DINode *, const DIType *>, codeview::TypeIndex> |
| 203 | TypeIndices; |
Reid Kleckner | f3b9ba4 | 2016-01-29 18:16:43 +0000 | [diff] [blame] | 204 | |
Reid Kleckner | a8d5740 | 2016-06-03 15:58:20 +0000 | [diff] [blame] | 205 | /// Map from DICompositeType* to complete type index. Non-record types are |
| 206 | /// always looked up in the normal TypeIndices map. |
| 207 | DenseMap<const DICompositeType *, codeview::TypeIndex> CompleteTypeIndices; |
| 208 | |
Reid Kleckner | 643dd83 | 2016-06-22 17:15:28 +0000 | [diff] [blame] | 209 | /// Complete record types to emit after all active type lowerings are |
| 210 | /// finished. |
| 211 | SmallVector<const DICompositeType *, 4> DeferredCompleteTypes; |
| 212 | |
| 213 | /// Number of type lowering frames active on the stack. |
| 214 | unsigned TypeEmissionLevel = 0; |
| 215 | |
Reid Kleckner | 9f7f3e1 | 2016-06-24 16:24:24 +0000 | [diff] [blame] | 216 | codeview::TypeIndex VBPType; |
| 217 | |
David Majnemer | 3128b10 | 2016-06-15 18:00:01 +0000 | [diff] [blame] | 218 | const DISubprogram *CurrentSubprogram = nullptr; |
| 219 | |
| 220 | // The UDTs we have seen while processing types; each entry is a pair of type |
| 221 | // index and type name. |
Zachary Turner | a7b0417 | 2017-08-28 18:49:04 +0000 | [diff] [blame] | 222 | std::vector<std::pair<std::string, const DIType *>> LocalUDTs; |
| 223 | std::vector<std::pair<std::string, const DIType *>> GlobalUDTs; |
David Majnemer | 3128b10 | 2016-06-15 18:00:01 +0000 | [diff] [blame] | 224 | |
Eugene Zelenko | fb69e66 | 2017-06-06 22:22:41 +0000 | [diff] [blame] | 225 | using FileToFilepathMapTy = std::map<const DIFile *, std::string>; |
Reid Kleckner | 9533af4 | 2016-01-16 00:09:09 +0000 | [diff] [blame] | 226 | FileToFilepathMapTy FileToFilepathMap; |
Eugene Zelenko | fb69e66 | 2017-06-06 22:22:41 +0000 | [diff] [blame] | 227 | |
Reid Kleckner | 9533af4 | 2016-01-16 00:09:09 +0000 | [diff] [blame] | 228 | StringRef getFullFilepath(const DIFile *S); |
Timur Iskhodzhanov | f166f6c | 2014-01-30 01:39:17 +0000 | [diff] [blame] | 229 | |
Reid Kleckner | 2214ed8 | 2016-01-29 00:49:42 +0000 | [diff] [blame] | 230 | unsigned maybeRecordFile(const DIFile *F); |
| 231 | |
Benjamin Kramer | bdc4956 | 2016-06-12 15:39:02 +0000 | [diff] [blame] | 232 | void maybeRecordLocation(const DebugLoc &DL, const MachineFunction *MF); |
Timur Iskhodzhanov | f166f6c | 2014-01-30 01:39:17 +0000 | [diff] [blame] | 233 | |
Amjad Aboud | 76c9eb9 | 2016-06-18 10:25:07 +0000 | [diff] [blame] | 234 | void clear(); |
David Majnemer | 3128b10 | 2016-06-15 18:00:01 +0000 | [diff] [blame] | 235 | |
| 236 | void setCurrentSubprogram(const DISubprogram *SP) { |
| 237 | CurrentSubprogram = SP; |
| 238 | LocalUDTs.clear(); |
Timur Iskhodzhanov | f166f6c | 2014-01-30 01:39:17 +0000 | [diff] [blame] | 239 | } |
| 240 | |
Reid Kleckner | 5d122f8 | 2016-05-25 23:16:12 +0000 | [diff] [blame] | 241 | /// Emit the magic version number at the start of a CodeView type or symbol |
Alexandre Ganea | d9e9674 | 2018-04-09 20:17:56 +0000 | [diff] [blame] | 242 | /// section. Appears at the front of every .debug$S or .debug$T or .debug$P |
| 243 | /// section. |
Reid Kleckner | 5d122f8 | 2016-05-25 23:16:12 +0000 | [diff] [blame] | 244 | void emitCodeViewMagicVersion(); |
| 245 | |
Reid Kleckner | f3b9ba4 | 2016-01-29 18:16:43 +0000 | [diff] [blame] | 246 | void emitTypeInformation(); |
| 247 | |
Zachary Turner | 048f8f9 | 2017-12-13 22:33:58 +0000 | [diff] [blame] | 248 | void emitTypeGlobalHashes(); |
| 249 | |
Adrian McCarthy | c64acfd | 2016-09-20 17:20:51 +0000 | [diff] [blame] | 250 | void emitCompilerInformation(); |
| 251 | |
Reid Kleckner | 5d122f8 | 2016-05-25 23:16:12 +0000 | [diff] [blame] | 252 | void emitInlineeLinesSubsection(); |
Reid Kleckner | 1fcd610 | 2016-02-02 17:41:18 +0000 | [diff] [blame] | 253 | |
Brock Wyma | 94ece8f | 2018-04-16 16:53:57 +0000 | [diff] [blame] | 254 | void emitDebugInfoForThunk(const Function *GV, |
| 255 | FunctionInfo &FI, |
| 256 | const MCSymbol *Fn); |
| 257 | |
Reid Kleckner | 2214ed8 | 2016-01-29 00:49:42 +0000 | [diff] [blame] | 258 | void emitDebugInfoForFunction(const Function *GV, FunctionInfo &FI); |
Timur Iskhodzhanov | f166f6c | 2014-01-30 01:39:17 +0000 | [diff] [blame] | 259 | |
Reid Kleckner | 6f3406d | 2016-06-07 00:02:03 +0000 | [diff] [blame] | 260 | void emitDebugInfoForGlobals(); |
| 261 | |
Hans Wennborg | b510b45 | 2016-06-23 16:33:53 +0000 | [diff] [blame] | 262 | void emitDebugInfoForRetainedTypes(); |
| 263 | |
Zachary Turner | a7b0417 | 2017-08-28 18:49:04 +0000 | [diff] [blame] | 264 | void |
| 265 | emitDebugInfoForUDTs(ArrayRef<std::pair<std::string, const DIType *>> UDTs); |
David Majnemer | 3128b10 | 2016-06-15 18:00:01 +0000 | [diff] [blame] | 266 | |
Peter Collingbourne | d4135bb | 2016-09-13 01:12:59 +0000 | [diff] [blame] | 267 | void emitDebugInfoForGlobal(const DIGlobalVariable *DIGV, |
| 268 | const GlobalVariable *GV, MCSymbol *GVSym); |
Reid Kleckner | 6f3406d | 2016-06-07 00:02:03 +0000 | [diff] [blame] | 269 | |
| 270 | /// Opens a subsection of the given kind in a .debug$S codeview section. |
| 271 | /// Returns an end label for use with endCVSubsection when the subsection is |
| 272 | /// finished. |
Zachary Turner | 8c099fe | 2017-05-30 16:36:15 +0000 | [diff] [blame] | 273 | MCSymbol *beginCVSubsection(codeview::DebugSubsectionKind Kind); |
Reid Kleckner | 6f3406d | 2016-06-07 00:02:03 +0000 | [diff] [blame] | 274 | |
| 275 | void endCVSubsection(MCSymbol *EndLabel); |
| 276 | |
Reid Kleckner | f3b9ba4 | 2016-01-29 18:16:43 +0000 | [diff] [blame] | 277 | void emitInlinedCallSite(const FunctionInfo &FI, const DILocation *InlinedAt, |
| 278 | const InlineSite &Site); |
| 279 | |
Eugene Zelenko | fb69e66 | 2017-06-06 22:22:41 +0000 | [diff] [blame] | 280 | using InlinedVariable = DbgValueHistoryMap::InlinedVariable; |
Reid Kleckner | 876330d | 2016-02-12 21:48:30 +0000 | [diff] [blame] | 281 | |
| 282 | void collectVariableInfo(const DISubprogram *SP); |
| 283 | |
Matthias Braun | ef331ef | 2016-11-30 23:48:50 +0000 | [diff] [blame] | 284 | void collectVariableInfoFromMFTable(DenseSet<InlinedVariable> &Processed); |
Reid Kleckner | 876330d | 2016-02-12 21:48:30 +0000 | [diff] [blame] | 285 | |
Reid Kleckner | 5a791ee | 2018-03-15 21:24:04 +0000 | [diff] [blame] | 286 | // Construct the lexical block tree for a routine, pruning emptpy lexical |
| 287 | // scopes, and populate it with local variables. |
| 288 | void collectLexicalBlockInfo(SmallVectorImpl<LexicalScope *> &Scopes, |
| 289 | SmallVectorImpl<LexicalBlock *> &Blocks, |
| 290 | SmallVectorImpl<LocalVariable> &Locals); |
| 291 | void collectLexicalBlockInfo(LexicalScope &Scope, |
| 292 | SmallVectorImpl<LexicalBlock *> &ParentBlocks, |
| 293 | SmallVectorImpl<LocalVariable> &ParentLocals); |
| 294 | |
Reid Kleckner | 876330d | 2016-02-12 21:48:30 +0000 | [diff] [blame] | 295 | /// Records information about a local variable in the appropriate scope. In |
| 296 | /// particular, locals from inlined code live inside the inlining site. |
Reid Kleckner | 5a791ee | 2018-03-15 21:24:04 +0000 | [diff] [blame] | 297 | void recordLocalVariable(LocalVariable &&Var, const LexicalScope *LS); |
Reid Kleckner | f9c275f | 2016-02-10 20:55:49 +0000 | [diff] [blame] | 298 | |
Reid Kleckner | 10dd55c | 2016-06-24 17:55:40 +0000 | [diff] [blame] | 299 | /// Emits local variables in the appropriate order. |
| 300 | void emitLocalVariableList(ArrayRef<LocalVariable> Locals); |
| 301 | |
| 302 | /// Emits an S_LOCAL record and its associated defined ranges. |
Reid Kleckner | f9c275f | 2016-02-10 20:55:49 +0000 | [diff] [blame] | 303 | void emitLocalVariable(const LocalVariable &Var); |
| 304 | |
Reid Kleckner | 5a791ee | 2018-03-15 21:24:04 +0000 | [diff] [blame] | 305 | /// Emits a sequence of lexical block scopes and their children. |
| 306 | void emitLexicalBlockList(ArrayRef<LexicalBlock *> Blocks, |
| 307 | const FunctionInfo& FI); |
| 308 | |
| 309 | /// Emit a lexical block scope and its children. |
| 310 | void emitLexicalBlock(const LexicalBlock &Block, const FunctionInfo& FI); |
| 311 | |
Reid Kleckner | 5acacbb | 2016-06-01 17:05:51 +0000 | [diff] [blame] | 312 | /// Translates the DIType to codeview if necessary and returns a type index |
| 313 | /// for it. |
Amjad Aboud | 76c9eb9 | 2016-06-18 10:25:07 +0000 | [diff] [blame] | 314 | codeview::TypeIndex getTypeIndex(DITypeRef TypeRef, |
| 315 | DITypeRef ClassTyRef = DITypeRef()); |
Reid Kleckner | 5acacbb | 2016-06-01 17:05:51 +0000 | [diff] [blame] | 316 | |
Bob Haarman | 223303c | 2017-08-29 20:59:25 +0000 | [diff] [blame] | 317 | codeview::TypeIndex getTypeIndexForReferenceTo(DITypeRef TypeRef); |
| 318 | |
Reid Kleckner | 0c5d874 | 2016-06-22 01:32:56 +0000 | [diff] [blame] | 319 | codeview::TypeIndex getMemberFunctionType(const DISubprogram *SP, |
| 320 | const DICompositeType *Class); |
| 321 | |
| 322 | codeview::TypeIndex getScopeIndex(const DIScope *Scope); |
| 323 | |
Reid Kleckner | 9f7f3e1 | 2016-06-24 16:24:24 +0000 | [diff] [blame] | 324 | codeview::TypeIndex getVBPTypeIndex(); |
| 325 | |
Zachary Turner | a7b0417 | 2017-08-28 18:49:04 +0000 | [diff] [blame] | 326 | void addToUDTs(const DIType *Ty); |
Hans Wennborg | 4b63a98 | 2016-06-23 22:57:25 +0000 | [diff] [blame] | 327 | |
Aaron Smith | 122d9e7 | 2018-03-06 18:20:22 +0000 | [diff] [blame] | 328 | void addUDTSrcLine(const DIType *Ty, codeview::TypeIndex TI); |
| 329 | |
Amjad Aboud | 76c9eb9 | 2016-06-18 10:25:07 +0000 | [diff] [blame] | 330 | codeview::TypeIndex lowerType(const DIType *Ty, const DIType *ClassTy); |
David Majnemer | d065e23 | 2016-06-02 06:21:37 +0000 | [diff] [blame] | 331 | codeview::TypeIndex lowerTypeAlias(const DIDerivedType *Ty); |
Adrian McCarthy | f3c3c13 | 2016-06-08 18:22:59 +0000 | [diff] [blame] | 332 | codeview::TypeIndex lowerTypeArray(const DICompositeType *Ty); |
Reid Kleckner | 5acacbb | 2016-06-01 17:05:51 +0000 | [diff] [blame] | 333 | codeview::TypeIndex lowerTypeBasic(const DIBasicType *Ty); |
Reid Kleckner | 3acdc67 | 2018-02-27 22:08:15 +0000 | [diff] [blame] | 334 | codeview::TypeIndex lowerTypePointer( |
| 335 | const DIDerivedType *Ty, |
| 336 | codeview::PointerOptions PO = codeview::PointerOptions::None); |
| 337 | codeview::TypeIndex lowerTypeMemberPointer( |
| 338 | const DIDerivedType *Ty, |
| 339 | codeview::PointerOptions PO = codeview::PointerOptions::None); |
Reid Kleckner | 5acacbb | 2016-06-01 17:05:51 +0000 | [diff] [blame] | 340 | codeview::TypeIndex lowerTypeModifier(const DIDerivedType *Ty); |
David Majnemer | 75c3ebf | 2016-06-02 17:13:53 +0000 | [diff] [blame] | 341 | codeview::TypeIndex lowerTypeFunction(const DISubroutineType *Ty); |
Reid Kleckner | 9dac473 | 2016-08-31 15:59:30 +0000 | [diff] [blame] | 342 | codeview::TypeIndex lowerTypeVFTableShape(const DIDerivedType *Ty); |
Amjad Aboud | 76c9eb9 | 2016-06-18 10:25:07 +0000 | [diff] [blame] | 343 | codeview::TypeIndex lowerTypeMemberFunction(const DISubroutineType *Ty, |
Reid Kleckner | 0c5d874 | 2016-06-22 01:32:56 +0000 | [diff] [blame] | 344 | const DIType *ClassTy, |
Adrian McCarthy | d91bf39 | 2017-09-13 20:53:55 +0000 | [diff] [blame] | 345 | int ThisAdjustment, |
| 346 | bool IsStaticMethod); |
David Majnemer | 979cb88 | 2016-06-16 21:32:16 +0000 | [diff] [blame] | 347 | codeview::TypeIndex lowerTypeEnum(const DICompositeType *Ty); |
Reid Kleckner | a8d5740 | 2016-06-03 15:58:20 +0000 | [diff] [blame] | 348 | codeview::TypeIndex lowerTypeClass(const DICompositeType *Ty); |
| 349 | codeview::TypeIndex lowerTypeUnion(const DICompositeType *Ty); |
| 350 | |
| 351 | /// Symbol records should point to complete types, but type records should |
| 352 | /// always point to incomplete types to avoid cycles in the type graph. Only |
| 353 | /// use this entry point when generating symbol records. The complete and |
| 354 | /// incomplete type indices only differ for record types. All other types use |
| 355 | /// the same index. |
| 356 | codeview::TypeIndex getCompleteTypeIndex(DITypeRef TypeRef); |
| 357 | |
| 358 | codeview::TypeIndex lowerCompleteTypeClass(const DICompositeType *Ty); |
| 359 | codeview::TypeIndex lowerCompleteTypeUnion(const DICompositeType *Ty); |
| 360 | |
Reid Kleckner | 643dd83 | 2016-06-22 17:15:28 +0000 | [diff] [blame] | 361 | struct TypeLoweringScope; |
| 362 | |
| 363 | void emitDeferredCompleteTypes(); |
| 364 | |
Amjad Aboud | 76c9eb9 | 2016-06-18 10:25:07 +0000 | [diff] [blame] | 365 | void collectMemberInfo(ClassInfo &Info, const DIDerivedType *DDTy); |
Reid Kleckner | 1ab7eac | 2016-06-22 16:06:42 +0000 | [diff] [blame] | 366 | ClassInfo collectClassInfo(const DICompositeType *Ty); |
Amjad Aboud | 76c9eb9 | 2016-06-18 10:25:07 +0000 | [diff] [blame] | 367 | |
Reid Kleckner | a8d5740 | 2016-06-03 15:58:20 +0000 | [diff] [blame] | 368 | /// Common record member lowering functionality for record types, which are |
| 369 | /// structs, classes, and unions. Returns the field list index and the member |
| 370 | /// count. |
Adrian McCarthy | 820ca54 | 2016-07-06 19:49:51 +0000 | [diff] [blame] | 371 | std::tuple<codeview::TypeIndex, codeview::TypeIndex, unsigned, bool> |
Reid Kleckner | a8d5740 | 2016-06-03 15:58:20 +0000 | [diff] [blame] | 372 | lowerRecordFieldList(const DICompositeType *Ty); |
| 373 | |
Amjad Aboud | 76c9eb9 | 2016-06-18 10:25:07 +0000 | [diff] [blame] | 374 | /// Inserts {{Node, ClassTy}, TI} into TypeIndices and checks for duplicates. |
Reid Kleckner | 0c5d874 | 2016-06-22 01:32:56 +0000 | [diff] [blame] | 375 | codeview::TypeIndex recordTypeIndexForDINode(const DINode *Node, |
| 376 | codeview::TypeIndex TI, |
| 377 | const DIType *ClassTy = nullptr); |
Amjad Aboud | 76c9eb9 | 2016-06-18 10:25:07 +0000 | [diff] [blame] | 378 | |
| 379 | unsigned getPointerSizeInBytes(); |
Reid Kleckner | 5acacbb | 2016-06-01 17:05:51 +0000 | [diff] [blame] | 380 | |
David Blaikie | b2fbb4b | 2017-02-16 18:48:33 +0000 | [diff] [blame] | 381 | protected: |
Adrian Prantl | 5f8f34e4 | 2018-05-01 15:54:18 +0000 | [diff] [blame^] | 382 | /// Gather pre-function debug information. |
David Blaikie | b2fbb4b | 2017-02-16 18:48:33 +0000 | [diff] [blame] | 383 | void beginFunctionImpl(const MachineFunction *MF) override; |
| 384 | |
Adrian Prantl | 5f8f34e4 | 2018-05-01 15:54:18 +0000 | [diff] [blame^] | 385 | /// Gather post-function debug information. |
David Blaikie | b2fbb4b | 2017-02-16 18:48:33 +0000 | [diff] [blame] | 386 | void endFunctionImpl(const MachineFunction *) override; |
| 387 | |
Timur Iskhodzhanov | f166f6c | 2014-01-30 01:39:17 +0000 | [diff] [blame] | 388 | public: |
Reid Kleckner | 70f5bc9 | 2016-01-14 19:25:04 +0000 | [diff] [blame] | 389 | CodeViewDebug(AsmPrinter *Asm); |
Timur Iskhodzhanov | f166f6c | 2014-01-30 01:39:17 +0000 | [diff] [blame] | 390 | |
Eugene Zelenko | fb69e66 | 2017-06-06 22:22:41 +0000 | [diff] [blame] | 391 | void setSymbolSize(const MCSymbol *, uint64_t) override {} |
Timur Iskhodzhanov | f166f6c | 2014-01-30 01:39:17 +0000 | [diff] [blame] | 392 | |
Adrian Prantl | 5f8f34e4 | 2018-05-01 15:54:18 +0000 | [diff] [blame^] | 393 | /// Emit the COFF section that holds the line table information. |
Craig Topper | 7b883b3 | 2014-03-08 06:31:39 +0000 | [diff] [blame] | 394 | void endModule() override; |
Timur Iskhodzhanov | f166f6c | 2014-01-30 01:39:17 +0000 | [diff] [blame] | 395 | |
Adrian Prantl | 5f8f34e4 | 2018-05-01 15:54:18 +0000 | [diff] [blame^] | 396 | /// Process beginning of an instruction. |
Craig Topper | 7b883b3 | 2014-03-08 06:31:39 +0000 | [diff] [blame] | 397 | void beginInstruction(const MachineInstr *MI) override; |
Timur Iskhodzhanov | f166f6c | 2014-01-30 01:39:17 +0000 | [diff] [blame] | 398 | }; |
Timur Iskhodzhanov | f166f6c | 2014-01-30 01:39:17 +0000 | [diff] [blame] | 399 | |
Eugene Zelenko | fb69e66 | 2017-06-06 22:22:41 +0000 | [diff] [blame] | 400 | } // end namespace llvm |
| 401 | |
| 402 | #endif // LLVM_LIB_CODEGEN_ASMPRINTER_CODEVIEWDEBUG_H |