X-Git-Url: https://git.draconx.ca/gitweb/liblbx.git/blobdiff_plain/cc58652bf808568b935ca755d4360aa0f1aedfed..c02dcc9610e66ebe6154e60dcd2f109c90eb09df:/src/lbx.c diff --git a/src/lbx.c b/src/lbx.c index fc52680..0e26315 100644 --- a/src/lbx.c +++ b/src/lbx.c @@ -1,7 +1,7 @@ /* * 2ooM: The Master of Orion II Reverse Engineering Project * Library for working with LBX archive files. - * Copyright (C) 2006-2008 Nick Bowler + * Copyright (C) 2006-2010 Nick Bowler * * 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 @@ -23,13 +23,9 @@ #include #include #include -#include #include #include -#include -#include - #include "pack.h" #include "misc.h" #include "lbx.h" @@ -40,15 +36,22 @@ int lbx_errno = 0; struct lbx_state { - const char *name; + char *name; - unsigned char *mem; - size_t memsize; - FILE *f; - long foff; + const struct lbx_file_ops *fops; + int (*dtor)(void *handle); + void *f; + + struct lbx_file_state *last_file; + + unsigned short nfiles; + unsigned long offsets[]; +}; - uint16_t nfiles; - uint32_t offsets[]; +struct lbx_file_state { + unsigned long base, limit, offset; + struct lbx_state *lbx; + int eof; }; static struct lbx_state *lbx_init(unsigned char hdr[static LBX_HDR_SIZE]) @@ -76,93 +79,110 @@ static struct lbx_state *lbx_init(unsigned char hdr[static LBX_HDR_SIZE]) return lbx; } -struct lbx_state *lbx_fopen(FILE *f, const char *name) +static char *str_dup(const char *s) +{ + char *buf; + + buf = malloc(strlen(s)+1); + if (buf) + strcpy(buf, s); + return buf; +} + +struct lbx_state *lbx_open(void *f, const struct lbx_file_ops *fops, + int (*destructor)(void *), const char *name) { unsigned char hdr_buf[LBX_HDR_SIZE]; - struct lbx_state *lbx; + struct lbx_state *lbx = NULL; + char *dupname = NULL; - if (fread(hdr_buf, 1, sizeof hdr_buf, f) != sizeof hdr_buf) { + dupname = str_dup(name); + if (!dupname) { lbx_errno = -errno; - return NULL; + goto err; + } + + if (fops->read(hdr_buf, sizeof hdr_buf, f) != sizeof hdr_buf) { + lbx_errno = -errno; + goto err; } lbx = lbx_init(hdr_buf); if (!lbx) - return NULL; + goto err; - lbx->name = name; + lbx->name = dupname; + lbx->dtor = destructor; + lbx->fops = fops; lbx->f = f; - lbx->foff = sizeof hdr_buf; for (unsigned i = 0; i <= lbx->nfiles; i++) { unsigned char buf[4]; - if (fread(buf, 1, sizeof buf, f) != sizeof buf) { + if (fops->read(buf, sizeof buf, f) != sizeof buf) { lbx_errno = -errno; - if (feof(f)) + if (fops->eof(f)) lbx_errno = LBX_EEOF; - free(lbx); - return NULL; + goto err; } lbx->offsets[i] = unpack_32_le(buf); - lbx->foff += sizeof buf; } return lbx; +err: + free(dupname); + free(lbx); + return NULL; } -static int _lbx_memcpy(void *dest, struct lbx_state *src, size_t size) +static int file_close(void *f) { - if (src->foff + size > src->memsize) - return -1; - memcpy(dest, src->mem + src->foff, size); - src->foff += size; - return 0; + return fclose((FILE *)f); } -struct lbx_state *lbx_mopen(void *_mem, size_t size, const char *name) +static int pipe_close(void *f) { - struct lbx_state *new = NULL; - struct lbx_state tmp = { .mem = _mem, .memsize = size }; - uint16_t nfiles, version; - uint32_t magic; + struct lbx_pipe_state *p = f; + int rc; - if (_lbx_memcpy(&nfiles, &tmp, sizeof nfiles) == -1) goto eof; - if (_lbx_memcpy(&magic, &tmp, sizeof magic) == -1) goto eof; - if (_lbx_memcpy(&version, &tmp, sizeof version) == -1) goto eof; + rc = fclose(p->f); + free(p); + return rc; +} - nfiles = letohs(nfiles); - magic = letohl(magic); - version = letohs(version); +static char *last_component(const char *name) +{ + char *c; - if (magic != LBX_MAGIC) { - lbx_errno = LBX_EMAGIC; + /* TODO: Handle other path separators. */ + c = strrchr(name, '/'); + if (!c) + return (char *)name; + return c+1; +} + +struct lbx_state *lbx_fopen(const char *file) +{ + const char *name = last_component(file); + struct lbx_pipe_state *p; + FILE *f; + + f = fopen(file, "rb"); + if (!f) return NULL; - } - - new = malloc(sizeof *new + (nfiles+1)*(sizeof *new->offsets)); - if (!new) { - lbx_errno = -errno; + + if (fseek(f, 0, SEEK_CUR) == 0) + return lbx_open(f, &lbx_default_fops, file_close, name); + + p = malloc(sizeof *p); + if (!p) { + fclose(f); return NULL; } - - *new = (struct lbx_state){ - .name = name, - .nfiles = nfiles, - .mem = _mem, - .memsize = size, - .foff = tmp.foff, - }; - - if (_lbx_memcpy(new->offsets, new, (nfiles+1)*(sizeof *new->offsets))) - goto eof; - - return new; -eof: - free(new); - lbx_errno = LBX_EEOF; - return NULL; + + *p = (struct lbx_pipe_state) { .f = f }; + return lbx_open(p, &lbx_pipe_fops, pipe_close, name); } size_t lbx_numfiles(struct lbx_state *lbx) @@ -170,100 +190,133 @@ size_t lbx_numfiles(struct lbx_state *lbx) return lbx->nfiles; } -int lbx_stat(struct lbx_state *lbx, size_t index, struct lbx_statbuf *buf) +int +lbx_file_stat(struct lbx_state *lbx, unsigned fileno, struct lbx_statbuf *buf) { static char str[256]; /* FIXME */ - if (index >= lbx->nfiles) { + if (fileno >= lbx->nfiles) { buf->name = NULL; lbx_errno = LBX_ERANGE; return -1; } - snprintf(str, sizeof str, "%s.%03zu", lbx->name, index); + snprintf(str, sizeof str, "%s.%03u", lbx->name, fileno); buf->name = str; - buf->size = lbx->offsets[index+1] - lbx->offsets[index]; + buf->size = lbx->offsets[fileno+1] - lbx->offsets[fileno]; return 0; } -static size_t -_lbx_mextract(struct lbx_state *lbx, size_t base, size_t len, FILE *of) +int lbx_close(struct lbx_state *lbx) { - size_t rc; - - assert(lbx->mem); - assert(base + len <= lbx->memsize); + int rc = 0; - rc = fwrite(lbx->mem + base, 1, len, of); - if (rc < len) - lbx_errno = -errno; + if (lbx && lbx->dtor) + rc = lbx->dtor(lbx->f); + free(lbx->name); + free(lbx); return rc; } -static size_t -_lbx_fextract(struct lbx_state *lbx, size_t base, size_t len, FILE *of) +struct lbx_file_state *lbx_file_open(struct lbx_state *lbx, unsigned fileno) { - unsigned char buf[1024]; - size_t rc, written = 0; - - assert(lbx->f); - - if (_lbx_fseek(lbx->f, &lbx->foff, base) == -1) - return 0; - - while (len) { - size_t amt = MIN(len, sizeof buf); - - rc = fread(buf, 1, amt, lbx->f); - lbx->foff += rc; - len -= rc; - if (rc < amt) { - if (feof(lbx->f)) lbx_errno = LBX_EEOF; - else lbx_errno = -errno; - break; - } + struct lbx_file_state *state; - rc = fwrite(buf, 1, amt, of); - written += rc; - if (rc < amt) { - lbx_errno = -errno; - break; - } + if (fileno >= lbx->nfiles) { + lbx_errno = LBX_ERANGE; + return NULL; + } + + lbx->last_file = NULL; + if (lbx->fops->seek(lbx->f, lbx->offsets[fileno], SEEK_SET) != 0) { + lbx_errno = -errno; + return NULL; + } + + state = malloc(sizeof *state); + if (!state) { + lbx_errno = -errno; + return NULL; } - return written; + *state = (struct lbx_file_state) { + .base = lbx->offsets[fileno], + .limit = lbx->offsets[fileno+1] - lbx->offsets[fileno], + .lbx = lbx, + }; + + lbx->last_file = state; + return state; } -size_t lbx_extract(struct lbx_state *lbx, size_t index, FILE *of) +size_t lbx_file_read(struct lbx_file_state *f, void *buf, size_t n) { - size_t base, len; + const struct lbx_file_ops *fops = f->lbx->fops; + size_t want = MIN(n, f->limit - f->offset); + size_t rc; - if (index >= lbx->nfiles) { - lbx_errno = LBX_ERANGE; - return 0; + if (f != f->lbx->last_file) { + f->lbx->last_file = NULL; + if (fops->seek(f->lbx->f, f->base + f->limit, SEEK_SET) != 0) + return 0; + f->lbx->last_file = f; } - - base = lbx->offsets[index]; - len = lbx->offsets[index+1] - lbx->offsets[index]; - if (lbx->mem) - return _lbx_mextract(lbx, base, len, of); - return _lbx_fextract(lbx, base, len, of); + rc = fops->read(buf, want, f->lbx->f); + f->offset += rc; + + if (want < n || (rc < want && fops->eof(f->lbx->f))) + f->eof = 1; + return rc; } -int lbx_close(struct lbx_state *lbx) +int lbx_file_seek(struct lbx_file_state *f, long offset, int whence) { - int rc = 0; + const struct lbx_file_ops *fops = f->lbx->fops; + unsigned long pos; + + switch (whence) { + case SEEK_CUR: + pos = f->offset + offset; + break; + case SEEK_SET: + pos = offset; + break; + case SEEK_END: + pos = f->limit + offset; + break; + } - if (!lbx) - return 0; + if (pos > f->limit) + return -1; - if (lbx->f) - rc = fclose(lbx->f); + f->lbx->last_file = NULL; + if (fops->seek(f->lbx->f, f->base + pos, SEEK_SET) != 0) + return -1; - free(lbx); - return rc; + f->offset = pos; + f->lbx->last_file = f; + f->eof = 0; + + return 0; +} + +long lbx_file_tell(struct lbx_file_state *f) +{ + return f->offset; +} + +int lbx_file_eof(struct lbx_file_state *f) +{ + return f->eof; +} + +void lbx_file_close(struct lbx_file_state *f) +{ + if (f->lbx->last_file == f) + f->lbx->last_file = NULL; + free(f); } const char *lbx_strerror(void)