mirror of
https://github.com/radareorg/radare2.git
synced 2025-01-25 07:15:19 +00:00
1237 lines
34 KiB
C
1237 lines
34 KiB
C
/* radare - LGPL - Copyright 2009-2020 - pancake */
|
|
|
|
#include <string.h>
|
|
#include "r_bin.h"
|
|
#include "r_config.h"
|
|
#include "r_cons.h"
|
|
#include "r_core.h"
|
|
#include "../bin/pdb/pdb_downloader.h"
|
|
|
|
static const char *help_msg_i[] = {
|
|
"Usage: i", "", "Get info from opened file (see rabin2's manpage)",
|
|
"Output mode:", "", "",
|
|
"'*'", "", "Output in radare commands",
|
|
"'j'", "", "Output in json",
|
|
"'q'", "", "Simple quiet output",
|
|
"Actions:", "", "",
|
|
"i|ij", "", "Show info of current file (in JSON)",
|
|
"iA", "", "List archs",
|
|
"ia", "", "Show all info (imports, exports, sections..)",
|
|
"ib", "", "Reload the current buffer for setting of the bin (use once only)",
|
|
"ic", "", "List classes, methods and fields",
|
|
"icc", "", "List classes, methods and fields in Header Format",
|
|
"icg", "", "List classes as agn/age commands to create class hirearchy graphs",
|
|
"icq", "", "List classes, in quiet mode (just the classname)",
|
|
"icqq", "", "List classes, in quieter mode (only show non-system classnames)",
|
|
"iC", "[j]", "Show signature info (entitlements, ...)",
|
|
"id", "", "Show DWARF source lines information",
|
|
"idp", " [file.pdb]", "Load pdb file information",
|
|
"idpi", " [file.pdb]", "Show pdb file information",
|
|
"idpi*", "", "Show symbols from pdb as flags (prefix with dot to import)",
|
|
"idpd", "", "Download pdb file on remote server",
|
|
"iD", " lang sym", "Demangle symbolname for given language",
|
|
"ie", "", "Entrypoint",
|
|
"iee", "", "Show Entry and Exit (preinit, init and fini)",
|
|
"iE", "", "Exports (global symbols)",
|
|
"iE.", "", "Current export",
|
|
"ih", "", "Headers (alias for iH)",
|
|
"iHH", "", "Verbose Headers in raw text",
|
|
"ii", "", "Imports",
|
|
"iI", "", "Binary info",
|
|
"ik", " [query]", "Key-value database from RBinObject",
|
|
"il", "", "Libraries",
|
|
"iL ", "[plugin]", "List all RBin plugins loaded or plugin details",
|
|
"im", "", "Show info about predefined memory allocation",
|
|
"iM", "", "Show main address",
|
|
"io", " [file]", "Load info from file (or last opened) use bin.baddr",
|
|
"iO", "[?]", "Perform binary operation (dump, resize, change sections, ...)",
|
|
"ir", "", "List the Relocations",
|
|
"iR", "", "List the Resources",
|
|
"is", "", "List the Symbols",
|
|
"is.", "", "Current symbol",
|
|
"iS ", "[entropy,sha1]", "Sections (choose which hash algorithm to use)",
|
|
"iS.", "", "Current section",
|
|
"iS=", "", "Show ascii-art color bars with the section ranges",
|
|
"iSS", "", "List memory segments (maps with om)",
|
|
"it", "", "File hashes",
|
|
"iT", "", "File signature",
|
|
"iV", "", "Display file version info",
|
|
"iw", "", "Show try/catch blocks",
|
|
"iX", "", "Display source files used (via dwarf)",
|
|
"iz|izj", "", "Strings in data sections (in JSON/Base64)",
|
|
"izz", "", "Search for Strings in the whole binary",
|
|
"izzz", "", "Dump Strings from whole binary to r2 shell (for huge files)",
|
|
"iz-", " [addr]", "Purge string via bin.str.purge",
|
|
"iZ", "", "Guess size of binary program",
|
|
NULL
|
|
};
|
|
|
|
// TODO: this command needs a refactoring
|
|
static const char *help_msg_id[] = {
|
|
"Usage: idp", "", "Debug information",
|
|
"id", "", "Show DWARF source lines information",
|
|
"idp", " [file.pdb]", "Load pdb file information",
|
|
"idpi", " [file.pdb]", "Show pdb file information",
|
|
"idpi*", "", "Show symbols from pdb as flags (prefix with dot to import)",
|
|
"idpd", "", "Download pdb file on remote server",
|
|
NULL
|
|
};
|
|
|
|
static void cmd_info_init(RCore *core, RCmdDesc *parent) {
|
|
DEFINE_CMD_DESCRIPTOR (core, i);
|
|
DEFINE_CMD_DESCRIPTOR (core, id);
|
|
}
|
|
|
|
#define PAIR_WIDTH 9
|
|
// TODO: reuse implementation in core/bin.c
|
|
static void pair(const char *a, const char *b) {
|
|
char ws[16];
|
|
int al = strlen (a);
|
|
if (!b) {
|
|
return;
|
|
}
|
|
memset (ws, ' ', sizeof (ws));
|
|
al = PAIR_WIDTH - al;
|
|
if (al < 0) {
|
|
al = 0;
|
|
}
|
|
ws[al] = 0;
|
|
r_cons_printf ("%s%s%s\n", a, ws, b);
|
|
}
|
|
|
|
static bool demangle_internal(RCore *core, const char *lang, const char *s) {
|
|
char *res = NULL;
|
|
int type = r_bin_demangle_type (lang);
|
|
switch (type) {
|
|
case R_BIN_NM_CXX: res = r_bin_demangle_cxx (core->bin->cur, s, 0); break;
|
|
case R_BIN_NM_JAVA: res = r_bin_demangle_java (s); break;
|
|
case R_BIN_NM_OBJC: res = r_bin_demangle_objc (NULL, s); break;
|
|
case R_BIN_NM_SWIFT: res = r_bin_demangle_swift (s, core->bin->demanglercmd); break;
|
|
case R_BIN_NM_DLANG: res = r_bin_demangle_plugin (core->bin, "dlang", s); break;
|
|
case R_BIN_NM_MSVC: res = r_bin_demangle_msvc (s); break;
|
|
case R_BIN_NM_RUST: res = r_bin_demangle_rust (core->bin->cur, s, 0); break;
|
|
default:
|
|
r_bin_demangle_list (core->bin);
|
|
return true;
|
|
}
|
|
if (res) {
|
|
if (*res) {
|
|
r_cons_printf ("%s\n", res);
|
|
}
|
|
free (res);
|
|
return false;
|
|
}
|
|
return true;
|
|
}
|
|
|
|
static bool demangle(RCore *core, const char *s) {
|
|
r_return_val_if_fail (core && s, false);
|
|
const char *ss = strchr (s, ' ');
|
|
if (!*s) {
|
|
return false;
|
|
}
|
|
if (!ss) {
|
|
const char *lang = r_config_get (core->config, "bin.lang");
|
|
demangle_internal (core, lang, s);
|
|
return true;
|
|
}
|
|
char *p = strdup (s);
|
|
char *q = p + (ss - s);
|
|
*q = 0;
|
|
demangle_internal (core, p, q + 1);
|
|
free (p);
|
|
return true;
|
|
}
|
|
|
|
static void cmd_info_here(RCore *core, PJ *pj, int mode) {
|
|
RCoreItem *item = r_core_item_at (core, core->offset);
|
|
// fixme: other modes
|
|
if (item && mode == R_MODE_JSON) {
|
|
pj_o (pj);
|
|
|
|
pj_ks (pj, "type", item->type);
|
|
pj_ks (pj, "perm", r_str_rwx_i (item->perm));
|
|
pj_kn (pj, "size", item->size);
|
|
pj_kn (pj, "addr", item->addr);
|
|
pj_kn (pj, "next", item->next);
|
|
pj_kn (pj, "prev", item->prev);
|
|
if (item->fcnname) {
|
|
pj_ks (pj, "fcnname", item->fcnname);
|
|
}
|
|
if (item->sectname) {
|
|
pj_ks (pj, "sectname", item->sectname);
|
|
}
|
|
if (item->comment) {
|
|
pj_ks (pj, "comment", item->comment);
|
|
}
|
|
RListIter *iter;
|
|
RAnalRef *ref;
|
|
if (item->data) {
|
|
pj_ks (pj, "data", item->data);
|
|
}
|
|
{
|
|
RList *refs = r_anal_refs_get (core->anal, core->offset);
|
|
if (refs && r_list_length (refs) > 0) {
|
|
pj_k (pj, "refs");
|
|
pj_a (pj);
|
|
r_list_foreach (refs, iter, ref) {
|
|
pj_o (pj);
|
|
pj_ks (pj, "type", r_anal_ref_type_tostring (ref->type));
|
|
pj_kn (pj, "addr", ref->addr);
|
|
pj_end (pj);
|
|
}
|
|
pj_end (pj);
|
|
}
|
|
}
|
|
{
|
|
RList *refs = r_anal_xrefs_get (core->anal, core->offset);
|
|
if (refs && r_list_length (refs) > 0) {
|
|
pj_k (pj, "xrefs");
|
|
pj_a (pj);
|
|
r_list_foreach (refs, iter, ref) {
|
|
pj_o (pj);
|
|
pj_ks (pj, "type", r_anal_ref_type_tostring (ref->type));
|
|
pj_kn (pj, "addr", ref->addr);
|
|
pj_end (pj);
|
|
}
|
|
pj_end (pj);
|
|
}
|
|
}
|
|
pj_end (pj);
|
|
r_core_item_free (item);
|
|
}
|
|
}
|
|
|
|
static void r_core_file_info(RCore *core, PJ *pj, int mode) {
|
|
const char *fn = NULL;
|
|
int dbg = r_config_get_i (core->config, "cfg.debug");
|
|
bool io_cache = r_config_get_i (core->config, "io.cache");
|
|
RBinInfo *info = r_bin_get_info (core->bin);
|
|
RBinFile *binfile = r_bin_cur (core->bin);
|
|
int fd = r_io_fd_get_current (core->io);
|
|
RIODesc *desc = r_io_desc_get (core->io, fd);
|
|
RBinPlugin *plugin = r_bin_file_cur_plugin (binfile);
|
|
if (mode == R_MODE_JSON) {
|
|
pj_o (pj);
|
|
}
|
|
if (mode == R_MODE_RADARE) {
|
|
return;
|
|
}
|
|
if (mode == R_MODE_SIMPLE) {
|
|
return;
|
|
}
|
|
if (info) {
|
|
fn = info->file;
|
|
if (mode == R_MODE_JSON) {
|
|
pj_ks (pj, "type", r_str_get (info->type));
|
|
}
|
|
} else {
|
|
fn = desc ? desc->name: NULL;
|
|
}
|
|
if (mode == R_MODE_JSON) {
|
|
const char *uri = fn;
|
|
if (!uri) {
|
|
if (desc && desc->uri && *desc->uri) {
|
|
uri = desc->uri;
|
|
} else {
|
|
uri = "";
|
|
}
|
|
}
|
|
pj_ks (pj, "file", uri);
|
|
if (dbg) {
|
|
dbg = R_PERM_WX;
|
|
}
|
|
if (desc) {
|
|
ut64 fsz = r_io_desc_size (desc);
|
|
pj_ki (pj, "fd", desc->fd);
|
|
if (fsz != UT64_MAX) {
|
|
char humansz[8];
|
|
pj_kN (pj, "size", fsz);
|
|
r_num_units (humansz, sizeof (humansz), fsz);
|
|
pj_ks (pj, "humansz", humansz);
|
|
}
|
|
pj_kb (pj, "iorw", io_cache || desc->perm & R_PERM_W);
|
|
pj_ks (pj, "mode", r_str_rwx_i (desc->perm & R_PERM_RWX));
|
|
if (desc->referer && *desc->referer) {
|
|
pj_ks (pj, "referer", desc->referer);
|
|
}
|
|
}
|
|
pj_ki (pj, "block", core->blocksize);
|
|
if (binfile) {
|
|
if (binfile->curxtr) {
|
|
pj_ks (pj, "packet", binfile->curxtr->name);
|
|
}
|
|
if (plugin) {
|
|
pj_ks (pj, "format", plugin->name);
|
|
}
|
|
}
|
|
pj_end (pj);
|
|
} else if (desc && mode != R_MODE_SIMPLE) {
|
|
//r_cons_printf ("# Core file info\n");
|
|
if (dbg) {
|
|
dbg = R_PERM_WX;
|
|
}
|
|
if (desc) {
|
|
pair ("fd", sdb_fmt ("%d", desc->fd));
|
|
}
|
|
if (fn || (desc && desc->uri)) {
|
|
char *escaped = r_str_escape_utf8_keep_printable (fn? fn: desc->uri, false, false);
|
|
if (escaped) {
|
|
pair ("file", escaped);
|
|
free (escaped);
|
|
}
|
|
}
|
|
if (desc) {
|
|
ut64 fsz = r_io_desc_size (desc);
|
|
if (fsz != UT64_MAX) {
|
|
char humansz[8];
|
|
pair ("size", sdb_itoca (fsz));
|
|
r_num_units (humansz, sizeof (humansz), fsz);
|
|
pair ("humansz", humansz);
|
|
}
|
|
}
|
|
if (desc) {
|
|
pair ("mode", r_str_rwx_i (desc->perm & R_PERM_RWX));
|
|
}
|
|
if (plugin) {
|
|
pair ("format", plugin->name);
|
|
}
|
|
if (desc) {
|
|
pair ("iorw", r_str_bool (io_cache || desc->perm & R_PERM_W));
|
|
}
|
|
pair ("block", sdb_fmt ("0x%x", core->blocksize));
|
|
|
|
if (binfile && binfile->curxtr) {
|
|
pair ("packet", binfile->curxtr->name);
|
|
}
|
|
if (desc && desc->referer && *desc->referer) {
|
|
pair ("referer", desc->referer);
|
|
}
|
|
|
|
if (info) {
|
|
pair ("type", info->type);
|
|
}
|
|
}
|
|
}
|
|
|
|
static int bin_is_executable(RBinObject *obj){
|
|
RListIter *it;
|
|
RBinSection *sec;
|
|
if (obj) {
|
|
if (obj->info && obj->info->arch) {
|
|
return true;
|
|
}
|
|
r_list_foreach (obj->sections, it, sec){
|
|
if (sec->perm & R_PERM_X) {
|
|
return true;
|
|
}
|
|
}
|
|
}
|
|
return false;
|
|
}
|
|
|
|
static void cmd_info_bin(RCore *core, int va, PJ *pj, int mode) {
|
|
RBinObject *obj = r_bin_cur_object (core->bin);
|
|
int array = 0;
|
|
if (core->file) {
|
|
if (mode & R_MODE_JSON) {
|
|
if (!(mode & R_MODE_ARRAY)) {
|
|
pj_o (pj);
|
|
} else {
|
|
array = 1;
|
|
}
|
|
mode = R_MODE_JSON;
|
|
pj_k (pj, "core");
|
|
}
|
|
r_core_file_info (core, pj, mode);
|
|
if (bin_is_executable (obj)) {
|
|
if ((mode & R_MODE_JSON)) {
|
|
pj_k (pj, "bin");
|
|
}
|
|
r_core_bin_info (core, R_CORE_BIN_ACC_INFO, pj, mode, va, NULL, NULL);
|
|
}
|
|
if ((mode & R_MODE_JSON) && array == 0) {
|
|
pj_end (pj);
|
|
}
|
|
} else {
|
|
eprintf ("No file selected\n");
|
|
}
|
|
}
|
|
|
|
static void playMsg(RCore *core, const char *n, int len) {
|
|
if (r_config_get_i (core->config, "scr.tts")) {
|
|
if (len > 0) {
|
|
char *s = r_str_newf ("%d %s", len, n);
|
|
r_sys_tts (s, true);
|
|
free (s);
|
|
} else if (len == 0) {
|
|
char *s = r_str_newf ("there are no %s", n);
|
|
r_sys_tts (s, true);
|
|
free (s);
|
|
}
|
|
}
|
|
}
|
|
|
|
static bool is_equal_file_hashes(RList *lfile_hashes, RList *rfile_hashes, bool *equal) {
|
|
r_return_val_if_fail (lfile_hashes, false);
|
|
r_return_val_if_fail (rfile_hashes, false);
|
|
r_return_val_if_fail (equal, false);
|
|
|
|
*equal = true;
|
|
RBinFileHash *fh_l, *fh_r;
|
|
RListIter *hiter_l, *hiter_r;
|
|
r_list_foreach (lfile_hashes, hiter_l, fh_l) {
|
|
r_list_foreach (rfile_hashes, hiter_r, fh_r) {
|
|
if (strcmp (fh_l->type, fh_r->type)) {
|
|
continue;
|
|
}
|
|
if (!!strcmp (fh_l->hex, fh_r->hex)) {
|
|
*equal = false;
|
|
return true;
|
|
}
|
|
}
|
|
}
|
|
return true;
|
|
}
|
|
|
|
static int __r_core_bin_reload(RCore *r, const char *file, ut64 baseaddr) {
|
|
int result = 0;
|
|
RCoreFile *cf = r_core_file_cur (r);
|
|
if (cf) {
|
|
RBinFile *bf = r_bin_file_find_by_fd (r->bin, cf->fd);
|
|
if (bf) {
|
|
result = r_bin_reload (r->bin, bf->id, baseaddr);
|
|
}
|
|
}
|
|
r_core_bin_set_env (r, r_bin_cur (r->bin));
|
|
return result;
|
|
}
|
|
|
|
static bool isKnownPackage(const char *cn) {
|
|
if (*cn == 'L') {
|
|
if (r_str_startswith (cn, "Lkotlin")) {
|
|
return true;
|
|
}
|
|
if (r_str_startswith (cn, "Lcom/google")) {
|
|
return true;
|
|
}
|
|
if (r_str_startswith (cn, "Lcom/facebook")) {
|
|
return true;
|
|
}
|
|
if (r_str_startswith (cn, "Lokhttp")) {
|
|
return true;
|
|
}
|
|
if (r_str_startswith (cn, "Landroid")) {
|
|
return true;
|
|
}
|
|
if (r_str_startswith (cn, "Lokio")) {
|
|
return true;
|
|
}
|
|
}
|
|
return false;
|
|
}
|
|
|
|
static int cmd_info(void *data, const char *input) {
|
|
RCore *core = (RCore *) data;
|
|
bool newline = r_cons_is_interactive ();
|
|
int fd = r_io_fd_get_current (core->io);
|
|
RIODesc *desc = r_io_desc_get (core->io, fd);
|
|
int i, va = core->io->va || r_config_get_i (core->config, "cfg.debug");
|
|
int mode = 0; //R_MODE_SIMPLE;
|
|
bool rdump = false;
|
|
int is_array = 0;
|
|
bool is_izzzj = false;
|
|
bool is_idpij = false;
|
|
Sdb *db;
|
|
PJ *pj = NULL;
|
|
|
|
for (i = 0; input[i] && input[i] != ' '; i++)
|
|
;
|
|
if (i > 0) {
|
|
switch (input[i - 1]) {
|
|
case '*': mode = R_MODE_RADARE; break;
|
|
case 'j': mode = R_MODE_JSON; break;
|
|
case 'q': mode = R_MODE_SIMPLE; break;
|
|
}
|
|
}
|
|
#define INIT_PJ()\
|
|
if (!pj) {\
|
|
pj = r_core_pj_new (core);\
|
|
if (!pj) {\
|
|
return 1;\
|
|
}\
|
|
}
|
|
if (mode == R_MODE_JSON) {
|
|
INIT_PJ ();
|
|
int suffix_shift = 0;
|
|
if (!strncmp (input, "SS", 2) || !strncmp (input, "ee", 2)
|
|
|| !strncmp (input, "zz", 2)) {
|
|
suffix_shift = 1;
|
|
}
|
|
if (strlen (input + 1 + suffix_shift) > 1) {
|
|
is_array = 1;
|
|
}
|
|
if (!strncmp (input, "zzz", 3)) {
|
|
is_izzzj = true;
|
|
}
|
|
if (!strncmp(input, "dpi", 3)) {
|
|
is_idpij = true;
|
|
}
|
|
}
|
|
if (is_array && !is_izzzj && !is_idpij) {
|
|
pj_o (pj);
|
|
}
|
|
if (!*input) {
|
|
cmd_info_bin (core, va, pj, mode);
|
|
}
|
|
/* i* is an alias for iI* */
|
|
if (!strcmp (input, "*")) {
|
|
input = "I*";
|
|
}
|
|
char *question = strchr (input, '?');
|
|
const char *space = strchr (input, ' ');
|
|
if (!space && question) {
|
|
space = question + 1;
|
|
}
|
|
if (question < space && question > input) {
|
|
question--;
|
|
char *prefix = strdup (input);
|
|
char *tmp = strchr (prefix, '?');
|
|
if (tmp) {
|
|
*tmp = 0;
|
|
}
|
|
r_core_cmdf (core, "i?~& i%s", prefix);
|
|
free (prefix);
|
|
goto done;
|
|
}
|
|
R_FREE (core->table_query);
|
|
if (space && *space == ' ') {
|
|
core->table_query = r_str_trim_dup (space + 1);
|
|
}
|
|
while (*input) {
|
|
if (*input == ' ') {
|
|
break;
|
|
}
|
|
switch (*input) {
|
|
case 'b': // "ib"
|
|
{
|
|
ut64 baddr = r_config_get_i (core->config, "bin.baddr");
|
|
if (input[1] == ' ') {
|
|
baddr = r_num_math (core->num, input + 1);
|
|
}
|
|
// XXX: this will reload the bin using the buffer.
|
|
// An assumption is made that assumes there is an underlying
|
|
// plugin that will be used to load the bin (e.g. malloc://)
|
|
// TODO: Might be nice to reload a bin at a specified offset?
|
|
__r_core_bin_reload (core, NULL, baddr);
|
|
r_core_block_read (core);
|
|
newline = false;
|
|
}
|
|
break;
|
|
case 'k': // "ik"
|
|
{
|
|
RBinObject *o = r_bin_cur_object (core->bin);
|
|
db = o? o->kv: NULL;
|
|
//:eprintf ("db = %p\n", db);
|
|
switch (input[1]) {
|
|
case 'v':
|
|
if (db) {
|
|
char *o = sdb_querys (db, NULL, 0, input + 3);
|
|
if (o && *o) {
|
|
r_cons_print (o);
|
|
}
|
|
free (o);
|
|
}
|
|
break;
|
|
case '*':
|
|
r_core_bin_export_info (core, R_MODE_RADARE);
|
|
break;
|
|
case '.':
|
|
case ' ':
|
|
if (db) {
|
|
char *o = sdb_querys (db, NULL, 0, input + 2);
|
|
if (o && *o) {
|
|
r_cons_print (o);
|
|
}
|
|
free (o);
|
|
}
|
|
break;
|
|
case '\0':
|
|
if (db) {
|
|
char *o = sdb_querys (db, NULL, 0, "*");
|
|
if (o && *o) {
|
|
r_cons_print (o);
|
|
}
|
|
free (o);
|
|
}
|
|
break;
|
|
case '?':
|
|
default:
|
|
eprintf ("Usage: ik [sdb-query]\n");
|
|
eprintf ("Usage: ik* # load all header information\n");
|
|
}
|
|
goto done;
|
|
}
|
|
break;
|
|
case 'o': // "io"
|
|
{
|
|
if (!desc) {
|
|
eprintf ("Core file not open\n");
|
|
return 0;
|
|
}
|
|
const char *fn = input[1] == ' '? input + 2: desc->name;
|
|
ut64 baddr = r_config_get_i (core->config, "bin.baddr");
|
|
r_core_bin_load (core, fn, baddr);
|
|
}
|
|
break;
|
|
#define RBININFO(n,x,y,z)\
|
|
if (is_array) {\
|
|
pj_k (pj, n);\
|
|
}\
|
|
if (z) { playMsg (core, n, z);}\
|
|
r_core_bin_info (core, x, pj, mode, va, NULL, y);
|
|
case 'A': // "iA"
|
|
if (input[1] == 'j') {
|
|
pj_o (pj);
|
|
r_bin_list_archs (core->bin, pj, 'j');
|
|
pj_end (pj);
|
|
} else {
|
|
r_bin_list_archs (core->bin, NULL, 1);
|
|
newline = false;
|
|
}
|
|
break;
|
|
case 'E': // "iE"
|
|
{
|
|
if (input[1] == 'j' && input[2] == '.') {
|
|
mode = R_MODE_JSON;
|
|
INIT_PJ ();
|
|
RBININFO ("exports", R_CORE_BIN_ACC_EXPORTS, input + 2, 0);
|
|
} else {
|
|
RBININFO ("exports", R_CORE_BIN_ACC_EXPORTS, input + 1, 0);
|
|
}
|
|
while (*(++input)) ;
|
|
input--;
|
|
break;
|
|
}
|
|
case 't': // "it"
|
|
{
|
|
ut64 limit = r_config_get_i (core->config, "bin.hashlimit");
|
|
RBinInfo *info = r_bin_get_info (core->bin);
|
|
if (!info) {
|
|
eprintf ("r_bin_get_info: Cannot get bin info\n");
|
|
return 0;
|
|
}
|
|
|
|
RList *new_hashes = r_bin_file_compute_hashes (core->bin, limit);
|
|
RList *old_hashes = r_bin_file_set_hashes (core->bin, new_hashes);
|
|
bool equal = true;
|
|
if (!r_list_empty (new_hashes) && !r_list_empty (old_hashes)) {
|
|
if (!is_equal_file_hashes (new_hashes, old_hashes, &equal)) {
|
|
eprintf ("is_equal_file_hashes: Cannot compare file hashes\n");
|
|
r_list_free (old_hashes);
|
|
return 0;
|
|
}
|
|
}
|
|
RBinFileHash *fh_old, *fh_new;
|
|
RListIter *hiter_old, *hiter_new;
|
|
const bool is_json = input[1] == 'j'; // "itj"
|
|
if (is_json) { // "itj"
|
|
pj_o (pj);
|
|
r_list_foreach (new_hashes, hiter_new, fh_new) {
|
|
pj_ks (pj, fh_new->type, fh_new->hex);
|
|
}
|
|
if (!equal) {
|
|
// print old hashes prefixed with `o` character like `omd5` and `isha1`
|
|
r_list_foreach (old_hashes, hiter_old, fh_old) {
|
|
char *key = r_str_newf ("o%s", fh_old->type);
|
|
pj_ks (pj, key, fh_old->hex);
|
|
free (key);
|
|
}
|
|
}
|
|
pj_end (pj);
|
|
} else { // "it"
|
|
if (!equal) {
|
|
eprintf ("File has been modified.\n");
|
|
hiter_new = r_list_iterator (new_hashes);
|
|
hiter_old = r_list_iterator (old_hashes);
|
|
while (r_list_iter_next (hiter_new) && r_list_iter_next (hiter_old)) {
|
|
fh_new = (RBinFileHash *)r_list_iter_get (hiter_new);
|
|
fh_old = (RBinFileHash *)r_list_iter_get (hiter_old);
|
|
if (strcmp (fh_new->type, fh_old->type)) {
|
|
eprintf ("Wrong file hashes structure");
|
|
}
|
|
if (!strcmp (fh_new->hex, fh_old->hex)) {
|
|
eprintf ("= %s %s\n", fh_new->type, fh_new->hex); // output one line because hash remains same `= hashtype hashval`
|
|
} else {
|
|
// output diff-like two lines, one with old hash val `- hashtype hashval` and one with new `+ hashtype hashval`
|
|
eprintf ("- %s %s\n+ %s %s\n",
|
|
fh_old->type, fh_old->hex,
|
|
fh_new->type, fh_new->hex);
|
|
}
|
|
}
|
|
} else { // hashes are equal
|
|
r_list_foreach (new_hashes, hiter_new, fh_new) {
|
|
r_cons_printf ("%s %s\n", fh_new->type, fh_new->hex);
|
|
}
|
|
}
|
|
newline = false;
|
|
}
|
|
r_list_free (old_hashes);
|
|
}
|
|
break;
|
|
case 'Z': // "iZ"
|
|
RBININFO ("size", R_CORE_BIN_ACC_SIZE, NULL, 0);
|
|
break;
|
|
case 'O': // "iO"
|
|
switch (input[1]) {
|
|
case ' ':
|
|
r_sys_cmdf ("rabin2 -O \"%s\" \"%s\"", r_str_trim_head_ro (input + 1), desc->name);
|
|
break;
|
|
default:
|
|
r_sys_cmdf ("rabin2 -O help");
|
|
break;
|
|
}
|
|
return 0;
|
|
case 'S': // "iS"
|
|
//we comes from ia or iS
|
|
if ((input[1] == 'm' && input[2] == 'z') || !input[1]) {
|
|
RBININFO ("sections", R_CORE_BIN_ACC_SECTIONS, NULL, 0);
|
|
} else if (input[1] == 'S' && !input[2]) { // "iSS"
|
|
RBININFO ("segments", R_CORE_BIN_ACC_SEGMENTS, NULL, 0);
|
|
} else { //iS/iSS entropy,sha1
|
|
const char *name = "sections";
|
|
int action = R_CORE_BIN_ACC_SECTIONS;
|
|
int param_shift = 0;
|
|
if (input[1] == 'S') {
|
|
name = "segments";
|
|
input++;
|
|
action = R_CORE_BIN_ACC_SEGMENTS;
|
|
}
|
|
// case for iS=
|
|
if (input[1] == '=') {
|
|
mode = R_MODE_EQUAL;
|
|
} else if (input[1] == '*') {
|
|
mode = R_MODE_RADARE;
|
|
} else if (input[1] == 'q' && input[2] == '.') {
|
|
mode = R_MODE_SIMPLE;
|
|
} else if (input[1] == 'j' && input[2] == '.') {
|
|
mode = R_MODE_JSON;
|
|
INIT_PJ ();
|
|
}
|
|
RBinObject *obj = r_bin_cur_object (core->bin);
|
|
if (mode == R_MODE_RADARE || mode == R_MODE_JSON || mode == R_MODE_SIMPLE) {
|
|
if (input[param_shift + 1]) {
|
|
param_shift ++;
|
|
}
|
|
}
|
|
RBININFO (name, action, input + 1 + param_shift,
|
|
(obj && obj->sections)? r_list_length (obj->sections): 0);
|
|
}
|
|
//we move input until get '\0'
|
|
while (*(++input)) ;
|
|
//input-- because we are inside a while that does input++
|
|
// oob read if not input--
|
|
input--;
|
|
break;
|
|
case 'H': // "iH"
|
|
if (input[1] == 'H') { // "iHH"
|
|
RBININFO ("header", R_CORE_BIN_ACC_HEADER, NULL, -1);
|
|
break;
|
|
}
|
|
case 'h': // "ih"
|
|
RBININFO ("fields", R_CORE_BIN_ACC_FIELDS, NULL, 0);
|
|
break;
|
|
case 'l': { // "il"
|
|
RBinObject *obj = r_bin_cur_object (core->bin);
|
|
RBININFO ("libs", R_CORE_BIN_ACC_LIBS, NULL, (obj && obj->libs)? r_list_length (obj->libs): 0);
|
|
break;
|
|
}
|
|
case 'L': { // "iL"
|
|
char *ptr = strchr (input, ' ');
|
|
int json = input[1] == 'j'? 'j': 0;
|
|
|
|
if (ptr && ptr[1]) {
|
|
const char *plugin_name = ptr + 1;
|
|
if (is_array) {
|
|
pj_k (pj, "plugin");
|
|
}
|
|
r_bin_list_plugin (core->bin, plugin_name, pj, json);
|
|
} else {
|
|
r_bin_list (core->bin, pj, json);
|
|
}
|
|
newline = false;
|
|
goto done;
|
|
}
|
|
case 's': { // "is"
|
|
RBinObject *obj = r_bin_cur_object (core->bin);
|
|
// Case for isj.
|
|
if (input[1] == 'j' && input[2] == '.') {
|
|
mode = R_MODE_JSON;
|
|
INIT_PJ ();
|
|
RBININFO ("symbols", R_CORE_BIN_ACC_SYMBOLS, input + 2, (obj && obj->symbols)? r_list_length (obj->symbols): 0);
|
|
} else if (input[1] == 'q' && input[2] == 'q') {
|
|
mode = R_MODE_SIMPLEST;
|
|
RBININFO ("symbols", R_CORE_BIN_ACC_SYMBOLS, input + 1, (obj && obj->symbols)? r_list_length (obj->symbols): 0);
|
|
} else if (input[1] == 'q' && input[2] == '.') {
|
|
mode = R_MODE_SIMPLE;
|
|
RBININFO ("symbols", R_CORE_BIN_ACC_SYMBOLS, input + 2, 0);
|
|
} else {
|
|
RBININFO ("symbols", R_CORE_BIN_ACC_SYMBOLS, input + 1, (obj && obj->symbols)? r_list_length (obj->symbols): 0);
|
|
}
|
|
while (*(++input)) ;
|
|
input--;
|
|
break;
|
|
}
|
|
case 'R': // "iR"
|
|
RBININFO ("resources", R_CORE_BIN_ACC_RESOURCES, NULL, 0);
|
|
break;
|
|
case 'r': // "ir"
|
|
RBININFO ("relocs", R_CORE_BIN_ACC_RELOCS, NULL, 0);
|
|
break;
|
|
case 'X': // "iX"
|
|
RBININFO ("source", R_CORE_BIN_ACC_SOURCE, NULL, 0);
|
|
break;
|
|
case 'd': // "id"
|
|
if (input[1] == 'p') { // "idp"
|
|
SPDBOptions pdbopts;
|
|
RBinInfo *info;
|
|
bool file_found;
|
|
char *filename;
|
|
|
|
switch (input[2]) {
|
|
case ' ': // "idp file.pdb"
|
|
r_core_cmdf (core, ".idpi* %s", input + 3);
|
|
while (input[2]) {
|
|
input++;
|
|
}
|
|
break;
|
|
case '\0': // "idp"
|
|
r_core_cmd0 (core, ".idpi*");
|
|
break;
|
|
case 'd': // "idpd"
|
|
pdbopts.user_agent = (char*) r_config_get (core->config, "pdb.useragent");
|
|
pdbopts.extract = r_config_get_i (core->config, "pdb.extract");
|
|
pdbopts.symbol_store_path = (char*) r_config_get (core->config, "pdb.symstore");
|
|
char *str = strdup (r_config_get (core->config, "pdb.server"));
|
|
RList *server_l = r_str_split_list (str, ";", 0);
|
|
RListIter *it;
|
|
char *server;
|
|
int r = 1;
|
|
r_list_foreach (server_l, it, server) {
|
|
pdbopts.symbol_server = server;
|
|
r = r_bin_pdb_download (core, pj, input[3] == 'j', &pdbopts);
|
|
if (!r) {
|
|
break;
|
|
}
|
|
}
|
|
if (r > 0) {
|
|
eprintf ("Error while downloading pdb file\n");
|
|
}
|
|
free (str);
|
|
r_list_free (server_l);
|
|
input++;
|
|
break;
|
|
case 'i': // "idpi"
|
|
info = r_bin_get_info (core->bin);
|
|
filename = strchr (input, ' ');
|
|
while (input[2]) input++;
|
|
if (filename) {
|
|
*filename++ = '\0';
|
|
filename = strdup (filename);
|
|
file_found = r_file_exists (filename);
|
|
} else {
|
|
/* Autodetect local file */
|
|
if (!info || !info->debug_file_name) {
|
|
eprintf ("Cannot get file's debug information\n");
|
|
break;
|
|
}
|
|
// Check raw path for debug filename
|
|
file_found = r_file_exists (r_file_basename (info->debug_file_name));
|
|
if (file_found) {
|
|
filename = strdup (r_file_basename (info->debug_file_name));
|
|
} else {
|
|
// Check debug filename basename in current directory
|
|
char* basename = (char*) r_file_basename (info->debug_file_name);
|
|
file_found = r_file_exists (basename);
|
|
if (!file_found) {
|
|
// Check if debug file is in file directory
|
|
char* dir = r_file_dirname (core->bin->cur->file);
|
|
filename = r_str_newf ("%s/%s", dir, basename);
|
|
file_found = r_file_exists (filename);
|
|
} else {
|
|
filename = strdup (basename);
|
|
}
|
|
}
|
|
|
|
// Last chance: Check if file is in downstream symbol store
|
|
if (!file_found) {
|
|
const char* symstore_path = r_config_get (core->config, "pdb.symstore");
|
|
const char *base_file = r_file_basename (info->debug_file_name);
|
|
char* pdb_path = r_str_newf ("%s" R_SYS_DIR "%s" R_SYS_DIR "%s" R_SYS_DIR "%s",
|
|
symstore_path, base_file, info->guid, base_file);
|
|
file_found = r_file_exists (pdb_path);
|
|
if (file_found) {
|
|
filename = pdb_path;
|
|
} else {
|
|
R_FREE(pdb_path);
|
|
}
|
|
}
|
|
}
|
|
|
|
if (!file_found) {
|
|
if (info->debug_file_name) {
|
|
const char *fn = r_file_basename (info->debug_file_name);
|
|
eprintf ("File '%s' not found in file directory or symbol store\n", fn);
|
|
} else {
|
|
eprintf ("Cannot open file\n");
|
|
}
|
|
free (filename);
|
|
break;
|
|
}
|
|
r_core_pdb_info (core, filename, pj, mode);
|
|
free (filename);
|
|
break;
|
|
case '?':
|
|
default:
|
|
r_core_cmd_help (core, help_msg_id);
|
|
input++;
|
|
break;
|
|
}
|
|
input++;
|
|
} else if (input[1] == '?') { // "id?"
|
|
r_core_cmd_help (core, help_msg_id);
|
|
input++;
|
|
} else { // "id"
|
|
RBININFO ("dwarf", R_CORE_BIN_ACC_DWARF, NULL, -1);
|
|
}
|
|
break;
|
|
case 'i': { // "ii"
|
|
RBinObject *obj = r_bin_cur_object (core->bin);
|
|
RBININFO ("imports", R_CORE_BIN_ACC_IMPORTS, NULL,
|
|
(obj && obj->imports)? r_list_length (obj->imports): 0);
|
|
break;
|
|
}
|
|
case 'I': // "iI"
|
|
RBININFO ("info", R_CORE_BIN_ACC_INFO, NULL, 0);
|
|
break;
|
|
case 'e': // "ie"
|
|
if (input[1] == 'e') {
|
|
RBININFO ("initfini", R_CORE_BIN_ACC_INITFINI, NULL, 0);
|
|
input++;
|
|
} else {
|
|
RBININFO ("entries", R_CORE_BIN_ACC_ENTRIES, NULL, 0);
|
|
}
|
|
break;
|
|
case 'M': // "iM"
|
|
RBININFO ("main", R_CORE_BIN_ACC_MAIN, NULL, 0);
|
|
break;
|
|
case 'm': // "im"
|
|
RBININFO ("memory", R_CORE_BIN_ACC_MEM, NULL, 0);
|
|
break;
|
|
case 'w': // "iw"
|
|
RBININFO ("trycatch", R_CORE_BIN_ACC_TRYCATCH, NULL, 0);
|
|
break;
|
|
case 'V': // "iV"
|
|
RBININFO ("versioninfo", R_CORE_BIN_ACC_VERSIONINFO, NULL, 0);
|
|
break;
|
|
case 'T': // "iT"
|
|
case 'C': // "iC" // rabin2 -C create // should be deprecated and just use iT (or find a better name)
|
|
RBININFO ("signature", R_CORE_BIN_ACC_SIGNATURE, NULL, 0);
|
|
break;
|
|
case 'z': // "iz"
|
|
if (input[1] == '-') { //iz-
|
|
char *strpurge = core->bin->strpurge;
|
|
ut64 addr = core->offset;
|
|
bool old_tmpseek = core->tmpseek;
|
|
input++;
|
|
if (input[1] == ' ') {
|
|
const char *argstr = r_str_trim_head_ro (input + 2);
|
|
ut64 arg = r_num_get (NULL, argstr);
|
|
input++;
|
|
if (arg != 0 || *argstr == '0') {
|
|
addr = arg;
|
|
}
|
|
}
|
|
core->tmpseek = false;
|
|
r_core_cmdf (core, "e bin.str.purge=%s%s0x%" PFMT64x,
|
|
r_str_get (strpurge),
|
|
strpurge && *strpurge ? "," : "",
|
|
addr);
|
|
core->tmpseek = old_tmpseek;
|
|
newline = false;
|
|
} else if (input[1] == 'z') { //izz
|
|
switch (input[2]) {
|
|
case 'z'://izzz
|
|
rdump = true;
|
|
break;
|
|
case '*':
|
|
mode = R_MODE_RADARE;
|
|
break;
|
|
case 'j':
|
|
mode = R_MODE_JSON;
|
|
INIT_PJ ();
|
|
break;
|
|
case 'q': //izzq
|
|
if (input[3] == 'q') { //izzqq
|
|
mode = R_MODE_SIMPLEST;
|
|
input++;
|
|
} else {
|
|
mode = R_MODE_SIMPLE;
|
|
}
|
|
break;
|
|
default:
|
|
mode = R_MODE_PRINT;
|
|
break;
|
|
}
|
|
input++;
|
|
if (rdump) {
|
|
RBinFile *bf = r_bin_cur (core->bin);
|
|
int min = r_config_get_i (core->config, "bin.minstr");
|
|
if (bf) {
|
|
bf->strmode = mode;
|
|
r_bin_dump_strings (bf, min, 2);
|
|
}
|
|
goto done;
|
|
}
|
|
RBININFO ("strings", R_CORE_BIN_ACC_RAW_STRINGS, NULL, 0);
|
|
} else {
|
|
RBinObject *obj = r_bin_cur_object (core->bin);
|
|
if (input[1] == 'q') {
|
|
mode = (input[2] == 'q')
|
|
? R_MODE_SIMPLEST
|
|
: R_MODE_SIMPLE;
|
|
input++;
|
|
}
|
|
if (obj) {
|
|
RBININFO ("strings", R_CORE_BIN_ACC_STRINGS, NULL,
|
|
(obj && obj->strings)? r_list_length (obj->strings): 0);
|
|
}
|
|
}
|
|
break;
|
|
case 'c': // "ic"
|
|
// XXX this is dupe of cbin.c:bin_classes()
|
|
if (input[1] == '?') {
|
|
eprintf ("Usage: ic[gljqc**] [class-index or name]\n");
|
|
} else if (input[1] == 'g') {
|
|
RBinClass *cls;
|
|
RListIter *iter;
|
|
RBinObject *obj = r_bin_cur_object (core->bin);
|
|
if (!obj) {
|
|
break;
|
|
}
|
|
bool fullGraph = true;
|
|
if (fullGraph) {
|
|
r_list_foreach (obj->classes, iter, cls) {
|
|
if (cls->super) {
|
|
r_cons_printf ("agn %s\n", cls->super);
|
|
r_cons_printf ("agn %s\n", cls->name);
|
|
r_cons_printf ("age %s %s\n", cls->super, cls->name);
|
|
} else {
|
|
r_cons_printf ("agn %s\n", cls->name);
|
|
}
|
|
}
|
|
} else {
|
|
r_list_foreach (obj->classes, iter, cls) {
|
|
if (cls->super && !strstr (cls->super, "NSObject")) {
|
|
r_cons_printf ("agn %s\n", cls->super);
|
|
r_cons_printf ("agn %s\n", cls->name);
|
|
r_cons_printf ("age %s %s\n", cls->super, cls->name);
|
|
}
|
|
}
|
|
}
|
|
goto done;
|
|
} else if (input[1] == ' ' || input[1] == 'q' || input[1] == 'j' || input[1] == 'l' || input[1] == 'c' || input[1] == '*') {
|
|
RBinClass *cls;
|
|
RBinSymbol *sym;
|
|
RListIter *iter, *iter2;
|
|
RBinObject *obj = r_bin_cur_object (core->bin);
|
|
if (!obj) {
|
|
break;
|
|
}
|
|
if (input[2] && input[2] != '*' && input[2] != 'j' && !strstr (input, "qq")) {
|
|
bool radare2 = strstr (input, "**") != NULL;
|
|
int idx = -1;
|
|
const char * cls_name = NULL;
|
|
if (radare2) {
|
|
input ++;
|
|
}
|
|
if (r_num_is_valid_input (core->num, input + 2)) {
|
|
idx = r_num_math (core->num, input + 2);
|
|
} else {
|
|
const char * first_char = input + ((input[1] == ' ') ? 1 : 2);
|
|
int not_space = strspn (first_char, " ");
|
|
if (first_char[not_space]) {
|
|
cls_name = first_char + not_space;
|
|
}
|
|
}
|
|
if (radare2) {
|
|
input++;
|
|
}
|
|
int count = 0;
|
|
int mode = input[1];
|
|
r_list_foreach (obj->classes, iter, cls) {
|
|
if (radare2) {
|
|
r_cons_printf ("ac %s\n", cls->name);
|
|
r_list_foreach (cls->methods, iter2, sym) {
|
|
r_cons_printf ("ac %s %s 0x%08"PFMT64x"\n", cls->name, sym->name, sym->vaddr);
|
|
}
|
|
continue;
|
|
}
|
|
if ((idx >= 0 && idx != count++) ||
|
|
(cls_name && *cls_name && strcmp (cls_name, cls->name) != 0)) {
|
|
continue;
|
|
}
|
|
switch (mode) {
|
|
case '*':
|
|
r_list_foreach (cls->methods, iter2, sym) {
|
|
r_cons_printf ("f sym.%s @ 0x%"PFMT64x "\n",
|
|
sym->name, sym->vaddr);
|
|
}
|
|
input++;
|
|
break;
|
|
case 'l':
|
|
r_list_foreach (cls->methods, iter2, sym) {
|
|
const char *comma = iter2->p? " ": "";
|
|
r_cons_printf ("%s0x%"PFMT64d, comma, sym->vaddr);
|
|
}
|
|
r_cons_newline ();
|
|
input++;
|
|
break;
|
|
case 'j':
|
|
input++;
|
|
pj_ks (pj, "class", cls->name);
|
|
pj_ka (pj, "methods");
|
|
r_list_foreach (cls->methods, iter2, sym) {
|
|
pj_o (pj);
|
|
pj_ks (pj, "name", sym->name);
|
|
if (sym->method_flags) {
|
|
char *flags = r_core_bin_method_flags_str (sym->method_flags, R_MODE_JSON);
|
|
pj_k (pj, "flags");
|
|
pj_j (pj, flags);
|
|
free (flags);
|
|
}
|
|
pj_kN (pj, "vaddr", sym->vaddr);
|
|
pj_end (pj);
|
|
}
|
|
pj_end (pj);
|
|
break;
|
|
default:
|
|
r_cons_printf ("class %s\n", cls->name);
|
|
r_list_foreach (cls->methods, iter2, sym) {
|
|
char *flags = r_core_bin_method_flags_str (sym->method_flags, 0);
|
|
r_cons_printf ("0x%08"PFMT64x " method %s %s %s\n",
|
|
sym->vaddr, cls->name, flags, sym->name);
|
|
R_FREE (flags);
|
|
}
|
|
break;
|
|
}
|
|
goto done;
|
|
}
|
|
goto done;
|
|
} else if (obj->classes) {
|
|
playMsg (core, "classes", r_list_length (obj->classes));
|
|
if (strstr (input, "qq")) { // "icqq"
|
|
r_list_foreach (obj->classes, iter, cls) {
|
|
if (!isKnownPackage (cls->name)) {
|
|
r_cons_printf ("%s\n", cls->name);
|
|
}
|
|
}
|
|
} else if (input[1] == 'l') { // "icl"
|
|
r_list_foreach (obj->classes, iter, cls) {
|
|
r_list_foreach (cls->methods, iter2, sym) {
|
|
const char *comma = iter2->p? " ": "";
|
|
r_cons_printf ("%s0x%"PFMT64d, comma, sym->vaddr);
|
|
}
|
|
if (!r_list_empty (cls->methods)) {
|
|
r_cons_newline ();
|
|
}
|
|
}
|
|
} else if (input[1] == 'c') { // "icc"
|
|
mode = R_MODE_CLASSDUMP;
|
|
if (input[2] == '*') {
|
|
mode |= R_MODE_RADARE;
|
|
}
|
|
RBININFO ("classes", R_CORE_BIN_ACC_CLASSES, NULL, r_list_length (obj->classes));
|
|
input = " ";
|
|
} else { // "icq"
|
|
if (input[2] == 'j') {
|
|
mode |= R_MODE_JSON; // default mode is R_MODE_SIMPLE
|
|
}
|
|
RBININFO ("classes", R_CORE_BIN_ACC_CLASSES, NULL, r_list_length (obj->classes));
|
|
}
|
|
goto done;
|
|
}
|
|
} else { // "ic"
|
|
RBinObject *obj = r_bin_cur_object (core->bin);
|
|
if (obj && obj->classes) {
|
|
int len = r_list_length (obj->classes);
|
|
RBININFO ("classes", R_CORE_BIN_ACC_CLASSES, NULL, len);
|
|
}
|
|
}
|
|
break;
|
|
case 'D': // "iD"
|
|
if (input[1] != ' ' || !demangle (core, input + 2)) {
|
|
eprintf ("|Usage: iD lang symbolname\n");
|
|
}
|
|
return 0;
|
|
case 'a': // "ia"
|
|
switch (mode) {
|
|
case R_MODE_RADARE: cmd_info (core, "IieEcsSmz*"); break;
|
|
case R_MODE_JSON: cmd_info (core, "IieEcsSmzj"); break;
|
|
case R_MODE_SIMPLE: cmd_info (core, "IieEcsSmzq"); break;
|
|
default: cmd_info (core, "IiEecsSmz"); break;
|
|
}
|
|
break;
|
|
case '?': // "i?"
|
|
r_core_cmd_help (core, help_msg_i);
|
|
goto redone;
|
|
case '*': // "i*"
|
|
if (mode == R_MODE_RADARE) {
|
|
// TODO:handle ** submodes
|
|
mode = R_MODE_RADARE;
|
|
} else {
|
|
mode = R_MODE_RADARE;
|
|
}
|
|
goto done;
|
|
case 'q': // "iq"
|
|
mode = R_MODE_SIMPLE;
|
|
cmd_info_bin (core, va, pj, mode);
|
|
goto done;
|
|
case 'j': // "ij"
|
|
mode = R_MODE_JSON;
|
|
if (is_array > 1) {
|
|
mode |= R_MODE_ARRAY;
|
|
}
|
|
cmd_info_bin (core, va, pj, mode);
|
|
goto done;
|
|
case '.': // "i."
|
|
cmd_info_here (core, pj, input[1]);
|
|
goto done;
|
|
default:
|
|
cmd_info_bin (core, va, pj, mode);
|
|
break;
|
|
}
|
|
// input can be overwritten like the 'input = " ";' a few lines above
|
|
if (input[0] != ' ') {
|
|
input++;
|
|
if ((*input == 'j' || *input == 'q') && (input[0] && !input[1])) {
|
|
break;
|
|
}
|
|
} else {
|
|
break;
|
|
}
|
|
}
|
|
done:
|
|
if (mode & R_MODE_JSON) {
|
|
if (is_array && !is_izzzj && !is_idpij) {
|
|
pj_end (pj);
|
|
}
|
|
r_cons_println (pj_string (pj));
|
|
pj_free (pj);
|
|
} else if (newline) {
|
|
r_cons_newline ();
|
|
}
|
|
redone:
|
|
return 0;
|
|
}
|