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" 232280f932SReid Kleckner #include "llvm/DebugInfo/CodeView/MemoryTypeTableBuilder.h" 24f3b9ba49SReid Kleckner #include "llvm/DebugInfo/CodeView/TypeIndex.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; 402280f932SReid Kleckner codeview::MemoryTypeTableBuilder 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 51876330d5SReid Kleckner /// Offset of the data into the user level struct. If zero, no splitting 52876330d5SReid Kleckner /// occurred. 53876330d5SReid Kleckner uint16_t StructOffset; 54876330d5SReid Kleckner 55876330d5SReid Kleckner /// Register containing the data or the register base of the memory 56876330d5SReid Kleckner /// location containing the data. 57876330d5SReid Kleckner uint16_t CVRegister; 58876330d5SReid Kleckner 59876330d5SReid Kleckner /// Compares all location fields. This includes all fields except the label 60876330d5SReid Kleckner /// ranges. 61876330d5SReid Kleckner bool isDifferentLocation(LocalVarDefRange &O) { 62876330d5SReid Kleckner return InMemory != O.InMemory || DataOffset != O.DataOffset || 63876330d5SReid Kleckner StructOffset != O.StructOffset || CVRegister != O.CVRegister; 64876330d5SReid Kleckner } 65876330d5SReid Kleckner 66876330d5SReid Kleckner SmallVector<std::pair<const MCSymbol *, const MCSymbol *>, 1> Ranges; 67876330d5SReid Kleckner }; 68876330d5SReid Kleckner 69876330d5SReid Kleckner static LocalVarDefRange createDefRangeMem(uint16_t CVRegister, int Offset); 70876330d5SReid Kleckner static LocalVarDefRange createDefRangeReg(uint16_t CVRegister); 71876330d5SReid Kleckner 72f9c275feSReid Kleckner /// Similar to DbgVariable in DwarfDebug, but not dwarf-specific. 73f9c275feSReid Kleckner struct LocalVariable { 74f9c275feSReid Kleckner const DILocalVariable *DIVar = nullptr; 75876330d5SReid Kleckner SmallVector<LocalVarDefRange, 1> DefRanges; 76f9c275feSReid Kleckner }; 7770f5bc99SReid Kleckner 78f3b9ba49SReid Kleckner struct InlineSite { 79f9c275feSReid Kleckner SmallVector<LocalVariable, 1> InlinedLocals; 80f9c275feSReid Kleckner SmallVector<const DILocation *, 1> ChildSites; 81f3b9ba49SReid Kleckner const DISubprogram *Inlinee = nullptr; 82c29b4f07SReid Kleckner 83c29b4f07SReid Kleckner /// The ID of the inline site or function used with .cv_loc. Not a type 84c29b4f07SReid Kleckner /// index. 85f3b9ba49SReid Kleckner unsigned SiteFuncId = 0; 86f3b9ba49SReid Kleckner }; 87f3b9ba49SReid Kleckner 8870f5bc99SReid Kleckner // For each function, store a vector of labels to its instructions, as well as 8970f5bc99SReid Kleckner // to the end of the function. 9070f5bc99SReid Kleckner struct FunctionInfo { 91f3b9ba49SReid Kleckner /// Map from inlined call site to inlined instructions and child inlined 92f3b9ba49SReid Kleckner /// call sites. Listed in program order. 93f9c275feSReid Kleckner std::unordered_map<const DILocation *, InlineSite> InlineSites; 94f9c275feSReid Kleckner 95f9c275feSReid Kleckner /// Ordered list of top-level inlined call sites. 96f9c275feSReid Kleckner SmallVector<const DILocation *, 1> ChildSites; 97f9c275feSReid Kleckner 98f9c275feSReid Kleckner SmallVector<LocalVariable, 1> Locals; 99f3b9ba49SReid Kleckner 1009533af4fSReid Kleckner DebugLoc LastLoc; 1011fcd610cSReid Kleckner const MCSymbol *Begin = nullptr; 1021fcd610cSReid Kleckner const MCSymbol *End = nullptr; 1032214ed89SReid Kleckner unsigned FuncId = 0; 104f3b9ba49SReid Kleckner unsigned LastFileId = 0; 1052214ed89SReid Kleckner bool HaveLineInfo = false; 1069533af4fSReid Kleckner }; 1079533af4fSReid Kleckner FunctionInfo *CurFn; 10870f5bc99SReid Kleckner 1095d122f87SReid Kleckner /// The set of comdat .debug$S sections that we've seen so far. Each section 1105d122f87SReid Kleckner /// must start with a magic version number that must only be emitted once. 1115d122f87SReid Kleckner /// This set tracks which sections we've already opened. 1125d122f87SReid Kleckner DenseSet<MCSectionCOFF *> ComdatDebugSections; 1135d122f87SReid Kleckner 1145d122f87SReid Kleckner /// Switch to the appropriate .debug$S section for GVSym. If GVSym, the symbol 1155d122f87SReid Kleckner /// of an emitted global value, is in a comdat COFF section, this will switch 1165d122f87SReid Kleckner /// to a new .debug$S section in that comdat. This method ensures that the 1175d122f87SReid Kleckner /// section starts with the magic version number on first use. If GVSym is 1185d122f87SReid Kleckner /// null, uses the main .debug$S section. 1195d122f87SReid Kleckner void switchToDebugSectionForSymbol(const MCSymbol *GVSym); 1205d122f87SReid Kleckner 121fbd7787dSReid Kleckner /// The next available function index for use with our .cv_* directives. Not 122fbd7787dSReid Kleckner /// to be confused with type indices for LF_FUNC_ID records. 1232214ed89SReid Kleckner unsigned NextFuncId = 0; 12470f5bc99SReid Kleckner 125876330d5SReid Kleckner InlineSite &getInlineSite(const DILocation *InlinedAt, 126876330d5SReid Kleckner const DISubprogram *Inlinee); 127f3b9ba49SReid Kleckner 12875c3ebfaSDavid Majnemer codeview::TypeIndex getFuncIdForSubprogram(const DISubprogram *SP); 1292280f932SReid Kleckner 1301fcd610cSReid Kleckner static void collectInlineSiteChildren(SmallVectorImpl<unsigned> &Children, 1311fcd610cSReid Kleckner const FunctionInfo &FI, 1321fcd610cSReid Kleckner const InlineSite &Site); 1331fcd610cSReid Kleckner 1342214ed89SReid Kleckner /// Remember some debug info about each function. Keep it in a stable order to 1352214ed89SReid Kleckner /// emit at the end of the TU. 1362214ed89SReid Kleckner MapVector<const Function *, FunctionInfo> FnDebugInfo; 13770f5bc99SReid Kleckner 1382214ed89SReid Kleckner /// Map from DIFile to .cv_file id. 1392214ed89SReid Kleckner DenseMap<const DIFile *, unsigned> FileIdMap; 14070f5bc99SReid Kleckner 141fbd7787dSReid Kleckner /// All inlined subprograms in the order they should be emitted. 1422280f932SReid Kleckner SmallSetVector<const DISubprogram *, 4> InlinedSubprograms; 143f3b9ba49SReid Kleckner 14476c9eb99SAmjad Aboud /// Map from a pair of DI metadata nodes and its DI type (or scope) that can 14576c9eb99SAmjad Aboud /// be nullptr, to CodeView type indices. Primarily indexed by 14676c9eb99SAmjad Aboud /// {DIType*, DIType*} and {DISubprogram*, DIType*}. 14776c9eb99SAmjad Aboud /// 14876c9eb99SAmjad Aboud /// The second entry in the key is needed for methods as DISubroutineType 14976c9eb99SAmjad Aboud /// representing static method type are shared with non-method function type. 15076c9eb99SAmjad Aboud DenseMap<std::pair<const DINode *, const DIType *>, codeview::TypeIndex> 15176c9eb99SAmjad Aboud TypeIndices; 152f3b9ba49SReid Kleckner 153a8d57407SReid Kleckner /// Map from DICompositeType* to complete type index. Non-record types are 154a8d57407SReid Kleckner /// always looked up in the normal TypeIndices map. 155a8d57407SReid Kleckner DenseMap<const DICompositeType *, codeview::TypeIndex> CompleteTypeIndices; 156a8d57407SReid Kleckner 157643dd836SReid Kleckner /// Complete record types to emit after all active type lowerings are 158643dd836SReid Kleckner /// finished. 159643dd836SReid Kleckner SmallVector<const DICompositeType *, 4> DeferredCompleteTypes; 160643dd836SReid Kleckner 161643dd836SReid Kleckner /// Number of type lowering frames active on the stack. 162643dd836SReid Kleckner unsigned TypeEmissionLevel = 0; 163643dd836SReid Kleckner 1649f7f3e1eSReid Kleckner codeview::TypeIndex VBPType; 1659f7f3e1eSReid Kleckner 1663128b10cSDavid Majnemer const DISubprogram *CurrentSubprogram = nullptr; 1673128b10cSDavid Majnemer 1683128b10cSDavid Majnemer // The UDTs we have seen while processing types; each entry is a pair of type 1693128b10cSDavid Majnemer // index and type name. 1703128b10cSDavid Majnemer std::vector<std::pair<std::string, codeview::TypeIndex>> LocalUDTs, 1713128b10cSDavid Majnemer GlobalUDTs; 1723128b10cSDavid Majnemer 1739533af4fSReid Kleckner typedef std::map<const DIFile *, std::string> FileToFilepathMapTy; 1749533af4fSReid Kleckner FileToFilepathMapTy FileToFilepathMap; 1759533af4fSReid Kleckner StringRef getFullFilepath(const DIFile *S); 17670f5bc99SReid Kleckner 1772214ed89SReid Kleckner unsigned maybeRecordFile(const DIFile *F); 1782214ed89SReid Kleckner 179bdc4956bSBenjamin Kramer void maybeRecordLocation(const DebugLoc &DL, const MachineFunction *MF); 18070f5bc99SReid Kleckner 18176c9eb99SAmjad Aboud void clear(); 1823128b10cSDavid Majnemer 1833128b10cSDavid Majnemer void setCurrentSubprogram(const DISubprogram *SP) { 1843128b10cSDavid Majnemer CurrentSubprogram = SP; 1853128b10cSDavid Majnemer LocalUDTs.clear(); 18670f5bc99SReid Kleckner } 18770f5bc99SReid Kleckner 1885d122f87SReid Kleckner /// Emit the magic version number at the start of a CodeView type or symbol 1895d122f87SReid Kleckner /// section. Appears at the front of every .debug$S or .debug$T section. 1905d122f87SReid Kleckner void emitCodeViewMagicVersion(); 1915d122f87SReid Kleckner 192f3b9ba49SReid Kleckner void emitTypeInformation(); 193f3b9ba49SReid Kleckner 1945d122f87SReid Kleckner void emitInlineeLinesSubsection(); 1951fcd610cSReid Kleckner 1962214ed89SReid Kleckner void emitDebugInfoForFunction(const Function *GV, FunctionInfo &FI); 19770f5bc99SReid Kleckner 1986f3406dfSReid Kleckner void emitDebugInfoForGlobals(); 1996f3406dfSReid Kleckner 200b510b458SHans Wennborg void emitDebugInfoForRetainedTypes(); 201b510b458SHans Wennborg 2023128b10cSDavid Majnemer void emitDebugInfoForUDTs( 2033128b10cSDavid Majnemer ArrayRef<std::pair<std::string, codeview::TypeIndex>> UDTs); 2043128b10cSDavid Majnemer 205*d4135bbcSPeter Collingbourne void emitDebugInfoForGlobal(const DIGlobalVariable *DIGV, 206*d4135bbcSPeter Collingbourne const GlobalVariable *GV, MCSymbol *GVSym); 2076f3406dfSReid Kleckner 2086f3406dfSReid Kleckner /// Opens a subsection of the given kind in a .debug$S codeview section. 2096f3406dfSReid Kleckner /// Returns an end label for use with endCVSubsection when the subsection is 2106f3406dfSReid Kleckner /// finished. 2116f3406dfSReid Kleckner MCSymbol *beginCVSubsection(codeview::ModuleSubstreamKind Kind); 2126f3406dfSReid Kleckner 2136f3406dfSReid Kleckner void endCVSubsection(MCSymbol *EndLabel); 2146f3406dfSReid Kleckner 215f3b9ba49SReid Kleckner void emitInlinedCallSite(const FunctionInfo &FI, const DILocation *InlinedAt, 216f3b9ba49SReid Kleckner const InlineSite &Site); 217f3b9ba49SReid Kleckner 218876330d5SReid Kleckner typedef DbgValueHistoryMap::InlinedVariable InlinedVariable; 219876330d5SReid Kleckner 220876330d5SReid Kleckner void collectVariableInfo(const DISubprogram *SP); 221876330d5SReid Kleckner 222876330d5SReid Kleckner void collectVariableInfoFromMMITable(DenseSet<InlinedVariable> &Processed); 223876330d5SReid Kleckner 224876330d5SReid Kleckner /// Records information about a local variable in the appropriate scope. In 225876330d5SReid Kleckner /// particular, locals from inlined code live inside the inlining site. 226876330d5SReid Kleckner void recordLocalVariable(LocalVariable &&Var, const DILocation *Loc); 227f9c275feSReid Kleckner 22810dd55c5SReid Kleckner /// Emits local variables in the appropriate order. 22910dd55c5SReid Kleckner void emitLocalVariableList(ArrayRef<LocalVariable> Locals); 23010dd55c5SReid Kleckner 23110dd55c5SReid Kleckner /// Emits an S_LOCAL record and its associated defined ranges. 232f9c275feSReid Kleckner void emitLocalVariable(const LocalVariable &Var); 233f9c275feSReid Kleckner 2345acacbb0SReid Kleckner /// Translates the DIType to codeview if necessary and returns a type index 2355acacbb0SReid Kleckner /// for it. 23676c9eb99SAmjad Aboud codeview::TypeIndex getTypeIndex(DITypeRef TypeRef, 23776c9eb99SAmjad Aboud DITypeRef ClassTyRef = DITypeRef()); 2385acacbb0SReid Kleckner 2390c5d874bSReid Kleckner codeview::TypeIndex getMemberFunctionType(const DISubprogram *SP, 2400c5d874bSReid Kleckner const DICompositeType *Class); 2410c5d874bSReid Kleckner 2420c5d874bSReid Kleckner codeview::TypeIndex getScopeIndex(const DIScope *Scope); 2430c5d874bSReid Kleckner 2449f7f3e1eSReid Kleckner codeview::TypeIndex getVBPTypeIndex(); 2459f7f3e1eSReid Kleckner 2464b63a98dSHans Wennborg void addToUDTs(const DIType *Ty, codeview::TypeIndex TI); 2474b63a98dSHans Wennborg 24876c9eb99SAmjad Aboud codeview::TypeIndex lowerType(const DIType *Ty, const DIType *ClassTy); 249d065e23dSDavid Majnemer codeview::TypeIndex lowerTypeAlias(const DIDerivedType *Ty); 250f3c3c132SAdrian McCarthy codeview::TypeIndex lowerTypeArray(const DICompositeType *Ty); 2515acacbb0SReid Kleckner codeview::TypeIndex lowerTypeBasic(const DIBasicType *Ty); 2525acacbb0SReid Kleckner codeview::TypeIndex lowerTypePointer(const DIDerivedType *Ty); 2535acacbb0SReid Kleckner codeview::TypeIndex lowerTypeMemberPointer(const DIDerivedType *Ty); 2545acacbb0SReid Kleckner codeview::TypeIndex lowerTypeModifier(const DIDerivedType *Ty); 25575c3ebfaSDavid Majnemer codeview::TypeIndex lowerTypeFunction(const DISubroutineType *Ty); 2569dac4731SReid Kleckner codeview::TypeIndex lowerTypeVFTableShape(const DIDerivedType *Ty); 25776c9eb99SAmjad Aboud codeview::TypeIndex lowerTypeMemberFunction(const DISubroutineType *Ty, 2580c5d874bSReid Kleckner const DIType *ClassTy, 2590c5d874bSReid Kleckner int ThisAdjustment); 260979cb888SDavid Majnemer codeview::TypeIndex lowerTypeEnum(const DICompositeType *Ty); 261a8d57407SReid Kleckner codeview::TypeIndex lowerTypeClass(const DICompositeType *Ty); 262a8d57407SReid Kleckner codeview::TypeIndex lowerTypeUnion(const DICompositeType *Ty); 263a8d57407SReid Kleckner 264a8d57407SReid Kleckner /// Symbol records should point to complete types, but type records should 265a8d57407SReid Kleckner /// always point to incomplete types to avoid cycles in the type graph. Only 266a8d57407SReid Kleckner /// use this entry point when generating symbol records. The complete and 267a8d57407SReid Kleckner /// incomplete type indices only differ for record types. All other types use 268a8d57407SReid Kleckner /// the same index. 269a8d57407SReid Kleckner codeview::TypeIndex getCompleteTypeIndex(DITypeRef TypeRef); 270a8d57407SReid Kleckner 271a8d57407SReid Kleckner codeview::TypeIndex lowerCompleteTypeClass(const DICompositeType *Ty); 272a8d57407SReid Kleckner codeview::TypeIndex lowerCompleteTypeUnion(const DICompositeType *Ty); 273a8d57407SReid Kleckner 274643dd836SReid Kleckner struct TypeLoweringScope; 275643dd836SReid Kleckner 276643dd836SReid Kleckner void emitDeferredCompleteTypes(); 277643dd836SReid Kleckner 27876c9eb99SAmjad Aboud void collectMemberInfo(ClassInfo &Info, const DIDerivedType *DDTy); 2791ab7eac8SReid Kleckner ClassInfo collectClassInfo(const DICompositeType *Ty); 28076c9eb99SAmjad Aboud 281a8d57407SReid Kleckner /// Common record member lowering functionality for record types, which are 282a8d57407SReid Kleckner /// structs, classes, and unions. Returns the field list index and the member 283a8d57407SReid Kleckner /// count. 284820ca540SAdrian McCarthy std::tuple<codeview::TypeIndex, codeview::TypeIndex, unsigned, bool> 285a8d57407SReid Kleckner lowerRecordFieldList(const DICompositeType *Ty); 286a8d57407SReid Kleckner 28776c9eb99SAmjad Aboud /// Inserts {{Node, ClassTy}, TI} into TypeIndices and checks for duplicates. 2880c5d874bSReid Kleckner codeview::TypeIndex recordTypeIndexForDINode(const DINode *Node, 2890c5d874bSReid Kleckner codeview::TypeIndex TI, 29076c9eb99SAmjad Aboud const DIType *ClassTy = nullptr); 29176c9eb99SAmjad Aboud 29276c9eb99SAmjad Aboud unsigned getPointerSizeInBytes(); 2935acacbb0SReid Kleckner 29470f5bc99SReid Kleckner public: 29570f5bc99SReid Kleckner CodeViewDebug(AsmPrinter *Asm); 29670f5bc99SReid Kleckner 29770f5bc99SReid Kleckner void setSymbolSize(const llvm::MCSymbol *, uint64_t) override {} 29870f5bc99SReid Kleckner 29970f5bc99SReid Kleckner /// \brief Emit the COFF section that holds the line table information. 30070f5bc99SReid Kleckner void endModule() override; 30170f5bc99SReid Kleckner 30270f5bc99SReid Kleckner /// \brief Gather pre-function debug information. 30370f5bc99SReid Kleckner void beginFunction(const MachineFunction *MF) override; 30470f5bc99SReid Kleckner 30570f5bc99SReid Kleckner /// \brief Gather post-function debug information. 30670f5bc99SReid Kleckner void endFunction(const MachineFunction *) override; 30770f5bc99SReid Kleckner 30870f5bc99SReid Kleckner /// \brief Process beginning of an instruction. 30970f5bc99SReid Kleckner void beginInstruction(const MachineInstr *MI) override; 31070f5bc99SReid Kleckner }; 31170f5bc99SReid Kleckner } // End of namespace llvm 31270f5bc99SReid Kleckner 31370f5bc99SReid Kleckner #endif 314