diff options
author | Alan Modra <amodra@gmail.com> | 2006-09-14 12:11:33 +0000 |
---|---|---|
committer | Alan Modra <amodra@gmail.com> | 2006-09-14 12:11:33 +0000 |
commit | 30288845d67c64d07905c1e4ca9de4768d3b2dd8 (patch) | |
tree | dbc93e964bb76b24a80197160527adddab2827e1 /binutils | |
parent | 87699a25a2e69dd4a5e7bc599ed37a88fd55ac88 (diff) | |
download | gdb-30288845d67c64d07905c1e4ca9de4768d3b2dd8.zip gdb-30288845d67c64d07905c1e4ca9de4768d3b2dd8.tar.gz gdb-30288845d67c64d07905c1e4ca9de4768d3b2dd8.tar.bz2 |
PR 3182
* elf.c (_bfd_elf_copy_private_header_data): Fix group members
that have had their SHT_GROUP section removed.
* objcopy.c (group_signature): New function, split out from..
(setup_section): ..here.
(is_strip_section): Return true for SHT_GROUP sections that are
going to lose their group signature symbol.
Diffstat (limited to 'binutils')
-rw-r--r-- | binutils/ChangeLog | 8 | ||||
-rw-r--r-- | binutils/objcopy.c | 71 |
2 files changed, 60 insertions, 19 deletions
diff --git a/binutils/ChangeLog b/binutils/ChangeLog index 08ea5c3..e1636e1 100644 --- a/binutils/ChangeLog +++ b/binutils/ChangeLog @@ -1,6 +1,14 @@ 2006-09-14 Alan Modra <amodra@bigpond.net.au> PR 3182 + * objcopy.c (group_signature): New function, split out from.. + (setup_section): ..here. + (is_strip_section): Return true for SHT_GROUP sections that are + going to lose their group signature symbol. + +2006-09-14 Alan Modra <amodra@bigpond.net.au> + + PR 3182 * objcopy.c (copy_object): Load symbols earlier. (setup_section): Mark group signature symbols with BSF_KEEP. diff --git a/binutils/objcopy.c b/binutils/objcopy.c index 1510ac4..e916fce 100644 --- a/binutils/objcopy.c +++ b/binutils/objcopy.c @@ -785,6 +785,30 @@ is_specified_symbol (const char *name, struct symlist *list) return FALSE; } +/* Return a pointer to the symbol used as a signature for GROUP. */ + +static asymbol * +group_signature (asection *group) +{ + bfd *abfd = group->owner; + Elf_Internal_Shdr *ghdr; + + if (bfd_get_flavour (abfd) != bfd_target_elf_flavour) + return NULL; + + ghdr = &elf_section_data (group)->this_hdr; + if (ghdr->sh_link < elf_numsections (abfd)) + { + const struct elf_backend_data *bed = get_elf_backend_data (abfd); + Elf_Internal_Shdr *symhdr = elf_elfsections (abfd) [ghdr->sh_link]; + + if (symhdr->sh_type == SHT_SYMTAB + && ghdr->sh_info < symhdr->sh_size / bed->s->sizeof_sym) + return isympp[ghdr->sh_info]; + } + return NULL; +} + /* See if a section is being removed. */ static bfd_boolean @@ -815,12 +839,30 @@ is_strip_section (bfd *abfd ATTRIBUTE_UNUSED, asection *sec) return FALSE; } - /* PR binutils/3166 - Group sections look like debugging sections but they are not. - (They have a non-zero size but they are not ALLOCated). */ - if ((bfd_get_section_flags (abfd, sec) & SEC_GROUP) != 0 - && strip_symbols == STRIP_NONDEBUG) - return TRUE; + if ((bfd_get_section_flags (abfd, sec) & SEC_GROUP) != 0) + { + asymbol *gsym; + const char *gname; + + /* PR binutils/3166 + Group sections look like debugging sections but they are not. + (They have a non-zero size but they are not ALLOCated). */ + if (strip_symbols == STRIP_NONDEBUG) + return TRUE; + + /* PR binutils/3181 + If we are going to strip the group signature symbol, then + strip the group section too. */ + gsym = group_signature (sec); + if (gsym != NULL) + gname = gsym->name; + else + gname = sec->name; + if ((strip_symbols == STRIP_ALL + && !is_specified_symbol (gname, keep_specific_list)) + || is_specified_symbol (gname, strip_specific_list)) + return TRUE; + } return FALSE; } @@ -2214,21 +2256,12 @@ setup_section (bfd *ibfd, sec_ptr isection, void *obfdarg) err = _("private data"); goto loser; } - else if ((isection->flags & SEC_GROUP) != 0 - && bfd_get_flavour (ibfd) == bfd_target_elf_flavour) + else if ((isection->flags & SEC_GROUP) != 0) { - Elf_Internal_Shdr *ghdr; + asymbol *gsym = group_signature (isection); - ghdr = &elf_section_data (isection)->this_hdr; - if (ghdr->sh_link < elf_numsections (ibfd)) - { - const struct elf_backend_data *bed = get_elf_backend_data (ibfd); - Elf_Internal_Shdr *symhdr = elf_elfsections (ibfd) [ghdr->sh_link]; - - if (symhdr->sh_type == SHT_SYMTAB - && ghdr->sh_info < symhdr->sh_size / bed->s->sizeof_sym) - isympp[ghdr->sh_info]->flags |= BSF_KEEP; - } + if (gsym != NULL) + gsym->flags |= BSF_KEEP; } /* All went well. */ |