radare2/test/unit/test_anal_hints.c
Florian Märkl a5e11a3f55
Refactor Anal Hints (#15876) ##anal
* Use RVector for Address Hints
* Add Arch/Bits Hint Trees
* Add Unit Tests for Addr, Arch and Bits Hints
* Add unset for newbits
* Fix jmptbl hint fetching
* Fix r_anal_*_bits_foreach
* Print grouped Anal Hints
* Fix arch bit affect on disasm
* Add Reset Hints to Commands
2020-02-13 11:48:24 +01:00

346 lines
9.9 KiB
C

#include <r_anal.h>
#include "minunit.h"
const RAnalHint empty_hint = {
.addr = UT64_MAX,
.ptr = 0,
.val = UT64_MAX,
.jump = UT64_MAX,
.fail = UT64_MAX,
.ret = UT64_MAX,
.arch = 0,
.opcode = NULL,
.syntax = NULL,
.esil = NULL,
.offset = NULL,
.type = 0,
.size = 0,
.bits = 0,
.new_bits = 0,
.immbase = 0,
.high = 0,
.nword = 0,
.stackframe = UT64_MAX,
};
bool hint_equals(const RAnalHint *a, const RAnalHint *b) {
#define CHECK_EQ(member) mu_assert_eq (a->member, b->member, "hint member " #member)
CHECK_EQ (ptr);
CHECK_EQ (val);
CHECK_EQ (jump);
CHECK_EQ (fail);
CHECK_EQ (ret);
CHECK_EQ (type);
CHECK_EQ (size);
CHECK_EQ (bits);
CHECK_EQ (new_bits);
CHECK_EQ (immbase);
CHECK_EQ (high);
CHECK_EQ (nword);
CHECK_EQ (stackframe);
#undef CHECK_EQ
#define CHECK_STREQ(member) mu_assert_nullable_streq (a->member, b->member, "hint member " #member)
CHECK_STREQ (arch);
CHECK_STREQ (opcode);
CHECK_STREQ (syntax);
CHECK_STREQ (esil);
CHECK_STREQ (offset);
#undef CHECK_STREQ
return true;
}
#define assert_hint_eq(actual, expected) do { \
if (actual == NULL) \
mu_assert ("hint", expected == &empty_hint); /* TODO: remove this part, only else should be used! */ \
else \
mu_assert ("hint", hint_equals (actual, expected)); \
} while (0)
bool test_r_anal_addr_hints() {
RAnal *anal = r_anal_new ();
RAnalHint *hint = r_anal_hint_get (anal, 0x1337);
assert_hint_eq (hint, &empty_hint);
r_anal_hint_free (hint);
RAnalHint cur = empty_hint;
#define CHECK \
hint = r_anal_hint_get (anal, 0x1337); \
assert_hint_eq (hint, &cur); \
r_anal_hint_free (hint);
hint = r_anal_hint_get (anal, 0x1338); \
assert_hint_eq (hint, &empty_hint); \
r_anal_hint_free (hint);
hint = r_anal_hint_get (anal, 0x1336); \
assert_hint_eq (hint, &empty_hint); \
r_anal_hint_free (hint);
// set --------
r_anal_hint_set_syntax (anal, 0x1337, "mysyntax");
cur.syntax = "mysyntax";
CHECK
r_anal_hint_set_type (anal, 0x1337, R_ANAL_OP_TYPE_RET);
cur.type = R_ANAL_OP_TYPE_RET;
CHECK
r_anal_hint_set_jump (anal, 0x1337, 0xdeadbeef);
cur.jump = 0xdeadbeef;
CHECK
r_anal_hint_set_fail (anal, 0x1337, 0xc0ffee);
cur.fail = 0xc0ffee;
CHECK
r_anal_hint_set_nword (anal, 0x1337, 42);
cur.nword = 42;
CHECK
r_anal_hint_set_offset (anal, 0x1337, "mytypeoff");
cur.offset = "mytypeoff";
CHECK
r_anal_hint_set_immbase (anal, 0x1337, 7);
cur.immbase = 7;
CHECK
r_anal_hint_set_size (anal, 0x1337, 0x123);
cur.size = 0x123;
CHECK
r_anal_hint_set_opcode (anal, 0x1337, "myopcode");
cur.opcode = "myopcode";
CHECK
r_anal_hint_set_esil (anal, 0x1337, "/,-rf,rm");
cur.esil = "/,-rf,rm";
CHECK
r_anal_hint_set_pointer (anal, 0x1337, 0x4242);
cur.ptr = 0x4242;
CHECK
r_anal_hint_set_ret (anal, 0x1337, 0xf00d);
cur.ret = 0xf00d;
CHECK
r_anal_hint_set_high (anal, 0x1337);
cur.high = true;
CHECK
r_anal_hint_set_stackframe (anal, 0x1337, 0x4321);
cur.stackframe = 0x4321;
CHECK
r_anal_hint_set_val (anal, 0x1337, 0x112358d);
cur.val = 0x112358d;
CHECK
r_anal_hint_set_newbits (anal, 0x1337, 16);
cur.new_bits = 16;
CHECK
// unset --------
r_anal_hint_unset_syntax (anal, 0x1337);
cur.syntax = NULL;
CHECK
r_anal_hint_unset_type (anal, 0x1337);
cur.type = 0;
CHECK
r_anal_hint_unset_jump (anal, 0x1337);
cur.jump = UT64_MAX;
CHECK
r_anal_hint_unset_fail (anal, 0x1337);
cur.fail = UT64_MAX;
CHECK
r_anal_hint_unset_nword (anal, 0x1337);
cur.nword = 0;
CHECK
r_anal_hint_unset_offset (anal, 0x1337);
cur.offset = NULL;
CHECK
r_anal_hint_unset_immbase (anal, 0x1337);
cur.immbase = 0;
CHECK
r_anal_hint_unset_size (anal, 0x1337);
cur.size = 0;
CHECK
r_anal_hint_unset_opcode (anal, 0x1337);
cur.opcode = NULL;
CHECK
r_anal_hint_unset_esil (anal, 0x1337);
cur.esil = NULL;
CHECK
r_anal_hint_unset_pointer (anal, 0x1337);
cur.ptr = 0;
CHECK
r_anal_hint_unset_ret (anal, 0x1337);
cur.ret = UT64_MAX;
CHECK
r_anal_hint_unset_high (anal, 0x1337);
cur.high = false;
CHECK
r_anal_hint_unset_stackframe (anal, 0x1337);
cur.stackframe = UT64_MAX;
CHECK
r_anal_hint_unset_val (anal, 0x1337);
cur.val = UT64_MAX;
CHECK
r_anal_hint_unset_newbits (anal, 0x1337);
cur.new_bits = 0;
//CHECK
hint = r_anal_hint_get (anal, 0x1337);
assert_hint_eq (hint, &empty_hint);
r_anal_hint_free (hint);
r_anal_free (anal);
mu_end;
#undef CHECK
}
#define RANGED_TEST(name, val, resetval, assert_val) \
bool test_r_anal_hints_##name() { \
RAnal *anal = r_anal_new (); \
\
ut64 hint_addr = 0xdead; \
assert_val (r_anal_hint_##name##_at (anal, 0x1337, &hint_addr), resetval, "no " #name ""); \
mu_assert_eq (hint_addr, UT64_MAX, "hint addr"); \
\
r_anal_hint_##name##_at (anal, 0x1337, NULL); /* make sure this does not null-deref */ \
\
/* -- */ \
r_anal_hint_set_##name (anal, 0x1337, val); \
\
hint_addr = 0xdead; \
assert_val (r_anal_hint_##name##_at (anal, 0x1337, &hint_addr), val, #name " at addr"); \
mu_assert_eq (hint_addr, 0x1337, "hint addr"); \
hint_addr = 0xdead; \
assert_val (r_anal_hint_##name##_at (anal, 0x1338, &hint_addr), val, #name " after addr"); \
mu_assert_eq (hint_addr, 0x1337, "hint addr"); \
hint_addr = 0xdead; \
assert_val (r_anal_hint_##name##_at (anal, UT64_MAX, &hint_addr), val, #name " after addr"); \
mu_assert_eq (hint_addr, 0x1337, "hint addr"); \
hint_addr = 0xdead; \
assert_val (r_anal_hint_##name##_at (anal, 0x1336, &hint_addr), resetval, "no " #name " before addr"); \
mu_assert_eq (hint_addr, UT64_MAX, "hint addr"); \
hint_addr = 0xdead; \
assert_val (r_anal_hint_##name##_at (anal, 0, &hint_addr), resetval, "no " #name " before addr"); \
mu_assert_eq (hint_addr, UT64_MAX, "hint addr"); \
\
r_anal_hint_##name##_at (anal, 0x1337, NULL); /* make sure this does not null-deref */ \
\
RAnalHint cur = empty_hint; \
cur.name = val; \
RAnalHint *hint = r_anal_hint_get (anal, 0x1337); \
assert_hint_eq (hint, &cur); \
r_anal_hint_free (hint); \
hint = r_anal_hint_get (anal, 0x1338); \
assert_hint_eq (hint, &cur); \
r_anal_hint_free (hint); \
hint = r_anal_hint_get (anal, 0x1336); \
assert_hint_eq (hint, &empty_hint); \
r_anal_hint_free (hint); \
\
/* -- */ \
hint_addr = 0xdead; \
assert_val (r_anal_hint_##name##_at (anal, 0xdeadbeef, &hint_addr), val, "before reset " #name " at addr"); \
mu_assert_eq (hint_addr, 0x1337, "hint addr"); \
r_anal_hint_set_##name (anal, 0xdeadbeef, resetval); \
hint_addr = 0xdead; \
assert_val (r_anal_hint_##name##_at (anal, 0xdeadbeef, &hint_addr), resetval, "reset " #name " at addr"); \
mu_assert_eq (hint_addr, 0xdeadbeef, "hint addr"); \
hint_addr = 0xdead; \
assert_val (r_anal_hint_##name##_at (anal, 0xdeadbeef + 1, &hint_addr), resetval, "reset " #name " after addr"); \
mu_assert_eq (hint_addr, 0xdeadbeef, "hint addr"); \
hint_addr = 0xdead; \
assert_val (r_anal_hint_##name##_at (anal, UT64_MAX, &hint_addr), resetval, "reset " #name " after addr"); \
mu_assert_eq (hint_addr, 0xdeadbeef, "hint addr"); \
hint_addr = 0xdead; \
assert_val (r_anal_hint_##name##_at (anal, 0xdeadbeef - 1, &hint_addr), val, "" #name " before addr"); \
mu_assert_eq (hint_addr, 0x1337, "hint addr"); \
\
/* -- */ \
r_anal_hint_unset_##name (anal, 0xdeadbeef); \
hint_addr = 0xdead; \
assert_val (r_anal_hint_##name##_at (anal, 0x1337, &hint_addr), val, #name " at addr"); \
mu_assert_eq (hint_addr, 0x1337, "hint addr"); \
hint_addr = 0xdead; \
assert_val (r_anal_hint_##name##_at (anal, 0x1338, &hint_addr), val, #name " after addr"); \
mu_assert_eq (hint_addr, 0x1337, "hint addr"); \
hint_addr = 0xdead; \
assert_val (r_anal_hint_##name##_at (anal, UT64_MAX, &hint_addr), val, #name " after addr"); \
mu_assert_eq (hint_addr, 0x1337, "hint addr"); \
hint_addr = 0xdead; \
assert_val (r_anal_hint_##name##_at (anal, 0x1336, &hint_addr), resetval, "no " #name " before addr"); \
mu_assert_eq (hint_addr, UT64_MAX, "hint addr"); \
hint_addr = 0xdead; \
assert_val (r_anal_hint_##name##_at (anal, 0, &hint_addr), resetval, "no " #name " before addr"); \
mu_assert_eq (hint_addr, UT64_MAX, "hint addr"); \
hint_addr = 0xdead; \
assert_val (r_anal_hint_##name##_at (anal, 0xdeadbeef, &hint_addr), val, "unset reset " #name " at addr"); \
mu_assert_eq (hint_addr, 0x1337, "hint addr"); \
hint_addr = 0xdead; \
assert_val (r_anal_hint_##name##_at (anal, 0xdeadbeef + 1, &hint_addr), val, "unset reset " #name " after addr"); \
mu_assert_eq (hint_addr, 0x1337, "hint addr"); \
hint_addr = 0xdead; \
assert_val (r_anal_hint_##name##_at (anal, UT64_MAX, &hint_addr), val, "unset reset " #name " after addr"); \
mu_assert_eq (hint_addr, 0x1337, "hint addr"); \
hint_addr = 0xdead; \
assert_val (r_anal_hint_##name##_at (anal, 0xdeadbeef - 1, &hint_addr), val, #name " before addr"); \
mu_assert_eq (hint_addr, 0x1337, "hint addr"); \
\
/* -- */ \
r_anal_hint_unset_##name (anal, 0x1337); \
hint_addr = 0xdead; \
assert_val (r_anal_hint_##name##_at (anal, 0x1336, &hint_addr), resetval, "unset " #name ""); \
mu_assert_eq (hint_addr, UT64_MAX, "hint addr"); \
hint_addr = 0xdead; \
assert_val (r_anal_hint_##name##_at (anal, 0, &hint_addr), resetval, "unset " #name ""); \
mu_assert_eq (hint_addr, UT64_MAX, "hint addr"); \
hint_addr = 0xdead; \
assert_val (r_anal_hint_##name##_at (anal, 0x1337, &hint_addr), resetval, "unset " #name ""); \
mu_assert_eq (hint_addr, UT64_MAX, "hint addr"); \
hint_addr = 0xdead; \
assert_val (r_anal_hint_##name##_at (anal, 0x1338, &hint_addr), resetval, "unset " #name ""); \
mu_assert_eq (hint_addr, UT64_MAX, "hint addr"); \
hint_addr = 0xdead; \
assert_val (r_anal_hint_##name##_at (anal, UT64_MAX, &hint_addr), resetval, "unset " #name ""); \
mu_assert_eq (hint_addr, UT64_MAX, "hint addr"); \
hint_addr = 0xdead; \
\
r_anal_free (anal); \
mu_end; \
}
RANGED_TEST(arch, "6502", NULL, mu_assert_nullable_streq)
RANGED_TEST(bits, 16, 0, mu_assert_eq)
bool all_tests() {
mu_run_test(test_r_anal_addr_hints);
mu_run_test(test_r_anal_hints_arch);
mu_run_test(test_r_anal_hints_bits);
return tests_passed != tests_run;
}
int main(int argc, char **argv) {
return all_tests();
}