aboutsummaryrefslogtreecommitdiff
diff options
context:
space:
mode:
-rw-r--r--engines/sci/scicore/resource.cpp191
-rw-r--r--engines/sci/scicore/resource.h3
-rw-r--r--engines/sci/scicore/resource_map.cpp135
-rw-r--r--engines/sci/scicore/resource_patch.cpp129
4 files changed, 194 insertions, 264 deletions
diff --git a/engines/sci/scicore/resource.cpp b/engines/sci/scicore/resource.cpp
index aca8e23592..389216e9d2 100644
--- a/engines/sci/scicore/resource.cpp
+++ b/engines/sci/scicore/resource.cpp
@@ -36,6 +36,7 @@
namespace Sci {
#undef SCI_REQUIRE_RESOURCE_FILES
+
//#define SCI_VERBOSE_RESMGR 1
const char *sci_version_types[] = {
@@ -723,4 +724,194 @@ int ResourceManager::detectVolVersion() {
return SCI_VERSION_1_1;
}
+// version-agnostic patch application
+void ResourceManager::processPatch(ResourceSource *source,
+ const char *filename, ResourceType restype, int resnumber) {
+ Common::File file;
+ Resource *newrsc;
+ uint32 resId = RESOURCE_HASH(restype, resnumber);
+ byte patchtype, patch_data_offset;
+ int fsize;
+
+ if (resnumber == -1)
+ return;
+ if (!file.open(filename)) {
+ perror("""__FILE__"": (""__LINE__""): failed to open");
+ return;
+ }
+ fsize = file.size();
+ if (fsize < 3) {
+ debug("Patching %s failed - file too small", filename);
+ return;
+ }
+
+ patchtype = file.readByte() & 0x7F;
+ patch_data_offset = file.readByte();
+
+ if (patchtype != restype) {
+ debug("Patching %s failed - resource type mismatch", filename);
+ return;
+ }
+ if (patch_data_offset + 2 >= fsize) {
+ debug("Patching %s failed - patch starting at offset %d can't be in file of size %d",
+ filename, patch_data_offset + 2, fsize);
+ return;
+ }
+ // Prepare destination, if neccessary
+ if (_resMap.contains(resId) == false) {
+ newrsc = new Resource;
+ _resMap.setVal(resId, newrsc);
+ } else
+ newrsc = _resMap.getVal(resId);
+ // Overwrite everything, because we're patching
+ newrsc->id = resId;
+ newrsc->number = resnumber;
+ newrsc->status = SCI_STATUS_NOMALLOC;
+ newrsc->type = restype;
+ newrsc->source = source;
+ newrsc->size = fsize - patch_data_offset - 2;
+ newrsc->file_offset = 2 + patch_data_offset;
+ debug("Patching %s - OK", filename);
+}
+
+
+void ResourceManager::readResourcePatches(ResourceSource *source) {
+// Note: since some SCI1 games(KQ5 floppy, SQ4) might use SCI0 naming scheme for patch files
+// this function tries to read patch file with any supported naming scheme,
+// regardless of _sciVersion value
+
+ Common::String mask, name;
+ Common::ArchiveMemberList files;
+ int number;
+ const char *szResType;
+ ResourceSource *psrcPatch;
+
+ for (int i = kResourceTypeView; i < kResourceTypeInvalid; i ++) {
+ files.clear();
+ szResType = getResourceTypeName((ResourceType)i);
+ // SCI0 naming - type.nnn
+ mask = szResType;
+ mask += ".???";
+ SearchMan.listMatchingMembers(files, mask);
+ // SCI1 and later naming - nnn.typ
+ mask = "*.";
+ mask += getResourceTypeSuffix((ResourceType)i);
+ SearchMan.listMatchingMembers(files, mask);
+ for (Common::ArchiveMemberList::const_iterator x = files.begin(); x != files.end(); x++) {
+ number = -1;
+ name = (*x)->getName();
+ if (isdigit(name[0])) {
+ // SCI1 scheme
+ number = atoi(name.c_str());
+ } else {
+ // SCI0 scheme
+ int resname_len = strlen(szResType);
+ if (scumm_strnicmp(name.c_str(), szResType, resname_len) == 0) {
+ number = atoi(name.c_str() + resname_len + 1);
+ }
+ }
+ psrcPatch = new ResourceSource;
+ psrcPatch->source_type = kSourcePatch;
+ psrcPatch->location_name = name;
+ processPatch(psrcPatch, name.c_str(), (ResourceType)i, number);
+ }
+ }
+}
+
+int ResourceManager::readResourceMapSCI0(ResourceSource *map) {
+ Common::File file;
+ Resource *res;
+ ResourceType type;
+ uint16 number, id;
+ uint32 offset;
+
+ if (!file.open(map->location_name))
+ return SCI_ERROR_RESMAP_NOT_FOUND;
+
+ file.seek(0, SEEK_SET);
+
+ byte bMask = _mapVersion == SCI_VERSION_01_VGA_ODD ? 0xF0: 0xFC;
+ byte bShift = _mapVersion == SCI_VERSION_01_VGA_ODD ? 28 : 26;
+
+ do {
+ id = file.readUint16LE();
+ offset = file.readUint32LE();
+
+ if (file.ioFailed()) {
+ warning("Error while reading %s: ", map->location_name.c_str());
+ perror("");
+ return SCI_ERROR_RESMAP_NOT_FOUND;
+ }
+ if(offset == 0xFFFFFFFF)
+ break;
+
+ type = (ResourceType)(id >> 11);
+ number = id & 0x7FF;
+ uint32 resId = RESOURCE_HASH(type, number);
+ // adding a new resource
+ if (_resMap.contains(resId) == false) {
+ res = new Resource;
+ res->id = id;
+ res->file_offset = offset & (((~bMask) << 24) | 0xFFFFFF);
+ res->number = number;
+ res->type = type;
+ res->source = getVolume(map, offset >> bShift);
+ _resMap.setVal(resId, res);
+ }
+ } while (!file.eos());
+ return 0;
+}
+
+int ResourceManager::readResourceMapSCI1(ResourceSource *map, ResourceSource *vol) {
+ Common::File file;
+ Resource *res;
+ if (!file.open(map->location_name))
+ return SCI_ERROR_RESMAP_NOT_FOUND;
+
+ resource_index_t resMap[kResourceTypeInvalid];
+ memset(resMap, 0, sizeof(resource_index_t) * kResourceTypeInvalid);
+ byte type = 0, prevtype = 0;
+ byte nEntrySize = _mapVersion == SCI_VERSION_1_1 ? SCI11_RESMAP_ENTRIES_SIZE : SCI1_RESMAP_ENTRIES_SIZE;
+ uint32 resId;
+
+ // Read resource type and offsets to resource offsets block from .MAP file
+ // The last entry has type=0xFF (0x1F) and offset equals to map file length
+ do {
+ type = file.readByte() & 0x1F;
+ resMap[type].wOffset = file.readUint16LE();
+ resMap[prevtype].wSize = (resMap[type].wOffset
+ - resMap[prevtype].wOffset) / nEntrySize;
+ prevtype = type;
+ } while (type != 0x1F); // the last entry is FF
+
+ // reading each type's offsets
+ uint32 off = 0;
+ for (type = 0; type < kResourceTypeInvalid; type++) {
+ if (resMap[type].wOffset == 0) // this resource does not exist in map
+ continue;
+ file.seek(resMap[type].wOffset);
+ for (int i = 0; i < resMap[type].wSize; i++) {
+ uint16 number = file.readUint16LE();
+ file.read(&off, nEntrySize - 2);
+ if (file.ioFailed()) {
+ warning("Error while reading %s: ", map->location_name.c_str());
+ perror("");
+ return SCI_ERROR_RESMAP_NOT_FOUND;
+ }
+ resId = RESOURCE_HASH(type, number);
+ // adding new resource only if it does not exist
+ if (_resMap.contains(resId) == false) {
+ res = new Resource;
+ _resMap.setVal(resId, res);
+ res->type = (ResourceType)type;
+ res->number = number;
+ res->id = res->number | (res->type << 16);
+ res->source = _mapVersion < SCI_VERSION_1_1 ? getVolume(map, off >> 28) : vol;
+ res->file_offset = _mapVersion < SCI_VERSION_1_1 ? off & 0x0FFFFFFF : off << 1;
+ }
+ }
+ }
+ return 0;
+}
+
} // End of namespace Sci
diff --git a/engines/sci/scicore/resource.h b/engines/sci/scicore/resource.h
index 6550514e25..a44ff0f0fa 100644
--- a/engines/sci/scicore/resource.h
+++ b/engines/sci/scicore/resource.h
@@ -94,6 +94,9 @@ enum ResSourceType {
#define RESSOURCE_ADDRESSING_MASK 128
#define RESOURCE_HASH(type, number) (uint32)((type<<16) | number)
+#define SCI0_RESMAP_ENTRIES_SIZE 6
+#define SCI1_RESMAP_ENTRIES_SIZE 6
+#define SCI11_RESMAP_ENTRIES_SIZE 5
extern const char *sci_error_types[];
extern const char *sci_version_types[];
diff --git a/engines/sci/scicore/resource_map.cpp b/engines/sci/scicore/resource_map.cpp
deleted file mode 100644
index 9279ca3b11..0000000000
--- a/engines/sci/scicore/resource_map.cpp
+++ /dev/null
@@ -1,135 +0,0 @@
-/* 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.
- *
- * $URL$
- * $Id$
- *
- */
-
-#include "sci/sci_memory.h"
-#include "sci/scicore/resource.h"
-#include "sci/tools.h"
-
-#include "common/util.h"
-#include "common/file.h"
-
-namespace Sci {
-
-#define SCI0_RESMAP_ENTRIES_SIZE 6
-#define SCI1_RESMAP_ENTRIES_SIZE 6
-#define SCI11_RESMAP_ENTRIES_SIZE 5
-
-int ResourceManager::readResourceMapSCI0(ResourceSource *map) {
- Common::File file;
- Resource *res;
- ResourceType type;
- uint16 number, id;
- uint32 offset;
-
- if (!file.open(map->location_name))
- return SCI_ERROR_RESMAP_NOT_FOUND;
-
- file.seek(0, SEEK_SET);
-
- byte bMask = _mapVersion == SCI_VERSION_01_VGA_ODD ? 0xF0: 0xFC;
- byte bShift = _mapVersion == SCI_VERSION_01_VGA_ODD ? 28 : 26;
-
- do {
- id = file.readUint16LE();
- offset = file.readUint32LE();
-
- if (file.ioFailed()) {
- warning("Error while reading %s: ", map->location_name.c_str());
- perror("");
- return SCI_ERROR_RESMAP_NOT_FOUND;
- }
- if(offset == 0xFFFFFFFF)
- break;
-
- type = (ResourceType)(id >> 11);
- number = id & 0x7FF;
- uint32 resId = RESOURCE_HASH(type, number);
- // adding a new resource
- if (_resMap.contains(resId) == false) {
- res = new Resource;
- res->id = id;
- res->file_offset = offset & (((~bMask) << 24) | 0xFFFFFF);
- res->number = number;
- res->type = type;
- res->source = getVolume(map, offset >> bShift);
- _resMap.setVal(resId, res);
- }
- } while (!file.eos());
- return 0;
-}
-
-int ResourceManager::readResourceMapSCI1(ResourceSource *map, ResourceSource *vol) {
- Common::File file;
- Resource *res;
- if (!file.open(map->location_name))
- return SCI_ERROR_RESMAP_NOT_FOUND;
-
- resource_index_t resMap[kResourceTypeInvalid];
- memset(resMap, 0, sizeof(resource_index_t) * kResourceTypeInvalid);
- byte type = 0, prevtype = 0;
- byte nEntrySize = _mapVersion == SCI_VERSION_1_1 ? SCI11_RESMAP_ENTRIES_SIZE : SCI1_RESMAP_ENTRIES_SIZE;
- uint32 resId;
-
- // Read resource type and offsets to resource offsets block from .MAP file
- // The last entry has type=0xFF (0x1F) and offset equals to map file length
- do {
- type = file.readByte() & 0x1F;
- resMap[type].wOffset = file.readUint16LE();
- resMap[prevtype].wSize = (resMap[type].wOffset
- - resMap[prevtype].wOffset) / nEntrySize;
- prevtype = type;
- } while (type != 0x1F); // the last entry is FF
-
- // reading each type's offsets
- uint32 off = 0;
- for (type = 0; type < kResourceTypeInvalid; type++) {
- if (resMap[type].wOffset == 0) // this resource does not exist in map
- continue;
- file.seek(resMap[type].wOffset);
- for (int i = 0; i < resMap[type].wSize; i++) {
- uint16 number = file.readUint16LE();
- file.read(&off, nEntrySize - 2);
- if (file.ioFailed()) {
- warning("Error while reading %s: ", map->location_name.c_str());
- perror("");
- return SCI_ERROR_RESMAP_NOT_FOUND;
- }
- resId = RESOURCE_HASH(type, number);
- // adding new resource only if it does not exist
- if (_resMap.contains(resId) == false) {
- res = new Resource;
- _resMap.setVal(resId, res);
- res->type = (ResourceType)type;
- res->number = number;
- res->id = res->number | (res->type << 16);
- res->source = _mapVersion < SCI_VERSION_1_1 ? getVolume(map, off >> 28) : vol;
- res->file_offset = _mapVersion < SCI_VERSION_1_1 ? off & 0x0FFFFFFF : off << 1;
- }
- }
- }
- return 0;
-}
-
-} // End of namespace Sci
diff --git a/engines/sci/scicore/resource_patch.cpp b/engines/sci/scicore/resource_patch.cpp
deleted file mode 100644
index 80762d9c66..0000000000
--- a/engines/sci/scicore/resource_patch.cpp
+++ /dev/null
@@ -1,129 +0,0 @@
-/* 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.
- *
- * $URL$
- * $Id$
- *
- */
-
-#include "common/archive.h"
-#include "common/file.h"
-#include "common/debug.h"
-
-#include "sci/scicore/resource.h"
-#include "sci/sci_memory.h"
-
-namespace Sci {
-
-// version-agnostic patch application
-void ResourceManager::processPatch(ResourceSource *source,
- const char *filename, ResourceType restype, int resnumber) {
- Common::File file;
- Resource *newrsc;
- uint32 resId = RESOURCE_HASH(restype, resnumber);
- byte patchtype, patch_data_offset;
- int fsize;
-
- if (resnumber == -1)
- return;
- if (!file.open(filename)) {
- perror("""__FILE__"": (""__LINE__""): failed to open");
- return;
- }
- fsize = file.size();
- if (fsize < 3) {
- debug("Patching %s failed - file too small", filename);
- return;
- }
-
- patchtype = file.readByte() & 0x7F;
- patch_data_offset = file.readByte();
-
- if (patchtype != restype) {
- debug("Patching %s failed - resource type mismatch", filename);
- return;
- }
- if (patch_data_offset + 2 >= fsize) {
- debug("Patching %s failed - patch starting at offset %d can't be in file of size %d",
- filename, patch_data_offset + 2, fsize);
- return;
- }
- // Prepare destination, if neccessary
- if (_resMap.contains(resId) == false) {
- newrsc = new Resource;
- _resMap.setVal(resId, newrsc);
- } else
- newrsc = _resMap.getVal(resId);
- // Overwrite everything, because we're patching
- newrsc->id = resId;
- newrsc->number = resnumber;
- newrsc->status = SCI_STATUS_NOMALLOC;
- newrsc->type = restype;
- newrsc->source = source;
- newrsc->size = fsize - patch_data_offset - 2;
- newrsc->file_offset = 2 + patch_data_offset;
- debug("Patching %s - OK", filename);
-}
-
-
-void ResourceManager::readResourcePatches(ResourceSource *source) {
-// Note: since some SCI1 games(KQ5 floppy, SQ4) might use SCI0 naming scheme for patch files
-// this function tries to read patch file with any supported naming scheme,
-// regardless of _sciVersion value
-
- Common::String mask, name;
- Common::ArchiveMemberList files;
- int number;
- const char *szResType;
- ResourceSource *psrcPatch;
-
- for (int i = kResourceTypeView; i < kResourceTypeInvalid; i ++) {
- files.clear();
- szResType = getResourceTypeName((ResourceType)i);
- // SCI0 naming - type.nnn
- mask = szResType;
- mask += ".???";
- SearchMan.listMatchingMembers(files, mask);
- // SCI1 and later naming - nnn.typ
- mask = "*.";
- mask += getResourceTypeSuffix((ResourceType)i);
- SearchMan.listMatchingMembers(files, mask);
- for (Common::ArchiveMemberList::const_iterator x = files.begin(); x != files.end(); x++) {
- number = -1;
- name = (*x)->getName();
- if (isdigit(name[0])) {
- // SCI1 scheme
- number = atoi(name.c_str());
- } else {
- // SCI0 scheme
- int resname_len = strlen(szResType);
- if (scumm_strnicmp(name.c_str(), szResType, resname_len) == 0) {
- number = atoi(name.c_str() + resname_len + 1);
- }
- }
- psrcPatch = new ResourceSource;
- psrcPatch->source_type = kSourcePatch;
- psrcPatch->location_name = name;
- processPatch(psrcPatch, name.c_str(), (ResourceType)i, number);
- }
- }
-}
-
-} // End of namespace Sci