zip: import initial module
This provides rudimentary support for parsing the contents of a ZIP file. For now this just supports uncompressed file entries but DEFLATE support is planned. This also only implements a low level public API so more work will be needed to make it usable by application code. Change-Id: Ia68b3078e5276666a0e5023f4bc7a9f94272738a
This commit is contained in:
parent
2acf8db3e1
commit
b87e75f768
3 changed files with 770 additions and 0 deletions
|
@ -268,6 +268,7 @@ common/structec.c
|
|||
common/timefuncs.c
|
||||
common/unicode.c
|
||||
common/vuprintf.c
|
||||
common/zip.c
|
||||
|
||||
/* Display */
|
||||
scroll_engine.c
|
||||
|
|
705
firmware/common/zip.c
Normal file
705
firmware/common/zip.c
Normal file
|
@ -0,0 +1,705 @@
|
|||
/***************************************************************************
|
||||
* __________ __ ___.
|
||||
* Open \______ \ ____ ____ | | _\_ |__ _______ ___
|
||||
* Source | _// _ \_/ ___\| |/ /| __ \ / _ \ \/ /
|
||||
* Jukebox | | ( <_> ) \___| < | \_\ ( <_> > < <
|
||||
* Firmware |____|_ /\____/ \___ >__|_ \|___ /\____/__/\_ \
|
||||
* \/ \/ \/ \/ \/
|
||||
* $Id$
|
||||
*
|
||||
* Copyright (C) 2021 by James Buren
|
||||
*
|
||||
* This program is free software; you can redistribute it and/or
|
||||
* modify it under the terms of the GNU General Public License
|
||||
* as published by the Free Software Foundation; either version 2
|
||||
* of the License, or (at your option) any later version.
|
||||
*
|
||||
* This software is distributed on an "AS IS" basis, WITHOUT WARRANTY OF ANY
|
||||
* KIND, either express or implied.
|
||||
*
|
||||
****************************************************************************/
|
||||
|
||||
#include "zip.h"
|
||||
#include <string.h>
|
||||
#include "file.h"
|
||||
#include "system.h"
|
||||
#include "core_alloc.h"
|
||||
#include "timefuncs.h"
|
||||
#include "crc32.h"
|
||||
#include "rbendian.h"
|
||||
|
||||
#define zip_core_alloc(N) core_alloc_ex("zip",(N),&dummy_ops)
|
||||
|
||||
static struct buflib_callbacks dummy_ops;
|
||||
|
||||
enum {
|
||||
ZIP_SIG_ED = 0x06054b50,
|
||||
ZIP_SIG_CD = 0x02014b50,
|
||||
ZIP_SIG_LF = 0x04034b50,
|
||||
ZIP_BIT_DD = 0x0008,
|
||||
ZIP_METHOD_STORE = 0x0000,
|
||||
ZIP_MAX_LENGTH = 0xffff,
|
||||
ZIP_BUFFER_SIZE = 4096,
|
||||
};
|
||||
|
||||
enum {
|
||||
ZIP_STATE_INITIAL,
|
||||
ZIP_STATE_ED_ENTER,
|
||||
ZIP_STATE_ED_EXIT,
|
||||
ZIP_STATE_CD_ENTER,
|
||||
ZIP_STATE_CD_EXIT,
|
||||
ZIP_STATE_LF_ENTER,
|
||||
ZIP_STATE_LF_EXIT,
|
||||
};
|
||||
|
||||
struct zip_ed_disk {
|
||||
uint32_t signature;
|
||||
uint16_t disk_number;
|
||||
uint16_t disk_number_cd;
|
||||
uint16_t disk_entries_cd;
|
||||
uint16_t cd_entries;
|
||||
uint32_t cd_size;
|
||||
uint32_t cd_offset;
|
||||
uint16_t comment_length;
|
||||
// comment block (variable length)
|
||||
} __attribute__((packed));
|
||||
|
||||
struct zip_ed {
|
||||
uint32_t cd_size;
|
||||
uint32_t cd_offset;
|
||||
uint16_t cd_entries;
|
||||
};
|
||||
|
||||
struct zip_cd_disk {
|
||||
uint32_t signature;
|
||||
uint16_t version_madeby;
|
||||
uint16_t version_needed;
|
||||
uint16_t flags;
|
||||
uint16_t method;
|
||||
uint16_t time;
|
||||
uint16_t date;
|
||||
uint32_t crc;
|
||||
uint32_t compressed_size;
|
||||
uint32_t uncompressed_size;
|
||||
uint16_t name_length;
|
||||
uint16_t extra_length;
|
||||
uint16_t comment_length;
|
||||
uint16_t disk_number;
|
||||
uint16_t internal_attributes;
|
||||
uint32_t external_attributes;
|
||||
uint32_t lf_offset;
|
||||
// name block (variable length)
|
||||
// extra block (variable length)
|
||||
// comment block (variable length)
|
||||
} __attribute__((packed));
|
||||
|
||||
struct zip_cd {
|
||||
uint32_t crc;
|
||||
uint32_t compressed_size;
|
||||
uint32_t uncompressed_size;
|
||||
uint32_t lf_offset;
|
||||
};
|
||||
|
||||
struct zip_lf_disk {
|
||||
uint32_t signature;
|
||||
uint16_t version_needed;
|
||||
uint16_t flags;
|
||||
uint16_t method;
|
||||
uint16_t time;
|
||||
uint16_t date;
|
||||
uint32_t crc;
|
||||
uint32_t compressed_size;
|
||||
uint32_t uncompressed_size;
|
||||
uint16_t name_length;
|
||||
uint16_t extra_length;
|
||||
// name block (variable length)
|
||||
// extra block (variable length)
|
||||
} __attribute__((packed));
|
||||
|
||||
struct zip_lf {
|
||||
uint16_t flags;
|
||||
uint16_t method;
|
||||
uint16_t time;
|
||||
uint16_t date;
|
||||
uint32_t crc;
|
||||
uint32_t compressed_size;
|
||||
uint32_t uncompressed_size;
|
||||
uint16_t name_length;
|
||||
char name[MAX_PATH];
|
||||
};
|
||||
|
||||
struct zip {
|
||||
ssize_t (*read) (struct zip*, void*, size_t);
|
||||
off_t (*seek) (struct zip*, off_t, int);
|
||||
off_t (*size) (struct zip*);
|
||||
void (*close) (struct zip*);
|
||||
int zip_handle;
|
||||
int state;
|
||||
zip_callback cb;
|
||||
struct zip_args args;
|
||||
void* ctx;
|
||||
struct zip_ed ed;
|
||||
int cds_handle;
|
||||
struct zip_cd* cds;
|
||||
struct zip_lf lf;
|
||||
};
|
||||
|
||||
struct zip_file {
|
||||
struct zip base;
|
||||
int file;
|
||||
};
|
||||
|
||||
struct zip_mem {
|
||||
struct zip base;
|
||||
int mem_handle;
|
||||
const uint8_t* mem;
|
||||
off_t mem_offset;
|
||||
off_t mem_size;
|
||||
};
|
||||
|
||||
static int zip_read_ed(struct zip* z) {
|
||||
const off_t file_size = z->size(z);
|
||||
const off_t max_size = sizeof(struct zip_ed_disk) + ZIP_MAX_LENGTH;
|
||||
const off_t read_size = MIN(file_size, max_size);
|
||||
const uint32_t sig = htole32(ZIP_SIG_ED);
|
||||
int mem_handle = -1;
|
||||
uint8_t* mem;
|
||||
off_t i = read_size - sizeof(struct zip_ed_disk);
|
||||
const struct zip_ed_disk* edd;
|
||||
uint16_t disk_number;
|
||||
uint16_t disk_number_cd;
|
||||
uint16_t disk_entries_cd;
|
||||
uint16_t cd_entries;
|
||||
struct zip_ed* ed = &z->ed;
|
||||
int rv;
|
||||
|
||||
z->state = ZIP_STATE_ED_ENTER;
|
||||
|
||||
if (file_size < (off_t) sizeof(struct zip_ed_disk)) {
|
||||
rv = -2;
|
||||
goto bail;
|
||||
}
|
||||
|
||||
if ((mem_handle = zip_core_alloc(read_size)) < 0) {
|
||||
rv = -3;
|
||||
goto bail;
|
||||
}
|
||||
|
||||
mem = core_get_data(mem_handle);
|
||||
|
||||
if (z->seek(z, -read_size, SEEK_END) < 0) {
|
||||
rv = -4;
|
||||
goto bail;
|
||||
}
|
||||
|
||||
if (z->read(z, mem, read_size) != read_size) {
|
||||
rv = -5;
|
||||
goto bail;
|
||||
}
|
||||
|
||||
for (; i >= 0; i--)
|
||||
if (memcmp(mem + i, &sig, sizeof(uint32_t)) == 0)
|
||||
break;
|
||||
|
||||
if (i < 0) {
|
||||
rv = -6;
|
||||
goto bail;
|
||||
}
|
||||
|
||||
edd = (struct zip_ed_disk*) (mem + i);
|
||||
disk_number = letoh16(edd->disk_number);
|
||||
disk_number_cd = letoh16(edd->disk_number_cd);
|
||||
disk_entries_cd = letoh16(edd->disk_entries_cd);
|
||||
cd_entries = letoh16(edd->cd_entries);
|
||||
|
||||
if (disk_number != 0 || disk_number_cd != 0 || disk_entries_cd != cd_entries) {
|
||||
rv = -7;
|
||||
goto bail;
|
||||
}
|
||||
|
||||
ed->cd_size = letoh32(edd->cd_size);
|
||||
ed->cd_offset = letoh32(edd->cd_offset);
|
||||
ed->cd_entries = cd_entries;
|
||||
|
||||
z->state = ZIP_STATE_ED_EXIT;
|
||||
rv = 0;
|
||||
|
||||
bail:
|
||||
if (mem_handle >= 0)
|
||||
core_free(mem_handle);
|
||||
return rv;
|
||||
}
|
||||
|
||||
static int zip_read_cd(struct zip* z, bool use_cb) {
|
||||
const struct zip_ed* ed = &z->ed;
|
||||
const uint32_t cd_size = ed->cd_size;
|
||||
const uint32_t cd_offset = ed->cd_offset;
|
||||
const uint16_t cd_entries = ed->cd_entries;
|
||||
const uint32_t sig = htole32(ZIP_SIG_CD);
|
||||
int cds_handle = -1;
|
||||
int mem_handle = -1;
|
||||
struct zip_cd* cds;
|
||||
uint8_t* mem;
|
||||
struct zip_lf* lf = &z->lf;
|
||||
struct zip_args* args = &z->args;
|
||||
struct zip_cd_disk* cdd;
|
||||
struct zip_cd* cd;
|
||||
uint16_t name_length;
|
||||
int rv;
|
||||
|
||||
z->state = ZIP_STATE_CD_ENTER;
|
||||
|
||||
if ((cds_handle = zip_core_alloc(sizeof(struct zip_cd) * cd_entries)) < 0) {
|
||||
rv = -7;
|
||||
goto bail;
|
||||
}
|
||||
|
||||
if ((mem_handle = zip_core_alloc(cd_size)) < 0) {
|
||||
rv = -8;
|
||||
goto bail;
|
||||
}
|
||||
|
||||
cds = core_get_data(cds_handle);
|
||||
mem = core_get_data(mem_handle);
|
||||
|
||||
if (z->seek(z, cd_offset, SEEK_SET) < 0) {
|
||||
rv = -9;
|
||||
goto bail;
|
||||
}
|
||||
|
||||
if (z->read(z, mem, cd_size) != (ssize_t) cd_size) {
|
||||
rv = -10;
|
||||
goto bail;
|
||||
}
|
||||
|
||||
if (use_cb) {
|
||||
args->entries = cd_entries;
|
||||
args->name = lf->name;
|
||||
args->block = NULL;
|
||||
args->block_size = 0;
|
||||
args->read_size = 0;
|
||||
}
|
||||
|
||||
cdd = (struct zip_cd_disk*) mem;
|
||||
|
||||
for (uint16_t i = 0; i < cd_entries; i++) {
|
||||
if (cdd->signature != sig) {
|
||||
rv = -11;
|
||||
goto bail;
|
||||
}
|
||||
|
||||
cd = &cds[i];
|
||||
|
||||
cd->crc = letoh32(cdd->crc);
|
||||
cd->compressed_size = letoh32(cdd->compressed_size);
|
||||
cd->uncompressed_size = letoh32(cdd->uncompressed_size);
|
||||
cd->lf_offset = letoh32(cdd->lf_offset);
|
||||
|
||||
mem += sizeof(struct zip_cd_disk);
|
||||
name_length = letoh16(cdd->name_length);
|
||||
if (use_cb) {
|
||||
if (name_length >= sizeof(lf->name)) {
|
||||
rv = -12;
|
||||
goto bail;
|
||||
}
|
||||
|
||||
args->entry = i + 1;
|
||||
args->file_size = cd->uncompressed_size;
|
||||
args->modts = dostime_mktime(letoh16(cdd->date), letoh16(cdd->time));
|
||||
|
||||
memcpy(lf->name, mem, name_length);
|
||||
lf->name[name_length] = '\0';
|
||||
|
||||
if ((rv = z->cb(args, ZIP_PASS_SHALLOW, z->ctx)) > 0)
|
||||
goto bail;
|
||||
}
|
||||
mem += name_length;
|
||||
mem += letoh16(cdd->extra_length);
|
||||
mem += letoh16(cdd->comment_length);
|
||||
cdd = (struct zip_cd_disk*) mem;
|
||||
}
|
||||
|
||||
z->cds_handle = cds_handle;
|
||||
z->cds = cds;
|
||||
z->state = ZIP_STATE_CD_EXIT;
|
||||
rv = 0;
|
||||
|
||||
bail:
|
||||
if (rv != 0 && cds_handle >= 0)
|
||||
core_free(cds_handle);
|
||||
if (mem_handle >= 0)
|
||||
core_free(mem_handle);
|
||||
return rv;
|
||||
}
|
||||
|
||||
static int zip_read_lf(struct zip* z, uint16_t i) {
|
||||
const uint32_t sig = htole32(ZIP_SIG_LF);
|
||||
const struct zip_cd* cd = &z->cds[i];
|
||||
struct zip_lf* lf = &z->lf;
|
||||
struct zip_lf_disk lfd;
|
||||
uint16_t name_length;
|
||||
|
||||
if (z->seek(z, cd->lf_offset, SEEK_SET) < 0)
|
||||
return -14;
|
||||
|
||||
if (z->read(z, &lfd, sizeof(struct zip_lf_disk)) != sizeof(struct zip_lf_disk))
|
||||
return -15;
|
||||
|
||||
if (lfd.signature != sig)
|
||||
return -16;
|
||||
|
||||
name_length = letoh16(lfd.name_length);
|
||||
|
||||
if (name_length >= sizeof(lf->name))
|
||||
return -17;
|
||||
|
||||
if (z->read(z, lf->name, name_length) != name_length)
|
||||
return -18;
|
||||
|
||||
if (z->seek(z, letoh16(lfd.extra_length), SEEK_CUR) < 0)
|
||||
return -19;
|
||||
|
||||
lf->flags = letoh16(lfd.flags);
|
||||
lf->method = letoh16(lfd.method);
|
||||
lf->time = letoh16(lfd.time);
|
||||
lf->date = letoh16(lfd.date);
|
||||
lf->crc = letoh32(lfd.crc);
|
||||
lf->compressed_size = letoh32(lfd.compressed_size);
|
||||
lf->uncompressed_size = letoh32(lfd.uncompressed_size);
|
||||
lf->name_length = name_length;
|
||||
lf->name[name_length] = '\0';
|
||||
|
||||
if ((lf->flags & ZIP_BIT_DD) == ZIP_BIT_DD) {
|
||||
lf->crc = cd->crc;
|
||||
lf->compressed_size = cd->compressed_size;
|
||||
lf->uncompressed_size = cd->uncompressed_size;
|
||||
}
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
static int zip_read_store(struct zip* z, void* mem, uint32_t mem_size) {
|
||||
const struct zip_lf* lf = &z->lf;
|
||||
struct zip_args* args = &z->args;
|
||||
uint32_t file_size = lf->uncompressed_size;
|
||||
uint32_t block_size = mem_size;
|
||||
uint32_t crc = 0xffffffff;
|
||||
int rv;
|
||||
|
||||
if (lf->compressed_size != lf->uncompressed_size)
|
||||
return -21;
|
||||
|
||||
args->block = mem;
|
||||
args->block_size = block_size;
|
||||
args->read_size = 0;
|
||||
|
||||
do {
|
||||
if (block_size > file_size) {
|
||||
args->block_size = block_size = file_size;
|
||||
}
|
||||
|
||||
if (z->read(z, mem, block_size) != (off_t) block_size)
|
||||
return -22;
|
||||
|
||||
args->read_size += block_size;
|
||||
crc = crc_32r(mem, block_size, crc);
|
||||
|
||||
if ((rv = z->cb(args, ZIP_PASS_DATA, z->ctx)) != 0)
|
||||
return (rv < 0) ? 0 : rv;
|
||||
|
||||
file_size -= block_size;
|
||||
} while (file_size > 0);
|
||||
|
||||
if (~crc != lf->crc)
|
||||
return -24;
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
static int zip_read_entry(struct zip* z, uint16_t i, void* mem, uint32_t mem_size) {
|
||||
const struct zip_lf* lf = &z->lf;
|
||||
struct zip_args* args = &z->args;
|
||||
int rv;
|
||||
|
||||
if ((rv = zip_read_lf(z, i)) != 0)
|
||||
return rv;
|
||||
|
||||
args->entry = i + 1;
|
||||
args->file_size = lf->uncompressed_size;
|
||||
args->modts = dostime_mktime(lf->date, lf->time);
|
||||
args->block = NULL;
|
||||
args->block_size = 0;
|
||||
args->read_size = 0;
|
||||
|
||||
if ((rv = z->cb(&z->args, ZIP_PASS_START, z->ctx)) != 0)
|
||||
return (rv < 0) ? 0 : rv;
|
||||
|
||||
if (lf->uncompressed_size == 0)
|
||||
goto skip_data;
|
||||
|
||||
if (lf->method == ZIP_METHOD_STORE) {
|
||||
if ((rv = zip_read_store(z, mem, mem_size)) != 0)
|
||||
return rv;
|
||||
} else {
|
||||
return -20;
|
||||
}
|
||||
|
||||
skip_data:
|
||||
args->block = NULL;
|
||||
args->block_size = 0;
|
||||
args->read_size = 0;
|
||||
|
||||
if ((rv = z->cb(args, ZIP_PASS_END, z->ctx)) != 0)
|
||||
return (rv < 0) ? 0 : rv;
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
static int zip_read_entries(struct zip* z) {
|
||||
const struct zip_ed* ed = &z->ed;
|
||||
const uint16_t cd_entries = ed->cd_entries;
|
||||
struct zip_lf* lf = &z->lf;
|
||||
struct zip_args* args = &z->args;
|
||||
uint32_t mem_size = ZIP_BUFFER_SIZE;
|
||||
int mem_handle;
|
||||
void* mem;
|
||||
int rv;
|
||||
|
||||
z->state = ZIP_STATE_LF_ENTER;
|
||||
|
||||
if ((mem_handle = zip_core_alloc(mem_size)) < 0) {
|
||||
rv = -13;
|
||||
goto bail;
|
||||
}
|
||||
|
||||
mem = core_get_data(mem_handle);
|
||||
|
||||
args->entries = cd_entries;
|
||||
args->name = lf->name;
|
||||
|
||||
for (uint16_t i = 0; i < cd_entries; i++)
|
||||
if ((rv = zip_read_entry(z, i, mem, mem_size)) > 0)
|
||||
goto bail;
|
||||
|
||||
z->state = ZIP_STATE_LF_EXIT;
|
||||
rv = 0;
|
||||
|
||||
bail:
|
||||
if (mem_handle >= 0)
|
||||
core_free(mem_handle);
|
||||
return rv;
|
||||
}
|
||||
|
||||
static void zip_init(struct zip* z, int zip_handle) {
|
||||
z->zip_handle = zip_handle;
|
||||
z->state = ZIP_STATE_INITIAL;
|
||||
z->cb = NULL;
|
||||
memset(&z->args, 0, sizeof(struct zip_args));
|
||||
z->ctx = NULL;
|
||||
memset(&z->ed, 0, sizeof(struct zip_ed));
|
||||
z->cds_handle = -1;
|
||||
z->cds = NULL;
|
||||
memset(&z->lf, 0, sizeof(struct zip_lf));
|
||||
}
|
||||
|
||||
static ssize_t zip_file_read(struct zip* zh, void* mem, size_t mem_size) {
|
||||
struct zip_file* z = (struct zip_file*) zh;
|
||||
|
||||
return read(z->file, mem, mem_size);
|
||||
}
|
||||
|
||||
static off_t zip_file_seek(struct zip* zh, off_t offset, int whence) {
|
||||
struct zip_file* z = (struct zip_file*) zh;
|
||||
|
||||
return lseek(z->file, offset, whence);
|
||||
}
|
||||
|
||||
static off_t zip_file_size(struct zip* zh) {
|
||||
struct zip_file* z = (struct zip_file*) zh;
|
||||
|
||||
return filesize(z->file);
|
||||
}
|
||||
|
||||
static void zip_file_close(struct zip* zh) {
|
||||
struct zip_file* z = (struct zip_file*) zh;
|
||||
|
||||
close(z->file);
|
||||
}
|
||||
|
||||
static void zip_file_init(struct zip_file* z, int zip_handle, int file) {
|
||||
struct zip* zh = &z->base;
|
||||
|
||||
zh->read = zip_file_read;
|
||||
zh->seek = zip_file_seek;
|
||||
zh->size = zip_file_size;
|
||||
zh->close = zip_file_close;
|
||||
zip_init(zh, zip_handle);
|
||||
|
||||
z->file = file;
|
||||
}
|
||||
|
||||
static ssize_t zip_mem_read(struct zip* zh, void* mem, size_t mem_size) {
|
||||
struct zip_mem* z = (struct zip_mem*) zh;
|
||||
off_t bytes = z->mem_size - z->mem_offset;
|
||||
off_t read_size = MIN(bytes, (off_t) mem_size);
|
||||
|
||||
memcpy(mem, z->mem + z->mem_offset, read_size);
|
||||
z->mem_offset += read_size;
|
||||
|
||||
return read_size;
|
||||
}
|
||||
|
||||
static off_t zip_mem_seek(struct zip* zh, off_t offset, int whence) {
|
||||
struct zip_mem* z = (struct zip_mem*) zh;
|
||||
off_t new_offset;
|
||||
|
||||
switch (whence) {
|
||||
case SEEK_SET:
|
||||
new_offset = offset;
|
||||
break;
|
||||
|
||||
case SEEK_CUR:
|
||||
new_offset = z->mem_offset + offset;
|
||||
break;
|
||||
|
||||
case SEEK_END:
|
||||
new_offset = z->mem_size + offset;
|
||||
break;
|
||||
|
||||
default:
|
||||
new_offset = -1;
|
||||
break;
|
||||
}
|
||||
|
||||
if (new_offset < 0 || new_offset > z->mem_size)
|
||||
return -1;
|
||||
|
||||
z->mem_offset = new_offset;
|
||||
|
||||
return new_offset;
|
||||
}
|
||||
|
||||
static off_t zip_mem_size(struct zip* zh) {
|
||||
struct zip_mem* z = (struct zip_mem*) zh;
|
||||
|
||||
return z->mem_size;
|
||||
}
|
||||
|
||||
static void zip_mem_close(struct zip* zh) {
|
||||
struct zip_mem* z = (struct zip_mem*) zh;
|
||||
|
||||
core_free(z->mem_handle);
|
||||
}
|
||||
|
||||
static void zip_mem_init(struct zip_mem* z, int zip_handle, int mem_handle, const void* mem, off_t mem_size) {
|
||||
struct zip* zh = &z->base;
|
||||
|
||||
zh->read = zip_mem_read;
|
||||
zh->seek = zip_mem_seek;
|
||||
zh->size = zip_mem_size;
|
||||
zh->close = zip_mem_close;
|
||||
zip_init(zh, zip_handle);
|
||||
|
||||
z->mem_handle = mem_handle;
|
||||
z->mem = mem;
|
||||
z->mem_offset = 0;
|
||||
z->mem_size = mem_size;
|
||||
}
|
||||
|
||||
struct zip* zip_open(const char* name, bool try_mem) {
|
||||
int file = -1;
|
||||
int mem_handle = -1;
|
||||
int zip_handle = -1;
|
||||
off_t mem_size;
|
||||
void* mem;
|
||||
void* zip;
|
||||
|
||||
if (name == NULL || name[0] == '\0')
|
||||
goto bail;
|
||||
|
||||
if ((file = open(name, O_RDONLY)) < 0)
|
||||
goto bail;
|
||||
|
||||
if (try_mem && (mem_handle = zip_core_alloc(mem_size = filesize(file))) >= 0) {
|
||||
if ((zip_handle = zip_core_alloc(sizeof(struct zip_mem))) < 0)
|
||||
goto bail;
|
||||
|
||||
mem = core_get_data(mem_handle);
|
||||
|
||||
if (read(file, mem, mem_size) != mem_size)
|
||||
goto bail;
|
||||
|
||||
close(file);
|
||||
|
||||
zip = core_get_data(zip_handle);
|
||||
|
||||
zip_mem_init(zip, zip_handle, mem_handle, mem, mem_size);
|
||||
} else {
|
||||
if ((zip_handle = zip_core_alloc(sizeof(struct zip_file))) < 0)
|
||||
goto bail;
|
||||
|
||||
zip = core_get_data(zip_handle);
|
||||
|
||||
zip_file_init(zip, zip_handle, file);
|
||||
}
|
||||
|
||||
return zip;
|
||||
|
||||
bail:
|
||||
if (file >= 0)
|
||||
close(file);
|
||||
if (mem_handle >= 0)
|
||||
core_free(mem_handle);
|
||||
if (zip_handle >= 0)
|
||||
core_free(zip_handle);
|
||||
return NULL;
|
||||
}
|
||||
|
||||
int zip_read_shallow(struct zip* z, zip_callback cb, void* ctx) {
|
||||
int rv;
|
||||
|
||||
if (z == NULL || z->state != ZIP_STATE_INITIAL || cb == NULL)
|
||||
return -1;
|
||||
|
||||
z->cb = cb;
|
||||
z->ctx = ctx;
|
||||
|
||||
if ((rv = zip_read_ed(z)) != 0)
|
||||
return rv;
|
||||
|
||||
return zip_read_cd(z, true);
|
||||
}
|
||||
|
||||
int zip_read_deep(struct zip* z, zip_callback cb, void* ctx) {
|
||||
int rv;
|
||||
|
||||
if (z == NULL || (z->state != ZIP_STATE_INITIAL && z->state != ZIP_STATE_CD_EXIT) || cb == NULL)
|
||||
return -1;
|
||||
|
||||
z->cb = cb;
|
||||
z->ctx = ctx;
|
||||
|
||||
if (z->state == ZIP_STATE_CD_EXIT)
|
||||
goto read_entries;
|
||||
|
||||
if ((rv = zip_read_ed(z)) != 0)
|
||||
return rv;
|
||||
|
||||
if ((rv = zip_read_cd(z, false)) != 0)
|
||||
return rv;
|
||||
|
||||
read_entries:
|
||||
return zip_read_entries(z);
|
||||
}
|
||||
|
||||
void zip_close(struct zip* z) {
|
||||
if (z == NULL)
|
||||
return;
|
||||
|
||||
z->close(z);
|
||||
|
||||
if (z->cds_handle >= 0)
|
||||
core_free(z->cds_handle);
|
||||
|
||||
core_free(z->zip_handle);
|
||||
}
|
64
firmware/include/zip.h
Normal file
64
firmware/include/zip.h
Normal file
|
@ -0,0 +1,64 @@
|
|||
/***************************************************************************
|
||||
* __________ __ ___.
|
||||
* Open \______ \ ____ ____ | | _\_ |__ _______ ___
|
||||
* Source | _// _ \_/ ___\| |/ /| __ \ / _ \ \/ /
|
||||
* Jukebox | | ( <_> ) \___| < | \_\ ( <_> > < <
|
||||
* Firmware |____|_ /\____/ \___ >__|_ \|___ /\____/__/\_ \
|
||||
* \/ \/ \/ \/ \/
|
||||
* $Id$
|
||||
*
|
||||
* Copyright (C) 2021 by James Buren
|
||||
*
|
||||
* This program is free software; you can redistribute it and/or
|
||||
* modify it under the terms of the GNU General Public License
|
||||
* as published by the Free Software Foundation; either version 2
|
||||
* of the License, or (at your option) any later version.
|
||||
*
|
||||
* This software is distributed on an "AS IS" basis, WITHOUT WARRANTY OF ANY
|
||||
* KIND, either express or implied.
|
||||
*
|
||||
****************************************************************************/
|
||||
|
||||
#ifndef _ZIP_H_
|
||||
#define _ZIP_H_
|
||||
|
||||
#include <stdbool.h>
|
||||
#include <stdint.h>
|
||||
#include <time.h>
|
||||
|
||||
enum {
|
||||
ZIP_PASS_SHALLOW,
|
||||
ZIP_PASS_START,
|
||||
ZIP_PASS_DATA,
|
||||
ZIP_PASS_END,
|
||||
};
|
||||
|
||||
struct zip;
|
||||
|
||||
struct zip_args {
|
||||
uint16_t entry;
|
||||
uint16_t entries;
|
||||
char* name;
|
||||
uint32_t file_size;
|
||||
time_t modts;
|
||||
void* block;
|
||||
uint32_t block_size;
|
||||
uint32_t read_size;
|
||||
};
|
||||
|
||||
typedef int (*zip_callback) (const struct zip_args* args, int pass, void* ctx);
|
||||
|
||||
// open a handle for the given full file name path
|
||||
struct zip* zip_open(const char* name, bool try_mem);
|
||||
|
||||
// quick read of only directory index
|
||||
int zip_read_shallow(struct zip* z, zip_callback cb, void* ctx);
|
||||
|
||||
// slow read of whole archive
|
||||
// this can also pickup where a successful shallow read leftoff
|
||||
int zip_read_deep(struct zip* z, zip_callback cb, void* ctx);
|
||||
|
||||
// returns system resources
|
||||
void zip_close(struct zip* z);
|
||||
|
||||
#endif
|
Loading…
Reference in a new issue