mirror of
https://github.com/llvm/llvm-project.git
synced 2025-04-26 11:06:06 +00:00

LLVM supports DWARF 5 linetable extension to store source files inline in DWARF. This is particularly useful for compiler-generated source code. This implementation tries to materialize them as temporary files lazily, so SBAPI clients don't need to be aware of them. rdar://110926168
265 lines
7.9 KiB
C++
265 lines
7.9 KiB
C++
//===-- SymbolFileCTF.h -----------------------------------------*- 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
|
|
//
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
#ifndef LLDB_SOURCE_PLUGINS_SYMBOLFILE_CTF_SYMBOLFILECTF_H
|
|
#define LLDB_SOURCE_PLUGINS_SYMBOLFILE_CTF_SYMBOLFILECTF_H
|
|
|
|
#include <map>
|
|
#include <optional>
|
|
#include <vector>
|
|
|
|
#include "CTFTypes.h"
|
|
#include "lldb/Symbol/CompileUnit.h"
|
|
#include "lldb/Symbol/SymbolFile.h"
|
|
|
|
namespace lldb_private {
|
|
|
|
class SymbolFileCTF : public lldb_private::SymbolFileCommon {
|
|
/// LLVM RTTI support.
|
|
static char ID;
|
|
|
|
public:
|
|
/// LLVM RTTI support.
|
|
/// \{
|
|
bool isA(const void *ClassID) const override {
|
|
return ClassID == &ID || SymbolFileCommon::isA(ClassID);
|
|
}
|
|
static bool classof(const SymbolFile *obj) { return obj->isA(&ID); }
|
|
/// \}
|
|
|
|
SymbolFileCTF(lldb::ObjectFileSP objfile_sp);
|
|
|
|
static void Initialize();
|
|
|
|
static void Terminate();
|
|
|
|
static llvm::StringRef GetPluginNameStatic() { return "CTF"; }
|
|
|
|
static llvm::StringRef GetPluginDescriptionStatic();
|
|
|
|
static lldb_private::SymbolFile *
|
|
CreateInstance(lldb::ObjectFileSP objfile_sp);
|
|
|
|
llvm::StringRef GetPluginName() override { return GetPluginNameStatic(); }
|
|
|
|
uint32_t CalculateAbilities() override;
|
|
|
|
void InitializeObject() override;
|
|
|
|
lldb::LanguageType ParseLanguage(CompileUnit &comp_unit) override {
|
|
return lldb::eLanguageTypeUnknown;
|
|
}
|
|
|
|
bool ParseHeader();
|
|
|
|
size_t ParseFunctions(CompileUnit &comp_unit) override;
|
|
|
|
size_t ParseObjects(CompileUnit &comp_unit);
|
|
|
|
bool ParseLineTable(CompileUnit &comp_unit) override { return false; }
|
|
|
|
bool ParseDebugMacros(CompileUnit &comp_unit) override { return false; }
|
|
|
|
bool ParseSupportFiles(CompileUnit &comp_unit,
|
|
SupportFileList &support_files) override {
|
|
return false;
|
|
}
|
|
|
|
size_t ParseTypes(CompileUnit &cu) override;
|
|
|
|
bool ParseImportedModules(
|
|
const SymbolContext &sc,
|
|
std::vector<lldb_private::SourceModule> &imported_modules) override {
|
|
return false;
|
|
}
|
|
|
|
size_t ParseBlocksRecursive(Function &func) override { return 0; }
|
|
|
|
size_t ParseVariablesForContext(const SymbolContext &sc) override;
|
|
|
|
uint32_t CalculateNumCompileUnits() override { return 0; }
|
|
|
|
lldb::CompUnitSP ParseCompileUnitAtIndex(uint32_t index) override;
|
|
|
|
Type *ResolveTypeUID(lldb::user_id_t type_uid) override;
|
|
std::optional<ArrayInfo> GetDynamicArrayInfoForUID(
|
|
lldb::user_id_t type_uid,
|
|
const lldb_private::ExecutionContext *exe_ctx) override {
|
|
return std::nullopt;
|
|
}
|
|
|
|
bool CompleteType(CompilerType &compiler_type) override;
|
|
|
|
uint32_t ResolveSymbolContext(const lldb_private::Address &so_addr,
|
|
lldb::SymbolContextItem resolve_scope,
|
|
lldb_private::SymbolContext &sc) override;
|
|
|
|
void AddSymbols(Symtab &symtab) override;
|
|
|
|
void GetTypes(lldb_private::SymbolContextScope *sc_scope,
|
|
lldb::TypeClass type_mask,
|
|
lldb_private::TypeList &type_list) override {}
|
|
|
|
void FindTypes(const lldb_private::TypeQuery &match,
|
|
lldb_private::TypeResults &results) override;
|
|
|
|
void FindTypesByRegex(const lldb_private::RegularExpression ®ex,
|
|
uint32_t max_matches, lldb_private::TypeMap &types);
|
|
|
|
void FindFunctions(const lldb_private::Module::LookupInfo &lookup_info,
|
|
const lldb_private::CompilerDeclContext &parent_decl_ctx,
|
|
bool include_inlines,
|
|
lldb_private::SymbolContextList &sc_list) override;
|
|
|
|
void FindFunctions(const lldb_private::RegularExpression ®ex,
|
|
bool include_inlines,
|
|
lldb_private::SymbolContextList &sc_list) override;
|
|
|
|
void
|
|
FindGlobalVariables(lldb_private::ConstString name,
|
|
const lldb_private::CompilerDeclContext &parent_decl_ctx,
|
|
uint32_t max_matches,
|
|
lldb_private::VariableList &variables) override;
|
|
|
|
void FindGlobalVariables(const lldb_private::RegularExpression ®ex,
|
|
uint32_t max_matches,
|
|
lldb_private::VariableList &variables) override;
|
|
|
|
enum TypeKind : uint32_t {
|
|
eUnknown = 0,
|
|
eInteger = 1,
|
|
eFloat = 2,
|
|
ePointer = 3,
|
|
eArray = 4,
|
|
eFunction = 5,
|
|
eStruct = 6,
|
|
eUnion = 7,
|
|
eEnum = 8,
|
|
eForward = 9,
|
|
eTypedef = 10,
|
|
eVolatile = 11,
|
|
eConst = 12,
|
|
eRestrict = 13,
|
|
eSlice = 14,
|
|
};
|
|
|
|
private:
|
|
enum Flags : uint32_t {
|
|
eFlagCompress = (1u << 0),
|
|
eFlagNewFuncInfo = (1u << 1),
|
|
eFlagIdxSorted = (1u << 2),
|
|
eFlagDynStr = (1u << 3),
|
|
};
|
|
|
|
enum IntEncoding : uint32_t {
|
|
eSigned = 0x1,
|
|
eChar = 0x2,
|
|
eBool = 0x4,
|
|
eVarArgs = 0x8,
|
|
};
|
|
|
|
struct ctf_preamble_t {
|
|
uint16_t magic;
|
|
uint8_t version;
|
|
uint8_t flags;
|
|
};
|
|
|
|
struct ctf_header_t {
|
|
ctf_preamble_t preamble;
|
|
uint32_t parlabel;
|
|
uint32_t parname;
|
|
uint32_t lbloff;
|
|
uint32_t objtoff;
|
|
uint32_t funcoff;
|
|
uint32_t typeoff;
|
|
uint32_t stroff;
|
|
uint32_t strlen;
|
|
};
|
|
|
|
struct ctf_type_t {
|
|
uint32_t name;
|
|
uint32_t info;
|
|
union {
|
|
uint32_t size;
|
|
uint32_t type;
|
|
};
|
|
uint32_t lsizehi;
|
|
uint32_t lsizelo;
|
|
};
|
|
|
|
struct ctf_stype_t {
|
|
uint32_t name;
|
|
uint32_t info;
|
|
union {
|
|
uint32_t size;
|
|
uint32_t type;
|
|
};
|
|
|
|
bool IsLargeType() const { return size == 0xffff; }
|
|
uint32_t GetStructSize() const {
|
|
if (IsLargeType())
|
|
return sizeof(ctf_type_t);
|
|
return sizeof(ctf_stype_t);
|
|
}
|
|
uint32_t GetType() const { return type; }
|
|
uint32_t GetSize() const { return size; }
|
|
};
|
|
|
|
llvm::Expected<std::unique_ptr<CTFType>> ParseType(lldb::offset_t &offset,
|
|
lldb::user_id_t uid);
|
|
|
|
llvm::Expected<lldb::TypeSP> CreateType(CTFType *ctf_type);
|
|
llvm::Expected<lldb::TypeSP> CreateInteger(const CTFInteger &ctf_integer);
|
|
llvm::Expected<lldb::TypeSP> CreateModifier(const CTFModifier &ctf_modifier);
|
|
llvm::Expected<lldb::TypeSP> CreateTypedef(const CTFTypedef &ctf_typedef);
|
|
llvm::Expected<lldb::TypeSP> CreateArray(const CTFArray &ctf_array);
|
|
llvm::Expected<lldb::TypeSP> CreateEnum(const CTFEnum &ctf_enum);
|
|
llvm::Expected<lldb::TypeSP> CreateFunction(const CTFFunction &ctf_function);
|
|
llvm::Expected<lldb::TypeSP> CreateRecord(const CTFRecord &ctf_record);
|
|
llvm::Expected<lldb::TypeSP> CreateForward(const CTFForward &ctf_forward);
|
|
|
|
llvm::StringRef ReadString(lldb::offset_t offset) const;
|
|
|
|
std::vector<uint16_t> GetFieldSizes(lldb::offset_t field_offset,
|
|
uint32_t fields, uint32_t struct_size);
|
|
|
|
DataExtractor m_data;
|
|
|
|
/// The start offset of the CTF body into m_data. If the body is uncompressed,
|
|
/// m_data contains the header and the body and the body starts after the
|
|
/// header. If the body is compressed, m_data only contains the body and the
|
|
/// offset is zero.
|
|
lldb::offset_t m_body_offset = 0;
|
|
|
|
TypeSystemClang *m_ast;
|
|
lldb::CompUnitSP m_comp_unit_sp;
|
|
|
|
std::optional<ctf_header_t> m_header;
|
|
|
|
/// Parsed CTF types.
|
|
llvm::DenseMap<lldb::user_id_t, std::unique_ptr<CTFType>> m_ctf_types;
|
|
|
|
/// Parsed LLDB types.
|
|
llvm::DenseMap<lldb::user_id_t, lldb::TypeSP> m_types;
|
|
|
|
/// To complete types, we need a way to map (imcomplete) compiler types back
|
|
/// to parsed CTF types.
|
|
llvm::DenseMap<lldb::opaque_compiler_type_t, const CTFType *>
|
|
m_compiler_types;
|
|
|
|
std::vector<lldb::FunctionSP> m_functions;
|
|
std::vector<lldb::VariableSP> m_variables;
|
|
|
|
static constexpr uint16_t g_ctf_magic = 0xcff1;
|
|
static constexpr uint8_t g_ctf_version = 4;
|
|
static constexpr uint16_t g_ctf_field_threshold = 0x2000;
|
|
};
|
|
} // namespace lldb_private
|
|
|
|
#endif // LLDB_SOURCE_PLUGINS_SYMBOLFILE_CTF_SYMBOLFILECTF_H
|