mirror of
https://github.com/radareorg/radare2.git
synced 2024-12-13 16:18:33 +00:00
698 lines
23 KiB
C
698 lines
23 KiB
C
#include <r_anal.h>
|
|
#include <r_core.h>
|
|
#include "minunit.h"
|
|
|
|
#include "test_anal_block_invars.inl"
|
|
#define check_invariants block_check_invariants
|
|
#define check_leaks block_check_leaks
|
|
|
|
static size_t blocks_count(RAnal *anal) {
|
|
size_t count = 0;
|
|
RBIter iter;
|
|
RAnalBlock *block;
|
|
r_rbtree_foreach(anal->bb_tree, iter, block, RAnalBlock, _rb) {
|
|
count++;
|
|
}
|
|
return count;
|
|
}
|
|
|
|
|
|
#define assert_invariants(anal) do { if (!check_invariants (anal)) { return false; } } while (0)
|
|
#define assert_leaks(anal) do { if (!check_leaks (anal)) { return false; } } while (0)
|
|
|
|
bool test_r_anal_block_create() {
|
|
RAnal *anal = r_anal_new ();
|
|
assert_invariants (anal);
|
|
|
|
mu_assert_eq (blocks_count (anal), 0, "initial count");
|
|
|
|
RAnalBlock *block = r_anal_create_block (anal, 0x1337, 42);
|
|
assert_invariants (anal);
|
|
mu_assert ("created block", block);
|
|
mu_assert_eq (block->addr, 0x1337, "created addr");
|
|
mu_assert_eq (block->size, 42, "created size");
|
|
mu_assert_eq (block->ref, 1, "created initial ref");
|
|
mu_assert_eq (blocks_count (anal), 1, "count after create");
|
|
|
|
RAnalBlock *block2 = r_anal_create_block (anal, 0x133f, 100);
|
|
assert_invariants (anal);
|
|
mu_assert ("created block (overlap)", block2);
|
|
mu_assert_eq (block2->addr, 0x133f, "created addr");
|
|
mu_assert_eq (block2->size, 100, "created size");
|
|
mu_assert_eq (block2->ref, 1, "created initial ref");
|
|
mu_assert_eq (blocks_count (anal), 2, "count after create");
|
|
|
|
RAnalBlock *block3 = r_anal_create_block (anal, 0x1337, 5);
|
|
assert_invariants (anal);
|
|
mu_assert ("no double create on same start", !block3);
|
|
mu_assert_eq (blocks_count (anal), 2, "count after failed create");
|
|
|
|
r_anal_block_unref (block);
|
|
r_anal_block_unref (block2);
|
|
|
|
assert_leaks (anal);
|
|
r_anal_free (anal);
|
|
mu_end;
|
|
}
|
|
|
|
bool test_r_anal_block_contains() {
|
|
RAnalBlock dummy = {0};
|
|
dummy.addr = 0x1337;
|
|
dummy.size = 42;
|
|
mu_assert ("contains before", !r_anal_block_contains (&dummy, 100));
|
|
mu_assert ("contains start", r_anal_block_contains (&dummy, 0x1337));
|
|
mu_assert ("contains inside", r_anal_block_contains (&dummy, 0x1339));
|
|
mu_assert ("contains last", r_anal_block_contains (&dummy, 0x1337 + 42 - 1));
|
|
mu_assert ("contains after", !r_anal_block_contains (&dummy, 0x1337 + 42));
|
|
mu_end;
|
|
}
|
|
|
|
bool test_r_anal_block_split() {
|
|
RAnal *anal = r_anal_new ();
|
|
assert_invariants (anal);
|
|
|
|
RAnalBlock *block = r_anal_create_block (anal, 0x1337, 42);
|
|
assert_invariants (anal);
|
|
mu_assert_eq (blocks_count (anal), 1, "count after create");
|
|
block->jump = 0xdeadbeef;
|
|
block->fail = 0xc0ffee;
|
|
block->ninstr = 5;
|
|
r_anal_bb_set_offset (block, 0, 0);
|
|
r_anal_bb_set_offset (block, 1, 1);
|
|
r_anal_bb_set_offset (block, 2, 2);
|
|
r_anal_bb_set_offset (block, 3, 4);
|
|
r_anal_bb_set_offset (block, 4, 30);
|
|
|
|
RAnalBlock *second = r_anal_block_split (block, 0x1337);
|
|
assert_invariants (anal);
|
|
mu_assert_ptreq (second, block, "nop split on first addr");
|
|
mu_assert_eq (blocks_count (anal), 1, "count after nop split");
|
|
mu_assert_eq (block->ref, 2, "ref after nop split");
|
|
r_anal_block_unref (block);
|
|
|
|
second = r_anal_block_split (block, 0x1339);
|
|
assert_invariants (anal);
|
|
mu_assert_ptrneq (second, block, "non-nop split");
|
|
mu_assert_eq (blocks_count (anal), 2, "count after non-nop split");
|
|
|
|
mu_assert_eq (block->addr, 0x1337, "first addr after split");
|
|
mu_assert_eq (block->size, 2, "first size after split");
|
|
mu_assert_eq (second->addr, 0x1339, "first addr after split");
|
|
mu_assert_eq (second->size, 40, "first size after split");
|
|
|
|
mu_assert_eq (block->jump, second->addr, "first jump");
|
|
mu_assert_eq (block->fail, UT64_MAX, "first fail");
|
|
mu_assert_eq (second->jump, 0xdeadbeef, "second jump");
|
|
mu_assert_eq (second->fail, 0xc0ffee, "second fail");
|
|
|
|
mu_assert_eq (block->ninstr, 2, "first ninstr after split");
|
|
mu_assert_eq (r_anal_bb_offset_inst (block, 0), 0, "first op_pos[0]");
|
|
mu_assert_eq (r_anal_bb_offset_inst (block, 1), 1, "first op_pos[1]");
|
|
|
|
mu_assert_eq (second->ninstr, 3, "second ninstr after split");
|
|
mu_assert_eq (r_anal_bb_offset_inst (second, 0), 0, "second op_pos[0]");
|
|
mu_assert_eq (r_anal_bb_offset_inst (second, 1), 2, "second op_pos[1]");
|
|
mu_assert_eq (r_anal_bb_offset_inst (second, 2), 28, "second op_pos[2]");
|
|
|
|
r_anal_block_unref (block);
|
|
r_anal_block_unref (second);
|
|
|
|
assert_leaks (anal);
|
|
r_anal_free (anal);
|
|
mu_end;
|
|
}
|
|
|
|
bool test_r_anal_block_split_in_function() {
|
|
RAnal *anal = r_anal_new ();
|
|
assert_invariants (anal);
|
|
|
|
RAnalFunction *fcn = r_anal_create_function (anal, "bbowner", 0x1337, 0, NULL);
|
|
assert_invariants (anal);
|
|
|
|
RAnalBlock *block = r_anal_create_block (anal, 0x1337, 42);
|
|
assert_invariants (anal);
|
|
mu_assert_eq (blocks_count (anal), 1, "count after create");
|
|
r_anal_function_add_block (fcn, block);
|
|
assert_invariants (anal);
|
|
mu_assert_eq (block->ref, 2, "block refs after adding to function");
|
|
|
|
RAnalBlock *second = r_anal_block_split (block, 0x1339);
|
|
assert_invariants (anal);
|
|
mu_assert_ptrneq (second, block, "non-nop split");
|
|
mu_assert_eq (blocks_count (anal), 2, "count after non-nop split");
|
|
mu_assert_eq (block->ref, 2, "first block refs after adding to function");
|
|
mu_assert_eq (second->ref, 2, "second block refs after adding to function");
|
|
|
|
mu_assert ("function has first block after split", r_list_contains (fcn->bbs, block));
|
|
mu_assert ("function has second block after split", r_list_contains (fcn->bbs, second));
|
|
mu_assert ("second block is in function after split", r_list_contains (block->fcns, fcn));
|
|
mu_assert ("second block is in function after split", r_list_contains (second->fcns, fcn));
|
|
|
|
r_anal_block_unref (block);
|
|
r_anal_block_unref (second);
|
|
|
|
assert_leaks (anal);
|
|
r_anal_free (anal);
|
|
mu_end;
|
|
}
|
|
|
|
bool test_r_anal_block_merge() {
|
|
RAnal *anal = r_anal_new ();
|
|
assert_invariants (anal);
|
|
|
|
RAnalBlock *first = r_anal_create_block (anal, 0x1337, 42);
|
|
RAnalBlock *second = r_anal_create_block (anal, 0x1337 + 42, 624);
|
|
assert_invariants (anal);
|
|
mu_assert_eq (blocks_count (anal), 2, "count after create");
|
|
second->jump = 0xdeadbeef;
|
|
second->fail = 0xc0ffee;
|
|
|
|
first->ninstr = 3;
|
|
r_anal_bb_set_offset (first, 0, 0);
|
|
r_anal_bb_set_offset (first, 1, 13);
|
|
r_anal_bb_set_offset (first, 2, 16);
|
|
|
|
second->ninstr = 4;
|
|
r_anal_bb_set_offset (second, 0, 0);
|
|
r_anal_bb_set_offset (second, 1, 4);
|
|
r_anal_bb_set_offset (second, 2, 9);
|
|
r_anal_bb_set_offset (second, 3, 30);
|
|
|
|
bool success = r_anal_block_merge (first, second);
|
|
assert_invariants (anal);
|
|
mu_assert ("merge success", success);
|
|
mu_assert_eq (blocks_count (anal), 1, "count after merge");
|
|
mu_assert_eq (first->addr, 0x1337, "addr after merge");
|
|
mu_assert_eq (first->size, 666, "size after merge");
|
|
mu_assert_eq (first->jump, 0xdeadbeef, "jump after merge");
|
|
mu_assert_eq (first->fail, 0xc0ffee, "fail after merge");
|
|
|
|
mu_assert_eq (first->ninstr, 3+4, "ninstr after merge");
|
|
mu_assert_eq (r_anal_bb_offset_inst (first, 0), 0, "offset 0 after merge");
|
|
mu_assert_eq (r_anal_bb_offset_inst (first, 1), 13, "offset 1 after merge");
|
|
mu_assert_eq (r_anal_bb_offset_inst (first, 2), 16, "offset 2 after merge");
|
|
mu_assert_eq (r_anal_bb_offset_inst (first, 3), 42+0, "offset 3 after merge");
|
|
mu_assert_eq (r_anal_bb_offset_inst (first, 4), 42+4, "offset 4 after merge");
|
|
mu_assert_eq (r_anal_bb_offset_inst (first, 5), 42+9, "offset 5 after merge");
|
|
mu_assert_eq (r_anal_bb_offset_inst (first, 6), 42+30, "offset 6 after merge");
|
|
|
|
r_anal_block_unref (first);
|
|
// second must be already freed by the merge!
|
|
|
|
assert_invariants (anal);
|
|
r_anal_free (anal);
|
|
mu_end;
|
|
}
|
|
|
|
bool test_r_anal_block_merge_in_function() {
|
|
RAnal *anal = r_anal_new ();
|
|
assert_invariants (anal);
|
|
|
|
RAnalFunction *fcn = r_anal_create_function (anal, "bbowner", 0x1337, 0, NULL);
|
|
|
|
RAnalBlock *first = r_anal_create_block (anal, 0x1337, 42);
|
|
RAnalBlock *second = r_anal_create_block (anal, 0x1337 + 42, 624);
|
|
assert_invariants (anal);
|
|
mu_assert_eq (blocks_count (anal), 2, "count after create");
|
|
|
|
r_anal_function_add_block (fcn, first);
|
|
assert_invariants (anal);
|
|
r_anal_function_add_block (fcn, second);
|
|
assert_invariants (anal);
|
|
|
|
bool success = r_anal_block_merge (first, second);
|
|
assert_invariants (anal);
|
|
mu_assert ("merge success", success);
|
|
mu_assert_eq (blocks_count (anal), 1, "count after merge");
|
|
mu_assert_eq (r_list_length (fcn->bbs), 1, "fcn bbs after merge");
|
|
mu_assert_eq (r_list_length (first->fcns), 1, "bb functions after merge");
|
|
mu_assert ("function has merged block", r_list_contains (fcn->bbs, first));
|
|
mu_assert ("merged block is in function", r_list_contains (first->fcns, fcn));
|
|
|
|
r_anal_block_unref (first);
|
|
// second must be already freed by the merge!
|
|
|
|
assert_invariants (anal);
|
|
r_anal_free (anal);
|
|
mu_end;
|
|
}
|
|
|
|
bool test_r_anal_block_delete() {
|
|
RAnal *anal = r_anal_new ();
|
|
assert_invariants (anal);
|
|
|
|
RAnalFunction *fcn = r_anal_create_function (anal, "bbowner", 0x1337, 0, NULL);
|
|
|
|
RAnalBlock *block = r_anal_create_block (anal, 0x1337, 42);
|
|
assert_invariants (anal);
|
|
mu_assert_eq (blocks_count (anal), 1, "count after create");
|
|
|
|
r_anal_function_add_block (fcn, block);
|
|
assert_invariants (anal);
|
|
mu_assert_eq (block->ref, 2, "refs after adding");
|
|
mu_assert_eq (r_list_length (fcn->bbs), 1, "fcn bbs after add");
|
|
mu_assert_eq (r_list_length (block->fcns), 1, "bb fcns after add");
|
|
|
|
r_anal_delete_block (block);
|
|
assert_invariants (anal);
|
|
mu_assert_eq (block->ref, 1, "refs after delete");
|
|
mu_assert_eq (r_list_length (fcn->bbs), 0, "fcn bbs after delete");
|
|
mu_assert_eq (r_list_length (block->fcns), 0, "bb fcns after delete");
|
|
|
|
r_anal_block_unref (block);
|
|
|
|
r_anal_free (anal);
|
|
mu_end;
|
|
}
|
|
|
|
bool test_r_anal_block_set_size() {
|
|
RAnal *anal = r_anal_new ();
|
|
assert_invariants (anal);
|
|
|
|
RAnalFunction *fcn = r_anal_create_function (anal, "bbowner", 0x1337, 0, NULL);
|
|
|
|
RAnalBlock *block = r_anal_create_block (anal, 0x1337, 42);
|
|
assert_invariants (anal);
|
|
|
|
r_anal_function_add_block (fcn, block);
|
|
assert_invariants (anal);
|
|
|
|
r_anal_block_set_size (block, 300);
|
|
assert_invariants (anal);
|
|
mu_assert_eq (block->size, 300, "size after set_size");
|
|
|
|
RAnalBlock *second = r_anal_create_block (anal, 0x1337+300, 100);
|
|
assert_invariants (anal);
|
|
r_anal_function_add_block (fcn, block);
|
|
assert_invariants (anal);
|
|
r_anal_function_linear_size (fcn); // trigger lazy calculation of min/max cache
|
|
assert_invariants (anal);
|
|
|
|
r_anal_block_set_size (second, 500);
|
|
assert_invariants (anal);
|
|
mu_assert_eq (second->size, 500, "size after set_size");
|
|
|
|
r_anal_block_set_size (block, 80);
|
|
assert_invariants (anal);
|
|
mu_assert_eq (block->size, 80, "size after set_size");
|
|
|
|
r_anal_block_unref (block);
|
|
r_anal_block_unref (second);
|
|
assert_invariants (anal);
|
|
|
|
r_anal_free (anal);
|
|
mu_end;
|
|
}
|
|
|
|
bool test_r_anal_block_relocate() {
|
|
RAnal *anal = r_anal_new ();
|
|
assert_invariants (anal);
|
|
|
|
RAnalFunction *fcn = r_anal_create_function (anal, "bbowner", 0x1337, 0, NULL);
|
|
|
|
RAnalBlock *block = r_anal_create_block (anal, 0x1337, 42);
|
|
assert_invariants (anal);
|
|
|
|
r_anal_function_add_block (fcn, block);
|
|
assert_invariants (anal);
|
|
r_anal_function_linear_size (fcn); // trigger lazy calculation of min/max cache
|
|
assert_invariants (anal);
|
|
|
|
bool success = r_anal_block_relocate (block, 0x200, 0x100);
|
|
mu_assert ("relocate success", success);
|
|
assert_invariants (anal);
|
|
mu_assert_eq (block->addr, 0x200, "addr after relocate");
|
|
mu_assert_eq (block->size, 0x100, "size after relocate");
|
|
|
|
RAnalBlock *second = r_anal_create_block (anal, 0x1337+300, 100);
|
|
assert_invariants (anal);
|
|
r_anal_function_add_block (fcn, second);
|
|
assert_invariants (anal);
|
|
|
|
success = r_anal_block_relocate (second, 0x400, 0x123);
|
|
mu_assert ("relocate success", success);
|
|
assert_invariants (anal);
|
|
mu_assert_eq (second->addr, 0x400, "addr after relocate");
|
|
mu_assert_eq (second->size, 0x123, "size after relocate");
|
|
r_anal_function_linear_size (fcn); // trigger lazy calculation of min/max cache
|
|
assert_invariants (anal);
|
|
|
|
success = r_anal_block_relocate (block, 0x400, 0x333);
|
|
mu_assert ("relocate fail on same addr", !success);
|
|
assert_invariants (anal);
|
|
mu_assert_eq (block->addr, 0x200, "addr after failed relocate");
|
|
mu_assert_eq (block->size, 0x100, "size after failed relocate");
|
|
r_anal_function_linear_size (fcn); // trigger lazy calculation of min/max cache
|
|
assert_invariants (anal);
|
|
|
|
// jump after the other block
|
|
success = r_anal_block_relocate (block, 0x500, 0x333);
|
|
mu_assert ("relocate success", success);
|
|
assert_invariants (anal);
|
|
mu_assert_eq (block->addr, 0x500, "addr after failed relocate");
|
|
mu_assert_eq (block->size, 0x333, "size after failed relocate");
|
|
r_anal_function_linear_size (fcn); // trigger lazy calculation of min/max cache
|
|
assert_invariants (anal);
|
|
|
|
// jump before the other block
|
|
success = r_anal_block_relocate (block, 0x10, 0x333);
|
|
mu_assert ("relocate success", success);
|
|
assert_invariants (anal);
|
|
mu_assert_eq (block->addr, 0x10, "addr after failed relocate");
|
|
mu_assert_eq (block->size, 0x333, "size after failed relocate");
|
|
|
|
r_anal_block_unref (block);
|
|
r_anal_block_unref (second);
|
|
assert_invariants (anal);
|
|
|
|
r_anal_free (anal);
|
|
mu_end;
|
|
}
|
|
|
|
bool test_r_anal_block_query() {
|
|
RAnal *anal = r_anal_new ();
|
|
assert_invariants (anal);
|
|
|
|
#define N 200
|
|
#define MAXSIZE 0x300
|
|
#define SPACE 0x10000
|
|
#define SAMPLES 300
|
|
|
|
RAnalBlock *blocks[N];
|
|
size_t i;
|
|
for (i = 0; i < N; i++) {
|
|
blocks[i] = r_anal_create_block (anal, rand () % SPACE, rand () % MAXSIZE); // may return null on duplicates
|
|
assert_invariants (anal);
|
|
}
|
|
|
|
// --
|
|
// test r_anal_get_block_at()
|
|
|
|
for (i = 0; i < N; i++) {
|
|
if (!blocks[i]) {
|
|
continue;
|
|
}
|
|
mu_assert_ptreq (r_anal_get_block_at (anal, blocks[i]->addr), blocks[i], "r_anal_get_block_at");
|
|
}
|
|
|
|
for (i = 0; i < SAMPLES; i++) {
|
|
ut64 addr = rand () % SPACE;
|
|
size_t j;
|
|
|
|
// goal is to check cases where r_anal_get_block_at() returns null,
|
|
// but since the addr is random, there may be a block sometimes too.
|
|
RAnalBlock *block = NULL;
|
|
for (j = 0; j < N; j++) {
|
|
if (!blocks[j]) {
|
|
continue;
|
|
}
|
|
if (blocks[j]->addr == addr) {
|
|
block = blocks[j];
|
|
break;
|
|
}
|
|
}
|
|
|
|
mu_assert_ptreq (r_anal_get_block_at (anal, addr), block, "r_anal_get_block_at");
|
|
}
|
|
|
|
// --
|
|
// test r_anal_get_blocks_in()
|
|
|
|
for (i = 0; i < SAMPLES; i++) {
|
|
ut64 addr = rand () % SPACE;
|
|
RList *in = r_anal_get_blocks_in (anal, addr);
|
|
|
|
RAnalBlock *block;
|
|
RListIter *it;
|
|
r_list_foreach (in, it, block) {
|
|
mu_assert_eq (block->ref, 2, "block refd in returned list");
|
|
}
|
|
|
|
size_t linear_found = 0;
|
|
size_t j;
|
|
for (j = 0; j < N; j++) {
|
|
if (!blocks[j]) {
|
|
continue;
|
|
}
|
|
if (r_anal_block_contains (blocks[j], addr)) {
|
|
linear_found++;
|
|
mu_assert ("intersect linear found in list", r_list_contains (in, blocks[j]));
|
|
}
|
|
}
|
|
mu_assert_eq ((size_t)r_list_length (in), linear_found, "r_anal_get_blocks_in count");
|
|
r_list_free (in);
|
|
}
|
|
|
|
// --
|
|
// test r_anal_get_blocks_intersect()
|
|
|
|
for (i = 0; i < SAMPLES; i++) {
|
|
ut64 addr = rand () % SPACE;
|
|
ut64 size = rand() % MAXSIZE;
|
|
RList *in = r_anal_get_blocks_intersect (anal, addr, size);
|
|
|
|
RAnalBlock *block;
|
|
RListIter *it;
|
|
r_list_foreach (in, it, block) {
|
|
mu_assert_eq (block->ref, 2, "block refd in returned list");
|
|
}
|
|
|
|
size_t linear_found = 0;
|
|
size_t j;
|
|
for (j = 0; j < N; j++) {
|
|
RAnalBlock *block = blocks[j];
|
|
if (!block || addr + size <= block->addr || addr >= block->addr + block->size) {
|
|
continue;
|
|
}
|
|
linear_found++;
|
|
mu_assert ("in linear found in list", r_list_contains (in, blocks[j]));
|
|
}
|
|
mu_assert_eq ((size_t)r_list_length (in), linear_found, "r_anal_get_blocks_intersect count");
|
|
r_list_free (in);
|
|
}
|
|
|
|
for (i = 0; i < N; i++) {
|
|
r_anal_block_unref (blocks[i]);
|
|
}
|
|
|
|
assert_leaks (anal);
|
|
r_anal_free (anal);
|
|
mu_end;
|
|
}
|
|
|
|
bool addr_list_cb(ut64 addr, void *user) {
|
|
RList *list = user;
|
|
r_list_push (list, (void *)addr);
|
|
return true;
|
|
}
|
|
|
|
bool test_r_anal_block_successors() {
|
|
RAnal *anal = r_anal_new ();
|
|
assert_invariants (anal);
|
|
|
|
RAnalBlock *blocks[10];
|
|
blocks[0] = r_anal_create_block (anal, 0x10, 0x10);
|
|
blocks[1] = r_anal_create_block (anal, 0x30, 0x10);
|
|
blocks[2] = r_anal_create_block (anal, 0x50, 0x10);
|
|
blocks[3] = r_anal_create_block (anal, 0x100, 0x10);
|
|
blocks[4] = r_anal_create_block (anal, 0x110, 0x10);
|
|
blocks[5] = r_anal_create_block (anal, 0x120, 0x10);
|
|
blocks[6] = r_anal_create_block (anal, 0x130, 0x10);
|
|
blocks[7] = r_anal_create_block (anal, 0x140, 0x10);
|
|
blocks[8] = r_anal_create_block (anal, 0xa0, 0x10);
|
|
blocks[9] = r_anal_create_block (anal, 0xc0, 0x10);
|
|
assert_invariants (anal);
|
|
|
|
blocks[0]->jump = 0x30;
|
|
blocks[0]->fail = 0x50;
|
|
blocks[1]->jump = 0x10;
|
|
blocks[1]->fail = 0x50;
|
|
blocks[2]->jump = 0x10;
|
|
|
|
RAnalSwitchOp *sop = r_anal_switch_op_new (0x55, 0x13, 0x15, 0x42);
|
|
mu_assert_eq (sop->addr, 0x55, "addr");
|
|
mu_assert_eq (sop->min_val, 0x13, "addr");
|
|
mu_assert_eq (sop->max_val, 0x15, "addr");
|
|
mu_assert_eq (sop->def_val, 0x42, "addr");
|
|
r_anal_switch_op_add_case (sop, 0x55, 1, 0x100);
|
|
r_anal_switch_op_add_case (sop, 0x55, 2, 0x110);
|
|
r_anal_switch_op_add_case (sop, 0x55, 3, 0x120);
|
|
r_anal_switch_op_add_case (sop, 0x55, 4, 0x130);
|
|
r_anal_switch_op_add_case (sop, 0x55, 5, 0x140);
|
|
blocks[2]->switch_op = sop;
|
|
|
|
RList *result = r_list_new ();
|
|
r_anal_block_successor_addrs_foreach (blocks[0], addr_list_cb, result);
|
|
mu_assert_eq (r_list_length (result), 2, "jump/fail successors count");
|
|
mu_assert ("jmp successor", r_list_contains (result, (void *)0x30));
|
|
mu_assert ("fail successor", r_list_contains (result, (void *)0x50));
|
|
r_list_purge (result);
|
|
|
|
r_anal_block_successor_addrs_foreach (blocks[2], addr_list_cb, result);
|
|
mu_assert_eq (r_list_length (result), 6, "switch successors count");
|
|
mu_assert ("jmp successor", r_list_contains (result, (void *)0x10));
|
|
mu_assert ("case successor", r_list_contains (result, (void *)0x100));
|
|
mu_assert ("case successor", r_list_contains (result, (void *)0x110));
|
|
mu_assert ("case successor", r_list_contains (result, (void *)0x120));
|
|
mu_assert ("case successor", r_list_contains (result, (void *)0x130));
|
|
mu_assert ("case successor", r_list_contains (result, (void *)0x140));
|
|
r_list_free (result);
|
|
|
|
result = r_anal_block_recurse_list (blocks[0]);
|
|
RAnalBlock *block;
|
|
RListIter *it;
|
|
r_list_foreach (result, it, block) {
|
|
mu_assert_eq (block->ref, 2, "block refd in returned list");
|
|
}
|
|
|
|
mu_assert_eq (r_list_length (result), 8, "recursive successors count");
|
|
mu_assert ("recursive successor", r_list_contains (result, blocks[0]));
|
|
mu_assert ("recursive successor", r_list_contains (result, blocks[1]));
|
|
mu_assert ("recursive successor", r_list_contains (result, blocks[2]));
|
|
mu_assert ("recursive successor", r_list_contains (result, blocks[3]));
|
|
mu_assert ("recursive successor", r_list_contains (result, blocks[4]));
|
|
mu_assert ("recursive successor", r_list_contains (result, blocks[5]));
|
|
mu_assert ("recursive successor", r_list_contains (result, blocks[6]));
|
|
mu_assert ("recursive successor", r_list_contains (result, blocks[7]));
|
|
|
|
r_list_free (result);
|
|
|
|
size_t i;
|
|
for (i = 0; i < sizeof (blocks) / sizeof (RAnalBlock *); i++) {
|
|
r_anal_block_unref (blocks[i]);
|
|
}
|
|
|
|
assert_leaks (anal);
|
|
r_anal_free (anal);
|
|
mu_end;
|
|
}
|
|
|
|
bool test_r_anal_block_automerge() {
|
|
size_t i;
|
|
for (i = 0; i < SAMPLES; i++) {
|
|
RAnal *anal = r_anal_new ();
|
|
assert_invariants (anal);
|
|
|
|
RAnalBlock *a = r_anal_create_block (anal, 0x100, 0x10);
|
|
|
|
RAnalBlock *b = r_anal_create_block (anal, 0x110, 0x10);
|
|
a->jump = b->addr;
|
|
|
|
RAnalBlock *c = r_anal_create_block (anal, 0x120, 0x10);
|
|
b->jump = c->addr;
|
|
c->fail = b->addr;
|
|
|
|
RAnalBlock *d = r_anal_create_block (anal, 0x130, 0x10);
|
|
c->jump = d->addr;
|
|
|
|
RAnalBlock *e = r_anal_create_block (anal, 0x140, 0x10);
|
|
d->jump = e->addr;
|
|
|
|
RAnalBlock *f = r_anal_create_block (anal, 0x150, 0x10);
|
|
e->jump = f->addr;
|
|
|
|
RAnalFunction *fa = r_anal_create_function (anal, "fcn", 0x100, R_ANAL_FCN_TYPE_FCN, NULL);
|
|
r_anal_function_add_block (fa, a);
|
|
r_anal_function_add_block (fa, c);
|
|
r_anal_function_add_block (fa, d);
|
|
r_anal_function_add_block (fa, e);
|
|
r_anal_function_add_block (fa, f);
|
|
|
|
RAnalFunction *fb = r_anal_create_function (anal, "fcn2", 0x110, R_ANAL_FCN_TYPE_FCN, NULL);
|
|
r_anal_function_add_block (fb, b);
|
|
r_anal_function_add_block (fb, c);
|
|
r_anal_function_add_block (fb, d);
|
|
r_anal_function_add_block (fb, e);
|
|
r_anal_function_add_block (fb, f);
|
|
|
|
RList *all_blocks = r_list_new ();
|
|
r_list_push (all_blocks, a);
|
|
r_list_push (all_blocks, b);
|
|
r_list_push (all_blocks, c);
|
|
r_list_push (all_blocks, d);
|
|
r_list_push (all_blocks, e);
|
|
r_list_push (all_blocks, f);
|
|
|
|
// Randomize the order in which we give the automerge the block.
|
|
// The outcome should always be the same but it can have some delicate implications on the algorithm inside.
|
|
RList *shuffled_blocks = r_list_newf ((RListFree)r_anal_block_unref);
|
|
while (!r_list_empty (all_blocks)) {
|
|
int n = rand () % r_list_length (all_blocks);
|
|
r_list_push (shuffled_blocks, r_list_get_n (all_blocks, n));
|
|
r_list_del_n (all_blocks, n);
|
|
}
|
|
r_list_free (all_blocks);
|
|
|
|
r_anal_block_automerge (shuffled_blocks);
|
|
assert_invariants (anal);
|
|
//mu_assert_eq (r_list_length (shuffled_blocks), 4, "length after automerge");
|
|
mu_assert ("remaining blocks a", r_list_contains (shuffled_blocks, a));
|
|
mu_assert ("remaining blocks b", r_list_contains (shuffled_blocks, b));
|
|
mu_assert ("remaining blocks c", r_list_contains (shuffled_blocks, c));
|
|
mu_assert ("remaining blocks d", r_list_contains (shuffled_blocks, d));
|
|
mu_assert_eq (blocks_count (anal), r_list_length (shuffled_blocks), "blocks in anal count");
|
|
RListIter *it;
|
|
RAnalBlock *block;
|
|
r_list_foreach (shuffled_blocks, it, block) {
|
|
mu_assert_ptreq (r_anal_get_block_at (anal, block->addr), block, "remaining blocks in anal");
|
|
}
|
|
r_list_free (shuffled_blocks);
|
|
|
|
assert_invariants (anal);
|
|
assert_leaks (anal);
|
|
r_anal_free (anal);
|
|
}
|
|
mu_end;
|
|
}
|
|
|
|
bool test_r_anal_block_chop_noreturn(void) {
|
|
RAnal *anal = r_anal_new ();
|
|
assert_invariants (anal);
|
|
|
|
RAnalBlock *a = r_anal_create_block (anal, 0x100, 0x10);
|
|
RAnalBlock *b = r_anal_create_block (anal, 0x110, 0x10);
|
|
RAnalBlock *c = r_anal_create_block (anal, 0x120, 0x10);
|
|
a->jump = c->addr;
|
|
b->jump = c->addr;
|
|
|
|
RAnalFunction *fa = r_anal_create_function (anal, "fcn", 0x100, R_ANAL_FCN_TYPE_FCN, NULL);
|
|
r_anal_function_add_block (fa, a);
|
|
r_anal_function_add_block (fa, b);
|
|
r_anal_function_add_block (fa, c);
|
|
|
|
RAnalFunction *fb = r_anal_create_function (anal, "fcn2", 0x130, R_ANAL_FCN_TYPE_FCN, NULL);
|
|
fb->is_noreturn = true;
|
|
|
|
r_anal_block_chop_noreturn (b, 0x111);
|
|
|
|
assert_invariants (anal);
|
|
r_anal_free (anal);
|
|
|
|
mu_end;
|
|
}
|
|
|
|
int all_tests() {
|
|
mu_run_test (test_r_anal_block_chop_noreturn);
|
|
mu_run_test (test_r_anal_block_create);
|
|
mu_run_test (test_r_anal_block_contains);
|
|
mu_run_test (test_r_anal_block_split);
|
|
mu_run_test (test_r_anal_block_split_in_function);
|
|
mu_run_test (test_r_anal_block_merge);
|
|
mu_run_test (test_r_anal_block_merge_in_function);
|
|
mu_run_test (test_r_anal_block_delete);
|
|
mu_run_test (test_r_anal_block_set_size);
|
|
mu_run_test (test_r_anal_block_relocate);
|
|
mu_run_test (test_r_anal_block_query);
|
|
mu_run_test (test_r_anal_block_successors);
|
|
mu_run_test (test_r_anal_block_automerge);
|
|
return tests_passed != tests_run;
|
|
}
|
|
|
|
int main(int argc, char **argv) {
|
|
struct timeval tv;
|
|
gettimeofday (&tv, NULL);
|
|
unsigned int seed = argc > 1 ? strtoul (argv[1], NULL, 0) : tv.tv_sec + tv.tv_usec;
|
|
printf("seed for test_anal_block: %u\n", seed);
|
|
return all_tests();
|
|
}
|