mirror of
https://github.com/llvm/llvm-project.git
synced 2025-05-01 03:36:07 +00:00
156 lines
5.6 KiB
C++
156 lines
5.6 KiB
C++
//===- 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();
|
|
}
|