2012-01-31 02:34:23 +00:00
|
|
|
/* radare - LGPL - Copyright 2008-2012 pancake<nopcode.org> nibble <.ds@gmail.com> */
|
2009-02-05 21:08:46 +00:00
|
|
|
|
|
|
|
#include "r_io.h"
|
|
|
|
|
2010-04-08 10:29:47 +00:00
|
|
|
R_API void r_io_section_init(RIO *io) {
|
2011-04-18 22:59:16 +00:00
|
|
|
io->next_section_id = 0;
|
2009-09-08 18:16:52 +00:00
|
|
|
io->enforce_rwx = 0; // do not enforce RWX section permissions by default
|
|
|
|
io->enforce_seek = 0; // do not limit seeks out of the file by default
|
2011-04-19 12:43:48 +00:00
|
|
|
io->sections = r_list_new ();
|
2009-09-08 02:14:19 +00:00
|
|
|
}
|
2009-02-05 21:08:46 +00:00
|
|
|
|
2012-02-16 00:26:50 +00:00
|
|
|
#if 0
|
2011-05-13 10:24:17 +00:00
|
|
|
static int cmpaddr (void *_a, void *_b) {
|
|
|
|
RIOSection *a = _a, *b = _b;
|
|
|
|
return (a->vaddr > b->vaddr);
|
|
|
|
}
|
2012-02-16 00:26:50 +00:00
|
|
|
#endif
|
2011-05-13 10:24:17 +00:00
|
|
|
|
2011-09-03 01:10:01 +00:00
|
|
|
R_API RIOSection *r_io_section_get_name(RIO *io, const char *name) {
|
|
|
|
RListIter *iter;
|
|
|
|
RIOSection *s;
|
2011-10-24 15:16:54 +00:00
|
|
|
if (name)
|
2011-09-03 01:10:01 +00:00
|
|
|
r_list_foreach (io->sections, iter, s) {
|
|
|
|
if (!strcmp (name, s->name))
|
|
|
|
return s;
|
|
|
|
}
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
2010-03-23 22:26:59 +00:00
|
|
|
R_API void r_io_section_add(RIO *io, ut64 offset, ut64 vaddr, ut64 size, ut64 vsize, int rwx, const char *name) {
|
2011-09-03 01:10:01 +00:00
|
|
|
int update = 0;
|
|
|
|
RIOSection *s = r_io_section_get_name (io, name);
|
2011-11-03 22:34:18 +00:00
|
|
|
if (s == NULL) {
|
2011-09-03 01:10:01 +00:00
|
|
|
s = R_NEW (RIOSection);
|
2011-11-03 22:34:18 +00:00
|
|
|
s->id = io->next_section_id++;
|
|
|
|
} else update = 1;
|
2012-01-31 02:34:23 +00:00
|
|
|
if (size>0xf00000) {
|
|
|
|
eprintf ("Invalid size for section at 0x%08"PFMT64x"\n", vaddr);
|
|
|
|
return;
|
|
|
|
}
|
2010-02-18 17:58:28 +00:00
|
|
|
s->offset = offset;
|
2009-02-05 21:08:46 +00:00
|
|
|
s->vaddr = vaddr;
|
2010-02-18 17:58:28 +00:00
|
|
|
s->size = size;
|
|
|
|
s->vsize = vsize;
|
2009-02-05 21:08:46 +00:00
|
|
|
s->rwx = rwx;
|
2011-09-03 01:10:01 +00:00
|
|
|
if (!update) {
|
2011-10-24 15:16:54 +00:00
|
|
|
if (name) strncpy (s->name, name, sizeof (s->name)-4);
|
2011-09-03 01:10:01 +00:00
|
|
|
else *s->name = '\0';
|
2011-10-25 18:30:05 +00:00
|
|
|
r_list_append (io->sections, s);
|
|
|
|
//r_list_prepend (io->sections, s);
|
|
|
|
//r_list_add_sorted (io->sections, s, cmpaddr);
|
2011-11-03 22:34:18 +00:00
|
|
|
} //else {
|
2011-09-03 01:10:01 +00:00
|
|
|
// This is a bottleneck.. the sorting must be done at append time
|
2011-11-03 22:34:18 +00:00
|
|
|
// r_list_sort (io->sections, cmpaddr);
|
|
|
|
//}
|
2009-02-05 21:08:46 +00:00
|
|
|
}
|
|
|
|
|
2010-04-08 10:29:47 +00:00
|
|
|
R_API RIOSection *r_io_section_get_i(RIO *io, int idx) {
|
2011-04-19 12:43:48 +00:00
|
|
|
RListIter *iter;
|
|
|
|
RIOSection *s;
|
|
|
|
r_list_foreach (io->sections, iter, s) {
|
2011-04-18 22:59:16 +00:00
|
|
|
if (s->id == idx)
|
2009-02-05 21:08:46 +00:00
|
|
|
return s;
|
|
|
|
}
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
2010-04-08 10:29:47 +00:00
|
|
|
R_API int r_io_section_rm(RIO *io, int idx) {
|
2011-04-19 12:43:48 +00:00
|
|
|
return r_list_del_n (io->sections, idx);
|
2009-02-05 21:08:46 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// TODO: implement as callback
|
2010-04-08 10:29:47 +00:00
|
|
|
R_API void r_io_section_list(RIO *io, ut64 offset, int rad) {
|
2009-02-05 21:08:46 +00:00
|
|
|
int i = 0;
|
2011-04-19 12:43:48 +00:00
|
|
|
RListIter *iter;
|
|
|
|
RIOSection *s;
|
2010-02-21 10:35:49 +00:00
|
|
|
|
2010-08-15 23:50:01 +00:00
|
|
|
if (io->va || io->debug)
|
2010-07-02 00:01:51 +00:00
|
|
|
offset = r_io_section_vaddr_to_offset (io, offset);
|
2011-10-25 18:30:05 +00:00
|
|
|
r_list_foreach (io->sections, iter, s) {
|
2012-01-31 02:34:23 +00:00
|
|
|
if (rad) io->printf ("S 0x%08"PFMT64x" 0x%08"PFMT64x" 0x%08"PFMT64x" 0x%08"PFMT64x" %s %s\n",
|
|
|
|
s->offset, s->vaddr, s->size, s->vsize, s->name, r_str_rwx_i (s->rwx));
|
2011-04-18 22:59:16 +00:00
|
|
|
else io->printf ("[%.2d] %c 0x%08"PFMT64x" %s va=0x%08"PFMT64x" sz=0x%08"PFMT64x" vsz=%08"PFMT64x" %s\n",
|
2011-04-19 12:43:48 +00:00
|
|
|
s->id, (offset>=s->offset && offset<s->offset+s->size)?'*':'.',
|
2010-04-08 10:29:47 +00:00
|
|
|
s->offset, r_str_rwx_i (s->rwx), s->vaddr, s->size, s->vsize, s->name);
|
2009-02-05 21:08:46 +00:00
|
|
|
i++;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2009-09-08 02:14:19 +00:00
|
|
|
/* TODO: move to print ??? support pretty print of ranges following an array of offsetof */
|
2010-04-08 10:29:47 +00:00
|
|
|
R_API void r_io_section_list_visual(RIO *io, ut64 seek, ut64 len) {
|
2011-04-19 12:43:48 +00:00
|
|
|
RListIter *iter;
|
|
|
|
RIOSection *s;
|
2009-07-08 11:49:55 +00:00
|
|
|
ut64 min = -1;
|
|
|
|
ut64 max = -1;
|
|
|
|
ut64 mul;
|
2011-07-03 15:45:44 +00:00
|
|
|
int j, i, width = 30; //config.width-30;
|
2009-02-05 21:08:46 +00:00
|
|
|
|
2010-08-15 23:50:01 +00:00
|
|
|
seek = (io->va || io->debug) ? r_io_section_vaddr_to_offset (io, seek) : seek;
|
2011-04-19 12:43:48 +00:00
|
|
|
r_list_foreach (io->sections, iter, s) {
|
2010-02-18 17:58:28 +00:00
|
|
|
if (min == -1 || s->offset < min)
|
|
|
|
min = s->offset;
|
|
|
|
if (max == -1 || s->offset+s->size > max)
|
|
|
|
max = s->offset+s->size;
|
2009-02-05 21:08:46 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
mul = (max-min) / width;
|
|
|
|
if (min != -1 && mul != 0) {
|
|
|
|
i = 0;
|
2011-10-25 18:30:05 +00:00
|
|
|
r_list_foreach (io->sections, iter, s) {
|
2010-08-15 23:50:01 +00:00
|
|
|
io->printf ("%02d%c 0x%08"PFMT64x" |",
|
|
|
|
i, (seek>=s->offset && seek<s->offset+s->size)?'*':' ', s->offset);
|
2010-08-15 19:30:59 +00:00
|
|
|
for (j=0; j<width; j++) {
|
2010-02-18 17:58:28 +00:00
|
|
|
if ((j*mul)+min >= s->offset && (j*mul)+min <=s->offset+s->size)
|
2011-10-25 18:30:05 +00:00
|
|
|
io->printf ("#");
|
|
|
|
else io->printf ("-");
|
2009-02-05 21:08:46 +00:00
|
|
|
}
|
2011-09-28 00:48:03 +00:00
|
|
|
io->printf ("| 0x%08"PFMT64x" %s %s\n", s->offset+s->size,
|
|
|
|
r_str_rwx_i (s->rwx), s->name);
|
2009-02-05 21:08:46 +00:00
|
|
|
i++;
|
|
|
|
}
|
|
|
|
/* current seek */
|
|
|
|
if (i>0 && len != 0) {
|
2011-10-25 18:30:05 +00:00
|
|
|
if (seek == UT64_MAX)
|
|
|
|
seek = 0;
|
|
|
|
//len = 8096;//r_io_size (io);
|
2010-04-14 11:02:23 +00:00
|
|
|
io->printf ("=> 0x%08"PFMT64x" |", seek);
|
2011-10-25 18:30:05 +00:00
|
|
|
for (j=0;j<width;j++) {
|
2010-04-08 10:29:47 +00:00
|
|
|
io->printf (
|
2009-09-05 23:58:02 +00:00
|
|
|
((j*mul)+min >= seek &&
|
|
|
|
(j*mul)+min <= seek+len)
|
2011-09-28 00:48:03 +00:00
|
|
|
?"^":"-");
|
2009-02-05 21:08:46 +00:00
|
|
|
}
|
2010-04-14 11:02:23 +00:00
|
|
|
io->printf ("| 0x%08"PFMT64x"\n", seek+len);
|
2009-02-05 21:08:46 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2010-04-08 10:29:47 +00:00
|
|
|
R_API RIOSection *r_io_section_get(RIO *io, ut64 offset) {
|
2011-04-19 12:43:48 +00:00
|
|
|
RListIter *iter;
|
|
|
|
RIOSection *s;
|
|
|
|
|
|
|
|
r_list_foreach (io->sections, iter, s) {
|
2011-10-25 18:30:05 +00:00
|
|
|
if (offset >= s->offset && offset <= s->offset + s->size) {
|
|
|
|
//eprintf ("SG: %llx %s\n", offset, s->name);
|
2009-02-05 21:08:46 +00:00
|
|
|
return s;
|
2011-10-25 18:30:05 +00:00
|
|
|
}
|
2009-02-05 21:08:46 +00:00
|
|
|
}
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
2010-04-08 10:29:47 +00:00
|
|
|
R_API ut64 r_io_section_get_offset(RIO *io, ut64 offset) {
|
2010-02-18 17:58:28 +00:00
|
|
|
RIOSection *s = r_io_section_get(io, offset);
|
2011-10-25 18:30:05 +00:00
|
|
|
return s? s->offset: -1;
|
2009-02-05 21:08:46 +00:00
|
|
|
}
|
|
|
|
|
2010-04-08 10:29:47 +00:00
|
|
|
R_API ut64 r_io_section_get_vaddr(RIO *io, ut64 offset) {
|
|
|
|
RIOSection *s = r_io_section_get (io, offset);
|
2011-10-25 18:30:05 +00:00
|
|
|
return s? s->vaddr: -1;
|
2009-02-05 21:08:46 +00:00
|
|
|
}
|
|
|
|
|
2010-04-08 10:29:47 +00:00
|
|
|
// TODO: deprecate
|
|
|
|
R_API int r_io_section_get_rwx(RIO *io, ut64 offset) {
|
|
|
|
RIOSection *s = r_io_section_get (io, offset);
|
|
|
|
eprintf ("r_io_section_get_rwx: must be deprecated\n");
|
2009-09-08 02:14:19 +00:00
|
|
|
return s?s->rwx:R_IO_READ|R_IO_WRITE|R_IO_EXEC;
|
|
|
|
}
|
|
|
|
|
2010-04-08 10:29:47 +00:00
|
|
|
R_API int r_io_section_overlaps(RIO *io, RIOSection *s) {
|
2009-02-05 21:08:46 +00:00
|
|
|
int i = 0;
|
2011-04-19 12:43:48 +00:00
|
|
|
RListIter *iter;
|
|
|
|
RIOSection *s2;
|
|
|
|
|
2011-10-25 18:30:05 +00:00
|
|
|
r_list_foreach (io->sections, iter, s2) {
|
2009-02-05 21:08:46 +00:00
|
|
|
if (s != s2) {
|
2010-02-18 17:58:28 +00:00
|
|
|
if (s->offset >= s2->offset) {
|
|
|
|
if (s2->offset+s2->size < s->offset)
|
2009-02-05 21:08:46 +00:00
|
|
|
return i;
|
|
|
|
} else {
|
2010-02-18 17:58:28 +00:00
|
|
|
if (s->offset+s->size < s2->offset)
|
2009-02-05 21:08:46 +00:00
|
|
|
return i;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
i++;
|
|
|
|
}
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
2010-04-08 10:29:47 +00:00
|
|
|
R_API ut64 r_io_section_vaddr_to_offset(RIO *io, ut64 vaddr) {
|
2011-04-19 12:43:48 +00:00
|
|
|
RListIter *iter;
|
|
|
|
RIOSection *s;
|
|
|
|
|
2011-10-25 18:30:05 +00:00
|
|
|
r_list_foreach (io->sections, iter, s) {
|
|
|
|
if (vaddr >= s->vaddr && vaddr < s->vaddr + s->vsize) {
|
|
|
|
// eprintf ("SG: %llx phys=%llx %s\n", vaddr, vaddr-s->vaddr+s->offset, s->name);
|
2011-04-19 12:43:48 +00:00
|
|
|
return (vaddr - s->vaddr + s->offset);
|
2011-10-25 18:30:05 +00:00
|
|
|
}
|
2009-02-05 21:08:46 +00:00
|
|
|
}
|
2010-04-08 12:04:34 +00:00
|
|
|
return -1;
|
2010-02-18 17:58:28 +00:00
|
|
|
}
|
|
|
|
|
2010-04-08 10:29:47 +00:00
|
|
|
R_API ut64 r_io_section_offset_to_vaddr(RIO *io, ut64 offset) {
|
2011-04-19 12:43:48 +00:00
|
|
|
RListIter *iter;
|
|
|
|
RIOSection *s;
|
|
|
|
|
2011-10-25 18:30:05 +00:00
|
|
|
r_list_foreach (io->sections, iter, s) {
|
2010-02-21 10:35:49 +00:00
|
|
|
if (offset >= s->offset && offset < s->offset + s->size)
|
2011-04-19 12:43:48 +00:00
|
|
|
return (s->vaddr + offset - s->offset);
|
2010-02-18 17:58:28 +00:00
|
|
|
}
|
2010-04-08 12:04:34 +00:00
|
|
|
return -1;
|
2009-02-05 21:08:46 +00:00
|
|
|
}
|
2011-10-25 18:30:05 +00:00
|
|
|
|
|
|
|
R_API ut64 r_io_section_next(RIO *io, ut64 o) {
|
|
|
|
RListIter *iter;
|
|
|
|
RIOSection *s;
|
|
|
|
|
|
|
|
r_list_foreach (io->sections, iter, s) {
|
|
|
|
//eprintf (" o=%llx (%llx) (%llx)\n", o, s->offset, s->size);
|
|
|
|
if (o >= s->vaddr && o < (s->vaddr + s->size)) {
|
|
|
|
ut64 n = s->vaddr + s->size;
|
|
|
|
if (n>o)
|
|
|
|
o = n;
|
|
|
|
#if 0
|
|
|
|
if (first) {
|
|
|
|
first = 0;
|
|
|
|
goto restart;
|
|
|
|
} else o = s->vaddr;
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return o;
|
|
|
|
}
|