xref: /aosp_15_r20/external/llvm/lib/DebugInfo/DWARF/DWARFUnitIndex.cpp (revision 9880d6810fe72a1726cb53787c6711e909410d58)
1*9880d681SAndroid Build Coastguard Worker //===-- DWARFUnitIndex.cpp ------------------------------------------------===//
2*9880d681SAndroid Build Coastguard Worker //
3*9880d681SAndroid Build Coastguard Worker //                     The LLVM Compiler Infrastructure
4*9880d681SAndroid Build Coastguard Worker //
5*9880d681SAndroid Build Coastguard Worker // This file is distributed under the University of Illinois Open Source
6*9880d681SAndroid Build Coastguard Worker // License. See LICENSE.TXT for details.
7*9880d681SAndroid Build Coastguard Worker //
8*9880d681SAndroid Build Coastguard Worker //===----------------------------------------------------------------------===//
9*9880d681SAndroid Build Coastguard Worker 
10*9880d681SAndroid Build Coastguard Worker #include "llvm/DebugInfo/DWARF/DWARFUnitIndex.h"
11*9880d681SAndroid Build Coastguard Worker 
12*9880d681SAndroid Build Coastguard Worker #include "llvm/ADT/StringRef.h"
13*9880d681SAndroid Build Coastguard Worker #include "llvm/Support/ErrorHandling.h"
14*9880d681SAndroid Build Coastguard Worker 
15*9880d681SAndroid Build Coastguard Worker namespace llvm {
16*9880d681SAndroid Build Coastguard Worker 
parse(DataExtractor IndexData,uint32_t * OffsetPtr)17*9880d681SAndroid Build Coastguard Worker bool DWARFUnitIndex::Header::parse(DataExtractor IndexData,
18*9880d681SAndroid Build Coastguard Worker                                    uint32_t *OffsetPtr) {
19*9880d681SAndroid Build Coastguard Worker   if (!IndexData.isValidOffsetForDataOfSize(*OffsetPtr, 16))
20*9880d681SAndroid Build Coastguard Worker     return false;
21*9880d681SAndroid Build Coastguard Worker   Version = IndexData.getU32(OffsetPtr);
22*9880d681SAndroid Build Coastguard Worker   NumColumns = IndexData.getU32(OffsetPtr);
23*9880d681SAndroid Build Coastguard Worker   NumUnits = IndexData.getU32(OffsetPtr);
24*9880d681SAndroid Build Coastguard Worker   NumBuckets = IndexData.getU32(OffsetPtr);
25*9880d681SAndroid Build Coastguard Worker   return Version <= 2;
26*9880d681SAndroid Build Coastguard Worker }
27*9880d681SAndroid Build Coastguard Worker 
dump(raw_ostream & OS) const28*9880d681SAndroid Build Coastguard Worker void DWARFUnitIndex::Header::dump(raw_ostream &OS) const {
29*9880d681SAndroid Build Coastguard Worker   OS << format("version = %u slots = %u\n\n", Version, NumBuckets);
30*9880d681SAndroid Build Coastguard Worker }
31*9880d681SAndroid Build Coastguard Worker 
parse(DataExtractor IndexData)32*9880d681SAndroid Build Coastguard Worker bool DWARFUnitIndex::parse(DataExtractor IndexData) {
33*9880d681SAndroid Build Coastguard Worker   bool b = parseImpl(IndexData);
34*9880d681SAndroid Build Coastguard Worker   if (!b) {
35*9880d681SAndroid Build Coastguard Worker     // Make sure we don't try to dump anything
36*9880d681SAndroid Build Coastguard Worker     Header.NumBuckets = 0;
37*9880d681SAndroid Build Coastguard Worker     // Release any partially initialized data.
38*9880d681SAndroid Build Coastguard Worker     ColumnKinds.reset();
39*9880d681SAndroid Build Coastguard Worker     Rows.reset();
40*9880d681SAndroid Build Coastguard Worker   }
41*9880d681SAndroid Build Coastguard Worker   return b;
42*9880d681SAndroid Build Coastguard Worker }
43*9880d681SAndroid Build Coastguard Worker 
parseImpl(DataExtractor IndexData)44*9880d681SAndroid Build Coastguard Worker bool DWARFUnitIndex::parseImpl(DataExtractor IndexData) {
45*9880d681SAndroid Build Coastguard Worker   uint32_t Offset = 0;
46*9880d681SAndroid Build Coastguard Worker   if (!Header.parse(IndexData, &Offset))
47*9880d681SAndroid Build Coastguard Worker     return false;
48*9880d681SAndroid Build Coastguard Worker 
49*9880d681SAndroid Build Coastguard Worker   if (!IndexData.isValidOffsetForDataOfSize(
50*9880d681SAndroid Build Coastguard Worker           Offset, Header.NumBuckets * (8 + 4) +
51*9880d681SAndroid Build Coastguard Worker                       (2 * Header.NumUnits + 1) * 4 * Header.NumColumns))
52*9880d681SAndroid Build Coastguard Worker     return false;
53*9880d681SAndroid Build Coastguard Worker 
54*9880d681SAndroid Build Coastguard Worker   Rows = llvm::make_unique<Entry[]>(Header.NumBuckets);
55*9880d681SAndroid Build Coastguard Worker   auto Contribs =
56*9880d681SAndroid Build Coastguard Worker       llvm::make_unique<Entry::SectionContribution *[]>(Header.NumUnits);
57*9880d681SAndroid Build Coastguard Worker   ColumnKinds = llvm::make_unique<DWARFSectionKind[]>(Header.NumColumns);
58*9880d681SAndroid Build Coastguard Worker 
59*9880d681SAndroid Build Coastguard Worker   // Read Hash Table of Signatures
60*9880d681SAndroid Build Coastguard Worker   for (unsigned i = 0; i != Header.NumBuckets; ++i)
61*9880d681SAndroid Build Coastguard Worker     Rows[i].Signature = IndexData.getU64(&Offset);
62*9880d681SAndroid Build Coastguard Worker 
63*9880d681SAndroid Build Coastguard Worker   // Read Parallel Table of Indexes
64*9880d681SAndroid Build Coastguard Worker   for (unsigned i = 0; i != Header.NumBuckets; ++i) {
65*9880d681SAndroid Build Coastguard Worker     auto Index = IndexData.getU32(&Offset);
66*9880d681SAndroid Build Coastguard Worker     if (!Index)
67*9880d681SAndroid Build Coastguard Worker       continue;
68*9880d681SAndroid Build Coastguard Worker     Rows[i].Index = this;
69*9880d681SAndroid Build Coastguard Worker     Rows[i].Contributions =
70*9880d681SAndroid Build Coastguard Worker         llvm::make_unique<Entry::SectionContribution[]>(Header.NumColumns);
71*9880d681SAndroid Build Coastguard Worker     Contribs[Index - 1] = Rows[i].Contributions.get();
72*9880d681SAndroid Build Coastguard Worker   }
73*9880d681SAndroid Build Coastguard Worker 
74*9880d681SAndroid Build Coastguard Worker   // Read the Column Headers
75*9880d681SAndroid Build Coastguard Worker   for (unsigned i = 0; i != Header.NumColumns; ++i) {
76*9880d681SAndroid Build Coastguard Worker     ColumnKinds[i] = static_cast<DWARFSectionKind>(IndexData.getU32(&Offset));
77*9880d681SAndroid Build Coastguard Worker     if (ColumnKinds[i] == InfoColumnKind) {
78*9880d681SAndroid Build Coastguard Worker       if (InfoColumn != -1)
79*9880d681SAndroid Build Coastguard Worker         return false;
80*9880d681SAndroid Build Coastguard Worker       InfoColumn = i;
81*9880d681SAndroid Build Coastguard Worker     }
82*9880d681SAndroid Build Coastguard Worker   }
83*9880d681SAndroid Build Coastguard Worker 
84*9880d681SAndroid Build Coastguard Worker   if (InfoColumn == -1)
85*9880d681SAndroid Build Coastguard Worker     return false;
86*9880d681SAndroid Build Coastguard Worker 
87*9880d681SAndroid Build Coastguard Worker   // Read Table of Section Offsets
88*9880d681SAndroid Build Coastguard Worker   for (unsigned i = 0; i != Header.NumUnits; ++i) {
89*9880d681SAndroid Build Coastguard Worker     auto *Contrib = Contribs[i];
90*9880d681SAndroid Build Coastguard Worker     for (unsigned i = 0; i != Header.NumColumns; ++i)
91*9880d681SAndroid Build Coastguard Worker       Contrib[i].Offset = IndexData.getU32(&Offset);
92*9880d681SAndroid Build Coastguard Worker   }
93*9880d681SAndroid Build Coastguard Worker 
94*9880d681SAndroid Build Coastguard Worker   // Read Table of Section Sizes
95*9880d681SAndroid Build Coastguard Worker   for (unsigned i = 0; i != Header.NumUnits; ++i) {
96*9880d681SAndroid Build Coastguard Worker     auto *Contrib = Contribs[i];
97*9880d681SAndroid Build Coastguard Worker     for (unsigned i = 0; i != Header.NumColumns; ++i)
98*9880d681SAndroid Build Coastguard Worker       Contrib[i].Length = IndexData.getU32(&Offset);
99*9880d681SAndroid Build Coastguard Worker   }
100*9880d681SAndroid Build Coastguard Worker 
101*9880d681SAndroid Build Coastguard Worker   return true;
102*9880d681SAndroid Build Coastguard Worker }
103*9880d681SAndroid Build Coastguard Worker 
getColumnHeader(DWARFSectionKind DS)104*9880d681SAndroid Build Coastguard Worker StringRef DWARFUnitIndex::getColumnHeader(DWARFSectionKind DS) {
105*9880d681SAndroid Build Coastguard Worker #define CASE(DS)                                                               \
106*9880d681SAndroid Build Coastguard Worker   case DW_SECT_##DS:                                                           \
107*9880d681SAndroid Build Coastguard Worker     return #DS;
108*9880d681SAndroid Build Coastguard Worker   switch (DS) {
109*9880d681SAndroid Build Coastguard Worker     CASE(INFO);
110*9880d681SAndroid Build Coastguard Worker     CASE(TYPES);
111*9880d681SAndroid Build Coastguard Worker     CASE(ABBREV);
112*9880d681SAndroid Build Coastguard Worker     CASE(LINE);
113*9880d681SAndroid Build Coastguard Worker     CASE(LOC);
114*9880d681SAndroid Build Coastguard Worker     CASE(STR_OFFSETS);
115*9880d681SAndroid Build Coastguard Worker     CASE(MACINFO);
116*9880d681SAndroid Build Coastguard Worker     CASE(MACRO);
117*9880d681SAndroid Build Coastguard Worker   }
118*9880d681SAndroid Build Coastguard Worker   llvm_unreachable("unknown DWARFSectionKind");
119*9880d681SAndroid Build Coastguard Worker }
120*9880d681SAndroid Build Coastguard Worker 
dump(raw_ostream & OS) const121*9880d681SAndroid Build Coastguard Worker void DWARFUnitIndex::dump(raw_ostream &OS) const {
122*9880d681SAndroid Build Coastguard Worker   if (!Header.NumBuckets)
123*9880d681SAndroid Build Coastguard Worker     return;
124*9880d681SAndroid Build Coastguard Worker 
125*9880d681SAndroid Build Coastguard Worker   Header.dump(OS);
126*9880d681SAndroid Build Coastguard Worker   OS << "Index Signature         ";
127*9880d681SAndroid Build Coastguard Worker   for (unsigned i = 0; i != Header.NumColumns; ++i)
128*9880d681SAndroid Build Coastguard Worker     OS << ' ' << left_justify(getColumnHeader(ColumnKinds[i]), 24);
129*9880d681SAndroid Build Coastguard Worker   OS << "\n----- ------------------";
130*9880d681SAndroid Build Coastguard Worker   for (unsigned i = 0; i != Header.NumColumns; ++i)
131*9880d681SAndroid Build Coastguard Worker     OS << " ------------------------";
132*9880d681SAndroid Build Coastguard Worker   OS << '\n';
133*9880d681SAndroid Build Coastguard Worker   for (unsigned i = 0; i != Header.NumBuckets; ++i) {
134*9880d681SAndroid Build Coastguard Worker     auto &Row = Rows[i];
135*9880d681SAndroid Build Coastguard Worker     if (auto *Contribs = Row.Contributions.get()) {
136*9880d681SAndroid Build Coastguard Worker       OS << format("%5u 0x%016" PRIx64 " ", i + 1, Row.Signature);
137*9880d681SAndroid Build Coastguard Worker       for (unsigned i = 0; i != Header.NumColumns; ++i) {
138*9880d681SAndroid Build Coastguard Worker         auto &Contrib = Contribs[i];
139*9880d681SAndroid Build Coastguard Worker         OS << format("[0x%08x, 0x%08x) ", Contrib.Offset,
140*9880d681SAndroid Build Coastguard Worker                      Contrib.Offset + Contrib.Length);
141*9880d681SAndroid Build Coastguard Worker       }
142*9880d681SAndroid Build Coastguard Worker       OS << '\n';
143*9880d681SAndroid Build Coastguard Worker     }
144*9880d681SAndroid Build Coastguard Worker   }
145*9880d681SAndroid Build Coastguard Worker }
146*9880d681SAndroid Build Coastguard Worker 
147*9880d681SAndroid Build Coastguard Worker const DWARFUnitIndex::Entry::SectionContribution *
getOffset(DWARFSectionKind Sec) const148*9880d681SAndroid Build Coastguard Worker DWARFUnitIndex::Entry::getOffset(DWARFSectionKind Sec) const {
149*9880d681SAndroid Build Coastguard Worker   uint32_t i = 0;
150*9880d681SAndroid Build Coastguard Worker   for (; i != Index->Header.NumColumns; ++i)
151*9880d681SAndroid Build Coastguard Worker     if (Index->ColumnKinds[i] == Sec)
152*9880d681SAndroid Build Coastguard Worker       return &Contributions[i];
153*9880d681SAndroid Build Coastguard Worker   return nullptr;
154*9880d681SAndroid Build Coastguard Worker }
155*9880d681SAndroid Build Coastguard Worker const DWARFUnitIndex::Entry::SectionContribution *
getOffset() const156*9880d681SAndroid Build Coastguard Worker DWARFUnitIndex::Entry::getOffset() const {
157*9880d681SAndroid Build Coastguard Worker   return &Contributions[Index->InfoColumn];
158*9880d681SAndroid Build Coastguard Worker }
159*9880d681SAndroid Build Coastguard Worker 
160*9880d681SAndroid Build Coastguard Worker const DWARFUnitIndex::Entry *
getFromOffset(uint32_t Offset) const161*9880d681SAndroid Build Coastguard Worker DWARFUnitIndex::getFromOffset(uint32_t Offset) const {
162*9880d681SAndroid Build Coastguard Worker   for (uint32_t i = 0; i != Header.NumBuckets; ++i)
163*9880d681SAndroid Build Coastguard Worker     if (const auto &Contribs = Rows[i].Contributions)
164*9880d681SAndroid Build Coastguard Worker       if (Contribs[InfoColumn].Offset == Offset)
165*9880d681SAndroid Build Coastguard Worker         return &Rows[i];
166*9880d681SAndroid Build Coastguard Worker   return nullptr;
167*9880d681SAndroid Build Coastguard Worker }
168*9880d681SAndroid Build Coastguard Worker }
169