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.
|
|
|
|
|
|
|
|
* 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$
|
|
|
|
*
|
|
|
|
*/
|
2009-02-15 06:10:59 +00:00
|
|
|
|
2009-02-27 02:23:00 +00:00
|
|
|
#ifndef SCI_SCICORE_RESOURCE_H
|
|
|
|
#define SCI_SCICORE_RESOURCE_H
|
2009-02-15 06:10:59 +00:00
|
|
|
|
2009-02-20 16:03:50 +00:00
|
|
|
#include "common/str.h"
|
2009-02-28 20:45:36 +00:00
|
|
|
#include "common/file.h"
|
|
|
|
#include "common/archive.h"
|
2009-02-20 16:03:50 +00:00
|
|
|
|
2009-04-25 08:50:42 +00:00
|
|
|
#include "sci/engine/vm.h" // for Object
|
2009-03-10 21:44:03 +00:00
|
|
|
#include "sci/scicore/decompressor.h"
|
|
|
|
|
2009-02-21 19:54:15 +00:00
|
|
|
namespace Common {
|
2009-03-11 01:40:08 +00:00
|
|
|
class ReadStream;
|
2009-02-21 19:54:15 +00:00
|
|
|
}
|
|
|
|
|
2009-02-21 10:23:36 +00:00
|
|
|
namespace Sci {
|
|
|
|
|
2009-02-21 19:27:06 +00:00
|
|
|
/** The maximum allowed size for a compressed or decompressed resource */
|
2009-02-15 06:10:59 +00:00
|
|
|
#define SCI_MAX_RESOURCE_SIZE 0x0400000
|
|
|
|
|
|
|
|
/*** 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,
|
|
|
|
kResStatusEnqueued, /* In the LRU queue */
|
|
|
|
kResStatusLocked /* Allocated and in use */
|
|
|
|
};
|
2009-02-15 06:10:59 +00:00
|
|
|
|
|
|
|
/*** INITIALIZATION RESULT TYPES ***/
|
|
|
|
#define SCI_ERROR_IO_ERROR 1
|
|
|
|
#define SCI_ERROR_EMPTY_OBJECT 2
|
|
|
|
#define SCI_ERROR_INVALID_RESMAP_ENTRY 3
|
|
|
|
/* Invalid resource.map entry */
|
|
|
|
#define SCI_ERROR_RESMAP_NOT_FOUND 4
|
|
|
|
#define SCI_ERROR_NO_RESOURCE_FILES_FOUND 5
|
|
|
|
/* No resource at all was found */
|
|
|
|
#define SCI_ERROR_UNKNOWN_COMPRESSION 6
|
|
|
|
#define SCI_ERROR_DECOMPRESSION_OVERFLOW 7
|
|
|
|
/* decompression failed: Buffer overflow (wrong SCI version?) */
|
|
|
|
#define SCI_ERROR_DECOMPRESSION_INSANE 8
|
|
|
|
/* sanity checks failed during decompression */
|
|
|
|
#define SCI_ERROR_RESOURCE_TOO_BIG 9
|
|
|
|
/* Resource size exceeds SCI_MAX_RESOURCE_SIZE */
|
|
|
|
#define SCI_ERROR_UNSUPPORTED_VERSION 10
|
|
|
|
#define SCI_ERROR_INVALID_SCRIPT_VERSION 11
|
|
|
|
|
|
|
|
#define SCI_ERROR_CRITICAL SCI_ERROR_NO_RESOURCE_FILES_FOUND
|
|
|
|
/* the first critical error number */
|
|
|
|
|
|
|
|
/*** SCI VERSION NUMBERS ***/
|
|
|
|
#define SCI_VERSION_AUTODETECT 0
|
|
|
|
#define SCI_VERSION_0 1
|
|
|
|
#define SCI_VERSION_01 2
|
|
|
|
#define SCI_VERSION_01_VGA 3
|
|
|
|
#define SCI_VERSION_01_VGA_ODD 4
|
|
|
|
#define SCI_VERSION_1_EARLY 5
|
|
|
|
#define SCI_VERSION_1_LATE 6
|
|
|
|
#define SCI_VERSION_1_1 7
|
2009-03-30 07:53:32 +00:00
|
|
|
#ifdef ENABLE_SCI32
|
2009-02-15 06:10:59 +00:00
|
|
|
#define SCI_VERSION_32 8
|
2009-03-30 07:53:32 +00:00
|
|
|
#endif
|
2009-02-15 06:10:59 +00:00
|
|
|
#define SCI_VERSION_LAST SCI_VERSION_1_LATE /* The last supported SCI version */
|
|
|
|
|
|
|
|
#define SCI_VERSION_1 SCI_VERSION_1_EARLY
|
|
|
|
|
2009-03-11 20:15:42 +00:00
|
|
|
#define MAX_OPENED_VOLUMES 5 // Max number of simultaneously opened volumes
|
|
|
|
|
2009-02-28 21:59:49 +00:00
|
|
|
enum ResSourceType {
|
2009-03-03 23:07:06 +00:00
|
|
|
kSourceDirectory = 0,
|
2009-03-05 23:27:02 +00:00
|
|
|
kSourcePatch = 1,
|
2009-03-03 23:07:06 +00:00
|
|
|
kSourceVolume = 2,
|
|
|
|
kSourceExtMap = 3,
|
|
|
|
kSourceIntMap = 4,
|
|
|
|
kSourceMask = 127
|
2009-02-22 04:22:53 +00:00
|
|
|
};
|
|
|
|
|
2009-02-15 06:10:59 +00:00
|
|
|
#define RESSOURCE_ADDRESSING_BASIC 0
|
|
|
|
#define RESSOURCE_ADDRESSING_EXTENDED 128
|
|
|
|
#define RESSOURCE_ADDRESSING_MASK 128
|
|
|
|
|
2009-03-11 01:40:08 +00:00
|
|
|
#define RESOURCE_HASH(type, number) (uint32)((type<<16) | number)
|
2009-03-07 04:49:34 +00:00
|
|
|
#define SCI0_RESMAP_ENTRIES_SIZE 6
|
|
|
|
#define SCI1_RESMAP_ENTRIES_SIZE 6
|
|
|
|
#define SCI11_RESMAP_ENTRIES_SIZE 5
|
2009-03-03 23:07:06 +00:00
|
|
|
|
2009-02-27 12:59:02 +00:00
|
|
|
extern const char *sci_error_types[];
|
|
|
|
extern const char *sci_version_types[];
|
2009-02-15 10:33:31 +00:00
|
|
|
extern const int sci_max_resource_nr[]; /* Highest possible resource numbers */
|
2009-02-15 06:10:59 +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,
|
|
|
|
kResourceTypeInvalid
|
2009-02-15 06:10:59 +00:00
|
|
|
};
|
|
|
|
|
2009-02-28 23:46:50 +00:00
|
|
|
const char *getResourceTypeName(ResourceType restype);
|
|
|
|
// Suffixes for SCI1 patch files
|
|
|
|
const char *getResourceTypeSuffix(ResourceType restype);
|
|
|
|
|
|
|
|
#define sci0_last_resource kResourceTypePatch
|
|
|
|
#define sci1_last_resource kResourceTypeHeap
|
2009-02-15 06:10:59 +00:00
|
|
|
/* Used for autodetection */
|
|
|
|
|
|
|
|
|
2009-03-11 20:15:42 +00:00
|
|
|
// resource type for SCI1 resource.map file
|
2009-03-07 04:42:59 +00:00
|
|
|
struct resource_index_t {
|
|
|
|
uint16 wOffset;
|
|
|
|
uint16 wSize;
|
2009-03-11 20:15:42 +00:00
|
|
|
};
|
2009-02-15 06:10:59 +00:00
|
|
|
|
2009-02-20 15:55:59 +00:00
|
|
|
struct ResourceSource {
|
2009-02-28 21:59:49 +00:00
|
|
|
ResSourceType source_type;
|
2009-02-20 16:03:50 +00:00
|
|
|
bool scanned;
|
|
|
|
Common::String location_name; // FIXME: Replace by FSNode ?
|
|
|
|
int volume_number;
|
2009-02-20 15:55:59 +00:00
|
|
|
ResourceSource *associated_map;
|
|
|
|
ResourceSource *next;
|
|
|
|
};
|
2009-02-15 06:10:59 +00:00
|
|
|
|
2009-03-03 23:07:06 +00:00
|
|
|
/** Class for storing resources in memory */
|
2009-02-28 21:59:49 +00:00
|
|
|
class Resource {
|
|
|
|
public:
|
2009-03-04 04:35:06 +00:00
|
|
|
Resource();
|
|
|
|
~Resource();
|
2009-03-03 23:07:06 +00:00
|
|
|
void unalloc();
|
|
|
|
|
2009-02-28 21:59:49 +00:00
|
|
|
// NOTE : Currently all member data has the same name and public visibility
|
|
|
|
// to let the rest of the engine compile without changes
|
2009-03-03 23:07:06 +00:00
|
|
|
public:
|
|
|
|
byte *data;
|
|
|
|
uint16 number;
|
2009-02-28 23:46:50 +00:00
|
|
|
ResourceType type;
|
2009-03-11 01:40:08 +00:00
|
|
|
uint32 id; // contains number and type.
|
2009-02-15 06:10:59 +00:00
|
|
|
unsigned int size;
|
|
|
|
unsigned int file_offset; /* Offset in file */
|
2009-03-10 21:44:03 +00:00
|
|
|
ResourceStatus status;
|
2009-02-15 06:10:59 +00:00
|
|
|
unsigned short lockers; /* Number of places where this resource was locked */
|
2009-02-28 21:59:49 +00:00
|
|
|
ResourceSource *source;
|
2009-02-22 13:11:43 +00:00
|
|
|
};
|
2009-02-15 06:10:59 +00:00
|
|
|
|
|
|
|
|
2009-02-28 20:45:36 +00:00
|
|
|
class ResourceManager {
|
2009-02-20 16:03:50 +00:00
|
|
|
public:
|
2009-02-28 22:19:22 +00:00
|
|
|
int _sciVersion; /* SCI resource version to use */
|
2009-03-07 00:59:38 +00:00
|
|
|
int _mapVersion; // RESOURCE.MAP version
|
|
|
|
int _volVersion; // RESOURCE.0xx version
|
2009-02-28 23:46:50 +00:00
|
|
|
|
2009-02-20 16:03:50 +00:00
|
|
|
/**
|
2009-04-03 08:10:58 +00:00
|
|
|
* Creates a new SCI resource manager.
|
2009-02-20 16:03:50 +00:00
|
|
|
* @param version The SCI version to look for; use SCI_VERSION_AUTODETECT
|
|
|
|
* in the default case.
|
|
|
|
* @param maxMemory Maximum number of bytes to allow 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.
|
|
|
|
*/
|
|
|
|
ResourceManager(int version, int maxMemory);
|
|
|
|
~ResourceManager();
|
2009-02-28 20:45:36 +00:00
|
|
|
|
|
|
|
//! Looks up a resource's data
|
|
|
|
/** @param type: The resource type to look for
|
|
|
|
* @param number: The resource number to search
|
|
|
|
* @param lock: non-zero iff the resource should be locked
|
2009-02-28 21:59:49 +00:00
|
|
|
* @return (Resource *): The resource, or NULL if it doesn't exist
|
2009-02-28 20:45:36 +00:00
|
|
|
* @note Locked resources are guaranteed not to have their contents freed until
|
|
|
|
* they are unlocked explicitly (by unlockResource).
|
|
|
|
*/
|
2009-02-28 23:46:50 +00:00
|
|
|
Resource *findResource(ResourceType type, int number, int lock);
|
|
|
|
|
2009-02-28 20:45:36 +00:00
|
|
|
/* Unlocks a previously locked resource
|
2009-02-28 21:59:49 +00:00
|
|
|
** (Resource *) res: The resource to free
|
2009-02-28 20:45:36 +00:00
|
|
|
** (int) number: Number of the resource to check (ditto)
|
2009-02-28 23:46:50 +00:00
|
|
|
** (ResourceType) type: Type of the resource to check (for error checking)
|
2009-02-28 20:45:36 +00:00
|
|
|
** Returns : (void)
|
|
|
|
*/
|
2009-02-28 23:46:50 +00:00
|
|
|
void unlockResource(Resource *res, int restype, ResourceType resnum);
|
|
|
|
|
2009-02-28 20:45:36 +00:00
|
|
|
/* Tests whether a resource exists
|
2009-02-28 23:46:50 +00:00
|
|
|
** (ResourceType) type: Type of the resource to check
|
2009-02-28 20:45:36 +00:00
|
|
|
** (int) number: Number of the resource to check
|
2009-02-28 21:59:49 +00:00
|
|
|
** Returns : (Resource *) non-NULL if the resource exists, NULL otherwise
|
2009-02-28 20:45:36 +00:00
|
|
|
** 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.
|
|
|
|
*/
|
2009-02-28 23:46:50 +00:00
|
|
|
Resource *testResource(ResourceType type, int number);
|
2009-02-28 20:45:36 +00:00
|
|
|
|
|
|
|
protected:
|
2009-03-11 20:15:42 +00:00
|
|
|
int _maxMemory; // Config option: Maximum total byte number allocated
|
2009-02-28 20:45:36 +00:00
|
|
|
ResourceSource *_sources;
|
|
|
|
int _memoryLocked; // Amount of resource bytes in locked memory
|
|
|
|
int _memoryLRU; // Amount of resource bytes under LRU control
|
2009-03-05 23:27:02 +00:00
|
|
|
Common::List<Resource *> _LRU; // Last Resource Used list
|
2009-03-03 23:07:06 +00:00
|
|
|
Common::HashMap<uint32, Resource *> _resMap;
|
2009-03-11 20:15:42 +00:00
|
|
|
Common::List<Common::File *> _volumeFiles; // list of opened volume files
|
|
|
|
|
|
|
|
/* Add a path to the resource manager's list of sources.
|
|
|
|
** Returns: A pointer to the added source structure, or NULL if an error occurred.
|
|
|
|
*/
|
|
|
|
ResourceSource *addPatchDir(const char *path);
|
2009-02-28 23:46:50 +00:00
|
|
|
|
2009-03-11 20:15:42 +00:00
|
|
|
ResourceSource *getVolume(ResourceSource *map, int volume_nr);
|
|
|
|
|
|
|
|
//! Add a volume to the resource manager's list of sources.
|
|
|
|
/** @param map The map associated with this volume
|
|
|
|
* @param filename The name of the volume to add
|
|
|
|
* @param extended_addressing 1 if this volume uses extended addressing,
|
|
|
|
* 0 otherwise.
|
|
|
|
* @return A pointer to the added source structure, or NULL if an error occurred.
|
|
|
|
*/
|
|
|
|
ResourceSource *addVolume(ResourceSource *map, const char *filename,
|
|
|
|
int number, int extended_addressing);
|
|
|
|
//! Add an external (i.e. separate file) map resource to the resource manager's list of sources.
|
|
|
|
/** @param file_name The name of the volume to add
|
|
|
|
* @return A pointer to the added source structure, or NULL if an error occurred.
|
|
|
|
*/
|
|
|
|
ResourceSource *addExternalMap(const char *file_name);
|
|
|
|
//! Scans newly registered resource sources for resources, earliest addition first.
|
|
|
|
/** @param detected_version: Pointer to the detected version number,
|
|
|
|
* used during startup. May be NULL.
|
|
|
|
* @return One of SCI_ERROR_*.
|
|
|
|
*/
|
|
|
|
int scanNewSources(ResourceSource *source);
|
2009-02-28 23:46:50 +00:00
|
|
|
int addAppropriateSources();
|
2009-02-28 20:45:36 +00:00
|
|
|
void freeResourceSources(ResourceSource *rss);
|
|
|
|
|
2009-03-11 20:15:42 +00:00
|
|
|
Common::File *getVolumeFile(const char *filename);
|
2009-03-05 23:27:02 +00:00
|
|
|
void loadResource(Resource *res);
|
2009-03-03 23:07:06 +00:00
|
|
|
bool loadFromPatchFile(Resource *res);
|
2009-02-28 20:45:36 +00:00
|
|
|
void freeOldResources(int last_invulnerable);
|
2009-03-10 21:44:03 +00:00
|
|
|
int decompress(Resource *res, Common::File *file);
|
|
|
|
int readResourceInfo(Resource *res, Common::File *file, uint32&szPacked, ResourceCompression &compression);
|
2009-02-28 20:45:36 +00:00
|
|
|
|
|
|
|
/**--- Resource map decoding functions ---*/
|
2009-03-07 00:59:38 +00:00
|
|
|
int detectMapVersion();
|
|
|
|
int detectVolVersion();
|
2009-02-28 20:45:36 +00:00
|
|
|
|
|
|
|
/* Reads the SCI0 resource.map file from a local directory
|
|
|
|
** Returns : (int) 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-02-28 20:45:36 +00:00
|
|
|
/* Reads the SCI1 resource.map file from a local directory
|
|
|
|
** Returns : (int) 0 on success, an SCI_ERROR_* code otherwise
|
|
|
|
*/
|
2009-03-26 13:11:30 +00:00
|
|
|
int readResourceMapSCI1(ResourceSource *map);
|
2009-02-28 23:46:50 +00:00
|
|
|
|
2009-02-28 20:45:36 +00:00
|
|
|
/**--- Patch management functions ---*/
|
|
|
|
|
2009-03-11 20:15:42 +00:00
|
|
|
//! Reads patch files from a local directory
|
2009-03-03 23:07:06 +00:00
|
|
|
/** @paramParameters: ResourceSource *source
|
|
|
|
*/
|
2009-03-05 22:19:29 +00:00
|
|
|
void readResourcePatches(ResourceSource *source);
|
2009-03-11 22:52:54 +00:00
|
|
|
void processPatch(ResourceSource *source, ResourceType restype, int resnumber);
|
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-02-20 15:55:59 +00:00
|
|
|
};
|
2009-02-15 06:10:59 +00:00
|
|
|
|
2009-04-25 08:50:42 +00:00
|
|
|
class ResourceSync : public Resource {
|
|
|
|
public:
|
|
|
|
ResourceSync() {}
|
|
|
|
~ResourceSync() {}
|
|
|
|
|
|
|
|
void startSync(Object *obj);
|
|
|
|
void nextSync(Object *obj);
|
|
|
|
void stopSync();
|
|
|
|
|
|
|
|
protected:
|
|
|
|
uint16 *_ptr;
|
|
|
|
uint16 _syncTime, _syncCue;
|
|
|
|
//bool _syncStarted; // not used
|
|
|
|
};
|
|
|
|
|
2009-02-21 10:23:36 +00:00
|
|
|
} // End of namespace Sci
|
2009-02-15 06:10:59 +00:00
|
|
|
|
2009-02-27 02:23:00 +00:00
|
|
|
#endif // SCI_SCICORE_RESOURCE_H
|