2014-05-18 00:19:11 +00:00
|
|
|
/* radare - LGPL - Copyright 2008-2014 - pancake */
|
|
|
|
|
2014-04-29 16:10:35 +00:00
|
|
|
#include <r_userconf.h>
|
|
|
|
#include <r_io.h>
|
|
|
|
#include <r_lib.h>
|
|
|
|
|
|
|
|
typedef struct r_io_mmo_t {
|
|
|
|
char * filename;
|
|
|
|
int mode;
|
2014-05-02 21:43:47 +00:00
|
|
|
int flags;
|
2014-04-29 16:10:35 +00:00
|
|
|
int fd;
|
|
|
|
int opened;
|
|
|
|
ut8 modified;
|
|
|
|
RBuffer *buf;
|
|
|
|
RIO * io_backref;
|
2014-05-18 00:19:11 +00:00
|
|
|
int rawio;
|
2014-04-29 16:10:35 +00:00
|
|
|
} RIOMMapFileObj;
|
|
|
|
|
|
|
|
static int r_io_def_mmap_refresh_def_mmap_buf(RIOMMapFileObj *mmo);
|
|
|
|
static void r_io_def_mmap_free (RIOMMapFileObj *mmo);
|
|
|
|
static int r_io_def_mmap_close(RIODesc *fd);
|
|
|
|
static int r_io_def_mmap_read(RIO *io, RIODesc *fd, ut8 *buf, int count);
|
|
|
|
static int r_io_def_mmap_write(RIO *io, RIODesc *fd, const ut8 *buf, int count);
|
2014-05-02 21:43:47 +00:00
|
|
|
static RIODesc *r_io_def_mmap_open(RIO *io, const char *file, int flags, int mode);
|
2014-04-29 16:10:35 +00:00
|
|
|
static ut64 r_io_def_mmap_seek(RIO *io, RIOMMapFileObj *mmo, ut64 offset, int whence);
|
|
|
|
static ut64 r_io_def_mmap_lseek(RIO *io, RIODesc *fd, ut64 offset, int whence);
|
|
|
|
static int r_io_def_mmap_truncate(RIOMMapFileObj *mmo, ut64 size);
|
|
|
|
static int r_io_def_mmap_resize(RIO *io, RIODesc *fd, ut64 size);
|
|
|
|
|
|
|
|
static int __plugin_open_default(RIO *io, const char *file, ut8 many);
|
2014-05-02 21:43:47 +00:00
|
|
|
static RIODesc *__open_default(RIO *io, const char *file, int flags, int mode);
|
2014-04-29 16:10:35 +00:00
|
|
|
static int __read(RIO *io, RIODesc *fd, ut8 *buf, int len);
|
|
|
|
static int __write(RIO *io, RIODesc *fd, const ut8 *buf, int len);
|
|
|
|
static ut64 __lseek(RIO *io, RIODesc *fd, ut64 offset, int whence);
|
|
|
|
static int __close(RIODesc *fd);
|
|
|
|
static int __resize(RIO *io, RIODesc *fd, ut64 newsize);
|
|
|
|
|
2014-05-18 00:19:11 +00:00
|
|
|
static int __io_posix_open (const char *file, int flags, int mode) {
|
|
|
|
int fd;
|
|
|
|
if (r_file_is_directory (file))
|
|
|
|
return -1;
|
|
|
|
#if __WINDOWS__
|
|
|
|
if (flags & R_IO_WRITE) {
|
|
|
|
fd = r_sandbox_open (file, O_BINARY | 1, 0);
|
|
|
|
if (fd == -1)
|
|
|
|
r_sandbox_creat (file, O_BINARY);
|
|
|
|
fd = r_sandbox_open (file, O_BINARY | 1, 0);
|
|
|
|
} else fd = r_sandbox_open (file, O_BINARY, 0);
|
|
|
|
#else
|
|
|
|
fd = r_sandbox_open (file, (flags&R_IO_WRITE)?
|
|
|
|
(O_RDWR|O_CREAT): O_RDONLY, mode);
|
|
|
|
#endif
|
|
|
|
return fd;
|
|
|
|
}
|
2014-04-29 16:10:35 +00:00
|
|
|
|
|
|
|
static int r_io_def_mmap_refresh_def_mmap_buf(RIOMMapFileObj *mmo) {
|
|
|
|
RIO* io = mmo->io_backref;
|
|
|
|
ut64 cur = mmo->buf ? mmo->buf->cur : 0;
|
|
|
|
if (mmo->buf) {
|
|
|
|
r_buf_free (mmo->buf);
|
|
|
|
mmo->buf = NULL;
|
|
|
|
}
|
2014-05-18 00:19:11 +00:00
|
|
|
if (mmo->rawio) {
|
|
|
|
mmo->fd = __io_posix_open (mmo->filename, mmo->flags, mmo->mode);
|
|
|
|
return (mmo->fd != -1);
|
|
|
|
}
|
2014-05-02 21:43:47 +00:00
|
|
|
mmo->buf = r_buf_mmap (mmo->filename, mmo->flags);
|
2014-04-29 16:10:35 +00:00
|
|
|
if (mmo->buf)
|
|
|
|
r_io_def_mmap_seek (io, mmo, cur, SEEK_SET);
|
|
|
|
return (mmo->buf ? R_TRUE : R_FALSE);
|
|
|
|
}
|
|
|
|
|
2014-05-02 21:43:47 +00:00
|
|
|
RIOMMapFileObj *r_io_def_mmap_create_new_file(RIO *io, const char *filename, int mode, int flags) {
|
2014-05-05 15:44:08 +00:00
|
|
|
RIOMMapFileObj *mmo = NULL;
|
|
|
|
if (!io)
|
|
|
|
return NULL;
|
|
|
|
|
|
|
|
mmo = R_NEW0 (RIOMMapFileObj);
|
|
|
|
if (!mmo)
|
2014-04-29 16:10:35 +00:00
|
|
|
return NULL;
|
|
|
|
|
|
|
|
mmo->filename = strdup (filename);
|
|
|
|
mmo->mode = mode;
|
2014-05-02 21:43:47 +00:00
|
|
|
mmo->flags = flags;
|
2014-04-29 16:10:35 +00:00
|
|
|
mmo->io_backref = io;
|
2014-06-06 20:34:03 +00:00
|
|
|
mmo->fd = r_sandbox_open (filename, O_CREAT|O_RDWR, mode);
|
2014-04-29 16:10:35 +00:00
|
|
|
|
|
|
|
if (!r_io_def_mmap_refresh_def_mmap_buf (mmo)) {
|
2014-05-18 00:19:11 +00:00
|
|
|
mmo->rawio = 1;
|
|
|
|
if (!r_io_def_mmap_refresh_def_mmap_buf (mmo)) {
|
|
|
|
r_io_def_mmap_free (mmo);
|
|
|
|
mmo = NULL;
|
|
|
|
}
|
2014-04-29 16:10:35 +00:00
|
|
|
}
|
|
|
|
return mmo;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void r_io_def_mmap_free (RIOMMapFileObj *mmo) {
|
|
|
|
free (mmo->filename);
|
|
|
|
r_buf_free (mmo->buf);
|
2014-05-18 00:19:11 +00:00
|
|
|
close (mmo->fd);
|
2014-05-20 23:46:01 +00:00
|
|
|
memset (mmo, 0, sizeof (RIOMMapFileObj));
|
2014-04-29 16:10:35 +00:00
|
|
|
free (mmo);
|
|
|
|
}
|
|
|
|
|
|
|
|
static int r_io_def_mmap_close(RIODesc *fd) {
|
|
|
|
if (!fd || !fd->data)
|
|
|
|
return -1;
|
|
|
|
r_io_def_mmap_free ( (RIOMMapFileObj *) fd->data);
|
|
|
|
fd->data = NULL;
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int r_io_def_mmap_check_default (const char *filename) {
|
|
|
|
char * peekaboo = strstr (filename, "://");
|
|
|
|
if ( (filename && !peekaboo) ||
|
|
|
|
( (peekaboo-filename) > 10 ) )
|
|
|
|
return 1;
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int r_io_def_mmap_read(RIO *io, RIODesc *fd, ut8 *buf, int count) {
|
|
|
|
RIOMMapFileObj *mmo = NULL;
|
|
|
|
if (!fd || !fd->data || !buf)
|
|
|
|
return -1;
|
2014-05-18 00:37:34 +00:00
|
|
|
if (io->off==UT64_MAX) {
|
|
|
|
memset (buf, 0xff, count);
|
|
|
|
return count;
|
|
|
|
}
|
2014-04-29 16:10:35 +00:00
|
|
|
mmo = fd->data;
|
2014-05-18 00:19:11 +00:00
|
|
|
if (mmo->rawio) {
|
|
|
|
return read (mmo->fd, buf, count);
|
|
|
|
}
|
2014-04-29 16:10:35 +00:00
|
|
|
if (mmo->buf->length < io->off)
|
|
|
|
io->off = mmo->buf->length;
|
|
|
|
return r_buf_read_at (mmo->buf, io->off, buf, count);
|
|
|
|
}
|
|
|
|
|
|
|
|
static int r_io_def_mmap_write(RIO *io, RIODesc *fd, const ut8 *buf, int count) {
|
|
|
|
RIOMMapFileObj *mmo;
|
|
|
|
int len = -1;
|
|
|
|
ut64 addr = io->off;
|
|
|
|
|
2014-05-02 21:43:47 +00:00
|
|
|
if (!fd || !fd->data || !buf) return -1;
|
2014-04-29 16:10:35 +00:00
|
|
|
|
|
|
|
mmo = fd->data;
|
|
|
|
|
2014-05-21 02:36:53 +00:00
|
|
|
if (mmo && mmo->buf) {
|
|
|
|
if (!(mmo->flags & R_IO_WRITE)) return -1;
|
|
|
|
if ( (count + addr > mmo->buf->length) || mmo->buf->empty) {
|
|
|
|
ut64 sz = count + addr;
|
|
|
|
r_file_truncate (mmo->filename, sz);
|
|
|
|
}
|
2014-04-29 16:10:35 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
len = r_file_mmap_write (mmo->filename, io->off, buf, count);
|
2014-06-06 20:34:03 +00:00
|
|
|
if (len != count) {
|
|
|
|
// aim to hack some corner cases?
|
|
|
|
lseek (fd->fd, addr, 0);
|
|
|
|
len = write (fd->fd, buf, count);
|
|
|
|
}
|
2014-04-29 16:10:35 +00:00
|
|
|
if (!r_io_def_mmap_refresh_def_mmap_buf (mmo) ) {
|
|
|
|
eprintf ("io_def_mmap: failed to refresh the def_mmap backed buffer.\n");
|
|
|
|
// XXX - not sure what needs to be done here (error handling).
|
|
|
|
}
|
|
|
|
return len;
|
|
|
|
}
|
|
|
|
|
2014-05-02 21:43:47 +00:00
|
|
|
static RIODesc *r_io_def_mmap_open(RIO *io, const char *file, int flags, int mode) {
|
2014-05-18 00:19:11 +00:00
|
|
|
RIOMMapFileObj *mmo = r_io_def_mmap_create_new_file (io, file, mode, flags);
|
2014-04-29 16:10:35 +00:00
|
|
|
if (!mmo) return NULL;
|
|
|
|
return r_io_desc_new (&r_io_plugin_default, mmo->fd,
|
2014-05-02 21:43:47 +00:00
|
|
|
mmo->filename, flags, mode, mmo);
|
2014-04-29 16:10:35 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
static ut64 r_io_def_mmap_seek(RIO *io, RIOMMapFileObj *mmo, ut64 offset, int whence) {
|
2014-05-18 00:37:34 +00:00
|
|
|
ut64 seek_val = UT64_MAX;
|
2014-05-19 01:28:12 +00:00
|
|
|
|
2014-05-18 00:37:34 +00:00
|
|
|
if (!mmo) return UT64_MAX;
|
2014-05-18 00:19:11 +00:00
|
|
|
if (mmo->rawio)
|
|
|
|
return lseek (mmo->fd, offset, whence);
|
2014-05-19 05:06:38 +00:00
|
|
|
if (!mmo->buf) return UT64_MAX;
|
2014-05-19 01:28:12 +00:00
|
|
|
|
|
|
|
seek_val = mmo->buf->cur;
|
2014-04-29 16:10:35 +00:00
|
|
|
switch (whence) {
|
2014-05-19 01:28:12 +00:00
|
|
|
case SEEK_SET:
|
|
|
|
seek_val = (mmo->buf->length < offset) ?
|
|
|
|
mmo->buf->length : offset;
|
|
|
|
break;
|
|
|
|
case SEEK_CUR:
|
|
|
|
seek_val = (mmo->buf->length < (offset + mmo->buf->cur)) ?
|
|
|
|
mmo->buf->length : offset + mmo->buf->cur;
|
|
|
|
break;
|
|
|
|
case SEEK_END:
|
|
|
|
seek_val = mmo->buf->length;
|
|
|
|
break;
|
2014-04-29 16:10:35 +00:00
|
|
|
}
|
2014-05-18 00:37:34 +00:00
|
|
|
mmo->buf->cur = io->off = seek_val;
|
2014-04-29 16:10:35 +00:00
|
|
|
return seek_val;
|
|
|
|
}
|
|
|
|
|
|
|
|
static ut64 r_io_def_mmap_lseek(RIO *io, RIODesc *fd, ut64 offset, int whence) {
|
|
|
|
RIOMMapFileObj *mmo;
|
|
|
|
if (!fd || !fd->data)
|
|
|
|
return -1;
|
|
|
|
mmo = fd->data;
|
|
|
|
return r_io_def_mmap_seek(io, mmo, offset, whence);
|
|
|
|
}
|
|
|
|
|
|
|
|
static int r_io_def_mmap_truncate(RIOMMapFileObj *mmo, ut64 size) {
|
|
|
|
int res = r_file_truncate (mmo->filename, size);
|
|
|
|
|
|
|
|
if (res && !r_io_def_mmap_refresh_def_mmap_buf (mmo) ) {
|
|
|
|
eprintf ("r_io_def_mmap_truncate: Error trying to refresh the def_mmap'ed file.");
|
|
|
|
res = R_FALSE;
|
|
|
|
}
|
|
|
|
else if (!res) eprintf ("r_io_def_mmap_truncate: Error trying to resize the file.");
|
|
|
|
return res;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int r_io_def_mmap_resize(RIO *io, RIODesc *fd, ut64 size) {
|
|
|
|
RIOMMapFileObj *mmo;
|
|
|
|
if (!fd || !fd->data)
|
|
|
|
return -1;
|
|
|
|
mmo = fd->data;
|
2014-05-19 00:52:51 +00:00
|
|
|
if (!(mmo->flags & R_IO_WRITE)) return -1;
|
2014-06-06 20:34:03 +00:00
|
|
|
return r_io_def_mmap_truncate (mmo, size);
|
2014-04-29 16:10:35 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
static int __plugin_open_default(RIO *io, const char *file, ut8 many) {
|
|
|
|
return r_io_def_mmap_check_default (file);
|
|
|
|
}
|
|
|
|
|
2014-05-02 21:43:47 +00:00
|
|
|
static RIODesc *__open_default(RIO *io, const char *file, int flags, int mode) {
|
2014-04-29 16:10:35 +00:00
|
|
|
if (!r_io_def_mmap_check_default (file) ) return NULL;
|
2014-05-02 21:43:47 +00:00
|
|
|
return r_io_def_mmap_open (io, file, flags, mode);
|
2014-04-29 16:10:35 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
static int __read(RIO *io, RIODesc *fd, ut8 *buf, int len) {
|
|
|
|
return r_io_def_mmap_read (io, fd, buf, len);
|
|
|
|
}
|
|
|
|
|
|
|
|
static int __write(RIO *io, RIODesc *fd, const ut8 *buf, int len) {
|
|
|
|
return r_io_def_mmap_write(io, fd, buf, len);
|
|
|
|
}
|
|
|
|
|
|
|
|
static ut64 __lseek(RIO *io, RIODesc *fd, ut64 offset, int whence) {
|
|
|
|
return r_io_def_mmap_lseek (io, fd, offset, whence);
|
|
|
|
}
|
|
|
|
|
|
|
|
static int __close(RIODesc *fd) {
|
|
|
|
return r_io_def_mmap_close (fd);
|
|
|
|
}
|
|
|
|
|
|
|
|
static int __resize(RIO *io, RIODesc *fd, ut64 size) {
|
|
|
|
return r_io_def_mmap_resize (io, fd, size);
|
|
|
|
}
|
|
|
|
|
|
|
|
struct r_io_plugin_t r_io_plugin_default = {
|
|
|
|
.name = "default",
|
|
|
|
.desc = "open local files using def_mmap://",
|
|
|
|
.license = "LGPL3",
|
|
|
|
.open = __open_default,
|
|
|
|
.close = __close,
|
|
|
|
.read = __read,
|
|
|
|
.plugin_open = __plugin_open_default,
|
|
|
|
.lseek = __lseek,
|
|
|
|
.write = __write,
|
|
|
|
.resize = __resize,
|
|
|
|
};
|
|
|
|
|
|
|
|
#ifndef CORELIB
|
|
|
|
struct r_lib_struct_t radare_plugin = {
|
|
|
|
.type = R_LIB_TYPE_IO,
|
|
|
|
.data = &r_io_plugin_default
|
|
|
|
};
|
2014-05-18 00:19:11 +00:00
|
|
|
#endif
|