radare2/libr/io/section.c
2018-01-24 10:47:33 +01:00

686 lines
16 KiB
C

/* radare2 - LGPL - Copyright 2017-2018 - condret , alvarofe */
#include <r_io.h>
#include <r_util.h>
#include <sdb.h>
#include <r_types.h>
#include <stdio.h>
#include <string.h>
static void section_free(void *p) {
RIOSection *s = (RIOSection *) p;
if (s) {
free (s->name);
free (s);
}
}
R_API void r_io_section_init(RIO *io) {
if (io) {
if (!io->sections) {
io->sections = ls_newf (section_free);
}
io->sec_ids = r_id_pool_new (0, UT32_MAX);
}
}
R_API void r_io_section_fini(RIO *io) {
if (!io) {
return;
}
ls_free (io->sections);
r_id_pool_free (io->sec_ids);
io->sections = NULL;
io->sec_ids = NULL;
}
R_API int r_io_section_exists_for_id(RIO *io, ut32 id) {
SdbListIter *iter;
RIOSection *sec;
if (!io || !io->sections) {
return false;
}
ls_foreach (io->sections, iter, sec) {
if (sec->id == id) {
return true;
}
}
return false;
}
R_API RIOSection *r_io_section_add(RIO *io, ut64 paddr, ut64 vaddr, ut64 size,
ut64 vsize, int flags, const char *name,
ut32 bin_id, int fd) {
if (!io || !io->sections || !io->sec_ids || !r_io_desc_get (io, fd) ||
UT64_ADD_OVFCHK (size, paddr) || UT64_ADD_OVFCHK (size, vaddr) || !vsize) {
return NULL;
}
RIOSection *sec = R_NEW0 (RIOSection);
if (sec) {
sec->paddr = paddr;
sec->vaddr = vaddr;
sec->size = size;
sec->vsize = vsize;
sec->flags = flags;
sec->bin_id = bin_id;
sec->fd = fd;
if (!name) {
sec->name = r_str_newf ("section.0x016%"PFMT64x "", vaddr);
} else {
sec->name = strdup (name);
}
if (!sec->name) {
free (sec);
return NULL;
}
if (!r_id_pool_grab_id (io->sec_ids, &sec->id)) {
free (sec->name);
free (sec);
return NULL;
}
ls_append (io->sections, sec);
}
return sec;
}
R_API RIOSection *r_io_section_get_i(RIO *io, ut32 id) {
SdbListIter *iter;
RIOSection *s;
if (!io || !io->sections) {
return NULL;
}
ls_foreach (io->sections, iter, s) {
if (s->id == id) {
return s;
}
}
return NULL;
}
R_API int r_io_section_rm(RIO *io, ut32 id) {
SdbListIter *iter;
RIOSection *s;
if (!io || !io->sections || !io->sec_ids) {
return false;
}
ls_foreach (io->sections, iter, s) {
if (s->id == id) {
ls_delete (io->sections, iter);
r_id_pool_kick_id (io->sec_ids, id);
return true;
}
}
return false;
}
//List return does not have assigned free function only
//the list will be freed. However, caller will might hold
//an old reference
R_API SdbList *r_io_section_bin_get(RIO *io, ut32 bin_id) {
SdbList *ret = NULL;
SdbListIter *iter;
RIOSection *s;
if (!io || !io->sections) {
return NULL;
}
ls_foreach (io->sections, iter, s) {
if (s->bin_id == bin_id) {
if (!ret) {
ret = ls_new ();
}
ls_prepend (ret, s);
}
}
return ret;
}
R_API bool r_io_section_bin_rm(RIO *io, ut32 bin_id) {
RIOSection *s;
SdbListIter *iter, *niter;
int length;
if (!io || !io->sections || !io->sections->head || !io->sec_ids) {
return false;
}
length = ls_length (io->sections);
ls_foreach_safe (io->sections, iter, niter, s) {
if (s->bin_id == bin_id) {
r_id_pool_kick_id (io->sec_ids, s->id);
ls_delete (io->sections, iter);
}
}
return length != ls_length (io->sections);
}
R_API RIOSection *r_io_section_get_name(RIO *io, const char *name) {
RIOSection *s;
SdbListIter *iter;
if (!io || !name || !io->sections) {
return NULL;
}
ls_foreach (io->sections, iter, s) {
if (s->name && (!strcmp (s->name, name))) {
return s;
}
}
return NULL;
}
R_API void r_io_section_cleanup(RIO *io) {
SdbListIter *iter, *ator;
RIOSection *s;
if (!io || !io->sections || !io->sec_ids) {
return;
}
if (!io->files) {
r_io_section_fini (io);
r_io_section_init (io);
return;
}
ls_foreach_safe (io->sections, iter, ator, s) {
if (!s) {
ls_delete (io->sections, iter);
} else if (!r_io_desc_get (io, s->fd)) {
r_id_pool_kick_id (io->sec_ids, s->id);
ls_delete (io->sections, iter);
} else {
if (!r_io_map_exists_for_id (io, s->filemap)) {
s->filemap = 0;
}
if (!r_io_map_exists_for_id (io, s->memmap)) {
s->memmap = 0;
}
}
}
}
R_API SdbList *r_io_sections_get(RIO *io, ut64 paddr) {
SdbList *ret = NULL;
SdbListIter *iter;
RIOSection *s;
if (!io || !io->sections) {
return NULL;
}
ls_foreach (io->sections, iter, s) {
if (paddr >= s->paddr && paddr < (s->paddr + s->size)) {
if (!ret) {
ret = ls_new ();
}
ls_prepend (ret, s);
}
}
return ret;
}
R_API SdbList *r_io_sections_vget(RIO *io, ut64 vaddr) {
SdbList *ret = NULL;
SdbListIter *iter;
RIOSection *s;
if (!io || !io->sections) {
return NULL;
}
ls_foreach (io->sections, iter, s) {
if (vaddr >= s->vaddr && vaddr < (s->vaddr + s->vsize)) {
if (!ret) {
ret = ls_new ();
}
ls_prepend (ret, s);
}
}
return ret;
}
R_API RIOSection* r_io_section_vget(RIO *io, ut64 vaddr) {
if (io) {
SdbList *sects = r_io_sections_vget (io, vaddr);
RIOSection *ret = NULL;
if (sects) {
if (ls_length (sects)) {
ret = ls_pop (sects);
}
}
ls_free (sects);
return ret;
}
return NULL;
}
R_API RIOSection* r_io_section_get(RIO *io, ut64 vaddr) {
if (io) {
SdbList *sects = r_io_sections_get (io, vaddr);
RIOSection *ret = NULL;
if (sects) {
if (ls_length (sects)) {
ret = ls_pop (sects);
}
}
ls_free (sects);
return ret;
}
return NULL;
}
R_API ut64 r_io_section_get_vaddr_at(RIO *io, ut64 paddr) {
if (io) {
SdbList *sects = r_io_sections_vget (io, paddr);
ut64 ret = UT64_MAX;
if (sects) {
if (ls_length (sects)) {
RIOSection *s = ls_pop (sects);
ret = s->vaddr;
}
}
ls_free (sects);
return ret;
}
return UT64_MAX;
}
R_API ut64 r_io_section_get_paddr_at(RIO *io, ut64 paddr) {
if (io) {
SdbList *sects = r_io_sections_get (io, paddr);
ut64 ret = UT64_MAX;
if (sects) {
if (ls_length (sects)) {
RIOSection *s = ls_pop (sects);
ret = s->paddr;
}
}
ls_free (sects);
return ret;
}
return UT64_MAX;
}
R_API int r_io_section_set_archbits(RIO *io, ut32 id, const char *arch, int bits) {
RIOSection *s;
if (!(s = r_io_section_get_i (io, id))) {
return false;
}
if (arch) {
s->arch = r_sys_arch_id (arch);
s->bits = bits;
} else {
s->arch = s->bits = 0;
}
return true;
}
R_API const char *r_io_section_get_archbits(RIO *io, ut64 vaddr, int *bits) {
//cache section so we avoid making so many looks up
// this may crash if we remove this section, we need a way to invalidate this cache
static RIOSection *s = NULL;
if (!s || (vaddr < s->vaddr) || (s->vaddr + s->vsize < vaddr)) {
s = r_io_section_vget (io, vaddr);
}
if (!s || !s->arch || !s->bits) {
return NULL;
}
if (s) {
if (bits) {
*bits = s->bits;
}
return r_sys_arch_str (s->arch);
}
return NULL;
}
R_API int r_io_section_bin_set_archbits(RIO *io, ut32 bin_id, const char *arch, int bits) {
SdbList *bin_sections;
SdbListIter *iter;
RIOSection *s;
if (!(bin_sections = r_io_section_bin_get (io, bin_id))) {
return false;
}
int a = arch? r_sys_arch_id (arch): 0;
if (a < 1) {
a = bits = 0;
}
ls_foreach (bin_sections, iter, s) {
s->arch = a;
s->bits = bits;
}
ls_free (bin_sections);
return true;
}
R_API bool r_io_section_priorize(RIO *io, ut32 id) {
SdbListIter *iter, *niter;
RIOSection *sec;
bool ret = false;
// assuming id = 0 is invalid
if (!id) {
return false;
}
if (!io || !io->sections) {
return false;
}
ls_foreach_safe (io->sections, iter, niter, sec) {
if (sec->id == id) {
ls_split_iter (io->sections, iter);
ls_append (io->sections, sec);
ret = true;
break;
}
}
if (!ret) {
return false;
}
if (sec->filemap) {
ret = r_io_map_priorize (io, sec->filemap);
if (!sec->memmap) {
return ret;
}
}
if (sec->filemap != sec->memmap) {
return ret & r_io_map_priorize (io, sec->memmap);
}
return false;
}
R_API bool r_io_section_priorize_bin(RIO *io, ut32 bin_id) {
SdbList *secs;
SdbListIter *iter;
RIOSection *sec;
r_io_section_cleanup (io);
if (!(secs = r_io_section_bin_get (io, bin_id))) {
return false;
}
ls_foreach (secs, iter, sec) {
r_io_map_priorize (io, sec->filemap);
r_io_map_priorize (io, sec->memmap);
}
ls_free (secs);
return true;
}
static bool _section_apply_for_anal_patch(RIO *io, RIOSection *sec, bool patch) {
ut64 at;
if (sec->vsize > sec->size) {
// in that case, we just have to allocate some memory of the size (vsize-size)
if (!sec->memmap) {
// offset,where the memory should be mapped to
at = sec->vaddr + sec->size;
// TODO: harden this, handle mapslit
// craft the uri for the null-fd
if (!r_io_create_mem_map (io, sec, at, true, false)) {
return false;
}
// we need to create this map for transfering the flags, no real remapping here
if (!r_io_create_file_map (io, sec, sec->size, patch, false)) {
return false;
}
return true;
} else {
// the section is already applied
return false;
}
} else {
// same as above
if (!sec->filemap && r_io_create_file_map (io, sec, sec->vsize, patch, false)) {
return true;
}
return false;
}
}
static bool _section_apply_for_emul(RIO *io, RIOSection *sec) {
RIODesc *desc, *oldesc;
RIOMap *map;
char *uri;
size_t size;
ut8 *buf = NULL;
// if the section doesn't allow writing, we don't need to initialize writeable memory
if (!(sec->flags & R_IO_WRITE)) {
return _section_apply_for_anal_patch (io, sec, R_IO_SECTION_APPLY_FOR_ANALYSIS);
}
if (sec->memmap) {
return false;
}
size = (size_t) (sec->size > sec->vsize)? sec->vsize: sec->size;
// allocate a buffer for copying from sec->fd to the malloc-map
buf = calloc (1, size + 1);
if (!buf) {
return false;
}
// save the current desc
oldesc = io->desc;
// copy to the buffer
r_io_use_fd (io, sec->fd);
r_io_pread_at (io, sec->paddr, buf, (int)size);
// craft the uri for the opening the malloc-fd
uri = r_str_newf ("malloc://%"PFMT64u "", sec->vsize);
// open the malloc-fd and map it to vaddr
desc = r_io_open_at (io, uri, sec->flags, 664, sec->vaddr);
if (!desc) {
free (buf);
return false;
}
io->desc = desc;
// copy from buffer to the malloc-fd
r_io_pwrite_at (io, 0LL, buf, (int) size);
free (buf);
// get the malloc-map
if ((map = r_io_map_get (io, sec->vaddr))) {
map->name = r_str_newf ("mmap.%s", sec->name);
// set the flags correctly
map->flags = sec->flags;
// restore old RIODesc
io->desc = oldesc;
// let the section refere to the map
sec->filemap = sec->memmap = map->id;
return true;
}
io->desc = oldesc;
return false;
}
static bool _section_apply(RIO *io, RIOSection *sec, RIOSectionApplyMethod method) {
switch (method) {
case R_IO_SECTION_APPLY_FOR_PATCH:
case R_IO_SECTION_APPLY_FOR_ANALYSIS:
return _section_apply_for_anal_patch (io, sec,
method == R_IO_SECTION_APPLY_FOR_ANALYSIS? false : true);
case R_IO_SECTION_APPLY_FOR_EMULATOR:
return _section_apply_for_emul (io, sec);
default:
return false;
}
}
R_API bool r_io_section_apply(RIO *io, ut32 id, RIOSectionApplyMethod method) {
RIOSection *sec = r_io_section_get_i (io, id);
return sec ? _section_apply (io, sec, method): false;
}
static bool _section_reapply_anal_or_patch(RIO *io, RIOSection *sec, RIOSectionApplyMethod method) {
SdbListIter *iter;
RIOMap *map;
RIODesc* desc;
if (!sec) {
return false;
}
if (sec->memmap) {
ls_foreach (io->maps, iter, map) {
if (map->id == sec->memmap) {
desc = r_io_desc_get (io, map->fd);
if (desc && desc->plugin && desc->plugin->close) { //we can't use r_io_desc_close here, bc it breaks the section-list
desc->plugin->close (desc);
r_io_desc_del (io, map->fd);
}
sec->memmap = 0;
r_io_map_cleanup (io);
break;
}
}
}
r_io_map_del (io, sec->filemap);
sec->filemap = 0;
return _section_apply (io, sec, method);
}
static bool _section_reapply_for_emul(RIO *io, RIOSection *sec) {
RIOMap *map = NULL;
SdbListIter *iter;
char *uri;
ut8 *buf = NULL;
size_t size;
RIODesc *desc, *oldesc;
// in this case the section was applied for patching
if (sec->filemap != sec->memmap) {
if (!sec->memmap) {
r_io_map_del (io, sec->filemap);
sec->filemap = 0;
return _section_apply (io, sec, R_IO_SECTION_APPLY_FOR_EMULATOR);
}
ls_foreach (io->maps, iter, map) {
if (map->id == sec->memmap) {
break;
}
}
if (!map) {
r_io_map_del (io, sec->filemap);
sec->filemap = sec->memmap = 0;
return _section_apply (io, sec, R_IO_SECTION_APPLY_FOR_EMULATOR);
}
size = map->itv.size;
buf = calloc (1, size + 1);
if (!buf) {
return false;
}
oldesc = io->desc;
r_io_use_fd (io, map->fd);
r_io_pread_at (io, map->delta, buf, (int) size);
r_io_fd_close (io, map->fd);
if (sec->size > sec->vsize) {
size = 0;
} else if (size > (size_t) (sec->vsize - sec->size)) {
size = (size_t) (sec->vsize - sec->size);
}
uri = r_str_newf ("malloc://%"PFMT64u, sec->vsize);
r_io_open_at (io, uri, sec->flags | R_IO_WRITE, 664, sec->vaddr);
map = r_io_map_get (io, sec->vaddr);
if (map) {
(void)r_io_use_fd (io, map->fd);
}
r_io_pwrite_at (io, sec->size, buf, (int) size);
free (buf);
if (sec->size > sec->vsize) {
size = (size_t) sec->vsize;
} else {
size = (size_t) sec->size;
}
buf = calloc (1, size + 1);
if (!buf) {
io->desc = oldesc;
return false;
}
r_io_use_fd (io, sec->fd);
r_io_pread_at (io, sec->paddr, buf, (int) size);
if (map) {
(void)r_io_use_fd (io, map->fd);
sec->filemap = sec->memmap = map->id;
}
r_io_pwrite_at (io, 0LL, buf, (int) size);
free (buf);
io->desc = oldesc;
return true;
}
if (!sec->filemap) {
return _section_apply (io, sec, R_IO_SECTION_APPLY_FOR_EMULATOR);
}
ls_foreach (io->maps, iter, map) {
if (map->id == sec->memmap) {
break;
}
}
if (!map) {
return _section_apply (io, sec, R_IO_SECTION_APPLY_FOR_EMULATOR);
}
size = map->itv.size;
//save desc to restore later
desc = io->desc;
r_io_use_fd (io, map->fd);
buf = calloc (1, size + 1);
if (!buf) {
io->desc = desc;
return false;
}
r_io_pread_at (io, map->delta, buf, (int) size);
r_io_fd_close (io, map->fd);
r_io_map_cleanup (io);
if (sec->vsize < (ut64) size) {
size = (size_t) sec->vsize;
}
uri = r_str_newf ("malloc://%"PFMT64u, sec->vsize);
r_io_open_at (io, uri, sec->flags | R_IO_WRITE, 664, sec->vaddr);
map = r_io_map_get (io, sec->vaddr);
r_io_use_fd (io, map->fd);
r_io_pwrite_at (io, 0LL, buf, (int) size);
free (buf);
map->flags = sec->flags;
io->desc = desc;
return true;
}
static bool _section_reapply(RIO *io, RIOSection *sec, RIOSectionApplyMethod method) {
r_io_map_cleanup (io);
switch (method) {
case R_IO_SECTION_APPLY_FOR_PATCH:
case R_IO_SECTION_APPLY_FOR_ANALYSIS:
return _section_reapply_anal_or_patch (io, sec, method);
case R_IO_SECTION_APPLY_FOR_EMULATOR:
return _section_reapply_for_emul (io, sec);
default:
return false;
}
}
R_API bool r_io_section_apply_bin(RIO *io, ut32 bin_id, RIOSectionApplyMethod method) {
RIOSection *sec;
SdbListIter *iter;
bool ret = false;
if (!io || !io->sections) {
return false;
}
ls_foreach_prev (io->sections, iter, sec) {
if (sec && (sec->bin_id == bin_id)) {
ret = true;
_section_apply (io, sec, method);
}
}
r_io_map_calculate_skyline (io);
return ret;
}
R_API bool r_io_section_reapply(RIO *io, ut32 id, RIOSectionApplyMethod method) {
RIOSection *sec;
if (!io || !io->sections || !io->maps) {
return false;
}
if (!(sec = r_io_section_get_i (io, id))) {
return false;
}
bool ret = _section_reapply (io, sec, method);
r_io_map_calculate_skyline (io);
return ret;
}
R_API bool r_io_section_reapply_bin(RIO *io, ut32 binid, RIOSectionApplyMethod method) {
RIOSection *sec;
SdbListIter *iter;
bool ret = false;
if (!io || !io->sections) {
return false;
}
ls_foreach_prev (io->sections, iter, sec) {
if (sec && (sec->bin_id == binid)) {
ret = true;
_section_reapply (io, sec, method);
}
}
r_io_map_calculate_skyline (io);
return ret;
}