scummvm/engines/scumm/script_v4.cpp
AndywinXp ba24698225 SCUMM: v3: Implement ability to rename savestate before loading on the original menu
As much as this seems bad design, it's a real possibility in the original interpreters, furthermore
confirmed by the menu messages which are shown when the user tries to do that, so here we are...
2022-12-01 00:04:46 +01:00

523 lines
14 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 3 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, see <http://www.gnu.org/licenses/>.
*
*/
#include "scumm/scumm_v4.h"
#include "scumm/object.h"
namespace Scumm {
#define OPCODE(i, x) _opcodes[i]._OPCODE(ScummEngine_v4, x)
void ScummEngine_v4::setupOpcodes() {
ScummEngine_v5::setupOpcodes();
OPCODE(0x25, o5_drawObject);
OPCODE(0x45, o5_drawObject);
OPCODE(0x65, o5_drawObject);
OPCODE(0xa5, o5_drawObject);
OPCODE(0xc5, o5_drawObject);
OPCODE(0xe5, o5_drawObject);
OPCODE(0x50, o4_pickupObject);
OPCODE(0xd0, o4_pickupObject);
OPCODE(0x5c, o4_oldRoomEffect);
OPCODE(0xdc, o4_oldRoomEffect);
OPCODE(0x0f, o4_ifState);
OPCODE(0x4f, o4_ifState);
OPCODE(0x8f, o4_ifState);
OPCODE(0xcf, o4_ifState);
OPCODE(0x2f, o4_ifNotState);
OPCODE(0x6f, o4_ifNotState);
OPCODE(0xaf, o4_ifNotState);
OPCODE(0xef, o4_ifNotState);
OPCODE(0xa7, o4_saveLoadVars);
OPCODE(0x22, o4_saveLoadGame);
OPCODE(0xa2, o4_saveLoadGame);
// Disable some opcodes which are unused in v4.
_opcodes[0x3b].setProc(nullptr, nullptr);
_opcodes[0x4c].setProc(nullptr, nullptr);
_opcodes[0xbb].setProc(nullptr, nullptr);
}
void ScummEngine_v4::o4_ifState() {
int a = getVarOrDirectWord(PARAM_1);
int b = getVarOrDirectByte(PARAM_2);
// WORKAROUND bug #5709 (also occurs in original): Some old versions of
// Indy3 sometimes fail to allocate IQ points correctly. To quote:
// "About the points error leaving Castle Brunwald: It seems to "reversed"!
// When you get caught, free yourself and escape, you get 25 IQ points even
// though you're not supposed to. However if you escape WITHOUT getting
// caught, you get 0 IQ points (supposed to get 25 IQ points)."
// This workaround is meant to address that.
if (_game.id == GID_INDY3 && a == 367 &&
vm.slot[_currentScript].number == 363 && _currentRoom == 25) {
b = 0;
}
jumpRelative(getState(a) == b);
}
void ScummEngine_v4::o4_ifNotState() {
int a = getVarOrDirectWord(PARAM_1);
int b = getVarOrDirectByte(PARAM_2);
jumpRelative(getState(a) != b);
}
void ScummEngine_v4::o4_pickupObject() {
int obj = getVarOrDirectWord(PARAM_1);
if (obj < 1) {
error("pickupObjectOld received invalid index %d (script %d)", obj, vm.slot[_currentScript].number);
}
if (getObjectIndex(obj) == -1)
return;
if (whereIsObject(obj) == WIO_INVENTORY) // Don't take an object twice
return;
// debug(0, "adding %d from %d to inventoryOld", obj, _currentRoom);
addObjectToInventory(obj, _roomResource);
markObjectRectAsDirty(obj);
putOwner(obj, VAR(VAR_EGO));
putClass(obj, kObjectClassUntouchable, 1);
putState(obj, 1);
clearDrawObjectQueue();
runInventoryScript(1);
}
void ScummEngine_v4::o4_oldRoomEffect() {
int a;
_opcode = fetchScriptByte();
if ((_opcode & 0x1F) == 3) {
a = getVarOrDirectWord(PARAM_1);
if (_game.platform == Common::kPlatformFMTowns && _game.version == 3) {
if (a == 4) {
_textSurface.fillRect(Common::Rect(0, 0, _textSurface.w * _textSurfaceMultiplier, _textSurface.h * _textSurfaceMultiplier), 0);
#ifndef DISABLE_TOWNS_DUAL_LAYER_MODE
if (_townsScreen)
_townsScreen->clearLayer(1);
#endif
return;
}
}
if (a) {
_switchRoomEffect = (byte)(a & 0xFF);
_switchRoomEffect2 = (byte)(a >> 8);
} else {
fadeIn(_newEffect);
}
}
}
void ScummEngine_v4::o4_saveLoadVars() {
if (fetchScriptByte() == 1)
saveVars();
else
loadVars();
}
enum StringIds {
// The string IDs used by Indy3 to store the episode resp. series IQ points.
// Note that we save the episode IQ points but load the series IQ points,
// which matches the original Indy3 save/load code. See also the notes
// on bug #7547.
STRINGID_IQ_EPISODE = 7,
STRINGID_IQ_SERIES = 9,
// The string IDs of the first savegame name, used as an offset to determine
// the IDs of all savenames.
// Loom is the only game whose savenames start with a different ID.
STRINGID_SAVENAME1 = 10,
STRINGID_SAVENAME1_LOOM = 9
};
void ScummEngine_v4::saveVars() {
int a, b;
while ((_opcode = fetchScriptByte()) != 0) {
switch (_opcode & 0x1F) {
case 0x01: // write a range of variables
getResultPos();
a = _resultVarNumber;
getResultPos();
b = _resultVarNumber;
debug(0, "stub saveVars: vars %d -> %d", a, b);
break;
case 0x02: // write a range of string variables
a = getVarOrDirectByte(PARAM_1);
b = getVarOrDirectByte(PARAM_2);
if (a == STRINGID_IQ_EPISODE && b == STRINGID_IQ_EPISODE) {
if (_game.id == GID_INDY3) {
saveIQPoints();
}
break;
}
// FIXME: changing savegame-names not supported
break;
case 0x03: // open file
a = resStrLen(_scriptPointer);
strncpy(_saveLoadVarsFilename, (const char *)_scriptPointer, a);
_saveLoadVarsFilename[a] = '\0';
_scriptPointer += a + 1;
break;
case 0x04:
return;
case 0x1F: // close file
_saveLoadVarsFilename[0] = '\0';
return;
default:
break;
}
}
}
void ScummEngine_v4::loadVars() {
int a, b;
while ((_opcode = fetchScriptByte()) != 0) {
switch (_opcode & 0x1F) {
case 0x01: // read a range of variables
getResultPos();
a = _resultVarNumber;
getResultPos();
b = _resultVarNumber;
debug(0, "stub loadVars: vars %d -> %d", a, b);
break;
case 0x02: // read a range of string variables
a = getVarOrDirectByte(PARAM_1);
b = getVarOrDirectByte(PARAM_2);
int slot;
int slotSize;
byte *slotContent;
int savegameId;
bool availSaves[100];
if (a == STRINGID_IQ_SERIES && b == STRINGID_IQ_SERIES) {
// Zak256 loads the IQ script-slot but does not use it -> ignore it
if (_game.id == GID_INDY3) {
byte *ptr = getResourceAddress(rtString, STRINGID_IQ_SERIES);
if (ptr) {
int size = getResourceSize(rtString, STRINGID_IQ_SERIES);
loadIQPoints(ptr, size);
}
}
break;
}
listSavegames(availSaves, ARRAYSIZE(availSaves));
for (slot = a; slot <= b; ++slot) {
slotSize = getResourceSize(rtString, slot);
slotContent = getResourceAddress(rtString, slot);
// load savegame names
savegameId = slot - a + 1;
Common::String name;
if (availSaves[savegameId] && getSavegameName(savegameId, name)) {
int pos;
const char *ptr = name.c_str();
// slotContent ends with {'\0','@'} -> max. length = slotSize-2
for (pos = 0; pos < slotSize - 2; ++pos) {
if (!ptr[pos])
break;
// replace special characters
if (ptr[pos] >= 32 && ptr[pos] <= 122 && ptr[pos] != 64)
slotContent[pos] = ptr[pos];
else
slotContent[pos] = '_';
}
slotContent[pos] = '\0';
} else {
slotContent[0] = '\0';
}
}
break;
case 0x03: // open file
a = resStrLen(_scriptPointer);
strncpy(_saveLoadVarsFilename, (const char *)_scriptPointer, a);
_saveLoadVarsFilename[a] = '\0';
_scriptPointer += a + 1;
break;
case 0x04:
return;
case 0x1F: // close file
_saveLoadVarsFilename[0] = '\0';
return;
default:
break;
}
}
}
/**
* IQ Point calculation for Indy3.
* The scripts that perform this task are
* - script-9 (save/load dialog initialization, loads room 14),
* - room-14-204 (load series IQ string),
* - room-14-205 (save series IQ string),
* - room-14-206 (calculate series IQ string).
* Unfortunately script-9 contains lots of GUI stuff so calling this script
* directly is not possible. The other scripts depend on script-9.
*/
void ScummEngine_v4::updateIQPoints() {
int seriesIQ;
// IQString[0..72] corresponds to each puzzle's IQ.
// IQString[73] indicates that the IQ-file was loaded successfully and is always 0 when
// the IQ is calculated, hence it will be ignored here.
const int NUM_PUZZLES = 73;
byte seriesIQString[NUM_PUZZLES];
byte *episodeIQString;
int episodeIQStringSize;
// load string with IQ points given per puzzle in any savegame
// IMPORTANT: the resource string STRINGID_IQ_SERIES is only valid while
// the original save/load dialog is executed, so do not use it here.
memset(seriesIQString, 0, sizeof(seriesIQString));
loadIQPoints(seriesIQString, sizeof(seriesIQString));
// string with IQ points given per puzzle in current savegame
episodeIQString = getResourceAddress(rtString, STRINGID_IQ_EPISODE);
if (!episodeIQString)
return;
episodeIQStringSize = getResourceSize(rtString, STRINGID_IQ_EPISODE);
if (episodeIQStringSize < NUM_PUZZLES)
return;
// merge episode and series IQ strings and calculate series IQ
seriesIQ = 0;
// iterate over puzzles
for (int i = 0; i < NUM_PUZZLES; ++i) {
byte puzzleIQ = seriesIQString[i];
// if puzzle is solved copy points to episode string
if (puzzleIQ > 0)
episodeIQString[i] = puzzleIQ;
// add puzzle's IQ-points to series IQ
seriesIQ += episodeIQString[i];
}
_scummVars[245] = seriesIQ;
// save series IQ string
saveIQPoints();
}
void ScummEngine_v4::saveIQPoints() {
// save Indy3 IQ-points
Common::OutSaveFile *file;
Common::String filename = _targetName + ".iq";
file = _saveFileMan->openForSaving(filename);
if (file != nullptr) {
byte *ptr = getResourceAddress(rtString, STRINGID_IQ_EPISODE);
if (ptr) {
int size = getResourceSize(rtString, STRINGID_IQ_EPISODE);
file->write(ptr, size);
}
delete file;
}
}
void ScummEngine_v4::loadIQPoints(byte *ptr, int size) {
// load Indy3 IQ-points
Common::InSaveFile *file;
Common::String filename = _targetName + ".iq";
file = _saveFileMan->openForLoading(filename);
if (file != nullptr) {
byte *tmp = (byte *)malloc(size);
int nread = file->read(tmp, size);
if (nread == size) {
memcpy(ptr, tmp, size);
}
free(tmp);
delete file;
}
}
void ScummEngine_v4::o4_saveLoadGame() {
getResultPos();
byte slot;
byte a = getVarOrDirectByte(PARAM_1);
byte result = 0;
Common::String dummyName;
int saveRoom = 50;
if (_game.id == GID_INDY3)
saveRoom = 14;
else if (_game.id == GID_LOOM)
saveRoom = 70;
_mainMenuIsActive = true;
if ((_game.id == GID_MANIAC && _game.version <= 1) || (_game.id == GID_ZAK && _game.platform == Common::kPlatformC64)) {
// Convert V0/V1 load/save screen (they support only one savegame per disk)
// 1 Load
// 2 Save
slot = 1;
switch (a) {
case 1:
_opcode = 0x40;
break;
case 2:
_opcode = 0x80;
break;
default:
error("o4_saveLoadGame: unknown param %d", a);
}
} else {
slot = a & 0x1F;
// Slot numbers in older games start with 0, in newer games with 1
if (_game.version <= 2)
slot++;
_opcode = a & 0xE0;
}
switch (_opcode) {
case 0x00: // Num slots available
result = 100;
break;
case 0x20: // Drive
if (_game.version <= 3) {
// 0 = ???
// [1,2] = Disk drive [A:,B:]
// 3 = Hard drive
result = 3;
} else {
// Set current drive
result = 1;
}
break;
case 0x40: // load
_lastLoadedRoom = -1;
// The original interpreter allowed you to change the name of the
// savegame before loading it. As weird as that is, let's allow it...
if (_game.version == 3) {
char *ptr;
int firstSlot = (_game.id == GID_LOOM) ? STRINGID_SAVENAME1_LOOM : STRINGID_SAVENAME1;
ptr = (char *)getStringAddress(slot + firstSlot - 1);
if (ptr) {
if (!changeSavegameName(slot, ptr)) {
warning("o4_saveLoadGame: Couldn't change savegame name");
}
}
}
if (loadState(slot, false))
result = 3; // Success
else
result = 5; // Failed to load
// If the loaded state loads a different room from the save menu room
// it means that we are loading a game saved from the GMM. To correctly
// handle this, we run the boot script, we reload the state, and then signal
// the ScummEngine_v3::scummLoop_handleSaveLoad() function that we need to
// perform the post load fixes.
if (result == 3 && _currentRoom != saveRoom) {
_loadFromLauncher = true;
runBootscript();
loadState(slot, false);
_mainMenuIsActive = false;
return;
}
break;
case 0x80: // Save
_lastLoadedRoom = -1;
if (_game.version <= 3) {
char name[32];
if (_game.version <= 2) {
// V2 and below use a hardcoded name for savestates
Common::sprintf_s(name, "Game %c", 'A' + slot - 1);
} else {
// Use the name entered by the user...
char* ptr;
int firstSlot = (_game.id == GID_LOOM) ? STRINGID_SAVENAME1_LOOM : STRINGID_SAVENAME1;
ptr = (char *)getStringAddress(slot + firstSlot - 1);
Common::strlcpy(name, ptr, sizeof(name));
}
_saveLoadDescription = name;
if (saveState(slot, false, dummyName))
result = 0;
else
result = 2;
} else {
result = 2; // Failed to save
}
break;
case 0xC0: // Test if the save file exists
{
Common::InSaveFile *file = nullptr;
bool availableSaves[100];
bool atLeastOneSaveExists = false;
listSavegames(availableSaves, ARRAYSIZE(availableSaves));
Common::String filename = makeSavegameName(slot, false);
for (int i = 0; i < ARRAYSIZE(availableSaves); i++) {
if (availableSaves[i]) {
atLeastOneSaveExists = true;
break;
}
}
if (availableSaves[slot] && (file = _saveFileMan->openForLoading(filename))) {
result = 6; // Save file exists
delete file;
} else if (_game.id == GID_LOOM && _game.platform == Common::kPlatformFMTowns && slot == 0 && atLeastOneSaveExists) {
// LOOM Towns explicitly sets the result to 6 if the selected slot is 0;
// also, it needs to have at least one savegame available, otherwise it would lead
// to the game reaching towards a non-existent string, and crashing as a consequence.
result = 6;
} else
result = (_game.id == GID_LOOM && _game.platform == Common::kPlatformFMTowns) ? 8 : 7; // Save file does not exist
}
break;
default:
error("o4_saveLoadGame: unknown subopcode %d", _opcode);
}
_mainMenuIsActive = false;
setResult(result);
// Did the script move? Update its state!
updateScriptPtr();
getScriptBaseAddress();
resetScriptPointer();
}
} // End of namespace Scumm