1*b5893f02SDimitry Andric #include "llvm/DebugInfo/PDB/Native/SymbolCache.h"
2*b5893f02SDimitry Andric
3*b5893f02SDimitry Andric #include "llvm/DebugInfo/CodeView/SymbolDeserializer.h"
4*b5893f02SDimitry Andric #include "llvm/DebugInfo/CodeView/TypeDeserializer.h"
5*b5893f02SDimitry Andric #include "llvm/DebugInfo/CodeView/TypeRecordHelpers.h"
6*b5893f02SDimitry Andric #include "llvm/DebugInfo/PDB/Native/DbiStream.h"
7*b5893f02SDimitry Andric #include "llvm/DebugInfo/PDB/Native/GlobalsStream.h"
8*b5893f02SDimitry Andric #include "llvm/DebugInfo/PDB/Native/NativeCompilandSymbol.h"
9*b5893f02SDimitry Andric #include "llvm/DebugInfo/PDB/Native/NativeEnumGlobals.h"
10*b5893f02SDimitry Andric #include "llvm/DebugInfo/PDB/Native/NativeEnumTypes.h"
11*b5893f02SDimitry Andric #include "llvm/DebugInfo/PDB/Native/NativeRawSymbol.h"
12*b5893f02SDimitry Andric #include "llvm/DebugInfo/PDB/Native/NativeSession.h"
13*b5893f02SDimitry Andric #include "llvm/DebugInfo/PDB/Native/NativeTypeArray.h"
14*b5893f02SDimitry Andric #include "llvm/DebugInfo/PDB/Native/NativeTypeBuiltin.h"
15*b5893f02SDimitry Andric #include "llvm/DebugInfo/PDB/Native/NativeTypeEnum.h"
16*b5893f02SDimitry Andric #include "llvm/DebugInfo/PDB/Native/NativeTypeFunctionSig.h"
17*b5893f02SDimitry Andric #include "llvm/DebugInfo/PDB/Native/NativeTypePointer.h"
18*b5893f02SDimitry Andric #include "llvm/DebugInfo/PDB/Native/NativeTypeTypedef.h"
19*b5893f02SDimitry Andric #include "llvm/DebugInfo/PDB/Native/NativeTypeUDT.h"
20*b5893f02SDimitry Andric #include "llvm/DebugInfo/PDB/Native/NativeTypeVTShape.h"
21*b5893f02SDimitry Andric #include "llvm/DebugInfo/PDB/Native/PDBFile.h"
22*b5893f02SDimitry Andric #include "llvm/DebugInfo/PDB/Native/SymbolStream.h"
23*b5893f02SDimitry Andric #include "llvm/DebugInfo/PDB/Native/TpiStream.h"
24*b5893f02SDimitry Andric #include "llvm/DebugInfo/PDB/PDBSymbol.h"
25*b5893f02SDimitry Andric #include "llvm/DebugInfo/PDB/PDBSymbolCompiland.h"
26*b5893f02SDimitry Andric #include "llvm/DebugInfo/PDB/PDBSymbolTypeEnum.h"
27*b5893f02SDimitry Andric
28*b5893f02SDimitry Andric using namespace llvm;
29*b5893f02SDimitry Andric using namespace llvm::codeview;
30*b5893f02SDimitry Andric using namespace llvm::pdb;
31*b5893f02SDimitry Andric
32*b5893f02SDimitry Andric // Maps codeview::SimpleTypeKind of a built-in type to the parameters necessary
33*b5893f02SDimitry Andric // to instantiate a NativeBuiltinSymbol for that type.
34*b5893f02SDimitry Andric static const struct BuiltinTypeEntry {
35*b5893f02SDimitry Andric codeview::SimpleTypeKind Kind;
36*b5893f02SDimitry Andric PDB_BuiltinType Type;
37*b5893f02SDimitry Andric uint32_t Size;
38*b5893f02SDimitry Andric } BuiltinTypes[] = {
39*b5893f02SDimitry Andric {codeview::SimpleTypeKind::None, PDB_BuiltinType::None, 0},
40*b5893f02SDimitry Andric {codeview::SimpleTypeKind::Void, PDB_BuiltinType::Void, 0},
41*b5893f02SDimitry Andric {codeview::SimpleTypeKind::HResult, PDB_BuiltinType::HResult, 4},
42*b5893f02SDimitry Andric {codeview::SimpleTypeKind::Int16Short, PDB_BuiltinType::Int, 2},
43*b5893f02SDimitry Andric {codeview::SimpleTypeKind::UInt16Short, PDB_BuiltinType::UInt, 2},
44*b5893f02SDimitry Andric {codeview::SimpleTypeKind::Int32, PDB_BuiltinType::Int, 4},
45*b5893f02SDimitry Andric {codeview::SimpleTypeKind::UInt32, PDB_BuiltinType::UInt, 4},
46*b5893f02SDimitry Andric {codeview::SimpleTypeKind::Int32Long, PDB_BuiltinType::Int, 4},
47*b5893f02SDimitry Andric {codeview::SimpleTypeKind::UInt32Long, PDB_BuiltinType::UInt, 4},
48*b5893f02SDimitry Andric {codeview::SimpleTypeKind::Int64Quad, PDB_BuiltinType::Int, 8},
49*b5893f02SDimitry Andric {codeview::SimpleTypeKind::UInt64Quad, PDB_BuiltinType::UInt, 8},
50*b5893f02SDimitry Andric {codeview::SimpleTypeKind::NarrowCharacter, PDB_BuiltinType::Char, 1},
51*b5893f02SDimitry Andric {codeview::SimpleTypeKind::WideCharacter, PDB_BuiltinType::WCharT, 2},
52*b5893f02SDimitry Andric {codeview::SimpleTypeKind::Character16, PDB_BuiltinType::Char16, 2},
53*b5893f02SDimitry Andric {codeview::SimpleTypeKind::Character32, PDB_BuiltinType::Char32, 4},
54*b5893f02SDimitry Andric {codeview::SimpleTypeKind::SignedCharacter, PDB_BuiltinType::Char, 1},
55*b5893f02SDimitry Andric {codeview::SimpleTypeKind::UnsignedCharacter, PDB_BuiltinType::UInt, 1},
56*b5893f02SDimitry Andric {codeview::SimpleTypeKind::Float32, PDB_BuiltinType::Float, 4},
57*b5893f02SDimitry Andric {codeview::SimpleTypeKind::Float64, PDB_BuiltinType::Float, 8},
58*b5893f02SDimitry Andric {codeview::SimpleTypeKind::Float80, PDB_BuiltinType::Float, 10},
59*b5893f02SDimitry Andric {codeview::SimpleTypeKind::Boolean8, PDB_BuiltinType::Bool, 1},
60*b5893f02SDimitry Andric // This table can be grown as necessary, but these are the only types we've
61*b5893f02SDimitry Andric // needed so far.
62*b5893f02SDimitry Andric };
63*b5893f02SDimitry Andric
SymbolCache(NativeSession & Session,DbiStream * Dbi)64*b5893f02SDimitry Andric SymbolCache::SymbolCache(NativeSession &Session, DbiStream *Dbi)
65*b5893f02SDimitry Andric : Session(Session), Dbi(Dbi) {
66*b5893f02SDimitry Andric // Id 0 is reserved for the invalid symbol.
67*b5893f02SDimitry Andric Cache.push_back(nullptr);
68*b5893f02SDimitry Andric
69*b5893f02SDimitry Andric if (Dbi)
70*b5893f02SDimitry Andric Compilands.resize(Dbi->modules().getModuleCount());
71*b5893f02SDimitry Andric }
72*b5893f02SDimitry Andric
73*b5893f02SDimitry Andric std::unique_ptr<IPDBEnumSymbols>
createTypeEnumerator(TypeLeafKind Kind)74*b5893f02SDimitry Andric SymbolCache::createTypeEnumerator(TypeLeafKind Kind) {
75*b5893f02SDimitry Andric return createTypeEnumerator(std::vector<TypeLeafKind>{Kind});
76*b5893f02SDimitry Andric }
77*b5893f02SDimitry Andric
78*b5893f02SDimitry Andric std::unique_ptr<IPDBEnumSymbols>
createTypeEnumerator(std::vector<TypeLeafKind> Kinds)79*b5893f02SDimitry Andric SymbolCache::createTypeEnumerator(std::vector<TypeLeafKind> Kinds) {
80*b5893f02SDimitry Andric auto Tpi = Session.getPDBFile().getPDBTpiStream();
81*b5893f02SDimitry Andric if (!Tpi) {
82*b5893f02SDimitry Andric consumeError(Tpi.takeError());
83*b5893f02SDimitry Andric return nullptr;
84*b5893f02SDimitry Andric }
85*b5893f02SDimitry Andric auto &Types = Tpi->typeCollection();
86*b5893f02SDimitry Andric return std::unique_ptr<IPDBEnumSymbols>(
87*b5893f02SDimitry Andric new NativeEnumTypes(Session, Types, std::move(Kinds)));
88*b5893f02SDimitry Andric }
89*b5893f02SDimitry Andric
90*b5893f02SDimitry Andric std::unique_ptr<IPDBEnumSymbols>
createGlobalsEnumerator(codeview::SymbolKind Kind)91*b5893f02SDimitry Andric SymbolCache::createGlobalsEnumerator(codeview::SymbolKind Kind) {
92*b5893f02SDimitry Andric return std::unique_ptr<IPDBEnumSymbols>(
93*b5893f02SDimitry Andric new NativeEnumGlobals(Session, {Kind}));
94*b5893f02SDimitry Andric }
95*b5893f02SDimitry Andric
createSimpleType(TypeIndex Index,ModifierOptions Mods)96*b5893f02SDimitry Andric SymIndexId SymbolCache::createSimpleType(TypeIndex Index,
97*b5893f02SDimitry Andric ModifierOptions Mods) {
98*b5893f02SDimitry Andric if (Index.getSimpleMode() != codeview::SimpleTypeMode::Direct)
99*b5893f02SDimitry Andric return createSymbol<NativeTypePointer>(Index);
100*b5893f02SDimitry Andric
101*b5893f02SDimitry Andric const auto Kind = Index.getSimpleKind();
102*b5893f02SDimitry Andric const auto It = std::find_if(
103*b5893f02SDimitry Andric std::begin(BuiltinTypes), std::end(BuiltinTypes),
104*b5893f02SDimitry Andric [Kind](const BuiltinTypeEntry &Builtin) { return Builtin.Kind == Kind; });
105*b5893f02SDimitry Andric if (It == std::end(BuiltinTypes))
106*b5893f02SDimitry Andric return 0;
107*b5893f02SDimitry Andric return createSymbol<NativeTypeBuiltin>(Mods, It->Type, It->Size);
108*b5893f02SDimitry Andric }
109*b5893f02SDimitry Andric
110*b5893f02SDimitry Andric SymIndexId
createSymbolForModifiedType(codeview::TypeIndex ModifierTI,codeview::CVType CVT)111*b5893f02SDimitry Andric SymbolCache::createSymbolForModifiedType(codeview::TypeIndex ModifierTI,
112*b5893f02SDimitry Andric codeview::CVType CVT) {
113*b5893f02SDimitry Andric ModifierRecord Record;
114*b5893f02SDimitry Andric if (auto EC = TypeDeserializer::deserializeAs<ModifierRecord>(CVT, Record)) {
115*b5893f02SDimitry Andric consumeError(std::move(EC));
116*b5893f02SDimitry Andric return 0;
117*b5893f02SDimitry Andric }
118*b5893f02SDimitry Andric
119*b5893f02SDimitry Andric if (Record.ModifiedType.isSimple())
120*b5893f02SDimitry Andric return createSimpleType(Record.ModifiedType, Record.Modifiers);
121*b5893f02SDimitry Andric
122*b5893f02SDimitry Andric // Make sure we create and cache a record for the unmodified type.
123*b5893f02SDimitry Andric SymIndexId UnmodifiedId = findSymbolByTypeIndex(Record.ModifiedType);
124*b5893f02SDimitry Andric NativeRawSymbol &UnmodifiedNRS = *Cache[UnmodifiedId];
125*b5893f02SDimitry Andric
126*b5893f02SDimitry Andric switch (UnmodifiedNRS.getSymTag()) {
127*b5893f02SDimitry Andric case PDB_SymType::Enum:
128*b5893f02SDimitry Andric return createSymbol<NativeTypeEnum>(
129*b5893f02SDimitry Andric static_cast<NativeTypeEnum &>(UnmodifiedNRS), std::move(Record));
130*b5893f02SDimitry Andric case PDB_SymType::UDT:
131*b5893f02SDimitry Andric return createSymbol<NativeTypeUDT>(
132*b5893f02SDimitry Andric static_cast<NativeTypeUDT &>(UnmodifiedNRS), std::move(Record));
133*b5893f02SDimitry Andric default:
134*b5893f02SDimitry Andric // No other types can be modified. (LF_POINTER, for example, records
135*b5893f02SDimitry Andric // its modifiers a different way.
136*b5893f02SDimitry Andric assert(false && "Invalid LF_MODIFIER record");
137*b5893f02SDimitry Andric break;
138*b5893f02SDimitry Andric }
139*b5893f02SDimitry Andric return 0;
140*b5893f02SDimitry Andric }
141*b5893f02SDimitry Andric
findSymbolByTypeIndex(codeview::TypeIndex Index)142*b5893f02SDimitry Andric SymIndexId SymbolCache::findSymbolByTypeIndex(codeview::TypeIndex Index) {
143*b5893f02SDimitry Andric // First see if it's already in our cache.
144*b5893f02SDimitry Andric const auto Entry = TypeIndexToSymbolId.find(Index);
145*b5893f02SDimitry Andric if (Entry != TypeIndexToSymbolId.end())
146*b5893f02SDimitry Andric return Entry->second;
147*b5893f02SDimitry Andric
148*b5893f02SDimitry Andric // Symbols for built-in types are created on the fly.
149*b5893f02SDimitry Andric if (Index.isSimple()) {
150*b5893f02SDimitry Andric SymIndexId Result = createSimpleType(Index, ModifierOptions::None);
151*b5893f02SDimitry Andric assert(TypeIndexToSymbolId.count(Index) == 0);
152*b5893f02SDimitry Andric TypeIndexToSymbolId[Index] = Result;
153*b5893f02SDimitry Andric return Result;
154*b5893f02SDimitry Andric }
155*b5893f02SDimitry Andric
156*b5893f02SDimitry Andric // We need to instantiate and cache the desired type symbol.
157*b5893f02SDimitry Andric auto Tpi = Session.getPDBFile().getPDBTpiStream();
158*b5893f02SDimitry Andric if (!Tpi) {
159*b5893f02SDimitry Andric consumeError(Tpi.takeError());
160*b5893f02SDimitry Andric return 0;
161*b5893f02SDimitry Andric }
162*b5893f02SDimitry Andric codeview::LazyRandomTypeCollection &Types = Tpi->typeCollection();
163*b5893f02SDimitry Andric codeview::CVType CVT = Types.getType(Index);
164*b5893f02SDimitry Andric
165*b5893f02SDimitry Andric if (isUdtForwardRef(CVT)) {
166*b5893f02SDimitry Andric Expected<TypeIndex> EFD = Tpi->findFullDeclForForwardRef(Index);
167*b5893f02SDimitry Andric
168*b5893f02SDimitry Andric if (!EFD)
169*b5893f02SDimitry Andric consumeError(EFD.takeError());
170*b5893f02SDimitry Andric else if (*EFD != Index) {
171*b5893f02SDimitry Andric assert(!isUdtForwardRef(Types.getType(*EFD)));
172*b5893f02SDimitry Andric SymIndexId Result = findSymbolByTypeIndex(*EFD);
173*b5893f02SDimitry Andric // Record a mapping from ForwardRef -> SymIndex of complete type so that
174*b5893f02SDimitry Andric // we'll take the fast path next time.
175*b5893f02SDimitry Andric assert(TypeIndexToSymbolId.count(Index) == 0);
176*b5893f02SDimitry Andric TypeIndexToSymbolId[Index] = Result;
177*b5893f02SDimitry Andric return Result;
178*b5893f02SDimitry Andric }
179*b5893f02SDimitry Andric }
180*b5893f02SDimitry Andric
181*b5893f02SDimitry Andric // At this point if we still have a forward ref udt it means the full decl was
182*b5893f02SDimitry Andric // not in the PDB. We just have to deal with it and use the forward ref.
183*b5893f02SDimitry Andric SymIndexId Id = 0;
184*b5893f02SDimitry Andric switch (CVT.kind()) {
185*b5893f02SDimitry Andric case codeview::LF_ENUM:
186*b5893f02SDimitry Andric Id = createSymbolForType<NativeTypeEnum, EnumRecord>(Index, std::move(CVT));
187*b5893f02SDimitry Andric break;
188*b5893f02SDimitry Andric case codeview::LF_ARRAY:
189*b5893f02SDimitry Andric Id = createSymbolForType<NativeTypeArray, ArrayRecord>(Index,
190*b5893f02SDimitry Andric std::move(CVT));
191*b5893f02SDimitry Andric break;
192*b5893f02SDimitry Andric case codeview::LF_CLASS:
193*b5893f02SDimitry Andric case codeview::LF_STRUCTURE:
194*b5893f02SDimitry Andric case codeview::LF_INTERFACE:
195*b5893f02SDimitry Andric Id = createSymbolForType<NativeTypeUDT, ClassRecord>(Index, std::move(CVT));
196*b5893f02SDimitry Andric break;
197*b5893f02SDimitry Andric case codeview::LF_UNION:
198*b5893f02SDimitry Andric Id = createSymbolForType<NativeTypeUDT, UnionRecord>(Index, std::move(CVT));
199*b5893f02SDimitry Andric break;
200*b5893f02SDimitry Andric case codeview::LF_POINTER:
201*b5893f02SDimitry Andric Id = createSymbolForType<NativeTypePointer, PointerRecord>(Index,
202*b5893f02SDimitry Andric std::move(CVT));
203*b5893f02SDimitry Andric break;
204*b5893f02SDimitry Andric case codeview::LF_MODIFIER:
205*b5893f02SDimitry Andric Id = createSymbolForModifiedType(Index, std::move(CVT));
206*b5893f02SDimitry Andric break;
207*b5893f02SDimitry Andric case codeview::LF_PROCEDURE:
208*b5893f02SDimitry Andric Id = createSymbolForType<NativeTypeFunctionSig, ProcedureRecord>(
209*b5893f02SDimitry Andric Index, std::move(CVT));
210*b5893f02SDimitry Andric break;
211*b5893f02SDimitry Andric case codeview::LF_MFUNCTION:
212*b5893f02SDimitry Andric Id = createSymbolForType<NativeTypeFunctionSig, MemberFunctionRecord>(
213*b5893f02SDimitry Andric Index, std::move(CVT));
214*b5893f02SDimitry Andric break;
215*b5893f02SDimitry Andric case codeview::LF_VTSHAPE:
216*b5893f02SDimitry Andric Id = createSymbolForType<NativeTypeVTShape, VFTableShapeRecord>(
217*b5893f02SDimitry Andric Index, std::move(CVT));
218*b5893f02SDimitry Andric break;
219*b5893f02SDimitry Andric default:
220*b5893f02SDimitry Andric Id = createSymbolPlaceholder();
221*b5893f02SDimitry Andric break;
222*b5893f02SDimitry Andric }
223*b5893f02SDimitry Andric if (Id != 0) {
224*b5893f02SDimitry Andric assert(TypeIndexToSymbolId.count(Index) == 0);
225*b5893f02SDimitry Andric TypeIndexToSymbolId[Index] = Id;
226*b5893f02SDimitry Andric }
227*b5893f02SDimitry Andric return Id;
228*b5893f02SDimitry Andric }
229*b5893f02SDimitry Andric
230*b5893f02SDimitry Andric std::unique_ptr<PDBSymbol>
getSymbolById(SymIndexId SymbolId) const231*b5893f02SDimitry Andric SymbolCache::getSymbolById(SymIndexId SymbolId) const {
232*b5893f02SDimitry Andric assert(SymbolId < Cache.size());
233*b5893f02SDimitry Andric
234*b5893f02SDimitry Andric // Id 0 is reserved.
235*b5893f02SDimitry Andric if (SymbolId == 0 || SymbolId >= Cache.size())
236*b5893f02SDimitry Andric return nullptr;
237*b5893f02SDimitry Andric
238*b5893f02SDimitry Andric // Make sure to handle the case where we've inserted a placeholder symbol
239*b5893f02SDimitry Andric // for types we don't yet suppport.
240*b5893f02SDimitry Andric NativeRawSymbol *NRS = Cache[SymbolId].get();
241*b5893f02SDimitry Andric if (!NRS)
242*b5893f02SDimitry Andric return nullptr;
243*b5893f02SDimitry Andric
244*b5893f02SDimitry Andric return PDBSymbol::create(Session, *NRS);
245*b5893f02SDimitry Andric }
246*b5893f02SDimitry Andric
getNativeSymbolById(SymIndexId SymbolId) const247*b5893f02SDimitry Andric NativeRawSymbol &SymbolCache::getNativeSymbolById(SymIndexId SymbolId) const {
248*b5893f02SDimitry Andric return *Cache[SymbolId];
249*b5893f02SDimitry Andric }
250*b5893f02SDimitry Andric
getNumCompilands() const251*b5893f02SDimitry Andric uint32_t SymbolCache::getNumCompilands() const {
252*b5893f02SDimitry Andric if (!Dbi)
253*b5893f02SDimitry Andric return 0;
254*b5893f02SDimitry Andric
255*b5893f02SDimitry Andric return Dbi->modules().getModuleCount();
256*b5893f02SDimitry Andric }
257*b5893f02SDimitry Andric
getOrCreateGlobalSymbolByOffset(uint32_t Offset)258*b5893f02SDimitry Andric SymIndexId SymbolCache::getOrCreateGlobalSymbolByOffset(uint32_t Offset) {
259*b5893f02SDimitry Andric auto Iter = GlobalOffsetToSymbolId.find(Offset);
260*b5893f02SDimitry Andric if (Iter != GlobalOffsetToSymbolId.end())
261*b5893f02SDimitry Andric return Iter->second;
262*b5893f02SDimitry Andric
263*b5893f02SDimitry Andric SymbolStream &SS = cantFail(Session.getPDBFile().getPDBSymbolStream());
264*b5893f02SDimitry Andric CVSymbol CVS = SS.readRecord(Offset);
265*b5893f02SDimitry Andric SymIndexId Id = 0;
266*b5893f02SDimitry Andric switch (CVS.kind()) {
267*b5893f02SDimitry Andric case SymbolKind::S_UDT: {
268*b5893f02SDimitry Andric UDTSym US = cantFail(SymbolDeserializer::deserializeAs<UDTSym>(CVS));
269*b5893f02SDimitry Andric Id = createSymbol<NativeTypeTypedef>(std::move(US));
270*b5893f02SDimitry Andric break;
271*b5893f02SDimitry Andric }
272*b5893f02SDimitry Andric default:
273*b5893f02SDimitry Andric Id = createSymbolPlaceholder();
274*b5893f02SDimitry Andric break;
275*b5893f02SDimitry Andric }
276*b5893f02SDimitry Andric if (Id != 0) {
277*b5893f02SDimitry Andric assert(GlobalOffsetToSymbolId.count(Offset) == 0);
278*b5893f02SDimitry Andric GlobalOffsetToSymbolId[Offset] = Id;
279*b5893f02SDimitry Andric }
280*b5893f02SDimitry Andric
281*b5893f02SDimitry Andric return Id;
282*b5893f02SDimitry Andric }
283*b5893f02SDimitry Andric
284*b5893f02SDimitry Andric std::unique_ptr<PDBSymbolCompiland>
getOrCreateCompiland(uint32_t Index)285*b5893f02SDimitry Andric SymbolCache::getOrCreateCompiland(uint32_t Index) {
286*b5893f02SDimitry Andric if (!Dbi)
287*b5893f02SDimitry Andric return nullptr;
288*b5893f02SDimitry Andric
289*b5893f02SDimitry Andric if (Index >= Compilands.size())
290*b5893f02SDimitry Andric return nullptr;
291*b5893f02SDimitry Andric
292*b5893f02SDimitry Andric if (Compilands[Index] == 0) {
293*b5893f02SDimitry Andric const DbiModuleList &Modules = Dbi->modules();
294*b5893f02SDimitry Andric Compilands[Index] =
295*b5893f02SDimitry Andric createSymbol<NativeCompilandSymbol>(Modules.getModuleDescriptor(Index));
296*b5893f02SDimitry Andric }
297*b5893f02SDimitry Andric
298*b5893f02SDimitry Andric return Session.getConcreteSymbolById<PDBSymbolCompiland>(Compilands[Index]);
299*b5893f02SDimitry Andric }
300