2017-02-28 07:43:14 +08:00
|
|
|
//===- DWARFDie.cpp -------------------------------------------------------===//
|
2016-12-14 02:25:19 +08:00
|
|
|
//
|
2019-01-19 16:50:56 +08:00
|
|
|
// 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
|
2016-12-14 02:25:19 +08:00
|
|
|
//
|
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
|
2017-06-06 19:49:48 +08:00
|
|
|
#include "llvm/DebugInfo/DWARF/DWARFDie.h"
|
2017-02-28 07:43:14 +08:00
|
|
|
#include "llvm/ADT/None.h"
|
|
|
|
#include "llvm/ADT/Optional.h"
|
2018-05-01 01:02:41 +08:00
|
|
|
#include "llvm/ADT/SmallSet.h"
|
2017-02-28 07:43:14 +08:00
|
|
|
#include "llvm/ADT/StringRef.h"
|
2017-06-07 11:48:56 +08:00
|
|
|
#include "llvm/BinaryFormat/Dwarf.h"
|
2017-02-28 07:43:14 +08:00
|
|
|
#include "llvm/DebugInfo/DWARF/DWARFAbbreviationDeclaration.h"
|
2016-12-14 02:25:19 +08:00
|
|
|
#include "llvm/DebugInfo/DWARF/DWARFContext.h"
|
2022-02-14 23:27:04 +08:00
|
|
|
#include "llvm/DebugInfo/DWARF/DWARFDebugLine.h"
|
|
|
|
#include "llvm/DebugInfo/DWARF/DWARFDebugLoc.h"
|
[dwarfdump] Pretty print location expressions and location lists
Summary:
Based on Fred's patch here: https://reviews.llvm.org/D6771
I can't seem to commandeer the old review, so I'm creating a new one.
With that change the locations exrpessions are pretty printed inline in the
DIE tree. The output looks like this for debug_loc entries:
DW_AT_location [DW_FORM_data4] (0x00000000
0x0000000000000001 - 0x000000000000000b: DW_OP_consts +3
0x000000000000000b - 0x0000000000000012: DW_OP_consts +7
0x0000000000000012 - 0x000000000000001b: DW_OP_reg0 RAX, DW_OP_piece 0x4
0x000000000000001b - 0x0000000000000024: DW_OP_breg5 RDI+0)
And like this for debug_loc.dwo entries:
DW_AT_location [DW_FORM_sec_offset] (0x00000000
Addr idx 2 (w/ length 190): DW_OP_consts +0, DW_OP_stack_value
Addr idx 3 (w/ length 23): DW_OP_reg0 RAX, DW_OP_piece 0x4)
Simple locations without ranges are printed inline:
DW_AT_location [DW_FORM_block1] (DW_OP_reg4 RSI, DW_OP_piece 0x4, DW_OP_bit_piece 0x20 0x0)
The debug_loc(.dwo) dumping in changed accordingly to factor the code.
Reviewers: dblaikie, aprantl, friss
Subscribers: mgorny, javed.absar, hiraditya, llvm-commits, JDevlieghere
Differential Revision: https://reviews.llvm.org/D37123
llvm-svn: 312042
2017-08-30 05:41:21 +08:00
|
|
|
#include "llvm/DebugInfo/DWARF/DWARFExpression.h"
|
2016-12-14 02:25:19 +08:00
|
|
|
#include "llvm/DebugInfo/DWARF/DWARFFormValue.h"
|
2022-02-14 23:27:04 +08:00
|
|
|
#include "llvm/DebugInfo/DWARF/DWARFTypeUnit.h"
|
2017-02-28 07:43:14 +08:00
|
|
|
#include "llvm/DebugInfo/DWARF/DWARFUnit.h"
|
2017-08-15 20:32:54 +08:00
|
|
|
#include "llvm/Object/ObjectFile.h"
|
2017-02-28 07:43:14 +08:00
|
|
|
#include "llvm/Support/DataExtractor.h"
|
2016-12-14 02:25:19 +08:00
|
|
|
#include "llvm/Support/Format.h"
|
[dwarf] Unify unknown dwarf enum formatting code
Summary:
We have had at least three pieces of code (in DWARFAbbreviationDeclaration,
DWARFAcceleratorTable and DWARFDie) that have hand-rolled support for
dumping unknown dwarf enum values. While not terrible, they are a bit
distracting and enable small differences to creep in (Unknown_ffff vs.
Unknown_0xffff). I ended up needing to add a fourth place
(DWARFVerifier), so it seems it would be a good time to centralize.
This patch creates an alternative to the XXXString dumping functions in
the BinaryFormat library, which formats an unknown value as
DW_TYPE_unknown_1234, instead of just an empty string. It is based on
the formatv function, as that allows us to avoid materializing the
string for unknown values (and because this way I don't have to invent a
name for the new functions :P).
In this patch I add formatters for dwarf attributes, forms, tags, and
index attributes as these are the ones in use currently, but adding
other enums is straight-forward.
Reviewers: dblaikie, JDevlieghere, aprantl
Subscribers: llvm-commits
Differential Revision: https://reviews.llvm.org/D44570
llvm-svn: 328090
2018-03-21 19:46:37 +08:00
|
|
|
#include "llvm/Support/FormatVariadic.h"
|
2017-02-28 07:43:14 +08:00
|
|
|
#include "llvm/Support/MathExtras.h"
|
2022-02-11 01:39:06 +08:00
|
|
|
#include "llvm/Support/ScopedPrinter.h"
|
2018-03-09 17:56:24 +08:00
|
|
|
#include "llvm/Support/WithColor.h"
|
2016-12-14 02:25:19 +08:00
|
|
|
#include "llvm/Support/raw_ostream.h"
|
2017-02-28 07:43:14 +08:00
|
|
|
#include <cassert>
|
|
|
|
#include <cinttypes>
|
|
|
|
#include <cstdint>
|
|
|
|
#include <string>
|
|
|
|
#include <utility>
|
2016-12-14 02:25:19 +08:00
|
|
|
|
|
|
|
using namespace llvm;
|
|
|
|
using namespace dwarf;
|
2017-08-15 20:32:54 +08:00
|
|
|
using namespace object;
|
2016-12-14 02:25:19 +08:00
|
|
|
|
2017-02-28 07:43:14 +08:00
|
|
|
static void dumpApplePropertyAttribute(raw_ostream &OS, uint64_t Val) {
|
2016-12-14 02:25:19 +08:00
|
|
|
OS << " (";
|
|
|
|
do {
|
|
|
|
uint64_t Shift = countTrailingZeros(Val);
|
|
|
|
assert(Shift < 64 && "undefined behavior");
|
|
|
|
uint64_t Bit = 1ULL << Shift;
|
|
|
|
auto PropName = ApplePropertyString(Bit);
|
|
|
|
if (!PropName.empty())
|
|
|
|
OS << PropName;
|
|
|
|
else
|
|
|
|
OS << format("DW_APPLE_PROPERTY_0x%" PRIx64, Bit);
|
|
|
|
if (!(Val ^= Bit))
|
|
|
|
break;
|
|
|
|
OS << ", ";
|
|
|
|
} while (true);
|
|
|
|
OS << ")";
|
|
|
|
}
|
|
|
|
|
2017-08-15 20:32:54 +08:00
|
|
|
static void dumpRanges(const DWARFObject &Obj, raw_ostream &OS,
|
|
|
|
const DWARFAddressRangesVector &Ranges,
|
|
|
|
unsigned AddressSize, unsigned Indent,
|
|
|
|
const DIDumpOptions &DumpOpts) {
|
2018-10-20 01:57:53 +08:00
|
|
|
if (!DumpOpts.ShowAddresses)
|
|
|
|
return;
|
|
|
|
|
2018-01-16 19:17:57 +08:00
|
|
|
for (const DWARFAddressRange &R : Ranges) {
|
2016-12-14 02:25:19 +08:00
|
|
|
OS << '\n';
|
|
|
|
OS.indent(Indent);
|
Re-commit "DWARF location lists: Add section index dumping"
This reapplies c0f6ad7d1f3ccb9d0b9ce9ef8dfa06409ccf1b3e with an
additional fix in test/DebugInfo/X86/constant-loclist.ll, which had a
slightly different output on windows targets. The test now accounts for
this difference.
The original commit message follows.
Summary:
As discussed in D70081, this adds the ability to dump section
names/indices to the location list dumper. It does this by moving the
range specific logic from DWARFDie.cpp:dumpRanges into the
DWARFAddressRange class.
The trickiest part of this patch is the backflip in the meanings of the
two dump flags for the location list sections.
The dumping of "raw" location list data is now controlled by
"DisplayRawContents" flag. This frees up the "Verbose" flag to be used
to control whether we print the section index. Additionally, the
DisplayRawContents flag is set for section-based dumps whenever the
--verbose option is passed, but this is not done for the "inline" dumps.
Also note that the index dumping currently does not work for the DWARF
v5 location lists, as the parser does not fill out the appropriate
fields. This will be done in a separate patch.
Reviewers: dblaikie, probinson, JDevlieghere, SouraVX
Subscribers: sdardis, hiraditya, jrtc27, atanasyan, arphaman, aprantl, llvm-commits
Tags: #llvm
Differential Revision: https://reviews.llvm.org/D70227
2019-11-14 18:22:00 +08:00
|
|
|
R.dump(OS, AddressSize, DumpOpts, &Obj);
|
2016-12-14 02:25:19 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-07-27 15:28:59 +08:00
|
|
|
static void dumpLocationList(raw_ostream &OS, const DWARFFormValue &FormValue,
|
|
|
|
DWARFUnit *U, unsigned Indent,
|
|
|
|
DIDumpOptions DumpOpts) {
|
|
|
|
assert(FormValue.isFormClass(DWARFFormValue::FC_SectionOffset) &&
|
|
|
|
"bad FORM for location list");
|
[dwarfdump] Pretty print location expressions and location lists
Summary:
Based on Fred's patch here: https://reviews.llvm.org/D6771
I can't seem to commandeer the old review, so I'm creating a new one.
With that change the locations exrpessions are pretty printed inline in the
DIE tree. The output looks like this for debug_loc entries:
DW_AT_location [DW_FORM_data4] (0x00000000
0x0000000000000001 - 0x000000000000000b: DW_OP_consts +3
0x000000000000000b - 0x0000000000000012: DW_OP_consts +7
0x0000000000000012 - 0x000000000000001b: DW_OP_reg0 RAX, DW_OP_piece 0x4
0x000000000000001b - 0x0000000000000024: DW_OP_breg5 RDI+0)
And like this for debug_loc.dwo entries:
DW_AT_location [DW_FORM_sec_offset] (0x00000000
Addr idx 2 (w/ length 190): DW_OP_consts +0, DW_OP_stack_value
Addr idx 3 (w/ length 23): DW_OP_reg0 RAX, DW_OP_piece 0x4)
Simple locations without ranges are printed inline:
DW_AT_location [DW_FORM_block1] (DW_OP_reg4 RSI, DW_OP_piece 0x4, DW_OP_bit_piece 0x20 0x0)
The debug_loc(.dwo) dumping in changed accordingly to factor the code.
Reviewers: dblaikie, aprantl, friss
Subscribers: mgorny, javed.absar, hiraditya, llvm-commits, JDevlieghere
Differential Revision: https://reviews.llvm.org/D37123
llvm-svn: 312042
2017-08-30 05:41:21 +08:00
|
|
|
DWARFContext &Ctx = U->getContext();
|
|
|
|
const MCRegisterInfo *MRI = Ctx.getRegisterInfo();
|
2021-07-27 15:28:59 +08:00
|
|
|
uint64_t Offset = *FormValue.getAsSectionOffset();
|
[dwarfdump] Pretty print location expressions and location lists
Summary:
Based on Fred's patch here: https://reviews.llvm.org/D6771
I can't seem to commandeer the old review, so I'm creating a new one.
With that change the locations exrpessions are pretty printed inline in the
DIE tree. The output looks like this for debug_loc entries:
DW_AT_location [DW_FORM_data4] (0x00000000
0x0000000000000001 - 0x000000000000000b: DW_OP_consts +3
0x000000000000000b - 0x0000000000000012: DW_OP_consts +7
0x0000000000000012 - 0x000000000000001b: DW_OP_reg0 RAX, DW_OP_piece 0x4
0x000000000000001b - 0x0000000000000024: DW_OP_breg5 RDI+0)
And like this for debug_loc.dwo entries:
DW_AT_location [DW_FORM_sec_offset] (0x00000000
Addr idx 2 (w/ length 190): DW_OP_consts +0, DW_OP_stack_value
Addr idx 3 (w/ length 23): DW_OP_reg0 RAX, DW_OP_piece 0x4)
Simple locations without ranges are printed inline:
DW_AT_location [DW_FORM_block1] (DW_OP_reg4 RSI, DW_OP_piece 0x4, DW_OP_bit_piece 0x20 0x0)
The debug_loc(.dwo) dumping in changed accordingly to factor the code.
Reviewers: dblaikie, aprantl, friss
Subscribers: mgorny, javed.absar, hiraditya, llvm-commits, JDevlieghere
Differential Revision: https://reviews.llvm.org/D37123
llvm-svn: 312042
2017-08-30 05:41:21 +08:00
|
|
|
|
2021-07-27 15:28:59 +08:00
|
|
|
if (FormValue.getForm() == DW_FORM_loclistx) {
|
|
|
|
FormValue.dump(OS, DumpOpts);
|
2019-11-23 22:34:36 +08:00
|
|
|
|
2021-07-27 15:28:59 +08:00
|
|
|
if (auto LoclistOffset = U->getLoclistOffset(Offset))
|
|
|
|
Offset = *LoclistOffset;
|
|
|
|
else
|
|
|
|
return;
|
[dwarfdump] Pretty print location expressions and location lists
Summary:
Based on Fred's patch here: https://reviews.llvm.org/D6771
I can't seem to commandeer the old review, so I'm creating a new one.
With that change the locations exrpessions are pretty printed inline in the
DIE tree. The output looks like this for debug_loc entries:
DW_AT_location [DW_FORM_data4] (0x00000000
0x0000000000000001 - 0x000000000000000b: DW_OP_consts +3
0x000000000000000b - 0x0000000000000012: DW_OP_consts +7
0x0000000000000012 - 0x000000000000001b: DW_OP_reg0 RAX, DW_OP_piece 0x4
0x000000000000001b - 0x0000000000000024: DW_OP_breg5 RDI+0)
And like this for debug_loc.dwo entries:
DW_AT_location [DW_FORM_sec_offset] (0x00000000
Addr idx 2 (w/ length 190): DW_OP_consts +0, DW_OP_stack_value
Addr idx 3 (w/ length 23): DW_OP_reg0 RAX, DW_OP_piece 0x4)
Simple locations without ranges are printed inline:
DW_AT_location [DW_FORM_block1] (DW_OP_reg4 RSI, DW_OP_piece 0x4, DW_OP_bit_piece 0x20 0x0)
The debug_loc(.dwo) dumping in changed accordingly to factor the code.
Reviewers: dblaikie, aprantl, friss
Subscribers: mgorny, javed.absar, hiraditya, llvm-commits, JDevlieghere
Differential Revision: https://reviews.llvm.org/D37123
llvm-svn: 312042
2017-08-30 05:41:21 +08:00
|
|
|
}
|
2021-07-27 15:28:59 +08:00
|
|
|
U->getLocationTable().dumpLocationList(&Offset, OS, U->getBaseAddress(), MRI,
|
|
|
|
Ctx.getDWARFObj(), U, DumpOpts,
|
|
|
|
Indent);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void dumpLocationExpr(raw_ostream &OS, const DWARFFormValue &FormValue,
|
|
|
|
DWARFUnit *U, unsigned Indent,
|
|
|
|
DIDumpOptions DumpOpts) {
|
|
|
|
assert((FormValue.isFormClass(DWARFFormValue::FC_Block) ||
|
|
|
|
FormValue.isFormClass(DWARFFormValue::FC_Exprloc)) &&
|
|
|
|
"bad FORM for location expression");
|
|
|
|
DWARFContext &Ctx = U->getContext();
|
|
|
|
const MCRegisterInfo *MRI = Ctx.getRegisterInfo();
|
|
|
|
ArrayRef<uint8_t> Expr = *FormValue.getAsBlock();
|
|
|
|
DataExtractor Data(StringRef((const char *)Expr.data(), Expr.size()),
|
|
|
|
Ctx.isLittleEndian(), 0);
|
|
|
|
DWARFExpression(Data, U->getAddressByteSize(), U->getFormParams().Format)
|
|
|
|
.print(OS, DumpOpts, MRI, U);
|
[dwarfdump] Pretty print location expressions and location lists
Summary:
Based on Fred's patch here: https://reviews.llvm.org/D6771
I can't seem to commandeer the old review, so I'm creating a new one.
With that change the locations exrpessions are pretty printed inline in the
DIE tree. The output looks like this for debug_loc entries:
DW_AT_location [DW_FORM_data4] (0x00000000
0x0000000000000001 - 0x000000000000000b: DW_OP_consts +3
0x000000000000000b - 0x0000000000000012: DW_OP_consts +7
0x0000000000000012 - 0x000000000000001b: DW_OP_reg0 RAX, DW_OP_piece 0x4
0x000000000000001b - 0x0000000000000024: DW_OP_breg5 RDI+0)
And like this for debug_loc.dwo entries:
DW_AT_location [DW_FORM_sec_offset] (0x00000000
Addr idx 2 (w/ length 190): DW_OP_consts +0, DW_OP_stack_value
Addr idx 3 (w/ length 23): DW_OP_reg0 RAX, DW_OP_piece 0x4)
Simple locations without ranges are printed inline:
DW_AT_location [DW_FORM_block1] (DW_OP_reg4 RSI, DW_OP_piece 0x4, DW_OP_bit_piece 0x20 0x0)
The debug_loc(.dwo) dumping in changed accordingly to factor the code.
Reviewers: dblaikie, aprantl, friss
Subscribers: mgorny, javed.absar, hiraditya, llvm-commits, JDevlieghere
Differential Revision: https://reviews.llvm.org/D37123
llvm-svn: 312042
2017-08-30 05:41:21 +08:00
|
|
|
}
|
|
|
|
|
2021-11-10 08:47:30 +08:00
|
|
|
static DWARFDie resolveReferencedType(DWARFDie D,
|
|
|
|
dwarf::Attribute Attr = DW_AT_type) {
|
|
|
|
return D.getAttributeValueAsReferencedDie(Attr).resolveTypeUnitReference();
|
|
|
|
}
|
|
|
|
static DWARFDie resolveReferencedType(DWARFDie D, DWARFFormValue F) {
|
|
|
|
return D.getAttributeValueAsReferencedDie(F).resolveTypeUnitReference();
|
|
|
|
}
|
|
|
|
|
2021-09-15 05:48:08 +08:00
|
|
|
namespace {
|
|
|
|
|
|
|
|
// FIXME: We should have pretty printers per language. Currently we print
|
|
|
|
// everything as if it was C++ and fall back to the TAG type name.
|
|
|
|
struct DWARFTypePrinter {
|
|
|
|
raw_ostream &OS;
|
|
|
|
bool Word = true;
|
|
|
|
bool EndedWithTemplate = false;
|
2017-10-10 22:15:25 +08:00
|
|
|
|
2021-09-15 05:48:08 +08:00
|
|
|
DWARFTypePrinter(raw_ostream &OS) : OS(OS) {}
|
|
|
|
|
|
|
|
/// Dump the name encoded in the type tag.
|
|
|
|
void appendTypeTagName(dwarf::Tag T) {
|
|
|
|
StringRef TagStr = TagString(T);
|
2021-09-20 08:33:56 +08:00
|
|
|
static constexpr StringRef Prefix = "DW_TAG_";
|
|
|
|
static constexpr StringRef Suffix = "_type";
|
|
|
|
if (!TagStr.startswith(Prefix) || !TagStr.endswith(Suffix))
|
2021-09-15 05:48:08 +08:00
|
|
|
return;
|
2021-09-20 08:33:56 +08:00
|
|
|
OS << TagStr.substr(Prefix.size(),
|
|
|
|
TagStr.size() - (Prefix.size() + Suffix.size()))
|
|
|
|
<< " ";
|
2021-09-15 05:48:08 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
void appendArrayType(const DWARFDie &D) {
|
2021-09-19 13:22:25 +08:00
|
|
|
for (const DWARFDie &C : D.children()) {
|
|
|
|
if (C.getTag() != DW_TAG_subrange_type)
|
|
|
|
continue;
|
|
|
|
Optional<uint64_t> LB;
|
|
|
|
Optional<uint64_t> Count;
|
|
|
|
Optional<uint64_t> UB;
|
|
|
|
Optional<unsigned> DefaultLB;
|
|
|
|
if (Optional<DWARFFormValue> L = C.find(DW_AT_lower_bound))
|
|
|
|
LB = L->getAsUnsignedConstant();
|
|
|
|
if (Optional<DWARFFormValue> CountV = C.find(DW_AT_count))
|
|
|
|
Count = CountV->getAsUnsignedConstant();
|
|
|
|
if (Optional<DWARFFormValue> UpperV = C.find(DW_AT_upper_bound))
|
|
|
|
UB = UpperV->getAsUnsignedConstant();
|
|
|
|
if (Optional<DWARFFormValue> LV =
|
|
|
|
D.getDwarfUnit()->getUnitDIE().find(DW_AT_language))
|
|
|
|
if (Optional<uint64_t> LC = LV->getAsUnsignedConstant())
|
|
|
|
if ((DefaultLB =
|
|
|
|
LanguageLowerBound(static_cast<dwarf::SourceLanguage>(*LC))))
|
|
|
|
if (LB && *LB == *DefaultLB)
|
|
|
|
LB = None;
|
|
|
|
if (!LB && !Count && !UB)
|
|
|
|
OS << "[]";
|
|
|
|
else if (!LB && (Count || UB) && DefaultLB)
|
|
|
|
OS << '[' << (Count ? *Count : *UB - *DefaultLB + 1) << ']';
|
|
|
|
else {
|
|
|
|
OS << "[[";
|
|
|
|
if (LB)
|
|
|
|
OS << *LB;
|
|
|
|
else
|
|
|
|
OS << '?';
|
|
|
|
OS << ", ";
|
|
|
|
if (Count)
|
llvm-dwarfdump: Improve/fix pretty printing of array dimensions
This is to address post-commit feedback from Paul Robinson on r348954.
The original commit misinterprets count and upper bound as the same thing (I thought I saw GCC producing an upper bound the same as Clang's count, but GCC correctly produces an upper bound that's one less than the count (in C, that is, where arrays are zero indexed)).
I want to preserve the C-like output for the common case, so in the absence of a lower bound the count (or one greater than the upper bound) is rendered between []. In the trickier cases, where a lower bound is specified, a half-open range is used (eg: lower bound 1, count 2 would be "[1, 3)" and an unknown parts use a '?' (eg: "[1, ?)" or "[?, 7)" or "[?, ? + 3)").
Reviewers: aprantl, probinson, JDevlieghere
Differential Revision: https://reviews.llvm.org/D55721
llvm-svn: 349670
2018-12-20 03:34:24 +08:00
|
|
|
if (LB)
|
2021-09-19 13:22:25 +08:00
|
|
|
OS << *LB + *Count;
|
2021-09-15 05:48:08 +08:00
|
|
|
else
|
2021-09-19 13:22:25 +08:00
|
|
|
OS << "? + " << *Count;
|
|
|
|
else if (UB)
|
|
|
|
OS << *UB + 1;
|
|
|
|
else
|
|
|
|
OS << '?';
|
|
|
|
OS << ")]";
|
llvm-dwarfdump: Improve/fix pretty printing of array dimensions
This is to address post-commit feedback from Paul Robinson on r348954.
The original commit misinterprets count and upper bound as the same thing (I thought I saw GCC producing an upper bound the same as Clang's count, but GCC correctly produces an upper bound that's one less than the count (in C, that is, where arrays are zero indexed)).
I want to preserve the C-like output for the common case, so in the absence of a lower bound the count (or one greater than the upper bound) is rendered between []. In the trickier cases, where a lower bound is specified, a half-open range is used (eg: lower bound 1, count 2 would be "[1, 3)" and an unknown parts use a '?' (eg: "[1, ?)" or "[?, 7)" or "[?, ? + 3)").
Reviewers: aprantl, probinson, JDevlieghere
Differential Revision: https://reviews.llvm.org/D55721
llvm-svn: 349670
2018-12-20 03:34:24 +08:00
|
|
|
}
|
2021-09-19 13:22:25 +08:00
|
|
|
}
|
2021-09-20 12:04:03 +08:00
|
|
|
EndedWithTemplate = false;
|
2021-09-15 05:48:08 +08:00
|
|
|
}
|
2021-09-04 03:18:13 +08:00
|
|
|
|
2021-09-20 03:52:06 +08:00
|
|
|
DWARFDie skipQualifiers(DWARFDie D) {
|
|
|
|
while (D && (D.getTag() == DW_TAG_const_type ||
|
|
|
|
D.getTag() == DW_TAG_volatile_type))
|
2021-11-10 08:47:30 +08:00
|
|
|
D = resolveReferencedType(D);
|
2021-09-20 03:52:06 +08:00
|
|
|
return D;
|
|
|
|
}
|
|
|
|
|
2021-09-19 13:54:57 +08:00
|
|
|
bool needsParens(DWARFDie D) {
|
2021-09-20 03:52:06 +08:00
|
|
|
D = skipQualifiers(D);
|
2021-09-19 13:54:57 +08:00
|
|
|
return D && (D.getTag() == DW_TAG_subroutine_type || D.getTag() == DW_TAG_array_type);
|
|
|
|
}
|
|
|
|
|
2021-09-15 05:48:08 +08:00
|
|
|
void appendPointerLikeTypeBefore(DWARFDie D, DWARFDie Inner, StringRef Ptr) {
|
2021-09-20 06:31:35 +08:00
|
|
|
appendQualifiedNameBefore(Inner);
|
2021-09-20 04:32:21 +08:00
|
|
|
if (Word)
|
|
|
|
OS << ' ';
|
2021-09-19 13:54:57 +08:00
|
|
|
if (needsParens(Inner))
|
2021-09-04 03:18:13 +08:00
|
|
|
OS << '(';
|
|
|
|
OS << Ptr;
|
2021-09-15 05:48:08 +08:00
|
|
|
Word = false;
|
2021-09-20 12:04:03 +08:00
|
|
|
EndedWithTemplate = false;
|
2017-10-10 22:15:25 +08:00
|
|
|
}
|
|
|
|
|
2021-09-20 12:04:03 +08:00
|
|
|
DWARFDie
|
|
|
|
appendUnqualifiedNameBefore(DWARFDie D,
|
|
|
|
std::string *OriginalFullName = nullptr) {
|
2021-09-15 05:48:08 +08:00
|
|
|
Word = true;
|
|
|
|
if (!D) {
|
|
|
|
OS << "void";
|
|
|
|
return DWARFDie();
|
2021-09-04 03:18:13 +08:00
|
|
|
}
|
2021-12-30 05:11:16 +08:00
|
|
|
DWARFDie InnerDIE;
|
|
|
|
auto Inner = [&] { return InnerDIE = resolveReferencedType(D); };
|
2021-09-15 05:48:08 +08:00
|
|
|
const dwarf::Tag T = D.getTag();
|
|
|
|
switch (T) {
|
|
|
|
case DW_TAG_pointer_type: {
|
2021-12-30 05:11:16 +08:00
|
|
|
appendPointerLikeTypeBefore(D, Inner(), "*");
|
2021-09-15 05:48:08 +08:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
case DW_TAG_subroutine_type: {
|
2021-12-30 05:11:16 +08:00
|
|
|
appendQualifiedNameBefore(Inner());
|
2021-09-15 05:48:08 +08:00
|
|
|
if (Word) {
|
|
|
|
OS << ' ';
|
2018-12-13 03:53:03 +08:00
|
|
|
}
|
2021-09-15 05:48:08 +08:00
|
|
|
Word = false;
|
|
|
|
break;
|
2018-12-13 03:53:03 +08:00
|
|
|
}
|
2021-09-15 05:48:08 +08:00
|
|
|
case DW_TAG_array_type: {
|
2021-12-30 05:11:16 +08:00
|
|
|
appendQualifiedNameBefore(Inner());
|
2021-09-15 05:48:08 +08:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
case DW_TAG_reference_type:
|
2021-12-30 05:11:16 +08:00
|
|
|
appendPointerLikeTypeBefore(D, Inner(), "&");
|
2021-09-15 05:48:08 +08:00
|
|
|
break;
|
|
|
|
case DW_TAG_rvalue_reference_type:
|
2021-12-30 05:11:16 +08:00
|
|
|
appendPointerLikeTypeBefore(D, Inner(), "&&");
|
2021-09-15 05:48:08 +08:00
|
|
|
break;
|
|
|
|
case DW_TAG_ptr_to_member_type: {
|
2021-12-30 05:11:16 +08:00
|
|
|
appendQualifiedNameBefore(Inner());
|
|
|
|
if (needsParens(InnerDIE))
|
2021-09-15 05:48:08 +08:00
|
|
|
OS << '(';
|
|
|
|
else if (Word)
|
|
|
|
OS << ' ';
|
2021-11-10 08:47:30 +08:00
|
|
|
if (DWARFDie Cont = resolveReferencedType(D, DW_AT_containing_type)) {
|
2021-09-20 06:31:35 +08:00
|
|
|
appendQualifiedName(Cont);
|
2022-02-17 03:18:31 +08:00
|
|
|
EndedWithTemplate = false;
|
2021-09-15 05:48:08 +08:00
|
|
|
OS << "::";
|
|
|
|
}
|
|
|
|
OS << "*";
|
|
|
|
Word = false;
|
|
|
|
break;
|
|
|
|
}
|
2021-09-20 03:52:06 +08:00
|
|
|
case DW_TAG_const_type:
|
|
|
|
case DW_TAG_volatile_type:
|
|
|
|
appendConstVolatileQualifierBefore(D);
|
|
|
|
break;
|
2021-09-20 06:31:35 +08:00
|
|
|
case DW_TAG_namespace: {
|
|
|
|
if (const char *Name = dwarf::toString(D.find(DW_AT_name), nullptr))
|
|
|
|
OS << Name;
|
|
|
|
else
|
|
|
|
OS << "(anonymous namespace)";
|
|
|
|
break;
|
|
|
|
}
|
2021-09-20 08:33:56 +08:00
|
|
|
case DW_TAG_unspecified_type: {
|
|
|
|
StringRef TypeName = D.getShortName();
|
|
|
|
if (TypeName == "decltype(nullptr)")
|
2021-09-22 02:32:17 +08:00
|
|
|
TypeName = "std::nullptr_t";
|
2021-09-20 08:33:56 +08:00
|
|
|
Word = true;
|
|
|
|
OS << TypeName;
|
|
|
|
EndedWithTemplate = false;
|
|
|
|
break;
|
|
|
|
}
|
2021-09-20 12:04:03 +08:00
|
|
|
/*
|
|
|
|
case DW_TAG_structure_type:
|
|
|
|
case DW_TAG_class_type:
|
|
|
|
case DW_TAG_enumeration_type:
|
|
|
|
case DW_TAG_base_type:
|
|
|
|
*/
|
|
|
|
default: {
|
2021-09-20 08:33:56 +08:00
|
|
|
const char *NamePtr = dwarf::toString(D.find(DW_AT_name), nullptr);
|
|
|
|
if (!NamePtr) {
|
2021-11-10 04:51:05 +08:00
|
|
|
appendTypeTagName(D.getTag());
|
2021-12-30 05:11:16 +08:00
|
|
|
return DWARFDie();
|
2021-09-20 12:04:03 +08:00
|
|
|
}
|
|
|
|
Word = true;
|
|
|
|
StringRef Name = NamePtr;
|
|
|
|
static constexpr StringRef MangledPrefix = "_STN";
|
|
|
|
if (Name.startswith(MangledPrefix)) {
|
|
|
|
Name = Name.drop_front(MangledPrefix.size());
|
|
|
|
auto Separator = Name.find('|');
|
|
|
|
assert(Separator != StringRef::npos);
|
|
|
|
StringRef BaseName = Name.substr(0, Separator);
|
|
|
|
StringRef TemplateArgs = Name.substr(Separator + 1);
|
|
|
|
if (OriginalFullName)
|
|
|
|
*OriginalFullName = (BaseName + TemplateArgs).str();
|
|
|
|
Name = BaseName;
|
|
|
|
} else
|
|
|
|
EndedWithTemplate = Name.endswith(">");
|
|
|
|
OS << Name;
|
2021-11-06 06:28:23 +08:00
|
|
|
// This check would be insufficient for operator overloads like
|
|
|
|
// "operator>>" - but for now Clang doesn't try to simplify them, so this
|
|
|
|
// is OK. Add more nuanced operator overload handling here if/when needed.
|
|
|
|
if (Name.endswith(">"))
|
|
|
|
break;
|
|
|
|
if (!appendTemplateParameters(D))
|
|
|
|
break;
|
|
|
|
|
|
|
|
if (EndedWithTemplate)
|
|
|
|
OS << ' ';
|
|
|
|
OS << '>';
|
|
|
|
EndedWithTemplate = true;
|
|
|
|
Word = true;
|
2021-09-15 05:48:08 +08:00
|
|
|
break;
|
|
|
|
}
|
2021-09-20 12:04:03 +08:00
|
|
|
}
|
2021-12-30 05:11:16 +08:00
|
|
|
return InnerDIE;
|
2018-12-13 02:46:25 +08:00
|
|
|
}
|
2021-09-15 05:48:08 +08:00
|
|
|
|
2021-09-16 05:45:12 +08:00
|
|
|
void appendUnqualifiedNameAfter(DWARFDie D, DWARFDie Inner,
|
2021-09-20 04:06:18 +08:00
|
|
|
bool SkipFirstParamIfArtificial = false) {
|
2021-09-15 05:48:08 +08:00
|
|
|
if (!D)
|
|
|
|
return;
|
|
|
|
switch (D.getTag()) {
|
|
|
|
case DW_TAG_subroutine_type: {
|
2021-09-20 03:52:06 +08:00
|
|
|
appendSubroutineNameAfter(D, Inner, SkipFirstParamIfArtificial, false,
|
|
|
|
false);
|
2021-09-15 05:48:08 +08:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
case DW_TAG_array_type: {
|
|
|
|
appendArrayType(D);
|
|
|
|
break;
|
|
|
|
}
|
2021-09-20 03:52:06 +08:00
|
|
|
case DW_TAG_const_type:
|
|
|
|
case DW_TAG_volatile_type:
|
|
|
|
appendConstVolatileQualifierAfter(D);
|
|
|
|
break;
|
2021-09-15 05:48:08 +08:00
|
|
|
case DW_TAG_ptr_to_member_type:
|
|
|
|
case DW_TAG_reference_type:
|
|
|
|
case DW_TAG_rvalue_reference_type:
|
|
|
|
case DW_TAG_pointer_type: {
|
2021-09-19 13:54:57 +08:00
|
|
|
if (needsParens(Inner))
|
2021-09-15 05:48:08 +08:00
|
|
|
OS << ')';
|
2021-11-10 08:47:30 +08:00
|
|
|
appendUnqualifiedNameAfter(Inner, resolveReferencedType(Inner),
|
|
|
|
/*SkipFirstParamIfArtificial=*/D.getTag() ==
|
|
|
|
DW_TAG_ptr_to_member_type);
|
2021-09-15 05:48:08 +08:00
|
|
|
break;
|
|
|
|
}
|
2021-09-20 12:04:03 +08:00
|
|
|
/*
|
|
|
|
case DW_TAG_structure_type:
|
|
|
|
case DW_TAG_class_type:
|
|
|
|
case DW_TAG_enumeration_type:
|
|
|
|
case DW_TAG_base_type:
|
|
|
|
case DW_TAG_namespace:
|
|
|
|
*/
|
2021-09-15 05:48:08 +08:00
|
|
|
default:
|
|
|
|
break;
|
|
|
|
}
|
2017-10-10 22:15:25 +08:00
|
|
|
}
|
|
|
|
|
2021-09-20 06:31:35 +08:00
|
|
|
void appendQualifiedName(DWARFDie D) {
|
|
|
|
if (D)
|
|
|
|
appendScopes(D.getParent());
|
|
|
|
appendUnqualifiedName(D);
|
|
|
|
}
|
|
|
|
DWARFDie appendQualifiedNameBefore(DWARFDie D) {
|
|
|
|
if (D)
|
|
|
|
appendScopes(D.getParent());
|
|
|
|
return appendUnqualifiedNameBefore(D);
|
|
|
|
}
|
2021-09-20 12:04:03 +08:00
|
|
|
bool appendTemplateParameters(DWARFDie D, bool *FirstParameter = nullptr) {
|
|
|
|
bool FirstParameterValue = true;
|
|
|
|
bool IsTemplate = false;
|
|
|
|
if (!FirstParameter)
|
|
|
|
FirstParameter = &FirstParameterValue;
|
|
|
|
for (const DWARFDie &C : D) {
|
|
|
|
auto Sep = [&] {
|
|
|
|
if (*FirstParameter)
|
|
|
|
OS << '<';
|
|
|
|
else
|
|
|
|
OS << ", ";
|
|
|
|
IsTemplate = true;
|
|
|
|
EndedWithTemplate = false;
|
|
|
|
*FirstParameter = false;
|
|
|
|
};
|
|
|
|
if (C.getTag() == dwarf::DW_TAG_GNU_template_parameter_pack) {
|
|
|
|
IsTemplate = true;
|
|
|
|
appendTemplateParameters(C, FirstParameter);
|
|
|
|
}
|
|
|
|
if (C.getTag() == dwarf::DW_TAG_template_value_parameter) {
|
2021-11-10 08:47:30 +08:00
|
|
|
DWARFDie T = resolveReferencedType(C);
|
2021-09-20 12:04:03 +08:00
|
|
|
Sep();
|
|
|
|
if (T.getTag() == DW_TAG_enumeration_type) {
|
|
|
|
auto V = C.find(DW_AT_const_value);
|
|
|
|
bool FoundEnumerator = false;
|
|
|
|
for (const DWARFDie &Enumerator : T) {
|
|
|
|
auto EV = Enumerator.find(DW_AT_const_value);
|
|
|
|
if (V && EV &&
|
|
|
|
V->getAsSignedConstant() == EV->getAsSignedConstant()) {
|
|
|
|
if (T.find(DW_AT_enum_class)) {
|
|
|
|
appendQualifiedName(T);
|
|
|
|
OS << "::";
|
|
|
|
} else
|
|
|
|
appendScopes(T.getParent());
|
|
|
|
OS << Enumerator.getShortName();
|
|
|
|
FoundEnumerator = true;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (FoundEnumerator)
|
|
|
|
continue;
|
|
|
|
OS << '(';
|
|
|
|
appendQualifiedName(T);
|
|
|
|
OS << ')';
|
|
|
|
OS << to_string(*V->getAsSignedConstant());
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
// /Maybe/ we could do pointer type parameters, looking for the
|
|
|
|
// symbol in the ELF symbol table to get back to the variable...
|
|
|
|
// but probably not worth it.
|
|
|
|
if (T.getTag() == DW_TAG_pointer_type)
|
|
|
|
continue;
|
|
|
|
const char *RawName = dwarf::toString(T.find(DW_AT_name), nullptr);
|
|
|
|
assert(RawName);
|
|
|
|
StringRef Name = RawName;
|
|
|
|
auto V = C.find(DW_AT_const_value);
|
|
|
|
bool IsQualifiedChar = false;
|
|
|
|
if (Name == "bool") {
|
|
|
|
OS << (*V->getAsUnsignedConstant() ? "true" : "false");
|
|
|
|
} else if (Name == "short") {
|
|
|
|
OS << "(short)";
|
|
|
|
OS << to_string(*V->getAsSignedConstant());
|
|
|
|
} else if (Name == "unsigned short") {
|
|
|
|
OS << "(unsigned short)";
|
|
|
|
OS << to_string(*V->getAsSignedConstant());
|
|
|
|
} else if (Name == "int")
|
|
|
|
OS << to_string(*V->getAsSignedConstant());
|
|
|
|
else if (Name == "long") {
|
|
|
|
OS << to_string(*V->getAsSignedConstant());
|
|
|
|
OS << "L";
|
|
|
|
} else if (Name == "long long") {
|
|
|
|
OS << to_string(*V->getAsSignedConstant());
|
|
|
|
OS << "LL";
|
|
|
|
} else if (Name == "unsigned int") {
|
|
|
|
OS << to_string(*V->getAsUnsignedConstant());
|
|
|
|
OS << "U";
|
|
|
|
} else if (Name == "unsigned long") {
|
|
|
|
OS << to_string(*V->getAsUnsignedConstant());
|
|
|
|
OS << "UL";
|
|
|
|
} else if (Name == "unsigned long long") {
|
|
|
|
OS << to_string(*V->getAsUnsignedConstant());
|
|
|
|
OS << "ULL";
|
|
|
|
} else if (Name == "char" ||
|
|
|
|
(IsQualifiedChar =
|
|
|
|
(Name == "unsigned char" || Name == "signed char"))) {
|
|
|
|
// FIXME: check T's DW_AT_type to see if it's signed or not (since
|
|
|
|
// char signedness is implementation defined).
|
|
|
|
auto Val = *V->getAsSignedConstant();
|
|
|
|
// Copied/hacked up from Clang's CharacterLiteral::print - incomplete
|
|
|
|
// (doesn't actually support different character types/widths, sign
|
|
|
|
// handling's not done, and doesn't correctly test if a character is
|
|
|
|
// printable or needs to use a numeric escape sequence instead)
|
|
|
|
if (IsQualifiedChar) {
|
|
|
|
OS << '(';
|
|
|
|
OS << Name;
|
|
|
|
OS << ')';
|
|
|
|
}
|
|
|
|
switch (Val) {
|
|
|
|
case '\\':
|
|
|
|
OS << "'\\\\'";
|
|
|
|
break;
|
|
|
|
case '\'':
|
|
|
|
OS << "'\\''";
|
|
|
|
break;
|
|
|
|
case '\a':
|
|
|
|
// TODO: K&R: the meaning of '\\a' is different in traditional C
|
|
|
|
OS << "'\\a'";
|
|
|
|
break;
|
|
|
|
case '\b':
|
|
|
|
OS << "'\\b'";
|
|
|
|
break;
|
|
|
|
case '\f':
|
|
|
|
OS << "'\\f'";
|
|
|
|
break;
|
|
|
|
case '\n':
|
|
|
|
OS << "'\\n'";
|
|
|
|
break;
|
|
|
|
case '\r':
|
|
|
|
OS << "'\\r'";
|
|
|
|
break;
|
|
|
|
case '\t':
|
|
|
|
OS << "'\\t'";
|
|
|
|
break;
|
|
|
|
case '\v':
|
|
|
|
OS << "'\\v'";
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
if ((Val & ~0xFFu) == ~0xFFu)
|
|
|
|
Val &= 0xFFu;
|
|
|
|
if (Val < 127 && Val >= 32) {
|
|
|
|
OS << "'";
|
|
|
|
OS << (char)Val;
|
|
|
|
OS << "'";
|
|
|
|
} else if (Val < 256)
|
|
|
|
OS << to_string(llvm::format("'\\x%02x'", Val));
|
|
|
|
else if (Val <= 0xFFFF)
|
|
|
|
OS << to_string(llvm::format("'\\u%04x'", Val));
|
|
|
|
else
|
|
|
|
OS << to_string(llvm::format("'\\U%08x'", Val));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
if (C.getTag() == dwarf::DW_TAG_GNU_template_template_param) {
|
|
|
|
const char *RawName =
|
|
|
|
dwarf::toString(C.find(DW_AT_GNU_template_name), nullptr);
|
|
|
|
assert(RawName);
|
|
|
|
StringRef Name = RawName;
|
|
|
|
Sep();
|
|
|
|
OS << Name;
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
if (C.getTag() != dwarf::DW_TAG_template_type_parameter)
|
|
|
|
continue;
|
|
|
|
auto TypeAttr = C.find(DW_AT_type);
|
|
|
|
Sep();
|
2021-11-10 08:47:30 +08:00
|
|
|
appendQualifiedName(TypeAttr ? resolveReferencedType(C, *TypeAttr)
|
|
|
|
: DWARFDie());
|
2021-09-20 12:04:03 +08:00
|
|
|
}
|
2022-02-17 03:18:31 +08:00
|
|
|
if (IsTemplate && *FirstParameter && FirstParameter == &FirstParameterValue) {
|
2021-09-20 12:04:03 +08:00
|
|
|
OS << '<';
|
2022-02-17 03:18:31 +08:00
|
|
|
EndedWithTemplate = false;
|
|
|
|
}
|
2021-09-20 12:04:03 +08:00
|
|
|
return IsTemplate;
|
|
|
|
}
|
2021-09-20 03:52:06 +08:00
|
|
|
void decomposeConstVolatile(DWARFDie &N, DWARFDie &T, DWARFDie &C,
|
|
|
|
DWARFDie &V) {
|
|
|
|
(N.getTag() == DW_TAG_const_type ? C : V) = N;
|
2021-11-10 08:47:30 +08:00
|
|
|
T = resolveReferencedType(N);
|
2021-09-20 03:52:06 +08:00
|
|
|
if (T) {
|
|
|
|
auto Tag = T.getTag();
|
|
|
|
if (Tag == DW_TAG_const_type) {
|
|
|
|
C = T;
|
2021-11-10 08:47:30 +08:00
|
|
|
T = resolveReferencedType(T);
|
2021-09-20 03:52:06 +08:00
|
|
|
} else if (Tag == DW_TAG_volatile_type) {
|
|
|
|
V = T;
|
2021-11-10 08:47:30 +08:00
|
|
|
T = resolveReferencedType(T);
|
2021-09-20 03:52:06 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
void appendConstVolatileQualifierAfter(DWARFDie N) {
|
|
|
|
DWARFDie C;
|
|
|
|
DWARFDie V;
|
|
|
|
DWARFDie T;
|
|
|
|
decomposeConstVolatile(N, T, C, V);
|
|
|
|
if (T && T.getTag() == DW_TAG_subroutine_type)
|
2021-11-10 08:47:30 +08:00
|
|
|
appendSubroutineNameAfter(T, resolveReferencedType(T), false, C.isValid(),
|
|
|
|
V.isValid());
|
2021-09-20 03:52:06 +08:00
|
|
|
else
|
2021-11-10 08:47:30 +08:00
|
|
|
appendUnqualifiedNameAfter(T, resolveReferencedType(T));
|
2021-09-20 03:52:06 +08:00
|
|
|
}
|
|
|
|
void appendConstVolatileQualifierBefore(DWARFDie N) {
|
|
|
|
DWARFDie C;
|
|
|
|
DWARFDie V;
|
|
|
|
DWARFDie T;
|
|
|
|
decomposeConstVolatile(N, T, C, V);
|
|
|
|
bool Subroutine = T && T.getTag() == DW_TAG_subroutine_type;
|
|
|
|
DWARFDie A = T;
|
|
|
|
while (A && A.getTag() == DW_TAG_array_type)
|
2021-11-10 08:47:30 +08:00
|
|
|
A = resolveReferencedType(A);
|
2021-09-20 03:52:06 +08:00
|
|
|
bool Leading =
|
|
|
|
(!A || (A.getTag() != DW_TAG_pointer_type &&
|
|
|
|
A.getTag() != llvm::dwarf::DW_TAG_ptr_to_member_type)) &&
|
|
|
|
!Subroutine;
|
|
|
|
if (Leading) {
|
|
|
|
if (C)
|
|
|
|
OS << "const ";
|
|
|
|
if (V)
|
|
|
|
OS << "volatile ";
|
|
|
|
}
|
2021-09-20 06:31:35 +08:00
|
|
|
appendQualifiedNameBefore(T);
|
2021-09-20 03:52:06 +08:00
|
|
|
if (!Leading && !Subroutine) {
|
|
|
|
Word = true;
|
|
|
|
if (C)
|
|
|
|
OS << "const";
|
|
|
|
if (V) {
|
|
|
|
if (C)
|
|
|
|
OS << ' ';
|
|
|
|
OS << "volatile";
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-09-15 05:48:08 +08:00
|
|
|
/// Recursively append the DIE type name when applicable.
|
2021-09-20 12:04:03 +08:00
|
|
|
void appendUnqualifiedName(DWARFDie D,
|
|
|
|
std::string *OriginalFullName = nullptr) {
|
2021-09-15 05:48:08 +08:00
|
|
|
// FIXME: We should have pretty printers per language. Currently we print
|
|
|
|
// everything as if it was C++ and fall back to the TAG type name.
|
2021-09-20 12:04:03 +08:00
|
|
|
DWARFDie Inner = appendUnqualifiedNameBefore(D, OriginalFullName);
|
2021-09-20 08:03:10 +08:00
|
|
|
appendUnqualifiedNameAfter(D, Inner);
|
2021-09-15 05:48:08 +08:00
|
|
|
}
|
2021-09-20 03:52:06 +08:00
|
|
|
|
|
|
|
void appendSubroutineNameAfter(DWARFDie D, DWARFDie Inner,
|
|
|
|
bool SkipFirstParamIfArtificial, bool Const,
|
|
|
|
bool Volatile) {
|
|
|
|
DWARFDie FirstParamIfArtificial;
|
|
|
|
OS << '(';
|
|
|
|
EndedWithTemplate = false;
|
|
|
|
bool First = true;
|
|
|
|
bool RealFirst = true;
|
|
|
|
for (DWARFDie P : D) {
|
2022-01-06 05:25:13 +08:00
|
|
|
if (P.getTag() != DW_TAG_formal_parameter &&
|
|
|
|
P.getTag() != DW_TAG_unspecified_parameters)
|
2021-09-20 03:52:06 +08:00
|
|
|
return;
|
2021-11-10 08:47:30 +08:00
|
|
|
DWARFDie T = resolveReferencedType(P);
|
2021-09-20 03:52:06 +08:00
|
|
|
if (SkipFirstParamIfArtificial && RealFirst && P.find(DW_AT_artificial)) {
|
|
|
|
FirstParamIfArtificial = T;
|
|
|
|
RealFirst = false;
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
if (!First) {
|
|
|
|
OS << ", ";
|
|
|
|
}
|
|
|
|
First = false;
|
2022-01-06 05:25:13 +08:00
|
|
|
if (P.getTag() == DW_TAG_unspecified_parameters)
|
|
|
|
OS << "...";
|
|
|
|
else
|
|
|
|
appendQualifiedName(T);
|
2021-09-20 03:52:06 +08:00
|
|
|
}
|
|
|
|
EndedWithTemplate = false;
|
|
|
|
OS << ')';
|
|
|
|
if (FirstParamIfArtificial) {
|
|
|
|
if (DWARFDie P = FirstParamIfArtificial) {
|
|
|
|
if (P.getTag() == DW_TAG_pointer_type) {
|
|
|
|
auto CVStep = [&](DWARFDie CV) {
|
2021-11-10 08:47:30 +08:00
|
|
|
if (DWARFDie U = resolveReferencedType(CV)) {
|
2022-03-18 06:59:18 +08:00
|
|
|
Const |= U.getTag() == DW_TAG_const_type;
|
|
|
|
Volatile |= U.getTag() == DW_TAG_volatile_type;
|
|
|
|
return U;
|
2021-09-20 03:52:06 +08:00
|
|
|
}
|
|
|
|
return DWARFDie();
|
|
|
|
};
|
|
|
|
if (DWARFDie CV = CVStep(P)) {
|
|
|
|
CVStep(CV);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2022-03-22 03:37:39 +08:00
|
|
|
|
|
|
|
if (auto CC = D.find(DW_AT_calling_convention)) {
|
|
|
|
switch (*CC->getAsUnsignedConstant()) {
|
|
|
|
case CallingConvention::DW_CC_BORLAND_stdcall:
|
|
|
|
OS << " __attribute__((stdcall))";
|
|
|
|
break;
|
|
|
|
case CallingConvention::DW_CC_BORLAND_msfastcall:
|
|
|
|
OS << " __attribute__((fastcall))";
|
|
|
|
break;
|
|
|
|
case CallingConvention::DW_CC_BORLAND_thiscall:
|
|
|
|
OS << " __attribute__((thiscall))";
|
|
|
|
break;
|
|
|
|
case CallingConvention::DW_CC_LLVM_vectorcall:
|
|
|
|
OS << " __attribute__((vectorcall))";
|
|
|
|
break;
|
|
|
|
case CallingConvention::DW_CC_BORLAND_pascal:
|
|
|
|
OS << " __attribute__((pascal))";
|
|
|
|
break;
|
|
|
|
case CallingConvention::DW_CC_LLVM_Win64:
|
|
|
|
OS << " __attribute__((ms_abi))";
|
|
|
|
break;
|
|
|
|
case CallingConvention::DW_CC_LLVM_X86_64SysV:
|
|
|
|
OS << " __attribute__((sysv_abi))";
|
|
|
|
break;
|
|
|
|
case CallingConvention::DW_CC_LLVM_AAPCS:
|
|
|
|
// AArch64VectorCall missing?
|
|
|
|
OS << " __attribute__((pcs(\"aapcs\")))";
|
|
|
|
break;
|
|
|
|
case CallingConvention::DW_CC_LLVM_AAPCS_VFP:
|
|
|
|
OS << " __attribute__((pcs(\"aapcs-vfp\")))";
|
|
|
|
break;
|
|
|
|
case CallingConvention::DW_CC_LLVM_IntelOclBicc:
|
|
|
|
OS << " __attribute__((intel_ocl_bicc))";
|
|
|
|
break;
|
|
|
|
case CallingConvention::DW_CC_LLVM_SpirFunction:
|
|
|
|
case CallingConvention::DW_CC_LLVM_OpenCLKernel:
|
|
|
|
// These aren't available as attributes, but maybe we should still
|
|
|
|
// render them somehow? (Clang doesn't render them, but that's an issue
|
|
|
|
// for template names too - since then the DWARF names of templates
|
|
|
|
// instantiated with function types with these calling conventions won't
|
|
|
|
// have distinct names - so we'd need to fix that too)
|
|
|
|
break;
|
|
|
|
case CallingConvention::DW_CC_LLVM_Swift:
|
|
|
|
// SwiftAsync missing
|
|
|
|
OS << " __attribute__((swiftcall))";
|
|
|
|
break;
|
|
|
|
case CallingConvention::DW_CC_LLVM_PreserveMost:
|
|
|
|
OS << " __attribute__((preserve_most))";
|
|
|
|
break;
|
|
|
|
case CallingConvention::DW_CC_LLVM_PreserveAll:
|
|
|
|
OS << " __attribute__((preserve_all))";
|
|
|
|
break;
|
|
|
|
case CallingConvention::DW_CC_LLVM_X86RegCall:
|
|
|
|
OS << " __attribute__((regcall))";
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-03-18 06:59:18 +08:00
|
|
|
if (Const)
|
|
|
|
OS << " const";
|
|
|
|
if (Volatile)
|
|
|
|
OS << " volatile";
|
2021-09-20 03:52:06 +08:00
|
|
|
if (D.find(DW_AT_reference))
|
|
|
|
OS << " &";
|
|
|
|
if (D.find(DW_AT_rvalue_reference))
|
|
|
|
OS << " &&";
|
2022-03-18 06:59:18 +08:00
|
|
|
|
2021-11-10 08:47:30 +08:00
|
|
|
appendUnqualifiedNameAfter(Inner, resolveReferencedType(Inner));
|
2021-09-20 03:52:06 +08:00
|
|
|
}
|
2021-09-20 06:31:35 +08:00
|
|
|
void appendScopes(DWARFDie D) {
|
|
|
|
if (D.getTag() == DW_TAG_compile_unit)
|
|
|
|
return;
|
|
|
|
if (D.getTag() == DW_TAG_type_unit)
|
|
|
|
return;
|
2021-11-10 04:51:05 +08:00
|
|
|
if (D.getTag() == DW_TAG_skeleton_unit)
|
2021-09-20 06:31:35 +08:00
|
|
|
return;
|
|
|
|
if (D.getTag() == DW_TAG_subprogram)
|
|
|
|
return;
|
2021-12-03 18:27:29 +08:00
|
|
|
if (D.getTag() == DW_TAG_lexical_block)
|
|
|
|
return;
|
2021-11-10 08:47:30 +08:00
|
|
|
D = D.resolveTypeUnitReference();
|
2021-09-20 06:31:35 +08:00
|
|
|
if (DWARFDie P = D.getParent())
|
|
|
|
appendScopes(P);
|
|
|
|
appendUnqualifiedName(D);
|
|
|
|
OS << "::";
|
|
|
|
}
|
2021-09-15 05:48:08 +08:00
|
|
|
};
|
|
|
|
} // anonymous namespace
|
2021-09-04 03:18:13 +08:00
|
|
|
|
2016-12-14 02:25:19 +08:00
|
|
|
static void dumpAttribute(raw_ostream &OS, const DWARFDie &Die,
|
2021-03-09 15:31:23 +08:00
|
|
|
const DWARFAttribute &AttrValue, unsigned Indent,
|
2017-06-07 07:28:45 +08:00
|
|
|
DIDumpOptions DumpOpts) {
|
2016-12-14 02:25:19 +08:00
|
|
|
if (!Die.isValid())
|
|
|
|
return;
|
|
|
|
const char BaseIndent[] = " ";
|
|
|
|
OS << BaseIndent;
|
2017-10-05 06:26:19 +08:00
|
|
|
OS.indent(Indent + 2);
|
2021-03-09 15:31:23 +08:00
|
|
|
dwarf::Attribute Attr = AttrValue.Attr;
|
[dwarf] Unify unknown dwarf enum formatting code
Summary:
We have had at least three pieces of code (in DWARFAbbreviationDeclaration,
DWARFAcceleratorTable and DWARFDie) that have hand-rolled support for
dumping unknown dwarf enum values. While not terrible, they are a bit
distracting and enable small differences to creep in (Unknown_ffff vs.
Unknown_0xffff). I ended up needing to add a fourth place
(DWARFVerifier), so it seems it would be a good time to centralize.
This patch creates an alternative to the XXXString dumping functions in
the BinaryFormat library, which formats an unknown value as
DW_TYPE_unknown_1234, instead of just an empty string. It is based on
the formatv function, as that allows us to avoid materializing the
string for unknown values (and because this way I don't have to invent a
name for the new functions :P).
In this patch I add formatters for dwarf attributes, forms, tags, and
index attributes as these are the ones in use currently, but adding
other enums is straight-forward.
Reviewers: dblaikie, JDevlieghere, aprantl
Subscribers: llvm-commits
Differential Revision: https://reviews.llvm.org/D44570
llvm-svn: 328090
2018-03-21 19:46:37 +08:00
|
|
|
WithColor(OS, HighlightColor::Attribute) << formatv("{0}", Attr);
|
2017-06-07 07:28:45 +08:00
|
|
|
|
2021-03-09 15:31:23 +08:00
|
|
|
dwarf::Form Form = AttrValue.Value.getForm();
|
[dwarf] Unify unknown dwarf enum formatting code
Summary:
We have had at least three pieces of code (in DWARFAbbreviationDeclaration,
DWARFAcceleratorTable and DWARFDie) that have hand-rolled support for
dumping unknown dwarf enum values. While not terrible, they are a bit
distracting and enable small differences to creep in (Unknown_ffff vs.
Unknown_0xffff). I ended up needing to add a fourth place
(DWARFVerifier), so it seems it would be a good time to centralize.
This patch creates an alternative to the XXXString dumping functions in
the BinaryFormat library, which formats an unknown value as
DW_TYPE_unknown_1234, instead of just an empty string. It is based on
the formatv function, as that allows us to avoid materializing the
string for unknown values (and because this way I don't have to invent a
name for the new functions :P).
In this patch I add formatters for dwarf attributes, forms, tags, and
index attributes as these are the ones in use currently, but adding
other enums is straight-forward.
Reviewers: dblaikie, JDevlieghere, aprantl
Subscribers: llvm-commits
Differential Revision: https://reviews.llvm.org/D44570
llvm-svn: 328090
2018-03-21 19:46:37 +08:00
|
|
|
if (DumpOpts.Verbose || DumpOpts.ShowForm)
|
|
|
|
OS << formatv(" [{0}]", Form);
|
2017-06-07 07:28:45 +08:00
|
|
|
|
2016-12-14 02:25:19 +08:00
|
|
|
DWARFUnit *U = Die.getDwarfUnit();
|
2021-03-09 15:31:23 +08:00
|
|
|
const DWARFFormValue &FormValue = AttrValue.Value;
|
2017-08-19 05:35:44 +08:00
|
|
|
|
2016-12-14 02:25:19 +08:00
|
|
|
OS << "\t(";
|
2017-08-19 05:35:44 +08:00
|
|
|
|
2016-12-14 02:25:19 +08:00
|
|
|
StringRef Name;
|
|
|
|
std::string File;
|
2018-03-09 17:56:24 +08:00
|
|
|
auto Color = HighlightColor::Enumerator;
|
2016-12-14 02:25:19 +08:00
|
|
|
if (Attr == DW_AT_decl_file || Attr == DW_AT_call_file) {
|
2018-03-09 17:56:24 +08:00
|
|
|
Color = HighlightColor::String;
|
2016-12-14 02:25:19 +08:00
|
|
|
if (const auto *LT = U->getContext().getLineTableForUnit(U))
|
2017-10-05 06:26:19 +08:00
|
|
|
if (LT->getFileNameByIndex(
|
2019-02-27 08:58:09 +08:00
|
|
|
FormValue.getAsUnsignedConstant().getValue(),
|
2017-10-05 06:26:19 +08:00
|
|
|
U->getCompilationDir(),
|
|
|
|
DILineInfoSpecifier::FileLineInfoKind::AbsoluteFilePath, File)) {
|
2016-12-14 02:25:19 +08:00
|
|
|
File = '"' + File + '"';
|
|
|
|
Name = File;
|
|
|
|
}
|
2019-02-27 08:58:09 +08:00
|
|
|
} else if (Optional<uint64_t> Val = FormValue.getAsUnsignedConstant())
|
2016-12-14 02:25:19 +08:00
|
|
|
Name = AttributeValueString(Attr, *Val);
|
2017-08-19 05:35:44 +08:00
|
|
|
|
2016-12-14 02:25:19 +08:00
|
|
|
if (!Name.empty())
|
|
|
|
WithColor(OS, Color) << Name;
|
|
|
|
else if (Attr == DW_AT_decl_line || Attr == DW_AT_call_line)
|
2019-02-27 08:58:09 +08:00
|
|
|
OS << *FormValue.getAsUnsignedConstant();
|
2020-10-05 03:28:37 +08:00
|
|
|
else if (Attr == DW_AT_low_pc &&
|
|
|
|
(FormValue.getAsAddress() ==
|
|
|
|
dwarf::computeTombstoneAddress(U->getAddressByteSize()))) {
|
|
|
|
if (DumpOpts.Verbose) {
|
|
|
|
FormValue.dump(OS, DumpOpts);
|
|
|
|
OS << " (";
|
|
|
|
}
|
|
|
|
OS << "dead code";
|
|
|
|
if (DumpOpts.Verbose)
|
|
|
|
OS << ')';
|
|
|
|
} else if (Attr == DW_AT_high_pc && !DumpOpts.ShowForm && !DumpOpts.Verbose &&
|
|
|
|
FormValue.getAsUnsignedConstant()) {
|
2017-12-09 07:32:47 +08:00
|
|
|
if (DumpOpts.ShowAddresses) {
|
|
|
|
// Print the actual address rather than the offset.
|
|
|
|
uint64_t LowPC, HighPC, Index;
|
|
|
|
if (Die.getLowAndHighPC(LowPC, HighPC, Index))
|
2020-10-05 06:42:03 +08:00
|
|
|
DWARFFormValue::dumpAddress(OS, U->getAddressByteSize(), HighPC);
|
2017-12-09 07:32:47 +08:00
|
|
|
else
|
2019-02-27 08:58:09 +08:00
|
|
|
FormValue.dump(OS, DumpOpts);
|
2017-12-09 07:32:47 +08:00
|
|
|
}
|
2021-07-27 15:28:59 +08:00
|
|
|
} else if (DWARFAttribute::mayHaveLocationList(Attr) &&
|
|
|
|
FormValue.isFormClass(DWARFFormValue::FC_SectionOffset))
|
|
|
|
dumpLocationList(OS, FormValue, U, sizeof(BaseIndent) + Indent + 4,
|
|
|
|
DumpOpts);
|
|
|
|
else if (FormValue.isFormClass(DWARFFormValue::FC_Exprloc) ||
|
|
|
|
(DWARFAttribute::mayHaveLocationExpr(Attr) &&
|
|
|
|
FormValue.isFormClass(DWARFFormValue::FC_Block)))
|
|
|
|
dumpLocationExpr(OS, FormValue, U, sizeof(BaseIndent) + Indent + 4,
|
|
|
|
DumpOpts);
|
2016-12-14 02:25:19 +08:00
|
|
|
else
|
2019-02-27 08:58:09 +08:00
|
|
|
FormValue.dump(OS, DumpOpts);
|
2017-08-19 05:35:44 +08:00
|
|
|
|
2018-09-05 00:21:37 +08:00
|
|
|
std::string Space = DumpOpts.ShowAddresses ? " " : "";
|
|
|
|
|
2016-12-14 02:25:19 +08:00
|
|
|
// We have dumped the attribute raw value. For some attributes
|
|
|
|
// having both the raw value and the pretty-printed value is
|
|
|
|
// interesting. These attributes are handled below.
|
2017-08-23 05:59:46 +08:00
|
|
|
if (Attr == DW_AT_specification || Attr == DW_AT_abstract_origin) {
|
2018-12-13 03:23:55 +08:00
|
|
|
if (const char *Name =
|
2019-02-27 08:58:09 +08:00
|
|
|
Die.getAttributeValueAsReferencedDie(FormValue).getName(
|
2018-12-13 03:23:55 +08:00
|
|
|
DINameKind::LinkageName))
|
2018-09-05 00:21:37 +08:00
|
|
|
OS << Space << "\"" << Name << '\"';
|
2017-10-10 22:15:25 +08:00
|
|
|
} else if (Attr == DW_AT_type) {
|
2021-11-10 08:47:30 +08:00
|
|
|
DWARFDie D = resolveReferencedType(Die, FormValue);
|
2021-09-20 08:03:10 +08:00
|
|
|
if (D && !D.isNULL()) {
|
|
|
|
OS << Space << "\"";
|
2022-01-12 23:42:48 +08:00
|
|
|
dumpTypeQualifiedName(D, OS);
|
2021-09-20 08:03:10 +08:00
|
|
|
OS << '"';
|
|
|
|
}
|
2016-12-14 02:25:19 +08:00
|
|
|
} else if (Attr == DW_AT_APPLE_property_attribute) {
|
2019-02-27 08:58:09 +08:00
|
|
|
if (Optional<uint64_t> OptVal = FormValue.getAsUnsignedConstant())
|
2016-12-14 02:25:19 +08:00
|
|
|
dumpApplePropertyAttribute(OS, *OptVal);
|
|
|
|
} else if (Attr == DW_AT_ranges) {
|
2017-08-15 20:32:54 +08:00
|
|
|
const DWARFObject &Obj = Die.getDwarfUnit()->getContext().getDWARFObj();
|
2018-05-19 04:12:54 +08:00
|
|
|
// For DW_FORM_rnglistx we need to dump the offset separately, since
|
|
|
|
// we have only dumped the index so far.
|
2019-02-27 08:58:09 +08:00
|
|
|
if (FormValue.getForm() == DW_FORM_rnglistx)
|
2018-05-19 04:12:54 +08:00
|
|
|
if (auto RangeListOffset =
|
2019-02-27 08:58:09 +08:00
|
|
|
U->getRnglistOffset(*FormValue.getAsSectionOffset())) {
|
|
|
|
DWARFFormValue FV = DWARFFormValue::createFromUValue(
|
|
|
|
dwarf::DW_FORM_sec_offset, *RangeListOffset);
|
2018-05-19 04:12:54 +08:00
|
|
|
FV.dump(OS, DumpOpts);
|
|
|
|
}
|
2018-06-21 06:56:37 +08:00
|
|
|
if (auto RangesOrError = Die.getAddressRanges())
|
|
|
|
dumpRanges(Obj, OS, RangesOrError.get(), U->getAddressByteSize(),
|
|
|
|
sizeof(BaseIndent) + Indent + 4, DumpOpts);
|
|
|
|
else
|
2020-02-14 23:57:21 +08:00
|
|
|
DumpOpts.RecoverableErrorHandler(createStringError(
|
|
|
|
errc::invalid_argument, "decoding address ranges: %s",
|
|
|
|
toString(RangesOrError.takeError()).c_str()));
|
2016-12-14 02:25:19 +08:00
|
|
|
}
|
2017-08-15 20:32:54 +08:00
|
|
|
|
2016-12-14 02:25:19 +08:00
|
|
|
OS << ")\n";
|
|
|
|
}
|
|
|
|
|
2021-09-20 12:04:03 +08:00
|
|
|
void DWARFDie::getFullName(raw_string_ostream &OS,
|
|
|
|
std::string *OriginalFullName) const {
|
|
|
|
const char *NamePtr = getShortName();
|
|
|
|
if (!NamePtr)
|
|
|
|
return;
|
2021-12-30 06:04:38 +08:00
|
|
|
if (getTag() == DW_TAG_GNU_template_parameter_pack)
|
|
|
|
return;
|
2022-01-12 23:42:48 +08:00
|
|
|
dumpTypeUnqualifiedName(*this, OS, OriginalFullName);
|
2021-09-20 12:04:03 +08:00
|
|
|
}
|
|
|
|
|
2017-10-05 06:26:19 +08:00
|
|
|
bool DWARFDie::isSubprogramDIE() const { return getTag() == DW_TAG_subprogram; }
|
2016-12-14 02:25:19 +08:00
|
|
|
|
|
|
|
bool DWARFDie::isSubroutineDIE() const {
|
|
|
|
auto Tag = getTag();
|
|
|
|
return Tag == DW_TAG_subprogram || Tag == DW_TAG_inlined_subroutine;
|
|
|
|
}
|
|
|
|
|
2017-10-05 06:26:19 +08:00
|
|
|
Optional<DWARFFormValue> DWARFDie::find(dwarf::Attribute Attr) const {
|
2016-12-14 07:20:56 +08:00
|
|
|
if (!isValid())
|
|
|
|
return None;
|
2016-12-14 02:25:19 +08:00
|
|
|
auto AbbrevDecl = getAbbreviationDeclarationPtr();
|
|
|
|
if (AbbrevDecl)
|
2016-12-14 07:20:56 +08:00
|
|
|
return AbbrevDecl->getAttributeValue(getOffset(), Attr, *U);
|
|
|
|
return None;
|
2016-12-14 02:25:19 +08:00
|
|
|
}
|
|
|
|
|
2017-01-14 06:32:12 +08:00
|
|
|
Optional<DWARFFormValue>
|
|
|
|
DWARFDie::find(ArrayRef<dwarf::Attribute> Attrs) const {
|
|
|
|
if (!isValid())
|
|
|
|
return None;
|
|
|
|
auto AbbrevDecl = getAbbreviationDeclarationPtr();
|
|
|
|
if (AbbrevDecl) {
|
|
|
|
for (auto Attr : Attrs) {
|
|
|
|
if (auto Value = AbbrevDecl->getAttributeValue(getOffset(), Attr, *U))
|
|
|
|
return Value;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return None;
|
|
|
|
}
|
|
|
|
|
|
|
|
Optional<DWARFFormValue>
|
|
|
|
DWARFDie::findRecursively(ArrayRef<dwarf::Attribute> Attrs) const {
|
2020-04-30 01:00:31 +08:00
|
|
|
SmallVector<DWARFDie, 3> Worklist;
|
2018-05-01 01:02:41 +08:00
|
|
|
Worklist.push_back(*this);
|
|
|
|
|
|
|
|
// Keep track if DIEs already seen to prevent infinite recursion.
|
|
|
|
// Empirically we rarely see a depth of more than 3 when dealing with valid
|
|
|
|
// DWARF. This corresponds to following the DW_AT_abstract_origin and
|
|
|
|
// DW_AT_specification just once.
|
|
|
|
SmallSet<DWARFDie, 3> Seen;
|
2019-03-27 17:38:05 +08:00
|
|
|
Seen.insert(*this);
|
2018-05-01 01:02:41 +08:00
|
|
|
|
|
|
|
while (!Worklist.empty()) {
|
2021-01-25 04:18:57 +08:00
|
|
|
DWARFDie Die = Worklist.pop_back_val();
|
2018-05-01 01:02:41 +08:00
|
|
|
|
|
|
|
if (!Die.isValid())
|
|
|
|
continue;
|
|
|
|
|
|
|
|
if (auto Value = Die.find(Attrs))
|
2017-11-28 06:12:44 +08:00
|
|
|
return Value;
|
2018-05-01 01:02:41 +08:00
|
|
|
|
|
|
|
if (auto D = Die.getAttributeValueAsReferencedDie(DW_AT_abstract_origin))
|
2019-03-27 17:38:05 +08:00
|
|
|
if (Seen.insert(D).second)
|
|
|
|
Worklist.push_back(D);
|
2018-05-01 01:02:41 +08:00
|
|
|
|
|
|
|
if (auto D = Die.getAttributeValueAsReferencedDie(DW_AT_specification))
|
2019-03-27 17:38:05 +08:00
|
|
|
if (Seen.insert(D).second)
|
|
|
|
Worklist.push_back(D);
|
2017-11-28 06:12:44 +08:00
|
|
|
}
|
2018-05-01 01:02:41 +08:00
|
|
|
|
2017-01-14 06:32:12 +08:00
|
|
|
return None;
|
|
|
|
}
|
|
|
|
|
2016-12-14 02:25:19 +08:00
|
|
|
DWARFDie
|
|
|
|
DWARFDie::getAttributeValueAsReferencedDie(dwarf::Attribute Attr) const {
|
2018-12-13 03:23:55 +08:00
|
|
|
if (Optional<DWARFFormValue> F = find(Attr))
|
|
|
|
return getAttributeValueAsReferencedDie(*F);
|
|
|
|
return DWARFDie();
|
|
|
|
}
|
|
|
|
|
|
|
|
DWARFDie
|
|
|
|
DWARFDie::getAttributeValueAsReferencedDie(const DWARFFormValue &V) const {
|
2021-11-10 08:47:30 +08:00
|
|
|
DWARFDie Result;
|
2019-05-11 03:15:29 +08:00
|
|
|
if (auto SpecRef = V.getAsRelativeReference()) {
|
|
|
|
if (SpecRef->Unit)
|
2021-11-10 08:47:30 +08:00
|
|
|
Result = SpecRef->Unit->getDIEForOffset(SpecRef->Unit->getOffset() +
|
|
|
|
SpecRef->Offset);
|
|
|
|
else if (auto SpecUnit =
|
|
|
|
U->getUnitVector().getUnitForOffset(SpecRef->Offset))
|
|
|
|
Result = SpecUnit->getDIEForOffset(SpecRef->Offset);
|
2016-12-14 02:25:19 +08:00
|
|
|
}
|
2021-11-10 08:47:30 +08:00
|
|
|
return Result;
|
|
|
|
}
|
|
|
|
|
|
|
|
DWARFDie DWARFDie::resolveTypeUnitReference() const {
|
|
|
|
if (auto Attr = find(DW_AT_signature)) {
|
|
|
|
if (Optional<uint64_t> Sig = Attr->getAsReferenceUVal()) {
|
2021-11-19 05:49:32 +08:00
|
|
|
if (DWARFTypeUnit *TU = U->getContext().getTypeUnitForHash(
|
|
|
|
U->getVersion(), *Sig, U->isDWOUnit()))
|
2021-11-10 08:47:30 +08:00
|
|
|
return TU->getDIEForOffset(TU->getTypeOffset() + TU->getOffset());
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return *this;
|
2016-12-14 02:25:19 +08:00
|
|
|
}
|
|
|
|
|
2017-10-05 06:26:19 +08:00
|
|
|
Optional<uint64_t> DWARFDie::getRangesBaseAttribute() const {
|
2017-01-14 06:32:12 +08:00
|
|
|
return toSectionOffset(find({DW_AT_rnglists_base, DW_AT_GNU_ranges_base}));
|
2016-12-14 02:25:19 +08:00
|
|
|
}
|
|
|
|
|
2019-11-13 06:15:37 +08:00
|
|
|
Optional<uint64_t> DWARFDie::getLocBaseAttribute() const {
|
|
|
|
return toSectionOffset(find(DW_AT_loclists_base));
|
|
|
|
}
|
|
|
|
|
2016-12-20 04:36:41 +08:00
|
|
|
Optional<uint64_t> DWARFDie::getHighPC(uint64_t LowPC) const {
|
2020-10-05 03:28:37 +08:00
|
|
|
uint64_t Tombstone = dwarf::computeTombstoneAddress(U->getAddressByteSize());
|
|
|
|
if (LowPC == Tombstone)
|
|
|
|
return None;
|
2017-01-14 05:08:18 +08:00
|
|
|
if (auto FormValue = find(DW_AT_high_pc)) {
|
2016-12-20 04:36:41 +08:00
|
|
|
if (auto Address = FormValue->getAsAddress()) {
|
|
|
|
// High PC is an address.
|
|
|
|
return Address;
|
|
|
|
}
|
|
|
|
if (auto Offset = FormValue->getAsUnsignedConstant()) {
|
|
|
|
// High PC is an offset from LowPC.
|
|
|
|
return LowPC + *Offset;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return None;
|
|
|
|
}
|
2016-12-15 06:38:08 +08:00
|
|
|
|
2017-05-28 02:10:23 +08:00
|
|
|
bool DWARFDie::getLowAndHighPC(uint64_t &LowPC, uint64_t &HighPC,
|
|
|
|
uint64_t &SectionIndex) const {
|
|
|
|
auto F = find(DW_AT_low_pc);
|
2018-12-23 06:20:40 +08:00
|
|
|
auto LowPcAddr = toSectionedAddress(F);
|
2016-12-20 04:36:41 +08:00
|
|
|
if (!LowPcAddr)
|
2016-12-15 06:38:08 +08:00
|
|
|
return false;
|
2018-12-23 06:20:40 +08:00
|
|
|
if (auto HighPcAddr = getHighPC(LowPcAddr->Address)) {
|
|
|
|
LowPC = LowPcAddr->Address;
|
2016-12-20 04:36:41 +08:00
|
|
|
HighPC = *HighPcAddr;
|
2018-12-23 06:20:40 +08:00
|
|
|
SectionIndex = LowPcAddr->SectionIndex;
|
2016-12-20 04:36:41 +08:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
return false;
|
2016-12-14 02:25:19 +08:00
|
|
|
}
|
|
|
|
|
2018-06-21 06:56:37 +08:00
|
|
|
Expected<DWARFAddressRangesVector> DWARFDie::getAddressRanges() const {
|
2016-12-14 02:25:19 +08:00
|
|
|
if (isNULL())
|
|
|
|
return DWARFAddressRangesVector();
|
|
|
|
// Single range specified by low/high PC.
|
2017-05-28 02:10:23 +08:00
|
|
|
uint64_t LowPC, HighPC, Index;
|
|
|
|
if (getLowAndHighPC(LowPC, HighPC, Index))
|
2018-06-21 06:56:37 +08:00
|
|
|
return DWARFAddressRangesVector{{LowPC, HighPC, Index}};
|
2017-05-16 20:30:59 +08:00
|
|
|
|
2018-05-19 04:12:54 +08:00
|
|
|
Optional<DWARFFormValue> Value = find(DW_AT_ranges);
|
|
|
|
if (Value) {
|
|
|
|
if (Value->getForm() == DW_FORM_rnglistx)
|
|
|
|
return U->findRnglistFromIndex(*Value->getAsSectionOffset());
|
|
|
|
return U->findRnglistFromOffset(*Value->getAsSectionOffset());
|
2016-12-14 02:25:19 +08:00
|
|
|
}
|
|
|
|
return DWARFAddressRangesVector();
|
|
|
|
}
|
|
|
|
|
|
|
|
bool DWARFDie::addressRangeContainsAddress(const uint64_t Address) const {
|
2018-06-21 06:56:37 +08:00
|
|
|
auto RangesOrError = getAddressRanges();
|
|
|
|
if (!RangesOrError) {
|
|
|
|
llvm::consumeError(RangesOrError.takeError());
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
for (const auto &R : RangesOrError.get())
|
2017-05-16 20:30:59 +08:00
|
|
|
if (R.LowPC <= Address && Address < R.HighPC)
|
2016-12-14 02:25:19 +08:00
|
|
|
return true;
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
Recommit "[DWARF] Add an api to get "interpreted" location lists"
This recommits 089c0f581492cd6e2a3d2927be3fbf60ea2d7e62, which was
reverted due to failing tests on big endian machines. It includes a fix
which I believe (I don't have BE machine) should fix this issue. The fix
consists of correcting the invocation DWARFYAML::EmitDebugSections,
which was missing one (default) function arguments, and so didn't
actually force the little-endian mode.
The original commit message follows.
Summary:
This patch adds DWARFDie::getLocations, which returns the location
expressions for a given attribute (typically DW_AT_location). It handles
both "inline" locations and references to the external location list
sections (currently only of the DW_FORM_sec_offset type). It is
implemented on top of DWARFUnit::findLoclistFromOffset, which is also
added in this patch. I tried to make their signatures similar to the
equivalent range list functionality.
The actual location list interpretation logic is in
DWARFLocationTable::visitAbsoluteLocationList. This part is not
equivalent to the range list code, but this deviation is motivated by a
desire to reuse the same location list parsing code within lldb.
The functionality is tested via a c++ unit test of the DWARFDie API.
Reviewers: dblaikie, JDevlieghere, SouraVX
Subscribers: mgorny, hiraditya, cmtice, probinson, llvm-commits, aprantl
Tags: #llvm
Differential Revision: https://reviews.llvm.org/D70394
2019-11-08 22:24:11 +08:00
|
|
|
Expected<DWARFLocationExpressionsVector>
|
|
|
|
DWARFDie::getLocations(dwarf::Attribute Attr) const {
|
|
|
|
Optional<DWARFFormValue> Location = find(Attr);
|
|
|
|
if (!Location)
|
|
|
|
return createStringError(inconvertibleErrorCode(), "No %s",
|
|
|
|
dwarf::AttributeString(Attr).data());
|
|
|
|
|
2019-11-27 09:34:04 +08:00
|
|
|
if (Optional<uint64_t> Off = Location->getAsSectionOffset()) {
|
|
|
|
uint64_t Offset = *Off;
|
|
|
|
|
|
|
|
if (Location->getForm() == DW_FORM_loclistx) {
|
|
|
|
if (auto LoclistOffset = U->getLoclistOffset(Offset))
|
|
|
|
Offset = *LoclistOffset;
|
|
|
|
else
|
|
|
|
return createStringError(inconvertibleErrorCode(),
|
|
|
|
"Loclist table not found");
|
|
|
|
}
|
|
|
|
return U->findLoclistFromOffset(Offset);
|
|
|
|
}
|
Recommit "[DWARF] Add an api to get "interpreted" location lists"
This recommits 089c0f581492cd6e2a3d2927be3fbf60ea2d7e62, which was
reverted due to failing tests on big endian machines. It includes a fix
which I believe (I don't have BE machine) should fix this issue. The fix
consists of correcting the invocation DWARFYAML::EmitDebugSections,
which was missing one (default) function arguments, and so didn't
actually force the little-endian mode.
The original commit message follows.
Summary:
This patch adds DWARFDie::getLocations, which returns the location
expressions for a given attribute (typically DW_AT_location). It handles
both "inline" locations and references to the external location list
sections (currently only of the DW_FORM_sec_offset type). It is
implemented on top of DWARFUnit::findLoclistFromOffset, which is also
added in this patch. I tried to make their signatures similar to the
equivalent range list functionality.
The actual location list interpretation logic is in
DWARFLocationTable::visitAbsoluteLocationList. This part is not
equivalent to the range list code, but this deviation is motivated by a
desire to reuse the same location list parsing code within lldb.
The functionality is tested via a c++ unit test of the DWARFDie API.
Reviewers: dblaikie, JDevlieghere, SouraVX
Subscribers: mgorny, hiraditya, cmtice, probinson, llvm-commits, aprantl
Tags: #llvm
Differential Revision: https://reviews.llvm.org/D70394
2019-11-08 22:24:11 +08:00
|
|
|
|
|
|
|
if (Optional<ArrayRef<uint8_t>> Expr = Location->getAsBlock()) {
|
|
|
|
return DWARFLocationExpressionsVector{
|
|
|
|
DWARFLocationExpression{None, to_vector<4>(*Expr)}};
|
|
|
|
}
|
|
|
|
|
|
|
|
return createStringError(
|
|
|
|
inconvertibleErrorCode(), "Unsupported %s encoding: %s",
|
|
|
|
dwarf::AttributeString(Attr).data(),
|
|
|
|
dwarf::FormEncodingString(Location->getForm()).data());
|
|
|
|
}
|
|
|
|
|
2017-10-05 06:26:19 +08:00
|
|
|
const char *DWARFDie::getSubroutineName(DINameKind Kind) const {
|
2016-12-14 02:25:19 +08:00
|
|
|
if (!isSubroutineDIE())
|
|
|
|
return nullptr;
|
|
|
|
return getName(Kind);
|
|
|
|
}
|
|
|
|
|
2017-10-05 06:26:19 +08:00
|
|
|
const char *DWARFDie::getName(DINameKind Kind) const {
|
2016-12-14 02:25:19 +08:00
|
|
|
if (!isValid() || Kind == DINameKind::None)
|
|
|
|
return nullptr;
|
|
|
|
// Try to get mangled name only if it was asked for.
|
|
|
|
if (Kind == DINameKind::LinkageName) {
|
2020-04-30 19:05:17 +08:00
|
|
|
if (auto Name = getLinkageName())
|
2017-01-14 05:08:18 +08:00
|
|
|
return Name;
|
2016-12-14 02:25:19 +08:00
|
|
|
}
|
2020-04-30 19:05:17 +08:00
|
|
|
return getShortName();
|
|
|
|
}
|
|
|
|
|
|
|
|
const char *DWARFDie::getShortName() const {
|
|
|
|
if (!isValid())
|
|
|
|
return nullptr;
|
|
|
|
|
|
|
|
return dwarf::toString(findRecursively(dwarf::DW_AT_name), nullptr);
|
|
|
|
}
|
|
|
|
|
|
|
|
const char *DWARFDie::getLinkageName() const {
|
|
|
|
if (!isValid())
|
|
|
|
return nullptr;
|
|
|
|
|
|
|
|
return dwarf::toString(findRecursively({dwarf::DW_AT_MIPS_linkage_name,
|
|
|
|
dwarf::DW_AT_linkage_name}),
|
|
|
|
nullptr);
|
2016-12-14 02:25:19 +08:00
|
|
|
}
|
|
|
|
|
2017-02-07 04:19:02 +08:00
|
|
|
uint64_t DWARFDie::getDeclLine() const {
|
|
|
|
return toUnsigned(findRecursively(DW_AT_decl_line), 0);
|
|
|
|
}
|
|
|
|
|
2020-09-09 05:05:20 +08:00
|
|
|
std::string
|
|
|
|
DWARFDie::getDeclFile(DILineInfoSpecifier::FileLineInfoKind Kind) const {
|
2021-08-21 03:10:21 +08:00
|
|
|
if (auto FormValue = findRecursively(DW_AT_decl_file))
|
|
|
|
if (auto OptString = FormValue->getAsFile(Kind))
|
|
|
|
return *OptString;
|
|
|
|
return {};
|
2020-09-09 05:05:20 +08:00
|
|
|
}
|
|
|
|
|
2016-12-14 02:25:19 +08:00
|
|
|
void DWARFDie::getCallerFrame(uint32_t &CallFile, uint32_t &CallLine,
|
2017-04-18 04:10:39 +08:00
|
|
|
uint32_t &CallColumn,
|
|
|
|
uint32_t &CallDiscriminator) const {
|
2017-01-14 05:08:18 +08:00
|
|
|
CallFile = toUnsigned(find(DW_AT_call_file), 0);
|
|
|
|
CallLine = toUnsigned(find(DW_AT_call_line), 0);
|
|
|
|
CallColumn = toUnsigned(find(DW_AT_call_column), 0);
|
2017-04-18 04:10:39 +08:00
|
|
|
CallDiscriminator = toUnsigned(find(DW_AT_GNU_discriminator), 0);
|
2016-12-14 02:25:19 +08:00
|
|
|
}
|
|
|
|
|
2017-09-19 05:27:44 +08:00
|
|
|
/// Helper to dump a DIE with all of its parents, but no siblings.
|
|
|
|
static unsigned dumpParentChain(DWARFDie Die, raw_ostream &OS, unsigned Indent,
|
2019-05-25 05:11:28 +08:00
|
|
|
DIDumpOptions DumpOpts, unsigned Depth = 0) {
|
2017-09-19 05:27:44 +08:00
|
|
|
if (!Die)
|
|
|
|
return Indent;
|
2019-05-25 05:11:28 +08:00
|
|
|
if (DumpOpts.ParentRecurseDepth > 0 && Depth >= DumpOpts.ParentRecurseDepth)
|
|
|
|
return Indent;
|
|
|
|
Indent = dumpParentChain(Die.getParent(), OS, Indent, DumpOpts, Depth + 1);
|
2017-09-21 01:44:00 +08:00
|
|
|
Die.dump(OS, Indent, DumpOpts);
|
2017-09-19 05:27:44 +08:00
|
|
|
return Indent + 2;
|
|
|
|
}
|
|
|
|
|
2017-09-21 01:44:00 +08:00
|
|
|
void DWARFDie::dump(raw_ostream &OS, unsigned Indent,
|
2017-06-07 07:28:45 +08:00
|
|
|
DIDumpOptions DumpOpts) const {
|
2016-12-14 02:25:19 +08:00
|
|
|
if (!isValid())
|
|
|
|
return;
|
2017-06-30 00:52:08 +08:00
|
|
|
DWARFDataExtractor debug_info_data = U->getDebugInfoExtractor();
|
2019-08-06 18:49:40 +08:00
|
|
|
const uint64_t Offset = getOffset();
|
|
|
|
uint64_t offset = Offset;
|
2017-09-19 05:27:44 +08:00
|
|
|
if (DumpOpts.ShowParents) {
|
2018-05-27 03:39:56 +08:00
|
|
|
DIDumpOptions ParentDumpOpts = DumpOpts;
|
|
|
|
ParentDumpOpts.ShowParents = false;
|
|
|
|
ParentDumpOpts.ShowChildren = false;
|
|
|
|
Indent = dumpParentChain(getParent(), OS, Indent, ParentDumpOpts);
|
2017-09-19 05:27:44 +08:00
|
|
|
}
|
2017-08-19 05:35:44 +08:00
|
|
|
|
2016-12-14 02:25:19 +08:00
|
|
|
if (debug_info_data.isValidOffset(offset)) {
|
|
|
|
uint32_t abbrCode = debug_info_data.getULEB128(&offset);
|
2017-12-09 07:32:47 +08:00
|
|
|
if (DumpOpts.ShowAddresses)
|
2018-03-09 17:56:24 +08:00
|
|
|
WithColor(OS, HighlightColor::Address).get()
|
2019-08-06 18:49:40 +08:00
|
|
|
<< format("\n0x%8.8" PRIx64 ": ", Offset);
|
2017-08-19 05:35:44 +08:00
|
|
|
|
2016-12-14 02:25:19 +08:00
|
|
|
if (abbrCode) {
|
|
|
|
auto AbbrevDecl = getAbbreviationDeclarationPtr();
|
|
|
|
if (AbbrevDecl) {
|
[dwarf] Unify unknown dwarf enum formatting code
Summary:
We have had at least three pieces of code (in DWARFAbbreviationDeclaration,
DWARFAcceleratorTable and DWARFDie) that have hand-rolled support for
dumping unknown dwarf enum values. While not terrible, they are a bit
distracting and enable small differences to creep in (Unknown_ffff vs.
Unknown_0xffff). I ended up needing to add a fourth place
(DWARFVerifier), so it seems it would be a good time to centralize.
This patch creates an alternative to the XXXString dumping functions in
the BinaryFormat library, which formats an unknown value as
DW_TYPE_unknown_1234, instead of just an empty string. It is based on
the formatv function, as that allows us to avoid materializing the
string for unknown values (and because this way I don't have to invent a
name for the new functions :P).
In this patch I add formatters for dwarf attributes, forms, tags, and
index attributes as these are the ones in use currently, but adding
other enums is straight-forward.
Reviewers: dblaikie, JDevlieghere, aprantl
Subscribers: llvm-commits
Differential Revision: https://reviews.llvm.org/D44570
llvm-svn: 328090
2018-03-21 19:46:37 +08:00
|
|
|
WithColor(OS, HighlightColor::Tag).get().indent(Indent)
|
|
|
|
<< formatv("{0}", getTag());
|
2021-11-08 23:34:32 +08:00
|
|
|
if (DumpOpts.Verbose) {
|
2017-06-07 07:28:45 +08:00
|
|
|
OS << format(" [%u] %c", abbrCode,
|
|
|
|
AbbrevDecl->hasChildren() ? '*' : ' ');
|
2021-11-08 23:34:32 +08:00
|
|
|
if (Optional<uint32_t> ParentIdx = Die->getParentIdx())
|
|
|
|
OS << format(" (0x%8.8" PRIx64 ")",
|
|
|
|
U->getDIEAtIndex(*ParentIdx).getOffset());
|
|
|
|
}
|
2017-06-07 07:28:45 +08:00
|
|
|
OS << '\n';
|
|
|
|
|
2016-12-14 02:25:19 +08:00
|
|
|
// Dump all data in the DIE for the attributes.
|
2021-03-09 16:26:17 +08:00
|
|
|
for (const DWARFAttribute &AttrValue : attributes())
|
2021-03-09 15:31:23 +08:00
|
|
|
dumpAttribute(OS, *this, AttrValue, Indent, DumpOpts);
|
2017-08-19 05:35:44 +08:00
|
|
|
|
2021-04-15 08:04:19 +08:00
|
|
|
if (DumpOpts.ShowChildren && DumpOpts.ChildRecurseDepth > 0) {
|
|
|
|
DWARFDie Child = getFirstChild();
|
2019-05-25 05:11:28 +08:00
|
|
|
DumpOpts.ChildRecurseDepth--;
|
2018-05-27 03:39:56 +08:00
|
|
|
DIDumpOptions ChildDumpOpts = DumpOpts;
|
|
|
|
ChildDumpOpts.ShowParents = false;
|
2021-04-15 08:04:19 +08:00
|
|
|
while (Child) {
|
|
|
|
Child.dump(OS, Indent + 2, ChildDumpOpts);
|
|
|
|
Child = Child.getSibling();
|
2016-12-14 02:25:19 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
OS << "Abbreviation code not found in 'debug_abbrev' class for code: "
|
2017-10-05 06:26:19 +08:00
|
|
|
<< abbrCode << '\n';
|
2016-12-14 02:25:19 +08:00
|
|
|
}
|
|
|
|
} else {
|
|
|
|
OS.indent(Indent) << "NULL\n";
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-08-17 01:43:01 +08:00
|
|
|
LLVM_DUMP_METHOD void DWARFDie::dump() const { dump(llvm::errs(), 0); }
|
|
|
|
|
2016-12-22 05:37:06 +08:00
|
|
|
DWARFDie DWARFDie::getParent() const {
|
|
|
|
if (isValid())
|
|
|
|
return U->getParent(Die);
|
|
|
|
return DWARFDie();
|
|
|
|
}
|
|
|
|
|
|
|
|
DWARFDie DWARFDie::getSibling() const {
|
|
|
|
if (isValid())
|
|
|
|
return U->getSibling(Die);
|
|
|
|
return DWARFDie();
|
|
|
|
}
|
2017-01-13 08:13:42 +08:00
|
|
|
|
2018-07-12 01:11:11 +08:00
|
|
|
DWARFDie DWARFDie::getPreviousSibling() const {
|
|
|
|
if (isValid())
|
|
|
|
return U->getPreviousSibling(Die);
|
|
|
|
return DWARFDie();
|
|
|
|
}
|
|
|
|
|
2017-10-25 18:23:49 +08:00
|
|
|
DWARFDie DWARFDie::getFirstChild() const {
|
|
|
|
if (isValid())
|
|
|
|
return U->getFirstChild(Die);
|
|
|
|
return DWARFDie();
|
|
|
|
}
|
|
|
|
|
2018-07-12 01:11:11 +08:00
|
|
|
DWARFDie DWARFDie::getLastChild() const {
|
|
|
|
if (isValid())
|
|
|
|
return U->getLastChild(Die);
|
|
|
|
return DWARFDie();
|
|
|
|
}
|
|
|
|
|
2017-10-05 06:26:19 +08:00
|
|
|
iterator_range<DWARFDie::attribute_iterator> DWARFDie::attributes() const {
|
2017-01-13 08:13:42 +08:00
|
|
|
return make_range(attribute_iterator(*this, false),
|
|
|
|
attribute_iterator(*this, true));
|
|
|
|
}
|
|
|
|
|
2017-10-05 06:26:19 +08:00
|
|
|
DWARFDie::attribute_iterator::attribute_iterator(DWARFDie D, bool End)
|
2019-07-16 14:53:06 +08:00
|
|
|
: Die(D), Index(0) {
|
2017-01-13 08:13:42 +08:00
|
|
|
auto AbbrDecl = Die.getAbbreviationDeclarationPtr();
|
|
|
|
assert(AbbrDecl && "Must have abbreviation declaration");
|
|
|
|
if (End) {
|
|
|
|
// This is the end iterator so we set the index to the attribute count.
|
|
|
|
Index = AbbrDecl->getNumAttributes();
|
|
|
|
} else {
|
|
|
|
// This is the begin iterator so we extract the value for this->Index.
|
|
|
|
AttrValue.Offset = D.getOffset() + AbbrDecl->getCodeByteSize();
|
|
|
|
updateForIndex(*AbbrDecl, 0);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void DWARFDie::attribute_iterator::updateForIndex(
|
|
|
|
const DWARFAbbreviationDeclaration &AbbrDecl, uint32_t I) {
|
|
|
|
Index = I;
|
2017-07-03 14:32:59 +08:00
|
|
|
// AbbrDecl must be valid before calling this function.
|
2017-01-13 08:13:42 +08:00
|
|
|
auto NumAttrs = AbbrDecl.getNumAttributes();
|
|
|
|
if (Index < NumAttrs) {
|
|
|
|
AttrValue.Attr = AbbrDecl.getAttrByIndex(Index);
|
|
|
|
// Add the previous byte size of any previous attribute value.
|
|
|
|
AttrValue.Offset += AttrValue.ByteSize;
|
2019-08-06 18:49:40 +08:00
|
|
|
uint64_t ParseOffset = AttrValue.Offset;
|
2021-03-09 16:26:17 +08:00
|
|
|
if (AbbrDecl.getAttrIsImplicitConstByIndex(Index))
|
|
|
|
AttrValue.Value = DWARFFormValue::createFromSValue(
|
|
|
|
AbbrDecl.getFormByIndex(Index),
|
|
|
|
AbbrDecl.getAttrImplicitConstValueByIndex(Index));
|
|
|
|
else {
|
|
|
|
auto U = Die.getDwarfUnit();
|
|
|
|
assert(U && "Die must have valid DWARF unit");
|
|
|
|
AttrValue.Value = DWARFFormValue::createFromUnit(
|
|
|
|
AbbrDecl.getFormByIndex(Index), U, &ParseOffset);
|
|
|
|
}
|
2017-01-13 08:13:42 +08:00
|
|
|
AttrValue.ByteSize = ParseOffset - AttrValue.Offset;
|
|
|
|
} else {
|
|
|
|
assert(Index == NumAttrs && "Indexes should be [0, NumAttrs) only");
|
2019-07-16 14:53:06 +08:00
|
|
|
AttrValue = {};
|
2017-01-13 08:13:42 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
DWARFDie::attribute_iterator &DWARFDie::attribute_iterator::operator++() {
|
|
|
|
if (auto AbbrDecl = Die.getAbbreviationDeclarationPtr())
|
|
|
|
updateForIndex(*AbbrDecl, Index + 1);
|
|
|
|
return *this;
|
|
|
|
}
|
2019-03-01 06:12:32 +08:00
|
|
|
|
2021-07-27 15:28:59 +08:00
|
|
|
bool DWARFAttribute::mayHaveLocationList(dwarf::Attribute Attr) {
|
|
|
|
switch(Attr) {
|
|
|
|
case DW_AT_location:
|
|
|
|
case DW_AT_string_length:
|
|
|
|
case DW_AT_return_addr:
|
|
|
|
case DW_AT_data_member_location:
|
|
|
|
case DW_AT_frame_base:
|
|
|
|
case DW_AT_static_link:
|
|
|
|
case DW_AT_segment:
|
|
|
|
case DW_AT_use_location:
|
|
|
|
case DW_AT_vtable_elem_location:
|
|
|
|
return true;
|
|
|
|
default:
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
bool DWARFAttribute::mayHaveLocationExpr(dwarf::Attribute Attr) {
|
2019-03-01 06:12:32 +08:00
|
|
|
switch (Attr) {
|
|
|
|
// From the DWARF v5 specification.
|
|
|
|
case DW_AT_location:
|
|
|
|
case DW_AT_byte_size:
|
2021-07-27 15:28:59 +08:00
|
|
|
case DW_AT_bit_offset:
|
2019-03-01 06:12:32 +08:00
|
|
|
case DW_AT_bit_size:
|
|
|
|
case DW_AT_string_length:
|
|
|
|
case DW_AT_lower_bound:
|
|
|
|
case DW_AT_return_addr:
|
|
|
|
case DW_AT_bit_stride:
|
|
|
|
case DW_AT_upper_bound:
|
|
|
|
case DW_AT_count:
|
|
|
|
case DW_AT_data_member_location:
|
|
|
|
case DW_AT_frame_base:
|
|
|
|
case DW_AT_segment:
|
|
|
|
case DW_AT_static_link:
|
|
|
|
case DW_AT_use_location:
|
|
|
|
case DW_AT_vtable_elem_location:
|
|
|
|
case DW_AT_allocated:
|
|
|
|
case DW_AT_associated:
|
2021-07-27 15:28:59 +08:00
|
|
|
case DW_AT_data_location:
|
2019-03-01 06:12:32 +08:00
|
|
|
case DW_AT_byte_stride:
|
|
|
|
case DW_AT_rank:
|
|
|
|
case DW_AT_call_value:
|
|
|
|
case DW_AT_call_origin:
|
|
|
|
case DW_AT_call_target:
|
|
|
|
case DW_AT_call_target_clobbered:
|
|
|
|
case DW_AT_call_data_location:
|
|
|
|
case DW_AT_call_data_value:
|
|
|
|
// Extensions.
|
|
|
|
case DW_AT_GNU_call_site_value:
|
2019-08-01 00:51:28 +08:00
|
|
|
case DW_AT_GNU_call_site_target:
|
2019-03-01 06:12:32 +08:00
|
|
|
return true;
|
|
|
|
default:
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
}
|
2022-01-12 23:42:48 +08:00
|
|
|
|
|
|
|
namespace llvm {
|
|
|
|
|
|
|
|
void dumpTypeQualifiedName(const DWARFDie &DIE, raw_ostream &OS) {
|
|
|
|
DWARFTypePrinter(OS).appendQualifiedName(DIE);
|
|
|
|
}
|
|
|
|
|
|
|
|
void dumpTypeUnqualifiedName(const DWARFDie &DIE, raw_ostream &OS,
|
|
|
|
std::string *OriginalFullName) {
|
|
|
|
DWARFTypePrinter(OS).appendUnqualifiedName(DIE, OriginalFullName);
|
|
|
|
}
|
|
|
|
|
|
|
|
} // namespace llvm
|