2009-02-17 15:05:16 +00:00
|
|
|
/* 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.
|
2014-02-18 02:34:24 +01:00
|
|
|
*
|
2009-02-17 15:05:16 +00:00
|
|
|
* 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.
|
2014-02-18 02:34:24 +01:00
|
|
|
*
|
2009-02-17 15:05:16 +00:00
|
|
|
* 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.
|
|
|
|
*
|
|
|
|
*/
|
2009-02-15 06:10:59 +00:00
|
|
|
|
2010-06-15 12:08:40 +00:00
|
|
|
#ifndef SCI_RESOURCE_H
|
|
|
|
#define SCI_RESOURCE_H
|
2009-02-15 06:10:59 +00:00
|
|
|
|
2010-05-27 08:09:32 +00:00
|
|
|
#include "common/str.h"
|
2010-06-15 12:20:03 +00:00
|
|
|
#include "common/list.h"
|
|
|
|
#include "common/hashmap.h"
|
2009-08-19 21:08:17 +00:00
|
|
|
|
2010-02-17 23:37:32 +00:00
|
|
|
#include "sci/graphics/helpers.h" // for ViewType
|
2009-05-15 14:07:45 +00:00
|
|
|
#include "sci/decompressor.h"
|
2009-08-30 01:37:27 +00:00
|
|
|
#include "sci/sci.h"
|
2009-03-10 21:44:03 +00:00
|
|
|
|
2009-02-21 19:54:15 +00:00
|
|
|
namespace Common {
|
2010-02-17 23:37:32 +00:00
|
|
|
class File;
|
2010-06-15 12:20:03 +00:00
|
|
|
class FSList;
|
|
|
|
class FSNode;
|
|
|
|
class WriteStream;
|
|
|
|
class SeekableReadStream;
|
2009-02-21 19:54:15 +00:00
|
|
|
}
|
|
|
|
|
2009-02-21 10:23:36 +00:00
|
|
|
namespace Sci {
|
|
|
|
|
2010-02-17 23:37:32 +00:00
|
|
|
enum {
|
|
|
|
/** The maximum allowed size for a compressed or decompressed resource */
|
|
|
|
SCI_MAX_RESOURCE_SIZE = 0x0400000
|
|
|
|
};
|
2009-02-15 06:10:59 +00:00
|
|
|
|
2009-05-28 11:15:09 +00:00
|
|
|
/** Resource status types */
|
2009-03-10 21:44:03 +00:00
|
|
|
enum ResourceStatus {
|
2009-03-11 01:40:08 +00:00
|
|
|
kResStatusNoMalloc = 0,
|
2009-03-10 21:44:03 +00:00
|
|
|
kResStatusAllocated,
|
2009-05-28 11:15:09 +00:00
|
|
|
kResStatusEnqueued, /**< In the LRU queue */
|
|
|
|
kResStatusLocked /**< Allocated and in use */
|
2009-03-10 21:44:03 +00:00
|
|
|
};
|
2009-02-15 06:10:59 +00:00
|
|
|
|
2012-05-15 03:05:08 +03:00
|
|
|
/** Resource error codes. Should be in sync with s_errorDescriptions */
|
|
|
|
enum ResourceErrorCodes {
|
|
|
|
SCI_ERROR_NONE = 0,
|
2009-05-28 11:15:09 +00:00
|
|
|
SCI_ERROR_IO_ERROR = 1,
|
2012-05-15 03:05:08 +03:00
|
|
|
SCI_ERROR_EMPTY_RESOURCE = 2,
|
|
|
|
SCI_ERROR_RESMAP_INVALID_ENTRY = 3, /**< Invalid resource.map entry */
|
|
|
|
SCI_ERROR_RESMAP_NOT_FOUND = 4,
|
|
|
|
SCI_ERROR_NO_RESOURCE_FILES_FOUND = 5, /**< No resource at all was found */
|
|
|
|
SCI_ERROR_UNKNOWN_COMPRESSION = 6,
|
|
|
|
SCI_ERROR_DECOMPRESSION_ERROR = 7, /**< sanity checks failed during decompression */
|
2009-05-28 11:15:09 +00:00
|
|
|
SCI_ERROR_RESOURCE_TOO_BIG = 8 /**< Resource size exceeds SCI_MAX_RESOURCE_SIZE */
|
|
|
|
};
|
2009-02-15 06:10:59 +00:00
|
|
|
|
2009-09-01 13:02:47 +00:00
|
|
|
enum {
|
2010-02-17 23:37:32 +00:00
|
|
|
MAX_OPENED_VOLUMES = 5 ///< Max number of simultaneously opened volumes
|
2009-09-01 13:02:47 +00:00
|
|
|
};
|
2009-03-11 20:15:42 +00:00
|
|
|
|
2009-02-28 23:46:50 +00:00
|
|
|
enum ResourceType {
|
|
|
|
kResourceTypeView = 0,
|
|
|
|
kResourceTypePic,
|
|
|
|
kResourceTypeScript,
|
|
|
|
kResourceTypeText,
|
|
|
|
kResourceTypeSound,
|
|
|
|
kResourceTypeMemory,
|
|
|
|
kResourceTypeVocab,
|
|
|
|
kResourceTypeFont,
|
|
|
|
kResourceTypeCursor,
|
|
|
|
kResourceTypePatch,
|
|
|
|
kResourceTypeBitmap,
|
|
|
|
kResourceTypePalette,
|
|
|
|
kResourceTypeCdAudio,
|
|
|
|
kResourceTypeAudio,
|
|
|
|
kResourceTypeSync,
|
|
|
|
kResourceTypeMessage,
|
|
|
|
kResourceTypeMap,
|
|
|
|
kResourceTypeHeap,
|
2009-05-19 02:10:58 +00:00
|
|
|
kResourceTypeAudio36,
|
|
|
|
kResourceTypeSync36,
|
2010-07-18 00:16:19 +00:00
|
|
|
kResourceTypeTranslation, // Currently unsupported
|
2010-11-13 00:20:23 +00:00
|
|
|
|
|
|
|
// SCI2.1+ Resources
|
2010-01-06 20:54:18 +00:00
|
|
|
kResourceTypeRobot,
|
2010-06-17 00:07:03 +00:00
|
|
|
kResourceTypeVMD,
|
2010-07-18 00:16:19 +00:00
|
|
|
kResourceTypeChunk,
|
2010-11-13 00:20:23 +00:00
|
|
|
kResourceTypeAnimation,
|
|
|
|
|
|
|
|
// SCI3 Resources
|
|
|
|
kResourceTypeEtc,
|
|
|
|
kResourceTypeDuck,
|
|
|
|
kResourceTypeClut,
|
|
|
|
kResourceTypeTGA,
|
|
|
|
kResourceTypeZZZ,
|
2010-05-24 17:21:11 +00:00
|
|
|
|
2010-07-18 00:16:19 +00:00
|
|
|
// Mac-only resources
|
|
|
|
kResourceTypeMacIconBarPictN, // IBIN resources (icon bar, not selected)
|
|
|
|
kResourceTypeMacIconBarPictS, // IBIS resources (icon bar, selected)
|
|
|
|
kResourceTypeMacPict, // PICT resources (inventory)
|
|
|
|
|
2013-12-10 01:40:46 +02:00
|
|
|
kResourceTypeRave, // KQ6 hires RAVE (special sync) resources
|
|
|
|
|
2010-07-18 00:16:19 +00:00
|
|
|
kResourceTypeInvalid
|
2009-02-15 06:10:59 +00:00
|
|
|
};
|
|
|
|
|
2009-02-28 23:46:50 +00:00
|
|
|
const char *getResourceTypeName(ResourceType restype);
|
|
|
|
|
2010-06-16 00:24:16 +00:00
|
|
|
enum ResVersion {
|
|
|
|
kResVersionUnknown,
|
|
|
|
kResVersionSci0Sci1Early,
|
|
|
|
kResVersionSci1Middle,
|
2011-02-15 21:47:29 -05:00
|
|
|
kResVersionKQ5FMT,
|
2010-06-16 00:24:16 +00:00
|
|
|
kResVersionSci1Late,
|
|
|
|
kResVersionSci11,
|
|
|
|
kResVersionSci11Mac,
|
2010-11-09 14:25:39 +00:00
|
|
|
kResVersionSci2,
|
|
|
|
kResVersionSci3
|
2010-06-16 00:24:16 +00:00
|
|
|
};
|
|
|
|
|
2009-06-04 22:16:31 +00:00
|
|
|
class ResourceManager;
|
2010-06-15 12:09:51 +00:00
|
|
|
class ResourceSource;
|
2009-06-04 22:16:31 +00:00
|
|
|
|
2009-06-07 02:18:38 +00:00
|
|
|
class ResourceId {
|
2010-06-15 12:31:16 +00:00
|
|
|
static inline ResourceType fixupType(ResourceType type) {
|
2010-07-18 00:18:27 +00:00
|
|
|
if (type >= kResourceTypeInvalid)
|
2010-06-15 12:31:16 +00:00
|
|
|
return kResourceTypeInvalid;
|
|
|
|
return type;
|
|
|
|
}
|
2009-06-07 02:18:38 +00:00
|
|
|
|
2010-06-15 12:31:16 +00:00
|
|
|
ResourceType _type;
|
|
|
|
uint16 _number;
|
|
|
|
uint32 _tuple; // Only used for audio36 and sync36
|
|
|
|
|
2013-12-10 01:40:46 +02:00
|
|
|
static Common::String intToBase36(uint32 number, int minChar) {
|
|
|
|
// Convert from an integer to a base36 string
|
|
|
|
Common::String string;
|
|
|
|
|
|
|
|
while (minChar--) {
|
|
|
|
int character = number % 36;
|
|
|
|
string = ((character < 10) ? (character + '0') : (character + 'A' - 10)) + string;
|
|
|
|
number /= 36;
|
|
|
|
}
|
|
|
|
|
|
|
|
return string;
|
|
|
|
}
|
|
|
|
|
2010-06-15 12:31:16 +00:00
|
|
|
public:
|
|
|
|
ResourceId() : _type(kResourceTypeInvalid), _number(0), _tuple(0) { }
|
2009-06-07 19:15:55 +00:00
|
|
|
|
2010-02-17 23:37:59 +00:00
|
|
|
ResourceId(ResourceType type_, uint16 number_, uint32 tuple_ = 0)
|
2010-06-15 12:31:16 +00:00
|
|
|
: _type(fixupType(type_)), _number(number_), _tuple(tuple_) {
|
2009-06-07 19:15:55 +00:00
|
|
|
}
|
|
|
|
|
2010-02-17 23:37:59 +00:00
|
|
|
ResourceId(ResourceType type_, uint16 number_, byte noun, byte verb, byte cond, byte seq)
|
2010-06-15 12:31:16 +00:00
|
|
|
: _type(fixupType(type_)), _number(number_) {
|
|
|
|
_tuple = (noun << 24) | (verb << 16) | (cond << 8) | seq;
|
2009-06-07 19:15:55 +00:00
|
|
|
}
|
2009-06-07 02:18:38 +00:00
|
|
|
|
2010-06-15 12:31:16 +00:00
|
|
|
Common::String toString() const {
|
2009-06-07 02:18:38 +00:00
|
|
|
char buf[32];
|
|
|
|
|
2010-07-26 07:40:51 +00:00
|
|
|
snprintf(buf, 32, "%s.%d", getResourceTypeName(_type), _number);
|
2009-06-07 02:18:38 +00:00
|
|
|
Common::String retStr = buf;
|
|
|
|
|
2010-06-15 12:31:16 +00:00
|
|
|
if (_tuple != 0) {
|
2010-07-26 07:40:51 +00:00
|
|
|
snprintf(buf, 32, "(%d, %d, %d, %d)", _tuple >> 24, (_tuple >> 16) & 0xff, (_tuple >> 8) & 0xff, _tuple & 0xff);
|
2009-06-07 02:18:38 +00:00
|
|
|
retStr += buf;
|
|
|
|
}
|
|
|
|
|
|
|
|
return retStr;
|
|
|
|
}
|
|
|
|
|
2013-12-10 01:40:46 +02:00
|
|
|
// Convert from a resource ID to a base36 patch name
|
|
|
|
Common::String toPatchNameBase36() {
|
|
|
|
Common::String output;
|
|
|
|
|
|
|
|
output += (getType() == kResourceTypeAudio36) ? '@' : '#'; // Identifier
|
|
|
|
output += intToBase36(getNumber(), 3); // Map
|
|
|
|
output += intToBase36(getTuple() >> 24, 2); // Noun
|
|
|
|
output += intToBase36((getTuple() >> 16) & 0xff, 2); // Verb
|
|
|
|
output += '.'; // Separator
|
|
|
|
output += intToBase36((getTuple() >> 8) & 0xff, 2); // Cond
|
|
|
|
output += intToBase36(getTuple() & 0xff, 1); // Seq
|
|
|
|
|
|
|
|
assert(output.size() == 12); // We should always get 12 characters in the end
|
|
|
|
return output;
|
|
|
|
}
|
|
|
|
|
2010-06-15 12:31:16 +00:00
|
|
|
inline ResourceType getType() const { return _type; }
|
|
|
|
inline uint16 getNumber() const { return _number; }
|
|
|
|
inline uint32 getTuple() const { return _tuple; }
|
2009-06-07 02:18:38 +00:00
|
|
|
|
2010-06-15 12:31:16 +00:00
|
|
|
inline uint hash() const {
|
|
|
|
return ((uint)((_type << 16) | _number)) ^ _tuple;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool operator==(const ResourceId &other) const {
|
|
|
|
return (_type == other._type) && (_number == other._number) && (_tuple == other._tuple);
|
|
|
|
}
|
2009-06-07 02:18:38 +00:00
|
|
|
|
2010-06-15 12:31:16 +00:00
|
|
|
bool operator<(const ResourceId &other) const {
|
|
|
|
return (_type < other._type) || ((_type == other._type) && (_number < other._number))
|
|
|
|
|| ((_type == other._type) && (_number == other._number) && (_tuple < other._tuple));
|
2009-06-07 19:15:55 +00:00
|
|
|
}
|
|
|
|
};
|
|
|
|
|
2010-06-15 12:31:16 +00:00
|
|
|
struct ResourceIdHash : public Common::UnaryFunction<ResourceId, uint> {
|
|
|
|
uint operator()(ResourceId val) const { return val.hash(); }
|
|
|
|
};
|
|
|
|
|
2009-03-03 23:07:06 +00:00
|
|
|
/** Class for storing resources in memory */
|
2009-02-28 21:59:49 +00:00
|
|
|
class Resource {
|
2009-06-04 22:16:31 +00:00
|
|
|
friend class ResourceManager;
|
2010-06-15 12:12:44 +00:00
|
|
|
|
|
|
|
// FIXME: These 'friend' declarations are meant to be a temporary hack to
|
|
|
|
// ease transition to the ResourceSource class system.
|
2010-06-15 12:13:29 +00:00
|
|
|
friend class ResourceSource;
|
2010-06-15 12:19:39 +00:00
|
|
|
friend class PatchResourceSource;
|
2010-06-15 12:14:15 +00:00
|
|
|
friend class WaveResourceSource;
|
|
|
|
friend class AudioVolumeResourceSource;
|
2010-06-15 12:12:44 +00:00
|
|
|
friend class MacResourceForkResourceSource;
|
2010-08-03 14:58:57 +00:00
|
|
|
#ifdef ENABLE_SCI32
|
|
|
|
friend class ChunkResourceSource;
|
|
|
|
#endif
|
2010-06-15 12:12:44 +00:00
|
|
|
|
2010-06-15 12:31:37 +00:00
|
|
|
// NOTE : Currently most member variables lack the underscore prefix and have
|
|
|
|
// public visibility to let the rest of the engine compile without changes.
|
2009-02-28 21:59:49 +00:00
|
|
|
public:
|
2009-03-03 23:07:06 +00:00
|
|
|
byte *data;
|
2009-06-04 22:16:31 +00:00
|
|
|
uint32 size;
|
2010-02-17 23:37:59 +00:00
|
|
|
byte *_header;
|
|
|
|
uint32 _headerSize;
|
|
|
|
|
2010-06-15 12:31:37 +00:00
|
|
|
public:
|
2010-08-03 14:58:57 +00:00
|
|
|
Resource(ResourceManager *resMan, ResourceId id);
|
2010-06-15 12:31:37 +00:00
|
|
|
~Resource();
|
|
|
|
void unalloc();
|
|
|
|
|
2010-06-15 12:31:16 +00:00
|
|
|
inline ResourceType getType() const { return _id.getType(); }
|
2010-06-15 12:35:17 +00:00
|
|
|
inline uint16 getNumber() const { return _id.getNumber(); }
|
2010-07-02 10:18:11 +00:00
|
|
|
bool isLocked() const { return _status == kResStatusLocked; }
|
2010-06-15 12:09:03 +00:00
|
|
|
/**
|
|
|
|
* Write the resource to the specified stream.
|
|
|
|
* This method is used only by the "dump" debugger command.
|
|
|
|
*/
|
2010-02-17 23:37:59 +00:00
|
|
|
void writeToStream(Common::WriteStream *stream) const;
|
2010-06-15 12:35:39 +00:00
|
|
|
|
2010-09-10 14:07:32 +00:00
|
|
|
const Common::String &getResourceLocation() const;
|
|
|
|
|
2010-06-15 12:35:39 +00:00
|
|
|
// FIXME: This audio specific method is a hack. After all, why should a
|
|
|
|
// Resource have audio specific methods? But for now we keep this, as it
|
|
|
|
// eases transition.
|
2010-06-15 12:17:47 +00:00
|
|
|
uint32 getAudioCompressionType() const;
|
2010-02-17 23:37:59 +00:00
|
|
|
|
2009-06-04 22:16:31 +00:00
|
|
|
protected:
|
2010-06-15 12:31:37 +00:00
|
|
|
ResourceId _id; // TODO: _id could almost be made const, only readResourceInfo() modifies it...
|
2010-02-17 23:37:59 +00:00
|
|
|
int32 _fileOffset; /**< Offset in file */
|
|
|
|
ResourceStatus _status;
|
|
|
|
uint16 _lockers; /**< Number of places where this resource was locked */
|
|
|
|
ResourceSource *_source;
|
2010-08-03 14:58:57 +00:00
|
|
|
ResourceManager *_resMan;
|
2010-06-15 12:19:39 +00:00
|
|
|
|
|
|
|
bool loadPatch(Common::SeekableReadStream *file);
|
|
|
|
bool loadFromPatchFile();
|
|
|
|
bool loadFromWaveFile(Common::SeekableReadStream *file);
|
|
|
|
bool loadFromAudioVolumeSCI1(Common::SeekableReadStream *file);
|
|
|
|
bool loadFromAudioVolumeSCI11(Common::SeekableReadStream *file);
|
2010-06-16 00:24:16 +00:00
|
|
|
int decompress(ResVersion volVersion, Common::SeekableReadStream *file);
|
|
|
|
int readResourceInfo(ResVersion volVersion, Common::SeekableReadStream *file, uint32 &szPacked, ResourceCompression &compression);
|
2009-02-22 13:11:43 +00:00
|
|
|
};
|
2009-02-15 06:10:59 +00:00
|
|
|
|
2010-06-15 12:31:16 +00:00
|
|
|
typedef Common::HashMap<ResourceId, Resource *, ResourceIdHash> ResourceMap;
|
2009-02-15 06:10:59 +00:00
|
|
|
|
2009-02-28 20:45:36 +00:00
|
|
|
class ResourceManager {
|
2010-06-15 12:12:21 +00:00
|
|
|
// FIXME: These 'friend' declarations are meant to be a temporary hack to
|
|
|
|
// ease transition to the ResourceSource class system.
|
2010-06-15 12:13:29 +00:00
|
|
|
friend class ResourceSource;
|
2010-06-15 12:12:21 +00:00
|
|
|
friend class DirectoryResourceSource;
|
2010-06-15 12:13:52 +00:00
|
|
|
friend class PatchResourceSource;
|
2010-06-15 12:12:21 +00:00
|
|
|
friend class ExtMapResourceSource;
|
|
|
|
friend class IntMapResourceSource;
|
2010-06-15 12:13:52 +00:00
|
|
|
friend class AudioVolumeResourceSource;
|
2010-06-15 12:12:21 +00:00
|
|
|
friend class ExtAudioMapResourceSource;
|
2010-06-15 12:13:52 +00:00
|
|
|
friend class WaveResourceSource;
|
2010-06-15 12:12:21 +00:00
|
|
|
friend class MacResourceForkResourceSource;
|
2010-08-03 14:58:57 +00:00
|
|
|
#ifdef ENABLE_SCI32
|
|
|
|
friend class ChunkResourceSource;
|
|
|
|
#endif
|
2010-06-15 12:12:21 +00:00
|
|
|
|
2009-02-20 16:03:50 +00:00
|
|
|
public:
|
|
|
|
/**
|
2009-04-03 08:10:58 +00:00
|
|
|
* Creates a new SCI resource manager.
|
2009-02-20 16:03:50 +00:00
|
|
|
*/
|
2009-08-18 09:12:41 +00:00
|
|
|
ResourceManager();
|
2009-02-20 16:03:50 +00:00
|
|
|
~ResourceManager();
|
2009-02-28 20:45:36 +00:00
|
|
|
|
2010-06-15 12:15:52 +00:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Initializes the resource manager.
|
|
|
|
*/
|
2010-11-09 17:07:34 +00:00
|
|
|
void init(bool initFromFallbackDetector = false);
|
2010-06-15 12:15:52 +00:00
|
|
|
|
|
|
|
int addAppropriateSources();
|
|
|
|
int addAppropriateSources(const Common::FSList &fslist); // TODO: Switch from FSList to Common::Archive?
|
|
|
|
|
2009-05-28 11:15:09 +00:00
|
|
|
/**
|
|
|
|
* Looks up a resource's data.
|
2010-06-15 12:15:27 +00:00
|
|
|
* @param id The resource type to look for
|
|
|
|
* @param lock non-zero iff the resource should be locked
|
|
|
|
* @return The resource, or NULL if it doesn't exist
|
2009-05-28 11:15:09 +00:00
|
|
|
* @note Locked resources are guaranteed not to have their contents freed until
|
|
|
|
* they are unlocked explicitly (by unlockResource).
|
2009-02-28 20:45:36 +00:00
|
|
|
*/
|
2009-06-07 19:15:55 +00:00
|
|
|
Resource *findResource(ResourceId id, bool lock);
|
2009-02-28 23:46:50 +00:00
|
|
|
|
2010-06-15 12:15:27 +00:00
|
|
|
/**
|
|
|
|
* Unlocks a previously locked resource.
|
|
|
|
* @param res The resource to free
|
|
|
|
*/
|
2009-06-07 19:15:55 +00:00
|
|
|
void unlockResource(Resource *res);
|
2009-02-28 23:46:50 +00:00
|
|
|
|
2010-06-15 12:15:27 +00:00
|
|
|
/**
|
|
|
|
* Tests whether a resource exists.
|
|
|
|
*
|
|
|
|
* This function may often be much faster than finding the resource
|
|
|
|
* and should be preferred for simple tests.
|
|
|
|
* The resource object returned is, indeed, the resource in question, but
|
|
|
|
* it should be used with care, as it may be unallocated.
|
|
|
|
* Use scir_find_resource() if you want to use the data contained in the resource.
|
|
|
|
*
|
|
|
|
* @param id Id of the resource to check
|
|
|
|
* @return non-NULL if the resource exists, NULL otherwise
|
|
|
|
*/
|
2009-06-07 19:15:55 +00:00
|
|
|
Resource *testResource(ResourceId id);
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Returns a list of all resources of the specified type.
|
2010-06-15 12:15:27 +00:00
|
|
|
* @param type The resource type to look for
|
|
|
|
* @param mapNumber For audio36 and sync36, limit search to this map
|
|
|
|
* @return The resource list
|
2009-06-07 19:15:55 +00:00
|
|
|
*/
|
2012-01-15 18:25:00 +01:00
|
|
|
Common::List<ResourceId> listResources(ResourceType type, int mapNumber = -1);
|
2009-02-28 20:45:36 +00:00
|
|
|
|
2009-06-09 19:18:48 +00:00
|
|
|
void setAudioLanguage(int language);
|
2010-02-17 23:37:32 +00:00
|
|
|
int getAudioLanguage() const;
|
2011-02-10 16:48:29 +00:00
|
|
|
void changeAudioDirectory(Common::String path);
|
2010-08-19 13:52:21 +00:00
|
|
|
bool isGMTrackIncluded();
|
2010-05-19 15:57:58 +00:00
|
|
|
bool isSci11Mac() const { return _volVersion == kResVersionSci11Mac; }
|
2010-01-03 20:18:36 +00:00
|
|
|
ViewType getViewType() const { return _viewType; }
|
|
|
|
const char *getMapVersionDesc() const { return versionDescription(_mapVersion); }
|
|
|
|
const char *getVolVersionDesc() const { return versionDescription(_volVersion); }
|
2010-06-15 12:21:29 +00:00
|
|
|
ResVersion getVolVersion() const { return _volVersion; }
|
2010-01-25 01:39:44 +00:00
|
|
|
|
2010-01-16 00:22:22 +00:00
|
|
|
/**
|
|
|
|
* Adds the appropriate GM patch from the Sierra MIDI utility as 4.pat, without
|
|
|
|
* requiring the user to rename the file to 4.pat. Thus, the original Sierra
|
|
|
|
* archive can be extracted in the extras directory, and the GM patches can be
|
2010-06-15 12:15:27 +00:00
|
|
|
* applied per game, if applicable.
|
2010-01-16 00:22:22 +00:00
|
|
|
*/
|
2010-06-25 16:16:29 +00:00
|
|
|
void addNewGMPatch(SciGameId gameId);
|
2009-06-09 19:18:48 +00:00
|
|
|
|
2010-08-03 14:58:57 +00:00
|
|
|
#ifdef ENABLE_SCI32
|
|
|
|
/**
|
|
|
|
* Parses all resources from a SCI2.1 chunk resource and adds them to the
|
|
|
|
* resource manager.
|
|
|
|
*/
|
|
|
|
void addResourcesFromChunk(uint16 id);
|
|
|
|
#endif
|
|
|
|
|
2010-01-28 21:01:57 +00:00
|
|
|
bool detectHires();
|
2010-04-17 20:43:09 +00:00
|
|
|
// Detects, if standard font of current game includes extended characters (>0x80)
|
|
|
|
bool detectFontExtended();
|
2010-06-25 10:04:31 +00:00
|
|
|
// Detects, if SCI1.1 game uses palette merging
|
2011-10-19 23:44:34 +03:00
|
|
|
bool detectPaletteMergingSci11();
|
2010-09-04 08:51:10 +00:00
|
|
|
// Detects, if SCI0EARLY game also has SCI0EARLY sound resources
|
|
|
|
bool detectEarlySound();
|
2010-01-28 21:01:57 +00:00
|
|
|
|
2010-05-23 10:28:03 +00:00
|
|
|
/**
|
2010-06-15 12:15:27 +00:00
|
|
|
* Finds the internal Sierra ID of the current game from script 0.
|
2010-05-23 10:28:03 +00:00
|
|
|
*/
|
|
|
|
Common::String findSierraGameId();
|
|
|
|
|
2010-05-29 14:03:08 +00:00
|
|
|
/**
|
2010-06-15 12:15:27 +00:00
|
|
|
* Finds the location of the game object from script 0.
|
|
|
|
* @param addSci11ScriptOffset Adjust the return value for SCI1.1 and newer
|
2010-05-29 14:03:08 +00:00
|
|
|
* games. Needs to be false when the heap is accessed directly inside
|
|
|
|
* findSierraGameId().
|
|
|
|
*/
|
|
|
|
reg_t findGameObject(bool addSci11ScriptOffset = true);
|
|
|
|
|
2010-07-18 00:16:19 +00:00
|
|
|
/**
|
|
|
|
* Converts a map resource type to our type
|
|
|
|
* @param sciType The type from the map/patch
|
|
|
|
* @return The ResourceType
|
|
|
|
*/
|
|
|
|
ResourceType convertResType(byte type);
|
|
|
|
|
2009-02-28 20:45:36 +00:00
|
|
|
protected:
|
2009-08-18 09:12:41 +00:00
|
|
|
// Maximum number of bytes to allow being allocated for resources
|
|
|
|
// Note: maxMemory will not be interpreted as a hard limit, only as a restriction
|
|
|
|
// for resources which are not explicitly locked. However, a warning will be
|
|
|
|
// issued whenever this limit is exceeded.
|
2009-09-01 13:02:47 +00:00
|
|
|
enum {
|
|
|
|
MAX_MEMORY = 256 * 1024 // 256KB
|
|
|
|
};
|
2009-08-18 09:12:41 +00:00
|
|
|
|
2009-08-16 19:18:19 +00:00
|
|
|
ViewType _viewType; // Used to determine if the game has EGA or VGA graphics
|
2009-06-08 12:20:36 +00:00
|
|
|
Common::List<ResourceSource *> _sources;
|
2009-10-08 21:28:57 +00:00
|
|
|
int _memoryLocked; ///< Amount of resource bytes in locked memory
|
|
|
|
int _memoryLRU; ///< Amount of resource bytes under LRU control
|
|
|
|
Common::List<Resource *> _LRU; ///< Last Resource Used list
|
2009-06-07 02:18:38 +00:00
|
|
|
ResourceMap _resMap;
|
2009-10-08 21:28:57 +00:00
|
|
|
Common::List<Common::File *> _volumeFiles; ///< list of opened volume files
|
|
|
|
ResourceSource *_audioMapSCI1; ///< Currently loaded audio map for SCI1
|
2010-06-01 22:06:52 +00:00
|
|
|
ResVersion _volVersion; ///< resource.0xx version
|
|
|
|
ResVersion _mapVersion; ///< resource.map version
|
2009-03-11 20:15:42 +00:00
|
|
|
|
2009-05-28 11:15:09 +00:00
|
|
|
/**
|
|
|
|
* Add a path to the resource manager's list of sources.
|
|
|
|
* @return a pointer to the added source structure, or NULL if an error occurred.
|
|
|
|
*/
|
2010-06-15 12:10:18 +00:00
|
|
|
ResourceSource *addPatchDir(const Common::String &path);
|
2009-02-28 23:46:50 +00:00
|
|
|
|
2010-06-15 12:11:56 +00:00
|
|
|
ResourceSource *findVolume(ResourceSource *map, int volume_nr);
|
2009-03-11 20:15:42 +00:00
|
|
|
|
2009-05-28 11:15:09 +00:00
|
|
|
/**
|
2009-06-06 00:07:18 +00:00
|
|
|
* Adds a source to the resource manager's list of sources.
|
2010-06-15 12:11:04 +00:00
|
|
|
* @param source The new source to add
|
2010-06-15 12:15:27 +00:00
|
|
|
* @return A pointer to the added source structure, or NULL if an error occurred.
|
2009-03-11 20:15:42 +00:00
|
|
|
*/
|
2010-06-15 12:14:39 +00:00
|
|
|
ResourceSource *addSource(ResourceSource *source);
|
2009-08-18 14:10:31 +00:00
|
|
|
|
2009-05-28 11:15:09 +00:00
|
|
|
/**
|
2010-06-15 12:15:27 +00:00
|
|
|
* Add an external (i.e., separate file) map resource to the resource
|
|
|
|
* manager's list of sources.
|
2010-06-15 12:20:24 +00:00
|
|
|
* @param filename The name of the volume to add
|
2010-01-27 04:26:28 +00:00
|
|
|
* @param volume_nr The volume number the map starts at, 0 for <SCI2.1
|
2009-05-28 11:15:09 +00:00
|
|
|
* @return A pointer to the added source structure, or NULL if an error occurred.
|
2009-03-11 20:15:42 +00:00
|
|
|
*/
|
2010-06-15 12:10:18 +00:00
|
|
|
ResourceSource *addExternalMap(const Common::String &filename, int volume_nr = 0);
|
2009-08-18 14:10:31 +00:00
|
|
|
|
2010-06-01 22:06:52 +00:00
|
|
|
ResourceSource *addExternalMap(const Common::FSNode *mapFile, int volume_nr = 0);
|
2009-08-18 14:10:31 +00:00
|
|
|
|
2009-05-28 11:15:09 +00:00
|
|
|
/**
|
|
|
|
* Scans newly registered resource sources for resources, earliest addition first.
|
2010-06-15 12:15:27 +00:00
|
|
|
* @param detected_version Pointer to the detected version number,
|
2009-03-11 20:15:42 +00:00
|
|
|
* used during startup. May be NULL.
|
|
|
|
* @return One of SCI_ERROR_*.
|
|
|
|
*/
|
2009-06-08 12:20:36 +00:00
|
|
|
void scanNewSources();
|
2010-06-15 12:15:52 +00:00
|
|
|
|
2011-02-10 12:57:43 +00:00
|
|
|
bool addAudioSources();
|
|
|
|
void addScriptChunkSources();
|
2009-06-08 12:20:36 +00:00
|
|
|
void freeResourceSources();
|
2009-02-28 20:45:36 +00:00
|
|
|
|
2009-08-15 00:28:59 +00:00
|
|
|
/**
|
2010-06-15 12:15:27 +00:00
|
|
|
* Returns a string describing a ResVersion.
|
|
|
|
* @param version The resource version
|
|
|
|
* @return The description of version
|
2009-08-15 00:28:59 +00:00
|
|
|
*/
|
|
|
|
const char *versionDescription(ResVersion version) const;
|
|
|
|
|
2010-06-06 13:04:24 +00:00
|
|
|
Common::SeekableReadStream *getVolumeFile(ResourceSource *source);
|
2009-03-05 23:27:02 +00:00
|
|
|
void loadResource(Resource *res);
|
2009-06-09 19:18:48 +00:00
|
|
|
void freeOldResources();
|
2009-06-07 19:15:55 +00:00
|
|
|
void addResource(ResourceId resId, ResourceSource *src, uint32 offset, uint32 size = 0);
|
2010-06-15 12:18:31 +00:00
|
|
|
Resource *updateResource(ResourceId resId, ResourceSource *src, uint32 size);
|
2009-06-09 19:18:48 +00:00
|
|
|
void removeAudioResource(ResourceId resId);
|
2009-02-28 20:45:36 +00:00
|
|
|
|
|
|
|
/**--- Resource map decoding functions ---*/
|
2009-08-15 00:28:59 +00:00
|
|
|
ResVersion detectMapVersion();
|
|
|
|
ResVersion detectVolVersion();
|
2009-02-28 20:45:36 +00:00
|
|
|
|
2009-05-28 11:15:09 +00:00
|
|
|
/**
|
|
|
|
* Reads the SCI0 resource.map file from a local directory.
|
2009-06-09 19:18:48 +00:00
|
|
|
* @param map The map
|
2009-05-28 11:15:09 +00:00
|
|
|
* @return 0 on success, an SCI_ERROR_* code otherwise
|
|
|
|
*/
|
2009-03-07 00:59:38 +00:00
|
|
|
int readResourceMapSCI0(ResourceSource *map);
|
2009-02-28 23:46:50 +00:00
|
|
|
|
2009-05-28 11:15:09 +00:00
|
|
|
/**
|
|
|
|
* Reads the SCI1 resource.map file from a local directory.
|
2009-06-09 19:18:48 +00:00
|
|
|
* @param map The map
|
2009-05-28 11:15:09 +00:00
|
|
|
* @return 0 on success, an SCI_ERROR_* code otherwise
|
|
|
|
*/
|
2009-03-26 13:11:30 +00:00
|
|
|
int readResourceMapSCI1(ResourceSource *map);
|
2010-06-15 12:15:27 +00:00
|
|
|
|
2009-06-06 00:07:18 +00:00
|
|
|
/**
|
2010-06-15 12:15:27 +00:00
|
|
|
* Reads SCI1.1 audio map resources.
|
2009-06-09 19:18:48 +00:00
|
|
|
* @param map The map
|
2009-06-06 00:07:18 +00:00
|
|
|
* @return 0 on success, an SCI_ERROR_* code otherwise
|
|
|
|
*/
|
2009-06-09 19:18:48 +00:00
|
|
|
int readAudioMapSCI11(ResourceSource *map);
|
|
|
|
|
|
|
|
/**
|
2010-06-15 12:15:27 +00:00
|
|
|
* Reads SCI1 audio map files.
|
2009-06-09 19:18:48 +00:00
|
|
|
* @param map The map
|
|
|
|
* @param unload Unload the map instead of loading it
|
|
|
|
* @return 0 on success, an SCI_ERROR_* code otherwise
|
|
|
|
*/
|
|
|
|
int readAudioMapSCI1(ResourceSource *map, bool unload = false);
|
2009-06-06 00:07:18 +00:00
|
|
|
|
2009-02-28 20:45:36 +00:00
|
|
|
/**--- Patch management functions ---*/
|
|
|
|
|
2009-05-28 11:15:09 +00:00
|
|
|
/**
|
|
|
|
* Reads patch files from a local directory.
|
|
|
|
*/
|
2010-06-15 12:19:39 +00:00
|
|
|
void readResourcePatches();
|
|
|
|
void readResourcePatchesBase36();
|
2010-06-07 21:55:57 +00:00
|
|
|
void processPatch(ResourceSource *source, ResourceType resourceType, uint16 resourceNr, uint32 tuple = 0);
|
2010-01-25 01:39:44 +00:00
|
|
|
|
2009-12-27 22:34:12 +00:00
|
|
|
/**
|
2010-06-15 12:15:27 +00:00
|
|
|
* Process wave files as patches for Audio resources.
|
2009-12-27 22:34:12 +00:00
|
|
|
*/
|
|
|
|
void readWaveAudioPatches();
|
2010-06-10 15:06:25 +00:00
|
|
|
void processWavePatch(ResourceId resourceId, Common::String name);
|
2009-02-28 20:45:36 +00:00
|
|
|
|
2013-11-23 11:02:24 +01:00
|
|
|
/**
|
2009-08-25 08:38:14 +00:00
|
|
|
* Applies to all versions before 0.000.395 (i.e. KQ4 old, XMAS 1988 and LSL2).
|
|
|
|
* Old SCI versions used two word header for script blocks (first word equal
|
|
|
|
* to 0x82, meaning of the second one unknown). New SCI versions used one
|
|
|
|
* word header.
|
|
|
|
* Also, old SCI versions assign 120 degrees to left & right, and 60 to up
|
|
|
|
* and down. Later versions use an even 90 degree distribution.
|
|
|
|
*/
|
|
|
|
bool hasOldScriptHeader();
|
|
|
|
|
2009-02-28 20:45:36 +00:00
|
|
|
void printLRU();
|
2009-02-28 21:59:49 +00:00
|
|
|
void addToLRU(Resource *res);
|
|
|
|
void removeFromLRU(Resource *res);
|
2009-06-04 22:16:31 +00:00
|
|
|
|
2009-08-16 19:18:19 +00:00
|
|
|
ResourceCompression getViewCompression();
|
|
|
|
ViewType detectViewType();
|
|
|
|
bool hasSci0Voc999();
|
|
|
|
bool hasSci1Voc900();
|
2009-09-23 10:55:35 +00:00
|
|
|
void detectSciVersion();
|
2009-02-20 15:55:59 +00:00
|
|
|
};
|
2009-02-15 06:10:59 +00:00
|
|
|
|
2009-12-19 16:19:53 +00:00
|
|
|
class SoundResource {
|
|
|
|
public:
|
2009-12-23 16:17:42 +00:00
|
|
|
struct Channel {
|
2009-12-19 16:19:53 +00:00
|
|
|
byte number;
|
2013-04-14 01:32:37 +02:00
|
|
|
byte flags;
|
2009-12-19 16:19:53 +00:00
|
|
|
byte poly;
|
2010-01-02 01:40:10 +00:00
|
|
|
uint16 prio;
|
2009-12-19 16:19:53 +00:00
|
|
|
uint16 size;
|
2009-12-23 16:17:42 +00:00
|
|
|
byte *data;
|
2010-06-26 19:55:49 +00:00
|
|
|
uint16 curPos;
|
2009-12-19 16:19:53 +00:00
|
|
|
long time;
|
|
|
|
byte prev;
|
|
|
|
};
|
|
|
|
|
2009-12-23 16:17:42 +00:00
|
|
|
struct Track {
|
2010-01-09 02:14:12 +00:00
|
|
|
byte type;
|
2009-12-23 16:17:42 +00:00
|
|
|
byte channelCount;
|
|
|
|
Channel *channels;
|
2009-12-27 12:33:47 +00:00
|
|
|
int16 digitalChannelNr;
|
2010-01-27 16:04:46 +00:00
|
|
|
uint16 digitalSampleRate;
|
|
|
|
uint16 digitalSampleSize;
|
|
|
|
uint16 digitalSampleStart;
|
|
|
|
uint16 digitalSampleEnd;
|
2009-12-19 16:19:53 +00:00
|
|
|
};
|
|
|
|
public:
|
2009-12-23 16:17:42 +00:00
|
|
|
SoundResource(uint32 resNumber, ResourceManager *resMan, SciVersion soundVersion);
|
2009-12-19 16:19:53 +00:00
|
|
|
~SoundResource();
|
2010-01-02 01:40:10 +00:00
|
|
|
#if 0
|
2009-12-23 16:17:42 +00:00
|
|
|
Track *getTrackByNumber(uint16 number);
|
2010-01-02 01:40:10 +00:00
|
|
|
#endif
|
2010-01-09 02:14:12 +00:00
|
|
|
Track *getTrackByType(byte type);
|
|
|
|
Track *getDigitalTrack();
|
2010-01-16 19:11:37 +00:00
|
|
|
int getChannelFilterMask(int hardwareMask, bool wantsRhythm);
|
2010-01-26 19:25:33 +00:00
|
|
|
byte getInitialVoiceCount(byte channel);
|
2009-12-19 16:19:53 +00:00
|
|
|
|
|
|
|
private:
|
2009-12-23 16:17:42 +00:00
|
|
|
SciVersion _soundVersion;
|
|
|
|
int _trackCount;
|
|
|
|
Track *_tracks;
|
2009-12-19 16:19:53 +00:00
|
|
|
Resource *_innerResource;
|
|
|
|
ResourceManager *_resMan;
|
|
|
|
};
|
|
|
|
|
2009-02-21 10:23:36 +00:00
|
|
|
} // End of namespace Sci
|
2009-02-15 06:10:59 +00:00
|
|
|
|
2010-06-15 12:08:40 +00:00
|
|
|
#endif // SCI_RESOURCE_H
|