mirror of
https://github.com/radareorg/radare2.git
synced 2025-01-15 02:00:31 +00:00
385 lines
8.7 KiB
C
385 lines
8.7 KiB
C
/* radare - LGPL - Copyright 2014-2017 - Fedor Sakharov */
|
|
|
|
#include <r_types.h>
|
|
#include <r_util.h>
|
|
#include <r_lib.h>
|
|
#include <r_bin.h>
|
|
|
|
#include "coff/coff.h"
|
|
|
|
static Sdb* get_sdb(RBinFile *bf) {
|
|
RBinObject *o = bf->o;
|
|
if (!o) {
|
|
return NULL;
|
|
}
|
|
struct r_bin_coff_obj *bin = (struct r_bin_coff_obj *) o->bin_obj;
|
|
if (bin->kv) {
|
|
return bin->kv;
|
|
}
|
|
return NULL;
|
|
}
|
|
|
|
static void * load_bytes(RBinFile *arch, const ut8 *buf, ut64 sz, ut64 loadaddr, Sdb *sdb) {
|
|
if (!buf || !sz || sz == UT64_MAX) {
|
|
return NULL;
|
|
}
|
|
RBuffer *tbuf = r_buf_new();
|
|
r_buf_set_bytes (tbuf, buf, sz);
|
|
void *res = r_bin_coff_new_buf (tbuf, arch->rbin->verbose);
|
|
r_buf_free (tbuf);
|
|
return res;
|
|
}
|
|
|
|
static bool load(RBinFile *arch) {
|
|
const ut8 *bytes = arch ? r_buf_buffer (arch->buf) : NULL;
|
|
ut64 sz = arch ? r_buf_size (arch->buf): 0;
|
|
|
|
if (!arch || !arch->o) {
|
|
return false;
|
|
}
|
|
arch->o->bin_obj = load_bytes (arch, bytes, sz, arch->o->loadaddr, arch->sdb);
|
|
return arch->o->bin_obj ? true: false;
|
|
}
|
|
|
|
static int destroy(RBinFile *arch) {
|
|
r_bin_coff_free((struct r_bin_coff_obj*)arch->o->bin_obj);
|
|
return true;
|
|
}
|
|
|
|
static ut64 baddr(RBinFile *arch) {
|
|
return 0;
|
|
}
|
|
|
|
static RBinAddr *binsym(RBinFile *arch, int sym) {
|
|
return NULL;
|
|
}
|
|
|
|
static bool _fill_bin_symbol(struct r_bin_coff_obj *bin, int idx, RBinSymbol **sym) {
|
|
RBinSymbol *ptr = *sym;
|
|
char * coffname = NULL;
|
|
struct coff_symbol *s = NULL;
|
|
if (idx < 0 || idx > bin->hdr.f_nsyms) {
|
|
return false;
|
|
}
|
|
if (!bin->symbols) {
|
|
return false;
|
|
}
|
|
s = &bin->symbols[idx];
|
|
coffname = r_coff_symbol_name (bin, s);
|
|
if (!coffname) {
|
|
return false;
|
|
}
|
|
ptr->name = strdup (coffname);
|
|
free (coffname);
|
|
ptr->forwarder = r_str_const ("NONE");
|
|
|
|
switch (s->n_sclass) {
|
|
case COFF_SYM_CLASS_FUNCTION:
|
|
ptr->type = r_str_const ("FUNC");
|
|
break;
|
|
case COFF_SYM_CLASS_FILE:
|
|
ptr->type = r_str_const ("FILE");
|
|
break;
|
|
case COFF_SYM_CLASS_SECTION:
|
|
ptr->type = r_str_const ("SECTION");
|
|
break;
|
|
case COFF_SYM_CLASS_EXTERNAL:
|
|
ptr->type = r_str_const ("EXTERNAL");
|
|
break;
|
|
case COFF_SYM_CLASS_STATIC:
|
|
ptr->type = r_str_const ("STATIC");
|
|
break;
|
|
default:
|
|
ptr->type = r_str_const (sdb_fmt (0, "%i", s->n_sclass));
|
|
break;
|
|
}
|
|
if (bin->symbols[idx].n_scnum < bin->hdr.f_nscns &&
|
|
bin->symbols[idx].n_scnum > 0) {
|
|
//first index is 0 that is why -1
|
|
ptr->paddr = bin->scn_hdrs[s->n_scnum - 1].s_scnptr + s->n_value;
|
|
}
|
|
ptr->size = 4;
|
|
ptr->ordinal = 0;
|
|
return true;
|
|
}
|
|
|
|
static RList *entries(RBinFile *arch) {
|
|
struct r_bin_coff_obj *obj = (struct r_bin_coff_obj*)arch->o->bin_obj;
|
|
RList *ret;
|
|
RBinAddr *ptr = NULL;
|
|
if (!(ret = r_list_newf (free))) {
|
|
return NULL;
|
|
}
|
|
ptr = r_coff_get_entry(obj);
|
|
r_list_append(ret, ptr);
|
|
return ret;
|
|
}
|
|
|
|
static RList *sections(RBinFile *arch) {
|
|
char *tmp, *coffname = NULL;
|
|
size_t i;
|
|
RList *ret = NULL;
|
|
RBinSection *ptr = NULL;
|
|
struct r_bin_coff_obj *obj = (struct r_bin_coff_obj*)arch->o->bin_obj;
|
|
|
|
ret = r_list_newf (free);
|
|
if (!ret) {
|
|
return NULL;
|
|
}
|
|
if (obj && obj->scn_hdrs) {
|
|
for (i = 0; i < obj->hdr.f_nscns; i++) {
|
|
free (coffname);
|
|
tmp = r_coff_symbol_name (obj, &obj->scn_hdrs[i]);
|
|
if (!tmp) {
|
|
r_list_free (ret);
|
|
return NULL;
|
|
}
|
|
//IO does not like sections with the same name append idx
|
|
//since it will update it
|
|
coffname = r_str_newf ("%s-%d", tmp, i);
|
|
free (tmp);
|
|
ptr = R_NEW0 (RBinSection);
|
|
if (!ptr) {
|
|
free (coffname);
|
|
return ret;
|
|
}
|
|
strncpy (ptr->name, coffname, R_BIN_SIZEOF_STRINGS);
|
|
if (strstr (ptr->name, "data")) {
|
|
ptr->is_data = true;
|
|
}
|
|
ptr->size = obj->scn_hdrs[i].s_size;
|
|
ptr->vsize = obj->scn_hdrs[i].s_size;
|
|
ptr->paddr = obj->scn_hdrs[i].s_scnptr;
|
|
ptr->add = true;
|
|
ptr->srwx = R_BIN_SCN_MAP;
|
|
if (obj->scn_hdrs[i].s_flags & COFF_SCN_MEM_READ) {
|
|
ptr->srwx |= R_BIN_SCN_READABLE;
|
|
}
|
|
if (obj->scn_hdrs[i].s_flags & COFF_SCN_MEM_WRITE) {
|
|
ptr->srwx |= R_BIN_SCN_WRITABLE;
|
|
}
|
|
if (obj->scn_hdrs[i].s_flags & COFF_SCN_MEM_EXECUTE) {
|
|
ptr->srwx |= R_BIN_SCN_EXECUTABLE;
|
|
}
|
|
r_list_append (ret, ptr);
|
|
}
|
|
}
|
|
free (coffname);
|
|
return ret;
|
|
}
|
|
|
|
static RList *symbols(RBinFile *arch) {
|
|
int i;
|
|
RList *ret = NULL;
|
|
RBinSymbol *ptr = NULL;
|
|
struct r_bin_coff_obj *obj = (struct r_bin_coff_obj*)arch->o->bin_obj;
|
|
if (!(ret = r_list_new ())) {
|
|
return ret;
|
|
}
|
|
ret->free = free;
|
|
if (obj->symbols) {
|
|
for (i = 0; i < obj->hdr.f_nsyms; i++) {
|
|
if (!(ptr = R_NEW0 (RBinSymbol))) {
|
|
break;
|
|
}
|
|
if (_fill_bin_symbol (obj, i, &ptr)) {
|
|
r_list_append (ret, ptr);
|
|
}
|
|
i += obj->symbols[i].n_numaux;
|
|
}
|
|
}
|
|
return ret;
|
|
}
|
|
|
|
static RList *imports(RBinFile *arch) {
|
|
return NULL;
|
|
}
|
|
|
|
static RList *libs(RBinFile *arch) {
|
|
return NULL;
|
|
}
|
|
|
|
static RList *relocs(RBinFile *arch) {
|
|
struct r_bin_coff_obj *bin = (struct r_bin_coff_obj*)arch->o->bin_obj;
|
|
RBinReloc *reloc;
|
|
struct coff_reloc *rel;
|
|
int j, i = 0;
|
|
RList *list_rel;
|
|
list_rel = r_list_new ();
|
|
if (!list_rel || !bin || !bin->scn_hdrs) {
|
|
r_list_free (list_rel);
|
|
return NULL;
|
|
}
|
|
for (i = 0; i < bin->hdr.f_nscns; i++) {
|
|
if (bin->scn_hdrs[i].s_nreloc) {
|
|
int len = 0, size = bin->scn_hdrs[i].s_nreloc * sizeof (struct coff_reloc);
|
|
if (size < 0) {
|
|
return list_rel;
|
|
}
|
|
rel = calloc (1, size + sizeof (struct coff_reloc));
|
|
if (!rel) {
|
|
return list_rel;
|
|
}
|
|
if (bin->scn_hdrs[i].s_relptr > bin->size ||
|
|
bin->scn_hdrs[i].s_relptr + size > bin->size) {
|
|
free (rel);
|
|
return list_rel;
|
|
}
|
|
len = r_buf_read_at (bin->b, bin->scn_hdrs[i].s_relptr, (ut8*)rel, size);
|
|
if (len != size) {
|
|
free (rel);
|
|
return list_rel;
|
|
}
|
|
for (j = 0; j < bin->scn_hdrs[i].s_nreloc; j++) {
|
|
RBinSymbol *symbol = R_NEW0 (RBinSymbol);
|
|
if (!symbol) {
|
|
continue;
|
|
}
|
|
if (!_fill_bin_symbol (bin, rel[j].r_symndx, &symbol)) {
|
|
free (symbol);
|
|
continue;
|
|
}
|
|
reloc = R_NEW0 (RBinReloc);
|
|
if (!reloc) {
|
|
free (symbol);
|
|
continue;
|
|
}
|
|
reloc->type = rel[j].r_type; //XXX the type if different from what r2 expects
|
|
reloc->symbol = symbol;
|
|
reloc->paddr = bin->scn_hdrs[i].s_scnptr + rel[j].r_vaddr;
|
|
reloc->vaddr = reloc->paddr;
|
|
r_list_append (list_rel, reloc);
|
|
}
|
|
free (rel);
|
|
}
|
|
}
|
|
return list_rel;
|
|
}
|
|
|
|
static RBinInfo *info(RBinFile *arch) {
|
|
RBinInfo *ret = R_NEW0(RBinInfo);
|
|
struct r_bin_coff_obj *obj = (struct r_bin_coff_obj*)arch->o->bin_obj;
|
|
|
|
ret->file = arch->file? strdup (arch->file): NULL;
|
|
ret->rclass = strdup ("coff");
|
|
ret->bclass = strdup ("coff");
|
|
ret->type = strdup ("COFF (Executable file)");
|
|
ret->os = strdup ("any");
|
|
ret->subsystem = strdup ("any");
|
|
ret->big_endian = obj->endian;
|
|
ret->has_va = false;
|
|
ret->dbg_info = 0;
|
|
|
|
if (r_coff_is_stripped (obj)) {
|
|
ret->dbg_info |= R_BIN_DBG_STRIPPED;
|
|
} else {
|
|
if (!(obj->hdr.f_flags & COFF_FLAGS_TI_F_RELFLG)) {
|
|
ret->dbg_info |= R_BIN_DBG_RELOCS;
|
|
}
|
|
if (!(obj->hdr.f_flags & COFF_FLAGS_TI_F_LNNO)) {
|
|
ret->dbg_info |= R_BIN_DBG_LINENUMS;
|
|
}
|
|
if (!(obj->hdr.f_flags & COFF_FLAGS_TI_F_EXEC)) {
|
|
ret->dbg_info |= R_BIN_DBG_SYMS;
|
|
}
|
|
}
|
|
|
|
switch (obj->hdr.f_magic) {
|
|
case COFF_FILE_MACHINE_I386:
|
|
ret->machine = strdup ("i386");
|
|
ret->arch = strdup ("x86");
|
|
ret->bits = 32;
|
|
break;
|
|
case COFF_FILE_MACHINE_AMD64:
|
|
ret->machine = strdup ("AMD64");
|
|
ret->arch = strdup ("x86");
|
|
ret->bits = 64;
|
|
break;
|
|
case COFF_FILE_MACHINE_H8300:
|
|
ret->machine = strdup ("H8300");
|
|
ret->arch = strdup ("h8300");
|
|
ret->bits = 16;
|
|
break;
|
|
case COFF_FILE_TI_COFF:
|
|
switch (obj->target_id) {
|
|
case COFF_FILE_MACHINE_TMS320C54:
|
|
ret->machine = strdup ("c54x");
|
|
ret->arch = strdup ("tms320");
|
|
ret->bits = 32;
|
|
break;
|
|
case COFF_FILE_MACHINE_TMS320C55:
|
|
ret->machine = strdup ("c55x");
|
|
ret->arch = strdup ("tms320");
|
|
ret->bits = 32;
|
|
break;
|
|
case COFF_FILE_MACHINE_TMS320C55PLUS:
|
|
ret->machine = strdup ("c55x+");
|
|
ret->arch = strdup ("tms320");
|
|
ret->bits = 32;
|
|
break;
|
|
}
|
|
break;
|
|
default:
|
|
ret->machine = strdup ("unknown");
|
|
}
|
|
|
|
return ret;
|
|
}
|
|
|
|
static RList *fields(RBinFile *arch) {
|
|
return NULL;
|
|
}
|
|
|
|
|
|
static ut64 size(RBinFile *arch) {
|
|
return 0;
|
|
}
|
|
|
|
static bool check_bytes(const ut8 *buf, ut64 length) {
|
|
#if 0
|
|
TODO: do more checks here to avoid false positives
|
|
|
|
ut16 MACHINE
|
|
ut16 NSECTIONS
|
|
ut32 DATE
|
|
ut32 PTRTOSYMTABLE
|
|
ut32 NUMOFSYMS
|
|
ut16 OPTHDRSIZE
|
|
ut16 CHARACTERISTICS
|
|
#endif
|
|
if (buf && length >= 20) {
|
|
return r_coff_supported_arch (buf);
|
|
}
|
|
return false;
|
|
}
|
|
|
|
RBinPlugin r_bin_plugin_coff = {
|
|
.name = "coff",
|
|
.desc = "COFF format r_bin plugin",
|
|
.license = "LGPL3",
|
|
.get_sdb = &get_sdb,
|
|
.load = &load,
|
|
.load_bytes = &load_bytes,
|
|
.destroy = &destroy,
|
|
.check_bytes = &check_bytes,
|
|
.baddr = &baddr,
|
|
.binsym = &binsym,
|
|
.entries = &entries,
|
|
.sections = §ions,
|
|
.symbols = &symbols,
|
|
.imports = &imports,
|
|
.info = &info,
|
|
.fields = &fields,
|
|
.size = &size,
|
|
.libs = &libs,
|
|
.relocs = &relocs,
|
|
};
|
|
|
|
#ifndef CORELIB
|
|
RLibStruct radare_plugin = {
|
|
.type = R_LIB_TYPE_BIN,
|
|
.data = &r_bin_plugin_coff,
|
|
.version = R2_VERSION
|
|
};
|
|
#endif
|