| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143 | 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;#if __BYTE_ORDER == __LITTLE_ENDIAN	byteswap = (epnt->e_ident[5] == ELFDATA2MSB) ? 1 : 0;#elif __BYTE_ORDER == __BIG_ENDIAN	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;					}				}			}		}		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;}
 |