2 * (C) GraÅžvydas "notaz" Ignotas, 2009-2010
4 * This work is licensed under the terms of any of these licenses
6 * - GNU GPL, version 2 or later.
7 * - GNU LGPL, version 2.1 or later.
9 * See the COPYING file in the top-level directory.
19 #include "host_dasm.h"
23 static struct disassemble_info di;
24 static disassembler_ftype print_insn_func;
27 //#define print_insn_func print_insn_little_arm
28 #define BFD_ARCH bfd_arch_arm
29 #define BFD_MACH bfd_mach_arm_unknown
30 #define DASM_OPTS "reg-names-std"
31 #elif defined __aarch64__
32 //#define print_insn_func print_insn_aarch64
33 #define BFD_ARCH bfd_arch_aarch64
34 #define BFD_MACH bfd_mach_aarch64
35 #define DASM_OPTS NULL
36 #elif defined __mips__
37 //#define print_insn_func print_insn_little_mips
38 #define BFD_ARCH bfd_arch_mips
39 #define BFD_MACH bfd_mach_mipsisa64r2
40 #define DASM_OPTS NULL
42 //#define print_insn_func print_insn_riscv
43 #define BFD_ARCH bfd_arch_riscv
44 #define BFD_MACH bfd_mach_riscv64
45 #define DASM_OPTS NULL
46 #elif defined __powerpc__
47 //#define print_insn_func print_insn_little_powerpc
48 #define BFD_ARCH bfd_arch_powerpc
49 #define BFD_MACH bfd_mach_ppc64
50 #define DASM_OPTS NULL
51 #elif defined(__x86_64__) || defined(__i386__)
52 //#define print_insn_func print_insn_i386_intel
53 #define BFD_ARCH bfd_arch_i386
55 #define BFD_MACH bfd_mach_x86_64_intel_syntax
57 #define BFD_MACH bfd_mach_i386_i386_intel_syntax
59 #define DASM_OPTS NULL
61 #error "missing arch support"
65 static asymbol **symbols;
66 static long symcount, symstorage;
69 /* Filter out (in place) symbols that are useless for disassembly.
70 COUNT is the number of elements in SYMBOLS.
71 Return the number of useful symbols. */
73 remove_useless_symbols (asymbol **symbols, long count)
75 asymbol **in_ptr = symbols, **out_ptr = symbols;
79 asymbol *sym = *in_ptr++;
81 if (sym->name == NULL || sym->name[0] == '\0' || sym->name[0] == '$')
83 if (sym->flags & (BSF_DEBUGGING | BSF_SECTION_SYM))
85 if (bfd_is_und_section (sym->section)
86 || bfd_is_com_section (sym->section))
88 if (sym->value + sym->section->vma == 0)
91 printf("sym: %08lx %04x %08x v %08x \"%s\"\n",
92 (unsigned int)sym->value, (unsigned int)sym->flags, (unsigned int)sym->udata.i,
93 (unsigned int)sym->section->vma, sym->name);
98 return out_ptr - symbols;
101 static void slurp_symtab(const char *filename)
107 abfd = bfd_openr(filename, NULL);
109 fprintf(stderr, "failed to open: %s\n", filename);
113 if (!bfd_check_format(abfd, bfd_object))
116 if (!(bfd_get_file_flags(abfd) & HAS_SYMS))
119 symstorage = bfd_get_symtab_upper_bound(abfd);
123 symbols = malloc(symstorage);
127 symcount = bfd_canonicalize_symtab(abfd, symbols);
131 symcount = remove_useless_symbols(symbols, symcount);
136 fprintf(stderr, "no symbols in %s\n", bfd_get_filename(abfd));
144 static const char *lookup_name(bfd_vma addr)
146 asymbol **sptr = symbols;
149 for (i = 0; i < symcount; i++) {
150 asymbol *sym = *sptr++;
152 if (addr == sym->value + sym->section->vma)
159 /* Like target_read_memory, but slightly different parameters. */
161 dis_asm_read_memory(bfd_vma memaddr, bfd_byte *myaddr, unsigned int len,
162 struct disassemble_info *info)
164 memcpy(myaddr, (void *)memaddr, len);
169 dis_asm_memory_error(int status, bfd_vma memaddr,
170 struct disassemble_info *info)
172 fprintf(stderr, "memory_error %p\n", (void *)(long)memaddr);
176 dis_asm_print_address(bfd_vma addr, struct disassemble_info *info)
180 printf("%08x", (int)addr);
182 name = lookup_name(addr);
184 printf(" <%s>", name);
187 static int insn_printf(void *f, const char *format, ...)
192 va_start(args, format);
193 n = vprintf(format, args);
199 static int print_insn_hex(bfd_vma addr, struct disassemble_info *info)
203 dis_asm_read_memory(addr, (bfd_byte *)&op, 4, info);
204 printf("%p %08lx",(void *)addr, (long)op);
208 #ifdef BFD_COMPRESS_ZSTD
209 static int insn_printf_styled(void *f, enum disassembler_style style, const char* format, ...)
214 va_start(args, format);
215 n = vprintf(format, args);
222 static void host_dasm_init(void)
225 if (g_argv && g_argv[0])
226 slurp_symtab(g_argv[0]);
228 #ifdef BFD_COMPRESS_ZSTD
229 init_disassemble_info(&di, NULL, insn_printf, insn_printf_styled);
231 init_disassemble_info(&di, NULL, insn_printf);
233 di.flavour = bfd_target_unknown_flavour;
234 di.memory_error_func = dis_asm_memory_error;
235 di.print_address_func = dis_asm_print_address;
236 // di.symbol_at_address_func = dis_asm_symbol_at_address;
237 di.read_memory_func = dis_asm_read_memory;
240 di.endian = BFD_ENDIAN_LITTLE;
241 di.disassembler_options = DASM_OPTS;
242 disassemble_init_for_target(&di);
243 #ifndef print_insn_func
244 print_insn_func = disassembler(BFD_ARCH, 0, BFD_MACH, NULL);
245 if (!print_insn_func) print_insn_func = print_insn_hex;
250 void host_dasm(void *addr, int len)
252 bfd_vma vma_end, vma = (bfd_vma)addr;
259 while (vma < vma_end) {
260 name = lookup_name(vma);
262 printf("%s:\n", name);
264 printf(" %08lx ", (long)vma);
265 vma += print_insn_func(vma, &di);
270 void host_dasm_new_symbol_(void *addr, const char *name)
272 bfd_vma vma = (bfd_vma)(long)addr;
279 if (symstorage <= symcount * sizeof(symbols[0])) {
280 tmp = realloc(symbols, symstorage * 2);
287 symbols[symcount] = calloc(sizeof(*symbols[0]), 1);
288 if (symbols[symcount] == NULL)
291 // a HACK (should use correct section), but ohwell
292 sym = symbols[symcount];
293 sym->section = symbols[0]->section;
294 sym->value = vma - sym->section->vma;