2016-02-03 12:29:32 +01:00
|
|
|
/* radare - LGPL - Copyright 2013-2016 - pancake */
|
2013-01-13 03:19:32 +01:00
|
|
|
|
2013-01-22 05:06:12 +01:00
|
|
|
#include <r_anal.h>
|
|
|
|
|
2014-03-11 02:47:10 +01:00
|
|
|
#define DB a->sdb_hints
|
2017-05-09 14:25:57 +02:00
|
|
|
#define setf(x,...) snprintf(x,sizeof(x)-1,##__VA_ARGS__)
|
2014-03-11 02:47:10 +01:00
|
|
|
|
2016-11-01 19:47:34 +01:00
|
|
|
R_API void r_anal_hint_clear(RAnal *a) {
|
2014-03-11 02:47:10 +01:00
|
|
|
sdb_reset (a->sdb_hints);
|
2013-01-22 05:06:12 +01:00
|
|
|
}
|
|
|
|
|
2016-11-01 19:47:34 +01:00
|
|
|
R_API void r_anal_hint_del(RAnal *a, ut64 addr, int size) {
|
2014-03-11 02:47:10 +01:00
|
|
|
char key[128];
|
2016-09-20 11:34:08 +02:00
|
|
|
if (size > 1) {
|
2014-03-11 02:47:10 +01:00
|
|
|
eprintf ("TODO: r_anal_hint_del: in range\n");
|
|
|
|
} else {
|
|
|
|
setf (key, "hint.0x%08"PFMT64x, addr);
|
|
|
|
sdb_unset (a->sdb_hints, key, 0);
|
2017-01-08 12:57:09 +01:00
|
|
|
a->bits_hints_changed = true;
|
2014-03-11 02:47:10 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-11-01 19:47:34 +01:00
|
|
|
static void unsetHint(RAnal *a, const char *type, ut64 addr) {
|
2015-09-28 01:00:06 +02:00
|
|
|
int idx;
|
|
|
|
char key[128];
|
2016-11-01 19:47:34 +01:00
|
|
|
setf (key, "hint.0x%08"PFMT64x, addr);
|
2015-09-28 01:00:06 +02:00
|
|
|
idx = sdb_array_indexof (DB, key, type, 0);
|
|
|
|
if (idx != -1) {
|
|
|
|
sdb_array_delete (DB, key, idx, 0);
|
|
|
|
sdb_array_delete (DB, key, idx, 0);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-11-01 19:47:34 +01:00
|
|
|
static void setHint(RAnal *a, const char *type, ut64 addr, const char *s, ut64 ptr) {
|
2014-03-11 02:47:10 +01:00
|
|
|
int idx;
|
|
|
|
char key[128], val[128], *nval = NULL;
|
2016-11-01 19:47:34 +01:00
|
|
|
setf (key, "hint.0x%08"PFMT64x, addr);
|
2014-03-26 04:16:03 +01:00
|
|
|
idx = sdb_array_indexof (DB, key, type, 0);
|
2016-11-01 19:47:34 +01:00
|
|
|
if (s) {
|
|
|
|
nval = sdb_encode ((const ut8*)s, -1);
|
|
|
|
} else {
|
|
|
|
nval = sdb_itoa (ptr, val, 16);
|
|
|
|
}
|
2014-03-11 02:47:10 +01:00
|
|
|
if (idx != -1) {
|
2016-11-01 19:47:34 +01:00
|
|
|
if (!s) {
|
|
|
|
nval = sdb_itoa (ptr, val, 16);
|
|
|
|
}
|
2016-09-20 11:34:08 +02:00
|
|
|
sdb_array_set (DB, key, idx + 1, nval, 0);
|
2014-03-11 02:47:10 +01:00
|
|
|
} else {
|
|
|
|
sdb_array_push (DB, key, nval, 0);
|
2014-03-29 02:49:49 +01:00
|
|
|
sdb_array_push (DB, key, type, 0);
|
2014-03-11 02:47:10 +01:00
|
|
|
}
|
2016-09-20 11:34:08 +02:00
|
|
|
if (s) {
|
|
|
|
free (nval);
|
|
|
|
}
|
2013-01-22 05:06:12 +01:00
|
|
|
}
|
|
|
|
|
2017-10-31 23:22:20 +08:00
|
|
|
R_API void r_anal_hint_set_offset(RAnal *a, ut64 addr, const char* typeoff) {
|
2018-01-08 03:29:29 +01:00
|
|
|
setHint (a, "Offset:", addr, r_str_trim_ro (typeoff), 0);
|
2017-10-31 23:22:20 +08:00
|
|
|
}
|
2017-11-09 12:57:51 +01:00
|
|
|
|
2016-11-01 19:47:34 +01:00
|
|
|
R_API void r_anal_hint_set_jump(RAnal *a, ut64 addr, ut64 ptr) {
|
2014-03-11 02:47:10 +01:00
|
|
|
setHint (a, "jump:", addr, NULL, ptr);
|
2013-10-25 02:06:00 +02:00
|
|
|
}
|
2017-11-09 12:57:51 +01:00
|
|
|
|
2018-02-27 19:13:12 +01:00
|
|
|
R_API void r_anal_hint_set_newbits(RAnal *a, ut64 addr, int bits) {
|
|
|
|
a->bits_hints_changed = true;
|
|
|
|
setHint (a, "Bits:", addr, NULL, bits);
|
|
|
|
}
|
|
|
|
|
|
|
|
// TOOD: add helpers for newendian and newbank
|
|
|
|
|
2013-10-25 02:06:00 +02:00
|
|
|
R_API void r_anal_hint_set_fail(RAnal *a, ut64 addr, ut64 ptr) {
|
2014-03-11 02:47:10 +01:00
|
|
|
setHint (a, "fail:", addr, NULL, ptr);
|
2013-10-25 02:06:00 +02:00
|
|
|
}
|
2017-11-09 12:57:51 +01:00
|
|
|
|
2017-11-14 23:36:04 +09:00
|
|
|
R_API void r_anal_hint_set_high(RAnal *a, ut64 addr) {
|
|
|
|
setHint (a, "high:", addr, NULL, 1);
|
|
|
|
}
|
|
|
|
|
2016-11-01 19:47:34 +01:00
|
|
|
R_API void r_anal_hint_set_immbase(RAnal *a, ut64 addr, int base) {
|
2015-11-23 23:04:49 +01:00
|
|
|
if (base) {
|
2015-09-28 01:00:06 +02:00
|
|
|
setHint (a, "immbase:", addr, NULL, (ut64)base);
|
2015-11-23 23:04:49 +01:00
|
|
|
} else {
|
|
|
|
unsetHint (a, "immbase:", addr);
|
2015-09-28 01:00:06 +02:00
|
|
|
}
|
|
|
|
}
|
2017-11-09 12:57:51 +01:00
|
|
|
|
2016-11-01 19:47:34 +01:00
|
|
|
R_API void r_anal_hint_set_pointer(RAnal *a, ut64 addr, ut64 ptr) {
|
2014-03-11 02:47:10 +01:00
|
|
|
setHint (a, "ptr:", addr, NULL, ptr);
|
2013-01-22 05:06:12 +01:00
|
|
|
}
|
2017-11-09 12:57:51 +01:00
|
|
|
|
2016-11-01 19:47:34 +01:00
|
|
|
R_API void r_anal_hint_set_arch(RAnal *a, ut64 addr, const char *arch) {
|
2018-01-08 03:29:29 +01:00
|
|
|
setHint (a, "arch:", addr, r_str_trim_ro (arch), 0);
|
2013-01-22 05:06:12 +01:00
|
|
|
}
|
2017-11-09 12:57:51 +01:00
|
|
|
|
2016-11-01 19:47:34 +01:00
|
|
|
R_API void r_anal_hint_set_syntax(RAnal *a, ut64 addr, const char *syn) {
|
2015-05-10 00:12:33 +02:00
|
|
|
setHint (a, "Syntax:", addr, syn, 0);
|
|
|
|
}
|
2017-11-09 12:57:51 +01:00
|
|
|
|
2016-11-01 19:47:34 +01:00
|
|
|
R_API void r_anal_hint_set_opcode(RAnal *a, ut64 addr, const char *opcode) {
|
2018-01-08 03:29:29 +01:00
|
|
|
setHint (a, "opcode:", addr, r_str_trim_ro (opcode), 0);
|
2013-01-22 05:06:12 +01:00
|
|
|
}
|
2017-11-09 12:57:51 +01:00
|
|
|
|
2016-11-01 19:47:34 +01:00
|
|
|
R_API void r_anal_hint_set_esil(RAnal *a, ut64 addr, const char *esil) {
|
2018-01-08 03:29:29 +01:00
|
|
|
setHint (a, "esil:", addr, r_str_trim_ro (esil), 0);
|
2013-01-22 05:06:12 +01:00
|
|
|
}
|
2017-11-09 12:57:51 +01:00
|
|
|
|
2016-11-01 19:47:34 +01:00
|
|
|
R_API void r_anal_hint_set_bits(RAnal *a, ut64 addr, int bits) {
|
2017-01-08 12:57:09 +01:00
|
|
|
a->bits_hints_changed = true;
|
2014-03-11 02:47:10 +01:00
|
|
|
setHint (a, "bits:", addr, NULL, bits);
|
2013-01-22 05:06:12 +01:00
|
|
|
}
|
2017-11-09 12:57:51 +01:00
|
|
|
|
2016-11-01 19:47:34 +01:00
|
|
|
R_API void r_anal_hint_set_size(RAnal *a, ut64 addr, int size) {
|
2014-03-11 02:47:10 +01:00
|
|
|
setHint (a, "size:", addr, NULL, size);
|
2013-01-22 05:06:12 +01:00
|
|
|
}
|
2017-11-09 12:57:51 +01:00
|
|
|
|
2016-11-01 19:47:34 +01:00
|
|
|
R_API void r_anal_hint_unset_size(RAnal *a, ut64 addr) {
|
|
|
|
unsetHint(a, "size:", addr);
|
|
|
|
}
|
2017-11-09 12:57:51 +01:00
|
|
|
|
2016-11-01 19:47:34 +01:00
|
|
|
R_API void r_anal_hint_unset_bits(RAnal *a, ut64 addr) {
|
2017-01-08 12:57:09 +01:00
|
|
|
a->bits_hints_changed = true;
|
2016-11-01 19:47:34 +01:00
|
|
|
unsetHint(a, "bits:", addr);
|
|
|
|
}
|
2017-11-09 12:57:51 +01:00
|
|
|
|
2016-11-01 19:47:34 +01:00
|
|
|
R_API void r_anal_hint_unset_esil(RAnal *a, ut64 addr) {
|
|
|
|
unsetHint(a, "esil:", addr);
|
|
|
|
}
|
2017-11-09 12:57:51 +01:00
|
|
|
|
2016-11-01 19:47:34 +01:00
|
|
|
R_API void r_anal_hint_unset_opcode(RAnal *a, ut64 addr) {
|
|
|
|
unsetHint(a, "opcode:", addr);
|
|
|
|
}
|
2017-11-09 12:57:51 +01:00
|
|
|
|
2017-11-14 23:36:04 +09:00
|
|
|
R_API void r_anal_hint_unset_high(RAnal *a, ut64 addr) {
|
|
|
|
unsetHint(a, "high:", addr);
|
|
|
|
}
|
|
|
|
|
2016-11-01 19:47:34 +01:00
|
|
|
R_API void r_anal_hint_unset_arch(RAnal *a, ut64 addr) {
|
|
|
|
unsetHint(a, "arch:", addr);
|
|
|
|
}
|
2017-11-09 12:57:51 +01:00
|
|
|
|
2016-11-01 19:47:34 +01:00
|
|
|
R_API void r_anal_hint_unset_syntax(RAnal *a, ut64 addr) {
|
|
|
|
unsetHint(a, "Syntax:", addr);
|
|
|
|
}
|
2017-11-09 12:57:51 +01:00
|
|
|
|
2016-11-01 19:47:34 +01:00
|
|
|
R_API void r_anal_hint_unset_pointer(RAnal *a, ut64 addr) {
|
|
|
|
unsetHint(a, "ptr:", addr);
|
|
|
|
}
|
2017-11-09 12:57:51 +01:00
|
|
|
|
2017-10-31 23:22:20 +08:00
|
|
|
R_API void r_anal_hint_unset_offset(RAnal *a, ut64 addr) {
|
|
|
|
unsetHint (a, "Offset:", addr);
|
|
|
|
}
|
2017-11-09 12:57:51 +01:00
|
|
|
|
2016-11-01 19:47:34 +01:00
|
|
|
R_API void r_anal_hint_unset_jump(RAnal *a, ut64 addr) {
|
|
|
|
unsetHint (a, "jump:", addr);
|
|
|
|
}
|
2017-11-09 12:57:51 +01:00
|
|
|
|
2016-11-01 19:47:34 +01:00
|
|
|
R_API void r_anal_hint_unset_fail(RAnal *a, ut64 addr) {
|
|
|
|
unsetHint (a, "fail:", addr);
|
|
|
|
}
|
2013-01-22 05:06:12 +01:00
|
|
|
|
2016-11-01 19:47:34 +01:00
|
|
|
R_API void r_anal_hint_free(RAnalHint *h) {
|
2016-09-20 11:34:08 +02:00
|
|
|
if (h) {
|
|
|
|
free (h->arch);
|
|
|
|
free (h->esil);
|
|
|
|
free (h->opcode);
|
|
|
|
free (h->syntax);
|
2017-10-31 23:22:20 +08:00
|
|
|
free (h->offset);
|
2016-09-20 11:34:08 +02:00
|
|
|
free (h);
|
|
|
|
}
|
2013-01-22 05:06:12 +01:00
|
|
|
}
|
|
|
|
|
2014-03-11 10:53:44 +01:00
|
|
|
R_API RAnalHint *r_anal_hint_from_string(RAnal *a, ut64 addr, const char *str) {
|
2016-12-12 13:22:03 +01:00
|
|
|
char *r, *nxt, *nxt2;
|
2014-03-11 02:47:10 +01:00
|
|
|
int token = 0;
|
2014-03-11 10:53:44 +01:00
|
|
|
RAnalHint *hint = R_NEW0 (RAnalHint);
|
2016-09-20 11:34:08 +02:00
|
|
|
if (!hint) {
|
2015-06-19 21:30:57 +02:00
|
|
|
return NULL;
|
2016-09-20 11:34:08 +02:00
|
|
|
}
|
2016-12-12 13:22:03 +01:00
|
|
|
hint->jump = UT64_MAX;
|
|
|
|
hint->fail = UT64_MAX;
|
2017-11-14 23:36:04 +09:00
|
|
|
char *s = strdup (str);
|
2015-06-19 21:30:57 +02:00
|
|
|
if (!s) {
|
2016-09-20 11:34:08 +02:00
|
|
|
free (hint);
|
2015-06-17 12:36:08 +02:00
|
|
|
return NULL;
|
|
|
|
}
|
2014-03-11 02:47:10 +01:00
|
|
|
hint->addr = addr;
|
2016-12-12 13:22:03 +01:00
|
|
|
token = *s;
|
|
|
|
for (r = s; ; r = nxt2) {
|
2014-03-26 04:16:03 +01:00
|
|
|
r = sdb_anext (r, &nxt);
|
2016-12-13 13:11:42 +01:00
|
|
|
if (!nxt) {
|
|
|
|
break;
|
|
|
|
}
|
2016-12-12 14:47:42 +01:00
|
|
|
sdb_anext (nxt, &nxt2); // tokenize value
|
2014-03-11 02:47:10 +01:00
|
|
|
if (token) {
|
|
|
|
switch (token) {
|
2016-12-12 13:22:03 +01:00
|
|
|
case 'i': hint->immbase = sdb_atoi (nxt); break;
|
|
|
|
case 'j': hint->jump = sdb_atoi (nxt); break;
|
|
|
|
case 'f': hint->fail = sdb_atoi (nxt); break;
|
|
|
|
case 'p': hint->ptr = sdb_atoi (nxt); break;
|
|
|
|
case 'b': hint->bits = sdb_atoi (nxt); break;
|
2018-02-27 19:13:12 +01:00
|
|
|
case 'B': hint->new_bits = sdb_atoi (nxt); break;
|
2016-12-12 13:22:03 +01:00
|
|
|
case 's': hint->size = sdb_atoi (nxt); break;
|
|
|
|
case 'S': hint->syntax = (char*)sdb_decode (nxt, 0); break;
|
|
|
|
case 'o': hint->opcode = (char*)sdb_decode (nxt, 0); break;
|
2017-10-31 23:22:20 +08:00
|
|
|
case 'O': hint->offset = (char*)sdb_decode (nxt, 0); break;
|
2016-12-12 13:22:03 +01:00
|
|
|
case 'e': hint->esil = (char*)sdb_decode (nxt, 0); break;
|
|
|
|
case 'a': hint->arch = (char*)sdb_decode (nxt, 0); break;
|
2017-11-14 23:36:04 +09:00
|
|
|
case 'h': hint->high = sdb_atoi (nxt); break;
|
2014-03-11 02:47:10 +01:00
|
|
|
}
|
2016-09-20 11:34:08 +02:00
|
|
|
}
|
2016-12-12 13:22:03 +01:00
|
|
|
if (!nxt || !nxt2) {
|
2014-03-11 02:47:10 +01:00
|
|
|
break;
|
2016-09-20 11:34:08 +02:00
|
|
|
}
|
2016-12-12 13:22:03 +01:00
|
|
|
token = *nxt2;
|
2013-01-22 05:06:12 +01:00
|
|
|
}
|
2014-04-30 02:11:51 +04:00
|
|
|
free (s);
|
2014-03-11 02:47:10 +01:00
|
|
|
return hint;
|
2013-01-22 05:06:12 +01:00
|
|
|
}
|
2014-03-11 10:53:44 +01:00
|
|
|
|
|
|
|
R_API RAnalHint *r_anal_hint_get(RAnal *a, ut64 addr) {
|
2016-12-12 13:22:03 +01:00
|
|
|
char key[64];
|
2016-11-01 19:47:34 +01:00
|
|
|
setf (key, "hint.0x%08"PFMT64x, addr);
|
2016-12-12 13:22:03 +01:00
|
|
|
const char *s = sdb_const_get (DB, key, 0);
|
2014-04-30 02:11:51 +04:00
|
|
|
if (!s) {
|
|
|
|
return NULL;
|
|
|
|
}
|
2016-12-12 13:22:03 +01:00
|
|
|
return r_anal_hint_from_string (a, addr, s);
|
2014-03-11 10:53:44 +01:00
|
|
|
}
|