mirror of
https://github.com/radareorg/radare2.git
synced 2025-02-10 00:02:39 +00:00
228 lines
5.6 KiB
C
228 lines
5.6 KiB
C
/* radare - LGPL - Copyright 2015 - pancake */
|
|
|
|
#include <r_types.h>
|
|
#include <r_util.h>
|
|
#include <r_lib.h>
|
|
#include <r_bin.h>
|
|
|
|
//__attribute__((__aligned__(4)))
|
|
typedef struct __packed art_header_t {
|
|
ut8 magic[4];
|
|
ut8 version[4];
|
|
ut32 image_base;
|
|
ut32 image_size;
|
|
ut32 bitmap_offset;
|
|
ut32 bitmap_size;
|
|
ut32 checksum; /* adler32 */
|
|
ut32 oat_begin;
|
|
ut32 oat_data_begin;
|
|
ut32 oat_data_end;
|
|
ut32 oat_end;
|
|
ut32 patch_delta;
|
|
ut32 image_roots;
|
|
} ARTHeader;
|
|
|
|
typedef struct {
|
|
Sdb *kv;
|
|
ARTHeader art;
|
|
} ArtObj;
|
|
|
|
|
|
static int art_header_load(ARTHeader *art, RBuffer *buf, Sdb *db) {
|
|
/* TODO: handle read errors here */
|
|
(void)r_buf_fread_at (buf, 0, (ut8*)art, "IIiiiiiiiiiii", 1);
|
|
sdb_set (db, "img.base", sdb_fmt (0, "0x%x", art->image_base), 0);
|
|
sdb_set (db, "img.size", sdb_fmt (0, "0x%x", art->image_size), 0);
|
|
sdb_set (db, "art.checksum", sdb_fmt (0, "0x%x", art->checksum), 0);
|
|
sdb_set (db, "art.version", sdb_fmt (0, "%c%c%c",
|
|
art->version[0], art->version[1], art->version[2]), 0);
|
|
sdb_set (db, "oat.begin", sdb_fmt (0, "0x%x", art->oat_begin), 0);
|
|
sdb_set (db, "oat.end", sdb_fmt (0, "0x%x", art->oat_end), 0);
|
|
sdb_set (db, "oat_data.begin", sdb_fmt (0, "0x%x", art->oat_data_begin), 0);
|
|
sdb_set (db, "oat_data.end", sdb_fmt (0, "0x%x", art->oat_data_end), 0);
|
|
sdb_set (db, "patch_delta", sdb_fmt (0, "0x%x", art->patch_delta), 0);
|
|
sdb_set (db, "image_roots", sdb_fmt (0, "0x%x", art->image_roots), 0);
|
|
return true;
|
|
}
|
|
|
|
static int check(RBinFile *arch);
|
|
static int check_bytes(const ut8 *buf, ut64 length);
|
|
|
|
static Sdb* get_sdb (RBinObject *o) {
|
|
ArtObj *ao;
|
|
if (!o) return NULL;
|
|
ao = o->bin_obj;
|
|
if (!ao) return NULL;
|
|
return ao->kv;
|
|
}
|
|
|
|
static void * load_bytes(RBinFile *arch, const ut8 *buf, ut64 sz, ut64 la, Sdb *sdb){
|
|
ArtObj *ao = R_NEW0 (ArtObj);
|
|
ao->kv = sdb_new0 ();
|
|
art_header_load (&ao->art, arch->buf, ao->kv);
|
|
sdb_ns_set (sdb, "info", ao->kv);
|
|
return ao;
|
|
}
|
|
|
|
static int load(RBinFile *arch) {
|
|
return true;
|
|
}
|
|
|
|
static int destroy(RBinFile *arch) {
|
|
return true;
|
|
}
|
|
|
|
static ut64 baddr(RBinFile *arch) {
|
|
ArtObj *ao = arch->o->bin_obj;
|
|
return ao?ao->art.image_base:0x00;
|
|
}
|
|
|
|
static RList *strings(RBinFile *arch) {
|
|
return NULL;
|
|
}
|
|
|
|
static RBinInfo* info(RBinFile *arch) {
|
|
ArtObj *ao = arch->o->bin_obj;
|
|
RBinInfo *ret;
|
|
if (!ao) return NULL;
|
|
|
|
ret = R_NEW0 (RBinInfo);
|
|
if (!ret) return NULL;
|
|
|
|
//art_header_load (&art, arch->buf);
|
|
|
|
ret->lang = NULL;
|
|
ret->file = arch->file? strdup (arch->file): NULL;
|
|
ret->type = strdup ("ART");
|
|
|
|
ret->bclass = malloc (5);
|
|
memcpy (ret->bclass, &ao->art.version, 4);
|
|
ret->bclass[3] = 0;
|
|
|
|
ret->rclass = strdup ("program");
|
|
ret->os = strdup ("android");
|
|
ret->subsystem = strdup ("unknown");
|
|
ret->machine = strdup ("arm");
|
|
ret->arch = strdup ("arm");
|
|
ret->has_va = 1;
|
|
ret->bits = 16; // 32? 64?
|
|
ret->big_endian = 0;
|
|
ret->dbg_info = 0;
|
|
return ret;
|
|
}
|
|
|
|
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) {
|
|
return (buf && !strncmp ((const char *)buf, "art\n", R_MIN (4, length)));
|
|
}
|
|
|
|
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)))
|
|
return ret;
|
|
ptr->paddr = ptr->vaddr = 0;
|
|
r_list_append (ret, ptr);
|
|
return ret;
|
|
}
|
|
|
|
static RList* sections(RBinFile *arch) {
|
|
ArtObj *ao = arch->o->bin_obj;
|
|
if (!ao) return NULL;
|
|
ARTHeader art = ao->art;
|
|
RList *ret = NULL;
|
|
RBinSection *ptr = NULL;
|
|
|
|
if (!(ret = r_list_new ()))
|
|
return NULL;
|
|
ret->free = free;
|
|
|
|
if (!(ptr = R_NEW0 (RBinSection)))
|
|
return ret;
|
|
strncpy (ptr->name, "load", R_BIN_SIZEOF_STRINGS);
|
|
ptr->size = arch->buf->length;
|
|
ptr->vsize = art.image_size; // TODO: align?
|
|
ptr->paddr = 0;
|
|
ptr->vaddr = art.image_base;
|
|
ptr->srwx = R_BIN_SCN_READABLE | R_BIN_SCN_MAP; // r--
|
|
r_list_append (ret, ptr);
|
|
|
|
if (!(ptr = R_NEW0 (RBinSection)))
|
|
return ret;
|
|
strncpy (ptr->name, "bitmap", R_BIN_SIZEOF_STRINGS);
|
|
ptr->size = art.bitmap_size;
|
|
ptr->vsize = art.bitmap_size;
|
|
ptr->paddr = art.bitmap_offset;
|
|
ptr->vaddr = art.image_base + art.bitmap_offset;
|
|
ptr->srwx = R_BIN_SCN_READABLE | R_BIN_SCN_EXECUTABLE | R_BIN_SCN_MAP; // r-x
|
|
r_list_append (ret, ptr);
|
|
|
|
if (!(ptr = R_NEW0 (RBinSection)))
|
|
return ret;
|
|
strncpy (ptr->name, "oat", R_BIN_SIZEOF_STRINGS);
|
|
ptr->paddr = art.bitmap_offset;
|
|
ptr->vaddr = art.oat_begin;
|
|
ptr->size = art.oat_end - art.oat_begin;
|
|
ptr->vsize = ptr->size;
|
|
ptr->srwx = R_BIN_SCN_READABLE | R_BIN_SCN_EXECUTABLE | R_BIN_SCN_MAP; // r-x
|
|
r_list_append (ret, ptr);
|
|
|
|
if (!(ptr = R_NEW0 (RBinSection)))
|
|
return ret;
|
|
strncpy (ptr->name, "oat_data", R_BIN_SIZEOF_STRINGS);
|
|
ptr->paddr = art.bitmap_offset;
|
|
ptr->vaddr = art.oat_data_begin;
|
|
ptr->size = art.oat_data_end - art.oat_data_begin;
|
|
ptr->vsize = ptr->size;
|
|
ptr->srwx = R_BIN_SCN_READABLE | R_BIN_SCN_MAP; // r--
|
|
r_list_append (ret, ptr);
|
|
|
|
return ret;
|
|
}
|
|
|
|
struct r_bin_plugin_t r_bin_plugin_art = {
|
|
.name = "art",
|
|
.desc = "Android Runtime",
|
|
.license = "LGPL3",
|
|
.init = NULL,
|
|
.fini = NULL,
|
|
.get_sdb = &get_sdb,
|
|
.load = &load,
|
|
.load_bytes = &load_bytes,
|
|
.destroy = &destroy,
|
|
.check = &check,
|
|
.check_bytes = &check_bytes,
|
|
.baddr = &baddr,
|
|
.sections = §ions,
|
|
.boffset = NULL,
|
|
.binsym = NULL,
|
|
.entries = entries,
|
|
.symbols = NULL,
|
|
.imports = NULL,
|
|
.strings = &strings,
|
|
.info = &info,
|
|
.fields = NULL,
|
|
.libs = NULL,
|
|
.relocs = NULL,
|
|
.dbginfo = NULL,
|
|
.write = NULL,
|
|
.demangle_type = NULL
|
|
};
|
|
|
|
#ifndef CORELIB
|
|
struct r_lib_struct_t radare_plugin = {
|
|
.type = R_LIB_TYPE_BIN,
|
|
.data = &r_bin_plugin_art,
|
|
.version = R2_VERSION
|
|
};
|
|
#endif
|