Skip to content

Commit bf72258

Browse files
committed
[lldb] Fix lookup of symbols with the same address range but different binding
This fixes a failing testcase on Fedora 30 x86_64 (regression Fedora 29->30): PASS: ./bin/lldb ./lldb-test-build.noindex/functionalities/unwind/noreturn/TestNoreturnUnwind.test_dwarf/a.out -o 'settings set symbols.enable-external-lookup false' -o r -o bt -o quit * frame #0: 0x00007ffff7aa6e75 libc.so.6`__GI_raise + 325 frame #1: 0x00007ffff7a91895 libc.so.6`__GI_abort + 295 frame #2: 0x0000000000401140 a.out`func_c at main.c:12:2 frame #3: 0x000000000040113a a.out`func_b at main.c:18:2 frame #4: 0x0000000000401134 a.out`func_a at main.c:26:2 frame #5: 0x000000000040112e a.out`main(argc=<unavailable>, argv=<unavailable>) at main.c:32:2 frame #6: 0x00007ffff7a92f33 libc.so.6`__libc_start_main + 243 frame #7: 0x000000000040106e a.out`_start + 46 vs. FAIL - unrecognized abort() function: ./bin/lldb ./lldb-test-build.noindex/functionalities/unwind/noreturn/TestNoreturnUnwind.test_dwarf/a.out -o 'settings set symbols.enable-external-lookup false' -o r -o bt -o quit * frame #0: 0x00007ffff7aa6e75 libc.so.6`.annobin_raise.c + 325 frame #1: 0x00007ffff7a91895 libc.so.6`.annobin_loadmsgcat.c_end.unlikely + 295 frame #2: 0x0000000000401140 a.out`func_c at main.c:12:2 frame #3: 0x000000000040113a a.out`func_b at main.c:18:2 frame #4: 0x0000000000401134 a.out`func_a at main.c:26:2 frame #5: 0x000000000040112e a.out`main(argc=<unavailable>, argv=<unavailable>) at main.c:32:2 frame #6: 0x00007ffff7a92f33 libc.so.6`.annobin_libc_start.c + 243 frame #7: 0x000000000040106e a.out`.annobin_init.c.hot + 46 The extra ELF symbols are there due to Annobin (I did not investigate why this problem happened specifically since F-30 and not since F-28). It is due to: Symbol table '.dynsym' contains 2361 entries: Valu e Size Type Bind Vis Name 0000000000022769 5 FUNC LOCAL DEFAULT _nl_load_domain.cold 000000000002276e 0 NOTYPE LOCAL HIDDEN .annobin_abort.c.unlikely ... 000000000002276e 0 NOTYPE LOCAL HIDDEN .annobin_loadmsgcat.c_end.unlikely ... 000000000002276e 0 NOTYPE LOCAL HIDDEN .annobin_textdomain.c_end.unlikely 000000000002276e 548 FUNC GLOBAL DEFAULT abort 000000000002276e 548 FUNC GLOBAL DEFAULT abort@@GLIBC_2.2.5 000000000002276e 548 FUNC LOCAL DEFAULT __GI_abort 0000000000022992 0 NOTYPE LOCAL HIDDEN .annobin_abort.c_end.unlikely GDB has some more complicated preferences between overlapping and/or sharing address symbols, I have made here so far the most simple fix for this case. Differential revision: https://reviews.llvm.org/D63540
1 parent af4adb0 commit bf72258

File tree

5 files changed

+70
-2
lines changed

5 files changed

+70
-2
lines changed

lldb/include/lldb/Symbol/Symtab.h

+23-1
Original file line numberDiff line numberDiff line change
@@ -143,7 +143,29 @@ class Symtab {
143143
typedef std::vector<Symbol> collection;
144144
typedef collection::iterator iterator;
145145
typedef collection::const_iterator const_iterator;
146-
typedef RangeDataVector<lldb::addr_t, lldb::addr_t, uint32_t>
146+
class FileRangeToIndexMapCompare {
147+
public:
148+
FileRangeToIndexMapCompare(const Symtab &symtab) : m_symtab(symtab) {}
149+
bool operator()(const uint32_t a_data, const uint32_t b_data) const {
150+
return rank(a_data) > rank(b_data);
151+
}
152+
153+
private:
154+
// How much preferred is this symbol?
155+
int rank(const uint32_t data) const {
156+
const Symbol &symbol = *m_symtab.SymbolAtIndex(data);
157+
if (symbol.IsExternal())
158+
return 3;
159+
if (symbol.IsWeak())
160+
return 2;
161+
if (symbol.IsDebug())
162+
return 0;
163+
return 1;
164+
}
165+
const Symtab &m_symtab;
166+
};
167+
typedef RangeDataVector<lldb::addr_t, lldb::addr_t, uint32_t, 0,
168+
FileRangeToIndexMapCompare>
147169
FileRangeToIndexMap;
148170
void InitNameIndexes();
149171
void InitAddressIndexes();

lldb/source/Plugins/ObjectFile/ELF/ObjectFileELF.cpp

+2
Original file line numberDiff line numberDiff line change
@@ -2258,6 +2258,8 @@ unsigned ObjectFileELF::ParseSymbols(Symtab *symtab, user_id_t start_id,
22582258
symbol_size_valid, // Symbol size is valid
22592259
has_suffix, // Contains linker annotations?
22602260
flags); // Symbol flags.
2261+
if (symbol.getBinding() == STB_WEAK)
2262+
dc_symbol.SetIsWeak(true);
22612263
symtab->AddSymbol(dc_symbol);
22622264
}
22632265
return i;

lldb/source/Symbol/Symtab.cpp

+1-1
Original file line numberDiff line numberDiff line change
@@ -28,7 +28,7 @@ using namespace lldb;
2828
using namespace lldb_private;
2929

3030
Symtab::Symtab(ObjectFile *objfile)
31-
: m_objfile(objfile), m_symbols(), m_file_addr_to_index(),
31+
: m_objfile(objfile), m_symbols(), m_file_addr_to_index(*this),
3232
m_name_to_index(), m_mutex(), m_file_addr_to_index_computed(false),
3333
m_name_indexes_computed(false) {}
3434

Original file line numberDiff line numberDiff line change
@@ -0,0 +1,22 @@
1+
.text
2+
.byte 0
3+
sizeless:
4+
sizeful:
5+
.byte 0
6+
.byte 0
7+
sizeend:
8+
.size sizeful, sizeend - sizeful
9+
.byte 0
10+
case1_local:
11+
case1_global:
12+
.globl case1_global
13+
.byte 0
14+
case2_local:
15+
case2_weak:
16+
.weak case2_weak
17+
.byte 0
18+
case3_weak:
19+
.weak case3_weak
20+
case3_global:
21+
.globl case3_global
22+
.byte 0
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,22 @@
1+
# Some targets do not have the .size directive.
2+
# RUN: %clang -target x86_64-unknown-unknown-elf %S/Inputs/symbol-binding.s -c -o %t.o
3+
# RUN: %lldb %t.o -s %s -o quit | FileCheck %s
4+
5+
image lookup --address 4
6+
# CHECK: Summary: symbol-binding.test.tmp.o`case1_global
7+
image lookup --address 5
8+
# CHECK: Summary: symbol-binding.test.tmp.o`case2_weak
9+
image lookup --address 6
10+
# CHECK: Summary: symbol-binding.test.tmp.o`case3_global
11+
image dump symtab
12+
# CHECK: Index UserID DSX Type File Address/Value Load Address Size Flags Name
13+
# CHECK-NEXT:------- ------ --- --------------- ------------------ ------------------ ------------------ ---------- ----------------------------------
14+
# CHECK-NEXT:[ 0] 1 Code 0x0000000000000004 0x0000000000000001 0x00000000 case1_local
15+
# CHECK-NEXT:[ 1] 2 Code 0x0000000000000005 0x0000000000000001 0x00000000 case2_local
16+
# CHECK-NEXT:[ 2] 3 Code 0x0000000000000003 0x0000000000000001 0x00000000 sizeend
17+
# CHECK-NEXT:[ 3] 4 Code 0x0000000000000001 0x0000000000000002 0x00000000 sizeful
18+
# CHECK-NEXT:[ 4] 5 Code 0x0000000000000001 0x0000000000000002 0x00000000 sizeless
19+
# CHECK-NEXT:[ 5] 6 X Code 0x0000000000000004 0x0000000000000001 0x00000010 case1_global
20+
# CHECK-NEXT:[ 6] 7 Code 0x0000000000000005 0x0000000000000001 0x00000020 case2_weak
21+
# CHECK-NEXT:[ 7] 8 X Code 0x0000000000000006 0x0000000000000001 0x00000010 case3_global
22+
# CHECK-NEXT:[ 8] 9 Code 0x0000000000000006 0x0000000000000001 0x00000020 case3_weak

0 commit comments

Comments
 (0)