2011-01-12 00:01:06 +01:00
|
|
|
/* radare - LGPL - Copyright 2011 pancake<nopcode.org> */
|
2011-01-07 18:22:02 +01:00
|
|
|
|
|
|
|
#include <r_fs.h>
|
2011-01-12 00:01:06 +01:00
|
|
|
#include "../config.h"
|
2011-01-07 18:22:02 +01:00
|
|
|
|
2011-01-14 01:02:20 +01:00
|
|
|
static RFSPlugin *fs_static_plugins[] = { R_FS_STATIC_PLUGINS };
|
2011-01-12 00:01:06 +01:00
|
|
|
|
|
|
|
/* lifecycle */
|
2011-01-07 18:22:02 +01:00
|
|
|
// TODO: needs much more love
|
|
|
|
R_API RFS *r_fs_new () {
|
2011-01-12 00:01:06 +01:00
|
|
|
int i;
|
|
|
|
RFSPlugin *static_plugin;
|
2011-01-07 18:22:02 +01:00
|
|
|
RFS *fs = R_NEW (RFS);
|
|
|
|
if (fs) {
|
2011-01-12 00:01:06 +01:00
|
|
|
fs->roots = r_list_new ();
|
|
|
|
fs->roots->free = (RListFree)r_fs_root_free;
|
2011-01-07 18:22:02 +01:00
|
|
|
fs->plugins = r_list_new ();
|
2011-01-12 00:01:06 +01:00
|
|
|
// XXX fs->roots->free = r_fs_plugin_free;
|
|
|
|
for (i=0; fs_static_plugins[i]; i++) {
|
|
|
|
static_plugin = R_NEW (RFSPlugin);
|
|
|
|
memcpy (static_plugin, fs_static_plugins[i], sizeof (RFSPlugin));
|
|
|
|
r_fs_add (fs, static_plugin);
|
|
|
|
}
|
2011-01-07 18:22:02 +01:00
|
|
|
}
|
|
|
|
return fs;
|
|
|
|
}
|
|
|
|
|
2011-01-12 00:01:06 +01:00
|
|
|
R_API RFSPlugin *r_fs_plugin_get (RFS *fs, const char *name) {
|
|
|
|
RListIter *iter;
|
|
|
|
RFSPlugin *p;
|
|
|
|
r_list_foreach (fs->plugins, iter, p) {
|
|
|
|
if (!strcmp (p->name, name))
|
|
|
|
return p;
|
|
|
|
}
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
2011-01-07 18:22:02 +01:00
|
|
|
R_API void r_fs_free (RFS* fs) {
|
2011-01-12 00:01:06 +01:00
|
|
|
r_list_free (fs->plugins);
|
|
|
|
r_list_free (fs->roots);
|
2011-01-07 18:22:02 +01:00
|
|
|
free (fs);
|
|
|
|
}
|
|
|
|
|
2011-01-12 00:01:06 +01:00
|
|
|
/* plugins */
|
|
|
|
R_API void r_fs_add (RFS *fs, RFSPlugin *p) {
|
2011-02-23 00:54:40 +01:00
|
|
|
// TODO: find coliding plugin name
|
|
|
|
if (p && p->init)
|
|
|
|
p->init ();
|
2011-01-12 00:01:06 +01:00
|
|
|
r_list_append (fs->plugins, p);
|
|
|
|
}
|
|
|
|
|
|
|
|
R_API void r_fs_del (RFS *fs, RFSPlugin *p) {
|
|
|
|
// TODO: implement
|
|
|
|
}
|
|
|
|
|
|
|
|
/* mountpoint */
|
|
|
|
|
2011-01-14 01:02:20 +01:00
|
|
|
R_API RFSRoot *r_fs_mount (RFS* fs, const char *fstype, const char *path, ut64 delta) {
|
2011-01-12 00:01:06 +01:00
|
|
|
RFSPlugin *p;
|
2011-01-14 01:02:20 +01:00
|
|
|
RFSRoot *root;
|
2011-02-18 01:43:31 +01:00
|
|
|
|
2011-01-12 00:01:06 +01:00
|
|
|
if (path[0] != '/') {
|
|
|
|
eprintf ("r_fs_mount: invalid mountpoint\n");
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
p = r_fs_plugin_get (fs, fstype);
|
|
|
|
if (p != NULL) {
|
2011-01-14 01:02:20 +01:00
|
|
|
root = r_fs_root_new (path, delta);
|
|
|
|
root->p = p;
|
|
|
|
//memcpy (&root->iob, &fs->iob, sizeof (root->iob));
|
|
|
|
root->iob = fs->iob;
|
|
|
|
p->mount (root);
|
2011-01-12 00:01:06 +01:00
|
|
|
r_list_append (fs->roots, root);
|
|
|
|
eprintf ("Mounted %s on %s at 0x%llx\n", fstype, path, 0LL);
|
|
|
|
} else eprintf ("r_fs_mount: Invalid filesystem type\n");
|
|
|
|
return root;
|
|
|
|
}
|
|
|
|
|
|
|
|
static inline int r_fs_match (const char *root, const char *path) {
|
|
|
|
return (!strncmp (path, root, strlen (path)));
|
|
|
|
}
|
|
|
|
|
|
|
|
R_API int r_fs_umount (RFS* fs, const char *path) {
|
|
|
|
RFSRoot *root;
|
|
|
|
RListIter *iter;
|
|
|
|
r_list_foreach (fs->roots, iter, root) {
|
|
|
|
if (r_fs_match (path, root->path)) {
|
|
|
|
r_list_delete (fs->roots, iter);
|
|
|
|
return R_TRUE;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return R_FALSE;
|
|
|
|
}
|
|
|
|
|
|
|
|
R_API RFSRoot *r_fs_root (RFS *fs, const char *path) {
|
|
|
|
RFSRoot *root;
|
|
|
|
RListIter *iter;
|
|
|
|
r_list_foreach (fs->roots, iter, root) {
|
|
|
|
if (r_fs_match (path, root->path))
|
|
|
|
return root;
|
|
|
|
}
|
2011-01-07 18:22:02 +01:00
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
2011-01-12 00:01:06 +01:00
|
|
|
/* filez */
|
2011-02-21 09:26:32 +01:00
|
|
|
R_API RFSFile *r_fs_open (RFS* fs, const char *p) {
|
2011-02-23 00:54:40 +01:00
|
|
|
RFSRoot *root;
|
2011-02-21 09:26:32 +01:00
|
|
|
char *path = strdup (p);
|
2011-02-18 01:43:31 +01:00
|
|
|
r_str_chop_path (path);
|
2011-02-23 00:54:40 +01:00
|
|
|
root = r_fs_root (fs, path);
|
2011-02-21 09:26:32 +01:00
|
|
|
if (root && root->p && root->p->open) {
|
|
|
|
RFSFile *f = root->p->open (root, path+strlen (root->path));
|
|
|
|
free (path);
|
|
|
|
return f;
|
|
|
|
} else eprintf ("r_fs_open: null root->p->open\n");
|
|
|
|
free (path);
|
2011-01-12 00:01:06 +01:00
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
2011-01-14 01:02:20 +01:00
|
|
|
// TODO: close or free?
|
2011-01-12 00:01:06 +01:00
|
|
|
R_API void r_fs_close (RFS* fs, RFSFile *file) {
|
2011-01-14 01:02:20 +01:00
|
|
|
if (fs && file && file->p && file->p->close)
|
|
|
|
file->p->close (file);
|
2011-01-12 00:01:06 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
R_API int r_fs_read (RFS* fs, RFSFile *file, ut64 addr, int len) {
|
2011-01-14 01:02:20 +01:00
|
|
|
if (len<1) {
|
|
|
|
eprintf ("r_fs_read: too short read\n");
|
2011-02-23 00:54:40 +01:00
|
|
|
return R_FALSE;
|
|
|
|
}
|
2011-01-12 00:01:06 +01:00
|
|
|
if (fs && file) {
|
|
|
|
free (file->data);
|
2011-01-14 01:02:20 +01:00
|
|
|
file->data = malloc (len+1);
|
|
|
|
if (file->p && file->p->read) {
|
|
|
|
file->p->read (file, addr, len);
|
|
|
|
return R_TRUE;
|
|
|
|
} else eprintf ("r_fs_read: file->p->read is null\n");
|
2011-01-12 00:01:06 +01:00
|
|
|
}
|
|
|
|
return R_FALSE;
|
|
|
|
}
|
|
|
|
|
2011-02-21 09:26:32 +01:00
|
|
|
R_API RList *r_fs_dir(RFS* fs, const char *p) {
|
2011-01-12 00:01:06 +01:00
|
|
|
if (fs) {
|
2011-02-21 09:26:32 +01:00
|
|
|
char *path = strdup (p);
|
2011-02-18 01:43:31 +01:00
|
|
|
r_str_chop_path (path);
|
2011-01-12 00:01:06 +01:00
|
|
|
RFSRoot *root = r_fs_root (fs, path);
|
2011-01-21 00:21:32 +01:00
|
|
|
if (root) {
|
|
|
|
const char *dir = path + strlen (root->path);
|
|
|
|
if (!*dir) dir = "/";
|
2011-02-21 09:26:32 +01:00
|
|
|
if (root) {
|
|
|
|
free (path);
|
2011-01-21 00:21:32 +01:00
|
|
|
return root->p->dir (root, dir);
|
2011-02-21 09:26:32 +01:00
|
|
|
}
|
2011-01-21 00:21:32 +01:00
|
|
|
}
|
2011-02-21 09:26:32 +01:00
|
|
|
free (path);
|
2011-02-21 18:10:22 +01:00
|
|
|
eprintf ("r_fs_dir: not mounted '%s'\n", path);
|
2011-01-12 00:01:06 +01:00
|
|
|
}
|
|
|
|
return NULL;
|
|
|
|
}
|
2011-01-14 01:02:20 +01:00
|
|
|
|
2011-01-14 01:05:23 +01:00
|
|
|
R_API RFSFile *r_fs_slurp(RFS* fs, const char *path) {
|
2011-01-14 01:02:20 +01:00
|
|
|
RFSFile *file = NULL;
|
|
|
|
RFSRoot *root = r_fs_root (fs, path);
|
|
|
|
if (root && root->p) {
|
|
|
|
if (root->p->open && root->p->read && root->p->close) {
|
|
|
|
file = root->p->open (root, path);
|
2011-02-23 00:54:40 +01:00
|
|
|
if (file) root->p->read (file, 0, file->size); //file->data
|
|
|
|
else eprintf ("r_fs_slurp: cannot open file\n");
|
2011-01-14 01:02:20 +01:00
|
|
|
} else {
|
2011-02-23 00:54:40 +01:00
|
|
|
if (root->p->slurp) return root->p->slurp (root, path);
|
2011-01-14 01:05:23 +01:00
|
|
|
else eprintf ("r_fs_slurp: null root->p->slurp\n");
|
2011-01-14 01:02:20 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
return file;
|
|
|
|
}
|
2011-01-14 14:41:56 +01:00
|
|
|
|
|
|
|
// TODO: move into grubfs
|
|
|
|
#include "p/grub/include/grubfs.h"
|
|
|
|
RList *list = NULL;
|
2011-02-21 15:20:33 +01:00
|
|
|
static int parhook (struct grub_disk *disk, struct grub_partition *par, void *closure) {
|
2011-01-14 14:41:56 +01:00
|
|
|
RFSPartition *p = r_fs_partition_new (r_list_length (list), par->start*512, 512*par->len);
|
|
|
|
p->type = par->msdostype;
|
|
|
|
r_list_append (list, p);
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2011-02-18 18:22:51 +01:00
|
|
|
R_API RList *r_fs_partitions (RFS *fs, const char *ptype, ut64 delta) {
|
2011-01-23 21:54:18 +01:00
|
|
|
struct grub_partition_map *gpm = NULL;
|
|
|
|
if (!strcmp (ptype, "msdos"))
|
|
|
|
gpm = &grub_msdos_partition_map;
|
|
|
|
else if (!strcmp (ptype, "apple"))
|
|
|
|
gpm = &grub_apple_partition_map;
|
|
|
|
else if (!strcmp (ptype, "sun"))
|
|
|
|
gpm = &grub_sun_partition_map;
|
|
|
|
else if (!strcmp (ptype, "sunpc"))
|
2011-01-23 23:58:46 +01:00
|
|
|
gpm = &grub_sun_pc_partition_map;
|
2011-01-23 21:54:18 +01:00
|
|
|
else if (!strcmp (ptype, "amiga"))
|
|
|
|
gpm = &grub_amiga_partition_map;
|
|
|
|
else if (!strcmp (ptype, "bsdlabel"))
|
|
|
|
gpm = &grub_bsdlabel_partition_map;
|
2011-02-21 09:26:32 +01:00
|
|
|
// XXX: In BURG all bsd partition map are in bsdlabel
|
|
|
|
// else if (!strcmp (ptype, "openbsdlabel"))
|
|
|
|
// gpm = &grub_openbsdlabel_partition_map;
|
|
|
|
// else if (!strcmp (ptype, "netbsdlabel"))
|
|
|
|
// gpm = &grub_netbsdlabel_partition_map;
|
2011-01-23 23:58:46 +01:00
|
|
|
// else if (!strcmp (ptype, "acorn"))
|
|
|
|
// gpm = &grub_acorn_partition_map;
|
2011-01-23 21:54:18 +01:00
|
|
|
else if (!strcmp (ptype, "gpt"))
|
|
|
|
gpm = &grub_gpt_partition_map;
|
|
|
|
|
|
|
|
if (gpm) {
|
|
|
|
list = r_list_new ();
|
|
|
|
list->free = (RListFree)r_fs_partition_free;
|
2011-01-14 14:41:56 +01:00
|
|
|
struct grub_disk *disk = grubfs_disk (&fs->iob);
|
2011-02-21 09:26:32 +01:00
|
|
|
gpm->iterate (disk, parhook, 0);
|
2011-01-23 21:54:18 +01:00
|
|
|
return list;
|
2011-01-14 14:41:56 +01:00
|
|
|
}
|
2011-02-23 00:54:40 +01:00
|
|
|
if (ptype&&*ptype)
|
|
|
|
eprintf ("Unknown partition type '%s'.\n", ptype);
|
|
|
|
eprintf ("Supported types:\n"
|
|
|
|
" msdos, apple, sun, sunpc, amiga, bsdlabel, acorn, gpt");
|
2011-01-23 21:54:18 +01:00
|
|
|
return NULL;
|
2011-01-14 14:41:56 +01:00
|
|
|
}
|
2011-02-18 18:22:51 +01:00
|
|
|
|
|
|
|
R_API int r_fs_prompt (RFS *fs, char *root) {
|
|
|
|
char buf[1024];
|
|
|
|
char path[1024];
|
|
|
|
char str[2048];
|
|
|
|
char *input;
|
|
|
|
RList *list;
|
|
|
|
RListIter *iter;
|
|
|
|
RFSFile *file;
|
|
|
|
|
2011-02-21 18:10:22 +01:00
|
|
|
if (root && *root) {
|
|
|
|
r_str_chop_path (root);
|
2011-02-23 03:01:26 +01:00
|
|
|
if (!r_fs_root (fs, root)) {
|
|
|
|
printf ("Unknown root\n");
|
|
|
|
return R_FALSE;
|
|
|
|
}
|
2011-02-21 18:10:22 +01:00
|
|
|
strncpy (path, root, sizeof (path)-1);
|
|
|
|
} else strcpy (path, "/");
|
2011-02-18 18:22:51 +01:00
|
|
|
|
|
|
|
for (;;) {
|
2011-02-21 18:10:22 +01:00
|
|
|
printf (Color_MAGENTA"[%s]> "Color_RESET, path);
|
2011-02-18 18:22:51 +01:00
|
|
|
fflush (stdout);
|
|
|
|
fgets (buf, sizeof (buf)-1, stdin);
|
|
|
|
if (feof (stdin)) break;
|
|
|
|
buf[strlen (buf)-1] = '\0';
|
|
|
|
if (!strcmp (buf, "q") || !strcmp (buf, "exit"))
|
|
|
|
return R_TRUE;
|
2011-02-21 18:10:22 +01:00
|
|
|
if (buf[0]=='!') {
|
|
|
|
system (buf+1);
|
|
|
|
} else
|
|
|
|
if (!strcmp (buf, "ls")) {
|
2011-02-18 18:22:51 +01:00
|
|
|
list = r_fs_dir (fs, path);
|
|
|
|
if (list) {
|
|
|
|
r_list_foreach (list, iter, file)
|
|
|
|
printf ("%c %s\n", file->type, file->name);
|
|
|
|
r_list_free (list);
|
2011-02-21 19:07:57 +01:00
|
|
|
} else printf ("Unknown path (%s)\n", path);
|
2011-02-21 18:10:22 +01:00
|
|
|
} else if (!strncmp (buf, "pwd", 3)) {
|
|
|
|
eprintf ("%s\n", path);
|
|
|
|
} else if (!memcmp (buf, "cd ", 3)) {
|
2011-02-18 18:22:51 +01:00
|
|
|
input = buf+3;
|
|
|
|
while (input[0] == ' ')
|
|
|
|
input++;
|
|
|
|
strncpy (str, path, sizeof(str)-1);
|
|
|
|
if (input[0] == '/')
|
|
|
|
strncpy (path, root, sizeof (path)-1);
|
|
|
|
strcat (path, "/");
|
|
|
|
strcat (path, input);
|
|
|
|
r_str_chop_path (path);
|
2011-02-18 22:34:50 +01:00
|
|
|
if (strlen (path) < strlen (root))
|
|
|
|
strncpy (path, root, sizeof (path)-1);
|
2011-02-18 18:22:51 +01:00
|
|
|
list = r_fs_dir (fs, path);
|
|
|
|
if (!r_list_empty (list))
|
|
|
|
r_list_free (list);
|
|
|
|
else {
|
|
|
|
strncpy (path, str, sizeof (path)-1);
|
|
|
|
printf ("Unknown path\n");
|
|
|
|
}
|
2011-02-21 18:10:22 +01:00
|
|
|
} else if (!memcmp (buf, "cat ", 4)) {
|
2011-02-18 18:22:51 +01:00
|
|
|
input = buf+3;
|
|
|
|
while (input[0] == ' ')
|
|
|
|
input++;
|
|
|
|
if (input[0] == '/')
|
|
|
|
strncpy (str, root, sizeof (str)-1);
|
2011-02-21 18:10:22 +01:00
|
|
|
else strncpy (str, path, sizeof (str)-1);
|
2011-02-18 18:22:51 +01:00
|
|
|
strcat (str, "/");
|
|
|
|
strcat (str, input);
|
|
|
|
file = r_fs_open (fs, str);
|
|
|
|
if (file) {
|
|
|
|
r_fs_read (fs, file, 0, file->size);
|
|
|
|
write (1, file->data, file->size);
|
|
|
|
r_fs_close (fs, file);
|
2011-02-21 18:10:22 +01:00
|
|
|
} else eprintf ("Cannot open file\n");
|
|
|
|
} else if (!memcmp (buf, "mount", 5)) {
|
|
|
|
RFSRoot *root;
|
|
|
|
r_list_foreach (fs->roots, iter, root) {
|
|
|
|
eprintf ("%s %s\n", root->path, root->p->name);
|
|
|
|
}
|
|
|
|
} else if (!memcmp (buf, "get ",4)) {
|
2011-02-18 18:22:51 +01:00
|
|
|
input = buf+3;
|
|
|
|
while (input[0] == ' ')
|
|
|
|
input++;
|
|
|
|
if (input[0] == '/')
|
|
|
|
strncpy (str, root, sizeof (str)-1);
|
2011-02-21 18:10:22 +01:00
|
|
|
else strncpy (str, path, sizeof (str)-1);
|
2011-02-18 18:22:51 +01:00
|
|
|
strcat (str, "/");
|
|
|
|
strcat (str, input);
|
|
|
|
file = r_fs_open (fs, str);
|
|
|
|
if (file) {
|
|
|
|
r_fs_read (fs, file, 0, file->size);
|
|
|
|
r_file_dump (input, file->data, file->size);
|
|
|
|
r_fs_close (fs, file);
|
|
|
|
} else printf ("Cannot open file\n");
|
2011-02-21 18:10:22 +01:00
|
|
|
} else if (!memcmp (buf, "help", 4) || !strcmp (buf, "?")) {
|
|
|
|
printf (
|
2011-02-18 18:22:51 +01:00
|
|
|
"Commands:\n"
|
2011-02-21 18:10:22 +01:00
|
|
|
" !cmd ; escape to system\n"
|
2011-02-18 18:22:51 +01:00
|
|
|
" ls ; list current directory\n"
|
|
|
|
" cd path ; change current directory\n"
|
|
|
|
" cat file ; print contents of file\n"
|
2011-02-21 18:10:22 +01:00
|
|
|
" get file ; dump file to disk\n"
|
|
|
|
" mount ; list mount points\n"
|
2011-02-18 18:22:51 +01:00
|
|
|
" q/exit ; leave prompt mode\n"
|
|
|
|
" ?/help ; show this help\n"
|
|
|
|
);
|
2011-02-18 22:34:50 +01:00
|
|
|
} else {
|
|
|
|
printf ("Unknown command %s\n", buf);
|
2011-02-18 18:22:51 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
clearerr (stdin);
|
|
|
|
printf ("\n");
|
|
|
|
return R_TRUE;
|
|
|
|
}
|