blob: 5cdd628312fe5c7313c7062e2f8ef2efff3f63b2 [file] [log] [blame]
Zachary Turner8ab7dd602018-09-07 00:12:34 +00001#include "llvm/DebugInfo/PDB/Native/SymbolCache.h"
2
Zachary Turnera5e3e022018-10-01 17:55:38 +00003#include "llvm/DebugInfo/CodeView/SymbolDeserializer.h"
Zachary Turner4d689512018-09-14 21:03:57 +00004#include "llvm/DebugInfo/CodeView/TypeDeserializer.h"
Zachary Turner355ffb02018-09-21 22:36:04 +00005#include "llvm/DebugInfo/CodeView/TypeRecordHelpers.h"
Zachary Turner8ab7dd602018-09-07 00:12:34 +00006#include "llvm/DebugInfo/PDB/Native/DbiStream.h"
Zachary Turnera5e3e022018-10-01 17:55:38 +00007#include "llvm/DebugInfo/PDB/Native/GlobalsStream.h"
Zachary Turner8ab7dd602018-09-07 00:12:34 +00008#include "llvm/DebugInfo/PDB/Native/NativeCompilandSymbol.h"
Zachary Turnera5e3e022018-10-01 17:55:38 +00009#include "llvm/DebugInfo/PDB/Native/NativeEnumGlobals.h"
Zachary Turner8ab7dd602018-09-07 00:12:34 +000010#include "llvm/DebugInfo/PDB/Native/NativeEnumTypes.h"
11#include "llvm/DebugInfo/PDB/Native/NativeRawSymbol.h"
12#include "llvm/DebugInfo/PDB/Native/NativeSession.h"
Zachary Turner518cb2d2018-09-30 16:19:18 +000013#include "llvm/DebugInfo/PDB/Native/NativeTypeArray.h"
Zachary Turner5d629962018-09-07 00:12:56 +000014#include "llvm/DebugInfo/PDB/Native/NativeTypeBuiltin.h"
15#include "llvm/DebugInfo/PDB/Native/NativeTypeEnum.h"
Zachary Turner6345e842018-09-21 22:36:28 +000016#include "llvm/DebugInfo/PDB/Native/NativeTypeFunctionSig.h"
Zachary Turnerda4b63a2018-09-07 23:21:33 +000017#include "llvm/DebugInfo/PDB/Native/NativeTypePointer.h"
Zachary Turnera5e3e022018-10-01 17:55:38 +000018#include "llvm/DebugInfo/PDB/Native/NativeTypeTypedef.h"
Zachary Turner355ffb02018-09-21 22:36:04 +000019#include "llvm/DebugInfo/PDB/Native/NativeTypeUDT.h"
Zachary Turner5c1873b2018-10-01 17:55:16 +000020#include "llvm/DebugInfo/PDB/Native/NativeTypeVTShape.h"
Zachary Turner8ab7dd602018-09-07 00:12:34 +000021#include "llvm/DebugInfo/PDB/Native/PDBFile.h"
Zachary Turnera5e3e022018-10-01 17:55:38 +000022#include "llvm/DebugInfo/PDB/Native/SymbolStream.h"
Zachary Turner8ab7dd602018-09-07 00:12:34 +000023#include "llvm/DebugInfo/PDB/Native/TpiStream.h"
24#include "llvm/DebugInfo/PDB/PDBSymbol.h"
25#include "llvm/DebugInfo/PDB/PDBSymbolCompiland.h"
26#include "llvm/DebugInfo/PDB/PDBSymbolTypeEnum.h"
27
28using namespace llvm;
Zachary Turner4d689512018-09-14 21:03:57 +000029using namespace llvm::codeview;
Zachary Turner8ab7dd602018-09-07 00:12:34 +000030using namespace llvm::pdb;
31
32// Maps codeview::SimpleTypeKind of a built-in type to the parameters necessary
33// to instantiate a NativeBuiltinSymbol for that type.
34static const struct BuiltinTypeEntry {
35 codeview::SimpleTypeKind Kind;
36 PDB_BuiltinType Type;
37 uint32_t Size;
38} BuiltinTypes[] = {
Zachary Turner5907a782018-09-20 16:12:05 +000039 {codeview::SimpleTypeKind::None, PDB_BuiltinType::None, 0},
Zachary Turner6345e842018-09-21 22:36:28 +000040 {codeview::SimpleTypeKind::Void, PDB_BuiltinType::Void, 0},
Zachary Turner5c1873b2018-10-01 17:55:16 +000041 {codeview::SimpleTypeKind::HResult, PDB_BuiltinType::HResult, 4},
Zachary Turnerbdf03812018-09-17 21:08:11 +000042 {codeview::SimpleTypeKind::Int16Short, PDB_BuiltinType::Int, 2},
43 {codeview::SimpleTypeKind::UInt16Short, PDB_BuiltinType::UInt, 2},
Zachary Turner8ab7dd602018-09-07 00:12:34 +000044 {codeview::SimpleTypeKind::Int32, PDB_BuiltinType::Int, 4},
45 {codeview::SimpleTypeKind::UInt32, PDB_BuiltinType::UInt, 4},
Zachary Turnerbdf03812018-09-17 21:08:11 +000046 {codeview::SimpleTypeKind::Int32Long, PDB_BuiltinType::Int, 4},
Zachary Turner8ab7dd602018-09-07 00:12:34 +000047 {codeview::SimpleTypeKind::UInt32Long, PDB_BuiltinType::UInt, 4},
Zachary Turnerbdf03812018-09-17 21:08:11 +000048 {codeview::SimpleTypeKind::Int64Quad, PDB_BuiltinType::Int, 8},
Zachary Turner8ab7dd602018-09-07 00:12:34 +000049 {codeview::SimpleTypeKind::UInt64Quad, PDB_BuiltinType::UInt, 8},
50 {codeview::SimpleTypeKind::NarrowCharacter, PDB_BuiltinType::Char, 1},
Zachary Turner5c1873b2018-10-01 17:55:16 +000051 {codeview::SimpleTypeKind::WideCharacter, PDB_BuiltinType::WCharT, 2},
52 {codeview::SimpleTypeKind::Character16, PDB_BuiltinType::Char16, 2},
53 {codeview::SimpleTypeKind::Character32, PDB_BuiltinType::Char32, 4},
Zachary Turner8ab7dd602018-09-07 00:12:34 +000054 {codeview::SimpleTypeKind::SignedCharacter, PDB_BuiltinType::Char, 1},
55 {codeview::SimpleTypeKind::UnsignedCharacter, PDB_BuiltinType::UInt, 1},
Zachary Turner5c1873b2018-10-01 17:55:16 +000056 {codeview::SimpleTypeKind::Float32, PDB_BuiltinType::Float, 4},
57 {codeview::SimpleTypeKind::Float64, PDB_BuiltinType::Float, 8},
58 {codeview::SimpleTypeKind::Float80, PDB_BuiltinType::Float, 10},
59 {codeview::SimpleTypeKind::Boolean8, PDB_BuiltinType::Bool, 1},
Zachary Turner8ab7dd602018-09-07 00:12:34 +000060 // This table can be grown as necessary, but these are the only types we've
61 // needed so far.
62};
63
64SymbolCache::SymbolCache(NativeSession &Session, DbiStream *Dbi)
65 : Session(Session), Dbi(Dbi) {
66 // Id 0 is reserved for the invalid symbol.
67 Cache.push_back(nullptr);
68
69 if (Dbi)
70 Compilands.resize(Dbi->modules().getModuleCount());
71}
72
Zachary Turner8ab7dd602018-09-07 00:12:34 +000073std::unique_ptr<IPDBEnumSymbols>
Zachary Turner355ffb02018-09-21 22:36:04 +000074SymbolCache::createTypeEnumerator(TypeLeafKind Kind) {
75 return createTypeEnumerator(std::vector<TypeLeafKind>{Kind});
76}
77
78std::unique_ptr<IPDBEnumSymbols>
79SymbolCache::createTypeEnumerator(std::vector<TypeLeafKind> Kinds) {
Zachary Turner8ab7dd602018-09-07 00:12:34 +000080 auto Tpi = Session.getPDBFile().getPDBTpiStream();
81 if (!Tpi) {
82 consumeError(Tpi.takeError());
83 return nullptr;
84 }
85 auto &Types = Tpi->typeCollection();
86 return std::unique_ptr<IPDBEnumSymbols>(
Zachary Turner355ffb02018-09-21 22:36:04 +000087 new NativeEnumTypes(Session, Types, std::move(Kinds)));
Zachary Turner8ab7dd602018-09-07 00:12:34 +000088}
89
Zachary Turnera5e3e022018-10-01 17:55:38 +000090std::unique_ptr<IPDBEnumSymbols>
91SymbolCache::createGlobalsEnumerator(codeview::SymbolKind Kind) {
92 return std::unique_ptr<IPDBEnumSymbols>(
93 new NativeEnumGlobals(Session, {Kind}));
94}
95
Zachary Turner4727ac22018-09-17 21:07:48 +000096SymIndexId SymbolCache::createSimpleType(TypeIndex Index,
97 ModifierOptions Mods) {
Zachary Turner6345e842018-09-21 22:36:28 +000098 if (Index.getSimpleMode() != codeview::SimpleTypeMode::Direct)
99 return createSymbol<NativeTypePointer>(Index);
Zachary Turner4727ac22018-09-17 21:07:48 +0000100
101 const auto Kind = Index.getSimpleKind();
102 const auto It = std::find_if(
103 std::begin(BuiltinTypes), std::end(BuiltinTypes),
104 [Kind](const BuiltinTypeEntry &Builtin) { return Builtin.Kind == Kind; });
105 if (It == std::end(BuiltinTypes))
106 return 0;
Zachary Turner6345e842018-09-21 22:36:28 +0000107 return createSymbol<NativeTypeBuiltin>(Mods, It->Type, It->Size);
Zachary Turner4727ac22018-09-17 21:07:48 +0000108}
109
110SymIndexId
111SymbolCache::createSymbolForModifiedType(codeview::TypeIndex ModifierTI,
112 codeview::CVType CVT) {
113 ModifierRecord Record;
114 if (auto EC = TypeDeserializer::deserializeAs<ModifierRecord>(CVT, Record)) {
115 consumeError(std::move(EC));
116 return 0;
117 }
118
119 if (Record.ModifiedType.isSimple())
120 return createSimpleType(Record.ModifiedType, Record.Modifiers);
121
Zachary Turner355ffb02018-09-21 22:36:04 +0000122 // Make sure we create and cache a record for the unmodified type.
123 SymIndexId UnmodifiedId = findSymbolByTypeIndex(Record.ModifiedType);
124 NativeRawSymbol &UnmodifiedNRS = *Cache[UnmodifiedId];
Zachary Turner4727ac22018-09-17 21:07:48 +0000125
Zachary Turner355ffb02018-09-21 22:36:04 +0000126 switch (UnmodifiedNRS.getSymTag()) {
127 case PDB_SymType::Enum:
128 return createSymbol<NativeTypeEnum>(
129 static_cast<NativeTypeEnum &>(UnmodifiedNRS), std::move(Record));
130 case PDB_SymType::UDT:
131 return createSymbol<NativeTypeUDT>(
132 static_cast<NativeTypeUDT &>(UnmodifiedNRS), std::move(Record));
Zachary Turner4727ac22018-09-17 21:07:48 +0000133 default:
134 // No other types can be modified. (LF_POINTER, for example, records
135 // its modifiers a different way.
136 assert(false && "Invalid LF_MODIFIER record");
137 break;
138 }
Zachary Turner355ffb02018-09-21 22:36:04 +0000139 return 0;
Zachary Turner4727ac22018-09-17 21:07:48 +0000140}
141
Zachary Turner8ab7dd602018-09-07 00:12:34 +0000142SymIndexId SymbolCache::findSymbolByTypeIndex(codeview::TypeIndex Index) {
143 // First see if it's already in our cache.
144 const auto Entry = TypeIndexToSymbolId.find(Index);
145 if (Entry != TypeIndexToSymbolId.end())
146 return Entry->second;
147
148 // Symbols for built-in types are created on the fly.
Zachary Turner6345e842018-09-21 22:36:28 +0000149 if (Index.isSimple()) {
150 SymIndexId Result = createSimpleType(Index, ModifierOptions::None);
151 assert(TypeIndexToSymbolId.count(Index) == 0);
152 TypeIndexToSymbolId[Index] = Result;
153 return Result;
154 }
Zachary Turner8ab7dd602018-09-07 00:12:34 +0000155
156 // We need to instantiate and cache the desired type symbol.
157 auto Tpi = Session.getPDBFile().getPDBTpiStream();
158 if (!Tpi) {
159 consumeError(Tpi.takeError());
160 return 0;
161 }
162 codeview::LazyRandomTypeCollection &Types = Tpi->typeCollection();
Zachary Turner4d689512018-09-14 21:03:57 +0000163 codeview::CVType CVT = Types.getType(Index);
Zachary Turner4727ac22018-09-17 21:07:48 +0000164
Zachary Turner355ffb02018-09-21 22:36:04 +0000165 if (isUdtForwardRef(CVT)) {
166 Expected<TypeIndex> EFD = Tpi->findFullDeclForForwardRef(Index);
167
168 if (!EFD)
169 consumeError(EFD.takeError());
170 else if (*EFD != Index) {
171 assert(!isUdtForwardRef(Types.getType(*EFD)));
172 SymIndexId Result = findSymbolByTypeIndex(*EFD);
173 // Record a mapping from ForwardRef -> SymIndex of complete type so that
174 // we'll take the fast path next time.
Zachary Turner6345e842018-09-21 22:36:28 +0000175 assert(TypeIndexToSymbolId.count(Index) == 0);
Zachary Turner355ffb02018-09-21 22:36:04 +0000176 TypeIndexToSymbolId[Index] = Result;
177 return Result;
178 }
179 }
180
181 // At this point if we still have a forward ref udt it means the full decl was
182 // not in the PDB. We just have to deal with it and use the forward ref.
183 SymIndexId Id = 0;
Zachary Turner8ab7dd602018-09-07 00:12:34 +0000184 switch (CVT.kind()) {
185 case codeview::LF_ENUM:
Zachary Turner4d689512018-09-14 21:03:57 +0000186 Id = createSymbolForType<NativeTypeEnum, EnumRecord>(Index, std::move(CVT));
Zachary Turner8ab7dd602018-09-07 00:12:34 +0000187 break;
Zachary Turner518cb2d2018-09-30 16:19:18 +0000188 case codeview::LF_ARRAY:
189 Id = createSymbolForType<NativeTypeArray, ArrayRecord>(Index,
190 std::move(CVT));
191 break;
Zachary Turner355ffb02018-09-21 22:36:04 +0000192 case codeview::LF_CLASS:
193 case codeview::LF_STRUCTURE:
194 case codeview::LF_INTERFACE:
195 Id = createSymbolForType<NativeTypeUDT, ClassRecord>(Index, std::move(CVT));
196 break;
197 case codeview::LF_UNION:
198 Id = createSymbolForType<NativeTypeUDT, UnionRecord>(Index, std::move(CVT));
199 break;
Zachary Turnerda4b63a2018-09-07 23:21:33 +0000200 case codeview::LF_POINTER:
Zachary Turner4d689512018-09-14 21:03:57 +0000201 Id = createSymbolForType<NativeTypePointer, PointerRecord>(Index,
202 std::move(CVT));
Zachary Turnerda4b63a2018-09-07 23:21:33 +0000203 break;
Zachary Turner4727ac22018-09-17 21:07:48 +0000204 case codeview::LF_MODIFIER:
205 Id = createSymbolForModifiedType(Index, std::move(CVT));
206 break;
Zachary Turner6345e842018-09-21 22:36:28 +0000207 case codeview::LF_PROCEDURE:
208 Id = createSymbolForType<NativeTypeFunctionSig, ProcedureRecord>(
209 Index, std::move(CVT));
210 break;
211 case codeview::LF_MFUNCTION:
212 Id = createSymbolForType<NativeTypeFunctionSig, MemberFunctionRecord>(
213 Index, std::move(CVT));
214 break;
Zachary Turner5c1873b2018-10-01 17:55:16 +0000215 case codeview::LF_VTSHAPE:
216 Id = createSymbolForType<NativeTypeVTShape, VFTableShapeRecord>(
217 Index, std::move(CVT));
218 break;
Zachary Turner8ab7dd602018-09-07 00:12:34 +0000219 default:
Zachary Turnerda4b63a2018-09-07 23:21:33 +0000220 Id = createSymbolPlaceholder();
221 break;
Zachary Turner8ab7dd602018-09-07 00:12:34 +0000222 }
Zachary Turner6345e842018-09-21 22:36:28 +0000223 if (Id != 0) {
224 assert(TypeIndexToSymbolId.count(Index) == 0);
Zachary Turner4d689512018-09-14 21:03:57 +0000225 TypeIndexToSymbolId[Index] = Id;
Zachary Turner6345e842018-09-21 22:36:28 +0000226 }
Zachary Turner8ab7dd602018-09-07 00:12:34 +0000227 return Id;
228}
229
230std::unique_ptr<PDBSymbol>
231SymbolCache::getSymbolById(SymIndexId SymbolId) const {
Zachary Turnerbdf03812018-09-17 21:08:11 +0000232 assert(SymbolId < Cache.size());
233
Zachary Turnerda4b63a2018-09-07 23:21:33 +0000234 // Id 0 is reserved.
Zachary Turnerc41ce832018-09-18 16:35:05 +0000235 if (SymbolId == 0 || SymbolId >= Cache.size())
Zachary Turnerda4b63a2018-09-07 23:21:33 +0000236 return nullptr;
237
Zachary Turnerc41ce832018-09-18 16:35:05 +0000238 // Make sure to handle the case where we've inserted a placeholder symbol
239 // for types we don't yet suppport.
240 NativeRawSymbol *NRS = Cache[SymbolId].get();
241 if (!NRS)
242 return nullptr;
243
244 return PDBSymbol::create(Session, *NRS);
Zachary Turner8ab7dd602018-09-07 00:12:34 +0000245}
246
247NativeRawSymbol &SymbolCache::getNativeSymbolById(SymIndexId SymbolId) const {
248 return *Cache[SymbolId];
249}
250
251uint32_t SymbolCache::getNumCompilands() const {
252 if (!Dbi)
253 return 0;
254
255 return Dbi->modules().getModuleCount();
256}
257
Zachary Turnera5e3e022018-10-01 17:55:38 +0000258SymIndexId SymbolCache::getOrCreateGlobalSymbolByOffset(uint32_t Offset) {
259 auto Iter = GlobalOffsetToSymbolId.find(Offset);
260 if (Iter != GlobalOffsetToSymbolId.end())
261 return Iter->second;
262
263 SymbolStream &SS = cantFail(Session.getPDBFile().getPDBSymbolStream());
264 CVSymbol CVS = SS.readRecord(Offset);
265 SymIndexId Id = 0;
266 switch (CVS.kind()) {
267 case SymbolKind::S_UDT: {
268 UDTSym US = cantFail(SymbolDeserializer::deserializeAs<UDTSym>(CVS));
269 Id = createSymbol<NativeTypeTypedef>(std::move(US));
270 break;
271 }
272 default:
273 Id = createSymbolPlaceholder();
274 break;
275 }
276 if (Id != 0) {
277 assert(GlobalOffsetToSymbolId.count(Offset) == 0);
278 GlobalOffsetToSymbolId[Offset] = Id;
279 }
280
281 return Id;
282}
283
Zachary Turner8ab7dd602018-09-07 00:12:34 +0000284std::unique_ptr<PDBSymbolCompiland>
285SymbolCache::getOrCreateCompiland(uint32_t Index) {
286 if (!Dbi)
287 return nullptr;
288
289 if (Index >= Compilands.size())
290 return nullptr;
291
292 if (Compilands[Index] == 0) {
293 const DbiModuleList &Modules = Dbi->modules();
294 Compilands[Index] =
295 createSymbol<NativeCompilandSymbol>(Modules.getModuleDescriptor(Index));
296 }
297
298 return Session.getConcreteSymbolById<PDBSymbolCompiland>(Compilands[Index]);
299}