GLK: Added Blorb container file handling

This commit is contained in:
Paul Gilbert 2018-11-10 21:22:57 -08:00 committed by Paul Gilbert
parent 4497a55a86
commit b8fcb62dd2
4 changed files with 692 additions and 1 deletions

543
engines/gargoyle/blorb.cpp Normal file

@ -0,0 +1,543 @@
/* ScummVM - Graphic Adventure Engine
*
* ScummVM is the legal property of its developers, whose names
* are too numerous to list here. Please refer to the COPYRIGHT
* file distributed with this source distribution.
*
* 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 program is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY{} without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU General Public License for more details.
*
* You should have received a copy of the GNU General Public License
* along with this program{} if not, write to the Free Software
* Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
*
*/
#include "gargoyle/blorb.h"
namespace Gargoyle {
#define giblorb_Inited_Magic 0xB7012BEDU
/**
* Describes one chunk of the Blorb file.
*/
struct giblorb_chunkdesc_struct {
glui32 type;
glui32 len;
glui32 startpos; ///< start of chunk header
glui32 datpos; ///< start of data (either startpos or startpos+8)
void *ptr; ///< pointer to malloc'd data, if loaded
int auxdatnum; ///< entry in the auxsound/auxpict array; -1 if none. This only applies to chunks that represent resources;
};
typedef giblorb_chunkdesc_struct giblorb_chunkdesc_t;
/**
* Describes one resource in the Blorb file.
*/
struct giblorb_resdesc_struct {
glui32 usage;
glui32 resnum;
glui32 chunknum;
};
/**
* Holds the complete description of an open Blorb file.
*/
struct giblorb_map_struct {
glui32 inited; ///< holds giblorb_Inited_Magic if the map structure is valid
Common::SeekableReadStream *file;
uint numchunks;
giblorb_chunkdesc_t *chunks; ///< list of chunk descriptors
int numresources;
giblorb_resdesc_t *resources; ///< list of resource descriptors
giblorb_resdesc_t **ressorted; ///< list of pointers to descriptors in map->resources -- sorted by usage and resource number.
};
/*--------------------------------------------------------------------------*/
giblorb_err_t Blorb::giblorb_initialize() {
_file = nullptr;
_map = nullptr;
return giblorb_err_None;
}
giblorb_err_t Blorb::giblorb_create_map(Common::SeekableReadStream *file, giblorb_map_t **newmap) {
giblorb_err_t err;
giblorb_map_t *map;
glui32 readlen;
glui32 nextpos, totallength;
giblorb_chunkdesc_t *chunks;
int chunks_size, numchunks;
char buffer[16];
*newmap = nullptr;
if (!_libInited) {
err = giblorb_initialize();
if (err)
return err;
_libInited = true;
}
/* First, chew through the file and index the chunks. */
file->seek(0);
readlen = file->read(buffer, 12);
if (readlen != 12)
return giblorb_err_Read;
if (READ_BE_INT32(buffer + 0) != giblorb_ID_FORM)
return giblorb_err_Format;
if (READ_BE_INT32(buffer + 8) != giblorb_ID_IFRS)
return giblorb_err_Format;
totallength = READ_BE_INT32(buffer + 4) + 8;
nextpos = 12;
chunks_size = 8;
numchunks = 0;
chunks = new giblorb_chunkdesc_t[chunks_size];
while (nextpos < totallength) {
glui32 type, len;
int chunum;
giblorb_chunkdesc_t *chu;
file->seek(nextpos);
readlen = file->read(buffer, 8);
if (readlen != 8) {
delete[] chunks;
return giblorb_err_Read;
}
type = READ_BE_INT32(buffer + 0);
len = READ_BE_INT32(buffer + 4);
if (numchunks >= chunks_size) {
chunks_size *= 2;
chunks = new giblorb_chunkdesc_t[chunks_size];
}
chunum = numchunks;
chu = &(chunks[chunum]);
numchunks++;
chu->type = type;
chu->startpos = nextpos;
if (type == giblorb_ID_FORM) {
chu->datpos = nextpos;
chu->len = len + 8;
} else {
chu->datpos = nextpos + 8;
chu->len = len;
}
chu->ptr = nullptr;
chu->auxdatnum = -1;
nextpos = nextpos + len + 8;
if (nextpos & 1)
nextpos++;
if (nextpos > totallength) {
delete[] chunks;
return giblorb_err_Format;
}
}
// The basic IFF structure seems to be ok, and we have a list of chunks.
// Now we allocate the map structure itself.
map = new giblorb_map_t();
if (!map) {
delete[] chunks;
return giblorb_err_Alloc;
}
map->inited = giblorb_Inited_Magic;
map->file = file;
map->chunks = chunks;
map->numchunks = numchunks;
map->resources = nullptr;
map->ressorted = nullptr;
map->numresources = 0;
// Now we do everything else involved in loading the Blorb file,
// such as building resource lists.
err = giblorb_initialize_map(map);
if (err) {
giblorb_destroy_map(map);
return err;
}
*newmap = map;
return giblorb_err_None;
}
giblorb_err_t Blorb::giblorb_initialize_map(giblorb_map_t *map) {
// It is important that the map structure be kept valid during this function.
// If this returns an error, giblorb_destroy_map() will be called.
uint ix, jx;
giblorb_result_t chunkres;
giblorb_err_t err;
char *ptr;
glui32 len;
glui32 numres;
int gotindex = false;
for (ix = 0; ix<map->numchunks; ix++) {
giblorb_chunkdesc_t *chu = &map->chunks[ix];
switch (chu->type) {
case giblorb_ID_RIdx:
// Resource index chunk: build the resource list and sort it.
if (gotindex)
return giblorb_err_Format; // duplicate index chunk
err = giblorb_load_chunk_by_number(map, giblorb_method_Memory, &chunkres, ix);
if (err)
return err;
ptr = (char *)chunkres.data.ptr;
len = chunkres.length;
numres = READ_BE_INT32(ptr + 0);
if (numres) {
uint ix2;
giblorb_resdesc_t *resources;
giblorb_resdesc_t **ressorted;
if (len != numres * 12 + 4)
return giblorb_err_Format; // bad length field
resources = new giblorb_resdesc_t[numres];
ressorted = new giblorb_resdesc_t *[numres];
if (!ressorted || !resources) {
delete[] resources;
delete[] ressorted;
return giblorb_err_Alloc;
}
ix2 = 0;
for (jx = 0; jx < numres; jx++) {
giblorb_resdesc_t *res = &(resources[jx]);
glui32 respos;
res->usage = READ_BE_INT32(ptr + jx * 12 + 4);
res->resnum = READ_BE_INT32(ptr + jx * 12 + 8);
respos = READ_BE_INT32(ptr + jx * 12 + 12);
while (ix2 < map->numchunks
&& map->chunks[ix2].startpos < respos)
ix2++;
if (ix2 >= map->numchunks
|| map->chunks[ix2].startpos != respos) {
delete[] resources;
delete[] ressorted;
return giblorb_err_Format; // start pos does not match a real chunk
}
res->chunknum = ix2;
ressorted[jx] = res;
}
// Sort a resource list (actually a list of pointers to structures in map->resources.)
// This makes it easy to find resources by usage and resource number.
giblorb_qsort(ressorted, numres);
map->numresources = numres;
map->resources = resources;
map->ressorted = ressorted;
}
giblorb_unload_chunk(map, ix);
gotindex = true;
break;
}
}
return giblorb_err_None;
}
void Blorb::giblorb_qsort(giblorb_resdesc_t **list, size_t len) {
int ix, jx, res;
giblorb_resdesc_t *tmpptr, *pivot;
if (len < 6) {
// The list is short enough for a bubble-sort.
for (jx = len - 1; jx>0; jx--) {
for (ix = 0; ix<jx; ix++) {
res = sortsplot(list[ix], list[ix + 1]);
if (res > 0) {
tmpptr = list[ix];
list[ix] = list[ix + 1];
list[ix + 1] = tmpptr;
}
}
}
} else {
// Split the list.
pivot = list[len / 2];
ix = 0;
jx = len;
for (;;) {
while (ix < jx - 1 && sortsplot(list[ix], pivot) < 0)
ix++;
while (ix < jx - 1 && sortsplot(list[jx - 1], pivot) > 0)
jx--;
if (ix >= jx - 1)
break;
tmpptr = list[ix];
list[ix] = list[jx - 1];
list[jx - 1] = tmpptr;
}
ix++;
// Sort the halves.
giblorb_qsort(list + 0, ix);
giblorb_qsort(list + ix, len - ix);
}
}
giblorb_resdesc_t *Blorb::giblorb_bsearch(giblorb_resdesc_t *sample,
giblorb_resdesc_t **list, int len) {
int top, bot, val, res;
bot = 0;
top = len;
while (bot < top) {
val = (top + bot) / 2;
res = sortsplot(list[val], sample);
if (res == 0)
return list[val];
if (res < 0) {
bot = val + 1;
} else {
top = val;
}
}
return nullptr;
}
int Blorb::sortsplot(giblorb_resdesc_t *v1, giblorb_resdesc_t *v2) {
if (v1->usage < v2->usage)
return -1;
if (v1->usage > v2->usage)
return 1;
if (v1->resnum < v2->resnum)
return -1;
if (v1->resnum > v2->resnum)
return 1;
return 0;
}
giblorb_err_t Blorb::giblorb_destroy_map(giblorb_map_t *map) {
if (!map || !map->chunks || map->inited != giblorb_Inited_Magic)
return giblorb_err_NotAMap;
for (uint ix = 0; ix<map->numchunks; ix++) {
giblorb_chunkdesc_t *chu = &(map->chunks[ix]);
if (chu->ptr) {
delete chu->ptr;
chu->ptr = nullptr;
}
}
if (map->chunks) {
delete[] map->chunks;
map->chunks = nullptr;
}
map->numchunks = 0;
if (map->resources) {
delete[] map->resources;
map->resources = nullptr;
}
if (map->ressorted) {
delete[] map->ressorted;
map->ressorted = nullptr;
}
map->numresources = 0;
map->file = nullptr;
map->inited = 0;
delete map;
return giblorb_err_None;
}
giblorb_err_t Blorb::giblorb_load_chunk_by_type(giblorb_map_t *map,
glui32 method, giblorb_result_t *res, glui32 chunktype, glui32 count) {
uint ix;
for (ix = 0; ix < map->numchunks; ix++) {
if (map->chunks[ix].type == chunktype) {
if (count == 0)
break;
count--;
}
}
if (ix >= map->numchunks) {
return giblorb_err_NotFound;
}
return giblorb_load_chunk_by_number(map, method, res, ix);
}
giblorb_err_t Blorb::giblorb_load_chunk_by_number(giblorb_map_t *map,
glui32 method, giblorb_result_t *res, glui32 chunknum) {
giblorb_chunkdesc_t *chu;
if (chunknum < 0 || chunknum >= map->numchunks)
return giblorb_err_NotFound;
chu = &(map->chunks[chunknum]);
switch (method) {
case giblorb_method_DontLoad:
// do nothing
break;
case giblorb_method_FilePos:
res->data.startpos = chu->datpos;
break;
case giblorb_method_Memory:
if (!chu->ptr) {
glui32 readlen;
byte *dat = new byte[chu->len];
if (!dat)
return giblorb_err_Alloc;
map->file->seek(chu->datpos);
readlen = map->file->read(dat, chu->len);
if (readlen != chu->len)
return giblorb_err_Read;
chu->ptr = dat;
}
res->data.ptr = chu->ptr;
break;
}
res->chunknum = chunknum;
res->length = chu->len;
res->chunktype = chu->type;
return giblorb_err_None;
}
giblorb_err_t Blorb::giblorb_unload_chunk(giblorb_map_t *map, glui32 chunknum) {
giblorb_chunkdesc_t *chu;
if (chunknum < 0 || chunknum >= map->numchunks)
return giblorb_err_NotFound;
chu = &(map->chunks[chunknum]);
if (chu->ptr) {
delete chu->ptr;
chu->ptr = nullptr;
}
return giblorb_err_None;
}
giblorb_err_t Blorb::giblorb_load_resource(giblorb_map_t *map, glui32 method,
giblorb_result_t *res, glui32 usage, glui32 resnum) {
giblorb_resdesc_t sample;
giblorb_resdesc_t *found;
sample.usage = usage;
sample.resnum = resnum;
found = giblorb_bsearch(&sample, map->ressorted, map->numresources);
if (!found)
return giblorb_err_NotFound;
return giblorb_load_chunk_by_number(map, method, res, found->chunknum);
}
giblorb_err_t Blorb::giblorb_count_resources(giblorb_map_t *map,
glui32 usage, glui32 *num, glui32 *min, glui32 *max) {
int ix;
int count;
glui32 val;
glui32 minval, maxval;
count = 0;
minval = 0;
maxval = 0;
for (ix = 0; ix<map->numresources; ix++) {
if (map->resources[ix].usage == usage) {
val = map->resources[ix].resnum;
if (count == 0) {
count++;
minval = val;
maxval = val;
}
else {
count++;
if (val < minval)
minval = val;
if (val > maxval)
maxval = val;
}
}
}
if (num)
*num = count;
if (min)
*min = minval;
if (max)
*max = maxval;
return giblorb_err_None;
}
giblorb_err_t Blorb::giblorb_set_resource_map(Common::SeekableReadStream *file) {
giblorb_err_t err;
err = giblorb_create_map(file, &_map);
if (err) {
_map = nullptr;
return err;
}
_file = file;
return giblorb_err_None;
}
giblorb_map_t *Blorb::giblorb_get_resource_map(void) {
return _map;
}
bool Blorb::giblorb_is_resource_map(void) const {
return _map != nullptr;
}
} // End of namespace Gargoyle

146
engines/gargoyle/blorb.h Normal file

@ -0,0 +1,146 @@
/* ScummVM - Graphic Adventure Engine
*
* ScummVM is the legal property of its developers, whose names
* are too numerous to list here. Please refer to the COPYRIGHT
* file distributed with this source distribution.
*
* 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 program is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU General Public License for more details.
*
* You should have received a copy of the GNU General Public License
* along with this program; if not, write to the Free Software
* Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
*
*/
#ifndef GARGOYLE_BLORB_H
#define GARGOYLE_BLORB_H
#include "gargoyle/glk_types.h"
#include "gargoyle/streams.h"
namespace Gargoyle {
/**
* Error type
*/
typedef glui32 giblorb_err_t;
/**
* Error codes
*/
enum giblorbError {
giblorb_err_None = 0,
giblorb_err_CompileTime = 1,
giblorb_err_Alloc = 2,
giblorb_err_Read = 3,
giblorb_err_NotAMap = 4,
giblorb_err_Format = 5,
giblorb_err_NotFound = 6
};
/**
* Methods for loading a chunk
*/
enum giblorbMethod {
giblorb_method_DontLoad = 0,
giblorb_method_Memory = 1,
giblorb_method_FilePos = 2
};
enum {
giblorb_ID_Snd = MKTAG('S', 'n', 'd', ' '),
giblorb_ID_Exec = MKTAG('E', 'x', 'e', 'c'),
giblorb_ID_Pict = MKTAG('P', 'i', 'c', 't'),
giblorb_ID_Copyright = MKTAG('(', 'c', ')', ' '),
giblorb_ID_AUTH = MKTAG('A', 'U', 'T', 'H'),
giblorb_ID_ANNO = MKTAG('A', 'N', 'N', 'O')
};
enum {
giblorb_ID_MOD = MKTAG('M', 'O', 'D', ' '),
giblorb_ID_FORM = MKTAG('F', 'O', 'R', 'M'),
giblorb_ID_IFRS = MKTAG('I', 'F', 'R', 'S'),
giblorb_ID_RIdx = MKTAG('R', 'I', 'd', 'x'),
giblorb_ID_OGG = MKTAG('O', 'G', 'G', 'V'),
// non-standard types
giblorb_ID_MIDI = MKTAG('M', 'I', 'D', 'I'),
giblorb_ID_MP3 = MKTAG('M', 'P', '3', ' '),
giblorb_ID_WAVE = MKTAG('W', 'A', 'V', 'E')
};
/**
* Holds the complete description of an open Blorb file.
* This type is opaque for normal interpreter use.
*/
typedef struct giblorb_map_struct giblorb_map_t;
/* giblorb_result_t: Result when you try to load a chunk. */
typedef struct giblorb_result_struct {
glui32 chunknum; // The chunk number (for use in giblorb_unload_chunk(), etc.)
union {
void *ptr; ///< A pointer to the data (if you used giblorb_method_Memory)
glui32 startpos; ///< The position in the file (if you used giblorb_method_FilePos)
} data;
glui32 length; ///< The length of the data
glui32 chunktype; ///< The type of the chunk.
} giblorb_result_t;
typedef struct giblorb_resdesc_struct giblorb_resdesc_t;
class Blorb {
private:
bool _libInited;
Common::SeekableReadStream *_file;
giblorb_map_t *_map;
private:
/**
* Initializes Blorb
*/
giblorb_err_t giblorb_initialize();
giblorb_err_t giblorb_initialize_map(giblorb_map_t *map);
void giblorb_qsort(giblorb_resdesc_t **list, size_t len);
giblorb_resdesc_t *giblorb_bsearch(giblorb_resdesc_t *sample,
giblorb_resdesc_t **list, int len);
int sortsplot(giblorb_resdesc_t *v1, giblorb_resdesc_t *v2);
public:
/**
* Constructor
*/
Blorb() : _libInited(false), _file(nullptr), _map(nullptr) {}
giblorb_err_t giblorb_set_resource_map(Common::SeekableReadStream *file);
giblorb_map_t *giblorb_get_resource_map(void);
bool giblorb_is_resource_map(void) const;
giblorb_err_t giblorb_create_map(Common::SeekableReadStream *file, giblorb_map_t **newmap);
giblorb_err_t giblorb_destroy_map(giblorb_map_t *map);
giblorb_err_t giblorb_load_chunk_by_type(giblorb_map_t *map,
glui32 method, giblorb_result_t *res, glui32 chunktype, glui32 count);
giblorb_err_t giblorb_load_chunk_by_number(giblorb_map_t *map,
glui32 method, giblorb_result_t *res, glui32 chunknum);
giblorb_err_t giblorb_unload_chunk(giblorb_map_t *map, glui32 chunknum);
giblorb_err_t giblorb_load_resource(giblorb_map_t *map, glui32 method,
giblorb_result_t *res, glui32 usage, glui32 resnum);
giblorb_err_t giblorb_count_resources(giblorb_map_t *map,
glui32 usage, glui32 *num, glui32 *min, glui32 *max);
};
} // End of namespace Gargoyle
#endif

@ -25,6 +25,7 @@
#include "gargoyle/gargoyle.h"
#include "gargoyle/glk_types.h"
#include "gargoyle/blorb.h"
#include "gargoyle/time.h"
#include "gargoyle/windows.h"
@ -33,7 +34,7 @@ namespace Gargoyle {
/**
* Implements the GLK interface
*/
class Glk : public GargoyleEngine {
class Glk : public GargoyleEngine, public Blorb {
private:
bool _gliFirstEvent;
unsigned char _charTolowerTable[256];

@ -1,6 +1,7 @@
MODULE := engines/gargoyle
MODULE_OBJS := \
blorb.o \
conf.o \
detection.o \
events.o \