1 | //===-- PdbIndex.cpp ------------------------------------------------------===// |
2 | // |
3 | // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions. |
4 | // See https://llvm.org/LICENSE.txt for license information. |
5 | // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception |
6 | // |
7 | //===----------------------------------------------------------------------===// |
8 | |
9 | #include "PdbIndex.h" |
10 | #include "PdbUtil.h" |
11 | |
12 | #include "llvm/DebugInfo/CodeView/SymbolDeserializer.h" |
13 | #include "llvm/DebugInfo/PDB/Native/DbiStream.h" |
14 | #include "llvm/DebugInfo/PDB/Native/GlobalsStream.h" |
15 | #include "llvm/DebugInfo/PDB/Native/ISectionContribVisitor.h" |
16 | #include "llvm/DebugInfo/PDB/Native/PDBFile.h" |
17 | #include "llvm/DebugInfo/PDB/Native/PublicsStream.h" |
18 | #include "llvm/DebugInfo/PDB/Native/SymbolStream.h" |
19 | #include "llvm/DebugInfo/PDB/Native/TpiStream.h" |
20 | #include "llvm/Object/COFF.h" |
21 | #include "llvm/Support/Error.h" |
22 | |
23 | #include "lldb/Utility/LLDBAssert.h" |
24 | #include "lldb/lldb-defines.h" |
25 | #include <optional> |
26 | |
27 | using namespace lldb_private; |
28 | using namespace lldb_private::npdb; |
29 | using namespace llvm::codeview; |
30 | using namespace llvm::pdb; |
31 | |
32 | PdbIndex::PdbIndex() : m_cus(*this), m_va_to_modi(m_allocator) {} |
33 | |
34 | #define ASSIGN_PTR_OR_RETURN(result_ptr, expr) \ |
35 | { \ |
36 | auto expected_result = expr; \ |
37 | if (!expected_result) \ |
38 | return expected_result.takeError(); \ |
39 | result_ptr = &expected_result.get(); \ |
40 | } |
41 | |
42 | llvm::Expected<std::unique_ptr<PdbIndex>> |
43 | PdbIndex::create(llvm::pdb::PDBFile *file) { |
44 | lldbassert(file); |
45 | |
46 | std::unique_ptr<PdbIndex> result(new PdbIndex()); |
47 | ASSIGN_PTR_OR_RETURN(result->m_dbi, file->getPDBDbiStream()); |
48 | ASSIGN_PTR_OR_RETURN(result->m_tpi, file->getPDBTpiStream()); |
49 | ASSIGN_PTR_OR_RETURN(result->m_ipi, file->getPDBIpiStream()); |
50 | ASSIGN_PTR_OR_RETURN(result->m_info, file->getPDBInfoStream()); |
51 | ASSIGN_PTR_OR_RETURN(result->m_publics, file->getPDBPublicsStream()); |
52 | ASSIGN_PTR_OR_RETURN(result->m_globals, file->getPDBGlobalsStream()); |
53 | ASSIGN_PTR_OR_RETURN(result->m_symrecords, file->getPDBSymbolStream()); |
54 | |
55 | result->m_tpi->buildHashMap(); |
56 | |
57 | result->m_file = file; |
58 | |
59 | return std::move(result); |
60 | } |
61 | |
62 | lldb::addr_t PdbIndex::MakeVirtualAddress(uint16_t segment, |
63 | uint32_t offset) const { |
64 | uint32_t max_section = dbi().getSectionHeaders().size(); |
65 | // Segment indices are 1-based. |
66 | // If this is an absolute symbol, it's indicated by the magic section index |
67 | // |max_section+1|. In this case, the offset is meaningless, so just return. |
68 | if (segment == 0 || segment > max_section) |
69 | return LLDB_INVALID_ADDRESS; |
70 | |
71 | const llvm::object::coff_section &cs = dbi().getSectionHeaders()[segment - 1]; |
72 | return m_load_address + static_cast<lldb::addr_t>(cs.VirtualAddress) + |
73 | static_cast<lldb::addr_t>(offset); |
74 | } |
75 | |
76 | std::optional<uint16_t> PdbIndex::GetModuleIndexForAddr(uint16_t segment, |
77 | uint32_t offset) const { |
78 | return GetModuleIndexForVa(va: MakeVirtualAddress(segment, offset)); |
79 | } |
80 | |
81 | std::optional<uint16_t> PdbIndex::GetModuleIndexForVa(lldb::addr_t va) const { |
82 | auto iter = m_va_to_modi.find(x: va); |
83 | if (iter == m_va_to_modi.end()) |
84 | return std::nullopt; |
85 | |
86 | return iter.value(); |
87 | } |
88 | |
89 | void PdbIndex::ParseSectionContribs() { |
90 | class Visitor : public ISectionContribVisitor { |
91 | PdbIndex &m_ctx; |
92 | llvm::IntervalMap<uint64_t, uint16_t> &m_imap; |
93 | |
94 | public: |
95 | Visitor(PdbIndex &ctx, llvm::IntervalMap<uint64_t, uint16_t> &imap) |
96 | : m_ctx(ctx), m_imap(imap) {} |
97 | |
98 | void visit(const SectionContrib &C) override { |
99 | if (C.Size == 0) |
100 | return; |
101 | |
102 | uint64_t va = m_ctx.MakeVirtualAddress(segment: C.ISect, offset: C.Off); |
103 | if (va == LLDB_INVALID_ADDRESS) |
104 | return; |
105 | uint64_t end = va + C.Size; |
106 | // IntervalMap's start and end represent a closed range, not a half-open |
107 | // range, so we have to subtract 1. |
108 | m_imap.insert(a: va, b: end - 1, y: C.Imod); |
109 | } |
110 | void visit(const SectionContrib2 &C) override { visit(C: C.Base); } |
111 | }; |
112 | Visitor v(*this, m_va_to_modi); |
113 | dbi().visitSectionContributions(Visitor&: v); |
114 | } |
115 | |
116 | void PdbIndex::BuildAddrToSymbolMap(CompilandIndexItem &cci) { |
117 | lldbassert(cci.m_symbols_by_va.empty() && |
118 | "Addr to symbol map is already built!" ); |
119 | uint16_t modi = cci.m_id.modi; |
120 | const CVSymbolArray &syms = cci.m_debug_stream.getSymbolArray(); |
121 | for (auto iter = syms.begin(); iter != syms.end(); ++iter) { |
122 | if (!SymbolHasAddress(sym: *iter)) |
123 | continue; |
124 | |
125 | SegmentOffset so = GetSegmentAndOffset(sym: *iter); |
126 | lldb::addr_t va = MakeVirtualAddress(segment: so.segment, offset: so.offset); |
127 | if (va == LLDB_INVALID_ADDRESS) |
128 | continue; |
129 | |
130 | PdbCompilandSymId cu_sym_id(modi, iter.offset()); |
131 | |
132 | // It's rare, but we could have multiple symbols with the same address |
133 | // because of identical comdat folding. Right now, the first one will win. |
134 | cci.m_symbols_by_va.insert(x: std::make_pair(x&: va, y: PdbSymUid(cu_sym_id))); |
135 | } |
136 | } |
137 | |
138 | std::vector<SymbolAndUid> PdbIndex::FindSymbolsByVa(lldb::addr_t va) { |
139 | std::vector<SymbolAndUid> result; |
140 | |
141 | std::optional<uint16_t> modi = GetModuleIndexForVa(va); |
142 | if (!modi) |
143 | return result; |
144 | |
145 | CompilandIndexItem &cci = compilands().GetOrCreateCompiland(modi: *modi); |
146 | if (cci.m_symbols_by_va.empty()) |
147 | BuildAddrToSymbolMap(cci); |
148 | |
149 | // The map is sorted by starting address of the symbol. So for example |
150 | // we could (in theory) have this situation |
151 | // |
152 | // [------------------] |
153 | // [----------] |
154 | // [-----------] |
155 | // [-------------] |
156 | // [----] |
157 | // [-----] |
158 | // ^ Address we're searching for |
159 | // In order to find this, we use the upper_bound of the key value which would |
160 | // be the first symbol whose starting address is higher than the element we're |
161 | // searching for. |
162 | |
163 | auto ub = cci.m_symbols_by_va.upper_bound(x: va); |
164 | |
165 | for (auto iter = cci.m_symbols_by_va.begin(); iter != ub; ++iter) { |
166 | PdbCompilandSymId cu_sym_id = iter->second.asCompilandSym(); |
167 | CVSymbol sym = ReadSymbolRecord(cu_sym: cu_sym_id); |
168 | |
169 | SegmentOffsetLength sol; |
170 | if (SymbolIsCode(sym)) |
171 | sol = GetSegmentOffsetAndLength(sym); |
172 | else |
173 | sol.so = GetSegmentAndOffset(sym); |
174 | |
175 | lldb::addr_t start = MakeVirtualAddress(segment: sol.so.segment, offset: sol.so.offset); |
176 | if (start == LLDB_INVALID_ADDRESS) |
177 | continue; |
178 | |
179 | lldb::addr_t end = start + sol.length; |
180 | if (va >= start && va < end) |
181 | result.push_back(x: {.sym: std::move(sym), .uid: iter->second}); |
182 | } |
183 | |
184 | return result; |
185 | } |
186 | |
187 | CVSymbol PdbIndex::ReadSymbolRecord(PdbCompilandSymId cu_sym) const { |
188 | const CompilandIndexItem *cci = compilands().GetCompiland(modi: cu_sym.modi); |
189 | auto iter = cci->m_debug_stream.getSymbolArray().at(Offset: cu_sym.offset); |
190 | lldbassert(iter != cci->m_debug_stream.getSymbolArray().end()); |
191 | return *iter; |
192 | } |
193 | |
194 | CVSymbol PdbIndex::ReadSymbolRecord(PdbGlobalSymId global) const { |
195 | return symrecords().readRecord(Offset: global.offset); |
196 | } |
197 | |