2016-03-20 02:24:17 +01:00
|
|
|
/* radare - LGPL - Copyright 2009-2016 - pancake */
|
2009-02-05 22:08:46 +01:00
|
|
|
|
|
|
|
#include <r_core.h>
|
2014-01-25 18:06:17 -06:00
|
|
|
#include <stdlib.h>
|
2015-10-14 23:50:16 +02:00
|
|
|
#include <string.h>
|
2009-02-05 22:08:46 +01:00
|
|
|
|
2016-03-18 13:39:45 +01:00
|
|
|
#define UPDATE_TIME(a) r->times->file_open_time = r_sys_now() - a
|
2016-03-14 22:12:54 +02:00
|
|
|
|
2014-04-23 18:04:25 -05:00
|
|
|
static int r_core_file_do_load_for_debug (RCore *r, ut64 loadaddr, const char *filenameuri);
|
|
|
|
static int r_core_file_do_load_for_io_plugin (RCore *r, ut64 baseaddr, ut64 loadaddr);
|
|
|
|
|
2014-10-17 00:36:00 +02:00
|
|
|
R_API int r_core_file_reopen(RCore *core, const char *args, int perm, int loadbin) {
|
2014-08-18 18:06:27 +02:00
|
|
|
int isdebug = r_config_get_i (core->config, "cfg.debug");
|
2011-12-05 02:42:06 +01:00
|
|
|
char *path;
|
2015-08-27 23:00:38 +02:00
|
|
|
ut64 ofrom = 0, laddr = r_config_get_i (core->config, "bin.laddr");
|
2014-08-18 18:06:27 +02:00
|
|
|
RCoreFile *file = NULL;
|
|
|
|
RCoreFile *ofile = core->file;
|
2016-06-06 22:40:34 +02:00
|
|
|
RBinFile *bf = (ofile && ofile->desc)
|
|
|
|
? r_bin_file_find_by_fd (core->bin, ofile->desc->fd)
|
|
|
|
: NULL;
|
2014-05-28 04:34:12 +02:00
|
|
|
RIODesc *odesc = ofile ? ofile->desc : NULL;
|
2014-11-07 10:52:44 +01:00
|
|
|
char *ofilepath = NULL, *obinfilepath = bf ? strdup (bf->file) : NULL;
|
2015-09-14 12:35:38 +02:00
|
|
|
int newpid, ret = false;
|
2014-10-17 00:04:52 +02:00
|
|
|
ut64 origoff = core->offset;
|
2014-11-03 10:47:02 +01:00
|
|
|
if (odesc) {
|
|
|
|
if (odesc->referer) {
|
|
|
|
ofilepath = odesc->referer;
|
|
|
|
} else if (odesc->uri) {
|
|
|
|
ofilepath = odesc->uri;
|
|
|
|
}
|
|
|
|
}
|
2014-11-03 11:47:51 +01:00
|
|
|
|
2012-10-22 10:43:10 +02:00
|
|
|
if (r_sandbox_enable (0)) {
|
|
|
|
eprintf ("Cannot reopen in sandbox\n");
|
2015-02-11 21:50:37 +01:00
|
|
|
free (obinfilepath);
|
2015-09-14 12:35:38 +02:00
|
|
|
return false;
|
2012-10-22 10:43:10 +02:00
|
|
|
}
|
2011-12-05 02:42:06 +01:00
|
|
|
if (!core->file) {
|
|
|
|
eprintf ("No file opened to reopen\n");
|
Fixed some issues in bin/dwarf.c and Fixed 1205194, 1205193, 1205192, 1205202, 1205203, 1205204, 1205205, 1205209, 1205208, 1205207, 1205206
2014-04-25 15:14:49 -05:00
|
|
|
free (ofilepath);
|
|
|
|
free (obinfilepath);
|
2015-09-14 12:35:38 +02:00
|
|
|
return false;
|
2011-12-05 02:42:06 +01:00
|
|
|
}
|
2014-04-23 18:04:25 -05:00
|
|
|
newpid = odesc ? odesc->fd : -1;
|
|
|
|
|
2014-08-18 18:06:27 +02:00
|
|
|
if (isdebug) {
|
2016-06-06 22:40:34 +02:00
|
|
|
r_debug_kill (core->dbg, core->dbg->pid, core->dbg->tid, 9); // KILL
|
2014-10-17 00:04:52 +02:00
|
|
|
perm = 7;
|
|
|
|
} else {
|
|
|
|
if (!perm) {
|
|
|
|
perm = 4; //R_IO_READ;
|
|
|
|
}
|
|
|
|
}
|
2014-11-03 11:47:51 +01:00
|
|
|
if (!ofilepath) {
|
2014-10-17 00:04:52 +02:00
|
|
|
eprintf ("Unknown file path");
|
2015-01-27 16:03:04 +01:00
|
|
|
free (obinfilepath);
|
2015-09-14 12:35:38 +02:00
|
|
|
return false;
|
2014-04-23 18:04:25 -05:00
|
|
|
}
|
2014-02-21 10:58:31 +01:00
|
|
|
|
|
|
|
// HACK: move last mapped address to higher place
|
2014-04-23 18:04:25 -05:00
|
|
|
// XXX - why does this hack work?
|
|
|
|
if (ofile->map) {
|
|
|
|
ofrom = ofile->map->from;
|
2014-10-17 00:04:52 +02:00
|
|
|
ofile->map->from = UT32_MAX;
|
2014-04-23 18:04:25 -05:00
|
|
|
}
|
2014-04-22 23:02:46 -05:00
|
|
|
// closing the file to make sure there are no collisions
|
|
|
|
// when the new memory maps are created.
|
2014-11-03 11:47:51 +01:00
|
|
|
path = strdup (ofilepath);
|
2015-04-19 02:38:41 +03:00
|
|
|
free (obinfilepath);
|
2016-06-06 22:40:34 +02:00
|
|
|
obinfilepath = strdup (ofilepath);
|
2014-11-03 11:47:51 +01:00
|
|
|
|
2015-08-27 23:00:38 +02:00
|
|
|
file = r_core_file_open (core, path, perm, laddr);
|
2011-12-05 02:42:06 +01:00
|
|
|
if (file) {
|
2016-06-07 10:03:04 +02:00
|
|
|
bool had_rbin_info = false;
|
2015-08-27 23:00:38 +02:00
|
|
|
|
2016-12-19 16:44:51 +01:00
|
|
|
if (ofile->map) {
|
|
|
|
ofile->map->from = ofrom;
|
|
|
|
}
|
2016-06-07 10:03:04 +02:00
|
|
|
if (ofile->desc) {
|
|
|
|
if (r_bin_file_delete (core->bin, ofile->desc->fd)) {
|
|
|
|
had_rbin_info = true;
|
|
|
|
}
|
2014-10-17 00:36:00 +02:00
|
|
|
}
|
2014-04-23 18:04:25 -05:00
|
|
|
r_core_file_close (core, ofile);
|
2014-04-25 14:21:20 -05:00
|
|
|
r_core_file_set_by_file (core, file);
|
2016-06-07 10:03:04 +02:00
|
|
|
if (file->desc) {
|
|
|
|
r_core_file_set_by_fd (core, file->desc->fd);
|
|
|
|
}
|
2014-04-23 18:04:25 -05:00
|
|
|
ofile = NULL;
|
|
|
|
odesc = NULL;
|
2014-10-17 00:04:52 +02:00
|
|
|
// core->file = file;
|
2012-10-22 10:12:13 +02:00
|
|
|
eprintf ("File %s reopened in %s mode\n", path,
|
2016-11-06 02:12:10 +01:00
|
|
|
(perm & R_IO_WRITE)? "read-write": "read-only");
|
2014-04-23 18:04:25 -05:00
|
|
|
|
2015-08-27 23:00:38 +02:00
|
|
|
if (loadbin && (loadbin == 2 || had_rbin_info)) {
|
|
|
|
ut64 baddr = r_config_get_i (core->config, "bin.baddr");
|
2014-10-17 00:36:00 +02:00
|
|
|
ret = r_core_bin_load (core, obinfilepath, baddr);
|
|
|
|
if (!ret) {
|
|
|
|
eprintf ("Error: Failed to reload rbin for: %s", path);
|
|
|
|
}
|
2014-04-23 18:04:25 -05:00
|
|
|
}
|
|
|
|
|
2016-06-06 22:40:34 +02:00
|
|
|
if (core->bin->cur && file->desc && !loadbin) {
|
|
|
|
//force here NULL because is causing uaf look this better in future XXX @alvarofe
|
|
|
|
core->bin->cur = NULL;
|
|
|
|
}
|
2012-10-22 10:12:13 +02:00
|
|
|
// close old file
|
2014-08-18 18:06:27 +02:00
|
|
|
} else if (ofile) {
|
2014-04-22 23:02:46 -05:00
|
|
|
eprintf ("r_core_file_reopen: Cannot reopen file: %s with perms 0x%04x,"
|
|
|
|
" attempting to open read-only.\n", path, perm);
|
2014-02-21 10:58:31 +01:00
|
|
|
// lower it down back
|
2014-04-22 23:02:46 -05:00
|
|
|
//ofile = r_core_file_open (core, path, R_IO_READ, addr);
|
2014-04-25 14:21:20 -05:00
|
|
|
r_core_file_set_by_file (core, ofile);
|
2016-10-20 15:02:25 +02:00
|
|
|
if (ofile->map) {
|
|
|
|
ofile->map->from = ofrom;
|
|
|
|
}
|
2014-10-17 00:04:52 +02:00
|
|
|
} else {
|
|
|
|
eprintf ("Cannot reopen\n");
|
2011-12-05 02:42:06 +01:00
|
|
|
}
|
2014-08-18 18:06:27 +02:00
|
|
|
if (isdebug) {
|
2016-08-26 23:51:56 +02:00
|
|
|
int newtid = newpid;
|
2014-04-23 18:04:25 -05:00
|
|
|
// XXX - select the right backend
|
2016-06-07 10:03:04 +02:00
|
|
|
if (core->file && core->file->desc) {
|
2014-05-28 04:34:12 +02:00
|
|
|
newpid = core->file->desc->fd;
|
2017-01-16 12:22:24 +01:00
|
|
|
#if __linux__
|
|
|
|
core->dbg->main_pid = newpid;
|
|
|
|
newtid = newpid;
|
|
|
|
#endif
|
2016-08-26 23:51:56 +02:00
|
|
|
#if __WINDOWS__
|
|
|
|
newpid = core->io->winpid;
|
|
|
|
newtid = core->io->wintid;
|
|
|
|
r_debug_select (core->dbg, newpid, newtid);
|
|
|
|
#endif
|
2016-06-07 10:03:04 +02:00
|
|
|
}
|
2016-01-02 23:25:20 +01:00
|
|
|
//reopen and attach
|
|
|
|
r_core_setup_debugger (core, "native", true);
|
2016-08-26 23:51:56 +02:00
|
|
|
r_debug_select (core->dbg, newpid, newtid);
|
2011-12-05 02:42:06 +01:00
|
|
|
}
|
2014-04-25 14:21:20 -05:00
|
|
|
if (core->file) {
|
|
|
|
RCoreFile * cf = core->file;
|
2014-05-28 04:34:12 +02:00
|
|
|
RIODesc *desc = cf ? cf->desc : NULL;
|
2014-10-17 00:04:52 +02:00
|
|
|
if (desc) {
|
|
|
|
#if 0
|
2014-04-27 02:06:50 -05:00
|
|
|
r_io_raise (core->io, desc->fd);
|
|
|
|
core->switch_file_view = 1;
|
2014-10-17 00:04:52 +02:00
|
|
|
#endif
|
2016-08-15 13:56:23 -05:00
|
|
|
r_core_block_read (core);
|
2014-04-27 02:06:50 -05:00
|
|
|
} else {
|
2014-09-09 17:01:04 +02:00
|
|
|
const char *name = (cf && cf->desc) ? cf->desc->name : "ERROR";
|
2014-04-27 02:06:50 -05:00
|
|
|
eprintf ("Error: Unable to switch the view to file: %s\n", name);
|
|
|
|
}
|
2014-04-23 18:04:25 -05:00
|
|
|
}
|
2014-10-20 22:48:50 +02:00
|
|
|
r_core_seek (core, origoff, 1);
|
2014-08-19 10:15:46 +02:00
|
|
|
if (isdebug) {
|
2015-01-29 02:25:00 +01:00
|
|
|
r_core_cmd0 (core, ".dm*");
|
2014-08-19 10:15:46 +02:00
|
|
|
r_core_cmd0 (core, ".dr*");
|
2015-10-31 01:57:52 +01:00
|
|
|
r_core_cmd0 (core, "sr PC");
|
2016-04-13 23:18:36 +02:00
|
|
|
} else {
|
|
|
|
ut64 gp = r_num_math (core->num, "loc._gp");
|
|
|
|
if (gp && gp != UT64_MAX) {
|
|
|
|
r_config_set_i (core->config, "anal.gp", gp);
|
|
|
|
}
|
2014-08-19 10:15:46 +02:00
|
|
|
}
|
2016-02-16 02:42:44 +01:00
|
|
|
// update anal io bind
|
|
|
|
r_io_bind (core->io, &(core->anal->iob));
|
2014-05-03 04:26:07 +04:00
|
|
|
// This is done to ensure that the file is correctly
|
2014-04-22 23:02:46 -05:00
|
|
|
// loaded into the view
|
2014-04-23 18:04:25 -05:00
|
|
|
free (obinfilepath);
|
2014-11-03 11:47:51 +01:00
|
|
|
//free (ofilepath);
|
2011-12-05 02:42:06 +01:00
|
|
|
free (path);
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
2011-02-04 11:30:08 +01:00
|
|
|
// NOTE: probably not all environment vars takes sesnse
|
|
|
|
// because they can be replaced by commands in the given
|
|
|
|
// command.. we should only expose the most essential and
|
|
|
|
// unidirectional ones.
|
2014-06-20 13:25:17 +02:00
|
|
|
R_API void r_core_sysenv_help(const RCore* core) {
|
|
|
|
const char* help_msg[] = {
|
2015-07-09 23:44:45 +02:00
|
|
|
"Usage:", "!<cmd>", " Run given command as in system(3)",
|
2014-06-20 13:25:17 +02:00
|
|
|
"!", "", "list all historic commands",
|
|
|
|
"!", "ls", "execute 'ls' in shell",
|
|
|
|
"!!", "", "save command history to hist file",
|
|
|
|
"!!", "ls~txt", "print output of 'ls' and grep for 'txt'",
|
2014-08-24 10:41:32 +02:00
|
|
|
".!", "rabin2 -rpsei ${FILE}", "run each output line as a r2 cmd",
|
2014-06-20 13:25:17 +02:00
|
|
|
"!", "echo $SIZE", "display file size",
|
2015-07-09 23:44:45 +02:00
|
|
|
"!=!", "", "enable remotecmd mode",
|
|
|
|
"=!=", "", "disable remotecmd mode",
|
2014-06-20 13:25:17 +02:00
|
|
|
"\nEnvironment:", "", "",
|
2016-08-28 13:39:15 +02:00
|
|
|
"R2_FILE", "", "file name",
|
|
|
|
"R2_OFFSET", "", "10base offset 64bit value",
|
|
|
|
"R2_BYTES", "", "TODO: variable with bytes in curblock",
|
|
|
|
"R2_XOFFSET", "", "same as above, but in 16 base",
|
|
|
|
"R2_BSIZE", "", "block size",
|
|
|
|
"R2_ENDIAN", "", "'big' or 'little'",
|
|
|
|
"R2_IOVA", "", "is io.va true? virtual addressing (1,0)",
|
|
|
|
"R2_DEBUG", "", "debug mode enabled? (1,0)",
|
|
|
|
"R2_BLOCK", "", "TODO: dump current block to tmp file",
|
|
|
|
"R2_SIZE", "","file size",
|
|
|
|
"R2_ARCH", "", "value of asm.arch",
|
2015-10-19 13:21:12 +02:00
|
|
|
"RABIN2_LANG", "", "assume this lang to demangle",
|
|
|
|
"RABIN2_DEMANGLE", "", "demangle or not",
|
2014-12-24 19:49:07 +01:00
|
|
|
"PDB_SERVER", "", "e pdb.server",
|
2014-06-20 13:25:17 +02:00
|
|
|
NULL
|
|
|
|
};
|
|
|
|
r_core_cmd_help (core, help_msg);
|
2011-02-04 11:30:08 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
R_API void r_core_sysenv_end(RCore *core, const char *cmd) {
|
|
|
|
// TODO: remove tmpfilez
|
2016-06-14 01:33:46 +02:00
|
|
|
if (strstr (cmd, "R2_BLOCK")) {
|
2011-02-04 11:30:08 +01:00
|
|
|
// remove temporary BLOCK file
|
2016-06-14 01:33:46 +02:00
|
|
|
char *f = r_sys_getenv ("R2_BLOCK");
|
2012-02-09 01:38:16 +01:00
|
|
|
if (f) {
|
|
|
|
r_file_rm (f);
|
2016-06-14 01:33:46 +02:00
|
|
|
r_sys_setenv ("R2_BLOCK", NULL);
|
2014-05-03 04:26:07 +04:00
|
|
|
free (f);
|
2012-02-09 01:38:16 +01:00
|
|
|
}
|
2011-02-04 11:30:08 +01:00
|
|
|
}
|
2016-06-14 01:33:46 +02:00
|
|
|
r_sys_setenv ("R2_FILE", NULL);
|
|
|
|
r_sys_setenv ("R2_BYTES", NULL);
|
|
|
|
r_sys_setenv ("R2_OFFSET", NULL);
|
2011-02-04 11:30:08 +01:00
|
|
|
}
|
|
|
|
|
2010-08-19 20:28:25 +02:00
|
|
|
#if DISCUSS
|
2016-03-20 02:24:17 +01:00
|
|
|
EDITOR r_sys_setenv ("EDITOR", r_config_get (core->config, "cfg.editor"));
|
|
|
|
CURSOR cursor position (offset from curseek)
|
|
|
|
VERBOSE cfg.verbose
|
2010-08-19 20:28:25 +02:00
|
|
|
#endif
|
2016-03-20 02:24:17 +01:00
|
|
|
R_API char *r_core_sysenv_begin(RCore *core, const char *cmd) {
|
|
|
|
char *f, *ret = strdup (cmd);
|
|
|
|
if (strstr (cmd, "R2_BYTES")) {
|
2012-02-09 01:38:16 +01:00
|
|
|
char *s = r_hex_bin2strdup (core->block, core->blocksize);
|
2016-03-20 02:24:17 +01:00
|
|
|
r_sys_setenv ("R2_BYTES", s);
|
2012-02-09 01:38:16 +01:00
|
|
|
free (s);
|
2011-02-04 11:30:08 +01:00
|
|
|
}
|
2014-12-24 19:49:07 +01:00
|
|
|
r_sys_setenv ("PDB_SERVER", r_config_get (core->config, "pdb.server"));
|
2014-09-26 17:58:17 +02:00
|
|
|
if (core->file && core->file->desc && core->file->desc->name) {
|
2016-03-20 02:24:17 +01:00
|
|
|
r_sys_setenv ("R2_FILE", core->file->desc->name);
|
|
|
|
r_sys_setenv ("R2_SIZE", sdb_fmt (0, "%"PFMT64d,
|
|
|
|
r_io_desc_size (core->io, core->file->desc)));
|
|
|
|
if (strstr (cmd, "R2_BLOCK")) {
|
2014-09-26 17:58:17 +02:00
|
|
|
// replace BLOCK in RET string
|
|
|
|
if ((f = r_file_temp ("r2block"))) {
|
2016-03-20 02:24:17 +01:00
|
|
|
if (r_file_dump (f, core->block, core->blocksize, 0)) {
|
|
|
|
r_sys_setenv ("R2_BLOCK", f);
|
|
|
|
}
|
2014-09-26 17:58:17 +02:00
|
|
|
free (f);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2015-10-19 13:21:12 +02:00
|
|
|
r_sys_setenv ("RABIN2_LANG", r_config_get (core->config, "bin.lang"));
|
|
|
|
r_sys_setenv ("RABIN2_DEMANGLE", r_config_get (core->config, "bin.demangle"));
|
2016-03-20 02:24:17 +01:00
|
|
|
r_sys_setenv ("R2_OFFSET", sdb_fmt (0, "%"PFMT64d, core->offset));
|
|
|
|
r_sys_setenv ("R2_XOFFSET", sdb_fmt (0, "0x%08"PFMT64x, core->offset));
|
|
|
|
r_sys_setenv ("R2_ENDIAN", core->assembler->big_endian? "big": "little");
|
|
|
|
r_sys_setenv ("R2_BSIZE", sdb_fmt (0, "%d", core->blocksize));
|
|
|
|
r_sys_setenv ("R2_ARCH", r_config_get (core->config, "asm.arch"));
|
2016-03-22 01:31:10 +01:00
|
|
|
r_sys_setenv ("R2_COLOR", r_config_get_i (core->config, "scr.color")? "1": "0");
|
2016-03-20 02:24:17 +01:00
|
|
|
r_sys_setenv ("R2_DEBUG", r_config_get_i (core->config, "cfg.debug")?"1":"0");
|
|
|
|
r_sys_setenv ("R2_IOVA", r_config_get_i (core->config, "io.va")?"1":"0");
|
2011-02-04 11:30:08 +01:00
|
|
|
return ret;
|
2010-08-19 20:28:25 +02:00
|
|
|
}
|
|
|
|
|
2015-05-28 11:09:13 +02:00
|
|
|
#if !__linux__
|
2013-12-17 02:10:13 +01:00
|
|
|
static ut64 get_base_from_maps(RCore *core, const char *file) {
|
|
|
|
RDebugMap *map;
|
|
|
|
RListIter *iter;
|
|
|
|
ut64 b = 0LL;
|
2014-01-09 16:57:04 -06:00
|
|
|
|
2013-12-17 02:10:13 +01:00
|
|
|
r_debug_map_sync (core->dbg); // update process memory maps
|
|
|
|
r_list_foreach (core->dbg->maps, iter, map) {
|
2016-03-20 02:24:17 +01:00
|
|
|
if ((map->perm & 5) == 5) {
|
2014-01-11 00:56:02 -06:00
|
|
|
// TODO: make this more flexible
|
|
|
|
// XXX - why "copy/" here?
|
|
|
|
if (map->name && strstr (map->name, "copy/")) return map->addr;
|
|
|
|
if (map->file && !strcmp (map->file, file)) return map->addr;
|
|
|
|
if (map->name && !strcmp (map->name, file)) return map->addr;
|
|
|
|
// XXX - Commented out, as this could unexpected results
|
|
|
|
//b = map->addr;
|
2013-12-17 02:10:13 +01:00
|
|
|
}
|
|
|
|
}
|
2017-01-10 00:55:38 +01:00
|
|
|
// fallback resolution copied from cmd_debug.c:r_debug_get_baddr
|
|
|
|
r_list_foreach (core->dbg->maps, iter, map) {
|
|
|
|
if (map->perm == 5) { // r-x
|
|
|
|
return map->addr;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-12-17 02:10:13 +01:00
|
|
|
return b;
|
|
|
|
}
|
2015-05-28 11:09:13 +02:00
|
|
|
#endif
|
2013-12-17 02:10:13 +01:00
|
|
|
|
2014-01-11 00:56:02 -06:00
|
|
|
R_API int r_core_bin_reload(RCore *r, const char *file, ut64 baseaddr) {
|
2014-01-09 16:57:04 -06:00
|
|
|
int result = 0;
|
2014-05-08 18:35:04 -05:00
|
|
|
RCoreFile *cf = r_core_file_cur (r);
|
2014-05-28 04:34:12 +02:00
|
|
|
RIODesc *desc = cf ? cf->desc : NULL;
|
2014-05-08 18:35:04 -05:00
|
|
|
RBinFile *bf = NULL;
|
2016-11-03 11:51:10 +01:00
|
|
|
if (desc) {
|
|
|
|
result = r_bin_reload (r->bin, desc, baseaddr);
|
|
|
|
}
|
2014-05-08 18:35:04 -05:00
|
|
|
bf = r_bin_cur (r->bin);
|
|
|
|
r_core_bin_set_env (r, bf);
|
2014-01-09 16:57:04 -06:00
|
|
|
return result;
|
2014-01-03 21:36:02 -06:00
|
|
|
}
|
|
|
|
|
2015-10-22 04:48:56 +02:00
|
|
|
static bool setbpint(RCore *r, const char *mode, const char *sym) {
|
|
|
|
RBreakpointItem *bp;
|
|
|
|
RFlagItem *fi = r_flag_get (r->flags, sym);
|
|
|
|
if (!fi) return false;
|
|
|
|
bp = r_bp_add_sw (r->dbg->bp, fi->offset, 1, R_BP_PROT_EXEC);
|
|
|
|
if (bp) {
|
|
|
|
bp->internal = true;
|
|
|
|
#if __linux__
|
2017-01-24 20:22:36 +01:00
|
|
|
bp->data = r_str_newf ("?e %s: %s", mode, sym);
|
2015-10-22 04:48:56 +02:00
|
|
|
#else
|
|
|
|
bp->data = r_str_newf ("?e %s: %s;ps@rdi", mode, sym);
|
|
|
|
#endif
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
eprintf ("Cannot set breakpoint at %s\n", sym);
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2014-04-27 02:06:50 -05:00
|
|
|
// XXX - need to handle index selection during debugging
|
2015-08-21 01:44:44 +02:00
|
|
|
static int r_core_file_do_load_for_debug (RCore *r, ut64 baseaddr, const char *filenameuri) {
|
2014-04-29 11:10:35 -05:00
|
|
|
RCoreFile *cf = r_core_file_cur (r);
|
2014-05-28 04:34:12 +02:00
|
|
|
RIODesc *desc = cf ? cf->desc : NULL;
|
2014-04-27 02:06:50 -05:00
|
|
|
RBinFile *binfile = NULL;
|
2014-05-04 10:03:15 -05:00
|
|
|
RBinPlugin *plugin;
|
2014-04-27 02:06:50 -05:00
|
|
|
int xtr_idx = 0; // if 0, load all if xtr is used
|
2015-09-14 12:35:38 +02:00
|
|
|
int treat_as_rawstr = false;
|
2014-04-23 18:04:25 -05:00
|
|
|
|
2017-02-04 17:35:09 +01:00
|
|
|
if (!strncmp ("dbg://", filenameuri, 6)) {
|
|
|
|
filenameuri += 6;
|
|
|
|
}
|
|
|
|
if (!desc) {
|
|
|
|
return false;
|
|
|
|
}
|
2014-04-29 11:10:35 -05:00
|
|
|
if (cf && desc) {
|
2014-04-23 18:04:25 -05:00
|
|
|
int newpid = desc->fd;
|
2015-11-24 23:28:50 +01:00
|
|
|
#if __WINDOWS__
|
|
|
|
r_debug_select (r->dbg, r->dbg->pid, r->dbg->tid);
|
|
|
|
#else
|
2014-04-23 18:04:25 -05:00
|
|
|
r_debug_select (r->dbg, newpid, newpid);
|
2015-11-24 23:28:50 +01:00
|
|
|
#endif
|
2014-04-23 18:04:25 -05:00
|
|
|
}
|
2015-08-21 01:44:44 +02:00
|
|
|
#if !__linux__
|
2015-11-24 23:28:50 +01:00
|
|
|
#if !__WINDOWS__
|
2014-04-23 18:04:25 -05:00
|
|
|
baseaddr = get_base_from_maps (r, filenameuri);
|
2015-11-24 23:28:50 +01:00
|
|
|
#endif
|
2014-08-25 03:58:22 +02:00
|
|
|
if (baseaddr != UT64_MAX) {
|
2015-08-21 01:44:44 +02:00
|
|
|
r_config_set_i (r->config, "bin.baddr", baseaddr);
|
2014-08-25 03:58:22 +02:00
|
|
|
}
|
2015-05-19 12:05:18 +02:00
|
|
|
#endif
|
2015-08-16 18:49:17 +02:00
|
|
|
// HACK if its a relative path, load from disk instead of memory
|
2015-08-16 22:41:58 +02:00
|
|
|
#if __APPLE__
|
2015-08-16 18:49:17 +02:00
|
|
|
int fd = (filenameuri[0] == '.')? -1: desc->fd;
|
2015-08-16 22:41:58 +02:00
|
|
|
#else
|
|
|
|
int fd = desc->fd;
|
|
|
|
#endif
|
2015-08-21 01:44:44 +02:00
|
|
|
if (!r_bin_load (r->bin, filenameuri, baseaddr, UT64_MAX, xtr_idx, fd, treat_as_rawstr)) {
|
2015-08-14 00:53:57 +02:00
|
|
|
eprintf ("Cannot open %s\n", filenameuri);
|
2014-09-08 23:29:01 +02:00
|
|
|
if (r_config_get_i (r->config, "bin.rawstr")) {
|
2015-09-14 12:35:38 +02:00
|
|
|
treat_as_rawstr = true;
|
2015-08-21 01:44:44 +02:00
|
|
|
if (!r_bin_load (r->bin, filenameuri, baseaddr, UT64_MAX, xtr_idx, desc->fd, treat_as_rawstr)) {
|
2015-09-14 12:35:38 +02:00
|
|
|
return false;
|
2014-09-08 23:29:01 +02:00
|
|
|
}
|
2014-05-04 10:03:15 -05:00
|
|
|
}
|
2014-04-23 18:04:25 -05:00
|
|
|
}
|
|
|
|
|
2015-10-22 03:11:03 +02:00
|
|
|
if (*r_config_get (r->config, "dbg.libs")) {
|
|
|
|
r_core_cmd0 (r, ".dmm*");
|
2015-10-22 04:48:56 +02:00
|
|
|
#if __linux__
|
2017-01-24 20:22:36 +01:00
|
|
|
setbpint (r, "dbg.libs", "sym.imp.dlopen");
|
|
|
|
setbpint (r, "dbg.libs", "sym.imp.dlmopen");
|
|
|
|
setbpint (r, "dbg.unlibs", "sym.imp.dlclose");
|
2015-10-22 04:48:56 +02:00
|
|
|
#elif __APPLE__
|
2017-01-24 20:22:36 +01:00
|
|
|
setbpint (r, "dbg.libs", "sym._dlopen");
|
|
|
|
setbpint (r, "dbg.libs", "sym._dlclose");
|
2015-10-22 04:48:56 +02:00
|
|
|
#endif
|
2015-10-22 03:11:03 +02:00
|
|
|
}
|
2014-05-04 10:03:15 -05:00
|
|
|
binfile = r_bin_cur (r->bin);
|
|
|
|
r_core_bin_set_env (r, binfile);
|
2014-05-08 18:35:04 -05:00
|
|
|
plugin = r_bin_file_cur_plugin (binfile);
|
2015-10-22 04:48:56 +02:00
|
|
|
if (plugin && !strncmp (plugin->name, "any", 5)) {
|
2014-05-04 10:03:15 -05:00
|
|
|
// set use of raw strings
|
2015-09-14 12:35:38 +02:00
|
|
|
r_config_set_i (r->config, "io.va", false);
|
2014-09-08 23:29:01 +02:00
|
|
|
//\\ r_config_set (r->config, "bin.rawstr", "true");
|
2014-05-04 10:03:15 -05:00
|
|
|
// get bin.minstr
|
|
|
|
r->bin->minstrlen = r_config_get_i (r->config, "bin.minstr");
|
2015-10-06 20:52:50 -04:00
|
|
|
r->bin->maxstrbuf = r_config_get_i (r->config, "bin.maxstrbuf");
|
2014-05-04 10:03:15 -05:00
|
|
|
} else if (binfile) {
|
|
|
|
RBinObject *obj = r_bin_get_object (r->bin);
|
|
|
|
RBinInfo * info = obj ? obj->info : NULL;
|
|
|
|
if (plugin && strcmp (plugin->name, "any") && info) {
|
2014-05-04 22:08:46 -05:00
|
|
|
r_core_bin_set_arch_bits (r, binfile->file, info->arch, info->bits);
|
2014-05-04 10:03:15 -05:00
|
|
|
}
|
2014-04-27 02:06:50 -05:00
|
|
|
}
|
|
|
|
|
2014-05-04 10:03:15 -05:00
|
|
|
if (plugin && !strcmp (plugin->name, "dex")) {
|
|
|
|
r_core_cmd0 (r, "\"(fix-dex,wx `#sha1 $s-32 @32` @12 ; wx `#adler32 $s-12 @12` @8)\"\n");
|
|
|
|
}
|
2015-09-14 12:35:38 +02:00
|
|
|
return true;
|
2014-04-23 18:04:25 -05:00
|
|
|
}
|
|
|
|
|
2016-12-04 20:41:56 +01:00
|
|
|
static int r_core_file_do_load_for_io_plugin(RCore *r, ut64 baseaddr, ut64 loadaddr) {
|
2014-04-29 11:10:35 -05:00
|
|
|
RCoreFile *cf = r_core_file_cur (r);
|
2014-05-28 04:34:12 +02:00
|
|
|
RIODesc *desc = cf ? cf->desc : NULL;
|
2014-04-23 18:04:25 -05:00
|
|
|
RBinFile *binfile = NULL;
|
2014-04-27 02:06:50 -05:00
|
|
|
int xtr_idx = 0; // if 0, load all if xtr is used
|
2014-05-04 10:03:15 -05:00
|
|
|
RBinPlugin * plugin;
|
2014-04-23 18:04:25 -05:00
|
|
|
|
2016-12-04 20:41:56 +01:00
|
|
|
if (!desc) {
|
|
|
|
return false;
|
|
|
|
}
|
2014-05-28 04:34:12 +02:00
|
|
|
r_io_use_desc (r->io, desc);
|
2015-11-09 15:14:41 +01:00
|
|
|
|
2016-11-07 19:33:49 +01:00
|
|
|
if (!r_bin_load_io (r->bin, desc, baseaddr, loadaddr, xtr_idx)) {
|
2014-04-29 11:10:35 -05:00
|
|
|
//eprintf ("Failed to load the bin with an IO Plugin.\n");
|
2015-09-14 12:35:38 +02:00
|
|
|
return false;
|
2014-04-27 02:06:50 -05:00
|
|
|
}
|
2014-05-04 10:03:15 -05:00
|
|
|
binfile = r_bin_cur (r->bin);
|
|
|
|
r_core_bin_set_env (r, binfile);
|
2014-05-08 18:35:04 -05:00
|
|
|
plugin = r_bin_file_cur_plugin (binfile);
|
2015-10-26 03:08:39 +01:00
|
|
|
if (plugin && !strcmp (plugin->name, "any") ) {
|
|
|
|
RBinObject *obj = r_bin_get_object (r->bin);
|
|
|
|
RBinInfo * info = obj ? obj->info : NULL;
|
2016-12-04 20:41:56 +01:00
|
|
|
if (!info) {
|
|
|
|
return false;
|
|
|
|
}
|
2014-04-23 18:04:25 -05:00
|
|
|
// set use of raw strings
|
2016-11-07 19:33:49 +01:00
|
|
|
r_core_bin_set_arch_bits (r, binfile->file, info->arch, info->bits);
|
2015-09-14 12:35:38 +02:00
|
|
|
r_config_set_i (r->config, "io.va", false);
|
2014-09-08 23:29:01 +02:00
|
|
|
// r_config_set (r->config, "bin.rawstr", "true");
|
2014-04-23 18:04:25 -05:00
|
|
|
// get bin.minstr
|
|
|
|
r->bin->minstrlen = r_config_get_i (r->config, "bin.minstr");
|
2015-10-06 20:52:50 -04:00
|
|
|
r->bin->maxstrbuf = r_config_get_i (r->config, "bin.maxstrbuf");
|
2014-05-04 10:03:15 -05:00
|
|
|
} else if (binfile) {
|
|
|
|
RBinObject *obj = r_bin_get_object (r->bin);
|
|
|
|
RBinInfo * info = obj ? obj->info : NULL;
|
2016-12-04 20:41:56 +01:00
|
|
|
if (!info) {
|
|
|
|
return false;
|
|
|
|
}
|
2014-05-04 10:03:15 -05:00
|
|
|
if (plugin && strcmp (plugin->name, "any") && info) {
|
2014-08-27 23:51:17 +02:00
|
|
|
r_core_bin_set_arch_bits (r, binfile->file,
|
|
|
|
info->arch, info->bits);
|
2014-05-04 10:03:15 -05:00
|
|
|
} else {
|
2015-09-14 12:35:38 +02:00
|
|
|
r_config_set_i (r->config, "io.va", false);
|
2014-05-04 10:03:15 -05:00
|
|
|
}
|
2014-04-23 18:04:25 -05:00
|
|
|
}
|
|
|
|
|
2014-05-04 10:03:15 -05:00
|
|
|
if (plugin && !strcmp (plugin->name, "dex")) {
|
|
|
|
r_core_cmd0 (r, "\"(fix-dex,wx `#sha1 $s-32 @32` @12 ; wx `#adler32 $s-12 @12` @8)\"\n");
|
2014-04-23 18:04:25 -05:00
|
|
|
}
|
2015-09-14 12:35:38 +02:00
|
|
|
return true;
|
2014-04-23 18:04:25 -05:00
|
|
|
}
|
|
|
|
|
2016-05-09 17:24:12 +02:00
|
|
|
static int try_loadlib(RCore *core, const char *lib, ut64 addr) {
|
|
|
|
RCoreFile *cf = r_core_file_open (core, lib, 0, addr);
|
|
|
|
if (!cf) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
R_API bool r_core_file_loadlib(RCore *core, const char *lib, ut64 libaddr) {
|
|
|
|
const char *ldlibrarypath[] = {
|
|
|
|
"/usr/local/lib",
|
|
|
|
"/usr/lib",
|
|
|
|
"/lib",
|
|
|
|
"./",
|
|
|
|
NULL
|
|
|
|
};
|
|
|
|
const char **libpath = (const char **)&ldlibrarypath;
|
|
|
|
|
|
|
|
if (*lib == '/') {
|
|
|
|
if (try_loadlib (core, lib, libaddr)) {
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
while (*libpath) {
|
|
|
|
bool ret = false;
|
|
|
|
char *s = r_str_newf ("%s/%s", *libpath, lib);
|
|
|
|
if (try_loadlib (core, s, libaddr)) {
|
|
|
|
ret = true;
|
|
|
|
}
|
|
|
|
free (s);
|
|
|
|
if (ret) {
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
libpath++;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2014-04-23 18:04:25 -05:00
|
|
|
R_API int r_core_bin_load(RCore *r, const char *filenameuri, ut64 baddr) {
|
|
|
|
const char *suppress_warning = r_config_get (r->config, "file.nowarn");
|
2014-04-29 11:10:35 -05:00
|
|
|
RCoreFile *cf = r_core_file_cur (r);
|
2014-04-23 18:04:25 -05:00
|
|
|
RBinFile *binfile = NULL;
|
2014-05-28 04:34:12 +02:00
|
|
|
RIODesc *desc = cf ? cf->desc : NULL;
|
2014-05-04 10:03:15 -05:00
|
|
|
RBinPlugin *plugin = NULL;
|
2016-05-24 21:22:15 +01:00
|
|
|
int is_io_load;
|
|
|
|
// NULL deref guard
|
|
|
|
if (!desc) {
|
|
|
|
is_io_load = false;
|
|
|
|
} else {
|
|
|
|
is_io_load = desc && desc->plugin;
|
|
|
|
}
|
2014-04-23 18:04:25 -05:00
|
|
|
|
2014-06-05 23:07:02 +02:00
|
|
|
if (cf) {
|
2016-09-19 13:44:47 +01:00
|
|
|
if (!filenameuri || !*filenameuri) {
|
2014-09-09 17:01:04 +02:00
|
|
|
filenameuri = cf->desc->name;
|
2015-05-19 12:05:18 +02:00
|
|
|
} else if (cf->desc->name && strcmp (filenameuri, cf->desc->name)) {
|
2014-06-05 23:07:02 +02:00
|
|
|
// XXX - this needs to be handled appropriately
|
|
|
|
// if the cf does not match the filenameuri then
|
|
|
|
// either that RCoreFIle * needs to be loaded or a
|
|
|
|
// new RCoreFile * should be opened.
|
2015-05-19 12:05:18 +02:00
|
|
|
if (!strcmp (suppress_warning, "false")) {
|
|
|
|
eprintf ("Error: The filenameuri '%s' is not the same as in RCoreFile: %s\n",
|
2014-09-09 17:01:04 +02:00
|
|
|
filenameuri, cf->desc->name);
|
2015-05-19 12:05:18 +02:00
|
|
|
}
|
2014-06-05 23:07:02 +02:00
|
|
|
}
|
2014-04-23 18:04:25 -05:00
|
|
|
}
|
2014-01-11 00:56:02 -06:00
|
|
|
|
2014-04-23 18:04:25 -05:00
|
|
|
if (!filenameuri) {
|
2013-04-24 09:46:57 +02:00
|
|
|
eprintf ("r_core_bin_load: no file specified\n");
|
2015-09-14 12:35:38 +02:00
|
|
|
return false;
|
2011-11-15 23:26:45 +01:00
|
|
|
}
|
2013-11-09 17:15:30 -06:00
|
|
|
|
2013-12-09 04:56:13 +01:00
|
|
|
r->bin->minstrlen = r_config_get_i (r->config, "bin.minstr");
|
2015-10-06 20:52:50 -04:00
|
|
|
r->bin->maxstrbuf = r_config_get_i (r->config, "bin.maxstrbuf");
|
2013-12-17 02:10:13 +01:00
|
|
|
if (is_io_load) {
|
2014-09-09 17:01:04 +02:00
|
|
|
// TODO? necessary to restore the desc back?
|
2014-06-05 23:07:02 +02:00
|
|
|
// RIODesc *oldesc = desc;
|
2014-01-11 00:56:02 -06:00
|
|
|
// Fix to select pid before trying to load the binary
|
2016-12-04 20:41:56 +01:00
|
|
|
if ((desc->plugin && desc->plugin->isdbg) || r_config_get_i (r->config, "cfg.debug")) {
|
2015-08-21 01:44:44 +02:00
|
|
|
r_core_file_do_load_for_debug (r, baddr, filenameuri);
|
2014-01-11 00:56:02 -06:00
|
|
|
} else {
|
2015-08-28 20:14:26 +02:00
|
|
|
ut64 laddr = r_config_get_i (r->config, "bin.laddr");
|
|
|
|
r_core_file_do_load_for_io_plugin (r, baddr, laddr);
|
2011-11-22 00:59:20 +01:00
|
|
|
}
|
2014-05-28 15:03:31 +02:00
|
|
|
// Restore original desc
|
|
|
|
r_io_use_desc (r->io, desc);
|
2014-05-04 10:03:15 -05:00
|
|
|
}
|
2014-04-23 18:04:25 -05:00
|
|
|
|
2016-11-07 19:33:49 +01:00
|
|
|
if (cf && binfile && desc) {
|
2014-08-28 03:11:13 +02:00
|
|
|
binfile->fd = desc->fd;
|
2016-11-07 19:33:49 +01:00
|
|
|
}
|
2014-05-04 10:03:15 -05:00
|
|
|
binfile = r_bin_cur (r->bin);
|
2015-10-29 18:41:21 +01:00
|
|
|
if (r->bin->cur && r->bin->cur->curplugin && r->bin->cur->curplugin->strfilter) {
|
2015-10-29 13:55:03 +01:00
|
|
|
char msg[2];
|
|
|
|
msg[0] = r->bin->cur->curplugin->strfilter;
|
|
|
|
msg[1] = 0;
|
|
|
|
r_config_set (r->config, "bin.strfilter", msg);
|
|
|
|
}
|
2014-05-04 10:03:15 -05:00
|
|
|
r_core_bin_set_env (r, binfile);
|
2014-05-08 18:35:04 -05:00
|
|
|
plugin = r_bin_file_cur_plugin (binfile);
|
2014-07-07 16:09:03 +02:00
|
|
|
if (plugin && plugin->name && !strncmp (plugin->name, "any", 3)) {
|
2014-05-04 10:03:15 -05:00
|
|
|
// set use of raw strings
|
2014-09-08 23:29:01 +02:00
|
|
|
//r_config_set (r->config, "bin.rawstr", "true");
|
2015-09-14 12:35:38 +02:00
|
|
|
r_config_set_i (r->config, "io.va", false);
|
2014-05-04 10:03:15 -05:00
|
|
|
// get bin.minstr
|
|
|
|
r->bin->minstrlen = r_config_get_i (r->config, "bin.minstr");
|
2015-10-06 20:52:50 -04:00
|
|
|
r->bin->maxstrbuf = r_config_get_i (r->config, "bin.maxstrbuf");
|
2014-05-04 10:03:15 -05:00
|
|
|
} else if (binfile) {
|
|
|
|
RBinObject *obj = r_bin_get_object (r->bin);
|
|
|
|
RBinInfo * info = obj ? obj->info : NULL;
|
2015-05-19 12:05:18 +02:00
|
|
|
if (plugin && plugin->name && info)
|
2015-10-26 03:08:39 +01:00
|
|
|
if (strcmp (plugin->name, "any")) {
|
2014-06-07 11:23:14 +02:00
|
|
|
r_core_bin_set_arch_bits (r, binfile->file,
|
|
|
|
info->arch, info->bits);
|
2015-10-26 03:08:39 +01:00
|
|
|
}
|
2013-04-24 09:46:57 +02:00
|
|
|
}
|
2014-07-07 16:09:03 +02:00
|
|
|
if (plugin && plugin->name && !strcmp (plugin->name, "dex")) {
|
2014-08-28 03:11:13 +02:00
|
|
|
r_core_cmd0 (r, "\"(fix-dex,wx `#sha1 $s-32 @32` @12 ;"
|
|
|
|
" wx `#adler32 $s-12 @12` @8)\"\n");
|
2013-04-12 01:15:00 +02:00
|
|
|
}
|
2016-04-13 23:18:36 +02:00
|
|
|
if (!r_config_get_i (r->config, "cfg.debug")) {
|
|
|
|
/* load GP for mips */
|
|
|
|
ut64 gp = r_num_math (r->num, "loc._gp");
|
|
|
|
if (gp && gp != UT64_MAX) {
|
|
|
|
r_config_set_i (r->config, "anal.gp", gp);
|
|
|
|
}
|
|
|
|
}
|
2016-05-09 17:24:12 +02:00
|
|
|
if (r_config_get_i (r->config, "bin.libs")) {
|
|
|
|
ut64 libaddr = (r->assembler->bits == 64)
|
2016-06-06 17:30:07 +02:00
|
|
|
? 0x00007fff00000000LL
|
2016-05-09 17:24:12 +02:00
|
|
|
: 0x7f000000;
|
|
|
|
const char *lib;
|
|
|
|
RListIter *iter;
|
|
|
|
RList *libs = r_bin_get_libs (r->bin);
|
|
|
|
r_list_foreach (libs, iter, lib) {
|
|
|
|
eprintf ("Opening %s\n", lib);
|
|
|
|
r_core_file_loadlib (r, lib, libaddr);
|
|
|
|
libaddr += 0x2000000;
|
|
|
|
}
|
|
|
|
}
|
2015-09-14 12:35:38 +02:00
|
|
|
return true;
|
2010-10-04 10:55:43 +02:00
|
|
|
}
|
2014-01-20 03:00:16 +01:00
|
|
|
|
2014-01-25 18:06:17 -06:00
|
|
|
R_API RIOMap *r_core_file_get_next_map (RCore *core, RCoreFile * fh, int mode, ut64 loadaddr) {
|
2014-03-25 17:11:11 +01:00
|
|
|
const char *loadmethod = r_config_get (core->config, "file.loadmethod");
|
2014-01-31 02:02:51 +01:00
|
|
|
const char *suppress_warning = r_config_get (core->config, "file.nowarn");
|
2014-03-25 17:11:11 +01:00
|
|
|
ut64 load_align = r_config_get_i (core->config, "file.loadalign");
|
2016-05-24 21:22:15 +01:00
|
|
|
if (!loadmethod || !suppress_warning) return NULL;
|
2014-03-25 17:11:11 +01:00
|
|
|
RIOMap *map = NULL;
|
|
|
|
if (!strcmp (loadmethod, "overwrite"))
|
2014-09-28 00:05:20 +02:00
|
|
|
map = r_io_map_new (core->io, fh->desc->fd, mode, 0, loadaddr, r_io_desc_size (core->io, fh->desc));
|
2014-03-25 17:11:11 +01:00
|
|
|
if (!strcmp (loadmethod, "fail"))
|
2014-09-28 00:05:20 +02:00
|
|
|
map = r_io_map_add (core->io, fh->desc->fd, mode, 0, loadaddr, r_io_desc_size (core->io, fh->desc));
|
2014-10-07 01:58:42 +02:00
|
|
|
if (!strcmp (loadmethod, "append") && load_align) {
|
2014-09-28 00:05:20 +02:00
|
|
|
map = r_io_map_add_next_available (core->io, fh->desc->fd, mode, 0, loadaddr, r_io_desc_size (core->io, fh->desc), load_align);
|
2014-10-07 01:58:42 +02:00
|
|
|
}
|
2014-03-25 17:11:11 +01:00
|
|
|
if (!strcmp (suppress_warning, "false")) {
|
2016-05-25 23:33:47 +02:00
|
|
|
if (!map) {
|
2014-03-25 17:11:11 +01:00
|
|
|
eprintf ("r_core_file_get_next_map: Unable to load specified file to 0x%08"PFMT64x"\n", loadaddr);
|
2016-05-25 23:33:47 +02:00
|
|
|
} else {
|
2014-03-25 17:11:11 +01:00
|
|
|
if (map->from != loadaddr)
|
|
|
|
eprintf ("r_core_file_get_next_map: Unable to load specified file to 0x%08"PFMT64x",\n"
|
|
|
|
"but loaded to 0x%08"PFMT64x"\n", loadaddr, map->from);
|
2014-01-25 18:06:17 -06:00
|
|
|
}
|
|
|
|
}
|
2016-05-25 23:33:47 +02:00
|
|
|
r_io_sort_maps (core->io); //necessary ???
|
2014-01-25 18:06:17 -06:00
|
|
|
return map;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2014-10-08 00:44:31 +02:00
|
|
|
R_API RCoreFile *r_core_file_open_many(RCore *r, const char *file, int flags, ut64 loadaddr) {
|
2016-05-25 23:33:47 +02:00
|
|
|
int openmany = r_config_get_i (r->config, "file.openmany"), opened_count = 0;
|
|
|
|
const char *suppress_warning = r_config_get (r->config, "file.nowarn");
|
|
|
|
ut64 current_loadaddr = loadaddr;
|
|
|
|
RCoreFile *fh, *top_file = NULL;
|
|
|
|
RListIter *fd_iter, *iter2;
|
|
|
|
char *loadmethod = NULL;
|
2014-10-17 00:04:52 +02:00
|
|
|
RList *list_fds = NULL;
|
|
|
|
const char *cp = NULL;
|
2016-05-25 23:33:47 +02:00
|
|
|
RIODesc *fd;
|
2014-04-29 11:10:35 -05:00
|
|
|
|
2014-10-08 00:44:31 +02:00
|
|
|
list_fds = r_io_open_many (r->io, file, flags, 0644);
|
2014-01-25 18:06:17 -06:00
|
|
|
|
2014-01-26 01:29:17 -06:00
|
|
|
if (!list_fds || r_list_length (list_fds) == 0 ) {
|
|
|
|
r_list_free (list_fds);
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
2014-01-25 18:06:17 -06:00
|
|
|
cp = r_config_get (r->config, "file.loadmethod");
|
|
|
|
if (cp) loadmethod = strdup (cp);
|
|
|
|
r_config_set (r->config, "file.loadmethod", "append");
|
2014-01-23 21:05:35 -06:00
|
|
|
|
2014-04-29 11:10:35 -05:00
|
|
|
r_list_foreach_safe (list_fds, fd_iter, iter2, fd) {
|
|
|
|
opened_count++;
|
|
|
|
if (opened_count > openmany) {
|
|
|
|
// XXX - Open Many should limit the number of files
|
|
|
|
// loaded in io plugin area this needs to be more premptive
|
|
|
|
// like down in the io plugin layer.
|
|
|
|
// start closing down descriptors
|
|
|
|
r_list_delete (list_fds, fd_iter);
|
|
|
|
continue;
|
|
|
|
}
|
2014-01-23 21:05:35 -06:00
|
|
|
fh = R_NEW0 (RCoreFile);
|
2014-01-27 08:24:44 -06:00
|
|
|
if (!fh) {
|
|
|
|
eprintf ("file.c:r_core_many failed to allocate new RCoreFile.\n");
|
|
|
|
break;
|
|
|
|
}
|
2014-04-22 23:02:46 -05:00
|
|
|
fh->alive = 1;
|
|
|
|
fh->core = r;
|
2014-05-28 04:34:12 +02:00
|
|
|
fh->desc = fd;
|
2014-01-23 21:05:35 -06:00
|
|
|
r->file = fh;
|
|
|
|
r->io->plugin = fd->plugin;
|
|
|
|
// XXX - load addr should be at a set offset
|
2014-10-08 00:44:31 +02:00
|
|
|
fh->map = r_core_file_get_next_map (r, fh, flags, current_loadaddr);
|
2014-01-25 18:06:17 -06:00
|
|
|
|
|
|
|
if (!fh->map) {
|
2014-11-03 13:36:58 +01:00
|
|
|
r_core_file_free (fh);
|
2016-05-25 23:33:47 +02:00
|
|
|
if (!strcmp (suppress_warning, "false")) {
|
|
|
|
eprintf ("Unable to load file due to failed mapping.\n");
|
|
|
|
}
|
2014-01-25 18:06:17 -06:00
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
current_loadaddr = fh->map->to;
|
|
|
|
if (!top_file) {
|
|
|
|
top_file = fh;
|
|
|
|
// check load addr to make sure its still valid
|
2014-01-27 08:24:44 -06:00
|
|
|
loadaddr = top_file->map->from;
|
2014-01-25 18:06:17 -06:00
|
|
|
}
|
2014-05-07 14:04:04 -05:00
|
|
|
r_bin_bind (r->bin, &(fh->binb));
|
2014-01-23 21:05:35 -06:00
|
|
|
r_list_append (r->files, fh);
|
2014-09-09 17:01:04 +02:00
|
|
|
r_core_bin_load (r, fh->desc->name, fh->map->from);
|
2014-01-23 21:05:35 -06:00
|
|
|
}
|
2014-01-27 08:24:44 -06:00
|
|
|
if (!top_file) {
|
|
|
|
free (loadmethod);
|
|
|
|
return top_file;
|
|
|
|
}
|
2014-01-23 21:05:35 -06:00
|
|
|
cp = r_config_get (r->config, "cmd.open");
|
2016-05-25 23:33:47 +02:00
|
|
|
if (cp && *cp) {
|
|
|
|
r_core_cmd (r, cp, 0);
|
|
|
|
}
|
2014-01-23 21:05:35 -06:00
|
|
|
|
2015-08-27 16:05:22 -04:00
|
|
|
r_config_set (r->config, "file.path", r_file_abspath (top_file->desc->name));
|
2014-09-28 00:05:20 +02:00
|
|
|
r_config_set_i (r->config, "zoom.to", top_file->map->from + r_io_desc_size (r->io, top_file->desc));
|
2014-01-25 18:06:17 -06:00
|
|
|
if (loadmethod) r_config_set (r->config, "file.loadmethod", loadmethod);
|
|
|
|
free (loadmethod);
|
2014-01-23 21:05:35 -06:00
|
|
|
|
|
|
|
return top_file;
|
|
|
|
}
|
|
|
|
|
2016-09-26 00:45:28 +02:00
|
|
|
/* loadaddr is r2 -m (mapaddr) */
|
2016-05-25 23:33:47 +02:00
|
|
|
R_API RCoreFile *r_core_file_open(RCore *r, const char *file, int flags, ut64 loadaddr) {
|
2016-03-18 13:39:45 +01:00
|
|
|
ut64 prev = r_sys_now();
|
2014-05-21 23:03:09 +02:00
|
|
|
const char *suppress_warning = r_config_get (r->config, "file.nowarn");
|
|
|
|
const int openmany = r_config_get_i (r->config, "file.openmany");
|
2010-02-01 11:55:56 +01:00
|
|
|
const char *cp;
|
2016-03-18 13:39:45 +01:00
|
|
|
RCoreFile *fh = NULL;
|
2012-02-07 00:44:46 +01:00
|
|
|
RIODesc *fd;
|
2014-01-25 18:06:17 -06:00
|
|
|
|
2016-03-14 22:12:54 +02:00
|
|
|
if (!file || !*file) {
|
2016-03-18 13:39:45 +01:00
|
|
|
goto beach;
|
2016-03-14 22:12:54 +02:00
|
|
|
}
|
2012-08-09 12:42:44 +02:00
|
|
|
if (!strcmp (file, "-")) {
|
2012-02-07 00:44:46 +01:00
|
|
|
file = "malloc://512";
|
2016-05-25 23:33:47 +02:00
|
|
|
flags = 4 | 2;
|
2012-08-09 12:42:44 +02:00
|
|
|
}
|
2012-05-30 01:35:41 +02:00
|
|
|
r->io->bits = r->assembler->bits; // TODO: we need an api for this
|
2014-10-08 00:44:31 +02:00
|
|
|
fd = r_io_open_nomap (r->io, file, flags, 0644);
|
2016-09-19 13:44:47 +01:00
|
|
|
if (!fd && openmany > 2) {
|
2014-01-23 21:05:35 -06:00
|
|
|
// XXX - make this an actual option somewhere?
|
2014-10-08 00:44:31 +02:00
|
|
|
fh = r_core_file_open_many (r, file, flags, loadaddr);
|
2016-09-26 00:45:28 +02:00
|
|
|
if (fh) {
|
|
|
|
goto beach;
|
|
|
|
}
|
2014-01-23 21:05:35 -06:00
|
|
|
}
|
2016-09-19 13:44:47 +01:00
|
|
|
if (!fd) {
|
2014-10-08 00:44:31 +02:00
|
|
|
if (flags & 2) {
|
2016-03-14 22:12:54 +02:00
|
|
|
if (!r_io_create (r->io, file, 0644, 0)) {
|
2016-03-18 13:39:45 +01:00
|
|
|
goto beach;
|
2016-03-14 22:12:54 +02:00
|
|
|
}
|
|
|
|
if (!(fd = r_io_open_nomap (r->io, file, flags, 0644))) {
|
2016-03-18 13:39:45 +01:00
|
|
|
goto beach;
|
2016-03-14 22:12:54 +02:00
|
|
|
}
|
|
|
|
} else {
|
2016-03-18 13:39:45 +01:00
|
|
|
goto beach;
|
2016-03-14 22:12:54 +02:00
|
|
|
}
|
2012-07-06 02:17:44 +02:00
|
|
|
}
|
2011-02-05 00:20:28 +01:00
|
|
|
if (r_io_is_listener (r->io)) {
|
2016-10-09 23:55:34 +02:00
|
|
|
r_io_desc_detach (r->io, fd);
|
2011-04-17 20:58:32 +02:00
|
|
|
r_core_serve (r, fd);
|
2016-10-09 23:55:34 +02:00
|
|
|
r_io_desc_free (fd);
|
2016-03-18 13:39:45 +01:00
|
|
|
goto beach;
|
2011-02-05 00:20:28 +01:00
|
|
|
}
|
2009-02-05 22:08:46 +01:00
|
|
|
|
2013-12-17 02:10:13 +01:00
|
|
|
fh = R_NEW0 (RCoreFile);
|
2014-01-27 08:24:44 -06:00
|
|
|
if (!fh) {
|
2014-05-28 18:58:53 +02:00
|
|
|
eprintf ("core/file.c: r_core_open failed to allocate RCoreFile.\n");
|
2016-03-18 13:39:45 +01:00
|
|
|
goto beach;
|
2014-01-27 08:24:44 -06:00
|
|
|
}
|
2014-04-22 23:02:46 -05:00
|
|
|
fh->alive = 1;
|
|
|
|
fh->core = r;
|
2014-05-28 04:34:12 +02:00
|
|
|
fh->desc = fd;
|
2009-02-18 01:43:57 +01:00
|
|
|
|
2010-05-20 00:59:42 +02:00
|
|
|
cp = r_config_get (r->config, "cmd.open");
|
2016-05-25 23:33:47 +02:00
|
|
|
if (cp && *cp) {
|
2010-02-01 11:55:56 +01:00
|
|
|
r_core_cmd (r, cp, 0);
|
2016-05-25 23:33:47 +02:00
|
|
|
}
|
2016-04-04 00:17:57 +02:00
|
|
|
{
|
|
|
|
char *absfile = r_file_abspath (file);
|
|
|
|
r_config_set (r->config, "file.path", absfile);
|
|
|
|
free (absfile);
|
|
|
|
}
|
2014-10-08 00:44:31 +02:00
|
|
|
fh->map = r_core_file_get_next_map (r, fh, flags, loadaddr);
|
2014-01-25 18:06:17 -06:00
|
|
|
if (!fh->map) {
|
2014-02-21 10:58:31 +01:00
|
|
|
r_core_file_free (fh);
|
2016-05-18 12:41:12 +03:00
|
|
|
fh = NULL;
|
2016-05-25 23:33:47 +02:00
|
|
|
if (!strcmp (suppress_warning, "false")) {
|
|
|
|
eprintf ("Unable to load file due to failed mapping.\n");
|
|
|
|
}
|
2016-03-18 13:39:45 +01:00
|
|
|
goto beach;
|
2014-01-25 18:06:17 -06:00
|
|
|
}
|
|
|
|
// check load addr to make sure its still valid
|
2014-05-07 14:04:04 -05:00
|
|
|
r_bin_bind (r->bin, &(fh->binb));
|
2014-01-25 18:06:17 -06:00
|
|
|
r_list_append (r->files, fh);
|
2014-04-27 02:06:50 -05:00
|
|
|
r_core_file_set_by_file (r, fh);
|
2014-09-28 00:05:20 +02:00
|
|
|
r_config_set_i (r->config, "zoom.to", fh->map->from + r_io_desc_size (r->io, fh->desc));
|
2016-04-22 10:13:54 +02:00
|
|
|
|
|
|
|
if (r_config_get_i (r->config, "cfg.debug")) {
|
|
|
|
bool swstep = true;
|
2016-05-25 23:33:47 +02:00
|
|
|
if (r->dbg->h && r->dbg->h->canstep) {
|
2016-04-22 10:13:54 +02:00
|
|
|
swstep = false;
|
2016-05-25 23:33:47 +02:00
|
|
|
}
|
2016-04-22 10:13:54 +02:00
|
|
|
r_config_set_i (r->config, "dbg.swstep", swstep);
|
|
|
|
}
|
2016-03-18 13:39:45 +01:00
|
|
|
beach:
|
|
|
|
r->times->file_open_time = r_sys_now() - prev;
|
2009-02-05 22:08:46 +01:00
|
|
|
return fh;
|
|
|
|
}
|
|
|
|
|
2016-01-02 23:25:20 +01:00
|
|
|
R_API int r_core_files_free(const RCore *core, RCoreFile *cf) {
|
2016-10-09 23:55:34 +02:00
|
|
|
if (!core || !core->files || !cf) {
|
|
|
|
return false;
|
|
|
|
}
|
2014-04-22 23:02:46 -05:00
|
|
|
return r_list_delete_data (core->files, cf);
|
|
|
|
}
|
|
|
|
|
2011-02-07 09:46:01 +01:00
|
|
|
R_API void r_core_file_free(RCoreFile *cf) {
|
2014-05-22 13:52:16 +02:00
|
|
|
int res = 1;
|
2016-10-09 23:55:34 +02:00
|
|
|
if (!cf || !cf->core) {
|
2014-11-03 13:36:58 +01:00
|
|
|
return;
|
2016-10-09 23:55:34 +02:00
|
|
|
}
|
2014-11-03 13:36:58 +01:00
|
|
|
if (cf) {
|
2014-05-22 13:52:16 +02:00
|
|
|
res = r_core_files_free (cf->core, cf);
|
2014-11-03 13:36:58 +01:00
|
|
|
}
|
|
|
|
//if (!res && cf && cf->alive) {
|
|
|
|
if (res && cf && cf->alive) {
|
2014-01-18 09:26:09 -06:00
|
|
|
// double free libr/io/io.c:70 performs free
|
2014-11-03 13:36:58 +01:00
|
|
|
RIO *io = NULL;
|
|
|
|
if (cf) {
|
|
|
|
io = (RIO*)(cf->desc ? cf->desc->io : NULL);
|
2014-11-07 10:52:44 +01:00
|
|
|
if (cf->map) {
|
2015-01-22 02:22:29 +01:00
|
|
|
r_io_map_del (io, cf->map->fd);
|
2014-11-07 10:52:44 +01:00
|
|
|
cf->map = NULL;
|
|
|
|
}
|
2014-11-07 10:37:54 +01:00
|
|
|
r_bin_file_deref_by_bind (&cf->binb);
|
2014-11-03 13:36:58 +01:00
|
|
|
r_io_close ((RIO *) io, cf->desc);
|
2014-11-07 10:37:54 +01:00
|
|
|
free (cf);
|
2014-11-03 13:36:58 +01:00
|
|
|
}
|
2013-11-09 17:15:30 -06:00
|
|
|
}
|
|
|
|
cf = NULL;
|
2011-02-07 09:46:01 +01:00
|
|
|
}
|
|
|
|
|
2014-02-21 10:58:31 +01:00
|
|
|
R_API int r_core_file_close(RCore *r, RCoreFile *fh) {
|
2014-11-03 11:47:51 +01:00
|
|
|
int ret;
|
2014-05-28 04:34:12 +02:00
|
|
|
RIODesc *desc = fh && fh->desc? fh->desc : NULL;
|
2014-07-22 00:24:37 +02:00
|
|
|
RCoreFile *prev_cf = r && r->file != fh ? r->file : NULL;
|
2014-09-16 21:58:02 +02:00
|
|
|
|
2015-09-30 13:10:49 +02:00
|
|
|
// TODO: This is not correclty done. because map and iodesc are
|
2014-09-16 21:58:02 +02:00
|
|
|
// still referenced // we need to fully clear all R_IO structs
|
|
|
|
// related to a file as well as the ones needed for RBin.
|
|
|
|
//
|
2014-04-22 23:02:46 -05:00
|
|
|
// XXX -these checks are intended to *try* and catch
|
|
|
|
// stale objects. Unfortunately, if the file handle
|
|
|
|
// (fh) is stale and freed, and there is more than 1
|
|
|
|
// fh in the r->files list, we are hosed. (design flaw)
|
|
|
|
// TODO maybe using sdb to keep track of the allocated and
|
|
|
|
// deallocated files might be a good solutions
|
2014-10-17 00:04:52 +02:00
|
|
|
if (!r || !desc || r_list_empty (r->files))
|
2015-09-14 12:35:38 +02:00
|
|
|
return false;
|
2014-04-22 23:02:46 -05:00
|
|
|
|
2014-04-23 18:04:25 -05:00
|
|
|
if (fh == r->file) r->file = NULL;
|
2014-10-17 00:04:52 +02:00
|
|
|
|
|
|
|
r_core_file_set_by_fd (r, fh->desc->fd);
|
|
|
|
r_core_bin_set_by_fd (r, fh->desc->fd);
|
|
|
|
|
|
|
|
/* delete filedescriptor from io descs here */
|
|
|
|
r_io_desc_del (r->io, fh->desc->fd);
|
|
|
|
|
2014-11-03 11:47:51 +01:00
|
|
|
// AVOID DOUBLE FREE HERE
|
|
|
|
r->files->free = NULL;
|
|
|
|
|
|
|
|
ret = r_list_delete_data (r->files, fh);
|
2014-04-22 23:02:46 -05:00
|
|
|
if (ret) {
|
2014-04-23 18:04:25 -05:00
|
|
|
if (!prev_cf && r_list_length (r->files) > 0)
|
|
|
|
prev_cf = (RCoreFile *) r_list_get_n (r->files, 0);
|
|
|
|
|
|
|
|
if (prev_cf) {
|
2014-05-28 04:34:12 +02:00
|
|
|
RIODesc *desc = prev_cf->desc;
|
2014-04-23 18:04:25 -05:00
|
|
|
if (!desc)
|
|
|
|
eprintf ("Error: RCoreFile's found with out a supporting RIODesc.\n");
|
|
|
|
ret = r_core_file_set_by_file (r, prev_cf);
|
2014-04-22 23:02:46 -05:00
|
|
|
}
|
|
|
|
}
|
2014-10-17 00:04:52 +02:00
|
|
|
#if 0
|
|
|
|
{
|
|
|
|
RListIter *iter;
|
|
|
|
RIODesc *iod;
|
|
|
|
RCoreFile *mcf;
|
|
|
|
r_list_foreach (r->files, iter, mcf) {
|
|
|
|
r_cons_printf ("[cf]--> %p %p %d\n", mcf, mcf->desc, mcf->desc->fd);
|
|
|
|
}
|
|
|
|
r_list_foreach (r->io->files, iter, iod) {
|
|
|
|
r_cons_printf ("[io]--> %p %d\n", iod, iod->fd);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
#endif
|
2009-02-05 22:08:46 +01:00
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
2014-04-27 02:06:50 -05:00
|
|
|
R_API RCoreFile *r_core_file_get_by_fd(RCore *core, int fd) {
|
2011-02-07 09:46:01 +01:00
|
|
|
RCoreFile *file;
|
|
|
|
RListIter *iter;
|
|
|
|
r_list_foreach (core->files, iter, file) {
|
2016-10-09 23:55:34 +02:00
|
|
|
if (file->desc->fd == fd) {
|
2010-02-01 11:55:56 +01:00
|
|
|
return file;
|
2016-10-09 23:55:34 +02:00
|
|
|
}
|
2009-02-05 22:08:46 +01:00
|
|
|
}
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
2014-12-19 13:46:04 +01:00
|
|
|
R_API int r_core_file_list(RCore *core, int mode) {
|
|
|
|
int overlapped, count = 0;
|
2011-02-07 09:46:01 +01:00
|
|
|
RCoreFile *f;
|
2014-12-19 13:46:04 +01:00
|
|
|
ut64 from;
|
2011-02-07 09:46:01 +01:00
|
|
|
RListIter *iter;
|
2016-11-02 03:27:37 +01:00
|
|
|
if (mode == 'j') {
|
2014-12-19 13:46:04 +01:00
|
|
|
r_cons_printf ("[");
|
2016-11-02 03:27:37 +01:00
|
|
|
}
|
2011-02-07 09:46:01 +01:00
|
|
|
r_list_foreach (core->files, iter, f) {
|
2014-10-07 03:07:45 +02:00
|
|
|
if (f->map) {
|
2014-12-19 13:46:04 +01:00
|
|
|
from = f->map->from;
|
|
|
|
overlapped = r_io_map_overlaps (core->io, f->desc, f->map);
|
|
|
|
} else {
|
|
|
|
from = 0LL;
|
2015-09-14 12:35:38 +02:00
|
|
|
overlapped = false;
|
2014-12-19 13:46:04 +01:00
|
|
|
}
|
|
|
|
switch (mode) {
|
|
|
|
case 'j':
|
|
|
|
r_cons_printf ("{\"raised\":%s,\"fd\":%d,\"uri\":\"%s\",\"from\":%"
|
|
|
|
PFMT64d",\"writable\":%s,\"size\":%d,\"overlaps\":%s}%s",
|
|
|
|
core->io->raised == f->desc->fd?"true":"false",
|
2015-01-24 23:26:17 +01:00
|
|
|
(int)f->desc->fd, f->desc->uri, (ut64)from,
|
2014-12-19 13:46:04 +01:00
|
|
|
f->desc->flags & R_IO_WRITE? "true": "false",
|
2015-01-24 23:26:17 +01:00
|
|
|
(int)r_io_desc_size (core->io, f->desc),
|
2014-12-19 13:46:04 +01:00
|
|
|
overlapped?"true":"false",
|
|
|
|
iter->n? ",":"");
|
|
|
|
break;
|
|
|
|
case '*':
|
|
|
|
case 'r':
|
2015-01-24 23:26:17 +01:00
|
|
|
r_cons_printf ("o %s 0x%"PFMT64x"\n", f->desc->uri, (ut64)from);
|
2014-12-19 13:46:04 +01:00
|
|
|
break;
|
|
|
|
default:
|
2016-10-09 22:03:02 +02:00
|
|
|
{
|
|
|
|
ut64 sz = r_io_desc_size (core->io, f->desc);
|
|
|
|
const char *fmt;
|
|
|
|
if (sz == UT64_MAX) {
|
2016-11-02 03:27:37 +01:00
|
|
|
fmt = "%c %d %d %s @ 0x%"PFMT64x" ; %s size=%"PFMT64d" %s\n";
|
2016-10-09 22:03:02 +02:00
|
|
|
} else {
|
2016-11-02 03:27:37 +01:00
|
|
|
fmt = "%c %d %d %s @ 0x%"PFMT64x" ; %s size=%"PFMT64u" %s\n";
|
2016-10-09 22:03:02 +02:00
|
|
|
}
|
|
|
|
r_cons_printf (fmt,
|
2014-12-19 13:46:04 +01:00
|
|
|
core->io->raised == f->desc->fd?'*':'-',
|
2016-11-02 03:27:37 +01:00
|
|
|
count,
|
2015-01-24 23:26:17 +01:00
|
|
|
(int)f->desc->fd, f->desc->uri, (ut64)from,
|
2014-12-19 13:46:04 +01:00
|
|
|
f->desc->flags & R_IO_WRITE? "rw": "r",
|
2016-07-05 16:16:28 +05:30
|
|
|
r_io_desc_size (core->io, f->desc),
|
2014-12-19 13:46:04 +01:00
|
|
|
overlapped?"overlaps":"");
|
2016-10-09 22:03:02 +02:00
|
|
|
}
|
2014-12-19 13:46:04 +01:00
|
|
|
break;
|
|
|
|
}
|
2010-02-01 11:55:56 +01:00
|
|
|
count++;
|
|
|
|
}
|
2016-10-09 23:55:34 +02:00
|
|
|
if (mode=='j') {
|
2014-12-19 13:46:04 +01:00
|
|
|
r_cons_printf ("]\n");
|
2016-10-09 23:55:34 +02:00
|
|
|
}
|
2010-02-01 11:55:56 +01:00
|
|
|
return count;
|
|
|
|
}
|
|
|
|
|
2014-05-08 18:35:04 -05:00
|
|
|
// XXX - needs to account for binfile index and bin object index
|
2014-04-27 02:06:50 -05:00
|
|
|
R_API int r_core_file_bin_raise (RCore *core, ut32 binfile_idx) {
|
|
|
|
RBin *bin = core->bin;
|
|
|
|
int v = binfile_idx > 1 ? binfile_idx : 1;
|
|
|
|
RBinFile *bf = r_list_get_n (bin->binfiles, v);
|
2015-09-14 12:35:38 +02:00
|
|
|
int res = false;
|
2014-04-27 02:06:50 -05:00
|
|
|
if (bf) {
|
|
|
|
res = r_bin_file_set_cur_binfile (bin, bf);
|
|
|
|
if (res) r_io_raise (core->io, bf->fd);
|
|
|
|
res = res ? r_core_file_set_by_fd (core, bf->fd) : res;
|
|
|
|
if (res) core->switch_file_view = 1;
|
|
|
|
}
|
|
|
|
return res;
|
|
|
|
}
|
|
|
|
|
|
|
|
R_API int r_core_file_binlist(RCore *core) {
|
|
|
|
int count = 0;
|
|
|
|
RListIter *iter;
|
2014-05-08 18:35:04 -05:00
|
|
|
RCoreFile *cur_cf = core->file, *cf = NULL;
|
2014-05-16 04:07:03 +02:00
|
|
|
RBinFile *binfile = NULL;
|
2014-04-27 02:06:50 -05:00
|
|
|
RBin *bin = core->bin;
|
|
|
|
const RList *binfiles = bin ? bin->binfiles: NULL;
|
|
|
|
|
2016-10-09 23:55:34 +02:00
|
|
|
if (!binfiles) {
|
|
|
|
return false;
|
|
|
|
}
|
2014-04-27 02:06:50 -05:00
|
|
|
r_list_foreach (binfiles, iter, binfile) {
|
2014-05-28 04:34:12 +02:00
|
|
|
int fd = binfile->fd;
|
2014-05-08 18:35:04 -05:00
|
|
|
cf = r_core_file_get_by_fd (core, fd);
|
|
|
|
if (cf && cf->map) {
|
2014-04-27 02:06:50 -05:00
|
|
|
r_cons_printf ("%c %d %s @ 0x%"PFMT64x" ; %s\n",
|
2014-05-28 04:34:12 +02:00
|
|
|
core->io->raised == cf->desc->fd?'*':'-',
|
2014-09-25 18:04:07 +02:00
|
|
|
fd, cf->desc->uri, cf->map->from,
|
2014-05-28 04:34:12 +02:00
|
|
|
cf->desc->flags & R_IO_WRITE? "rw": "r");
|
2014-04-27 02:06:50 -05:00
|
|
|
}
|
|
|
|
}
|
|
|
|
r_core_file_set_by_file (core, cur_cf);
|
2014-05-04 10:03:15 -05:00
|
|
|
//r_core_bin_bind (core, cur_bf);
|
2014-04-27 02:06:50 -05:00
|
|
|
return count;
|
|
|
|
}
|
|
|
|
|
2011-02-07 09:46:01 +01:00
|
|
|
R_API int r_core_file_close_fd(RCore *core, int fd) {
|
|
|
|
RCoreFile *file;
|
|
|
|
RListIter *iter;
|
|
|
|
r_list_foreach (core->files, iter, file) {
|
2015-01-22 02:22:29 +01:00
|
|
|
if (file->desc->fd == fd || fd == -1) {
|
2014-04-22 23:02:46 -05:00
|
|
|
r_core_file_close (core, file);
|
2014-11-07 10:37:54 +01:00
|
|
|
if (file == core->file) {
|
|
|
|
core->file = NULL; // deref
|
|
|
|
}
|
2014-01-21 15:40:10 +01:00
|
|
|
#if 0
|
2011-02-07 09:46:01 +01:00
|
|
|
if (r_list_empty (core->files))
|
|
|
|
core->file = NULL;
|
2014-01-21 15:40:10 +01:00
|
|
|
#endif
|
2015-09-14 12:35:38 +02:00
|
|
|
return true;
|
2011-02-07 09:46:01 +01:00
|
|
|
}
|
|
|
|
}
|
2015-09-14 12:35:38 +02:00
|
|
|
return false;
|
2009-02-05 22:08:46 +01:00
|
|
|
}
|
2011-07-13 17:41:26 +02:00
|
|
|
|
|
|
|
R_API int r_core_hash_load(RCore *r, const char *file) {
|
|
|
|
const ut8 *md5, *sha1;
|
|
|
|
char hash[128], *p;
|
2014-09-28 01:37:56 +02:00
|
|
|
int i;
|
2014-09-30 12:06:37 +02:00
|
|
|
int buf_len = 0;
|
2013-02-07 09:41:05 +01:00
|
|
|
ut8 *buf = NULL;
|
2011-07-13 17:41:26 +02:00
|
|
|
RHash *ctx;
|
|
|
|
ut64 limit;
|
2014-04-29 11:10:35 -05:00
|
|
|
RCoreFile *cf = r_core_file_cur (r);
|
2015-08-25 12:31:04 +02:00
|
|
|
if (!file && cf && cf->desc) {
|
|
|
|
file = cf->desc->name;
|
|
|
|
}
|
|
|
|
if (!file) {
|
2015-09-14 12:35:38 +02:00
|
|
|
return false;
|
2015-08-25 12:31:04 +02:00
|
|
|
}
|
2011-07-13 17:41:26 +02:00
|
|
|
|
|
|
|
limit = r_config_get_i (r->config, "cfg.hashlimit");
|
2017-01-14 21:45:13 +01:00
|
|
|
if (cf && r_io_desc_size (r->io, cf->desc) > limit) {
|
2015-09-14 12:35:38 +02:00
|
|
|
return false;
|
2017-01-14 21:45:13 +01:00
|
|
|
}
|
2011-11-14 00:21:25 +01:00
|
|
|
buf = (ut8*)r_file_slurp (file, &buf_len);
|
2017-01-14 21:45:13 +01:00
|
|
|
if (!buf) {
|
2015-09-14 12:35:38 +02:00
|
|
|
return false;
|
2017-01-14 21:45:13 +01:00
|
|
|
}
|
2015-09-14 12:35:38 +02:00
|
|
|
ctx = r_hash_new (true, R_HASH_MD5);
|
2011-07-13 17:41:26 +02:00
|
|
|
md5 = r_hash_do_md5 (ctx, buf, buf_len);
|
|
|
|
p = hash;
|
2016-11-02 03:27:37 +01:00
|
|
|
for (i = 0; i < R_HASH_SIZE_MD5; i++) {
|
2011-07-13 17:41:26 +02:00
|
|
|
sprintf (p, "%02x", md5[i]);
|
2011-11-11 17:14:09 +01:00
|
|
|
p += 2;
|
2011-07-13 17:41:26 +02:00
|
|
|
}
|
2011-11-11 17:14:09 +01:00
|
|
|
*p = 0;
|
2011-07-13 17:41:26 +02:00
|
|
|
r_config_set (r->config, "file.md5", hash);
|
|
|
|
r_hash_free (ctx);
|
2015-09-14 12:35:38 +02:00
|
|
|
ctx = r_hash_new (true, R_HASH_SHA1);
|
2011-07-13 17:41:26 +02:00
|
|
|
sha1 = r_hash_do_sha1 (ctx, buf, buf_len);
|
|
|
|
p = hash;
|
2016-11-02 03:27:37 +01:00
|
|
|
for (i = 0; i < R_HASH_SIZE_SHA1; i++) {
|
2011-07-13 17:41:26 +02:00
|
|
|
sprintf (p, "%02x", sha1[i]);
|
2012-08-04 23:48:06 +02:00
|
|
|
p += 2;
|
2011-07-13 17:41:26 +02:00
|
|
|
}
|
2012-08-04 23:48:06 +02:00
|
|
|
*p = 0;
|
2011-07-13 17:41:26 +02:00
|
|
|
r_config_set (r->config, "file.sha1", hash);
|
|
|
|
r_hash_free (ctx);
|
2011-11-11 17:14:09 +01:00
|
|
|
free (buf);
|
2015-09-14 12:35:38 +02:00
|
|
|
return true;
|
2011-07-13 17:41:26 +02:00
|
|
|
}
|
2014-04-23 18:04:25 -05:00
|
|
|
|
|
|
|
R_API RCoreFile * r_core_file_find_by_fd (RCore *core, ut64 fd) {
|
|
|
|
RListIter *iter;
|
|
|
|
RCoreFile *cf = NULL;
|
|
|
|
r_list_foreach (core->files, iter, cf) {
|
2016-11-02 03:27:37 +01:00
|
|
|
if (cf && cf->desc && cf->desc->fd == fd) {
|
|
|
|
break;
|
|
|
|
}
|
2014-04-23 18:04:25 -05:00
|
|
|
cf = NULL;
|
|
|
|
}
|
|
|
|
return cf;
|
|
|
|
}
|
|
|
|
|
|
|
|
R_API RCoreFile * r_core_file_find_by_name (RCore * core, const char * name) {
|
|
|
|
RListIter *iter;
|
|
|
|
RCoreFile *cf = NULL;
|
|
|
|
|
|
|
|
r_list_foreach (core->files, iter, cf) {
|
2016-11-02 03:27:37 +01:00
|
|
|
if (cf && cf->desc && !strcmp (cf->desc->name, name)) {
|
|
|
|
break;
|
|
|
|
}
|
2014-04-23 18:04:25 -05:00
|
|
|
cf = NULL;
|
|
|
|
}
|
|
|
|
return cf;
|
|
|
|
}
|
|
|
|
|
|
|
|
R_API int r_core_file_set_by_fd (RCore * core, ut64 fd) {
|
|
|
|
RCoreFile *cf = r_core_file_find_by_fd (core, fd);
|
|
|
|
return r_core_file_set_by_file (core, cf);
|
|
|
|
}
|
|
|
|
|
|
|
|
R_API int r_core_file_set_by_name (RCore * core, const char * name) {
|
|
|
|
RCoreFile *cf = r_core_file_find_by_name (core, name);
|
|
|
|
return r_core_file_set_by_file (core, cf);
|
|
|
|
}
|
|
|
|
|
|
|
|
R_API int r_core_file_set_by_file (RCore * core, RCoreFile *cf) {
|
|
|
|
if (cf) {
|
2014-05-28 04:34:12 +02:00
|
|
|
RIODesc *desc = cf->desc;
|
2014-10-17 00:04:52 +02:00
|
|
|
core->offset = cf && cf->map ? cf->map->from : 0LL;
|
2014-04-23 18:04:25 -05:00
|
|
|
core->file = cf;
|
|
|
|
if (desc) {
|
2014-09-09 16:04:57 +02:00
|
|
|
r_io_use_desc (core->io, desc);
|
2014-04-23 18:04:25 -05:00
|
|
|
r_core_bin_set_by_fd (core, desc->fd);
|
|
|
|
}
|
2015-09-14 12:35:38 +02:00
|
|
|
return true;
|
2014-04-23 18:04:25 -05:00
|
|
|
}
|
2015-09-14 12:35:38 +02:00
|
|
|
return false;
|
2014-04-23 18:04:25 -05:00
|
|
|
}
|
2014-04-27 02:06:50 -05:00
|
|
|
|
|
|
|
R_API ut32 r_core_file_cur_fd (RCore *core) {
|
2014-05-28 04:34:12 +02:00
|
|
|
RIODesc *desc = core->file ? core->file->desc : NULL;
|
2014-05-08 18:35:04 -05:00
|
|
|
if (desc) {
|
2014-04-27 02:06:50 -05:00
|
|
|
return desc->fd;
|
2014-05-08 18:35:04 -05:00
|
|
|
}
|
2016-11-02 03:27:37 +01:00
|
|
|
return UT32_MAX;
|
2014-04-29 11:10:35 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
R_API RCoreFile * r_core_file_cur (RCore *r) {
|
|
|
|
// Add any locks here
|
|
|
|
return r->file;
|
2014-05-16 04:07:03 +02:00
|
|
|
}
|