mirror of
https://github.com/libretro/scummvm.git
synced 2025-04-02 23:01:42 +00:00
Implemented all variants of IF script instruction. Program class has been changed to store an Array of instruction instead of a List, so that references to instructions are integers.
svn-id: r39631
This commit is contained in:
parent
ca993d8b00
commit
d18274d0ee
@ -37,18 +37,18 @@ void ProgramExec::runScript(ProgramPtr script, AnimationPtr a) {
|
||||
_ctxt._suspend = false;
|
||||
_ctxt._modCounter = _modCounter;
|
||||
|
||||
InstructionList::iterator inst;
|
||||
InstructionPtr inst;
|
||||
for ( ; (a->_flags & kFlagsActing) ; ) {
|
||||
|
||||
inst = _ctxt._ip;
|
||||
inst = script->_instructions[_ctxt._ip];
|
||||
_ctxt._inst = inst;
|
||||
++_ctxt._ip;
|
||||
|
||||
debugC(9, kDebugExec, "inst [%02i] %s\n", (*inst)->_index, _instructionNames[(*inst)->_index - 1]);
|
||||
debugC(9, kDebugExec, "inst [%02i] %s\n", inst->_index, _instructionNames[inst->_index - 1]);
|
||||
|
||||
script->_status = kProgramRunning;
|
||||
|
||||
(*_opcodes[(*inst)->_index])(_ctxt);
|
||||
(*_opcodes[inst->_index])(_ctxt);
|
||||
|
||||
if (_ctxt._suspend)
|
||||
break;
|
||||
|
@ -64,8 +64,8 @@ typedef Common::Array<const CommandOpcode*> CommandOpcodeSet;
|
||||
struct ProgramContext {
|
||||
AnimationPtr _anim;
|
||||
ProgramPtr _program;
|
||||
InstructionList::iterator _inst;
|
||||
InstructionList::iterator _ip;
|
||||
InstructionPtr _inst;
|
||||
uint32 _ip;
|
||||
uint16 _modCounter;
|
||||
bool _suspend;
|
||||
};
|
||||
|
@ -312,7 +312,7 @@ DECLARE_COMMAND_OPCODE(offsave) {
|
||||
}
|
||||
|
||||
DECLARE_INSTRUCTION_OPCODE(invalid) {
|
||||
error("Can't execute invalid opcode %i", (*ctxt._inst)->_index);
|
||||
error("Can't execute invalid opcode %i", ctxt._inst->_index);
|
||||
}
|
||||
|
||||
DECLARE_COMMAND_OPCODE(clear) {
|
||||
@ -367,24 +367,23 @@ DECLARE_COMMAND_OPCODE(set) {
|
||||
|
||||
|
||||
DECLARE_INSTRUCTION_OPCODE(on) {
|
||||
_vm->showZone((*ctxt._inst)->_z, true);
|
||||
_vm->showZone(ctxt._inst->_z, true);
|
||||
}
|
||||
|
||||
|
||||
DECLARE_INSTRUCTION_OPCODE(off) {
|
||||
_vm->showZone((*ctxt._inst)->_z, false);
|
||||
_vm->showZone(ctxt._inst->_z, false);
|
||||
}
|
||||
|
||||
|
||||
DECLARE_INSTRUCTION_OPCODE(set) {
|
||||
InstructionPtr inst = *ctxt._inst;
|
||||
inst->_opA.setValue(inst->_opB.getValue());
|
||||
ctxt._inst->_opA.setValue(ctxt._inst->_opB.getValue());
|
||||
}
|
||||
|
||||
|
||||
|
||||
DECLARE_INSTRUCTION_OPCODE(inc) {
|
||||
InstructionPtr inst = *ctxt._inst;
|
||||
InstructionPtr inst = ctxt._inst;
|
||||
|
||||
int16 rvalue = inst->_opB.getValue();
|
||||
|
||||
@ -437,25 +436,25 @@ DECLARE_INSTRUCTION_OPCODE(wait) {
|
||||
|
||||
|
||||
DECLARE_INSTRUCTION_OPCODE(start) {
|
||||
(*ctxt._inst)->_z->_flags |= kFlagsActing;
|
||||
ctxt._inst->_z->_flags |= kFlagsActing;
|
||||
}
|
||||
|
||||
|
||||
DECLARE_INSTRUCTION_OPCODE(process) {
|
||||
_vm->_activeZone2 = (*ctxt._inst)->_z;
|
||||
_vm->_activeZone2 = ctxt._inst->_z;
|
||||
}
|
||||
|
||||
|
||||
DECLARE_INSTRUCTION_OPCODE(move) {
|
||||
// NOTE: I couldn't find evidence of scripts containing this instruction being used
|
||||
InstructionPtr inst =*ctxt._inst;
|
||||
InstructionPtr inst = ctxt._inst;
|
||||
_vm->scheduleWalk(inst->_opA.getValue(), inst->_opB.getValue(), false);
|
||||
ctxt._suspend = true;
|
||||
}
|
||||
|
||||
|
||||
DECLARE_INSTRUCTION_OPCODE(color) {
|
||||
InstructionPtr inst = *ctxt._inst;
|
||||
InstructionPtr inst = ctxt._inst;
|
||||
_vm->_gfx->_palette.setEntry(inst->_opB.getValue(), inst->_colors[0], inst->_colors[1], inst->_colors[2]);
|
||||
}
|
||||
|
||||
@ -477,33 +476,45 @@ DECLARE_INSTRUCTION_OPCODE(print) {
|
||||
}
|
||||
|
||||
DECLARE_INSTRUCTION_OPCODE(text) {
|
||||
InstructionPtr inst = (*ctxt._inst);
|
||||
InstructionPtr inst = ctxt._inst;
|
||||
_vm->setupSubtitles(inst->_text, inst->_text2, inst->_y);
|
||||
}
|
||||
|
||||
|
||||
DECLARE_INSTRUCTION_OPCODE(ifeq) {
|
||||
warning("Parallaction_br::instOp_ifeq not yet implemented");
|
||||
InstructionPtr inst = ctxt._inst;
|
||||
bool cond = inst->_opA.getValue() == inst->_opB.getValue();
|
||||
if (!cond) {
|
||||
ctxt._ip = inst->_endif;
|
||||
}
|
||||
}
|
||||
|
||||
|
||||
DECLARE_INSTRUCTION_OPCODE(iflt) {
|
||||
warning("Parallaction_br::instOp_iflt not yet implemented");
|
||||
InstructionPtr inst = ctxt._inst;
|
||||
bool cond = inst->_opA.getValue() < inst->_opB.getValue();
|
||||
if (!cond) {
|
||||
ctxt._ip = inst->_endif;
|
||||
}
|
||||
}
|
||||
|
||||
|
||||
DECLARE_INSTRUCTION_OPCODE(ifgt) {
|
||||
warning("Parallaction_br::instOp_ifgt not yet implemented");
|
||||
InstructionPtr inst = ctxt._inst;
|
||||
bool cond = inst->_opA.getValue() > inst->_opB.getValue();
|
||||
if (!cond) {
|
||||
ctxt._ip = inst->_endif;
|
||||
}
|
||||
}
|
||||
|
||||
|
||||
DECLARE_INSTRUCTION_OPCODE(endif) {
|
||||
warning("Parallaction_br::instOp_endif not yet implemented");
|
||||
// nothing to do here
|
||||
}
|
||||
|
||||
|
||||
DECLARE_INSTRUCTION_OPCODE(stop) {
|
||||
ZonePtr z = (*ctxt._inst)->_z;
|
||||
ZonePtr z = ctxt._inst->_z;
|
||||
if (ACTIONTYPE(z) == kZoneHear) {
|
||||
warning("Parallaction_br::instOp_stop not yet implemented for HEAR zones");
|
||||
// TODO: stop music or sound effects generated by a zone.
|
||||
@ -513,7 +524,7 @@ DECLARE_INSTRUCTION_OPCODE(stop) {
|
||||
}
|
||||
|
||||
DECLARE_INSTRUCTION_OPCODE(loop) {
|
||||
InstructionPtr inst = *ctxt._inst;
|
||||
InstructionPtr inst = ctxt._inst;
|
||||
|
||||
ctxt._program->_loopCounter = inst->_opB.getValue();
|
||||
ctxt._program->_loopStart = ctxt._ip;
|
||||
@ -531,7 +542,7 @@ DECLARE_INSTRUCTION_OPCODE(show) {
|
||||
}
|
||||
|
||||
DECLARE_INSTRUCTION_OPCODE(call) {
|
||||
_vm->callFunction((*ctxt._inst)->_immediate, 0);
|
||||
_vm->callFunction(ctxt._inst->_immediate, 0);
|
||||
}
|
||||
|
||||
|
||||
@ -542,7 +553,7 @@ DECLARE_INSTRUCTION_OPCODE(endscript) {
|
||||
ctxt._program->_status = kProgramDone;
|
||||
}
|
||||
|
||||
ctxt._ip = ctxt._program->_instructions.begin();
|
||||
ctxt._ip = 0;
|
||||
ctxt._suspend = true;
|
||||
}
|
||||
|
||||
|
@ -67,7 +67,7 @@ extern const char *_instructionNamesRes_ns[];
|
||||
|
||||
|
||||
DECLARE_INSTRUCTION_OPCODE(on) {
|
||||
InstructionPtr inst = *ctxt._inst;
|
||||
InstructionPtr inst = ctxt._inst;
|
||||
|
||||
inst->_a->_flags |= kFlagsActive;
|
||||
inst->_a->_flags &= ~kFlagsRemove;
|
||||
@ -75,12 +75,12 @@ DECLARE_INSTRUCTION_OPCODE(on) {
|
||||
|
||||
|
||||
DECLARE_INSTRUCTION_OPCODE(off) {
|
||||
(*ctxt._inst)->_a->_flags |= kFlagsRemove;
|
||||
ctxt._inst->_a->_flags |= kFlagsRemove;
|
||||
}
|
||||
|
||||
|
||||
DECLARE_INSTRUCTION_OPCODE(loop) {
|
||||
InstructionPtr inst = *ctxt._inst;
|
||||
InstructionPtr inst = ctxt._inst;
|
||||
|
||||
ctxt._program->_loopCounter = inst->_opB.getValue();
|
||||
ctxt._program->_loopStart = ctxt._ip;
|
||||
@ -94,7 +94,7 @@ DECLARE_INSTRUCTION_OPCODE(endloop) {
|
||||
}
|
||||
|
||||
DECLARE_INSTRUCTION_OPCODE(inc) {
|
||||
InstructionPtr inst = *ctxt._inst;
|
||||
InstructionPtr inst = ctxt._inst;
|
||||
int16 _si = inst->_opB.getValue();
|
||||
|
||||
if (inst->_flags & kInstMod) { // mod
|
||||
@ -118,13 +118,12 @@ DECLARE_INSTRUCTION_OPCODE(inc) {
|
||||
|
||||
|
||||
DECLARE_INSTRUCTION_OPCODE(set) {
|
||||
InstructionPtr inst = *ctxt._inst;
|
||||
inst->_opA.setValue(inst->_opB.getValue());
|
||||
ctxt._inst->_opA.setValue(ctxt._inst->_opB.getValue());
|
||||
}
|
||||
|
||||
|
||||
DECLARE_INSTRUCTION_OPCODE(put) {
|
||||
InstructionPtr inst = *ctxt._inst;
|
||||
InstructionPtr inst = ctxt._inst;
|
||||
Common::Rect r;
|
||||
inst->_a->getFrameRect(r);
|
||||
|
||||
@ -145,11 +144,11 @@ DECLARE_INSTRUCTION_OPCODE(show) {
|
||||
}
|
||||
|
||||
DECLARE_INSTRUCTION_OPCODE(invalid) {
|
||||
error("Can't execute invalid opcode %i", (*ctxt._inst)->_index);
|
||||
error("Can't execute invalid opcode %i", ctxt._inst->_index);
|
||||
}
|
||||
|
||||
DECLARE_INSTRUCTION_OPCODE(call) {
|
||||
_vm->callFunction((*ctxt._inst)->_immediate, 0);
|
||||
_vm->callFunction(ctxt._inst->_immediate, 0);
|
||||
}
|
||||
|
||||
|
||||
@ -162,17 +161,17 @@ DECLARE_INSTRUCTION_OPCODE(wait) {
|
||||
|
||||
|
||||
DECLARE_INSTRUCTION_OPCODE(start) {
|
||||
(*ctxt._inst)->_a->_flags |= (kFlagsActing | kFlagsActive);
|
||||
ctxt._inst->_a->_flags |= (kFlagsActing | kFlagsActive);
|
||||
}
|
||||
|
||||
|
||||
DECLARE_INSTRUCTION_OPCODE(sound) {
|
||||
_vm->_activeZone = (*ctxt._inst)->_z;
|
||||
_vm->_activeZone = ctxt._inst->_z;
|
||||
}
|
||||
|
||||
|
||||
DECLARE_INSTRUCTION_OPCODE(move) {
|
||||
InstructionPtr inst = (*ctxt._inst);
|
||||
InstructionPtr inst = ctxt._inst;
|
||||
|
||||
int16 x = inst->_opA.getValue();
|
||||
int16 y = inst->_opB.getValue();
|
||||
@ -187,7 +186,7 @@ DECLARE_INSTRUCTION_OPCODE(endscript) {
|
||||
ctxt._program->_status = kProgramDone;
|
||||
}
|
||||
|
||||
ctxt._ip = ctxt._program->_instructions.begin();
|
||||
ctxt._ip = 0;
|
||||
ctxt._suspend = true;
|
||||
}
|
||||
|
||||
|
@ -25,7 +25,6 @@
|
||||
|
||||
#include "parallaction/parallaction.h"
|
||||
#include "parallaction/objects.h"
|
||||
#include "parallaction/parser.h"
|
||||
|
||||
namespace Parallaction {
|
||||
|
||||
|
@ -49,7 +49,7 @@ typedef Common::SharedPtr<Animation> AnimationPtr;
|
||||
typedef Common::List<AnimationPtr> AnimationList;
|
||||
|
||||
typedef Common::SharedPtr<Instruction> InstructionPtr;
|
||||
typedef Common::List<InstructionPtr> InstructionList;
|
||||
typedef Common::Array<InstructionPtr> InstructionList;
|
||||
|
||||
typedef Common::List<Common::Point> PointList;
|
||||
|
||||
@ -468,7 +468,7 @@ struct Instruction {
|
||||
char *_text;
|
||||
char *_text2;
|
||||
int _y;
|
||||
InstructionList::iterator _endif;
|
||||
uint32 _endif;
|
||||
|
||||
Instruction();
|
||||
~Instruction();
|
||||
@ -483,16 +483,14 @@ enum {
|
||||
|
||||
struct Program {
|
||||
AnimationPtr _anim;
|
||||
|
||||
LocalVariable *_locals;
|
||||
|
||||
uint16 _loopCounter;
|
||||
uint16 _loopCounter;
|
||||
uint16 _numLocals;
|
||||
|
||||
uint16 _numLocals;
|
||||
|
||||
InstructionList::iterator _ip;
|
||||
InstructionList::iterator _loopStart;
|
||||
InstructionList _instructions;
|
||||
uint32 _ip;
|
||||
uint32 _loopStart;
|
||||
InstructionList _instructions;
|
||||
|
||||
uint32 _status;
|
||||
|
||||
|
@ -328,14 +328,13 @@ protected:
|
||||
OpcodeSet _instructionParsers;
|
||||
Table *_instructionNames;
|
||||
|
||||
uint32 _currentInstruction; // index of the instruction being parsed
|
||||
|
||||
struct ParserContext {
|
||||
bool end;
|
||||
AnimationPtr a;
|
||||
InstructionPtr inst;
|
||||
LocalVariable *locals;
|
||||
|
||||
// BRA specific
|
||||
InstructionPtr openIf;
|
||||
} ctxt;
|
||||
|
||||
DECLARE_UNQUALIFIED_INSTRUCTION_PARSER(defLocal);
|
||||
@ -377,7 +376,7 @@ public:
|
||||
clearSet(_instructionParsers);
|
||||
}
|
||||
|
||||
void parse(Script *script, ProgramPtr program);
|
||||
virtual void parse(Script *script, ProgramPtr program);
|
||||
|
||||
};
|
||||
|
||||
@ -395,6 +394,10 @@ protected:
|
||||
DECLARE_UNQUALIFIED_INSTRUCTION_PARSER(if_op);
|
||||
DECLARE_UNQUALIFIED_INSTRUCTION_PARSER(endif);
|
||||
|
||||
int32 _openIfStatement;
|
||||
void beginIfStatement();
|
||||
void endIfStatement();
|
||||
|
||||
virtual void parseRValue(ScriptVar &var, const char *str);
|
||||
|
||||
public:
|
||||
@ -402,7 +405,7 @@ public:
|
||||
}
|
||||
|
||||
virtual void init();
|
||||
|
||||
virtual void parse(Script *script, ProgramPtr program);
|
||||
};
|
||||
|
||||
|
||||
|
@ -1058,9 +1058,7 @@ DECLARE_INSTRUCTION_PARSER(text) {
|
||||
DECLARE_INSTRUCTION_PARSER(if_op) {
|
||||
debugC(7, kDebugParser, "INSTRUCTION_PARSER(if_op) ");
|
||||
|
||||
|
||||
if (ctxt.openIf)
|
||||
error("cannot nest 'if' blocks");
|
||||
beginIfStatement();
|
||||
|
||||
parseLValue(ctxt.inst->_opA, _tokens[1]);
|
||||
parseRValue(ctxt.inst->_opB, _tokens[3]);
|
||||
@ -1075,20 +1073,27 @@ DECLARE_INSTRUCTION_PARSER(if_op) {
|
||||
ctxt.inst->_index = INST_IFLT;
|
||||
} else
|
||||
error("unknown test operator '%s' in if-clause", _tokens[2]);
|
||||
}
|
||||
|
||||
ctxt.openIf = ctxt.inst;
|
||||
void ProgramParser_br::beginIfStatement() {
|
||||
if (_openIfStatement != -1)
|
||||
error("cannot nest 'if' statements");
|
||||
|
||||
_openIfStatement = _currentInstruction;
|
||||
}
|
||||
|
||||
void ProgramParser_br::endIfStatement() {
|
||||
if (_openIfStatement == -1)
|
||||
error("unexpected 'endif' in script");
|
||||
|
||||
_program->_instructions[_openIfStatement]->_endif = _currentInstruction;
|
||||
_openIfStatement = -1;
|
||||
}
|
||||
|
||||
|
||||
DECLARE_INSTRUCTION_PARSER(endif) {
|
||||
debugC(7, kDebugParser, "INSTRUCTION_PARSER(endif) ");
|
||||
|
||||
if (!ctxt.openIf)
|
||||
error("unexpected 'endif'");
|
||||
|
||||
// ctxt.openIf->_endif = ctxt.inst;
|
||||
ctxt.openIf = InstructionPtr();
|
||||
endIfStatement();
|
||||
ctxt.inst->_index = _parser->_lookup;
|
||||
}
|
||||
|
||||
@ -1142,6 +1147,11 @@ void ProgramParser_br::parseRValue(ScriptVar &v, const char *str) {
|
||||
|
||||
}
|
||||
|
||||
void ProgramParser_br::parse(Script *script, ProgramPtr program) {
|
||||
_openIfStatement = -1;
|
||||
ProgramParser_ns::parse(script, program);
|
||||
}
|
||||
|
||||
|
||||
void LocationParser_br::init() {
|
||||
|
||||
|
@ -327,6 +327,9 @@ void ProgramParser_ns::parseInstruction() {
|
||||
|
||||
InstructionPtr inst(new Instruction);
|
||||
ctxt.inst = inst;
|
||||
|
||||
_currentInstruction = _program->_instructions.size();
|
||||
|
||||
_parser->parseStatement();
|
||||
_program->_instructions.push_back(inst);
|
||||
|
||||
@ -337,7 +340,6 @@ void ProgramParser_ns::parse(Script *script, ProgramPtr program) {
|
||||
_script = script;
|
||||
_program = program;
|
||||
|
||||
ctxt.openIf = InstructionPtr();
|
||||
ctxt.end = false;
|
||||
ctxt.locals = program->_locals;
|
||||
|
||||
@ -348,7 +350,7 @@ void ProgramParser_ns::parse(Script *script, ProgramPtr program) {
|
||||
} while (!ctxt.end);
|
||||
_parser->popTables();
|
||||
|
||||
program->_ip = program->_instructions.begin();
|
||||
program->_ip = 0;
|
||||
}
|
||||
|
||||
void Parallaction_ns::loadProgram(AnimationPtr a, const char *filename) {
|
||||
|
Loading…
x
Reference in New Issue
Block a user