diff options
author | Ulrich Drepper <drepper@redhat.com> | 2006-07-12 05:22:32 +0000 |
---|---|---|
committer | Ulrich Drepper <drepper@redhat.com> | 2006-07-12 05:22:32 +0000 |
commit | 8ae5814209558f51c93b583c9ae6fdd482f0cbb2 (patch) | |
tree | 23f4d8f5670ab48d07c71e49d9cf11f05bf2246e /src | |
parent | c87b6e76285a0f6da401a5de2aa7b096b9ec731a (diff) | |
download | elfutils-8ae5814209558f51c93b583c9ae6fdd482f0cbb2.tar.gz |
Adjust for final version of GNU-style hash table format.
Diffstat (limited to 'src')
-rw-r--r-- | src/ChangeLog | 5 | ||||
-rw-r--r-- | src/elflint.c | 95 | ||||
-rw-r--r-- | src/readelf.c | 55 |
3 files changed, 117 insertions, 38 deletions
diff --git a/src/ChangeLog b/src/ChangeLog index 1c153b00..7ac79785 100644 --- a/src/ChangeLog +++ b/src/ChangeLog @@ -1,3 +1,8 @@ +2006-07-11 Ulrich Drepper <drepper@redhat.com> + + * readelf.c (handle_gnu_hash): Adjust for final form of .gnu.hash. + * elflint.c (check_gnu_hash): Likewise. + 2006-07-06 Ulrich Drepper <drepper@redhat.com> * elflint.c: Adjust for latest new hash table format. diff --git a/src/elflint.c b/src/elflint.c index a679acc4..b0aa9ab0 100644 --- a/src/elflint.c +++ b/src/elflint.c @@ -1897,17 +1897,34 @@ check_gnu_hash (Ebl *ebl, GElf_Shdr *shdr, Elf_Data *data, int idx, { Elf32_Word nbuckets = ((Elf32_Word *) data->d_buf)[0]; Elf32_Word symbias = ((Elf32_Word *) data->d_buf)[1]; + Elf32_Word bitmask_words = ((Elf32_Word *) data->d_buf)[2]; - if (shdr->sh_size < (2 + 2 * nbuckets) * sizeof (Elf32_Word)) + if (!powerof2 (bitmask_words)) + ERROR (gettext ("\ +section [%2d] '%s': bitmask size not power of 2: %u\n"), + idx, section_name (ebl, idx), bitmask_words); + + size_t bitmask_idxmask = bitmask_words - 1; + if (gelf_getclass (ebl->elf) == ELFCLASS64) + bitmask_words *= 2; + Elf32_Word shift = ((Elf32_Word *) data->d_buf)[3]; + + if (shdr->sh_size < (4 + bitmask_words + nbuckets) * sizeof (Elf32_Word)) { ERROR (gettext ("\ section [%2d] '%s': hash table section is too small (is %ld, expected at least%ld)\n"), idx, section_name (ebl, idx), (long int) shdr->sh_size, - (long int) ((2 + 2 * nbuckets) * sizeof (Elf32_Word))); + (long int) ((4 + bitmask_words + nbuckets) * sizeof (Elf32_Word))); return; } - size_t maxidx = shdr->sh_size / sizeof (Elf32_Word) - (2 + 2 * nbuckets); + if (shift > 31) + ERROR (gettext ("\ +section [%2d] '%s': 2nd hash function shift too big: %u\n"), + idx, section_name (ebl, idx), shift); + + size_t maxidx = shdr->sh_size / sizeof (Elf32_Word) - (4 + bitmask_words + + nbuckets); if (symshdr != NULL) maxidx = MIN (maxidx, symshdr->sh_size / symshdr->sh_entsize); @@ -1915,35 +1932,36 @@ section [%2d] '%s': hash table section is too small (is %ld, expected at least%l /* We need the symbol section data. */ Elf_Data *symdata = elf_getdata (elf_getscn (ebl->elf, shdr->sh_link), NULL); + union + { + Elf32_Word *p32; + Elf64_Xword *p64; + } bitmask = { .p32 = &((Elf32_Word *) data->d_buf)[4] }, + collected = { .p32 = xcalloc (bitmask_words, sizeof (Elf32_Word)) }; + + size_t classbits = gelf_getclass (ebl->elf) == ELFCLASS32 ? 32 : 64; + size_t cnt; - for (cnt = 2; cnt < 2 + 2 * nbuckets; cnt += 2) + for (cnt = 4 + bitmask_words; cnt < 4 + bitmask_words + nbuckets; ++cnt) { - Elf32_Word bitset = ((Elf32_Word *) data->d_buf)[cnt]; - Elf32_Word symidx = ((Elf32_Word *) data->d_buf)[cnt + 1]; + Elf32_Word symidx = ((Elf32_Word *) data->d_buf)[cnt]; if (symidx == 0) - { - /* Nothing in here. No bit in the bitset should be set either. */ - if (bitset != 0) - ERROR (gettext ("\ -section [%2d] '%s': hash chain for bucket %zu empty but bitset is not\n"), - idx, section_name (ebl, idx), cnt / 2 - 1); - - continue; - } + continue; if (symidx < symbias) { ERROR (gettext ("\ section [%2d] '%s': hash chain for bucket %zu lower than symbol index bias\n"), - idx, section_name (ebl, idx), cnt / 2 - 1); + idx, section_name (ebl, idx), cnt - (4 + bitmask_words)); continue; } - Elf32_Word collected_bitset = 0; while (symidx - symbias < maxidx) { - Elf32_Word chainhash = ((Elf32_Word *) data->d_buf)[2 + 2 * nbuckets + Elf32_Word chainhash = ((Elf32_Word *) data->d_buf)[4 + + bitmask_words + + nbuckets + symidx - symbias]; @@ -1966,11 +1984,26 @@ section [%2d] '%s': symbol %u referenced in chain for bucket %zu is undefined\n" ERROR (gettext ("\ section [%2d] '%s': hash value for symbol %u in chain for bucket %zu wrong\n"), idx, section_name (ebl, idx), symidx, cnt / 2 - 1); + + /* Set the bits in the bitmask. */ + size_t maskidx = (hval / classbits) & bitmask_idxmask; + if (classbits == 32) + { + collected.p32[maskidx] + |= UINT32_C (1) << (hval & (classbits - 1)); + collected.p32[maskidx] + |= UINT32_C (1) << ((hval >> shift) & (classbits - 1)); + } + else + { + collected.p64[maskidx] + |= UINT64_C (1) << (hval & (classbits - 1)); + collected.p64[maskidx] + |= UINT64_C (1) << ((hval >> shift) & (classbits - 1)); + } } } - collected_bitset |= 1 << ((chainhash >> 5) & 31); - if ((chainhash & 1) != 0) break; @@ -1986,13 +2019,14 @@ section [%2d] '%s': hash chain for bucket %zu out of bounds\n"), ERROR (gettext ("\ section [%2d] '%s': symbol reference in chain for bucket %zu out of bounds\n"), idx, section_name (ebl, idx), cnt / 2 - 1); - - if (bitset != collected_bitset) - ERROR (gettext ("\ -section [%2d] '%s': bitset for bucket %zu does not match chain entries: computed %#x, reported %#x\n"), - idx, section_name (ebl, idx), cnt / 2 - 1, - collected_bitset, bitset); } + + if (memcmp (collected.p32, bitmask.p32, bitmask_words * sizeof (Elf32_Word))) + ERROR (gettext ("\ +section [%2d] '%s': bitmask does not match names in the hash table\n"), + idx, section_name (ebl, idx)); + + free (collected.p32); } @@ -2024,7 +2058,8 @@ section [%2d] '%s': hash table not for dynamic symbol table\n"), idx, section_name (ebl, idx)); if (shdr->sh_entsize != (tag == SHT_GNU_HASH - ? sizeof (Elf32_Word) + ? (gelf_getclass (ebl->elf) == ELFCLASS32 + ? sizeof (Elf32_Word) : 0) : (size_t) ebl_sysvhash_entrysize (ebl))) ERROR (gettext ("\ section [%2d] '%s': hash table entry size incorrect\n"), @@ -2034,10 +2069,10 @@ section [%2d] '%s': hash table entry size incorrect\n"), ERROR (gettext ("section [%2d] '%s': not marked to be allocated\n"), idx, section_name (ebl, idx)); - if (shdr->sh_size < 2 * shdr->sh_entsize) + if (shdr->sh_size < (tag == SHT_GNU_HASH ? 4 : 2) * (shdr->sh_entsize ?: 4)) { ERROR (gettext ("\ -section [%2d] '%s': hash table has not even room for initial two administrative entries\n"), +section [%2d] '%s': hash table has not even room for initial administrative entries\n"), idx, section_name (ebl, idx)); return; } @@ -2056,7 +2091,7 @@ section [%2d] '%s': hash table has not even room for initial two administrative break; default: - assert (! "should not happen"); + assert (! "should not happen"); } } diff --git a/src/readelf.c b/src/readelf.c index 9fbc24d6..eba6dd81 100644 --- a/src/readelf.c +++ b/src/readelf.c @@ -2323,7 +2323,7 @@ handle_versym (Ebl *ebl, Elf_Scn *scn, GElf_Shdr *shdr) static void print_hash_info (Ebl *ebl, Elf_Scn *scn, GElf_Shdr *shdr, size_t shstrndx, uint_fast32_t maxlength, Elf32_Word nbucket, - uint_fast32_t nsyms, uint32_t *lengths) + uint_fast32_t nsyms, uint32_t *lengths, const char *extrastr) { uint32_t *counts = (uint32_t *) xcalloc (maxlength + 1, sizeof (uint32_t)); @@ -2347,6 +2347,9 @@ print_hash_info (Ebl *ebl, Elf_Scn *scn, GElf_Shdr *shdr, size_t shstrndx, gelf_getshdr (elf_getscn (ebl->elf, shdr->sh_link), &glink)->sh_name)); + if (extrastr != NULL) + fputs (extrastr, stdout); + if (nbucket > 0) { uint64_t success = 0; @@ -2419,7 +2422,7 @@ handle_sysv_hash (Ebl *ebl, Elf_Scn *scn, GElf_Shdr *shdr, size_t shstrndx) } print_hash_info (ebl, scn, shdr, shstrndx, maxlength, nbucket, nsyms, - lengths); + lengths, NULL); free (lengths); } @@ -2461,7 +2464,7 @@ handle_sysv_hash64 (Ebl *ebl, Elf_Scn *scn, GElf_Shdr *shdr, size_t shstrndx) } print_hash_info (ebl, scn, shdr, shstrndx, maxlength, nbucket, nsyms, - lengths); + lengths, NULL); free (lengths); } @@ -2481,17 +2484,30 @@ handle_gnu_hash (Ebl *ebl, Elf_Scn *scn, GElf_Shdr *shdr, size_t shstrndx) Elf32_Word nbucket = ((Elf32_Word *) data->d_buf)[0]; Elf32_Word symbias = ((Elf32_Word *) data->d_buf)[1]; - Elf32_Word *bucket = &((Elf32_Word *) data->d_buf)[2]; - Elf32_Word *chain = &((Elf32_Word *) data->d_buf)[2 + 2 * nbucket]; + + /* Next comes the size of the bitmap. It's measured in words for + the architecture. It's 32 bits for 32 bit archs, and 64 bits for + 64 bit archs. */ + Elf32_Word bitmask_words = ((Elf32_Word *) data->d_buf)[2]; + if (gelf_getclass (ebl->elf) == ELFCLASS64) + bitmask_words *= 2; + + Elf32_Word shift = ((Elf32_Word *) data->d_buf)[3]; uint32_t *lengths = (uint32_t *) xcalloc (nbucket, sizeof (uint32_t)); + Elf32_Word *bitmask = &((Elf32_Word *) data->d_buf)[4]; + Elf32_Word *bucket = &((Elf32_Word *) data->d_buf)[4 + bitmask_words]; + Elf32_Word *chain = &((Elf32_Word *) data->d_buf)[4 + bitmask_words + + nbucket]; + + /* Compute distribution of chain lengths. */ uint_fast32_t maxlength = 0; uint_fast32_t nsyms = 0; for (Elf32_Word cnt = 0; cnt < nbucket; ++cnt) - if (bucket[2 * cnt + 1] != 0) + if (bucket[cnt] != 0) { - Elf32_Word inner = bucket[2 * cnt + 1] - symbias; + Elf32_Word inner = bucket[cnt] - symbias; do { ++nsyms; @@ -2501,9 +2517,32 @@ handle_gnu_hash (Ebl *ebl, Elf_Scn *scn, GElf_Shdr *shdr, size_t shstrndx) while ((chain[inner++] & 1) == 0); } + /* Count bits in bitmask. */ + uint_fast32_t nbits = 0; + for (Elf32_Word cnt = 0; cnt < bitmask_words; ++cnt) + { + uint_fast32_t word = bitmask[cnt]; + + word = (word & 0x55555555) + ((word >> 1) & 0x55555555); + word = (word & 0x33333333) + ((word >> 2) & 0x33333333); + word = (word & 0x0f0f0f0f) + ((word >> 4) & 0x0f0f0f0f); + word = (word & 0x00ff00ff) + ((word >> 8) & 0x00ff00ff); + nbits += (word & 0x0000ffff) + ((word >> 16) & 0x0000ffff); + } + + char *str; + if (asprintf (&str, gettext ("\ + Symbol Bias: %u\n\ + Bitmask Size: %zu bytes %" PRIuFAST32 "%% bits set 2nd hash shift: %u\n"), + symbias, bitmask_words * sizeof (Elf32_Word), + (nbits * 100 + 50) / (bitmask_words * sizeof (Elf32_Word) * 8), + shift) == -1) + error (EXIT_FAILURE, 0, gettext ("memory exhausted")); + print_hash_info (ebl, scn, shdr, shstrndx, maxlength, nbucket, nsyms, - lengths); + lengths, str); + free (str); free (lengths); } |