/* radare - Apache 2.0 - Copyright 2013 - Adam Pridgen */ #include #include #include #include #include "../config.h" #ifdef IFDBG #undef IFDBG #endif #define DO_THE_DBG 0 #define IFDBG if(DO_THE_DBG) #define IFINT if(0) static void kv_anal_bb_free(HtKv *kv) { if (kv) { free (kv->key); r_anal_bb_free (kv->value); free (kv); } } R_API RAnalState * r_anal_state_new(ut64 start, ut8* buffer, ut64 len) { RAnalState *state = R_NEW0 (RAnalState); if (!state) { return NULL; } state->start = start; state->end = start + len; state->buffer = buffer; state->len = len; state->current_op = NULL; state->current_bb = NULL; state->current_fcn = NULL; state->ht = ht_new (NULL, (HtKvFreeFunc)kv_anal_bb_free, NULL); state->ht_sz = 512; state->bbs = r_list_newf ((RListFree)r_anal_bb_free); state->max_depth = 50; state->current_depth = 0; return state; } R_API void r_anal_state_set_depth(RAnalState *state, ut32 depth) { state->current_depth = depth; } R_API void r_anal_state_insert_bb(RAnalState* state, RAnalBlock *bb) { if (!state || !bb) { return; } if (!r_anal_state_search_bb (state, bb->addr) && state->current_fcn) { r_list_append (state->current_fcn->bbs, bb); state->bytes_consumed += state->current_bb->op_sz; const char *key = sdb_fmt (0, "0x%08"PFMT64x, bb->addr); if (!ht_insert (state->ht, key, bb)) { eprintf ("Inserted bb 0x%04"PFMT64x" failure\n", bb->addr); } } } R_API RAnalBlock * r_anal_state_search_bb(RAnalState* state, ut64 addr) { /* * Return 0 if no rehash is needed, otherwise return 1 */ const char *key = sdb_fmt (0, "0x%08"PFMT64x, addr); RAnalBlock *tmp_bb = ht_find (state->ht, key, NULL); return tmp_bb; } R_API void r_anal_state_free(RAnalState * state) { r_list_free (state->bbs); ht_free (state->ht); free (state); } R_API ut64 r_anal_state_get_len(RAnalState *state, ut64 addr) { ut64 result = 0; if (r_anal_state_addr_is_valid (state, addr)) { result = state->len - (addr - state->start); } return result; } R_API const ut8 * r_anal_state_get_buf_by_addr(RAnalState *state, ut64 addr) { if (r_anal_state_addr_is_valid (state, addr)) { ut64 offset = addr - state->start; return state->buffer + offset; } return NULL; } R_API bool r_anal_state_addr_is_valid(RAnalState *state, ut64 addr) { return (addr < state->end && addr >= state->start); } R_API void r_anal_state_merge_bb_list (RAnalState *state, RList* bbs) { RListIter *iter; RAnalBlock *bb; r_list_foreach (bbs, iter, bb) { IFDBG eprintf ("Inserting bb from 0x%04"PFMT64x"\n", bb->addr); r_anal_state_insert_bb (state, bb); } }