mirror of
https://github.com/radareorg/radare2.git
synced 2024-12-15 09:21:00 +00:00
536 lines
15 KiB
C
536 lines
15 KiB
C
/* radare - LGPL - Copyright 2013-2020 - pancake */
|
|
|
|
#include <r_anal.h>
|
|
|
|
// Common base-struct for hints which affect an entire range as opposed to only one single address
|
|
// They are saved in a RBTree per hint type.
|
|
// Each ranged record in a tree affects every address address greater or equal to its specified address until
|
|
// the next record or the end of the address space.
|
|
typedef struct r_anal_ranged_hint_record_base_t {
|
|
RBNode rb;
|
|
ut64 addr;
|
|
} RAnalRangedHintRecordBase;
|
|
|
|
typedef struct r_anal_arch_hint_record_t {
|
|
RAnalRangedHintRecordBase base; // MUST be the first member!
|
|
char *arch; // NULL => reset to global
|
|
} RAnalArchHintRecord;
|
|
|
|
typedef struct r_anal_bits_hint_record_t {
|
|
RAnalRangedHintRecordBase base; // MUST be the first member!
|
|
int bits; // 0 => reset to global
|
|
} RAnalBitsHintRecord;
|
|
|
|
static int ranged_hint_record_cmp(const void *incoming, const RBNode *in_tree, void *user) {
|
|
ut64 addr = *(const ut64 *)incoming;
|
|
const RAnalRangedHintRecordBase *in_tree_record = container_of (in_tree, const RAnalRangedHintRecordBase, rb);
|
|
if (addr < in_tree_record->addr) {
|
|
return -1;
|
|
} else if (addr > in_tree_record->addr) {
|
|
return 1;
|
|
}
|
|
return 0;
|
|
}
|
|
|
|
static void addr_hint_record_fini(void *element, void *user) {
|
|
(void)user;
|
|
RAnalAddrHintRecord *record = element;
|
|
switch (record->type) {
|
|
case R_ANAL_ADDR_HINT_TYPE_TYPE_OFFSET:
|
|
free (record->type_offset);
|
|
break;
|
|
case R_ANAL_ADDR_HINT_TYPE_SYNTAX:
|
|
free (record->syntax);
|
|
break;
|
|
case R_ANAL_ADDR_HINT_TYPE_OPCODE:
|
|
free (record->opcode);
|
|
break;
|
|
case R_ANAL_ADDR_HINT_TYPE_ESIL:
|
|
free (record->esil);
|
|
break;
|
|
default:
|
|
break;
|
|
}
|
|
}
|
|
|
|
static void addr_hint_record_ht_free(HtUPKv *kv) {
|
|
r_vector_free (kv->value);
|
|
}
|
|
|
|
static void bits_hint_record_free_rb(RBNode *node, void *user) {
|
|
free (container_of (node, RAnalRangedHintRecordBase, rb));
|
|
}
|
|
|
|
static void arch_hint_record_free_rb(RBNode *node, void *user) {
|
|
RAnalArchHintRecord *record = (RAnalArchHintRecord *)container_of (node, RAnalRangedHintRecordBase, rb);
|
|
free (record->arch);
|
|
free (record);
|
|
}
|
|
|
|
// used in anal.c, but no API needed
|
|
void r_anal_hint_storage_init(RAnal *a) {
|
|
a->addr_hints = ht_up_new (NULL, addr_hint_record_ht_free, NULL);
|
|
a->arch_hints = NULL;
|
|
a->bits_hints = NULL;
|
|
}
|
|
|
|
// used in anal.c, but no API needed
|
|
void r_anal_hint_storage_fini(RAnal *a) {
|
|
ht_up_free (a->addr_hints);
|
|
r_rbtree_free (a->arch_hints, arch_hint_record_free_rb, NULL);
|
|
r_rbtree_free (a->bits_hints, bits_hint_record_free_rb, NULL);
|
|
}
|
|
|
|
R_API void r_anal_hint_clear(RAnal *a) {
|
|
r_anal_hint_storage_fini (a);
|
|
r_anal_hint_storage_init (a);
|
|
}
|
|
|
|
typedef struct {
|
|
HtUP *ht;
|
|
ut64 addr;
|
|
ut64 size;
|
|
} DeleteRangeCtx;
|
|
|
|
static bool addr_hint_range_delete_cb(void *user, const ut64 key, const void *value) {
|
|
DeleteRangeCtx *ctx = user;
|
|
if (key < ctx->addr || key >= ctx->addr + ctx->size) {
|
|
return true;
|
|
}
|
|
ht_up_delete (ctx->ht, key);
|
|
return true;
|
|
}
|
|
|
|
R_API void r_anal_hint_del(RAnal *a, ut64 addr, ut64 size) {
|
|
if (size <= 1) {
|
|
// only single address
|
|
ht_up_delete (a->addr_hints, addr);
|
|
r_anal_hint_unset_arch (a, addr);
|
|
r_anal_hint_unset_bits (a, addr);
|
|
return;
|
|
}
|
|
// ranged delete
|
|
DeleteRangeCtx ctx = { a->addr_hints, addr, size };
|
|
ht_up_foreach (a->addr_hints, addr_hint_range_delete_cb, &ctx);
|
|
while (true) { // arch
|
|
RBNode *node = r_rbtree_lower_bound (a->arch_hints, &addr, ranged_hint_record_cmp, NULL);
|
|
if (!node) {
|
|
return;
|
|
}
|
|
RAnalRangedHintRecordBase *base = container_of (node, RAnalRangedHintRecordBase, rb);
|
|
if (base->addr >= addr + size) {
|
|
break;
|
|
}
|
|
r_anal_hint_unset_arch (a, base->addr);
|
|
}
|
|
while (true) { // bits
|
|
RBNode *node = r_rbtree_lower_bound (a->bits_hints, &addr, ranged_hint_record_cmp, NULL);
|
|
if (!node) {
|
|
return;
|
|
}
|
|
RAnalRangedHintRecordBase *base = container_of (node, RAnalRangedHintRecordBase, rb);
|
|
if (base->addr >= addr + size) {
|
|
break;
|
|
}
|
|
r_anal_hint_unset_bits (a, base->addr);
|
|
}
|
|
}
|
|
|
|
static void unset_addr_hint_record(RAnal *anal, RAnalAddrHintType type, ut64 addr) {
|
|
RVector *records = ht_up_find (anal->addr_hints, addr, NULL);
|
|
if (!records) {
|
|
return;
|
|
}
|
|
size_t i;
|
|
for (i = 0; i < records->len; i++) {
|
|
RAnalAddrHintRecord *record = r_vector_index_ptr (records, i);
|
|
if (record->type == type) {
|
|
addr_hint_record_fini (record, NULL);
|
|
r_vector_remove_at (records, i, NULL);
|
|
return;
|
|
}
|
|
}
|
|
}
|
|
|
|
// create or return the existing addr hint record of the given type at addr
|
|
static RAnalAddrHintRecord *ensure_addr_hint_record(RAnal *anal, RAnalAddrHintType type, ut64 addr) {
|
|
RVector *records = ht_up_find (anal->addr_hints, addr, NULL);
|
|
if (!records) {
|
|
records = r_vector_new (sizeof (RAnalAddrHintRecord), addr_hint_record_fini, NULL);
|
|
if (!records) {
|
|
return NULL;
|
|
}
|
|
ht_up_insert (anal->addr_hints, addr, records);
|
|
}
|
|
void *pos;
|
|
r_vector_foreach (records, pos) {
|
|
RAnalAddrHintRecord *record = pos;
|
|
if (record->type == type) {
|
|
return record;
|
|
}
|
|
}
|
|
RAnalAddrHintRecord *record = r_vector_push (records, NULL);
|
|
memset (record, 0, sizeof (*record));
|
|
record->type = type;
|
|
return record;
|
|
}
|
|
|
|
#define SET_HINT(type, setcode) do { \
|
|
RAnalAddrHintRecord *r = ensure_addr_hint_record (a, type, addr); \
|
|
if (!r) { \
|
|
break; \
|
|
} \
|
|
setcode \
|
|
} while (0)
|
|
|
|
static RAnalRangedHintRecordBase *ensure_ranged_hint_record(RBTree *tree, ut64 addr, size_t sz) {
|
|
RBNode *node = r_rbtree_find (*tree, &addr, ranged_hint_record_cmp, NULL);
|
|
if (node) {
|
|
return container_of (node, RAnalRangedHintRecordBase, rb);
|
|
}
|
|
RAnalRangedHintRecordBase *record = malloc (sz);
|
|
memset (record, 0, sz);
|
|
if (!record) {
|
|
return NULL;
|
|
}
|
|
record->addr = addr;
|
|
r_rbtree_insert (tree, &addr, &record->rb, ranged_hint_record_cmp, NULL);
|
|
return record;
|
|
}
|
|
|
|
R_API void r_anal_hint_set_offset(RAnal *a, ut64 addr, const char *typeoff) {
|
|
SET_HINT (R_ANAL_ADDR_HINT_TYPE_TYPE_OFFSET,
|
|
free (r->type_offset);
|
|
r->type_offset = strdup (typeoff);
|
|
);
|
|
}
|
|
|
|
R_API void r_anal_hint_set_nword(RAnal *a, ut64 addr, int nword) {
|
|
SET_HINT (R_ANAL_ADDR_HINT_TYPE_NWORD, r->nword = nword;);
|
|
}
|
|
|
|
R_API void r_anal_hint_set_jump(RAnal *a, ut64 addr, ut64 jump) {
|
|
SET_HINT (R_ANAL_ADDR_HINT_TYPE_JUMP, r->jump = jump;);
|
|
}
|
|
|
|
R_API void r_anal_hint_set_fail(RAnal *a, ut64 addr, ut64 fail) {
|
|
SET_HINT (R_ANAL_ADDR_HINT_TYPE_FAIL, r->fail = fail;);
|
|
}
|
|
|
|
R_API void r_anal_hint_set_newbits(RAnal *a, ut64 addr, int bits) {
|
|
SET_HINT (R_ANAL_ADDR_HINT_TYPE_NEW_BITS, r->newbits = bits;);
|
|
}
|
|
|
|
R_API void r_anal_hint_set_high(RAnal *a, ut64 addr) {
|
|
SET_HINT (R_ANAL_ADDR_HINT_TYPE_HIGH,);
|
|
}
|
|
|
|
R_API void r_anal_hint_set_immbase(RAnal *a, ut64 addr, int base) {
|
|
if (base) {
|
|
SET_HINT (R_ANAL_ADDR_HINT_TYPE_IMMBASE, r->immbase = base;);
|
|
} else {
|
|
unset_addr_hint_record (a, R_ANAL_ADDR_HINT_TYPE_IMMBASE, addr);
|
|
}
|
|
}
|
|
|
|
R_API void r_anal_hint_set_pointer(RAnal *a, ut64 addr, ut64 ptr) {
|
|
SET_HINT (R_ANAL_ADDR_HINT_TYPE_PTR, r->ptr = ptr;);
|
|
}
|
|
|
|
R_API void r_anal_hint_set_ret(RAnal *a, ut64 addr, ut64 val) {
|
|
SET_HINT (R_ANAL_ADDR_HINT_TYPE_RET, r->retval = val;);
|
|
}
|
|
|
|
R_API void r_anal_hint_set_syntax(RAnal *a, ut64 addr, const char *syn) {
|
|
SET_HINT (R_ANAL_ADDR_HINT_TYPE_SYNTAX,
|
|
free (r->syntax);
|
|
r->syntax = strdup (syn);
|
|
);
|
|
}
|
|
|
|
R_API void r_anal_hint_set_opcode(RAnal *a, ut64 addr, const char *opcode) {
|
|
SET_HINT (R_ANAL_ADDR_HINT_TYPE_OPCODE,
|
|
free (r->opcode);
|
|
r->opcode = strdup (opcode);
|
|
);
|
|
}
|
|
|
|
R_API void r_anal_hint_set_esil(RAnal *a, ut64 addr, const char *esil) {
|
|
SET_HINT (R_ANAL_ADDR_HINT_TYPE_ESIL,
|
|
free (r->esil);
|
|
r->esil = strdup (esil);
|
|
);
|
|
}
|
|
|
|
R_API void r_anal_hint_set_type(RAnal *a, ut64 addr, int type) {
|
|
SET_HINT (R_ANAL_ADDR_HINT_TYPE_OPTYPE, r->optype = type;);
|
|
}
|
|
|
|
R_API void r_anal_hint_set_size(RAnal *a, ut64 addr, ut64 size) {
|
|
SET_HINT (R_ANAL_ADDR_HINT_TYPE_SIZE, r->size = size;);
|
|
}
|
|
|
|
R_API void r_anal_hint_set_stackframe(RAnal *a, ut64 addr, ut64 size) {
|
|
SET_HINT (R_ANAL_ADDR_HINT_TYPE_STACKFRAME, r->stackframe = size;);
|
|
}
|
|
|
|
R_API void r_anal_hint_set_val(RAnal *a, ut64 addr, ut64 v) {
|
|
SET_HINT (R_ANAL_ADDR_HINT_TYPE_VAL, r->val = v;);
|
|
}
|
|
|
|
R_API void r_anal_hint_set_arch(RAnal *a, ut64 addr, const char *arch) {
|
|
RAnalArchHintRecord *record = (RAnalArchHintRecord *)ensure_ranged_hint_record (&a->arch_hints, addr, sizeof (RAnalArchHintRecord));
|
|
if (!record) {
|
|
return;
|
|
}
|
|
free (record->arch);
|
|
record->arch = arch ? strdup (arch) : NULL;
|
|
}
|
|
|
|
R_API void r_anal_hint_set_bits(RAnal *a, ut64 addr, int bits) {
|
|
RAnalBitsHintRecord *record = (RAnalBitsHintRecord *)ensure_ranged_hint_record (&a->bits_hints, addr, sizeof (RAnalBitsHintRecord));
|
|
if (!record) {
|
|
return;
|
|
}
|
|
record->bits = bits;
|
|
if (a->hint_cbs.on_bits) {
|
|
a->hint_cbs.on_bits (a, addr, bits, true);
|
|
}
|
|
}
|
|
|
|
R_API void r_anal_hint_unset_size(RAnal *a, ut64 addr) {
|
|
unset_addr_hint_record (a, R_ANAL_ADDR_HINT_TYPE_SIZE, addr);
|
|
}
|
|
|
|
R_API void r_anal_hint_unset_esil(RAnal *a, ut64 addr) {
|
|
unset_addr_hint_record (a, R_ANAL_ADDR_HINT_TYPE_ESIL, addr);
|
|
}
|
|
|
|
R_API void r_anal_hint_unset_opcode(RAnal *a, ut64 addr) {
|
|
unset_addr_hint_record (a, R_ANAL_ADDR_HINT_TYPE_OPCODE, addr);
|
|
}
|
|
|
|
R_API void r_anal_hint_unset_high(RAnal *a, ut64 addr) {
|
|
unset_addr_hint_record (a, R_ANAL_ADDR_HINT_TYPE_HIGH, addr);
|
|
}
|
|
|
|
R_API void r_anal_hint_unset_immbase(RAnal *a, ut64 addr) {
|
|
unset_addr_hint_record (a, R_ANAL_ADDR_HINT_TYPE_IMMBASE, addr);
|
|
}
|
|
|
|
R_API void r_anal_hint_unset_nword(RAnal *a, ut64 addr) {
|
|
unset_addr_hint_record (a, R_ANAL_ADDR_HINT_TYPE_NWORD, addr);
|
|
}
|
|
|
|
R_API void r_anal_hint_unset_syntax(RAnal *a, ut64 addr) {
|
|
unset_addr_hint_record (a, R_ANAL_ADDR_HINT_TYPE_SYNTAX, addr);
|
|
}
|
|
|
|
R_API void r_anal_hint_unset_pointer(RAnal *a, ut64 addr) {
|
|
unset_addr_hint_record (a, R_ANAL_ADDR_HINT_TYPE_PTR, addr);
|
|
}
|
|
|
|
R_API void r_anal_hint_unset_ret(RAnal *a, ut64 addr) {
|
|
unset_addr_hint_record (a, R_ANAL_ADDR_HINT_TYPE_RET, addr);
|
|
}
|
|
|
|
R_API void r_anal_hint_unset_offset(RAnal *a, ut64 addr) {
|
|
unset_addr_hint_record (a, R_ANAL_ADDR_HINT_TYPE_TYPE_OFFSET, addr);
|
|
}
|
|
|
|
R_API void r_anal_hint_unset_jump(RAnal *a, ut64 addr) {
|
|
unset_addr_hint_record (a, R_ANAL_ADDR_HINT_TYPE_JUMP, addr);
|
|
}
|
|
|
|
R_API void r_anal_hint_unset_fail(RAnal *a, ut64 addr) {
|
|
unset_addr_hint_record (a, R_ANAL_ADDR_HINT_TYPE_FAIL, addr);
|
|
}
|
|
|
|
R_API void r_anal_hint_unset_newbits(RAnal *a, ut64 addr) {
|
|
unset_addr_hint_record (a, R_ANAL_ADDR_HINT_TYPE_NEW_BITS, addr);
|
|
}
|
|
|
|
R_API void r_anal_hint_unset_val(RAnal *a, ut64 addr) {
|
|
unset_addr_hint_record (a, R_ANAL_ADDR_HINT_TYPE_VAL, addr);
|
|
}
|
|
|
|
R_API void r_anal_hint_unset_type(RAnal *a, ut64 addr) {
|
|
unset_addr_hint_record (a, R_ANAL_ADDR_HINT_TYPE_OPTYPE, addr);
|
|
}
|
|
|
|
R_API void r_anal_hint_unset_stackframe(RAnal *a, ut64 addr) {
|
|
unset_addr_hint_record (a, R_ANAL_ADDR_HINT_TYPE_STACKFRAME, addr);
|
|
}
|
|
|
|
R_API void r_anal_hint_unset_arch(RAnal *a, ut64 addr) {
|
|
r_rbtree_delete (&a->arch_hints, &addr, ranged_hint_record_cmp, NULL, arch_hint_record_free_rb, NULL);
|
|
}
|
|
|
|
R_API void r_anal_hint_unset_bits(RAnal *a, ut64 addr) {
|
|
r_rbtree_delete (&a->bits_hints, &addr, ranged_hint_record_cmp, NULL, bits_hint_record_free_rb, NULL);
|
|
}
|
|
|
|
R_API void r_anal_hint_free(RAnalHint *h) {
|
|
if (h) {
|
|
free (h->arch);
|
|
free (h->esil);
|
|
free (h->opcode);
|
|
free (h->syntax);
|
|
free (h->offset);
|
|
free (h);
|
|
}
|
|
}
|
|
|
|
R_API R_NULLABLE R_BORROW const char *r_anal_hint_arch_at(RAnal *anal, ut64 addr, R_NULLABLE ut64 *hint_addr) {
|
|
RBNode *node = r_rbtree_upper_bound (anal->arch_hints, &addr, ranged_hint_record_cmp, NULL);
|
|
if (!node) {
|
|
if (hint_addr) {
|
|
*hint_addr = UT64_MAX;
|
|
}
|
|
return NULL;
|
|
}
|
|
RAnalArchHintRecord *record = (RAnalArchHintRecord *)container_of (node, RAnalRangedHintRecordBase, rb);
|
|
if (hint_addr) {
|
|
*hint_addr = record->base.addr;
|
|
}
|
|
return record->arch;
|
|
}
|
|
|
|
R_API int r_anal_hint_bits_at(RAnal *anal, ut64 addr, R_NULLABLE ut64 *hint_addr) {
|
|
RBNode *node = r_rbtree_upper_bound (anal->bits_hints, &addr, ranged_hint_record_cmp, NULL);
|
|
if (!node) {
|
|
if (hint_addr) {
|
|
*hint_addr = UT64_MAX;
|
|
}
|
|
return 0;
|
|
}
|
|
RAnalBitsHintRecord *record = (RAnalBitsHintRecord *)container_of (node, RAnalRangedHintRecordBase, rb);
|
|
if (hint_addr) {
|
|
*hint_addr = record->base.addr;
|
|
}
|
|
return record->bits;
|
|
}
|
|
|
|
R_API R_NULLABLE const RVector/*<const RAnalAddrHintRecord>*/ *r_anal_addr_hints_at(RAnal *anal, ut64 addr) {
|
|
return ht_up_find (anal->addr_hints, addr, NULL);
|
|
}
|
|
|
|
typedef struct {
|
|
RAnalAddrHintRecordsCb cb;
|
|
void *user;
|
|
} AddrHintForeachCtx;
|
|
|
|
static bool addr_hint_foreach_cb(void *user, const ut64 key, const void *value) {
|
|
AddrHintForeachCtx *ctx = user;
|
|
return ctx->cb (key, value, ctx->user);
|
|
}
|
|
|
|
R_API void r_anal_addr_hints_foreach(RAnal *anal, RAnalAddrHintRecordsCb cb, void *user) {
|
|
AddrHintForeachCtx ctx = { cb, user };
|
|
ht_up_foreach (anal->addr_hints, addr_hint_foreach_cb, &ctx);
|
|
}
|
|
|
|
R_API void r_anal_arch_hints_foreach(RAnal *anal, RAnalArchHintCb cb, void *user) {
|
|
RBIter iter;
|
|
RAnalRangedHintRecordBase *record;
|
|
r_rbtree_foreach (anal->arch_hints, iter, record, RAnalRangedHintRecordBase, rb) {
|
|
bool cont = cb (record->addr, ((RAnalArchHintRecord *)record)->arch, user);
|
|
if (!cont) {
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
|
|
R_API void r_anal_bits_hints_foreach(RAnal *anal, RAnalBitsHintCb cb, void *user) {
|
|
RBIter iter;
|
|
RAnalRangedHintRecordBase *record;
|
|
r_rbtree_foreach (anal->bits_hints, iter, record, RAnalRangedHintRecordBase, rb) {
|
|
bool cont = cb (record->addr, ((RAnalBitsHintRecord *)record)->bits, user);
|
|
if (!cont) {
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
|
|
static void hint_merge(RAnalHint *hint, RAnalAddrHintRecord *record) {
|
|
switch (record->type) {
|
|
case R_ANAL_ADDR_HINT_TYPE_IMMBASE:
|
|
hint->immbase = record->immbase;
|
|
break;
|
|
case R_ANAL_ADDR_HINT_TYPE_JUMP:
|
|
hint->jump = record->jump;
|
|
break;
|
|
case R_ANAL_ADDR_HINT_TYPE_FAIL:
|
|
hint->fail = record->fail;
|
|
break;
|
|
case R_ANAL_ADDR_HINT_TYPE_STACKFRAME:
|
|
hint->stackframe = record->stackframe;
|
|
break;
|
|
case R_ANAL_ADDR_HINT_TYPE_PTR:
|
|
hint->ptr = record->ptr;
|
|
break;
|
|
case R_ANAL_ADDR_HINT_TYPE_NWORD:
|
|
hint->nword = record->nword;
|
|
break;
|
|
case R_ANAL_ADDR_HINT_TYPE_RET:
|
|
hint->ret = record->retval;
|
|
break;
|
|
case R_ANAL_ADDR_HINT_TYPE_NEW_BITS:
|
|
hint->new_bits = record->newbits;
|
|
break;
|
|
case R_ANAL_ADDR_HINT_TYPE_SIZE:
|
|
hint->size = record->size;
|
|
break;
|
|
case R_ANAL_ADDR_HINT_TYPE_SYNTAX:
|
|
hint->syntax = record->syntax ? strdup (record->syntax) : NULL;
|
|
break;
|
|
case R_ANAL_ADDR_HINT_TYPE_OPTYPE:
|
|
hint->type = record->optype;
|
|
break;
|
|
case R_ANAL_ADDR_HINT_TYPE_OPCODE:
|
|
hint->opcode = record->opcode ? strdup (record->opcode) : NULL;
|
|
break;
|
|
case R_ANAL_ADDR_HINT_TYPE_TYPE_OFFSET:
|
|
hint->offset = record->type_offset ? strdup (record->type_offset) : NULL;
|
|
break;
|
|
case R_ANAL_ADDR_HINT_TYPE_ESIL:
|
|
hint->esil = record->esil ? strdup (record->esil) : NULL;
|
|
break;
|
|
case R_ANAL_ADDR_HINT_TYPE_HIGH:
|
|
hint->high = true;
|
|
break;
|
|
case R_ANAL_ADDR_HINT_TYPE_VAL:
|
|
hint->val = record->val;
|
|
break;
|
|
}
|
|
}
|
|
|
|
R_API RAnalHint *r_anal_hint_get(RAnal *a, ut64 addr) {
|
|
RAnalHint *hint = R_NEW0 (RAnalHint);
|
|
if (!hint) {
|
|
return NULL;
|
|
}
|
|
hint->addr = addr;
|
|
hint->jump = UT64_MAX;
|
|
hint->fail = UT64_MAX;
|
|
hint->ret = UT64_MAX;
|
|
hint->val = UT64_MAX;
|
|
hint->stackframe = UT64_MAX;
|
|
const RVector *records = r_anal_addr_hints_at (a, addr);
|
|
if (records) {
|
|
RAnalAddrHintRecord *record;
|
|
r_vector_foreach (records, record) {
|
|
hint_merge (hint, record);
|
|
}
|
|
}
|
|
const char *arch = r_anal_hint_arch_at (a, addr, NULL);
|
|
hint->arch = arch ? strdup (arch) : NULL;
|
|
hint->bits = r_anal_hint_bits_at (a, addr, NULL);
|
|
if ((!records || r_vector_empty (records)) && !hint->arch && !hint->bits) {
|
|
// no hints found
|
|
free (hint);
|
|
return NULL;
|
|
}
|
|
return hint;
|
|
}
|