1654 lines
53 KiB
C++
1654 lines
53 KiB
C++
|
//
|
||
|
// Copyright 2020 Electronic Arts Inc.
|
||
|
//
|
||
|
// TiberianDawn.DLL and RedAlert.dll and corresponding source code 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.
|
||
|
|
||
|
// TiberianDawn.DLL and RedAlert.dll and corresponding source code is distributed
|
||
|
// in the hope that it will be useful, but with permitted additional restrictions
|
||
|
// under Section 7 of the GPL. See the GNU General Public License in LICENSE.TXT
|
||
|
// distributed with this program. You should have received a copy of the
|
||
|
// GNU General Public License along with permitted additional restrictions
|
||
|
// with this program. If not, see https://github.com/electronicarts/CnC_Remastered_Collection
|
||
|
|
||
|
/* $Header: /counterstrike/SAVELOAD.CPP 9 3/17/97 1:04a Steve_tall $ */
|
||
|
/***********************************************************************************************
|
||
|
*** C O N F I D E N T I A L --- W E S T W O O D S T U D I O S ***
|
||
|
***********************************************************************************************
|
||
|
* *
|
||
|
* Project Name : Command & Conquer *
|
||
|
* *
|
||
|
* File Name : SAVELOAD.CPP *
|
||
|
* *
|
||
|
* Programmer : Joe L. Bostic *
|
||
|
* *
|
||
|
* Start Date : August 23, 1994 *
|
||
|
* *
|
||
|
* Last Update : July 8, 1996 [JLB] *
|
||
|
* *
|
||
|
*---------------------------------------------------------------------------------------------*
|
||
|
* Functions: *
|
||
|
* Code_All_Pointers -- Code all pointers. *
|
||
|
* Decode_All_Pointers -- Decodes all pointers. *
|
||
|
* Get_Savefile_Info -- gets description, scenario #, house *
|
||
|
* Load_Game -- loads a saved game *
|
||
|
* Load_MPlayer_Values -- Loads multiplayer-specific values *
|
||
|
* Load_Misc_Values -- loads miscellaneous variables *
|
||
|
* MPlayer_Save_Message -- pops up a "saving..." message *
|
||
|
* Put_All -- Store all save game data to the pipe. *
|
||
|
* Reconcile_Players -- Reconciles loaded data with the 'Players' vector *
|
||
|
* Save_Game -- saves a game to disk *
|
||
|
* Save_MPlayer_Values -- Saves multiplayer-specific values *
|
||
|
* Save_Misc_Values -- saves miscellaneous variables *
|
||
|
* - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - */
|
||
|
|
||
|
#include "function.h"
|
||
|
#include "vortex.h"
|
||
|
#ifdef WIN32
|
||
|
#include "tcpip.h"
|
||
|
#include "ccdde.h"
|
||
|
|
||
|
//#include "WolDebug.h"
|
||
|
|
||
|
extern bool DLLSave(Pipe &file);
|
||
|
extern bool DLLLoad(Straw &file);
|
||
|
|
||
|
extern bool SpawnedFromWChat;
|
||
|
#endif
|
||
|
|
||
|
//#define SAVE_BLOCK_SIZE 512
|
||
|
#define SAVE_BLOCK_SIZE 4096
|
||
|
//#define SAVE_BLOCK_SIZE 1024
|
||
|
|
||
|
/*
|
||
|
********************************** Defines **********************************
|
||
|
*/
|
||
|
#define SAVEGAME_VERSION (DESCRIP_MAX + \
|
||
|
0x01000006 + ( \
|
||
|
sizeof(AircraftClass) + \
|
||
|
sizeof(AircraftTypeClass) + \
|
||
|
sizeof(AnimClass) + \
|
||
|
sizeof(AnimTypeClass) + \
|
||
|
sizeof(BaseClass) + \
|
||
|
sizeof(BuildingClass) + \
|
||
|
sizeof(BuildingTypeClass) + \
|
||
|
sizeof(BulletClass) + \
|
||
|
sizeof(BulletTypeClass) + \
|
||
|
sizeof(CellClass) + \
|
||
|
sizeof(FactoryClass) + \
|
||
|
sizeof(HouseClass) + \
|
||
|
sizeof(HouseTypeClass) + \
|
||
|
sizeof(InfantryClass) + \
|
||
|
sizeof(InfantryTypeClass) + \
|
||
|
sizeof(LayerClass) + \
|
||
|
sizeof(MouseClass) + \
|
||
|
sizeof(OverlayClass) + \
|
||
|
sizeof(OverlayTypeClass) + \
|
||
|
sizeof(SmudgeClass) + \
|
||
|
sizeof(SmudgeTypeClass) + \
|
||
|
sizeof(TeamClass) + \
|
||
|
sizeof(TeamTypeClass) + \
|
||
|
sizeof(TemplateClass) + \
|
||
|
sizeof(TemplateTypeClass) + \
|
||
|
sizeof(TerrainClass) + \
|
||
|
sizeof(TerrainTypeClass) + \
|
||
|
sizeof(TriggerClass) + \
|
||
|
sizeof(TriggerTypeClass) + \
|
||
|
sizeof(UnitClass) + \
|
||
|
sizeof(UnitTypeClass) + \
|
||
|
sizeof(VesselClass) + \
|
||
|
sizeof(ScenarioClass) + \
|
||
|
sizeof(ChronalVortexClass)))
|
||
|
// sizeof(Waypoint)))
|
||
|
|
||
|
|
||
|
static int Reconcile_Players(void);
|
||
|
extern bool Is_Mission_Counterstrike (char *file_name);
|
||
|
#ifdef FIXIT_CSII // checked - ajw 9/28/98
|
||
|
extern bool Is_Mission_Aftermath (char *file_name);
|
||
|
#endif
|
||
|
|
||
|
/***********************************************************************************************
|
||
|
* Put_All -- Store all save game data to the pipe. *
|
||
|
* *
|
||
|
* This is the bulk processor of the game related save game data. All the game object *
|
||
|
* and state data is stored to the pipe specified. *
|
||
|
* *
|
||
|
* INPUT: pipe -- Reference to the pipe that will receive the save game data. *
|
||
|
* *
|
||
|
* OUTPUT: none *
|
||
|
* *
|
||
|
* WARNINGS: none *
|
||
|
* *
|
||
|
* HISTORY: *
|
||
|
* 07/08/1996 JLB : Created. *
|
||
|
*=============================================================================================*/
|
||
|
static void Put_All(Pipe & pipe, int save_net)
|
||
|
{
|
||
|
/*
|
||
|
** Save the scenario global information.
|
||
|
*/
|
||
|
pipe.Put(&Scen, sizeof(Scen));
|
||
|
|
||
|
/*
|
||
|
** Save the map. The map must be saved first, since it saves the Theater.
|
||
|
*/
|
||
|
if (!save_net) Call_Back();
|
||
|
Map.Save(pipe);
|
||
|
|
||
|
if (!save_net) Call_Back();
|
||
|
|
||
|
/*
|
||
|
** Save all game objects. This code saves every object that's stored in a
|
||
|
** TFixedIHeap class.
|
||
|
*/
|
||
|
Houses.Save(pipe);
|
||
|
if (!save_net) Call_Back();
|
||
|
TeamTypes.Save(pipe);
|
||
|
if (!save_net) Call_Back();
|
||
|
Teams.Save(pipe);
|
||
|
if (!save_net) Call_Back();
|
||
|
TriggerTypes.Save(pipe);
|
||
|
if (!save_net) Call_Back();
|
||
|
Triggers.Save(pipe);
|
||
|
if (!save_net) Call_Back();
|
||
|
Aircraft.Save(pipe);
|
||
|
if (!save_net) Call_Back();
|
||
|
Anims.Save(pipe);
|
||
|
|
||
|
if (!save_net) Call_Back();
|
||
|
|
||
|
Buildings.Save(pipe);
|
||
|
if (!save_net) Call_Back();
|
||
|
Bullets.Save(pipe);
|
||
|
if (!save_net) Call_Back();
|
||
|
Infantry.Save(pipe);
|
||
|
if (!save_net) Call_Back();
|
||
|
Overlays.Save(pipe);
|
||
|
if (!save_net) Call_Back();
|
||
|
Smudges.Save(pipe);
|
||
|
if (!save_net) Call_Back();
|
||
|
Templates.Save(pipe);
|
||
|
if (!save_net) Call_Back();
|
||
|
Terrains.Save(pipe);
|
||
|
if (!save_net) Call_Back();
|
||
|
Units.Save(pipe);
|
||
|
if (!save_net) Call_Back();
|
||
|
Factories.Save(pipe);
|
||
|
if (!save_net) Call_Back();
|
||
|
Vessels.Save(pipe);
|
||
|
|
||
|
if (!save_net) Call_Back();
|
||
|
|
||
|
/*
|
||
|
** Save the Logic & Map layers
|
||
|
*/
|
||
|
Logic.Save(pipe);
|
||
|
|
||
|
int count = MapTriggers.Count();
|
||
|
pipe.Put(&count, sizeof(count));
|
||
|
int index;
|
||
|
for (int index = 0; index < MapTriggers.Count(); index++) {
|
||
|
TARGET target = MapTriggers[index]->As_Target();
|
||
|
pipe.Put(&target, sizeof(target));
|
||
|
}
|
||
|
if (!save_net) Call_Back();
|
||
|
count = LogicTriggers.Count();
|
||
|
pipe.Put(&count, sizeof(count));
|
||
|
for (index = 0; index < LogicTriggers.Count(); index++) {
|
||
|
TARGET target = LogicTriggers[index]->As_Target();
|
||
|
pipe.Put(&target, sizeof(target));
|
||
|
}
|
||
|
if (!save_net) Call_Back();
|
||
|
for (HousesType h = HOUSE_FIRST; h < HOUSE_COUNT; h++) {
|
||
|
count = HouseTriggers[h].Count();
|
||
|
pipe.Put(&count, sizeof(count));
|
||
|
for (index = 0; index < HouseTriggers[h].Count(); index++) {
|
||
|
TARGET target = HouseTriggers[h][index]->As_Target();
|
||
|
pipe.Put(&target, sizeof(target));
|
||
|
}
|
||
|
}
|
||
|
if (!save_net) Call_Back();
|
||
|
|
||
|
for (int i = 0; i < LAYER_COUNT; i++) {
|
||
|
Map.Layer[i].Save(pipe);
|
||
|
}
|
||
|
|
||
|
if (!save_net) Call_Back();
|
||
|
|
||
|
/*
|
||
|
** Save the Score
|
||
|
*/
|
||
|
pipe.Put(&Score, sizeof(Score));
|
||
|
if (!save_net) Call_Back();
|
||
|
|
||
|
/*
|
||
|
** Save the AI Base
|
||
|
*/
|
||
|
Base.Save(pipe);
|
||
|
if (!save_net) Call_Back();
|
||
|
|
||
|
/*
|
||
|
** Save out the carry over list (if present). First see how
|
||
|
** many carry over objects are in the list.
|
||
|
*/
|
||
|
int carry_count = 0;
|
||
|
CarryoverClass const * cptr = Carryover;
|
||
|
while (cptr != NULL) {
|
||
|
carry_count++;
|
||
|
cptr = (CarryoverClass const *)cptr->Get_Next();
|
||
|
}
|
||
|
|
||
|
if (!save_net) Call_Back();
|
||
|
|
||
|
/*
|
||
|
** Save out the number of objects in the list.
|
||
|
*/
|
||
|
pipe.Put(&carry_count, sizeof(carry_count));
|
||
|
if (!save_net) Call_Back();
|
||
|
|
||
|
/*
|
||
|
** Now write out the objects themselves.
|
||
|
*/
|
||
|
CarryoverClass const * object_to_write = Carryover;
|
||
|
while (object_to_write != NULL) {
|
||
|
pipe.Put(object_to_write, sizeof(*object_to_write));
|
||
|
object_to_write = (CarryoverClass const *)object_to_write->Get_Next();
|
||
|
}
|
||
|
if (!save_net) Call_Back();
|
||
|
|
||
|
/*
|
||
|
** Save miscellaneous variables.
|
||
|
*/
|
||
|
Save_Misc_Values(pipe);
|
||
|
|
||
|
if (!save_net) Call_Back();
|
||
|
|
||
|
/*
|
||
|
** Save multiplayer values
|
||
|
*/
|
||
|
pipe.Put(&save_net, sizeof(save_net)); // Write out whether we saved the net values so we know if we have to load them again. ST - 10/22/2019 2:10PM
|
||
|
if (save_net) {
|
||
|
Save_MPlayer_Values(pipe);
|
||
|
}
|
||
|
|
||
|
pipe.Flush();
|
||
|
}
|
||
|
|
||
|
|
||
|
/***************************************************************************
|
||
|
* Save_Game -- saves a game to disk *
|
||
|
* *
|
||
|
* Saving the Map: *
|
||
|
* DisplayClass::Save() invokes CellClass's Write() for every cell *
|
||
|
* that needs to be saved. A cell needs to be saved if it contains *
|
||
|
* any special data at all, such as a TIcon, or an Occupier. *
|
||
|
* The cell saves its own CellTrigger pointer, converted to a TARGET. *
|
||
|
* *
|
||
|
* Saving game objects: *
|
||
|
* - Any object stored in an ArrayOf class needs to be saved. The ArrayOf*
|
||
|
* Save() routine invokes each object's Write() routine, if that *
|
||
|
* object's IsActive is set. *
|
||
|
* *
|
||
|
* Saving the layers: *
|
||
|
* The Map's Layers (Ground, Air, etc) of things that are on the map, *
|
||
|
* and the Logic's Layer of things to process both need to be saved. *
|
||
|
* LayerClass::Save() writes the entire layer array to disk *
|
||
|
* *
|
||
|
* Saving the houses: *
|
||
|
* Each house needs to be saved, to record its Credits, Power, etc. *
|
||
|
* *
|
||
|
* Saving miscellaneous data: *
|
||
|
* There are a lot of miscellaneous variables to save, such as the *
|
||
|
* map's dimensions, the player's house, etc. *
|
||
|
* *
|
||
|
* INPUT: *
|
||
|
* id numerical ID, for the file extension *
|
||
|
* *
|
||
|
* OUTPUT: *
|
||
|
* true = OK, false = error *
|
||
|
* *
|
||
|
* WARNINGS: *
|
||
|
* none. *
|
||
|
* *
|
||
|
* HISTORY: *
|
||
|
* 12/28/1994 BR : Created. *
|
||
|
* 02/27/1996 JLB : Uses simpler game control value save operation. *
|
||
|
*=========================================================================*/
|
||
|
bool Save_Game(int id, char const * descr, bool )
|
||
|
{
|
||
|
char name[_MAX_FNAME+_MAX_EXT];
|
||
|
|
||
|
/*
|
||
|
** Generate the filename to save. If 'id' is -1, it means save a
|
||
|
** network/modem game; otherwise, use 'id' as the file extension.
|
||
|
*/
|
||
|
if (id==-1) {
|
||
|
strcpy(name, NET_SAVE_FILE_NAME);
|
||
|
//save_net = 1;
|
||
|
} else {
|
||
|
sprintf(name, "SAVEGAME.%03d", id);
|
||
|
}
|
||
|
|
||
|
return Save_Game(name, descr);
|
||
|
}
|
||
|
|
||
|
|
||
|
|
||
|
/*
|
||
|
** Version that takes file name. ST - 9/9/2019 11:10AM
|
||
|
*/
|
||
|
bool NowSavingGame = false; // TEMP MBL: Need to discuss better solution with Steve
|
||
|
bool Save_Game(const char *file_name, const char *descr)
|
||
|
{
|
||
|
NowSavingGame = true; // TEMP MBL: Need to discuss better solution with Steve
|
||
|
|
||
|
int save_net = 0; // 1 = save network/modem game
|
||
|
|
||
|
if (Session.Type == GAME_GLYPHX_MULTIPLAYER) {
|
||
|
save_net = 1;
|
||
|
}
|
||
|
|
||
|
unsigned scenario;
|
||
|
HousesType house;
|
||
|
|
||
|
scenario = Scen.Scenario; // get current scenario #
|
||
|
house = PlayerPtr->Class->House; // get current house
|
||
|
|
||
|
/*
|
||
|
** Code everybody's pointers
|
||
|
*/
|
||
|
Code_All_Pointers();
|
||
|
|
||
|
/*
|
||
|
** Open the file
|
||
|
*/
|
||
|
BufferIOFileClass file(file_name);
|
||
|
|
||
|
FilePipe fpipe(&file);
|
||
|
|
||
|
/*
|
||
|
** Save the DLLs variables first, so we can do a version check in the DLL when we begin the load
|
||
|
*/
|
||
|
if (RunningAsDLL) {
|
||
|
DLLSave(fpipe);
|
||
|
}
|
||
|
|
||
|
/*
|
||
|
** Save the description, scenario #, and house
|
||
|
** (scenario # & house are saved separately from the actual Scenario &
|
||
|
** PlayerPtr globals for convenience; we can quickly find out which
|
||
|
** house & scenario this save-game file is for by reading these values.
|
||
|
** Also, PlayerPtr is stored in a coded form in Save_Misc_Values(),
|
||
|
** which may or may not be a HousesType number; so, saving 'house'
|
||
|
** here ensures we can always pull out the house for this file.)
|
||
|
*/
|
||
|
char descr_buf[DESCRIP_MAX];
|
||
|
memset(descr_buf, '\0', sizeof(descr_buf));
|
||
|
sprintf(descr_buf, "%s\r\n", descr); // put CR-LF after text
|
||
|
//descr_buf[strlen(descr_buf) + 1] = 26; // put CTRL-Z after NULL
|
||
|
fpipe.Put(descr_buf, DESCRIP_MAX);
|
||
|
|
||
|
fpipe.Put(&scenario, sizeof(scenario));
|
||
|
|
||
|
fpipe.Put(&house, sizeof(house));
|
||
|
|
||
|
/*
|
||
|
** Save the save-game version, for loading verification
|
||
|
*/
|
||
|
unsigned long version = SAVEGAME_VERSION;
|
||
|
#ifdef FIXIT_CSII // checked - ajw 9/28/98
|
||
|
version++;
|
||
|
#endif
|
||
|
fpipe.Put(&version, sizeof(version));
|
||
|
|
||
|
int pos = file.Seek(0, SEEK_CUR);
|
||
|
|
||
|
/*
|
||
|
** Store a dummy message digest.
|
||
|
*/
|
||
|
char digest[20];
|
||
|
fpipe.Put(digest, sizeof(digest));
|
||
|
|
||
|
|
||
|
/*
|
||
|
** Dump the save game data to the file. The data is compressed
|
||
|
** and then encrypted. The message digest is calculated in the
|
||
|
** process by using the data just as it is written to disk.
|
||
|
*/
|
||
|
SHAPipe sha;
|
||
|
BlowPipe bpipe(BlowPipe::ENCRYPT);
|
||
|
LZOPipe pipe(LZOPipe::COMPRESS, SAVE_BLOCK_SIZE);
|
||
|
// LZWPipe pipe(LZWPipe::COMPRESS, SAVE_BLOCK_SIZE);
|
||
|
// LCWPipe pipe(LCWPipe::COMPRESS, SAVE_BLOCK_SIZE);
|
||
|
bpipe.Key(&FastKey, BlowfishEngine::MAX_KEY_LENGTH);
|
||
|
|
||
|
sha.Put_To(fpipe);
|
||
|
bpipe.Put_To(sha);
|
||
|
pipe.Put_To(bpipe);
|
||
|
Put_All(pipe, save_net);
|
||
|
|
||
|
/*
|
||
|
** Output the real final message digest. This is the one that is of
|
||
|
** the data image as it exists on the disk.
|
||
|
*/
|
||
|
pipe.Flush();
|
||
|
file.Seek(pos, SEEK_SET);
|
||
|
sha.Result(digest);
|
||
|
fpipe.Put(digest, sizeof(digest));
|
||
|
|
||
|
pipe.End();
|
||
|
|
||
|
Decode_All_Pointers();
|
||
|
|
||
|
NowSavingGame = false; // TEMP MBL: Need to discuss better solution with Steve
|
||
|
|
||
|
return(true);
|
||
|
}
|
||
|
|
||
|
|
||
|
/***************************************************************************
|
||
|
* Load_Game -- loads a saved game *
|
||
|
* *
|
||
|
* This routine loads the data in the same way it was saved out. *
|
||
|
* *
|
||
|
* Loading the Map: *
|
||
|
* - DisplayClass::Load() invokes CellClass's Load() for every cell *
|
||
|
* that was saved. *
|
||
|
* - The cell loads its own CellTrigger pointer. *
|
||
|
* *
|
||
|
* Loading game objects: *
|
||
|
* - IHeap's Load() routine loads the # of objects stored, and loads *
|
||
|
* each object. *
|
||
|
* - Triggers: Add themselves to the HouseTriggers if they're associated *
|
||
|
* with a house *
|
||
|
* *
|
||
|
* Loading the layers: *
|
||
|
* LayerClass::Load() reads the entire layer array to disk *
|
||
|
* *
|
||
|
* Loading the houses: *
|
||
|
* Each house is loaded in its entirety. *
|
||
|
* *
|
||
|
* Loading miscellaneous data: *
|
||
|
* There are a lot of miscellaneous variables to load, such as the *
|
||
|
* map's dimensions, the player's house, etc. *
|
||
|
* *
|
||
|
* INPUT: *
|
||
|
* id numerical ID, for the file extension *
|
||
|
* *
|
||
|
* OUTPUT: *
|
||
|
* true = OK, false = error *
|
||
|
* *
|
||
|
* WARNINGS: *
|
||
|
* If this routine returns false, the entire game will be in an *
|
||
|
* unknown state, so the scenario will have to be re-initialized. *
|
||
|
* *
|
||
|
* HISTORY: *
|
||
|
* 12/28/1994 BR : Created. *
|
||
|
* 1/20/97 V.Grippi Added expansion CD check *
|
||
|
*=========================================================================*/
|
||
|
bool Load_Game(int id)
|
||
|
{
|
||
|
char name[_MAX_FNAME+_MAX_EXT];
|
||
|
|
||
|
/*
|
||
|
** Generate the filename to load. If 'id' is -1, it means save a
|
||
|
** network/modem game; otherwise, use 'id' as the file extension.
|
||
|
*/
|
||
|
if (id == -1) {
|
||
|
strcpy(name, NET_SAVE_FILE_NAME);
|
||
|
//load_net = 1;
|
||
|
} else {
|
||
|
sprintf(name, "SAVEGAME.%03d", id);
|
||
|
}
|
||
|
return Load_Game(name);
|
||
|
}
|
||
|
|
||
|
|
||
|
/*
|
||
|
** Version that takes a file name instead. ST - 9/9/2019 11:13AM
|
||
|
*/
|
||
|
bool Load_Game(const char *file_name)
|
||
|
{
|
||
|
int i;
|
||
|
unsigned scenario;
|
||
|
HousesType house;
|
||
|
char descr_buf[DESCRIP_MAX];
|
||
|
int load_net = 0; // 1 = save network/modem game
|
||
|
|
||
|
/*
|
||
|
** Open the file
|
||
|
*/
|
||
|
RawFileClass file(file_name);
|
||
|
if (!file.Is_Available()) {
|
||
|
return(false);
|
||
|
}
|
||
|
|
||
|
FileStraw fstraw(file);
|
||
|
|
||
|
Call_Back();
|
||
|
|
||
|
/*
|
||
|
** Load the DLLs variables first, in case we need to do something different based on version
|
||
|
*/
|
||
|
if (RunningAsDLL) {
|
||
|
DLLLoad(fstraw);
|
||
|
}
|
||
|
|
||
|
/*
|
||
|
** Read & discard the save-game's header info
|
||
|
*/
|
||
|
if (fstraw.Get(descr_buf, DESCRIP_MAX) != DESCRIP_MAX) {
|
||
|
return(false);
|
||
|
}
|
||
|
|
||
|
if (fstraw.Get(&scenario, sizeof(scenario)) != sizeof(scenario)) {
|
||
|
return(false);
|
||
|
}
|
||
|
|
||
|
if (fstraw.Get(&house, sizeof(house)) != sizeof(house)) {
|
||
|
return(false);
|
||
|
}
|
||
|
|
||
|
/*
|
||
|
** Read in & verify the save-game ID code
|
||
|
*/
|
||
|
unsigned long version;
|
||
|
if (fstraw.Get(&version, sizeof(version)) != sizeof(version)) {
|
||
|
return(false);
|
||
|
}
|
||
|
GameVersion = version;
|
||
|
#ifdef FIXIT_CSII // checked - ajw 9/28/98
|
||
|
if (version != SAVEGAME_VERSION && ((version-1) != SAVEGAME_VERSION) ) {
|
||
|
return(false);
|
||
|
}
|
||
|
#else
|
||
|
if (version != SAVEGAME_VERSION /*&& version != 0x0100616D*/){
|
||
|
return(false);
|
||
|
}
|
||
|
#endif
|
||
|
/*
|
||
|
** Get the message digest that is embedded in the file.
|
||
|
*/
|
||
|
char digest[20];
|
||
|
fstraw.Get(digest, sizeof(digest));
|
||
|
|
||
|
/*
|
||
|
** Remember the file position since we must seek back here to
|
||
|
** perform the real saved game read.
|
||
|
*/
|
||
|
long pos = file.Seek(0, SEEK_CUR);
|
||
|
|
||
|
/*
|
||
|
** Pass the rest of the file through the hash straw so that
|
||
|
** the digest can be compaired to the one in the file.
|
||
|
*/
|
||
|
SHAStraw sha;
|
||
|
sha.Get_From(fstraw);
|
||
|
for (;;) {
|
||
|
if (sha.Get(_staging_buffer, sizeof(_staging_buffer)) != sizeof(_staging_buffer)) break;
|
||
|
}
|
||
|
char actual[20];
|
||
|
sha.Result(actual);
|
||
|
sha.Get_From(NULL);
|
||
|
|
||
|
Call_Back();
|
||
|
|
||
|
/*
|
||
|
** Compare the two digests. If they differ then return a failure condition
|
||
|
** before any damage could be done.
|
||
|
*/
|
||
|
if (memcmp(actual, digest, sizeof(digest)) != 0) {
|
||
|
return(false);
|
||
|
}
|
||
|
|
||
|
/*
|
||
|
** Set up the pipe so that the scenario data can be read.
|
||
|
*/
|
||
|
file.Seek(pos, SEEK_SET);
|
||
|
BlowStraw bstraw(BlowStraw::DECRYPT);
|
||
|
LZOStraw straw(LZOStraw::DECOMPRESS, SAVE_BLOCK_SIZE);
|
||
|
// LZWStraw straw(LZWStraw::DECOMPRESS, SAVE_BLOCK_SIZE);
|
||
|
// LCWStraw straw(LCWStraw::DECOMPRESS, SAVE_BLOCK_SIZE);
|
||
|
|
||
|
bstraw.Key(&FastKey, BlowfishEngine::MAX_KEY_LENGTH);
|
||
|
bstraw.Get_From(fstraw);
|
||
|
straw.Get_From(bstraw);
|
||
|
|
||
|
/*
|
||
|
** Clear the scenario so we start fresh; this calls the Init_Clear() routine
|
||
|
** for the Map, and all object arrays. It has the following important
|
||
|
** effects:
|
||
|
** - Every cell is cleared to 0's, via MapClass::Init_Clear()
|
||
|
** - All heap elements' are cleared
|
||
|
** - The Houses are Initialized, which also clears their HouseTriggers
|
||
|
** array
|
||
|
** - The map's Layers & Logic Layer are cleared to empty
|
||
|
** - The list of currently-selected objects is cleared
|
||
|
*/
|
||
|
Clear_Scenario();
|
||
|
|
||
|
/*
|
||
|
** Load the scenario global information.
|
||
|
*/
|
||
|
straw.Get(&Scen, sizeof(Scen));
|
||
|
|
||
|
/*
|
||
|
** Fixup the Sessionclass scenario info so we can work out which
|
||
|
** CD to request later
|
||
|
*/
|
||
|
if ( load_net ){
|
||
|
|
||
|
CCFileClass scenario_file (Scen.ScenarioName);
|
||
|
if ( !scenario_file.Is_Available() ){
|
||
|
|
||
|
int cd = -1;
|
||
|
if (Is_Mission_Counterstrike (Scen.ScenarioName)) {
|
||
|
cd = 2;
|
||
|
#ifdef FIXIT_CSII // checked - ajw 9/28/98
|
||
|
if (Expansion_AM_Present()) {
|
||
|
int current_drive = CCFileClass::Get_CD_Drive();
|
||
|
int index = Get_CD_Index(current_drive, 1*60);
|
||
|
if (index == 3) cd = 3;
|
||
|
}
|
||
|
#endif
|
||
|
}
|
||
|
#ifdef FIXIT_CSII // checked - ajw 9/28/98
|
||
|
if (Is_Mission_Aftermath (Scen.ScenarioName)) {
|
||
|
cd = 3;
|
||
|
#ifdef BOGUSCD
|
||
|
cd = -1;
|
||
|
#endif
|
||
|
}
|
||
|
#endif
|
||
|
RequiredCD = cd;
|
||
|
if (!Force_CD_Available (RequiredCD)) {
|
||
|
Emergency_Exit(EXIT_FAILURE);
|
||
|
}
|
||
|
|
||
|
/*
|
||
|
** Update the internal list of scenarios to include the counterstrike
|
||
|
** list.
|
||
|
*/
|
||
|
Session.Read_Scenario_Descriptions();
|
||
|
} else {
|
||
|
/*
|
||
|
** The scenario is available so set RequiredCD to whatever is currently
|
||
|
** in the drive.
|
||
|
*/
|
||
|
int current_drive = CCFileClass::Get_CD_Drive();
|
||
|
RequiredCD = Get_CD_Index(current_drive, 1*60);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
/*
|
||
|
** Load the map. The map comes first, since it loads the Theater & init's
|
||
|
** mixfiles. The map calls all the type-class's Init routines, telling them
|
||
|
** what the Theater is; this must be done before any objects are created, so
|
||
|
** they'll be properly created.
|
||
|
*/
|
||
|
Map.Load(straw);
|
||
|
|
||
|
Call_Back();
|
||
|
|
||
|
/*
|
||
|
** Load the object data.
|
||
|
*/
|
||
|
Houses.Load(straw);
|
||
|
TeamTypes.Load(straw);
|
||
|
Teams.Load(straw);
|
||
|
TriggerTypes.Load(straw);
|
||
|
Triggers.Load(straw);
|
||
|
Aircraft.Load(straw);
|
||
|
Anims.Load(straw);
|
||
|
Buildings.Load(straw);
|
||
|
Bullets.Load(straw);
|
||
|
|
||
|
Call_Back();
|
||
|
|
||
|
Infantry.Load(straw);
|
||
|
Overlays.Load(straw);
|
||
|
Smudges.Load(straw);
|
||
|
Templates.Load(straw);
|
||
|
Terrains.Load(straw);
|
||
|
Units.Load(straw);
|
||
|
Factories.Load(straw);
|
||
|
Vessels.Load(straw);
|
||
|
|
||
|
/*
|
||
|
** Load the Logic & Map Layers
|
||
|
*/
|
||
|
Logic.Load(straw);
|
||
|
|
||
|
int count;
|
||
|
straw.Get(&count, sizeof(count));
|
||
|
MapTriggers.Clear();
|
||
|
int index;
|
||
|
for (index = 0; index < count; index++) {
|
||
|
TARGET target;
|
||
|
straw.Get(&target, sizeof(target));
|
||
|
MapTriggers.Add(As_Trigger(target));
|
||
|
}
|
||
|
|
||
|
straw.Get(&count, sizeof(count));
|
||
|
LogicTriggers.Clear();
|
||
|
for (index = 0; index < count; index++) {
|
||
|
TARGET target;
|
||
|
straw.Get(&target, sizeof(target));
|
||
|
LogicTriggers.Add(As_Trigger(target));
|
||
|
}
|
||
|
|
||
|
for (HousesType h = HOUSE_FIRST; h < HOUSE_COUNT; h++) {
|
||
|
straw.Get(&count, sizeof(count));
|
||
|
HouseTriggers[h].Clear();
|
||
|
for (index = 0; index < count; index++) {
|
||
|
TARGET target;
|
||
|
straw.Get(&target, sizeof(target));
|
||
|
HouseTriggers[h].Add(As_Trigger(target));
|
||
|
}
|
||
|
}
|
||
|
|
||
|
for (i = 0; i < LAYER_COUNT; i++) {
|
||
|
Map.Layer[i].Load(straw);
|
||
|
}
|
||
|
|
||
|
Call_Back();
|
||
|
|
||
|
/*
|
||
|
** Load the Score
|
||
|
*/
|
||
|
straw.Get(&Score, sizeof(Score));
|
||
|
new(&Score) ScoreClass(NoInitClass());
|
||
|
|
||
|
/*
|
||
|
** Load the AI Base
|
||
|
*/
|
||
|
Base.Load(straw);
|
||
|
|
||
|
/*
|
||
|
** Delete any carryover pseudo-saved game list.
|
||
|
*/
|
||
|
while (Carryover != NULL) {
|
||
|
CarryoverClass * cptr = (CarryoverClass *)Carryover->Get_Next();
|
||
|
Carryover->Remove();
|
||
|
delete Carryover;
|
||
|
Carryover = cptr;
|
||
|
}
|
||
|
|
||
|
/*
|
||
|
** Load any carryover pseudo-saved game list.
|
||
|
*/
|
||
|
int carry_count = 0;
|
||
|
straw.Get(&carry_count, sizeof(carry_count));
|
||
|
while (carry_count) {
|
||
|
CarryoverClass * cptr = new CarryoverClass;
|
||
|
assert(cptr != NULL);
|
||
|
|
||
|
straw.Get(cptr, sizeof(CarryoverClass));
|
||
|
new (cptr) CarryoverClass(NoInitClass());
|
||
|
cptr->Zap();
|
||
|
|
||
|
if (!Carryover) {
|
||
|
Carryover = cptr;
|
||
|
} else {
|
||
|
cptr->Add_Tail(*Carryover);
|
||
|
}
|
||
|
carry_count--;
|
||
|
}
|
||
|
|
||
|
Call_Back();
|
||
|
|
||
|
/*
|
||
|
** Load miscellaneous variables, including the map size & the Theater
|
||
|
*/
|
||
|
Load_Misc_Values(straw);
|
||
|
|
||
|
/*
|
||
|
** Load multiplayer values
|
||
|
*/
|
||
|
straw.Get(&load_net, sizeof(load_net));
|
||
|
if (load_net) {
|
||
|
Load_MPlayer_Values(straw);
|
||
|
}
|
||
|
|
||
|
file.Close();
|
||
|
Decode_All_Pointers();
|
||
|
Map.Init_IO();
|
||
|
Map.Flag_To_Redraw(true);
|
||
|
|
||
|
/*
|
||
|
** Fixup any expediency data that can be inferred from the physical
|
||
|
** data loaded.
|
||
|
*/
|
||
|
Post_Load_Game(load_net);
|
||
|
|
||
|
/*
|
||
|
** Re-init unit trackers. They will be garbage pointers after the load
|
||
|
*/
|
||
|
for (HousesType house = HOUSE_FIRST; house < HOUSE_COUNT; house++) {
|
||
|
HouseClass * hptr = HouseClass::As_Pointer(house);
|
||
|
if (hptr && hptr->IsActive) {
|
||
|
hptr->Init_Unit_Trackers();
|
||
|
}
|
||
|
}
|
||
|
|
||
|
Call_Back();
|
||
|
|
||
|
/*
|
||
|
** Set the required CD to be in the drive according to the scenario
|
||
|
** loaded.
|
||
|
*/
|
||
|
if (RequiredCD != -2 && !load_net) {
|
||
|
|
||
|
#ifdef FIXIT_ANTS
|
||
|
/*
|
||
|
** Determines if this an ant mission. Since the ant mission looks no different from
|
||
|
** a regular mission, examining of the scenario name is the only way to tell.
|
||
|
*/
|
||
|
if (toupper(Scen.ScenarioName[0]) == 'S' &&
|
||
|
toupper(Scen.ScenarioName[1]) == 'C' &&
|
||
|
toupper(Scen.ScenarioName[2]) == 'A' &&
|
||
|
toupper(Scen.ScenarioName[3]) == '0' &&
|
||
|
toupper(Scen.ScenarioName[5]) == 'E' &&
|
||
|
toupper(Scen.ScenarioName[6]) == 'A') {
|
||
|
|
||
|
AntsEnabled = true;
|
||
|
} else{
|
||
|
AntsEnabled = false;
|
||
|
}
|
||
|
#endif
|
||
|
|
||
|
if (Scen.Scenario == 1) {
|
||
|
RequiredCD = -1;
|
||
|
} else {
|
||
|
#ifdef FIXIT_ANTS
|
||
|
if (Scen.Scenario > 19 || AntsEnabled) {
|
||
|
RequiredCD = 2;
|
||
|
#ifdef FIXIT_CSII // checked - ajw 9/28/98
|
||
|
if(Scen.Scenario >= 36) {
|
||
|
RequiredCD = 3;
|
||
|
#ifdef BOGUSCD
|
||
|
RequiredCD = -1;
|
||
|
#endif
|
||
|
}
|
||
|
#endif
|
||
|
} else {
|
||
|
#endif //FIXIT_ANTS
|
||
|
if (PlayerPtr->Class->House != HOUSE_USSR && PlayerPtr->Class->House != HOUSE_UKRAINE) {
|
||
|
RequiredCD = 0;
|
||
|
} else {
|
||
|
RequiredCD = 1;
|
||
|
}
|
||
|
#ifdef FIXIT_ANTS
|
||
|
}
|
||
|
#endif //FIXIT_ANTS
|
||
|
}
|
||
|
|
||
|
}else{
|
||
|
|
||
|
if ( load_net ){
|
||
|
|
||
|
CCFileClass scenario_file (Scen.ScenarioName);
|
||
|
|
||
|
/*
|
||
|
** Fix up the session class variables
|
||
|
*/
|
||
|
for ( int s=0 ; s<Session.Scenarios.Count() ; s++ ) {
|
||
|
if (Session.Scenarios[s]->Description() == Scen.Description){
|
||
|
|
||
|
memcpy (Session.Options.ScenarioDescription, Scen.Description,
|
||
|
sizeof (Session.Options.ScenarioDescription));
|
||
|
memcpy (Session.ScenarioFileName, Scen.ScenarioName,
|
||
|
sizeof (Session.ScenarioFileName));
|
||
|
Session.ScenarioFileLength = scenario_file.Size();
|
||
|
memcpy (Session.ScenarioDigest, Session.Scenarios[s]->Get_Digest(),
|
||
|
sizeof (Session.ScenarioDigest));
|
||
|
Session.ScenarioIsOfficial = Session.Scenarios[s]->Get_Official();
|
||
|
Scen.Scenario = s;
|
||
|
Session.Options.ScenarioIndex = s;
|
||
|
break;
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
if (!Force_CD_Available(RequiredCD)) {
|
||
|
Prog_End("Load_Game Force_CD_Available failed", true);
|
||
|
Emergency_Exit(EXIT_FAILURE);
|
||
|
}
|
||
|
|
||
|
ScenarioInit = 0;
|
||
|
|
||
|
if (load_net) {
|
||
|
|
||
|
// Removed as this is ensured by the GlyphX & DLL save/load code. ST - 10/22/2019 5:20PM
|
||
|
//if (!Reconcile_Players()) { // (must do after Decode pointers)
|
||
|
// return(false);
|
||
|
//}
|
||
|
|
||
|
//!!!!!!!!!! put Fixup_Player_Units() here
|
||
|
Session.LoadGame = true;
|
||
|
}
|
||
|
|
||
|
Map.Reload_Sidebar(); // re-load sidebar art.
|
||
|
|
||
|
/*
|
||
|
** Rescan the scenario file for any rules updates.
|
||
|
*/
|
||
|
CCINIClass ini;
|
||
|
int result = ini.Load(CCFileClass(Scen.ScenarioName), true);
|
||
|
|
||
|
/*
|
||
|
** Reset the rules values to their initial settings.
|
||
|
*/
|
||
|
Rule.General(RuleINI);
|
||
|
Rule.Recharge(RuleINI);
|
||
|
Rule.AI(RuleINI);
|
||
|
Rule.Powerups(RuleINI);
|
||
|
Rule.Land_Types(RuleINI);
|
||
|
Rule.Themes(RuleINI);
|
||
|
Rule.IQ(RuleINI);
|
||
|
Rule.Objects(RuleINI);
|
||
|
Rule.Difficulty(RuleINI);
|
||
|
#ifdef FIXIT_CSII // checked - ajw 9/28/98 - But does this incorporate *changes*? - NO.
|
||
|
Rule.General(AftermathINI);
|
||
|
Rule.Recharge(AftermathINI);
|
||
|
Rule.AI(AftermathINI);
|
||
|
Rule.Powerups(AftermathINI);
|
||
|
Rule.Land_Types(AftermathINI);
|
||
|
Rule.Themes(AftermathINI);
|
||
|
Rule.IQ(AftermathINI);
|
||
|
Rule.Objects(AftermathINI);
|
||
|
Rule.Difficulty(AftermathINI);
|
||
|
#endif
|
||
|
|
||
|
/*
|
||
|
** Override any rules values specified in this
|
||
|
** particular scenario file.
|
||
|
*/
|
||
|
Rule.General(ini);
|
||
|
Rule.Recharge(ini);
|
||
|
Rule.AI(ini);
|
||
|
Rule.Powerups(ini);
|
||
|
Rule.Land_Types(ini); \
|
||
|
Rule.Themes(ini);
|
||
|
Rule.IQ(ini);
|
||
|
Rule.Objects(ini);
|
||
|
Rule.Difficulty(ini);
|
||
|
#ifdef FIXIT_CSII // ajw - Added runtime check for Aftermath to skirmish mode.
|
||
|
if (load_net) {
|
||
|
bool readini = false;
|
||
|
switch(Session.Type) {
|
||
|
case GAME_NORMAL:
|
||
|
readini = false;
|
||
|
break;
|
||
|
case GAME_SKIRMISH:
|
||
|
readini = Is_Aftermath_Installed();
|
||
|
break;
|
||
|
default:
|
||
|
#ifdef FIXIT_VERSION_3
|
||
|
readini = bAftermathMultiplayer;
|
||
|
#else
|
||
|
if (PlayingAgainstVersion >= VERSION_AFTERMATH_CS) {
|
||
|
readini = true;
|
||
|
}
|
||
|
#endif
|
||
|
break;
|
||
|
}
|
||
|
if(readini) {
|
||
|
/*
|
||
|
** Find out if the CD in the current drive is the Aftermath disc.
|
||
|
*/
|
||
|
if(Get_CD_Index(CCFileClass::Get_CD_Drive(), 1*60) != 3) {
|
||
|
GamePalette.Set(FADE_PALETTE_FAST, Call_Back);
|
||
|
if (!Force_CD_Available(3)) { // force Aftermath CD in drive.
|
||
|
Prog_End("Load_Game Force_CD_Available(3) failed", true);
|
||
|
#ifndef FIXIT_VERSION_3 // WChat eliminated.
|
||
|
#ifdef WIN32
|
||
|
if(Special.IsFromWChat || SpawnedFromWChat) {
|
||
|
char packet[10] = {"Hello"};
|
||
|
Send_Data_To_DDE_Server (packet, strlen(packet), DDEServerClass::DDE_CONNECTION_FAILED);
|
||
|
}
|
||
|
#endif
|
||
|
#endif
|
||
|
Emergency_Exit(EXIT_FAILURE);
|
||
|
}
|
||
|
}
|
||
|
CCINIClass mpini;
|
||
|
if (mpini.Load(CCFileClass("MPLAYER.INI"), false)) {
|
||
|
Rule.General(mpini);
|
||
|
Rule.Recharge(mpini);
|
||
|
Rule.AI(mpini);
|
||
|
Rule.Powerups(mpini);
|
||
|
Rule.Land_Types(mpini);
|
||
|
Rule.Themes(mpini);
|
||
|
Rule.IQ(mpini);
|
||
|
Rule.Objects(mpini);
|
||
|
Rule.Difficulty(mpini);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
}
|
||
|
#endif
|
||
|
|
||
|
if (Scen.TransitTheme == THEME_NONE) {
|
||
|
Theme.Queue_Song(THEME_FIRST);
|
||
|
} else {
|
||
|
Theme.Queue_Song(Scen.TransitTheme);
|
||
|
}
|
||
|
|
||
|
if (Session.Type == GAME_GLYPHX_MULTIPLAYER) {
|
||
|
Rule.IsSmartDefense = true;
|
||
|
}
|
||
|
|
||
|
return(true);
|
||
|
}
|
||
|
|
||
|
|
||
|
/***************************************************************************
|
||
|
* Save_Misc_Values -- saves miscellaneous variables *
|
||
|
* *
|
||
|
* INPUT: *
|
||
|
* file file to use for writing *
|
||
|
* *
|
||
|
* OUTPUT: *
|
||
|
* true = success, false = failure *
|
||
|
* *
|
||
|
* WARNINGS: *
|
||
|
* none. *
|
||
|
* *
|
||
|
* HISTORY: *
|
||
|
* 12/29/1994 BR : Created. *
|
||
|
* 03/12/1996 JLB : Simplified. *
|
||
|
*=========================================================================*/
|
||
|
bool Save_Misc_Values(Pipe & file)
|
||
|
{
|
||
|
int i, j;
|
||
|
int count; // # ptrs in 'CurrentObject'
|
||
|
ObjectClass * ptr; // for saving 'CurrentObject' ptrs
|
||
|
|
||
|
/*
|
||
|
** Player's House.
|
||
|
*/
|
||
|
int x = PlayerPtr->Class->House;
|
||
|
file.Put(&x, sizeof(x));
|
||
|
|
||
|
/*
|
||
|
** Save frame #.
|
||
|
*/
|
||
|
file.Put(&Frame, sizeof(Frame));
|
||
|
|
||
|
/*
|
||
|
** Save currently-selected objects list.
|
||
|
** Save the # of ptrs in the list.
|
||
|
*/
|
||
|
for (i = 0; i < SelectedObjectsType::COUNT; i++) {
|
||
|
DynamicVectorClass<ObjectClass *>& selection = CurrentObject.Raw(i);
|
||
|
count = selection.Count();
|
||
|
file.Put(&count, sizeof(count));
|
||
|
|
||
|
/*
|
||
|
** Save the pointers.
|
||
|
*/
|
||
|
for (j = 0; j < count; j++) {
|
||
|
ptr = selection[j];
|
||
|
file.Put(&ptr, sizeof(ptr));
|
||
|
}
|
||
|
}
|
||
|
|
||
|
/*
|
||
|
** Save the chronal vortex
|
||
|
*/
|
||
|
ChronalVortex.Save(file);
|
||
|
|
||
|
/*
|
||
|
** Save Tanya flags.
|
||
|
*/
|
||
|
file.Put(&IsTanyaDead, sizeof(IsTanyaDead));
|
||
|
file.Put(&SaveTanya, sizeof(SaveTanya));
|
||
|
|
||
|
return(true);
|
||
|
}
|
||
|
|
||
|
|
||
|
/***********************************************************************************************
|
||
|
* Load_Misc_Values -- Loads miscellaneous variables. *
|
||
|
* *
|
||
|
* INPUT: file -- The file to load the misc values from. *
|
||
|
* *
|
||
|
* OUTPUT: Was the misc load process successful? *
|
||
|
* *
|
||
|
* WARNINGS: none *
|
||
|
* *
|
||
|
* HISTORY: *
|
||
|
* 06/24/1995 BRR : Created. *
|
||
|
* 03/12/1996 JLB : Simplified. *
|
||
|
*=============================================================================================*/
|
||
|
bool Load_Misc_Values(Straw & file)
|
||
|
{
|
||
|
ObjectClass * ptr; // for loading 'CurrentObject' ptrs
|
||
|
|
||
|
/*
|
||
|
** Player's House.
|
||
|
*/
|
||
|
int x;
|
||
|
file.Get(&x, sizeof(x));
|
||
|
// file.Get(&PlayerPtr, sizeof(PlayerPtr));
|
||
|
PlayerPtr = HouseClass::As_Pointer((HousesType)x);
|
||
|
|
||
|
/*
|
||
|
** Load frame #.
|
||
|
*/
|
||
|
file.Get(&Frame, sizeof(Frame));
|
||
|
|
||
|
for (int i = 0; i < SelectedObjectsType::COUNT; i++) {
|
||
|
/*
|
||
|
** Load currently-selected objects list.
|
||
|
** Load the # of ptrs in the list.
|
||
|
*/
|
||
|
DynamicVectorClass<ObjectClass *>& selection = CurrentObject.Raw(i);
|
||
|
int count; // # ptrs in 'CurrentObject'
|
||
|
file.Get(&count, sizeof(count));
|
||
|
|
||
|
/*
|
||
|
** Load the pointers.
|
||
|
*/
|
||
|
for (int j = 0; j < count; j++) {
|
||
|
file.Get(&ptr, sizeof(ptr));
|
||
|
selection.Add(ptr); // add to the list
|
||
|
}
|
||
|
}
|
||
|
|
||
|
/*
|
||
|
** Load the chronal vortex
|
||
|
*/
|
||
|
ChronalVortex.Load(file);
|
||
|
|
||
|
/*
|
||
|
** Save Tanya flags.
|
||
|
*/
|
||
|
file.Get(&IsTanyaDead, sizeof(IsTanyaDead));
|
||
|
file.Get(&SaveTanya, sizeof(SaveTanya));
|
||
|
|
||
|
return(true);
|
||
|
}
|
||
|
|
||
|
|
||
|
/***************************************************************************
|
||
|
* Save_MPlayer_Values -- Saves multiplayer-specific values *
|
||
|
* *
|
||
|
* This routine saves multiplayer values that need to be restored for a *
|
||
|
* save game. In addition to saving the random # seed for this scenario, *
|
||
|
* it saves the contents of the actual random number generator; this *
|
||
|
* ensures that the random # sequencer will pick up where it left off when *
|
||
|
* the game was saved. *
|
||
|
* This routine also saves the header for a Recording file, so it must *
|
||
|
* save some data not needed specifically by a save-game file (ie Seed). *
|
||
|
* *
|
||
|
* INPUT: *
|
||
|
* file file to save to *
|
||
|
* *
|
||
|
* OUTPUT: *
|
||
|
* true = success, false = failure *
|
||
|
* *
|
||
|
* WARNINGS: *
|
||
|
* none. *
|
||
|
* *
|
||
|
* HISTORY: *
|
||
|
* 09/28/1995 BRR : Created. *
|
||
|
*=========================================================================*/
|
||
|
bool Save_MPlayer_Values(Pipe & file)
|
||
|
{
|
||
|
Session.Save(file);
|
||
|
file.Put(&BuildLevel, sizeof(BuildLevel));
|
||
|
file.Put(&Debug_Unshroud, sizeof(Debug_Unshroud));
|
||
|
file.Put(&Seed, sizeof(Seed));
|
||
|
file.Put(&Whom, sizeof(Whom));
|
||
|
file.Put(&Special, sizeof(SpecialClass));
|
||
|
file.Put(&Options, sizeof(GameOptionsClass));
|
||
|
|
||
|
return (true);
|
||
|
}
|
||
|
|
||
|
|
||
|
/***************************************************************************
|
||
|
* Load_MPlayer_Values -- Loads multiplayer-specific values *
|
||
|
* *
|
||
|
* INPUT: *
|
||
|
* file file to load from *
|
||
|
* *
|
||
|
* OUTPUT: *
|
||
|
* true = success, false = failure *
|
||
|
* *
|
||
|
* WARNINGS: *
|
||
|
* none. *
|
||
|
* *
|
||
|
* HISTORY: *
|
||
|
* 09/28/1995 BRR : Created. *
|
||
|
*=========================================================================*/
|
||
|
bool Load_MPlayer_Values(Straw & file)
|
||
|
{
|
||
|
Session.Load(file);
|
||
|
file.Get(&BuildLevel, sizeof(BuildLevel));
|
||
|
file.Get(&Debug_Unshroud, sizeof(Debug_Unshroud));
|
||
|
file.Get(&Seed, sizeof(Seed));
|
||
|
file.Get(&Whom, sizeof(Whom));
|
||
|
file.Get(&Special, sizeof(SpecialClass));
|
||
|
file.Get(&Options, sizeof(GameOptionsClass));
|
||
|
|
||
|
return (true);
|
||
|
}
|
||
|
|
||
|
/*
|
||
|
** ST - 9/26/2019 11:43AM
|
||
|
*/
|
||
|
extern void DLL_Code_Pointers(void);
|
||
|
extern void DLL_Decode_Pointers(void);
|
||
|
|
||
|
/***********************************************************************************************
|
||
|
* Code_All_Pointers -- Code all pointers. *
|
||
|
* *
|
||
|
* INPUT: none *
|
||
|
* *
|
||
|
* OUTPUT: none *
|
||
|
* *
|
||
|
* WARNINGS: none *
|
||
|
* *
|
||
|
* HISTORY: *
|
||
|
* 06/24/1995 BRR : Created. *
|
||
|
*=============================================================================================*/
|
||
|
void Code_All_Pointers(void)
|
||
|
{
|
||
|
int i, j;
|
||
|
|
||
|
/*
|
||
|
** The Map.
|
||
|
*/
|
||
|
Map.Code_Pointers();
|
||
|
|
||
|
/*
|
||
|
** The ArrayOf's.
|
||
|
*/
|
||
|
TeamTypes.Code_Pointers();
|
||
|
Teams.Code_Pointers();
|
||
|
Triggers.Code_Pointers();
|
||
|
Aircraft.Code_Pointers();
|
||
|
Anims.Code_Pointers();
|
||
|
Buildings.Code_Pointers();
|
||
|
Bullets.Code_Pointers();
|
||
|
Infantry.Code_Pointers();
|
||
|
Overlays.Code_Pointers();
|
||
|
Smudges.Code_Pointers();
|
||
|
Templates.Code_Pointers();
|
||
|
Terrains.Code_Pointers();
|
||
|
Units.Code_Pointers();
|
||
|
Factories.Code_Pointers();
|
||
|
Vessels.Code_Pointers();
|
||
|
|
||
|
/*
|
||
|
** The Layers.
|
||
|
*/
|
||
|
Logic.Code_Pointers();
|
||
|
for (i = 0; i < LAYER_COUNT; i++) {
|
||
|
Map.Layer[i].Code_Pointers();
|
||
|
}
|
||
|
|
||
|
/*
|
||
|
** The Score.
|
||
|
*/
|
||
|
Score.Code_Pointers();
|
||
|
|
||
|
/*
|
||
|
** The Base.
|
||
|
*/
|
||
|
Base.Code_Pointers();
|
||
|
|
||
|
/*
|
||
|
** PlayerPtr.
|
||
|
*/
|
||
|
// PlayerPtr = (HouseClass *)(PlayerPtr->Class->House);
|
||
|
|
||
|
/*
|
||
|
** Currently-selected objects.
|
||
|
*/
|
||
|
for (i = 0; i < SelectedObjectsType::COUNT; i++) {
|
||
|
DynamicVectorClass<ObjectClass *>& selection = CurrentObject.Raw(i);
|
||
|
for (j = 0; j < selection.Count(); j++) {
|
||
|
selection[j] = (ObjectClass *)selection[j]->As_Target();
|
||
|
}
|
||
|
}
|
||
|
|
||
|
/*
|
||
|
** DLL data
|
||
|
*/
|
||
|
DLL_Code_Pointers();
|
||
|
|
||
|
/*
|
||
|
** Houses must be coded last, because the Class->House member of the HouseClass
|
||
|
** is used to code HouseClass pointers for all other objects, and if Class is
|
||
|
** coded, it will point to a meaningless value.
|
||
|
*/
|
||
|
Houses.Code_Pointers();
|
||
|
}
|
||
|
|
||
|
|
||
|
/***********************************************************************************************
|
||
|
* Decode_All_Pointers -- Decodes all pointers. *
|
||
|
* *
|
||
|
* INPUT: none *
|
||
|
* *
|
||
|
* OUTPUT: none *
|
||
|
* *
|
||
|
* WARNINGS: none *
|
||
|
* *
|
||
|
* HISTORY: *
|
||
|
* 06/24/1995 BRR : Created. *
|
||
|
*=============================================================================================*/
|
||
|
void Decode_All_Pointers(void)
|
||
|
{
|
||
|
/*
|
||
|
** The Map.
|
||
|
*/
|
||
|
Map.Decode_Pointers();
|
||
|
|
||
|
/*
|
||
|
** Decode houses first, so we can properly decode all other objects'
|
||
|
** House pointers
|
||
|
*/
|
||
|
Houses.Decode_Pointers();
|
||
|
|
||
|
/*
|
||
|
** DLL data
|
||
|
*/
|
||
|
DLL_Decode_Pointers();
|
||
|
|
||
|
/*
|
||
|
** The ArrayOf's.
|
||
|
*/
|
||
|
TeamTypes.Decode_Pointers();
|
||
|
Teams.Decode_Pointers();
|
||
|
Triggers.Decode_Pointers();
|
||
|
Aircraft.Decode_Pointers();
|
||
|
Anims.Decode_Pointers();
|
||
|
Buildings.Decode_Pointers();
|
||
|
Bullets.Decode_Pointers();
|
||
|
Infantry.Decode_Pointers();
|
||
|
Overlays.Decode_Pointers();
|
||
|
Smudges.Decode_Pointers();
|
||
|
Templates.Decode_Pointers();
|
||
|
Terrains.Decode_Pointers();
|
||
|
Units.Decode_Pointers();
|
||
|
Factories.Decode_Pointers();
|
||
|
Vessels.Decode_Pointers();
|
||
|
|
||
|
/*
|
||
|
** The Layers.
|
||
|
*/
|
||
|
Logic.Decode_Pointers();
|
||
|
for (int i = 0; i < LAYER_COUNT; i++) {
|
||
|
Map.Layer[i].Decode_Pointers();
|
||
|
}
|
||
|
|
||
|
/*
|
||
|
** The Score.
|
||
|
*/
|
||
|
Score.Decode_Pointers();
|
||
|
|
||
|
/*
|
||
|
** The Base.
|
||
|
*/
|
||
|
Base.Decode_Pointers();
|
||
|
|
||
|
/*
|
||
|
** PlayerPtr.
|
||
|
*/
|
||
|
// PlayerPtr = HouseClass::As_Pointer((HousesType)PlayerPtr);
|
||
|
Whom = PlayerPtr->Class->House;
|
||
|
assert(PlayerPtr != NULL);
|
||
|
|
||
|
/*
|
||
|
** Currently-selected objects.
|
||
|
*/
|
||
|
for (int index = 0; index < SelectedObjectsType::COUNT; index++) {
|
||
|
DynamicVectorClass<ObjectClass *>& selection = CurrentObject.Raw(index);
|
||
|
for (int j = 0; j < selection.Count(); j++) {
|
||
|
selection[j] = As_Object((TARGET)selection[j]);
|
||
|
assert(selection[j] != NULL);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
/*
|
||
|
** Last-Minute Fixups; to resolve these pointers properly requires all other
|
||
|
** pointers to be loaded & decoded.
|
||
|
*/
|
||
|
if (Map.PendingObjectPtr) {
|
||
|
Map.PendingObject = &Map.PendingObjectPtr->Class_Of();
|
||
|
assert(Map.PendingObject != NULL);
|
||
|
Map.Set_Cursor_Shape(Map.PendingObject->Occupy_List(true));
|
||
|
#ifdef BG
|
||
|
Map.Set_Placement_List(Map.PendingObject->Placement_List(true));
|
||
|
#endif
|
||
|
} else {
|
||
|
Map.PendingObject = 0;
|
||
|
Map.Set_Cursor_Shape(0);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
|
||
|
/***************************************************************************
|
||
|
* Get_Savefile_Info -- gets description, scenario #, house *
|
||
|
* *
|
||
|
* INPUT: *
|
||
|
* id numerical ID, for the file extension *
|
||
|
* buf buffer to store description in *
|
||
|
* scenp ptr to variable to hold scenario *
|
||
|
* housep ptr to variable to hold house *
|
||
|
* *
|
||
|
* OUTPUT: *
|
||
|
* true = OK, false = error (save-game file invalid) *
|
||
|
* *
|
||
|
* WARNINGS: *
|
||
|
* none. *
|
||
|
* *
|
||
|
* HISTORY: *
|
||
|
* 01/12/1995 BR : Created. *
|
||
|
*=========================================================================*/
|
||
|
bool Get_Savefile_Info(int id, char * buf, unsigned * scenp, HousesType * housep)
|
||
|
{
|
||
|
char name[_MAX_FNAME+_MAX_EXT];
|
||
|
unsigned long version;
|
||
|
char descr_buf[DESCRIP_MAX];
|
||
|
|
||
|
/*
|
||
|
** Generate the filename to load
|
||
|
*/
|
||
|
sprintf(name, "SAVEGAME.%03d", id);
|
||
|
BufferIOFileClass file(name);
|
||
|
|
||
|
FileStraw straw(file);
|
||
|
|
||
|
/*
|
||
|
** Read in the description, scenario #, and the house
|
||
|
*/
|
||
|
if (straw.Get(descr_buf, DESCRIP_MAX) != DESCRIP_MAX) {
|
||
|
return(false);
|
||
|
}
|
||
|
|
||
|
descr_buf[strlen(descr_buf) - 2] = '\0'; // trim off CR/LF
|
||
|
strcpy(buf, descr_buf);
|
||
|
|
||
|
if (straw.Get(scenp, sizeof(unsigned)) != sizeof(unsigned)) {
|
||
|
return(false);
|
||
|
}
|
||
|
|
||
|
if (straw.Get(housep, sizeof(HousesType)) != sizeof(HousesType)) {
|
||
|
return(false);
|
||
|
}
|
||
|
|
||
|
/*
|
||
|
** Read & verify the save-game version #
|
||
|
*/
|
||
|
if (straw.Get(&version, sizeof(version)) != sizeof(version)) {
|
||
|
return(false);
|
||
|
}
|
||
|
#ifdef FIXIT_CSII // checked - ajw 9/28/98
|
||
|
if (version != SAVEGAME_VERSION && ((version-1 != SAVEGAME_VERSION)) ) {
|
||
|
#else
|
||
|
if (version != SAVEGAME_VERSION) {
|
||
|
#endif
|
||
|
return(false);
|
||
|
}
|
||
|
return(true);
|
||
|
}
|
||
|
|
||
|
|
||
|
/***************************************************************************
|
||
|
* Reconcile_Players -- Reconciles loaded data with the 'Players' vector *
|
||
|
* *
|
||
|
* This function is for supporting loading a saved multiplayer game. *
|
||
|
* When the game is loaded, we have to figure out which house goes with *
|
||
|
* which entry in the Players vector. We also have to figure out if *
|
||
|
* everyone who was originally in the game is still with us, and if not, *
|
||
|
* turn their stuff over to the computer. *
|
||
|
* *
|
||
|
* So, this function does the following: *
|
||
|
* - For every name in 'Players', makes sure that name is in the House *
|
||
|
* array; if not, it's a fatal error. *
|
||
|
* - For every human-controlled house, makes sure there's a player *
|
||
|
* with that name; if not, it turns that house over to the computer. *
|
||
|
* - Fills in the Player's house ID *
|
||
|
* *
|
||
|
* This assumes that each player MUST keep their name the same as it was *
|
||
|
* when the game was saved! It's also assumed that the network *
|
||
|
* connections have not been formed yet, since Player[i]->Player.ID will *
|
||
|
* be invalid until this routine has been called. *
|
||
|
* *
|
||
|
* INPUT: *
|
||
|
* none. *
|
||
|
* *
|
||
|
* OUTPUT: *
|
||
|
* true = OK, false = error *
|
||
|
* *
|
||
|
* WARNINGS: *
|
||
|
* none. *
|
||
|
* *
|
||
|
* HISTORY: *
|
||
|
* 09/29/1995 BRR : Created. *
|
||
|
*=========================================================================*/
|
||
|
static int Reconcile_Players(void)
|
||
|
{
|
||
|
int i;
|
||
|
int found;
|
||
|
HousesType house;
|
||
|
HouseClass * housep;
|
||
|
|
||
|
/*
|
||
|
** If there are no players, there's nothing to do.
|
||
|
*/
|
||
|
if (Session.Players.Count()==0)
|
||
|
return (true);
|
||
|
|
||
|
/*
|
||
|
** Make sure every name we're connected to can be found in a House
|
||
|
*/
|
||
|
for (i = 0; i < Session.Players.Count(); i++) {
|
||
|
found = 0;
|
||
|
for (house = HOUSE_MULTI1; house < HOUSE_MULTI1 +
|
||
|
Session.MaxPlayers; house++) {
|
||
|
|
||
|
housep = HouseClass::As_Pointer(house);
|
||
|
if (!housep) {
|
||
|
continue;
|
||
|
}
|
||
|
|
||
|
if (!stricmp(Session.Players[i]->Name, housep->IniName)) {
|
||
|
found = 1;
|
||
|
break;
|
||
|
}
|
||
|
}
|
||
|
if (!found)
|
||
|
return (false);
|
||
|
}
|
||
|
|
||
|
//
|
||
|
// Loop through all Houses; if we find a human-owned house that we're
|
||
|
// not connected to, turn it over to the computer.
|
||
|
//
|
||
|
for (house = HOUSE_MULTI1; house < HOUSE_MULTI1 +
|
||
|
Session.MaxPlayers; house++) {
|
||
|
housep = HouseClass::As_Pointer(house);
|
||
|
if (!housep) {
|
||
|
continue;
|
||
|
}
|
||
|
|
||
|
//
|
||
|
// Skip this house if it wasn't human to start with.
|
||
|
//
|
||
|
if (!housep->IsHuman) {
|
||
|
continue;
|
||
|
}
|
||
|
|
||
|
//
|
||
|
// Try to find this name in the Players vector; if it's found, set
|
||
|
// its ID to this house.
|
||
|
//
|
||
|
found = 0;
|
||
|
for (i = 0; i < Session.Players.Count(); i++) {
|
||
|
if (!stricmp(Session.Players[i]->Name, housep->IniName)) {
|
||
|
found = 1;
|
||
|
Session.Players[i]->Player.ID = house;
|
||
|
break;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
/*
|
||
|
** If this name wasn't found, remove it
|
||
|
*/
|
||
|
if (!found) {
|
||
|
|
||
|
/*
|
||
|
** Turn the player's house over to the computer's AI
|
||
|
*/
|
||
|
housep->IsHuman = false;
|
||
|
housep->IsStarted = true;
|
||
|
// housep->Smartness = IQ_MENSA;
|
||
|
housep->IQ = Rule.MaxIQ;
|
||
|
strcpy (housep->IniName, Text_String(TXT_COMPUTER));
|
||
|
|
||
|
Session.NumPlayers--;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
//
|
||
|
// If all went well, our Session.NumPlayers value should now equal the value
|
||
|
// from the saved game, minus any players we removed.
|
||
|
//
|
||
|
if (Session.NumPlayers == Session.Players.Count()) {
|
||
|
return (true);
|
||
|
} else {
|
||
|
return (false);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
|
||
|
/***************************************************************************
|
||
|
* MPlayer_Save_Message -- pops up a "saving..." message *
|
||
|
* *
|
||
|
* INPUT: *
|
||
|
* none. *
|
||
|
* *
|
||
|
* OUTPUT: *
|
||
|
* none. *
|
||
|
* *
|
||
|
* WARNINGS: *
|
||
|
* none. *
|
||
|
* *
|
||
|
* HISTORY: *
|
||
|
* 10/30/1995 BRR : Created. *
|
||
|
*=========================================================================*/
|
||
|
void MPlayer_Save_Message(void)
|
||
|
{
|
||
|
//char *txt = Text_String(
|
||
|
}
|