mirror of
https://github.com/radareorg/radare2.git
synced 2025-01-17 03:13:16 +00:00
996c9d22eb
Making type matching works on each basic block on its own thus increasing code coverage. Also reducing the verbosity level, and removing many useless printed data
278 lines
9.0 KiB
C
278 lines
9.0 KiB
C
/* radare - LGPL - Copyright 2016 - oddcoder */
|
|
/* type matching - type propagation */
|
|
|
|
#include <r_anal.h>
|
|
#include <r_util.h>
|
|
#include <r_core.h>
|
|
#define LOOP_MAX 10
|
|
|
|
enum {
|
|
ROMEM = 0,
|
|
ASM_TRACE,
|
|
ANAL_TRACE,
|
|
DBG_TRACE,
|
|
NONULL,
|
|
STATES_SIZE
|
|
};
|
|
|
|
static bool r_anal_emul_init(RCore *core, RConfigHold *hc) {
|
|
r_config_save_num (hc, "esil.romem", "asm.trace", "anal.trace",
|
|
"dbg.trace", "esil.nonull", NULL);
|
|
r_config_set (core->config, "esil.romem", "true");
|
|
r_config_set (core->config, "asm.trace", "true");
|
|
r_config_set (core->config, "anal.trace", "true");
|
|
r_config_set (core->config, "dbg.trace", "true");
|
|
r_config_set (core->config, "esil.nonull", "true");
|
|
const char *bp = r_reg_get_name (core->anal->reg, R_REG_NAME_BP);
|
|
const char *sp = r_reg_get_name (core->anal->reg, R_REG_NAME_SP);
|
|
if ((bp && !r_reg_getv (core->anal->reg, bp)) && (sp && !r_reg_getv (core->anal->reg, sp))) {
|
|
eprintf ("Stack isn't initiatized.\n");
|
|
eprintf ("Try running aei and aeim commands before aftm for default stack initialization\n");
|
|
return false;
|
|
}
|
|
return (core->anal->esil != NULL);
|
|
}
|
|
|
|
static void r_anal_emul_restore(RCore *core, RConfigHold *hc) {
|
|
sdb_reset (core->anal->esil->db_trace);
|
|
r_config_restore (hc);
|
|
r_config_hold_free (hc);
|
|
}
|
|
|
|
static void type_match(RCore *core, ut64 addr, char *name) {
|
|
Sdb *trace = core->anal->esil->db_trace;
|
|
RAnal *anal = core->anal;
|
|
RAnalVar *v;
|
|
char *fcn_name;
|
|
if (r_anal_type_func_exist (anal, name)) {
|
|
fcn_name = strdup (name);
|
|
} else if (!(fcn_name = r_anal_type_func_guess (anal, name))) {
|
|
//eprintf ("can't find function prototype for %s\n", name);
|
|
return;
|
|
}
|
|
const char* cc = r_anal_type_func_cc (anal, fcn_name);
|
|
if (!cc || !r_anal_cc_exist (anal, cc)) {
|
|
//eprintf ("can't find %s calling convention %s\n", fcn_name, cc);
|
|
return;
|
|
}
|
|
int i, j, max = r_anal_type_func_args_count (anal, fcn_name);
|
|
int size = 0, idx = sdb_num_get (trace, "idx", 0);
|
|
const char *sp_name = r_reg_get_name (anal->reg, R_REG_NAME_SP);
|
|
const char *bp_name = r_reg_get_name (anal->reg, R_REG_NAME_BP);
|
|
ut64 sp = r_reg_getv (anal->reg, sp_name);
|
|
ut64 bp = r_reg_getv (anal->reg, bp_name);
|
|
r_cons_break_push (NULL, NULL);
|
|
for (i = 0; i < max; i++) {
|
|
if (r_cons_is_breaked ()) {
|
|
goto out_function;
|
|
}
|
|
char *type = r_anal_type_func_args_type (anal, fcn_name, i);
|
|
const char *name = r_anal_type_func_args_name (anal, fcn_name, i);
|
|
const char *place = r_anal_cc_arg (anal, cc, i + 1);
|
|
if (!strcmp (place, "stack")) {
|
|
// type_match_stack ();
|
|
for (j = idx; j >= 0; j--) {
|
|
if (r_cons_is_breaked ()) {
|
|
goto out_function;
|
|
}
|
|
ut64 write_addr = sdb_num_get (trace, sdb_fmt (-1, "%d.mem.write", j), 0);
|
|
if (write_addr == sp + size) {
|
|
ut64 instr_addr = sdb_num_get (trace, sdb_fmt (-1, "%d.addr", j), 0);
|
|
r_meta_set_string (core->anal, R_META_TYPE_COMMENT, instr_addr,
|
|
sdb_fmt (-1, "%s %s", type, name));
|
|
char *tmp = sdb_fmt (-1, "%d.mem.read", j);
|
|
int i2, array_size = sdb_array_size (trace, tmp);
|
|
for (i2 = 0; i2 < array_size; i2++) {
|
|
if (bp_name) {
|
|
int bp_idx = sdb_array_get_num (trace, tmp, i2, 0) - bp;
|
|
if ((v = r_anal_var_get (anal, addr, R_ANAL_VAR_KIND_BPV, 1, bp_idx))) {
|
|
r_anal_var_retype (anal, addr, 1, bp_idx, R_ANAL_VAR_KIND_BPV, type, -1, v->name);
|
|
r_anal_var_free (v);
|
|
}
|
|
}
|
|
int sp_idx = sdb_array_get_num (trace, tmp, i2, 0) - sp;
|
|
if ((v = r_anal_var_get (anal, addr, R_ANAL_VAR_KIND_SPV, 1, sp_idx))) {
|
|
r_anal_var_retype (anal, addr, 1, sp_idx, R_ANAL_VAR_KIND_SPV, type, -1, v->name);
|
|
r_anal_var_free (v);
|
|
}
|
|
}
|
|
break;
|
|
}
|
|
}
|
|
size += r_anal_type_get_size (anal, type) / 8;
|
|
} else if (!strcmp (place , "stack_rev")) {
|
|
// type_match_stack_rev ();
|
|
free (type);
|
|
int k;
|
|
for ( k = max -1; k >=i; k--) {
|
|
if (r_cons_is_breaked ()) {
|
|
goto out_function;
|
|
}
|
|
type = r_anal_type_func_args_type (anal, fcn_name, k);
|
|
name = r_anal_type_func_args_name (anal, fcn_name, k);
|
|
place = r_anal_cc_arg (anal, cc, k + 1);
|
|
if (strcmp (place ,"stack_rev")) {
|
|
break;
|
|
}
|
|
for (j = idx; j >= 0; j--) {
|
|
if (r_cons_is_breaked ()) {
|
|
goto out_function;
|
|
}
|
|
ut64 write_addr = sdb_num_get (trace, sdb_fmt (-1, "%d.mem.write", j), 0);
|
|
if (write_addr == sp + size) {
|
|
ut64 instr_addr = sdb_num_get (trace, sdb_fmt (-1, "%d.addr", j), 0);
|
|
r_meta_set_string (core->anal, R_META_TYPE_COMMENT, instr_addr,
|
|
sdb_fmt (-1, "%s %s", type, name));
|
|
char *tmp = sdb_fmt (-1, "%d.mem.read", j);
|
|
int i2, array_size = sdb_array_size (trace, tmp);
|
|
for (i2 = 0; i2 < array_size; i2++) {
|
|
if (bp_name) {
|
|
int bp_idx = sdb_array_get_num (trace, tmp, i2, 0) - bp;
|
|
if ((v = r_anal_var_get (anal, addr, R_ANAL_VAR_KIND_BPV, 1, bp_idx))) {
|
|
r_anal_var_retype (anal, addr, 1, bp_idx, R_ANAL_VAR_KIND_BPV, type, -1, v->name);
|
|
r_anal_var_free (v);
|
|
}
|
|
}
|
|
int sp_idx = sdb_array_get_num (trace, tmp, i2, 0) - sp;
|
|
if ((v =r_anal_var_get (anal, addr, R_ANAL_VAR_KIND_SPV, 1, sp_idx))) {
|
|
r_anal_var_retype (anal, addr, 1, sp_idx, R_ANAL_VAR_KIND_SPV, type, -1, v->name);
|
|
r_anal_var_free (v);
|
|
}
|
|
}
|
|
break;
|
|
}
|
|
|
|
}
|
|
size += r_anal_type_get_size (anal, type) / 8;
|
|
}
|
|
break;
|
|
} else {
|
|
// type_match_reg ();
|
|
for (j = idx; j >= 0; j--) {
|
|
if (r_cons_is_breaked ()) {
|
|
goto out_function;
|
|
}
|
|
if (sdb_array_contains (trace, sdb_fmt (-1, "%d.reg.write", j), place, 0)) {
|
|
ut64 instr_addr = sdb_num_get (trace, sdb_fmt (-1, "%d.addr", j), 0);
|
|
r_meta_set_string (core->anal, R_META_TYPE_COMMENT, instr_addr,
|
|
sdb_fmt (-1, "%s %s", type, name));
|
|
char *tmp = sdb_fmt (-1, "%d.mem.read", j);
|
|
int i2, array_size = sdb_array_size (trace, tmp);
|
|
for (i2 = 0; i2 < array_size; i2++) {
|
|
if (r_cons_is_breaked ()) {
|
|
goto out_function;
|
|
}
|
|
if (bp_name) {
|
|
int bp_idx = sdb_array_get_num (trace, tmp, i2, 0) - bp;
|
|
if ((v = r_anal_var_get (anal, addr, R_ANAL_VAR_KIND_BPV, 1, bp_idx))) {
|
|
r_anal_var_retype (anal, addr, 1, bp_idx, R_ANAL_VAR_KIND_BPV, type, -1, v->name);
|
|
r_anal_var_free (v);
|
|
}
|
|
}
|
|
int sp_idx = sdb_array_get_num (trace, tmp, i2, 0) - sp;
|
|
if ((v = r_anal_var_get (anal, addr, R_ANAL_VAR_KIND_SPV, 1, sp_idx))) {
|
|
r_anal_var_retype (anal, addr, 1, sp_idx, R_ANAL_VAR_KIND_SPV, type, -1, v->name);
|
|
r_anal_var_free (v);
|
|
}
|
|
}
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
free (type);
|
|
}
|
|
out_function:
|
|
r_cons_break_pop ();
|
|
free (fcn_name);
|
|
}
|
|
|
|
static int stack_clean (RCore *core, ut64 addr, RAnalFunction *fcn) {
|
|
int offset, ret;
|
|
char *tmp, *str, *sig;
|
|
RAnalOp *op = r_core_anal_op (core, addr);
|
|
if (!op) {
|
|
return 0;
|
|
}
|
|
str = strdup (r_strbuf_get (&op->esil));
|
|
if (!str) {
|
|
return 0;
|
|
}
|
|
tmp = strchr (str, ',');
|
|
if (!tmp) {
|
|
free (str);
|
|
return 0;
|
|
}
|
|
*tmp++ = 0;
|
|
|
|
offset = r_num_math (core->num, str);
|
|
const char *sp = r_reg_get_name (core->anal->reg, R_REG_NAME_SP);
|
|
sig = sdb_fmt (-1, "%s,+=", sp);
|
|
ret = 0;
|
|
if (!strncmp (tmp, sig, strlen (sig))) {
|
|
const char *esil = sdb_fmt (-1, "%d,%s,-=", offset, sp);
|
|
r_anal_esil_parse (core->anal->esil, esil);
|
|
r_anal_esil_stack_free (core->anal->esil);
|
|
r_core_esil_step (core, UT64_MAX, NULL);
|
|
ret = op->size;
|
|
}
|
|
r_anal_op_free (op);
|
|
free (str);
|
|
return ret;
|
|
}
|
|
|
|
R_API void r_core_anal_type_match(RCore *core, RAnalFunction *fcn) {
|
|
RConfigHold *hc = NULL;
|
|
RAnalBlock *bb;
|
|
RListIter *it;
|
|
|
|
if (!core|| !fcn) {
|
|
return;
|
|
}
|
|
hc = r_config_hold_new (core->config);
|
|
if (!hc) {
|
|
return;
|
|
}
|
|
if (!r_anal_emul_init (core, hc) || !fcn) {
|
|
r_anal_emul_restore (core, hc);
|
|
return;
|
|
}
|
|
const char *pc = r_reg_get_name (core->anal->reg, R_REG_NAME_PC);
|
|
r_list_foreach (fcn->bbs, it,bb) {
|
|
ut64 addr = bb->addr;
|
|
r_reg_setv (core->dbg->reg, pc, bb->addr);
|
|
r_debug_reg_sync (core->dbg, R_REG_TYPE_ALL, true);
|
|
r_cons_break_push (NULL, NULL);
|
|
while (!r_cons_is_breaked ()) {
|
|
RAnalOp *op = r_core_anal_op (core, addr);
|
|
int loop_count = sdb_num_get (core->anal->esil->db_trace, sdb_fmt (-1, "0x%"PFMT64x".count", addr), 0);
|
|
if (loop_count > LOOP_MAX || !op || op->type == R_ANAL_OP_TYPE_RET || addr >= bb->addr + bb->size || addr < bb->addr) {
|
|
break;
|
|
}
|
|
sdb_num_set (core->anal->esil->db_trace, sdb_fmt (-1, "0x%"PFMT64x".count", addr), loop_count + 1, 0);
|
|
if (op->type == R_ANAL_OP_TYPE_CALL) {
|
|
RAnalFunction *fcn_call = r_anal_get_fcn_in (core->anal, op->jump, -1);
|
|
if (fcn_call) {
|
|
type_match (core, addr, fcn_call->name);
|
|
}
|
|
addr += op->size;
|
|
r_anal_op_free (op);
|
|
r_reg_setv (core->dbg->reg, pc, addr);
|
|
r_debug_reg_sync (core->dbg, R_REG_TYPE_ALL, true);
|
|
r_anal_esil_set_pc (core->anal->esil, addr);
|
|
addr += stack_clean (core, addr, fcn);
|
|
r_reg_setv (core->dbg->reg, pc, addr);
|
|
r_debug_reg_sync (core->dbg, R_REG_TYPE_ALL, true);
|
|
r_anal_esil_set_pc (core->anal->esil, addr);
|
|
} else {
|
|
r_core_esil_step (core, UT64_MAX, NULL);
|
|
r_anal_op_free (op);
|
|
r_core_cmd0 (core, ".ar*");
|
|
addr = r_reg_getv (core->anal->reg, pc);
|
|
}
|
|
}
|
|
}
|
|
r_cons_break_pop ();
|
|
r_anal_emul_restore (core, hc);
|
|
|
|
}
|