#include #include #include #include #include #include #include "host_dasm.h" extern char **g_argv; static struct disassemble_info di; #ifdef ARM #define print_insn_func print_insn_little_arm #define BFD_ARCH bfd_arch_arm #define BFD_MACH bfd_mach_arm_4T #else #define print_insn_func print_insn_i386_intel #define BFD_ARCH bfd_arch_i386 #define BFD_MACH bfd_mach_i386_i386_intel_syntax #endif /* symbols */ static asymbol **symbols; static long symcount, symstorage; static int init_done; /* Filter out (in place) symbols that are useless for disassembly. COUNT is the number of elements in SYMBOLS. Return the number of useful symbols. */ static long remove_useless_symbols (asymbol **symbols, long count) { asymbol **in_ptr = symbols, **out_ptr = symbols; while (--count >= 0) { asymbol *sym = *in_ptr++; if (sym->name == NULL || sym->name[0] == '\0' || sym->name[0] == '$') continue; if (sym->flags & (BSF_DEBUGGING | BSF_SECTION_SYM)) continue; if (bfd_is_und_section (sym->section) || bfd_is_com_section (sym->section)) continue; if (sym->value + sym->section->vma == 0) continue; /* printf("sym: %08lx %04x %08x v %08x \"%s\"\n", (unsigned int)sym->value, (unsigned int)sym->flags, (unsigned int)sym->udata.i, (unsigned int)sym->section->vma, sym->name); */ *out_ptr++ = sym; } return out_ptr - symbols; } static void slurp_symtab(const char *filename) { bfd *abfd; symcount = 0; abfd = bfd_openr(filename, NULL); if (abfd == NULL) { fprintf(stderr, "failed to open: %s\n", filename); goto no_symbols; } if (!bfd_check_format(abfd, bfd_object)) goto no_symbols; if (!(bfd_get_file_flags(abfd) & HAS_SYMS)) goto no_symbols; symstorage = bfd_get_symtab_upper_bound(abfd); if (symstorage <= 0) goto no_symbols; symbols = malloc(symstorage); if (symbols == NULL) goto no_symbols; symcount = bfd_canonicalize_symtab(abfd, symbols); if (symcount < 0) goto no_symbols; symcount = remove_useless_symbols(symbols, symcount); // bfd_close(abfd); return; no_symbols: fprintf(stderr, "no symbols in %s\n", bfd_get_filename(abfd)); if (symbols != NULL) free(symbols); symbols = NULL; if (abfd != NULL) bfd_close(abfd); } static const char *lookup_name(bfd_vma addr) { asymbol **sptr = symbols; int i; for (i = 0; i < symcount; i++) { asymbol *sym = *sptr++; if (addr == sym->value + sym->section->vma) return sym->name; } return NULL; } /* Like target_read_memory, but slightly different parameters. */ static int dis_asm_read_memory(bfd_vma memaddr, bfd_byte *myaddr, unsigned int len, struct disassemble_info *info) { memcpy(myaddr, (void *)(int)memaddr, len); return 0; } static void dis_asm_memory_error(int status, bfd_vma memaddr, struct disassemble_info *info) { fprintf(stderr, "memory_error %p\n", (void *)(int)memaddr); } static void dis_asm_print_address(bfd_vma addr, struct disassemble_info *info) { const char *name; printf("%08x", (int)addr); name = lookup_name(addr); if (name != NULL) printf(" <%s>", name); } static int insn_printf(void *f, const char *format, ...) { va_list args; size_t n; va_start(args, format); n = vprintf(format, args); va_end(args); return n; } static void host_dasm_init(void) { bfd_init(); slurp_symtab(g_argv[0]); init_disassemble_info(&di, NULL, insn_printf); di.flavour = bfd_target_unknown_flavour; di.memory_error_func = dis_asm_memory_error; di.print_address_func = dis_asm_print_address; // di.symbol_at_address_func = dis_asm_symbol_at_address; di.read_memory_func = dis_asm_read_memory; di.arch = BFD_ARCH; di.mach = BFD_MACH; di.endian = BFD_ENDIAN_LITTLE; disassemble_init_for_target(&di); init_done = 1; } void host_dasm(void *addr, int len) { bfd_vma vma_end, vma = (bfd_vma)(long)addr; const char *name; if (!init_done) host_dasm_init(); vma_end = vma + len; while (vma < vma_end) { name = lookup_name(vma); if (name != NULL) printf("%s:\n", name); printf(" %p ", (void *)(long)vma); vma += print_insn_func(vma, &di); printf("\n"); } } void host_dasm_new_symbol_(void *addr, const char *name) { bfd_vma vma = (bfd_vma)(long)addr; asymbol *sym, **tmp; if (!init_done) host_dasm_init(); if (symbols == NULL) return; if (symstorage <= symcount * sizeof(symbols[0])) { tmp = realloc(symbols, symstorage * 2); if (tmp == NULL) return; symstorage *= 2; symbols = tmp; } symbols[symcount] = calloc(sizeof(*symbols[0]), 1); if (symbols[symcount] == NULL) return; // a HACK (should use correct section), but ohwell sym = symbols[symcount]; sym->section = symbols[0]->section; sym->value = vma - sym->section->vma; sym->name = name; symcount++; }