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"
|
|
|
|
|
|
|
|
#include "fullpipe/utils.h"
|
|
|
|
#include "fullpipe/objects.h"
|
|
|
|
|
|
|
|
namespace Fullpipe {
|
|
|
|
|
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);
|
|
|
|
|
|
|
|
return tmp;
|
|
|
|
}
|
|
|
|
|
|
|
|
int MfcArchive::readCount() {
|
|
|
|
int count = readUint16LE();
|
|
|
|
|
|
|
|
if (count == 0xffff)
|
|
|
|
count = readUint32LE();
|
|
|
|
|
|
|
|
return count;
|
|
|
|
}
|
|
|
|
|
2013-06-07 23:56:40 +03:00
|
|
|
enum {
|
2013-06-09 00:27:42 +03:00
|
|
|
kCInteraction = 0,
|
|
|
|
kMessageQueue = 1,
|
2013-06-09 12:51:23 +03:00
|
|
|
kExCommand = 2,
|
|
|
|
kCObjstateCommand = 3
|
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-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-07 23:56:40 +03:00
|
|
|
MfcArchive::MfcArchive() {
|
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-09 12:26:38 +03:00
|
|
|
|
|
|
|
_objectMap.push_back(0);
|
2013-06-07 23:56:40 +03:00
|
|
|
}
|
|
|
|
|
2013-06-07 00:49:50 +03:00
|
|
|
CObject *MfcArchive::parseClass() {
|
2013-06-07 23:56:40 +03:00
|
|
|
char *name;
|
|
|
|
int objectId;
|
|
|
|
|
|
|
|
uint obTag = readUint16LE();
|
|
|
|
|
2013-06-09 13:33:22 +03:00
|
|
|
debug(0, "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-08 17:05:51 +03:00
|
|
|
debug(0, "parseClass::schema = %d", schema);
|
|
|
|
|
|
|
|
name = readPascalString(true);
|
2013-06-09 00:27:42 +03:00
|
|
|
debug(0, "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];
|
|
|
|
_objectMap.push_back(objectId);
|
2013-06-09 13:22:10 +03:00
|
|
|
debug(0, "tag: %d (%x)", _objectMap.size() - 1, objectId);
|
2013-06-09 00:27:42 +03:00
|
|
|
|
2013-06-09 13:22:10 +03:00
|
|
|
objectId = _classMap[name];
|
2013-06-07 23:56:40 +03:00
|
|
|
} else {
|
2013-06-09 13:22:10 +03:00
|
|
|
if ((obTag & 0x8000) == 0) {
|
|
|
|
error("Wrong object index format: %d at 0x%08x", obTag, pos() - 2);
|
|
|
|
}
|
|
|
|
|
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-09 13:33:22 +03:00
|
|
|
debug(0, "parseClass::obTag <%s>", lookupObjectId(_objectMap[obTag]));
|
|
|
|
|
2013-06-09 12:26:38 +03:00
|
|
|
objectId = _objectMap[obTag];
|
2013-06-07 23:56:40 +03:00
|
|
|
}
|
2013-06-08 17:05:51 +03:00
|
|
|
|
2013-06-09 13:22:10 +03:00
|
|
|
_objectMap.push_back(objectId);
|
|
|
|
|
2013-06-08 17:05:51 +03:00
|
|
|
debug(0, "objectId: %d", objectId);
|
2013-06-07 00:49:50 +03:00
|
|
|
|
2013-06-07 23:56:40 +03:00
|
|
|
switch (objectId) {
|
|
|
|
case kCInteraction:
|
|
|
|
return new CInteraction();
|
2013-06-09 00:27:42 +03:00
|
|
|
case kMessageQueue:
|
|
|
|
return new MessageQueue();
|
|
|
|
case kExCommand:
|
|
|
|
return new ExCommand();
|
2013-06-09 12:51:23 +03:00
|
|
|
case kCObjstateCommand:
|
|
|
|
return new CObjstateCommand();
|
2013-06-07 23:56:40 +03:00
|
|
|
default:
|
|
|
|
error("Unknown objectId: %d", objectId);
|
|
|
|
}
|
2013-06-07 00:49:50 +03:00
|
|
|
|
2013-06-07 23:56:40 +03:00
|
|
|
return 0;
|
2013-06-07 00:49:50 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
} // End of namespace Fullpipe
|