1
0
Fork 0
mirror of https://github.com/tobast/libunwind-eh_elf.git synced 2025-01-25 01:30:30 +01:00

ARM extbl cleanup.

Move code that does not necessarily need to reside in the ex_tables.h header
file into ex_tables.c. Add comments and remove unused code.

Signed-off-by: Ken Werner <ken.werner@linaro.org>
This commit is contained in:
Ken Werner 2011-03-23 15:55:01 +00:00 committed by Arun Sharma
parent 6e25c11505
commit fb325c895e
2 changed files with 42 additions and 187 deletions

View file

@ -41,7 +41,6 @@ struct arm_exidx_table {
#endif #endif
}; };
typedef enum arm_exbuf_cmd { typedef enum arm_exbuf_cmd {
ARM_EXIDX_CMD_FINISH, ARM_EXIDX_CMD_FINISH,
ARM_EXIDX_CMD_DATA_PUSH, ARM_EXIDX_CMD_DATA_PUSH,
@ -55,47 +54,21 @@ typedef enum arm_exbuf_cmd {
ARM_EXIDX_CMD_REFUSED, ARM_EXIDX_CMD_REFUSED,
} arm_exbuf_cmd_t; } arm_exbuf_cmd_t;
enum arm_exbuf_cmd_flags {
ARM_EXIDX_VFP_SHIFT_16 = 1 << 16,
ARM_EXIDX_VFP_DOUBLE = 1 << 17,
};
#define ARM_EXBUF_START(x) (((x) >> 4) & 0x0f)
#define ARM_EXBUF_COUNT(x) ((x) & 0x0f)
#define ARM_EXBUF_END(x) (ARM_EXBUF_START(x) + ARM_EXBUF_COUNT(x))
struct arm_exbuf_data struct arm_exbuf_data
{ {
arm_exbuf_cmd_t cmd; arm_exbuf_cmd_t cmd;
uint32_t data; uint32_t data;
}; };
static inline void *
prel31_to_addr (void *addr)
{
uint32_t offset = ((long)*(uint32_t *)addr << 1) >> 1;
return (char *)addr + offset;
}
int arm_exidx_init_local (const char *appname); int arm_exidx_init_local (const char *appname);
int arm_exidx_table_add (const char *name, struct arm_exidx_entry *start,
int arm_exidx_table_add (const char *name, struct arm_exidx_entry *end);
struct arm_exidx_entry *start, struct arm_exidx_entry *end);
struct arm_exidx_table *arm_exidx_table_find (void *pc); struct arm_exidx_table *arm_exidx_table_find (void *pc);
struct arm_exidx_entry *arm_exidx_table_lookup ( struct arm_exidx_entry *arm_exidx_table_lookup (struct arm_exidx_table *table,
struct arm_exidx_table *table, void *pc); void *pc);
void arm_exidx_section_find (struct elf_image *ei,
Elf_W (Shdr) **exidx, Elf_W (Shdr) **extbl);
int arm_exidx_entry_lookup (struct elf_image *ei,
Elf_W (Shdr) *exidx, uint32_t ip);
int arm_exidx_entry_extract (struct elf_image *ei,
Elf_W (Shdr) *exidx, Elf_W (Shdr) *extbl,
unsigned i, uint8_t *buf);
int arm_exidx_extract (struct arm_exidx_entry *entry, uint8_t *buf); int arm_exidx_extract (struct arm_exidx_entry *entry, uint8_t *buf);
int arm_exidx_decode (const uint8_t *buf, uint8_t len, struct dwarf_cursor *c);
int arm_exidx_decode (const uint8_t *buf, uint8_t len,
struct dwarf_cursor *c);
int arm_exidx_apply_cmd (struct arm_exbuf_data *edata, struct dwarf_cursor *c); int arm_exidx_apply_cmd (struct arm_exbuf_data *edata, struct dwarf_cursor *c);
#endif // ARM_EX_TABLES_H #endif // ARM_EX_TABLES_H

View file

@ -24,11 +24,20 @@ WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE. */
#include "libunwind_i.h" #include "libunwind_i.h"
#define ARM_EXBUF_START(x) (((x) >> 4) & 0x0f)
#define ARM_EXBUF_COUNT(x) ((x) & 0x0f)
#define ARM_EXBUF_END(x) (ARM_EXBUF_START(x) + ARM_EXBUF_COUNT(x))
#define ARM_EXIDX_CANT_UNWIND 0x00000001 #define ARM_EXIDX_CANT_UNWIND 0x00000001
#define ARM_EXIDX_COMPACT 0x80000000 #define ARM_EXIDX_COMPACT 0x80000000
#define ARM_EXTBL_OP_FINISH 0xb0 #define ARM_EXTBL_OP_FINISH 0xb0
enum arm_exbuf_cmd_flags {
ARM_EXIDX_VFP_SHIFT_16 = 1 << 16,
ARM_EXIDX_VFP_DOUBLE = 1 << 17,
};
#ifdef ARM_EXIDX_TABLE_MALLOC #ifdef ARM_EXIDX_TABLE_MALLOC
static struct arm_exidx_table *arm_exidx_table_list; static struct arm_exidx_table *arm_exidx_table_list;
#else #else
@ -38,6 +47,19 @@ static unsigned arm_exidx_table_count = 0;
#endif #endif
static const char *arm_exidx_appname; static const char *arm_exidx_appname;
static inline uint32_t
prel31_read (uint32_t prel31)
{
return ((int32_t)prel31 << 1) >> 1;
}
static inline void *
prel31_to_addr (void *addr)
{
uint32_t offset = ((long)*(uint32_t *)addr << 1) >> 1;
return (char *)addr + offset;
}
static void static void
arm_exidx_table_reset_all (void) arm_exidx_table_reset_all (void)
{ {
@ -79,7 +101,7 @@ arm_exidx_table_add (const char *name,
} }
/** /**
* Located the appropriate unwind table * Locate the appropriate unwind table from the given PC.
*/ */
HIDDEN struct arm_exidx_table * HIDDEN struct arm_exidx_table *
arm_exidx_table_find (void *pc) arm_exidx_table_find (void *pc)
@ -100,7 +122,9 @@ arm_exidx_table_find (void *pc)
return NULL; return NULL;
} }
/* Based on Linux arm/arm/kernel/unwind.c:search_index() */ /**
* Finds the corresponding arm_exidx_entry from a given index table and PC.
*/
HIDDEN struct arm_exidx_entry * HIDDEN struct arm_exidx_entry *
arm_exidx_table_lookup (struct arm_exidx_table *table, void *pc) arm_exidx_table_lookup (struct arm_exidx_table *table, void *pc)
{ {
@ -190,7 +214,10 @@ arm_exidx_apply_cmd (struct arm_exbuf_data *edata, struct dwarf_cursor *c)
return ret; return ret;
} }
/**
* Decodes the given unwind instructions into arm_exbuf_data and calls
* arm_exidx_apply_cmd that applies the command onto the dwarf_cursor.
*/
HIDDEN int HIDDEN int
arm_exidx_decode (const uint8_t *buf, uint8_t len, struct dwarf_cursor *c) arm_exidx_decode (const uint8_t *buf, uint8_t len, struct dwarf_cursor *c)
{ {
@ -323,59 +350,11 @@ arm_exidx_decode (const uint8_t *buf, uint8_t len, struct dwarf_cursor *c)
return 0; return 0;
} }
static inline uint32_t
prel31_read (uint32_t prel31)
{
return ((int32_t)prel31 << 1) >> 1;
}
/** /**
* Finds the index of the table entry for the given address * Reads the given entry and extracts the unwind instructions into buf.
* using a binary search. * Returns the number of the extracted unwind insns or -UNW_ESTOPUNWIND
* @returns The table index (or -1 if not found). * if the special bit pattern ARM_EXIDX_CANT_UNWIND (0x1) was found.
*/ */
HIDDEN int
arm_exidx_lookup (uint32_t *exidx_data, uint32_t exidx_size, uint32_t ip)
{
unsigned lo = 0, hi = exidx_size / 8;
while (lo < hi)
{
unsigned mid = (lo + hi) / 2;
uint32_t base = (uint32_t)(exidx_data + mid * 2);
uint32_t addr = base + prel31_read (exidx_data[mid * 2]);
if (ip < addr)
hi = mid;
else
lo = mid + 1;
}
return hi - 1;
}
HIDDEN int
arm_exidx_entry_lookup (struct elf_image *ei,
Elf_W (Shdr) *exidx, uint32_t ip)
{
#if 1
return arm_exidx_lookup (ei->image + exidx->sh_offset, exidx->sh_size, ip);
#else
unsigned n_entries = exidx->sh_size / 8;
uint32_t *exidx_data = ei->image + exidx->sh_offset;
unsigned lo = 0, hi = n_entries;
while (lo < hi)
{
unsigned mid = (lo + hi) / 2;
uint32_t base = exidx->sh_addr + mid * 8;
uint32_t addr = base + prel31_read (exidx_data[mid * 2]);
if (ip < addr)
hi = mid;
else
lo = mid + 1;
}
return hi - 1;
#endif
}
HIDDEN int HIDDEN int
arm_exidx_extract (struct arm_exidx_entry *entry, uint8_t *buf) arm_exidx_extract (struct arm_exidx_entry *entry, uint8_t *buf)
{ {
@ -417,7 +396,7 @@ arm_exidx_extract (struct arm_exidx_entry *entry, uint8_t *buf)
else else
{ {
void *pers = prel31_to_addr (extbl_data); void *pers = prel31_to_addr (extbl_data);
Debug (2, "%p Personality routine: %8.8x\n", addr, pers); Debug (2, "%p Personality routine: %8p\n", addr, pers);
n_table_words = extbl_data[1] >> 24; n_table_words = extbl_data[1] >> 24;
buf[nbuf++] = extbl_data[1] >> 16; buf[nbuf++] = extbl_data[1] >> 16;
buf[nbuf++] = extbl_data[1] >> 8; buf[nbuf++] = extbl_data[1] >> 8;
@ -442,105 +421,8 @@ arm_exidx_extract (struct arm_exidx_entry *entry, uint8_t *buf)
return nbuf; return nbuf;
} }
HIDDEN int
arm_exidx_entry_extract (struct elf_image *ei,
Elf_W (Shdr) *exidx, Elf_W (Shdr) *extbl,
unsigned i, uint8_t *buf)
{
int nbuf = 0;
uint32_t *exidx_data = ei->image + exidx->sh_offset;
uint32_t base = exidx->sh_addr + i * 8;
uint32_t one = base + prel31_read (exidx_data[i * 2]);
uint32_t two = exidx_data[i * 2 + 1];
if (two == ARM_EXIDX_CANT_UNWIND)
printf ("0x1 [can't unwind]\n");
else if (two & ARM_EXIDX_COMPACT)
{
printf ("compact model %d [%8.8x]\n", (two >> 24) & 0x7f, two);
buf[nbuf++] = two >> 16;
buf[nbuf++] = two >> 8;
buf[nbuf++] = two;
}
else
{
assert (NULL != extbl);
two = base + prel31_read (two) + 4;
uint32_t *extbl_data = ei->image + extbl->sh_offset
+ (two - extbl->sh_addr);
two = extbl_data[0];
unsigned int n_table_words = 0;
if (two & ARM_EXIDX_COMPACT)
{
int pers = (two >> 24) & 0x0f;
printf ("compact model %d [%8.8x]\n", pers, two);
if (pers == 1 || pers == 2)
{
n_table_words = (two >> 16) & 0xff;
extbl_data += 1;
}
else
buf[nbuf++] = two >> 16;
buf[nbuf++] = two >> 8;
buf[nbuf++] = two;
}
else
{
uint32_t pers = prel31_read (extbl_data[0]);
pers += extbl->sh_addr + i * 8 + 4;
printf ("Personality routine: %8.8x\n", pers);
n_table_words = extbl_data[1] >> 24;
buf[nbuf++] = extbl_data[1] >> 16;
buf[nbuf++] = extbl_data[1] >> 8;
buf[nbuf++] = extbl_data[1];
extbl_data += 2;
}
assert (n_table_words <= 5);
unsigned j;
for (j = 0; j < n_table_words; j++)
{
two = *extbl_data++;
buf[nbuf++] = two >> 24;
buf[nbuf++] = two >> 16;
buf[nbuf++] = two >> 8;
buf[nbuf++] = two >> 0;
}
}
if (nbuf > 0 && buf[nbuf - 1] != ARM_EXTBL_OP_FINISH)
buf[nbuf++] = ARM_EXTBL_OP_FINISH;
return nbuf;
}
HIDDEN void
arm_exidx_section_find (struct elf_image *ei,
Elf_W (Shdr) **exidx, Elf_W (Shdr) **extbl)
{
Elf_W (Ehdr) *ehdr = ei->image;
Elf_W (Shdr) *shdr;
shdr = (Elf_W (Shdr) *)((char *)ei->image + ehdr->e_shoff);
char *names = (char *)ei->image + shdr[ehdr->e_shstrndx].sh_offset;
int i = 0;
for (i = 0; i < ehdr->e_shnum; i++)
{
char *name = names + shdr[i].sh_name;
switch (shdr[i].sh_type)
{
case SHT_ARM_EXIDX:
*exidx = shdr + i;
break;
case SHT_PROGBITS:
if (strcmp (name, ".ARM.extab") != 0)
break;
*extbl = shdr + i;
break;
}
}
}
/** /**
* Callback to dl_iterate_phr to find each library's unwind table. * Callback to dl_iterate_phdr to find the unwind tables.
* If found, calls arm_exidx_table_add to remember it for later lookups. * If found, calls arm_exidx_table_add to remember it for later lookups.
*/ */
static int static int
@ -570,13 +452,13 @@ arm_exidx_init_local_cb (struct dl_phdr_info *info, size_t size, void *data)
} }
/** /**
* Must be called for local process lookups. * Traverse the program headers of the executable and its loaded
* shared objects to collect the unwind tables.
*/ */
HIDDEN int HIDDEN int
arm_exidx_init_local (const char *appname) arm_exidx_init_local (const char *appname)
{ {
arm_exidx_appname = appname; arm_exidx_appname = appname;
// traverse all libraries and find unwind tables
arm_exidx_table_reset_all(); arm_exidx_table_reset_all();
return dl_iterate_phdr (&arm_exidx_init_local_cb, NULL); return dl_iterate_phdr (&arm_exidx_init_local_cb, NULL);
} }