170f5bc99SReid Kleckner //===-- llvm/lib/CodeGen/AsmPrinter/CodeViewDebug.h ----*- C++ -*--===// 270f5bc99SReid Kleckner // 370f5bc99SReid Kleckner // The LLVM Compiler Infrastructure 470f5bc99SReid Kleckner // 570f5bc99SReid Kleckner // This file is distributed under the University of Illinois Open Source 670f5bc99SReid Kleckner // License. See LICENSE.TXT for details. 770f5bc99SReid Kleckner // 870f5bc99SReid Kleckner //===----------------------------------------------------------------------===// 970f5bc99SReid Kleckner // 1070f5bc99SReid Kleckner // This file contains support for writing Microsoft CodeView debug info. 1170f5bc99SReid Kleckner // 1270f5bc99SReid Kleckner //===----------------------------------------------------------------------===// 1370f5bc99SReid Kleckner 1470f5bc99SReid Kleckner #ifndef LLVM_LIB_CODEGEN_ASMPRINTER_CODEVIEWDEBUG_H 1570f5bc99SReid Kleckner #define LLVM_LIB_CODEGEN_ASMPRINTER_CODEVIEWDEBUG_H 1670f5bc99SReid Kleckner 17f9c275feSReid Kleckner #include "DebugHandlerBase.h" 1870f5bc99SReid Kleckner #include "llvm/ADT/DenseMap.h" 1970f5bc99SReid Kleckner #include "llvm/ADT/StringMap.h" 2070f5bc99SReid Kleckner #include "llvm/CodeGen/AsmPrinter.h" 2170f5bc99SReid Kleckner #include "llvm/CodeGen/MachineFunction.h" 2270f5bc99SReid Kleckner #include "llvm/CodeGen/MachineModuleInfo.h" 23f3b9ba49SReid Kleckner #include "llvm/DebugInfo/CodeView/TypeIndex.h" 244efa0a42SZachary Turner #include "llvm/DebugInfo/CodeView/TypeTableBuilder.h" 2570f5bc99SReid Kleckner #include "llvm/IR/DebugInfo.h" 2670f5bc99SReid Kleckner #include "llvm/IR/DebugLoc.h" 2770f5bc99SReid Kleckner #include "llvm/MC/MCStreamer.h" 2870f5bc99SReid Kleckner #include "llvm/Target/TargetLoweringObjectFile.h" 2970f5bc99SReid Kleckner 3070f5bc99SReid Kleckner namespace llvm { 31f9c275feSReid Kleckner 32b550cb17SMehdi Amini class StringRef; 33f9c275feSReid Kleckner class LexicalScope; 3476c9eb99SAmjad Aboud struct ClassInfo; 35f9c275feSReid Kleckner 3670f5bc99SReid Kleckner /// \brief Collects and handles line tables information in a CodeView format. 37f9c275feSReid Kleckner class LLVM_LIBRARY_VISIBILITY CodeViewDebug : public DebugHandlerBase { 38dac21b43SReid Kleckner MCStreamer &OS; 39c6d54da8SZachary Turner llvm::BumpPtrAllocator Allocator; 404efa0a42SZachary Turner codeview::TypeTableBuilder TypeTable; 41f9c275feSReid Kleckner 42876330d5SReid Kleckner /// Represents the most general definition range. 43876330d5SReid Kleckner struct LocalVarDefRange { 44876330d5SReid Kleckner /// Indicates that variable data is stored in memory relative to the 45876330d5SReid Kleckner /// specified register. 46876330d5SReid Kleckner int InMemory : 1; 47876330d5SReid Kleckner 48876330d5SReid Kleckner /// Offset of variable data in memory. 49876330d5SReid Kleckner int DataOffset : 31; 50876330d5SReid Kleckner 512b3e6428SReid Kleckner /// Non-zero if this is a piece of an aggregate. 522b3e6428SReid Kleckner uint16_t IsSubfield : 1; 532b3e6428SReid Kleckner 542b3e6428SReid Kleckner /// Offset into aggregate. 552b3e6428SReid Kleckner uint16_t StructOffset : 15; 56876330d5SReid Kleckner 57876330d5SReid Kleckner /// Register containing the data or the register base of the memory 58876330d5SReid Kleckner /// location containing the data. 59876330d5SReid Kleckner uint16_t CVRegister; 60876330d5SReid Kleckner 61876330d5SReid Kleckner /// Compares all location fields. This includes all fields except the label 62876330d5SReid Kleckner /// ranges. 63876330d5SReid Kleckner bool isDifferentLocation(LocalVarDefRange &O) { 64876330d5SReid Kleckner return InMemory != O.InMemory || DataOffset != O.DataOffset || 652b3e6428SReid Kleckner IsSubfield != O.IsSubfield || StructOffset != O.StructOffset || 662b3e6428SReid Kleckner CVRegister != O.CVRegister; 67876330d5SReid Kleckner } 68876330d5SReid Kleckner 69876330d5SReid Kleckner SmallVector<std::pair<const MCSymbol *, const MCSymbol *>, 1> Ranges; 70876330d5SReid Kleckner }; 71876330d5SReid Kleckner 72876330d5SReid Kleckner static LocalVarDefRange createDefRangeMem(uint16_t CVRegister, int Offset); 732b3e6428SReid Kleckner static LocalVarDefRange createDefRangeGeneral(uint16_t CVRegister, 742b3e6428SReid Kleckner bool InMemory, int Offset, 752b3e6428SReid Kleckner bool IsSubfield, 762b3e6428SReid Kleckner uint16_t StructOffset); 77876330d5SReid Kleckner 78f9c275feSReid Kleckner /// Similar to DbgVariable in DwarfDebug, but not dwarf-specific. 79f9c275feSReid Kleckner struct LocalVariable { 80f9c275feSReid Kleckner const DILocalVariable *DIVar = nullptr; 81876330d5SReid Kleckner SmallVector<LocalVarDefRange, 1> DefRanges; 82f9c275feSReid Kleckner }; 8370f5bc99SReid Kleckner 84f3b9ba49SReid Kleckner struct InlineSite { 85f9c275feSReid Kleckner SmallVector<LocalVariable, 1> InlinedLocals; 86f9c275feSReid Kleckner SmallVector<const DILocation *, 1> ChildSites; 87f3b9ba49SReid Kleckner const DISubprogram *Inlinee = nullptr; 88c29b4f07SReid Kleckner 89c29b4f07SReid Kleckner /// The ID of the inline site or function used with .cv_loc. Not a type 90c29b4f07SReid Kleckner /// index. 91f3b9ba49SReid Kleckner unsigned SiteFuncId = 0; 92f3b9ba49SReid Kleckner }; 93f3b9ba49SReid Kleckner 9470f5bc99SReid Kleckner // For each function, store a vector of labels to its instructions, as well as 9570f5bc99SReid Kleckner // to the end of the function. 9670f5bc99SReid Kleckner struct FunctionInfo { 97f3b9ba49SReid Kleckner /// Map from inlined call site to inlined instructions and child inlined 98f3b9ba49SReid Kleckner /// call sites. Listed in program order. 99f9c275feSReid Kleckner std::unordered_map<const DILocation *, InlineSite> InlineSites; 100f9c275feSReid Kleckner 101f9c275feSReid Kleckner /// Ordered list of top-level inlined call sites. 102f9c275feSReid Kleckner SmallVector<const DILocation *, 1> ChildSites; 103f9c275feSReid Kleckner 104f9c275feSReid Kleckner SmallVector<LocalVariable, 1> Locals; 105f3b9ba49SReid Kleckner 1069533af4fSReid Kleckner DebugLoc LastLoc; 1071fcd610cSReid Kleckner const MCSymbol *Begin = nullptr; 1081fcd610cSReid Kleckner const MCSymbol *End = nullptr; 1092214ed89SReid Kleckner unsigned FuncId = 0; 110f3b9ba49SReid Kleckner unsigned LastFileId = 0; 1112214ed89SReid Kleckner bool HaveLineInfo = false; 1129533af4fSReid Kleckner }; 1139533af4fSReid Kleckner FunctionInfo *CurFn; 11470f5bc99SReid Kleckner 1155d122f87SReid Kleckner /// The set of comdat .debug$S sections that we've seen so far. Each section 1165d122f87SReid Kleckner /// must start with a magic version number that must only be emitted once. 1175d122f87SReid Kleckner /// This set tracks which sections we've already opened. 1185d122f87SReid Kleckner DenseSet<MCSectionCOFF *> ComdatDebugSections; 1195d122f87SReid Kleckner 1205d122f87SReid Kleckner /// Switch to the appropriate .debug$S section for GVSym. If GVSym, the symbol 1215d122f87SReid Kleckner /// of an emitted global value, is in a comdat COFF section, this will switch 1225d122f87SReid Kleckner /// to a new .debug$S section in that comdat. This method ensures that the 1235d122f87SReid Kleckner /// section starts with the magic version number on first use. If GVSym is 1245d122f87SReid Kleckner /// null, uses the main .debug$S section. 1255d122f87SReid Kleckner void switchToDebugSectionForSymbol(const MCSymbol *GVSym); 1265d122f87SReid Kleckner 127fbd7787dSReid Kleckner /// The next available function index for use with our .cv_* directives. Not 128fbd7787dSReid Kleckner /// to be confused with type indices for LF_FUNC_ID records. 1292214ed89SReid Kleckner unsigned NextFuncId = 0; 13070f5bc99SReid Kleckner 131876330d5SReid Kleckner InlineSite &getInlineSite(const DILocation *InlinedAt, 132876330d5SReid Kleckner const DISubprogram *Inlinee); 133f3b9ba49SReid Kleckner 13475c3ebfaSDavid Majnemer codeview::TypeIndex getFuncIdForSubprogram(const DISubprogram *SP); 1352280f932SReid Kleckner 1361fcd610cSReid Kleckner static void collectInlineSiteChildren(SmallVectorImpl<unsigned> &Children, 1371fcd610cSReid Kleckner const FunctionInfo &FI, 1381fcd610cSReid Kleckner const InlineSite &Site); 1391fcd610cSReid Kleckner 1402214ed89SReid Kleckner /// Remember some debug info about each function. Keep it in a stable order to 1412214ed89SReid Kleckner /// emit at the end of the TU. 1422214ed89SReid Kleckner MapVector<const Function *, FunctionInfo> FnDebugInfo; 14370f5bc99SReid Kleckner 1442214ed89SReid Kleckner /// Map from DIFile to .cv_file id. 1452214ed89SReid Kleckner DenseMap<const DIFile *, unsigned> FileIdMap; 14670f5bc99SReid Kleckner 147fbd7787dSReid Kleckner /// All inlined subprograms in the order they should be emitted. 1482280f932SReid Kleckner SmallSetVector<const DISubprogram *, 4> InlinedSubprograms; 149f3b9ba49SReid Kleckner 15076c9eb99SAmjad Aboud /// Map from a pair of DI metadata nodes and its DI type (or scope) that can 15176c9eb99SAmjad Aboud /// be nullptr, to CodeView type indices. Primarily indexed by 15276c9eb99SAmjad Aboud /// {DIType*, DIType*} and {DISubprogram*, DIType*}. 15376c9eb99SAmjad Aboud /// 15476c9eb99SAmjad Aboud /// The second entry in the key is needed for methods as DISubroutineType 15576c9eb99SAmjad Aboud /// representing static method type are shared with non-method function type. 15676c9eb99SAmjad Aboud DenseMap<std::pair<const DINode *, const DIType *>, codeview::TypeIndex> 15776c9eb99SAmjad Aboud TypeIndices; 158f3b9ba49SReid Kleckner 159a8d57407SReid Kleckner /// Map from DICompositeType* to complete type index. Non-record types are 160a8d57407SReid Kleckner /// always looked up in the normal TypeIndices map. 161a8d57407SReid Kleckner DenseMap<const DICompositeType *, codeview::TypeIndex> CompleteTypeIndices; 162a8d57407SReid Kleckner 163643dd836SReid Kleckner /// Complete record types to emit after all active type lowerings are 164643dd836SReid Kleckner /// finished. 165643dd836SReid Kleckner SmallVector<const DICompositeType *, 4> DeferredCompleteTypes; 166643dd836SReid Kleckner 167643dd836SReid Kleckner /// Number of type lowering frames active on the stack. 168643dd836SReid Kleckner unsigned TypeEmissionLevel = 0; 169643dd836SReid Kleckner 1709f7f3e1eSReid Kleckner codeview::TypeIndex VBPType; 1719f7f3e1eSReid Kleckner 1723128b10cSDavid Majnemer const DISubprogram *CurrentSubprogram = nullptr; 1733128b10cSDavid Majnemer 1743128b10cSDavid Majnemer // The UDTs we have seen while processing types; each entry is a pair of type 1753128b10cSDavid Majnemer // index and type name. 1763128b10cSDavid Majnemer std::vector<std::pair<std::string, codeview::TypeIndex>> LocalUDTs, 1773128b10cSDavid Majnemer GlobalUDTs; 1783128b10cSDavid Majnemer 1799533af4fSReid Kleckner typedef std::map<const DIFile *, std::string> FileToFilepathMapTy; 1809533af4fSReid Kleckner FileToFilepathMapTy FileToFilepathMap; 1819533af4fSReid Kleckner StringRef getFullFilepath(const DIFile *S); 18270f5bc99SReid Kleckner 1832214ed89SReid Kleckner unsigned maybeRecordFile(const DIFile *F); 1842214ed89SReid Kleckner 185bdc4956bSBenjamin Kramer void maybeRecordLocation(const DebugLoc &DL, const MachineFunction *MF); 18670f5bc99SReid Kleckner 18776c9eb99SAmjad Aboud void clear(); 1883128b10cSDavid Majnemer 1893128b10cSDavid Majnemer void setCurrentSubprogram(const DISubprogram *SP) { 1903128b10cSDavid Majnemer CurrentSubprogram = SP; 1913128b10cSDavid Majnemer LocalUDTs.clear(); 19270f5bc99SReid Kleckner } 19370f5bc99SReid Kleckner 1945d122f87SReid Kleckner /// Emit the magic version number at the start of a CodeView type or symbol 1955d122f87SReid Kleckner /// section. Appears at the front of every .debug$S or .debug$T section. 1965d122f87SReid Kleckner void emitCodeViewMagicVersion(); 1975d122f87SReid Kleckner 198f3b9ba49SReid Kleckner void emitTypeInformation(); 199f3b9ba49SReid Kleckner 200c64acfd4SAdrian McCarthy void emitCompilerInformation(); 201c64acfd4SAdrian McCarthy 2025d122f87SReid Kleckner void emitInlineeLinesSubsection(); 2031fcd610cSReid Kleckner 2042214ed89SReid Kleckner void emitDebugInfoForFunction(const Function *GV, FunctionInfo &FI); 20570f5bc99SReid Kleckner 2066f3406dfSReid Kleckner void emitDebugInfoForGlobals(); 2076f3406dfSReid Kleckner 208b510b458SHans Wennborg void emitDebugInfoForRetainedTypes(); 209b510b458SHans Wennborg 2103128b10cSDavid Majnemer void emitDebugInfoForUDTs( 2113128b10cSDavid Majnemer ArrayRef<std::pair<std::string, codeview::TypeIndex>> UDTs); 2123128b10cSDavid Majnemer 213d4135bbcSPeter Collingbourne void emitDebugInfoForGlobal(const DIGlobalVariable *DIGV, 214d4135bbcSPeter Collingbourne const GlobalVariable *GV, MCSymbol *GVSym); 2156f3406dfSReid Kleckner 2166f3406dfSReid Kleckner /// Opens a subsection of the given kind in a .debug$S codeview section. 2176f3406dfSReid Kleckner /// Returns an end label for use with endCVSubsection when the subsection is 2186f3406dfSReid Kleckner /// finished. 2196f3406dfSReid Kleckner MCSymbol *beginCVSubsection(codeview::ModuleSubstreamKind Kind); 2206f3406dfSReid Kleckner 2216f3406dfSReid Kleckner void endCVSubsection(MCSymbol *EndLabel); 2226f3406dfSReid Kleckner 223f3b9ba49SReid Kleckner void emitInlinedCallSite(const FunctionInfo &FI, const DILocation *InlinedAt, 224f3b9ba49SReid Kleckner const InlineSite &Site); 225f3b9ba49SReid Kleckner 226876330d5SReid Kleckner typedef DbgValueHistoryMap::InlinedVariable InlinedVariable; 227876330d5SReid Kleckner 228876330d5SReid Kleckner void collectVariableInfo(const DISubprogram *SP); 229876330d5SReid Kleckner 230ef331effSMatthias Braun void collectVariableInfoFromMFTable(DenseSet<InlinedVariable> &Processed); 231876330d5SReid Kleckner 232876330d5SReid Kleckner /// Records information about a local variable in the appropriate scope. In 233876330d5SReid Kleckner /// particular, locals from inlined code live inside the inlining site. 234876330d5SReid Kleckner void recordLocalVariable(LocalVariable &&Var, const DILocation *Loc); 235f9c275feSReid Kleckner 23610dd55c5SReid Kleckner /// Emits local variables in the appropriate order. 23710dd55c5SReid Kleckner void emitLocalVariableList(ArrayRef<LocalVariable> Locals); 23810dd55c5SReid Kleckner 23910dd55c5SReid Kleckner /// Emits an S_LOCAL record and its associated defined ranges. 240f9c275feSReid Kleckner void emitLocalVariable(const LocalVariable &Var); 241f9c275feSReid Kleckner 2425acacbb0SReid Kleckner /// Translates the DIType to codeview if necessary and returns a type index 2435acacbb0SReid Kleckner /// for it. 24476c9eb99SAmjad Aboud codeview::TypeIndex getTypeIndex(DITypeRef TypeRef, 24576c9eb99SAmjad Aboud DITypeRef ClassTyRef = DITypeRef()); 2465acacbb0SReid Kleckner 2470c5d874bSReid Kleckner codeview::TypeIndex getMemberFunctionType(const DISubprogram *SP, 2480c5d874bSReid Kleckner const DICompositeType *Class); 2490c5d874bSReid Kleckner 2500c5d874bSReid Kleckner codeview::TypeIndex getScopeIndex(const DIScope *Scope); 2510c5d874bSReid Kleckner 2529f7f3e1eSReid Kleckner codeview::TypeIndex getVBPTypeIndex(); 2539f7f3e1eSReid Kleckner 2544b63a98dSHans Wennborg void addToUDTs(const DIType *Ty, codeview::TypeIndex TI); 2554b63a98dSHans Wennborg 25676c9eb99SAmjad Aboud codeview::TypeIndex lowerType(const DIType *Ty, const DIType *ClassTy); 257d065e23dSDavid Majnemer codeview::TypeIndex lowerTypeAlias(const DIDerivedType *Ty); 258f3c3c132SAdrian McCarthy codeview::TypeIndex lowerTypeArray(const DICompositeType *Ty); 2595acacbb0SReid Kleckner codeview::TypeIndex lowerTypeBasic(const DIBasicType *Ty); 2605acacbb0SReid Kleckner codeview::TypeIndex lowerTypePointer(const DIDerivedType *Ty); 2615acacbb0SReid Kleckner codeview::TypeIndex lowerTypeMemberPointer(const DIDerivedType *Ty); 2625acacbb0SReid Kleckner codeview::TypeIndex lowerTypeModifier(const DIDerivedType *Ty); 26375c3ebfaSDavid Majnemer codeview::TypeIndex lowerTypeFunction(const DISubroutineType *Ty); 2649dac4731SReid Kleckner codeview::TypeIndex lowerTypeVFTableShape(const DIDerivedType *Ty); 26576c9eb99SAmjad Aboud codeview::TypeIndex lowerTypeMemberFunction(const DISubroutineType *Ty, 2660c5d874bSReid Kleckner const DIType *ClassTy, 2670c5d874bSReid Kleckner int ThisAdjustment); 268979cb888SDavid Majnemer codeview::TypeIndex lowerTypeEnum(const DICompositeType *Ty); 269a8d57407SReid Kleckner codeview::TypeIndex lowerTypeClass(const DICompositeType *Ty); 270a8d57407SReid Kleckner codeview::TypeIndex lowerTypeUnion(const DICompositeType *Ty); 271a8d57407SReid Kleckner 272a8d57407SReid Kleckner /// Symbol records should point to complete types, but type records should 273a8d57407SReid Kleckner /// always point to incomplete types to avoid cycles in the type graph. Only 274a8d57407SReid Kleckner /// use this entry point when generating symbol records. The complete and 275a8d57407SReid Kleckner /// incomplete type indices only differ for record types. All other types use 276a8d57407SReid Kleckner /// the same index. 277a8d57407SReid Kleckner codeview::TypeIndex getCompleteTypeIndex(DITypeRef TypeRef); 278a8d57407SReid Kleckner 279a8d57407SReid Kleckner codeview::TypeIndex lowerCompleteTypeClass(const DICompositeType *Ty); 280a8d57407SReid Kleckner codeview::TypeIndex lowerCompleteTypeUnion(const DICompositeType *Ty); 281a8d57407SReid Kleckner 282643dd836SReid Kleckner struct TypeLoweringScope; 283643dd836SReid Kleckner 284643dd836SReid Kleckner void emitDeferredCompleteTypes(); 285643dd836SReid Kleckner 28676c9eb99SAmjad Aboud void collectMemberInfo(ClassInfo &Info, const DIDerivedType *DDTy); 2871ab7eac8SReid Kleckner ClassInfo collectClassInfo(const DICompositeType *Ty); 28876c9eb99SAmjad Aboud 289a8d57407SReid Kleckner /// Common record member lowering functionality for record types, which are 290a8d57407SReid Kleckner /// structs, classes, and unions. Returns the field list index and the member 291a8d57407SReid Kleckner /// count. 292820ca540SAdrian McCarthy std::tuple<codeview::TypeIndex, codeview::TypeIndex, unsigned, bool> 293a8d57407SReid Kleckner lowerRecordFieldList(const DICompositeType *Ty); 294a8d57407SReid Kleckner 29576c9eb99SAmjad Aboud /// Inserts {{Node, ClassTy}, TI} into TypeIndices and checks for duplicates. 2960c5d874bSReid Kleckner codeview::TypeIndex recordTypeIndexForDINode(const DINode *Node, 2970c5d874bSReid Kleckner codeview::TypeIndex TI, 29876c9eb99SAmjad Aboud const DIType *ClassTy = nullptr); 29976c9eb99SAmjad Aboud 30076c9eb99SAmjad Aboud unsigned getPointerSizeInBytes(); 3015acacbb0SReid Kleckner 302*b2fbb4b2SDavid Blaikie protected: 303*b2fbb4b2SDavid Blaikie /// \brief Gather pre-function debug information. 304*b2fbb4b2SDavid Blaikie void beginFunctionImpl(const MachineFunction *MF) override; 305*b2fbb4b2SDavid Blaikie 306*b2fbb4b2SDavid Blaikie /// \brief Gather post-function debug information. 307*b2fbb4b2SDavid Blaikie void endFunctionImpl(const MachineFunction *) override; 308*b2fbb4b2SDavid Blaikie 30970f5bc99SReid Kleckner public: 31070f5bc99SReid Kleckner CodeViewDebug(AsmPrinter *Asm); 31170f5bc99SReid Kleckner 31270f5bc99SReid Kleckner void setSymbolSize(const llvm::MCSymbol *, uint64_t) override {} 31370f5bc99SReid Kleckner 31470f5bc99SReid Kleckner /// \brief Emit the COFF section that holds the line table information. 31570f5bc99SReid Kleckner void endModule() override; 31670f5bc99SReid Kleckner 31770f5bc99SReid Kleckner /// \brief Process beginning of an instruction. 31870f5bc99SReid Kleckner void beginInstruction(const MachineInstr *MI) override; 31970f5bc99SReid Kleckner }; 32070f5bc99SReid Kleckner } // End of namespace llvm 32170f5bc99SReid Kleckner 32270f5bc99SReid Kleckner #endif 323