2010-03-12 03:05:20 +01:00
|
|
|
/* radare - LGPL - Copyright 2010 */
|
|
|
|
/* nibble<.ds@gmail.com> */
|
|
|
|
/* pancake<nopcode.org> */
|
|
|
|
|
|
|
|
#include <r_anal.h>
|
|
|
|
#include <r_util.h>
|
|
|
|
#include <r_list.h>
|
|
|
|
|
2010-05-21 01:46:26 +02:00
|
|
|
R_API RAnalOp *r_anal_aop_new() {
|
2010-06-16 21:44:19 +02:00
|
|
|
RAnalOp *aop = R_NEW (RAnalOp);
|
2010-05-20 17:40:58 +02:00
|
|
|
if (aop) {
|
2010-05-21 01:46:26 +02:00
|
|
|
memset (aop, 0, sizeof (RAnalOp));
|
2010-06-03 11:31:23 +02:00
|
|
|
aop->mnemonic = NULL;
|
2010-05-20 17:40:58 +02:00
|
|
|
aop->addr = -1;
|
|
|
|
aop->jump = -1;
|
|
|
|
aop->fail = -1;
|
|
|
|
}
|
|
|
|
return aop;
|
2010-03-12 03:05:20 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
R_API RList *r_anal_aop_list_new() {
|
|
|
|
RList *list = r_list_new ();
|
2010-06-14 18:52:49 +02:00
|
|
|
list->free = &r_anal_aop_free;
|
2010-03-12 03:05:20 +01:00
|
|
|
return list;
|
|
|
|
}
|
|
|
|
|
2010-06-16 21:44:19 +02:00
|
|
|
R_API void r_anal_aop_free(void *_aop) {
|
|
|
|
if (_aop) {
|
|
|
|
RAnalOp *aop = _aop;
|
|
|
|
r_anal_value_free (aop->src[0]);
|
|
|
|
r_anal_value_free (aop->src[1]);
|
|
|
|
r_anal_value_free (aop->src[2]);
|
|
|
|
r_anal_value_free (aop->dst);
|
|
|
|
free (aop->mnemonic);
|
|
|
|
free (aop);
|
|
|
|
}
|
2010-03-12 03:05:20 +01:00
|
|
|
}
|
|
|
|
|
2010-05-21 01:46:26 +02:00
|
|
|
R_API int r_anal_aop(RAnal *anal, RAnalOp *aop, ut64 addr, const ut8 *data, int len) {
|
2010-03-12 03:05:20 +01:00
|
|
|
if (anal && aop && anal->cur && anal->cur->aop)
|
2010-03-26 17:01:25 +01:00
|
|
|
return anal->cur->aop (anal, aop, addr, data, len);
|
2010-06-14 18:29:52 +02:00
|
|
|
return 0;
|
2010-03-12 03:05:20 +01:00
|
|
|
}
|
2010-06-21 11:55:48 +02:00
|
|
|
|
|
|
|
R_API char *r_anal_aop_to_string(RAnal *anal, RAnalOp *op) {
|
|
|
|
int retsz = 128;
|
|
|
|
char *cstr, *ret = malloc (128);
|
|
|
|
char *r0 = r_anal_value_to_string (op->dst);
|
|
|
|
char *a0 = r_anal_value_to_string (op->src[0]);
|
|
|
|
char *a1 = r_anal_value_to_string (op->src[1]);
|
|
|
|
|
|
|
|
switch (op->type) {
|
|
|
|
case R_ANAL_OP_TYPE_MOV:
|
|
|
|
snprintf (ret, retsz, "%s = %s", r0, a0);
|
|
|
|
break;
|
|
|
|
case R_ANAL_OP_TYPE_ADD:
|
|
|
|
if (a1 == NULL || !strcmp (a0, a1))
|
|
|
|
snprintf (ret, retsz, "%s += %s", r0, a0);
|
|
|
|
else snprintf (ret, retsz, "%s = %s + %s", r0, a0, a1);
|
|
|
|
break;
|
|
|
|
case R_ANAL_OP_TYPE_MUL:
|
|
|
|
if (a1 == NULL || !strcmp (a0, a1))
|
|
|
|
snprintf (ret, retsz, "%s *= %s", r0, a0);
|
|
|
|
else snprintf (ret, retsz, "%s = %s * %s", r0, a0, a1);
|
|
|
|
break;
|
|
|
|
case R_ANAL_OP_TYPE_DIV:
|
|
|
|
if (a1 == NULL || !strcmp (a0, a1))
|
|
|
|
snprintf (ret, retsz, "%s /= %s", r0, a0);
|
|
|
|
else snprintf (ret, retsz, "%s = %s / %s", r0, a0, a1);
|
|
|
|
break;
|
|
|
|
case R_ANAL_OP_TYPE_CJMP:
|
2010-06-23 03:33:41 +02:00
|
|
|
// XXX: op->cond is not a valid pointer
|
|
|
|
//cstr = r_anal_cond_to_string (op->cond);
|
|
|
|
cstr = (char*)"";
|
2010-06-21 11:55:48 +02:00
|
|
|
snprintf (ret, retsz, "if (%s) goto 0x%08llx", cstr, op->jump);
|
|
|
|
//free (cstr);
|
|
|
|
break;
|
|
|
|
case R_ANAL_OP_TYPE_JMP:
|
|
|
|
snprintf (ret, retsz, "goto 0x%08llx", op->jump);
|
|
|
|
break;
|
|
|
|
case R_ANAL_OP_TYPE_CALL:
|
|
|
|
// XXX: resolve flag name
|
|
|
|
snprintf (ret, retsz, "0x%08llx()", op->jump);
|
|
|
|
break;
|
|
|
|
case R_ANAL_OP_TYPE_SUB:
|
|
|
|
if (a1 == NULL || !strcmp (a0, a1))
|
|
|
|
snprintf (ret, retsz, "%s -= %s", r0, a0);
|
|
|
|
else snprintf (ret, retsz, "%s = %s - %s", r0, a0, a1);
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
sprintf (ret, "// ?");
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
free (r0);
|
|
|
|
free (a0);
|
|
|
|
free (a1);
|
|
|
|
return ret;
|
|
|
|
}
|