radare2/libr/core/cmd_magic.c

206 lines
4.7 KiB
C
Raw Normal View History

2018-04-24 13:50:09 +02:00
/* radare - LGPL - Copyright 2009-2018 - pancake */
2016-05-02 22:52:41 -04:00
#include "r_core.h"
2012-02-27 02:40:27 +01:00
/* ugly global vars */
static int magicdepth = 99; //XXX: do not use global var here
static RMagic *ck = NULL; // XXX: Use RCore->magic
2015-06-21 10:00:41 +02:00
static char *ofile = NULL;
2018-04-24 13:50:09 +02:00
static int kw_count = 0;
static void r_core_magic_reset(RCore *core) {
kw_count = 0;
}
2018-09-21 16:32:25 +02:00
static int r_core_magic_at(RCore *core, const char *file, ut64 addr, int depth, int v, bool json, int *hits) {
2012-02-27 02:40:27 +01:00
const char *fmt;
char *q, *p;
const char *str;
int found = 0, delta = 0, adelta = 0, ret;
ut64 curoffset = core->offset;
2018-09-21 16:32:25 +02:00
int maxHits = r_config_get_i (core->config, "search.maxhits");
if (maxHits > 0 && *hits >= maxHits) {
return 0;
}
#define NAH 32
2012-02-27 02:40:27 +01:00
if (--depth<0) {
ret = 0;
goto seek_exit;
}
if (addr != core->offset) {
#if 1
if (addr >= core->offset && (addr+NAH) < (core->offset + core->blocksize)) {
delta = addr - core->offset;
} else {
2015-09-14 12:35:38 +02:00
r_core_seek (core, addr, true);
}
#endif
}
if (core->search->align) {
int mod = addr % core->search->align;
if (mod) {
eprintf ("Unaligned search at %d\n", mod);
ret = mod;
goto seek_exit;
}
}
if (((addr&7)==0) && ((addr&(7<<8))==0))
2018-04-09 02:10:14 +05:30
if (!json) {
eprintf ("0x%08"PFMT64x"\r", addr);
}
2012-02-27 02:40:27 +01:00
if (file) {
if (*file == ' ') file++;
if (!*file) file = NULL;
}
if (file && ofile && file != ofile) {
if (strcmp (file, ofile)) {
r_magic_free (ck);
ck = NULL;
}
}
2018-04-24 13:50:09 +02:00
if (!ck) {
2012-02-27 02:40:27 +01:00
// TODO: Move RMagic into RCore
r_magic_free (ck);
// allocate once
2012-02-27 02:40:27 +01:00
ck = r_magic_new (0);
if (file) {
2015-06-21 10:00:41 +02:00
free (ofile);
ofile = strdup (file);
if (r_magic_load (ck, file) == -1) {
eprintf ("failed r_magic_load (\"%s\") %s\n", file, r_magic_error (ck));
ck = NULL;
ret = -1;
goto seek_exit;
}
} else {
const char *magicpath = r_config_get (core->config, "dir.magic");
if (r_magic_load (ck, magicpath) == -1) {
ck = NULL;
eprintf ("failed r_magic_load (dir.magic) %s\n", r_magic_error (ck));
ret = -1;
goto seek_exit;
}
}
2012-02-27 02:40:27 +01:00
}
2014-10-25 00:54:22 +02:00
//repeat:
2012-02-27 02:40:27 +01:00
//if (v) r_cons_printf (" %d # pm %s @ 0x%"PFMT64x"\n", depth, file? file: "", addr);
2018-04-24 13:50:09 +02:00
if (delta + 2 > core->blocksize) {
eprintf ("EOB\n");
ret = -1;
goto seek_exit;
}
2018-04-24 13:50:09 +02:00
str = r_magic_buffer (ck, core->block+delta, core->blocksize - delta);
2012-02-27 02:40:27 +01:00
if (str) {
const char *cmdhit;
#if USE_LIB_MAGIC
if (!v && (!strcmp (str, "data") || strstr(str, "ASCII") || strstr(str, "ISO") || strstr(str, "no line terminator"))) {
#else
if (!v && (!strcmp (str, "data"))) {
#endif
int mod = core->search->align;
2018-04-24 13:50:09 +02:00
if (mod < 1) {
mod = 1;
}
//r_magic_free (ck);
//ck = NULL;
//return -1;
ret = mod + 1;
goto seek_exit;
2014-08-29 16:42:00 +02:00
}
2012-02-27 02:40:27 +01:00
p = strdup (str);
fmt = p;
// processing newlinez
for (q=p; *q; q++) {
2012-02-27 02:40:27 +01:00
if (q[0]=='\\' && q[1]=='n') {
*q = '\n';
2018-04-24 13:50:09 +02:00
strcpy (q + 1, q + ((q[2] == ' ')? 3: 2));
2012-02-27 02:40:27 +01:00
}
}
2018-09-21 16:32:25 +02:00
(*hits)++;
cmdhit = r_config_get (core->config, "cmd.hit");
2014-12-17 12:41:51 +01:00
if (cmdhit && *cmdhit) {
r_core_cmd0 (core, cmdhit);
}
2018-04-24 13:50:09 +02:00
{
const char *searchprefix = r_config_get (core->config, "search.prefix");
const char *flag = sdb_fmt ("%s%d_%d", searchprefix, 0, kw_count++);
r_flag_set (core->flags, flag, addr + adelta, 1);
}
2012-02-27 02:40:27 +01:00
// TODO: This must be a callback .. move this into RSearch?
2018-04-09 02:10:14 +05:30
if (!json) {
r_cons_printf ("0x%08"PFMT64x" %d %s\n", addr + adelta, magicdepth-depth, p);
} else {
if (found >= 1) {
r_cons_printf (",");
}
r_cons_printf ("{\"offset\":%"PFMT64d ",\"depth\":%d,\"info\":\"%s\"}",
addr + adelta, magicdepth-depth, p);
}
r_cons_clear_line (1);
2018-04-09 02:10:14 +05:30
//eprintf ("0x%08"PFMT64x" 0x%08"PFMT64x" %d %s\n", addr+adelta, addr+adelta, magicdepth-depth, p);
2012-02-27 02:40:27 +01:00
// walking children
2018-04-24 13:50:09 +02:00
for (q = p; *q; q++) {
2012-02-27 02:40:27 +01:00
switch (*q) {
case ' ':
2018-04-24 13:50:09 +02:00
fmt = q + 1;
2012-02-27 02:40:27 +01:00
break;
case '@':
{
ut64 addr = 0LL;
*q = 0;
2018-04-24 13:50:09 +02:00
if (!strncmp (q + 1, "0x", 2)) {
sscanf (q + 3, "%"PFMT64x, &addr);
} else {
sscanf (q + 1, "%"PFMT64d, &addr);
}
if (!fmt || !*fmt) {
fmt = file;
}
2018-09-21 16:32:25 +02:00
r_core_magic_at (core, fmt, addr, depth, 1, json, hits);
*q = '@';
}
break;
2012-02-27 02:40:27 +01:00
}
}
free (p);
2014-08-29 16:42:00 +02:00
r_magic_free (ck);
ck = NULL;
found ++;
// return adelta+1;
2012-02-27 02:40:27 +01:00
}
adelta ++;
delta ++;
#if 0
if((core->blocksize-delta)>16)
goto repeat;
#endif
#if 0
2014-08-29 16:42:00 +02:00
r_magic_free (ck);
ck = NULL;
#endif
{
int mod = core->search->align;
if (mod) {
ret = mod; //adelta%addr + deR_ABS(mod-adelta)+1;
goto seek_exit;
}
}
ret = adelta; //found;
seek_exit:
r_core_seek (core, curoffset, true);
return ret;
2012-02-27 02:40:27 +01:00
}
static void r_core_magic(RCore *core, const char *file, int v) {
ut64 addr = core->offset;
2018-09-21 16:32:25 +02:00
int hits = 0;
2012-02-27 02:40:27 +01:00
magicdepth = r_config_get_i (core->config, "magic.depth"); // TODO: do not use global var here
2018-09-21 16:32:25 +02:00
r_core_magic_at (core, file, addr, magicdepth, v, false, &hits);
2018-04-24 13:50:09 +02:00
if (addr != core->offset) {
2015-09-14 12:35:38 +02:00
r_core_seek (core, addr, true);
2018-04-24 13:50:09 +02:00
}
2012-02-27 02:40:27 +01:00
}