* commit '746f128602772997b57e24fc8546473d01d6f955': Use more types than just 'unsigned' in the linker.
This commit is contained in:
commit
60414ba165
@ -215,7 +215,7 @@ soinfo libdl_info = {
|
|||||||
|
|
||||||
phdr: 0, phnum: 0,
|
phdr: 0, phnum: 0,
|
||||||
entry: 0, base: 0, size: 0,
|
entry: 0, base: 0, size: 0,
|
||||||
unused: 0, dynamic: 0, unused2: 0, unused3: 0,
|
unused1: 0, dynamic: 0, unused2: 0, unused3: 0,
|
||||||
next: 0,
|
next: 0,
|
||||||
|
|
||||||
flags: FLAG_LINKED,
|
flags: FLAG_LINKED,
|
||||||
@ -238,7 +238,7 @@ soinfo libdl_info = {
|
|||||||
mips_symtabno: 0, mips_local_gotno: 0, mips_gotsym: 0,
|
mips_symtabno: 0, mips_local_gotno: 0, mips_gotsym: 0,
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
refcount: 0,
|
ref_count: 0,
|
||||||
{ l_addr: 0, l_name: 0, l_ld: 0, l_next: 0, l_prev: 0, },
|
{ l_addr: 0, l_name: 0, l_ld: 0, l_next: 0, l_prev: 0, },
|
||||||
constructors_called: false,
|
constructors_called: false,
|
||||||
load_bias: 0,
|
load_bias: 0,
|
||||||
|
@ -174,13 +174,13 @@ extern "C" void __attribute__((noinline)) __attribute__((visibility("default")))
|
|||||||
|
|
||||||
static r_debug _r_debug = {1, NULL, &rtld_db_dlactivity,
|
static r_debug _r_debug = {1, NULL, &rtld_db_dlactivity,
|
||||||
RT_CONSISTENT, 0};
|
RT_CONSISTENT, 0};
|
||||||
static link_map* r_debug_tail = 0;
|
static link_map_t* r_debug_tail = 0;
|
||||||
|
|
||||||
static pthread_mutex_t gDebugMutex = PTHREAD_MUTEX_INITIALIZER;
|
static pthread_mutex_t gDebugMutex = PTHREAD_MUTEX_INITIALIZER;
|
||||||
|
|
||||||
static void insert_soinfo_into_debug_map(soinfo * info) {
|
static void insert_soinfo_into_debug_map(soinfo * info) {
|
||||||
// Copy the necessary fields into the debug structure.
|
// Copy the necessary fields into the debug structure.
|
||||||
link_map* map = &(info->linkmap);
|
link_map_t* map = &(info->link_map);
|
||||||
map->l_addr = info->base;
|
map->l_addr = info->base;
|
||||||
map->l_name = (char*) info->name;
|
map->l_name = (char*) info->name;
|
||||||
map->l_ld = (uintptr_t)info->dynamic;
|
map->l_ld = (uintptr_t)info->dynamic;
|
||||||
@ -203,7 +203,7 @@ static void insert_soinfo_into_debug_map(soinfo * info) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
static void remove_soinfo_from_debug_map(soinfo* info) {
|
static void remove_soinfo_from_debug_map(soinfo* info) {
|
||||||
link_map* map = &(info->linkmap);
|
link_map_t* map = &(info->link_map);
|
||||||
|
|
||||||
if (r_debug_tail == map) {
|
if (r_debug_tail == map) {
|
||||||
r_debug_tail = map->l_prev;
|
r_debug_tail = map->l_prev;
|
||||||
@ -315,7 +315,7 @@ static soinfo* soinfo_alloc(const char* name) {
|
|||||||
sonext->next = si;
|
sonext->next = si;
|
||||||
sonext = si;
|
sonext = si;
|
||||||
|
|
||||||
TRACE("name %s: allocated soinfo @ %p\n", name, si);
|
TRACE("name %s: allocated soinfo @ %p", name, si);
|
||||||
return si;
|
return si;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -327,7 +327,7 @@ static void soinfo_free(soinfo* si)
|
|||||||
|
|
||||||
soinfo *prev = NULL, *trav;
|
soinfo *prev = NULL, *trav;
|
||||||
|
|
||||||
TRACE("name %s: freeing soinfo @ %p\n", si->name, si);
|
TRACE("name %s: freeing soinfo @ %p", si->name, si);
|
||||||
|
|
||||||
for (trav = solist; trav != NULL; trav = trav->next) {
|
for (trav = solist; trav != NULL; trav = trav->next) {
|
||||||
if (trav == si)
|
if (trav == si)
|
||||||
@ -425,8 +425,8 @@ dl_iterate_phdr(int (*cb)(dl_phdr_info *info, size_t size, void *data),
|
|||||||
int rv = 0;
|
int rv = 0;
|
||||||
for (soinfo* si = solist; si != NULL; si = si->next) {
|
for (soinfo* si = solist; si != NULL; si = si->next) {
|
||||||
dl_phdr_info dl_info;
|
dl_phdr_info dl_info;
|
||||||
dl_info.dlpi_addr = si->linkmap.l_addr;
|
dl_info.dlpi_addr = si->link_map.l_addr;
|
||||||
dl_info.dlpi_name = si->linkmap.l_name;
|
dl_info.dlpi_name = si->link_map.l_name;
|
||||||
dl_info.dlpi_phdr = si->phdr;
|
dl_info.dlpi_phdr = si->phdr;
|
||||||
dl_info.dlpi_phnum = si->phnum;
|
dl_info.dlpi_phnum = si->phnum;
|
||||||
rv = cb(&dl_info, sizeof(dl_phdr_info), data);
|
rv = cb(&dl_info, sizeof(dl_phdr_info), data);
|
||||||
@ -445,7 +445,7 @@ static Elf32_Sym* soinfo_elf_lookup(soinfo* si, unsigned hash, const char* name)
|
|||||||
const char* strtab = si->strtab;
|
const char* strtab = si->strtab;
|
||||||
unsigned n;
|
unsigned n;
|
||||||
|
|
||||||
TRACE_TYPE(LOOKUP, "SEARCH %s in %s@0x%08x %08x %d\n",
|
TRACE_TYPE(LOOKUP, "SEARCH %s in %s@0x%08x %08x %d",
|
||||||
name, si->name, si->base, hash, hash % si->nbucket);
|
name, si->name, si->base, hash, hash % si->nbucket);
|
||||||
n = hash % si->nbucket;
|
n = hash % si->nbucket;
|
||||||
|
|
||||||
@ -461,7 +461,7 @@ static Elf32_Sym* soinfo_elf_lookup(soinfo* si, unsigned hash, const char* name)
|
|||||||
continue;
|
continue;
|
||||||
}
|
}
|
||||||
|
|
||||||
TRACE_TYPE(LOOKUP, "FOUND %s in %s (%08x) %d\n",
|
TRACE_TYPE(LOOKUP, "FOUND %s in %s (%08x) %d",
|
||||||
name, si->name, s->st_value, s->st_size);
|
name, si->name, s->st_value, s->st_size);
|
||||||
return s;
|
return s;
|
||||||
}
|
}
|
||||||
@ -510,7 +510,7 @@ static Elf32_Sym* soinfo_do_lookup(soinfo* si, const char* name, soinfo** lsi, s
|
|||||||
*/
|
*/
|
||||||
|
|
||||||
if (!si->has_DT_SYMBOLIC) {
|
if (!si->has_DT_SYMBOLIC) {
|
||||||
DEBUG("%s: looking up %s in executable %s\n",
|
DEBUG("%s: looking up %s in executable %s",
|
||||||
si->name, name, somain->name);
|
si->name, name, somain->name);
|
||||||
s = soinfo_elf_lookup(somain, elf_hash, name);
|
s = soinfo_elf_lookup(somain, elf_hash, name);
|
||||||
if (s != NULL) {
|
if (s != NULL) {
|
||||||
@ -541,7 +541,7 @@ static Elf32_Sym* soinfo_do_lookup(soinfo* si, const char* name, soinfo** lsi, s
|
|||||||
*/
|
*/
|
||||||
|
|
||||||
if (si->has_DT_SYMBOLIC) {
|
if (si->has_DT_SYMBOLIC) {
|
||||||
DEBUG("%s: looking up %s in executable %s after local scope\n",
|
DEBUG("%s: looking up %s in executable %s after local scope",
|
||||||
si->name, name, somain->name);
|
si->name, name, somain->name);
|
||||||
s = soinfo_elf_lookup(somain, elf_hash, name);
|
s = soinfo_elf_lookup(somain, elf_hash, name);
|
||||||
if (s != NULL) {
|
if (s != NULL) {
|
||||||
@ -562,7 +562,7 @@ static Elf32_Sym* soinfo_do_lookup(soinfo* si, const char* name, soinfo** lsi, s
|
|||||||
}
|
}
|
||||||
|
|
||||||
for (int i = 0; needed[i] != NULL; i++) {
|
for (int i = 0; needed[i] != NULL; i++) {
|
||||||
DEBUG("%s: looking up %s in %s\n",
|
DEBUG("%s: looking up %s in %s",
|
||||||
si->name, name, needed[i]->name);
|
si->name, name, needed[i]->name);
|
||||||
s = soinfo_elf_lookup(needed[i], elf_hash, name);
|
s = soinfo_elf_lookup(needed[i], elf_hash, name);
|
||||||
if (s != NULL) {
|
if (s != NULL) {
|
||||||
@ -574,7 +574,7 @@ static Elf32_Sym* soinfo_do_lookup(soinfo* si, const char* name, soinfo** lsi, s
|
|||||||
done:
|
done:
|
||||||
if (s != NULL) {
|
if (s != NULL) {
|
||||||
TRACE_TYPE(LOOKUP, "si %s sym %s s->st_value = 0x%08x, "
|
TRACE_TYPE(LOOKUP, "si %s sym %s s->st_value = 0x%08x, "
|
||||||
"found in %s, base = 0x%08x, load bias = 0x%08x\n",
|
"found in %s, base = 0x%08x, load bias = 0x%08x",
|
||||||
si->name, name, s->st_value,
|
si->name, name, s->st_value,
|
||||||
(*lsi)->name, (*lsi)->base, (*lsi)->load_bias);
|
(*lsi)->name, (*lsi)->base, (*lsi)->load_bias);
|
||||||
return s;
|
return s;
|
||||||
@ -619,7 +619,7 @@ Elf32_Sym* dlsym_linear_lookup(const char* name, soinfo** found, soinfo* start)
|
|||||||
}
|
}
|
||||||
|
|
||||||
if (s != NULL) {
|
if (s != NULL) {
|
||||||
TRACE_TYPE(LOOKUP, "%s s->st_value = 0x%08x, found->base = 0x%08x\n",
|
TRACE_TYPE(LOOKUP, "%s s->st_value = 0x%08x, found->base = 0x%08x",
|
||||||
name, s->st_value, (*found)->base);
|
name, s->st_value, (*found)->base);
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -658,7 +658,7 @@ static void dump(soinfo* si)
|
|||||||
{
|
{
|
||||||
Elf32_Sym* s = si->symtab;
|
Elf32_Sym* s = si->symtab;
|
||||||
for (unsigned n = 0; n < si->nchain; n++) {
|
for (unsigned n = 0; n < si->nchain; n++) {
|
||||||
TRACE("%04d> %08x: %02x %04x %08x %08x %s\n", n, s,
|
TRACE("%04d> %08x: %02x %04x %08x %08x %s", n, s,
|
||||||
s->st_info, s->st_shndx, s->st_value, s->st_size,
|
s->st_info, s->st_shndx, s->st_value, s->st_size,
|
||||||
si->strtab + s->st_name);
|
si->strtab + s->st_name);
|
||||||
s++;
|
s++;
|
||||||
@ -671,7 +671,7 @@ static int open_library_on_path(const char* name, const char* const paths[]) {
|
|||||||
for (size_t i = 0; paths[i] != NULL; ++i) {
|
for (size_t i = 0; paths[i] != NULL; ++i) {
|
||||||
int n = __libc_format_buffer(buf, sizeof(buf), "%s/%s", paths[i], name);
|
int n = __libc_format_buffer(buf, sizeof(buf), "%s/%s", paths[i], name);
|
||||||
if (n < 0 || n >= static_cast<int>(sizeof(buf))) {
|
if (n < 0 || n >= static_cast<int>(sizeof(buf))) {
|
||||||
PRINT("Warning: ignoring very long library path: %s/%s\n", paths[i], name);
|
PRINT("Warning: ignoring very long library path: %s/%s", paths[i], name);
|
||||||
continue;
|
continue;
|
||||||
}
|
}
|
||||||
int fd = TEMP_FAILURE_RETRY(open(buf, O_RDONLY | O_CLOEXEC));
|
int fd = TEMP_FAILURE_RETRY(open(buf, O_RDONLY | O_CLOEXEC));
|
||||||
@ -683,7 +683,7 @@ static int open_library_on_path(const char* name, const char* const paths[]) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
static int open_library(const char* name) {
|
static int open_library(const char* name) {
|
||||||
TRACE("[ opening %s ]\n", name);
|
TRACE("[ opening %s ]", name);
|
||||||
|
|
||||||
// If the name contains a slash, we should attempt to open it directly and not search the paths.
|
// If the name contains a slash, we should attempt to open it directly and not search the paths.
|
||||||
if (strchr(name, '/') != NULL) {
|
if (strchr(name, '/') != NULL) {
|
||||||
@ -765,7 +765,7 @@ static soinfo* find_library_internal(const char* name) {
|
|||||||
return NULL;
|
return NULL;
|
||||||
}
|
}
|
||||||
|
|
||||||
TRACE("[ '%s' has not been loaded yet. Locating...]\n", name);
|
TRACE("[ '%s' has not been loaded yet. Locating...]", name);
|
||||||
si = load_library(name);
|
si = load_library(name);
|
||||||
if (si == NULL) {
|
if (si == NULL) {
|
||||||
return NULL;
|
return NULL;
|
||||||
@ -773,7 +773,7 @@ static soinfo* find_library_internal(const char* name) {
|
|||||||
|
|
||||||
// At this point we know that whatever is loaded @ base is a valid ELF
|
// At this point we know that whatever is loaded @ base is a valid ELF
|
||||||
// shared library whose segments are properly mapped in.
|
// shared library whose segments are properly mapped in.
|
||||||
TRACE("[ init_library base=0x%08x sz=0x%08x name='%s') ]\n",
|
TRACE("[ init_library base=0x%08x sz=0x%08x name='%s' ]",
|
||||||
si->base, si->size, si->name);
|
si->base, si->size, si->name);
|
||||||
|
|
||||||
if (!soinfo_link_image(si)) {
|
if (!soinfo_link_image(si)) {
|
||||||
@ -788,14 +788,14 @@ static soinfo* find_library_internal(const char* name) {
|
|||||||
static soinfo* find_library(const char* name) {
|
static soinfo* find_library(const char* name) {
|
||||||
soinfo* si = find_library_internal(name);
|
soinfo* si = find_library_internal(name);
|
||||||
if (si != NULL) {
|
if (si != NULL) {
|
||||||
si->refcount++;
|
si->ref_count++;
|
||||||
}
|
}
|
||||||
return si;
|
return si;
|
||||||
}
|
}
|
||||||
|
|
||||||
static int soinfo_unload(soinfo* si) {
|
static int soinfo_unload(soinfo* si) {
|
||||||
if (si->refcount == 1) {
|
if (si->ref_count == 1) {
|
||||||
TRACE("unloading '%s'\n", si->name);
|
TRACE("unloading '%s'", si->name);
|
||||||
si->CallDestructors();
|
si->CallDestructors();
|
||||||
|
|
||||||
for (Elf32_Dyn* d = si->dynamic; d->d_tag != DT_NULL; ++d) {
|
for (Elf32_Dyn* d = si->dynamic; d->d_tag != DT_NULL; ++d) {
|
||||||
@ -803,7 +803,7 @@ static int soinfo_unload(soinfo* si) {
|
|||||||
const char* library_name = si->strtab + d->d_un.d_val;
|
const char* library_name = si->strtab + d->d_un.d_val;
|
||||||
soinfo* lsi = find_loaded_library(library_name);
|
soinfo* lsi = find_loaded_library(library_name);
|
||||||
if (lsi != NULL) {
|
if (lsi != NULL) {
|
||||||
TRACE("%s needs to unload %s\n", si->name, lsi->name);
|
TRACE("%s needs to unload %s", si->name, lsi->name);
|
||||||
soinfo_unload(lsi);
|
soinfo_unload(lsi);
|
||||||
} else {
|
} else {
|
||||||
// TODO: should we return -1 in this case?
|
// TODO: should we return -1 in this case?
|
||||||
@ -815,10 +815,10 @@ static int soinfo_unload(soinfo* si) {
|
|||||||
munmap(reinterpret_cast<void*>(si->base), si->size);
|
munmap(reinterpret_cast<void*>(si->base), si->size);
|
||||||
notify_gdb_of_unload(si);
|
notify_gdb_of_unload(si);
|
||||||
soinfo_free(si);
|
soinfo_free(si);
|
||||||
si->refcount = 0;
|
si->ref_count = 0;
|
||||||
} else {
|
} else {
|
||||||
si->refcount--;
|
si->ref_count--;
|
||||||
TRACE("not unloading '%s', decrementing refcount to %d\n", si->name, si->refcount);
|
TRACE("not unloading '%s', decrementing ref_count to %d", si->name, si->ref_count);
|
||||||
}
|
}
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
@ -870,7 +870,7 @@ static int soinfo_relocate(soinfo* si, Elf32_Rel* rel, unsigned count,
|
|||||||
Elf32_Addr sym_addr = 0;
|
Elf32_Addr sym_addr = 0;
|
||||||
char* sym_name = NULL;
|
char* sym_name = NULL;
|
||||||
|
|
||||||
DEBUG("Processing '%s' relocation at index %d\n", si->name, idx);
|
DEBUG("Processing '%s' relocation at index %d", si->name, idx);
|
||||||
if (type == 0) { // R_*_NONE
|
if (type == 0) { // R_*_NONE
|
||||||
continue;
|
continue;
|
||||||
}
|
}
|
||||||
@ -957,25 +957,25 @@ static int soinfo_relocate(soinfo* si, Elf32_Rel* rel, unsigned count,
|
|||||||
case R_ARM_JUMP_SLOT:
|
case R_ARM_JUMP_SLOT:
|
||||||
count_relocation(kRelocAbsolute);
|
count_relocation(kRelocAbsolute);
|
||||||
MARK(rel->r_offset);
|
MARK(rel->r_offset);
|
||||||
TRACE_TYPE(RELO, "RELO JMP_SLOT %08x <- %08x %s\n", reloc, sym_addr, sym_name);
|
TRACE_TYPE(RELO, "RELO JMP_SLOT %08x <- %08x %s", reloc, sym_addr, sym_name);
|
||||||
*reinterpret_cast<Elf32_Addr*>(reloc) = sym_addr;
|
*reinterpret_cast<Elf32_Addr*>(reloc) = sym_addr;
|
||||||
break;
|
break;
|
||||||
case R_ARM_GLOB_DAT:
|
case R_ARM_GLOB_DAT:
|
||||||
count_relocation(kRelocAbsolute);
|
count_relocation(kRelocAbsolute);
|
||||||
MARK(rel->r_offset);
|
MARK(rel->r_offset);
|
||||||
TRACE_TYPE(RELO, "RELO GLOB_DAT %08x <- %08x %s\n", reloc, sym_addr, sym_name);
|
TRACE_TYPE(RELO, "RELO GLOB_DAT %08x <- %08x %s", reloc, sym_addr, sym_name);
|
||||||
*reinterpret_cast<Elf32_Addr*>(reloc) = sym_addr;
|
*reinterpret_cast<Elf32_Addr*>(reloc) = sym_addr;
|
||||||
break;
|
break;
|
||||||
case R_ARM_ABS32:
|
case R_ARM_ABS32:
|
||||||
count_relocation(kRelocAbsolute);
|
count_relocation(kRelocAbsolute);
|
||||||
MARK(rel->r_offset);
|
MARK(rel->r_offset);
|
||||||
TRACE_TYPE(RELO, "RELO ABS %08x <- %08x %s\n", reloc, sym_addr, sym_name);
|
TRACE_TYPE(RELO, "RELO ABS %08x <- %08x %s", reloc, sym_addr, sym_name);
|
||||||
*reinterpret_cast<Elf32_Addr*>(reloc) += sym_addr;
|
*reinterpret_cast<Elf32_Addr*>(reloc) += sym_addr;
|
||||||
break;
|
break;
|
||||||
case R_ARM_REL32:
|
case R_ARM_REL32:
|
||||||
count_relocation(kRelocRelative);
|
count_relocation(kRelocRelative);
|
||||||
MARK(rel->r_offset);
|
MARK(rel->r_offset);
|
||||||
TRACE_TYPE(RELO, "RELO REL32 %08x <- %08x - %08x %s\n",
|
TRACE_TYPE(RELO, "RELO REL32 %08x <- %08x - %08x %s",
|
||||||
reloc, sym_addr, rel->r_offset, sym_name);
|
reloc, sym_addr, rel->r_offset, sym_name);
|
||||||
*reinterpret_cast<Elf32_Addr*>(reloc) += sym_addr - rel->r_offset;
|
*reinterpret_cast<Elf32_Addr*>(reloc) += sym_addr - rel->r_offset;
|
||||||
break;
|
break;
|
||||||
@ -983,20 +983,20 @@ static int soinfo_relocate(soinfo* si, Elf32_Rel* rel, unsigned count,
|
|||||||
case R_386_JMP_SLOT:
|
case R_386_JMP_SLOT:
|
||||||
count_relocation(kRelocAbsolute);
|
count_relocation(kRelocAbsolute);
|
||||||
MARK(rel->r_offset);
|
MARK(rel->r_offset);
|
||||||
TRACE_TYPE(RELO, "RELO JMP_SLOT %08x <- %08x %s\n", reloc, sym_addr, sym_name);
|
TRACE_TYPE(RELO, "RELO JMP_SLOT %08x <- %08x %s", reloc, sym_addr, sym_name);
|
||||||
*reinterpret_cast<Elf32_Addr*>(reloc) = sym_addr;
|
*reinterpret_cast<Elf32_Addr*>(reloc) = sym_addr;
|
||||||
break;
|
break;
|
||||||
case R_386_GLOB_DAT:
|
case R_386_GLOB_DAT:
|
||||||
count_relocation(kRelocAbsolute);
|
count_relocation(kRelocAbsolute);
|
||||||
MARK(rel->r_offset);
|
MARK(rel->r_offset);
|
||||||
TRACE_TYPE(RELO, "RELO GLOB_DAT %08x <- %08x %s\n", reloc, sym_addr, sym_name);
|
TRACE_TYPE(RELO, "RELO GLOB_DAT %08x <- %08x %s", reloc, sym_addr, sym_name);
|
||||||
*reinterpret_cast<Elf32_Addr*>(reloc) = sym_addr;
|
*reinterpret_cast<Elf32_Addr*>(reloc) = sym_addr;
|
||||||
break;
|
break;
|
||||||
#elif defined(ANDROID_MIPS_LINKER)
|
#elif defined(ANDROID_MIPS_LINKER)
|
||||||
case R_MIPS_REL32:
|
case R_MIPS_REL32:
|
||||||
count_relocation(kRelocAbsolute);
|
count_relocation(kRelocAbsolute);
|
||||||
MARK(rel->r_offset);
|
MARK(rel->r_offset);
|
||||||
TRACE_TYPE(RELO, "RELO REL32 %08x <- %08x %s\n",
|
TRACE_TYPE(RELO, "RELO REL32 %08x <- %08x %s",
|
||||||
reloc, sym_addr, (sym_name) ? sym_name : "*SECTIONHDR*");
|
reloc, sym_addr, (sym_name) ? sym_name : "*SECTIONHDR*");
|
||||||
if (s) {
|
if (s) {
|
||||||
*reinterpret_cast<Elf32_Addr*>(reloc) += sym_addr;
|
*reinterpret_cast<Elf32_Addr*>(reloc) += sym_addr;
|
||||||
@ -1017,7 +1017,7 @@ static int soinfo_relocate(soinfo* si, Elf32_Rel* rel, unsigned count,
|
|||||||
DL_ERR("odd RELATIVE form...");
|
DL_ERR("odd RELATIVE form...");
|
||||||
return -1;
|
return -1;
|
||||||
}
|
}
|
||||||
TRACE_TYPE(RELO, "RELO RELATIVE %08x <- +%08x\n", reloc, si->base);
|
TRACE_TYPE(RELO, "RELO RELATIVE %08x <- +%08x", reloc, si->base);
|
||||||
*reinterpret_cast<Elf32_Addr*>(reloc) += si->base;
|
*reinterpret_cast<Elf32_Addr*>(reloc) += si->base;
|
||||||
break;
|
break;
|
||||||
|
|
||||||
@ -1026,14 +1026,14 @@ static int soinfo_relocate(soinfo* si, Elf32_Rel* rel, unsigned count,
|
|||||||
count_relocation(kRelocRelative);
|
count_relocation(kRelocRelative);
|
||||||
MARK(rel->r_offset);
|
MARK(rel->r_offset);
|
||||||
|
|
||||||
TRACE_TYPE(RELO, "RELO R_386_32 %08x <- +%08x %s\n", reloc, sym_addr, sym_name);
|
TRACE_TYPE(RELO, "RELO R_386_32 %08x <- +%08x %s", reloc, sym_addr, sym_name);
|
||||||
*reinterpret_cast<Elf32_Addr*>(reloc) += sym_addr;
|
*reinterpret_cast<Elf32_Addr*>(reloc) += sym_addr;
|
||||||
break;
|
break;
|
||||||
|
|
||||||
case R_386_PC32:
|
case R_386_PC32:
|
||||||
count_relocation(kRelocRelative);
|
count_relocation(kRelocRelative);
|
||||||
MARK(rel->r_offset);
|
MARK(rel->r_offset);
|
||||||
TRACE_TYPE(RELO, "RELO R_386_PC32 %08x <- +%08x (%08x - %08x) %s\n",
|
TRACE_TYPE(RELO, "RELO R_386_PC32 %08x <- +%08x (%08x - %08x) %s",
|
||||||
reloc, (sym_addr - reloc), sym_addr, reloc, sym_name);
|
reloc, (sym_addr - reloc), sym_addr, reloc, sym_name);
|
||||||
*reinterpret_cast<Elf32_Addr*>(reloc) += (sym_addr - reloc);
|
*reinterpret_cast<Elf32_Addr*>(reloc) += (sym_addr - reloc);
|
||||||
break;
|
break;
|
||||||
@ -1058,7 +1058,7 @@ static int soinfo_relocate(soinfo* si, Elf32_Rel* rel, unsigned count,
|
|||||||
}
|
}
|
||||||
count_relocation(kRelocCopy);
|
count_relocation(kRelocCopy);
|
||||||
MARK(rel->r_offset);
|
MARK(rel->r_offset);
|
||||||
TRACE_TYPE(RELO, "RELO %08x <- %d @ %08x %s\n", reloc, s->st_size, sym_addr, sym_name);
|
TRACE_TYPE(RELO, "RELO %08x <- %d @ %08x %s", reloc, s->st_size, sym_addr, sym_name);
|
||||||
if (reloc == sym_addr) {
|
if (reloc == sym_addr) {
|
||||||
Elf32_Sym *src = soinfo_do_lookup(NULL, sym_name, &lsi, needed);
|
Elf32_Sym *src = soinfo_do_lookup(NULL, sym_name, &lsi, needed);
|
||||||
|
|
||||||
@ -1176,37 +1176,33 @@ static int mips_relocate_got(soinfo* si, soinfo* needed[]) {
|
|||||||
*
|
*
|
||||||
* DT_FINI_ARRAY must be parsed in reverse order.
|
* DT_FINI_ARRAY must be parsed in reverse order.
|
||||||
*/
|
*/
|
||||||
void soinfo::CallArray(const char* array_name UNUSED, unsigned* array, int count, bool reverse) {
|
void soinfo::CallArray(const char* array_name UNUSED, linker_function_t* functions, size_t count, bool reverse) {
|
||||||
if (array == NULL) {
|
if (functions == NULL) {
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
int step = 1;
|
TRACE("[ Calling %s (size %d) @ %p for '%s' ]", array_name, count, functions, name);
|
||||||
if (reverse) {
|
|
||||||
array += (count-1);
|
int begin = reverse ? (count - 1) : 0;
|
||||||
step = -1;
|
int end = reverse ? -1 : count;
|
||||||
|
int step = reverse ? -1 : 1;
|
||||||
|
|
||||||
|
for (int i = begin; i != end; i += step) {
|
||||||
|
TRACE("[ %s[%d] == %p ]", array_name, i, functions[i]);
|
||||||
|
CallFunction("function", functions[i]);
|
||||||
}
|
}
|
||||||
|
|
||||||
TRACE("[ Calling %s @ %p [%d] for '%s' ]\n", array_name, array, count, name);
|
TRACE("[ Done calling %s for '%s' ]", array_name, name);
|
||||||
|
|
||||||
for (int n = count; n > 0; n--) {
|
|
||||||
TRACE("[ Looking at %s[%d] *%p == 0x%08x ]\n", array_name, n, array, *array);
|
|
||||||
void (*func)() = (void (*)()) *array;
|
|
||||||
array += step;
|
|
||||||
CallFunction("function", func);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
TRACE("[ Done calling %s for '%s' ]\n", array_name, name);
|
void soinfo::CallFunction(const char* function_name UNUSED, linker_function_t function) {
|
||||||
}
|
|
||||||
|
|
||||||
void soinfo::CallFunction(const char* function_name UNUSED, void (*function)()) {
|
|
||||||
if (function == NULL || reinterpret_cast<uintptr_t>(function) == static_cast<uintptr_t>(-1)) {
|
if (function == NULL || reinterpret_cast<uintptr_t>(function) == static_cast<uintptr_t>(-1)) {
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
TRACE("[ Calling %s @ %p for '%s' ]\n", function_name, function, name);
|
TRACE("[ Calling %s @ %p for '%s' ]", function_name, function, name);
|
||||||
function();
|
function();
|
||||||
TRACE("[ Done calling %s for '%s' ]\n", function_name, name);
|
TRACE("[ Done calling %s @ %p for '%s' ]", function_name, function, name);
|
||||||
|
|
||||||
// The function may have called dlopen(3) or dlclose(3), so we need to ensure our data structures
|
// The function may have called dlopen(3) or dlclose(3), so we need to ensure our data structures
|
||||||
// are still writable. This happens with our debug malloc (see http://b/7941716).
|
// are still writable. This happens with our debug malloc (see http://b/7941716).
|
||||||
@ -1273,7 +1269,7 @@ static int nullify_closed_stdio() {
|
|||||||
DL_ERR("cannot open /dev/null: %s", strerror(errno));
|
DL_ERR("cannot open /dev/null: %s", strerror(errno));
|
||||||
return -1;
|
return -1;
|
||||||
}
|
}
|
||||||
TRACE("[ Opened /dev/null file-descriptor=%d]\n", dev_null);
|
TRACE("[ Opened /dev/null file-descriptor=%d]", dev_null);
|
||||||
|
|
||||||
/* If any of the stdio file descriptors is valid and not associated
|
/* If any of the stdio file descriptors is valid and not associated
|
||||||
with /dev/null, dup /dev/null to it. */
|
with /dev/null, dup /dev/null to it. */
|
||||||
@ -1283,7 +1279,7 @@ static int nullify_closed_stdio() {
|
|||||||
continue;
|
continue;
|
||||||
}
|
}
|
||||||
|
|
||||||
TRACE("[ Nullifying stdio file descriptor %d]\n", i);
|
TRACE("[ Nullifying stdio file descriptor %d]", i);
|
||||||
status = TEMP_FAILURE_RETRY(fcntl(i, F_GETFL));
|
status = TEMP_FAILURE_RETRY(fcntl(i, F_GETFL));
|
||||||
|
|
||||||
/* If file is opened, we are good. */
|
/* If file is opened, we are good. */
|
||||||
@ -1312,7 +1308,7 @@ static int nullify_closed_stdio() {
|
|||||||
|
|
||||||
/* If /dev/null is not one of the stdio file descriptors, close it. */
|
/* If /dev/null is not one of the stdio file descriptors, close it. */
|
||||||
if (dev_null > 2) {
|
if (dev_null > 2) {
|
||||||
TRACE("[ Closing /dev/null file-descriptor=%d]\n", dev_null);
|
TRACE("[ Closing /dev/null file-descriptor=%d]", dev_null);
|
||||||
status = TEMP_FAILURE_RETRY(close(dev_null));
|
status = TEMP_FAILURE_RETRY(close(dev_null));
|
||||||
if (status == -1) {
|
if (status == -1) {
|
||||||
DL_ERR("close failed: %s", strerror(errno));
|
DL_ERR("close failed: %s", strerror(errno));
|
||||||
@ -1332,8 +1328,8 @@ static bool soinfo_link_image(soinfo* si) {
|
|||||||
|
|
||||||
/* We can't debug anything until the linker is relocated */
|
/* We can't debug anything until the linker is relocated */
|
||||||
if (!relocating_linker) {
|
if (!relocating_linker) {
|
||||||
INFO("[ linking %s ]\n", si->name);
|
INFO("[ linking %s ]", si->name);
|
||||||
DEBUG("si->base = 0x%08x si->flags = 0x%08x\n", si->base, si->flags);
|
DEBUG("si->base = 0x%08x si->flags = 0x%08x", si->base, si->flags);
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Extract dynamic section */
|
/* Extract dynamic section */
|
||||||
@ -1348,7 +1344,7 @@ static bool soinfo_link_image(soinfo* si) {
|
|||||||
return false;
|
return false;
|
||||||
} else {
|
} else {
|
||||||
if (!relocating_linker) {
|
if (!relocating_linker) {
|
||||||
DEBUG("dynamic = %p\n", si->dynamic);
|
DEBUG("dynamic = %p", si->dynamic);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -1360,7 +1356,7 @@ static bool soinfo_link_image(soinfo* si) {
|
|||||||
/* extract useful information from dynamic section */
|
/* extract useful information from dynamic section */
|
||||||
uint32_t needed_count = 0;
|
uint32_t needed_count = 0;
|
||||||
for (Elf32_Dyn* d = si->dynamic; d->d_tag != DT_NULL; ++d) {
|
for (Elf32_Dyn* d = si->dynamic; d->d_tag != DT_NULL; ++d) {
|
||||||
DEBUG("d = %p, d[0](tag) = 0x%08x d[1](val) = 0x%08x\n", d, d->d_tag, d->d_un.d_val);
|
DEBUG("d = %p, d[0](tag) = 0x%08x d[1](val) = 0x%08x", d, d->d_tag, d->d_un.d_val);
|
||||||
switch(d->d_tag){
|
switch(d->d_tag){
|
||||||
case DT_HASH:
|
case DT_HASH:
|
||||||
si->nbucket = ((unsigned *) (base + d->d_un.d_ptr))[0];
|
si->nbucket = ((unsigned *) (base + d->d_un.d_ptr))[0];
|
||||||
@ -1407,30 +1403,30 @@ static bool soinfo_link_image(soinfo* si) {
|
|||||||
DL_ERR("unsupported DT_RELA in \"%s\"", si->name);
|
DL_ERR("unsupported DT_RELA in \"%s\"", si->name);
|
||||||
return false;
|
return false;
|
||||||
case DT_INIT:
|
case DT_INIT:
|
||||||
si->init_func = (void (*)(void))(base + d->d_un.d_ptr);
|
si->init_func = reinterpret_cast<linker_function_t>(base + d->d_un.d_ptr);
|
||||||
DEBUG("%s constructors (init func) found at %p\n", si->name, si->init_func);
|
DEBUG("%s constructors (init func) found at %p", si->name, si->init_func);
|
||||||
break;
|
break;
|
||||||
case DT_FINI:
|
case DT_FINI:
|
||||||
si->fini_func = (void (*)(void))(base + d->d_un.d_ptr);
|
si->fini_func = reinterpret_cast<linker_function_t>(base + d->d_un.d_ptr);
|
||||||
DEBUG("%s destructors (fini func) found at %p\n", si->name, si->fini_func);
|
DEBUG("%s destructors (fini func) found at %p", si->name, si->fini_func);
|
||||||
break;
|
break;
|
||||||
case DT_INIT_ARRAY:
|
case DT_INIT_ARRAY:
|
||||||
si->init_array = (unsigned *)(base + d->d_un.d_ptr);
|
si->init_array = reinterpret_cast<linker_function_t*>(base + d->d_un.d_ptr);
|
||||||
DEBUG("%s constructors (init_array) found at %p\n", si->name, si->init_array);
|
DEBUG("%s constructors (init_array) found at %p", si->name, si->init_array);
|
||||||
break;
|
break;
|
||||||
case DT_INIT_ARRAYSZ:
|
case DT_INIT_ARRAYSZ:
|
||||||
si->init_array_count = ((unsigned)d->d_un.d_val) / sizeof(Elf32_Addr);
|
si->init_array_count = ((unsigned)d->d_un.d_val) / sizeof(Elf32_Addr);
|
||||||
break;
|
break;
|
||||||
case DT_FINI_ARRAY:
|
case DT_FINI_ARRAY:
|
||||||
si->fini_array = (unsigned *)(base + d->d_un.d_ptr);
|
si->fini_array = reinterpret_cast<linker_function_t*>(base + d->d_un.d_ptr);
|
||||||
DEBUG("%s destructors (fini_array) found at %p\n", si->name, si->fini_array);
|
DEBUG("%s destructors (fini_array) found at %p", si->name, si->fini_array);
|
||||||
break;
|
break;
|
||||||
case DT_FINI_ARRAYSZ:
|
case DT_FINI_ARRAYSZ:
|
||||||
si->fini_array_count = ((unsigned)d->d_un.d_val) / sizeof(Elf32_Addr);
|
si->fini_array_count = ((unsigned)d->d_un.d_val) / sizeof(Elf32_Addr);
|
||||||
break;
|
break;
|
||||||
case DT_PREINIT_ARRAY:
|
case DT_PREINIT_ARRAY:
|
||||||
si->preinit_array = (unsigned *)(base + d->d_un.d_ptr);
|
si->preinit_array = reinterpret_cast<linker_function_t*>(base + d->d_un.d_ptr);
|
||||||
DEBUG("%s constructors (preinit_array) found at %p\n", si->name, si->preinit_array);
|
DEBUG("%s constructors (preinit_array) found at %p", si->name, si->preinit_array);
|
||||||
break;
|
break;
|
||||||
case DT_PREINIT_ARRAYSZ:
|
case DT_PREINIT_ARRAYSZ:
|
||||||
si->preinit_array_count = ((unsigned)d->d_un.d_val) / sizeof(Elf32_Addr);
|
si->preinit_array_count = ((unsigned)d->d_un.d_val) / sizeof(Elf32_Addr);
|
||||||
@ -1486,13 +1482,13 @@ static bool soinfo_link_image(soinfo* si) {
|
|||||||
break;
|
break;
|
||||||
|
|
||||||
default:
|
default:
|
||||||
DEBUG("Unused DT entry: type 0x%08x arg 0x%08x\n", d->d_tag, d->d_un.d_val);
|
DEBUG("Unused DT entry: type 0x%08x arg 0x%08x", d->d_tag, d->d_un.d_val);
|
||||||
break;
|
break;
|
||||||
#endif
|
#endif
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
DEBUG("si->base = 0x%08x, si->strtab = %p, si->symtab = %p\n",
|
DEBUG("si->base = 0x%08x, si->strtab = %p, si->symtab = %p",
|
||||||
si->base, si->strtab, si->symtab);
|
si->base, si->strtab, si->symtab);
|
||||||
|
|
||||||
// Sanity checks.
|
// Sanity checks.
|
||||||
@ -1534,7 +1530,7 @@ static bool soinfo_link_image(soinfo* si) {
|
|||||||
for (Elf32_Dyn* d = si->dynamic; d->d_tag != DT_NULL; ++d) {
|
for (Elf32_Dyn* d = si->dynamic; d->d_tag != DT_NULL; ++d) {
|
||||||
if (d->d_tag == DT_NEEDED) {
|
if (d->d_tag == DT_NEEDED) {
|
||||||
const char* library_name = si->strtab + d->d_un.d_val;
|
const char* library_name = si->strtab + d->d_un.d_val;
|
||||||
DEBUG("%s needs %s\n", si->name, library_name);
|
DEBUG("%s needs %s", si->name, library_name);
|
||||||
soinfo* lsi = find_library(library_name);
|
soinfo* lsi = find_library(library_name);
|
||||||
if (lsi == NULL) {
|
if (lsi == NULL) {
|
||||||
strlcpy(tmp_err_buf, linker_get_error_buffer(), sizeof(tmp_err_buf));
|
strlcpy(tmp_err_buf, linker_get_error_buffer(), sizeof(tmp_err_buf));
|
||||||
@ -1561,13 +1557,13 @@ static bool soinfo_link_image(soinfo* si) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
if (si->plt_rel != NULL) {
|
if (si->plt_rel != NULL) {
|
||||||
DEBUG("[ relocating %s plt ]\n", si->name );
|
DEBUG("[ relocating %s plt ]", si->name );
|
||||||
if (soinfo_relocate(si, si->plt_rel, si->plt_rel_count, needed)) {
|
if (soinfo_relocate(si, si->plt_rel, si->plt_rel_count, needed)) {
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
if (si->rel != NULL) {
|
if (si->rel != NULL) {
|
||||||
DEBUG("[ relocating %s ]\n", si->name );
|
DEBUG("[ relocating %s ]", si->name );
|
||||||
if (soinfo_relocate(si, si->rel, si->rel_count, needed)) {
|
if (soinfo_relocate(si, si->rel, si->rel_count, needed)) {
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
@ -1580,7 +1576,7 @@ static bool soinfo_link_image(soinfo* si) {
|
|||||||
#endif
|
#endif
|
||||||
|
|
||||||
si->flags |= FLAG_LINKED;
|
si->flags |= FLAG_LINKED;
|
||||||
DEBUG("[ finished linking %s ]\n", si->name);
|
DEBUG("[ finished linking %s ]", si->name);
|
||||||
|
|
||||||
if (si->has_text_relocations) {
|
if (si->has_text_relocations) {
|
||||||
/* All relocations are done, we can protect our segments back to
|
/* All relocations are done, we can protect our segments back to
|
||||||
@ -1649,7 +1645,7 @@ static Elf32_Addr __linker_init_post_relocation(KernelArgumentBlock& args, Elf32
|
|||||||
ldpreload_env = linker_env_get("LD_PRELOAD");
|
ldpreload_env = linker_env_get("LD_PRELOAD");
|
||||||
}
|
}
|
||||||
|
|
||||||
INFO("[ android linker & debugger ]\n");
|
INFO("[ android linker & debugger ]");
|
||||||
|
|
||||||
soinfo* si = soinfo_alloc(args.argv[0]);
|
soinfo* si = soinfo_alloc(args.argv[0]);
|
||||||
if (si == NULL) {
|
if (si == NULL) {
|
||||||
@ -1658,7 +1654,7 @@ static Elf32_Addr __linker_init_post_relocation(KernelArgumentBlock& args, Elf32
|
|||||||
|
|
||||||
/* bootstrap the link map, the main exe always needs to be first */
|
/* bootstrap the link map, the main exe always needs to be first */
|
||||||
si->flags |= FLAG_EXE;
|
si->flags |= FLAG_EXE;
|
||||||
link_map* map = &(si->linkmap);
|
link_map_t* map = &(si->link_map);
|
||||||
|
|
||||||
map->l_addr = 0;
|
map->l_addr = 0;
|
||||||
map->l_name = args.argv[0];
|
map->l_name = args.argv[0];
|
||||||
@ -1706,7 +1702,7 @@ static Elf32_Addr __linker_init_post_relocation(KernelArgumentBlock& args, Elf32
|
|||||||
si->base = 0;
|
si->base = 0;
|
||||||
si->size = phdr_table_get_load_size(si->phdr, si->phnum);
|
si->size = phdr_table_get_load_size(si->phdr, si->phnum);
|
||||||
si->load_bias = 0;
|
si->load_bias = 0;
|
||||||
for (int i = 0; i < si->phnum; ++i) {
|
for (size_t i = 0; i < si->phnum; ++i) {
|
||||||
if (si->phdr[i].p_type == PT_PHDR) {
|
if (si->phdr[i].p_type == PT_PHDR) {
|
||||||
si->load_bias = reinterpret_cast<Elf32_Addr>(si->phdr) - si->phdr[i].p_vaddr;
|
si->load_bias = reinterpret_cast<Elf32_Addr>(si->phdr) - si->phdr[i].p_vaddr;
|
||||||
si->base = reinterpret_cast<Elf32_Addr>(si->phdr) - si->phdr[i].p_offset;
|
si->base = reinterpret_cast<Elf32_Addr>(si->phdr) - si->phdr[i].p_offset;
|
||||||
@ -1714,7 +1710,7 @@ static Elf32_Addr __linker_init_post_relocation(KernelArgumentBlock& args, Elf32
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
si->dynamic = NULL;
|
si->dynamic = NULL;
|
||||||
si->refcount = 1;
|
si->ref_count = 1;
|
||||||
|
|
||||||
// Use LD_LIBRARY_PATH and LD_PRELOAD (but only if we aren't setuid/setgid).
|
// Use LD_LIBRARY_PATH and LD_PRELOAD (but only if we aren't setuid/setgid).
|
||||||
parse_LD_LIBRARY_PATH(ldpath_env);
|
parse_LD_LIBRARY_PATH(ldpath_env);
|
||||||
@ -1743,13 +1739,13 @@ static Elf32_Addr __linker_init_post_relocation(KernelArgumentBlock& args, Elf32
|
|||||||
|
|
||||||
#if TIMING
|
#if TIMING
|
||||||
gettimeofday(&t1,NULL);
|
gettimeofday(&t1,NULL);
|
||||||
PRINT("LINKER TIME: %s: %d microseconds\n", args.argv[0], (int) (
|
PRINT("LINKER TIME: %s: %d microseconds", args.argv[0], (int) (
|
||||||
(((long long)t1.tv_sec * 1000000LL) + (long long)t1.tv_usec) -
|
(((long long)t1.tv_sec * 1000000LL) + (long long)t1.tv_usec) -
|
||||||
(((long long)t0.tv_sec * 1000000LL) + (long long)t0.tv_usec)
|
(((long long)t0.tv_sec * 1000000LL) + (long long)t0.tv_usec)
|
||||||
));
|
));
|
||||||
#endif
|
#endif
|
||||||
#if STATS
|
#if STATS
|
||||||
PRINT("RELO STATS: %s: %d abs, %d rel, %d copy, %d symbol\n", args.argv[0],
|
PRINT("RELO STATS: %s: %d abs, %d rel, %d copy, %d symbol", args.argv[0],
|
||||||
linker_stats.count[kRelocAbsolute],
|
linker_stats.count[kRelocAbsolute],
|
||||||
linker_stats.count[kRelocRelative],
|
linker_stats.count[kRelocRelative],
|
||||||
linker_stats.count[kRelocCopy],
|
linker_stats.count[kRelocCopy],
|
||||||
@ -1771,7 +1767,7 @@ static Elf32_Addr __linker_init_post_relocation(KernelArgumentBlock& args, Elf32
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
PRINT("PAGES MODIFIED: %s: %d (%dKB)\n", args.argv[0], count, count * 4);
|
PRINT("PAGES MODIFIED: %s: %d (%dKB)", args.argv[0], count, count * 4);
|
||||||
}
|
}
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
@ -1779,7 +1775,7 @@ static Elf32_Addr __linker_init_post_relocation(KernelArgumentBlock& args, Elf32
|
|||||||
fflush(stdout);
|
fflush(stdout);
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
TRACE("[ Ready to execute '%s' @ 0x%08x ]\n", si->name, si->entry);
|
TRACE("[ Ready to execute '%s' @ 0x%08x ]", si->name, si->entry);
|
||||||
return si->entry;
|
return si->entry;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -57,12 +57,12 @@
|
|||||||
|
|
||||||
// Magic shared structures that GDB knows about.
|
// Magic shared structures that GDB knows about.
|
||||||
|
|
||||||
struct link_map {
|
struct link_map_t {
|
||||||
uintptr_t l_addr;
|
uintptr_t l_addr;
|
||||||
char* l_name;
|
char* l_name;
|
||||||
uintptr_t l_ld;
|
uintptr_t l_ld;
|
||||||
struct link_map* l_next;
|
link_map_t* l_next;
|
||||||
struct link_map* l_prev;
|
link_map_t* l_prev;
|
||||||
};
|
};
|
||||||
|
|
||||||
// Values for r_debug->state
|
// Values for r_debug->state
|
||||||
@ -74,7 +74,7 @@ enum {
|
|||||||
|
|
||||||
struct r_debug {
|
struct r_debug {
|
||||||
int32_t r_version;
|
int32_t r_version;
|
||||||
struct link_map* r_map;
|
link_map_t* r_map;
|
||||||
void (*r_brk)(void);
|
void (*r_brk)(void);
|
||||||
int32_t r_state;
|
int32_t r_state;
|
||||||
uintptr_t r_ldbase;
|
uintptr_t r_ldbase;
|
||||||
@ -86,20 +86,23 @@ struct r_debug {
|
|||||||
|
|
||||||
#define SOINFO_NAME_LEN 128
|
#define SOINFO_NAME_LEN 128
|
||||||
|
|
||||||
|
typedef void (*linker_function_t)();
|
||||||
|
|
||||||
struct soinfo {
|
struct soinfo {
|
||||||
|
public:
|
||||||
char name[SOINFO_NAME_LEN];
|
char name[SOINFO_NAME_LEN];
|
||||||
const Elf32_Phdr* phdr;
|
const Elf32_Phdr* phdr;
|
||||||
int phnum;
|
size_t phnum;
|
||||||
Elf32_Addr entry;
|
Elf32_Addr entry;
|
||||||
Elf32_Addr base;
|
Elf32_Addr base;
|
||||||
unsigned size;
|
unsigned size;
|
||||||
|
|
||||||
int unused; // DO NOT USE, maintained for compatibility.
|
uint32_t unused1; // DO NOT USE, maintained for compatibility.
|
||||||
|
|
||||||
Elf32_Dyn* dynamic;
|
Elf32_Dyn* dynamic;
|
||||||
|
|
||||||
unsigned unused2; // DO NOT USE, maintained for compatibility
|
uint32_t unused2; // DO NOT USE, maintained for compatibility
|
||||||
unsigned unused3; // DO NOT USE, maintained for compatibility
|
uint32_t unused3; // DO NOT USE, maintained for compatibility
|
||||||
|
|
||||||
soinfo* next;
|
soinfo* next;
|
||||||
unsigned flags;
|
unsigned flags;
|
||||||
@ -107,42 +110,42 @@ struct soinfo {
|
|||||||
const char* strtab;
|
const char* strtab;
|
||||||
Elf32_Sym* symtab;
|
Elf32_Sym* symtab;
|
||||||
|
|
||||||
unsigned nbucket;
|
size_t nbucket;
|
||||||
unsigned nchain;
|
size_t nchain;
|
||||||
unsigned* bucket;
|
unsigned* bucket;
|
||||||
unsigned* chain;
|
unsigned* chain;
|
||||||
|
|
||||||
unsigned* plt_got;
|
unsigned* plt_got;
|
||||||
|
|
||||||
Elf32_Rel* plt_rel;
|
Elf32_Rel* plt_rel;
|
||||||
unsigned plt_rel_count;
|
size_t plt_rel_count;
|
||||||
|
|
||||||
Elf32_Rel* rel;
|
Elf32_Rel* rel;
|
||||||
unsigned rel_count;
|
size_t rel_count;
|
||||||
|
|
||||||
unsigned* preinit_array;
|
linker_function_t* preinit_array;
|
||||||
unsigned preinit_array_count;
|
size_t preinit_array_count;
|
||||||
|
|
||||||
unsigned* init_array;
|
linker_function_t* init_array;
|
||||||
unsigned init_array_count;
|
size_t init_array_count;
|
||||||
unsigned* fini_array;
|
linker_function_t* fini_array;
|
||||||
unsigned fini_array_count;
|
size_t fini_array_count;
|
||||||
|
|
||||||
void (*init_func)();
|
linker_function_t init_func;
|
||||||
void (*fini_func)();
|
linker_function_t fini_func;
|
||||||
|
|
||||||
#if defined(ANDROID_ARM_LINKER)
|
#if defined(ANDROID_ARM_LINKER)
|
||||||
// ARM EABI section used for stack unwinding.
|
// ARM EABI section used for stack unwinding.
|
||||||
unsigned* ARM_exidx;
|
unsigned* ARM_exidx;
|
||||||
unsigned ARM_exidx_count;
|
size_t ARM_exidx_count;
|
||||||
#elif defined(ANDROID_MIPS_LINKER)
|
#elif defined(ANDROID_MIPS_LINKER)
|
||||||
unsigned mips_symtabno;
|
unsigned mips_symtabno;
|
||||||
unsigned mips_local_gotno;
|
unsigned mips_local_gotno;
|
||||||
unsigned mips_gotsym;
|
unsigned mips_gotsym;
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
unsigned refcount;
|
size_t ref_count;
|
||||||
struct link_map linkmap;
|
link_map_t link_map;
|
||||||
|
|
||||||
bool constructors_called;
|
bool constructors_called;
|
||||||
|
|
||||||
@ -158,8 +161,8 @@ struct soinfo {
|
|||||||
void CallPreInitConstructors();
|
void CallPreInitConstructors();
|
||||||
|
|
||||||
private:
|
private:
|
||||||
void CallArray(const char* array_name, unsigned* array, int count, bool reverse);
|
void CallArray(const char* array_name, linker_function_t* functions, size_t count, bool reverse);
|
||||||
void CallFunction(const char* function_name, void (*function)());
|
void CallFunction(const char* function_name, linker_function_t function);
|
||||||
};
|
};
|
||||||
|
|
||||||
extern soinfo libdl_info;
|
extern soinfo libdl_info;
|
||||||
|
@ -70,7 +70,7 @@ __LIBC_HIDDEN__ extern int gLdDebugVerbosity;
|
|||||||
#else /* !LINKER_DEBUG_TO_LOG */
|
#else /* !LINKER_DEBUG_TO_LOG */
|
||||||
#define _PRINTVF(v,x...) \
|
#define _PRINTVF(v,x...) \
|
||||||
do { \
|
do { \
|
||||||
if (gLdDebugVerbosity > (v)) __libc_format_fd(1, x); \
|
if (gLdDebugVerbosity > (v)) { __libc_format_fd(1, x); write(1, "\n", 1); } \
|
||||||
} while (0)
|
} while (0)
|
||||||
#endif /* !LINKER_DEBUG_TO_LOG */
|
#endif /* !LINKER_DEBUG_TO_LOG */
|
||||||
|
|
||||||
|
Loading…
Reference in New Issue
Block a user