/* radare2 - LGPL - Copyright 2009-2015 - nibble, pancake */ #include #include #include #include #include "../format/p9/p9bin.h" static int check(RBinFile *arch); static int check_bytes(const ut8 *buf, ut64 length); static int check(RBinFile *arch) { const ut8 *bytes = arch ? r_buf_buffer (arch->buf) : NULL; ut64 sz = arch ? r_buf_size (arch->buf): 0; return check_bytes (bytes, sz); } static int check_bytes(const ut8 *buf, ut64 length) { if (buf && length >= 4) return (r_bin_p9_get_arch (buf, NULL, NULL)); return false; } static Sdb* get_sdb (RBinObject *o) { if (!o) return NULL; //struct r_bin_[NAME]_obj_t *bin = (struct r_bin_r_bin_[NAME]_obj_t *) o->bin_obj; //if (bin->kv) return kv; return NULL; } static void * load_bytes(RBinFile *arch, const ut8 *buf, ut64 sz, ut64 loadaddr, Sdb *sdb){ return (void*)(size_t)check_bytes (buf, sz); } static int load(RBinFile *arch) { return check(arch); } 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_NEW (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; int big_endian = 0; ut64 textsize, datasize, symssize, spszsize, pcszsize; if (!arch->o->info) return NULL; 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->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, 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->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, 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->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, 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->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, 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->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)) == NULL) 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 int size(RBinFile *arch) { ut64 text, data, syms, spsz; int big_endian; if (!arch->o->info) arch->o->info = info (arch); if (!arch->o->info) return 0; 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", .license = "LGPL3", .get_sdb = &get_sdb, .load = &load, .load_bytes = &load_bytes, .size = &size, .destroy = &destroy, .check = &check, .check_bytes = &check_bytes, .baddr = &baddr, .binsym = &binsym, .entries = &entries, .sections = §ions, .symbols = &symbols, .imports = &imports, .info = &info, .libs = &libs, .create = &create, }; #ifndef CORELIB struct r_lib_struct_t radare_plugin = { .type = R_LIB_TYPE_BIN, .data = &r_bin_plugin_p9, .version = R2_VERSION }; #endif #endif