xref: /src/contrib/llvm-project/llvm/lib/DebugInfo/CodeView/CVSymbolVisitor.cpp (revision 81ad626541db97eb356e2c1d4a20eb2a26a766ab) !
1b915e9e0SDimitry Andric //===- CVSymbolVisitor.cpp --------------------------------------*- C++ -*-===//
2b915e9e0SDimitry Andric //
3e6d15924SDimitry Andric // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
4e6d15924SDimitry Andric // See https://llvm.org/LICENSE.txt for license information.
5e6d15924SDimitry Andric // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
6b915e9e0SDimitry Andric //
7b915e9e0SDimitry Andric //===----------------------------------------------------------------------===//
8b915e9e0SDimitry Andric 
9b915e9e0SDimitry Andric #include "llvm/DebugInfo/CodeView/CVSymbolVisitor.h"
10b915e9e0SDimitry Andric 
11145449b1SDimitry Andric #include "llvm/DebugInfo/CodeView/CodeView.h"
12145449b1SDimitry Andric #include "llvm/DebugInfo/CodeView/SymbolRecord.h"
13145449b1SDimitry Andric #include "llvm/DebugInfo/CodeView/SymbolRecordHelpers.h"
14b915e9e0SDimitry Andric #include "llvm/DebugInfo/CodeView/SymbolVisitorCallbacks.h"
15145449b1SDimitry Andric #include "llvm/Support/BinaryStreamArray.h"
16145449b1SDimitry Andric #include "llvm/Support/ErrorHandling.h"
17b915e9e0SDimitry Andric 
18b915e9e0SDimitry Andric using namespace llvm;
19b915e9e0SDimitry Andric using namespace llvm::codeview;
20b915e9e0SDimitry Andric 
CVSymbolVisitor(SymbolVisitorCallbacks & Callbacks)21b915e9e0SDimitry Andric CVSymbolVisitor::CVSymbolVisitor(SymbolVisitorCallbacks &Callbacks)
22b915e9e0SDimitry Andric     : Callbacks(Callbacks) {}
23b915e9e0SDimitry Andric 
24b915e9e0SDimitry Andric template <typename T>
visitKnownRecord(CVSymbol & Record,SymbolVisitorCallbacks & Callbacks)25b915e9e0SDimitry Andric static Error visitKnownRecord(CVSymbol &Record,
26b915e9e0SDimitry Andric                               SymbolVisitorCallbacks &Callbacks) {
27e6d15924SDimitry Andric   SymbolRecordKind RK = static_cast<SymbolRecordKind>(Record.kind());
28b915e9e0SDimitry Andric   T KnownRecord(RK);
29b915e9e0SDimitry Andric   if (auto EC = Callbacks.visitKnownRecord(Record, KnownRecord))
30b915e9e0SDimitry Andric     return EC;
31b915e9e0SDimitry Andric   return Error::success();
32b915e9e0SDimitry Andric }
33b915e9e0SDimitry Andric 
finishVisitation(CVSymbol & Record,SymbolVisitorCallbacks & Callbacks)349df3605dSDimitry Andric static Error finishVisitation(CVSymbol &Record,
359df3605dSDimitry Andric                               SymbolVisitorCallbacks &Callbacks) {
36e6d15924SDimitry Andric   switch (Record.kind()) {
37b915e9e0SDimitry Andric   default:
38b915e9e0SDimitry Andric     if (auto EC = Callbacks.visitUnknownSymbol(Record))
39b915e9e0SDimitry Andric       return EC;
40b915e9e0SDimitry Andric     break;
41b915e9e0SDimitry Andric #define SYMBOL_RECORD(EnumName, EnumVal, Name)                                 \
42b915e9e0SDimitry Andric   case EnumName: {                                                             \
43b915e9e0SDimitry Andric     if (auto EC = visitKnownRecord<Name>(Record, Callbacks))                   \
44b915e9e0SDimitry Andric       return EC;                                                               \
45b915e9e0SDimitry Andric     break;                                                                     \
46b915e9e0SDimitry Andric   }
47b915e9e0SDimitry Andric #define SYMBOL_RECORD_ALIAS(EnumName, EnumVal, Name, AliasName)                \
48b915e9e0SDimitry Andric   SYMBOL_RECORD(EnumVal, EnumVal, AliasName)
49f382538dSDimitry Andric #include "llvm/DebugInfo/CodeView/CodeViewSymbols.def"
50b915e9e0SDimitry Andric   }
51b915e9e0SDimitry Andric 
52b915e9e0SDimitry Andric   if (auto EC = Callbacks.visitSymbolEnd(Record))
53b915e9e0SDimitry Andric     return EC;
54b915e9e0SDimitry Andric 
55b915e9e0SDimitry Andric   return Error::success();
56b915e9e0SDimitry Andric }
57b915e9e0SDimitry Andric 
visitSymbolRecord(CVSymbol & Record)589df3605dSDimitry Andric Error CVSymbolVisitor::visitSymbolRecord(CVSymbol &Record) {
599df3605dSDimitry Andric   if (auto EC = Callbacks.visitSymbolBegin(Record))
609df3605dSDimitry Andric     return EC;
619df3605dSDimitry Andric   return finishVisitation(Record, Callbacks);
629df3605dSDimitry Andric }
639df3605dSDimitry Andric 
visitSymbolRecord(CVSymbol & Record,uint32_t Offset)649df3605dSDimitry Andric Error CVSymbolVisitor::visitSymbolRecord(CVSymbol &Record, uint32_t Offset) {
659df3605dSDimitry Andric   if (auto EC = Callbacks.visitSymbolBegin(Record, Offset))
669df3605dSDimitry Andric     return EC;
679df3605dSDimitry Andric   return finishVisitation(Record, Callbacks);
689df3605dSDimitry Andric }
699df3605dSDimitry Andric 
visitSymbolStream(const CVSymbolArray & Symbols)70b915e9e0SDimitry Andric Error CVSymbolVisitor::visitSymbolStream(const CVSymbolArray &Symbols) {
71b915e9e0SDimitry Andric   for (auto I : Symbols) {
72b915e9e0SDimitry Andric     if (auto EC = visitSymbolRecord(I))
73b915e9e0SDimitry Andric       return EC;
74b915e9e0SDimitry Andric   }
75b915e9e0SDimitry Andric   return Error::success();
76b915e9e0SDimitry Andric }
779df3605dSDimitry Andric 
visitSymbolStream(const CVSymbolArray & Symbols,uint32_t InitialOffset)789df3605dSDimitry Andric Error CVSymbolVisitor::visitSymbolStream(const CVSymbolArray &Symbols,
799df3605dSDimitry Andric                                          uint32_t InitialOffset) {
809df3605dSDimitry Andric   for (auto I : Symbols) {
81d8e91e46SDimitry Andric     if (auto EC = visitSymbolRecord(I, InitialOffset + Symbols.skew()))
829df3605dSDimitry Andric       return EC;
839df3605dSDimitry Andric     InitialOffset += I.length();
849df3605dSDimitry Andric   }
859df3605dSDimitry Andric   return Error::success();
869df3605dSDimitry Andric }
87145449b1SDimitry Andric 
visitSymbolStreamFiltered(const CVSymbolArray & Symbols,const FilterOptions & Filter)88145449b1SDimitry Andric Error CVSymbolVisitor::visitSymbolStreamFiltered(const CVSymbolArray &Symbols,
89145449b1SDimitry Andric                                                  const FilterOptions &Filter) {
90145449b1SDimitry Andric   if (!Filter.SymbolOffset)
91145449b1SDimitry Andric     return visitSymbolStream(Symbols);
92145449b1SDimitry Andric   uint32_t SymbolOffset = *Filter.SymbolOffset;
93145449b1SDimitry Andric   uint32_t ParentRecurseDepth = Filter.ParentRecursiveDepth.value_or(0);
94145449b1SDimitry Andric   uint32_t ChildrenRecurseDepth = Filter.ChildRecursiveDepth.value_or(0);
95145449b1SDimitry Andric   if (!Symbols.isOffsetValid(SymbolOffset))
96145449b1SDimitry Andric     return createStringError(inconvertibleErrorCode(), "Invalid symbol offset");
97145449b1SDimitry Andric   CVSymbol Sym = *Symbols.at(SymbolOffset);
98145449b1SDimitry Andric   uint32_t SymEndOffset =
99145449b1SDimitry Andric       symbolOpensScope(Sym.kind()) ? getScopeEndOffset(Sym) : 0;
100145449b1SDimitry Andric 
101145449b1SDimitry Andric   std::vector<uint32_t> ParentOffsets;
102145449b1SDimitry Andric   std::vector<uint32_t> ParentEndOffsets;
103145449b1SDimitry Andric   uint32_t ChildrenDepth = 0;
104145449b1SDimitry Andric   for (auto Begin = Symbols.begin(), End = Symbols.end(); Begin != End;
105145449b1SDimitry Andric        ++Begin) {
106145449b1SDimitry Andric     uint32_t BeginOffset = Begin.offset();
107145449b1SDimitry Andric     CVSymbol BeginSym = *Begin;
108145449b1SDimitry Andric     if (BeginOffset < SymbolOffset) {
109145449b1SDimitry Andric       if (symbolOpensScope(Begin->kind())) {
110145449b1SDimitry Andric         uint32_t EndOffset = getScopeEndOffset(BeginSym);
111145449b1SDimitry Andric         if (SymbolOffset < EndOffset) {
112145449b1SDimitry Andric           ParentOffsets.push_back(BeginOffset);
113145449b1SDimitry Andric           ParentEndOffsets.push_back(EndOffset);
114145449b1SDimitry Andric         }
115145449b1SDimitry Andric       }
116145449b1SDimitry Andric     } else if (BeginOffset == SymbolOffset) {
117145449b1SDimitry Andric       // Found symbol at offset. Visit its parent up to ParentRecurseDepth.
118145449b1SDimitry Andric       if (ParentRecurseDepth >= ParentOffsets.size())
119145449b1SDimitry Andric         ParentRecurseDepth = ParentOffsets.size();
120145449b1SDimitry Andric       uint32_t StartIndex = ParentOffsets.size() - ParentRecurseDepth;
121145449b1SDimitry Andric       while (StartIndex < ParentOffsets.size()) {
122145449b1SDimitry Andric         if (!Symbols.isOffsetValid(ParentOffsets[StartIndex]))
123145449b1SDimitry Andric           break;
124145449b1SDimitry Andric         CVSymbol Parent = *Symbols.at(ParentOffsets[StartIndex]);
125145449b1SDimitry Andric         if (auto EC = visitSymbolRecord(Parent, ParentOffsets[StartIndex]))
126145449b1SDimitry Andric           return EC;
127145449b1SDimitry Andric         ++StartIndex;
128145449b1SDimitry Andric       }
129145449b1SDimitry Andric       if (auto EC = visitSymbolRecord(Sym, SymbolOffset))
130145449b1SDimitry Andric         return EC;
131145449b1SDimitry Andric     } else if (BeginOffset <= SymEndOffset) {
132145449b1SDimitry Andric       if (ChildrenRecurseDepth) {
133145449b1SDimitry Andric         // Visit children.
134145449b1SDimitry Andric         if (symbolEndsScope(Begin->kind()))
135145449b1SDimitry Andric           --ChildrenDepth;
136145449b1SDimitry Andric         if (ChildrenDepth < ChildrenRecurseDepth ||
137145449b1SDimitry Andric             BeginOffset == SymEndOffset) {
138145449b1SDimitry Andric           if (auto EC = visitSymbolRecord(BeginSym, BeginOffset))
139145449b1SDimitry Andric             return EC;
140145449b1SDimitry Andric         }
141145449b1SDimitry Andric         if (symbolOpensScope(Begin->kind()))
142145449b1SDimitry Andric           ++ChildrenDepth;
143145449b1SDimitry Andric       }
144145449b1SDimitry Andric     } else {
145145449b1SDimitry Andric       // Visit parents' ends.
146145449b1SDimitry Andric       if (ParentRecurseDepth && BeginOffset == ParentEndOffsets.back()) {
147145449b1SDimitry Andric         if (auto EC = visitSymbolRecord(BeginSym, BeginOffset))
148145449b1SDimitry Andric           return EC;
149145449b1SDimitry Andric         ParentEndOffsets.pop_back();
150145449b1SDimitry Andric         --ParentRecurseDepth;
151145449b1SDimitry Andric       }
152145449b1SDimitry Andric     }
153145449b1SDimitry Andric   }
154145449b1SDimitry Andric   return Error::success();
155145449b1SDimitry Andric }
156