summaryrefslogtreecommitdiff
path: root/utils/readsoname2.c
diff options
context:
space:
mode:
Diffstat (limited to 'utils/readsoname2.c')
-rw-r--r--utils/readsoname2.c268
1 files changed, 133 insertions, 135 deletions
diff --git a/utils/readsoname2.c b/utils/readsoname2.c
index 029a2bbc1..6a9f00775 100644
--- a/utils/readsoname2.c
+++ b/utils/readsoname2.c
@@ -1,145 +1,143 @@
char *readsonameXX(char *name, FILE *infile, int expected_type, int *type)
{
- ElfW(Ehdr) *epnt;
- ElfW(Phdr) *ppnt;
- unsigned int i, j;
- char *header;
- ElfW(Addr) dynamic_addr = 0;
- ElfW(Addr) dynamic_size = 0;
- unsigned long page_size = getpagesize();
- ElfW(Addr) strtab_val = 0;
- ElfW(Addr) needed_val;
- ElfW(Addr) loadaddr = -1;
- ElfW(Dyn) *dpnt;
- struct stat st;
- char *needed;
- char *soname = NULL;
- int multi_libcs = 0;
-
- if(expected_type == LIB_DLL)
- {
- warn("%s does not match type specified for directory!", name);
- expected_type = LIB_ANY;
- }
-
- *type = LIB_ELF;
-
- if (fstat(fileno(infile), &st))
- return NULL;
- header = mmap(0, st.st_size, PROT_READ|PROT_WRITE, MAP_PRIVATE, fileno(infile), 0);
- if (header == (caddr_t)-1)
- return NULL;
-
- epnt = (ElfW(Ehdr) *)header;
- if ((char *)(epnt+1) > (char *)(header + st.st_size))
- goto skip;
+ ElfW(Ehdr) *epnt;
+ ElfW(Phdr) *ppnt;
+ unsigned int i, j;
+ char *header;
+ ElfW(Addr) dynamic_addr = 0;
+ ElfW(Addr) dynamic_size = 0;
+ unsigned long page_size = getpagesize();
+ ElfW(Addr) strtab_val = 0;
+ ElfW(Addr) needed_val;
+ ElfW(Addr) loadaddr = -1;
+ ElfW(Dyn) *dpnt;
+ struct stat st;
+ char *needed;
+ char *soname = NULL;
+ int multi_libcs = 0;
+
+ if (expected_type == LIB_DLL) {
+ warn("%s does not match type specified for directory!", name);
+ expected_type = LIB_ANY;
+ }
+
+ *type = LIB_ELF;
+
+ if (fstat(fileno(infile), &st))
+ return NULL;
+ header =
+ mmap(0, st.st_size, PROT_READ | PROT_WRITE, MAP_PRIVATE,
+ fileno(infile), 0);
+ if (header == (caddr_t) - 1)
+ return NULL;
+
+ epnt = (ElfW(Ehdr) *) header;
+ if ((char *)(epnt + 1) > (char *)(header + st.st_size))
+ goto skip;
#if __BYTE_ORDER == __LITTLE_ENDIAN
- byteswap = (epnt->e_ident[5] == ELFDATA2MSB) ? 1 : 0;
+ byteswap = (epnt->e_ident[5] == ELFDATA2MSB) ? 1 : 0;
#elif __BYTE_ORDER == __BIG_ENDIAN
- byteswap = (epnt->e_ident[5] == ELFDATA2LSB) ? 1 : 0;
+ byteswap = (epnt->e_ident[5] == ELFDATA2LSB) ? 1 : 0;
#else
#error Unknown host byte order!
#endif
- /* Be very lazy, and only byteswap the stuff we use */
- if (byteswap==1) {
- epnt->e_phoff=bswap_32(epnt->e_phoff);
- epnt->e_phnum=bswap_16(epnt->e_phnum);
- }
-
- ppnt = (ElfW(Phdr) *)&header[epnt->e_phoff];
- if ((char *)ppnt < (char *)header ||
- (char *)(ppnt+epnt->e_phnum) > (char *)(header + st.st_size))
- goto skip;
-
- for(i = 0; i < epnt->e_phnum; i++)
- {
- /* Be very lazy, and only byteswap the stuff we use */
- if (byteswap==1) {
- ppnt->p_type=bswap_32(ppnt->p_type);
- ppnt->p_vaddr=bswap_32(ppnt->p_vaddr);
- ppnt->p_offset=bswap_32(ppnt->p_offset);
- ppnt->p_filesz=bswap_32(ppnt->p_filesz);
- }
-
- if (loadaddr == (ElfW(Addr))-1 && ppnt->p_type == PT_LOAD)
- loadaddr = (ppnt->p_vaddr & ~(page_size-1)) -
- (ppnt->p_offset & ~(page_size-1));
- if(ppnt->p_type == 2)
- {
- dynamic_addr = ppnt->p_offset;
- dynamic_size = ppnt->p_filesz;
- };
- ppnt++;
- };
-
- dpnt = (ElfW(Dyn) *) &header[dynamic_addr];
- dynamic_size = dynamic_size / sizeof(ElfW(Dyn));
- if ((char *)dpnt < (char *)header ||
- (char *)(dpnt+dynamic_size) > (char *)(header + st.st_size))
- goto skip;
-
- if (byteswap==1) {
- dpnt->d_tag=bswap_32(dpnt->d_tag);
- dpnt->d_un.d_val=bswap_32(dpnt->d_un.d_val);
- }
-
- while (dpnt->d_tag != DT_NULL)
- {
- if (dpnt->d_tag == DT_STRTAB)
- strtab_val = dpnt->d_un.d_val;
- dpnt++;
- if (byteswap==1) {
- dpnt->d_tag=bswap_32(dpnt->d_tag);
- dpnt->d_un.d_val=bswap_32(dpnt->d_un.d_val);
- }
- };
-
- if (!strtab_val)
- goto skip;
-
- dpnt = (ElfW(Dyn) *) &header[dynamic_addr];
- while (dpnt->d_tag != DT_NULL)
- {
- if (dpnt->d_tag == DT_SONAME || dpnt->d_tag == DT_NEEDED)
- {
- needed_val = dpnt->d_un.d_val;
- if (needed_val + strtab_val >= loadaddr ||
- needed_val + strtab_val < st.st_size - loadaddr)
- {
- needed = (char *) (header - loadaddr + strtab_val + needed_val);
-
- if (dpnt->d_tag == DT_SONAME)
- soname = xstrdup(needed);
-
- for (j = 0; needed_tab[j].soname != NULL; j++)
- {
- if (strcmp(needed, needed_tab[j].soname) == 0)
- {
- if (*type != LIB_ELF && *type != needed_tab[j].type)
- multi_libcs = 1;
- *type = needed_tab[j].type;
- }
+ /* Be very lazy, and only byteswap the stuff we use */
+ if (byteswap == 1) {
+ epnt->e_phoff = bswap_32(epnt->e_phoff);
+ epnt->e_phnum = bswap_16(epnt->e_phnum);
+ }
+
+ ppnt = (ElfW(Phdr) *) & header[epnt->e_phoff];
+ if ((char *)ppnt < (char *)header ||
+ (char *)(ppnt + epnt->e_phnum) > (char *)(header + st.st_size))
+ goto skip;
+
+ for (i = 0; i < epnt->e_phnum; i++) {
+ /* Be very lazy, and only byteswap the stuff we use */
+ if (byteswap == 1) {
+ ppnt->p_type = bswap_32(ppnt->p_type);
+ ppnt->p_vaddr = bswap_32(ppnt->p_vaddr);
+ ppnt->p_offset = bswap_32(ppnt->p_offset);
+ ppnt->p_filesz = bswap_32(ppnt->p_filesz);
+ }
+
+ if (loadaddr == (ElfW(Addr)) - 1 && ppnt->p_type == PT_LOAD)
+ loadaddr = (ppnt->p_vaddr & ~(page_size - 1)) -
+ (ppnt->p_offset & ~(page_size - 1));
+ if (ppnt->p_type == 2) {
+ dynamic_addr = ppnt->p_offset;
+ dynamic_size = ppnt->p_filesz;
+ };
+ ppnt++;
+ };
+
+ dpnt = (ElfW(Dyn) *) & header[dynamic_addr];
+ dynamic_size = dynamic_size / sizeof(ElfW(Dyn));
+ if ((char *)dpnt < (char *)header ||
+ (char *)(dpnt + dynamic_size) > (char *)(header + st.st_size))
+ goto skip;
+
+ if (byteswap == 1) {
+ dpnt->d_tag = bswap_32(dpnt->d_tag);
+ dpnt->d_un.d_val = bswap_32(dpnt->d_un.d_val);
+ }
+
+ while (dpnt->d_tag != DT_NULL) {
+ if (dpnt->d_tag == DT_STRTAB)
+ strtab_val = dpnt->d_un.d_val;
+ dpnt++;
+ if (byteswap == 1) {
+ dpnt->d_tag = bswap_32(dpnt->d_tag);
+ dpnt->d_un.d_val = bswap_32(dpnt->d_un.d_val);
+ }
+ };
+
+ if (!strtab_val)
+ goto skip;
+
+ dpnt = (ElfW(Dyn) *) & header[dynamic_addr];
+ while (dpnt->d_tag != DT_NULL) {
+ if (dpnt->d_tag == DT_SONAME || dpnt->d_tag == DT_NEEDED) {
+ needed_val = dpnt->d_un.d_val;
+ if (needed_val + strtab_val >= loadaddr ||
+ needed_val + strtab_val < st.st_size - loadaddr) {
+ needed =
+ (char *)(header - loadaddr + strtab_val +
+ needed_val);
+
+ if (dpnt->d_tag == DT_SONAME)
+ soname = xstrdup(needed);
+
+ for (j = 0; needed_tab[j].soname != NULL; j++) {
+ if (strcmp(needed, needed_tab[j].soname)
+ == 0) {
+ if (*type != LIB_ELF
+ && *type !=
+ needed_tab[j].type)
+ multi_libcs = 1;
+ *type = needed_tab[j].type;
+ }
+ }
+ }
+ }
+ dpnt++;
+ };
+
+ if (multi_libcs)
+ warn("%s appears to be for multiple libc's", name);
+
+ /* If we could not deduce the libc type, and we know what to expect, set the type */
+ if (*type == LIB_ELF && expected_type != LIB_ANY)
+ *type = expected_type;
+
+ if (expected_type != LIB_ANY && expected_type != LIB_ELF &&
+ expected_type != *type) {
+ warn("%s does not match type specified for directory!", name);
}
- }
- }
- dpnt++;
- };
-
- if (multi_libcs)
- warn("%s appears to be for multiple libc's", name);
-
- /* If we could not deduce the libc type, and we know what to expect, set the type */
- if(*type == LIB_ELF && expected_type != LIB_ANY) *type = expected_type;
-
- if(expected_type != LIB_ANY && expected_type != LIB_ELF &&
- expected_type != *type)
- {
- warn("%s does not match type specified for directory!", name);
- }
-
- skip:
- munmap(header, st.st_size);
-
- return soname;
+
+ skip:
+ munmap(header, st.st_size);
+
+ return soname;
}