mirror of
https://github.com/radareorg/radare2.git
synced 2025-02-12 17:48:33 +00:00
232 lines
5.7 KiB
C
232 lines
5.7 KiB
C
/* radare - LGPL - Copyright 2009-2012 nibble<.ds@gmail.com>, pancake<nopcode.org> */
|
|
|
|
#include <r_types.h>
|
|
#include <r_util.h>
|
|
#include <r_lib.h>
|
|
#include <r_bin.h>
|
|
#include "../format/p9/p9bin.h"
|
|
|
|
static int check(RBinArch *arch) {
|
|
if (arch && arch->buf && arch->buf->buf)
|
|
return (r_bin_p9_get_arch (arch->buf->buf, NULL, NULL));
|
|
return R_FALSE;
|
|
}
|
|
|
|
static int load(RBinArch *arch) {
|
|
return check(arch);
|
|
}
|
|
|
|
static int destroy (RBinArch *arch) {
|
|
return R_TRUE;
|
|
}
|
|
|
|
static ut64 baddr(RBinArch *arch) {
|
|
return 0x1000000; // XXX
|
|
}
|
|
|
|
static RBinAddr* binsym(RBinArch *arch, int type) {
|
|
return NULL; // TODO
|
|
}
|
|
|
|
static RList* entries(RBinArch *arch) {
|
|
RList* ret;
|
|
RBinAddr *ptr = NULL;
|
|
|
|
if (!(ret = r_list_new ()))
|
|
return NULL;
|
|
ret->free = free;
|
|
if ((ptr = R_NEW (RBinAddr))) {
|
|
ptr->offset = 8*4;
|
|
ptr->rva = 8*4;// + baddr (arch);
|
|
r_list_append (ret, ptr);
|
|
}
|
|
return ret;
|
|
}
|
|
|
|
static RList* sections(RBinArch *arch) {
|
|
RList *ret = NULL;
|
|
RBinSection *ptr = NULL;
|
|
ut64 textsize, datasize, symssize, spszsize, pcszsize;
|
|
int big_endian = arch->o->info->big_endian;
|
|
|
|
if (!(ret = r_list_new ()))
|
|
return NULL;
|
|
ret->free = free;
|
|
|
|
// add text segment
|
|
textsize = r_mem_get_num (arch->buf->buf+4, 4, big_endian);
|
|
if (!(ptr = R_NEW0 (RBinSection)))
|
|
return ret;
|
|
strncpy (ptr->name, "text", R_BIN_SIZEOF_STRINGS);
|
|
ptr->size = textsize;
|
|
ptr->vsize = textsize + (textsize%4096);
|
|
ptr->offset = 8*4;
|
|
ptr->rva = ptr->offset;
|
|
ptr->srwx = 5; // r-x
|
|
r_list_append (ret, ptr);
|
|
// add data segment
|
|
datasize = r_mem_get_num (arch->buf->buf+8, 4, big_endian);
|
|
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->offset = textsize+(8*4);
|
|
ptr->rva = ptr->offset;
|
|
ptr->srwx = 6; // rw-
|
|
r_list_append (ret, ptr);
|
|
}
|
|
// ignore bss or what
|
|
// add syms segment
|
|
symssize = r_mem_get_num (arch->buf->buf+16, 4, big_endian);
|
|
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->offset = datasize+textsize+(8*4);
|
|
ptr->rva = ptr->offset;
|
|
ptr->srwx = 4; // r--
|
|
r_list_append (ret, ptr);
|
|
}
|
|
// add spsz segment
|
|
spszsize = r_mem_get_num (arch->buf->buf+24, 4, big_endian);
|
|
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->offset = symssize+datasize+textsize+(8*4);
|
|
ptr->rva = ptr->offset;
|
|
ptr->srwx = 4; // r--
|
|
r_list_append (ret, ptr);
|
|
}
|
|
// add pcsz segment
|
|
pcszsize = r_mem_get_num (arch->buf->buf+24, 4, big_endian);
|
|
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->offset = spszsize+symssize+datasize+textsize+(8*4);
|
|
ptr->rva = ptr->offset;
|
|
ptr->srwx = 4; // r--
|
|
r_list_append (ret, ptr);
|
|
}
|
|
return ret;
|
|
}
|
|
|
|
static RList* symbols(RBinArch *arch) {
|
|
// TODO: parse symbol table
|
|
return NULL;
|
|
}
|
|
|
|
static RList* imports(RBinArch *arch) {
|
|
return NULL;
|
|
}
|
|
|
|
static RList* libs(RBinArch *arch) {
|
|
return NULL;
|
|
}
|
|
|
|
static RBinInfo* info(RBinArch *arch) {
|
|
const char *archstr;
|
|
RBinInfo *ret = NULL;
|
|
int big_endian = 0;
|
|
int bits = 32;
|
|
int bina;
|
|
|
|
if (!(bina = r_bin_p9_get_arch (arch->buf->buf, &bits, &big_endian)))
|
|
return NULL;
|
|
if ((ret = R_NEW0 (RBinInfo)) == NULL)
|
|
return NULL;
|
|
strncpy (ret->file, arch->file, R_BIN_SIZEOF_STRINGS);
|
|
strncpy (ret->rpath, "NONE", R_BIN_SIZEOF_STRINGS);
|
|
strncpy (ret->bclass, "program", R_BIN_SIZEOF_STRINGS);
|
|
strncpy (ret->rclass, "p9", R_BIN_SIZEOF_STRINGS);
|
|
strncpy (ret->os, "plan9", R_BIN_SIZEOF_STRINGS);
|
|
archstr = r_sys_arch_str (bina);
|
|
strncpy (ret->arch, archstr, R_BIN_SIZEOF_STRINGS);
|
|
strncpy (ret->machine, archstr, R_BIN_SIZEOF_STRINGS);
|
|
strncpy (ret->subsystem, "plan9", R_BIN_SIZEOF_STRINGS);
|
|
strncpy (ret->type, "EXEC (Executable file)", R_BIN_SIZEOF_STRINGS);
|
|
ret->bits = bits;
|
|
ret->has_va = R_TRUE;
|
|
ret->big_endian = big_endian;
|
|
ret->dbg_info = 0;
|
|
ret->dbg_info = 0;
|
|
return ret;
|
|
}
|
|
|
|
static int size(RBinArch *arch) {
|
|
ut64 text, data, syms, spsz;
|
|
int big_endian;
|
|
if (!arch->o->info)
|
|
arch->o->info = info (arch);
|
|
big_endian = arch->o->info->big_endian;
|
|
// TODO: reuse section list
|
|
text = r_mem_get_num (arch->buf->buf+4, 4, big_endian);
|
|
data = r_mem_get_num (arch->buf->buf+8, 4, big_endian);
|
|
syms = r_mem_get_num (arch->buf->buf+16, 4, big_endian);
|
|
spsz = r_mem_get_num (arch->buf->buf+24, 4, big_endian);
|
|
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",
|
|
.init = NULL,
|
|
.fini = NULL,
|
|
.load = &load,
|
|
.size = &size,
|
|
.destroy = &destroy,
|
|
.check = &check,
|
|
.baddr = &baddr,
|
|
.binsym = &binsym,
|
|
.entries = &entries,
|
|
.sections = §ions,
|
|
.symbols = &symbols,
|
|
.imports = &imports,
|
|
.strings = NULL,
|
|
.info = &info,
|
|
.fields = NULL,
|
|
.libs = &libs,
|
|
.relocs = NULL,
|
|
.meta = NULL,
|
|
.write = NULL,
|
|
.create = &create,
|
|
};
|
|
|
|
#ifndef CORELIB
|
|
struct r_lib_struct_t radare_plugin = {
|
|
.type = R_LIB_TYPE_BIN,
|
|
.data = &r_bin_plugin_pe
|
|
};
|
|
#endif
|
|
#endif
|