1 //===- TypeIndexDiscovery.cpp -----------------------------------*- C++ -*-===//
2 //
3 // The LLVM Compiler Infrastructure
4 //
5 // This file is distributed under the University of Illinois Open Source
6 // License. See LICENSE.TXT for details.
7 //
8 //===----------------------------------------------------------------------===//
9 #include "llvm/DebugInfo/CodeView/TypeIndexDiscovery.h"
10
11 #include "llvm/ADT/ArrayRef.h"
12 #include "llvm/Support/Endian.h"
13
14 using namespace llvm;
15 using namespace llvm::codeview;
16
getMethodKind(uint16_t Attrs)17 static inline MethodKind getMethodKind(uint16_t Attrs) {
18 Attrs &= uint16_t(MethodOptions::MethodKindMask);
19 Attrs >>= 2;
20 return MethodKind(Attrs);
21 }
22
isIntroVirtual(uint16_t Attrs)23 static inline bool isIntroVirtual(uint16_t Attrs) {
24 MethodKind MK = getMethodKind(Attrs);
25 return MK == MethodKind::IntroducingVirtual ||
26 MK == MethodKind::PureIntroducingVirtual;
27 }
28
getPointerMode(uint32_t Attrs)29 static inline PointerMode getPointerMode(uint32_t Attrs) {
30 return static_cast<PointerMode>((Attrs >> PointerRecord::PointerModeShift) &
31 PointerRecord::PointerModeMask);
32 }
33
isMemberPointer(uint32_t Attrs)34 static inline bool isMemberPointer(uint32_t Attrs) {
35 PointerMode Mode = getPointerMode(Attrs);
36 return Mode == PointerMode::PointerToDataMember ||
37 Mode == PointerMode::PointerToMemberFunction;
38 }
39
getEncodedIntegerLength(ArrayRef<uint8_t> Data)40 static inline uint32_t getEncodedIntegerLength(ArrayRef<uint8_t> Data) {
41 uint16_t N = support::endian::read16le(Data.data());
42 if (N < LF_NUMERIC)
43 return 2;
44
45 assert(N <= LF_UQUADWORD);
46
47 constexpr uint32_t Sizes[] = {
48 1, // LF_CHAR
49 2, // LF_SHORT
50 2, // LF_USHORT
51 4, // LF_LONG
52 4, // LF_ULONG
53 4, // LF_REAL32
54 8, // LF_REAL64
55 10, // LF_REAL80
56 16, // LF_REAL128
57 8, // LF_QUADWORD
58 8, // LF_UQUADWORD
59 };
60
61 return 2 + Sizes[N - LF_NUMERIC];
62 }
63
getCStringLength(ArrayRef<uint8_t> Data)64 static inline uint32_t getCStringLength(ArrayRef<uint8_t> Data) {
65 const char *S = reinterpret_cast<const char *>(Data.data());
66 return strlen(S) + 1;
67 }
68
handleMethodOverloadList(ArrayRef<uint8_t> Content,SmallVectorImpl<TiReference> & Refs)69 static void handleMethodOverloadList(ArrayRef<uint8_t> Content,
70 SmallVectorImpl<TiReference> &Refs) {
71 uint32_t Offset = 0;
72
73 while (!Content.empty()) {
74 // Array of:
75 // 0: Attrs
76 // 2: Padding
77 // 4: TypeIndex
78 // if (isIntroVirtual())
79 // 8: VFTableOffset
80
81 // At least 8 bytes are guaranteed. 4 extra bytes come iff function is an
82 // intro virtual.
83 uint32_t Len = 8;
84
85 uint16_t Attrs = support::endian::read16le(Content.data());
86 Refs.push_back({TiRefKind::TypeRef, Offset + 4, 1});
87
88 if (LLVM_UNLIKELY(isIntroVirtual(Attrs)))
89 Len += 4;
90 Offset += Len;
91 Content = Content.drop_front(Len);
92 }
93 }
94
handleBaseClass(ArrayRef<uint8_t> Data,uint32_t Offset,SmallVectorImpl<TiReference> & Refs)95 static uint32_t handleBaseClass(ArrayRef<uint8_t> Data, uint32_t Offset,
96 SmallVectorImpl<TiReference> &Refs) {
97 // 0: Kind
98 // 2: Padding
99 // 4: TypeIndex
100 // 8: Encoded Integer
101 Refs.push_back({TiRefKind::TypeRef, Offset + 4, 1});
102 return 8 + getEncodedIntegerLength(Data.drop_front(8));
103 }
104
handleEnumerator(ArrayRef<uint8_t> Data,uint32_t Offset,SmallVectorImpl<TiReference> & Refs)105 static uint32_t handleEnumerator(ArrayRef<uint8_t> Data, uint32_t Offset,
106 SmallVectorImpl<TiReference> &Refs) {
107 // 0: Kind
108 // 2: Padding
109 // 4: Encoded Integer
110 // <next>: Name
111 uint32_t Size = 4 + getEncodedIntegerLength(Data.drop_front(4));
112 return Size + getCStringLength(Data.drop_front(Size));
113 }
114
handleDataMember(ArrayRef<uint8_t> Data,uint32_t Offset,SmallVectorImpl<TiReference> & Refs)115 static uint32_t handleDataMember(ArrayRef<uint8_t> Data, uint32_t Offset,
116 SmallVectorImpl<TiReference> &Refs) {
117 // 0: Kind
118 // 2: Padding
119 // 4: TypeIndex
120 // 8: Encoded Integer
121 // <next>: Name
122 Refs.push_back({TiRefKind::TypeRef, Offset + 4, 1});
123 uint32_t Size = 8 + getEncodedIntegerLength(Data.drop_front(8));
124 return Size + getCStringLength(Data.drop_front(Size));
125 }
126
handleOverloadedMethod(ArrayRef<uint8_t> Data,uint32_t Offset,SmallVectorImpl<TiReference> & Refs)127 static uint32_t handleOverloadedMethod(ArrayRef<uint8_t> Data, uint32_t Offset,
128 SmallVectorImpl<TiReference> &Refs) {
129 // 0: Kind
130 // 2: Padding
131 // 4: TypeIndex
132 // 8: Name
133 Refs.push_back({TiRefKind::TypeRef, Offset + 4, 1});
134 return 8 + getCStringLength(Data.drop_front(8));
135 }
136
handleOneMethod(ArrayRef<uint8_t> Data,uint32_t Offset,SmallVectorImpl<TiReference> & Refs)137 static uint32_t handleOneMethod(ArrayRef<uint8_t> Data, uint32_t Offset,
138 SmallVectorImpl<TiReference> &Refs) {
139 // 0: Kind
140 // 2: Attributes
141 // 4: Type
142 // if (isIntroVirtual)
143 // 8: VFTableOffset
144 // <next>: Name
145 uint32_t Size = 8;
146 Refs.push_back({TiRefKind::TypeRef, Offset + 4, 1});
147
148 uint16_t Attrs = support::endian::read16le(Data.drop_front(2).data());
149 if (LLVM_UNLIKELY(isIntroVirtual(Attrs)))
150 Size += 4;
151
152 return Size + getCStringLength(Data.drop_front(Size));
153 }
154
handleNestedType(ArrayRef<uint8_t> Data,uint32_t Offset,SmallVectorImpl<TiReference> & Refs)155 static uint32_t handleNestedType(ArrayRef<uint8_t> Data, uint32_t Offset,
156 SmallVectorImpl<TiReference> &Refs) {
157 // 0: Kind
158 // 2: Padding
159 // 4: TypeIndex
160 // 8: Name
161 Refs.push_back({TiRefKind::TypeRef, Offset + 4, 1});
162 return 8 + getCStringLength(Data.drop_front(8));
163 }
164
handleStaticDataMember(ArrayRef<uint8_t> Data,uint32_t Offset,SmallVectorImpl<TiReference> & Refs)165 static uint32_t handleStaticDataMember(ArrayRef<uint8_t> Data, uint32_t Offset,
166 SmallVectorImpl<TiReference> &Refs) {
167 // 0: Kind
168 // 2: Padding
169 // 4: TypeIndex
170 // 8: Name
171 Refs.push_back({TiRefKind::TypeRef, Offset + 4, 1});
172 return 8 + getCStringLength(Data.drop_front(8));
173 }
174
handleVirtualBaseClass(ArrayRef<uint8_t> Data,uint32_t Offset,bool IsIndirect,SmallVectorImpl<TiReference> & Refs)175 static uint32_t handleVirtualBaseClass(ArrayRef<uint8_t> Data, uint32_t Offset,
176 bool IsIndirect,
177 SmallVectorImpl<TiReference> &Refs) {
178 // 0: Kind
179 // 2: Attrs
180 // 4: TypeIndex
181 // 8: TypeIndex
182 // 12: Encoded Integer
183 // <next>: Encoded Integer
184 uint32_t Size = 12;
185 Refs.push_back({TiRefKind::TypeRef, Offset + 4, 2});
186 Size += getEncodedIntegerLength(Data.drop_front(Size));
187 Size += getEncodedIntegerLength(Data.drop_front(Size));
188 return Size;
189 }
190
handleVFPtr(ArrayRef<uint8_t> Data,uint32_t Offset,SmallVectorImpl<TiReference> & Refs)191 static uint32_t handleVFPtr(ArrayRef<uint8_t> Data, uint32_t Offset,
192 SmallVectorImpl<TiReference> &Refs) {
193 // 0: Kind
194 // 2: Padding
195 // 4: TypeIndex
196 Refs.push_back({TiRefKind::TypeRef, Offset + 4, 1});
197 return 8;
198 }
199
handleListContinuation(ArrayRef<uint8_t> Data,uint32_t Offset,SmallVectorImpl<TiReference> & Refs)200 static uint32_t handleListContinuation(ArrayRef<uint8_t> Data, uint32_t Offset,
201 SmallVectorImpl<TiReference> &Refs) {
202 // 0: Kind
203 // 2: Padding
204 // 4: TypeIndex
205 Refs.push_back({TiRefKind::TypeRef, Offset + 4, 1});
206 return 8;
207 }
208
handleFieldList(ArrayRef<uint8_t> Content,SmallVectorImpl<TiReference> & Refs)209 static void handleFieldList(ArrayRef<uint8_t> Content,
210 SmallVectorImpl<TiReference> &Refs) {
211 uint32_t Offset = 0;
212 uint32_t ThisLen = 0;
213 while (!Content.empty()) {
214 TypeLeafKind Kind =
215 static_cast<TypeLeafKind>(support::endian::read16le(Content.data()));
216 switch (Kind) {
217 case LF_BCLASS:
218 ThisLen = handleBaseClass(Content, Offset, Refs);
219 break;
220 case LF_ENUMERATE:
221 ThisLen = handleEnumerator(Content, Offset, Refs);
222 break;
223 case LF_MEMBER:
224 ThisLen = handleDataMember(Content, Offset, Refs);
225 break;
226 case LF_METHOD:
227 ThisLen = handleOverloadedMethod(Content, Offset, Refs);
228 break;
229 case LF_ONEMETHOD:
230 ThisLen = handleOneMethod(Content, Offset, Refs);
231 break;
232 case LF_NESTTYPE:
233 ThisLen = handleNestedType(Content, Offset, Refs);
234 break;
235 case LF_STMEMBER:
236 ThisLen = handleStaticDataMember(Content, Offset, Refs);
237 break;
238 case LF_VBCLASS:
239 case LF_IVBCLASS:
240 ThisLen =
241 handleVirtualBaseClass(Content, Offset, Kind == LF_VBCLASS, Refs);
242 break;
243 case LF_VFUNCTAB:
244 ThisLen = handleVFPtr(Content, Offset, Refs);
245 break;
246 case LF_INDEX:
247 ThisLen = handleListContinuation(Content, Offset, Refs);
248 break;
249 default:
250 return;
251 }
252 Content = Content.drop_front(ThisLen);
253 Offset += ThisLen;
254 if (!Content.empty()) {
255 uint8_t Pad = Content.front();
256 if (Pad >= LF_PAD0) {
257 uint32_t Skip = Pad & 0x0F;
258 Content = Content.drop_front(Skip);
259 Offset += Skip;
260 }
261 }
262 }
263 }
264
handlePointer(ArrayRef<uint8_t> Content,SmallVectorImpl<TiReference> & Refs)265 static void handlePointer(ArrayRef<uint8_t> Content,
266 SmallVectorImpl<TiReference> &Refs) {
267 Refs.push_back({TiRefKind::TypeRef, 0, 1});
268
269 uint32_t Attrs = support::endian::read32le(Content.drop_front(4).data());
270 if (isMemberPointer(Attrs))
271 Refs.push_back({TiRefKind::TypeRef, 8, 1});
272 }
273
discoverTypeIndices(ArrayRef<uint8_t> Content,TypeLeafKind Kind,SmallVectorImpl<TiReference> & Refs)274 static void discoverTypeIndices(ArrayRef<uint8_t> Content, TypeLeafKind Kind,
275 SmallVectorImpl<TiReference> &Refs) {
276 uint32_t Count;
277 // FIXME: In the future it would be nice if we could avoid hardcoding these
278 // values. One idea is to define some structures representing these types
279 // that would allow the use of offsetof().
280 switch (Kind) {
281 case TypeLeafKind::LF_FUNC_ID:
282 Refs.push_back({TiRefKind::IndexRef, 0, 1});
283 Refs.push_back({TiRefKind::TypeRef, 4, 1});
284 break;
285 case TypeLeafKind::LF_MFUNC_ID:
286 Refs.push_back({TiRefKind::TypeRef, 0, 2});
287 break;
288 case TypeLeafKind::LF_STRING_ID:
289 Refs.push_back({TiRefKind::IndexRef, 0, 1});
290 break;
291 case TypeLeafKind::LF_SUBSTR_LIST:
292 Count = support::endian::read32le(Content.data());
293 if (Count > 0)
294 Refs.push_back({TiRefKind::IndexRef, 4, Count});
295 break;
296 case TypeLeafKind::LF_BUILDINFO:
297 Count = support::endian::read16le(Content.data());
298 if (Count > 0)
299 Refs.push_back({TiRefKind::IndexRef, 2, Count});
300 break;
301 case TypeLeafKind::LF_UDT_SRC_LINE:
302 Refs.push_back({TiRefKind::TypeRef, 0, 1});
303 Refs.push_back({TiRefKind::IndexRef, 4, 1});
304 break;
305 case TypeLeafKind::LF_UDT_MOD_SRC_LINE:
306 Refs.push_back({TiRefKind::TypeRef, 0, 1});
307 break;
308 case TypeLeafKind::LF_MODIFIER:
309 Refs.push_back({TiRefKind::TypeRef, 0, 1});
310 break;
311 case TypeLeafKind::LF_PROCEDURE:
312 Refs.push_back({TiRefKind::TypeRef, 0, 1});
313 Refs.push_back({TiRefKind::TypeRef, 8, 1});
314 break;
315 case TypeLeafKind::LF_MFUNCTION:
316 Refs.push_back({TiRefKind::TypeRef, 0, 3});
317 Refs.push_back({TiRefKind::TypeRef, 16, 1});
318 break;
319 case TypeLeafKind::LF_ARGLIST:
320 Count = support::endian::read32le(Content.data());
321 if (Count > 0)
322 Refs.push_back({TiRefKind::TypeRef, 4, Count});
323 break;
324 case TypeLeafKind::LF_ARRAY:
325 Refs.push_back({TiRefKind::TypeRef, 0, 2});
326 break;
327 case TypeLeafKind::LF_CLASS:
328 case TypeLeafKind::LF_STRUCTURE:
329 case TypeLeafKind::LF_INTERFACE:
330 Refs.push_back({TiRefKind::TypeRef, 4, 3});
331 break;
332 case TypeLeafKind::LF_UNION:
333 Refs.push_back({TiRefKind::TypeRef, 4, 1});
334 break;
335 case TypeLeafKind::LF_ENUM:
336 Refs.push_back({TiRefKind::TypeRef, 4, 2});
337 break;
338 case TypeLeafKind::LF_BITFIELD:
339 Refs.push_back({TiRefKind::TypeRef, 0, 1});
340 break;
341 case TypeLeafKind::LF_VFTABLE:
342 Refs.push_back({TiRefKind::TypeRef, 0, 2});
343 break;
344 case TypeLeafKind::LF_VTSHAPE:
345 break;
346 case TypeLeafKind::LF_METHODLIST:
347 handleMethodOverloadList(Content, Refs);
348 break;
349 case TypeLeafKind::LF_FIELDLIST:
350 handleFieldList(Content, Refs);
351 break;
352 case TypeLeafKind::LF_POINTER:
353 handlePointer(Content, Refs);
354 break;
355 default:
356 break;
357 }
358 }
359
discoverTypeIndices(ArrayRef<uint8_t> Content,SymbolKind Kind,SmallVectorImpl<TiReference> & Refs)360 static bool discoverTypeIndices(ArrayRef<uint8_t> Content, SymbolKind Kind,
361 SmallVectorImpl<TiReference> &Refs) {
362 uint32_t Count;
363 // FIXME: In the future it would be nice if we could avoid hardcoding these
364 // values. One idea is to define some structures representing these types
365 // that would allow the use of offsetof().
366 switch (Kind) {
367 case SymbolKind::S_GPROC32:
368 case SymbolKind::S_LPROC32:
369 case SymbolKind::S_GPROC32_ID:
370 case SymbolKind::S_LPROC32_ID:
371 case SymbolKind::S_LPROC32_DPC:
372 case SymbolKind::S_LPROC32_DPC_ID:
373 Refs.push_back({TiRefKind::IndexRef, 24, 1}); // LF_FUNC_ID
374 break;
375 case SymbolKind::S_UDT:
376 Refs.push_back({TiRefKind::TypeRef, 0, 1}); // UDT
377 break;
378 case SymbolKind::S_GDATA32:
379 case SymbolKind::S_LDATA32:
380 Refs.push_back({TiRefKind::TypeRef, 0, 1}); // Type
381 break;
382 case SymbolKind::S_BUILDINFO:
383 Refs.push_back({TiRefKind::IndexRef, 0, 1}); // Compile flags
384 break;
385 case SymbolKind::S_LTHREAD32:
386 case SymbolKind::S_GTHREAD32:
387 Refs.push_back({TiRefKind::TypeRef, 0, 1}); // Type
388 break;
389 case SymbolKind::S_FILESTATIC:
390 Refs.push_back({TiRefKind::TypeRef, 0, 1}); // Type
391 break;
392 case SymbolKind::S_LOCAL:
393 Refs.push_back({TiRefKind::TypeRef, 0, 1}); // Type
394 break;
395 case SymbolKind::S_REGISTER:
396 Refs.push_back({TiRefKind::TypeRef, 0, 1}); // Type
397 break;
398 case SymbolKind::S_CONSTANT:
399 Refs.push_back({TiRefKind::TypeRef, 0, 1}); // Type
400 break;
401 case SymbolKind::S_BPREL32:
402 case SymbolKind::S_REGREL32:
403 Refs.push_back({TiRefKind::TypeRef, 4, 1}); // Type
404 break;
405 case SymbolKind::S_CALLSITEINFO:
406 Refs.push_back({TiRefKind::TypeRef, 8, 1}); // Call signature
407 break;
408 case SymbolKind::S_CALLERS:
409 case SymbolKind::S_CALLEES:
410 case SymbolKind::S_INLINEES:
411 // The record is a count followed by an array of type indices.
412 Count = *reinterpret_cast<const ulittle32_t *>(Content.data());
413 Refs.push_back({TiRefKind::IndexRef, 4, Count}); // Callees
414 break;
415 case SymbolKind::S_INLINESITE:
416 Refs.push_back({TiRefKind::IndexRef, 8, 1}); // ID of inlinee
417 break;
418 case SymbolKind::S_HEAPALLOCSITE:
419 Refs.push_back({TiRefKind::TypeRef, 8, 1}); // UDT allocated
420 break;
421
422 // Defranges don't have types, just registers and code offsets.
423 case SymbolKind::S_DEFRANGE_REGISTER:
424 case SymbolKind::S_DEFRANGE_REGISTER_REL:
425 case SymbolKind::S_DEFRANGE_FRAMEPOINTER_REL:
426 case SymbolKind::S_DEFRANGE_FRAMEPOINTER_REL_FULL_SCOPE:
427 case SymbolKind::S_DEFRANGE_SUBFIELD_REGISTER:
428 case SymbolKind::S_DEFRANGE_SUBFIELD:
429 break;
430
431 // No type references.
432 case SymbolKind::S_LABEL32:
433 case SymbolKind::S_OBJNAME:
434 case SymbolKind::S_COMPILE:
435 case SymbolKind::S_COMPILE2:
436 case SymbolKind::S_COMPILE3:
437 case SymbolKind::S_ENVBLOCK:
438 case SymbolKind::S_BLOCK32:
439 case SymbolKind::S_FRAMEPROC:
440 case SymbolKind::S_THUNK32:
441 case SymbolKind::S_FRAMECOOKIE:
442 case SymbolKind::S_UNAMESPACE:
443 break;
444 // Scope ending symbols.
445 case SymbolKind::S_END:
446 case SymbolKind::S_INLINESITE_END:
447 case SymbolKind::S_PROC_ID_END:
448 break;
449 default:
450 return false; // Unknown symbol.
451 }
452 return true;
453 }
454
discoverTypeIndices(const CVType & Type,SmallVectorImpl<TiReference> & Refs)455 void llvm::codeview::discoverTypeIndices(const CVType &Type,
456 SmallVectorImpl<TiReference> &Refs) {
457 ::discoverTypeIndices(Type.content(), Type.kind(), Refs);
458 }
459
resolveTypeIndexReferences(ArrayRef<uint8_t> RecordData,ArrayRef<TiReference> Refs,SmallVectorImpl<TypeIndex> & Indices)460 static void resolveTypeIndexReferences(ArrayRef<uint8_t> RecordData,
461 ArrayRef<TiReference> Refs,
462 SmallVectorImpl<TypeIndex> &Indices) {
463 Indices.clear();
464
465 if (Refs.empty())
466 return;
467
468 RecordData = RecordData.drop_front(sizeof(RecordPrefix));
469
470 BinaryStreamReader Reader(RecordData, support::little);
471 for (const auto &Ref : Refs) {
472 Reader.setOffset(Ref.Offset);
473 FixedStreamArray<TypeIndex> Run;
474 cantFail(Reader.readArray(Run, Ref.Count));
475 Indices.append(Run.begin(), Run.end());
476 }
477 }
478
discoverTypeIndices(const CVType & Type,SmallVectorImpl<TypeIndex> & Indices)479 void llvm::codeview::discoverTypeIndices(const CVType &Type,
480 SmallVectorImpl<TypeIndex> &Indices) {
481 return discoverTypeIndices(Type.RecordData, Indices);
482 }
483
discoverTypeIndices(ArrayRef<uint8_t> RecordData,SmallVectorImpl<TypeIndex> & Indices)484 void llvm::codeview::discoverTypeIndices(ArrayRef<uint8_t> RecordData,
485 SmallVectorImpl<TypeIndex> &Indices) {
486 SmallVector<TiReference, 4> Refs;
487 discoverTypeIndices(RecordData, Refs);
488 resolveTypeIndexReferences(RecordData, Refs, Indices);
489 }
490
discoverTypeIndices(ArrayRef<uint8_t> RecordData,SmallVectorImpl<TiReference> & Refs)491 void llvm::codeview::discoverTypeIndices(ArrayRef<uint8_t> RecordData,
492 SmallVectorImpl<TiReference> &Refs) {
493 const RecordPrefix *P =
494 reinterpret_cast<const RecordPrefix *>(RecordData.data());
495 TypeLeafKind K = static_cast<TypeLeafKind>(uint16_t(P->RecordKind));
496 ::discoverTypeIndices(RecordData.drop_front(sizeof(RecordPrefix)), K, Refs);
497 }
498
discoverTypeIndicesInSymbol(const CVSymbol & Sym,SmallVectorImpl<TiReference> & Refs)499 bool llvm::codeview::discoverTypeIndicesInSymbol(
500 const CVSymbol &Sym, SmallVectorImpl<TiReference> &Refs) {
501 SymbolKind K = Sym.kind();
502 return ::discoverTypeIndices(Sym.content(), K, Refs);
503 }
504
discoverTypeIndicesInSymbol(ArrayRef<uint8_t> RecordData,SmallVectorImpl<TiReference> & Refs)505 bool llvm::codeview::discoverTypeIndicesInSymbol(
506 ArrayRef<uint8_t> RecordData, SmallVectorImpl<TiReference> &Refs) {
507 const RecordPrefix *P =
508 reinterpret_cast<const RecordPrefix *>(RecordData.data());
509 SymbolKind K = static_cast<SymbolKind>(uint16_t(P->RecordKind));
510 return ::discoverTypeIndices(RecordData.drop_front(sizeof(RecordPrefix)), K,
511 Refs);
512 }
513
discoverTypeIndicesInSymbol(ArrayRef<uint8_t> RecordData,SmallVectorImpl<TypeIndex> & Indices)514 bool llvm::codeview::discoverTypeIndicesInSymbol(
515 ArrayRef<uint8_t> RecordData, SmallVectorImpl<TypeIndex> &Indices) {
516 SmallVector<TiReference, 2> Refs;
517 if (!discoverTypeIndicesInSymbol(RecordData, Refs))
518 return false;
519 resolveTypeIndexReferences(RecordData, Refs, Indices);
520 return true;
521 }
522