123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155 |
- //===- CVSymbolVisitor.cpp --------------------------------------*- C++ -*-===//
- //
- // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
- // See https://llvm.org/LICENSE.txt for license information.
- // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
- //
- //===----------------------------------------------------------------------===//
- #include "llvm/DebugInfo/CodeView/CVSymbolVisitor.h"
- #include "llvm/DebugInfo/CodeView/CodeView.h"
- #include "llvm/DebugInfo/CodeView/SymbolRecord.h"
- #include "llvm/DebugInfo/CodeView/SymbolRecordHelpers.h"
- #include "llvm/DebugInfo/CodeView/SymbolVisitorCallbacks.h"
- #include "llvm/Support/BinaryStreamArray.h"
- #include "llvm/Support/ErrorHandling.h"
- using namespace llvm;
- using namespace llvm::codeview;
- CVSymbolVisitor::CVSymbolVisitor(SymbolVisitorCallbacks &Callbacks)
- : Callbacks(Callbacks) {}
- template <typename T>
- static Error visitKnownRecord(CVSymbol &Record,
- SymbolVisitorCallbacks &Callbacks) {
- SymbolRecordKind RK = static_cast<SymbolRecordKind>(Record.kind());
- T KnownRecord(RK);
- if (auto EC = Callbacks.visitKnownRecord(Record, KnownRecord))
- return EC;
- return Error::success();
- }
- static Error finishVisitation(CVSymbol &Record,
- SymbolVisitorCallbacks &Callbacks) {
- switch (Record.kind()) {
- default:
- if (auto EC = Callbacks.visitUnknownSymbol(Record))
- return EC;
- break;
- #define SYMBOL_RECORD(EnumName, EnumVal, Name) \
- case EnumName: { \
- if (auto EC = visitKnownRecord<Name>(Record, Callbacks)) \
- return EC; \
- break; \
- }
- #define SYMBOL_RECORD_ALIAS(EnumName, EnumVal, Name, AliasName) \
- SYMBOL_RECORD(EnumVal, EnumVal, AliasName)
- #include "llvm/DebugInfo/CodeView/CodeViewSymbols.def"
- }
- if (auto EC = Callbacks.visitSymbolEnd(Record))
- return EC;
- return Error::success();
- }
- Error CVSymbolVisitor::visitSymbolRecord(CVSymbol &Record) {
- if (auto EC = Callbacks.visitSymbolBegin(Record))
- return EC;
- return finishVisitation(Record, Callbacks);
- }
- Error CVSymbolVisitor::visitSymbolRecord(CVSymbol &Record, uint32_t Offset) {
- if (auto EC = Callbacks.visitSymbolBegin(Record, Offset))
- return EC;
- return finishVisitation(Record, Callbacks);
- }
- Error CVSymbolVisitor::visitSymbolStream(const CVSymbolArray &Symbols) {
- for (auto I : Symbols) {
- if (auto EC = visitSymbolRecord(I))
- return EC;
- }
- return Error::success();
- }
- Error CVSymbolVisitor::visitSymbolStream(const CVSymbolArray &Symbols,
- uint32_t InitialOffset) {
- for (auto I : Symbols) {
- if (auto EC = visitSymbolRecord(I, InitialOffset + Symbols.skew()))
- return EC;
- InitialOffset += I.length();
- }
- return Error::success();
- }
- Error CVSymbolVisitor::visitSymbolStreamFiltered(const CVSymbolArray &Symbols,
- const FilterOptions &Filter) {
- if (!Filter.SymbolOffset)
- return visitSymbolStream(Symbols);
- uint32_t SymbolOffset = *Filter.SymbolOffset;
- uint32_t ParentRecurseDepth = Filter.ParentRecursiveDepth.value_or(0);
- uint32_t ChildrenRecurseDepth = Filter.ChildRecursiveDepth.value_or(0);
- if (!Symbols.isOffsetValid(SymbolOffset))
- return createStringError(inconvertibleErrorCode(), "Invalid symbol offset");
- CVSymbol Sym = *Symbols.at(SymbolOffset);
- uint32_t SymEndOffset =
- symbolOpensScope(Sym.kind()) ? getScopeEndOffset(Sym) : 0;
- std::vector<uint32_t> ParentOffsets;
- std::vector<uint32_t> ParentEndOffsets;
- uint32_t ChildrenDepth = 0;
- for (auto Begin = Symbols.begin(), End = Symbols.end(); Begin != End;
- ++Begin) {
- uint32_t BeginOffset = Begin.offset();
- CVSymbol BeginSym = *Begin;
- if (BeginOffset < SymbolOffset) {
- if (symbolOpensScope(Begin->kind())) {
- uint32_t EndOffset = getScopeEndOffset(BeginSym);
- if (SymbolOffset < EndOffset) {
- ParentOffsets.push_back(BeginOffset);
- ParentEndOffsets.push_back(EndOffset);
- }
- }
- } else if (BeginOffset == SymbolOffset) {
- // Found symbol at offset. Visit its parent up to ParentRecurseDepth.
- if (ParentRecurseDepth >= ParentOffsets.size())
- ParentRecurseDepth = ParentOffsets.size();
- uint32_t StartIndex = ParentOffsets.size() - ParentRecurseDepth;
- while (StartIndex < ParentOffsets.size()) {
- if (!Symbols.isOffsetValid(ParentOffsets[StartIndex]))
- break;
- CVSymbol Parent = *Symbols.at(ParentOffsets[StartIndex]);
- if (auto EC = visitSymbolRecord(Parent, ParentOffsets[StartIndex]))
- return EC;
- ++StartIndex;
- }
- if (auto EC = visitSymbolRecord(Sym, SymbolOffset))
- return EC;
- } else if (BeginOffset <= SymEndOffset) {
- if (ChildrenRecurseDepth) {
- // Visit children.
- if (symbolEndsScope(Begin->kind()))
- --ChildrenDepth;
- if (ChildrenDepth < ChildrenRecurseDepth ||
- BeginOffset == SymEndOffset) {
- if (auto EC = visitSymbolRecord(BeginSym, BeginOffset))
- return EC;
- }
- if (symbolOpensScope(Begin->kind()))
- ++ChildrenDepth;
- }
- } else {
- // Visit parents' ends.
- if (ParentRecurseDepth && BeginOffset == ParentEndOffsets.back()) {
- if (auto EC = visitSymbolRecord(BeginSym, BeginOffset))
- return EC;
- ParentEndOffsets.pop_back();
- --ParentRecurseDepth;
- }
- }
- }
- return Error::success();
- }
|