2017-11-23 21:47:45 +01:00
|
|
|
/* radare - LGPL - Copyright 2017 - pancake, defragger */
|
2017-02-10 00:02:39 +01:00
|
|
|
|
|
|
|
#include <r_core.h>
|
|
|
|
|
2017-03-30 10:53:40 +02:00
|
|
|
typedef enum bb_type {
|
|
|
|
TRAP,
|
|
|
|
NORMAL,
|
|
|
|
JUMP,
|
|
|
|
FAIL,
|
|
|
|
CALL,
|
2017-04-15 16:45:16 +02:00
|
|
|
END,
|
2017-03-30 10:53:40 +02:00
|
|
|
} bb_type_t;
|
|
|
|
|
|
|
|
typedef struct bb {
|
|
|
|
ut64 start;
|
|
|
|
ut64 end;
|
|
|
|
ut64 jump;
|
|
|
|
ut64 fail;
|
|
|
|
int score;
|
|
|
|
int called;
|
|
|
|
int reached;
|
|
|
|
bb_type_t type;
|
|
|
|
} bb_t;
|
|
|
|
|
2017-04-15 16:45:16 +02:00
|
|
|
typedef struct fcn {
|
|
|
|
ut64 addr;
|
|
|
|
ut64 size;
|
|
|
|
RList *bbs;
|
|
|
|
st64 score;
|
|
|
|
ut64 ends;
|
|
|
|
} fcn_t;
|
|
|
|
|
2020-01-15 10:58:30 +01:00
|
|
|
static bool __is_data_block_cb(RAnalBlock *block, void *user) {
|
|
|
|
bool *block_exists = user;
|
|
|
|
*block_exists = true;
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2019-09-03 00:00:06 +02:00
|
|
|
static int __isdata(RCore *core, ut64 addr) {
|
|
|
|
if (!r_io_is_valid_offset (core->io, addr, false)) {
|
|
|
|
// eprintf ("Warning: Invalid memory address at 0x%08"PFMT64x"\n", addr);
|
|
|
|
return 4;
|
|
|
|
}
|
2020-01-15 10:58:30 +01:00
|
|
|
|
|
|
|
bool block_exists = false;
|
|
|
|
// This will just set block_exists = true if there is any basic block at this addr
|
|
|
|
r_anal_blocks_foreach_in (core->anal, addr, __is_data_block_cb, &block_exists);
|
|
|
|
if (block_exists) {
|
|
|
|
return 1;
|
2019-09-03 00:00:06 +02:00
|
|
|
}
|
2020-01-15 10:58:30 +01:00
|
|
|
|
2019-09-03 00:00:06 +02:00
|
|
|
RList *list = r_meta_find_list_in (core->anal, addr, -1, 4);
|
|
|
|
RListIter *iter;
|
|
|
|
RAnalMetaItem *meta;
|
2019-09-07 20:24:14 +03:00
|
|
|
int result = 0;
|
2019-09-03 00:00:06 +02:00
|
|
|
r_list_foreach (list, iter, meta) {
|
|
|
|
switch (meta->type) {
|
|
|
|
case R_META_TYPE_DATA:
|
|
|
|
case R_META_TYPE_STRING:
|
|
|
|
case R_META_TYPE_FORMAT:
|
2019-09-07 20:24:14 +03:00
|
|
|
result = meta->size - (addr - meta->from);
|
|
|
|
goto exit;
|
2019-09-03 00:00:06 +02:00
|
|
|
}
|
|
|
|
}
|
2019-09-07 20:24:14 +03:00
|
|
|
exit:
|
2019-09-03 00:00:06 +02:00
|
|
|
r_list_free (list);
|
2019-09-07 20:24:14 +03:00
|
|
|
return result;
|
2019-09-03 00:00:06 +02:00
|
|
|
}
|
|
|
|
|
2017-04-15 16:45:16 +02:00
|
|
|
static bool fcnAddBB (fcn_t *fcn, bb_t* block) {
|
|
|
|
if (!fcn) {
|
|
|
|
eprintf ("No function given to add a basic block\n");
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
fcn->score += block->score;
|
2017-11-23 21:47:45 +01:00
|
|
|
fcn->size += block->end - block->start;
|
2017-04-15 16:45:16 +02:00
|
|
|
if (block->type == END) {
|
|
|
|
fcn->ends++;
|
|
|
|
}
|
|
|
|
if (!fcn->bbs) {
|
|
|
|
eprintf ("Block list not initialized\n");
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
r_list_append (fcn->bbs, block);
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
static fcn_t* fcnNew (bb_t *block) {
|
|
|
|
fcn_t* fcn = R_NEW0 (fcn_t);
|
|
|
|
if (!fcn) {
|
|
|
|
eprintf ("Failed to allocate memory for function\n");
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
fcn->addr = block->start;
|
|
|
|
fcn->bbs = r_list_new ();
|
|
|
|
if (!fcnAddBB (fcn, block)) {
|
|
|
|
eprintf ("Failed to add block to function\n");
|
|
|
|
}
|
|
|
|
return fcn;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void fcnFree (fcn_t *fcn) {
|
|
|
|
r_list_free (fcn->bbs);
|
|
|
|
free (fcn);
|
|
|
|
}
|
|
|
|
|
|
|
|
static int bbCMP (void *_a, void *_b) {
|
2017-03-30 10:53:40 +02:00
|
|
|
bb_t *a = (bb_t*)_a;
|
|
|
|
bb_t *b = (bb_t*)_b;
|
|
|
|
return b->start - a->start;
|
|
|
|
}
|
|
|
|
|
2017-04-15 16:45:16 +02:00
|
|
|
static void initBB (bb_t *bb, ut64 start, ut64 end, ut64 jump, ut64 fail, bb_type_t type, int score, int reached, int called) {
|
2017-03-30 10:53:40 +02:00
|
|
|
if (bb) {
|
|
|
|
bb->start = start;
|
|
|
|
bb->end = end;
|
|
|
|
bb->jump = jump;
|
|
|
|
bb->fail = fail;
|
|
|
|
bb->type = type;
|
|
|
|
bb->score = score;
|
|
|
|
bb->reached = reached;
|
|
|
|
bb->called = called;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-09-03 00:00:06 +02:00
|
|
|
static bool addBB(RList *block_list, ut64 start, ut64 end, ut64 jump, ut64 fail, bb_type_t type, int score) {
|
2017-03-30 10:53:40 +02:00
|
|
|
bb_t *bb = (bb_t*) R_NEW0 (bb_t);
|
|
|
|
if (!bb) {
|
|
|
|
eprintf ("Failed to calloc mem for new basic block!\n");
|
|
|
|
return false;
|
|
|
|
}
|
2017-04-15 16:45:16 +02:00
|
|
|
initBB (bb, start, end, jump, fail, type, score, 0, 0);
|
2017-03-30 10:53:40 +02:00
|
|
|
if (jump < UT64_MAX) {
|
|
|
|
bb_t *jump_bb = (bb_t*) R_NEW0 (bb_t);
|
|
|
|
if (!jump_bb) {
|
|
|
|
eprintf ("Failed to allocate memory for jump block\n");
|
|
|
|
free (bb);
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
if (type == CALL) {
|
2017-04-15 16:45:16 +02:00
|
|
|
initBB (jump_bb, jump, UT64_MAX, UT64_MAX, UT64_MAX, CALL, 0, 1, 1);
|
2017-03-30 10:53:40 +02:00
|
|
|
} else {
|
2017-04-15 16:45:16 +02:00
|
|
|
initBB (jump_bb, jump, UT64_MAX, UT64_MAX, UT64_MAX, JUMP, 0, 1, 0);
|
2017-03-30 10:53:40 +02:00
|
|
|
}
|
|
|
|
r_list_append (block_list, jump_bb);
|
|
|
|
}
|
|
|
|
if (fail < UT64_MAX) {
|
|
|
|
bb_t *fail_bb = (bb_t*) R_NEW0 (bb_t);
|
|
|
|
if (!fail_bb) {
|
|
|
|
eprintf ("Failed to allocate memory for fail block\n");
|
|
|
|
free (bb);
|
|
|
|
return false;
|
|
|
|
}
|
2017-04-15 16:45:16 +02:00
|
|
|
initBB (fail_bb, fail, UT64_MAX, UT64_MAX, UT64_MAX, FAIL, 0, 1, 0);
|
2017-03-30 10:53:40 +02:00
|
|
|
r_list_append (block_list, fail_bb);
|
|
|
|
}
|
|
|
|
r_list_append (block_list, bb);
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
void dump_block(bb_t *block) {
|
2017-04-15 16:45:16 +02:00
|
|
|
eprintf ("s: 0x%"PFMT64x" e: 0x%"PFMT64x" j: 0x%"PFMT64x" f: 0x%"PFMT64x" t: %d\n"
|
|
|
|
, block->start, block->end, block->jump, block->fail, block->type);
|
2017-03-30 10:53:40 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
void dump_blocks (RList* list) {
|
|
|
|
RListIter *iter;
|
|
|
|
bb_t *block = NULL;
|
|
|
|
r_list_foreach (list, iter, block) {
|
|
|
|
dump_block(block);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-06-06 18:28:43 +02:00
|
|
|
static bool checkFunction(fcn_t *fcn) {
|
2017-11-23 21:47:45 +01:00
|
|
|
if (fcn && fcn->ends > 0 && fcn->size > 0) {
|
2017-06-06 18:28:43 +02:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void printFunctionCommands(RCore *core, fcn_t* fcn, const char *name) {
|
|
|
|
if (!fcn) {
|
|
|
|
eprintf ("No function given to print\n");
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
RListIter *fcn_iter;
|
|
|
|
bb_t *cur = NULL;
|
|
|
|
const char *pfx = r_config_get (core->config, "anal.fcnprefix");
|
|
|
|
if (!pfx) {
|
|
|
|
pfx = "fcn";
|
|
|
|
}
|
|
|
|
|
|
|
|
char *_name = name? (char *) name: r_str_newf ("%s.%" PFMT64x, pfx, fcn->addr);
|
|
|
|
r_cons_printf ("af+ 0x%08" PFMT64x " %s\n", fcn->addr, _name);
|
|
|
|
if (!name) {
|
|
|
|
free (_name);
|
|
|
|
}
|
|
|
|
|
|
|
|
r_list_foreach (fcn->bbs, fcn_iter, cur) {
|
|
|
|
r_cons_printf ("afb+ 0x%08" PFMT64x " 0x%08" PFMT64x " %llu 0x%08"PFMT64x" 0x%08"PFMT64x"\n"
|
|
|
|
, fcn->addr, cur->start, cur->end - cur->start, cur->jump, cur->fail);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static void createFunction(RCore *core, fcn_t* fcn, const char *name) {
|
|
|
|
if (!fcn) {
|
|
|
|
eprintf ("No function given to create\n");
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
RListIter *fcn_iter;
|
|
|
|
bb_t *cur = NULL;
|
|
|
|
const char *pfx = r_config_get (core->config, "anal.fcnprefix");
|
|
|
|
if (!pfx) {
|
|
|
|
pfx = "fcn";
|
|
|
|
}
|
|
|
|
|
2020-01-15 10:58:30 +01:00
|
|
|
RAnalFunction *f = r_anal_function_new (core->anal);
|
2017-06-06 18:28:43 +02:00
|
|
|
if (!f) {
|
|
|
|
eprintf ("Failed to create new function\n");
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2018-01-03 02:28:15 +01:00
|
|
|
f->name = name? strdup (name): r_str_newf ("%s.%" PFMT64x, pfx, fcn->addr);
|
2017-06-06 18:28:43 +02:00
|
|
|
f->addr = fcn->addr;
|
|
|
|
f->bits = core->anal->bits;
|
2019-10-20 12:31:45 +02:00
|
|
|
f->cc = r_str_constpool_get (&core->anal->constpool, r_anal_cc_default (core->anal));
|
2017-06-06 18:28:43 +02:00
|
|
|
f->type = R_ANAL_FCN_TYPE_FCN;
|
|
|
|
|
|
|
|
r_list_foreach (fcn->bbs, fcn_iter, cur) {
|
2019-09-03 00:00:06 +02:00
|
|
|
if (__isdata (core, cur->start)) {
|
|
|
|
continue;
|
|
|
|
}
|
2020-02-14 17:50:10 +01:00
|
|
|
r_anal_fcn_add_bb (core->anal, f, cur->start, (cur->end - cur->start), cur->jump, cur->fail, NULL);
|
2017-06-06 18:28:43 +02:00
|
|
|
}
|
2020-01-15 10:58:30 +01:00
|
|
|
if (!r_anal_add_function (core->anal, f)) {
|
2018-03-15 00:46:12 +01:00
|
|
|
// eprintf ("Failed to insert function\n");
|
2020-01-15 10:58:30 +01:00
|
|
|
r_anal_function_free (f);
|
2017-06-06 18:28:43 +02:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-04-10 23:52:47 +02:00
|
|
|
#define Fhandled(x) sdb_fmt("handled.%"PFMT64x"", x)
|
2017-06-06 18:28:43 +02:00
|
|
|
R_API bool core_anal_bbs(RCore *core, const char* input) {
|
2017-08-23 22:19:57 +00:00
|
|
|
if (!r_io_is_valid_offset (core->io, core->offset, false)) {
|
2017-04-15 16:45:16 +02:00
|
|
|
eprintf ("No valid offset given to analyze\n");
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
Sdb *sdb = NULL;
|
2017-03-30 10:53:40 +02:00
|
|
|
ut64 cur = 0;
|
|
|
|
ut64 start = core->offset;
|
2017-08-18 06:05:47 -07:00
|
|
|
ut64 size = input[0] ? r_num_math (core->num, input + 1) : core->blocksize;
|
2017-03-30 10:53:40 +02:00
|
|
|
ut64 b_start = start;
|
|
|
|
RAnalOp *op;
|
|
|
|
RListIter *iter;
|
|
|
|
int block_score = 0;
|
|
|
|
RList *block_list;
|
|
|
|
bb_t *block = NULL;
|
2017-04-15 16:45:16 +02:00
|
|
|
int invalid_instruction_barrier = -20000;
|
2017-11-23 21:47:45 +01:00
|
|
|
bool debug = r_config_get_i (core->config, "cfg.debug");
|
2018-03-07 11:03:17 +01:00
|
|
|
bool nopskip = r_config_get_i (core->config, "anal.nopskip");
|
2017-03-30 10:53:40 +02:00
|
|
|
|
2017-04-15 16:45:16 +02:00
|
|
|
block_list = r_list_new ();
|
|
|
|
if (!block_list) {
|
|
|
|
eprintf ("Failed to create block_list\n");
|
|
|
|
}
|
2017-03-30 10:53:40 +02:00
|
|
|
|
2017-11-23 21:47:45 +01:00
|
|
|
if (debug) {
|
|
|
|
eprintf ("Analyzing [0x%08"PFMT64x"-0x%08"PFMT64x"]\n", start, start + size);
|
|
|
|
eprintf ("Creating basic blocks\b");
|
|
|
|
}
|
2019-09-03 00:00:06 +02:00
|
|
|
ut64 base = cur;
|
|
|
|
while (cur >= base && cur < size) {
|
2017-07-10 22:45:24 +02:00
|
|
|
if (r_cons_is_breaked ()) {
|
|
|
|
break;
|
|
|
|
}
|
2017-04-15 16:45:16 +02:00
|
|
|
// magic number to fix huge section of invalid code fuzz files
|
|
|
|
if (block_score < invalid_instruction_barrier) {
|
|
|
|
break;
|
|
|
|
}
|
2019-09-03 00:00:06 +02:00
|
|
|
ut64 dst = start + cur;
|
|
|
|
if (dst < start) {
|
|
|
|
// fix underflow issue
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
int dsize = __isdata (core, dst);
|
|
|
|
if (dsize > 0) {
|
|
|
|
cur += dsize;
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
op = r_core_anal_op (core, dst, R_ANAL_OP_MASK_BASIC | R_ANAL_OP_MASK_DISASM);
|
2017-03-30 10:53:40 +02:00
|
|
|
|
|
|
|
if (!op || !op->mnemonic) {
|
|
|
|
block_score -= 10;
|
2017-04-15 16:45:16 +02:00
|
|
|
cur++;
|
2017-03-30 10:53:40 +02:00
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (op->mnemonic[0] == '?') {
|
2017-11-23 22:37:44 +01:00
|
|
|
eprintf ("? Bad op at: 0x%08"PFMT64x"\n", cur + start);
|
2019-03-27 10:19:18 +01:00
|
|
|
eprintf ("Cannot analyze opcode at 0x%"PFMT64x"\n", start + cur);
|
2017-03-30 10:53:40 +02:00
|
|
|
block_score -= 10;
|
2017-04-15 16:45:16 +02:00
|
|
|
cur++;
|
2017-03-30 10:53:40 +02:00
|
|
|
continue;
|
|
|
|
}
|
|
|
|
switch (op->type) {
|
|
|
|
case R_ANAL_OP_TYPE_NOP:
|
2018-03-07 11:03:17 +01:00
|
|
|
if (nopskip && b_start == start + cur) {
|
|
|
|
b_start = start + cur + op->size;
|
|
|
|
}
|
2017-03-30 10:53:40 +02:00
|
|
|
break;
|
|
|
|
case R_ANAL_OP_TYPE_CALL:
|
2017-05-23 10:13:51 +02:00
|
|
|
if (r_anal_noreturn_at (core->anal, op->jump)) {
|
|
|
|
addBB (block_list, b_start, start + cur + op->size, UT64_MAX, UT64_MAX, END, block_score);
|
|
|
|
b_start = start + cur + op->size;
|
|
|
|
block_score = 0;
|
|
|
|
} else {
|
|
|
|
addBB (block_list, op->jump, UT64_MAX, UT64_MAX, UT64_MAX, CALL, block_score);
|
|
|
|
}
|
2017-03-30 10:53:40 +02:00
|
|
|
break;
|
|
|
|
case R_ANAL_OP_TYPE_JMP:
|
2017-04-15 16:45:16 +02:00
|
|
|
addBB (block_list, b_start, start + cur + op->size, op->jump, UT64_MAX, END, block_score);
|
2017-03-30 10:53:40 +02:00
|
|
|
b_start = start + cur + op->size;
|
|
|
|
block_score = 0;
|
|
|
|
break;
|
|
|
|
case R_ANAL_OP_TYPE_TRAP:
|
2019-06-20 12:45:00 +08:00
|
|
|
// we don't want to add trap stuff
|
2017-03-30 10:53:40 +02:00
|
|
|
if (b_start < start + cur) {
|
2017-04-15 16:45:16 +02:00
|
|
|
addBB (block_list, b_start, start + cur, UT64_MAX, UT64_MAX, NORMAL, block_score);
|
2017-03-30 10:53:40 +02:00
|
|
|
}
|
|
|
|
b_start = start + cur + op->size;
|
|
|
|
block_score = 0;
|
|
|
|
break;
|
|
|
|
case R_ANAL_OP_TYPE_RET:
|
2017-04-15 16:45:16 +02:00
|
|
|
addBB (block_list, b_start, start + cur + op->size, UT64_MAX, UT64_MAX, END, block_score);
|
2017-03-30 10:53:40 +02:00
|
|
|
b_start = start + cur + op->size;
|
|
|
|
block_score = 0;
|
|
|
|
break;
|
|
|
|
case R_ANAL_OP_TYPE_CJMP:
|
2017-04-15 16:45:16 +02:00
|
|
|
addBB (block_list, b_start, start + cur + op->size, op->jump, start + cur + op->size, NORMAL, block_score);
|
2017-03-30 10:53:40 +02:00
|
|
|
b_start = start + cur + op->size;
|
|
|
|
block_score = 0;
|
|
|
|
break;
|
|
|
|
case R_ANAL_OP_TYPE_UNK:
|
|
|
|
case R_ANAL_OP_TYPE_ILL:
|
|
|
|
block_score -= 10;
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
cur += op->size;
|
|
|
|
r_anal_op_free (op);
|
|
|
|
op = NULL;
|
|
|
|
}
|
|
|
|
|
2017-11-23 21:47:45 +01:00
|
|
|
if (debug) {
|
|
|
|
eprintf ("Found %d basic blocks\n", block_list->length);
|
|
|
|
}
|
|
|
|
|
2017-03-30 10:53:40 +02:00
|
|
|
RList *result = r_list_newf (free);
|
|
|
|
if (!result) {
|
2017-04-15 16:45:16 +02:00
|
|
|
r_list_free (block_list);
|
2017-03-30 10:53:40 +02:00
|
|
|
eprintf ("Failed to create resulting list\n");
|
2017-04-15 16:45:16 +02:00
|
|
|
return false;
|
2017-03-30 10:53:40 +02:00
|
|
|
}
|
|
|
|
|
2017-04-15 16:45:16 +02:00
|
|
|
sdb = sdb_new0 ();
|
|
|
|
if (!sdb) {
|
|
|
|
eprintf ("Failed to initialize sdb db\n");
|
|
|
|
r_list_free (block_list);
|
|
|
|
r_list_free (result);
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
r_list_sort (block_list, (RListComparator)bbCMP);
|
|
|
|
|
2017-11-23 21:47:45 +01:00
|
|
|
if (debug) {
|
|
|
|
eprintf ("Sorting all blocks done\n");
|
|
|
|
eprintf ("Creating the complete graph\n");
|
|
|
|
}
|
|
|
|
|
2017-03-30 10:53:40 +02:00
|
|
|
while (block_list->length > 0) {
|
|
|
|
block = r_list_pop (block_list);
|
|
|
|
if (!block) {
|
|
|
|
eprintf ("Failed to get next block from list\n");
|
|
|
|
continue;
|
|
|
|
}
|
2017-07-10 22:45:24 +02:00
|
|
|
if (r_cons_is_breaked ()) {
|
|
|
|
break;
|
|
|
|
}
|
2017-03-30 10:53:40 +02:00
|
|
|
|
|
|
|
if (block_list->length > 0) {
|
|
|
|
bb_t *next_block = (bb_t*) r_list_iter_get_data (block_list->tail);
|
|
|
|
if (!next_block) {
|
|
|
|
eprintf ("No next block to compare with!\n");
|
|
|
|
}
|
|
|
|
|
|
|
|
// current block is just a split block
|
|
|
|
if (block->start == next_block->start && block->end == UT64_MAX) {
|
|
|
|
if (block->type != CALL && next_block->type != CALL) {
|
2017-04-15 16:45:16 +02:00
|
|
|
next_block->reached = block->reached + 1;
|
2017-03-30 10:53:40 +02:00
|
|
|
}
|
|
|
|
free (block);
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
// block and next_block share the same start so we copy the
|
|
|
|
// contenct of the block into the next_block and skip the current one
|
|
|
|
if (block->start == next_block->start && next_block->end == UT64_MAX) {
|
|
|
|
if (next_block->type != CALL) {
|
2017-04-15 16:45:16 +02:00
|
|
|
block->reached += 1;
|
2017-03-30 10:53:40 +02:00
|
|
|
}
|
2017-04-15 16:45:16 +02:00
|
|
|
*next_block = *block;
|
2017-03-30 10:53:40 +02:00
|
|
|
free (block);
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (block->end < UT64_MAX && next_block->start < block->end && next_block->start > block->start) {
|
|
|
|
if (next_block->jump == UT64_MAX) {
|
|
|
|
next_block->jump = block->jump;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (next_block->fail == UT64_MAX) {
|
|
|
|
next_block->fail = block->fail;
|
|
|
|
}
|
|
|
|
|
|
|
|
next_block->end = block->end;
|
|
|
|
block->end = next_block->start;
|
|
|
|
block->jump = next_block->start;
|
|
|
|
block->fail = UT64_MAX;
|
|
|
|
next_block->type = block->type;
|
|
|
|
if (next_block->type != CALL) {
|
|
|
|
next_block->reached += 1;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-04-10 23:52:47 +02:00
|
|
|
sdb_ptr_set (sdb, sdb_fmt ("bb.0x%08"PFMT64x, block->start), block, 0);
|
2017-04-15 16:45:16 +02:00
|
|
|
r_list_append (result, block);
|
2017-03-30 10:53:40 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
// finally search for functions
|
|
|
|
// we simply assume that non reached blocks or called blocks
|
|
|
|
// are functions
|
2017-11-23 21:47:45 +01:00
|
|
|
if (debug) {
|
|
|
|
eprintf ("Trying to create functions\n");
|
|
|
|
}
|
|
|
|
|
2017-03-30 10:53:40 +02:00
|
|
|
r_list_foreach (result, iter, block) {
|
2017-07-10 22:45:24 +02:00
|
|
|
if (r_cons_is_breaked ()) {
|
|
|
|
break;
|
|
|
|
}
|
2017-03-30 10:53:40 +02:00
|
|
|
if (block && (block->reached == 0 || block->called >= 1)) {
|
2017-04-15 16:45:16 +02:00
|
|
|
fcn_t* current_function = fcnNew (block);
|
|
|
|
RStack *stack = r_stack_new (100);
|
2017-03-30 10:53:40 +02:00
|
|
|
bb_t *jump = NULL;
|
|
|
|
bb_t *fail = NULL;
|
2017-04-15 16:45:16 +02:00
|
|
|
bb_t *cur = NULL;
|
2017-03-30 10:53:40 +02:00
|
|
|
|
|
|
|
if (!r_stack_push (stack, (void*)block)) {
|
|
|
|
eprintf ("Failed to push initial block\n");
|
|
|
|
}
|
|
|
|
|
|
|
|
while (!r_stack_is_empty (stack)) {
|
|
|
|
cur = (bb_t*) r_stack_pop (stack);
|
|
|
|
if (!cur) {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
sdb_num_set (sdb, Fhandled(cur->start), 1, 0);
|
2017-04-15 16:45:16 +02:00
|
|
|
if (cur->score < 0) {
|
2017-11-23 21:47:45 +01:00
|
|
|
fcnFree (current_function);
|
|
|
|
current_function = NULL;
|
2017-04-15 16:45:16 +02:00
|
|
|
break;
|
|
|
|
}
|
2017-03-30 10:53:40 +02:00
|
|
|
// we ignore negative blocks
|
|
|
|
if ((st64)(cur->end - cur->start) < 0) {
|
2017-04-15 16:45:16 +02:00
|
|
|
break;
|
2017-03-30 10:53:40 +02:00
|
|
|
}
|
2017-04-15 16:45:16 +02:00
|
|
|
|
|
|
|
fcnAddBB (current_function, cur);
|
2017-03-30 10:53:40 +02:00
|
|
|
|
|
|
|
if (cur->jump < UT64_MAX && !sdb_num_get (sdb, Fhandled(cur->jump), NULL)) {
|
2018-04-10 23:52:47 +02:00
|
|
|
jump = sdb_ptr_get (sdb, sdb_fmt ("bb.0x%08"PFMT64x, cur->jump), NULL);
|
2017-03-30 10:53:40 +02:00
|
|
|
if (!jump) {
|
2017-04-15 16:45:16 +02:00
|
|
|
eprintf ("Failed to get jump block at 0x%"PFMT64x"\n", cur->jump);
|
2017-03-30 10:53:40 +02:00
|
|
|
continue;
|
|
|
|
}
|
|
|
|
if (!r_stack_push (stack, (void*)jump)) {
|
|
|
|
eprintf ("Failed to push jump block to stack\n");
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (cur->fail < UT64_MAX && !sdb_num_get (sdb, Fhandled(cur->fail), NULL)) {
|
2018-04-10 23:52:47 +02:00
|
|
|
fail = sdb_ptr_get (sdb, sdb_fmt ("bb.0x%08" PFMT64x, cur->fail), NULL);
|
2017-03-30 10:53:40 +02:00
|
|
|
if (!fail) {
|
2017-04-15 16:45:16 +02:00
|
|
|
eprintf ("Failed to get fail block at 0x%"PFMT64x"\n", cur->fail);
|
2017-03-30 10:53:40 +02:00
|
|
|
continue;
|
|
|
|
}
|
|
|
|
if (!r_stack_push (stack, (void*)fail)) {
|
|
|
|
eprintf ("Failed to push jump block to stack\n");
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2017-04-15 16:45:16 +02:00
|
|
|
|
2017-06-06 18:28:43 +02:00
|
|
|
// function creation complete
|
2017-11-23 21:47:45 +01:00
|
|
|
if (current_function) {
|
|
|
|
if (checkFunction (current_function)) {
|
|
|
|
if (input[0] == '*') {
|
|
|
|
printFunctionCommands (core, current_function, NULL);
|
|
|
|
} else {
|
|
|
|
createFunction (core, current_function, NULL);
|
|
|
|
}
|
2017-04-15 16:45:16 +02:00
|
|
|
}
|
2017-11-23 21:47:45 +01:00
|
|
|
fcnFree (current_function);
|
2017-04-15 16:45:16 +02:00
|
|
|
}
|
2017-06-06 18:28:43 +02:00
|
|
|
|
2017-03-30 10:53:40 +02:00
|
|
|
r_stack_free (stack);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-04-15 16:45:16 +02:00
|
|
|
sdb_free (sdb);
|
2017-04-05 10:41:19 +02:00
|
|
|
r_list_free (result);
|
2017-04-15 16:45:16 +02:00
|
|
|
r_list_free (block_list);
|
2017-03-30 10:53:40 +02:00
|
|
|
return true;
|
|
|
|
}
|
2018-01-31 17:59:05 +01:00
|
|
|
|
|
|
|
R_API bool core_anal_bbs_range (RCore *core, const char* input) {
|
|
|
|
if (!r_io_is_valid_offset (core->io, core->offset, false)) {
|
|
|
|
eprintf ("No valid offset given to analyze\n");
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
Sdb *sdb = NULL;
|
|
|
|
ut64 cur = 0;
|
|
|
|
ut64 start = core->offset;
|
|
|
|
ut64 size = input[0] ? r_num_math (core->num, input + 1) : core->blocksize;
|
|
|
|
ut64 b_start = start;
|
|
|
|
RAnalOp *op;
|
|
|
|
RListIter *iter;
|
|
|
|
int block_score = 0;
|
|
|
|
RList *block_list;
|
|
|
|
bb_t *block = NULL;
|
|
|
|
int invalid_instruction_barrier = -20000;
|
|
|
|
bool debug = r_config_get_i (core->config, "cfg.debug");
|
|
|
|
ut64 lista[1024] = { 0 };
|
|
|
|
int idx = 0;
|
2018-02-01 16:35:08 +01:00
|
|
|
int x;
|
2018-01-31 17:59:05 +01:00
|
|
|
|
|
|
|
block_list = r_list_new ();
|
|
|
|
if (!block_list) {
|
|
|
|
eprintf ("Failed to create block_list\n");
|
|
|
|
}
|
|
|
|
if (debug) {
|
|
|
|
eprintf ("Analyzing [0x%08"PFMT64x"-0x%08"PFMT64x"]\n", start, start + size);
|
|
|
|
eprintf ("Creating basic blocks\b");
|
|
|
|
}
|
|
|
|
lista[idx++] = b_start;
|
2018-02-01 16:35:08 +01:00
|
|
|
for (x = 0; x < 1024; x++) {
|
2018-01-31 17:59:05 +01:00
|
|
|
if (lista[x] != 0) {
|
|
|
|
cur =0;
|
|
|
|
b_start = lista[x];
|
|
|
|
lista[x] = 0;
|
|
|
|
while (cur < size) {
|
|
|
|
if (r_cons_is_breaked ()) {
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
// magic number to fix huge section of invalid code fuzz files
|
|
|
|
if (block_score < invalid_instruction_barrier) {
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool bFound = false;
|
2019-06-20 12:45:00 +08:00
|
|
|
// check if offset don't have into block_list, to end branch analisys
|
2018-01-31 17:59:05 +01:00
|
|
|
r_list_foreach (block_list, iter, block) {
|
|
|
|
if ( (block->type == END || block->type == NORMAL) && b_start + cur == block->start ) {
|
|
|
|
bFound = true;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!bFound) {
|
2019-04-14 17:33:49 +02:00
|
|
|
op = r_core_anal_op (core, b_start + cur, R_ANAL_OP_MASK_BASIC | R_ANAL_OP_MASK_DISASM);
|
2018-01-31 17:59:05 +01:00
|
|
|
|
|
|
|
if (!op || !op->mnemonic) {
|
|
|
|
block_score -= 10;
|
|
|
|
cur++;
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (op->mnemonic[0] == '?') {
|
|
|
|
eprintf ("? Bad op at: 0x%08"PFMT64x"\n", cur + b_start);
|
|
|
|
eprintf ("Cannot analyze opcode at %"PFMT64x"\n", b_start + cur);
|
|
|
|
block_score -= 10;
|
|
|
|
cur++;
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
//eprintf ("0x%08"PFMT64x" %s\n", b_start + cur, op->mnemonic);
|
|
|
|
switch (op->type) {
|
|
|
|
case R_ANAL_OP_TYPE_RET:
|
|
|
|
addBB (block_list, b_start, b_start + cur + op->size, UT64_MAX, UT64_MAX, END, block_score);
|
|
|
|
cur = size;
|
|
|
|
break;
|
|
|
|
case R_ANAL_OP_TYPE_UJMP:
|
|
|
|
case R_ANAL_OP_TYPE_IRJMP:
|
|
|
|
addBB (block_list, b_start, b_start + cur + op->size, op->jump, UT64_MAX, END, block_score);
|
|
|
|
cur = size;
|
|
|
|
break;
|
|
|
|
case R_ANAL_OP_TYPE_JMP:
|
|
|
|
addBB (block_list, b_start, b_start + cur + op->size, op->jump, UT64_MAX, END, block_score);
|
|
|
|
b_start = op->jump;
|
|
|
|
cur = 0;
|
|
|
|
block_score = 0;
|
|
|
|
break;
|
|
|
|
case R_ANAL_OP_TYPE_CJMP:
|
|
|
|
//eprintf ("bb_b 0x%08"PFMT64x" - 0x%08"PFMT64x"\n", b_start, b_start + cur + op->size);
|
|
|
|
addBB (block_list, b_start, b_start + cur + op->size, op->jump, b_start + cur + op->size, NORMAL, block_score);
|
|
|
|
b_start = b_start + cur + op->size;
|
|
|
|
cur = 0;
|
|
|
|
if (idx < 1024) {
|
|
|
|
lista[idx++] = op->jump;
|
|
|
|
}
|
|
|
|
block_score = 0;
|
|
|
|
break;
|
|
|
|
case R_ANAL_OP_TYPE_TRAP:
|
|
|
|
case R_ANAL_OP_TYPE_UNK:
|
|
|
|
case R_ANAL_OP_TYPE_ILL:
|
|
|
|
block_score -= 10;
|
|
|
|
cur += op->size;
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
cur += op->size;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
r_anal_op_free (op);
|
|
|
|
op = NULL;
|
|
|
|
}
|
|
|
|
else {
|
2019-06-20 12:45:00 +08:00
|
|
|
// we have this offset into previous analyzed block, exit from this path flow.
|
2018-01-31 17:59:05 +01:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (debug) {
|
|
|
|
eprintf ("Found %d basic blocks\n", block_list->length);
|
|
|
|
}
|
|
|
|
|
|
|
|
RList *result = r_list_newf (free);
|
|
|
|
if (!result) {
|
|
|
|
r_list_free (block_list);
|
|
|
|
eprintf ("Failed to create resulting list\n");
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
sdb = sdb_new0 ();
|
|
|
|
if (!sdb) {
|
|
|
|
eprintf ("Failed to initialize sdb db\n");
|
|
|
|
r_list_free (block_list);
|
|
|
|
r_list_free (result);
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
r_list_sort (block_list, (RListComparator)bbCMP);
|
|
|
|
|
|
|
|
if (debug) {
|
|
|
|
eprintf ("Sorting all blocks done\n");
|
|
|
|
eprintf ("Creating the complete graph\n");
|
|
|
|
}
|
|
|
|
|
|
|
|
while (block_list->length > 0) {
|
|
|
|
block = r_list_pop (block_list);
|
|
|
|
if (!block) {
|
|
|
|
eprintf ("Failed to get next block from list\n");
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
if (r_cons_is_breaked ()) {
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (block_list->length > 0) {
|
|
|
|
bb_t *next_block = (bb_t*)r_list_iter_get_data (block_list->tail);
|
|
|
|
if (!next_block) {
|
|
|
|
eprintf ("No next block to compare with!\n");
|
|
|
|
}
|
|
|
|
|
|
|
|
// current block is just a split block
|
|
|
|
if (block->start == next_block->start && block->end == UT64_MAX) {
|
|
|
|
if (block->type != CALL && next_block->type != CALL) {
|
|
|
|
next_block->reached = block->reached + 1;
|
|
|
|
}
|
|
|
|
free (block);
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
// block and next_block share the same start so we copy the
|
|
|
|
// contenct of the block into the next_block and skip the current one
|
|
|
|
if (block->start == next_block->start && next_block->end == UT64_MAX) {
|
|
|
|
if (next_block->type != CALL) {
|
|
|
|
block->reached += 1;
|
|
|
|
}
|
|
|
|
*next_block = *block;
|
|
|
|
free (block);
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (block->end < UT64_MAX && next_block->start < block->end && next_block->start > block->start) {
|
|
|
|
if (next_block->jump == UT64_MAX) {
|
|
|
|
next_block->jump = block->jump;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (next_block->fail == UT64_MAX) {
|
|
|
|
next_block->fail = block->fail;
|
|
|
|
}
|
|
|
|
|
|
|
|
next_block->end = block->end;
|
|
|
|
block->end = next_block->start;
|
|
|
|
block->jump = next_block->start;
|
|
|
|
block->fail = UT64_MAX;
|
|
|
|
next_block->type = block->type;
|
|
|
|
if (next_block->type != CALL) {
|
|
|
|
next_block->reached += 1;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-04-10 23:52:47 +02:00
|
|
|
sdb_ptr_set (sdb, sdb_fmt ("bb.0x%08"PFMT64x, block->start), block, 0);
|
2018-01-31 17:59:05 +01:00
|
|
|
r_list_append (result, block);
|
|
|
|
}
|
|
|
|
|
2019-06-20 12:45:00 +08:00
|
|
|
// finally add bb to function
|
2018-01-31 17:59:05 +01:00
|
|
|
// we simply assume that non reached blocks
|
2019-06-20 12:45:00 +08:00
|
|
|
// don't are part of the created function
|
2018-01-31 17:59:05 +01:00
|
|
|
if (debug) {
|
|
|
|
eprintf ("Trying to create functions\n");
|
|
|
|
}
|
|
|
|
|
|
|
|
r_list_foreach (result, iter, block) {
|
|
|
|
if (r_cons_is_breaked ()) {
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
if (block && (block->reached == 0)) {
|
|
|
|
fcn_t* current_function = fcnNew (block);
|
|
|
|
RStack *stack = r_stack_new (100);
|
|
|
|
bb_t *jump = NULL;
|
|
|
|
bb_t *fail = NULL;
|
|
|
|
bb_t *cur = NULL;
|
|
|
|
|
|
|
|
if (!r_stack_push (stack, (void*)block)) {
|
|
|
|
eprintf ("Failed to push initial block\n");
|
|
|
|
}
|
|
|
|
|
|
|
|
while (!r_stack_is_empty (stack)) {
|
|
|
|
cur = (bb_t*)r_stack_pop (stack);
|
|
|
|
if (!cur) {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
sdb_num_set (sdb, Fhandled (cur->start), 1, 0);
|
|
|
|
if (cur->score < 0) {
|
|
|
|
fcnFree (current_function);
|
|
|
|
current_function = NULL;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
// we ignore negative blocks
|
|
|
|
if ((st64)(cur->end - cur->start) < 0) {
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
fcnAddBB (current_function, cur);
|
|
|
|
|
|
|
|
if (cur->jump < UT64_MAX && !sdb_num_get (sdb, Fhandled (cur->jump), NULL)) {
|
2018-04-10 23:52:47 +02:00
|
|
|
jump = sdb_ptr_get (sdb, sdb_fmt ("bb.0x%08"PFMT64x, cur->jump), NULL);
|
2018-01-31 17:59:05 +01:00
|
|
|
if (!jump) {
|
|
|
|
eprintf ("Failed to get jump block at 0x%"PFMT64x"\n", cur->jump);
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
if (!r_stack_push (stack, (void*)jump)) {
|
|
|
|
eprintf ("Failed to push jump block to stack\n");
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (cur->fail < UT64_MAX && !sdb_num_get (sdb, Fhandled (cur->fail), NULL)) {
|
2018-04-10 23:52:47 +02:00
|
|
|
fail = sdb_ptr_get (sdb, sdb_fmt ("bb.0x%08" PFMT64x, cur->fail), NULL);
|
2018-01-31 17:59:05 +01:00
|
|
|
if (!fail) {
|
|
|
|
eprintf ("Failed to get fail block at 0x%"PFMT64x"\n", cur->fail);
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
if (!r_stack_push (stack, (void*)fail)) {
|
|
|
|
eprintf ("Failed to push jump block to stack\n");
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// function creation complete
|
|
|
|
if (current_function) {
|
2019-06-20 12:45:00 +08:00
|
|
|
// check for supply function address match with current block
|
2018-01-31 17:59:05 +01:00
|
|
|
if (current_function->addr == start) {
|
2019-06-20 12:45:00 +08:00
|
|
|
// set supply function size
|
2018-01-31 17:59:05 +01:00
|
|
|
current_function->size = size;
|
|
|
|
if (checkFunction (current_function)) {
|
|
|
|
if (input[0] == '*') {
|
|
|
|
printFunctionCommands (core, current_function, NULL);
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
createFunction (core, current_function, NULL);
|
|
|
|
}
|
|
|
|
fcnFree (current_function);
|
|
|
|
r_stack_free (stack);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
fcnFree (current_function);
|
|
|
|
}
|
|
|
|
r_stack_free (stack);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
sdb_free (sdb);
|
|
|
|
r_list_free (result);
|
|
|
|
r_list_free (block_list);
|
|
|
|
return true;
|
|
|
|
}
|