radare2/libr/io/p/io_zip.c
2016-05-24 22:22:15 +02:00

645 lines
17 KiB
C

/* radare - LGPL - Copyright 2012-2014 - pancake
io_zip.c rewrite: Adam Pridgen <dso@rice.edu || adam.pridgen@thecoverofnight.com>
*/
// TODO: wrap with r_sandbox api
#include <r_io.h>
#include <r_lib.h>
#include <r_util.h>
#include <r_cons.h>
#include <zip.h>
typedef enum {
R_IO_PARENT_ZIP = 0x0001,
R_IO_CHILD_FILE = 0x0002,
R_IO_NEW_FILE = 0x0004,
R_IO_EXISTING_FILE = 0x0008,
R_IO_MODIFIED_FILE = 0x0010,
R_IO_DELETED_FILE = 0x0020,
} R_IO_ZIP_ARCHIVE_TYPE;
typedef struct r_io_zip_uri_const_t {
const char *name;
ut32 len;
} RIOZipConstURI;
static RIOZipConstURI ZIP_URIS[] = {
{"zip://", 6},
{"apk://", 6},
{"jar://", 6},
{NULL, 0}
};
static RIOZipConstURI ZIP_ALL_URIS[] = {
{"zipall://", 9},
{"apkall://", 9},
{"jarall://", 9},
{NULL, 0}
};
typedef struct r_io_zfo_t {
char * name;
char * archivename;
int mode;
int rw;
int fd;
int opened;
ut64 entry;
int flags;
ut8 modified;
RBuffer *b;
char *password;
ut8 encryption_value;
RIO * io_backref;
} RIOZipFileObj;
static int r_io_zip_realloc_buf(RIOZipFileObj *zfo, int count);
static int r_io_zip_truncate_buf(RIOZipFileObj *zfo, int size);
int r_io_zip_slurp_file(RIOZipFileObj *zfo);
//static int r_io_zip_check_file(const char *file);
int r_io_zip_open_zip_file(RIOZipFileObj * zfo);
void r_io_zip_free_zipfileobj(RIOZipFileObj *zfo);
RIODesc *check_zip_file_open(RIO *io, const char* filename);
RList *r_io_zip_get_files(char *archivename, ut32 flags, int mode, int rw);
RIOZipFileObj * r_io_zip_create_new_file(const char *archivename, const char *filename, struct zip_stat *sb, ut32 flags, int mode, int rw);
RIOZipFileObj *r_io_zip_alloc_zipfileobj(const char *archive_name, const char *filename, ut32 flags, int mode, int rw);
static int r_io_zip_init();
static int r_io_zip_plugin_open(RIO *io, const char *file, ut8 many);
static int r_io_zip_has_uri_substr(const char *file);
static int r_io_zip_check_uri(const char *file);
static int r_io_zip_flush_file(RIOZipFileObj *zfo);
static int r_io_zip_read(RIO *io, RIODesc *fd, ut8 *buf, int count);
static int r_io_zip_write(RIO *io, RIODesc *fd, const ut8 *buf, int count);
static int r_io_zip_close(RIODesc *desc);
static char * r_io_zip_get_by_file_idx(const char * archivename, const char *idx, ut32 flags, int mode, int rw);
static RIODesc * r_io_zip_open(RIO *io, const char *file, int rw, int mode);
static RList *r_io_zip_open_many(RIO *io, const char *file, int rw, int mode);
static int r_io_zip_check_uri_many(const char *file);
static int r_io_zip_resize(RIO *io, RIODesc *fd, ut64 size);
static int r_io_zip_init(RIO *io) {
return true;
}
static int r_io_zip_has_uri_substr(const char *file) {
return (file && strstr (file, "://"));
}
static int r_io_zip_check_uri(const char *file) {
int res = false;
int i = 0;
if (r_io_zip_has_uri_substr (file)) {
for (i = 0; ZIP_URIS[i].name != NULL; i++) {
if (!memcmp (file, ZIP_URIS[i].name, ZIP_URIS[i].len) && file[ZIP_URIS[i].len]) {
res = true;
break;
}
}
}
return res;
}
static int r_io_zip_check_uri_many(const char *file) {
int res = false;
int i = 0;
if (r_io_zip_has_uri_substr (file)) {
for (i = 0; ZIP_ALL_URIS[i].name != NULL; i++) {
if (!memcmp (file, ZIP_ALL_URIS[i].name, ZIP_ALL_URIS[i].len) && file[ZIP_ALL_URIS[i].len]) {
res = true;
break;
}
}
}
return res;
}
int r_io_zip_open_zip_file(RIOZipFileObj * zfo) {
return zfo->opened;
}
struct zip * r_io_zip_open_archive(const char *archivename, ut32 flags, int mode, int rw) {
struct zip * zipArch = NULL;
int zip_errorp;
if (!archivename)
return zipArch;
zipArch = zip_open (archivename, flags, &zip_errorp);
if (!zipArch) {
if (zip_errorp == ZIP_ER_INVAL) {
eprintf("ZIP File Error: Invalid file name (NULL).\n");
} else if (zip_errorp == ZIP_ER_OPEN) {
eprintf ("ZIP File Error: File could not be opened file name.\n");
} else if (zip_errorp == ZIP_ER_NOENT) {
eprintf ("ZIP File Error: File does not exist.\n");
} else if (zip_errorp == ZIP_ER_READ) {
eprintf ("ZIP File Error: Read error occurred.\n");
} else if (zip_errorp == ZIP_ER_NOZIP) {
eprintf ("ZIP File Error: File is not a valid ZIP archive.\n");
} else if (zip_errorp == ZIP_ER_INCONS) {
eprintf ("ZIP File Error: ZIP file had some inconsistencies archive.\n");
} else eprintf ("ZIP File Error: Something bad happened, get your debug on.\n");
}
return zipArch;
}
#if 0
static int r_io_zip_check_file(const char *file) {
int res = false;
ut8 buf[10];
FILE * fp = r_sandbox_fopen (file, "rb");
if (file && fp) {
fread (buf, 1, 10, fp);
if (!memcmp (buf, "\x50\x4b\x03\x04", 4))
res = true;
fclose (fp);
}
return res;
}
#endif
int r_io_zip_slurp_file(RIOZipFileObj *zfo) {
int res = false;
struct zip_stat sb;
struct zip_file *zFile = NULL;
struct zip * zipArch ;
if (!zfo) return res;
zipArch = r_io_zip_open_archive (
zfo->archivename, zfo->flags,
zfo->mode, zfo->rw);
//eprintf("Slurping file");
if (zipArch && zfo && zfo->entry != -1) {
zFile = zip_fopen_index (zipArch, zfo->entry, 0);
if (!zfo->b)
zfo->b = r_buf_new ();
zip_stat_init (&sb);
if (zFile && zfo->b && !zip_stat_index(zipArch,
zfo->entry, 0, &sb) ) {
ut8 *buf = malloc (sb.size);
memset (buf, 0, sb.size);
if (buf) {
zip_fread (zFile, buf, sb.size);
r_buf_set_bytes (zfo->b, buf, sb.size);
res = zfo->opened = true;
free (buf);
}
}
zip_fclose (zFile);
}
zip_close (zipArch);
return res;
}
RList * r_io_zip_get_files(char *archivename, ut32 flags, int mode, int rw) {
ut64 num_entries = 0, i = 0;
struct zip *zipArch = r_io_zip_open_archive (archivename, flags, mode, rw);
RList *files = NULL;
struct zip_stat sb;
char *name;
//eprintf("Slurping file");
if (zipArch) {
files = r_list_new ();
if (!files) return NULL;
files->free = free;
num_entries = zip_get_num_files (zipArch);
for (i=0; i < num_entries; i++) {
zip_stat_init (&sb);
zip_stat_index (zipArch, i, 0, &sb);
//eprintf("Comparing %s == %s = %d\n", sb.name, filename, strcmp(sb.name, filename));
if ((name = strdup (sb.name)))
r_list_append (files, name);
}
}
zip_close (zipArch);
return files;
}
int r_io_zip_flush_file(RIOZipFileObj *zfo) {
int res = false;
struct zip * zipArch;
if (!zfo) return res;
zipArch = r_io_zip_open_archive (
zfo->archivename, zfo->flags, zfo->mode, zfo->rw);
if (!zipArch)
return res;
struct zip_source *s = zip_source_buffer (zipArch, zfo->b->buf, zfo->b->length, 0);
if (s && zfo->entry != -1) {
if (zip_replace(zipArch, zfo->entry, s) == 0) {
res = true;
}
} else if (s && zfo->name) {
if (zip_add (zipArch, zfo->name, s) == 0) {
zfo->entry = zip_name_locate (zipArch, zfo->name, 0);
res = true;
}
}
// s (zip_source) is freed when the archive is closed, i think - dso
zip_close (zipArch);
if (s) zip_source_free (s);
return res;
}
void r_io_zip_free_zipfileobj(RIOZipFileObj *zfo) {
if (!zfo) return;
if (zfo->modified)
r_io_zip_flush_file (zfo);
free (zfo->name);
free (zfo->password);
r_buf_free (zfo->b);
free (zfo);
}
// Below this line are the r_io_zip plugin APIs
static RList *r_io_zip_open_many(RIO *io, const char *file, int rw, int mode) {
RList *list_fds = NULL;
RListIter *iter;
RList *filenames = NULL;
RIODesc *res = NULL;
RIOZipFileObj *zfo = NULL;
char *filename_in_zipfile, *zip_filename = NULL, *zip_uri;
if (!r_io_zip_plugin_open (io, file, 1))
return NULL;
zip_uri = strdup (file);
if (!zip_uri) return NULL;
// 1) Tokenize to the '//' and find the base file directory ('/')
zip_filename = strstr(zip_uri, "//");
if (zip_filename && zip_filename[2]) {
if (zip_filename[0] && zip_filename[0] == '/' &&
zip_filename[1] && zip_filename[1] == '/' ) {
*zip_filename++ = 0;
}
*zip_filename++ = 0;
} else {
free (zip_uri);
return NULL;
}
filenames = r_io_zip_get_files(zip_filename, 0, mode, rw );
if (!filenames) {
free (zip_uri);
return NULL;
}
list_fds = r_list_new ();
r_list_foreach (filenames, iter, filename_in_zipfile) {
size_t v = strlen (filename_in_zipfile);
if (filename_in_zipfile[v-1] == '/') continue;
zfo = r_io_zip_alloc_zipfileobj (zip_filename,
filename_in_zipfile, ZIP_CREATE, mode, rw);
if (zfo && zfo->entry == -1)
eprintf ("Warning: File did not exist, creating a new one.\n");
if (zfo) {
zfo->io_backref = io;
res = r_io_desc_new (&r_io_plugin_zip, zfo->fd,
zfo->name, rw, mode, zfo);
}
r_list_append (list_fds, res);
}
free(zip_uri);
r_list_free (filenames);
return list_fds;
}
static RIODesc *r_io_zip_open(RIO *io, const char *file, int rw, int mode) {
RIODesc *res = NULL;
RIOZipFileObj *zfo = NULL;
char *zip_uri = NULL, *zip_filename = NULL, *filename_in_zipfile = NULL;
if (!r_io_zip_plugin_open (io, file, 0))
return res;
zip_uri = strdup (file);
if (!zip_uri) return NULL;
// 1) Tokenize to the '//' and find the base file directory ('/')
zip_filename = strstr(zip_uri, "//");
if (zip_filename && zip_filename[2]) {
if (zip_filename[0] && zip_filename[0] == '/' &&
zip_filename[1] && zip_filename[1] == '/' ) {
*zip_filename++ = 0;
}
*zip_filename++ = 0;
// check for // for file in the archive
if ( (filename_in_zipfile = strstr(zip_filename, "//")) &&
filename_in_zipfile[2]) {
// null terminating uri to filename here.
*filename_in_zipfile++ = 0;
*filename_in_zipfile++ = 0;
filename_in_zipfile = strdup (filename_in_zipfile);
// check for :: index
} else if ((filename_in_zipfile = strstr (zip_filename, "::")) &&
filename_in_zipfile[2]) {
// null terminating uri to filename here.
*filename_in_zipfile++ = 0;
*filename_in_zipfile++ = 0;
filename_in_zipfile = r_io_zip_get_by_file_idx (
zip_filename, filename_in_zipfile,
ZIP_CREATE, mode, rw);
}
}
if (!zip_filename) {// && !filename_in_zipfile) {
free (zip_uri);
eprintf ("usage: zip:///path/to/archive//filepath\n"
"usage: zip:///path/to/archive::[number]\n"
"Archive was not found.\n");
return res;
}
// Failed to find the file name the archive.
if (!filename_in_zipfile) {
RList *files = NULL;
RListIter *iter, *iter_tmp;
char *name;
//eprintf("usage: zip:///path/to/archive//filepath\n");
files = r_io_zip_get_files (zip_filename, 0, mode, rw );
if (files) {
ut32 i = 0;
r_list_foreach_safe (files, iter, iter_tmp, name) {
io->cb_printf ("%d %s\n", i, name);
i++;
}
r_list_free (files);
}
free (zip_uri);
return res;
}
//eprintf("After parsing the given uri: %s\n", file);
//eprintf("Zip filename the given uri: %s\n", zip_filename);
//eprintf("File in the zip: %s\n", filename_in_zipfile);
zfo = r_io_zip_alloc_zipfileobj (zip_filename,
filename_in_zipfile, ZIP_CREATE, mode, rw);
if (zfo) {
if (zfo->entry == -1)
eprintf ("Warning: File did not exist, creating a new one.\n");
zfo->io_backref = io;
res = r_io_desc_new (&r_io_plugin_zip, zfo->fd,
zfo->name, rw, mode, zfo);
}
if (!res) {
eprintf ("Failed to open the archive %s and file %s\n",
zip_filename, filename_in_zipfile);
//free (zfo); zfo is already freed by r_io_desc_new
r_io_desc_free (res);
res = NULL;
}
free (zip_uri);
free (filename_in_zipfile);
return res;
}
static ut64 r_io_zip_lseek(RIO *io, RIODesc *fd, ut64 offset, int whence) {
RIOZipFileObj *zfo;
ut64 seek_val = 0;
if (!fd || !fd->data)
return -1;
zfo = fd->data;
seek_val = zfo->b->cur;
switch (whence) {
case SEEK_SET:
seek_val = (zfo->b->length < offset) ?
zfo->b->length : offset;
zfo->b->cur = io->off = seek_val;
return seek_val;
case SEEK_CUR:
seek_val = (zfo->b->length < (offset + zfo->b->cur)) ?
zfo->b->length : offset + zfo->b->cur;
zfo->b->cur = io->off = seek_val;
return seek_val;
case SEEK_END:
seek_val = zfo->b->length;
zfo->b->cur = io->off = seek_val;
return seek_val;
}
return seek_val;
}
static int r_io_zip_read(RIO *io, RIODesc *fd, ut8 *buf, int count) {
RIOZipFileObj *zfo = NULL;
if (!fd || !fd->data || !buf)
return -1;
zfo = fd->data;
if (zfo->b->length < io->off)
io->off = zfo->b->length;
return r_buf_read_at (zfo->b, io->off, buf, count);
}
static int r_io_zip_truncate_buf(RIOZipFileObj *zfo, int size) {
if (zfo->b->length < size)
return r_io_zip_realloc_buf(zfo, size - zfo->b->length);
if (size > 0){
ut8 *buf = malloc (size);
if (!buf) return false;
memcpy(buf, zfo->b->buf, size);
free (zfo->b->buf);
zfo->b->buf = buf;
zfo->b->length = size;
} else {
memset (zfo->b->buf, 0, zfo->b->length);
zfo->b->length = 0;
}
return true;
}
static int r_io_zip_realloc_buf(RIOZipFileObj *zfo, int count) {
int res = false;
if (zfo->b->cur + count > zfo->b->length) {
RBuffer *buffer = r_buf_new();
if (!buffer) return false;
buffer->buf = malloc (zfo->b->cur + count );
if (!buffer->buf) {
r_buf_free(buffer);
return false;
}
buffer->length = zfo->b->cur + count;
memcpy (buffer->buf, zfo->b->buf, zfo->b->length);
memset (buffer->buf+zfo->b->length, 0, count);
buffer->cur = zfo->b->cur;
r_buf_free (zfo->b);
zfo->b = buffer;
res = true;
}
return res;
}
static int r_io_zip_resize(RIO *io, RIODesc *fd, ut64 size) {
RIOZipFileObj *zfo;
int res = false;
ut64 cur_off = io->off;
if (!fd || !fd->data)
return -1;
zfo = fd->data;
res = r_io_zip_truncate_buf(zfo, size);
if (res == true) {
// XXX - Implement a flush of some sort, but until then, lets
// just write through
zfo->modified = 1;
r_io_zip_flush_file (zfo);
}
io->off = cur_off;
return res;
}
static int r_io_zip_write(RIO *io, RIODesc *fd, const ut8 *buf, int count) {
RIOZipFileObj *zfo;
int ret = 0;
if ( !fd || !fd->data || !buf)
return -1;
zfo = fd->data;
if ( !(zfo->flags & R_IO_WRITE)) return -1;
if (zfo->b->cur + count >= zfo->b->length)
r_io_zip_realloc_buf (zfo, count);
if (zfo->b->length < io->off)
io->off = zfo->b->length;
zfo->modified = 1;
ret = r_buf_write_at (zfo->b, io->off, buf, count);
// XXX - Implement a flush of some sort, but until then, lets
// just write through
r_io_zip_flush_file (zfo);
return ret;
}
static int r_io_zip_close(RIODesc *fd) {
RIOZipFileObj *zfo = NULL;
//eprintf("Am I called 2x?\n");
// this api will be called multiple times :/
if (!fd || !fd->data)
return -1;
zfo = fd->data;
r_io_zip_free_zipfileobj (zfo);
zfo = fd->data = NULL;
return 0;
}
char * r_io_zip_get_by_file_idx(const char * archivename, const char *idx, ut32 flags, int mode, int rw) {
char *filename = NULL;
ut64 i, num_entries;
ut32 file_idx = -1;
struct zip_stat sb;
struct zip * zipArch = r_io_zip_open_archive (archivename,
flags, mode, rw);
if (!idx || !zipArch) {
zip_close (zipArch);
return filename;
}
num_entries = zip_get_num_files (zipArch);
// filename starts with ::
file_idx = atoi (idx);
if ((file_idx == 0 && idx[0] != '0') || (file_idx >= num_entries)) {
zip_close (zipArch);
return filename;
}
for (i=0; i < num_entries; i++) {
zip_stat_init (&sb);
zip_stat_index (zipArch, i, 0, &sb );
//eprintf("Comparing %s == %s = %d\n", sb.name, filename, strcmp(sb.name, filename));
// filename starts with ::[number]
if (file_idx == i) {
filename = strdup (sb.name);
break;
}
}
zip_close (zipArch);
return filename;
}
/* The file can be a file in the archive or ::[num]. */
RIOZipFileObj* r_io_zip_alloc_zipfileobj(const char *archivename, const char *filename, ut32 flags, int mode, int rw) {
RIOZipFileObj *zfo = NULL;
ut64 i, num_entries;
struct zip_stat sb;
struct zip *zipArch = r_io_zip_open_archive (archivename, flags, mode, rw);
if (!zipArch) return NULL;
num_entries = zip_get_num_files (zipArch);
for (i=0; i < num_entries; i++) {
zip_stat_init (&sb);
zip_stat_index (zipArch, i, 0, &sb);
if (sb.name != NULL) {
if (strcmp (sb.name, filename) == 0) {
zfo = r_io_zip_create_new_file (
archivename, filename, &sb,
flags, mode, rw);
r_io_zip_slurp_file (zfo);
break;
}
}
}
if (!zfo)
zfo = r_io_zip_create_new_file (archivename,
filename, NULL, flags, mode, rw);
zip_close (zipArch);
return zfo;
}
RIOZipFileObj *r_io_zip_create_new_file(const char *archivename, const char *filename, struct zip_stat *sb, ut32 flags, int mode, int rw) {
RIOZipFileObj *zfo = R_NEW0 (RIOZipFileObj);
if (!zfo)
return NULL;
zfo->b = r_buf_new ();
zfo->archivename = strdup (archivename);
zfo->name = strdup (sb?sb->name:filename);
zfo->entry = sb == NULL ? -1 : sb->index;
zfo->fd = r_num_rand (0xFFFF); // XXX: Use r_io_fd api
zfo->flags = flags;
zfo->mode = mode;
zfo->rw = rw;
return zfo;
}
static int r_io_zip_plugin_open(RIO *io, const char *file, ut8 many) {
if (many) return (io && file) && (r_io_zip_check_uri_many (file));
return (io && file) && (r_io_zip_check_uri (file));
}
RIOPlugin r_io_plugin_zip = {
.name = "zip",
.desc = "Open zip files apk://foo.apk//MANIFEST or zip://foo.apk//theclass/fun.class, show files with: zip://foo.apk/, open all files with zipall://",
.license = "BSD",
.open = r_io_zip_open,
.open_many = r_io_zip_open_many,
.write = r_io_zip_write,
.read = r_io_zip_read,
.close = r_io_zip_close,
.lseek = r_io_zip_lseek,
.plugin_open = r_io_zip_plugin_open,
.resize = r_io_zip_resize,
.init = r_io_zip_init
};
#ifndef CORELIB
struct r_lib_struct_t radare_plugin = {
.type = R_LIB_TYPE_IO,
.data = &r_io_plugin_zip,
.version = R2_VERSION
};
#endif