//===- DebugInfo.h - Debug Information Helpers ------------------*- C++ -*-===// // // The LLVM Compiler Infrastructure // // This file is distributed under the University of Illinois Open Source // License. See LICENSE.TXT for details. // //===----------------------------------------------------------------------===// // // This file defines a bunch of datatypes that are useful for creating and // walking debug info in LLVM IR form. They essentially provide wrappers around // the information in the global variables that's needed when constructing the // DWARF information. // //===----------------------------------------------------------------------===// #ifndef LLVM_IR_DEBUGINFO_H #define LLVM_IR_DEBUGINFO_H #include "llvm/ADT/DenseMap.h" #include "llvm/ADT/SmallPtrSet.h" #include "llvm/ADT/SmallVector.h" #include "llvm/ADT/StringRef.h" #include "llvm/ADT/iterator_range.h" #include "llvm/IR/DebugInfoMetadata.h" #include "llvm/Support/Casting.h" #include "llvm/Support/Dwarf.h" #include "llvm/Support/ErrorHandling.h" #include namespace llvm { class BasicBlock; class Constant; class Function; class GlobalVariable; class Module; class Type; class Value; class DbgDeclareInst; class DbgValueInst; class Instruction; class Metadata; class MDNode; class MDString; class NamedMDNode; class LLVMContext; class raw_ostream; class DIVariable; class DIObjCProperty; /// \brief Maps from type identifier to the actual MDNode. typedef DenseMap DITypeIdentifierMap; #define DECLARE_SIMPLIFY_DESCRIPTOR(DESC) \ class DESC; \ template <> struct simplify_type; \ template <> struct simplify_type; DECLARE_SIMPLIFY_DESCRIPTOR(DISubrange) DECLARE_SIMPLIFY_DESCRIPTOR(DIEnumerator) DECLARE_SIMPLIFY_DESCRIPTOR(DITemplateTypeParameter) DECLARE_SIMPLIFY_DESCRIPTOR(DITemplateValueParameter) DECLARE_SIMPLIFY_DESCRIPTOR(DIGlobalVariable) DECLARE_SIMPLIFY_DESCRIPTOR(DIVariable) DECLARE_SIMPLIFY_DESCRIPTOR(DIExpression) DECLARE_SIMPLIFY_DESCRIPTOR(DILocation) DECLARE_SIMPLIFY_DESCRIPTOR(DIObjCProperty) DECLARE_SIMPLIFY_DESCRIPTOR(DIImportedEntity) #undef DECLARE_SIMPLIFY_DESCRIPTOR typedef DebugNodeArray DIArray; typedef MDTypeRefArray DITypeArray; class DISubrange { MDSubrange *N; public: DISubrange(const MDSubrange *N = nullptr) : N(const_cast(N)) {} operator MDSubrange *() const { return N; } MDSubrange *operator->() const { return N; } MDSubrange &operator*() const { return *N; } }; class DIEnumerator { MDEnumerator *N; public: DIEnumerator(const MDEnumerator *N = nullptr) : N(const_cast(N)) {} operator MDEnumerator *() const { return N; } MDEnumerator *operator->() const { return N; } MDEnumerator &operator*() const { return *N; } }; class DITemplateTypeParameter { MDTemplateTypeParameter *N; public: DITemplateTypeParameter(const MDTemplateTypeParameter *N = nullptr) : N(const_cast(N)) {} operator MDTemplateTypeParameter *() const { return N; } MDTemplateTypeParameter *operator->() const { return N; } MDTemplateTypeParameter &operator*() const { return *N; } }; class DITemplateValueParameter { MDTemplateValueParameter *N; public: DITemplateValueParameter(const MDTemplateValueParameter *N = nullptr) : N(const_cast(N)) {} operator MDTemplateValueParameter *() const { return N; } MDTemplateValueParameter *operator->() const { return N; } MDTemplateValueParameter &operator*() const { return *N; } }; class DIGlobalVariable { MDGlobalVariable *N; public: DIGlobalVariable(const MDGlobalVariable *N = nullptr) : N(const_cast(N)) {} operator MDGlobalVariable *() const { return N; } MDGlobalVariable *operator->() const { return N; } MDGlobalVariable &operator*() const { return *N; } }; class DIVariable { MDLocalVariable *N; public: DIVariable(const MDLocalVariable *N = nullptr) : N(const_cast(N)) {} operator MDLocalVariable *() const { return N; } MDLocalVariable *operator->() const { return N; } MDLocalVariable &operator*() const { return *N; } }; class DIExpression { MDExpression *N; public: DIExpression(const MDExpression *N = nullptr) : N(const_cast(N)) {} operator MDExpression *() const { return N; } MDExpression *operator->() const { return N; } MDExpression &operator*() const { return *N; } }; class DILocation { MDLocation *N; public: DILocation(const MDLocation *N = nullptr) : N(const_cast(N)) {} operator MDLocation *() const { return N; } MDLocation *operator->() const { return N; } MDLocation &operator*() const { return *N; } }; class DIObjCProperty { MDObjCProperty *N; public: DIObjCProperty(const MDObjCProperty *N = nullptr) : N(const_cast(N)) {} operator MDObjCProperty *() const { return N; } MDObjCProperty *operator->() const { return N; } MDObjCProperty &operator*() const { return *N; } }; class DIImportedEntity { MDImportedEntity *N; public: DIImportedEntity(const MDImportedEntity *N = nullptr) : N(const_cast(N)) {} operator MDImportedEntity *() const { return N; } MDImportedEntity *operator->() const { return N; } MDImportedEntity &operator*() const { return *N; } }; #define SIMPLIFY_DESCRIPTOR(DESC) \ template <> struct simplify_type { \ typedef Metadata *SimpleType; \ static SimpleType getSimplifiedValue(const DESC &DI) { return DI; } \ }; \ template <> struct simplify_type : simplify_type {}; SIMPLIFY_DESCRIPTOR(DISubrange) SIMPLIFY_DESCRIPTOR(DIEnumerator) SIMPLIFY_DESCRIPTOR(DITemplateTypeParameter) SIMPLIFY_DESCRIPTOR(DITemplateValueParameter) SIMPLIFY_DESCRIPTOR(DIGlobalVariable) SIMPLIFY_DESCRIPTOR(DIVariable) SIMPLIFY_DESCRIPTOR(DIExpression) SIMPLIFY_DESCRIPTOR(DILocation) SIMPLIFY_DESCRIPTOR(DIObjCProperty) SIMPLIFY_DESCRIPTOR(DIImportedEntity) #undef SIMPLIFY_DESCRIPTOR /// \brief Find subprogram that is enclosing this scope. MDSubprogram *getDISubprogram(const MDNode *Scope); /// \brief Find debug info for a given function. /// /// \returns a valid subprogram, if found. Otherwise, return \c nullptr. MDSubprogram *getDISubprogram(const Function *F); /// \brief Find underlying composite type. MDCompositeTypeBase *getDICompositeType(MDType *T); /// \brief Generate map by visiting all retained types. DITypeIdentifierMap generateDITypeIdentifierMap(const NamedMDNode *CU_Nodes); /// \brief Strip debug info in the module if it exists. /// /// To do this, we remove all calls to the debugger intrinsics and any named /// metadata for debugging. We also remove debug locations for instructions. /// Return true if module is modified. bool StripDebugInfo(Module &M); bool stripDebugInfo(Function &F); /// \brief Return Debug Info Metadata Version by checking module flags. unsigned getDebugMetadataVersionFromModule(const Module &M); /// \brief Utility to find all debug info in a module. /// /// DebugInfoFinder tries to list all debug info MDNodes used in a module. To /// list debug info MDNodes used by an instruction, DebugInfoFinder uses /// processDeclare, processValue and processLocation to handle DbgDeclareInst, /// DbgValueInst and DbgLoc attached to instructions. processModule will go /// through all DICompileUnits in llvm.dbg.cu and list debug info MDNodes /// used by the CUs. class DebugInfoFinder { public: DebugInfoFinder() : TypeMapInitialized(false) {} /// \brief Process entire module and collect debug info anchors. void processModule(const Module &M); /// \brief Process DbgDeclareInst. void processDeclare(const Module &M, const DbgDeclareInst *DDI); /// \brief Process DbgValueInst. void processValue(const Module &M, const DbgValueInst *DVI); /// \brief Process DILocation. void processLocation(const Module &M, const MDLocation *Loc); /// \brief Clear all lists. void reset(); private: void InitializeTypeMap(const Module &M); void processType(MDType *DT); void processSubprogram(MDSubprogram *SP); void processScope(MDScope *Scope); bool addCompileUnit(MDCompileUnit *CU); bool addGlobalVariable(MDGlobalVariable *DIG); bool addSubprogram(MDSubprogram *SP); bool addType(MDType *DT); bool addScope(MDScope *Scope); public: typedef SmallVectorImpl::const_iterator compile_unit_iterator; typedef SmallVectorImpl::const_iterator subprogram_iterator; typedef SmallVectorImpl::const_iterator global_variable_iterator; typedef SmallVectorImpl::const_iterator type_iterator; typedef SmallVectorImpl::const_iterator scope_iterator; iterator_range compile_units() const { return iterator_range(CUs.begin(), CUs.end()); } iterator_range subprograms() const { return iterator_range(SPs.begin(), SPs.end()); } iterator_range global_variables() const { return iterator_range(GVs.begin(), GVs.end()); } iterator_range types() const { return iterator_range(TYs.begin(), TYs.end()); } iterator_range scopes() const { return iterator_range(Scopes.begin(), Scopes.end()); } unsigned compile_unit_count() const { return CUs.size(); } unsigned global_variable_count() const { return GVs.size(); } unsigned subprogram_count() const { return SPs.size(); } unsigned type_count() const { return TYs.size(); } unsigned scope_count() const { return Scopes.size(); } private: SmallVector CUs; SmallVector SPs; SmallVector GVs; SmallVector TYs; SmallVector Scopes; SmallPtrSet NodesSeen; DITypeIdentifierMap TypeIdentifierMap; /// \brief Specify if TypeIdentifierMap is initialized. bool TypeMapInitialized; }; DenseMap makeSubprogramMap(const Module &M); } // end namespace llvm #endif