mirror of
https://github.com/radareorg/radare2.git
synced 2025-01-07 05:41:43 +00:00
118 lines
2.9 KiB
C
118 lines
2.9 KiB
C
/* radare - LGPL - Copyright 2012-2016 - pancake */
|
|
|
|
#include <stdio.h>
|
|
#include <stdarg.h>
|
|
#include <string.h>
|
|
#include <r_types.h>
|
|
#include <r_lib.h>
|
|
#include <r_util.h>
|
|
#include <r_asm.h>
|
|
#include "dis-asm.h"
|
|
#include <mybfd.h>
|
|
|
|
/* extern */
|
|
int decodeInstr (bfd_vma address, disassemble_info * info);
|
|
int ARCTangent_decodeInstr (bfd_vma address, disassemble_info * info);
|
|
int ARCompact_decodeInstr (bfd_vma address, disassemble_info * info);
|
|
|
|
/* ugly globals */
|
|
static ut32 Offset = 0;
|
|
static char *buf_global = NULL;
|
|
static int buf_len = 0;
|
|
static ut8 bytes[32] = {0};
|
|
|
|
static int arc_buffer_read_memory (bfd_vma memaddr, bfd_byte *myaddr, unsigned int length, struct disassemble_info *info) {
|
|
int delta = (memaddr - Offset);
|
|
if (delta < 0) {
|
|
return -1; // disable backward reads
|
|
}
|
|
if ((delta + length) > sizeof (bytes)) {
|
|
return -1;
|
|
}
|
|
memcpy (myaddr, bytes + delta, R_MIN (buf_len - delta, length));
|
|
return 0;
|
|
}
|
|
|
|
static int symbol_at_address(bfd_vma addr, struct disassemble_info * info) {
|
|
return 0;
|
|
}
|
|
|
|
static void memory_error_func(int status, bfd_vma memaddr, struct disassemble_info *info) {
|
|
//--
|
|
}
|
|
|
|
static void print_address(bfd_vma address, struct disassemble_info *info) {
|
|
char tmp[64];
|
|
if (buf_global) {
|
|
sprintf (tmp, "0x%08"PFMT64x"", (ut64)address);
|
|
strcat (buf_global, tmp);
|
|
}
|
|
}
|
|
|
|
static int buf_fprintf(void *stream, const char *format, ...) {
|
|
va_list ap;
|
|
char tmp[1024];
|
|
if (!buf_global) {
|
|
return 0;
|
|
}
|
|
va_start (ap, format);
|
|
vsnprintf (tmp, sizeof (tmp), format, ap);
|
|
strcat (buf_global, tmp);
|
|
va_end (ap);
|
|
return 0;
|
|
}
|
|
|
|
static int disassemble(RAsm *a, RAsmOp *op, const ut8 *buf, int len) {
|
|
static struct disassemble_info disasm_obj;
|
|
if (len < 2) {
|
|
return -1;
|
|
}
|
|
buf_global = op->buf_asm;
|
|
Offset = a->pc;
|
|
if (len > sizeof (bytes)) {
|
|
len = sizeof (bytes);
|
|
}
|
|
memcpy (bytes, buf, len); // TODO handle compact
|
|
buf_len = len;
|
|
/* prepare disassembler */
|
|
memset (&disasm_obj,'\0', sizeof (struct disassemble_info));
|
|
disasm_obj.buffer = bytes;
|
|
disasm_obj.buffer_length = len;
|
|
disasm_obj.read_memory_func = &arc_buffer_read_memory;
|
|
disasm_obj.symbol_at_address_func = &symbol_at_address;
|
|
disasm_obj.memory_error_func = &memory_error_func;
|
|
disasm_obj.print_address_func = &print_address;
|
|
disasm_obj.endian = !a->big_endian;
|
|
disasm_obj.fprintf_func = &buf_fprintf;
|
|
disasm_obj.stream = stdout;
|
|
disasm_obj.mach = 0;
|
|
op->buf_asm[0] = '\0';
|
|
if (a->bits == 16) {
|
|
op->size = ARCompact_decodeInstr ((bfd_vma)Offset, &disasm_obj);
|
|
} else {
|
|
op->size = ARCTangent_decodeInstr ((bfd_vma)Offset, &disasm_obj);
|
|
}
|
|
if (op->size == -1) {
|
|
strncpy (op->buf_asm, " (data)", R_ASM_BUFSIZE);
|
|
}
|
|
return op->size;
|
|
}
|
|
|
|
RAsmPlugin r_asm_plugin_arc = {
|
|
.name = "arc",
|
|
.arch = "arc",
|
|
.bits = 16 | 32,
|
|
.endian = R_SYS_ENDIAN_LITTLE | R_SYS_ENDIAN_BIG,
|
|
.desc = "Argonaut RISC Core",
|
|
.disassemble = &disassemble,
|
|
.license = "GPL3"
|
|
};
|
|
|
|
#ifndef CORELIB
|
|
RLibStruct radare_plugin = {
|
|
.type = R_LIB_TYPE_ASM,
|
|
.data = &r_asm_plugin_arc,
|
|
.version = R2_VERSION
|
|
};
|
|
#endif
|