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