mirror of
https://github.com/RPCSX/llvm.git
synced 2025-01-05 19:29:01 +00:00
[dsymutil] Correctly clone address attributes.
DW_AT_low_pc on functions is taken care of by the relocation processing, but DW_AT_high_pc and DW_AT_low_pc on other lexical scopes need special handling. git-svn-id: https://llvm.org/svn/llvm-project/llvm/trunk@231955 91177308-0d34-0410-b5e6-96231b3b80d8
This commit is contained in:
parent
ca75f1df99
commit
89393bbba6
@ -25,6 +25,7 @@ CHECK: DW_AT_type [DW_FORM_ref4] (cu + 0x0063 => {0x00000063})
|
|||||||
CHECK: DW_AT_external [DW_FORM_flag] (0x01)
|
CHECK: DW_AT_external [DW_FORM_flag] (0x01)
|
||||||
CHECK: DW_AT_accessibility [DW_FORM_data1] (DW_ACCESS_public)
|
CHECK: DW_AT_accessibility [DW_FORM_data1] (DW_ACCESS_public)
|
||||||
CHECK: DW_AT_low_pc [DW_FORM_addr] (0x0000000100000ea0)
|
CHECK: DW_AT_low_pc [DW_FORM_addr] (0x0000000100000ea0)
|
||||||
|
CHECK: DW_AT_high_pc [DW_FORM_addr] (0x0000000100000ec4)
|
||||||
CHECK: DW_AT_frame_base [DW_FORM_block1] (<0x01> 56 )
|
CHECK: DW_AT_frame_base [DW_FORM_block1] (<0x01> 56 )
|
||||||
CHECK: DW_TAG_formal_parameter [3]
|
CHECK: DW_TAG_formal_parameter [3]
|
||||||
CHECK: DW_AT_name [DW_FORM_strp] ( .debug_str[0x00000056] = "argc")
|
CHECK: DW_AT_name [DW_FORM_strp] ( .debug_str[0x00000056] = "argc")
|
||||||
@ -75,6 +76,7 @@ CHECK: DW_TAG_subprogram [2] *
|
|||||||
CHECK: DW_AT_name [DW_FORM_strp] ( .debug_str[0x00000082] = "foo")
|
CHECK: DW_AT_name [DW_FORM_strp] ( .debug_str[0x00000082] = "foo")
|
||||||
CHECK: DW_AT_type [DW_FORM_ref4] (cu + 0x0026 => {0x000000a7})
|
CHECK: DW_AT_type [DW_FORM_ref4] (cu + 0x0026 => {0x000000a7})
|
||||||
CHECK: DW_AT_low_pc [DW_FORM_addr] (0x0000000100000ed0)
|
CHECK: DW_AT_low_pc [DW_FORM_addr] (0x0000000100000ed0)
|
||||||
|
CHECK: DW_AT_high_pc [DW_FORM_addr] (0x0000000100000f19)
|
||||||
CHECK: DW_AT_frame_base [DW_FORM_block1] (<0x01> 56 )
|
CHECK: DW_AT_frame_base [DW_FORM_block1] (<0x01> 56 )
|
||||||
CHECK: DW_TAG_formal_parameter [3]
|
CHECK: DW_TAG_formal_parameter [3]
|
||||||
CHECK: DW_AT_name [DW_FORM_strp] ( .debug_str[0x00000086] = "arg")
|
CHECK: DW_AT_name [DW_FORM_strp] ( .debug_str[0x00000086] = "arg")
|
||||||
@ -85,6 +87,7 @@ CHECK: DW_TAG_subprogram [8]
|
|||||||
CHECK: DW_AT_name [DW_FORM_strp] ( .debug_str[0x0000008a] = "inc")
|
CHECK: DW_AT_name [DW_FORM_strp] ( .debug_str[0x0000008a] = "inc")
|
||||||
CHECK: DW_AT_type [DW_FORM_ref4] (cu + 0x0026 => {0x000000a7})
|
CHECK: DW_AT_type [DW_FORM_ref4] (cu + 0x0026 => {0x000000a7})
|
||||||
CHECK: DW_AT_low_pc [DW_FORM_addr] (0x0000000100000f20)
|
CHECK: DW_AT_low_pc [DW_FORM_addr] (0x0000000100000f20)
|
||||||
|
CHECK: DW_AT_high_pc [DW_FORM_addr] (0x0000000100000f37)
|
||||||
CHECK: DW_AT_frame_base [DW_FORM_block1] (<0x01> 56 )
|
CHECK: DW_AT_frame_base [DW_FORM_block1] (<0x01> 56 )
|
||||||
CHECK: NULL
|
CHECK: NULL
|
||||||
|
|
||||||
@ -107,6 +110,7 @@ CHECK: DW_TAG_subprogram [2] *
|
|||||||
CHECK: DW_AT_name [DW_FORM_strp] ( .debug_str[0x0000009b] = "bar")
|
CHECK: DW_AT_name [DW_FORM_strp] ( .debug_str[0x0000009b] = "bar")
|
||||||
CHECK: DW_AT_type [DW_FORM_ref4] (cu + 0x0041 => {0x00000167})
|
CHECK: DW_AT_type [DW_FORM_ref4] (cu + 0x0041 => {0x00000167})
|
||||||
CHECK: DW_AT_low_pc [DW_FORM_addr] (0x0000000100000f40)
|
CHECK: DW_AT_low_pc [DW_FORM_addr] (0x0000000100000f40)
|
||||||
|
CHECK: DW_AT_high_pc [DW_FORM_addr] (0x0000000100000f84)
|
||||||
CHECK: DW_AT_frame_base [DW_FORM_block1] (<0x01> 56 )
|
CHECK: DW_AT_frame_base [DW_FORM_block1] (<0x01> 56 )
|
||||||
CHECK: DW_TAG_formal_parameter [3]
|
CHECK: DW_TAG_formal_parameter [3]
|
||||||
CHECK: DW_AT_name [DW_FORM_strp] ( .debug_str[0x00000086] = "arg")
|
CHECK: DW_AT_name [DW_FORM_strp] ( .debug_str[0x00000086] = "arg")
|
||||||
@ -117,6 +121,7 @@ CHECK: DW_TAG_subprogram [8]
|
|||||||
CHECK: DW_AT_name [DW_FORM_strp] ( .debug_str[0x0000008a] = "inc")
|
CHECK: DW_AT_name [DW_FORM_strp] ( .debug_str[0x0000008a] = "inc")
|
||||||
CHECK: DW_AT_type [DW_FORM_ref4] (cu + 0x0041 => {0x00000167})
|
CHECK: DW_AT_type [DW_FORM_ref4] (cu + 0x0041 => {0x00000167})
|
||||||
CHECK: DW_AT_low_pc [DW_FORM_addr] (0x0000000100000f90)
|
CHECK: DW_AT_low_pc [DW_FORM_addr] (0x0000000100000f90)
|
||||||
|
CHECK: DW_AT_high_pc [DW_FORM_addr] (0x0000000100000fa9)
|
||||||
CHECK: DW_AT_frame_base [DW_FORM_block1] (<0x01> 56 )
|
CHECK: DW_AT_frame_base [DW_FORM_block1] (<0x01> 56 )
|
||||||
|
|
||||||
CHECK: NULL
|
CHECK: NULL
|
||||||
|
@ -19,6 +19,7 @@ CHECK: DW_AT_type [DW_FORM_ref4] (cu + 0x0063 => {0x00000063})
|
|||||||
CHECK: DW_AT_external [DW_FORM_flag] (0x01)
|
CHECK: DW_AT_external [DW_FORM_flag] (0x01)
|
||||||
CHECK: DW_AT_accessibility [DW_FORM_data1] (DW_ACCESS_public)
|
CHECK: DW_AT_accessibility [DW_FORM_data1] (DW_ACCESS_public)
|
||||||
CHECK: DW_AT_low_pc [DW_FORM_addr] (0x0000000100000f40)
|
CHECK: DW_AT_low_pc [DW_FORM_addr] (0x0000000100000f40)
|
||||||
|
CHECK: DW_AT_high_pc [DW_FORM_addr] (0x0000000100000f4b)
|
||||||
CHECK: DW_AT_frame_base [DW_FORM_block1] (<0x01> 56 )
|
CHECK: DW_AT_frame_base [DW_FORM_block1] (<0x01> 56 )
|
||||||
CHECK: DW_TAG_formal_parameter [3]
|
CHECK: DW_TAG_formal_parameter [3]
|
||||||
CHECK: DW_AT_name [DW_FORM_strp] ( .debug_str[0x00000056] = "argc")
|
CHECK: DW_AT_name [DW_FORM_strp] ( .debug_str[0x00000056] = "argc")
|
||||||
@ -63,12 +64,15 @@ CHECK: DW_TAG_subprogram [8] *
|
|||||||
CHECK: DW_AT_name [DW_FORM_strp] ( .debug_str[0x00000082] = "foo")
|
CHECK: DW_AT_name [DW_FORM_strp] ( .debug_str[0x00000082] = "foo")
|
||||||
CHECK: DW_AT_type [DW_FORM_ref_addr] (0x0000000000000063)
|
CHECK: DW_AT_type [DW_FORM_ref_addr] (0x0000000000000063)
|
||||||
CHECK: DW_AT_low_pc [DW_FORM_addr] (0x0000000100000f50)
|
CHECK: DW_AT_low_pc [DW_FORM_addr] (0x0000000100000f50)
|
||||||
|
CHECK: DW_AT_high_pc [DW_FORM_addr] (0x0000000100000f89)
|
||||||
CHECK: DW_AT_frame_base [DW_FORM_block1] (<0x01> 56 )
|
CHECK: DW_AT_frame_base [DW_FORM_block1] (<0x01> 56 )
|
||||||
CHECK: DW_TAG_formal_parameter [9]
|
CHECK: DW_TAG_formal_parameter [9]
|
||||||
CHECK: DW_AT_name [DW_FORM_strp] ( .debug_str[0x00000086] = "arg")
|
CHECK: DW_AT_name [DW_FORM_strp] ( .debug_str[0x00000086] = "arg")
|
||||||
CHECK: DW_AT_type [DW_FORM_ref_addr] (0x0000000000000063)
|
CHECK: DW_AT_type [DW_FORM_ref_addr] (0x0000000000000063)
|
||||||
CHECK: DW_TAG_inlined_subroutine [10]
|
CHECK: DW_TAG_inlined_subroutine [10]
|
||||||
CHECK: DW_AT_abstract_origin [DW_FORM_ref4] (cu + 0x00a7 => {0x00000128} "inc")
|
CHECK: DW_AT_abstract_origin [DW_FORM_ref4] (cu + 0x00a7 => {0x00000128} "inc")
|
||||||
|
CHECK: DW_AT_low_pc [DW_FORM_addr] (0x0000000100000f63)
|
||||||
|
CHECK: DW_AT_high_pc [DW_FORM_addr] (0x0000000100000f72)
|
||||||
CHECK: DW_AT_call_line [DW_FORM_data1] (20)
|
CHECK: DW_AT_call_line [DW_FORM_data1] (20)
|
||||||
CHECK: NULL
|
CHECK: NULL
|
||||||
CHECK: DW_TAG_subprogram [11]
|
CHECK: DW_TAG_subprogram [11]
|
||||||
@ -93,11 +97,14 @@ CHECK: DW_TAG_subprogram [8] *
|
|||||||
CHECK: DW_AT_name [DW_FORM_strp] ( .debug_str[0x0000009b] = "bar")
|
CHECK: DW_AT_name [DW_FORM_strp] ( .debug_str[0x0000009b] = "bar")
|
||||||
CHECK: DW_AT_type [DW_FORM_ref_addr] (0x0000000000000063)
|
CHECK: DW_AT_type [DW_FORM_ref_addr] (0x0000000000000063)
|
||||||
CHECK: DW_AT_low_pc [DW_FORM_addr] (0x0000000100000f90)
|
CHECK: DW_AT_low_pc [DW_FORM_addr] (0x0000000100000f90)
|
||||||
|
CHECK: DW_AT_high_pc [DW_FORM_addr] (0x0000000100000fb4)
|
||||||
CHECK: DW_AT_frame_base [DW_FORM_block1] (<0x01> 56 )
|
CHECK: DW_AT_frame_base [DW_FORM_block1] (<0x01> 56 )
|
||||||
CHECK: DW_TAG_formal_parameter [9]
|
CHECK: DW_TAG_formal_parameter [9]
|
||||||
CHECK: DW_AT_name [DW_FORM_strp] ( .debug_str[0x00000086] = "arg")
|
CHECK: DW_AT_name [DW_FORM_strp] ( .debug_str[0x00000086] = "arg")
|
||||||
CHECK: DW_AT_type [DW_FORM_ref_addr] (0x0000000000000063)
|
CHECK: DW_AT_type [DW_FORM_ref_addr] (0x0000000000000063)
|
||||||
CHECK: DW_TAG_lexical_block [14] *
|
CHECK: DW_TAG_lexical_block [14] *
|
||||||
|
CHECK: DW_AT_low_pc [DW_FORM_addr] (0x0000000100000f94)
|
||||||
|
CHECK DW_AT_high_pc [DW_FORM_addr] (0x0000000100000fa7)
|
||||||
CHECK: DW_TAG_inlined_subroutine [15]
|
CHECK: DW_TAG_inlined_subroutine [15]
|
||||||
CHECK: DW_AT_abstract_origin [DW_FORM_ref4] (cu + 0x009a => {0x000001d4} "inc")
|
CHECK: DW_AT_abstract_origin [DW_FORM_ref4] (cu + 0x009a => {0x000001d4} "inc")
|
||||||
|
|
||||||
|
@ -54,7 +54,7 @@ class CompileUnit {
|
|||||||
public:
|
public:
|
||||||
/// \brief Information gathered about a DIE in the object file.
|
/// \brief Information gathered about a DIE in the object file.
|
||||||
struct DIEInfo {
|
struct DIEInfo {
|
||||||
uint64_t Address; ///< Linked address of the described entity.
|
int64_t AddrAdjust; ///< Address offset to apply to the described entity.
|
||||||
DIE *Clone; ///< Cloned version of that DIE.
|
DIE *Clone; ///< Cloned version of that DIE.
|
||||||
uint32_t ParentIdx; ///< The index of this DIE's parent.
|
uint32_t ParentIdx; ///< The index of this DIE's parent.
|
||||||
bool Keep; ///< Is the DIE part of the linked output?
|
bool Keep; ///< Is the DIE part of the linked output?
|
||||||
@ -529,17 +529,29 @@ private:
|
|||||||
///
|
///
|
||||||
/// \param OutOffset is the offset the cloned DIE in the output
|
/// \param OutOffset is the offset the cloned DIE in the output
|
||||||
/// compile unit.
|
/// compile unit.
|
||||||
|
/// \param PCOffset (while cloning a function scope) is the offset
|
||||||
|
/// applied to the entry point of the function to get the linked address.
|
||||||
///
|
///
|
||||||
/// \returns the root of the cloned tree.
|
/// \returns the root of the cloned tree.
|
||||||
DIE *cloneDIE(const DWARFDebugInfoEntryMinimal &InputDIE, CompileUnit &U,
|
DIE *cloneDIE(const DWARFDebugInfoEntryMinimal &InputDIE, CompileUnit &U,
|
||||||
uint32_t OutOffset);
|
int64_t PCOffset, uint32_t OutOffset);
|
||||||
|
|
||||||
typedef DWARFAbbreviationDeclaration::AttributeSpec AttributeSpec;
|
typedef DWARFAbbreviationDeclaration::AttributeSpec AttributeSpec;
|
||||||
|
|
||||||
|
/// \brief Information gathered and exchanged between the various
|
||||||
|
/// clone*Attributes helpers about the attributes of a particular DIE.
|
||||||
|
struct AttributesInfo {
|
||||||
|
uint64_t OrigHighPc; ///< Value of AT_high_pc in the input DIE
|
||||||
|
int64_t PCOffset; ///< Offset to apply to PC addresses inside a function.
|
||||||
|
|
||||||
|
AttributesInfo() : OrigHighPc(0), PCOffset(0) {}
|
||||||
|
};
|
||||||
|
|
||||||
/// \brief Helper for cloneDIE.
|
/// \brief Helper for cloneDIE.
|
||||||
unsigned cloneAttribute(DIE &Die, const DWARFDebugInfoEntryMinimal &InputDIE,
|
unsigned cloneAttribute(DIE &Die, const DWARFDebugInfoEntryMinimal &InputDIE,
|
||||||
CompileUnit &U, const DWARFFormValue &Val,
|
CompileUnit &U, const DWARFFormValue &Val,
|
||||||
const AttributeSpec AttrSpec, unsigned AttrSize);
|
const AttributeSpec AttrSpec, unsigned AttrSize,
|
||||||
|
AttributesInfo &AttrInfo);
|
||||||
|
|
||||||
/// \brief Helper for cloneDIE.
|
/// \brief Helper for cloneDIE.
|
||||||
unsigned cloneStringAttribute(DIE &Die, AttributeSpec AttrSpec,
|
unsigned cloneStringAttribute(DIE &Die, AttributeSpec AttrSpec,
|
||||||
@ -556,6 +568,11 @@ private:
|
|||||||
unsigned cloneBlockAttribute(DIE &Die, AttributeSpec AttrSpec,
|
unsigned cloneBlockAttribute(DIE &Die, AttributeSpec AttrSpec,
|
||||||
const DWARFFormValue &Val, unsigned AttrSize);
|
const DWARFFormValue &Val, unsigned AttrSize);
|
||||||
|
|
||||||
|
/// \brief Helper for cloneDIE.
|
||||||
|
unsigned cloneAddressAttribute(DIE &Die, AttributeSpec AttrSpec,
|
||||||
|
const DWARFFormValue &Val,
|
||||||
|
const CompileUnit &Unit, AttributesInfo &Info);
|
||||||
|
|
||||||
/// \brief Helper for cloneDIE.
|
/// \brief Helper for cloneDIE.
|
||||||
unsigned cloneScalarAttribute(DIE &Die,
|
unsigned cloneScalarAttribute(DIE &Die,
|
||||||
const DWARFDebugInfoEntryMinimal &InputDIE,
|
const DWARFDebugInfoEntryMinimal &InputDIE,
|
||||||
@ -831,8 +848,9 @@ bool DwarfLinker::hasValidRelocation(uint32_t StartOffset, uint32_t EndOffset,
|
|||||||
ValidReloc.Mapping->getValue().ObjectAddress,
|
ValidReloc.Mapping->getValue().ObjectAddress,
|
||||||
ValidReloc.Mapping->getValue().BinaryAddress);
|
ValidReloc.Mapping->getValue().BinaryAddress);
|
||||||
|
|
||||||
Info.Address =
|
Info.AddrAdjust = int64_t(ValidReloc.Mapping->getValue().BinaryAddress) +
|
||||||
ValidReloc.Mapping->getValue().BinaryAddress + ValidReloc.Addend;
|
ValidReloc.Addend -
|
||||||
|
ValidReloc.Mapping->getValue().ObjectAddress;
|
||||||
Info.InDebugMap = true;
|
Info.InDebugMap = true;
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
@ -1189,6 +1207,30 @@ unsigned DwarfLinker::cloneBlockAttribute(DIE &Die, AttributeSpec AttrSpec,
|
|||||||
return AttrSize;
|
return AttrSize;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/// \brief Clone an address attribute and add it to \p Die.
|
||||||
|
/// \returns the size of the new attribute.
|
||||||
|
unsigned DwarfLinker::cloneAddressAttribute(DIE &Die, AttributeSpec AttrSpec,
|
||||||
|
const DWARFFormValue &Val,
|
||||||
|
const CompileUnit &Unit,
|
||||||
|
AttributesInfo &Info) {
|
||||||
|
int64_t Addr = *Val.getAsAddress(&Unit.getOrigUnit());
|
||||||
|
if (AttrSpec.Attr == dwarf::DW_AT_low_pc) {
|
||||||
|
if (Die.getTag() == dwarf::DW_TAG_inlined_subroutine ||
|
||||||
|
Die.getTag() == dwarf::DW_TAG_lexical_block)
|
||||||
|
Addr += Info.PCOffset;
|
||||||
|
} else if (AttrSpec.Attr == dwarf::DW_AT_high_pc) {
|
||||||
|
// If we have a high_pc recorded for the input DIE, use
|
||||||
|
// it. Otherwise (when no relocations where applied) just use the
|
||||||
|
// one we just decoded.
|
||||||
|
Addr = (Info.OrigHighPc ? Info.OrigHighPc : Addr) + Info.PCOffset;
|
||||||
|
}
|
||||||
|
|
||||||
|
Die.addValue(static_cast<dwarf::Attribute>(AttrSpec.Attr),
|
||||||
|
static_cast<dwarf::Form>(AttrSpec.Form),
|
||||||
|
new (DIEAlloc) DIEInteger(Addr));
|
||||||
|
return Unit.getOrigUnit().getAddressByteSize();
|
||||||
|
}
|
||||||
|
|
||||||
/// \brief Clone a scalar attribute and add it to \p Die.
|
/// \brief Clone a scalar attribute and add it to \p Die.
|
||||||
/// \returns the size of the new attribute.
|
/// \returns the size of the new attribute.
|
||||||
unsigned DwarfLinker::cloneScalarAttribute(
|
unsigned DwarfLinker::cloneScalarAttribute(
|
||||||
@ -1199,8 +1241,6 @@ unsigned DwarfLinker::cloneScalarAttribute(
|
|||||||
Value = *Val.getAsSectionOffset();
|
Value = *Val.getAsSectionOffset();
|
||||||
else if (AttrSpec.Form == dwarf::DW_FORM_sdata)
|
else if (AttrSpec.Form == dwarf::DW_FORM_sdata)
|
||||||
Value = *Val.getAsSignedConstant();
|
Value = *Val.getAsSignedConstant();
|
||||||
else if (AttrSpec.Form == dwarf::DW_FORM_addr)
|
|
||||||
Value = *Val.getAsAddress(&U);
|
|
||||||
else if (auto OptionalValue = Val.getAsUnsignedConstant())
|
else if (auto OptionalValue = Val.getAsUnsignedConstant())
|
||||||
Value = *OptionalValue;
|
Value = *OptionalValue;
|
||||||
else {
|
else {
|
||||||
@ -1221,7 +1261,7 @@ unsigned DwarfLinker::cloneAttribute(DIE &Die,
|
|||||||
CompileUnit &Unit,
|
CompileUnit &Unit,
|
||||||
const DWARFFormValue &Val,
|
const DWARFFormValue &Val,
|
||||||
const AttributeSpec AttrSpec,
|
const AttributeSpec AttrSpec,
|
||||||
unsigned AttrSize) {
|
unsigned AttrSize, AttributesInfo &Info) {
|
||||||
const DWARFUnit &U = Unit.getOrigUnit();
|
const DWARFUnit &U = Unit.getOrigUnit();
|
||||||
|
|
||||||
switch (AttrSpec.Form) {
|
switch (AttrSpec.Form) {
|
||||||
@ -1242,6 +1282,7 @@ unsigned DwarfLinker::cloneAttribute(DIE &Die,
|
|||||||
case dwarf::DW_FORM_exprloc:
|
case dwarf::DW_FORM_exprloc:
|
||||||
return cloneBlockAttribute(Die, AttrSpec, Val, AttrSize);
|
return cloneBlockAttribute(Die, AttrSpec, Val, AttrSize);
|
||||||
case dwarf::DW_FORM_addr:
|
case dwarf::DW_FORM_addr:
|
||||||
|
return cloneAddressAttribute(Die, AttrSpec, Val, Unit, Info);
|
||||||
case dwarf::DW_FORM_data1:
|
case dwarf::DW_FORM_data1:
|
||||||
case dwarf::DW_FORM_data2:
|
case dwarf::DW_FORM_data2:
|
||||||
case dwarf::DW_FORM_data4:
|
case dwarf::DW_FORM_data4:
|
||||||
@ -1312,7 +1353,8 @@ bool DwarfLinker::applyValidRelocs(MutableArrayRef<char> Data,
|
|||||||
///
|
///
|
||||||
/// \returns the cloned DIE object or null if nothing was selected.
|
/// \returns the cloned DIE object or null if nothing was selected.
|
||||||
DIE *DwarfLinker::cloneDIE(const DWARFDebugInfoEntryMinimal &InputDIE,
|
DIE *DwarfLinker::cloneDIE(const DWARFDebugInfoEntryMinimal &InputDIE,
|
||||||
CompileUnit &Unit, uint32_t OutOffset) {
|
CompileUnit &Unit, int64_t PCOffset,
|
||||||
|
uint32_t OutOffset) {
|
||||||
DWARFUnit &U = Unit.getOrigUnit();
|
DWARFUnit &U = Unit.getOrigUnit();
|
||||||
unsigned Idx = U.getDIEIndex(&InputDIE);
|
unsigned Idx = U.getDIEIndex(&InputDIE);
|
||||||
CompileUnit::DIEInfo &Info = Unit.getInfo(Idx);
|
CompileUnit::DIEInfo &Info = Unit.getInfo(Idx);
|
||||||
@ -1333,6 +1375,7 @@ DIE *DwarfLinker::cloneDIE(const DWARFDebugInfoEntryMinimal &InputDIE,
|
|||||||
// Extract and clone every attribute.
|
// Extract and clone every attribute.
|
||||||
DataExtractor Data = U.getDebugInfoExtractor();
|
DataExtractor Data = U.getDebugInfoExtractor();
|
||||||
uint32_t NextOffset = U.getDIEAtIndex(Idx + 1)->getOffset();
|
uint32_t NextOffset = U.getDIEAtIndex(Idx + 1)->getOffset();
|
||||||
|
AttributesInfo AttrInfo;
|
||||||
|
|
||||||
// We could copy the data only if we need to aply a relocation to
|
// We could copy the data only if we need to aply a relocation to
|
||||||
// it. After testing, it seems there is no performance downside to
|
// it. After testing, it seems there is no performance downside to
|
||||||
@ -1340,7 +1383,17 @@ DIE *DwarfLinker::cloneDIE(const DWARFDebugInfoEntryMinimal &InputDIE,
|
|||||||
SmallString<40> DIECopy(Data.getData().substr(Offset, NextOffset - Offset));
|
SmallString<40> DIECopy(Data.getData().substr(Offset, NextOffset - Offset));
|
||||||
Data = DataExtractor(DIECopy, Data.isLittleEndian(), Data.getAddressSize());
|
Data = DataExtractor(DIECopy, Data.isLittleEndian(), Data.getAddressSize());
|
||||||
// Modify the copy with relocated addresses.
|
// Modify the copy with relocated addresses.
|
||||||
applyValidRelocs(DIECopy, Offset, Data.isLittleEndian());
|
if (applyValidRelocs(DIECopy, Offset, Data.isLittleEndian())) {
|
||||||
|
// If we applied relocations, we store the value of high_pc that was
|
||||||
|
// potentially stored in the input DIE. If high_pc is an address
|
||||||
|
// (Dwarf version == 2), then it might have been relocated to a
|
||||||
|
// totally unrelated value (because the end address in the object
|
||||||
|
// file might be start address of another function which got moved
|
||||||
|
// independantly by the linker). The computation of the actual
|
||||||
|
// high_pc value is done in cloneAddressAttribute().
|
||||||
|
AttrInfo.OrigHighPc =
|
||||||
|
InputDIE.getAttributeValueAsAddress(&U, dwarf::DW_AT_high_pc, 0);
|
||||||
|
}
|
||||||
|
|
||||||
// Reset the Offset to 0 as we will be working on the local copy of
|
// Reset the Offset to 0 as we will be working on the local copy of
|
||||||
// the data.
|
// the data.
|
||||||
@ -1349,13 +1402,19 @@ DIE *DwarfLinker::cloneDIE(const DWARFDebugInfoEntryMinimal &InputDIE,
|
|||||||
const auto *Abbrev = InputDIE.getAbbreviationDeclarationPtr();
|
const auto *Abbrev = InputDIE.getAbbreviationDeclarationPtr();
|
||||||
Offset += getULEB128Size(Abbrev->getCode());
|
Offset += getULEB128Size(Abbrev->getCode());
|
||||||
|
|
||||||
|
// We are entering a subprogram. Get and propagate the PCOffset.
|
||||||
|
if (Die->getTag() == dwarf::DW_TAG_subprogram)
|
||||||
|
PCOffset = Info.AddrAdjust;
|
||||||
|
AttrInfo.PCOffset = PCOffset;
|
||||||
|
|
||||||
for (const auto &AttrSpec : Abbrev->attributes()) {
|
for (const auto &AttrSpec : Abbrev->attributes()) {
|
||||||
DWARFFormValue Val(AttrSpec.Form);
|
DWARFFormValue Val(AttrSpec.Form);
|
||||||
uint32_t AttrSize = Offset;
|
uint32_t AttrSize = Offset;
|
||||||
Val.extractValue(Data, &Offset, &U);
|
Val.extractValue(Data, &Offset, &U);
|
||||||
AttrSize = Offset - AttrSize;
|
AttrSize = Offset - AttrSize;
|
||||||
|
|
||||||
OutOffset += cloneAttribute(*Die, InputDIE, Unit, Val, AttrSpec, AttrSize);
|
OutOffset +=
|
||||||
|
cloneAttribute(*Die, InputDIE, Unit, Val, AttrSpec, AttrSize, AttrInfo);
|
||||||
}
|
}
|
||||||
|
|
||||||
DIEAbbrev &NewAbbrev = Die->getAbbrev();
|
DIEAbbrev &NewAbbrev = Die->getAbbrev();
|
||||||
@ -1379,7 +1438,7 @@ DIE *DwarfLinker::cloneDIE(const DWARFDebugInfoEntryMinimal &InputDIE,
|
|||||||
// Recursively clone children.
|
// Recursively clone children.
|
||||||
for (auto *Child = InputDIE.getFirstChild(); Child && !Child->isNULL();
|
for (auto *Child = InputDIE.getFirstChild(); Child && !Child->isNULL();
|
||||||
Child = Child->getSibling()) {
|
Child = Child->getSibling()) {
|
||||||
if (DIE *Clone = cloneDIE(*Child, Unit, OutOffset)) {
|
if (DIE *Clone = cloneDIE(*Child, Unit, PCOffset, OutOffset)) {
|
||||||
Die->addChild(std::unique_ptr<DIE>(Clone));
|
Die->addChild(std::unique_ptr<DIE>(Clone));
|
||||||
OutOffset = Clone->getOffset() + Clone->getSize();
|
OutOffset = Clone->getOffset() + Clone->getSize();
|
||||||
}
|
}
|
||||||
@ -1460,8 +1519,8 @@ bool DwarfLinker::link(const DebugMap &Map) {
|
|||||||
for (auto &CurrentUnit : Units) {
|
for (auto &CurrentUnit : Units) {
|
||||||
const auto *InputDIE = CurrentUnit.getOrigUnit().getCompileUnitDIE();
|
const auto *InputDIE = CurrentUnit.getOrigUnit().getCompileUnitDIE();
|
||||||
CurrentUnit.setStartOffset(OutputDebugInfoSize);
|
CurrentUnit.setStartOffset(OutputDebugInfoSize);
|
||||||
DIE *OutputDIE =
|
DIE *OutputDIE = cloneDIE(*InputDIE, CurrentUnit, 0 /* PCOffset */,
|
||||||
cloneDIE(*InputDIE, CurrentUnit, 11 /* Unit Header size */);
|
11 /* Unit Header size */);
|
||||||
CurrentUnit.setOutputUnitDIE(OutputDIE);
|
CurrentUnit.setOutputUnitDIE(OutputDIE);
|
||||||
OutputDebugInfoSize = CurrentUnit.computeNextUnitOffset();
|
OutputDebugInfoSize = CurrentUnit.computeNextUnitOffset();
|
||||||
}
|
}
|
||||||
|
Loading…
Reference in New Issue
Block a user