mirror of
https://github.com/libretro/scummvm.git
synced 2025-01-13 05:00:59 +00:00
294 lines
11 KiB
C++
294 lines
11 KiB
C++
/* 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.
|
|
*
|
|
*/
|
|
|
|
/*
|
|
* Based on
|
|
* WebVenture (c) 2010, Sean Kasun
|
|
* https://github.com/mrkite/webventure, http://seancode.com/webventure/
|
|
*
|
|
* Used with explicit permission from the author
|
|
*/
|
|
|
|
#ifndef MACVENTURE_SCRIPT_H
|
|
#define MACVENTURE_SCRIPT_H
|
|
|
|
#include "macventure/container.h"
|
|
#include "macventure/world.h"
|
|
#include "macventure/macventure.h"
|
|
#include "macventure/controls.h"
|
|
|
|
namespace MacVenture {
|
|
|
|
class Container;
|
|
class World;
|
|
|
|
typedef uint32 ObjID;
|
|
|
|
class ScriptAsset {
|
|
public:
|
|
ScriptAsset(ObjID id, Container *container);
|
|
~ScriptAsset() {}
|
|
|
|
void reset();
|
|
uint8 fetch();
|
|
bool hasNext();
|
|
void branch(int16 amount);
|
|
|
|
ObjID getId();
|
|
|
|
private:
|
|
|
|
void loadInstructions();
|
|
|
|
private:
|
|
ObjID _id;
|
|
Container *_container;
|
|
|
|
Common::Array<uint8> _instructions;
|
|
uint32 _ip; // Instruction pointer
|
|
};
|
|
|
|
class EngineState {
|
|
public:
|
|
EngineState() {
|
|
clear();
|
|
}
|
|
|
|
void push(int16 data) {
|
|
sp--;
|
|
stack[sp] = unneg16(data);
|
|
}
|
|
|
|
int16 pop() {
|
|
int16 v = stack[sp];
|
|
sp++;
|
|
return v;
|
|
}
|
|
|
|
int16 peek(int16 off) {
|
|
return stack[sp + off];
|
|
}
|
|
|
|
void poke(int16 off, int16 val) {
|
|
stack[sp + off] = unneg16(val);
|
|
}
|
|
|
|
void clear() {
|
|
sp = 0x80;
|
|
for (int i = 0; i < sp; i++) {
|
|
stack[i] = 0;
|
|
}
|
|
}
|
|
|
|
int16 size() {
|
|
return 0x80 - sp;
|
|
}
|
|
|
|
private:
|
|
int16 unneg16(int16 data) {
|
|
if (data < 0)
|
|
data = ((-data) ^ 0xFFFF) + 1;
|
|
|
|
return data;
|
|
}
|
|
|
|
private:
|
|
|
|
int16 stack[0x80];
|
|
int16 sp;
|
|
};
|
|
|
|
struct FunCall {
|
|
int16 func;
|
|
int16 rank;
|
|
|
|
FunCall(int16 f, int16 r) {
|
|
func = f;
|
|
rank = r;
|
|
}
|
|
};
|
|
|
|
struct EngineFrame {
|
|
ControlAction action;
|
|
ObjID src;
|
|
ObjID dest;
|
|
int x;
|
|
int y;
|
|
EngineState state;
|
|
Common::List<ScriptAsset> scripts;
|
|
Common::Array<FunCall> saves;
|
|
uint32 familyIdx;
|
|
|
|
bool haltedInFirst;
|
|
bool haltedInFamily;
|
|
bool haltedInSaves;
|
|
};
|
|
|
|
class ScriptEngine {
|
|
public:
|
|
ScriptEngine(MacVentureEngine *engine, World *world);
|
|
~ScriptEngine();
|
|
|
|
public:
|
|
bool runControl(ControlAction action, ObjID source, ObjID destination, Common::Point delta);
|
|
bool resume(bool execAll);
|
|
void reset();
|
|
|
|
private:
|
|
bool execFrame(bool execAll);
|
|
bool loadScript(EngineFrame *frame, uint32 scriptID);
|
|
bool resumeFunc(EngineFrame *frame);
|
|
bool runFunc(EngineFrame *frame);
|
|
|
|
private:
|
|
|
|
// Aux
|
|
int16 neg16(int16 val);
|
|
int16 neg8(int16 val);
|
|
int16 sumChildrenAttr(int16 obj, int16 attr, bool recursive);
|
|
void ensureNonzeroDivisor(int16 divisor, byte opcode);
|
|
|
|
// Opcodes
|
|
void op80GATT(EngineState *state, EngineFrame *frame); //get attribute
|
|
void op81SATT(EngineState *state, EngineFrame *frame); //set attribute
|
|
void op82SUCH(EngineState *state, EngineFrame *frame); //sum children attribute
|
|
void op83PUCT(EngineState *state, EngineFrame *frame); //push selected control
|
|
void op84PUOB(EngineState *state, EngineFrame *frame); //push selected object
|
|
void op85PUTA(EngineState *state, EngineFrame *frame); //push target
|
|
void op86PUDX(EngineState *state, EngineFrame *frame); //push deltax
|
|
void op87PUDY(EngineState *state, EngineFrame *frame); //push deltay
|
|
void op88PUIB(EngineState *state, EngineFrame *frame, ScriptAsset *script);//push immediate.b
|
|
void op89PUI(EngineState *state, EngineFrame *frame, ScriptAsset *script);//push immediate
|
|
void op8aGGLO(EngineState *state, EngineFrame *frame); //get global
|
|
void op8bSGLO(EngineState *state, EngineFrame *frame); //set global
|
|
void op8cRAND(EngineState *state, EngineFrame *frame); //random
|
|
void op8dCOPY(EngineState *state, EngineFrame *frame); //copy
|
|
void op8eCOPYN(EngineState *state, EngineFrame *frame); //copyn
|
|
void op8fSWAP(EngineState *state, EngineFrame *frame); //swap
|
|
|
|
void op90SWAPN(EngineState *state, EngineFrame *frame); //swapn
|
|
void op91POP(EngineState *state, EngineFrame *frame); //pop
|
|
void op92COPYP(EngineState *state, EngineFrame *frame); //copy+1
|
|
void op93COPYPN(EngineState *state, EngineFrame *frame);//copy+n
|
|
void op94SHUFF(EngineState *state, EngineFrame *frame); //shuffle
|
|
void op95SORT(EngineState *state, EngineFrame *frame); //sort
|
|
void op96CLEAR(EngineState *state, EngineFrame *frame); //clear stack
|
|
void op97SIZE(EngineState *state, EngineFrame *frame); //get stack size
|
|
void op98ADD(EngineState *state, EngineFrame *frame); //add
|
|
void op99SUB(EngineState *state, EngineFrame *frame); //subtract
|
|
void op9aMUL(EngineState *state, EngineFrame *frame); //multiply
|
|
void op9bDIV(EngineState *state, EngineFrame *frame); //divide
|
|
void op9cMOD(EngineState *state, EngineFrame *frame); //mod
|
|
void op9dDMOD(EngineState *state, EngineFrame *frame); //divmod
|
|
void op9eABS(EngineState *state, EngineFrame *frame); //abs
|
|
void op9fNEG(EngineState *state, EngineFrame *frame); //neg
|
|
|
|
void opa0AND(EngineState *state, EngineFrame *frame); //and
|
|
void opa1OR(EngineState *state, EngineFrame *frame); //or
|
|
void opa2XOR(EngineState *state, EngineFrame *frame); //xor
|
|
void opa3NOT(EngineState *state, EngineFrame *frame); //not
|
|
void opa4LAND(EngineState *state, EngineFrame *frame); //logical and
|
|
void opa5LOR(EngineState *state, EngineFrame *frame); //logical or
|
|
void opa6LXOR(EngineState *state, EngineFrame *frame); //logical xor
|
|
void opa7LNOT(EngineState *state, EngineFrame *frame); //logical not
|
|
void opa8GTU(EngineState *state, EngineFrame *frame); //gt? unsigned
|
|
void opa9LTU(EngineState *state, EngineFrame *frame); //lt? unsigned
|
|
void opaaGTS(EngineState *state, EngineFrame *frame); //gt? signed
|
|
void opabLTS(EngineState *state, EngineFrame *frame); //lt? signed
|
|
void opacEQ(EngineState *state, EngineFrame *frame); //eq?
|
|
void opadEQS(EngineState *state, EngineFrame *frame); //eq string?
|
|
void opaeCONT(EngineState *state, EngineFrame *frame); //contains
|
|
void opafCONTW(EngineState *state, EngineFrame *frame); //contains word
|
|
|
|
void opb0BRA(EngineState *state, EngineFrame *frame, ScriptAsset *script); //bra
|
|
void opb1BRAB(EngineState *state, EngineFrame *frame, ScriptAsset *script); //bra.b
|
|
void opb2BEQ(EngineState *state, EngineFrame *frame, ScriptAsset *script); //beq
|
|
void opb3BEQB(EngineState *state, EngineFrame *frame, ScriptAsset *script); //beq.b
|
|
void opb4BNE(EngineState *state, EngineFrame *frame, ScriptAsset *script); //bne
|
|
void opb5BNEB(EngineState *state, EngineFrame *frame, ScriptAsset *script); //bne.b
|
|
void opb6CLAT(EngineState *state, EngineFrame *frame); //call later
|
|
void opb7CCA(EngineState *state, EngineFrame *frame); //cancel call
|
|
void opb8CLOW(EngineState *state, EngineFrame *frame); //cancel low priority
|
|
void opb9CHI(EngineState *state, EngineFrame *frame); //cancel high priority
|
|
void opbaCRAN(EngineState *state, EngineFrame *frame); //cancel priority range
|
|
bool opbbFORK(EngineState *state, EngineFrame *frame); //fork
|
|
bool opbcCALL(EngineState *state, EngineFrame *frame, ScriptAsset &script); //call
|
|
void opbdFOOB(EngineState *state, EngineFrame *frame); //focus object
|
|
void opbeSWOB(EngineState *state, EngineFrame *frame); //swap objects
|
|
void opbfSNOB(EngineState *state, EngineFrame *frame); //snap object
|
|
|
|
void opc0TEXI(EngineState *state, EngineFrame *frame); //toggle exits
|
|
void opc1PTXT(EngineState *state, EngineFrame *frame); //print text
|
|
void opc2PNEW(EngineState *state, EngineFrame *frame); //print newline
|
|
void opc3PTNE(EngineState *state, EngineFrame *frame); //print text+nl
|
|
void opc4PNTN(EngineState *state, EngineFrame *frame); //print nl+text+nl
|
|
void opc5PNUM(EngineState *state, EngineFrame *frame); //print number
|
|
void opc6P2(EngineState *state, EngineFrame *frame); //push 2
|
|
void opc7PLBG(EngineState *state, EngineFrame *frame); //play sound in background
|
|
void opc8PLAW(EngineState *state, EngineFrame *frame); //play sound and wait
|
|
void opc9WAIT(EngineState *state, EngineFrame *frame); //wait for sound to finish?
|
|
void opcaTIME(EngineState *state, EngineFrame *frame); //get current time
|
|
void opcbDAY(EngineState *state, EngineFrame *frame); //get current day
|
|
void opccCHLD(EngineState *state, EngineFrame *frame); //get children
|
|
void opcdNCHLD(EngineState *state, EngineFrame *frame); //get num children
|
|
void opceVERS(EngineState *state, EngineFrame *frame); //get engine version
|
|
void opcfPSCE(EngineState *state, EngineFrame *frame); //push scenario number
|
|
|
|
void opd0P1(EngineState *state, EngineFrame *frame); //push 1
|
|
void opd1GOBD(EngineState *state, EngineFrame *frame); //get object dimensions
|
|
void opd2GOVP(EngineState *state, EngineFrame *frame); //get overlap percent
|
|
void opd3CAPC(EngineState *state, EngineFrame *frame); //capture children
|
|
void opd4RELC(EngineState *state, EngineFrame *frame); //release children
|
|
void opd5DLOG(EngineState *state, EngineFrame *frame); //show speech dialog
|
|
void opd6ACMD(EngineState *state, EngineFrame *frame); //activate command
|
|
void opd7LOSE(EngineState *state, EngineFrame *frame); //lose game
|
|
void opd8WIN(EngineState *state, EngineFrame *frame); //win game
|
|
void opd9SLEEP(EngineState *state, EngineFrame *frame); //sleep
|
|
void opdaCLICK(EngineState *state, EngineFrame *frame); //click to continue
|
|
void opdbROBQ(EngineState *state, EngineFrame *frame); //run queue
|
|
void opdcRSQ(EngineState *state, EngineFrame *frame); //run sound queue
|
|
void opddRTQ(EngineState *state, EngineFrame *frame); //run text queue
|
|
void opdeUPSC(EngineState *state, EngineFrame *frame); //update screen
|
|
void opdfFMAI(EngineState *state, EngineFrame *frame); //flash main window
|
|
|
|
void ope0CHGR(EngineState *state, EngineFrame *frame); //cache graphic and object
|
|
void ope1CHSO(EngineState *state, EngineFrame *frame); //cache sound
|
|
void ope2MDIV(EngineState *state, EngineFrame *frame); //muldiv
|
|
void ope3UPOB(EngineState *state, EngineFrame *frame); //update object
|
|
void ope4PLEV(EngineState *state, EngineFrame *frame); //currently playing event?
|
|
void ope5WEV(EngineState *state, EngineFrame *frame); //wait for event to finish
|
|
void ope6GFIB(EngineState *state, EngineFrame *frame); //get fibonacci (joke)
|
|
void ope7CFIB(EngineState *state, EngineFrame *frame); //calc fibonacci
|
|
|
|
void op00NOOP(byte op);
|
|
|
|
private:
|
|
MacVentureEngine *_engine;
|
|
World *_world;
|
|
Common::List<EngineFrame> _frames;
|
|
Container *_scripts;
|
|
};
|
|
|
|
} // End of namespace MacVenture
|
|
|
|
#endif
|