2004-04-12 21:40:49 +00:00
|
|
|
/* ScummVM - Scumm Interpreter
|
2005-01-01 16:20:17 +00:00
|
|
|
* Copyright (C) 2004-2005 The ScummVM project
|
2004-04-12 21:40:49 +00:00
|
|
|
*
|
|
|
|
* The ReInherit Engine is (C)2000-2003 by Daniel Balsom.
|
|
|
|
*
|
|
|
|
* 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., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA.
|
|
|
|
*
|
|
|
|
* $Header$
|
|
|
|
*
|
|
|
|
*/
|
|
|
|
|
2004-05-01 13:19:15 +00:00
|
|
|
// RSC Resource file management module
|
2004-08-02 16:20:35 +00:00
|
|
|
#include "saga/saga.h"
|
2004-04-12 21:40:49 +00:00
|
|
|
|
2004-08-02 16:20:35 +00:00
|
|
|
#include "saga/rscfile.h"
|
2004-12-15 00:24:12 +00:00
|
|
|
#include "saga/stream.h"
|
2004-04-12 21:40:49 +00:00
|
|
|
|
|
|
|
namespace Saga {
|
|
|
|
|
2005-07-19 19:05:52 +00:00
|
|
|
Resource::Resource(SagaEngine *vm): _vm(vm) {
|
|
|
|
_contexts = NULL;
|
|
|
|
_contextsCount = 0;
|
2004-04-12 21:40:49 +00:00
|
|
|
}
|
|
|
|
|
2005-07-19 19:05:52 +00:00
|
|
|
Resource::~Resource() {
|
|
|
|
clearContexts();
|
2004-04-12 21:40:49 +00:00
|
|
|
}
|
|
|
|
|
2005-07-19 19:05:52 +00:00
|
|
|
bool Resource::loadContext(ResourceContext *context) {
|
|
|
|
size_t i;
|
|
|
|
int j;
|
|
|
|
bool result;
|
|
|
|
byte tableInfo[RSC_TABLEINFO_SIZE];
|
|
|
|
uint32 resourceTableOffset;
|
|
|
|
GamePatchDescription *patchDescription;
|
|
|
|
ResourceData *resourceData;
|
|
|
|
byte *tableBuffer;
|
|
|
|
size_t tableSize;
|
2004-04-12 21:40:49 +00:00
|
|
|
|
2005-07-19 19:05:52 +00:00
|
|
|
if (!context->file->open(context->fileName)) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
context->isBigEndian = _vm->isBigEndian();
|
|
|
|
|
|
|
|
if (!context->isBigEndian) {
|
|
|
|
context->isBigEndian = ((_vm->getFeatures() & GF_BIG_ENDIAN_VOICES) != 0) && ((context->fileType & GAME_VOICEFILE) != 0);
|
2004-04-12 21:40:49 +00:00
|
|
|
}
|
|
|
|
|
2005-07-19 19:05:52 +00:00
|
|
|
if (context->file->size() < RSC_MIN_FILESIZE) {
|
|
|
|
return false;
|
2004-04-12 21:40:49 +00:00
|
|
|
}
|
|
|
|
|
2004-05-01 13:19:15 +00:00
|
|
|
// Read resource table info from the rear end of file
|
2005-07-19 19:05:52 +00:00
|
|
|
context->file->seek((long)(-RSC_TABLEINFO_SIZE), SEEK_END);
|
2004-04-12 21:40:49 +00:00
|
|
|
|
2005-07-19 19:05:52 +00:00
|
|
|
if (context->file->read(tableInfo, RSC_TABLEINFO_SIZE) != RSC_TABLEINFO_SIZE) {
|
|
|
|
return false;
|
2004-04-12 21:40:49 +00:00
|
|
|
}
|
|
|
|
|
2005-07-19 19:05:52 +00:00
|
|
|
MemoryReadStreamEndian readS(tableInfo, RSC_TABLEINFO_SIZE, context->isBigEndian);
|
2004-04-12 21:40:49 +00:00
|
|
|
|
2005-07-19 19:05:52 +00:00
|
|
|
resourceTableOffset = readS.readUint32();
|
|
|
|
context->count = readS.readUint32();
|
2004-04-12 21:40:49 +00:00
|
|
|
|
2005-07-19 19:05:52 +00:00
|
|
|
// Check for sane table offset
|
|
|
|
if (resourceTableOffset != context->file->size() - RSC_TABLEINFO_SIZE - RSC_TABLEENTRY_SIZE * context->count) {
|
|
|
|
return false;
|
2004-04-12 21:40:49 +00:00
|
|
|
}
|
|
|
|
|
2004-05-01 13:19:15 +00:00
|
|
|
// Load resource table
|
2005-07-19 19:05:52 +00:00
|
|
|
tableSize = RSC_TABLEENTRY_SIZE * context->count;
|
2004-04-12 21:40:49 +00:00
|
|
|
|
2005-07-19 19:05:52 +00:00
|
|
|
tableBuffer = (byte *)malloc(tableSize);
|
2004-04-12 21:40:49 +00:00
|
|
|
|
2005-07-19 19:05:52 +00:00
|
|
|
context->file->seek((long)resourceTableOffset, SEEK_SET);
|
2004-04-12 21:40:49 +00:00
|
|
|
|
2005-07-19 19:05:52 +00:00
|
|
|
result = (context->file->read(tableBuffer, tableSize) == tableSize);
|
|
|
|
if (result) {
|
|
|
|
context->table = (ResourceData *)calloc(context->count, sizeof(*context->table));
|
|
|
|
|
|
|
|
MemoryReadStreamEndian readS1(tableBuffer, tableSize, context->isBigEndian);
|
|
|
|
|
|
|
|
for (i = 0; i < context->count; i++) {
|
|
|
|
resourceData = &context->table[i];
|
|
|
|
resourceData->offset = readS1.readUint32();
|
|
|
|
resourceData->size = readS1.readUint32();
|
|
|
|
//sanity check
|
|
|
|
if ((resourceData->offset > context->file->size()) || (resourceData->size > context->file->size())) {
|
|
|
|
result = false;
|
|
|
|
break;
|
|
|
|
}
|
2004-04-12 21:40:49 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2005-07-19 19:05:52 +00:00
|
|
|
free(tableBuffer);
|
|
|
|
|
|
|
|
//process patch files
|
|
|
|
if (result) {
|
|
|
|
for (j = 0; j < _vm->getGameDescription()->patchsCount; j++) {
|
|
|
|
patchDescription = &_vm->getGameDescription()->patchDescriptions[j];
|
|
|
|
if ((patchDescription->fileType & context->fileType) != 0) {
|
|
|
|
if (patchDescription->resourceId < context->count) {
|
|
|
|
//TODO|fix: should we convert this ID? or make separate patch list for MAC version?
|
|
|
|
resourceData = &context->table[patchDescription->resourceId];
|
|
|
|
resourceData->patchFile = new Common::File();
|
|
|
|
if (resourceData->patchFile->open(patchDescription->fileName)) {
|
|
|
|
resourceData->offset = 0;
|
|
|
|
resourceData->size = resourceData->patchFile->size();
|
|
|
|
} else {
|
|
|
|
warning("loadContext: patch file not found %s", patchDescription->fileName);
|
|
|
|
delete resourceData->patchFile;
|
|
|
|
resourceData->patchFile = NULL;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2004-04-12 21:40:49 +00:00
|
|
|
}
|
|
|
|
|
2005-07-19 19:05:52 +00:00
|
|
|
return result;
|
2004-04-12 21:40:49 +00:00
|
|
|
}
|
|
|
|
|
2005-07-19 19:05:52 +00:00
|
|
|
bool Resource::createContexts() {
|
2005-07-23 15:23:01 +00:00
|
|
|
int i;
|
2005-07-19 19:05:52 +00:00
|
|
|
ResourceContext *context;
|
|
|
|
_contextsCount = _vm->getGameDescription()->filesCount;
|
|
|
|
_contexts = (ResourceContext*)calloc(_contextsCount, sizeof(*_contexts));
|
|
|
|
|
|
|
|
for (i = 0; i < _contextsCount; i++) {
|
|
|
|
context = &_contexts[i];
|
|
|
|
context->file = new Common::File();
|
|
|
|
context->fileName = _vm->getGameDescription()->filesDescriptions[i].fileName;
|
|
|
|
context->fileType = _vm->getGameDescription()->filesDescriptions[i].fileType;
|
2005-07-23 15:23:01 +00:00
|
|
|
context->serial = 0;
|
|
|
|
|
|
|
|
// IHNM has serveral different voice files, so we need to allow
|
|
|
|
// multiple resource contexts of the same type. We tell them
|
|
|
|
// apart by assigning each of the duplicates an unique serial
|
|
|
|
// number. The default behaviour when requesting a context will
|
|
|
|
// be to look for serial number 0.
|
2005-07-19 19:05:52 +00:00
|
|
|
|
2005-07-23 15:23:01 +00:00
|
|
|
for (int j = i - 1; j >= 0; j--) {
|
|
|
|
if (_contexts[j].fileType & context->fileType) {
|
|
|
|
context->serial = _contexts[j].serial + 1;
|
|
|
|
break;
|
2005-07-19 19:05:52 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!loadContext(context)) {
|
|
|
|
return false;
|
|
|
|
}
|
2004-04-12 21:40:49 +00:00
|
|
|
}
|
2005-07-19 19:05:52 +00:00
|
|
|
return true;
|
2004-04-12 21:40:49 +00:00
|
|
|
}
|
|
|
|
|
2005-07-19 19:05:52 +00:00
|
|
|
void Resource::clearContexts() {
|
|
|
|
int i;
|
|
|
|
size_t j;
|
|
|
|
ResourceContext *context;
|
|
|
|
if (_contexts == NULL) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
for(i = 0; i < _contextsCount; i++) {
|
|
|
|
context = &_contexts[i];
|
|
|
|
delete context->file;
|
|
|
|
if (context->table != NULL) {
|
|
|
|
for(j = 0; j < context->count; j++) {
|
|
|
|
delete context->table[j].patchFile;
|
|
|
|
}
|
2005-01-02 20:29:27 +00:00
|
|
|
}
|
2005-07-19 19:05:52 +00:00
|
|
|
free(context->table);
|
2005-06-21 13:55:18 +00:00
|
|
|
}
|
2005-07-19 19:05:52 +00:00
|
|
|
free(_contexts);
|
|
|
|
_contexts = NULL;
|
|
|
|
}
|
2005-01-02 20:29:27 +00:00
|
|
|
|
2005-07-19 19:05:52 +00:00
|
|
|
uint32 Resource::convertResourceId(uint32 resourceId) {
|
|
|
|
|
|
|
|
if ((_vm->getGameType() == GType_ITE) && (_vm->getFeatures() & GF_MAC_RESOURCES)) {
|
|
|
|
if (resourceId > 1537) {
|
|
|
|
return resourceId - 2;
|
2005-01-02 20:29:27 +00:00
|
|
|
} else {
|
2005-07-19 19:05:52 +00:00
|
|
|
if (resourceId == 1535 || resourceId == 1536) {
|
|
|
|
error ("Wrong resource number %d for Mac ITE", resourceId);
|
|
|
|
}
|
2005-01-02 20:29:27 +00:00
|
|
|
}
|
2004-04-12 21:40:49 +00:00
|
|
|
}
|
|
|
|
|
2005-07-19 19:05:52 +00:00
|
|
|
return resourceId;
|
|
|
|
}
|
2004-04-12 21:40:49 +00:00
|
|
|
|
2005-07-19 19:05:52 +00:00
|
|
|
void Resource::loadResource(ResourceContext *context, uint32 resourceId, byte*&resourceBuffer, size_t &resourceSize) {
|
|
|
|
Common::File *file;
|
|
|
|
uint32 resourceOffset;
|
|
|
|
ResourceData *resourceData;
|
2004-04-12 21:40:49 +00:00
|
|
|
|
2005-07-19 19:05:52 +00:00
|
|
|
debug(8, "loadResource %d", resourceId);
|
|
|
|
|
|
|
|
resourceData = getResourceData(context, resourceId);
|
|
|
|
|
|
|
|
file = context->getFile(resourceData);
|
2004-04-12 21:40:49 +00:00
|
|
|
|
2005-07-19 19:05:52 +00:00
|
|
|
resourceOffset = resourceData->offset;
|
|
|
|
resourceSize = resourceData->size;
|
2004-04-12 21:40:49 +00:00
|
|
|
|
2005-07-19 19:05:52 +00:00
|
|
|
resourceBuffer = (byte*)malloc(resourceSize);
|
2004-04-12 21:40:49 +00:00
|
|
|
|
2005-07-19 19:05:52 +00:00
|
|
|
file->seek((long)resourceOffset, SEEK_SET);
|
2004-12-15 00:24:12 +00:00
|
|
|
|
2005-07-19 19:05:52 +00:00
|
|
|
if (file->read(resourceBuffer, resourceSize) != resourceSize) {
|
|
|
|
error("Resource::loadResource() failed to read");
|
2004-12-15 00:24:12 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2004-04-12 21:40:49 +00:00
|
|
|
} // End of namespace Saga
|