2013-06-07 00:49:50 +03: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.
|
|
|
|
*
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include "fullpipe/fullpipe.h"
|
|
|
|
|
|
|
|
#include "common/file.h"
|
2013-06-21 21:30:38 -04:00
|
|
|
#include "common/memstream.h"
|
2013-06-07 00:49:50 +03:00
|
|
|
|
|
|
|
#include "fullpipe/objects.h"
|
2013-06-16 15:11:18 +03:00
|
|
|
#include "fullpipe/motion.h"
|
2013-06-21 21:30:38 -04:00
|
|
|
#include "fullpipe/ngiarchive.h"
|
2013-06-07 00:49:50 +03:00
|
|
|
|
|
|
|
namespace Fullpipe {
|
|
|
|
|
2013-06-20 14:44:59 -04:00
|
|
|
bool CObject::loadFile(const char *fname) {
|
2013-06-20 17:21:37 -04:00
|
|
|
Common::File file;
|
2013-06-20 14:44:59 -04:00
|
|
|
|
|
|
|
if (!file.open(fname))
|
|
|
|
return false;
|
|
|
|
|
2013-06-20 17:21:37 -04:00
|
|
|
MfcArchive archive(&file);
|
|
|
|
|
|
|
|
return load(archive);
|
2013-06-20 14:44:59 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
bool CObList::load(MfcArchive &file) {
|
|
|
|
int count = file.readCount();
|
|
|
|
|
|
|
|
debug(9, "CObList::count: %d:", count);
|
|
|
|
|
|
|
|
for (int i = 0; i < count; i++) {
|
|
|
|
debug(9, "CObList::[%d]", i);
|
|
|
|
CObject *t = file.readClass();
|
|
|
|
|
|
|
|
push_back(*t);
|
|
|
|
}
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool CObArray::load(MfcArchive &file) {
|
|
|
|
int count = file.readCount();
|
|
|
|
|
|
|
|
resize(count);
|
|
|
|
|
|
|
|
for (int i = 0; i < count; i++) {
|
|
|
|
CObject *t = file.readClass();
|
|
|
|
|
|
|
|
push_back(*t);
|
|
|
|
}
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool CDWordArray::load(MfcArchive &file) {
|
|
|
|
int count = file.readCount();
|
|
|
|
|
|
|
|
debug(9, "CDWordArray::count: %d", count);
|
|
|
|
|
|
|
|
resize(count);
|
|
|
|
|
|
|
|
for (int i = 0; i < count; i++) {
|
|
|
|
int32 t = file.readUint32LE();
|
|
|
|
|
|
|
|
push_back(t);
|
|
|
|
}
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2013-06-08 17:05:51 +03:00
|
|
|
char *MfcArchive::readPascalString(bool twoByte) {
|
2013-06-07 00:49:50 +03:00
|
|
|
char *tmp;
|
2013-06-08 17:05:51 +03:00
|
|
|
int len;
|
|
|
|
|
|
|
|
if (twoByte)
|
|
|
|
len = readUint16LE();
|
|
|
|
else
|
|
|
|
len = readByte();
|
|
|
|
|
2013-06-07 00:49:50 +03:00
|
|
|
tmp = (char *)calloc(len + 1, 1);
|
|
|
|
read(tmp, len);
|
|
|
|
|
2013-06-11 01:17:11 +03:00
|
|
|
debug(9, "readPascalString: %d <%s>", len, tmp);
|
2013-06-09 23:51:33 +03:00
|
|
|
|
2013-06-07 00:49:50 +03:00
|
|
|
return tmp;
|
|
|
|
}
|
|
|
|
|
2013-06-21 21:30:38 -04:00
|
|
|
MemoryObject::MemoryObject() {
|
|
|
|
_filename = 0;
|
|
|
|
_field_8 = 0;
|
|
|
|
_field_C = 0;
|
|
|
|
_field_10 = -1;
|
|
|
|
_field_14 = 1;
|
|
|
|
_dataSize = 0;
|
|
|
|
_flags = 0;
|
|
|
|
_libHandle = 0;
|
|
|
|
_data = 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool MemoryObject::load(MfcArchive &file) {
|
|
|
|
_filename = file.readPascalString();
|
|
|
|
|
|
|
|
if (g_fullpipe->_currArchive) {
|
|
|
|
_field_14 = 0;
|
|
|
|
_libHandle = g_fullpipe->_currArchive;
|
|
|
|
}
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
void MemoryObject::loadFile(char *filename) {
|
|
|
|
if (!_data) {
|
|
|
|
if (g_fullpipe->_currArchive != _libHandle) {
|
|
|
|
assert(0);
|
|
|
|
}
|
|
|
|
|
|
|
|
Common::SeekableReadStream *s = _libHandle->createReadStreamForMember(filename);
|
|
|
|
|
|
|
|
if (s) {
|
|
|
|
debug(0, "Reading %s", filename);
|
|
|
|
assert(s->size() > 0);
|
|
|
|
_data = calloc(s->size(), 1);
|
|
|
|
s->read(_data, s->size());
|
|
|
|
|
|
|
|
delete s;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
MemoryObject2::MemoryObject2() {
|
|
|
|
_data2 = 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool MemoryObject2::load(MfcArchive &file) {
|
|
|
|
MemoryObject::load(file);
|
|
|
|
|
|
|
|
_flags |= 1;
|
|
|
|
|
|
|
|
if (_filename) {
|
|
|
|
MemoryObject::loadFile(_filename);
|
|
|
|
}
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2013-06-07 00:49:50 +03:00
|
|
|
int MfcArchive::readCount() {
|
|
|
|
int count = readUint16LE();
|
|
|
|
|
|
|
|
if (count == 0xffff)
|
|
|
|
count = readUint32LE();
|
|
|
|
|
|
|
|
return count;
|
|
|
|
}
|
|
|
|
|
2013-06-16 16:10:46 +03:00
|
|
|
double MfcArchive::readDouble() {
|
|
|
|
// FIXME: This is utterly cruel and unportable
|
|
|
|
|
|
|
|
union {
|
|
|
|
struct {
|
|
|
|
int32 a;
|
|
|
|
int32 b;
|
|
|
|
} i;
|
|
|
|
double d;
|
|
|
|
} tmp;
|
|
|
|
|
|
|
|
tmp.i.a = readUint32LE();
|
|
|
|
tmp.i.b = readUint32LE();
|
|
|
|
|
|
|
|
return tmp.d;
|
|
|
|
}
|
|
|
|
|
2013-06-07 23:56:40 +03:00
|
|
|
enum {
|
2013-06-13 00:57:54 +03:00
|
|
|
kNullObject,
|
|
|
|
kCInteraction,
|
|
|
|
kMessageQueue,
|
|
|
|
kExCommand,
|
|
|
|
kCObjstateCommand,
|
2013-06-15 01:19:45 +03:00
|
|
|
kCGameVar,
|
2013-06-16 15:07:33 +03:00
|
|
|
kCMctlCompound,
|
2013-06-16 16:10:46 +03:00
|
|
|
kCMovGraph,
|
|
|
|
kCMovGraphLink,
|
2013-06-18 10:04:29 -04:00
|
|
|
kCMovGraphNode,
|
2013-06-18 10:23:49 -04:00
|
|
|
kCReactParallel,
|
|
|
|
kCReactPolygonal
|
2013-06-07 23:56:40 +03:00
|
|
|
};
|
|
|
|
|
|
|
|
const struct {
|
|
|
|
const char *name;
|
|
|
|
int id;
|
|
|
|
} classMap[] = {
|
2013-06-09 00:27:42 +03:00
|
|
|
{ "CInteraction", kCInteraction },
|
|
|
|
{ "MessageQueue", kMessageQueue },
|
|
|
|
{ "ExCommand", kExCommand },
|
2013-06-09 12:51:23 +03:00
|
|
|
{ "CObjstateCommand", kCObjstateCommand },
|
2013-06-10 01:03:15 +03:00
|
|
|
{ "CGameVar", kCGameVar },
|
2013-06-15 01:19:45 +03:00
|
|
|
{ "CMctlCompound", kCMctlCompound },
|
2013-06-16 15:07:33 +03:00
|
|
|
{ "CMovGraph", kCMovGraph },
|
2013-06-16 16:10:46 +03:00
|
|
|
{ "CMovGraphLink", kCMovGraphLink },
|
|
|
|
{ "CMovGraphNode", kCMovGraphNode },
|
2013-06-18 10:04:29 -04:00
|
|
|
{ "CReactParallel", kCReactParallel },
|
2013-06-18 10:23:49 -04:00
|
|
|
{ "CReactPolygonal", kCReactPolygonal },
|
2013-06-07 23:56:40 +03:00
|
|
|
{ 0, 0 }
|
|
|
|
};
|
|
|
|
|
2013-06-09 13:33:22 +03:00
|
|
|
static const char *lookupObjectId(int id) {
|
|
|
|
for (int i = 0; classMap[i].name; i++) {
|
|
|
|
if (classMap[i].id == id)
|
|
|
|
return classMap[i].name;
|
|
|
|
}
|
|
|
|
|
|
|
|
return "";
|
|
|
|
}
|
|
|
|
|
2013-06-11 01:17:11 +03:00
|
|
|
static CObject *createObject(int objectId) {
|
|
|
|
switch (objectId) {
|
|
|
|
case kNullObject:
|
|
|
|
return 0;
|
|
|
|
case kCInteraction:
|
|
|
|
return new CInteraction();
|
|
|
|
case kMessageQueue:
|
|
|
|
return new MessageQueue();
|
|
|
|
case kExCommand:
|
|
|
|
return new ExCommand();
|
|
|
|
case kCObjstateCommand:
|
|
|
|
return new CObjstateCommand();
|
|
|
|
case kCGameVar:
|
|
|
|
return new CGameVar();
|
2013-06-15 01:19:45 +03:00
|
|
|
case kCMctlCompound:
|
|
|
|
return new CMctlCompound();
|
2013-06-16 15:07:33 +03:00
|
|
|
case kCMovGraph:
|
|
|
|
return new CMovGraph();
|
2013-06-16 16:10:46 +03:00
|
|
|
case kCMovGraphLink:
|
|
|
|
return new CMovGraphLink();
|
|
|
|
case kCMovGraphNode:
|
|
|
|
return new CMovGraphNode();
|
2013-06-18 10:04:29 -04:00
|
|
|
case kCReactParallel:
|
|
|
|
return new CReactParallel();
|
2013-06-18 10:23:49 -04:00
|
|
|
case kCReactPolygonal:
|
|
|
|
return new CReactPolygonal();
|
2013-06-11 01:17:11 +03:00
|
|
|
default:
|
|
|
|
error("Unknown objectId: %d", objectId);
|
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
2013-06-09 13:33:22 +03:00
|
|
|
|
2013-06-20 17:21:37 -04:00
|
|
|
MfcArchive::MfcArchive(Common::SeekableReadStream *stream) {
|
2013-06-09 13:33:22 +03:00
|
|
|
for (int i = 0; classMap[i].name; i++) {
|
2013-06-07 23:56:40 +03:00
|
|
|
_classMap[classMap[i].name] = classMap[i].id;
|
|
|
|
}
|
|
|
|
|
|
|
|
_lastIndex = 1;
|
2013-06-11 01:34:37 +03:00
|
|
|
_level = 0;
|
2013-06-09 12:26:38 +03:00
|
|
|
|
2013-06-20 17:21:37 -04:00
|
|
|
_stream = stream;
|
|
|
|
|
2013-06-11 01:17:11 +03:00
|
|
|
_objectMap.push_back(0);
|
|
|
|
_objectIdMap.push_back(kNullObject);
|
2013-06-07 23:56:40 +03:00
|
|
|
}
|
|
|
|
|
2013-06-10 01:03:15 +03:00
|
|
|
CObject *MfcArchive::readClass() {
|
2013-06-11 01:17:11 +03:00
|
|
|
bool isCopyReturned;
|
|
|
|
CObject *res = parseClass(&isCopyReturned);
|
2013-06-10 01:03:15 +03:00
|
|
|
|
2013-06-11 01:17:11 +03:00
|
|
|
if (res && !isCopyReturned)
|
2013-06-10 01:03:15 +03:00
|
|
|
res->load(*this);
|
|
|
|
|
|
|
|
return res;
|
|
|
|
}
|
|
|
|
|
2013-06-11 01:17:11 +03:00
|
|
|
CObject *MfcArchive::parseClass(bool *isCopyReturned) {
|
2013-06-07 23:56:40 +03:00
|
|
|
char *name;
|
2013-06-11 01:17:11 +03:00
|
|
|
int objectId = 0;
|
|
|
|
CObject *res = 0;
|
2013-06-07 23:56:40 +03:00
|
|
|
|
|
|
|
uint obTag = readUint16LE();
|
|
|
|
|
2013-06-11 01:17:11 +03:00
|
|
|
debug(7, "parseClass::obTag = %d (%04x) at 0x%08x", obTag, obTag, pos() - 2);
|
2013-06-08 17:05:51 +03:00
|
|
|
|
2013-06-07 23:56:40 +03:00
|
|
|
if (obTag == 0xffff) {
|
|
|
|
int schema = readUint16LE();
|
|
|
|
|
2013-06-11 01:17:11 +03:00
|
|
|
debug(7, "parseClass::schema = %d", schema);
|
2013-06-08 17:05:51 +03:00
|
|
|
|
|
|
|
name = readPascalString(true);
|
2013-06-11 01:17:11 +03:00
|
|
|
debug(7, "parseClass::class <%s>", name);
|
2013-06-07 23:56:40 +03:00
|
|
|
|
|
|
|
if (!_classMap.contains(name)) {
|
2013-06-08 17:05:51 +03:00
|
|
|
error("Unknown class in MfcArchive: <%s>", name);
|
2013-06-07 23:56:40 +03:00
|
|
|
}
|
|
|
|
|
2013-06-08 17:05:51 +03:00
|
|
|
objectId = _classMap[name];
|
2013-06-09 00:27:42 +03:00
|
|
|
|
2013-06-11 01:17:11 +03:00
|
|
|
debug(7, "tag: %d 0x%x (%x)", _objectMap.size() - 1, _objectMap.size() - 1, objectId);
|
|
|
|
|
|
|
|
res = createObject(objectId);
|
|
|
|
_objectMap.push_back(res);
|
|
|
|
_objectIdMap.push_back(objectId);
|
|
|
|
|
|
|
|
_objectMap.push_back(res); // Basically a hack, but behavior is all correct
|
|
|
|
_objectIdMap.push_back(objectId);
|
|
|
|
|
|
|
|
*isCopyReturned = false;
|
2013-06-09 23:51:33 +03:00
|
|
|
} else if ((obTag & 0x8000) == 0) {
|
2013-06-11 01:17:11 +03:00
|
|
|
if (_objectMap.size() < obTag) {
|
|
|
|
error("Object index too big: %d at 0x%08x", obTag, pos() - 2);
|
|
|
|
}
|
|
|
|
res = _objectMap[obTag];
|
|
|
|
|
|
|
|
*isCopyReturned = true;
|
2013-06-07 23:56:40 +03:00
|
|
|
} else {
|
2013-06-09 13:22:10 +03:00
|
|
|
|
2013-06-07 23:56:40 +03:00
|
|
|
obTag &= ~0x8000;
|
|
|
|
|
|
|
|
if (_objectMap.size() < obTag) {
|
2013-06-09 00:27:42 +03:00
|
|
|
error("Object index too big: %d at 0x%08x", obTag, pos() - 2);
|
2013-06-07 23:56:40 +03:00
|
|
|
}
|
|
|
|
|
2013-06-11 01:17:11 +03:00
|
|
|
debug(7, "parseClass::obTag <%s>", lookupObjectId(_objectIdMap[obTag]));
|
2013-06-09 13:33:22 +03:00
|
|
|
|
2013-06-11 01:17:11 +03:00
|
|
|
objectId = _objectIdMap[obTag];
|
2013-06-09 13:22:10 +03:00
|
|
|
|
2013-06-11 01:17:11 +03:00
|
|
|
res = createObject(objectId);
|
|
|
|
_objectMap.push_back(res);
|
|
|
|
_objectIdMap.push_back(objectId);
|
2013-06-07 00:49:50 +03:00
|
|
|
|
2013-06-11 01:17:11 +03:00
|
|
|
*isCopyReturned = false;
|
2013-06-07 23:56:40 +03:00
|
|
|
}
|
2013-06-07 00:49:50 +03:00
|
|
|
|
2013-06-11 01:17:11 +03:00
|
|
|
return res;
|
2013-06-07 00:49:50 +03:00
|
|
|
}
|
|
|
|
|
2013-06-20 16:39:05 -04:00
|
|
|
char *genFileName(int superId, int sceneId, const char *ext) {
|
|
|
|
char *s = (char *)calloc(256, 1);
|
|
|
|
|
|
|
|
if (superId) {
|
|
|
|
snprintf(s, 255, "%04d%04d.%s", superId, sceneId, ext);
|
|
|
|
} else {
|
|
|
|
snprintf(s, 255, "%04d.%s", sceneId, ext);
|
|
|
|
}
|
|
|
|
|
|
|
|
return s;
|
|
|
|
}
|
|
|
|
|
2013-06-07 00:49:50 +03:00
|
|
|
} // End of namespace Fullpipe
|