radare2/libr/anal/esil_dfg.c

2140 lines
72 KiB
C

/* radare - LGPL - Copyright 2019-2024 - condret */
#include <r_anal.h>
#define R_ANAL_ESIL_DFG_TAG_LI_MASK (R_ANAL_ESIL_DFG_TAG_VAR | R_ANAL_ESIL_DFG_TAG_CONST \
| R_ANAL_ESIL_DFG_TAG_GENERATIVE | R_ANAL_ESIL_DFG_TAG_RESULT | R_ANAL_ESIL_DFG_TAG_PTR)
typedef enum {
VAR_TYPE_REG = 0,
VAR_TYPE_MEM,
} EsilDFGVarType;
typedef struct esil_dfg_var_t {
ut64 from;
ut64 to;
RGraphNode *node;
EsilDFGVarType type;
} EsilDFGVar;
typedef struct r_anal_esil_dfg_filter_t {
RAnalEsilDFG *dfg;
RRBTree *tree;
Sdb *results;
} RAnalEsilDFGFilter;
typedef struct r_anal_esil_dfg_const_reducer_t {
RAnalEsilDFGFilter filter;
RRBTree *const_result_gnodes;
} RAnalEsilDFGConstReducer;
// TODO: simple const propagation - use node->type of srcs to propagate consts of pushed vars
R_API RAnalEsilDFGNode *r_anal_esil_dfg_node_new(RAnalEsilDFG *edf, R_NULLABLE const char *c) {
R_RETURN_VAL_IF_FAIL (edf, NULL);
RAnalEsilDFGNode *ret = R_NEW0 (RAnalEsilDFGNode);
ret->content = r_strbuf_new (c);
ret->idx = edf->idx++;
return ret;
}
static void _dfg_node_free(RAnalEsilDFGNode *free_me) {
if (free_me) {
r_strbuf_free (free_me->content);
free (free_me);
}
}
static int _rv_del_alloc_cmp(void *incoming, void *in, void *user) {
EsilDFGVar *rv_incoming = (EsilDFGVar *)incoming;
EsilDFGVar *rv_in = (EsilDFGVar *)in;
RAnalEsilDFG *dfg = (RAnalEsilDFG *)user;
if (dfg->malloc_failed) {
return -1;
}
if (rv_incoming->type < rv_in->type) {
return -1;
}
if (rv_incoming->type > rv_in->type) {
return 1;
}
// first handle the simple cases without intersection
if (rv_incoming->to < rv_in->from) {
return -1;
}
if (rv_in->to < rv_incoming->from) {
return 1;
}
if (rv_in->from == rv_incoming->from && rv_in->to == rv_incoming->to) {
return 0;
}
/*
the following cases are about intersection, here some ascii-art, so you understand what I do
=incoming=
=========in=========
split in into 2 and reinsert the second half (in2)
shrink first half (in1)
=incoming=
=in1= =in2=
*/
if (rv_in->from < rv_incoming->from && rv_incoming->to < rv_in->to) {
EsilDFGVar *rv = R_NEW (EsilDFGVar);
if (!rv) {
dfg->malloc_failed = true;
return -1;
}
rv[0] = rv_in[0];
rv_in->to = rv_incoming->from - 1;
rv->from = rv_incoming->to + 1;
dfg->insert = rv;
return 1;
}
/*
=incoming=
=in=
enqueue the non-intersecting ends in the todo-queue
*/
if (rv_incoming->from < rv_in->from && rv_in->to < rv_incoming->to) {
// lower part
EsilDFGVar *rv = R_NEW (EsilDFGVar);
if (!rv) {
dfg->malloc_failed = true;
return -1;
}
rv[0] = rv_incoming[0];
rv->to = rv_in->from - 1;
r_queue_enqueue (dfg->todo, rv);
// upper part
rv = R_NEW (EsilDFGVar);
if (!rv) {
dfg->malloc_failed = true;
return -1;
}
rv[0] = rv_incoming[0];
rv->from = rv_in->to + 1;
r_queue_enqueue (dfg->todo, rv);
return 0;
}
/*
=incoming=
=in=
similar to the previous case, but this time only enqueue 1 half
*/
if (rv_incoming->from == rv_in->from && rv_in->to < rv_incoming->to) {
EsilDFGVar *rv = R_NEW (EsilDFGVar);
if (!rv) {
dfg->malloc_failed = true;
return -1;
}
rv[0] = rv_incoming[0];
rv->from = rv_in->to + 1;
r_queue_enqueue (dfg->todo, rv);
return 0;
}
/*
=incoming=
=in=
*/
if (rv_incoming->from < rv_in->from && rv_in->to == rv_incoming->to) {
EsilDFGVar *rv = R_NEW (EsilDFGVar);
if (!rv) {
dfg->malloc_failed = true;
return -1;
}
rv[0] = rv_incoming[0];
rv->to = rv_in->from - 1;
r_queue_enqueue (dfg->todo, rv);
return 0;
}
/*
=incoming=
===in===
shrink in
=incoming=
=in=
*/
if (rv_in->to <= rv_incoming->to) {
rv_in->to = rv_incoming->from - 1;
return 1;
}
/*
=incoming=
===in===
up-shrink in
=incoming=
==in==
*/
rv_in->from = rv_incoming->to + 1;
return -1;
}
static int _rv_ins_cmp(void *incoming, void *in, void *user) {
EsilDFGVar *rv_incoming = (EsilDFGVar *)incoming;
EsilDFGVar *rv_in = (EsilDFGVar *)in;
if (rv_incoming->type < rv_in->type) {
return -1;
}
if (rv_incoming->type > rv_in->type) {
return 1;
}
return rv_incoming->from - rv_in->from;
}
static bool _edf_reg_set(RAnalEsilDFG *dfg, const char *reg, RGraphNode *node) {
R_RETURN_VAL_IF_FAIL (dfg && !dfg->malloc_failed && reg, false);
char *_reg = r_str_newf ("reg.%s", reg);
if (!sdb_num_exists (dfg->regs, _reg)) {
//no assert to prevent memleaks
free (_reg);
return false;
}
EsilDFGVar *rv = R_NEW0 (EsilDFGVar);
if (!rv) {
free (_reg);
return false;
}
const ut64 v = sdb_num_get (dfg->regs, _reg, NULL);
free (_reg);
rv->from = (v & (UT64_MAX ^ UT32_MAX)) >> 32;
rv->to = v & UT32_MAX;
r_queue_enqueue (dfg->todo, rv);
while (!r_queue_is_empty (dfg->todo) && !dfg->malloc_failed) {
// rbtree api does sadly not allow deleting multiple items at once :(
rv = r_queue_dequeue (dfg->todo);
r_crbtree_delete (dfg->vars, rv, _rv_del_alloc_cmp, dfg);
if (dfg->insert && !dfg->malloc_failed) {
r_crbtree_insert (dfg->vars, dfg->insert, _rv_ins_cmp, NULL);
dfg->insert = NULL;
}
free (rv);
}
if (dfg->malloc_failed) {
while (!r_queue_is_empty (dfg->todo)) {
free (r_queue_dequeue (dfg->todo));
}
return false;
}
rv = R_NEW0 (EsilDFGVar);
rv->from = (v & (UT64_MAX ^ UT32_MAX)) >> 32;
rv->to = v & UT32_MAX;
rv->node = node;
r_crbtree_insert (dfg->vars, rv, _rv_ins_cmp, NULL);
return true;
}
static bool _edf_mem_set(RAnalEsilDFG *dfg, ut64 addr, ut32 size, RGraphNode *node) {
R_RETURN_VAL_IF_FAIL (dfg && !dfg->malloc_failed && size, false);
EsilDFGVar *mv = R_NEW0 (EsilDFGVar);
if (!mv) {
return false;
}
mv->from = addr;
mv->to = addr + size - 1;
mv->type = VAR_TYPE_MEM;
r_queue_enqueue (dfg->todo, mv);
while (!r_queue_is_empty (dfg->todo) && !dfg->malloc_failed) {
// rbtree api does sadly not allow deleting multiple items at once :(
mv = r_queue_dequeue (dfg->todo);
r_crbtree_delete (dfg->vars, mv, _rv_del_alloc_cmp, dfg);
if (dfg->insert && !dfg->malloc_failed) {
r_crbtree_insert (dfg->vars, dfg->insert, _rv_ins_cmp, NULL);
dfg->insert = NULL;
}
free (mv);
}
if (dfg->malloc_failed) {
while (!r_queue_is_empty (dfg->todo)) {
free (r_queue_dequeue (dfg->todo));
}
return false;
}
mv = R_NEW0 (EsilDFGVar);
mv->from = addr;
mv->to = addr + size - 1;
mv->type = VAR_TYPE_MEM;
mv->node = node;
r_crbtree_insert (dfg->vars, mv, _rv_ins_cmp, NULL);
return true;
}
static int _rv_find_cmp(void *incoming, void *in, void *user) {
EsilDFGVar *rv_incoming = (EsilDFGVar *)incoming;
EsilDFGVar *rv_in = (EsilDFGVar *)in;
RAnalEsilDFG *dfg = (RAnalEsilDFG *)user;
if (dfg->malloc_failed) {
return -1;
}
if (rv_incoming->type < rv_in->type) {
return -1;
}
if (rv_incoming->type > rv_in->type) {
return 1;
}
// first handle the simple cases without intersection
if (rv_incoming->to < rv_in->from) {
return -1;
}
if (rv_in->to < rv_incoming->from) {
return 1;
}
/*
=incoming=
=========in=========
*/
if (rv_in->from <= rv_incoming->from && rv_incoming->to <= rv_in->to) {
return 0;
}
/*
=incoming=
=in=
enqueue the non-intersecting ends in the todo-queue
*/
if (rv_incoming->from < rv_in->from && rv_in->to < rv_incoming->to) {
// lower part
EsilDFGVar *rv = R_NEW (EsilDFGVar);
if (!rv) {
dfg->malloc_failed = true;
return -1;
}
rv[0] = rv_incoming[0];
rv->to = rv_in->from - 1;
r_queue_enqueue (dfg->todo, rv);
// upper part
rv = R_NEW (EsilDFGVar);
if (!rv) {
dfg->malloc_failed = true;
return -1;
}
rv[0] = rv_incoming[0];
rv->from = rv_in->to + 1;
r_queue_enqueue (dfg->todo, rv);
return 0;
}
/*
=incoming=
=in=
similar to the previous case, but this time only enqueue 1 half
*/
if (rv_in->from <= rv_incoming->from && rv_in->to < rv_incoming->to) {
EsilDFGVar *rv = R_NEW (EsilDFGVar);
if (!rv) {
dfg->malloc_failed = true;
return -1;
}
rv[0] = rv_incoming[0];
rv->from = rv_in->to + 1;
r_queue_enqueue (dfg->todo, rv);
return 0;
}
/*
=incoming=
=in=
*/
EsilDFGVar *rv = R_NEW (EsilDFGVar);
if (!rv) {
dfg->malloc_failed = true;
return -1;
}
rv[0] = rv_incoming[0];
rv->to = rv_in->from - 1;
r_queue_enqueue (dfg->todo, rv);
return 0;
}
static RGraphNode *_edf_origin_reg_get(RAnalEsilDFG *dfg, const char *reg) {
R_RETURN_VAL_IF_FAIL (dfg && reg, NULL);
char *_reg = r_str_newf ("reg.%s", reg);
if (!sdb_num_exists (dfg->regs, _reg)) {
free (_reg);
return NULL;
}
free (_reg);
char *origin_reg = r_str_newf ("ori.%s", reg);
RGraphNode *origin_reg_node = sdb_ptr_get (dfg->regs, origin_reg, 0);
if (origin_reg_node) {
free (origin_reg);
return origin_reg_node;
}
RGraphNode *reg_node = r_graph_add_node (dfg->flow, r_anal_esil_dfg_node_new (dfg, reg));
RAnalEsilDFGNode *_origin_reg_node = r_anal_esil_dfg_node_new (dfg, reg);
r_strbuf_appendf (_origin_reg_node->content, ":var_%d", dfg->idx++);
_origin_reg_node->type = R_ANAL_ESIL_DFG_TAG_VAR | R_ANAL_ESIL_DFG_TAG_REG;
origin_reg_node = r_graph_add_node (dfg->flow, _origin_reg_node);
r_graph_add_edge (dfg->flow, reg_node, origin_reg_node);
sdb_ptr_set (dfg->regs, origin_reg, origin_reg_node, 0);
free (origin_reg);
return origin_reg_node;
}
static RGraphNode *_edf_reg_get(RAnalEsilDFG *dfg, const char *reg) {
R_RETURN_VAL_IF_FAIL (dfg && reg, NULL);
char *_reg = r_str_newf ("reg.%s", reg);
if (!sdb_num_exists (dfg->regs, _reg)) {
free (_reg);
return NULL;
}
EsilDFGVar *rv = R_NEW0 (EsilDFGVar);
if (!rv) {
free (_reg);
return NULL;
}
const ut64 v = sdb_num_get (dfg->regs, _reg, NULL);
free (_reg);
rv->from = (v & (UT64_MAX ^ UT32_MAX)) >> 32;
rv->to = v & UT32_MAX;
RQueue *parts = r_queue_new (8);
if (!parts) {
free (rv);
return NULL;
}
r_queue_enqueue (dfg->todo, rv);
// log2((search_rv.to + 1) - search_rv.from) maybe better?
// wat du if this fails?
RGraphNode *reg_node = NULL;
while (!r_queue_is_empty (dfg->todo)) {
rv = r_queue_dequeue (dfg->todo);
EsilDFGVar *part_rv = r_crbtree_find (dfg->vars, rv, _rv_find_cmp, dfg);
if (part_rv) {
r_queue_enqueue (parts, part_rv->node);
} else if (!reg_node) {
reg_node = _edf_origin_reg_get (dfg, reg);
//insert in the gap
part_rv = R_NEW (EsilDFGVar);
if (!part_rv) {
R_FREE (rv);
dfg->malloc_failed = true;
break;
}
part_rv[0] = rv[0];
part_rv->node = reg_node;
r_crbtree_insert (dfg->vars, part_rv, _rv_ins_cmp, NULL);
//enqueue for later merge
r_queue_enqueue (parts, reg_node);
} else {
//initial regnode was already created
//only need to insert in the tree
part_rv = R_NEW (EsilDFGVar);
if (!part_rv) {
R_FREE (rv);
dfg->malloc_failed = true;
break;
}
part_rv[0] = rv[0];
part_rv->node = reg_node;
r_crbtree_insert (dfg->vars, part_rv, _rv_ins_cmp, NULL);
}
free (rv);
}
reg_node = NULL; // is this needed?
if (dfg->malloc_failed) {
while (!r_queue_is_empty (dfg->todo)) {
free (r_queue_dequeue (dfg->todo));
}
goto beach; // Outside loop!
}
switch (parts->size) {
case 0:
break;
case 1:
reg_node = r_queue_dequeue (parts);
break;
default: {
RAnalEsilDFGNode *_reg_node = r_anal_esil_dfg_node_new (dfg, "merge to ");
if (!_reg_node) {
while (!r_queue_is_empty (dfg->todo)) {
free (r_queue_dequeue (dfg->todo));
}
dfg->malloc_failed = true;
goto beach;
}
r_strbuf_appendf (_reg_node->content, "%s:var_%d", reg, dfg->idx++);
reg_node = r_graph_add_node (dfg->flow, _reg_node);
if (!reg_node) {
_dfg_node_free (_reg_node);
while (!r_queue_is_empty (dfg->todo)) {
free (r_queue_dequeue (dfg->todo));
}
dfg->malloc_failed = true;
goto beach;
}
_reg_node->type = R_ANAL_ESIL_DFG_TAG_MERGE | R_ANAL_ESIL_DFG_TAG_REG;
}
do {
r_graph_add_edge (dfg->flow, r_queue_dequeue (parts), reg_node);
} while (!r_queue_is_empty (parts));
break;
}
beach:
r_queue_free (parts);
return reg_node;
}
static RGraphNode *_edf_uninitialized_mem_get(RAnalEsilDFG *dfg, ut64 addr, ut32 size) {
R_RETURN_VAL_IF_FAIL (dfg && size, NULL);
char *content = r_str_newf ("[%d]@0x%"PFMT64x, size, addr);
RGraphNode *orig_mem_gnode = r_graph_add_node (dfg->flow, r_anal_esil_dfg_node_new (dfg, content));
free (content);
content = r_str_newf ("[%d]@<0x%"PFMT64x">:uninitialized_mem_var_%d", size, addr, dfg->idx + 1);
RAnalEsilDFGNode *mem_node = r_anal_esil_dfg_node_new (dfg, content);
free (content);
dfg->idx++;
mem_node->type = R_ANAL_ESIL_DFG_TAG_VAR | R_ANAL_ESIL_DFG_TAG_MEM;
if (dfg->use_map_info) {
RIOMap *map = dfg->iob.map_get_at (dfg->iob.io, addr);
if (map && !(map->perm & R_PERM_W)) {
mem_node->type |= R_ANAL_ESIL_DFG_TAG_CONST;
}
}
RGraphNode *mem_gnode = r_graph_add_node (dfg->flow, mem_node);
r_graph_add_edge (dfg->flow, orig_mem_gnode, mem_gnode);
return mem_gnode;
}
static RGraphNode *_edf_mem_get(RAnalEsilDFG *dfg, ut64 addr, ut32 size) {
R_RETURN_VAL_IF_FAIL (dfg && size, NULL);
EsilDFGVar *mv = R_NEW0 (EsilDFGVar);
if (!mv) {
return NULL;
}
mv->from = addr;
mv->to = addr + size - 1;
mv->type = VAR_TYPE_MEM;
RQueue *parts = r_queue_new (size);
if (!parts) {
free (mv);
return NULL;
}
r_queue_enqueue (dfg->todo, mv);
// log2((search_rv.to + 1) - search_rv.from) maybe better?
// wat du if this fails?
RGraphNode *mem_node = NULL;
while (!r_queue_is_empty (dfg->todo)) {
mv = r_queue_dequeue (dfg->todo);
EsilDFGVar *part_mv = r_crbtree_find (dfg->vars, mv, _rv_find_cmp, dfg);
if (part_mv) {
r_queue_enqueue (parts, part_mv->node);
} else if (!mem_node) {
mem_node = _edf_uninitialized_mem_get (dfg, mv->from, (ut32)(mv->to - mv->from + 1));
if (!mem_node) {
dfg->malloc_failed = true;
break;
}
//insert in the gap
part_mv = R_NEW (EsilDFGVar);
if (!part_mv) {
R_FREE (mv);
dfg->malloc_failed = true;
break;
}
part_mv[0] = mv[0];
part_mv->node = mem_node;
r_crbtree_insert (dfg->vars, part_mv, _rv_ins_cmp, NULL);
//enqueue for later merge
r_queue_enqueue (parts, mem_node);
} else {
//initial regnode was already created
//only need to insert in the tree
part_mv = R_NEW (EsilDFGVar);
if (!part_mv) {
R_FREE (mv);
dfg->malloc_failed = true;
break;
}
part_mv[0] = mv[0];
part_mv->node = mem_node;
r_crbtree_insert (dfg->vars, part_mv, _rv_ins_cmp, NULL);
}
free (mv);
}
mem_node = NULL; // is this needed?
if (dfg->malloc_failed) {
while (!r_queue_is_empty (dfg->todo)) {
free (r_queue_dequeue (dfg->todo));
}
goto beach; // Outside loop!
}
switch (parts->size) {
case 0:
break;
case 1:
mem_node = r_queue_dequeue (parts);
break;
default: {
RAnalEsilDFGNode *_mem_node = r_anal_esil_dfg_node_new (dfg, "merge to ");
if (!_mem_node) {
while (!r_queue_is_empty (dfg->todo)) {
free (r_queue_dequeue (dfg->todo));
}
dfg->malloc_failed = true;
goto beach;
}
r_strbuf_appendf (_mem_node->content, "<0x%"PFMT64x">:mem_var_%d", addr, dfg->idx++);
mem_node = r_graph_add_node (dfg->flow, _mem_node);
if (!mem_node) {
_dfg_node_free (_mem_node);
while (!r_queue_is_empty (dfg->todo)) {
free (r_queue_dequeue (dfg->todo));
}
dfg->malloc_failed = true;
goto beach;
}
_mem_node->type = R_ANAL_ESIL_DFG_TAG_MERGE | R_ANAL_ESIL_DFG_TAG_MEM;
}
do {
r_graph_add_edge (dfg->flow, r_queue_dequeue (parts), mem_node);
} while (!r_queue_is_empty (parts));
break;
}
beach:
r_queue_free (parts);
return mem_node;
}
static RGraphNode *_edf_const_get(RAnalEsilDFG *dfg, char *const_value) {
RGraphNode *orig_value_gnode = r_graph_add_node (dfg->flow, r_anal_esil_dfg_node_new (dfg, const_value));
RAnalEsilDFGNode *value_node = r_anal_esil_dfg_node_new (dfg, const_value);
value_node->type = R_ANAL_ESIL_DFG_TAG_CONST;
r_strbuf_appendf (value_node->content, ":const_%d", dfg->idx++);
RGraphNode *ret = r_graph_add_node (dfg->flow, value_node);
r_graph_add_edge (dfg->flow, orig_value_gnode, ret);
return ret;
}
static bool _edf_var_set(RAnalEsilDFG *dfg, const char *var, RGraphNode *node) {
R_RETURN_VAL_IF_FAIL (dfg && var, false);
char *_var = r_str_newf ("var.%s", var);
const bool ret = !sdb_ptr_set (dfg->regs, _var, node, 0);
free (_var);
return ret;
}
static RGraphNode *_edf_var_get(RAnalEsilDFG *dfg, const char *var) {
R_RETURN_VAL_IF_FAIL (dfg && var, NULL);
char *k = r_str_newf ("var.%s", var);
RGraphNode *ret = sdb_ptr_get (dfg->regs, k, NULL);
free (k);
return ret;
}
static bool edf_consume_2_set_reg(REsil *esil);
static bool edf_consume_2_push_1(REsil *esil);
static bool edf_consume_1_push_1(REsil *esil);
typedef void (*AddConstraintStringUseNewCB) (RStrBuf *result, const char *new_node_str);
static bool edf_use_new_push_1(REsil *esil, const char *op_string, AddConstraintStringUseNewCB cb);
typedef void (*AddConstraintStringConsume1UseOldNewCB) (RStrBuf *result, const char *consume_str, const char *old_node_str, const char *new_node_str);
static bool edf_consume_1_use_old_new_push_1(REsil *esil, const char *op_string, AddConstraintStringConsume1UseOldNewCB cb);
static bool edf_eq_weak(REsil *esil) {
RAnalEsilDFG *edf = (RAnalEsilDFG *)esil->user;
RGraphNode *o_old = edf->old; //node for esil->old
RGraphNode *o_new = edf->cur; //node for esil->cur
if (!edf_consume_2_set_reg (esil)) {
return false;
}
//work-around
edf->old = o_old ? o_old : NULL;
edf->cur = o_new ? o_new : NULL;
return true;
}
static void edf_zf_constraint(RStrBuf *result, const char *new_node_str) {
r_strbuf_appendf (result, ":(%s==0)", new_node_str);
}
static bool edf_zf(REsil *esil) {
return edf_use_new_push_1 (esil, "$z", edf_zf_constraint);
}
static void edf_sf_constraint(RStrBuf *result, const char *new_node_str) {
r_strbuf_appendf (result, ":(%s<0)", new_node_str);
}
static bool edf_sf(REsil *esil) {
char *bitsize = r_esil_pop (esil);
R_LOG_DEBUG ("bitsize not yet implemented for sf (%s)", bitsize);
return edf_use_new_push_1 (esil, "$s", edf_sf_constraint);
}
static void edf_pf_constraint(RStrBuf *result, const char *new_node_str) {
r_strbuf_appendf (result, ":parity_of(%s)", new_node_str);
}
static bool edf_pf(REsil *esil) {
return edf_use_new_push_1 (esil, "$p", edf_pf_constraint);
}
static void edf_cf_constraint(RStrBuf *result, const char *consume, const char *o, const char *n) {
r_strbuf_appendf (result, ":((%s&mask(%s&0x3f))<(%s&mask(%s&0x3f)))",
n, consume, o, consume);
}
static bool edf_cf(REsil *esil) {
return edf_consume_1_use_old_new_push_1 (esil, "$c", edf_cf_constraint);
}
static void edf_bf_constraint(RStrBuf *result, const char *consume, const char *o, const char *n) {
r_strbuf_appendf (result, ":((%s&mask((%s+0x3f)&0x3f))<(%s& mask((%s+0x3f)&0x3f)))",
o, consume, n, consume);
}
static bool edf_bf(REsil *esil) {
return edf_consume_1_use_old_new_push_1 (esil, "$b", edf_bf_constraint);
}
static bool _edf_consume_2_set_reg(REsil *esil, const bool use_origin) {
const char *op_string = esil->current_opstr;
RAnalEsilDFG *edf = (RAnalEsilDFG *)esil->user;
char *dst = r_esil_pop (esil);
char *src = r_esil_pop (esil);
if (!src || !dst) {
free (dst);
free (src);
return false;
}
int dst_type = r_esil_get_parm_type (esil, dst);
if (dst_type == R_ESIL_PARM_INVALID) {
free (dst);
free (src);
return false;
}
const int src_type = r_esil_get_parm_type (esil, src);
RGraphNode *src_node = NULL;
if (src_type == R_ESIL_PARM_REG) {
src_node = _edf_reg_get (edf, src);
} else if (src_type == R_ESIL_PARM_NUM) {
src_node = _edf_const_get (edf, src);
} else {
src_node = _edf_var_get (edf, src);
}
RGraphNode *dst_node = use_origin ? _edf_origin_reg_get (edf, dst) : _edf_reg_get (edf, dst);
RGraphNode *old_dst_node = dst_node;
if (!src_node || !dst_node) {
free (src);
free (dst);
return false;
}
RAnalEsilDFGNode *eop_node = r_anal_esil_dfg_node_new (edf, src);
r_strbuf_appendf (eop_node->content, ",%s,%s", dst, op_string);
eop_node->type = R_ANAL_ESIL_DFG_TAG_GENERATIVE;
free (src);
RGraphNode *op_node = r_graph_add_node (edf->flow, eop_node);
r_graph_add_edge (edf->flow, dst_node, op_node);
r_graph_add_edge (edf->flow, src_node, op_node);
edf->old = old_dst_node;
RAnalEsilDFGNode *result = r_anal_esil_dfg_node_new (edf, dst);
result->type = R_ANAL_ESIL_DFG_TAG_RESULT | R_ANAL_ESIL_DFG_TAG_VAR | R_ANAL_ESIL_DFG_TAG_REG;
if (use_origin) {
if (((RAnalEsilDFGNode *)(src_node->data))->type & R_ANAL_ESIL_DFG_TAG_CONST) {
result->type |= R_ANAL_ESIL_DFG_TAG_CONST;
}
} else {
if ((((RAnalEsilDFGNode *)(src_node->data))->type & R_ANAL_ESIL_DFG_TAG_CONST) &&
(((RAnalEsilDFGNode *)(dst_node->data))->type & R_ANAL_ESIL_DFG_TAG_CONST)) {
result->type |= R_ANAL_ESIL_DFG_TAG_CONST;
}
}
r_strbuf_appendf (result->content, ":var_%d", edf->idx++);
dst_node = r_graph_add_node (edf->flow, result);
r_graph_add_edge (edf->flow, op_node, dst_node);
_edf_reg_set (edf, dst, dst_node);
edf->cur = dst_node;
free (dst);
return true;
}
static bool edf_consume_2_use_set_reg(REsil *esil) {
return _edf_consume_2_set_reg (esil, false);
}
static bool edf_consume_2_set_reg(REsil *esil) {
return _edf_consume_2_set_reg (esil, true);
}
// TODO: not properly implemented
static bool edf_pop(REsil *esil) {
const char *op_string = esil->current_opstr;
RAnalEsilDFG *edf = (RAnalEsilDFG *)esil->user;
char *src = r_esil_pop (esil);
if (!src) {
return false;
}
const int src_type = r_esil_get_parm_type (esil, src);
RGraphNode *src_node = NULL;
if (src_type == R_ESIL_PARM_REG) {
src_node = _edf_reg_get (edf, src);
} else if (src_type == R_ESIL_PARM_NUM) {
src_node = _edf_const_get (edf, src);
} else {
src_node = _edf_var_get (edf, src);
}
if (!src_node) {
free (src);
return false;
}
RAnalEsilDFGNode *eop_node = r_anal_esil_dfg_node_new (edf, src);
r_strbuf_appendf (eop_node->content, ",%s", op_string);
eop_node->type = R_ANAL_ESIL_DFG_TAG_GENERATIVE;
free (src);
RGraphNode *op_node = r_graph_add_node (edf->flow, eop_node);
r_graph_add_edge (edf->flow, src_node, op_node);
return true;
}
#if 1
// TODO: kill DUP
static bool edf_dup(REsil *esil) {
char *src = r_esil_pop (esil);
if (!src) {
return false;
}
const int src_type = r_esil_get_parm_type (esil, src);
if (src_type == R_ESIL_PARM_REG || src_type == R_ESIL_PARM_NUM) {
// this is a optimization to reduce needless DUPs
r_esil_push (esil, src);
return r_esil_push (esil, strdup (src));
}
const char *op_string = esil->current_opstr;
RAnalEsilDFG *edf = (RAnalEsilDFG *)esil->user;
RGraphNode *src_node = _edf_var_get (edf, src);
RAnalEsilDFGNode *eop_node = r_anal_esil_dfg_node_new (edf, src);
r_strbuf_appendf (eop_node->content, ",%s", op_string);
eop_node->type = R_ANAL_ESIL_DFG_TAG_GENERATIVE;
RGraphNode *op_node = r_graph_add_node (edf->flow, eop_node);
r_graph_add_edge (edf->flow, src_node, op_node);
const bool const_result = !!(((RAnalEsilDFGNode *)src_node->data)->type & R_ANAL_ESIL_DFG_TAG_CONST);
RAnalEsilDFGNode *result = r_anal_esil_dfg_node_new (edf, "result_");
result->type = R_ANAL_ESIL_DFG_TAG_RESULT | R_ANAL_ESIL_DFG_TAG_SIBLING;
if (const_result) {
result->type |= R_ANAL_ESIL_DFG_TAG_CONST;
}
r_strbuf_appendf (result->content, "%d", edf->idx++);
RGraphNode *result_node = r_graph_add_node (edf->flow, result);
r_graph_add_edge (edf->flow, op_node, result_node);
_edf_var_set (edf, r_strbuf_get (result->content), result_node);
r_esil_push (esil, r_strbuf_get (result->content));
result = r_anal_esil_dfg_node_new (edf, "result_");
result->type = R_ANAL_ESIL_DFG_TAG_RESULT | R_ANAL_ESIL_DFG_TAG_SIBLING;
if (const_result) {
result->type |= R_ANAL_ESIL_DFG_TAG_CONST;
}
r_strbuf_appendf (result->content, "%d", edf->idx++);
result_node = r_graph_add_node (edf->flow, result);
r_graph_add_edge (edf->flow, op_node, result_node);
_edf_var_set (edf, r_strbuf_get (result->content), result_node);
return r_esil_push (esil, r_strbuf_get (result->content));
}
#endif
static bool edf_consume_2_push_1(REsil *esil) {
const char *op_string = esil->current_opstr;
RAnalEsilDFG *edf = (RAnalEsilDFG *)esil->user;
char *src[2] = { r_esil_pop (esil), r_esil_pop (esil) };
if (!src[0] || !src[1]) {
free (src[0]);
free (src[1]);
return false;
}
RAnalEsilDFGNode *eop_node = r_anal_esil_dfg_node_new (edf, src[1]);
r_strbuf_appendf (eop_node->content, ",%s,%s", src[0], op_string);
eop_node->type = R_ANAL_ESIL_DFG_TAG_RESULT | R_ANAL_ESIL_DFG_TAG_GENERATIVE;
// eop_node->type = R_ANAL_ESIL_DFG_TAG_GENERATIVE;
RGraphNode *op_node = r_graph_add_node (edf->flow, eop_node);
RGraphNode *src_node[2];
bool const_result = true;
ut32 i;
for (i = 0; i < 2; i++) {
const int src_type = r_esil_get_parm_type (esil, src[i]);
if (src_type == R_ESIL_PARM_REG) {
src_node[i] = _edf_reg_get (edf, src[i]);
RAnalEsilDFGNode *ec_node = (RAnalEsilDFGNode *)src_node[i]->data;
const_result &= !!(ec_node->type & R_ANAL_ESIL_DFG_TAG_CONST);
// const_result = false;
} else if (src_type == R_ESIL_PARM_NUM) {
src_node[i] = _edf_const_get (edf, src[i]);
// todo: check op_type, not relevant for now since this is always OP_MATH atm
const_result &= true;
} else {
src_node[i] = _edf_var_get (edf, src[i]);
if (src_node[i]) {
RAnalEsilDFGNode *ec_node = (RAnalEsilDFGNode *)src_node[i]->data;
const_result &= !!(ec_node->type & R_ANAL_ESIL_DFG_TAG_CONST);
} else {
R_LOG_WARN ("Invalid node");
}
}
r_graph_add_edge (edf->flow, src_node[i], op_node);
}
free (src[0]);
free (src[1]);
RAnalEsilDFGNode *result = r_anal_esil_dfg_node_new (edf, "result_");
result->type = R_ANAL_ESIL_DFG_TAG_RESULT;
if (const_result) {
result->type |= R_ANAL_ESIL_DFG_TAG_CONST;
}
r_strbuf_appendf (result->content, "%d", edf->idx++);
RGraphNode *result_node = r_graph_add_node (edf->flow, result);
r_graph_add_edge (edf->flow, op_node, result_node);
_edf_var_set (edf, r_strbuf_get (result->content), result_node);
r_esil_push (esil, r_strbuf_get (result->content));
return true;
}
static bool edf_consume_1_push_1(REsil *esil) {
const char *op_string = esil->current_opstr;
RAnalEsilDFG *edf = (RAnalEsilDFG *)esil->user;
char *src = r_esil_pop (esil);
if (!src) {
return false;
}
RAnalEsilDFGNode *eop_node = r_anal_esil_dfg_node_new (edf, src);
r_strbuf_appendf (eop_node->content, ",%s", op_string);
eop_node->type = R_ANAL_ESIL_DFG_TAG_RESULT | R_ANAL_ESIL_DFG_TAG_GENERATIVE;
// eop_node->type = R_ANAL_ESIL_DFG_TAG_GENERATIVE;
// esil operation node
RGraphNode *op_node = r_graph_add_node (edf->flow, eop_node);
// operation node, but in the rgraph
const int src_type = r_esil_get_parm_type (esil, src);
RGraphNode *src_node = NULL;
bool const_result = false;
// is the result a const value?
// e.g.: 42,!,!,! => 0,!,! => 1,! => 0 => const_result
// 0xaabbccdd,[1] => not const result, bc memory read
const ut32 eop_type = ((REsilOp *)ht_pp_find (esil->ops, op_string, NULL))->type;
// no need to check pointer here, bc this cannot fail if this function got called
if (src_type == R_ESIL_PARM_REG) {
src_node = _edf_reg_get (edf, src);
RAnalEsilDFGNode *ec_node = (RAnalEsilDFGNode *)src_node->data;
const_result = (!!(ec_node->type & R_ANAL_ESIL_DFG_TAG_CONST)) & (eop_type == R_ESIL_OP_TYPE_MATH);
} else if (src_type == R_ESIL_PARM_NUM) {
src_node = _edf_const_get (edf, src);
const_result = (eop_type == R_ESIL_OP_TYPE_MATH);
} else {
src_node = _edf_var_get (edf, src);
// cannot fail, bc src cannot be NULL
RAnalEsilDFGNode *ec_node = (RAnalEsilDFGNode *)src_node->data;
const_result = (eop_type == R_ESIL_OP_TYPE_MATH) & !!(ec_node->type & R_ANAL_ESIL_DFG_TAG_CONST);
}
free (src);
r_graph_add_edge (edf->flow, src_node, op_node);
RAnalEsilDFGNode *result = r_anal_esil_dfg_node_new (edf, "result_");
result->type = R_ANAL_ESIL_DFG_TAG_RESULT;
if (const_result) {
result->type |= R_ANAL_ESIL_DFG_TAG_CONST;
}
r_strbuf_appendf (result->content, "%d", edf->idx++);
RGraphNode *result_node = r_graph_add_node (edf->flow, result);
r_graph_add_edge (edf->flow, op_node, result_node);
_edf_var_set (edf, r_strbuf_get (result->content), result_node);
r_esil_push (esil, r_strbuf_get (result->content));
return true;
}
static RStrBuf *filter_gnode_expr(RAnalEsilDFG *dfg, RGraphNode *gnode);
#if THIS_FUNCTION_IS_UNUSED
static void _edf_check_stack_or_mem_const_node_cb(RGraphNode *gnode, RGraphVisitor *vi) {
bool *is_const = (bool *)vi->data;
RAnalEsilDFGNode *enode = (RAnalEsilDFGNode *)gnode->data;
is_const[0] &= (!((enode->type & R_ANAL_ESIL_DFG_TAG_VAR) &&
((enode->type & (R_ANAL_ESIL_DFG_TAG_CONST | R_ANAL_ESIL_DFG_TAG_MEM)) !=
(R_ANAL_ESIL_DFG_TAG_CONST | R_ANAL_ESIL_DFG_TAG_MEM))));
}
static bool _edf_is_stack_or_mem_const_node(RAnalEsilDFG *dfg, RGraphNode *gnode) {
RAnalEsilDFGNode *enode = (RAnalEsilDFGNode *)gnode->data;
if (enode->type & R_ANAL_ESIL_DFG_TAG_VAR &&
(enode->type & (R_ANAL_ESIL_DFG_TAG_CONST | R_ANAL_ESIL_DFG_TAG_MEM)) !=
(R_ANAL_ESIL_DFG_TAG_CONST | R_ANAL_ESIL_DFG_TAG_MEM)) {
return false;
}
bool ret = true;
RGraphVisitor vi = { _edf_check_stack_or_mem_const_node_cb, NULL, NULL, NULL, NULL, &ret };
r_graph_dfs_node_reverse (dfg->flow, gnode, &vi);
return ret;
}
#endif
static bool edf_consume_1_get_mem_push_1(REsil *esil) {
const char *op_string = esil->current_opstr;
RAnalEsilDFG *edf = (RAnalEsilDFG *)esil->user;
char *src = r_esil_pop (esil);
if (!src) {
return false;
}
RAnalEsilDFGNode *eop_node = r_anal_esil_dfg_node_new (edf, src);
r_strbuf_appendf (eop_node->content, ",%s", op_string);
// eop_node->type = R_ANAL_ESIL_DFG_TAG_RESULT | R_ANAL_ESIL_DFG_TAG_GENERATIVE;
eop_node->type = R_ANAL_ESIL_DFG_TAG_GENERATIVE;
// esil operation node
RGraphNode *op_node = r_graph_add_node (edf->flow, eop_node);
// operation node, but in the rgraph
ut32 mem_size = 0;
if (r_str_endswith (op_string, "[1]")) {
mem_size = 1;
} else if (r_str_endswith (op_string, "[2]")) {
mem_size = 2;
} else if (r_str_endswith (op_string, "[4]")) {
mem_size = 4;
} else if (r_str_endswith (op_string, "[8]")) {
mem_size = 8;
}
const int src_type = r_esil_get_parm_type (esil, src);
RGraphNode *src_node = NULL;
RGraphNode *mem_src_node = NULL;
// const ut32 eop_type = ((REsilOp *)ht_pp_find (esil->ops, op_string, NULL))->type;
// no need to check pointer here, bc this cannot fail if this function got called
if (src_type == R_ESIL_PARM_REG) {
src_node = _edf_reg_get (edf, src);
if (((RAnalEsilDFGNode *)src_node->data)->type & R_ANAL_ESIL_DFG_TAG_CONST) {
RStrBuf *expr = filter_gnode_expr(edf, src_node);
r_esil_parse (edf->esil, r_strbuf_get (expr));
ut64 src_addr = r_reg_getv (edf->reg, src);
R_LOG_DEBUG ("resolved: %s => 0x%"PFMT64x, r_strbuf_get (expr), src_addr);
r_strbuf_free (expr);
r_esil_stack_free (edf->esil);
edf->iob.system (edf->iob.io, "reset");
mem_src_node = _edf_mem_get (edf, src_addr, mem_size);
}
} else if (src_type == R_ESIL_PARM_NUM) {
src_node = _edf_const_get (edf, src);
ut64 src_addr;
r_esil_get_parm (esil, src, &src_addr);
mem_src_node = _edf_mem_get (edf, src_addr, mem_size);
} else {
src_node = _edf_var_get (edf, src);
// cannot fail, bc src cannot be NULL
if (((RAnalEsilDFGNode *)src_node->data)->type & R_ANAL_ESIL_DFG_TAG_CONST) {
// if (_edf_is_stack_or_mem_const_node (edf, src_node)) {
RStrBuf *expr = filter_gnode_expr(edf, src_node);
r_esil_parse (edf->esil, r_strbuf_get (expr));
char *src_addr_str = r_esil_pop (edf->esil);
R_LOG_DEBUG ("resolved: %s => %s", r_strbuf_get (expr), src_addr_str);
r_strbuf_free (expr);
ut64 src_addr;
r_esil_get_parm (esil, src_addr_str, &src_addr);
free (src_addr_str);
r_esil_stack_free (edf->esil);
edf->iob.system (edf->iob.io, "reset");
mem_src_node = _edf_mem_get (edf, src_addr, mem_size);
}
}
free (src);
r_graph_add_edge (edf->flow, src_node, op_node);
if (mem_src_node) {
r_graph_add_edge (edf->flow, mem_src_node, op_node);
}
RAnalEsilDFGNode *result = r_anal_esil_dfg_node_new (edf, "result_");
// result->type = R_ANAL_ESIL_DFG_TAG_RESULT | R_ANAL_ESIL_DFG_TAG_MEM;
result->type = R_ANAL_ESIL_DFG_TAG_RESULT;
if (mem_src_node && (((RAnalEsilDFGNode *)mem_src_node->data)->type & R_ANAL_ESIL_DFG_TAG_CONST)) {
result->type |= R_ANAL_ESIL_DFG_TAG_CONST;
}
r_strbuf_appendf (result->content, "%d", edf->idx++);
RGraphNode *result_node = r_graph_add_node (edf->flow, result);
r_graph_add_edge (edf->flow, op_node, result_node);
_edf_var_set (edf, r_strbuf_get (result->content), result_node);
r_esil_push (esil, r_strbuf_get (result->content));
return true;
}
static bool edf_consume_2_set_mem(REsil *esil) {
const char *op_string = esil->current_opstr;
RAnalEsilDFG *edf = (RAnalEsilDFG *)esil->user;
char *dst = r_esil_pop (esil);
char *src = r_esil_pop (esil);
if (!src || !dst) {
free (dst);
free (src);
return 0;
}
ut32 mem_size = 0;
if (r_str_endswith (op_string, "[1]")) {
mem_size = 1;
} else if (r_str_endswith (op_string, "[2]")) {
mem_size = 2;
} else if (r_str_endswith (op_string, "[4]")) {
mem_size = 4;
} else if (r_str_endswith (op_string, "[8]")) {
mem_size = 8;
}
int dst_type = r_esil_get_parm_type (esil, dst);
const int src_type = r_esil_get_parm_type (esil, src);
RGraphNode *src_node = NULL;
if (src_type == R_ESIL_PARM_REG) {
src_node = _edf_reg_get (edf, src);
} else if (src_type == R_ESIL_PARM_NUM) {
src_node = _edf_const_get (edf, src);
} else {
src_node = _edf_var_get (edf, src);
}
RGraphNode *dst_node = NULL;
bool write_result = false;
ut64 dst_addr = 0;
if (dst_type == R_ESIL_PARM_REG) {
dst_node = _edf_reg_get (edf, dst);
RAnalEsilDFGNode *ev_node = (RAnalEsilDFGNode *)dst_node->data;
ev_node->type |= R_ANAL_ESIL_DFG_TAG_PTR;
if (ev_node->type & R_ANAL_ESIL_DFG_TAG_CONST) {
RStrBuf *expr = filter_gnode_expr(edf, dst_node);
r_esil_parse (edf->esil, r_strbuf_get (expr));
dst_addr = r_reg_getv (edf->reg, dst);
R_LOG_DEBUG ("resolved: %s => 0x%"PFMT64x, r_strbuf_get (expr), dst_addr);
r_strbuf_free (expr);
r_esil_stack_free (edf->esil);
edf->iob.system (edf->iob.io, "reset");
write_result = true;
}
// TODO: try to resolve addr here
} else if (dst_type == R_ESIL_PARM_NUM) {
// dst_addr = r_num_get (NULL, dst);
r_esil_get_parm (esil, dst, &dst_addr);
RGraphNode *orig_value_gnode = r_graph_add_node (edf->flow, r_anal_esil_dfg_node_new (edf, dst));
RAnalEsilDFGNode *value_node = r_anal_esil_dfg_node_new (edf, dst);
value_node->type = R_ANAL_ESIL_DFG_TAG_CONST | R_ANAL_ESIL_DFG_TAG_PTR;
r_strbuf_appendf (value_node->content, ":const_ptr_%d", edf->idx++);
dst_node = r_graph_add_node (edf->flow, value_node);
r_graph_add_edge (edf->flow, orig_value_gnode, dst_node);
write_result = true;
} else {
dst_node = _edf_var_get (edf, dst);
// TODO: try to resolve addr here
// if (_edf_is_stack_or_mem_const_node (edf, dst_node)) {
if (((RAnalEsilDFGNode *)dst_node->data)->type & R_ANAL_ESIL_DFG_TAG_CONST) {
RStrBuf *expr = filter_gnode_expr(edf, dst_node);
r_esil_parse (edf->esil, r_strbuf_get (expr));
char *dst_addr_str = r_esil_pop (edf->esil);
R_LOG_DEBUG ("resolved: %s => %s", r_strbuf_get (expr), dst_addr_str);
r_strbuf_free (expr);
r_esil_get_parm (esil, dst_addr_str, &dst_addr);
free (dst_addr_str);
r_esil_stack_free (edf->esil);
edf->iob.system (edf->iob.io, "reset");
write_result = true;
}
RAnalEsilDFGNode *ev_node = (RAnalEsilDFGNode *)dst_node->data;
ev_node->type |= R_ANAL_ESIL_DFG_TAG_PTR;
}
if (!src_node || !dst_node) {
free (src);
free (dst);
return false;
}
RAnalEsilDFGNode *eop_node = r_anal_esil_dfg_node_new (edf, src);
R_FREE (src);
r_strbuf_appendf (eop_node->content, ",%s,%s", dst, op_string);
eop_node->type = R_ANAL_ESIL_DFG_TAG_GENERATIVE;
RGraphNode *op_node = r_graph_add_node (edf->flow, eop_node);
r_graph_add_edge (edf->flow, dst_node, op_node);
r_graph_add_edge (edf->flow, src_node, op_node);
char *content = r_str_newf ("[%d]@<%s>:mem_var_%d", mem_size,
dst_type == R_ESIL_PARM_REG? r_strbuf_get (((RAnalEsilDFGNode *)dst_node->data)->content): dst,
edf->idx + 1);
RAnalEsilDFGNode *result = r_anal_esil_dfg_node_new (edf, content);
free (content);
result->type = R_ANAL_ESIL_DFG_TAG_RESULT | R_ANAL_ESIL_DFG_TAG_VAR |
(((RAnalEsilDFGNode *)src_node->data)->type & R_ANAL_ESIL_DFG_TAG_CONST) | R_ANAL_ESIL_DFG_TAG_MEM;
dst_node = r_graph_add_node (edf->flow, result);
if (write_result) {
_edf_mem_set (edf, dst_addr, mem_size, dst_node);
}
r_graph_add_edge (edf->flow, op_node, dst_node);
free (dst);
return true;
}
static bool edf_consume_2_use_set_mem(REsil *esil) {
const char *op_string = esil->current_opstr;
RAnalEsilDFG *edf = (RAnalEsilDFG *)esil->user;
char *dst = r_esil_pop (esil);
char *src = r_esil_pop (esil);
if (!src || !dst) {
free (dst);
free (src);
return 0;
}
ut32 mem_size = 0;
if (r_str_endswith (op_string, "[1]")) {
mem_size = 1;
} else if (r_str_endswith (op_string, "[2]")) {
mem_size = 2;
} else if (r_str_endswith (op_string, "[4]")) {
mem_size = 4;
} else if (r_str_endswith (op_string, "[8]")) {
mem_size = 8;
}
int dst_type = r_esil_get_parm_type (esil, dst);
const int src_type = r_esil_get_parm_type (esil, src);
RGraphNode *src_node = NULL;
if (src_type == R_ESIL_PARM_REG) {
src_node = _edf_reg_get (edf, src);
} else if (src_type == R_ESIL_PARM_NUM) {
src_node = _edf_const_get (edf, src);
} else {
src_node = _edf_var_get (edf, src);
}
RGraphNode *dst_node = NULL;
bool write_result = false;
ut64 dst_addr = 0;
if (dst_type == R_ESIL_PARM_REG) {
dst_node = _edf_reg_get (edf, dst);
RAnalEsilDFGNode *ev_node = (RAnalEsilDFGNode *)dst_node->data;
ev_node->type |= R_ANAL_ESIL_DFG_TAG_PTR;
if (ev_node->type & R_ANAL_ESIL_DFG_TAG_CONST) {
RStrBuf *expr = filter_gnode_expr(edf, dst_node);
r_esil_parse (edf->esil, r_strbuf_get (expr));
dst_addr = r_reg_getv (edf->reg, dst);
R_LOG_DEBUG ("resolved: %s => 0x%"PFMT64x, r_strbuf_get (expr), dst_addr);
r_strbuf_free (expr);
r_esil_stack_free (edf->esil);
edf->iob.system (edf->iob.io, "reset");
write_result = true;
}
// TODO: try to resolve addr here
} else if (dst_type == R_ESIL_PARM_NUM) {
// dst_addr = r_num_get (NULL, dst);
r_esil_get_parm (esil, dst, &dst_addr);
RGraphNode *orig_value_gnode = r_graph_add_node (edf->flow, r_anal_esil_dfg_node_new (edf, dst));
RAnalEsilDFGNode *value_node = r_anal_esil_dfg_node_new (edf, dst);
value_node->type = R_ANAL_ESIL_DFG_TAG_CONST | R_ANAL_ESIL_DFG_TAG_PTR;
r_strbuf_appendf (value_node->content, ":const_ptr_%d", edf->idx++);
dst_node = r_graph_add_node (edf->flow, value_node);
r_graph_add_edge (edf->flow, orig_value_gnode, dst_node);
write_result = true;
} else {
dst_node = _edf_var_get (edf, dst);
// TODO: try to resolve addr here
// if (_edf_is_stack_or_mem_const_node (edf, dst_node)) {
if (((RAnalEsilDFGNode *)dst_node->data)->type & R_ANAL_ESIL_DFG_TAG_CONST) {
RStrBuf *expr = filter_gnode_expr(edf, dst_node);
r_esil_parse (edf->esil, r_strbuf_get (expr));
char *dst_addr_str = r_esil_pop (edf->esil);
R_LOG_DEBUG ("resolved: %s => %s", r_strbuf_get (expr), dst_addr_str);
r_strbuf_free (expr);
r_esil_get_parm (esil, dst_addr_str, &dst_addr);
free (dst_addr_str);
r_esil_stack_free (edf->esil);
edf->iob.system (edf->iob.io, "reset");
write_result = true;
}
RAnalEsilDFGNode *ev_node = (RAnalEsilDFGNode *)dst_node->data;
ev_node->type |= R_ANAL_ESIL_DFG_TAG_PTR;
}
if (!src_node || !dst_node) {
free (src);
free (dst);
return false;
}
RAnalEsilDFGNode *eop_node = r_anal_esil_dfg_node_new (edf, src);
R_FREE (src);
r_strbuf_appendf (eop_node->content, ",%s,%s", dst, op_string);
eop_node->type = R_ANAL_ESIL_DFG_TAG_GENERATIVE;
RGraphNode *op_node = r_graph_add_node (edf->flow, eop_node);
r_graph_add_edge (edf->flow, dst_node, op_node);
r_graph_add_edge (edf->flow, src_node, op_node);
RAnalEsilDFGNode *result = NULL;
if (write_result) {
RGraphNode *o_dst_node = _edf_mem_get (edf, dst_addr, mem_size);
r_graph_add_edge (edf->flow, o_dst_node, op_node);
char *content = r_str_newf ("[%d]@<%s>:mem_var_%d", mem_size,
dst_type == R_ESIL_PARM_REG? r_strbuf_get (((RAnalEsilDFGNode *)dst_node->data)->content):
dst, edf->idx + 1);
result = r_anal_esil_dfg_node_new (edf, content);
free (content);
result->type = R_ANAL_ESIL_DFG_TAG_RESULT | R_ANAL_ESIL_DFG_TAG_VAR | R_ANAL_ESIL_DFG_TAG_MEM |
((((RAnalEsilDFGNode *)src_node->data)->type & R_ANAL_ESIL_DFG_TAG_CONST) &
(((RAnalEsilDFGNode *)o_dst_node->data)->type & R_ANAL_ESIL_DFG_TAG_CONST));
dst_node = r_graph_add_node (edf->flow, result);
_edf_mem_set (edf, dst_addr, mem_size, dst_node);
} else {
char *content = r_str_newf ("[%d]@<%s>:mem_var_%d", mem_size,
dst_type == R_ESIL_PARM_REG? r_strbuf_get (((RAnalEsilDFGNode *)dst_node->data)->content):
dst, edf->idx + 1);
result = r_anal_esil_dfg_node_new (edf, content);
free (content);
result->type = R_ANAL_ESIL_DFG_TAG_RESULT | R_ANAL_ESIL_DFG_TAG_VAR | R_ANAL_ESIL_DFG_TAG_MEM;
dst_node = r_graph_add_node (edf->flow, result);
}
r_graph_add_edge (edf->flow, op_node, dst_node);
free (dst);
return true;
}
static bool edf_consume_1_set_mem(REsil *esil) {
const char *op_string = esil->current_opstr;
RAnalEsilDFG *edf = (RAnalEsilDFG *)esil->user;
char *dst = r_esil_pop (esil);
if (!dst) {
free (dst);
return 0;
}
ut32 mem_size = 0;
if (r_str_endswith (op_string, "[1]")) {
mem_size = 1;
} else if (r_str_endswith (op_string, "[2]")) {
mem_size = 2;
} else if (r_str_endswith (op_string, "[4]")) {
mem_size = 4;
} else if (r_str_endswith (op_string, "[8]")) {
mem_size = 8;
}
int dst_type = r_esil_get_parm_type (esil, dst);
RGraphNode *dst_node = NULL;
bool write_result = false;
ut64 dst_addr = 0;
if (dst_type == R_ESIL_PARM_REG) {
dst_node = _edf_reg_get (edf, dst);
RAnalEsilDFGNode *ev_node = (RAnalEsilDFGNode *)dst_node->data;
ev_node->type |= R_ANAL_ESIL_DFG_TAG_PTR;
if (ev_node->type & R_ANAL_ESIL_DFG_TAG_CONST) {
RStrBuf *expr = filter_gnode_expr(edf, dst_node);
r_esil_parse (edf->esil, r_strbuf_get (expr));
dst_addr = r_reg_getv (edf->reg, dst);
R_LOG_DEBUG ("resolved: %s => 0x%"PFMT64x, r_strbuf_get (expr), dst_addr);
r_strbuf_free (expr);
r_esil_stack_free (edf->esil);
edf->iob.system (edf->iob.io, "reset");
write_result = true;
}
// TODO: try to resolve addr here
} else if (dst_type == R_ESIL_PARM_NUM) {
// dst_addr = r_num_get (NULL, dst);
r_esil_get_parm (esil, dst, &dst_addr);
RGraphNode *orig_value_gnode = r_graph_add_node (edf->flow, r_anal_esil_dfg_node_new (edf, dst));
RAnalEsilDFGNode *value_node = r_anal_esil_dfg_node_new (edf, dst);
value_node->type = R_ANAL_ESIL_DFG_TAG_CONST | R_ANAL_ESIL_DFG_TAG_PTR;
r_strbuf_appendf (value_node->content, ":const_ptr_%d", edf->idx++);
dst_node = r_graph_add_node (edf->flow, value_node);
r_graph_add_edge (edf->flow, orig_value_gnode, dst_node);
write_result = true;
} else {
dst_node = _edf_var_get (edf, dst);
// TODO: try to resolve addr here
// if (_edf_is_stack_or_mem_const_node (edf, dst_node)) {
if (((RAnalEsilDFGNode *)dst_node->data)->type & R_ANAL_ESIL_DFG_TAG_CONST) {
RStrBuf *expr = filter_gnode_expr(edf, dst_node);
r_esil_parse (edf->esil, r_strbuf_get (expr));
char *dst_addr_str = r_esil_pop (edf->esil);
R_LOG_DEBUG ("resolved: %s => %s", r_strbuf_get (expr), dst_addr_str);
r_strbuf_free (expr);
r_esil_get_parm (esil, dst_addr_str, &dst_addr);
free (dst_addr_str);
r_esil_stack_free (edf->esil);
edf->iob.system (edf->iob.io, "reset");
write_result = true;
}
RAnalEsilDFGNode *ev_node = (RAnalEsilDFGNode *)dst_node->data;
ev_node->type |= R_ANAL_ESIL_DFG_TAG_PTR;
}
if (!dst_node) {
free (dst);
return false;
}
RAnalEsilDFGNode *eop_node = r_anal_esil_dfg_node_new (edf, dst);
r_strbuf_appendf (eop_node->content, ",%s", op_string);
eop_node->type = R_ANAL_ESIL_DFG_TAG_GENERATIVE;
RGraphNode *op_node = r_graph_add_node (edf->flow, eop_node);
r_graph_add_edge (edf->flow, dst_node, op_node);
RAnalEsilDFGNode *result = NULL;
if (write_result) {
RGraphNode *o_dst_node = _edf_mem_get (edf, dst_addr, mem_size);
r_graph_add_edge (edf->flow, o_dst_node, op_node);
char *content = r_str_newf ("[%d]@<%s>:mem_var_%d", mem_size,
dst_type == R_ESIL_PARM_REG? r_strbuf_get (((RAnalEsilDFGNode *)dst_node->data)->content):
dst, edf->idx + 1);
result = r_anal_esil_dfg_node_new (edf, content);
free (content);
result->type = R_ANAL_ESIL_DFG_TAG_RESULT | R_ANAL_ESIL_DFG_TAG_VAR | R_ANAL_ESIL_DFG_TAG_MEM |
(((RAnalEsilDFGNode *)o_dst_node->data)->type & R_ANAL_ESIL_DFG_TAG_CONST);
dst_node = r_graph_add_node (edf->flow, result);
_edf_mem_set (edf, dst_addr, mem_size, dst_node);
} else {
char *content = r_str_newf ("[%d]@<%s>:mem_var_%d", mem_size,
dst_type == R_ESIL_PARM_REG? r_strbuf_get (((RAnalEsilDFGNode *)dst_node->data)->content):
dst, edf->idx + 1);
result = r_anal_esil_dfg_node_new (edf, content);
free (content);
result->type = R_ANAL_ESIL_DFG_TAG_RESULT | R_ANAL_ESIL_DFG_TAG_VAR | R_ANAL_ESIL_DFG_TAG_MEM;
dst_node = r_graph_add_node (edf->flow, result);
}
r_graph_add_edge (edf->flow, op_node, dst_node);
free (dst);
return true;
}
static bool edf_use_new_push_1(REsil *esil, const char *op_string, AddConstraintStringUseNewCB cb) {
RAnalEsilDFG *edf = (RAnalEsilDFG *)esil->user;
RGraphNode *op_node = r_graph_add_node (edf->flow, r_anal_esil_dfg_node_new (edf, op_string));
RGraphNode *latest_new = edf->cur;
if (!latest_new) {
return false;
}
RAnalEsilDFGNode *result = r_anal_esil_dfg_node_new (edf, "result_");
result->type = R_ANAL_ESIL_DFG_TAG_RESULT; // is this generative?
r_strbuf_appendf (result->content, "%d", edf->idx++);
if (cb) {
RAnalEsilDFGNode *e_new_node = (RAnalEsilDFGNode *)latest_new->data;
cb (result->content, r_strbuf_get (e_new_node->content));
}
RGraphNode *result_node = r_graph_add_node (edf->flow, result);
_edf_var_set (edf, r_strbuf_get (result->content), result_node);
r_graph_add_edge (edf->flow, latest_new, op_node);
r_graph_add_edge (edf->flow, op_node, result_node);
return r_esil_push (esil, r_strbuf_get (result->content));
}
static bool edf_consume_1_use_old_new_push_1(REsil *esil, const char *op_string, AddConstraintStringConsume1UseOldNewCB cb) {
RAnalEsilDFG *edf = (RAnalEsilDFG *)esil->user;
char *src = r_esil_pop (esil);
if (!src) {
return false;
}
RAnalEsilDFGNode *eop_node = r_anal_esil_dfg_node_new (edf, src);
#if 0
eop_node->type = R_ANAL_ESIL_DFG_TAG_GENERATIVE;
#endif
r_strbuf_appendf (eop_node->content, ",%s", op_string);
RGraphNode *op_node = r_graph_add_node (edf->flow, eop_node);
const int src_type = r_esil_get_parm_type (esil, src);
RGraphNode *src_node = NULL;
if (src_type == R_ESIL_PARM_REG) {
src_node = _edf_reg_get (edf, src);
} else if (src_type == R_ESIL_PARM_NUM) {
src_node = _edf_const_get (edf, src);
} else {
src_node = _edf_var_get (edf, src);
}
free (src);
r_graph_add_edge (edf->flow, src_node, op_node);
RGraphNode *latest_new = edf->cur;
RGraphNode *latest_old = edf->old;
RAnalEsilDFGNode *result = r_anal_esil_dfg_node_new (edf, "result_");
result->type = R_ANAL_ESIL_DFG_TAG_RESULT; // propagate type here
r_strbuf_appendf (result->content, "%d", edf->idx++);
if (cb) {
RAnalEsilDFGNode *e_src_node = (RAnalEsilDFGNode *)src_node->data;
RAnalEsilDFGNode *e_new_node = (RAnalEsilDFGNode *)latest_new->data;
RAnalEsilDFGNode *e_old_node = (RAnalEsilDFGNode *)latest_old->data;
cb (result->content, r_strbuf_get (e_src_node->content),
r_strbuf_get (e_new_node->content), r_strbuf_get (e_old_node->content));
}
RGraphNode *result_node = r_graph_add_node (edf->flow, result);
_edf_var_set (edf, r_strbuf_get (result->content), result_node);
r_graph_add_edge (edf->flow, latest_new, op_node);
r_graph_add_edge (edf->flow, latest_old, op_node);
r_graph_add_edge (edf->flow, op_node, result_node);
return r_esil_push (esil, r_strbuf_get (result->content));
}
static bool _dfg_mem_read (REsil *esil, ut64 addr, ut8 *buf, int len) {
RAnalEsilDFG *dfg = (RAnalEsilDFG *)esil->user;
addr &= esil->addrmask;
if (dfg->use_maps) {
RIOMap *map = dfg->iob.map_get_at (dfg->iob.io, addr);
if (map && (map->perm & R_PERM_RW) == R_PERM_R) {
return dfg->iob.read_at (dfg->iob.io, addr, buf, len);
}
}
return (dfg->iob.fd_read_at (dfg->iob.io, dfg->fd, addr, buf, len) > 0);
}
static bool _dfg_mem_write (REsil *esil, ut64 addr, const ut8 *buf, int len) {
RAnalEsilDFG *dfg = (RAnalEsilDFG *)esil->user;
addr &= esil->addrmask;
if (dfg->use_maps) {
RIOMap *map = dfg->iob.map_get_at (dfg->iob.io, addr);
if (map && (map->perm & R_PERM_RW) == R_PERM_R) {
return true;
}
}
return (dfg->iob.fd_write_at (dfg->iob.io, dfg->fd, addr, buf, len) > 0);
}
R_API RAnalEsilDFG *r_anal_esil_dfg_new(RAnal* anal, bool use_map_info, bool use_maps) {
R_RETURN_VAL_IF_FAIL (anal && anal->reg, NULL);
RAnalEsilDFG *dfg = R_NEW0 (RAnalEsilDFG);
if (!dfg) {
return NULL;
}
dfg->use_map_info = use_map_info;
dfg->use_maps = use_maps;
if (anal->iob.io) {
const bool autofd = anal->iob.io->autofd;
anal->iob.io->autofd = false;
dfg->fd = anal->iob.fd_open (anal->iob.io, "treebuf://", R_PERM_RW, 0);
if (dfg->fd >= 0) {
memcpy (&dfg->iob, &anal->iob, sizeof (RIOBind));
}
anal->iob.io->autofd = autofd;
}
dfg->reg = r_reg_new ();
if (!dfg->reg) {
free (dfg);
return NULL;
}
dfg->esil = r_esil_new (4096, 0, 1);
if (!dfg->esil) {
r_reg_free (dfg->reg);
free (dfg);
return NULL;
}
dfg->flow = r_graph_new ();
if (!dfg->flow) {
r_esil_free (dfg->esil);
r_reg_free (dfg->reg);
free (dfg);
return NULL;
}
dfg->regs = sdb_new0 ();
if (!dfg->regs) {
r_graph_free (dfg->flow);
r_esil_free (dfg->esil);
r_reg_free (dfg->reg);
free (dfg);
return NULL;
}
// rax, eax, ax, ah, al => 8 should be enough
dfg->todo = r_queue_new (8);
if (!dfg->todo) {
sdb_free (dfg->regs);
r_graph_free (dfg->flow);
r_esil_free (dfg->esil);
r_reg_free (dfg->reg);
free (dfg);
return NULL;
}
dfg->vars = r_crbtree_new (free);
if (!dfg->vars) {
r_queue_free (dfg->todo);
sdb_free (dfg->regs);
r_graph_free (dfg->flow);
r_esil_free (dfg->esil);
r_reg_free (dfg->reg);
free (dfg);
return NULL;
}
// this is not exactly necessary
// could use RReg-API directly in the dfg gen,
// but sdb as transition table is probably faster
RRegItem *ri;
RListIter *ator;
r_list_foreach (anal->reg->allregs, ator, ri) {
const ut32 from = ri->offset;
const ut32 to = from + ri->size - 1; // closed intervals because of FUCK YOU
const ut64 v = to | (((ut64)from) << 32);
char *reg = r_str_newf ("reg.%s", ri->name);
sdb_num_set (dfg->regs, reg, v, 0);
free (reg);
}
r_reg_set_profile_string (dfg->reg, anal->reg->reg_profile_str);
r_esil_setup(dfg->esil, anal, 0, 0, 0);
if (dfg->iob.io && dfg->fd >= 0) {
dfg->esil->user = dfg;
dfg->esil->cb.mem_read = _dfg_mem_read;
dfg->esil->cb.mem_write = _dfg_mem_write;
}
return dfg;
}
R_API void r_anal_esil_dfg_free(RAnalEsilDFG *dfg) {
if (dfg) {
if (dfg->flow) {
RGraphNode *n;
RListIter *iter;
r_list_foreach (r_graph_get_nodes (dfg->flow), iter, n) {
n->free = (RListFree)_dfg_node_free;
}
r_graph_free (dfg->flow);
}
sdb_free (dfg->regs);
r_crbtree_free (dfg->vars);
r_queue_free (dfg->todo);
r_esil_free (dfg->esil);
r_reg_free (dfg->reg);
if (dfg->iob.io && dfg->fd >= 0) {
dfg->iob.fd_close (dfg->iob.io, dfg->fd);
}
free (dfg);
}
}
R_API RAnalEsilDFG *r_anal_esil_dfg_expr(RAnal *anal, R_NULLABLE RAnalEsilDFG *dfg, const char *expr,
bool use_map_info, bool use_maps) {
R_RETURN_VAL_IF_FAIL (anal && expr, NULL);
REsil *esil = r_esil_new (4096, 0, 1);
if (!esil) {
return NULL;
}
esil->anal = anal;
RAnalEsilDFG *edf = dfg ? dfg : r_anal_esil_dfg_new (anal, use_map_info, use_maps);
if (!edf) {
r_esil_free (esil);
return NULL;
}
r_esil_set_op (esil, "=", edf_consume_2_set_reg, 0, 2, R_ESIL_OP_TYPE_REG_WRITE);
r_esil_set_op (esil, ":=", edf_eq_weak, 0, 2, R_ESIL_OP_TYPE_REG_WRITE);
r_esil_set_op (esil, "$s", edf_sf, 1, 0, R_ESIL_OP_TYPE_UNKNOWN); // XXX TODO
r_esil_set_op (esil, "$z", edf_zf, 1, 0, R_ESIL_OP_TYPE_UNKNOWN);
r_esil_set_op (esil, "$p", edf_pf, 1, 0, R_ESIL_OP_TYPE_UNKNOWN);
r_esil_set_op (esil, "$c", edf_cf, 1, 1, R_ESIL_OP_TYPE_UNKNOWN);
r_esil_set_op (esil, "$b", edf_bf, 1, 1, R_ESIL_OP_TYPE_UNKNOWN);
r_esil_set_op (esil, "^=", edf_consume_2_use_set_reg, 0, 2, R_ESIL_OP_TYPE_MATH | R_ESIL_OP_TYPE_REG_WRITE);
r_esil_set_op (esil, "-=", edf_consume_2_use_set_reg, 0, 2, R_ESIL_OP_TYPE_MATH | R_ESIL_OP_TYPE_REG_WRITE);
r_esil_set_op (esil, "+=", edf_consume_2_use_set_reg, 0, 2, R_ESIL_OP_TYPE_MATH | R_ESIL_OP_TYPE_REG_WRITE);
r_esil_set_op (esil, "*=", edf_consume_2_use_set_reg, 0, 2, R_ESIL_OP_TYPE_MATH | R_ESIL_OP_TYPE_REG_WRITE);
r_esil_set_op (esil, "/=", edf_consume_2_use_set_reg, 0, 2, R_ESIL_OP_TYPE_MATH | R_ESIL_OP_TYPE_REG_WRITE);
r_esil_set_op (esil, "&=", edf_consume_2_use_set_reg, 0, 2, R_ESIL_OP_TYPE_MATH | R_ESIL_OP_TYPE_REG_WRITE);
r_esil_set_op (esil, "|=", edf_consume_2_use_set_reg, 0, 2, R_ESIL_OP_TYPE_MATH | R_ESIL_OP_TYPE_REG_WRITE);
r_esil_set_op (esil, "^=", edf_consume_2_use_set_reg, 0, 2, R_ESIL_OP_TYPE_MATH | R_ESIL_OP_TYPE_REG_WRITE);
r_esil_set_op (esil, "+", edf_consume_2_push_1, 1, 2, R_ESIL_OP_TYPE_MATH);
r_esil_set_op (esil, "-", edf_consume_2_push_1, 1, 2, R_ESIL_OP_TYPE_MATH);
r_esil_set_op (esil, "&", edf_consume_2_push_1, 1, 2, R_ESIL_OP_TYPE_MATH);
r_esil_set_op (esil, "|", edf_consume_2_push_1, 1, 2, R_ESIL_OP_TYPE_MATH);
r_esil_set_op (esil, "^", edf_consume_2_push_1, 1, 2, R_ESIL_OP_TYPE_MATH);
r_esil_set_op (esil, "%", edf_consume_2_push_1, 1, 2, R_ESIL_OP_TYPE_MATH);
r_esil_set_op (esil, "*", edf_consume_2_push_1, 1, 2, R_ESIL_OP_TYPE_MATH);
r_esil_set_op (esil, "/", edf_consume_2_push_1, 1, 2, R_ESIL_OP_TYPE_MATH);
r_esil_set_op (esil, ">>", edf_consume_2_push_1, 1, 2, R_ESIL_OP_TYPE_MATH);
r_esil_set_op (esil, "POP", edf_pop, 1, 0, R_ESIL_OP_TYPE_UNKNOWN);
#if 1
r_esil_set_op (esil, "DUP", edf_dup, 1, 2, R_ESIL_OP_TYPE_UNKNOWN);
#endif
r_esil_set_op (esil, "<<", edf_consume_2_push_1, 1, 2, R_ESIL_OP_TYPE_MATH);
r_esil_set_op (esil, ">>>", edf_consume_2_push_1, 1, 2, R_ESIL_OP_TYPE_MATH);
r_esil_set_op (esil, ">>>", edf_consume_2_push_1, 1, 2, R_ESIL_OP_TYPE_MATH);
r_esil_set_op (esil, "!", edf_consume_1_push_1, 1, 1, R_ESIL_OP_TYPE_MATH);
r_esil_set_op (esil, "++", edf_consume_1_push_1, 1, 1, R_ESIL_OP_TYPE_MATH);
r_esil_set_op (esil, "--", edf_consume_1_push_1, 1, 1, R_ESIL_OP_TYPE_MATH);
r_esil_set_op (esil, "[1]", edf_consume_1_get_mem_push_1, 1, 1, R_ESIL_OP_TYPE_MEM_READ);
r_esil_set_op (esil, "[2]", edf_consume_1_get_mem_push_1, 1, 1, R_ESIL_OP_TYPE_MEM_READ);
r_esil_set_op (esil, "[4]", edf_consume_1_get_mem_push_1, 1, 1, R_ESIL_OP_TYPE_MEM_READ);
r_esil_set_op (esil, "[8]", edf_consume_1_get_mem_push_1, 1, 1, R_ESIL_OP_TYPE_MEM_READ);
// r_esil_set_op (esil, "[16]", edf_consume_1_push_1, 1, 1, R_ESIL_OP_TYPE_MEM_READ);
r_esil_set_op (esil, "=[1]", edf_consume_2_set_mem, 0, 2, R_ESIL_OP_TYPE_MEM_WRITE);
r_esil_set_op (esil, "=[2]", edf_consume_2_set_mem, 0, 2, R_ESIL_OP_TYPE_MEM_WRITE);
r_esil_set_op (esil, "=[4]", edf_consume_2_set_mem, 0, 2, R_ESIL_OP_TYPE_MEM_WRITE);
r_esil_set_op (esil, "=[8]", edf_consume_2_set_mem, 0, 2, R_ESIL_OP_TYPE_MEM_WRITE);
r_esil_set_op (esil, "|=[1]", edf_consume_2_use_set_mem, 0, 2,
R_ESIL_OP_TYPE_MATH | R_ESIL_OP_TYPE_MEM_READ | R_ESIL_OP_TYPE_MEM_WRITE);
r_esil_set_op (esil, "|=[2]", edf_consume_2_use_set_mem, 0, 2,
R_ESIL_OP_TYPE_MATH | R_ESIL_OP_TYPE_MEM_READ | R_ESIL_OP_TYPE_MEM_WRITE);
r_esil_set_op (esil, "|=[4]", edf_consume_2_use_set_mem, 0, 2,
R_ESIL_OP_TYPE_MATH | R_ESIL_OP_TYPE_MEM_READ | R_ESIL_OP_TYPE_MEM_WRITE);
r_esil_set_op (esil, "|=[8]", edf_consume_2_use_set_mem, 0, 2,
R_ESIL_OP_TYPE_MATH | R_ESIL_OP_TYPE_MEM_READ | R_ESIL_OP_TYPE_MEM_WRITE);
r_esil_set_op (esil, "^=[1]", edf_consume_2_use_set_mem, 0, 2,
R_ESIL_OP_TYPE_MATH | R_ESIL_OP_TYPE_MEM_READ | R_ESIL_OP_TYPE_MEM_WRITE);
r_esil_set_op (esil, "^=[2]", edf_consume_2_use_set_mem, 0, 2,
R_ESIL_OP_TYPE_MATH | R_ESIL_OP_TYPE_MEM_READ | R_ESIL_OP_TYPE_MEM_WRITE);
r_esil_set_op (esil, "^=[4]", edf_consume_2_use_set_mem, 0, 2,
R_ESIL_OP_TYPE_MATH | R_ESIL_OP_TYPE_MEM_READ | R_ESIL_OP_TYPE_MEM_WRITE);
r_esil_set_op (esil, "^=[8]", edf_consume_2_use_set_mem, 0, 2,
R_ESIL_OP_TYPE_MATH | R_ESIL_OP_TYPE_MEM_READ | R_ESIL_OP_TYPE_MEM_WRITE);
r_esil_set_op (esil, "&=[1]", edf_consume_2_use_set_mem, 0, 2,
R_ESIL_OP_TYPE_MATH | R_ESIL_OP_TYPE_MEM_READ | R_ESIL_OP_TYPE_MEM_WRITE);
r_esil_set_op (esil, "&=[2]", edf_consume_2_use_set_mem, 0, 2,
R_ESIL_OP_TYPE_MATH | R_ESIL_OP_TYPE_MEM_READ | R_ESIL_OP_TYPE_MEM_WRITE);
r_esil_set_op (esil, "&=[4]", edf_consume_2_use_set_mem, 0, 2,
R_ESIL_OP_TYPE_MATH | R_ESIL_OP_TYPE_MEM_READ | R_ESIL_OP_TYPE_MEM_WRITE);
r_esil_set_op (esil, "&=[8]", edf_consume_2_use_set_mem, 0, 2,
R_ESIL_OP_TYPE_MATH | R_ESIL_OP_TYPE_MEM_READ | R_ESIL_OP_TYPE_MEM_WRITE);
r_esil_set_op (esil, "+=[1]", edf_consume_2_use_set_mem, 0, 2,
R_ESIL_OP_TYPE_MATH | R_ESIL_OP_TYPE_MEM_READ | R_ESIL_OP_TYPE_MEM_WRITE);
r_esil_set_op (esil, "+=[2]", edf_consume_2_use_set_mem, 0, 2,
R_ESIL_OP_TYPE_MATH | R_ESIL_OP_TYPE_MEM_READ | R_ESIL_OP_TYPE_MEM_WRITE);
r_esil_set_op (esil, "+=[4]", edf_consume_2_use_set_mem, 0, 2,
R_ESIL_OP_TYPE_MATH | R_ESIL_OP_TYPE_MEM_READ | R_ESIL_OP_TYPE_MEM_WRITE);
r_esil_set_op (esil, "+=[8]", edf_consume_2_use_set_mem, 0, 2,
R_ESIL_OP_TYPE_MATH | R_ESIL_OP_TYPE_MEM_READ | R_ESIL_OP_TYPE_MEM_WRITE);
r_esil_set_op (esil, "-=[1]", edf_consume_2_use_set_mem, 0, 2,
R_ESIL_OP_TYPE_MATH | R_ESIL_OP_TYPE_MEM_READ | R_ESIL_OP_TYPE_MEM_WRITE);
r_esil_set_op (esil, "-=[2]", edf_consume_2_use_set_mem, 0, 2,
R_ESIL_OP_TYPE_MATH | R_ESIL_OP_TYPE_MEM_READ | R_ESIL_OP_TYPE_MEM_WRITE);
r_esil_set_op (esil, "-=[4]", edf_consume_2_use_set_mem, 0, 2,
R_ESIL_OP_TYPE_MATH | R_ESIL_OP_TYPE_MEM_READ | R_ESIL_OP_TYPE_MEM_WRITE);
r_esil_set_op (esil, "-=[8]", edf_consume_2_use_set_mem, 0, 2,
R_ESIL_OP_TYPE_MATH | R_ESIL_OP_TYPE_MEM_READ | R_ESIL_OP_TYPE_MEM_WRITE);
r_esil_set_op (esil, "%=[1]", edf_consume_2_use_set_mem, 0, 2,
R_ESIL_OP_TYPE_MATH | R_ESIL_OP_TYPE_MEM_READ | R_ESIL_OP_TYPE_MEM_WRITE);
r_esil_set_op (esil, "%=[2]", edf_consume_2_use_set_mem, 0, 2,
R_ESIL_OP_TYPE_MATH | R_ESIL_OP_TYPE_MEM_READ | R_ESIL_OP_TYPE_MEM_WRITE);
r_esil_set_op (esil, "%=[4]", edf_consume_2_use_set_mem, 0, 2,
R_ESIL_OP_TYPE_MATH | R_ESIL_OP_TYPE_MEM_READ | R_ESIL_OP_TYPE_MEM_WRITE);
r_esil_set_op (esil, "%=[8]", edf_consume_2_use_set_mem, 0, 2,
R_ESIL_OP_TYPE_MATH | R_ESIL_OP_TYPE_MEM_READ | R_ESIL_OP_TYPE_MEM_WRITE);
r_esil_set_op (esil, "/=[1]", edf_consume_2_use_set_mem, 0, 2,
R_ESIL_OP_TYPE_MATH | R_ESIL_OP_TYPE_MEM_READ | R_ESIL_OP_TYPE_MEM_WRITE);
r_esil_set_op (esil, "/=[2]", edf_consume_2_use_set_mem, 0, 2,
R_ESIL_OP_TYPE_MATH | R_ESIL_OP_TYPE_MEM_READ | R_ESIL_OP_TYPE_MEM_WRITE);
r_esil_set_op (esil, "/=[4]", edf_consume_2_use_set_mem, 0, 2,
R_ESIL_OP_TYPE_MATH | R_ESIL_OP_TYPE_MEM_READ | R_ESIL_OP_TYPE_MEM_WRITE);
r_esil_set_op (esil, "/=[8]", edf_consume_2_use_set_mem, 0, 2,
R_ESIL_OP_TYPE_MATH | R_ESIL_OP_TYPE_MEM_READ | R_ESIL_OP_TYPE_MEM_WRITE);
r_esil_set_op (esil, "*=[1]", edf_consume_2_use_set_mem, 0, 2,
R_ESIL_OP_TYPE_MATH | R_ESIL_OP_TYPE_MEM_READ | R_ESIL_OP_TYPE_MEM_WRITE);
r_esil_set_op (esil, "*=[2]", edf_consume_2_use_set_mem, 0, 2,
R_ESIL_OP_TYPE_MATH | R_ESIL_OP_TYPE_MEM_READ | R_ESIL_OP_TYPE_MEM_WRITE);
r_esil_set_op (esil, "*=[4]", edf_consume_2_use_set_mem, 0, 2,
R_ESIL_OP_TYPE_MATH | R_ESIL_OP_TYPE_MEM_READ | R_ESIL_OP_TYPE_MEM_WRITE);
r_esil_set_op (esil, "*=[8]", edf_consume_2_use_set_mem, 0, 2,
R_ESIL_OP_TYPE_MATH | R_ESIL_OP_TYPE_MEM_READ | R_ESIL_OP_TYPE_MEM_WRITE);
r_esil_set_op (esil, ">>=[1]", edf_consume_2_use_set_mem, 0, 2,
R_ESIL_OP_TYPE_MATH | R_ESIL_OP_TYPE_MEM_READ | R_ESIL_OP_TYPE_MEM_WRITE);
r_esil_set_op (esil, ">>=[2]", edf_consume_2_use_set_mem, 0, 2,
R_ESIL_OP_TYPE_MATH | R_ESIL_OP_TYPE_MEM_READ | R_ESIL_OP_TYPE_MEM_WRITE);
r_esil_set_op (esil, ">>=[4]", edf_consume_2_use_set_mem, 0, 2,
R_ESIL_OP_TYPE_MATH | R_ESIL_OP_TYPE_MEM_READ | R_ESIL_OP_TYPE_MEM_WRITE);
r_esil_set_op (esil, ">>=[8]", edf_consume_2_use_set_mem, 0, 2,
R_ESIL_OP_TYPE_MATH | R_ESIL_OP_TYPE_MEM_READ | R_ESIL_OP_TYPE_MEM_WRITE);
r_esil_set_op (esil, "<<=[1]", edf_consume_2_use_set_mem, 0, 2,
R_ESIL_OP_TYPE_MATH | R_ESIL_OP_TYPE_MEM_READ | R_ESIL_OP_TYPE_MEM_WRITE);
r_esil_set_op (esil, "<<=[2]", edf_consume_2_use_set_mem, 0, 2,
R_ESIL_OP_TYPE_MATH | R_ESIL_OP_TYPE_MEM_READ | R_ESIL_OP_TYPE_MEM_WRITE);
r_esil_set_op (esil, "<<=[4]", edf_consume_2_use_set_mem, 0, 2,
R_ESIL_OP_TYPE_MATH | R_ESIL_OP_TYPE_MEM_READ | R_ESIL_OP_TYPE_MEM_WRITE);
r_esil_set_op (esil, "<<=[8]", edf_consume_2_use_set_mem, 0, 2,
R_ESIL_OP_TYPE_MATH | R_ESIL_OP_TYPE_MEM_READ | R_ESIL_OP_TYPE_MEM_WRITE);
r_esil_set_op (esil, "++=[1]", edf_consume_1_set_mem, 0, 1,
R_ESIL_OP_TYPE_MATH | R_ESIL_OP_TYPE_MEM_READ | R_ESIL_OP_TYPE_MEM_WRITE);
r_esil_set_op (esil, "++=[2]", edf_consume_1_set_mem, 0, 1,
R_ESIL_OP_TYPE_MATH | R_ESIL_OP_TYPE_MEM_READ | R_ESIL_OP_TYPE_MEM_WRITE);
r_esil_set_op (esil, "++=[4]", edf_consume_1_set_mem, 0, 1,
R_ESIL_OP_TYPE_MATH | R_ESIL_OP_TYPE_MEM_READ | R_ESIL_OP_TYPE_MEM_WRITE);
r_esil_set_op (esil, "++=[8]", edf_consume_1_set_mem, 0, 1,
R_ESIL_OP_TYPE_MATH | R_ESIL_OP_TYPE_MEM_READ | R_ESIL_OP_TYPE_MEM_WRITE);
r_esil_set_op (esil, "--=[1]", edf_consume_1_set_mem, 0, 1,
R_ESIL_OP_TYPE_MATH | R_ESIL_OP_TYPE_MEM_READ | R_ESIL_OP_TYPE_MEM_WRITE);
r_esil_set_op (esil, "--=[2]", edf_consume_1_set_mem, 0, 1,
R_ESIL_OP_TYPE_MATH | R_ESIL_OP_TYPE_MEM_READ | R_ESIL_OP_TYPE_MEM_WRITE);
r_esil_set_op (esil, "--=[4]", edf_consume_1_set_mem, 0, 1,
R_ESIL_OP_TYPE_MATH | R_ESIL_OP_TYPE_MEM_READ | R_ESIL_OP_TYPE_MEM_WRITE);
r_esil_set_op (esil, "--=[8]", edf_consume_1_set_mem, 0, 1,
R_ESIL_OP_TYPE_MATH | R_ESIL_OP_TYPE_MEM_READ | R_ESIL_OP_TYPE_MEM_WRITE);
esil->user = edf;
RReg *reg = edf->esil->anal->reg;
edf->esil->anal->reg = edf->reg;
r_esil_parse (esil, expr);
edf->esil->anal->reg = reg;
r_esil_free (esil);
return edf;
}
static int _dfg_node_filter_insert_cmp(void *incoming, void *in, void *user) {
RAnalEsilDFGNode *incoming_node = (RAnalEsilDFGNode *)incoming;
RAnalEsilDFGNode *in_node = (RAnalEsilDFGNode *)in;
return incoming_node->idx - in_node->idx;
}
static int _dfg_gnode_reducer_insert_cmp(void *incoming, void *in, void *user) {
RGraphNode *incoming_gnode = (RGraphNode *)incoming;
RGraphNode *in_gnode = (RGraphNode *)in;
RAnalEsilDFGNode *incoming_node = (RAnalEsilDFGNode *)incoming_gnode->data;
RAnalEsilDFGNode *in_node = (RAnalEsilDFGNode *)in_gnode->data;
return in_node->idx - incoming_node->idx;
}
static void _dfg_filter_rev_dfs(RGraphNode *n, RAnalEsilDFGFilter *filter) {
RAnalEsilDFGNode *node = (RAnalEsilDFGNode *)n->data;
if (node->type & R_ANAL_ESIL_DFG_TAG_RESULT) {
RGraphNode *previous = (RGraphNode *)r_list_last (n->in_nodes);
if (!previous) {
return;
}
if (node->type & (R_ANAL_ESIL_DFG_TAG_REG | R_ANAL_ESIL_DFG_TAG_MEM)) {
node = (RAnalEsilDFGNode *)previous->data;
if (node->type & R_ANAL_ESIL_DFG_TAG_GENERATIVE) {
r_crbtree_insert (filter->tree, node, _dfg_node_filter_insert_cmp, NULL);
}
return;
}
sdb_ptr_set (filter->results, r_strbuf_get (node->content), previous, 0);
}
}
static void _dfg_filter_rev_dfs_cb(RGraphNode *n, RGraphVisitor *vi) {
_dfg_filter_rev_dfs (n, (RAnalEsilDFGFilter *)vi->data);
}
static void _dfg_const_reducer_rev_dfs_cb(RGraphNode *n, RGraphVisitor *vi) {
RAnalEsilDFGConstReducer *reducer = (RAnalEsilDFGConstReducer *)vi->data;
RAnalEsilDFGNode *enode = (RAnalEsilDFGNode *)n->data;
_dfg_filter_rev_dfs (n, &reducer->filter);
r_queue_enqueue (reducer->filter.dfg->todo, n);
if ((enode->type & R_ANAL_ESIL_DFG_TAG_LI_MASK) == (R_ANAL_ESIL_DFG_TAG_CONST | R_ANAL_ESIL_DFG_TAG_RESULT)) {
// n can only exist in the tree, if it is a const-result
r_crbtree_delete (reducer->const_result_gnodes, n, _dfg_gnode_reducer_insert_cmp, NULL);
}
}
static char *condrets_strtok(char *str, const char tok) {
if (!str) {
return NULL;
}
ut32 i = 0;
while (1 == 1) {
if (!str[i]) {
break;
}
if (str[i] == tok) {
str[i] = '\0';
return &str[i + 1];
}
i++;
}
return NULL;
}
static RStrBuf *get_resolved_expr(RAnalEsilDFGFilter *filter, RAnalEsilDFGNode *node) {
char *expr = strdup (r_strbuf_get (node->content));
RStrBuf *res = r_strbuf_new ("");
if (!expr) { //empty expressions. can this happen?
return res;
}
char *p, *q;
// we can do this bc every generative node MUST end with an operator
for (p = expr; (q = condrets_strtok (p, ',')); p = q) {
RGraphNode *gn = sdb_ptr_get (filter->results, p, 0);
if (!gn) {
r_strbuf_appendf (res, ",%s,", p);
} else {
bool cont = false;
char *c = p;
while (r_list_length (gn->out_nodes) == 2) {
// check if p refers to a node generated by DUP
RListIter *iter;
RGraphNode *outgn = NULL;
RAnalEsilDFGNode *sib = NULL;
r_list_foreach (gn->out_nodes, iter, outgn) {
RAnalEsilDFGNode *enode = (RAnalEsilDFGNode *)outgn->data;
if (enode->type & R_ANAL_ESIL_DFG_TAG_SIBLING) {
if (!strcmp (c, r_strbuf_get (enode->content))) {
node = enode;
} else {
sib = enode;
}
}
}
if (sib) {
if (sdb_ptr_get (filter->results, r_strbuf_get (sib->content), 0)) {
if (node->idx < sib->idx) {
cont = true;
break;
}
} else {
gn = (RGraphNode *)r_list_first (gn->in_nodes);
c = r_strbuf_get (((RAnalEsilDFGNode *)gn->data)->content);
gn = (RGraphNode *)r_list_first (gn->in_nodes);
}
}
}
if (cont) {
continue;
}
RStrBuf *r = get_resolved_expr (filter, (RAnalEsilDFGNode *)gn->data);
r_strbuf_appendf (res, ",%s,", r_strbuf_get (r));
r_strbuf_free (r);
}
}
r_strbuf_appendf (res, "%s", p);
free (expr);
return res;
}
static RStrBuf *filter_gnode_expr(RAnalEsilDFG *dfg, RGraphNode *gnode) { //TODO: find better name
RAnalEsilDFGFilter filter = { dfg, r_crbtree_new (NULL), sdb_new0 () };
RStrBuf *filtered = r_strbuf_new ("");
RGraphVisitor vi = { _dfg_filter_rev_dfs_cb, NULL, NULL, NULL, NULL, &filter };
RAnalEsilDFGNode *node = (RAnalEsilDFGNode *)gnode->data;
if ((node->type & (R_ANAL_ESIL_DFG_TAG_RESULT |
R_ANAL_ESIL_DFG_TAG_REG | R_ANAL_ESIL_DFG_TAG_MEM)) == R_ANAL_ESIL_DFG_TAG_RESULT) {
RGraphNode *previous = (RGraphNode *)r_list_last (gnode->in_nodes);
if (((RAnalEsilDFGNode *)previous->data)->type & R_ANAL_ESIL_DFG_TAG_GENERATIVE) {
r_crbtree_insert (filter.tree, previous->data, _dfg_node_filter_insert_cmp, NULL);
}
}
// reverse dfs the graph starting at node of register
r_graph_dfs_node_reverse (dfg->flow, gnode, &vi);
if (filter.tree->root) {
RRBNode *ator = r_crbtree_first_node (filter.tree);
while (ator) {
node = (RAnalEsilDFGNode *)ator->data;
// resolve results to opstr here
RStrBuf *resolved = get_resolved_expr (&filter, node);
r_strbuf_append (filtered, r_strbuf_get (resolved));
r_strbuf_free (resolved);
ator = r_rbnode_next (ator);
}
}
{
char *sanitized = r_str_replace (r_str_replace (strdup (r_strbuf_get (filtered)), ",,", ",", 1), ",,", ",", 1);
r_strbuf_set (filtered, (sanitized[0] == ',') ? &sanitized[1] : sanitized);
free (sanitized);
}
r_crbtree_free (filter.tree);
sdb_free (filter.results);
return filtered;
}
R_API void r_anal_esil_dfg_fold_const(RAnal *anal, RAnalEsilDFG *dfg) {
// sorted RRBTree for graph-nodes that contain edf-nodes with const-result as type
RAnalEsilDFGConstReducer reducer = { { dfg, NULL, NULL }, r_crbtree_new (NULL) };
RListIter *iter;
RGraphNode *gnode;
r_list_foreach (dfg->flow->nodes, iter, gnode) {
RAnalEsilDFGNode *enode = (RAnalEsilDFGNode *)gnode->data;
// insert const-result-nodes into the tree
// sort key is enode->idx
if (enode->type == (R_ANAL_ESIL_DFG_TAG_CONST | R_ANAL_ESIL_DFG_TAG_RESULT)) {
r_crbtree_insert (reducer.const_result_gnodes, gnode, _dfg_gnode_reducer_insert_cmp, NULL);
}
}
REsil *esil = r_esil_new (4096, 0, 1);
r_esil_setup (esil, anal, 1, 0, 0);
RGraphVisitor vi = { _dfg_const_reducer_rev_dfs_cb, NULL, NULL, NULL, NULL, &reducer };
RRBNode *first_node;
while ((first_node = r_crbtree_first_node (reducer.const_result_gnodes))) {
if (!first_node->data) {
break;
}
gnode = (RGraphNode *)first_node->data;
// filter, remove gnodes from const-result-tree
// during rdfs, run in esil, replace subtree with
// const-nodes and fix str-reference if outnode exists
// filter
reducer.filter.tree = r_crbtree_new (NULL);
reducer.filter.results = sdb_new0 (); // I guess this can be done better
r_graph_dfs_node_reverse (dfg->flow, gnode, &vi);
// ok, so gnode here cannot contain a generative node, only const-results
// get_resolved_expr expects a generative node
// the predecessor of a const-result node is always a generative node
RGraphNode *previous_gnode = (RGraphNode *)r_list_last (gnode->in_nodes);
// it can never be NULL
RAnalEsilDFGNode *enode = (RAnalEsilDFGNode *)previous_gnode->data;
RStrBuf *filtered = get_resolved_expr (&reducer.filter, enode);
{
char *sanitized = r_str_replace (r_str_replace (strdup (r_strbuf_get (filtered)), ",,", ",", 1), ",,", ",", 1);
r_strbuf_set (filtered, (sanitized[0] == ',') ? &sanitized[1] : sanitized);
free (sanitized);
}
r_crbtree_free (reducer.filter.tree);
sdb_free (reducer.filter.results);
// running filtered const-expression in esil
r_esil_parse (esil, r_strbuf_get (filtered));
char *reduced_const = r_esil_pop (esil);
r_strbuf_free (filtered);
// this part needs some explanation:
// in _dfg_const_reducer_rev_dfs_cb all nodes that are traversed during
// reverse dfs are enqueued in dfg->todo. reverse dfs in this context
// ALWAYS starts with a const-result-node. Result-nodes ALWAYS have a
// generative node as predecessor, which represent to operation.
// An operation in esil, with the exception of $z and some custom operations,
// have at least one operand, which is represented by at least one node in the dfg.
// As a consequence of this, we can safely dequeue 2 nodes from the dfg->todo
// without any checks and reuse them
gnode = (RGraphNode *)r_queue_dequeue (dfg->todo);
enode = (RAnalEsilDFGNode *)gnode->data;
RGraphNode *next_gnode = (RGraphNode *)r_list_last (gnode->out_nodes);
if (next_gnode) {
// Cannot assume that there is another operation
// Fix string reference
RAnalEsilDFGNode *next_enode = (RAnalEsilDFGNode *)next_gnode->data;
char *fixed = r_str_replace (strdup (r_strbuf_get (next_enode->content)),
r_strbuf_get (enode->content), reduced_const, 0);
r_strbuf_set (next_enode->content, fixed);
free (fixed);
}
// replace subtree with const-nodes
r_strbuf_setf (enode->content, "%s:const_%d", reduced_const, enode->idx);
gnode = (RGraphNode *)r_queue_dequeue (dfg->todo);
enode = (RAnalEsilDFGNode *)gnode->data;
r_strbuf_set (enode->content, reduced_const);
free (reduced_const);
while (!r_queue_is_empty (dfg->todo)) {
gnode = (RGraphNode *)r_queue_dequeue (dfg->todo);
enode = (RAnalEsilDFGNode *)gnode->data;
_dfg_node_free (enode);
r_graph_del_node (dfg->flow, gnode);
}
}
r_esil_free (esil);
r_crbtree_free (reducer.const_result_gnodes);
}
R_API RStrBuf *r_anal_esil_dfg_filter(RAnalEsilDFG *dfg, const char *reg) {
R_RETURN_VAL_IF_FAIL (dfg && reg, NULL);
RGraphNode *resolve_me = _edf_reg_get (dfg, reg);
return resolve_me? filter_gnode_expr (dfg, resolve_me): NULL;
}
R_API RStrBuf *r_anal_esil_dfg_filter_expr(RAnal *anal, const char *expr, const char *reg,
bool use_map_info, bool use_maps) {
R_RETURN_VAL_IF_FAIL (anal && expr && reg, NULL);
RAnalEsilDFG *dfg = r_anal_esil_dfg_expr (anal, NULL, expr, use_map_info, use_maps);
if (!dfg) {
return NULL;
}
RStrBuf *filtered = r_anal_esil_dfg_filter (dfg, reg);
r_anal_esil_dfg_free (dfg);
return filtered;
}
R_API bool r_anal_esil_dfg_reg_is_const(RAnalEsilDFG *dfg, const char *reg) {
R_RETURN_VAL_IF_FAIL (dfg && reg, false);
char *_reg = r_str_newf ("reg.%s", reg);
if (!sdb_num_exists (dfg->regs, _reg)) {
// reg is actually not part of the current reg-profile
free (_reg);
return false;
}
EsilDFGVar *rv = R_NEW0 (EsilDFGVar);
if (!rv) {
free (_reg);
eprintf ("Allocation failed\n");
return false;
}
const ut64 v = sdb_num_get (dfg->regs, _reg, NULL);
free (_reg);
rv->from = (v & (UT64_MAX ^ UT32_MAX)) >> 32;
rv->to = v & UT32_MAX;
r_queue_enqueue (dfg->todo, rv);
while (!r_queue_is_empty (dfg->todo)) {
rv = r_queue_dequeue (dfg->todo);
EsilDFGVar *part_rv = r_crbtree_find (dfg->vars, rv, _rv_find_cmp, dfg);
R_FREE (rv);
if (part_rv) {
RAnalEsilDFGNode *edf_node = (RAnalEsilDFGNode *)part_rv->node->data;
if (!edf_node) {
eprintf ("edf_node is NULL\n");
goto beach;
}
if (!(edf_node->type & R_ANAL_ESIL_DFG_TAG_CONST)) {
goto beach;
}
} else {
if (dfg->malloc_failed) {
eprintf ("Allocation failed\n");
}
goto beach;
}
}
return true;
beach:
while (!r_queue_is_empty (dfg->todo)) {
free (r_queue_dequeue (dfg->todo));
}
return false;
}