247 lines
6.4 KiB
C++
247 lines
6.4 KiB
C++
/* 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 3 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, see <http://www.gnu.org/licenses/>.
|
|
*
|
|
*/
|
|
|
|
#include "common/file.h"
|
|
#include "common/textconsole.h"
|
|
#include "m4/mem/res.h"
|
|
#include "m4/mem/memman.h"
|
|
#include "m4/mem/reloc.h"
|
|
#include "m4/core/errors.h"
|
|
#include "m4/vars.h"
|
|
|
|
namespace M4 {
|
|
|
|
#define PRIME 8179
|
|
#define HASHSIZE MAX_RESOURCES
|
|
#define FULLY_BUFFERED 1
|
|
#define MARKED_PURGE 0x80
|
|
|
|
Resources::~Resources() {
|
|
delete _fp;
|
|
}
|
|
|
|
|
|
Resources::Entry *Resources::findAndSetResEntry(const Common::String &resourceName) {
|
|
int orig_hash_val;
|
|
int hash_val;
|
|
Entry *res = nullptr;
|
|
|
|
Common::String resName = resourceName;
|
|
if (_useLowercase)
|
|
resName.toLowercase();
|
|
orig_hash_val = hash_val = hash(resName);
|
|
|
|
// If empty slot at this hash, then we're done
|
|
if (!_resources[hash_val].Flags)
|
|
goto got_one;
|
|
|
|
// Flags is set, so scan until Flags is clear, or the resource name strings match
|
|
while ((_resources[hash_val].Flags & FULLY_BUFFERED)
|
|
&& !resName.equals(_resources[hash_val].name)) {
|
|
// if we searched every entry to no avail:
|
|
if ((hash_val = (hash_val + 1) & (HASHSIZE - 1)) == orig_hash_val)
|
|
goto test4;
|
|
}
|
|
|
|
goto got_one;
|
|
|
|
test4:
|
|
hash_val = orig_hash_val;
|
|
while (!(_resources[hash_val].Flags & MARKED_PURGE))
|
|
// if we searched every entry to no avail:
|
|
if ((hash_val = (hash_val + 1) & (HASHSIZE - 1)) == orig_hash_val) {
|
|
error("Out of resource space");
|
|
}
|
|
|
|
res = &_resources[hash_val];
|
|
delete[] res->RHandle;
|
|
res->RHandle = nullptr;
|
|
|
|
got_one:
|
|
res = &_resources[hash_val];
|
|
res->name = resName;
|
|
res->Flags = FULLY_BUFFERED;
|
|
|
|
return res;
|
|
}
|
|
|
|
int Resources::hash(const Common::String &sym) const {
|
|
uint ret_val = 0;
|
|
|
|
const char *s = sym.c_str();
|
|
while (*s) {
|
|
ret_val += *s++;
|
|
ret_val *= PRIME;
|
|
}
|
|
|
|
return ret_val & (HASHSIZE - 1);
|
|
}
|
|
|
|
MemHandle Resources::rget(const Common::String &resourceName, int32 *resourceSize) {
|
|
Entry *resEntry;
|
|
|
|
if (resourceSize)
|
|
*resourceSize = 0;
|
|
|
|
if (!(resEntry = findAndSetResEntry(resourceName))) {
|
|
term_message("rget:%s -> failed!", resourceName.c_str());
|
|
return nullptr;
|
|
}
|
|
|
|
// Check if resource is fully buffered
|
|
// All resources are currently fully buffered!
|
|
if (!(resEntry->Flags & FULLY_BUFFERED)) {
|
|
term_message("rget:%s -> failed!", resourceName.c_str());
|
|
return nullptr;
|
|
}
|
|
|
|
// If there's a handle with some memory in it
|
|
// Resource is cached; unmark purge flag and return handle
|
|
if (resEntry->RHandle && *resEntry->RHandle) {
|
|
if (resourceSize)
|
|
*resourceSize = resEntry->Size;
|
|
|
|
HNoPurge(resEntry->RHandle);
|
|
resEntry->Flags &= ~MARKED_PURGE;
|
|
term_message("rget:%s -> from memory", resourceName.c_str());
|
|
return resEntry->RHandle;
|
|
}
|
|
|
|
// Check if size known
|
|
if ((resEntry->Size = get_file(resEntry->name)) == -1)
|
|
error("Error getting entry %s", resEntry->name.c_str());
|
|
|
|
// Check if buffer size is set
|
|
if (resEntry->BufferSize != resEntry->Size)
|
|
resEntry->BufferSize = resEntry->Size;
|
|
|
|
// Check if resource handle allocated
|
|
if (!resEntry->RHandle)
|
|
resEntry->RHandle = MakeNewHandle(resEntry->BufferSize, resEntry->name.c_str());
|
|
|
|
// Check if resource handle has valid memory block allocated to it
|
|
if (!*resEntry->RHandle)
|
|
mem_ReallocateHandle(resEntry->RHandle, resEntry->BufferSize, resEntry->name.c_str());
|
|
|
|
if (!do_file(resEntry->RHandle))
|
|
error("rget: do_file -> %s", resourceName.c_str());
|
|
|
|
if (resourceSize) // xi change
|
|
*resourceSize = resEntry->Size;
|
|
|
|
term_message("rget:%s -> from disk", resourceName.c_str());
|
|
return resEntry->RHandle;
|
|
}
|
|
|
|
void Resources::rtoss(const Common::String &resourceName) {
|
|
int hash_val;
|
|
Entry *resEntry = nullptr;
|
|
Common::String resName = resourceName;
|
|
|
|
if (_useLowercase)
|
|
resName.toLowercase();
|
|
hash_val = hash(resName);
|
|
|
|
// Check if resource is in resource table
|
|
if (_resources[hash_val].Flags) {
|
|
for (int ctr = 0; ctr <= HASHSIZE && _resources[hash_val].Flags;
|
|
++ctr, hash_val = (hash_val + 1) % HASHSIZE) {
|
|
if (resName.equals(_resources[hash_val].name)) {
|
|
resEntry = &_resources[hash_val];
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
|
|
if (!resEntry)
|
|
error_show(FL, 'RIOU', "rtoss: %s", resourceName.c_str());
|
|
else if (!(resEntry->Flags & FULLY_BUFFERED))
|
|
return;
|
|
|
|
if (!resEntry || !*resEntry->RHandle) {
|
|
term_message("bad rtoss no memory %s", resourceName.c_str());
|
|
return;
|
|
}
|
|
|
|
if (resEntry->Flags & MARKED_PURGE)
|
|
// This is actually okay. In Riddle, for example, "show script"
|
|
// is assigned to multiple MACH slots, and in ClearWSAssets will
|
|
// pass it to rtoss for each entry
|
|
term_message("multiple rtoss: %s", resourceName.c_str());
|
|
else
|
|
term_message("rtossing: %s", resourceName.c_str());
|
|
|
|
HUnLock(resEntry->RHandle);
|
|
HPurge(resEntry->RHandle);
|
|
resEntry->Flags |= MARKED_PURGE;
|
|
}
|
|
|
|
int32 Resources::get_file(const Common::String &name) {
|
|
assert(!_fp);
|
|
_fp = new SysFile(name);
|
|
assert(_fp);
|
|
|
|
if (!_fp->exists())
|
|
error("get_file - getting %s", name.c_str());
|
|
|
|
return _fp->size();
|
|
}
|
|
|
|
bool Resources::do_file(MemHandle buffer) {
|
|
_fp->seek(0);
|
|
|
|
bool result = true;
|
|
|
|
// read in resource from file
|
|
if (!_fp->read(buffer)) {
|
|
term_message("unable to read: %s", _fp->filename.c_str());
|
|
result = false;
|
|
}
|
|
|
|
delete _fp;
|
|
_fp = nullptr;
|
|
|
|
return result;
|
|
}
|
|
|
|
void Resources::dumpResources() {
|
|
if (gDebugLevel >= 2) {
|
|
debug(2, "List of active resources:");
|
|
for (int i = 0; i < MAX_RESOURCES; ++i) {
|
|
Entry &entry = _resources[i];
|
|
if (entry.Flags != 0 && (entry.Flags & MARKED_PURGE) == 0)
|
|
debug(2, "#%d - %s", i, entry.name.c_str());
|
|
}
|
|
|
|
debugN(2, "\n");
|
|
}
|
|
}
|
|
|
|
MemHandle rget(const Common::String &resourceName, int32 *resourceSize) {
|
|
return _G(resources).rget(resourceName, resourceSize);
|
|
}
|
|
|
|
void rtoss(const Common::String &resourceName) {
|
|
_G(resources).rtoss(resourceName);
|
|
}
|
|
|
|
} // namespace M4
|