radare2/libr/bin/p/bin_p9.c

256 lines
6.1 KiB
C

/* radare2 - LGPL - Copyright 2009-2017 - nibble, pancake */
#include <r_types.h>
#include <r_util.h>
#include <r_lib.h>
#include <r_bin.h>
#include "../format/p9/p9bin.h"
static bool check_bytes(const ut8 *buf, ut64 length) {
if (buf && length >= 4) {
return (r_bin_p9_get_arch (buf, NULL, NULL));
}
return false;
}
static void *load_bytes(RBinFile *arch, const ut8 *buf, ut64 sz, ut64 loadaddr, Sdb *sdb){
return (void *) (size_t) check_bytes (buf, sz);
}
static bool load(RBinFile *arch) {
const ut8 *bytes = arch? r_buf_buffer (arch->buf): NULL;
ut64 sz = arch? r_buf_size (arch->buf): 0;
ut64 la = (arch && arch->o)? arch->o->loadaddr: 0;
return load_bytes (arch, bytes, sz, la, arch? arch->sdb: NULL);
}
static int destroy(RBinFile *arch) {
return true;
}
static ut64 baddr(RBinFile *arch) {
return 0x1000000; // XXX
}
static RBinAddr *binsym(RBinFile *arch, int type) {
return NULL; // TODO
}
static RList *entries(RBinFile *arch) {
RList *ret;
RBinAddr *ptr = NULL;
if (!(ret = r_list_new ())) {
return NULL;
}
ret->free = free;
if ((ptr = R_NEW0 (RBinAddr))) {
ptr->paddr = 8 * 4;
ptr->vaddr = 8 * 4;// + baddr (arch);
r_list_append (ret, ptr);
}
return ret;
}
static RList *sections(RBinFile *arch) {
RList *ret = NULL;
RBinSection *ptr = NULL;
ut64 textsize, datasize, symssize, spszsize, pcszsize;
if (!arch->o->info) {
return NULL;
}
if (!(ret = r_list_newf ((RListFree)free))) {
return NULL;
}
if (r_buf_size (arch->buf) < 28) {
r_list_free (ret);
return NULL;
}
// add text segment
textsize = r_mem_get_num (arch->buf->buf + 4, 4);
if (!(ptr = R_NEW0 (RBinSection))) {
r_list_free (ret);
return NULL;
}
strncpy (ptr->name, "text", R_BIN_SIZEOF_STRINGS);
ptr->size = textsize;
ptr->vsize = textsize + (textsize % 4096);
ptr->paddr = 8 * 4;
ptr->vaddr = ptr->paddr;
ptr->srwx = R_BIN_SCN_READABLE | R_BIN_SCN_EXECUTABLE | R_BIN_SCN_MAP; // r-x
ptr->add = true;
r_list_append (ret, ptr);
// add data segment
datasize = r_mem_get_num (arch->buf->buf + 8, 4);
if (datasize > 0) {
if (!(ptr = R_NEW0 (RBinSection))) {
return ret;
}
strncpy (ptr->name, "data", R_BIN_SIZEOF_STRINGS);
ptr->size = datasize;
ptr->vsize = datasize + (datasize % 4096);
ptr->paddr = textsize + (8 * 4);
ptr->vaddr = ptr->paddr;
ptr->srwx = R_BIN_SCN_READABLE | R_BIN_SCN_WRITABLE | R_BIN_SCN_MAP; // rw-
ptr->add = true;
r_list_append (ret, ptr);
}
// ignore bss or what
// add syms segment
symssize = r_mem_get_num (arch->buf->buf + 16, 4);
if (symssize) {
if (!(ptr = R_NEW0 (RBinSection))) {
return ret;
}
strncpy (ptr->name, "syms", R_BIN_SIZEOF_STRINGS);
ptr->size = symssize;
ptr->vsize = symssize + (symssize % 4096);
ptr->paddr = datasize + textsize + (8 * 4);
ptr->vaddr = ptr->paddr;
ptr->srwx = R_BIN_SCN_READABLE | R_BIN_SCN_MAP; // r--
ptr->add = true;
r_list_append (ret, ptr);
}
// add spsz segment
spszsize = r_mem_get_num (arch->buf->buf + 24, 4);
if (spszsize) {
if (!(ptr = R_NEW0 (RBinSection))) {
return ret;
}
strncpy (ptr->name, "spsz", R_BIN_SIZEOF_STRINGS);
ptr->size = spszsize;
ptr->vsize = spszsize + (spszsize % 4096);
ptr->paddr = symssize + datasize + textsize + (8 * 4);
ptr->vaddr = ptr->paddr;
ptr->srwx = R_BIN_SCN_READABLE | R_BIN_SCN_MAP; // r--
ptr->add = true;
r_list_append (ret, ptr);
}
// add pcsz segment
pcszsize = r_mem_get_num (arch->buf->buf + 24, 4);
if (pcszsize) {
if (!(ptr = R_NEW0 (RBinSection))) {
return ret;
}
strncpy (ptr->name, "pcsz", R_BIN_SIZEOF_STRINGS);
ptr->size = pcszsize;
ptr->vsize = pcszsize + (pcszsize % 4096);
ptr->paddr = spszsize + symssize + datasize + textsize + (8 * 4);
ptr->vaddr = ptr->paddr;
ptr->srwx = R_BIN_SCN_READABLE | R_BIN_SCN_MAP; // r--
ptr->add = true;
r_list_append (ret, ptr);
}
return ret;
}
static RList *symbols(RBinFile *arch) {
// TODO: parse symbol table
return NULL;
}
static RList *imports(RBinFile *arch) {
return NULL;
}
static RList *libs(RBinFile *arch) {
return NULL;
}
static RBinInfo *info(RBinFile *arch) {
RBinInfo *ret = NULL;
int bits = 32, bina, big_endian = 0;
if (!(bina = r_bin_p9_get_arch (arch->buf->buf, &bits, &big_endian))) {
return NULL;
}
if (!(ret = R_NEW0 (RBinInfo))) {
return NULL;
}
ret->file = strdup (arch->file);
ret->bclass = strdup ("program");
ret->rclass = strdup ("p9");
ret->os = strdup ("Plan9");
ret->arch = strdup (r_sys_arch_str (bina));
ret->machine = strdup (ret->arch);
ret->subsystem = strdup ("plan9");
ret->type = strdup ("EXEC (executable file)");
ret->bits = bits;
ret->has_va = true;
ret->big_endian = big_endian;
ret->dbg_info = 0;
ret->dbg_info = 0;
return ret;
}
static ut64 size(RBinFile *arch) {
ut64 text, data, syms, spsz;
if (!arch->o->info) {
arch->o->info = info (arch);
}
if (!arch->o->info) {
return 0;
}
// TODO: reuse section list
if (r_buf_size (arch->buf) < 28) {
return 0;
}
text = r_mem_get_num (arch->buf->buf + 4, 4);
data = r_mem_get_num (arch->buf->buf + 8, 4);
syms = r_mem_get_num (arch->buf->buf + 16, 4);
spsz = r_mem_get_num (arch->buf->buf + 24, 4);
return text + data + syms + spsz + (6 * 4);
}
#if !R_BIN_P9
/* inspired in http://www.phreedom.org/solar/code/tinype/tiny.97/tiny.asm */
static RBuffer *create(RBin *bin, const ut8 *code, int codelen, const ut8 *data, int datalen) {
RBuffer *buf = r_buf_new ();
#define B(x, y) r_buf_append_bytes (buf, (const ut8 *) x, y)
#define D(x) r_buf_append_ut32 (buf, x)
D (I_MAGIC); // i386 only atm
D (codelen);
D (datalen);
D (4096); // bss
D (0); // syms
D (8 * 4); // entry
D (4096); // spsz
D (4096); // pcsz
B (code, codelen);
if (datalen > 0) {
B (data, datalen);
}
return buf;
}
struct r_bin_plugin_t r_bin_plugin_p9 = {
.name = "p9",
.desc = "Plan9 bin plugin",
.license = "LGPL3",
.load = &load,
.load_bytes = &load_bytes,
.size = &size,
.destroy = &destroy,
.check_bytes = &check_bytes,
.baddr = &baddr,
.binsym = &binsym,
.entries = &entries,
.sections = &sections,
.symbols = &symbols,
.imports = &imports,
.info = &info,
.libs = &libs,
.create = &create,
};
#ifndef CORELIB
RLibStruct radare_plugin = {
.type = R_LIB_TYPE_BIN,
.data = &r_bin_plugin_p9,
.version = R2_VERSION
};
#endif
#endif