scummvm/scumm/script_v90he.cpp

1487 lines
26 KiB
C++
Raw Normal View History

/* ScummVM - Scumm Interpreter
* Copyright (C) 2001 Ludvig Strigeus
* Copyright (C) 2001-2004 The ScummVM project
*
* 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., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA.
*
* $Header$
*
*/
#include "stdafx.h"
#include "common/config-manager.h"
#include "scumm/actor.h"
#include "scumm/charset.h"
#include "scumm/intern.h"
#include "scumm/object.h"
#include "scumm/resource.h"
#include "scumm/resource_v7he.h"
#include "scumm/scumm.h"
#include "scumm/sound.h"
namespace Scumm {
#define OPCODE(x) _OPCODE(ScummEngine_v90he, x)
void ScummEngine_v90he::setupOpcodes() {
static const OpcodeEntryV90he opcodes[256] = {
/* 00 */
OPCODE(o6_pushByte),
OPCODE(o6_pushWord),
OPCODE(o72_pushDWord),
OPCODE(o6_pushWordVar),
/* 04 */
OPCODE(o72_addMessageToStack),
OPCODE(o6_invalid),
OPCODE(o6_invalid),
2004-09-20 19:19:57 +00:00
OPCODE(o6_wordArrayRead),
/* 08 */
OPCODE(o6_invalid),
OPCODE(o6_invalid),
OPCODE(o90_dup),
2004-09-20 19:19:57 +00:00
OPCODE(o6_wordArrayIndexedRead),
/* 0C */
OPCODE(o6_dup),
OPCODE(o6_not),
OPCODE(o6_eq),
OPCODE(o6_neq),
/* 10 */
OPCODE(o6_gt),
OPCODE(o6_lt),
OPCODE(o6_le),
OPCODE(o6_ge),
/* 14 */
OPCODE(o6_add),
OPCODE(o6_sub),
OPCODE(o6_mul),
OPCODE(o6_div),
/* 18 */
OPCODE(o6_land),
OPCODE(o6_lor),
OPCODE(o6_pop),
OPCODE(o72_isAnyOf),
/* 1C */
OPCODE(o90_wizImageOps),
2004-10-08 17:56:08 +00:00
OPCODE(o90_min),
OPCODE(o90_max),
2004-10-03 19:57:19 +00:00
OPCODE(o90_sin),
/* 20 */
2004-10-03 19:57:19 +00:00
OPCODE(o90_cos),
2004-10-08 17:56:08 +00:00
OPCODE(o90_sqrt),
OPCODE(o90_atan2),
OPCODE(o90_getSegmentAngle),
/* 24 */
OPCODE(o90_getDistanceBetweenPoints),
OPCODE(o90_unknown25), // o90_getSpriteInfo
OPCODE(o90_unknown26), // o90_setSpriteInfo
OPCODE(o90_unknown27), // o90_getSpriteGroupInfo
/* 28 */
OPCODE(o90_unknown28), // o90_setSpriteGroupInfo
OPCODE(o90_getWizData),
OPCODE(o6_invalid),
2004-10-10 05:49:30 +00:00
OPCODE(o90_startScriptUnk),
/* 2C */
2004-10-10 05:49:30 +00:00
OPCODE(o90_jumpToScriptUnk),
OPCODE(o6_invalid),
OPCODE(o6_invalid),
2004-09-10 14:31:20 +00:00
OPCODE(o90_unknown2F),
/* 30 */
OPCODE(o90_mod),
2004-11-28 05:33:33 +00:00
OPCODE(o90_shl),
OPCODE(o90_shr),
OPCODE(o6_invalid),
/* 34 */
OPCODE(o90_findAllObjectsWithClassOf),
OPCODE(o90_getPolygonOverlap),
OPCODE(o90_cond),
2004-10-04 01:34:29 +00:00
OPCODE(o90_dim2dim2Array),
/* 38 */
OPCODE(o6_invalid),
OPCODE(o6_invalid),
OPCODE(o90_sortArray),
OPCODE(o6_invalid),
/* 3C */
OPCODE(o6_invalid),
OPCODE(o6_invalid),
OPCODE(o6_invalid),
OPCODE(o6_invalid),
/* 40 */
OPCODE(o6_invalid),
OPCODE(o6_invalid),
OPCODE(o6_invalid),
OPCODE(o6_writeWordVar),
/* 44 */
OPCODE(o90_getObjectData),
2004-11-23 19:47:54 +00:00
OPCODE(o80_loadSBNG),
OPCODE(o6_invalid),
2004-09-20 19:19:57 +00:00
OPCODE(o6_wordArrayWrite),
/* 48 */
2004-11-23 19:47:54 +00:00
OPCODE(o80_stringToInt),
2004-10-17 06:32:58 +00:00
OPCODE(o80_getSoundVar),
2004-10-10 05:49:30 +00:00
OPCODE(o80_localizeArrayToRoom),
2004-09-20 19:19:57 +00:00
OPCODE(o6_wordArrayIndexedWrite),
/* 4C */
OPCODE(o6_invalid),
OPCODE(o80_readConfigFile),
OPCODE(o80_writeConfigFile),
OPCODE(o6_wordVarInc),
/* 50 */
OPCODE(o72_resetCutscene),
OPCODE(o6_invalid),
OPCODE(o72_findObjectWithClassOf),
2004-10-13 10:55:50 +00:00
OPCODE(o6_wordArrayInc),
/* 54 */
OPCODE(o72_getObjectImageX),
OPCODE(o72_getObjectImageY),
2004-10-08 22:58:49 +00:00
OPCODE(o72_captureWizImage),
OPCODE(o6_wordVarDec),
/* 58 */
OPCODE(o72_getTimer),
OPCODE(o72_setTimer),
OPCODE(o72_unknown5A),
2004-10-13 10:55:50 +00:00
OPCODE(o6_wordArrayDec),
/* 5C */
OPCODE(o6_if),
OPCODE(o6_ifNot),
OPCODE(o72_startScript),
OPCODE(o6_startScriptQuick),
/* 60 */
OPCODE(o72_startObject),
OPCODE(o72_drawObject),
OPCODE(o72_printWizImage),
OPCODE(o72_getArrayDimSize),
/* 64 */
OPCODE(o72_getNumFreeArrays),
OPCODE(o6_stopObjectCode),
OPCODE(o6_stopObjectCode),
OPCODE(o6_endCutscene),
/* 68 */
OPCODE(o6_cutscene),
2004-09-22 02:07:14 +00:00
OPCODE(o6_invalid),
OPCODE(o6_freezeUnfreeze),
OPCODE(o80_cursorCommand),
/* 6C */
OPCODE(o6_breakHere),
OPCODE(o6_ifClassOfIs),
OPCODE(o6_setClass),
OPCODE(o6_getState),
/* 70 */
2004-09-06 01:54:24 +00:00
OPCODE(o80_setState),
OPCODE(o6_setOwner),
OPCODE(o6_getOwner),
OPCODE(o6_jump),
/* 74 */
2004-09-21 01:00:30 +00:00
OPCODE(o70_startSound),
OPCODE(o6_stopSound),
2004-10-09 05:15:12 +00:00
OPCODE(o6_invalid),
OPCODE(o6_stopObjectScript),
/* 78 */
OPCODE(o6_panCameraTo),
OPCODE(o6_actorFollowCamera),
OPCODE(o6_setCameraAt),
OPCODE(o6_loadRoom),
/* 7C */
OPCODE(o6_stopScript),
OPCODE(o6_walkActorToObj),
OPCODE(o6_walkActorTo),
OPCODE(o6_putActorAtXY),
/* 80 */
OPCODE(o6_putActorAtObject),
OPCODE(o6_faceActor),
OPCODE(o6_animateActor),
OPCODE(o6_doSentence),
/* 84 */
2004-09-21 01:00:30 +00:00
OPCODE(o70_pickupObject),
OPCODE(o6_loadRoomWithEgo),
OPCODE(o6_invalid),
OPCODE(o6_getRandomNumber),
/* 88 */
OPCODE(o6_getRandomNumberRange),
OPCODE(o6_invalid),
OPCODE(o6_getActorMoving),
OPCODE(o6_isScriptRunning),
/* 8C */
2004-09-21 01:00:30 +00:00
OPCODE(o70_getActorRoom),
OPCODE(o6_getObjectX),
OPCODE(o6_getObjectY),
OPCODE(o6_getObjectOldDir),
/* 90 */
OPCODE(o6_getActorWalkBox),
OPCODE(o6_getActorCostume),
OPCODE(o6_findInventory),
OPCODE(o6_getInventoryCount),
/* 94 */
OPCODE(o90_getPaletteData),
OPCODE(o6_beginOverride),
OPCODE(o6_endOverride),
OPCODE(o6_setObjectName),
/* 98 */
OPCODE(o6_isSoundRunning),
OPCODE(o6_setBoxFlags),
OPCODE(o6_invalid),
2004-09-21 01:00:30 +00:00
OPCODE(o70_resourceRoutines),
/* 9C */
OPCODE(o72_roomOps),
OPCODE(o72_actorOps),
OPCODE(o90_paletteOps),
OPCODE(o6_getActorFromXY),
/* A0 */
OPCODE(o70_findObject),
OPCODE(o6_pseudoRoom),
OPCODE(o6_getActorElevation),
OPCODE(o6_getVerbEntrypoint),
/* A4 */
OPCODE(o72_arrayOps),
2004-10-01 23:56:39 +00:00
OPCODE(o90_unknownA5),
OPCODE(o80_drawBox),
OPCODE(o6_pop),
/* A8 */
OPCODE(o6_getActorWidth),
2004-09-21 01:00:30 +00:00
OPCODE(o60_wait),
OPCODE(o6_getActorScaleX),
OPCODE(o90_getActorAnimProgress),
/* AC */
OPCODE(o80_drawWizPolygon),
OPCODE(o6_isAnyOf),
2004-09-21 01:00:30 +00:00
OPCODE(o70_quitPauseRestart),
OPCODE(o6_isActorInBox),
/* B0 */
OPCODE(o6_delay),
OPCODE(o6_delaySeconds),
OPCODE(o6_delayMinutes),
OPCODE(o6_stopSentence),
/* B4 */
OPCODE(o6_printLine),
OPCODE(o6_printCursor),
OPCODE(o6_printDebug),
OPCODE(o6_printSystem),
/* B8 */
OPCODE(o6_printActor),
OPCODE(o6_printEgo),
OPCODE(o6_talkActor),
OPCODE(o6_talkEgo),
/* BC */
OPCODE(o72_dimArray),
2004-09-15 12:40:49 +00:00
OPCODE(o6_stopObjectCode),
OPCODE(o6_startObjectQuick),
OPCODE(o6_startScriptQuick2),
/* C0 */
OPCODE(o72_dim2dimArray),
2004-09-21 01:00:30 +00:00
OPCODE(o72_traceStatus),
OPCODE(o6_invalid),
OPCODE(o6_invalid),
/* C4 */
OPCODE(o6_abs),
OPCODE(o6_distObjectObject),
OPCODE(o6_distObjectPt),
OPCODE(o6_distPtPt),
/* C8 */
2004-10-04 00:01:06 +00:00
OPCODE(o72_kernelGetFunctions),
2004-09-21 01:00:30 +00:00
OPCODE(o70_kernelSetFunctions),
OPCODE(o6_delayFrames),
OPCODE(o6_pickOneOf),
/* CC */
OPCODE(o6_pickOneOfDefault),
OPCODE(o6_stampObject),
OPCODE(o72_drawWizImage),
2004-10-02 00:12:09 +00:00
OPCODE(o72_unknownCF),
/* D0 */
OPCODE(o6_getDateTime),
OPCODE(o6_stopTalking),
OPCODE(o6_getAnimateVariable),
OPCODE(o6_invalid),
/* D4 */
2004-09-20 19:48:08 +00:00
OPCODE(o6_shuffle),
OPCODE(o72_jumpToScript),
OPCODE(o6_band),
OPCODE(o6_bor),
/* D8 */
OPCODE(o6_isRoomScriptRunning),
2004-09-21 01:00:30 +00:00
OPCODE(o60_closeFile),
OPCODE(o72_openFile),
OPCODE(o72_readFile),
/* DC */
OPCODE(o72_writeFile),
OPCODE(o72_findAllObjects),
OPCODE(o72_deleteFile),
2004-09-21 01:00:30 +00:00
OPCODE(o60_rename),
/* E0 */
2004-10-03 23:37:34 +00:00
OPCODE(o80_unknownE0),
OPCODE(o72_getPixel),
2004-10-10 05:49:30 +00:00
OPCODE(o60_localizeArrayToScript),
OPCODE(o80_pickVarRandom),
/* E4 */
OPCODE(o6_setBoxSet),
OPCODE(o6_invalid),
OPCODE(o6_invalid),
OPCODE(o6_invalid),
/* E8 */
OPCODE(o6_invalid),
2004-09-21 01:00:30 +00:00
OPCODE(o60_seekFilePos),
OPCODE(o72_redimArray),
2004-09-21 01:00:30 +00:00
OPCODE(o60_readFilePos),
/* EC */
OPCODE(o72_copyString),
2004-11-23 19:47:54 +00:00
OPCODE(o70_getStringWidth),
OPCODE(o70_getStringLen),
OPCODE(o72_appendString),
/* F0 */
OPCODE(o72_concatString),
OPCODE(o70_compareString),
2004-09-07 09:25:37 +00:00
OPCODE(o72_checkGlobQueue),
OPCODE(o72_readINI),
/* F4 */
OPCODE(o72_writeINI),
OPCODE(o70_getStringLenForWidth),
OPCODE(o70_getCharIndexInString),
OPCODE(o6_invalid),
/* F8 */
OPCODE(o72_getResourceSize),
2004-09-10 10:55:43 +00:00
OPCODE(o72_setFilePath),
2004-10-13 09:39:59 +00:00
OPCODE(o72_setWindowCaption),
2004-09-21 01:00:30 +00:00
OPCODE(o70_polygonOps),
/* FC */
2004-09-21 01:00:30 +00:00
OPCODE(o70_polygonHit),
OPCODE(o6_invalid),
OPCODE(o6_invalid),
OPCODE(o6_invalid),
};
_opcodesV90he = opcodes;
}
void ScummEngine_v90he::executeOpcode(byte i) {
OpcodeProcV90he op = _opcodesV90he[i].proc;
(this->*op) ();
}
const char *ScummEngine_v90he::getOpcodeDesc(byte i) {
return _opcodesV90he[i].desc;
}
void ScummEngine_v90he::o90_dup() {
int a, num;
num = fetchScriptWord();
for (int i = 0; i < num; i++) {
a = pop();
push(a);
push(a);
}
}
2004-10-08 17:56:08 +00:00
void ScummEngine_v90he::o90_min() {
2004-09-12 00:48:16 +00:00
int a = pop();
int b = pop();
if (b < a) {
push(b);
} else {
push(a);
}
}
2004-10-08 17:56:08 +00:00
void ScummEngine_v90he::o90_max() {
2004-09-12 00:48:16 +00:00
int a = pop();
int b = pop();
if (b > a) {
push(b);
} else {
push(a);
}
}
2004-10-03 19:57:19 +00:00
void ScummEngine_v90he::o90_sin() {
double a = pop() * PI / 180.;
2004-10-03 19:57:19 +00:00
push((int)(sin(a) * 100000));
}
2004-10-03 19:57:19 +00:00
void ScummEngine_v90he::o90_cos() {
double a = pop() * PI / 180.;
2004-10-03 19:57:19 +00:00
push((int)(cos(a) * 100000));
}
2004-10-08 17:56:08 +00:00
void ScummEngine_v90he::o90_sqrt() {
int i = pop();
if (i < 2) {
push(i);
} else {
2004-10-16 20:38:37 +00:00
push((int)sqrt((double)(i + 1)));
2004-10-08 17:56:08 +00:00
}
}
void ScummEngine_v90he::o90_atan2() {
int y = pop();
int x = pop();
2004-10-16 20:38:37 +00:00
int a = (int)(atan2((double)y, (double)x) * 180. / PI);
2004-10-08 17:56:08 +00:00
if (a < 0) {
a += 360;
}
push(a);
}
void ScummEngine_v90he::o90_getSegmentAngle() {
int y1 = pop();
int x1 = pop();
int dy = y1 - pop();
int dx = x1 - pop();
2004-10-16 20:38:37 +00:00
int a = (int)(atan2((double)dy, (double)dx) * 180. / PI);
2004-10-08 17:56:08 +00:00
if (a < 0) {
a += 360;
}
push(a);
}
2004-10-10 05:49:30 +00:00
void ScummEngine_v90he::o90_startScriptUnk() {
2004-12-19 09:27:08 +00:00
int args[25];
2004-10-10 05:49:30 +00:00
int script, cycle;
byte flags;
getStackList(args, ARRAYSIZE(args));
cycle = pop();
script = pop();
flags = fetchScriptByte();
runScript(script, (flags == 199 || flags == 200), (flags == 195 || flags == 200), args);
}
void ScummEngine_v90he::o90_jumpToScriptUnk() {
2004-12-19 09:27:08 +00:00
int args[25];
2004-10-10 05:49:30 +00:00
int script, cycle;
byte flags;
2004-09-12 00:48:16 +00:00
getStackList(args, ARRAYSIZE(args));
2004-10-10 05:49:30 +00:00
cycle = pop();
2004-09-12 00:48:16 +00:00
script = pop();
flags = fetchScriptByte();
2004-10-10 05:49:30 +00:00
stopObjectCode();
2004-09-12 00:48:16 +00:00
runScript(script, (flags == 199 || flags == 200), (flags == 195 || flags == 200), args);
}
void ScummEngine_v90he::o90_wizImageOps() {
int a, b;
2004-10-10 11:09:23 +00:00
int subOp = fetchScriptByte();
2004-09-06 08:53:19 +00:00
subOp -= 46;
2004-09-06 08:53:19 +00:00
switch (subOp) {
case -14: // HE99+
2004-10-08 22:58:49 +00:00
_wizParams.processFlags |= 0x20;
pop();
break;
case -13: // HE99+
2004-10-08 22:58:49 +00:00
_wizParams.processFlags |= 0x40;
pop();
break;
case 0:
pop();
break;
case 1:
_wizParams.box.bottom = pop();
_wizParams.box.right = pop();
_wizParams.box.top = pop();
_wizParams.box.left = pop();
break;
case 2:
2004-10-08 22:58:49 +00:00
_wizParams.processMode = 1;
break;
case 3:
2004-10-08 22:58:49 +00:00
_wizParams.processFlags |= 0x800;
_wizParams.processMode = 3;
copyScriptString(_wizParams.filename);
break;
case 4:
2004-10-08 22:58:49 +00:00
_wizParams.processFlags |= 0x800;
_wizParams.processMode = 4;
copyScriptString(_wizParams.filename);
_wizParams.unk_14C = pop();
break;
case 5:
2004-10-08 22:58:49 +00:00
_wizParams.processFlags |= 0x300;
_wizParams.processMode = 2;
_wizParams.box.bottom = pop();
_wizParams.box.right = pop();
_wizParams.box.top = pop();
_wizParams.box.left = pop();
_wizParams.compType = pop();
break;
case 6:
2004-10-08 22:58:49 +00:00
_wizParams.processFlags |= 0x400;
_wizParams.img.state = pop();
break;
case 7:
2004-10-08 22:58:49 +00:00
_wizParams.processFlags |= 0x10;
_wizParams.angle = pop();
break;
case 8:
2004-10-08 22:58:49 +00:00
_wizParams.processFlags |= 0x20;
_wizParams.img.flags = pop();
break;
case 10:
_wizParams.img.flags = pop();
_wizParams.img.state = pop();
_wizParams.img.y1 = pop();
_wizParams.img.x1 = pop();
_wizParams.img.resNum = pop();
displayWizImage(&_wizParams.img);
break;
case 11:
_wizParams.img.resNum = pop();
2004-10-08 22:58:49 +00:00
_wizParams.processMode = 0;
_wizParams.processFlags = 0;
_wizParams.remapNum = 0;
_wizParams.img.flags = 0;
break;
case 16: // HE99+
2004-10-08 22:58:49 +00:00
_wizParams.processFlags |= 0x80000;
pop();
break;
case 19:
2004-10-08 22:58:49 +00:00
_wizParams.processFlags |= 1;
_wizParams.img.y1 = pop();
_wizParams.img.x1 = pop();
break;
case 20:
case 203: // HE98+
b = pop();
a = pop();
2004-10-08 22:58:49 +00:00
_wizParams.processFlags |= 0x40;
_wizParams.processMode = 6;
if (_wizParams.remapNum == 0) {
memset(_wizParams.remapIndex, 0, sizeof(_wizParams.remapIndex));
} else {
assert(_wizParams.remapNum < ARRAYSIZE(_wizParams.remapIndex));
_wizParams.remapIndex[_wizParams.remapNum] = a;
_wizParams.remapColor[a] = b;
++_wizParams.remapNum;
}
break;
2004-09-21 01:00:30 +00:00
case 21:
2004-10-08 22:58:49 +00:00
_wizParams.processFlags |= 0x200;
_wizParams.box.bottom = pop();
_wizParams.box.right = pop();
_wizParams.box.top = pop();
_wizParams.box.left = pop();
2004-09-21 01:00:30 +00:00
break;
case 31: // HE99+
2004-10-08 22:58:49 +00:00
_wizParams.processFlags |= 2;
break;
2004-09-22 09:48:40 +00:00
case 40: // HE99+
2004-10-08 22:58:49 +00:00
_wizParams.processFlags |= 0x80;
2004-09-22 09:48:40 +00:00
pop();
break;
case 46:
2004-10-08 22:58:49 +00:00
_wizParams.processFlags |= 8;
_wizParams.zoom = pop();
break;
case 52:
2004-10-08 22:58:49 +00:00
_wizParams.processFlags |= 4;
_wizParams.unk_15C = pop();
break;
case 85: // HE99+
2004-10-08 22:58:49 +00:00
_wizParams.processFlags |= 0x1102;
_wizParams.processMode = 7;
2004-10-17 08:02:42 +00:00
pop();
pop();
_wizParams.compType = pop();
break;
case 87: // HE99+
2004-10-08 22:58:49 +00:00
_wizParams.processFlags |= 0x60000;
_wizParams.processMode = 9;
pop();
pop();
pop();
pop();
pop();
break;
case 88: // HE99+
2004-10-08 22:58:49 +00:00
_wizParams.processFlags |= 0x60000;
_wizParams.processMode = 10;
pop();
pop();
pop();
pop();
pop();
break;
case 89: // HE99+
2004-10-08 22:58:49 +00:00
_wizParams.processFlags |= 0x60000;
_wizParams.processMode = 11;
pop();
pop();
pop();
break;
case 90: // HE99+
2004-10-08 22:58:49 +00:00
_wizParams.processFlags |= 0x60000;
_wizParams.processMode = 12;
pop();
pop();
pop();
break;
case 91: // HE99+
2004-10-08 22:58:49 +00:00
_wizParams.processFlags |= 0x10000;
pop();
break;
case 108:
2004-10-08 22:58:49 +00:00
_wizParams.processFlags |= 1;
_wizParams.img.y1 = pop();
_wizParams.img.x1 = pop();
break;
case 171: // HE99+
2004-10-08 22:58:49 +00:00
_wizParams.processMode = 8;
break;
2004-09-20 09:16:49 +00:00
case 200:
2004-10-08 22:58:49 +00:00
_wizParams.processFlags |= 0x23;
_wizParams.img.flags |= 0x40;
_wizParams.unk_160 = _wizParams.img.y1 = _wizParams.img.x1 = pop();
2004-09-20 09:16:49 +00:00
break;
case 209:
2004-10-08 22:58:49 +00:00
processWizImage(&_wizParams);
break;
default:
error("o90_wizImageOps: unhandled case %d", subOp);
}
debug(1,"o90_wizImageOps stub (%d)", subOp);
}
void ScummEngine_v90he::o90_getDistanceBetweenPoints() {
2004-10-10 06:32:02 +00:00
byte subOp = fetchScriptByte();
int x1, y1, z1, x2, y2, z2, dx, dy, dz, d;
2004-10-10 06:32:02 +00:00
switch (subOp) {
case 28:
y2 = pop();
x2 = pop();
y1 = pop();
x1 = pop();
dx = x2 - x1;
dy = y2 - y1;
d = dx * dx + dy * dy;
if (d < 2) {
push(d);
} else {
push((int)sqrt((double)(d + 1)));
}
2004-10-10 06:32:02 +00:00
break;
case 29:
z2 = pop();
y2 = pop();
x2 = pop();
z1 = pop();
y1 = pop();
x1 = pop();
dx = x2 - x1;
dy = y2 - y1;
dz = z2 - z1;
d = dx * dx + dy * dy + dz * dz;
if (d < 2) {
push(d);
} else {
push((int)sqrt((double)(d + 1)));
}
2004-10-10 06:32:02 +00:00
break;
default:
error("o90_getDistanceBetweenPoints: Unknown case %d", subOp);
2004-10-10 06:32:02 +00:00
}
}
void ScummEngine_v90he::o90_unknown25() {
int args[16];
2004-10-10 00:24:15 +00:00
byte subOp = fetchScriptByte();
2004-09-12 00:48:16 +00:00
subOp -= 30;
debug(1,"o90_unknown25 stub (%d)", subOp);
switch (subOp) {
case 0:
pop();
break;
case 1:
pop();
break;
case 2:
pop();
break;
case 3:
pop();
break;
case 4:
pop();
break;
case 5:
pop();
break;
case 6:
pop();
break;
case 7:
pop();
break;
case 8:
pop();
break;
case 9:
pop();
break;
case 12:
pop();
pop();
break;
case 13:
pop();
break;
case 15:
if (_heversion == 99) {
getStackList(args, ARRAYSIZE(args));
2004-09-12 00:48:16 +00:00
pop();
pop();
pop();
pop();
} else if (_heversion == 98) {
pop();
pop();
pop();
pop();
} else {
pop();
2004-09-12 00:48:16 +00:00
pop();
pop();
}
break;
case 22:
pop();
break;
case 33:
pop();
break;
case 38:
pop();
break;
case 52:
pop();
break;
case 62:
pop();
break;
case 67:
pop();
break;
case 68:
pop();
break;
case 94:
pop();
break;
case 95:
getStackList(args, ARRAYSIZE(args));
pop();
break;
case 168:
pop();
pop();
break;
default:
error("o90_unknown25: Unknown case %d", subOp);
}
push(0);
}
void ScummEngine_v90he::o90_unknown26() {
2004-09-07 12:56:12 +00:00
int args[16];
2004-10-10 00:24:15 +00:00
byte subOp = fetchScriptByte();
subOp -= 34;
switch (subOp) {
case 0:
pop();
break;
case 1:
pop();
break;
case 3:
pop();
break;
case 8:
pop();
pop();
break;
case 9:
pop();
break;
case 10:
pop();
pop();
break;
case 18:
pop();
break;
case 19:
pop();
break;
case 23:
pop();
if (_gameId == GID_FREDDI4 || _heversion >= 99)
2004-09-07 03:42:15 +00:00
pop();
break;
case 28: // HE99+
pop();
break;
case 29:
pop();
break;
case 31:
pop();
pop();
break;
case 34:
pop();
break;
case 43:
pop();
pop();
break;
case 48:
pop();
break;
case 52: // HE 98+
pop();
break;
2004-10-02 07:46:06 +00:00
case 58: // HE 99+
pop();
break;
case 63: // HE 98+
pop();
break;
case 64:
pop();
break;
case 90:
pop();
break;
case 91:
getStackList(args, ARRAYSIZE(args));
break;
2004-10-10 06:32:02 +00:00
case 105: // HE 99+
pop();
pop();
2004-10-10 06:32:02 +00:00
break;
case 106: // HE 99+
pop();
break;
case 124:
break;
case 164:
pop();
pop();
break;
case 183:
break;
default:
error("o90_unknown26: Unknown case %d", subOp);
2004-09-06 01:54:24 +00:00
}
2004-09-07 03:42:15 +00:00
debug(1,"o90_unknown26 stub (%d)", subOp);
}
2004-09-06 08:53:19 +00:00
void ScummEngine_v90he::o90_unknown27() {
2004-10-10 00:24:15 +00:00
byte subOp = fetchScriptByte();
2004-09-06 08:53:19 +00:00
switch (subOp) {
2004-09-30 10:24:26 +00:00
case 8: // HE 99+
pop();
break;
case 30:
pop();
break;
case 31:
pop();
break;
2004-09-30 10:24:26 +00:00
case 42: // HE 99+
pop();
pop();
break;
case 43:
pop();
break;
2004-09-30 10:24:26 +00:00
case 63: // HE 99+
pop();
break;
case 139: // HE 99+
pop();
pop();
break;
default:
error("o90_unknown27: Unknown case %d", subOp);
2004-09-06 08:53:19 +00:00
}
push(0);
debug(1,"o90_unknown27 stub (%d)", subOp);
}
2004-09-06 02:39:02 +00:00
void ScummEngine_v90he::o90_unknown28() {
2004-10-10 00:24:15 +00:00
byte subOp = fetchScriptByte();
subOp -= 37;
2004-09-06 02:39:02 +00:00
switch (subOp) {
2004-10-17 09:17:32 +00:00
case 0:
switch (pop()) {
case 1:
pop();
pop();
break;
case 2:
pop();
break;
case 3:
pop();
break;
case 4:
pop();
break;
case 5:
break;
case 6:
pop();
break;
case 7:
pop();
break;
case 8:
pop();
break;
default:
2004-10-17 09:29:52 +00:00
error("o90_unknown28 subOp 0: Unknown case %d", subOp);
2004-10-17 09:17:32 +00:00
}
break;
2004-10-10 06:32:02 +00:00
case 5:
pop();
pop();
break;
case 6:
pop();
break;
case 7:
pop();
pop();
break;
case 20:
pop();
break;
2004-10-17 09:17:32 +00:00
case 26:
pop();
break;
case 28:
pop();
pop();
break;
case 30:
pop();
pop();
pop();
pop();
break;
2004-10-17 09:17:32 +00:00
case 56:
break;
case 180:
break;
default:
error("o90_unknown28: Unknown case %d", subOp);
2004-09-06 02:39:02 +00:00
}
2004-09-07 03:42:15 +00:00
debug(1,"o90_unknown28 stub (%d)", subOp);
}
void ScummEngine_v90he::o90_getWizData() {
2004-09-22 00:22:32 +00:00
int state, resId;
2004-10-16 20:38:37 +00:00
int32 w, h;
int16 x, y;
2004-09-22 00:22:32 +00:00
2004-10-10 00:24:15 +00:00
byte subOp = fetchScriptByte();
2004-09-07 03:42:15 +00:00
subOp -= 30;
switch (subOp) {
case 0:
2004-09-22 00:22:32 +00:00
state = pop();
resId = pop();
loadImgSpot(resId, state, x, y);
push(x);
break;
case 1:
2004-09-22 00:22:32 +00:00
state = pop();
resId = pop();
loadImgSpot(resId, state, x, y);
push(y);
break;
case 2:
2004-09-22 00:22:32 +00:00
state = pop();
resId = pop();
getWizImageDim(resId, state, w, h);
push(w);
break;
case 3:
2004-09-22 00:22:32 +00:00
state = pop();
resId = pop();
getWizImageDim(resId, state, w, h);
push(h);
break;
case 6:
resId = pop();
push(getWizImageStates(resId));
break;
case 15:
2004-11-21 21:31:28 +00:00
y = pop();
x = pop();
state = pop();
resId = pop();
push(isWizPixelNonTransparent(rtImage, resId, state, x, y, 0));
break;
case 36:
2004-11-21 21:31:28 +00:00
y = pop();
x = pop();
state = pop();
resId = pop();
push(getWizPixelColor(rtImage, resId, state, x, y, 0));
break;
case 100:
h = pop();
w = pop();
y = pop();
x = pop();
state = pop();
resId = pop();
if (x == -1 && y == -1 && w == -1 && h == -1) {
getWizImageDim(resId, state, w, h);
x = 0;
y = 0;
}
push(computeWizHistogram(resId, state, x, y, w, h));
break;
2004-11-13 04:05:35 +00:00
case 109:
pop();
pop();
push(0);
break;
default:
error("o90_getWizData: Unknown case %d", subOp);
2004-09-07 03:42:15 +00:00
}
2004-09-06 02:39:02 +00:00
}
2004-09-10 14:31:20 +00:00
void ScummEngine_v90he::o90_unknown2F() {
2004-10-10 00:24:15 +00:00
byte subOp = fetchScriptByte();
2004-09-10 14:31:20 +00:00
subOp -= 54;
switch (subOp) {
case 0:
pop();
break;
case 3:
break;
case 11:
pop();
pop();
break;
case 12:
pop();
break;
case 13:
pop();
pop();
pop();
pop();
break;
case 201:
break;
default:
2004-10-04 01:34:29 +00:00
error("o90_unknown2F: Unknown case %d", subOp);
2004-09-10 14:31:20 +00:00
}
debug(1,"o90_unknown2F stub (%d)", subOp);
}
2004-11-28 05:33:33 +00:00
void ScummEngine_v90he::o90_shl() {
int a = pop();
push(pop() << a);
}
2004-11-28 05:33:33 +00:00
void ScummEngine_v90he::o90_shr() {
int a = pop();
push(pop() >> a);
}
void ScummEngine_v90he::o90_mod() {
2004-09-10 14:31:20 +00:00
int a = pop();
if (a == 0)
error("modulus by zero");
push(pop() % a);
}
void ScummEngine_v90he::o90_findAllObjectsWithClassOf() {
int args[16];
int cond, num, cls, tmp;
bool b;
num = getStackList(args, ARRAYSIZE(args));
int room = pop();
2004-10-03 05:34:48 +00:00
int j = 1;
if (room != _currentRoom)
warning("o90_findAllObjectsWithClassOf: current room is not %d", room);
writeVar(0, 0);
defineArray(0, kDwordArray, 0, 0, 0, _numLocalObjects + 1);
2004-10-03 05:34:48 +00:00
for (int i = 1; i < _numLocalObjects; i++) {
cond = 1;
tmp = num;
while (--tmp >= 0) {
cls = args[tmp];
b = getClass(i, cls);
if ((cls & 0x80 && !b) || (!(cls & 0x80) && b))
cond = 0;
}
if (cond)
2004-10-03 05:34:48 +00:00
writeArray(0, 0, j++, _objs[i].obj_nr);
}
2004-10-03 05:34:48 +00:00
writeArray(0, 0, 0, j);
push(readVar(0));
2004-09-10 14:31:20 +00:00
}
void ScummEngine_v90he::o90_getPolygonOverlap() {
int args1[32];
2004-09-30 09:54:45 +00:00
int args2[32];
int n1 = getStackList(args1, ARRAYSIZE(args1));
2004-09-30 09:54:45 +00:00
getStackList(args2, ARRAYSIZE(args2));
int subOp = pop();
switch (subOp) {
case 1:
{
Common::Rect r(args1[0], args1[1], args1[2] + 1, args1[3] + 1);
Common::Point p(args2[0], args2[1]);
push(r.contains(p) ? 1 : 0);
}
break;
2004-09-30 09:54:45 +00:00
case 2:
{
int dx = args2[0] - args1[0];
int dy = args2[1] - args1[1];
int dist = dx * dx + dy * dy;
if (dist >= 2) {
2004-10-16 20:38:37 +00:00
dist = (int)sqrt((double)(dist + 1));
}
push((dist > args1[2]) ? 1 : 0);
}
break;
2004-09-30 09:54:45 +00:00
case 3:
{
Common::Rect r1(args1[0], args1[1], args1[2] + 1, args1[3] + 1);
Common::Rect r2(args2[0], args2[1], args2[2] + 1, args2[3] + 1);
push(r2.intersects(r1) ? 1 : 0);
}
break;
2004-09-30 09:54:45 +00:00
case 4:
{
int dx = args2[0] - args1[0];
int dy = args2[1] - args1[1];
int dist = dx * dx + dy * dy;
if (dist >= 2) {
2004-10-16 20:38:37 +00:00
dist = (int)sqrt((double)(dist + 1));
}
push((dist < args1[2] && dist < args2[2]) ? 1 : 0);
}
break;
2004-09-30 09:54:45 +00:00
case 5:
{
assert((n1 & 1) == 0);
n1 /= 2;
if (n1 == 0) {
push(0);
} else {
WizPolygon wp;
memset(&wp, 0, sizeof(wp));
wp.numVerts = n1;
assert(n1 < ARRAYSIZE(wp.vert));
for (int i = 0; i < n1; ++i) {
wp.vert[i].x = args1[i * 2 + 0];
wp.vert[i].y = args1[i * 2 + 1];
}
push(_wiz.polygonContains(wp, args2[0], args2[1]) ? 1 : 0);
}
}
break;
2004-09-30 09:54:45 +00:00
// HE 98+
case 6:
case 7:
case 8:
case 9:
push(0);
2004-09-30 09:54:45 +00:00
break;
default:
error("o90_getPolygonOverlap: default case %d", subOp);
2004-09-30 09:54:45 +00:00
}
}
void ScummEngine_v90he::o90_cond() {
2004-09-12 11:15:26 +00:00
int a = pop();
int b = pop();
int c = pop();
if (!c)
b = a;
push(b);
}
2004-10-04 01:34:29 +00:00
void ScummEngine_v90he::o90_dim2dim2Array() {
2004-09-07 12:56:12 +00:00
int data, dim1start, dim1end, dim2start, dim2end;
int type = fetchScriptByte();
switch (type) {
case 2: // SO_BIT_ARRAY
data = kBitArray;
break;
case 3: // SO_NIBBLE_ARRAY
data = kNibbleArray;
break;
case 4: // SO_BYTE_ARRAY
data = kByteArray;
break;
case 5: // SO_INT_ARRAY
data = kIntArray;
break;
case 6:
data = kDwordArray;
break;
case 7: // SO_STRING_ARRAY
data = kStringArray;
break;
default:
2004-10-04 01:34:29 +00:00
error("o90_dim2dim2Array: default case %d", type);
2004-09-07 12:56:12 +00:00
}
if (pop() == 2) {
dim1end = pop();
dim1start = pop();
dim2end = pop();
dim2start = pop();
} else {
dim2end = pop();
dim2start = pop();
dim1end = pop();
dim1start = pop();
}
defineArray(fetchScriptWord(), data, dim2start, dim2end, dim1start, dim1end);
}
void ScummEngine_v90he::o90_sortArray() {
// Sorts array via qsort
2004-10-10 00:24:15 +00:00
byte subOp = fetchScriptByte();
2004-09-13 05:25:35 +00:00
switch (subOp) {
case 129:
fetchScriptWord();
pop();
pop();
pop();
pop();
pop();
break;
default:
error("o90_sortArray: Unknown case %d", subOp);
2004-09-13 05:25:35 +00:00
}
debug(1,"o90_sortArray stub (%d)", subOp);
2004-09-13 05:25:35 +00:00
}
void ScummEngine_v90he::o90_getObjectData() {
2004-10-02 01:40:34 +00:00
// Object releated
2004-10-10 00:24:15 +00:00
byte subOp = fetchScriptByte();
2004-10-02 01:40:34 +00:00
subOp -= 32;
switch (subOp) {
case 0:
if (_heObjectNum == -1)
push(0);
else
push(_objs[_heObjectNum].width);
break;
case 1:
if (_heObjectNum == -1)
push(0);
else
push(_objs[_heObjectNum].height);
break;
case 4:
push(0);
break;
case 6:
if (_heObjectNum == -1)
push(0);
else
push(_objs[_heObjectNum].x_pos);
break;
case 7:
if (_heObjectNum == -1)
push(0);
else
push(_objs[_heObjectNum].y_pos);
break;
case 20:
push(getState(_heObject));
break;
case 25:
_heObject = pop();
_heObjectNum = getObjectIndex(_heObject);
break;
case 107:
// Dummy case
pop();
push(0);
break;
default:
error("o90_getObjectData: Unknown case %d", subOp);
2004-10-02 01:40:34 +00:00
}
debug(1,"o90_getObjectData stub (%d)", subOp);
2004-10-02 01:40:34 +00:00
}
void ScummEngine_v90he::o90_getPaletteData() {
2004-10-10 00:24:15 +00:00
byte subOp = fetchScriptByte();
2004-09-18 04:12:14 +00:00
subOp -= 45;
switch (subOp) {
case 0:
pop();
pop();
pop();
pop();
pop();
pop();
break;
case 7:
pop();
pop();
pop();
break;
case 21:
pop();
pop();
break;
case 87:
pop();
pop();
break;
case 172:
pop();
pop();
pop();
break;
default:
error("o90_getPaletteData: Unknown case %d", subOp);
2004-09-18 04:12:14 +00:00
}
push(0);
debug(1,"o90_getPaletteData stub (%d)", subOp);
2004-09-18 04:12:14 +00:00
}
void ScummEngine_v90he::o90_paletteOps() {
int idx, state;
byte subOp = fetchScriptByte();
subOp -= 57;
switch (subOp) {
case 0:
_hePaletteNum = pop();
break;
case 6:
{
state = pop();
idx = pop();
const uint8 *dataPtr = getResourceAddress(rtImage, idx);
const uint8 *pal = findWrappedBlock(MKID('RGBS'), dataPtr, state, 0);
assert(pal);
}
break;
case 9:
pop();
pop();
pop();
pop();
pop();
break;
case 13:
pop();
pop();
pop();
break;
case 19: //HE99+
pop();
break;
case 29:
pop();
break;
case 118:
pop();
pop();
break;
case 160:
break;
case 198:
_hePaletteNum = 0;
break;
default:
error("o90_paletteOps: Unknown case %d", subOp);
}
debug(0,"o90_paletteOps stub (%d)", subOp);
}
2004-10-01 23:56:39 +00:00
void ScummEngine_v90he::o90_unknownA5() {
2004-10-24 13:43:36 +00:00
// Font related
byte string[80];
int a;
2004-10-10 00:24:15 +00:00
byte subOp = fetchScriptByte();
2004-10-01 23:56:39 +00:00
switch (subOp) {
case 42:
a = pop();
2004-10-24 13:43:36 +00:00
if (a == 2) {
copyScriptString(string);
2004-10-24 13:43:36 +00:00
push(-1);
} else if (a == 1) {
2004-10-01 23:56:39 +00:00
pop();
2004-10-24 13:43:36 +00:00
writeVar(0, 0);
defineArray(0, kStringArray, 0, 0, 0, 0);
writeArray(0, 0, 0, 0);
push(readVar(0));
}
2004-10-01 23:56:39 +00:00
break;
case 57:
2004-10-24 13:43:36 +00:00
push(1);
2004-10-01 23:56:39 +00:00
break;
default:
error("o90_unknownA5: Unknown case %d", subOp);
}
debug(1,"o90_unknownA5 stub (%d)", subOp);
}
void ScummEngine_v90he::o90_getActorAnimProgress() {
Actor *a = derefActor(pop(), "o90_getActorAnimProgress");
push(a->getAnimProgress());
}
} // End of namespace Scumm