scummvm/engines/lilliput/lilliput.cpp

489 lines
14 KiB
C++
Raw Normal View History

2012-03-30 18:03:59 +02: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 "common/system.h"
#include "common/random.h"
#include "common/error.h"
#include "common/events.h"
#include "common/debug-channels.h"
#include "common/config-manager.h"
#include "common/textconsole.h"
#include "common/memstream.h"
2012-03-30 18:03:59 +02:00
#include "lilliput/lilliput.h"
#include "engines/util.h"
#include "lilliput/script.h"
2012-03-30 18:03:59 +02:00
namespace Lilliput {
LilliputEngine *LilliputEngine::s_Engine = 0;
static const byte _basisPalette[768] = {
0, 0, 0, 0, 0, 42, 0, 42, 0, 0, 42, 42,
42, 0, 0, 42, 0, 42, 42, 21, 0, 42, 42, 42,
21, 21, 21, 21, 21, 63, 21, 63, 21, 21, 63, 63,
63, 21, 21, 63, 21, 63, 63, 63, 21, 63, 63, 63,
63, 63, 63, 59, 59, 59, 54, 54, 54, 50, 50, 50,
46, 46, 46, 42, 42, 42, 38, 38, 38, 33, 33, 33,
29, 29, 29, 25, 25, 25, 21, 21, 21, 17, 17, 17,
13, 13, 13, 8, 8, 8, 4, 4, 4, 0, 0, 0,
63, 54, 54, 63, 46, 46, 63, 39, 39, 63, 31, 31,
63, 23, 23, 63, 16, 16, 63, 8, 8, 63, 0, 0,
57, 0, 0, 51, 0, 0, 45, 0, 0, 39, 0, 0,
33, 0, 0, 28, 0, 0, 22, 0, 0, 16, 0, 0,
63, 58, 54, 63, 54, 46, 63, 50, 39, 63, 46, 31,
63, 42, 23, 63, 38, 16, 63, 34, 8, 63, 30, 0,
57, 27, 0, 51, 24, 0, 45, 21, 0, 39, 19, 0,
33, 16, 0, 28, 14, 0, 22, 11, 0, 16, 8, 0,
63, 63, 54, 63, 63, 46, 63, 63, 39, 63, 63, 31,
63, 62, 23, 63, 61, 16, 63, 61, 8, 63, 61, 0,
57, 54, 0, 51, 49, 0, 45, 43, 0, 39, 39, 0,
33, 33, 0, 28, 27, 0, 22, 21, 0, 16, 16, 0,
62, 63, 54, 59, 61, 47, 56, 59, 42, 53, 58, 36,
50, 56, 32, 47, 54, 26, 44, 52, 22, 41, 50, 17,
36, 46, 14, 32, 42, 11, 28, 37, 8, 24, 33, 6,
20, 29, 4, 16, 25, 2, 13, 20, 1, 10, 16, 0,
54, 63, 54, 48, 61, 48, 43, 59, 43, 38, 58, 38,
33, 56, 33, 29, 54, 29, 25, 52, 24, 21, 50, 20,
16, 46, 16, 14, 42, 13, 10, 37, 9, 8, 33, 7,
6, 29, 4, 4, 25, 2, 2, 20, 1, 1, 16, 0,
59, 63, 63, 53, 63, 63, 47, 62, 63, 41, 61, 62,
35, 60, 62, 30, 59, 62, 24, 57, 62, 18, 55, 62,
20, 52, 56, 15, 47, 50, 11, 42, 45, 8, 37, 39,
5, 32, 33, 3, 27, 27, 1, 22, 22, 0, 16, 16,
54, 59, 63, 46, 56, 63, 39, 53, 63, 31, 50, 63,
23, 47, 63, 16, 44, 63, 8, 42, 63, 0, 39, 63,
0, 35, 57, 0, 31, 51, 0, 27, 45, 0, 23, 39,
0, 19, 33, 0, 16, 28, 0, 12, 22, 0, 9, 16,
54, 54, 63, 46, 47, 63, 39, 39, 63, 31, 32, 63,
23, 24, 63, 16, 16, 63, 8, 9, 63, 0, 1, 63,
0, 1, 57, 0, 1, 51, 0, 0, 45, 0, 0, 39,
0, 0, 33, 0, 0, 28, 0, 0, 22, 0, 0, 16,
54, 63, 54, 47, 63, 46, 39, 63, 39, 32, 63, 31,
24, 63, 23, 16, 63, 16, 8, 63, 8, 0, 63, 0,
0, 56, 0, 0, 49, 0, 0, 43, 0, 0, 36, 0,
0, 30, 0, 0, 23, 0, 0, 16, 0, 0, 10, 0,
63, 54, 63, 63, 46, 63, 63, 39, 63, 63, 31, 63,
63, 23, 63, 63, 16, 63, 63, 8, 63, 63, 0, 63,
56, 0, 57, 50, 0, 51, 45, 0, 45, 39, 0, 39,
33, 0, 33, 27, 0, 28, 22, 0, 22, 16, 0, 16,
63, 58, 55, 63, 56, 52, 63, 54, 49, 63, 53, 47,
63, 51, 44, 63, 49, 41, 63, 47, 39, 63, 46, 36,
63, 44, 32, 63, 41, 28, 63, 39, 24, 60, 37, 23,
58, 35, 22, 55, 34, 21, 52, 32, 20, 50, 31, 19,
47, 30, 18, 45, 28, 17, 42, 26, 16, 40, 25, 15,
39, 24, 14, 36, 23, 13, 34, 22, 12, 32, 20, 11,
29, 19, 10, 27, 18, 9, 23, 16, 8, 21, 15, 7,
18, 14, 6, 16, 12, 6, 14, 11, 5, 10, 8, 3,
63, 63, 63, 63, 63, 63, 63, 63, 63, 63, 63, 63,
63, 63, 63, 63, 63, 63, 63, 63, 63, 63, 63, 63,
63, 63, 63, 63, 63, 63, 63, 63, 63, 63, 63, 63,
63, 63, 63, 63, 63, 63, 63, 63, 63, 63, 63, 63,
63, 63, 63, 63, 63, 63, 63, 63, 63, 63, 63, 63,
63, 63, 63, 63, 63, 63, 63, 63, 63, 63, 63, 63,
63, 63, 63, 63, 63, 63, 63, 63, 63, 63, 63, 63,
63, 63, 63, 63, 63, 63, 63, 63, 63, 63, 63, 63
};
2012-03-30 18:03:59 +02:00
LilliputEngine::LilliputEngine(OSystem *syst, const LilliputGameDescription *gd) : Engine(syst), _gameDescription(gd)
{
_system = syst;
DebugMan.addDebugChannel(kDebugEngine, "Engine", "Engine debug level");
DebugMan.addDebugChannel(kDebugScript, "Script", "Script debug level");
2012-03-30 18:03:59 +02:00
_console = new LilliputConsole(this);
_rnd = 0;
2012-04-10 00:57:59 +02:00
_mouseX = 0;
_mouseY = 0;
_mouseButton = 0;
_scriptHandler = new LilliputScript(this);
_byte1714E = 0;
_rulesBuffer2PrevIndx = 0;
_word16EFA = 0;
2012-04-08 12:54:56 +02:00
_saveFlag = false;
for (int i = 0; i < 40; i++) {
_byte10999[i] = 0;
_byte109C1[i] = 0;
}
2012-03-30 18:03:59 +02:00
}
LilliputEngine::~LilliputEngine() {
DebugMan.clearAllDebugChannels();
delete _console;
delete _rnd;
}
GUI::Debugger *LilliputEngine::getDebugger() {
return _console;
}
bool LilliputEngine::hasFeature(EngineFeature f) const {
return (f == kSupportsRTL) || (f == kSupportsLoadingDuringRuntime) || (f == kSupportsSavingDuringRuntime);
}
const char *LilliputEngine::getCopyrightString() const {
return "copyright S.L.Grand, Brainware, 1991";
2012-03-30 18:03:59 +02:00
}
GameType LilliputEngine::getGameType() const {
return _gameType;
}
Common::Platform LilliputEngine::getPlatform() const {
return _platform;
}
2012-04-10 00:57:59 +02:00
void LilliputEngine::getMouseEvent() {
Common::EventManager *_event = _system->getEventManager();
Common::Event event;
while (_event->pollEvent(event) && !_shouldQuit)
;
_mouseX = _event->getMousePos().x;
_mouseY = _event->getMousePos().y;
_mouseButton = _event->getButtonState();
}
byte *LilliputEngine::loadVGA(Common::String filename, bool loadPal) {
Common::File f;
if (!f.open(filename))
error("Missing game file %s", filename.c_str());
int remainingSize = f.size();
if (loadPal) {
for (int i = 0; i < 768; ++i)
_curPalette[i] = f.readByte();
remainingSize -= 768;
fixPaletteEntries(_curPalette, 256);
_system->getPaletteManager()->setPalette(_curPalette, 0, 256);
}
uint8 curByte;
byte decodeBuffer[100000];
int size = 0;
for (;remainingSize > 0;) {
curByte = f.readByte();
--remainingSize;
if (curByte == 0xFF)
break;
if (curByte & 0x80) {
// Compressed
int compSize = (curByte & 0x7F);
curByte = f.readByte();
--remainingSize;
for (int i = 0; i < compSize; ++i) {
decodeBuffer[size] = curByte;
++size;
}
} else {
// Not compressed
int rawSize = (curByte & 0xF);
for (int i = 0; i < rawSize; ++i) {
decodeBuffer[size] = f.readByte();
--remainingSize;
++size;
}
}
}
f.close();
byte *res = (byte *)malloc(sizeof(byte) * size);
memcpy(res, decodeBuffer, size);
return res;
}
byte *LilliputEngine::loadRaw(Common::String filename) {
Common::File f;
if (!f.open(filename))
error("Missing game file %s", filename.c_str());
int size = f.size();
byte *res = (byte *)malloc(sizeof(byte) * size);
for (int i = 0; i < size; ++i)
res[i] = f.readByte();
f.close();
return res;
}
void LilliputEngine::loadRules() {
static const byte _rulesXlatArray[26] = {30, 48, 46, 32, 18, 33, 34, 35, 23, 36, 37, 38, 50, 49, 24, 25, 16, 19, 31, 20, 22, 47, 17, 45, 21, 44};
Common::File f;
uint16 curWord;
if (!f.open("ERULES.PRG"))
error("Missing game file ERULES.PRG");
_word10800_ERULES = f.readUint16LE();
// Chunk 1
int size = f.readUint16LE();
_rulesChunk1 = (byte *)malloc(sizeof(byte) * size);
for (int i = 0; i < size; ++i)
_rulesChunk1[i] = f.readByte();
// Chunk 2
_word10807_ERULES = f.readSint16LE();
assert(_word10807_ERULES <= 40);
for (int i = _word10807_ERULES, j = 0; i != 0; i--, j++) {
curWord = f.readUint16LE();
if (curWord != 0xFFFF)
curWord = (curWord << 3) + 4;
_rulesBuffer2_1[j] = curWord;
curWord = f.readUint16LE();
if (curWord != 0xFFFF)
curWord = (curWord << 3) + 4;
_rulesBuffer2_2[j] = curWord;
_rulesBuffer2_3[j] = (f.readUint16LE() & 0xFF);
_rulesBuffer2_4[j] = f.readUint16LE();
_rulesBuffer2_5[j] = f.readByte();
_rulesBuffer2_6[j] = f.readByte();
_rulesBuffer2_7[j] = f.readByte();
_rulesBuffer2_8[j] = f.readByte();
_rulesBuffer2_9[j] = f.readByte();
_rulesBuffer2_10[j] = f.readByte();
_rulesBuffer2_11[j] = f.readByte();
_rulesBuffer2_12[j] = f.readByte();
_rulesBuffer2_13[j] = f.readByte();
_rulesBuffer2_14[j] = f.readByte();
for (int k = 0; k < 32; k++)
_rulesBuffer2_15[(j * 32) + k] = f.readByte();
for (int k = 0; k < 32; k++)
_rulesBuffer2_16[(j * 32) + k] = f.readByte();
}
// Chunk 3 & 4
_rulesChunk3_size = f.readSint16LE();
curWord = f.readSint16LE();
_rulesChunk3 = (int *)malloc(sizeof(int) * _rulesChunk3_size);
for (int i = 0; i < _rulesChunk3_size; ++i)
_rulesChunk3[i] = f.readUint16LE();
_rulesChunk4 = (byte *)malloc(sizeof(byte) * curWord);
for (int i = 0; i < curWord; ++i)
_rulesChunk4[i] = f.readByte();
// Chunk 5: Scripts
// Use byte instead of int, therefore multiply by two the size.
// This is for changing that into a memory read stream
_rulesScript_size = f.readUint16LE() * 2;
_rulesScript = (byte *)malloc(sizeof(byte) * _rulesScript_size);
for (int i = 0; i < _rulesScript_size; ++i)
_rulesScript[i] = f.readByte();
// Chunk 6
_rulesChunk6_size = f.readUint16LE();
_rulesChunk6 = (int *)malloc(sizeof(int) * _rulesChunk6_size);
for (int i = 0; i < _rulesChunk6_size; ++i)
_rulesChunk6[i] = f.readUint16LE();
// Chunk 7 & 8
_rulesChunk7_size = f.readUint16LE();
_rulesChunk7 = (int *)malloc(sizeof(int) * _rulesChunk7_size);
for (int i = 0; i < _rulesChunk7_size; ++i)
_rulesChunk7[i] = f.readUint16LE();
curWord = f.readUint16LE();
_rulesChunk8 = (byte *)malloc(sizeof(byte) * curWord);
for (int i = 0; i < curWord; ++i)
_rulesChunk8[i] = f.readByte();
// Chunk 9
for (int i = 0; i < 60; i++)
_rulesChunk9[i] = f.readByte();
// Chunk 10 & 11
_rulesChunk10_size = f.readByte();
assert(_rulesChunk10_size <= 20);
if (_rulesChunk10_size != 0) {
_rulesChunk10 = (int *)malloc(sizeof(int) * _rulesChunk10_size);
int totalSize = 0;
for (int i = 0; i < _rulesChunk10_size; ++i) {
_rulesChunk10[i] = totalSize;
totalSize += f.readByte();
}
if (totalSize != 0) {
_rulesChunk11 = (byte *)malloc(sizeof(byte) * totalSize);
for (int i = 0; i < totalSize; i++)
_rulesChunk11[i] = f.readByte();
}
}
// Chunk 12
_rulesChunk12_size = f.readUint16LE();
assert(_rulesChunk12_size <= 40);
for (int i = 0; i < _rulesChunk12_size; i++) {
_rulesBuffer12_1[i] = f.readUint16LE();
_rulesBuffer12_2[i] = f.readUint16LE();
_rulesBuffer12_3[i] = f.readUint16LE();
_rulesBuffer12_4[i] = f.readUint16LE();
}
// Chunk 13
_word12F68_ERULES = f.readUint16LE();
for (int i = 0 ; i < 20; i++)
_rulesBuffer13_1[i] = f.readByte();
for (int i = 0 ; i < 20; i++)
_rulesBuffer13_2[i] = f.readUint16LE();
for (int i = 0 ; i < 20; i++)
_rulesBuffer13_3[i] = f.readUint16LE();
for (int i = 0; i < 20; i++) {
byte curByte = f.readByte();
if (curByte == 0x20)
_rulesBuffer13_4[i] = 0x39;
else if (curByte == 0xD)
_rulesBuffer13_4[i] = 0x1C;
// Hack to avoid xlat out of bounds
else if (curByte == 0xFF)
_rulesBuffer13_4[i] = 0x21;
// Hack to avoid xlat out of bounds
else if (curByte == 0x00)
_rulesBuffer13_4[i] = 0xB4;
else {
assert((curByte > 0x40) && (curByte <= 0x41 + 26));
_rulesBuffer13_4[i] = _rulesXlatArray[curByte - 0x41];
}
}
f.close();
// Skipped: Load Savegame
}
void LilliputEngine::displayVGAFile(Common::String fileName) {
warning("TODO: display function #4");
byte *buffer = loadVGA(fileName, true);
memcpy(_mainSurface->getPixels(), buffer, 320*200);
_system->copyRectToScreen((byte *)_mainSurface->getPixels(), 320, 0, 0, 320, 200);
_system->updateScreen();
warning("TODO: display function #5");
}
void LilliputEngine::fixPaletteEntries(uint8 *palette, int num) {
// Color values are coded on 6bits ( for old 6bits DAC )
for (int32 i = 0; i < num * 3; i++) {
int32 a = palette[i];
assert(a < 64);
a = (a << 2) | (a >> 4);
if (a > 255)
a = 255;
palette[i] = a;
}
}
void LilliputEngine::initPalette() {
for (int i = 0; i < 768; i++)
_curPalette[i] = _basisPalette[i];
fixPaletteEntries(_curPalette, 256);
_system->getPaletteManager()->setPalette(_curPalette, 0, 256);
}
2012-03-30 18:03:59 +02:00
Common::Error LilliputEngine::run() {
s_Engine = this;
initGraphics(320, 200);
_mainSurface = new Graphics::Surface();
_mainSurface->create(320, 200, Graphics::PixelFormat::createFormatCLUT8());
2012-03-30 18:03:59 +02:00
// Setup mixer
syncSoundSettings();
initPalette();
// Load files. In the original, the size was hardcoded
_bufferIdeogram = loadVGA("IDEOGRAM.VGA", false);
_bufferMen = loadVGA("MEN.VGA", false);
_bufferMen2 = loadVGA("MEN2.VGA", false);
_bufferIsoChars = loadVGA("ISOCHARS.VGA", false);
_bufferIsoMap = loadRaw("ISOMAP.DTA");
//TODO: Init mouse handler
loadRules();
//TODO: Init sound/music player
_scriptHandler->runScript(Common::MemoryReadStream(_rulesScript, _rulesScript_size));
//TODO: Main loop
2012-03-30 18:03:59 +02:00
return Common::kNoError;
}
void LilliputEngine::initialize() {
debugC(1, kDebugEngine, "initialize");
_rnd = new Common::RandomSource("robin");
_rnd->setSeed(42); // Kick random number generator
2012-04-10 00:57:59 +02:00
_shouldQuit = false;
for (int i = 0; i < 4; i++) {
_arr18560[i]._field0 = 0;
_arr18560[i]._field1 = 0;
_arr18560[i]._field3 = 0;
for (int j = 0; j < 8; j ++)
_arr18560[i]._field5[j] = 0;
}
2012-03-30 18:03:59 +02:00
}
void LilliputEngine::syncSoundSettings() {
Engine::syncSoundSettings();
// _sound->syncVolume();
}
Common::String LilliputEngine::getSavegameFilename(int slot) {
return _targetName + Common::String::format("-%02d.SAV", slot);
}
byte LilliputEngine::_keyboard_getch() {
return ' ';
}
2012-03-30 18:03:59 +02:00
} // End of namespace Lilliput