• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
1 //===-- DWARFContext.h ------------------------------------------*- 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 #ifndef LLVM_DEBUGINFO_DWARFCONTEXT_H
11 #define LLVM_DEBUGINFO_DWARFCONTEXT_H
12 
13 #include "DWARFCompileUnit.h"
14 #include "DWARFDebugAranges.h"
15 #include "DWARFDebugFrame.h"
16 #include "DWARFDebugLine.h"
17 #include "DWARFDebugLoc.h"
18 #include "DWARFDebugRangeList.h"
19 #include "DWARFTypeUnit.h"
20 #include "llvm/ADT/MapVector.h"
21 #include "llvm/ADT/SmallVector.h"
22 #include "llvm/DebugInfo/DIContext.h"
23 
24 namespace llvm {
25 
26 /// DWARFContext
27 /// This data structure is the top level entity that deals with dwarf debug
28 /// information parsing. The actual data is supplied through pure virtual
29 /// methods that a concrete implementation provides.
30 class DWARFContext : public DIContext {
31   typedef SmallVector<std::unique_ptr<DWARFCompileUnit>, 1> CUVector;
32   typedef SmallVector<std::unique_ptr<DWARFTypeUnit>, 1> TUVector;
33 
34   CUVector CUs;
35   TUVector TUs;
36   std::unique_ptr<DWARFDebugAbbrev> Abbrev;
37   std::unique_ptr<DWARFDebugLoc> Loc;
38   std::unique_ptr<DWARFDebugAranges> Aranges;
39   std::unique_ptr<DWARFDebugLine> Line;
40   std::unique_ptr<DWARFDebugFrame> DebugFrame;
41 
42   CUVector DWOCUs;
43   TUVector DWOTUs;
44   std::unique_ptr<DWARFDebugAbbrev> AbbrevDWO;
45   std::unique_ptr<DWARFDebugLocDWO> LocDWO;
46 
47   DWARFContext(DWARFContext &) LLVM_DELETED_FUNCTION;
48   DWARFContext &operator=(DWARFContext &) LLVM_DELETED_FUNCTION;
49 
50   /// Read compile units from the debug_info section (if necessary)
51   /// and store them in CUs.
52   void parseCompileUnits();
53 
54   /// Read type units from the debug_types sections (if necessary)
55   /// and store them in TUs.
56   void parseTypeUnits();
57 
58   /// Read compile units from the debug_info.dwo section (if necessary)
59   /// and store them in DWOCUs.
60   void parseDWOCompileUnits();
61 
62   /// Read type units from the debug_types.dwo section (if necessary)
63   /// and store them in DWOTUs.
64   void parseDWOTypeUnits();
65 
66 public:
67   struct Section {
68     StringRef Data;
69     RelocAddrMap Relocs;
70   };
71 
DWARFContext()72   DWARFContext() : DIContext(CK_DWARF) {}
73 
classof(const DIContext * DICtx)74   static bool classof(const DIContext *DICtx) {
75     return DICtx->getKind() == CK_DWARF;
76   }
77 
78   void dump(raw_ostream &OS, DIDumpType DumpType = DIDT_All) override;
79 
80   typedef iterator_range<CUVector::iterator> cu_iterator_range;
81   typedef iterator_range<TUVector::iterator> tu_iterator_range;
82 
83   /// Get compile units in this context.
compile_units()84   cu_iterator_range compile_units() {
85     parseCompileUnits();
86     return cu_iterator_range(CUs.begin(), CUs.end());
87   }
88 
89   /// Get type units in this context.
type_units()90   tu_iterator_range type_units() {
91     parseTypeUnits();
92     return tu_iterator_range(TUs.begin(), TUs.end());
93   }
94 
95   /// Get compile units in the DWO context.
dwo_compile_units()96   cu_iterator_range dwo_compile_units() {
97     parseDWOCompileUnits();
98     return cu_iterator_range(DWOCUs.begin(), DWOCUs.end());
99   }
100 
101   /// Get type units in the DWO context.
dwo_type_units()102   tu_iterator_range dwo_type_units() {
103     parseDWOTypeUnits();
104     return tu_iterator_range(DWOTUs.begin(), DWOTUs.end());
105   }
106 
107   /// Get the number of compile units in this context.
getNumCompileUnits()108   unsigned getNumCompileUnits() {
109     parseCompileUnits();
110     return CUs.size();
111   }
112 
113   /// Get the number of compile units in this context.
getNumTypeUnits()114   unsigned getNumTypeUnits() {
115     parseTypeUnits();
116     return TUs.size();
117   }
118 
119   /// Get the number of compile units in the DWO context.
getNumDWOCompileUnits()120   unsigned getNumDWOCompileUnits() {
121     parseDWOCompileUnits();
122     return DWOCUs.size();
123   }
124 
125   /// Get the number of compile units in the DWO context.
getNumDWOTypeUnits()126   unsigned getNumDWOTypeUnits() {
127     parseDWOTypeUnits();
128     return DWOTUs.size();
129   }
130 
131   /// Get the compile unit at the specified index for this compile unit.
getCompileUnitAtIndex(unsigned index)132   DWARFCompileUnit *getCompileUnitAtIndex(unsigned index) {
133     parseCompileUnits();
134     return CUs[index].get();
135   }
136 
137   /// Get the compile unit at the specified index for the DWO compile units.
getDWOCompileUnitAtIndex(unsigned index)138   DWARFCompileUnit *getDWOCompileUnitAtIndex(unsigned index) {
139     parseDWOCompileUnits();
140     return DWOCUs[index].get();
141   }
142 
143   /// Get a pointer to the parsed DebugAbbrev object.
144   const DWARFDebugAbbrev *getDebugAbbrev();
145 
146   /// Get a pointer to the parsed DebugLoc object.
147   const DWARFDebugLoc *getDebugLoc();
148 
149   /// Get a pointer to the parsed dwo abbreviations object.
150   const DWARFDebugAbbrev *getDebugAbbrevDWO();
151 
152   /// Get a pointer to the parsed DebugLoc object.
153   const DWARFDebugLocDWO *getDebugLocDWO();
154 
155   /// Get a pointer to the parsed DebugAranges object.
156   const DWARFDebugAranges *getDebugAranges();
157 
158   /// Get a pointer to the parsed frame information object.
159   const DWARFDebugFrame *getDebugFrame();
160 
161   /// Get a pointer to a parsed line table corresponding to a compile unit.
162   const DWARFDebugLine::LineTable *
163   getLineTableForCompileUnit(DWARFCompileUnit *cu);
164 
165   DILineInfo getLineInfoForAddress(uint64_t Address,
166       DILineInfoSpecifier Specifier = DILineInfoSpecifier()) override;
167   DILineInfoTable getLineInfoForAddressRange(uint64_t Address, uint64_t Size,
168       DILineInfoSpecifier Specifier = DILineInfoSpecifier()) override;
169   DIInliningInfo getInliningInfoForAddress(uint64_t Address,
170       DILineInfoSpecifier Specifier = DILineInfoSpecifier()) override;
171 
172   virtual bool isLittleEndian() const = 0;
173   virtual uint8_t getAddressSize() const = 0;
174   virtual const Section &getInfoSection() = 0;
175   typedef MapVector<object::SectionRef, Section,
176                     std::map<object::SectionRef, unsigned> > TypeSectionMap;
177   virtual const TypeSectionMap &getTypesSections() = 0;
178   virtual StringRef getAbbrevSection() = 0;
179   virtual const Section &getLocSection() = 0;
180   virtual const Section &getLocDWOSection() = 0;
181   virtual StringRef getARangeSection() = 0;
182   virtual StringRef getDebugFrameSection() = 0;
183   virtual const Section &getLineSection() = 0;
184   virtual const Section &getLineDWOSection() = 0;
185   virtual StringRef getStringSection() = 0;
186   virtual StringRef getRangeSection() = 0;
187   virtual StringRef getPubNamesSection() = 0;
188   virtual StringRef getPubTypesSection() = 0;
189   virtual StringRef getGnuPubNamesSection() = 0;
190   virtual StringRef getGnuPubTypesSection() = 0;
191 
192   // Sections for DWARF5 split dwarf proposal.
193   virtual const Section &getInfoDWOSection() = 0;
194   virtual const TypeSectionMap &getTypesDWOSections() = 0;
195   virtual StringRef getAbbrevDWOSection() = 0;
196   virtual StringRef getStringDWOSection() = 0;
197   virtual StringRef getStringOffsetDWOSection() = 0;
198   virtual StringRef getRangeDWOSection() = 0;
199   virtual StringRef getAddrSection() = 0;
200 
isSupportedVersion(unsigned version)201   static bool isSupportedVersion(unsigned version) {
202     return version == 2 || version == 3 || version == 4;
203   }
204 private:
205   /// Return the compile unit that includes an offset (relative to .debug_info).
206   DWARFCompileUnit *getCompileUnitForOffset(uint32_t Offset);
207 
208   /// Return the compile unit which contains instruction with provided
209   /// address.
210   DWARFCompileUnit *getCompileUnitForAddress(uint64_t Address);
211 };
212 
213 /// DWARFContextInMemory is the simplest possible implementation of a
214 /// DWARFContext. It assumes all content is available in memory and stores
215 /// pointers to it.
216 class DWARFContextInMemory : public DWARFContext {
217   virtual void anchor();
218   bool IsLittleEndian;
219   uint8_t AddressSize;
220   Section InfoSection;
221   TypeSectionMap TypesSections;
222   StringRef AbbrevSection;
223   Section LocSection;
224   Section LocDWOSection;
225   StringRef ARangeSection;
226   StringRef DebugFrameSection;
227   Section LineSection;
228   Section LineDWOSection;
229   StringRef StringSection;
230   StringRef RangeSection;
231   StringRef PubNamesSection;
232   StringRef PubTypesSection;
233   StringRef GnuPubNamesSection;
234   StringRef GnuPubTypesSection;
235 
236   // Sections for DWARF5 split dwarf proposal.
237   Section InfoDWOSection;
238   TypeSectionMap TypesDWOSections;
239   StringRef AbbrevDWOSection;
240   StringRef StringDWOSection;
241   StringRef StringOffsetDWOSection;
242   StringRef RangeDWOSection;
243   StringRef AddrSection;
244 
245   SmallVector<SmallString<32>, 4> UncompressedSections;
246 
247 public:
248   DWARFContextInMemory(object::ObjectFile *);
isLittleEndian()249   bool isLittleEndian() const override { return IsLittleEndian; }
getAddressSize()250   uint8_t getAddressSize() const override { return AddressSize; }
getInfoSection()251   const Section &getInfoSection() override { return InfoSection; }
getTypesSections()252   const TypeSectionMap &getTypesSections() override { return TypesSections; }
getAbbrevSection()253   StringRef getAbbrevSection() override { return AbbrevSection; }
getLocSection()254   const Section &getLocSection() override { return LocSection; }
getLocDWOSection()255   const Section &getLocDWOSection() override { return LocDWOSection; }
getARangeSection()256   StringRef getARangeSection() override { return ARangeSection; }
getDebugFrameSection()257   StringRef getDebugFrameSection() override { return DebugFrameSection; }
getLineSection()258   const Section &getLineSection() override { return LineSection; }
getLineDWOSection()259   const Section &getLineDWOSection() override { return LineDWOSection; }
getStringSection()260   StringRef getStringSection() override { return StringSection; }
getRangeSection()261   StringRef getRangeSection() override { return RangeSection; }
getPubNamesSection()262   StringRef getPubNamesSection() override { return PubNamesSection; }
getPubTypesSection()263   StringRef getPubTypesSection() override { return PubTypesSection; }
getGnuPubNamesSection()264   StringRef getGnuPubNamesSection() override { return GnuPubNamesSection; }
getGnuPubTypesSection()265   StringRef getGnuPubTypesSection() override { return GnuPubTypesSection; }
266 
267   // Sections for DWARF5 split dwarf proposal.
getInfoDWOSection()268   const Section &getInfoDWOSection() override { return InfoDWOSection; }
getTypesDWOSections()269   const TypeSectionMap &getTypesDWOSections() override {
270     return TypesDWOSections;
271   }
getAbbrevDWOSection()272   StringRef getAbbrevDWOSection() override { return AbbrevDWOSection; }
getStringDWOSection()273   StringRef getStringDWOSection() override { return StringDWOSection; }
getStringOffsetDWOSection()274   StringRef getStringOffsetDWOSection() override {
275     return StringOffsetDWOSection;
276   }
getRangeDWOSection()277   StringRef getRangeDWOSection() override { return RangeDWOSection; }
getAddrSection()278   StringRef getAddrSection() override {
279     return AddrSection;
280   }
281 };
282 
283 }
284 
285 #endif
286