diff options
author | H.J. Lu <hjl.tools@gmail.com> | 2003-05-05 03:33:09 +0000 |
---|---|---|
committer | H.J. Lu <hjl.tools@gmail.com> | 2003-05-05 03:33:09 +0000 |
commit | c424e0e973d0cee8125e328fa2716473961afad4 (patch) | |
tree | b331ec3a3c3a2845f0889e3a292ef0014f51db29 /bfd/elflink.h | |
parent | d31e3dfee9bea01a1d860d6ceae2052f3ded8b83 (diff) | |
download | gdb-c424e0e973d0cee8125e328fa2716473961afad4.zip gdb-c424e0e973d0cee8125e328fa2716473961afad4.tar.gz gdb-c424e0e973d0cee8125e328fa2716473961afad4.tar.bz2 |
bfd/
2003-05-04 H.J. Lu <hjl@gnu.org>
* elflink.h (elf_merge_symbol): Correctly handle weak definiton.
ld/testsuite/
2003-05-04 H.J. Lu <hjl@gnu.org>
* ld-elfvers/vers18.dsym: Updated for weak definiton change.
* ld-elfvers/vers18.ver: Likewise.
* ld-elfvers/vers19.ver: Likewise.
* ld-elfweak/elfweak.exp: Remove xfail.
Diffstat (limited to 'bfd/elflink.h')
-rw-r--r-- | bfd/elflink.h | 77 |
1 files changed, 62 insertions, 15 deletions
diff --git a/bfd/elflink.h b/bfd/elflink.h index 3aec4e0..dfcce7d 100644 --- a/bfd/elflink.h +++ b/bfd/elflink.h @@ -484,6 +484,7 @@ elf_merge_symbol (abfd, info, name, sym, psec, pvalue, sym_hash, skip, int bind; bfd *oldbfd; bfd_boolean newdyn, olddyn, olddef, newdef, newdyncommon, olddyncommon; + bfd_boolean newweakdef, oldweakdef, newweakundef, oldweakundef; *skip = FALSE; *override = FALSE; @@ -670,6 +671,49 @@ elf_merge_symbol (abfd, info, name, sym, psec, pvalue, sym_hash, skip, return TRUE; } + /* We need to treat weak definiton right, depending on if there is a + definition from a dynamic object. */ + if (bind == STB_WEAK) + { + if (olddef) + { + newweakdef = TRUE; + newweakundef = FALSE; + } + else + { + newweakdef = FALSE; + newweakundef = TRUE; + } + } + else + newweakdef = newweakundef = FALSE; + + /* If the new weak definition comes from a relocatable file and the + old symbol comes from a dynamic object, we treat the new one as + strong. */ + if (newweakdef && !newdyn && olddyn) + newweakdef = FALSE; + + if (h->root.type == bfd_link_hash_defweak) + { + oldweakdef = TRUE; + oldweakundef = FALSE; + } + else if (h->root.type == bfd_link_hash_undefweak) + { + oldweakdef = FALSE; + oldweakundef = TRUE; + } + else + oldweakdef = oldweakundef = FALSE; + + /* If the old weak definition comes from a relocatable file and the + new symbol comes from a dynamic object, we treat the old one as + strong. */ + if (oldweakdef && !olddyn && newdyn) + oldweakdef = FALSE; + /* NEWDYNCOMMON and OLDDYNCOMMON indicate whether the new or old symbol, respectively, appears to be a common symbol in a dynamic object. If a symbol appears in an uninitialized section, and is @@ -698,7 +742,8 @@ elf_merge_symbol (abfd, info, name, sym, psec, pvalue, sym_hash, skip, && (sec->flags & SEC_ALLOC) != 0 && (sec->flags & SEC_LOAD) == 0 && sym->st_size > 0 - && bind != STB_WEAK + && !newweakdef + && !newweakundef && ELF_ST_TYPE (sym->st_info) != STT_FUNC) newdyncommon = TRUE; else @@ -721,9 +766,10 @@ elf_merge_symbol (abfd, info, name, sym, psec, pvalue, sym_hash, skip, a shared object, in which case, the DT_NEEDED entry may not be required at the run time. */ - if ((! dt_needed && h->root.type == bfd_link_hash_defweak) - || h->root.type == bfd_link_hash_undefweak - || bind == STB_WEAK) + if ((! dt_needed && oldweakdef) + || oldweakundef + || newweakdef + || newweakundef) *type_change_ok = TRUE; /* It's OK to change the size if either the existing symbol or the @@ -781,11 +827,13 @@ elf_merge_symbol (abfd, info, name, sym, psec, pvalue, sym_hash, skip, && newdef && (olddef || (h->root.type == bfd_link_hash_common - && (bind == STB_WEAK + && (newweakdef + || newweakundef || ELF_ST_TYPE (sym->st_info) == STT_FUNC))) - && (h->root.type != bfd_link_hash_defweak + && (!oldweakdef || dt_needed - || bind == STB_WEAK)) + || newweakdef + || newweakundef)) { *override = TRUE; newdef = FALSE; @@ -838,13 +886,11 @@ elf_merge_symbol (abfd, info, name, sym, psec, pvalue, sym_hash, skip, if (! newdyn && (newdef || (bfd_is_com_section (sec) - && (h->root.type == bfd_link_hash_defweak - || h->type == STT_FUNC))) + && (oldweakdef || h->type == STT_FUNC))) && olddyn && olddef && (h->elf_link_hash_flags & ELF_LINK_HASH_DEF_DYNAMIC) != 0 - && (bind != STB_WEAK - || h->root.type == bfd_link_hash_defweak)) + && ((!newweakdef && !newweakundef) || oldweakdef)) { /* Change the hash table entry to undefined, and let _bfd_generic_link_add_one_symbol do the right thing with the @@ -940,10 +986,11 @@ elf_merge_symbol (abfd, info, name, sym, psec, pvalue, sym_hash, skip, the DT_NEEDED entry may not be required at the run time. */ if (olddef && ! dt_needed - && h->root.type == bfd_link_hash_defweak + && oldweakdef && newdef && newdyn - && bind != STB_WEAK) + && !newweakdef + && !newweakundef) { /* To make this work we have to frob the flags so that the rest of the code does not think we are using the regular @@ -969,10 +1016,10 @@ elf_merge_symbol (abfd, info, name, sym, psec, pvalue, sym_hash, skip, as a definition. */ if (olddef && olddyn - && h->root.type != bfd_link_hash_defweak + && !oldweakdef && newdef && ! newdyn - && bind == STB_WEAK) + && (newweakdef || newweakundef)) *override = TRUE; return TRUE; |