| /* Return symbol table of archive. |
| Copyright (C) 1998, 1999, 2000, 2002 Red Hat, Inc. |
| Written by Ulrich Drepper <drepper@redhat.com>, 1998. |
| |
| This program is free software; you can redistribute it and/or modify |
| it under the terms of the GNU General Public License as published by |
| the Free Software Foundation, version 2. |
| |
| This program is distributed in the hope that it will be useful, |
| but WITHOUT ANY WARRANTY; without even the implied warranty of |
| MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the |
| GNU General Public License for more details. |
| |
| You should have received a copy of the GNU General Public License |
| along with this program; if not, write to the Free Software Foundation, |
| Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA. */ |
| |
| #ifdef HAVE_CONFIG_H |
| # include <config.h> |
| #endif |
| |
| #include <assert.h> |
| //#include <byteswap.h> |
| //#include <endian.h> |
| #include <stdint.h> |
| #include <stdlib.h> |
| #include <string.h> |
| #include <unistd.h> |
| |
| #include <dl-hash.h> |
| #include "libelfP.h" |
| |
| |
| Elf_Arsym * |
| elf_getarsym (elf, ptr) |
| Elf *elf; |
| size_t *ptr; |
| { |
| Elf_Arsym *result; |
| |
| if (elf->kind != ELF_K_AR) |
| { |
| /* This is no archive. */ |
| __libelf_seterrno (ELF_E_NO_ARCHIVE); |
| return NULL; |
| } |
| |
| if (ptr != NULL) |
| /* In case of an error or when we know the value store the expected |
| value now. Doing this allows us easier exits in an error case. */ |
| *ptr = elf->state.ar.ar_sym_num; |
| |
| if (elf->state.ar.ar_sym == (Elf_Arsym *) -1l) |
| { |
| /* There is no index. */ |
| __libelf_seterrno (ELF_E_NO_INDEX); |
| return NULL; |
| } |
| |
| result = elf->state.ar.ar_sym; |
| if (result == NULL) |
| { |
| /* We have not yet read the index. */ |
| struct ar_hdr *index_hdr; |
| uint32_t n; |
| size_t index_size; |
| char tmpbuf[17]; |
| size_t ar_sym_len; |
| Elf_Arsym *arsym; |
| size_t cnt; |
| |
| rwlock_wrlock (elf->lock); |
| |
| /* In case we find no index remember this for the next call. */ |
| elf->state.ar.ar_sym = (Elf_Arsym *) -1l; |
| |
| if (elf->map_address == NULL) |
| { |
| /* We must read index from the file. */ |
| assert (elf->fildes != -1); |
| if (pread (elf->fildes, &elf->state.ar.ar_hdr, |
| sizeof (struct ar_hdr), elf->start_offset + SARMAG) |
| != sizeof (struct ar_hdr)) |
| { |
| /* It is not possible to read the index. Maybe it does not |
| exist. */ |
| __libelf_seterrno (ELF_E_READ_ERROR); |
| goto out; |
| } |
| |
| index_hdr = &elf->state.ar.ar_hdr; |
| } |
| else |
| { |
| if (SARMAG + sizeof (struct ar_hdr) > elf->maximum_size) |
| { |
| /* There is no room for the full archive. */ |
| __libelf_seterrno (ELF_E_NO_INDEX); |
| goto out; |
| } |
| |
| index_hdr = (struct ar_hdr *) (elf->map_address |
| + elf->start_offset + SARMAG); |
| } |
| |
| /* Now test whether this really is an archive. */ |
| if (memcmp (index_hdr->ar_fmag, ARFMAG, 2) != 0) |
| { |
| /* Invalid magic bytes. */ |
| __libelf_seterrno (ELF_E_ARCHIVE_FMAG); |
| goto out; |
| } |
| |
| /* Now test whether this is the index. It is denoted by the |
| name being "/ ". |
| XXX This is not entirely true. There are some more forms. |
| Which of them shall we handle? */ |
| if (memcmp (index_hdr->ar_name, "/ ", 16) != 0) |
| { |
| /* If the index is not the first entry, there is no index. |
| |
| XXX Is this true? */ |
| __libelf_seterrno (ELF_E_NO_INDEX); |
| goto out; |
| } |
| |
| /* We have an archive. The first word in there is the number of |
| entries in the table. */ |
| if (elf->map_address == NULL) |
| { |
| if (pread (elf->fildes, &n, sizeof (n), |
| elf->start_offset + SARMAG + sizeof (struct ar_hdr)) |
| != sizeof (n)) |
| { |
| /* Cannot read the number of entries. */ |
| __libelf_seterrno (ELF_E_NO_INDEX); |
| goto out; |
| } |
| } |
| else |
| n = *(uint32_t *) (elf->map_address + elf->start_offset |
| + SARMAG + sizeof (struct ar_hdr)); |
| |
| if (__BYTE_ORDER == __LITTLE_ENDIAN) |
| n = bswap_32 (n); |
| |
| /* Now we can perform some first tests on whether all the data |
| needed for the index is available. */ |
| memcpy (tmpbuf, index_hdr->ar_size, 10); |
| tmpbuf[10] = '\0'; |
| index_size = atol (tmpbuf); |
| |
| if (SARMAG + sizeof (struct ar_hdr) + index_size > elf->maximum_size |
| || n * sizeof (uint32_t) > index_size) |
| { |
| /* This index table cannot be right since it does not fit into |
| the file. */ |
| __libelf_seterrno (ELF_E_NO_INDEX); |
| goto out; |
| } |
| |
| /* Now we can allocate the arrays needed to store the index. */ |
| ar_sym_len = (n + 1) * sizeof (Elf_Arsym); |
| elf->state.ar.ar_sym = (Elf_Arsym *) malloc (ar_sym_len); |
| if (elf->state.ar.ar_sym != NULL) |
| { |
| uint32_t *file_data; |
| char *str_data; |
| |
| if (elf->map_address == NULL) |
| { |
| char *new_str; |
| Elf_Arsym *newp; |
| |
| file_data = (uint32_t *) alloca (n * sizeof (uint32_t)); |
| |
| ar_sym_len += index_size - n * sizeof (uint32_t); |
| newp = (Elf_Arsym *) realloc (elf->state.ar.ar_sym, |
| ar_sym_len); |
| if (newp == NULL) |
| { |
| free (elf->state.ar.ar_sym); |
| elf->state.ar.ar_sym = NULL; |
| __libelf_seterrno (ELF_E_NOMEM); |
| goto out; |
| } |
| elf->state.ar.ar_sym = newp; |
| |
| new_str = (char *) (elf->state.ar.ar_sym + n + 1); |
| |
| /* Now read the data from the file. */ |
| if ((size_t) pread (elf->fildes, file_data, |
| n * sizeof (uint32_t), elf->start_offset |
| + SARMAG + sizeof (struct ar_hdr) |
| + sizeof (uint32_t)) != n * sizeof (uint32_t) |
| || ((size_t) pread (elf->fildes, new_str, |
| index_size - n * sizeof (uint32_t), |
| elf->start_offset |
| + SARMAG + sizeof (struct ar_hdr) |
| + (n + 1) * sizeof (uint32_t)) |
| != index_size - n * sizeof (uint32_t))) |
| { |
| /* We were not able to read the data. */ |
| free (elf->state.ar.ar_sym); |
| elf->state.ar.ar_sym = NULL; |
| __libelf_seterrno (ELF_E_NO_INDEX); |
| goto out; |
| } |
| |
| str_data = (char *) new_str; |
| } |
| else |
| { |
| file_data = (uint32_t *) (elf->map_address + elf->start_offset |
| + SARMAG + sizeof (struct ar_hdr) |
| + sizeof (uint32_t)); |
| str_data = (char *) &file_data[n]; |
| } |
| |
| /* Now we can build the data structure. */ |
| arsym = elf->state.ar.ar_sym; |
| for (cnt = 0; cnt < n; ++cnt) |
| { |
| arsym[cnt].as_name = str_data; |
| if (__BYTE_ORDER == __LITTLE_ENDIAN) |
| arsym[cnt].as_off = bswap_32 (file_data[cnt]); |
| else |
| arsym[cnt].as_off = file_data[cnt]; |
| arsym[cnt].as_hash = _dl_elf_hash (str_data); |
| str_data = rawmemchr (str_data, '\0') + 1; |
| } |
| /* At the end a special entry. */ |
| arsym[n].as_name = NULL; |
| arsym[n].as_off = 0; |
| arsym[n].as_hash = ~0UL; |
| |
| /* Tell the caller how many entries we have. */ |
| elf->state.ar.ar_sym_num = n + 1; |
| } |
| |
| result = elf->state.ar.ar_sym; |
| |
| out: |
| rwlock_unlock (elf->lock); |
| } |
| |
| if (ptr != NULL) |
| *ptr = elf->state.ar.ar_sym_num; |
| |
| return result; |
| } |