radare2/libr/core/cmd_zign.inc.c

1422 lines
36 KiB
C

/* radare - LGPL - Copyright 2009-2024 - pancake, nibble, Dennis Goodlett */
#if R_INCLUDE_BEGIN
static RCoreHelpMessage help_msg_z = {
"Usage:", "z[*j-aof/cs] [args] ", "# Manage zignatures",
"z", " ([addr])", "show/list zignatures", // TODO: rename to zl ?
"z.", "", "find matching zignatures in current offset", // rename to 'z' ?
"z,", "([:help])", "list zignatures loaded in table format (see z,:help)",
"zb", "[?][n=5]", "search for best match",
"zd", "zignature", "diff current function and signature",
"z*", " ([addr])", "show zignatures in radare format",
"zq", " ([addr])", "show zignatures in quiet mode",
"zj", " ([addr])", "show zignatures in json format",
"zk", "", "show zignatures in sdb format",
"z-", "zignature", "delete zignature",
"z-", "*", "delete all zignatures",
"za", "[?]", "add zignature",
"zg", "", "generate zignatures (alias for zaF)",
"zo", "[?]", "manage zignature files",
"zf", "[?]", "manage FLIRT signatures",
"z/", "[?]", "search zignatures",
"zc", "[?]", "compare current zignspace zignatures with another one",
"zs", "[?]", "manage zignspaces",
"zi", " [text]", "show zignatures matching information",
NULL
};
static RCoreHelpMessage help_msg_zb = {
"Usage:", "zb[r?] [args]", "# search for closest matching signatures",
"zb ", "[n]", "find n closest matching zignatures to function at current offset",
"zbr ", "zigname [n]", "search for n most similar functions to zigname",
NULL
};
static RCoreHelpMessage help_msg_z_slash = {
"Usage:", "z/[f*] ", "# Search signatures (see 'e?search' for options)",
"z/ ", "", "search zignatures on range and flag matches",
"z/f ", "", "zignature search on known functions",
"z/* ", "", "search zignatures on range and output radare commands",
NULL
};
static RCoreHelpMessage help_msg_za = {
"Usage:", "za[fFM?] [args] ", "# Add zignature",
"za ", "zigname type params", "add zignature",
"zac ", "", "Compute collisions between signatures",
"zaf ", "[fcnname] [zigname]", "create zignature for function",
"zaF ", "", "generate zignatures for all functions",
"zaM ", "", "Same as zaF but merge signatures of same name",
"za?? ", "", "show extended help",
NULL
};
static RCoreHelpMessage help_msg_zf = {
"Usage:", "zf[dsz] filename ", "# Manage FLIRT signatures",
"zfd ", "filename", "open FLIRT file and dump",
"zfs ", "filename", "open FLIRT file and scan",
"zfs ", "/path/**.sig", "recursively search for FLIRT files and scan them (see dir.depth)",
"zfz ", "filename", "open FLIRT file and get sig commands (zfz flirt_file > zignatures.sig)",
NULL
};
static RCoreHelpMessage help_msg_zo = {
"Usage:", "zo[zs] filename ", "# Manage zignature files (see dir.zigns)",
"zo ", "filename", "load zinatures from sdb file",
"zoz ", "filename", "load zinatures from gzipped sdb file",
"zos ", "filename", "save zignatures to sdb file (merge if file exists)",
NULL
};
static RCoreHelpMessage help_msg_zs = {
"Usage:", "zs[+-*] [namespace] ", "# Manage zignspaces",
"zs", "", "display zignspaces",
"zs ", "zignspace", "select zignspace",
"zs ", "*", "select all zignspaces",
"zs-", "zignspace", "delete zignspace",
"zs-", "*", "delete all zignspaces",
"zs+", "zignspace", "push previous zignspace and set",
"zs-", "", "pop to the previous zignspace",
"zsr ", "newname", "rename selected zignspace",
NULL
};
static RCoreHelpMessage help_msg_zc = {
"Usage:", "zc[n!] other_space ", "# Compare zignspaces, match >= threshold (e zign.diff.*)",
"zc", " other_space", "compare all current space with other_space",
"zcn", " other_space", "compare current space with zigns with same name on other_space",
"zcn!", " other_space", "same as above but show the ones not matching",
NULL
};
static bool addGraphZign(RCore *core, const char *name, RList *args) {
RSignGraph graph = { .cc = -1, .nbbs = -1, .edges = -1, .ebbs = -1, .bbsum = 0 };
char *ptr;
RListIter *iter;
r_list_foreach (args, iter, ptr) {
if (r_str_startswith (ptr, "cc=")) {
graph.cc = atoi (ptr + 3);
} else if (r_str_startswith (ptr, "nbbs=")) {
graph.nbbs = atoi (ptr + 5);
} else if (r_str_startswith (ptr, "edges=")) {
graph.edges = atoi (ptr + 6);
} else if (r_str_startswith (ptr, "ebbs=")) {
graph.ebbs = atoi (ptr + 5);
} else if (r_str_startswith (ptr, "bbsum=")) {
graph.bbsum = atoi (ptr + 6);
} else {
return false;
}
}
return r_sign_add_graph (core->anal, name, graph);
}
// R2_600 - move to libr/anal/sign.c as public api
static RSignItem *item_new_named(RAnal *a, const char *n) {
RSignItem *it = r_sign_item_new ();
if (it && (it->name = strdup (n))) {
it->space = r_spaces_current (&a->zign_spaces);
return it;
}
r_sign_item_free (it);
return NULL;
}
static bool r_sign_add_next(RAnal *a, const char *name, const char *nextname) {
R_RETURN_VAL_IF_FAIL (a && name && nextname, false);
bool retval = false;
RSignItem *it = item_new_named (a, name);
if (it) {
it->next = strdup (nextname);
retval = r_sign_add_item (a, it);
r_sign_item_free (it);
}
return retval;
}
static inline bool za_add(RCore *core, const char *input) {
char *args = r_str_trim_dup (input + 1);
if (!args) {
return false;
}
char *save_ptr = NULL;
char *name = r_str_tok_r (args, " ", &save_ptr);
char *stype = r_str_tok_r (NULL, " ", &save_ptr);
char *sig = r_str_tok_r (NULL, "", &save_ptr);
if (!stype || !sig || stype[1] != '\0') {
R_LOG_ERROR ("Invalid input");
free (args);
return false;
}
char t = *stype;
RList *lst = NULL;
bool ret = false;
switch (t) {
case R_SIGN_BYTES:
ret = r_sign_add_bytes (core->anal, name, sig);
break;
case R_SIGN_ANAL:
ret = r_sign_add_anal (core->anal, name, sig);
break;
case R_SIGN_GRAPH:
lst = r_str_split_list (sig, " ", 0);
ret = addGraphZign (core, name, lst);
break;
case R_SIGN_COMMENT:
ret = r_sign_add_comment (core->anal, name, sig);
break;
case R_SIGN_NAME:
ret = r_sign_add_name (core->anal, name, sig);
break;
case R_SIGN_TYPES:
ret = r_sign_add_types (core->anal, name, sig);
break;
case R_SIGN_OFFSET:
{
ut64 offset = r_num_get (core->num, sig);
ret = r_sign_add_addr (core->anal, name, offset);
}
break;
case R_SIGN_REFS:
lst = r_str_split_list (sig, " ", 0);
r_sign_add_refs (core->anal, name, lst);
break;
case R_SIGN_XREFS:
lst = r_str_split_list (sig, " ", 0);
r_sign_add_xrefs (core->anal, name, lst);
break;
case R_SIGN_VARS:
r_sign_add_vars (core->anal, name, sig);
break;
case R_SIGN_BBHASH:
ret = r_sign_add_hash (core->anal, name, t, sig, strlen (sig));
break;
case R_SIGN_NEXT:
// TODO
ret = r_sign_add_next (core->anal, name, sig);
break;
default:
R_LOG_ERROR ("Unknown zignature type: %s", stype);
}
r_list_free (lst);
free (args);
return ret;
}
static void cmd_za_detailed_help(void) {
r_cons_printf ("Adding Zignatures (examples and documentation)\n\n"
"Zignature types:\n"
" a: bytes pattern, r2 creates mask from analysis\n"
" b: bytes pattern\n"
" c: base64 comment\n"
" n: real function name\n"
" g: graph metrics\n"
" o: original offset\n"
" r: references\n"
" x: cross references\n"
" h: bbhash (hashing of fcn basic blocks)\n"
" v: vars (and args)\n"
"Bytes patterns:\n"
" bytes can contain '..' (dots) to specify a binary mask\n\n"
"Graph metrics:\n"
" cc: cyclomatic complexity\n"
" edges: number of edges\n"
" nbbs: number of basic blocks\n"
" ebbs: number of end basic blocks\n\n"
"Examples:\n"
" za foo b 558bec..e8........\n"
" za foo a e811223344\n"
" za foo g cc=2 nbbs=3 edges=3 ebbs=1\n"
" za foo g nbbs=3 edges=3\n"
" za foo v b-32 b-48 b-64\n"
" za foo o 0x08048123\n"
" za foo c this is a comment (base64?)\n"
" za foo r sym.imp.strcpy sym.imp.sprintf sym.imp.strlen\n"
" za foo h 2c26b46b68ffc68ff99b453c1d30413413422d706483bfa0f98a5e886266e7ae\n");
}
static int cmd_za(void *data, const char *input) {
RCore *core = (RCore *)data;
if (r_str_endswith (input, "??")) {
cmd_za_detailed_help ();
return 0;
}
if (*input && input[1] == '?') {
char two[3] = { input[0], input[1], 0 };
r_core_cmd_help_contains (core, help_msg_za, two);
return 0;
}
switch (*input) {
case ' ': // "za"
return za_add (core, input);
case 'f': // "zaf"
{
char *args = r_str_trim_dup (input + 1);
int n = r_str_word_set0 (args);
if (input[1] == '?' || n > 2) {
r_core_cmd_help_contains (core, help_msg_za, "zaf");
free (args);
return false;
}
RAnalFunction *fcni = NULL;
const char *zigname = (n == 2)? r_str_word_get0 (args, 1): NULL;
if (n > 0) {
fcni = r_anal_get_function_byname (core->anal, r_str_word_get0 (args, 0));
} else {
fcni = r_anal_get_function_at (core->anal, core->offset);
}
if (fcni) {
r_sign_add_func (core->anal, fcni, zigname);
}
free (args);
if (!fcni) {
R_LOG_ERROR ("Could not find function");
return false;
}
}
break;
case 'c': // "zac"
if (input[1] == '?') {
r_core_cmd_help_contains (core, help_msg_za, "zac");
} else {
r_cons_break_push (NULL, NULL);
r_sign_resolve_collisions (core->anal);
r_cons_break_pop ();
}
break;
case 'F': // "zaF"
if (input[1] == '?') {
r_core_cmd_help_contains (core, help_msg_za, "zaF");
} else {
int count = r_sign_all_functions (core->anal, false);
R_LOG_INFO ("generated zignatures: %d", count);
}
break;
case 'M': // "zaM"
if (input[1] == '?') {
r_core_cmd_help_contains (core, help_msg_za, "zaM");
} else {
int count = r_sign_all_functions (core->anal, true);
R_LOG_INFO ("generated zignatures: %d", count);
}
break;
case '?': // “za?”
r_core_cmd_help (core, help_msg_za);
break;
default:
r_core_return_invalid_command (core, "za", *input);
return false;
}
return true;
}
static int cmd_zo(void *data, const char *input) {
RCore *core = (RCore *)data;
switch (*input) {
case ' ': // "zo"
if (input[1]) {
return r_sign_load (core->anal, r_str_trim_head_ro (input + 1), false);
}
r_core_cmd_help_contains (core, help_msg_zo, "zo");
return false;
case 's': // "zos"
if (input[1] == ' ' && input[2]) {
return r_sign_save (core->anal, r_str_trim_head_ro (input + 2));
}
r_core_cmd_help_contains (core, help_msg_zo, "zos");
return false;
case 'z': // "zoz"
if (input[1] == ' ' && input[2]) {
return r_sign_load_gz (core->anal, input + 2, false);
}
r_core_cmd_help_contains (core, help_msg_zo, "zoz");
return false;
case '?': // "zo?"
r_core_cmd_help (core, help_msg_zo);
break;
default:
r_core_return_invalid_command (core, "zo", *input);
return false;
}
return true;
}
static int cmd_zs(void *data, const char *input) {
RCore *core = (RCore *) data;
RSpaces *zs = &core->anal->zign_spaces;
switch (*input) {
case '+': // "zs+"
if (!input[1]) {
r_core_cmd_help_contains (core, help_msg_zs, "zs+");
return false;
}
char *sp = r_str_trim_dup (input + 1);
if (sp) {
r_spaces_push (zs, sp);
free (sp);
}
break;
case 'r': // "zsr"
if (input[1] != ' ' || !input[2]) {
r_core_cmd_help_contains (core, help_msg_zs, "zsr");
return false;
}
r_spaces_rename (zs, NULL, input + 2);
break;
case '-': // "zs-"
if (input[1] == '\x00') {
r_spaces_pop (zs);
} else if (input[1] == '*') {
r_spaces_unset (zs, NULL);
} else {
r_spaces_unset (zs, input + 1);
}
break;
case 'j': // "zsj"
case '*': // "zs*"
case '\0':
spaces_list (core, zs, input[0]);
break;
case ' ': // "zs"
if (!input[1]) {
r_core_cmd_help (core, help_msg_zs);
return false;
}
r_spaces_set (zs, input + 1);
break;
case '?': // "zs?"
r_core_cmd_help (core, help_msg_zs);
break;
default:
r_core_return_invalid_command (core, "zs", *input);
return false;
}
return true;
}
static int cmd_zf(void *data, const char *input) {
RCore *core = (RCore *)data;
switch (*input) {
case 'd': // "zfd"
// TODO
if (input[1] != ' ') {
r_core_cmd_help_contains (core, help_msg_zf, "zfd");
return false;
}
r_sign_flirt_dump (core->anal, input + 2);
break;
case 's': // "zfs"
// TODO
if (input[1] != ' ') {
r_core_cmd_help_contains (core, help_msg_zf, "zfs");
return false;
}
int depth = r_config_get_i (core->config, "dir.depth");
char *file;
RListIter *iter;
RList *files = r_file_glob (input + 2, depth);
r_list_foreach (files, iter, file) {
r_sign_flirt_scan (core->anal, file);
}
r_list_free (files);
break;
case 'z': // "zfz"
// TODO
break;
case '?': // "zf?"
r_core_cmd_help (core, help_msg_zf);
break;
default:
r_core_return_invalid_command (core, "zf", *input);
return false;
}
return true;
}
struct ctxSearchCB {
RCore *core;
bool bytes_only;
bool rad;
int collisions;
int newfuncs;
int count;
int bytes_count;
int graph_count;
int offset_count;
int refs_count;
int types_count;
int bbhash_count;
int next_count;
};
static void apply_name(RCore *core, RAnalFunction *fcn, RSignItem *it, bool rad) {
R_RETURN_IF_FAIL (core && fcn && it && it->name);
const char *name = it->realname? it->realname: it->name;
if (rad) {
char *tmp = r_name_filter_dup (name);
if (tmp) {
r_cons_printf ("'@0x%08"PFMT64x"'afn %s\n", fcn->addr, tmp);
free (tmp);
}
return;
}
RFlagItem *flag = r_flag_get (core->flags, fcn->name);
if (flag && flag->space && strcmp (flag->space->name, R_FLAGS_FS_FUNCTIONS)) {
r_flag_rename (core->flags, flag, name);
}
r_anal_function_rename (fcn, name);
if (core->anal->cb.on_fcn_rename) {
core->anal->cb.on_fcn_rename (core->anal, core->anal->user, fcn, name);
}
}
static void apply_flag(RCore *core, RSignItem *it, ut64 addr, int size, int count, const char *prefix, bool rad) {
const char *zign_prefix = r_config_get (core->config, "zign.prefix");
char *name = r_str_newf ("%s.%s.%s_%d", zign_prefix, prefix, it->name, count);
if (name) {
if (rad) {
char *tmp = r_name_filter_dup (name);
if (tmp) {
r_cons_printf ("f %s %d @ 0x%08" PFMT64x "\n", tmp, size, addr);
free (tmp);
}
} else {
r_flag_set (core->flags, name, addr, size);
}
free (name);
}
}
static int searchBytesHitCB(RSignItem *it, RSearchKeyword *kw, ut64 addr, void *user) {
struct ctxSearchCB *ctx = (struct ctxSearchCB *)user;
RAnalFunction *fcn = r_anal_get_fcn_in (ctx->core->anal, addr, 0);
if (!fcn) {
r_core_af (ctx->core, addr, NULL, false);
fcn = r_anal_get_fcn_in (ctx->core->anal, addr, 0);
ctx->newfuncs++;
ctx->count++;
}
apply_flag (ctx->core, it, addr, kw->keyword_length, kw->count, "bytes", ctx->rad);
if (ctx->bytes_only) {
if (fcn) {
char *tmp = NULL;
apply_name (ctx->core, fcn, it, ctx->rad);
if (it->types && (tmp = r_str_newf ("%s;", it->types))) { // apply types
r_anal_str_to_fcn (ctx->core->anal, fcn, tmp);
free (tmp);
}
if (it->vars) {
r_anal_function_set_var_prot (fcn, it->vars);
}
}
ctx->bytes_count++;
}
return 1;
}
static int fcnMatchCB(RSignItem *it, RAnalFunction *fcn, RSignType *types, void *user, RList *col) {
R_RETURN_VAL_IF_FAIL (types && *types != R_SIGN_END, 1);
struct ctxSearchCB *ctx = (struct ctxSearchCB *)user;
ut64 sz = r_anal_function_realsize (fcn);
RSignType t;
bool collides = false;
if (!col || !r_list_empty (col)) {
// NULL col implies collision computation err, so assume collides
collides = true;
ctx->collisions++;
}
int i = 0;
while ((t = types[i++]) != R_SIGN_END) {
const char *prefix = NULL;
switch (t) {
case R_SIGN_BYTES:
ctx->bytes_count++;
prefix = "bytes_func";
break;
case R_SIGN_GRAPH:
prefix = "graph";
ctx->graph_count++;
break;
case R_SIGN_OFFSET:
prefix = "offset";
ctx->offset_count++;
break;
case R_SIGN_REFS:
prefix = "refs";
ctx->refs_count++;
break;
case R_SIGN_TYPES:
prefix = "types";
ctx->types_count++;
break;
case R_SIGN_BBHASH:
prefix = "bbhash";
ctx->bbhash_count++;
break;
case R_SIGN_NEXT:
prefix = "next";
ctx->next_count++;
break;
default:
r_warn_if_reached ();
break;
}
if (prefix) {
char *tmp = NULL;
if (collides && (tmp = r_str_newf ("%s_collision", prefix))) {
apply_flag (ctx->core, it, fcn->addr, sz, ctx->count, tmp, ctx->rad);
free (tmp);
} else {
apply_flag (ctx->core, it, fcn->addr, sz, ctx->count, prefix, ctx->rad);
}
ctx->count++;
}
}
if (!collides) {
char *tmp = NULL;
apply_name (ctx->core, fcn, it, ctx->rad);
if (it->types && (tmp = r_str_newf ("%s;", it->types))) { // apply types
r_anal_str_to_fcn (ctx->core->anal, fcn, tmp);
free (tmp);
}
if (it->vars) {
r_anal_function_set_var_prot (fcn, it->vars);
}
}
return 1;
}
static bool searchRange(RCore *core, ut64 from, ut64 to, bool rad, struct ctxSearchCB *ctx) {
ut8 *buf = malloc (core->blocksize);
ut64 at;
int rlen;
bool retval = true;
int minsz = r_config_get_i (core->config, "zign.minsz");
if (!buf) {
return false;
}
RSignSearch *ss = r_sign_search_new ();
ss->search->align = r_config_get_i (core->config, "search.align");
r_sign_search_init (core->anal, ss, minsz, searchBytesHitCB, ctx);
r_cons_break_push (NULL, NULL);
for (at = from; at < to; at += core->blocksize) {
if (r_cons_is_breaked ()) {
retval = false;
break;
}
rlen = R_MIN (core->blocksize, to - at);
if (!r_io_is_valid_offset (core->io, at, 0)) {
retval = false;
break;
}
(void)r_io_read_at (core->io, at, buf, rlen);
if (r_sign_search_update (core->anal, ss, &at, buf, rlen) == -1) {
R_LOG_ERROR ("search update read error at 0x%08" PFMT64x, at);
retval = false;
break;
}
}
r_cons_break_pop ();
free (buf);
r_sign_search_free (ss);
return retval;
}
static void search_add_to_types(RCore *c, RSignSearchMetrics *sm, RSignType t, const char *str, unsigned int *i) {
unsigned int count = *i;
R_RETURN_IF_FAIL (count < sizeof (sm->stypes) / sizeof (RSignType) - 1);
if (r_config_get_i (c->config, str)) {
sm->stypes[count++] = t;
sm->stypes[count] = 0;
*i = count;
}
}
static bool fill_search_metrics(RSignSearchMetrics *sm, RCore *c, void *user) {
unsigned int i = 0;
sm->stypes[0] = R_SIGN_END;
search_add_to_types (c, sm, R_SIGN_BYTES, "zign.bytes", &i);
search_add_to_types (c, sm, R_SIGN_GRAPH, "zign.graph", &i);
search_add_to_types (c, sm, R_SIGN_OFFSET, "zign.offset", &i);
search_add_to_types (c, sm, R_SIGN_REFS, "zign.refs", &i);
search_add_to_types (c, sm, R_SIGN_BBHASH, "zign.hash", &i);
search_add_to_types (c, sm, R_SIGN_TYPES, "zign.types", &i);
#if 0
// untested
search_add_to_types(c, sm, R_SIGN_VARS, "zign.vars", &i);
#endif
sm->mincc = r_config_get_i (c->config, "zign.mincc");
sm->minsz = r_config_get_i (c->config, "zign.minsz");
sm->anal = c->anal;
sm->cb = fcnMatchCB;
sm->user = user;
return (i > 0);
}
static void print_ctx_hits(struct ctxSearchCB *ctx) {
int prints = 0;
if (ctx->newfuncs) {
eprintf ("New functions: %d\n", ctx->newfuncs);
}
if (ctx->collisions) {
eprintf ("collisions: %d\n", ctx->collisions);
}
if (ctx->bytes_count) {
eprintf ("bytes: %d\n", ctx->bytes_count);
prints++;
}
if (ctx->graph_count) {
eprintf ("graph: %d\n", ctx->graph_count);
prints++;
}
if (ctx->offset_count) {
eprintf ("offset: %d\n", ctx->offset_count);
prints++;
}
if (ctx->refs_count) {
eprintf ("refs: %d\n", ctx->refs_count);
prints++;
}
if (ctx->types_count) {
eprintf ("types: %d\n", ctx->types_count);
prints++;
}
if (ctx->bbhash_count) {
eprintf ("bbhash: %d\n", ctx->bbhash_count);
prints++;
}
if (ctx->next_count) {
eprintf ("next: %d\n", ctx->next_count);
prints++;
}
if (prints > 1) {
eprintf ("total: %d\n", ctx->count);
}
}
static bool search(RCore *core, bool rad, bool only_func) {
const char *zign_prefix = r_config_get (core->config, "zign.prefix");
if (rad) {
r_cons_printf ("fs+%s\n", zign_prefix);
} else {
if (!r_flag_space_push (core->flags, zign_prefix)) {
R_LOG_ERROR ("cannot create flagspace");
return false;
}
}
struct ctxSearchCB ctx;
memset (&ctx, 0, sizeof (struct ctxSearchCB));
ctx.rad = rad;
ctx.core = core;
RSignSearchMetrics sm;
bool metsearch = fill_search_metrics (&sm, core, (void *)&ctx);
if (!metsearch) {
R_LOG_ERROR ("No zign types enabled");
return false;
}
if (sm.stypes[0] == R_SIGN_BYTES && sm.stypes[1] == R_SIGN_END) {
ctx.bytes_only = true;
}
// Bytes search
if (r_config_get_i (core->config, "zign.bytes") && !only_func) {
const char *mode = r_config_get (core->config, "search.in");
RList *list = r_core_get_boundaries_prot (core, -1, mode, "search");
if (!list) {
return false;
}
RListIter *iter;
RIOMap *map;
r_list_foreach (list, iter, map) {
R_LOG_INFO ("searching 0x%08"PFMT64x" - 0x%08"PFMT64x, r_io_map_begin (map), r_io_map_end (map));
searchRange (core, r_io_map_begin (map), r_io_map_end (map), rad, &ctx);
}
r_list_free (list);
}
// Function search
if (!ctx.bytes_only) {
R_LOG_INFO ("searching function metrics");
r_sign_metric_search (core->anal, &sm);
}
if (rad) {
r_cons_printf ("fs-\n");
} else {
if (!r_flag_space_pop (core->flags)) {
R_LOG_ERROR ("cannot restore flagspace");
return false;
}
}
print_ctx_hits (&ctx);
return ctx.count > 0;
}
static void print_possible_matches(RList *list, bool json, RCore *core) {
RListIter *itr;
RSignCloseMatch *row;
if (json) {
PJ *pj = core->anal->coreb.pjWithEncoding (core);
pj_a (pj);
r_list_foreach (list, itr, row) {
pj_o (pj);
pj_ks (pj, "name", row->item->name);
pj_kd (pj, "similarity", row->score);
pj_kd (pj, "byte similarity", row->bscore);
pj_kd (pj, "graph similarity", row->gscore);
pj_end (pj);
}
pj_end (pj);
r_cons_printf ("%s\n", pj_string (pj));
pj_free (pj);
} else {
r_list_foreach (list, itr, row) {
// total score
if (row->bscore > 0.0 && row->gscore > 0.0) {
r_cons_printf ("%02.5lf ", row->score);
}
if (row->bscore > 0.0) {
r_cons_printf ("%02.5lf B ", row->bscore);
}
if (row->gscore > 0.0) {
r_cons_printf ("%02.5lf G ", row->gscore);
}
r_cons_printf (" %s\n", row->item->name);
}
}
}
static RSignItem *item_frm_signame(RAnal *a, const char *signame) {
// example zign|*|sym.unlink_blk
const RSpace *space = r_spaces_current (&a->zign_spaces);
char *k = r_str_newf ("zign|%s|%s", space? space->name: "*", signame);
char *value = sdb_querys (a->sdb_zigns, NULL, 0, k);
if (!value) {
free (k);
return NULL;
}
RSignItem *it = r_sign_item_new ();
if (!it) {
free (k);
free (value);
return NULL;
}
if (!r_sign_deserialize (a, it, k, value)) {
r_sign_item_free (it);
it = NULL;
}
free (k);
free (value);
return it;
}
static double get_zb_threshold(RCore *core) {
const char *th = r_config_get (core->config, "zign.threshold");
double thresh = r_num_get_double (NULL, th);
if (thresh < 0.0 || thresh > 1.0) {
R_LOG_ERROR ("Invalid zign.threshold %s, using 0.0", th);
thresh = 0.0;
}
return thresh;
}
static bool bestmatch_fcn(RCore *core, const char *input, bool json) {
R_RETURN_VAL_IF_FAIL (input && core, false);
char *argv = strdup (input);
if (!argv) {
return false;
}
int count = 5;
char *save_ptr = NULL;
char *zigname = r_str_tok_r (argv, " ", &save_ptr);
if (!zigname) {
R_LOG_ERROR ("Need a signature");
free (argv);
return false;
}
char *cs = r_str_tok_r (NULL, " ", &save_ptr);
if (cs) {
if ((count = atoi (cs)) <= 0) {
free (argv);
R_LOG_ERROR ("Invalid count");
return false;
}
if (r_str_tok_r (NULL, " ", &save_ptr)) {
free (argv);
R_LOG_ERROR ("Too many parameters");
return false;
}
}
RSignItem *it = item_frm_signame (core->anal, zigname);
if (!it) {
R_LOG_ERROR ("Couldn't get signature for %s", zigname);
free (argv);
return false;
}
free (argv);
if (!r_config_get_b (core->config, "zign.bytes")) {
r_sign_bytes_free (it->bytes);
it->bytes = NULL;
}
if (!r_config_get_b (core->config, "zign.graph")) {
r_sign_graph_free (it->graph);
it->graph = NULL;
}
double thresh = get_zb_threshold (core);
RList *list = r_sign_find_closest_fcn (core->anal, it, count, thresh);
r_sign_item_free (it);
if (list) {
print_possible_matches (list, json, core);
r_list_free (list);
return true;
}
return false;
}
static bool bestmatch_sig(RCore *core, const char *input, bool json) {
R_RETURN_VAL_IF_FAIL (input && core, false);
int count = 5;
if (!R_STR_ISEMPTY (input)) {
count = atoi (input);
if (count <= 0) {
R_LOG_ERROR ("invalid number %s", input);
return false;
}
}
RAnalFunction *fcn = r_anal_get_fcn_in (core->anal, core->offset, 0);
if (!fcn) {
R_LOG_ERROR ("No function at 0x%08" PFMT64x, core->offset);
return false;
}
RSignItem *item = r_sign_item_new ();
if (!item) {
return false;
}
if (r_config_get_b (core->config, "zign.bytes")) {
r_sign_addto_item (core->anal, item, fcn, R_SIGN_BYTES);
RSignBytes *b = item->bytes;
int minsz = r_config_get_i (core->config, "zign.minsz");
if (b && b->size < minsz) {
R_LOG_WARN ("Function signature is too small (%d < %d) See e zign.minsz", b->size, minsz);
}
}
if (r_config_get_b (core->config, "zign.graph")) {
r_sign_addto_item (core->anal, item, fcn, R_SIGN_GRAPH);
}
double th = get_zb_threshold (core);
bool found = false;
if (item->graph || item->bytes) {
r_cons_break_push (NULL, NULL);
RList *list = r_sign_find_closest_sig (core->anal, item, count, th);
if (list) {
found = true;
print_possible_matches (list, json, core);
r_list_free (list);
}
r_cons_break_pop ();
} else {
R_LOG_WARN ("no signatures types available for testing");
}
r_sign_item_free (item);
return found;
}
static bool cmd_zb(void *data, const char *input) {
R_RETURN_VAL_IF_FAIL (data && input, false);
bool json = false;
RCore *core = (RCore *)data;
switch (input[0]) {
case 'r': // "zbr"
input++;
if (*input == 'j') {
input++;
json = true;
}
return bestmatch_fcn (core, input, json);
case 'j': // "zbj"
json = true;
/* fallthrough */
case ' ': // "zb"
input++;
/* fallthrough */
case '\x00':
return bestmatch_sig (core, input, json);
case '?': // "zb?"
default:
r_core_cmd_help (core, help_msg_zb);
return false;
}
}
static bool _sig_bytediff_cb(RLevBuf *va, RLevBuf *vb, ut32 ia, ut32 ib) {
RSignBytes *a = (RSignBytes *)va->buf;
RSignBytes *b = (RSignBytes *)vb->buf;
if ((a->bytes[ia] & a->mask[ia]) == (b->bytes[ib] & b->mask[ib])) {
return false;
}
return true;
}
#define lines_addbytesmask(l, sig, index, add, col) \
l.bytes = r_str_appendf (l.bytes, " %s%s%02x%s", r_str_get (col), r_str_get (add), sig->bytes[index], col? Color_RESET: ""); \
l.mask = r_str_appendf (l.mask, " %s%s%02x%s", r_str_get (col), r_str_get (add), sig->mask[index], col? Color_RESET: ""); \
l.land = r_str_appendf (l.land, " %s%s%02x%s", r_str_get (col), r_str_get (add), sig->bytes[index] & sig->mask[index], col?Color_RESET: ""); \
index++;
#define lines_addblnk(l) \
l.bytes = r_str_append (l.bytes, " "); \
l.mask = r_str_append (l.mask, " "); \
l.land = r_str_append (l.land, " ");
#define freelines(x) \
free (x.bytes); \
free (x.mask); \
free (x.land); \
memset (&x, 0, sizeof (x));
static void print_zig_diff(RCore *c, RSignBytes *ab, RSignBytes *bb, RLevOp *ops) {
struct lines {
char *mask, *bytes, *land;
} al, bl;
memset (&al, 0, sizeof (al));
memset (&bl, 0, sizeof (bl));
char *colsub, *coladd, *coldel;
colsub = coladd = coldel = NULL;
if (r_config_get_b (c->config, "scr.color")) {
coldel = "\x1b[1;31m";
coladd = "\x1b[1;32m";
colsub = "\x1b[1;33m";
}
int i, ia, ib, iastart, ibstart;
ia = ib = iastart = ibstart = 0;
bool printb = false;
for (i = 0; ops[i] != LEVEND; i++) {
switch (ops[i]) {
case LEVNOP:
// lines_addbytesmask macro does ia++ so test must before
if (!printb && (ab->bytes[ia] != bb->bytes[ib] || ab->mask[ia] != bb->mask[ib])) {
printb = true;
}
lines_addbytesmask (al, ab, ia, " ", (const char *)NULL);
lines_addbytesmask (bl, bb, ib, " ", (const char *)NULL);
break;
case LEVSUB:
lines_addbytesmask (al, ab, ia, " ", colsub);
lines_addbytesmask (bl, bb, ib, "^", colsub);
printb = true;
break;
case LEVADD:
lines_addblnk (al);
lines_addbytesmask (bl, bb, ib, "+", coladd);
printb = true;
break;
case LEVDEL:
lines_addbytesmask (al, ab, ia, "-", coldel);
lines_addblnk (bl);
printb = true;
break;
default:
r_warn_if_reached ();
freelines (al);
freelines (bl);
return;
}
// when alloc fails
if (!(al.bytes && al.mask && al.land && bl.bytes && bl.mask && bl.land)) {
freelines (al);
freelines (bl);
return;
}
if (i % 16 == 15 || ops[i + 1] == LEVEND) {
if (i > 16) {
r_cons_printf ("\n");
}
r_cons_printf ("Fnc cmp 0x%04x %s\n", iastart, al.land);
if (printb) {
r_cons_printf ("Sig cmp 0x%04x %s\n", ibstart, bl.land);
}
r_cons_printf ("Fnc Mask 0x%04x %s\n", iastart, al.mask);
if (printb) {
r_cons_printf ("Sig Mask 0x%04x %s\n", ibstart, bl.mask);
}
r_cons_printf ("Fnc Bytes 0x%04x %s\n", iastart, al.bytes);
if (printb) {
r_cons_printf ("Sig Bytes 0x%04x %s\n", ibstart, bl.bytes);
} else {
r_cons_printf ("== Signature was same ==\n");
}
freelines (al);
freelines (bl);
iastart = ia;
ibstart = ib;
printb = false;
}
}
}
#undef lines_addbytesmask
#undef lines_addblnk
#undef freelines
static bool cmd_zd(void *data, const char *input) {
R_RETURN_VAL_IF_FAIL (data && input, false);
RCore *core = (RCore *)data;
RAnalFunction *fcn = r_anal_get_fcn_in (core->anal, core->offset, 0);
if (!fcn) {
R_LOG_ERROR ("No function at 0x%08" PFMT64x, core->offset);
return false;
}
char *argv = strdup (input);
if (!argv) {
return false;
}
char *save_ptr = NULL;
char *zigname = r_str_tok_r (argv, " ", &save_ptr);
if (!zigname) {
R_LOG_ERROR ("Need a signature");
free (argv);
return false;
}
if (r_str_tok_r (NULL, " ", &save_ptr)) {
R_LOG_ERROR ("too many arguments");
free (argv);
return false;
}
RSignItem *it = item_frm_signame (core->anal, zigname);
if (!it) {
R_LOG_ERROR ("Couldn't get signature for %s", zigname);
free (argv);
return false;
}
free (argv);
if (!it->bytes) {
R_LOG_ERROR ("Signature %s missing bytes", it->name);
return false;
}
RLevBuf b;
b.buf = it->bytes;
b.len = it->bytes->size;
RSignItem *fit = r_sign_item_new ();
if (!fit) {
r_sign_item_free (it);
return false;
}
r_sign_addto_item (core->anal, fit, fcn, R_SIGN_BYTES);
RLevBuf a;
a.buf = fit->bytes;
a.len = fit->bytes->size;
RLevOp *ops = NULL;
if (r_diff_levenshtein_path (&a, &b, UT32_MAX, _sig_bytediff_cb, &ops) < 0) {
R_LOG_ERROR ("Levenshtein diff has failed");
} else {
print_zig_diff (core, fit->bytes, it->bytes, ops);
}
free (ops);
r_sign_item_free (fit);
r_sign_item_free (it);
return false;
}
static int cmd_zc(void *data, const char *input) {
int result = true;
RCore *core = (RCore *)data;
const char *raw_bytes_thresh = r_config_get (core->config, "zign.diff.bthresh");
const char *raw_graph_thresh = r_config_get (core->config, "zign.diff.gthresh");
RSignOptions *options = r_sign_options_new (raw_bytes_thresh, raw_graph_thresh);
switch (*input) {
case ' ': // "zc"
if (!input[1]) {
r_core_cmd_help (core, help_msg_zc);
result = false;
break;
}
result = r_sign_diff (core->anal, options, input + 1);
break;
case 'n':
switch (input[1]) {
case ' ': // "zcn"
if (!input[2]) {
r_core_cmd_help_contains (core, help_msg_zc, "zcn");
result = false;
break;
}
result = r_sign_diff_by_name (core->anal, options, input + 2, false);
break;
case '!': // "zcn!"
if (input[2] != ' ' || !input[3]) {
r_core_cmd_help_contains (core, help_msg_zc, "zcn");
result = false;
break;
}
result = r_sign_diff_by_name (core->anal, options, input + 3, true);
break;
default:
r_core_cmd_help_contains (core, help_msg_zc, "zcn");
result = false;
}
break;
case '?': // "zc?"
r_core_cmd_help (core, help_msg_zc);
break;
default:
r_core_return_invalid_command (core, "zc", *input);
result = false;
break;
}
r_sign_options_free (options);
return result;
}
static int cmd_zdot(void *data, const char *input) {
RCore *core = (RCore *) data;
struct ctxSearchCB ctx = {
.rad = input[0] == '*',
.core = core
};
RSignSearchMetrics sm;
if (!fill_search_metrics (&sm, core, (void *)&ctx)) {
R_LOG_INFO ("Nothing to search for");
return 0;
}
const char *zign_prefix = r_config_get (core->config, "zign.prefix");
if (ctx.rad) {
r_cons_printf ("fs+%s\n", zign_prefix);
} else {
if (!r_flag_space_push (core->flags, zign_prefix)) {
R_LOG_ERROR ("cannot create flagspace");
return false;
}
}
R_LOG_INFO ("searching function metrics");
RAnalFunction *fcn = r_anal_get_function_at (core->anal, core->offset);
if (fcn) {
r_cons_break_push (NULL, NULL);
r_sign_fcn_match_metrics (&sm, fcn);
r_cons_break_pop ();
} else {
R_LOG_ERROR ("No function at 0x%08" PFMT64x, core->offset);
}
if (ctx.rad) {
r_cons_printf ("fs-\n");
} else {
if (!r_flag_space_pop (core->flags)) {
R_LOG_ERROR ("cannot restore flagspace");
return false;
}
}
print_ctx_hits (&ctx);
return ctx.count;
}
static int cmd_zslash(void *data, const char *input) {
RCore *core = (RCore *) data;
switch (*input) {
case 0:
case '*': // "z/*"
return search (core, input[0] == '*', false);
case 'f': // "z/f"
switch (input[1]) {
case 0:
case '*':
return search (core, input[1] == '*', true);
default:
r_core_cmd_help_contains (core, help_msg_z_slash, "z/f");
return false;
}
case '?': // "z/?"
r_core_cmd_help (core, help_msg_z_slash);
break;
default:
r_core_return_invalid_command (core, "z/", *input);
return false;
}
return true;
}
static int cmd_zi(void *data, const char *input) {
if (!data || !input) {
return false;
}
RCore *core = (RCore *) data;
const char i0 = *input;
switch (i0) {
case '?':
r_core_cmd_help_contains (core, help_msg_z, "zi");
return false;
case 0:
case ' ':
r_flag_space_push (core->flags, R_FLAGS_FS_SIGNS);
r_flag_list (core->flags, *input, input[0] ? input + 1: "");
r_flag_space_pop (core->flags);
break;
default:
r_core_return_invalid_command (core, "zi", i0);
break;
}
return true;
}
struct ctxListCB {
RAnal *anal;
RTable *t;
};
// R2_600 - move into anal/sign.c
static bool listCB(RSignItem *it, void *user) {
struct ctxListCB *ctx = (struct ctxListCB *)user;
#if 0
RAnal *a = ctx->anal;
if (!validate_item (it)) {
return true;
}
#endif
// End item
char *bytes = r_hex_bin2strdup (it->bytes->bytes, it->bytes->size);
char *masks = r_hex_bin2strdup (it->bytes->mask, it->bytes->size);
r_table_add_rowf (ctx->t, "dssss", it->addr, it->name, it->hash->bbhash, bytes, masks);
free (bytes);
free (masks);
return true;
}
static int csvZignatures(RCore *core, const char *arg) {
RTable *t = r_table_new ("pxr");
RTableColumnType *n = r_table_type ("number");
RTableColumnType *s = r_table_type ("string");
r_table_add_column (t, n, "addr", 0);
r_table_add_column (t, s, "name", 1);
r_table_add_column (t, s, "hash", 2);
r_table_add_column (t, s, "data", 3);
r_table_add_column (t, s, "mask", 4);
struct ctxListCB ctx = { core->anal, t };
r_sign_foreach (core->anal, listCB, &ctx);
if (r_table_query (t, arg)) {
char *ts = r_table_tostring (t);
r_cons_printf ("%s", ts); // \n?
free (ts);
}
r_table_free (t);
return 0;
}
static int cmd_zign(void *data, const char *input) {
RCore *core = (RCore *) data;
const char *arg = input + 1;
switch (*input) {
case '\0':
case ' ': // "z"
case '*': // "z*"
case 'q': // "zq"
case 'j': // "zj"
{
ut64 naddr = UT64_MAX;
if (*input == ' ' || (*input && input[1] == ' ')) {
naddr = r_num_math (core->num, input + 1);
if (naddr == 0) {
naddr = UT64_MAX;
}
}
ut64 oaddr = core->offset;
core->offset = naddr; // XXX R2_600 - this is a hack because we cant break the abi
r_sign_list (core->anal, *input);
core->offset = oaddr;
}
break;
case ',': // "z,"
return csvZignatures (core, arg);
case 'k': // "zk"
r_core_cmd0 (core, "k anal/zigns/*");
break;
case '-': // "z-"
r_sign_delete (core->anal, arg);
break;
case '.': // "z."
return cmd_zdot (data, arg);
case 'b': // "zb"
return cmd_zb (data, arg);
case 'd': // "zd"
return cmd_zd (data, arg);
case 'o': // "zo"
return cmd_zo (data, arg);
case 'g': // "zg"
return cmd_za (data, "F");
case 'a': // "za"
return cmd_za (data, arg);
case 'f': // "zf"
return cmd_zf (data, arg);
case '/': // "z/"
return cmd_zslash (data, arg);
case 'c': // "zc"
return cmd_zc (data, arg);
case 's': // "zs"
return cmd_zs (data, arg);
case 'i': // "zi"
return cmd_zi (data, arg);
case '?': // "z?"
r_core_cmd_help (core, help_msg_z);
break;
default:
r_core_return_invalid_command (core, "z", *input);
return false;
}
return true;
}
#endif