mirror of
https://github.com/radareorg/radare2.git
synced 2024-12-15 09:21:00 +00:00
173 lines
3.9 KiB
C
173 lines
3.9 KiB
C
/* radare - LGPL - Copyright 2015-2016 - ampotos */
|
|
|
|
#include <r_types.h>
|
|
#include <r_util.h>
|
|
#include <r_lib.h>
|
|
#include <r_bin.h>
|
|
#include "omf/omf.h"
|
|
|
|
static void *load_bytes(RBinFile *arch, const ut8 *buf, ut64 size, ut64 loadaddrn, Sdb *sdb) {
|
|
if (!buf || !size || size == UT64_MAX) {
|
|
return NULL;
|
|
}
|
|
return r_bin_internal_omf_load ((char *)buf, size);
|
|
}
|
|
|
|
static int load(RBinFile *arch) {
|
|
const ut8 *byte = arch ? r_buf_buffer (arch->buf) : NULL;
|
|
ut64 size = arch ? r_buf_size (arch->buf) : 0;
|
|
if (!arch || !arch->o) {
|
|
return false;
|
|
}
|
|
arch->o->bin_obj = load_bytes(arch, byte, size, arch->o->loadaddr, arch->sdb);
|
|
return arch->o->bin_obj != NULL;
|
|
}
|
|
|
|
static int destroy(RBinFile *arch) {
|
|
r_bin_free_all_omf_obj (arch->o->bin_obj);
|
|
arch->o->bin_obj = NULL;
|
|
return true;
|
|
}
|
|
|
|
static int check_bytes(const ut8 *buf, ut64 length) {
|
|
int i;
|
|
if (!buf) {
|
|
return false;
|
|
}
|
|
if ((*buf != 0x80 && *buf != 0x82) || length < 4) {
|
|
return false;
|
|
}
|
|
ut16 rec_size = ut8p_bw (buf + 1);
|
|
ut8 str_size = *(buf + 3);
|
|
if (str_size + 2 != rec_size || length < rec_size + 3) {
|
|
return false;
|
|
}
|
|
// check that the string is ASCII
|
|
for (i = 4; i < str_size + 4; ++i) {
|
|
if (buf[i] > 0x7f) {
|
|
return false;
|
|
}
|
|
}
|
|
return r_bin_checksum_omf_ok ((char *)buf, 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 ut64 baddr(RBinFile *arch) {
|
|
return OMF_BASE_ADDR;
|
|
}
|
|
|
|
static RList *entries(RBinFile *arch) {
|
|
RList *ret;
|
|
RBinAddr *addr;
|
|
|
|
if (!(ret = r_list_newf (free))) {
|
|
return NULL;
|
|
}
|
|
if (!(addr = R_NEW0 (RBinAddr))) {
|
|
r_list_free (ret);
|
|
return NULL;
|
|
}
|
|
if (!r_bin_omf_get_entry(arch->o->bin_obj, addr)) {
|
|
R_FREE (addr);
|
|
} else {
|
|
r_list_append (ret, addr);
|
|
}
|
|
return ret;
|
|
}
|
|
|
|
static RList *sections(RBinFile *arch) {
|
|
RList *ret;
|
|
ut32 ct_omf_sect = 0;
|
|
r_bin_omf_obj *obj = arch->o->bin_obj;
|
|
|
|
if (!(ret = r_list_new ()))
|
|
return NULL;
|
|
|
|
while (ct_omf_sect < obj->nb_section)
|
|
if (!r_bin_omf_send_sections (ret, \
|
|
obj->sections[ct_omf_sect++], arch->o->bin_obj))
|
|
return ret;
|
|
return ret;
|
|
}
|
|
|
|
static RList *symbols(RBinFile *arch) {
|
|
RList *ret;
|
|
RBinSymbol *sym;
|
|
OMF_symbol *sym_omf;
|
|
int ct_sym = 0;
|
|
|
|
if (!(ret = r_list_new()))
|
|
return NULL;
|
|
|
|
ret->free = free;
|
|
|
|
while (ct_sym < ((r_bin_omf_obj *)arch->o->bin_obj)->nb_symbol) {
|
|
if (!(sym = R_NEW0 (RBinSymbol)))
|
|
return ret;
|
|
sym_omf = ((r_bin_omf_obj *)arch->o->bin_obj)->symbols[ct_sym++];
|
|
sym->name = strdup (sym_omf->name);
|
|
sym->forwarder = r_str_const ("NONE");
|
|
sym->paddr = r_bin_omf_get_paddr_sym(arch->o->bin_obj, sym_omf);
|
|
sym->vaddr = r_bin_omf_get_vaddr_sym(arch->o->bin_obj, sym_omf);
|
|
sym->ordinal = ct_sym;
|
|
sym->size = 0;
|
|
r_list_append (ret, sym);
|
|
}
|
|
return ret;
|
|
}
|
|
|
|
static RBinInfo *info(RBinFile *arch) {
|
|
RBinInfo *ret;
|
|
|
|
if(!(ret = R_NEW0 (RBinInfo)))
|
|
return NULL;
|
|
ret->file = strdup (arch->file);
|
|
ret->bclass = strdup ("OMF");
|
|
ret->rclass = strdup ("omf");
|
|
// the "E" is here to made rva return the same value for 16 bit en 32 bits files
|
|
ret->type = strdup ("E OMF (Relocatable Object Module Format)");
|
|
ret->os = strdup ("any");
|
|
ret->machine = strdup ("i386");
|
|
ret->arch = strdup ("x86");
|
|
ret->big_endian = false;
|
|
ret->has_va = true;
|
|
ret->bits = r_bin_omf_get_bits (arch->o->bin_obj);
|
|
ret->dbg_info = 0;
|
|
ret->has_nx = false;
|
|
return ret;
|
|
}
|
|
|
|
static ut64 get_vaddr (RBinFile *arch, ut64 baddr, ut64 paddr, ut64 vaddr) {
|
|
return vaddr;
|
|
}
|
|
|
|
struct r_bin_plugin_t r_bin_plugin_omf = {
|
|
.name = "omf",
|
|
.desc = "omf bin plugin",
|
|
.license = "LGPL3",
|
|
.load = &load,
|
|
.load_bytes = &load_bytes,
|
|
.destroy = &destroy,
|
|
.check = &check,
|
|
.check_bytes = &check_bytes,
|
|
.baddr = &baddr,
|
|
.entries = &entries,
|
|
.sections = §ions,
|
|
.symbols = &symbols,
|
|
.info = &info,
|
|
.get_vaddr = &get_vaddr,
|
|
};
|
|
|
|
#ifndef CORELIB
|
|
struct r_lib_struct_t radare_plugin = {
|
|
.type = R_LIB_TYPE_BIN,
|
|
.data = &r_bin_plugin_omf,
|
|
.version = R2_VERSION
|
|
};
|
|
#endif
|