radare2/libr/io/io.c

325 lines
7.8 KiB
C
Raw Normal View History

/* radare - LGPL - Copyright 2008-2010 pancake<nopcode.org> */
#include "r_io.h"
#include "r_util.h"
#include <stdio.h>
// TODO: R_API int r_io_fetch(struct r_io_t *io, ut8 *buf, int len)
// --- check for EXEC perms in section (use cached read to accelerate)
R_API struct r_io_t *r_io_init(struct r_io_t *io) {
if (!io) return NULL;
io->write_mask_fd = -1;
io->redirect = NULL;
io->printf = (void*) printf;
r_io_cache_init(io);
r_io_map_init(io);
r_io_section_init(io);
r_io_handle_init(io);
r_io_desc_init(io);
r_io_undo_init(io);
return io;
}
R_API struct r_io_t *r_io_new() {
return r_io_init (R_NEW (struct r_io_t));
}
R_API RBuffer *r_io_read_buf(struct r_io_t *io, ut64 addr, int len) {
RBuffer *b = R_NEW (RBuffer);
b->buf = malloc (len);
len = r_io_read_at (io, addr, b->buf, len);
if (len<0) len = 0;
b->length = len;
return b;
}
R_API int r_io_write_buf(struct r_io_t *io, struct r_buf_t *b) {
return r_io_write_at(io, b->base, b->buf, b->length);
}
R_API struct r_io_t *r_io_free(struct r_io_t *io) {
/* TODO: properly free inner nfo */
free (io);
return NULL;
}
/* used by uri handler plugins */
R_API int r_io_redirect(struct r_io_t *io, const char *file) {
free (io->redirect);
io->redirect = file?strdup (file):NULL;
return 0;
}
R_API int r_io_open_as(struct r_io_t *io, const char *urihandler, const char *file, int flags, int mode) {
int ret;
char *uri;
int urilen = strlen (urihandler);
uri = malloc (strlen (urihandler)+strlen (file)+5);
if (uri == NULL)
return -1;
if (urilen>0)
sprintf (uri, "%s://", urihandler);
else *uri = '\0';
strcpy (uri+urilen, file);
ret = r_io_open (io, uri, flags, mode);
free (uri);
return ret;
}
R_API int r_io_open(struct r_io_t *io, const char *file, int flags, int mode) {
int fd = -2;
char *uri = strdup (file);
struct r_io_handle_t *plugin;
if (io != NULL) {
for (;;) {
plugin = r_io_handle_resolve (io, uri);
if (plugin) {
fd = plugin->open (io, uri, flags, mode);
if (io->redirect) {
free ((void *)uri);
uri = strdup (io->redirect);
r_io_redirect (io, NULL);
continue;
}
if (fd != -1)
r_io_handle_open (io, fd, plugin);
if (fd != io->fd)
io->plugin = plugin;
}
break;
}
}
2010-01-15 12:02:54 +00:00
if (fd == -2) {
#if __WINDOWS__
fd = open (file, 0);
#else
if (flags & R_IO_WRITE)
fd = open (file, O_RDWR, mode);
else fd = open (file, O_RDONLY, mode);
2010-01-15 12:02:54 +00:00
#endif
}
if (fd >= 0) {
r_io_set_fd (io, fd);
r_io_desc_add (io, fd, file, flags, io->plugin);
} else fd = -1;
free((void *)uri);
return fd;
}
// TODO: Rename to use_fd ?
R_API int r_io_set_fd(struct r_io_t *io, int fd)
{
if (fd != -1 && fd != io->fd) {
io->plugin = r_io_handle_resolve_fd (io, fd);
io->fd = fd;
}
return io->fd;
}
R_API int r_io_read(struct r_io_t *io, ut8 *buf, int len) {
int ret;
/* check section permissions */
if (io->enforce_rwx && !(r_io_section_get_rwx (io, io->off) & R_IO_READ))
return -1;
if (io->cached) {
ret = r_io_cache_read (io, io->off, buf, len);
if (ret == len)
return len;
if (ret > 0) {
len -= ret;
buf += ret;
}
// partial reads
if (ret == len)
return len;
}
ret = r_io_map_read_at (io, io->off, buf, len);
// partial reads
if (ret != len) {
if (ret != -1) {
len -= ret;
buf += len;
}
if (io->plugin && io->plugin->read) {
if (io->plugin->read != NULL)
ret = io->plugin->read(io, io->fd, buf, len);
else eprintf ("IO handler for fd=%d has no read()\n", io->fd);
} else ret = read (io->fd, buf, len);
if (ret > 0 && ret<len) //ret != len)
memset (buf+ret, 0xff, len-ret);
}
/* if read is cached. cache it :) */
if (ret != -1 && ret == len && io->cached_read)
r_io_cache_write (io, io->off, buf, len);
return ret;
}
R_API int r_io_read_at(struct r_io_t *io, ut64 addr, ut8 *buf, int len) {
if (r_io_seek (io, addr, R_IO_SEEK_SET)==-1)
return -1;
return r_io_read (io, buf, len);
}
R_API ut64 r_io_read_i(struct r_io_t *io, ut64 addr, int sz, int endian) {
ut64 ret = 0LL;
int err;
ut8 buf[8];
if (sz > 8) sz = 8;
if (sz < 0) sz = 1;
err = r_io_read_at (io, addr, buf, sz);
if (err == sz) r_mem_copyendian ((ut8*)&ret, buf, sz, endian);
else perror("Cannot read");
return ret;
}
R_API int r_io_resize(struct r_io_t *io, const char *file, int flags, int mode) {
// XXX not implemented
#if 0
/* TODO */
struct r_io_handle_t *plugin = r_io_handle_resolve(file);
if (plugin && io->plugin->resize) {
int fd = plugin->resize(file, flags, mode);
if (fd != -1)
r_io_handle_open(fd, plugin);
return fd;
}
#endif
return -1;
}
R_API int r_io_set_write_mask(struct r_io_t *io, const ut8 *buf, int len) {
int ret = R_FALSE;
if (len) {
io->write_mask_fd = io->fd;
io->write_mask_buf = (ut8 *)malloc (len);
memcpy (io->write_mask_buf, buf, len);
io->write_mask_len = len;
ret = R_TRUE;
} else io->write_mask_fd = -1;
return ret;
}
R_API int r_io_write(struct r_io_t *io, const ut8 *buf, int len) {
int i, ret = -1;
/* check section permissions */
if (io->enforce_rwx && !(r_io_section_get_rwx (io, io->off) & R_IO_WRITE))
return -1;
if (io->cached) {
ret = r_io_cache_write (io, io->off, buf, len);
if (ret == len)
return len;
if (ret > 0) {
len -= ret;
buf += ret;
}
}
/* TODO: implement IO cache here. to avoid dupping work on vm for example */
/* apply write binary mask */
if (io->write_mask_fd != -1) {
ut8 *data = alloca(len);
r_io_seek (io, io->off, R_IO_SEEK_SET);
r_io_read (io, data, len);
r_io_seek (io, io->off, R_IO_SEEK_SET);
for (i=0;i<len;i++) {
data[i] = buf[i] & \
io->write_mask_buf[i%io->write_mask_len];
}
buf = data;
}
if (r_io_map_write_at (io, io->off, buf, len) != 0)
return len;
if (io->plugin) {
if (io->plugin->write) {
ret = io->plugin->write (io, io->fd, buf, len);
} else eprintf ("r_io_write: io handler with no write callback\n");
} else ret = write (io->fd, buf, len);
if (ret == -1)
eprintf ("r_io_write: cannot write on fd %d\n", io->fd);
return ret;
}
R_API int r_io_write_at(struct r_io_t *io, ut64 addr, const ut8 *buf, int len) {
if (r_io_seek (io, addr, R_IO_SEEK_SET)<0)
return -1;
return r_io_write (io, buf, len);
}
R_API ut64 r_io_seek(struct r_io_t *io, ut64 offset, int whence) {
int posix_whence = SEEK_SET;
switch(whence) {
case R_IO_SEEK_SET:
posix_whence = SEEK_SET;
break;
case R_IO_SEEK_CUR:
offset += io->off;
posix_whence = SEEK_CUR;
break;
case R_IO_SEEK_END:
offset = UT64_MAX; // XXX: depending on io bits?
posix_whence = SEEK_END;
break;
}
offset = io->va ? r_io_section_vaddr_to_offset (io, offset) : offset;
// TODO: implement io->enforce_seek here!
if (io->plugin && io->plugin->lseek)
io->off = io->plugin->lseek (io, io->fd, offset, whence);
// XXX can be problematic on w32..so no 64 bit offset?
else io->off = lseek (io->fd, offset, posix_whence);
r_io_sundo_push (io);
return io->va ? r_io_section_offset_to_vaddr (io, io->off) : io->off;
}
R_API ut64 r_io_size(struct r_io_t *io, int fd) {
ut64 size, here;
fd = r_io_set_fd (io, fd);
here = r_io_seek (io, 0, R_IO_SEEK_CUR);
size = r_io_seek (io, 0, R_IO_SEEK_END);
r_io_seek (io, here, R_IO_SEEK_SET);
return size;
}
R_API int r_io_system(struct r_io_t *io, const char *cmd) {
int ret = -1;
if (io->plugin && io->plugin->system)
ret = io->plugin->system (io, io->fd, cmd);
return ret;
}
// TODO: remove int fd here???
R_API int r_io_close(struct r_io_t *io, int fd) {
fd = r_io_set_fd (io, fd);
if (fd != -1 && io->plugin) {
r_io_desc_del (io, fd);
r_io_map_del (io, fd);
r_io_handle_close (io, fd, io->plugin);
if (io->plugin->close)
return io->plugin->close (io, fd);
}
io->fd = -1; // unset current fd
return close (fd);
}
R_API int r_io_bind(struct r_io_t *io, struct r_io_bind_t *bnd) {
bnd->io = io;
bnd->init = R_TRUE;
bnd->read_at = r_io_read_at;
bnd->write_at = r_io_write_at;
//bnd->fd = io->fd;// do we need to store ptr to fd??
return R_TRUE;
}