mirror of
https://github.com/radareorg/radare2.git
synced 2025-01-09 06:50:49 +00:00
392 lines
9.3 KiB
C
392 lines
9.3 KiB
C
/* radare - LGPL - Copyright 2009-2018 // pancake */
|
|
|
|
#define ms_argc (sizeof (ms_argv)/sizeof(const char*))
|
|
static const char *ms_argv[] = {
|
|
"?", "!", "ls", "cd", "cat", "get", "mount", "help", "q", "exit", NULL
|
|
};
|
|
|
|
static const char *help_msg_m[] = {
|
|
"Usage:", "m[-?*dgy] [...] ", "Mountpoints management",
|
|
"m", "", "List all mountpoints in human readable format",
|
|
"m*", "", "Same as above, but in r2 commands",
|
|
"ml", "", "List filesystem plugins",
|
|
"m", " /mnt", "Mount fs at /mnt with autodetect fs and current offset",
|
|
"m", " /mnt ext2 0", "Mount ext2 fs at /mnt with delta 0 on IO",
|
|
"m-/", "", "Umount given path (/)",
|
|
"md", " /", "List directory contents for path",
|
|
"mf", "[?] [o|n]", "Search files for given filename or for offset",
|
|
"mg", " /foo", "Get contents of file/dir dumped to disk (XXX?)",
|
|
"mo", " /foo/bar", "Open given file into a malloc://",
|
|
"mi", " /foo/bar", "Get offset and size of given file",
|
|
"mp", "", "List all supported partition types",
|
|
"mp", " msdos 0", "Show partitions in msdos format at offset 0",
|
|
"ms", " /mnt", "Open filesystem prompt at /mnt",
|
|
"my", "", "Yank contents of file into clipboard",
|
|
//"TODO: support multiple mountpoints and RFile IO's (need io+core refactorn",
|
|
NULL
|
|
};
|
|
|
|
static void cmd_mount_init(RCore *core) {
|
|
DEFINE_CMD_DESCRIPTOR (core, m);
|
|
}
|
|
|
|
static int cmd_mkdir(void *data, const char *input) {
|
|
char *res = r_syscmd_mkdir (input);
|
|
if (res) {
|
|
r_cons_print (res);
|
|
free (res);
|
|
}
|
|
return 0;
|
|
}
|
|
|
|
static int cmd_mv(void *data, const char *input) {
|
|
return r_syscmd_mv (input)? 1: 0;
|
|
}
|
|
|
|
static char *cwd = NULL;
|
|
static char * av[1024] = {NULL};
|
|
#define av_max 1024
|
|
|
|
static char **getFilesFor(RLine *line, const char *path, int *ac) {
|
|
RCore *core = line->user;
|
|
RFS *fs = core->fs;
|
|
RListIter *iter;
|
|
RFSFile *file;
|
|
char *full_path;
|
|
char *lpath = strdup (path);
|
|
|
|
if (!lpath) {
|
|
return NULL;
|
|
}
|
|
|
|
r_str_trim_head (lpath);
|
|
if (lpath[0] != '/') {
|
|
full_path = r_str_newf ("%s/%s", cwd, lpath);
|
|
} else {
|
|
full_path = strdup (lpath);
|
|
}
|
|
free (lpath);
|
|
|
|
//eprintf ("autocompleting for path '%s'\n", full_path);
|
|
|
|
RList *list = r_fs_dir (fs, full_path);
|
|
int count = 0;
|
|
if (list) {
|
|
r_list_foreach (list, iter, file) {
|
|
eprintf ("==> %c %s\n", file->type, file->name);
|
|
if (count >= av_max) {
|
|
break;
|
|
}
|
|
av[count++] = file->name;
|
|
}
|
|
r_list_free (list);
|
|
}
|
|
// autocomplete mountpoints
|
|
// mountpoints if any
|
|
RFSRoot *r;
|
|
char *me = strdup (full_path);
|
|
r_list_foreach (fs->roots, iter, r) {
|
|
char *base = strdup (r->path);
|
|
char *ls = (char *) r_str_lchr (base, '/');
|
|
if (ls) {
|
|
ls++;
|
|
*ls = 0;
|
|
}
|
|
// TODO: adjust contents between //
|
|
if (!strcmp (me, base)) {
|
|
//eprintf ("m %s\n", (r->path && r->path[0]) ? r->path + 1: "");
|
|
if (count >= av_max) {
|
|
break;
|
|
}
|
|
av[count++] = r->path;
|
|
}
|
|
free (base);
|
|
}
|
|
free (me);
|
|
free (full_path);
|
|
av[count] = NULL;
|
|
if (ac) {
|
|
*ac = count;
|
|
}
|
|
av[3] = NULL;
|
|
return av;
|
|
}
|
|
|
|
static int ms_autocomplete(RLine *line) {
|
|
const char *data = line->buffer.data;
|
|
line->completion.argc = ms_argc;
|
|
line->completion.argv = ms_argv;
|
|
if (!strncmp (data, "ls ", 3)
|
|
|| !strncmp (data, "cd ", 3)
|
|
|| !strncmp (data, "cat ", 4)
|
|
|| !strncmp (data, "get ", 4)) {
|
|
const char *file = strchr (data, ' ');
|
|
if (file++) {
|
|
//eprintf ("FILE (%s)\n", file);
|
|
int tmp_argc = 0;
|
|
// TODO: handle abs vs rel
|
|
char **tmp_argv = getFilesFor (line, file, &tmp_argc);
|
|
line->completion.argc = tmp_argc;
|
|
line->completion.argv = (const char **)tmp_argv;
|
|
}
|
|
return true;
|
|
}
|
|
return false;
|
|
}
|
|
|
|
static int cmd_mount(void *data, const char *_input) {
|
|
ut64 off = 0;
|
|
char *input, *oinput, *ptr, *ptr2;
|
|
RList *list;
|
|
RListIter *iter;
|
|
RFSFile *file;
|
|
RFSRoot *root;
|
|
RFSPlugin *plug;
|
|
RFSPartition *part;
|
|
RCore *core = (RCore *)data;
|
|
|
|
if (!strncmp ("kdir", _input, 4)) {
|
|
return cmd_mkdir (data, _input);
|
|
}
|
|
if (!strncmp ("v", _input, 1)) {
|
|
return cmd_mv (data, _input);
|
|
}
|
|
input = oinput = strdup (_input);
|
|
|
|
switch (*input) {
|
|
case ' ':
|
|
input++;
|
|
if (input[0]==' ') {
|
|
input++;
|
|
}
|
|
ptr = strchr (input, ' ');
|
|
if (ptr) {
|
|
*ptr = 0;
|
|
ptr++;
|
|
ptr2 = strchr (ptr, ' ');
|
|
if (ptr2) {
|
|
*ptr2 = 0;
|
|
off = r_num_math (core->num, ptr2+1);
|
|
}
|
|
if (!r_fs_mount (core->fs, ptr, input, off)) {
|
|
eprintf ("Cannot mount %s\n", input);
|
|
}
|
|
} else {
|
|
if (!(ptr = r_fs_name (core->fs, core->offset))) {
|
|
eprintf ("Unknown filesystem type\n");
|
|
}
|
|
if (!r_fs_mount (core->fs, ptr, input, core->offset)) {
|
|
eprintf ("Cannot mount %s\n", input);
|
|
}
|
|
free (ptr);
|
|
}
|
|
break;
|
|
case '-':
|
|
r_fs_umount (core->fs, input+1);
|
|
break;
|
|
case '*':
|
|
eprintf ("List commands in radare format\n");
|
|
r_list_foreach (core->fs->roots, iter, root) {
|
|
r_cons_printf ("m %s %s 0x%"PFMT64x"\n",
|
|
root-> path, root->p->name, root->delta);
|
|
}
|
|
break;
|
|
case '\0':
|
|
r_list_foreach (core->fs->roots, iter, root) {
|
|
r_cons_printf ("%s\t0x%"PFMT64x"\t%s\n",
|
|
root->p->name, root->delta, root->path);
|
|
}
|
|
break;
|
|
case 'L': // "ml" list of plugins .. should be mL
|
|
case 'l': // "ml" list of plugins .. should be mL
|
|
r_list_foreach (core->fs->plugins, iter, plug) {
|
|
r_cons_printf ("%10s %s\n", plug->name, plug->desc);
|
|
}
|
|
break;
|
|
case 'd': // "md"
|
|
input++;
|
|
if (input[0] == ' ') {
|
|
input++;
|
|
}
|
|
list = r_fs_dir (core->fs, input);
|
|
if (list) {
|
|
r_list_foreach (list, iter, file) {
|
|
r_cons_printf ("%c %s\n", file->type, file->name);
|
|
}
|
|
r_list_free (list);
|
|
}
|
|
const char *path = *input? input: "/";
|
|
r_list_foreach (core->fs->roots, iter, root) {
|
|
// TODO: adjust contents between //
|
|
if (!strncmp (path, root->path, strlen (path))) {
|
|
char *base = strdup (root->path);
|
|
char *ls = (char *)r_str_lchr (base, '/');
|
|
if (ls) {
|
|
ls++;
|
|
*ls = 0;
|
|
}
|
|
// TODO: adjust contents between //
|
|
if (!strcmp (path, base)) {
|
|
r_cons_printf ("m %s\n", (root->path && root->path[0]) ? root->path + 1: "");
|
|
}
|
|
free (base);
|
|
}
|
|
}
|
|
break;
|
|
case 'p':
|
|
input++;
|
|
if (*input == ' ') {
|
|
input++;
|
|
}
|
|
ptr = strchr (input, ' ');
|
|
if (ptr) {
|
|
*ptr = 0;
|
|
off = r_num_math (core->num, ptr+1);
|
|
}
|
|
list = r_fs_partitions (core->fs, input, off);
|
|
if (list) {
|
|
r_list_foreach (list, iter, part) {
|
|
r_cons_printf ("%d %02x 0x%010"PFMT64x" 0x%010"PFMT64x"\n",
|
|
part->number, part->type,
|
|
part->start, part->start+part->length);
|
|
}
|
|
r_list_free (list);
|
|
} else {
|
|
eprintf ("Cannot read partition\n");
|
|
}
|
|
break;
|
|
case 'o': //"mo"
|
|
input++;
|
|
if (input[0]==' ') {
|
|
input++;
|
|
}
|
|
file = r_fs_open (core->fs, input);
|
|
if (file) {
|
|
r_fs_read (core->fs, file, 0, file->size);
|
|
char *uri = r_str_newf ("malloc://%d", file->size);
|
|
RIODesc *fd = r_io_open (core->io, uri, R_IO_READ | R_IO_WRITE, 0);
|
|
if (fd) {
|
|
r_io_desc_write (fd, file->data, file->size);
|
|
}
|
|
} else {
|
|
eprintf ("Cannot open file\n");
|
|
}
|
|
break;
|
|
case 'i':
|
|
input++;
|
|
if (input[0]==' ') {
|
|
input++;
|
|
}
|
|
file = r_fs_open (core->fs, input);
|
|
if (file) {
|
|
// XXX: dump to file or just pipe?
|
|
r_fs_read (core->fs, file, 0, file->size);
|
|
r_cons_printf ("f file %d 0x%08"PFMT64x"\n", file->size, file->off);
|
|
r_fs_close (core->fs, file);
|
|
} else {
|
|
eprintf ("Cannot open file\n");
|
|
}
|
|
break;
|
|
case 'g': // "mg"
|
|
input++;
|
|
if (*input == ' ') {
|
|
input++;
|
|
}
|
|
ptr = strchr (input, ' ');
|
|
if (ptr) {
|
|
*ptr++ = 0;
|
|
} else {
|
|
ptr = "./";
|
|
}
|
|
file = r_fs_open (core->fs, input);
|
|
if (file) {
|
|
r_fs_read (core->fs, file, 0, file->size);
|
|
r_cons_memcat ((const char *)file->data, file->size);
|
|
r_fs_close (core->fs, file);
|
|
r_cons_memcat ("\n", 1);
|
|
} else if (!r_fs_dir_dump (core->fs, input, ptr)) {
|
|
eprintf ("Cannot open file\n");
|
|
}
|
|
break;
|
|
case 'f':
|
|
input++;
|
|
switch (*input) {
|
|
case '?':
|
|
r_cons_printf (
|
|
"Usage: mf[no] [...]\n"
|
|
" mfn /foo *.c ; search files by name in /foo path\n"
|
|
" mfo /foo 0x5e91 ; search files by offset in /foo path\n"
|
|
);
|
|
break;
|
|
case 'n':
|
|
input++;
|
|
if (*input == ' ')
|
|
input++;
|
|
ptr = strchr (input, ' ');
|
|
if (ptr) {
|
|
*ptr++ = 0;
|
|
list = r_fs_find_name (core->fs, input, ptr);
|
|
r_list_foreach (list, iter, ptr) {
|
|
r_str_trim_path (ptr);
|
|
printf ("%s\n", ptr);
|
|
}
|
|
//XXX: r_list_purge (list);
|
|
} else eprintf ("Unknown store path\n");
|
|
break;
|
|
case 'o':
|
|
input++;
|
|
if (*input == ' ')
|
|
input++;
|
|
ptr = strchr (input, ' ');
|
|
if (ptr) {
|
|
*ptr++ = 0;
|
|
ut64 off = r_num_math (core->num, ptr);
|
|
list = r_fs_find_off (core->fs, input, off);
|
|
r_list_foreach (list, iter, ptr) {
|
|
r_str_trim_path (ptr);
|
|
printf ("%s\n", ptr);
|
|
}
|
|
//XXX: r_list_purge (list);
|
|
} else eprintf ("Unknown store path\n");
|
|
break;
|
|
}
|
|
break;
|
|
case 's':
|
|
if (core->http_up) {
|
|
free (oinput);
|
|
return false;
|
|
}
|
|
input++;
|
|
if (input[0] == ' ') {
|
|
input++;
|
|
}
|
|
r_cons_set_raw (false);
|
|
{
|
|
RFSShell shell = {
|
|
.cwd = &cwd,
|
|
.set_prompt = r_line_set_prompt,
|
|
.readline = r_line_readline,
|
|
.hist_add = r_line_hist_add
|
|
};
|
|
RLine *rli = r_line_singleton ();
|
|
RLineCompletion c;
|
|
memcpy (&c, &rli->completion, sizeof (c));
|
|
rli->completion.run = ms_autocomplete;
|
|
rli->completion.argc = ms_argc;
|
|
rli->completion.argv = ms_argv;
|
|
r_fs_shell_prompt (&shell, core->fs, input);
|
|
free (cwd);
|
|
memcpy (&rli->completion, &c, sizeof (c));
|
|
}
|
|
break;
|
|
case 'y':
|
|
eprintf ("TODO\n");
|
|
break;
|
|
case '?':
|
|
r_core_cmd_help (core, help_msg_m);
|
|
break;
|
|
}
|
|
free (oinput);
|
|
return 0;
|
|
}
|