radare2/libr/bin/p/bin_dyldcache.c
2018-09-15 22:52:12 +02:00

1324 lines
33 KiB
C

/* radare2 - LGPL - Copyright 2018 - pancake */
#include <r_types.h>
#include <r_util.h>
#include <r_lib.h>
#include <r_bin.h>
#include <r_core.h>
#include <r_io.h>
// #include "../format/mach0/mach0_defines.h"
#define R_BIN_MACH064 1
#include "../format/mach0/mach0.h"
#include "objc/mach0_classes.h"
typedef struct {
ut64 start_of_data;
ut16 *page_starts;
ut32 page_starts_count;
ut16 *page_extras;
ut32 page_extras_count;
ut64 delta_mask;
ut64 value_mask;
ut32 delta_shift;
ut64 value_add;
ut32 page_size;
ut64 slide;
ut8 *one_page_buf;
} RDyldRebaseInfo;
typedef struct _r_dyldcache {
ut8 magic[8];
RList *bins;
RBuffer *buf;
int (*original_io_read)(RIO *io, RIODesc *fd, ut8 *buf, int count);
RDyldRebaseInfo *rebase_info;
cache_hdr_t *hdr;
cache_map_t *maps;
cache_accel_t *accel;
} RDyldCache;
typedef struct _r_bin_image {
char *file;
ut64 header_at;
} RDyldBinImage;
static void free_bin(RDyldBinImage *bin) {
if (!bin) {
return;
}
if (bin->file) {
free (bin->file);
bin->file = NULL;
}
R_FREE (bin);
}
static void rebase_info_free(RDyldRebaseInfo *rebase_info) {
if (!rebase_info) {
return;
}
if (rebase_info->page_starts) {
free (rebase_info->page_starts);
rebase_info->page_starts = NULL;
}
if (rebase_info->page_extras) {
free (rebase_info->page_extras);
rebase_info->page_extras = NULL;
}
if (rebase_info->one_page_buf) {
free (rebase_info->one_page_buf);
rebase_info->one_page_buf = NULL;
}
R_FREE (rebase_info);
}
static void r_dyldcache_free(RDyldCache *cache) {
if (!cache) {
return;
}
if (cache->bins) {
r_list_free (cache->bins);
cache->bins = NULL;
}
if (cache->buf) {
r_buf_free (cache->buf);
cache->buf = NULL;
}
if (cache->rebase_info) {
rebase_info_free (cache->rebase_info);
cache->rebase_info = NULL;
}
if (cache->hdr) {
free (cache->hdr);
cache->hdr = NULL;
}
if (cache->maps) {
free (cache->maps);
cache->maps = NULL;
}
if (cache->accel) {
free (cache->accel);
cache->accel = NULL;
}
R_FREE (cache);
}
static ut64 va2pa(uint64_t addr, cache_hdr_t *hdr, cache_map_t *maps, RBuffer *cache_buf, ut64 slide, ut32 *offset, ut32 *left) {
ut64 res = UT64_MAX;
uint32_t i;
addr -= slide;
for (i = 0; i < hdr->mappingCount; ++i) {
if (addr >= maps[i].address && addr < maps[i].address + maps[i].size) {
res = maps[i].fileOffset + addr - maps[i].address;
if (offset) {
*offset = addr - maps[i].address;
}
if (left) {
*left = maps[i].size - (addr - maps[i].address);
}
break;
}
}
return res;
}
static ut64 bin_obj_va2pa(ut64 p, ut32 *offset, ut32 *left, RBinFile *bf) {
if (!bf || !bf->o || !bf->o->bin_obj) {
return 0;
}
RDyldCache *cache = (RDyldCache*) ((struct MACH0_(obj_t)*)bf->o->bin_obj)->user;
if (!cache) {
return 0;
}
ut64 res = va2pa (p, cache->hdr, cache->maps, cache->buf, cache->rebase_info->slide, offset, left);
if (res == UT64_MAX) {
res = 0;
}
return res;
}
static struct MACH0_(obj_t) *bin_to_mach0(RBinFile *bf, RDyldBinImage *bin) {
if (!bin || !bf) {
return NULL;
}
RDyldCache *cache = (RDyldCache*) bf->o->bin_obj;
if (!cache) {
return NULL;
}
struct MACH0_(opts_t) opts;
MACH0_(opts_set_default) (&opts, bf);
opts.header_at = bin->header_at;
struct MACH0_(obj_t) *mach0 = MACH0_(new_buf) (cache->buf, &opts);
mach0->user = cache;
mach0->va2pa = &bin_obj_va2pa;
return mach0;
}
static int prot2perm(int x) {
int r = 0;
if (x & 1) {
r |= 4;
}
if (x & 2) {
r |= 2;
}
if (x & 4) {
r |= 1;
}
return r;
}
static ut32 dumb_ctzll(ut64 x) {
ut64 result = 0;
int i,j;
for (i = 0; i < 64; i += 8) {
ut8 byte = (x >> i) & 0xff;
if (!byte) {
result += 8;
} else {
for (j = 0; j < 8; j++) {
if (!((byte >> j) & 1)) {
result++;
} else {
break;
}
}
break;
}
}
return result;
}
static ut64 estimate_slide(RBinFile *bf, RDyldCache *cache, ut64 value_mask) {
ut64 slide = 0;
ut64 *classlist = malloc (64);
if (!classlist) {
goto beach;
}
RListIter *iter;
RDyldBinImage *bin;
r_list_foreach (cache->bins, iter, bin) {
bool found_sample = false;
struct MACH0_(opts_t) opts;
opts.verbose = bf->rbin->verbose;
opts.header_at = bin->header_at;
struct MACH0_(obj_t) *mach0 = MACH0_(new_buf) (cache->buf, &opts);
if (!mach0) {
goto beach;
}
struct section_t *sections = NULL;
if (!(sections = MACH0_(get_sections) (mach0))) {
MACH0_(mach0_free) (mach0);
goto beach;
}
int i;
int incomplete = 2;
int classlist_idx, data_idx;
for (i = 0; !sections[i].last && incomplete; i++) {
if (sections[i].size == 0) {
continue;
}
if (strstr (sections[i].name, "__objc_classlist")) {
incomplete--;
classlist_idx = i;
continue;
}
if (strstr (sections[i].name, "__objc_data")) {
incomplete--;
data_idx = i;
continue;
}
}
if (incomplete) {
goto next_bin;
}
int classlist_sample_size = R_MIN (64, sections[classlist_idx].size);
int n_classes = classlist_sample_size / 8;
if (r_buf_fread_at (cache->buf, sections[classlist_idx].offset, (ut8*) classlist, "l", n_classes) < classlist_sample_size) {
goto next_bin;
}
ut64 data_addr = sections[data_idx].addr;
for (i = 0; i < n_classes; i++) {
if ((classlist[i] & 0xfff) == (data_addr & 0xfff)) {
slide = (classlist[i] & value_mask) - (data_addr & value_mask);
found_sample = true;
break;
}
}
next_bin:
MACH0_(mach0_free) (mach0);
R_FREE (sections);
if (found_sample) {
break;
}
}
beach:
R_FREE (classlist);
return slide;
}
static RDyldRebaseInfo *get_rebase_info(RBinFile *bf, RDyldCache *cache) {
ut16 *page_starts = NULL;
ut16 *page_extras = NULL;
ut8 *one_page_buf = NULL;
RBuffer *cache_buf = cache->buf;
ut64 start_of_data = 0;
int i;
for (i = 0; i < cache->hdr->mappingCount; ++i) {
int perm = prot2perm (cache->maps[i].initProt);
if (!(perm & R_BIN_SCN_EXECUTABLE)) {
start_of_data = cache->maps[i].fileOffset;// + bf->o->boffset;
break;
}
}
if (!start_of_data) {
return NULL;
}
cache_slide2_t slide_info;
ut64 offset = cache->hdr->slideInfoOffset;
ut64 size = sizeof (cache_slide2_t);
if (r_buf_fread_at (cache_buf, offset, (ut8*) &slide_info, "6i2l", 1) != size) {
return NULL;
}
if (slide_info.version != 2) {
return NULL;
}
if (slide_info.page_starts_offset == 0 ||
slide_info.page_starts_offset > cache->hdr->slideInfoSize ||
slide_info.page_starts_offset + slide_info.page_starts_count * 2 > cache->hdr->slideInfoSize) {
return NULL;
}
if (slide_info.page_extras_offset == 0 ||
slide_info.page_extras_offset > cache->hdr->slideInfoSize ||
slide_info.page_extras_offset + slide_info.page_extras_count * 2 > cache->hdr->slideInfoSize) {
return NULL;
}
if (slide_info.page_starts_count > 0) {
ut64 size = slide_info.page_starts_count * 2;
ut64 at = cache->hdr->slideInfoOffset + slide_info.page_starts_offset;
page_starts = malloc (size);
if (r_buf_fread_at (cache_buf, at, (ut8*) page_starts, "s", slide_info.page_starts_count) != size) {
R_FREE (page_starts);
return NULL;
}
}
if (slide_info.page_extras_count > 0) {
ut64 size = slide_info.page_extras_count * 2;
ut64 at = cache->hdr->slideInfoOffset + slide_info.page_extras_offset;
page_extras = malloc (size);
if (r_buf_fread_at (cache_buf, at, (ut8*) page_extras, "s", slide_info.page_extras_count) != size) {
R_FREE (page_starts);
R_FREE (page_extras);
return NULL;
}
}
if (slide_info.page_size > 0) {
one_page_buf = malloc (slide_info.page_size);
if (!one_page_buf) {
goto beach;
}
}
RDyldRebaseInfo *rebase_info = R_NEW0 (RDyldRebaseInfo);
if (!rebase_info) {
goto beach;
}
rebase_info->start_of_data = start_of_data;
rebase_info->page_starts = page_starts;
rebase_info->page_starts_count = slide_info.page_starts_count;
rebase_info->page_extras = page_extras;
rebase_info->page_extras_count = slide_info.page_extras_count;
rebase_info->value_add = slide_info.value_add;
rebase_info->delta_mask = slide_info.delta_mask;
rebase_info->value_mask = ~rebase_info->delta_mask;
rebase_info->delta_shift = dumb_ctzll (rebase_info->delta_mask) - 2;
rebase_info->page_size = slide_info.page_size;
rebase_info->one_page_buf = one_page_buf;
rebase_info->slide = estimate_slide (bf, cache, rebase_info->value_mask);
if (rebase_info->slide) {
eprintf ("dyldcache is slid: 0x%"PFMT64x"\n", rebase_info->slide);
}
return rebase_info;
beach:
R_FREE (page_starts);
R_FREE (page_extras);
R_FREE (one_page_buf);
return NULL;
}
static bool check_bytes(const ut8 *buf, ut64 length) {
bool rc = false;
if (buf && length >= 32) {
char arch[9] = { 0 };
strncpy (arch, (const char *) buf + 9, R_MIN (length, sizeof (arch) - 1));
rc = !memcmp (buf, "dyld", 4);
if (rc) {
if (*arch) {
eprintf ("Arch: %s\n", arch);
if (!strstr (arch, "arm64")) {
return false;
}
}
}
}
return rc;
}
static cache_img_t *read_cache_images(RBuffer *cache_buf, cache_hdr_t *hdr) {
if (!cache_buf || !hdr || !hdr->imagesCount || !hdr->imagesOffset) {
return NULL;
}
ut64 size = sizeof (cache_img_t) * hdr->imagesCount;
cache_img_t *images = R_NEWS0 (cache_img_t, hdr->imagesCount);
if (!images) {
return NULL;
}
if (r_buf_fread_at (cache_buf, hdr->imagesOffset, (ut8*) images, "3l2i", hdr->imagesCount) != size) {
R_FREE (images);
return NULL;
}
return images;
}
static cache_imgxtr_t *read_cache_imgextra(RBuffer *cache_buf, cache_hdr_t *hdr, cache_accel_t *accel) {
if (!cache_buf || !hdr || !hdr->imagesCount || !accel || !accel->imageExtrasCount || !accel->imagesExtrasOffset) {
return NULL;
}
ut64 size = sizeof (cache_imgxtr_t) * accel->imageExtrasCount;
cache_imgxtr_t *images = R_NEWS0 (cache_imgxtr_t, accel->imageExtrasCount);
if (!images) {
return NULL;
}
if (r_buf_fread_at (cache_buf, accel->imagesExtrasOffset, (ut8*) images, "ll4i", accel->imageExtrasCount) != size) {
R_FREE (images);
return NULL;
}
return images;
}
static char *get_lib_name(RBuffer *cache_buf, cache_img_t *img) {
char file[256];
char *lib_name = file;
if (r_buf_read_at (cache_buf, img->pathFileOffset, (ut8*) &file, sizeof (file)) == sizeof (file)) {
file[255] = 0;
/*char * last_slash = strrchr (file, '/');
if (last_slash && *last_slash) {
lib_name = last_slash + 1;
}*/
return strdup (lib_name);
}
return strdup ("FAIL");
}
static int string_contains(const void *a, const void *b) {
return !strstr ((const char*) a, (const char*) b);
}
static RList *create_cache_bins(RBinFile *bf, RBuffer *cache_buf, cache_hdr_t *hdr, cache_map_t *maps, cache_accel_t *accel) {
RList *bins = r_list_newf ((RListFree)free_bin);
if (!bins) {
return NULL;
}
cache_img_t *img = read_cache_images (cache_buf, hdr);
if (!img) {
r_list_free (bins);
return NULL;
}
int i;
int *deps = NULL;
char *target_libs = NULL;
target_libs = r_sys_getenv ("R_DYLDCACHE_FILTER");
if (target_libs) {
RList *target_lib_names = r_str_split_list (target_libs, ":");
if (!target_lib_names) {
R_FREE (target_libs);
r_list_free (bins);
R_FREE (img);
return NULL;
}
deps = R_NEWS0 (int, hdr->imagesCount);
if (!deps) {
r_list_free (target_lib_names);
R_FREE (target_libs);
r_list_free (bins);
R_FREE (img);
return NULL;
}
ut16 *depArray = R_NEWS0 (ut16, accel->depListCount);
if (!depArray) {
r_list_free (target_lib_names);
R_FREE (target_libs);
r_list_free (bins);
R_FREE (deps);
R_FREE (img);
return NULL;
}
if (r_buf_fread_at (cache_buf, accel->depListOffset, (ut8*) depArray, "s", accel->depListCount) != accel->depListCount * 2) {
r_list_free (target_lib_names);
R_FREE (target_libs);
r_list_free (bins);
R_FREE (deps);
R_FREE (depArray);
R_FREE (img);
return NULL;
}
cache_imgxtr_t *extras = read_cache_imgextra (cache_buf, hdr, accel);
if (!extras) {
r_list_free (target_lib_names);
R_FREE (target_libs);
r_list_free (bins);
R_FREE (deps);
R_FREE (depArray);
R_FREE (img);
return NULL;
}
for (i = 0; i < hdr->imagesCount; i++) {
char *lib_name = get_lib_name (cache_buf, &img[i]);
if (!r_list_find (target_lib_names, lib_name, string_contains)) {
R_FREE (lib_name);
continue;
}
eprintf ("FILTER: %s\n", lib_name);
R_FREE (lib_name);
deps[i]++;
ut32 j;
for (j = extras[i].dependentsStartArrayIndex; depArray[j] != 0xffff; j++) {
bool upward = depArray[j] & 0x8000;
ut16 dep_index = depArray[j] & 0x7fff;
if (!upward) {
deps[dep_index]++;
char *dep_name = get_lib_name (cache_buf, &img[dep_index]);
eprintf ("-> %s\n", dep_name);
R_FREE (dep_name);
}
}
}
R_FREE (depArray);
R_FREE (extras);
R_FREE (target_libs);
r_list_free (target_lib_names);
}
for (i = 0; i < hdr->imagesCount; i++) {
if (deps && !deps[i]) {
continue;
}
ut64 pa = va2pa (img[i].address, hdr, maps, cache_buf, 0, NULL, NULL);
if (pa == UT64_MAX) {
continue;
}
ut8 magicbytes[4];
r_buf_read_at (cache_buf, pa, magicbytes, 4);
int magic = r_read_le32 (magicbytes);
switch (magic) {
case MH_MAGIC:
// parse_mach0 (ret, *ptr, bf);
break;
case MH_MAGIC_64:
{
char file[256];
RDyldBinImage *bin = R_NEW0 (RDyldBinImage);
if (!bin) {
r_list_free (bins);
R_FREE (img);
return NULL;
}
bin->header_at = pa;
if (r_buf_read_at (cache_buf, img[i].pathFileOffset, (ut8*) &file, sizeof (file)) == sizeof (file)) {
file[255] = 0;
char *last_slash = strrchr (file, '/');
if (last_slash && *last_slash) {
if (last_slash > file) {
char *scan = last_slash - 1;
while (scan > file && *scan != '/') {
scan--;
}
if (*scan == '/') {
bin->file = strdup (scan + 1);
} else {
bin->file = strdup (last_slash + 1);
}
} else {
bin->file = strdup (last_slash + 1);
}
} else {
bin->file = strdup (file);
}
}
r_list_append (bins, bin);
break;
}
default:
eprintf ("Unknown sub-bin\n");
break;
}
}
R_FREE (deps);
R_FREE (img);
return bins;
}
static void rebase_bytes(RDyldRebaseInfo *rebase_info, ut8 *buf, ut64 offset, int count, ut64 start_of_write) {
if (!rebase_info || !buf) {
return;
}
int in_buf = 0;
while (in_buf < count) {
ut64 offset_in_data = offset - rebase_info->start_of_data;
ut64 page_index = offset_in_data / rebase_info->page_size;
ut64 page_offset = offset_in_data % rebase_info->page_size;
ut64 to_next_page = rebase_info->page_size - page_offset;
if (page_index >= rebase_info->page_starts_count) {
goto next_page;
}
ut16 page_flag = rebase_info->page_starts[page_index];
if (page_flag == DYLD_CACHE_SLIDE_PAGE_ATTR_NO_REBASE) {
goto next_page;
}
if (!(page_flag & DYLD_CACHE_SLIDE_PAGE_ATTR_EXTRA)) {
ut64 first_rebase_off = rebase_info->page_starts[page_index] * 4;
if (first_rebase_off >= page_offset && first_rebase_off < page_offset + count) {
ut32 delta = 1;
while (delta) {
ut64 position = in_buf + first_rebase_off - page_offset;
if (position >= count) {
break;
}
ut64 raw_value = r_read_le64 (buf + position);
delta = ((raw_value & rebase_info->delta_mask) >> rebase_info->delta_shift);
if (position >= start_of_write) {
ut64 new_value = raw_value & rebase_info->value_mask;
if (new_value != 0) {
new_value += rebase_info->value_add;
new_value += rebase_info->slide;
}
r_write_le64 (buf + position, new_value);
}
first_rebase_off += delta;
}
}
}
next_page:
in_buf += to_next_page;
offset += to_next_page;
}
}
static int dyldcache_io_read(RIO *io, RIODesc *fd, ut8 *buf, int count) {
if (!io) {
return -1;
}
RCore *core = (RCore*) io->user;
if (!core || !core->bin || !core->bin->binfiles) {
return -1;
}
RDyldCache *cache = NULL;
RListIter *iter;
RBinFile *bf;
r_list_foreach (core->bin->binfiles, iter, bf) {
if (bf->fd == fd->fd ) {
if (!strncmp ((char*) bf->o->bin_obj, "dyldcac", 7)) {
cache = bf->o->bin_obj;
} else {
cache = ((struct MACH0_(obj_t)*) bf->o->bin_obj)->user;
}
break;
}
}
if (!cache || !cache->original_io_read) {
return fd->plugin->read (io, fd, buf, count);
}
bool includes_data = io->off > cache->rebase_info->start_of_data ||
(io->off + count) > cache->rebase_info->start_of_data;
int result = 0;
if (includes_data && count > 0) {
RDyldRebaseInfo *rebase_info = cache->rebase_info;
ut64 offset_in_data = io->off - rebase_info->start_of_data;
ut64 page_offset = offset_in_data % rebase_info->page_size;
ut64 internal_offset = io->off & ~(rebase_info->page_size - 1);
ut64 internal_end = io->off + count;
int rounded_count = internal_end - internal_offset;
ut8 *internal_buf = rebase_info->one_page_buf;
if (rounded_count > rebase_info->page_size) {
internal_buf = malloc (rounded_count);
}
ut64 original_off = io->off;
io->off = internal_offset;
int internal_result = cache->original_io_read (io, fd, internal_buf, rounded_count);
io->off = original_off;
if (internal_result >= page_offset + count) {
rebase_bytes (cache->rebase_info, internal_buf, internal_offset, internal_result, page_offset);
result = R_MIN (count, internal_result);
memcpy (buf, internal_buf + page_offset, result);
} else {
eprintf ("ERROR rebasing\n");
result = cache->original_io_read (io, fd, buf, count);
}
if (internal_buf != rebase_info->one_page_buf) {
R_FREE (internal_buf);
}
} else {
result = cache->original_io_read (io, fd, buf, count);
}
return result;
}
static void swizzle_io_read(RDyldCache *cache, RIO *io) {
if (!io || !io->desc || !io->desc->plugin) {
return;
}
RIOPlugin *plugin = io->desc->plugin;
cache->original_io_read = plugin->read;
plugin->read = &dyldcache_io_read;
}
static cache_hdr_t *read_cache_header(RBuffer *cache_buf) {
if (!cache_buf) {
return NULL;
}
cache_hdr_t *hdr = R_NEW0 (cache_hdr_t);
if (!hdr) {
return NULL;
}
ut64 size = sizeof (cache_hdr_t);
if (r_buf_fread_at (cache_buf, 0, (ut8*) hdr, "16c4i7l16clii4l", 1) != size) {
R_FREE (hdr);
return NULL;
}
return hdr;
}
static cache_map_t *read_cache_maps(RBuffer *cache_buf, cache_hdr_t *hdr) {
if (!cache_buf || !hdr || !hdr->mappingCount || !hdr->mappingOffset) {
return NULL;
}
ut64 size = sizeof (cache_map_t) * hdr->mappingCount;
cache_map_t *maps = R_NEWS0 (cache_map_t, hdr->mappingCount);
if (!maps) {
return NULL;
}
if (r_buf_fread_at (cache_buf, hdr->mappingOffset, (ut8*) maps, "3l2i", hdr->mappingCount) != size) {
R_FREE (maps);
return NULL;
}
return maps;
}
static cache_accel_t *read_cache_accel(RBuffer *cache_buf, cache_hdr_t *hdr, cache_map_t *maps) {
if (!cache_buf || !hdr || !hdr->accelerateInfoSize || !hdr->accelerateInfoAddr) {
return NULL;
}
ut64 offset = va2pa (hdr->accelerateInfoAddr, hdr, maps, cache_buf, 0, NULL, NULL);
if (!offset) {
return NULL;
}
ut64 size = sizeof (cache_accel_t);
cache_accel_t *accel = R_NEW0 (cache_accel_t);
if (!accel) {
return NULL;
}
if (r_buf_fread_at (cache_buf, offset, (ut8*) accel, "16il", 1) != size) {
R_FREE (accel);
return NULL;
}
accel->imagesExtrasOffset += offset;
accel->bottomUpListOffset += offset;
accel->dylibTrieOffset += offset;
accel->initializersOffset += offset;
accel->dofSectionsOffset += offset;
accel->reExportListOffset += offset;
accel->depListOffset += offset;
accel->rangeTableOffset += offset;
return accel;
}
static void *load_buffer(RBinFile *bf, RBuffer *buf, ut64 loadaddr, Sdb *sdb) {
RBuffer *fbuf = r_buf_new_with_io (&bf->rbin->iob, bf->fd);
ut8 bytes_to_check[32];
r_buf_read_at (fbuf, 0, (ut8*)bytes_to_check, 32);
if (!check_bytes (bytes_to_check, 32)) {
r_buf_free (fbuf);
return NULL;
}
RDyldCache *cache = R_NEW0 (RDyldCache);
strncpy ((char*) cache->magic, "dyldcac", 7);
cache->buf = fbuf;
cache->hdr = read_cache_header (fbuf);
if (!cache->hdr) {
r_dyldcache_free (cache);
return NULL;
}
cache->maps = read_cache_maps (fbuf, cache->hdr);
if (!cache->maps) {
r_dyldcache_free (cache);
return NULL;
}
cache->accel = read_cache_accel (fbuf, cache->hdr, cache->maps);
if (!cache->accel) {
r_dyldcache_free (cache);
return NULL;
}
cache->bins = create_cache_bins (bf, fbuf, cache->hdr, cache->maps, cache->accel);
if (!cache->bins) {
r_dyldcache_free (cache);
return NULL;
}
cache->rebase_info = get_rebase_info (bf, cache);
if (!cache->rebase_info) {
r_dyldcache_free (cache);
return NULL;
}
if (!cache->rebase_info->slide) {
swizzle_io_read (cache, bf->rbin->iob.io);
}
return cache;
}
static void *load_bytes(RBinFile *bf, const ut8 *buf, ut64 sz, ut64 loadaddr, Sdb *sdb) {
return (void *) (size_t) check_bytes (buf, sz);
}
static bool load(RBinFile *bf) {
const ut8 *bytes = bf ? r_buf_buffer (bf->buf) : NULL;
ut64 sz = bf ? r_buf_size (bf->buf): 0;
ut64 la = (bf && bf->o) ? bf->o->loadaddr: 0;
return load_bytes (bf, bytes, sz, la, bf? bf->sdb: NULL) != NULL;
}
static RList *entries(RBinFile *bf) {
RBinAddr *ptr = NULL;
RList *ret = r_list_newf (free);
if (!ret) {
return NULL;
}
if ((ptr = R_NEW0 (RBinAddr))) {
r_list_append (ret, ptr);
}
return ret;
}
static RBinInfo *info(RBinFile *bf) {
RBinInfo *ret = NULL;
if (!bf || !bf->o) {
return NULL;
}
RDyldCache *cache = (RDyldCache*) bf->o->bin_obj;
if (!cache) {
return NULL;
}
bool big_endian = 0;
if (!(ret = R_NEW0 (RBinInfo))) {
return NULL;
}
ret->file = strdup (bf->file);
ret->bclass = strdup ("dyldcache");
ret->rclass = strdup ("ios");
ret->os = strdup ("iOS");
ret->arch = strdup ("arm"); // XXX
ret->machine = strdup (ret->arch);
ret->subsystem = strdup ("xnu");
ret->type = strdup ("library-cache");
bool dyld64 = strstr(cache->hdr->magic, "arm64") != NULL;
ret->bits = dyld64? 64: 32;
ret->has_va = true;
ret->big_endian = big_endian;
ret->dbg_info = 0;
return ret;
}
#if 0
static void parse_mach0 (RList *ret, ut64 paddr, RBinFile *bf) {
// TODO
}
#endif
static ut64 baddr(RBinFile *bf) {
// XXX hardcoded
return 0x180000000;
}
void symbols_from_bin(RList *ret, RBinFile *bf, RDyldBinImage *bin) {
struct MACH0_(obj_t) *mach0 = bin_to_mach0 (bf, bin);
if (!mach0) {
return;
}
struct symbol_t *symbols = MACH0_(get_symbols) (mach0);
if (!symbols) {
return;
}
int i;
for (i = 0; !symbols[i].last; i++) {
if (!symbols[i].name[0] || symbols[i].addr < 100) {
continue;
}
if (strstr (symbols[i].name, "<redacted>")) {
continue;
}
RBinSymbol *sym = R_NEW0 (RBinSymbol);
if (!sym) {
break;
}
sym->name = strdup (symbols[i].name);
sym->vaddr = symbols[i].addr;
if (sym->name[0] == '_') {
char *dn = r_bin_demangle (bf, sym->name, sym->name, sym->vaddr);
if (dn) {
sym->dname = dn;
char *p = strchr (dn, '.');
if (p) {
if (IS_UPPER (sym->name[0])) {
sym->classname = strdup (sym->name);
sym->classname[p - sym->name] = 0;
} else if (IS_UPPER (p[1])) {
sym->classname = strdup (p + 1);
p = strchr (sym->classname, '.');
if (p) {
*p = 0;
}
}
}
}
}
sym->forwarder = r_str_const ("NONE");
sym->bind = r_str_const ((symbols[i].type == R_BIN_MACH0_SYMBOL_TYPE_LOCAL)?
R_BIN_BIND_LOCAL_STR: R_BIN_BIND_GLOBAL_STR);
sym->type = r_str_const (R_BIN_TYPE_FUNC_STR);
sym->paddr = symbols[i].offset + bf->o->boffset;
sym->size = symbols[i].size;
sym->ordinal = i;
r_list_append (ret, sym);
}
free (symbols);
MACH0_(mach0_free) (mach0);
}
static void handle_data_sections(RBinSection *sect) {
if (strstr (sect->name, "_cstring")) {
sect->is_data = true;
} else if (strstr (sect->name, "_os_log")) {
sect->is_data = true;
} else if (strstr (sect->name, "_objc_methname")) {
sect->is_data = true;
} else if (strstr (sect->name, "_objc_classname")) {
sect->is_data = true;
} else if (strstr (sect->name, "_objc_methtype")) {
sect->is_data = true;
}
}
static void sections_from_bin(RList *ret, RBinFile *bf, RDyldBinImage *bin) {
struct MACH0_(obj_t) *mach0 = bin_to_mach0 (bf, bin);
if (!mach0) {
return;
}
struct section_t *sections = NULL;
if (!(sections = MACH0_(get_sections) (mach0))) {
return;
}
int i;
for (i = 0; !sections[i].last; i++) {
RBinSection *ptr;
if (!(ptr = R_NEW0 (RBinSection))) {
break;
}
if (bin->file) {
r_snprintf (ptr->name, R_BIN_SIZEOF_STRINGS, "%s.%s", bin->file, (char*)sections[i].name);
} else {
r_snprintf (ptr->name, R_BIN_SIZEOF_STRINGS, "%s", (char*)sections[i].name);
}
if (strstr (ptr->name, "la_symbol_ptr")) {
int len = sections[i].size / 8;
ptr->format = r_str_newf ("Cd %d[%d]", 8, len);
}
ptr->name[R_BIN_SIZEOF_STRINGS] = 0;
handle_data_sections (ptr);
ptr->size = sections[i].size;
ptr->vsize = sections[i].vsize;
ptr->paddr = sections[i].offset + bf->o->boffset;
ptr->vaddr = sections[i].addr;
ptr->add = true;
if (!ptr->vaddr) {
ptr->vaddr = ptr->paddr;
}
ptr->srwx = sections[i].srwx;
r_list_append (ret, ptr);
}
free (sections);
MACH0_(mach0_free) (mach0);
}
static RList *sections(RBinFile *bf) {
RDyldCache *cache = (RDyldCache*) bf->o->bin_obj;
if (!cache) {
return NULL;
}
RList *ret = r_list_newf (free);
if (!ret) {
return NULL;
}
RListIter *iter;
RDyldBinImage *bin;
r_list_foreach (cache->bins, iter, bin) {
sections_from_bin (ret, bf, bin);
}
RBinSection *ptr = NULL;
int i;
for (i = 0; i < cache->hdr->mappingCount; ++i) {
if (!(ptr = R_NEW0 (RBinSection))) {
return NULL;
}
r_snprintf (ptr->name, R_BIN_SIZEOF_STRINGS, "cache_map.%d", i);
ptr->size = cache->maps[i].size;
ptr->vsize = ptr->size;
ptr->paddr = cache->maps[i].fileOffset;// + bf->o->boffset;
ptr->vaddr = cache->maps[i].address;
ptr->add = true;
ptr->srwx = prot2perm (cache->maps[i].initProt);
r_list_append (ret, ptr);
}
if (cache->rebase_info->slide > 0) {
RBinSection *section;
r_list_foreach (ret, iter, section) {
section->vaddr += cache->rebase_info->slide;
}
}
return ret;
}
static RList *symbols(RBinFile *bf) {
RDyldCache *cache = (RDyldCache*) bf->o->bin_obj;
if (!cache) {
return NULL;
}
RList *ret = r_list_newf (free);
if (!ret) {
return NULL;
}
RListIter *iter;
RDyldBinImage *bin;
r_list_foreach (cache->bins, iter, bin) {
symbols_from_bin (ret, bf, bin);
}
if (cache->rebase_info->slide > 0) {
RBinSymbol *sym;
r_list_foreach (ret, iter, sym) {
sym->vaddr += cache->rebase_info->slide;
}
}
return ret;
}
/* static void unswizzle_io_read(RDyldCache *cache, RIO *io) {
if (!io || !io->desc || !io->desc->plugin || !cache->original_io_read) {
return;
}
RIOPlugin *plugin = io->desc->plugin;
plugin->read = cache->original_io_read;
cache->original_io_read = NULL;
} */
static int destroy(RBinFile *bf) {
RDyldCache *cache = (RDyldCache*) bf->o->bin_obj;
// unswizzle_io_read (cache, bf->rbin->iob.io); // XXX io may be dead here
r_dyldcache_free (cache);
return true;
}
static RList *classes(RBinFile *bf) {
RDyldCache *cache = (RDyldCache*) bf->o->bin_obj;
if (!cache) {
return NULL;
}
RList *ret = r_list_newf (free);
if (!ret) {
return NULL;
}
RListIter *iter;
RDyldBinImage *bin;
RBuffer *orig_buf = bf->buf;
ut32 num_of_unnamed_class = 0;
r_list_foreach (cache->bins, iter, bin) {
struct MACH0_(obj_t) *mach0 = bin_to_mach0 (bf, bin);
if (!mach0) {
goto beach;
}
struct section_t *sections = NULL;
if (!(sections = MACH0_(get_sections) (mach0))) {
MACH0_(mach0_free) (mach0);
goto beach;
}
int i;
for (i = 0; !sections[i].last; i++) {
if (sections[i].size == 0) {
continue;
}
if (!strstr (sections[i].name, "__objc_classlist")) {
continue;
}
ut8 *pointers = malloc (sections[i].size);
if (r_buf_read_at (cache->buf, sections[i].offset, pointers, sections[i].size) < sections[i].size) {
R_FREE (pointers);
continue;
}
ut8 *cursor = pointers;
ut8 *pointers_end = pointers + sections[i].size;
for (; cursor < pointers_end; cursor += 8) {
ut64 pointer_to_class = r_read_le64 (cursor);
RBinClass *klass;
if (!(klass = R_NEW0 (RBinClass)) ||
!(klass->methods = r_list_new ()) ||
!(klass->fields = r_list_new ())) {
R_FREE (klass);
R_FREE (pointers);
R_FREE (sections);
MACH0_(mach0_free) (mach0);
goto beach;
}
bf->o->bin_obj = mach0;
bf->buf = cache->buf;
MACH0_(get_class_t) ((ut64) pointer_to_class, bf, klass, false);
bf->o->bin_obj = cache;
bf->buf = orig_buf;
if (!klass->name) {
klass->name = r_str_newf ("UnnamedClass%" PFMT64d, num_of_unnamed_class);
if (!klass->name) {
R_FREE (klass);
R_FREE (pointers);
R_FREE (sections);
MACH0_(mach0_free) (mach0);
goto beach;
}
num_of_unnamed_class++;
}
r_list_append (ret, klass);
}
R_FREE (pointers);
}
R_FREE (sections);
MACH0_(mach0_free) (mach0);
}
return ret;
beach:
r_list_free (ret);
return NULL;
}
static void header(RBinFile *bf) {
if (!bf || !bf->o) {
return;
}
RDyldCache *cache = (RDyldCache*) bf->o->bin_obj;
if (!cache) {
return;
}
RBin *bin = bf->rbin;
ut64 slide = cache->rebase_info->slide;
bin->cb_printf ("dyld cache header:\n");
bin->cb_printf ("magic: %s\n", cache->hdr->magic);
bin->cb_printf ("mappingOffset: 0x%"PFMT64x"\n", cache->hdr->mappingOffset);
bin->cb_printf ("mappingCount: 0x%"PFMT64x"\n", cache->hdr->mappingCount);
bin->cb_printf ("imagesOffset: 0x%"PFMT64x"\n", cache->hdr->imagesOffset);
bin->cb_printf ("imagesCount: 0x%"PFMT64x"\n", cache->hdr->imagesCount);
bin->cb_printf ("dyldBaseAddress: 0x%"PFMT64x"\n", cache->hdr->dyldBaseAddress);
bin->cb_printf ("codeSignatureOffset: 0x%"PFMT64x"\n", cache->hdr->codeSignatureOffset);
bin->cb_printf ("codeSignatureSize: 0x%"PFMT64x"\n", cache->hdr->codeSignatureSize);
bin->cb_printf ("slideInfoOffset: 0x%"PFMT64x"\n", cache->hdr->slideInfoOffset);
bin->cb_printf ("slideInfoSize: 0x%"PFMT64x"\n", cache->hdr->slideInfoSize);
bin->cb_printf ("localSymbolsOffset: 0x%"PFMT64x"\n", cache->hdr->localSymbolsSize);
char uuidstr[128];
r_hex_bin2str ((ut8*)cache->hdr->uuid, 16, uuidstr);
bin->cb_printf ("uuid: %s\n", uuidstr);
bin->cb_printf ("cacheType: 0x%"PFMT64x"\n", cache->hdr->cacheType);
bin->cb_printf ("branchPoolsOffset: 0x%"PFMT64x"\n", cache->hdr->branchPoolsOffset);
bin->cb_printf ("branchPoolsCount: 0x%"PFMT64x"\n", cache->hdr->branchPoolsCount);
bin->cb_printf ("accelerateInfoAddr: 0x%"PFMT64x"\n", cache->hdr->accelerateInfoAddr + slide);
bin->cb_printf ("accelerateInfoSize: 0x%"PFMT64x"\n", cache->hdr->accelerateInfoSize);
bin->cb_printf ("imagesTextOffset: 0x%"PFMT64x"\n", cache->hdr->imagesTextOffset);
bin->cb_printf ("imagesTextCount: 0x%"PFMT64x"\n", cache->hdr->imagesTextCount);
bin->cb_printf ("\nacceleration info:\n");
bin->cb_printf ("version: 0x%"PFMT64x"\n", cache->accel->version);
bin->cb_printf ("imageExtrasCount: 0x%"PFMT64x"\n", cache->accel->imageExtrasCount);
bin->cb_printf ("imagesExtrasOffset: 0x%"PFMT64x"\n", cache->accel->imagesExtrasOffset);
bin->cb_printf ("bottomUpListOffset: 0x%"PFMT64x"\n", cache->accel->bottomUpListOffset);
bin->cb_printf ("dylibTrieOffset: 0x%"PFMT64x"\n", cache->accel->dylibTrieOffset);
bin->cb_printf ("dylibTrieSize: 0x%"PFMT64x"\n", cache->accel->dylibTrieSize);
bin->cb_printf ("initializersOffset: 0x%"PFMT64x"\n", cache->accel->initializersOffset);
bin->cb_printf ("initializersCount: 0x%"PFMT64x"\n", cache->accel->initializersCount);
bin->cb_printf ("dofSectionsOffset: 0x%"PFMT64x"\n", cache->accel->dofSectionsOffset);
bin->cb_printf ("dofSectionsCount: 0x%"PFMT64x"\n", cache->accel->dofSectionsCount);
bin->cb_printf ("reExportListOffset: 0x%"PFMT64x"\n", cache->accel->reExportListOffset);
bin->cb_printf ("reExportCount: 0x%"PFMT64x"\n", cache->accel->reExportCount);
bin->cb_printf ("depListOffset: 0x%"PFMT64x"\n", cache->accel->depListOffset);
bin->cb_printf ("depListCount: 0x%"PFMT64x"\n", cache->accel->depListCount);
bin->cb_printf ("rangeTableOffset: 0x%"PFMT64x"\n", cache->accel->rangeTableOffset);
bin->cb_printf ("rangeTableCount: 0x%"PFMT64x"\n", cache->accel->rangeTableCount);
bin->cb_printf ("dyldSectionAddr: 0x%"PFMT64x"\n", cache->accel->dyldSectionAddr + slide);
bin->cb_printf ("\nslide info:\n");
bin->cb_printf ("page_starts_count: 0x%"PFMT64x"\n", cache->rebase_info->page_starts_count);
bin->cb_printf ("page_extras_count: 0x%"PFMT64x"\n", cache->rebase_info->page_extras_count);
bin->cb_printf ("delta_mask: 0x%"PFMT64x"\n", cache->rebase_info->delta_mask);
bin->cb_printf ("value_mask: 0x%"PFMT64x"\n", cache->rebase_info->value_mask);
bin->cb_printf ("delta_shift: 0x%"PFMT64x"\n", cache->rebase_info->delta_shift);
bin->cb_printf ("page_size: 0x%"PFMT64x"\n", cache->rebase_info->page_size);
bin->cb_printf ("slide: 0x%"PFMT64x"\n", slide);
}
RBinPlugin r_bin_plugin_dyldcache = {
.name = "dyldcache",
.desc = "dyldcache bin plugin",
.license = "LGPL3",
.load = &load,
.load_bytes = &load_bytes,
.load_buffer = &load_buffer,
.entries = &entries,
.baddr = &baddr,
.symbols = &symbols,
.sections = &sections,
.check_bytes = &check_bytes,
.destroy = &destroy,
.classes = &classes,
.header = &header,
.info = &info,
};
#ifndef CORELIB
R_API RLibStruct radare_plugin = {
.type = R_LIB_TYPE_BIN,
.data = &r_bin_plugin_dyldcache,
.version = R2_VERSION
};
#endif