2008-04-20 14:47:37 +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$
|
|
|
|
*
|
|
|
|
*/
|
|
|
|
|
|
|
|
#ifndef M4_WOODSCRIPT_H
|
|
|
|
#define M4_WOODSCRIPT_H
|
|
|
|
|
|
|
|
#include "common/scummsys.h"
|
|
|
|
#include "common/util.h"
|
|
|
|
#include "common/str.h"
|
|
|
|
#include "common/array.h"
|
|
|
|
#include "common/stream.h"
|
|
|
|
#include "graphics/surface.h"
|
|
|
|
|
|
|
|
#include "m4/globals.h"
|
|
|
|
#include "m4/assets.h"
|
|
|
|
#include "m4/resource.h"
|
|
|
|
#include "m4/sprite.h"
|
|
|
|
#include "m4/m4.h"
|
|
|
|
#include "m4/graphics.h"
|
|
|
|
#include "m4/viewmgr.h"
|
|
|
|
|
|
|
|
namespace M4 {
|
|
|
|
|
2010-01-30 09:21:07 +00:00
|
|
|
class MadsM4Engine;
|
2008-04-20 14:47:37 +00:00
|
|
|
class WoodScript;
|
|
|
|
class Machine;
|
|
|
|
class Sequence;
|
|
|
|
class AssetManager;
|
|
|
|
class View;
|
|
|
|
|
|
|
|
struct Instruction {
|
|
|
|
int32 instr;
|
|
|
|
long *argp[3];
|
|
|
|
long argv[3];
|
|
|
|
int argc;
|
|
|
|
// Helper method; many opcode functions can get either a defined value or a random value
|
|
|
|
long getValue() {
|
|
|
|
if (argc == 3)
|
|
|
|
return _vm->imath_ranged_rand16(argv[1], argv[2]);
|
|
|
|
else
|
|
|
|
return argv[1];
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
class Bytecode {
|
|
|
|
public:
|
|
|
|
Bytecode(WoodScript *ws, byte *code, int32 codeSize, Sequence *seq);
|
|
|
|
~Bytecode();
|
|
|
|
int loadInstruction(Instruction &instruction);
|
|
|
|
void jumpAbsolute(int32 ofs);
|
|
|
|
void jumpRelative(int32 ofs);
|
|
|
|
void setSequence(Sequence *sequence);
|
|
|
|
void setCode(byte *code, int32 codeSize);
|
|
|
|
Sequence *sequence() const;
|
|
|
|
uint32 pos() const { return _code->pos() / 4; }
|
|
|
|
protected:
|
|
|
|
WoodScript *_ws;
|
|
|
|
Common::MemoryReadStream *_code;
|
|
|
|
Sequence *_sequence;
|
|
|
|
static int32 _dataFormats[];
|
|
|
|
bool decodeArgument(int32 format, int32 data, long *&arg, long &value);
|
|
|
|
};
|
|
|
|
|
|
|
|
struct EndOfSequenceRequestItem {
|
|
|
|
int32 codeOffset, count;
|
|
|
|
EndOfSequenceRequestItem() : codeOffset(-1) {}
|
|
|
|
bool isValid() const { return codeOffset >= 0; }
|
|
|
|
};
|
|
|
|
|
|
|
|
typedef Common::Array<EndOfSequenceRequestItem> EndOfSequenceRequestList;
|
|
|
|
|
|
|
|
class Sequence {
|
|
|
|
public:
|
|
|
|
Sequence(WoodScript *ws, Machine *machine, int32 sequenceHash);
|
|
|
|
~Sequence();
|
|
|
|
|
|
|
|
void pause();
|
|
|
|
void resume();
|
|
|
|
void issueEndOfSequenceRequest(int32 codeOffset, int32 count);
|
|
|
|
void cancelEndOfSequenceRequest();
|
|
|
|
|
|
|
|
bool runProgram();
|
|
|
|
|
|
|
|
bool changeProgram(int32 sequenceHash);
|
|
|
|
|
|
|
|
void clearVars();
|
|
|
|
|
|
|
|
long *getVarPtr(int index);
|
|
|
|
long *getParentVarPtr(int index);
|
|
|
|
long *getDataPtr(int index);
|
2009-01-01 15:06:43 +00:00
|
|
|
|
2008-04-20 14:47:37 +00:00
|
|
|
void setActive(bool active) { _active = active; }
|
|
|
|
bool isActive() const { return _active; }
|
|
|
|
|
|
|
|
bool isTerminated() const { return _terminated; }
|
|
|
|
|
|
|
|
void draw(M4Surface *surface, const Common::Rect &clipRect, Common::Rect &updateRect);
|
2009-01-01 15:06:43 +00:00
|
|
|
|
2008-04-20 14:47:37 +00:00
|
|
|
bool s1_end(Instruction &instruction);
|
|
|
|
bool s1_clearVars(Instruction &instruction);
|
|
|
|
bool s1_set(Instruction &instruction);
|
|
|
|
bool s1_compare(Instruction &instruction);
|
|
|
|
bool s1_add(Instruction &instruction);
|
|
|
|
bool s1_sub(Instruction &instruction);
|
|
|
|
bool s1_mul(Instruction &instruction);
|
|
|
|
bool s1_div(Instruction &instruction);
|
|
|
|
bool s1_and(Instruction &instruction);
|
|
|
|
bool s1_or(Instruction &instruction);
|
|
|
|
bool s1_not(Instruction &instruction);
|
|
|
|
bool s1_sin(Instruction &instruction);
|
|
|
|
bool s1_cos(Instruction &instruction);
|
|
|
|
bool s1_abs(Instruction &instruction);
|
|
|
|
bool s1_min(Instruction &instruction);
|
|
|
|
bool s1_max(Instruction &instruction);
|
|
|
|
bool s1_mod(Instruction &instruction);
|
|
|
|
bool s1_floor(Instruction &instruction);
|
|
|
|
bool s1_round(Instruction &instruction);
|
|
|
|
bool s1_ceil(Instruction &instruction);
|
|
|
|
bool s1_point(Instruction &instruction);
|
|
|
|
bool s1_dist2d(Instruction &instruction);
|
|
|
|
bool s1_crunch(Instruction &instruction);
|
|
|
|
bool s1_branch(Instruction &instruction);
|
|
|
|
bool s1_setFrame(Instruction &instruction);
|
|
|
|
bool s1_sendMessage(Instruction &instruction);
|
|
|
|
bool s1_push(Instruction &instruction);
|
|
|
|
bool s1_pop(Instruction &instruction);
|
|
|
|
bool s1_jumpSub(Instruction &instruction);
|
|
|
|
bool s1_return(Instruction &instruction);
|
|
|
|
bool s1_getFrameCount(Instruction &instruction);
|
|
|
|
bool s1_getFrameRate(Instruction &instruction);
|
|
|
|
bool s1_getCelsPixSpeed(Instruction &instruction);
|
|
|
|
bool s1_setIndex(Instruction &instruction);
|
|
|
|
bool s1_setLayer(Instruction &instruction);
|
|
|
|
bool s1_setDepth(Instruction &instruction);
|
|
|
|
bool s1_setData(Instruction &instruction);
|
|
|
|
bool s1_openStream(Instruction &instruction);
|
|
|
|
bool s1_streamNextFrame(Instruction &instruction);
|
|
|
|
bool s1_closeStream(Instruction &instruction);
|
|
|
|
|
|
|
|
int32 indexReg() const { return _indexReg; }
|
|
|
|
|
|
|
|
EndOfSequenceRequestItem getEndOfSequenceRequestItem() const { return _endOfSequenceRequest; }
|
|
|
|
bool hasEndOfSequenceRequestPending() const { return _endOfSequenceRequest.isValid(); }
|
|
|
|
void resetEndOfSequenceRequest() { _endOfSequenceRequest.codeOffset = -1; }
|
|
|
|
|
|
|
|
Machine *getMachine() const { return _machine; }
|
|
|
|
|
|
|
|
|
|
|
|
protected:
|
|
|
|
WoodScript *_ws;
|
|
|
|
Bytecode *_code;
|
|
|
|
|
|
|
|
long *_vars;
|
|
|
|
bool _active, _terminated;
|
|
|
|
Machine *_machine;
|
|
|
|
Sequence *_parentSequence;
|
|
|
|
int32 _layer;
|
|
|
|
int32 _startTime, _switchTime;
|
|
|
|
long *_dataRow;
|
|
|
|
int32 _localVarCount;
|
|
|
|
int32 _cmpFlags;
|
|
|
|
|
|
|
|
EndOfSequenceRequestItem _endOfSequenceRequest;
|
|
|
|
|
|
|
|
int32 _indexReg;
|
2009-01-01 15:06:43 +00:00
|
|
|
|
2008-04-20 14:47:37 +00:00
|
|
|
M4Sprite *_curFrame;
|
2009-01-01 15:06:43 +00:00
|
|
|
|
2008-04-20 14:47:37 +00:00
|
|
|
int32 _sequenceHash;
|
|
|
|
|
|
|
|
int32 _returnHashes[8]; //FIXME: Use constant instead of 8
|
|
|
|
uint32 _returnOffsets[8];
|
|
|
|
int32 _returnStackIndex;
|
|
|
|
|
|
|
|
Common::SeekableReadStream *_stream;
|
|
|
|
SpriteAsset *_streamSpriteAsset;
|
|
|
|
|
|
|
|
bool streamOpen();
|
|
|
|
bool streamNextFrame();
|
|
|
|
void streamClose();
|
|
|
|
|
|
|
|
};
|
|
|
|
|
|
|
|
class Machine {
|
|
|
|
public:
|
|
|
|
Machine(WoodScript *ws, int32 machineHash, Sequence *parentSeq, int32 dataHash,
|
|
|
|
int32 dataRowIndex, int callbackHandler, Common::String machineName, int32 id);
|
|
|
|
~Machine();
|
|
|
|
|
|
|
|
void clearMessages();
|
|
|
|
void clearPersistentMessages();
|
|
|
|
void restorePersistentMessages();
|
|
|
|
void sendMessage(uint32 messageHash, long messageValue, Machine *sender);
|
|
|
|
void resetSwitchTime();
|
|
|
|
bool changeSequenceProgram(int32 sequenceHash);
|
|
|
|
|
|
|
|
bool searchMessages(uint32 messageHash, uint32 messageValue, Machine *sender);
|
|
|
|
bool searchPersistentMessages(uint32 messageHash, uint32 messageValue, Machine *sender);
|
|
|
|
|
|
|
|
void enterState();
|
|
|
|
int32 execInstruction();
|
|
|
|
void execBlock(int32 offset, int32 count);
|
|
|
|
int32 getState() { return _currentState; }
|
|
|
|
|
|
|
|
int32 getId() const { return _id; }
|
|
|
|
|
|
|
|
bool m1_gotoState(Instruction &instruction);
|
|
|
|
bool m1_jump(Instruction &instruction);
|
|
|
|
bool m1_terminate(Instruction &instruction);
|
|
|
|
bool m1_startSequence(Instruction &instruction);
|
|
|
|
bool m1_pauseSequence(Instruction &instruction);
|
|
|
|
bool m1_resumeSequence(Instruction &instruction);
|
|
|
|
bool m1_storeValue(Instruction &instruction);
|
|
|
|
bool m1_sendMessage(Instruction &instruction);
|
|
|
|
bool m1_broadcastMessage(Instruction &instruction);
|
|
|
|
bool m1_replyMessage(Instruction &instruction);
|
|
|
|
bool m1_sendSystemMessage(Instruction &instruction);
|
|
|
|
bool m1_createMachine(Instruction &instruction);
|
|
|
|
bool m1_createMachineEx(Instruction &instruction);
|
|
|
|
bool m1_clearVars(Instruction &instruction);
|
|
|
|
|
|
|
|
void m1_onEndSequence(Instruction &instruction);
|
|
|
|
void m1_onMessage(Instruction &instruction);
|
|
|
|
void m1_switchLt(Instruction &instruction);
|
|
|
|
void m1_switchLe(Instruction &instruction);
|
|
|
|
void m1_switchEq(Instruction &instruction);
|
|
|
|
void m1_switchNe(Instruction &instruction);
|
|
|
|
void m1_switchGe(Instruction &instruction);
|
|
|
|
void m1_switchGt(Instruction &instruction);
|
|
|
|
|
|
|
|
long *dataRow() const { return _dataRow; }
|
|
|
|
Sequence *parentSequence() const { return _parentSequence; }
|
|
|
|
Common::String name() const { return _name; }
|
|
|
|
|
|
|
|
protected:
|
|
|
|
WoodScript *_ws;
|
|
|
|
Bytecode *_code;
|
|
|
|
|
|
|
|
Common::String _name;
|
|
|
|
Sequence *_sequence, *_parentSequence;
|
|
|
|
byte *_mach;
|
|
|
|
int32 _machHash, _machineCodeOffset;
|
|
|
|
int32 _stateCount, _stateTableOffset;
|
|
|
|
long *_dataRow;
|
|
|
|
int32 _id, _recursionLevel, _currentState, _targetCount;
|
|
|
|
/* TODO:
|
|
|
|
m->msgReplyXM = NULL;
|
|
|
|
m->CintrMsg = CintrMsg;
|
|
|
|
_walkPath
|
|
|
|
_messages
|
|
|
|
_persistentMessages
|
|
|
|
_usedPersistentMessages
|
|
|
|
*/
|
|
|
|
};
|
|
|
|
|
|
|
|
class WoodScript {
|
|
|
|
public:
|
|
|
|
|
2010-01-30 09:21:07 +00:00
|
|
|
WoodScript(MadsM4Engine *vm);
|
2008-04-20 14:47:37 +00:00
|
|
|
~WoodScript();
|
|
|
|
|
|
|
|
Machine *createMachine(int32 machineHash, Sequence *parentSeq, int32 dataHash, int32 dataRowIndex, int callbackHandler, const char *machineName);
|
|
|
|
Sequence *createSequence(Machine *machine, int32 sequenceHash);
|
|
|
|
|
|
|
|
void runSequencePrograms();
|
|
|
|
void runEndOfSequenceRequests();
|
|
|
|
void runTimerSequenceRequests();
|
|
|
|
|
|
|
|
/* Series */
|
|
|
|
// Move to own class, e.g. SeriesPlayer
|
|
|
|
int32 loadSeries(const char* seriesName, int32 hash, RGB8* palette);
|
|
|
|
void unloadSeries(int32 hash);
|
|
|
|
void setSeriesFramerate(Machine *machine, int32 frameRate);
|
|
|
|
Machine *playSeries(const char *seriesName, long layer, uint32 flags, int32 triggerNum,
|
|
|
|
int32 frameRate, int32 loopCount, int32 s, int32 x, int32 y,
|
|
|
|
int32 firstFrame, int32 lastFrame);
|
|
|
|
Machine *showSeries(const char *seriesName, long layer, uint32 flags, int32 triggerNum,
|
|
|
|
int32 duration, int32 index, int32 s, int32 x, int32 y);
|
|
|
|
Machine *streamSeries(const char *seriesName, int32 frameRate, long layer, int32 triggerNum);
|
|
|
|
|
|
|
|
void update();
|
|
|
|
void clear();
|
|
|
|
|
|
|
|
/* Misc */
|
|
|
|
void setDepthTable(int16 *depthTable);
|
2009-01-01 15:06:43 +00:00
|
|
|
|
2008-04-20 14:47:37 +00:00
|
|
|
long *getGlobalPtr(int index);
|
|
|
|
long getGlobal(int index);
|
|
|
|
void setGlobal(int index, long value);
|
|
|
|
|
|
|
|
AssetManager *assets() const { return _assets; }
|
|
|
|
|
|
|
|
// Sets the untouched, clean surface which contains the room background
|
|
|
|
void setBackgroundSurface(M4Surface *backgroundSurface);
|
|
|
|
// Sets the view which is used for drawing
|
|
|
|
void setSurfaceView(View *view);
|
2009-01-01 15:06:43 +00:00
|
|
|
|
2008-04-20 14:47:37 +00:00
|
|
|
RGB8 *getMainPalette() const;
|
2009-01-01 15:06:43 +00:00
|
|
|
|
2010-03-14 05:39:47 +00:00
|
|
|
void setInverseColourTable(byte *inverseColourTable) { _inverseColourTable = inverseColourTable; }
|
|
|
|
byte *getInverseColourTable() const { return _inverseColourTable; }
|
2008-04-20 14:47:37 +00:00
|
|
|
|
|
|
|
protected:
|
2010-01-30 09:21:07 +00:00
|
|
|
MadsM4Engine *_vm;
|
2008-04-20 14:47:37 +00:00
|
|
|
AssetManager *_assets;
|
|
|
|
|
|
|
|
Common::Array<Sequence*> _sequences, _layers;
|
|
|
|
Common::Array<Machine*> _machines;
|
|
|
|
int32 _machineId;
|
|
|
|
|
|
|
|
long *_globals;
|
2009-01-01 15:06:43 +00:00
|
|
|
|
2008-04-20 14:47:37 +00:00
|
|
|
Common::Array<Sequence*> _endOfSequenceRequestList;
|
|
|
|
|
|
|
|
int32 _indexReg;
|
|
|
|
|
|
|
|
/* Misc */
|
|
|
|
int16 *_depthTable;
|
2010-03-14 05:39:47 +00:00
|
|
|
byte *_inverseColourTable;
|
2008-04-20 14:47:37 +00:00
|
|
|
M4Surface *_backgroundSurface;
|
|
|
|
View *_surfaceView;
|
|
|
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
} // End of namespace M4
|
|
|
|
|
|
|
|
|
|
|
|
#endif
|