radare2/libr/asm/asm.c
pancake 1b1599c6df * Support for some push/pop arm opcodes with 2 regs
* More fixes for rarc2 on ARM
* Implement .fill directive in r_asm
* Added asm.profile in r_core
* Added r_str_bits as a helper to display register flags
2010-09-14 11:22:31 +02:00

418 lines
11 KiB
C

/* radare - LGPL - Copyright 2009-2010 nibble<.ds@gmail.com> */
#include <stdio.h>
#include <r_types.h>
#include <r_util.h>
#include <r_asm.h>
#include <list.h>
#include "../config.h"
static RAsmPlugin *asm_static_plugins[] =
{ R_ASM_STATIC_PLUGINS };
static int r_asm_pseudo_align(struct r_asm_aop_t *aop, char *input) {
eprintf ("TODO: .align\n"); // Must add padding for labels and others.. but this is from RAsm, not RAsmAop
return 0;
}
static int r_asm_pseudo_string(struct r_asm_aop_t *aop, char *input) {
int len = strlen (input)+1;
r_hex_bin2str ((ut8*)input, len, aop->buf_hex);
strncpy ((char*)aop->buf, input, R_ASM_BUFSIZE);
return len;
}
static inline int r_asm_pseudo_arch(RAsm *a, char *input) {
if (!r_asm_use (a, input)) {
eprintf ("Error: Unknown plugin\n");
return -1;
}
return 0;
}
static inline int r_asm_pseudo_bits(RAsm *a, char *input) {
if (!(r_asm_set_bits (a, r_num_math (NULL, input))))
eprintf ("Error: Unsupported bits value\n");
else return 0;
return -1;
}
static inline int r_asm_pseudo_org(RAsm *a, char *input) {
r_asm_set_pc (a, r_num_math (NULL, input));
return 0;
}
static inline int r_asm_pseudo_hex(struct r_asm_aop_t *aop, char *input) {
int len = r_hex_str2bin (input, aop->buf);
strncpy (aop->buf_hex, r_str_trim (input), R_ASM_BUFSIZE);
return len;
}
// TODO: add .short, .word, .dword, ...
static inline int r_asm_pseudo_byte(struct r_asm_aop_t *aop, char *input) {
int i, len = 0;
r_str_subchr (input, ',', ' ');
len = r_str_word_count (input);
r_str_word_set0 (input);
for(i=0;i<len;i++) {
char *word = r_str_word_get0 (input, i);
int num = (int)r_num_math (NULL, word);
aop->buf[i] = num;
}
r_hex_bin2str (aop->buf, len, aop->buf_hex);
return len;
}
static inline int r_asm_pseudo_fill(struct r_asm_aop_t *aop, char *input) {
int i, repeat, size, value;
sscanf (input, "%d,%d,%d", &repeat, &size, &value); // use r_num?
size *= repeat;
if (size>0) {
for (i=0; i<size; i++)
aop->buf[i] = value;
r_hex_bin2str (aop->buf, size, aop->buf_hex);
} else size = 0;
return size;
}
R_API RAsm *r_asm_new() {
int i;
RAsmPlugin *static_plugin;
RAsm *a = R_NEW (RAsm);
if (a) {
a->user = NULL;
a->cur = NULL;
a->bits = 32;
a->big_endian = 0;
a->pc = 0;
a->syntax = R_ASM_SYNTAX_INTEL;
a->plugins = r_list_new ();
for (i=0; asm_static_plugins[i]; i++) {
static_plugin = R_NEW (RAsmPlugin);
memcpy (static_plugin, asm_static_plugins[i], sizeof (RAsmPlugin));
r_asm_add (a, static_plugin);
}
}
return a;
}
R_API void r_asm_free(RAsm *a) {
// TODO: free plugins and so on
free (a);
}
/* return fastcall register argument 'idx' for a syscall with 'num' args */
R_API const char *r_asm_fastcall(RAsm *a, int idx, int num) {
struct r_asm_fastcall_t *fastcall;
const char *ret = NULL;
if (a && a->cur && a->cur->fastcall)
fastcall = *a->cur->fastcall;
if (fastcall && idx<=num)
ret = fastcall[num].arg[idx];
return ret;
}
R_API void r_asm_set_user_ptr(RAsm *a, void *user) {
a->user = user;
}
R_API int r_asm_add(RAsm *a, RAsmPlugin *foo) {
RListIter *iter;
RAsmPlugin *h;
// TODO: cache foo->name length and use memcmp instead of strcmp
if (foo->init)
foo->init (a->user);
r_list_foreach (a->plugins, iter, h)
if (!strcmp (h->name, foo->name))
return R_FALSE;
r_list_append (a->plugins, foo);
return R_TRUE;
}
R_API int r_asm_del(RAsm *a, const char *name) {
/* TODO: Implement r_asm_del */
return R_FALSE;
}
// TODO: this can be optimized using r_str_hash()
R_API int r_asm_use(RAsm *a, const char *name) {
RAsmPlugin *h;
RListIter *iter;
r_list_foreach (a->plugins, iter, h)
if (!strcmp (h->name, name)) {
a->cur = h;
return R_TRUE;
}
return R_FALSE;
}
R_API int r_asm_set_subarch(RAsm *a, const char *name) {
int ret = R_FALSE;
if (a->cur && a->cur->set_subarch)
ret = a->cur->set_subarch(a, name);
return ret;
}
static int has_bits(RAsmPlugin *h, int bits) {
int i;
if (h && h->bits)
for(i=0; h->bits[i]; i++)
if (bits == h->bits[i])
return R_TRUE;
return R_FALSE;
}
R_API int r_asm_set_bits(RAsm *a, int bits) {
if (has_bits (a->cur, bits)) {
a->bits = bits;
return R_TRUE;
}
return R_FALSE;
}
R_API int r_asm_set_big_endian(RAsm *a, int boolean) {
a->big_endian = boolean;
return R_TRUE;
}
R_API int r_asm_set_syntax(RAsm *a, int syntax) {
switch (syntax) {
case R_ASM_SYNTAX_INTEL:
case R_ASM_SYNTAX_ATT:
a->syntax = syntax;
return R_TRUE;
default:
return R_FALSE;
}
}
R_API int r_asm_set_pc(RAsm *a, ut64 pc) {
a->pc = pc;
return R_TRUE;
}
R_API int r_asm_disassemble(RAsm *a, struct r_asm_aop_t *aop, ut8 *buf, ut64 len) {
int ret = 0;
if (a->cur && a->cur->disassemble)
ret = a->cur->disassemble(a, aop, buf, len);
if (ret > 0) {
memcpy (aop->buf, buf, ret);
r_hex_bin2str (buf, ret, aop->buf_hex);
}
return ret;
}
R_API int r_asm_assemble(RAsm *a, struct r_asm_aop_t *aop, const char *buf) {
int ret = 0;
RAsmPlugin *h;
RListIter *iter;
if (a->cur) {
if (!a->cur->assemble) {
/* find callback if no assembler support in current plugin */
r_list_foreach (a->plugins, iter, h) {
if (h->arch && h->assemble && has_bits(h, a->bits)
&& !strcmp(a->cur->arch, h->arch)) {
ret = h->assemble(a, aop, buf);
break;
}
}
} else ret = a->cur->assemble (a, aop, buf);
}
if (aop && ret > 0) {
r_hex_bin2str (aop->buf, ret, aop->buf_hex);
aop->inst_len = ret;
aop->buf_hex[ret*2]=0;
strncpy (aop->buf_asm, buf, R_ASM_BUFSIZE);
}
return ret;
}
R_API RAsmCode* r_asm_mdisassemble(RAsm *a, ut8 *buf, ut64 len) {
struct r_asm_aop_t aop;
RAsmCode *acode;
int ret, slen;
ut64 idx;
if (!(acode = r_asm_code_new()))
return NULL;
if (!(acode->buf = malloc (len)))
return r_asm_code_free (acode);
memcpy (acode->buf, buf, len);
if (!(acode->buf_hex = malloc(2*len+1)))
return r_asm_code_free(acode);
r_hex_bin2str(buf, len, acode->buf_hex);
if (!(acode->buf_asm = malloc(2)))
return r_asm_code_free(acode);
for (idx = ret = slen = 0, acode->buf_asm[0] = '\0'; idx < len; idx+=ret) {
r_asm_set_pc(a, a->pc + ret);
ret = r_asm_disassemble (a, &aop, buf+idx, len-idx);
if (ret<1) {
eprintf ("error disassemble at offset %"PFMT64d"\n", idx);
return r_asm_code_free (acode);
}
slen += strlen (aop.buf_asm) + 2;
if(!(acode->buf_asm = realloc (acode->buf_asm, slen)))
return r_asm_code_free (acode);
strcat (acode->buf_asm, aop.buf_asm);
strcat (acode->buf_asm, "\n");
}
acode->len = idx;
return acode;
}
R_API RAsmCode* r_asm_mdisassemble_hexstr(RAsm *a, const char *hexstr) {
RAsmCode *ret;
ut8 *buf;
int len;
if (!(buf = malloc (strlen (hexstr))))
return NULL;
len = r_hex_str2bin (hexstr, buf);
ret = r_asm_mdisassemble (a, buf, len);
free (buf);
return ret;
}
R_API RAsmCode* r_asm_massemble(RAsm *a, const char *buf) {
char *lbuf = NULL, *ptr2, *ptr = NULL, *ptr_start = NULL,
*tokens[R_ASM_BUFSIZE], buf_token[R_ASM_BUFSIZE];
int labels = 0, stage, ret, idx, ctr, i, j;
struct r_asm_aop_t aop;
ut64 off;
RAsmCode *acode = NULL;
if (buf == NULL)
return NULL;
if (!(acode = r_asm_code_new ()))
return NULL;
if (!(acode->buf_asm = malloc (strlen (buf)+16)))
return r_asm_code_free (acode);
strcpy (acode->buf_asm, buf);
if (!(acode->buf_hex = malloc (64)))
return r_asm_code_free (acode);
acode->buf_hex[0]=0;
if (!(acode->buf = malloc (64)))
return r_asm_code_free (acode);
lbuf = strdup (buf);
if (strchr (lbuf, ':'))
labels = 1;
/* Tokenize */
for (tokens[0] = lbuf, ctr = 0;
(ptr = strchr (tokens[ctr], ';')) ||
(ptr = strchr (tokens[ctr], '\n')) ||
(ptr = strchr (tokens[ctr], '\r'));
tokens[++ctr] = ptr+1)
*ptr = '\0';
/* Stage 0-1: Parse labels*/
/* Stage 2: Assemble */
for (stage = 0; stage < 3; stage++) {
if (stage < 2 && !labels)
continue;
for (idx = ret = i = j = 0, off = a->pc, acode->buf_hex[0] = '\0';
i <= ctr; i++, idx += ret) {
strncpy (buf_token, tokens[i], R_ASM_BUFSIZE);
for (ptr_start = buf_token; *ptr_start &&
isseparator (*ptr_start); ptr_start++);
ptr = strchr (ptr_start, '#'); /* Comments */
if (ptr) *ptr = '\0';
if (stage == 2) {
r_asm_set_pc (a, a->pc + ret);
off = a->pc;
} else off +=ret;
ret = 0;
if (!*ptr_start)
continue;
//eprintf ("LINE %d %s\n", stage, ptr_start);
if (labels) /* Labels */
if ((ptr = strchr (ptr_start, ':'))) {
char food[64];
if (stage != 2) {
*ptr = 0;
snprintf (food, sizeof (food), "0x%"PFMT64x"", off);
r_asm_code_set_equ (acode, ptr_start, food);
}
ptr_start = ptr + 1;
}
if (*ptr_start == '\0') {
ret = 0;
continue;
} else if (*ptr_start == '.') { /* pseudo */
ptr = ptr_start;
if (!memcmp (ptr, ".intel_syntax", 13))
a->syntax = R_ASM_SYNTAX_INTEL;
else if (!memcmp (ptr, ".att_syntax", 10))
a->syntax = R_ASM_SYNTAX_ATT;
else if (!memcmp (ptr, ".string ", 8))
ret = r_asm_pseudo_string (&aop, ptr+8);
else if (!memcmp (ptr, ".align", 7))
ret = r_asm_pseudo_align (&aop, ptr+7);
else if (!memcmp (ptr, ".arch ", 6))
ret = r_asm_pseudo_arch (a, ptr+6);
else if (!memcmp (ptr, ".bits ", 6))
ret = r_asm_pseudo_bits (a, ptr+6);
else if (!memcmp (ptr, ".fill ", 6))
ret = r_asm_pseudo_fill (&aop, ptr+6);
else if (!memcmp (ptr, ".hex ", 5))
ret = r_asm_pseudo_hex (&aop, ptr+5);
else if (!memcmp (ptr, ".byte ", 6))
ret = r_asm_pseudo_byte (&aop, ptr+6);
else if (!memcmp (ptr, ".glob", 5)) { // .global .globl
// eprintf (".global directive not yet implemented\n");
ret = 0;
continue;
} else if (!memcmp (ptr, ".equ ", 5)) {
ptr2 = strchr (ptr+5, ',');
if (ptr2) {
*ptr2 = '\0';
r_asm_code_set_equ (acode, ptr+5, ptr2+1);
} else eprintf ("TODO: undef equ\n");
} else if (!memcmp (ptr, ".org ", 5)) {
ret = r_asm_pseudo_org (a, ptr+5);
off = a->pc;
} else {
eprintf ("Unknown keyword (%s)\n", ptr);
return r_asm_code_free (acode);
}
if (!ret)
continue;
if (ret < 0) {
eprintf ("!!! Oops\n");
return r_asm_code_free (acode);
}
} else { /* Instruction */
if (acode->equs) {
char *str = r_asm_code_equ_replace (acode, strdup (ptr_start));
ret = r_asm_assemble (a, &aop, str);
free (str);
} else ret = r_asm_assemble (a, &aop, ptr_start);
}
if (stage == 2) {
if (ret < 1) {
printf ("Cannot assemble '%s'\n", ptr_start);
return r_asm_code_free (acode);
}
acode->len = idx + ret;
if (!(acode->buf = realloc (acode->buf, (idx+ret)*2)))
return r_asm_code_free (acode);
if (!(acode->buf_hex = realloc (acode->buf_hex, (acode->len*2)+1)))
return r_asm_code_free (acode);
memcpy (acode->buf+idx, aop.buf, ret);
strcat (acode->buf_hex, aop.buf_hex);
}
}
}
return acode;
}
R_API int r_asm_modify(RAsm *a, ut8 *buf, int field, ut64 val) {
int ret = R_FALSE;
if (a->cur && a->cur->modify)
ret = a->cur->modify (a, buf, field, val);
return ret;
}