#define _GNU_SOURCE #include <link.h> #include <stdint.h> struct find_exidx_data { uintptr_t pc, exidx_start; int exidx_len; }; static int find_exidx(struct dl_phdr_info *info, size_t size, void *ptr) { struct find_exidx_data *data = ptr; const ElfW(Phdr) *phdr = info->dlpi_phdr; uintptr_t addr, exidx_start = 0; int i, match = 0, exidx_len = 0; for (i = info->dlpi_phnum; i > 0; i--, phdr++) { addr = info->dlpi_addr + phdr->p_vaddr; switch (phdr->p_type) { case PT_LOAD: match |= data->pc >= addr && data->pc < addr + phdr->p_memsz; break; case PT_ARM_EXIDX: exidx_start = addr; exidx_len = phdr->p_memsz; break; } } data->exidx_start = exidx_start; data->exidx_len = exidx_len; return match; } uintptr_t __gnu_Unwind_Find_exidx(uintptr_t pc, int *pcount) { struct find_exidx_data data; data.pc = pc; if (dl_iterate_phdr(find_exidx, &data) <= 0) return 0; *pcount = data.exidx_len / 8; return data.exidx_start; }