mirror of
https://github.com/RPCSX/llvm.git
synced 2024-11-30 23:20:54 +00:00
Added partial verification for .apple_names accelerator table in llvm-dwarfdump output.
This patch adds code which verifies that each bucket in the .apple_names accelerator table is either empty or has a valid hash index. Differential Revision: https://reviews.llvm.org/D34177 git-svn-id: https://llvm.org/svn/llvm-project/llvm/trunk@305344 91177308-0d34-0410-b5e6-96231b3b80d8
This commit is contained in:
parent
f52e7b3036
commit
2940ab652c
@ -50,6 +50,10 @@ public:
|
||||
: AccelSection(AccelSection), StringSection(StringSection), Relocs(Relocs) {}
|
||||
|
||||
bool extract();
|
||||
uint32_t getNumBuckets();
|
||||
uint32_t getNumHashes();
|
||||
uint32_t getSizeHdr();
|
||||
uint32_t getHeaderDataLength();
|
||||
void dump(raw_ostream &OS) const;
|
||||
};
|
||||
|
||||
|
@ -20,6 +20,7 @@ struct DWARFAttribute;
|
||||
class DWARFContext;
|
||||
class DWARFDie;
|
||||
class DWARFUnit;
|
||||
class DWARFAcceleratorTable;
|
||||
|
||||
/// A class that verifies DWARF debug information given a DWARF Context.
|
||||
class DWARFVerifier {
|
||||
@ -31,6 +32,7 @@ class DWARFVerifier {
|
||||
std::map<uint64_t, std::set<uint32_t>> ReferenceToDIEOffsets;
|
||||
uint32_t NumDebugInfoErrors;
|
||||
uint32_t NumDebugLineErrors;
|
||||
uint32_t NumAppleNamesErrors;
|
||||
|
||||
/// Verifies the attribute's DWARF attribute and its value.
|
||||
///
|
||||
@ -75,7 +77,8 @@ class DWARFVerifier {
|
||||
|
||||
public:
|
||||
DWARFVerifier(raw_ostream &S, DWARFContext &D)
|
||||
: OS(S), DCtx(D), NumDebugInfoErrors(0), NumDebugLineErrors(0) {}
|
||||
: OS(S), DCtx(D), NumDebugInfoErrors(0), NumDebugLineErrors(0),
|
||||
NumAppleNamesErrors(0) {}
|
||||
/// Verify the information in the .debug_info section.
|
||||
///
|
||||
/// Any errors are reported to the stream that was this object was
|
||||
@ -91,6 +94,14 @@ public:
|
||||
///
|
||||
/// @return True if the .debug_line verifies successfully, false otherwise.
|
||||
bool handleDebugLine();
|
||||
|
||||
/// Verify the information in the .apple_names accelerator table.
|
||||
///
|
||||
/// Any errors are reported to the stream that was this object was
|
||||
/// constructed with.
|
||||
///
|
||||
/// @return True if the .apple_names verifies successfully, false otherwise.
|
||||
bool handleAppleNames();
|
||||
};
|
||||
|
||||
} // end namespace llvm
|
||||
|
@ -55,6 +55,13 @@ bool DWARFAcceleratorTable::extract() {
|
||||
return true;
|
||||
}
|
||||
|
||||
uint32_t DWARFAcceleratorTable::getNumBuckets() { return Hdr.NumBuckets; }
|
||||
uint32_t DWARFAcceleratorTable::getNumHashes() { return Hdr.NumHashes; }
|
||||
uint32_t DWARFAcceleratorTable::getSizeHdr() { return sizeof(Hdr); }
|
||||
uint32_t DWARFAcceleratorTable::getHeaderDataLength() {
|
||||
return Hdr.HeaderDataLength;
|
||||
}
|
||||
|
||||
LLVM_DUMP_METHOD void DWARFAcceleratorTable::dump(raw_ostream &OS) const {
|
||||
// Dump the header.
|
||||
OS << "Magic = " << format("0x%08x", Hdr.Magic) << '\n'
|
||||
|
@ -678,8 +678,13 @@ bool DWARFContext::verify(raw_ostream &OS, DIDumpType DumpType) {
|
||||
if (!verifier.handleDebugLine())
|
||||
Success = false;
|
||||
}
|
||||
if (DumpType == DIDT_All || DumpType == DIDT_AppleNames) {
|
||||
if (!verifier.handleAppleNames())
|
||||
Success = false;
|
||||
}
|
||||
return Success;
|
||||
}
|
||||
|
||||
const DWARFUnitIndex &DWARFContext::getCUIndex() {
|
||||
if (CUIndex)
|
||||
return *CUIndex;
|
||||
|
@ -14,6 +14,7 @@
|
||||
#include "llvm/DebugInfo/DWARF/DWARFDie.h"
|
||||
#include "llvm/DebugInfo/DWARF/DWARFFormValue.h"
|
||||
#include "llvm/DebugInfo/DWARF/DWARFSection.h"
|
||||
#include "llvm/DebugInfo/DWARF/DWARFAcceleratorTable.h"
|
||||
#include "llvm/Support/raw_ostream.h"
|
||||
#include <map>
|
||||
#include <set>
|
||||
@ -275,3 +276,36 @@ bool DWARFVerifier::handleDebugLine() {
|
||||
verifyDebugLineRows();
|
||||
return NumDebugLineErrors == 0;
|
||||
}
|
||||
|
||||
bool DWARFVerifier::handleAppleNames() {
|
||||
NumAppleNamesErrors = 0;
|
||||
OS << "Verifying .apple_names...\n";
|
||||
|
||||
DataExtractor AppleNamesSection(DCtx.getAppleNamesSection().Data,
|
||||
DCtx.isLittleEndian(), 0);
|
||||
DataExtractor StrData(DCtx.getStringSection(), DCtx.isLittleEndian(), 0);
|
||||
DWARFAcceleratorTable AppleNames(AppleNamesSection, StrData,
|
||||
DCtx.getAppleNamesSection().Relocs);
|
||||
|
||||
if (!AppleNames.extract()) {
|
||||
OS << "error: cannot extract .apple_names accelerator table\n";
|
||||
return false;
|
||||
}
|
||||
|
||||
// Verify that all buckets have a valid hash index or are empty
|
||||
uint32_t NumBuckets = AppleNames.getNumBuckets();
|
||||
uint32_t NumHashes = AppleNames.getNumHashes();
|
||||
|
||||
uint32_t BucketsOffset =
|
||||
AppleNames.getSizeHdr() + AppleNames.getHeaderDataLength();
|
||||
|
||||
for (uint32_t BucketIdx = 0; BucketIdx < NumBuckets; ++BucketIdx) {
|
||||
uint32_t HashIdx = AppleNamesSection.getU32(&BucketsOffset);
|
||||
if (HashIdx >= NumHashes && HashIdx != UINT32_MAX) {
|
||||
OS << format("error: Bucket[%d] has invalid hash index: [%d]\n",
|
||||
BucketIdx, HashIdx);
|
||||
++NumAppleNamesErrors;
|
||||
}
|
||||
}
|
||||
return NumAppleNamesErrors == 0;
|
||||
}
|
||||
|
192
test/tools/llvm-dwarfdump/X86/apple_names_verify_buckets.s
Normal file
192
test/tools/llvm-dwarfdump/X86/apple_names_verify_buckets.s
Normal file
@ -0,0 +1,192 @@
|
||||
# RUN: llvm-mc %s -filetype obj -triple x86_64-apple-darwin -o - \
|
||||
# RUN: | not llvm-dwarfdump -verify - \
|
||||
# RUN: | FileCheck %s
|
||||
|
||||
# CHECK: Verifying .apple_names...
|
||||
# CHECK-NEXT: error:
|
||||
|
||||
# This test is meant to verify that the -verify option
|
||||
# in llvm-dwarfdump, correctly identifies
|
||||
# an invalid hash index for bucket[0] in the .apple_names section.
|
||||
|
||||
.section __TEXT,__text,regular,pure_instructions
|
||||
.file 1 "basic.c"
|
||||
.comm _i,4,2 ## @i
|
||||
.section __DWARF,__debug_str,regular,debug
|
||||
Linfo_string:
|
||||
.asciz "basic.c" ## string offset=42
|
||||
.asciz "i" ## string offset=84
|
||||
.asciz "int" ## string offset=86
|
||||
.section __DWARF,__debug_loc,regular,debug
|
||||
Lsection_debug_loc:
|
||||
.section __DWARF,__debug_abbrev,regular,debug
|
||||
Lsection_abbrev:
|
||||
.byte 1 ## Abbreviation Code
|
||||
.byte 17 ## DW_TAG_compile_unit
|
||||
.byte 1 ## DW_CHILDREN_yes
|
||||
.byte 37 ## DW_AT_producer
|
||||
.byte 14 ## DW_FORM_strp
|
||||
.byte 19 ## DW_AT_language
|
||||
.byte 5 ## DW_FORM_data2
|
||||
.byte 3 ## DW_AT_name
|
||||
.byte 14 ## DW_FORM_strp
|
||||
.byte 16 ## DW_AT_stmt_list
|
||||
.byte 23 ## DW_FORM_sec_offset
|
||||
.byte 27 ## DW_AT_comp_dir
|
||||
.byte 14 ## DW_FORM_strp
|
||||
.byte 0 ## EOM(1)
|
||||
.byte 0 ## EOM(2)
|
||||
.byte 2 ## Abbreviation Code
|
||||
.byte 52 ## DW_TAG_variable
|
||||
.byte 0 ## DW_CHILDREN_no
|
||||
.byte 3 ## DW_AT_name
|
||||
.byte 14 ## DW_FORM_strp
|
||||
.byte 73 ## DW_AT_type
|
||||
.byte 19 ## DW_FORM_ref4
|
||||
.byte 63 ## DW_AT_external
|
||||
.byte 25 ## DW_FORM_flag_present
|
||||
.byte 58 ## DW_AT_decl_file
|
||||
.byte 11 ## DW_FORM_data1
|
||||
.byte 59 ## DW_AT_decl_line
|
||||
.byte 11 ## DW_FORM_data1
|
||||
.byte 2 ## DW_AT_location
|
||||
.byte 24 ## DW_FORM_exprloc
|
||||
.byte 0 ## EOM(1)
|
||||
.byte 0 ## EOM(2)
|
||||
.byte 3 ## Abbreviation Code
|
||||
.byte 36 ## DW_TAG_base_type
|
||||
.byte 0 ## DW_CHILDREN_no
|
||||
.byte 3 ## DW_AT_name
|
||||
.byte 14 ## DW_FORM_strp
|
||||
.byte 62 ## DW_AT_encoding
|
||||
.byte 11 ## DW_FORM_data1
|
||||
.byte 11 ## DW_AT_byte_size
|
||||
.byte 11 ## DW_FORM_data1
|
||||
.byte 0 ## EOM(1)
|
||||
.byte 0 ## EOM(2)
|
||||
.byte 0 ## EOM(3)
|
||||
.section __DWARF,__debug_info,regular,debug
|
||||
Lsection_info:
|
||||
Lcu_begin0:
|
||||
.long 55 ## Length of Unit
|
||||
.short 4 ## DWARF version number
|
||||
Lset0 = Lsection_abbrev-Lsection_abbrev ## Offset Into Abbrev. Section
|
||||
.long Lset0
|
||||
.byte 8 ## Address Size (in bytes)
|
||||
.byte 1 ## Abbrev [1] 0xb:0x30 DW_TAG_compile_unit
|
||||
.long 0 ## DW_AT_producer
|
||||
.short 12 ## DW_AT_language
|
||||
.long 42 ## DW_AT_name
|
||||
Lset1 = Lline_table_start0-Lsection_line ## DW_AT_stmt_list
|
||||
.long Lset1
|
||||
.long 50 ## DW_AT_comp_dir
|
||||
.byte 2 ## Abbrev [2] 0x1e:0x15 DW_TAG_variable
|
||||
.long 84 ## DW_AT_name
|
||||
.long 51 ## DW_AT_type
|
||||
## DW_AT_external
|
||||
.byte 1 ## DW_AT_decl_file
|
||||
.byte 1 ## DW_AT_decl_line
|
||||
.byte 9 ## DW_AT_location
|
||||
.byte 3
|
||||
.quad _i
|
||||
.byte 3 ## Abbrev [3] 0x33:0x7 DW_TAG_base_type
|
||||
.long 86 ## DW_AT_name
|
||||
.byte 5 ## DW_AT_encoding
|
||||
.byte 4 ## DW_AT_byte_size
|
||||
.byte 0 ## End Of Children Mark
|
||||
.section __DWARF,__debug_ranges,regular,debug
|
||||
Ldebug_range:
|
||||
.section __DWARF,__debug_macinfo,regular,debug
|
||||
Ldebug_macinfo:
|
||||
Lcu_macro_begin0:
|
||||
.byte 0 ## End Of Macro List Mark
|
||||
.section __DWARF,__apple_names,regular,debug
|
||||
Lnames_begin:
|
||||
.long 1212240712 ## Header Magic
|
||||
.short 1 ## Header Version
|
||||
.short 0 ## Header Hash Function
|
||||
.long 1 ## Header Bucket Count
|
||||
.long 1 ## Header Hash Count
|
||||
.long 12 ## Header Data Length
|
||||
.long 0 ## HeaderData Die Offset Base
|
||||
.long 1 ## HeaderData Atom Count
|
||||
.short 1 ## DW_ATOM_die_offset
|
||||
.short 6 ## DW_FORM_data4
|
||||
.long -2 ## Bucket 0 -- error: Bucket[0] has invalid hash index: [-2]
|
||||
.long 177678 ## Hash in Bucket 0
|
||||
.long LNames0-Lnames_begin ## Offset in Bucket 0
|
||||
LNames0:
|
||||
.long 84 ## i
|
||||
.long 1 ## Num DIEs
|
||||
.long 30
|
||||
.long 0
|
||||
.section __DWARF,__apple_objc,regular,debug
|
||||
Lobjc_begin:
|
||||
.long 1212240712 ## Header Magic
|
||||
.short 1 ## Header Version
|
||||
.short 0 ## Header Hash Function
|
||||
.long 1 ## Header Bucket Count
|
||||
.long 0 ## Header Hash Count
|
||||
.long 12 ## Header Data Length
|
||||
.long 0 ## HeaderData Die Offset Base
|
||||
.long 1 ## HeaderData Atom Count
|
||||
.short 1 ## DW_ATOM_die_offset
|
||||
.short 6 ## DW_FORM_data4
|
||||
.long -1 ## Bucket 0
|
||||
.section __DWARF,__apple_namespac,regular,debug
|
||||
Lnamespac_begin:
|
||||
.long 1212240712 ## Header Magic
|
||||
.short 1 ## Header Version
|
||||
.short 0 ## Header Hash Function
|
||||
.long 1 ## Header Bucket Count
|
||||
.long 0 ## Header Hash Count
|
||||
.long 12 ## Header Data Length
|
||||
.long 0 ## HeaderData Die Offset Base
|
||||
.long 1 ## HeaderData Atom Count
|
||||
.short 1 ## DW_ATOM_die_offset
|
||||
.short 6 ## DW_FORM_data4
|
||||
.long -1 ## Bucket 0
|
||||
.section __DWARF,__apple_types,regular,debug
|
||||
Ltypes_begin:
|
||||
.long 1212240712 ## Header Magic
|
||||
.short 1 ## Header Version
|
||||
.short 0 ## Header Hash Function
|
||||
.long 1 ## Header Bucket Count
|
||||
.long 1 ## Header Hash Count
|
||||
.long 20 ## Header Data Length
|
||||
.long 0 ## HeaderData Die Offset Base
|
||||
.long 3 ## HeaderData Atom Count
|
||||
.short 1 ## DW_ATOM_die_offset
|
||||
.short 6 ## DW_FORM_data4
|
||||
.short 3 ## DW_ATOM_die_tag
|
||||
.short 5 ## DW_FORM_data2
|
||||
.short 4 ## DW_ATOM_type_flags
|
||||
.short 11 ## DW_FORM_data1
|
||||
.long 0 ## Bucket 0
|
||||
.long 193495088 ## Hash in Bucket 0
|
||||
.long Ltypes0-Ltypes_begin ## Offset in Bucket 0
|
||||
Ltypes0:
|
||||
.long 86 ## int
|
||||
.long 1 ## Num DIEs
|
||||
.long 51
|
||||
.short 36
|
||||
.byte 0
|
||||
.long 0
|
||||
.section __DWARF,__apple_exttypes,regular,debug
|
||||
Lexttypes_begin:
|
||||
.long 1212240712 ## Header Magic
|
||||
.short 1 ## Header Version
|
||||
.short 0 ## Header Hash Function
|
||||
.long 1 ## Header Bucket Count
|
||||
.long 0 ## Header Hash Count
|
||||
.long 12 ## Header Data Length
|
||||
.long 0 ## HeaderData Die Offset Base
|
||||
.long 1 ## HeaderData Atom Count
|
||||
.short 7 ## DW_ATOM_ext_types
|
||||
.short 6 ## DW_FORM_data4
|
||||
.long -1 ## Bucket 0
|
||||
|
||||
.subsections_via_symbols
|
||||
.section __DWARF,__debug_line,regular,debug
|
||||
Lsection_line:
|
||||
Lline_table_start0:
|
Loading…
Reference in New Issue
Block a user