CnC_Remastered_Collection/REDALERT/INIT.CPP
PG-SteveT ae72fce5dd August 6th Patch Update
Accumulated DLL source code changes since June 22nd patch
2020-08-06 09:44:54 -07:00

3873 lines
127 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/INIT.CPP 8 3/14/97 5:15p Joe_b $ */
/***********************************************************************************************
*** 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 : INIT.CPP *
* *
* Programmer : Joe L. Bostic *
* *
* Start Date : January 20, 1992 *
* *
*---------------------------------------------------------------------------------------------*
* Functions: *
* Anim_Init -- Initialize the VQ animation control structure. *
* Bootstrap -- Perform the initial bootstrap procedure. *
* Calculate_CRC -- Calculates a one-way hash from a data block. *
* Init_Authorization -- Verifies that the player is authorized to play the game. *
* Init_Bootstrap_Mixfiles -- Registers and caches any mixfiles needed for bootstrapping. *
* Init_Bulk_Data -- Initialize the time-consuming mixfile caching. *
* Init_CDROM_Access -- Initialize the CD-ROM access handler. *
* Init_Color_Remaps -- Initialize the text remap tables. *
* Init_Expansion_Files -- Fetch any override expansion mixfiles. *
* Init_Fonts -- Initialize all the game font pointers. *
* Init_Game -- Main game initialization routine. *
* Init_Heaps -- Initialize the game heaps and buffers. *
* Init_Keys -- Initialize the cryptographic keys. *
* Init_Mouse -- Initialize the mouse system. *
* Init_One_Time_Systems -- Initialize internal pointers to the bulk data. *
* Init_Random -- Initializes the random-number generator *
* Init_Secondary_Mixfiles -- Register and cache secondary mixfiles. *
* Load_Recording_Values -- Loads recording values from recording file *
* Load_Title_Page -- Load the background art for the title page. *
* Obfuscate -- Sufficiently transform parameter to thwart casual hackers. *
* Parse_Command_Line -- Parses the command line parameters. *
* Parse_INI_File -- Parses CONQUER.INI for special options *
* Play_Intro -- plays the introduction & logo movies *
* Save_Recording_Values -- Saves recording values to a recording file *
* Select_Game -- The game's main menu *
* Load_Prolog_Page -- Loads the special pre-prolog "please wait" page. *
* - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - */
#include "function.h"
#include "loaddlg.h"
#ifdef WIN32
#ifdef WINSOCK_IPX
#include "WSProto.h"
#include "WSPUDP.h"
#include "WSPIPX.h"
#include "internet.h"
#else //WINSOCK_IPX
#include "tcpip.h"
#endif //WINSOCK_IPX
#endif
#include <conio.h>
#include <dos.h>
#ifndef WIN32
#include <sys\timeb.h>
#endif
#include "ccdde.h"
#include <time.h>
#ifdef DONGLE
#include "cbn_.h"
#endif
#ifdef MPEGMOVIE // Denzil 6/25/98
#include "mpgset.h"
#endif
RemapControlType SidebarScheme;
//#include "WolDebug.h"
#ifdef CHEAT_KEYS
extern bool bNoMovies;
#endif
/****************************************
** Function prototypes for this module **
*****************************************/
static void Play_Intro(bool sequenced=false);
static void Init_Color_Remaps(void);
static void Init_Heaps(void);
static void Init_Expansion_Files(void);
static void Init_One_Time_Systems(void);
static void Init_Fonts(void);
static void Init_CDROM_Access(void);
static void Init_Bootstrap_Mixfiles(void);
static void Init_Secondary_Mixfiles(void);
static void Init_Mouse(void);
static void Bootstrap(void);
//static void Init_Authorization(void);
static void Init_Bulk_Data(void);
static void Init_Keys(void);
extern int UnitBuildPenalty;
extern "C" {
extern long RandNumb;
}
#ifndef WIN32
static int UsePageFaultHandler = 1; // 1 = install PFH
#endif //WIN32
//extern int SimRandIndex;
void Init_Random(void);
#define ATTRACT_MODE_TIMEOUT 3600 // timeout for attract mode
bool Load_Recording_Values(CCFileClass & file);
bool Save_Recording_Values(CCFileClass & file);
#ifdef WOLAPI_INTEGRATION
extern int WOL_Main();
#include "WolapiOb.h"
extern WolapiObject* pWolapi;
#endif
#ifdef FIXIT_VERSION_3
bool Expansion_Dialog( bool bCounterstrike );
#endif
extern bool Is_Mission_Counterstrike (char *file_name);
/***********************************************************************************************
* Load_Prolog_Page -- Loads the special pre-prolog "please wait" page. *
* *
* This loads and displays the prolog page that is displayed before the prolog movie *
* is played. This page is necessary because there is much loading that occurs before *
* the prolog movie is played and looking at a picture is better than looking at a blank *
* screen. *
* *
* INPUT: none *
* *
* OUTPUT: none *
* *
* WARNINGS: none *
* *
* HISTORY: *
* 11/03/1996 JLB : Created. *
*=============================================================================================*/
static void Load_Prolog_Page(void)
{
Hide_Mouse();
#ifdef WIN32
Load_Title_Screen("PROLOG.PCX", &HidPage, (unsigned char*)CCPalette.Get_Data());
HidPage.Blit(SeenPage);
#else
Load_Picture("PROLOG.CPS", HidPage, HidPage, CCPalette, BM_DEFAULT);
HidPage.Blit(SeenPage);
#endif
CCPalette.Set();
Show_Mouse();
}
/***********************************************************************************************
* Init_Game -- Main game initialization routine. *
* *
* Perform all one-time game initializations here. This includes all *
* allocations and table setups. The intro and other one-time startup *
* tasks are also performed here. *
* *
* INPUT: argc,argv -- Command line arguments. *
* *
* OUTPUT: none *
* *
* WARNINGS: Only call this ONCE! *
* *
* HISTORY: *
* 10/07/1992 JLB : Created. *
*=============================================================================================*/
#include "sha.h"
//#include <locale.h>
bool Init_Game(int , char * [])
{
/*
** Allocate the benchmark tracking objects only if the machine and
** compile flags indicate.
*/
#ifdef CHEAT_KEYS
if (Processor() >= 2) {
Benches = new Benchmark [BENCH_COUNT];
}
#endif
/*
** Initialize the encryption keys.
*/
Init_Keys();
/*
** Bootstrap as much as possible before error-prone initializations are
** performed. This bootstrap process will enable the error message
** handler to function.
*/
Bootstrap();
////////////////////////////////////////
// The editor needs to not start the mouse up. - 7/22/2019 JAS
if (!RunningFromEditor)
{
/*
** Check for an initialize a working mouse pointer. Display error and bail if
** no mouse driver is installed.
*/
Init_Mouse();
}
/*
** Initialize access to the CD-ROM and ensure that the CD is inserted. This can, and
** most likely will, result in a visible prompt.
*/
#if (0) //PG
Init_CDROM_Access();
#endif
if (Special.IsFromInstall) {
Load_Prolog_Page();
}
/*
** Register and cache any secondary mixfiles.
*/
Init_Secondary_Mixfiles();
/*
** This is a special hack to initialize the heaps that must be in place before the
** rules file is processed. These heaps should properly be allocated as a consequence
** of processing the rules.ini file, but that is a bit beyond the capabilities of
** the rule parser routine (currently).
*/
HouseTypes.Set_Heap(HOUSE_COUNT);
BuildingTypes.Set_Heap(STRUCT_COUNT);
AircraftTypes.Set_Heap(AIRCRAFT_COUNT);
InfantryTypes.Set_Heap(INFANTRY_COUNT);
BulletTypes.Set_Heap(BULLET_COUNT);
AnimTypes.Set_Heap(ANIM_COUNT);
UnitTypes.Set_Heap(UNIT_COUNT);
VesselTypes.Set_Heap(VESSEL_COUNT);
TemplateTypes.Set_Heap(TEMPLATE_COUNT);
TerrainTypes.Set_Heap(TERRAIN_COUNT);
OverlayTypes.Set_Heap(OVERLAY_COUNT);
SmudgeTypes.Set_Heap(SMUDGE_COUNT);
HouseTypeClass::Init_Heap();
BuildingTypeClass::Init_Heap();
AircraftTypeClass::Init_Heap();
InfantryTypeClass::Init_Heap();
BulletTypeClass::Init_Heap();
AnimTypeClass::Init_Heap();
UnitTypeClass::Init_Heap();
VesselTypeClass::Init_Heap();
TemplateTypeClass::Init_Heap();
TerrainTypeClass::Init_Heap();
OverlayTypeClass::Init_Heap();
SmudgeTypeClass::Init_Heap();
// Heap init moved here from globals.cpp. ST - 5/20/2019
CCPtr<AircraftClass>::Set_Heap(&Aircraft);
CCPtr<AnimClass>::Set_Heap(&Anims);
CCPtr<BuildingClass>::Set_Heap(&Buildings);
CCPtr<BulletClass>::Set_Heap(&Bullets);
CCPtr<FactoryClass>::Set_Heap(&Factories);
CCPtr<HouseClass>::Set_Heap(&Houses);
CCPtr<InfantryClass>::Set_Heap(&Infantry);
CCPtr<OverlayClass>::Set_Heap(&Overlays);
CCPtr<SmudgeClass>::Set_Heap(&Smudges);
CCPtr<TeamClass>::Set_Heap(&Teams);
CCPtr<TeamTypeClass>::Set_Heap(&TeamTypes);
CCPtr<TemplateClass>::Set_Heap(&Templates);
CCPtr<TerrainClass>::Set_Heap(&Terrains);
CCPtr<TriggerClass>::Set_Heap(&Triggers);
CCPtr<TriggerTypeClass>::Set_Heap(&TriggerTypes);
CCPtr<HouseTypeClass>::Set_Heap(&HouseTypes);
CCPtr<BuildingTypeClass>::Set_Heap(&BuildingTypes);
CCPtr<AircraftTypeClass>::Set_Heap(&AircraftTypes);
CCPtr<InfantryTypeClass>::Set_Heap(&InfantryTypes);
CCPtr<BulletTypeClass>::Set_Heap(&BulletTypes);
CCPtr<AnimTypeClass>::Set_Heap(&AnimTypes);
CCPtr<UnitTypeClass>::Set_Heap(&UnitTypes);
CCPtr<VesselTypeClass>::Set_Heap(&VesselTypes);
CCPtr<TemplateTypeClass>::Set_Heap(&TemplateTypes);
CCPtr<TerrainTypeClass>::Set_Heap(&TerrainTypes);
CCPtr<OverlayTypeClass>::Set_Heap(&OverlayTypes);
CCPtr<SmudgeTypeClass>::Set_Heap(&SmudgeTypes);
/*
** Find and process any rules for this game.
*/
if (RuleINI.Load(CCFileClass("RULES.INI"), false)) {
Rule.Process(RuleINI);
}
#ifdef FIXIT_CSII // checked - ajw 9/28/98
// Aftermath runtime change 9/29/98
// This is safe to do, as only rules for aftermath units are included in this ini.
if (Is_Aftermath_Installed() == true) {
if (AftermathINI.Load(CCFileClass("AFTRMATH.INI"), false)) {
Rule.Process(AftermathINI);
}
}
#endif
Session.MaxPlayers = Rule.MaxPlayers;
/*
** Initialize the game object heaps as well as other rules-dependant buffer allocations.
*/
Init_Heaps();
/*
** Initialize the animation system.
*/
Anim_Init();
#ifndef FIXIT_VERSION_3 // WChat eliminated.
#ifdef WIN32
if (SpawnedFromWChat){
Special.IsFromWChat = true;
}
#endif
#endif
#ifdef MPEGMOVIE // Denzil 6/15/98
if( Using_DVD() )
{
#ifdef MCIMPEG
MciMovie = new MCIMovie(MainWindow);
#endif
MpgSettings = new MPGSettings(NULL); //RawFileClass(CONFIG_FILE_NAME));
}
#endif
/*
** Play the startup animation.
*/
if (!Special.IsFromInstall && !Special.IsFromWChat) {
VisiblePage.Clear();
// Mono_Printf("Playing Intro\n");
Play_Intro();
memset(CurrentPalette, 0x01, 768);
WhitePalette.Set();
} else {
memset(CurrentPalette, 0x01, 768);
}
/*
** Initialize the text remap tables.
*/
Init_Color_Remaps();
/*
** Get authorization to access the game.
*/
// Init_Authorization();
// Show_Mouse();
/*
** Set the logic page to the seenpage.
*/
Set_Logic_Page(SeenBuff);
/*
** If not automatically launching into the intro, then display the title
** page while the bulk data is cached.
*/
if (!Special.IsFromInstall) {
Load_Title_Page(true);
Hide_Mouse();
Fancy_Text_Print(TXT_STAND_BY, 160*RESFACTOR, 120*RESFACTOR, &ColorRemaps[PCOLOR_DIALOG_BLUE], TBLACK, TPF_CENTER|TPF_TEXT|TPF_DROPSHADOW);
Show_Mouse();
CCPalette.Set(FADE_PALETTE_SLOW);
Call_Back();
}
/*
** Initialize the bulk data. This takes the longest time and must be performed once
** before the regular game starts.
*/
Init_Bulk_Data();
/*
** Initialize the multiplayer score values
*/
Session.GamesPlayed = 0;
Session.NumScores = 0;
Session.CurGame = 0;
for (int i = 0; i < MAX_MULTI_NAMES; i++) {
Session.Score[i].Name[0] = '\0';
Session.Score[i].Wins = 0;
for (int j = 0; j < MAX_MULTI_GAMES; j++) {
Session.Score[i].Kills[j] = -1; // -1 = this player didn't play this round
}
}
/*
** Copy the title screen's palette into the GamePalette & OriginalPalette,
** because the options Load routine uses these palettes to set the brightness, etc.
*/
GamePalette = CCPalette;
// InGamePalette = CCPalette;
OriginalPalette = CCPalette;
/*
** Read game options, so the GameSpeed is initialized when multiplayer
** dialogs are invoked. (GameSpeed must be synchronized between systems.)
*/
Options.Load_Settings();
/*
** Initialise the color lookup tables for the chronal vortex
*/
ChronalVortex.Stop();
ChronalVortex.Setup_Remap_Tables(Scen.Theater);
/*
** Clear out name overrides array
*/
#ifdef FIXIT_NAME_OVERRIDE
for (int index = 0; index < ARRAY_SIZE(NameOverride); index++) {
NameOverride[index] = NULL;
NameIDOverride[index] = 0;
}
#endif //FIXIT_NAME_OVERRIDE
return(true);
}
#ifdef WINSOCK_IPX // Steve Tall missed this one - ajw
extern bool Get_Broadcast_Addresses (void);
#endif
/***********************************************************************************************
* Select_Game -- The game's main menu *
* *
* INPUT: *
* fade if true, will fade the palette in gradually *
* *
* OUTPUT: *
* none. *
* *
* WARNINGS: *
* none. *
* *
* HISTORY: *
* 06/05/1995 BRR : Created. *
*=============================================================================================*/
bool Select_Game(bool fade)
{
// Enums in Select_Game() must match order of buttons in Main_Menu().
#ifdef FIXIT_VERSION_3
enum {
SEL_TIMEOUT = -1, // main menu timeout--go into attract mode
SEL_NEW_SCENARIO_CS, // Expansion scenario to play.
SEL_NEW_SCENARIO_AM, // Expansion scenario to play.
SEL_START_NEW_GAME, // start a new game
SEL_LOAD_MISSION, // load a saved game
SEL_MULTIPLAYER_GAME, // play modem/null-modem/network game
SEL_INTRO, // couch-potato mode
SEL_EXIT, // exit to DOS
SEL_FAME, // view the hall o' fame
SEL_NONE, // placeholder default value
};
#else // FIXIT_VERSION_3
enum {
SEL_TIMEOUT = -1, // main menu timeout--go into attract mode
SEL_NEW_SCENARIO, // Expansion scenario to play.
SEL_START_NEW_GAME, // start a new game
#if defined(WIN32) && !defined(INTERNET_OFF) // Denzil 5/1/98 - Internet play
SEL_INTERNET,
#endif //WIN32
//#if defined(MPEGMOVIE) // Denzil 6/25/98
// SEL_MOVIESETTINGS,
//#endif
SEL_LOAD_MISSION, // load a saved game
SEL_MULTIPLAYER_GAME, // play modem/null-modem/network game
SEL_INTRO, // couch-potato mode
SEL_EXIT, // exit to DOS
SEL_FAME, // view the hall o' fame
SEL_NONE, // placeholder default value
};
#endif // FIXIT_VERSION_3
bool gameloaded=false; // Has the game been loaded from the menu?
int selection; // the default selection
bool process = true; // false = break out of while loop
bool display = true;
#ifdef DONGLE
/* These where added by ColinM for the dongle checking */
short iRet = 0;
unsigned short iPortNr = 1; /* automatic port scan enabled */
unsigned char cSCodeSER[] = "\x41\x42";
unsigned long ulIdRet = 0;
unsigned char cBoxName[]= "\x00\x00";
#endif
#ifdef FIXIT_CSII // checked - ajw 9/28/98
int cdcheck = 0;
bool cs = Is_Counterstrike_Installed();
#endif
// #ifndef DVD // Denzil - We want the menu screen ajw No we don't
// if (Special.IsFromInstall) {
// display = false;
// }
// #endif
Show_Mouse();
#ifdef FIXIT_CSII // checked - ajw 9/28/98
NewUnitsEnabled = SecretUnitsEnabled = 0; // Assume new units disabled, unless specifically .INI enabled or multiplayer negotiations enable it.
#endif
#ifndef WOLAPI_INTEGRATION
#ifdef WIN32
/*
** Enable the DDE Server so we can get internet start game packets from WChat
*/
DDEServer.Enable();
#endif //WIN32
#endif // !WOLAPI_INTEGRATION
/*
** [Re]set any globals that need it, in preparation for a new scenario
*/
GameActive = true;
DoList.Init();
#ifdef MIRROR_QUEUE
MirrorList.Init();
#endif
OutList.Init();
Frame = 0;
Scen.MissionTimer = 0;
Scen.MissionTimer.Stop();
Scen.CDifficulty = DIFF_NORMAL;
Scen.Difficulty = DIFF_NORMAL;
PlayerWins = false;
PlayerLoses = false;
Session.ObiWan = false;
Debug_Unshroud = false;
Map.Set_Cursor_Shape(0);
Map.PendingObjectPtr = 0;
Map.PendingObject = 0;
Map.PendingHouse = HOUSE_NONE;
Session.ProcessTicks = 0;
Session.ProcessFrames = 0;
Session.DesiredFrameRate = 30;
#if(TIMING_FIX)
NewMaxAheadFrame1 = 0;
NewMaxAheadFrame2 = 0;
#endif
/* ColinM added to check for dongle */
#ifdef DONGLE
iRet = CbN_BoxReady( iPortNr , cBoxName);
if (cBoxName[0] != 0xc5 && cBoxName[0] != 0xc9)
{
WWMessageBox().Process("Please ensure dongle is attached. Run the dongle batch file too.", TXT_OK);
Emergency_Exit(EXIT_FAILURE);
}
iRet = CbN_ReadSER( iPortNr, cSCodeSER, &ulIdRet);
if (ulIdRet != 0xa0095)
{
WWMessageBox().Process("Please ensure dongle is attached. Run the dongle batch file too.", TXT_OK);
Emergency_Exit(EXIT_FAILURE);
}
#endif
/*
** Init multiplayer game scores. Let Wins accumulate; just init the current
** Kills for this game. Kills of -1 means this player didn't play this round.
*/
for (int i = 0 ; i < MAX_MULTI_GAMES; i++) {
Session.Score[i].Kills[Session.CurGame] = -1;
}
/*
** Set default mouse shape
*/
Map.Set_Default_Mouse(MOUSE_NORMAL, false);
/*
** If the last game we played was a multiplayer game, jump right to that
** menu by pre-setting 'selection'.
*/
if (Session.Type == GAME_NORMAL) {
selection = SEL_NONE;
} else {
selection = SEL_MULTIPLAYER_GAME;
}
/*
** Main menu processing; only do this if we're not in editor mode.
*/
if (!Debug_Map) {
/*
** Menu selection processing loop
*/
Theme.Queue_Song(THEME_CRUS);
/*
** If we're playing back a recording, load all pertinent values & skip
** the menu loop. Hide the now-useless mouse pointer.
*/
if (Session.Play && Session.RecordFile.Is_Available()) {
if (Session.RecordFile.Open(READ)) {
Load_Recording_Values(Session.RecordFile);
process = false;
Theme.Fade_Out();
} else
Session.Play = false;
}
#ifndef FIXIT_VERSION_3
#if defined(WIN32) && !defined(INTERNET_OFF) // Denzil 5/1/98 - Internet play
/*
** Handle case where we were spawned from Wchat
*/
if (SpawnedFromWChat) {
Special.IsFromInstall = false; //Dont play intro if we were spawned from wchat
selection = SEL_INTERNET;
Theme.Queue_Song(THEME_QUIET);
Session.Type = GAME_INTERNET;
display = false;
Set_Logic_Page(SeenBuff);
}
#endif //WIN32
#endif
while (process) {
/*
** Redraw the title page if needed
*/
if (display) {
Hide_Mouse();
/*
** Display the title page; fade it in if this is the first time
** through the loop, and the 'fade' flag is true
*/
Load_Title_Page();
GamePalette = CCPalette;
HidPage.Blit(SeenPage);
// if (fade) {
// WhitePalette.Set();
// CCPalette.Set(FADE_PALETTE_SLOW, Call_Back);
// fade = false;
// } else {
CCPalette.Set();
// }
Set_Logic_Page(SeenBuff);
display = false;
Show_Mouse();
}
else {
if (RunningAsDLL) { //PG
return true;;
}
}
/*
** Display menu and fetch selection from player.
*/
if (Special.IsFromInstall) selection = SEL_START_NEW_GAME;
#ifndef WOLAPI_INTEGRATION
#if defined(WIN32) && !defined(INTERNET_OFF) // Denzil 5/1/98 - Internet play
/*
** Handle case where we were spawned from Wchat and our start game
** packet has already arrived
*/
if (Special.IsFromWChat && DDEServer.Get_MPlayer_Game_Info()) {
Check_From_WChat(NULL);
selection = SEL_MULTIPLAYER_GAME;
Theme.Queue_Song(THEME_QUIET);
Session.Type = GAME_INTERNET;
} else {
/*
** We werent spawned but we could still receive a DDE packet from wchat
*/
if (DDEServer.Get_MPlayer_Game_Info()) {
Check_From_WChat(NULL);
/*
** Make sure top and bottom of screen are clear in 640x480 mode
*/
if (ScreenHeight == 480) {
VisiblePage.Fill_Rect (0, 0, 639, 40, 0);
VisiblePage.Fill_Rect (0, 440, 639, 479, 0);
}
}
}
#endif //WIN32
#endif
#ifdef WOLAPI_INTEGRATION
if( pWolapi )
selection = SEL_MULTIPLAYER_GAME; // We are returning from a game.
#endif
if (selection == SEL_NONE) {
#ifdef FIXIT_ANTS
AntsEnabled = false;
#endif
selection = Main_Menu(ATTRACT_MODE_TIMEOUT);
}
Call_Back();
switch (selection) {
/*
** Pick an expansion scenario.
*/
#ifdef FIXIT_VERSION_3
case SEL_NEW_SCENARIO_CS:
case SEL_NEW_SCENARIO_AM:
#else // FIXIT_VERSION_3
case SEL_NEW_SCENARIO:
#endif // FIXIT_VERSION_3
Scen.CarryOverMoney = 0;
IsTanyaDead = false;
SaveTanya = false;
#ifdef FIXIT_VERSION_3
if( selection == SEL_NEW_SCENARIO_CS )
{
if(!Force_CD_Available(2)) {
selection = SEL_NONE;
break;
}
if(!Expansion_Dialog( true )){
selection = SEL_NONE;
break;
}
}
else
{
if(!Force_CD_Available(3)) {
selection = SEL_NONE;
break;
}
if(!Expansion_Dialog( false )){
selection = SEL_NONE;
break;
}
}
#else // FIXIT_VERSION_3
#ifdef FIXIT_CSII // checked - ajw 9/28/98
if (cs) {
cdcheck = 2;
}
if (Is_Aftermath_Installed()) {
if (!cdcheck) {
cdcheck = 3;
} else {
cdcheck = 4; // special case: means check for 3 or 4
}
}
if(!Force_CD_Available(cdcheck)) {
return(false);
}
#else
if(!Force_CD_Available(2)) {
return(false);
}
#endif
if(!Expansion_Dialog()){
selection = SEL_NONE;
break;
}
#endif // FIXIT_VERSION_3
#ifdef FIXIT_DIFFICULTY
#ifdef FIXIT_CSII // checked - ajw 9/28/98
switch (Fetch_Difficulty(cdcheck >= 3)) {
#else
switch (Fetch_Difficulty()) {
#endif
case 0:
Scen.CDifficulty = DIFF_HARD;
Scen.Difficulty = DIFF_EASY;
break;
case 1:
Scen.CDifficulty = DIFF_HARD;
Scen.Difficulty = DIFF_NORMAL;
break;
case 2:
Scen.CDifficulty = DIFF_NORMAL;
Scen.Difficulty = DIFF_NORMAL;
break;
case 3:
Scen.CDifficulty = DIFF_EASY;
Scen.Difficulty = DIFF_NORMAL;
break;
case 4:
Scen.CDifficulty = DIFF_EASY;
Scen.Difficulty = DIFF_HARD;
break;
}
#endif
Theme.Fade_Out();
Theme.Queue_Song(THEME_FIRST);
Session.Type = GAME_NORMAL;
process = false;
break;
/*
** SEL_START_NEW_GAME: Play the game
*/
case SEL_START_NEW_GAME:
if (Special.IsFromInstall) {
Scen.CDifficulty = DIFF_NORMAL;
Scen.Difficulty = DIFF_NORMAL;
} else {
switch (Fetch_Difficulty()) {
case 0:
Scen.CDifficulty = DIFF_HARD;
Scen.Difficulty = DIFF_EASY;
break;
case 1:
Scen.CDifficulty = DIFF_HARD;
Scen.Difficulty = DIFF_NORMAL;
break;
case 2:
Scen.CDifficulty = DIFF_NORMAL;
Scen.Difficulty = DIFF_NORMAL;
break;
case 3:
Scen.CDifficulty = DIFF_EASY;
Scen.Difficulty = DIFF_NORMAL;
break;
case 4:
Scen.CDifficulty = DIFF_EASY;
Scen.Difficulty = DIFF_HARD;
break;
}
}
Scen.CarryOverMoney = 0;
BuildLevel = 10;
IsTanyaDead = false;
SaveTanya = false;
Whom = HOUSE_GOOD;
if (!Special.IsFromInstall) {
#ifdef FIXIT_ANTS
if (AntsEnabled) {
Scen.Set_Scenario_Name("SCA01EA.INI");
} else {
#endif
#ifdef FIXIT_VERSION_3
switch (WWMessageBox().Process(TXT_CHOOSE, TXT_ALLIES, TXT_CANCEL, TXT_SOVIET)) {
case 2:
Scen.Set_Scenario_Name("SCU01EA.INI");
break;
default:
selection = SEL_NONE;
continue;
#else
switch (WWMessageBox().Process(TXT_CHOOSE, TXT_ALLIES, TXT_SOVIET)) {
case 1:
Scen.Set_Scenario_Name("SCU01EA.INI");
break;
default:
#endif
case 0:
Scen.Set_Scenario_Name("SCG01EA.INI");
break;
}
#ifdef FIXIT_ANTS
}
#endif
Theme.Fade_Out();
Load_Title_Page();
} else {
Theme.Fade_Out();
#ifdef DVD // Denzil ajw Presumably a bug fix.
Choose_Side();
Hide_Mouse();
#else
Hide_Mouse();
Choose_Side();
#endif
if (CurrentCD == 0) {
Scen.Set_Scenario_Name("SCG01EA.INI");
} else {
Scen.Set_Scenario_Name("SCU01EA.INI");
}
}
Session.Type = GAME_NORMAL;
process = false;
break;
#ifndef FIXIT_VERSION_3 // Removed button from main menu.
#if defined(WIN32) && !defined(INTERNET_OFF) // Denzil 5/1/98 - Internet play
/*
** Internet game is requested
*/
case SEL_INTERNET:
/*
** Only call up the internet menu code if we dont already have connect info from WChat
*/
if (!DDEServer.Get_MPlayer_Game_Info()) {
if (Do_The_Internet_Menu_Thang() && DDEServer.Get_MPlayer_Game_Info()) {
Check_From_WChat(NULL);
selection = SEL_MULTIPLAYER_GAME;
display = false;
Session.Type = GAME_INTERNET;
} else {
selection = SEL_NONE;
display = true;
}
} else {
Check_From_WChat(NULL);
display = false;
Session.Type = GAME_INTERNET;
selection = SEL_MULTIPLAYER_GAME;
}
break;
#endif //WIN32
#endif
// #if defined(MPEGMOVIE) // Denzil 6/25/98
// case SEL_MOVIESETTINGS:
// MpgSettings->Dialog();
// display = true;
// selection = SEL_NONE;
// break;
// #endif
/*
** Load a saved game.
*/
case SEL_LOAD_MISSION:
if (LoadOptionsClass(LoadOptionsClass::LOAD).Process()) {
Theme.Queue_Song(THEME_FIRST);
process = false;
gameloaded = true;
} else {
display = true;
selection = SEL_NONE;
}
break;
/*
** SEL_MULTIPLAYER_GAME: set 'Session.Type' to NULL-modem, modem, or
** network play.
*/
case SEL_MULTIPLAYER_GAME:
#ifdef WOLAPI_INTEGRATION
if( !pWolapi )
{
#endif
switch (Session.Type) {
/*
** If 'Session.Type' isn't already set up for a multiplayer game,
** we must prompt the user for which type of multiplayer game
** they want.
*/
case GAME_NORMAL:
Session.Type = Select_MPlayer_Game();
if (Session.Type == GAME_NORMAL) { // 'Cancel'
display = true;
selection = SEL_NONE;
}
break;
case GAME_SKIRMISH:
#if (0)//PG
if ( !Com_Scenario_Dialog(true) ) {
Session.Type = Select_MPlayer_Game();
if (Session.Type == GAME_NORMAL) { // user hit Cancel
display = true;
selection = SEL_NONE;
}
}
else
{
// Ever hits? Session.Type set to GAME_SKIRMISH without user selecting in Select_MPlayer_Game()?
#ifdef FIXIT_VERSION_3
// If mission is Counterstrike, CS CD will be required. But aftermath units require AM CD.
bAftermathMultiplayer = Is_Aftermath_Installed() && !Is_Mission_Counterstrike( Scen.ScenarioName );
// ajw I'll bet this was needed before also...
Session.ScenarioIsOfficial = Session.Scenarios[Session.Options.ScenarioIndex]->Get_Official();
#endif
}
#endif//PG
break;
case GAME_NULL_MODEM:
case GAME_MODEM:
#if (0)
if ( Session.Type != GAME_SKIRMISH && NullModem.Num_Connections() ) {
NullModem.Init_Send_Queue();
if ((Session.Type == GAME_NULL_MODEM &&
Session.ModemType == MODEM_NULL_HOST) ||
(Session.Type == GAME_MODEM &&
Session.ModemType == MODEM_DIALER) ) {
if ( !Com_Scenario_Dialog() ) {
Session.Type = Select_Serial_Dialog();
if (Session.Type == GAME_NORMAL) { // user hit Cancel
display = true;
selection = SEL_NONE;
}
}
} else {
if ( !Com_Show_Scenario_Dialog() ) {
Session.Type = Select_Serial_Dialog();
if (Session.Type == GAME_NORMAL) { // user hit Cancel
display = true;
selection = SEL_NONE;
}
}
}
} else {
Session.Type = Select_MPlayer_Game();
if (Session.Type == GAME_NORMAL) { // 'Cancel'
display = true;
selection = SEL_NONE;
}
}
#endif
break;
#ifndef WOLAPI_INTEGRATION
#if defined(WIN32) && !defined(INTERNET_OFF) // Denzil 5/1/98 - Internet play
/*
** Handle being spawned from WChat. Internet play based on IPX code.
*/
case GAME_INTERNET: // ajw No longer hit.
if (Special.IsFromWChat) {
/*
** Give myself focus.
*/
SetForegroundWindow ( MainWindow );
ShowWindow ( MainWindow, ShowCommand );
#ifdef WINSOCK_IPX
if (PacketTransport ) delete PacketTransport;
PacketTransport = new UDPInterfaceClass;
assert ( PacketTransport != NULL);
if (PacketTransport->Init()) {
WWDebugString ("RA95 - About to read multiplayer settings.\n");
Session.Read_MultiPlayer_Settings ();
WWDebugString ("RA95 - About to call Start_Server or Start_Client.\n");
PacketTransport->Start_Listening();
/*
** Flush out any pending packets from a previous game.
*/
PacketTransport->Discard_In_Buffers();
PacketTransport->Discard_Out_Buffers();
} else {
delete PacketTransport;
PacketTransport = NULL;
WWDebugString ("RA95 - Winsock failed to initialise.\n");
Session.Type = GAME_NORMAL;
selection = SEL_EXIT;
Special.IsFromWChat = false;
break;
}
WWDebugString ("RA95 - About to call Init_Network.\n");
Init_Network();
#else //WINSOCK_IPX
WWDebugString ("RA95 - About to initialise Winsock.\n");
if (Winsock.Init()) {
WWDebugString ("RA95 - About to read multiplayer settings.\n");
Session.Read_MultiPlayer_Settings ();
Server = PlanetWestwoodIsHost;
WWDebugString ("RA95 - About to set addresses.\n");
Winsock.Set_Host_Address(PlanetWestwoodIPAddress);
WWDebugString ("RA95 - About to call Start_Server or Start_Client.\n");
if (Server) {
Winsock.Start_Server();
} else {
Winsock.Start_Client();
}
/*
** Flush out any pending packets from a previous game.
*/
WWDebugString ("RA95 - About to flush packet queue.\n");
WWDebugString ("RA95 - Allocating scrap memory.\n");
char *temp_buffer = new char[1024];
WWDebugString ("RA95 - Creating timer class instance.\n");
CountDownTimerClass ptimer;
WWDebugString ("RA95 - Entering read loop.\n");
while (Winsock.Read(temp_buffer, 1024)) {
WWDebugString ("RA95 - Discarding a packet.\n");
ptimer.Set (30, true);
while (ptimer.Time()) {};
WWDebugString ("RA95 - Ready to check for more packets.\n");
}
WWDebugString ("RA95 - About to delete scrap memory.\n");
delete temp_buffer;
} else {
WWDebugString ("RA95 - Winsock failed to initialise.\n");
Session.Type = GAME_NORMAL;
selection = SEL_EXIT;
Special.IsFromWChat = false;
break;
}
#endif //WINSOCK_IPX
WWDebugString ("RA95 - About to call Init_Network.\n");
Init_Network();
if (DDEServer.Get_MPlayer_Game_Info()) {
WWDebugString ("RA95 - About to call Read_Game_Options.\n");
Read_Game_Options( NULL );
} else {
Read_Game_Options( "C&CSPAWN.INI" );
}
#ifdef WINSOCK_IPX
WWDebugString ("RA95 - About to set addresses.\n");
PacketTransport->Set_Broadcast_Address (PlanetWestwoodIPAddress);
#endif //WINSOCK_IPX
if (PlanetWestwoodIsHost) {
WWDebugString ("RA95 - About to call Server_Remote_Connect.\n");
if (Server_Remote_Connect()) {
WWDebugString ("RA95 - Server_Remote_Connect returned success.\n");
break;
} else {
/*
** We failed to connect to the other player
*/
#ifdef WINSOCK_IPX
delete PacketTransport;
PacketTransport = NULL;
#else //WINSOCK_IPX
Winsock.Close();
#endif //WINSOCK_IPX
Session.Type = GAME_NORMAL;
selection = SEL_NONE;
DDEServer.Delete_MPlayer_Game_Info(); // Make sure we dont go round in an infinite loop
break;
}
} else {
WWDebugString ("RA95 - About to call Client_Remote_Connect.\n");
if (Client_Remote_Connect()) {
WWDebugString ("RA95 - Client_Remote_Connect returned success.\n");
break;
} else {
/*
** We failed to connect to the other player
*/
#ifdef WINSOCK_IPX
delete PacketTransport;
PacketTransport = NULL;
#else //WINSOCK_IPX
Winsock.Close();
#endif //WINSOCK_IPX
Session.Type = GAME_NORMAL;
selection = SEL_NONE;
DDEServer.Delete_MPlayer_Game_Info(); // Make sure we dont go round in an infinite loop
break;
}
}
} else {
Session.Type = Select_MPlayer_Game();
if (Session.Type == GAME_NORMAL) { // 'Cancel'
display = true;
selection = SEL_NONE;
}
}
break;
#endif //WIN32
#endif // !WOLAPI_INTEGRATION
}
#ifdef WOLAPI_INTEGRATION
} // if( !pWolapi )
if( pWolapi )
Session.Type = GAME_INTERNET;
#endif
//debugprint( "Session.Type = %i\n", Session.Type );
switch (Session.Type) {
/*
** Modem, Null-Modem or internet
*/
case GAME_MODEM:
case GAME_NULL_MODEM:
#ifndef WOLAPI_INTEGRATION
case GAME_INTERNET:
#endif
case GAME_SKIRMISH:
Theme.Fade_Out();
process = false;
#ifdef FIXIT_VERSION_3
Options.ScoreVolume = Options.MultiScoreVolume;
#else
Options.ScoreVolume = 0;
#endif
break;
#ifdef WOLAPI_INTEGRATION // implies also WINSOCK_IPX
case GAME_INTERNET:
if( PacketTransport )
delete PacketTransport;
PacketTransport = new UDPInterfaceClass;
assert( PacketTransport != NULL );
if( PacketTransport->Init() )
{
switch( WOL_Main() )
{
case 1:
// Start game.
#ifdef FIXIT_VERSION_3
Options.ScoreVolume = Options.MultiScoreVolume;
#else
Options.ScoreVolume = 0;
#endif
process = false;
Theme.Fade_Out();
break;
case 0:
// User cancelled.
Session.Type = GAME_NORMAL;
display = true;
selection = SEL_MULTIPLAYER_GAME; //SEL_NONE;
delete PacketTransport;
PacketTransport = NULL;
break;
case -1:
// Patch was downloaded. Exit app.
Theme.Fade_Out();
BlackPalette.Set( FADE_PALETTE_SLOW );
return false;
}
}
else
{
Session.Type = GAME_NORMAL;
display = true;
selection = SEL_MULTIPLAYER_GAME; //SEL_NONE;
delete PacketTransport;
PacketTransport = NULL;
}
break;
#endif
/*
** Network (IPX): start a new network game.
*/
case GAME_IPX:
WWDebugString ("RA95 - Game type is IPX.\n");
/*
** Init network system & remote-connect
*/
#ifdef WINSOCK_IPX
if (PacketTransport ) delete PacketTransport;
// if (WWMessageBox().Process("Select a protocol to use for network play.", "UDP", "IPX")) {
PacketTransport = new IPXInterfaceClass;
assert ( PacketTransport != NULL);
// }else{
// PacketTransport = new UDPInterfaceClass; //IPXInterfaceClass;
// assert ( PacketTransport != NULL);
// if (!Get_Broadcast_Addresses()) {
// Session.Type = GAME_NORMAL;
// display = true;
// selection = SEL_NONE;
// delete PacketTransport;
// PacketTransport = NULL;
// break;
// }
// }
#endif //WINSOCK_IPX
WWDebugString ("RA95 - About to call Init_Network.\n");
if (Session.Type == GAME_IPX && Init_Network() && Remote_Connect()) {
#ifdef FIXIT_VERSION_3
Options.ScoreVolume = Options.MultiScoreVolume;
#else
Options.ScoreVolume = 0;
#endif
process = false;
Theme.Fade_Out();
} else { // user hit cancel, or init failed
Session.Type = GAME_NORMAL;
display = true;
selection = SEL_NONE;
#ifdef WINSOCK_IPX
delete PacketTransport;
PacketTransport = NULL;
#endif //WINSOCK_IPX
}
break;
#if(TEN)
/*
** TEN: jump straight into the game
*/
case GAME_TEN:
if (Init_TEN()) {
#ifdef FIXIT_VERSION_3
Options.ScoreVolume = Options.MultiScoreVolume;
#else
Options.ScoreVolume = 0;
#endif
process = false;
Theme.Fade_Out();
} else {
WWMessageBox().Process("Unable to initialize TEN!");
//Prog_End();
Emergency_Exit(1);
}
break;
#endif // TEN
#if(MPATH)
/*
** MPATH: jump straight into the game
*/
case GAME_MPATH:
if (Init_MPATH()) {
#ifdef FIXIT_VERSION_3
Options.ScoreVolume = Options.MultiScoreVolume;
#else
Options.ScoreVolume = 0;
#endif
process = false;
Theme.Fade_Out();
} else {
WWMessageBox().Process("Unable to initialize MPATH!");
//Prog_End();
Emergency_Exit(1);
}
break;
#endif // MPATH
}
break;
/*
** Play a VQ
*/
case SEL_INTRO:
Theme.Fade_Out();
if (Debug_Flag) {
Play_Intro(Debug_Flag);
} else {
Hide_Mouse();
VisiblePage.Clear();
Show_Mouse();
Play_Movie(VQ_INTRO_MOVIE, THEME_NONE, true); // no transition picture to briefing
Keyboard->Clear();
Play_Movie(VQ_SIZZLE, THEME_NONE, true);
Play_Movie(VQ_SIZZLE2, THEME_NONE, true);
// Play_Movie(VQ_INTRO_MOVIE, THEME_NONE, false); // has transitino picture to briefing
}
Theme.Queue_Song(THEME_CRUS);
display = true;
fade = true;
selection = SEL_NONE;
break;
/*
** Exit to DOS.
*/
case SEL_EXIT:
Theme.Fade_Out();
BlackPalette.Set(FADE_PALETTE_SLOW);
return(false);
/*
** Display the hall of fame.
*/
case SEL_FAME:
break;
case SEL_TIMEOUT:
if (Session.Attract && Session.RecordFile.Is_Available()) {
Session.Play = true;
if (Session.RecordFile.Open(READ)) {
Load_Recording_Values(Session.RecordFile);
process = false;
Theme.Fade_Out();
} else {
Session.Play = false;
selection = SEL_NONE;
}
} else {
selection = SEL_NONE;
}
break;
default:
break;
}
}
} else {
/*
** For Debug_Map (editor) mode to load scenario
*/
Scen.Set_Scenario_Name("SCG01EA.INI");
}
/*
** Don't carry stray keystrokes into game.
*/
Keyboard->Clear();
/*
** Initialize the random number generator(s)
*/
Init_Random();
/*
** Save initialization values if we're recording this game.
*/
if (Session.Record) {
if (Session.RecordFile.Open(WRITE)) {
Save_Recording_Values(Session.RecordFile);
} else {
Session.Record = false;
}
}
#ifdef FIXIT_CSII // checked - ajw 9/28/98
switch(Session.Type) {
case GAME_MODEM:
case GAME_NULL_MODEM:
case GAME_IPX:
#ifdef FIXIT_VERSION_3
if( !bAftermathMultiplayer ) {
#else
if (PlayingAgainstVersion < VERSION_AFTERMATH_CS) {
#endif
NewUnitsEnabled = SecretUnitsEnabled = false;
} else {
NewUnitsEnabled = true;
}
// debugprint( "Non Internet game: NewUnitsEnabled = %i\n", NewUnitsEnabled );
break;
case GAME_INTERNET:
#if (0)
if( !pWolapi )
{
// debugprint( "pWolapi is null on internet game!" );
Fatal( "pWolapi is null on internet game!" );
}
//if( pWolapi->bEnableNewAftermathUnits )
if( bAftermathMultiplayer )
NewUnitsEnabled = true;
else
NewUnitsEnabled = SecretUnitsEnabled = false;
// debugprint( "Internet game: NewUnitsEnabled = %i\n", NewUnitsEnabled );
#endif
break;
default:
break;
}
#endif
/*
** Load the scenario. Specify variation 'A' for the editor; for the game,
** don't specify a variation, to make 'Set_Scenario_Name()' pick a random one.
** Skip this if we've already loaded a save-game.
*/
if (!gameloaded && !Session.LoadGame) {
// if (Debug_Map) {
// Set_Scenario_Name(Scen.ScenarioName, Scen.Scenario, Scen.ScenPlayer, Scen.ScenDir, SCEN_VAR_A);
// } else {
// Set_Scenario_Name(Scen.ScenarioName, Scen.Scenario, Scen.ScenPlayer, Scen.ScenDir);
// }
/*
** Start_Scenario() changes the palette; so, fade out & clear the screen
** before calling it.
*/
Hide_Mouse();
if (selection != SEL_START_NEW_GAME) {
BlackPalette.Set(FADE_PALETTE_MEDIUM, Call_Back);
#ifdef WIN32
HiddenPage.Clear();
VisiblePage.Clear();
#else
HidPage.Clear();
SeenPage.Clear();
#endif //WIN32
}
Show_Mouse();
//Mono_Printf("About to call Start Scenario with %s\n", Scen.ScenarioName);
if (!Start_Scenario(Scen.ScenarioName)) {
return(false);
}
if (Special.IsFromInstall) Show_Mouse();
Special.IsFromInstall = false;
}
/*
** For multiplayer games, initialize the inter-player message system.
** Do this after loading the scenario, so the map's upper-left corner is
** properly set.
*/
Session.Messages.Init(
Map.TacPixelX, Map.TacPixelY, // x,y for messages
6, // max # msgs
MAX_MESSAGE_LENGTH-14, // max msg length
7 * RESFACTOR, // font height in pixels
-1, -1, // x,y for edit line (appears above msgs)
0,//BG 1, // enable edit overflow
20, // min,
MAX_MESSAGE_LENGTH - 14, // max for trimming overflow
#ifdef WIN32
Lepton_To_Pixel(Map.TacLeptonWidth)); // Width in pixels of buffer
#else
(320-SIDEBAR_WID)); // Width in pixels of buffer
#endif
if (Session.Type != GAME_NORMAL && Session.Type != GAME_SKIRMISH &&
!Session.Play) {
if (Session.Type == GAME_TEN) {
#if(TEN)
Session.Create_TEN_Connections();
#endif // TEN
} else if (Session.Type == GAME_MPATH) {
#if(MPATH)
Session.Create_MPATH_Connections();
#endif
} else {
Session.Create_Connections();
}
}
/*
** If this isnt an internet game that set the unit build rate to its default value
*/
if (Session.Type != GAME_INTERNET){
UnitBuildPenalty = 100;
}
/*
** Hide the SeenPage; force the map to render one frame. The caller can
** then fade the palette in.
** (If we loaded a game, this step will fade out the title screen. If we
** started a scenario, Start_Scenario() will have played a couple of VQ
** movies, which will have cleared the screen to black already.)
*/
Call_Back();
Hide_Mouse();
BlackPalette.Set(FADE_PALETTE_MEDIUM, Call_Back);
// Fade_Palette_To(BlackPalette, FADE_PALETTE_MEDIUM, Call_Back);
#ifdef WIN32
HiddenPage.Clear();
VisiblePage.Clear();
#else
HidPage.Clear();
SeenPage.Clear();
#endif //WIN32
Show_Mouse();
Set_Logic_Page(SeenBuff);
#ifdef WIN32
/*
** Sidebar is always active in hi-res.
*/
if (!Debug_Map) {
Map.SidebarClass::Activate(1);
}
#endif //WIN32
Map.Flag_To_Redraw();
Call_Back();
Map.Render();
#ifdef WOLAPI_INTEGRATION
//ajw debugging only
// debugprint( "Debugging Session...\n" );
// debugprint( "Session.Players count is %i.\n", Session.Players.Count() );
for (i = 0; i < Session.Players.Count(); i++)
{
NetNumType net;
NetNodeType node;
Session.Players[i]->Address.Get_Address( net, node );
// debugprint( "Player %i, %s, color %i, ip %i.%i.%i.%i.%i.%i\n", i, Session.Players[i]->Name,
// Session.Players[i]->Player.Color, node[0], node[1], node[2], node[3], node[4], node[5] );
}
// debugprint( "PlanetWestwoodPortNumber is %i\n", PlanetWestwoodPortNumber );
#endif
return(true);
}
/***********************************************************************************************
* Play_Intro -- plays the introduction & logo movies *
* *
* INPUT: *
* *
* OUTPUT: *
* none. *
* *
* WARNINGS: *
* none. *
* *
* HISTORY: *
* 06/06/1995 BRR : Created. *
* 05/08/1996 JLB : Modified for Red Alert and direction control. *
*=============================================================================================*/
static void Play_Intro(bool sequenced)
{
static VQType _counter = VQ_FIRST;
Keyboard->Clear();
if (sequenced) {
if (_counter <= VQ_FIRST) _counter = VQ_COUNT;
if (_counter == VQ_COUNT) _counter--;
if (_counter == VQ_REDINTRO) _counter--;
if (_counter == VQ_TITLE) _counter--;
Hide_Mouse();
VisiblePage.Clear();
Show_Mouse();
Play_Movie(VQType(_counter--), THEME_NONE);
// Show_Mouse();
} else {
Hide_Mouse();
VisiblePage.Clear();
Show_Mouse();
#ifdef WIN32
Play_Movie(VQ_REDINTRO, THEME_NONE, false);
#else
Play_Movie(VQ_TITLE, THEME_NONE, false);
#endif
}
}
/***********************************************************************************************
* Anim_Init -- Initialize the VQ animation control structure. *
* *
* VQ animations are controlled by a structure passed to the VQ player. This routine *
* initializes the structure to values required by C&C. *
* *
* INPUT: none *
* *
* OUTPUT: none *
* *
* WARNINGS: Only need to call this routine once at the beginning of the game. *
* *
* HISTORY: *
* 12/20/1994 JLB : Created. *
*=============================================================================================*/
#ifdef WIN32
#ifdef MOVIE640
//GraphicBufferClass VQ640(711, 400, (void *)NULL);
#endif
#endif
void Anim_Init(void)
{
#if (0)
#ifdef WIN32
/* Configure player with INI file */
VQA_DefaultConfig(&AnimControl);
AnimControl.DrawFlags = VQACFGF_TOPLEFT;
AnimControl.DrawFlags |= VQACFGF_BUFFER;
//AnimControl.DrawFlags |= VQACFGF_NODRAW;
//BG - M. Grayford says turn this off AnimControl.DrawFlags |= VQACFGF_NOSKIP;
AnimControl.DrawFlags |= VQACFGF_NOSKIP;
AnimControl.FrameRate = -1;
AnimControl.DrawRate = -1;
AnimControl.DrawerCallback = VQ_Call_Back;
AnimControl.ImageWidth = 320;
AnimControl.ImageHeight = 200;
AnimControl.ImageBuf = (unsigned char *)SysMemPage.Get_Offset();
#ifdef MOVIE640
if(IsVQ640) {
AnimControl.ImageWidth = 711;
AnimControl.ImageHeight = 400;
AnimControl.ImageBuf = (unsigned char *)VQ640.Get_Offset();
}
#endif
AnimControl.Vmode = 0;
AnimControl.OptionFlags |= VQAOPTF_CAPTIONS|VQAOPTF_EVA;
if (SlowPalette) {
AnimControl.OptionFlags |= VQAOPTF_SLOWPAL;
}
AnimControl.SoundObject = SoundObject;
AnimControl.PrimaryBufferPtr = PrimaryBufferPtr;
if (MonoClass::Is_Enabled()) {
AnimControl.OptionFlags |= VQAOPTF_MONO;
}
#else //WIN32
/* Configure player with INI file */
VQA_DefaultConfig(&AnimControl);
// void const * font = Load_Font(FONT8);
// AnimControl.EVAFont = (char *)font;
// AnimControl.CapFont = (char *)font;
AnimControl.DrawerCallback = VQ_Call_Back;
AnimControl.ImageWidth = 320;
AnimControl.ImageHeight = 200;
AnimControl.Vmode = MCGA_MODE;
AnimControl.VBIBit = VertBlank;
AnimControl.DrawFlags |= VQACFGF_TOPLEFT;
AnimControl.OptionFlags |= VQAOPTF_HMIINIT|VQAOPTF_CAPTIONS|VQAOPTF_EVA;
// AnimControl.AudioBuf = (unsigned char *)HidPage.Get_Buffer();
// AnimControl.AudioBufSize = 32768U;
AnimControl.DigiCard = NewConfig.DigitCard;
AnimControl.HMIBufSize = 8192;
AnimControl.DigiHandle = Get_Digi_Handle();
AnimControl.Volume = 0x00FF;
AnimControl.AudioRate = 22050;
// if (NewConfig.Speed) AnimControl.AudioRate = 11025;
if (!Debug_Quiet && Get_Digi_Handle() != -1) {
AnimControl.OptionFlags |= VQAOPTF_AUDIO;
}
if (MonoClass::Is_Enabled()) {
AnimControl.OptionFlags |= VQAOPTF_MONO;
}
#endif //WIN32
#endif
}
/***********************************************************************************************
* Parse_Command_Line -- Parses the command line parameters. *
* *
* This routine should be called before the graphic mode is initialized. It examines the *
* command line parameters and sets the appropriate globals. If there is an error, then *
* it outputs a command summary and then returns false. *
* *
* INPUT: argc -- The number of command line arguments. *
* *
* argv -- Pointer to character string array that holds the individual arguments. *
* *
* OUTPUT: bool; Was the command line parsed successfully? *
* *
* WARNINGS: none *
* *
* HISTORY: *
* 03/18/1995 JLB : Created. *
*=============================================================================================*/
bool Parse_Command_Line(int argc, char * argv[])
{
/*
** Parse the command line and set globals to reflect the parameters
** passed in.
*/
Whom = HOUSE_GOOD;
Special.Init();
Debug_Map = false;
Debug_Unshroud = false;
for (int index = 1; index < argc; index++) {
char * string; // Pointer to argument.
long code = 0;
char arg_string[512];
int str_len = strlen(argv[index]);
char *src = argv[index];
char *dest = arg_string;
for (int i=0 ; i<str_len ; i++) {
if (*src == '\"') {
src++;
} else {
*dest++ = *src++;
}
}
*dest++ = 0;
string = arg_string;
strupr(string);
//string = strupr(argv[index]);
/*
** Print usage text only if requested.
*/
if (stricmp("/?", string) == 0 || stricmp("-?", string) == 0 || stricmp("-h", string) == 0 || stricmp("/h", string) == 0) {
/*
** Unrecognized command line parameter... Display usage
** and then exit.
*/
puts(TEXT_OPTIONS);
return(false);
}
bool processed = true;
long ob = Obfuscate(string);
/*
** Check to see if the parameter is a cheat enabling one.
*/
long const * optr = (const long*)&CheatCodes[0];
while (*optr) {
if (*optr++ == ob) {
Debug_Playtest = true;
Debug_Flag = true;
break;
}
}
/*
** Check to see if the parameter is a cheat enabling one.
*/
optr = (const long*)&PlayCodes[0];
while (*optr) {
if (*optr++ == ob) {
Debug_Playtest = true;
Debug_Flag = true;
break;
}
}
/*
** Check to see if the parameter is a scenario editor
** enabling one.
*/
optr = (const long*)&EditorCodes[0];
while (*optr) {
if (*optr++ == ob) {
Debug_Map = true;
Debug_Unshroud = true;
Debug_Flag = true;
Debug_Playtest = true;
break;
}
}
switch (ob) {
#ifdef VIRGIN_CHEAT_KEYS
case PARM_PLAYTEST:
Debug_Playtest = true;
break;
#endif
/*
** Special flag - is C&C being run from the install program?
*/
case PARM_INSTALL:
Special.IsFromInstall = true;
// If uncommented, will disable the <ESC> key during the first movie run.
// BreakoutAllowed = false;
break;
#if(TEN)
case PARM_ALLOW_SOLO:
Session.AllowSolo = 1;
break;
#endif
#if(MPATH)
case PARM_ALLOW_SOLO:
Session.AllowSolo = 1;
break;
#endif
default:
processed = false;
break;
}
if (processed) continue;
#ifdef CHEAT_KEYS
/*
** Scenario Editor Mode
*/
if (stricmp(string, "-CHECKMAP") == 0) {
Debug_Check_Map = true;
continue;
}
#endif
/*
** File search path override.
*/
if (strstr(string, "-CD")) {
CCFileClass::Set_Search_Drives(&string[3]);
continue;
}
#if (0)
/*
** Build speed modifier
*/
if (strstr (string, "-UNITRATE:")){
int unit_rate;
sscanf (string, "-UNITRATE:%d", &unit_rate);
UnitBuildPenalty = unit_rate;
}
#endif //(0)
/*
** Specify destination connection for network play
*/
if (strstr(string, "-DESTNET")) {
NetNumType net;
NetNodeType node;
/*
** Scan the command-line string, pulling off each address piece
*/
int i = 0;
char * p = strtok(string + 8, ".");
while (p) {
int x;
sscanf(p, "%x", &x); // convert from hex string to int
if (i < 4) {
net[i] = (char)x; // fill NetNum
} else {
node[i-4] = (char)x; // fill NetNode
}
i++;
p = strtok(NULL, ".");
}
/*
** If all the address components were successfully read, fill in the
** BridgeNet with a broadcast address to the network across the bridge.
*/
if (i >= 4) {
Session.IsBridge = 1;
memset(node, 0xff, 6);
Session.BridgeNet = IPXAddressClass(net, node);
}
continue;
}
/*
** Specify socket ID, as an offset from 0x4000.
*/
if (strstr(string, "-SOCKET")) {
unsigned short socket;
socket = (unsigned short)(atoi(string + strlen("SOCKET")));
socket += 0x4000;
if (socket >= 0x4000 && socket < 0x8000) {
Ipx.Set_Socket (socket);
}
continue;
}
/*
** Set the Net Stealth option
*/
if (strstr(string, "-STEALTH")) {
Session.NetStealth = true;
continue;
}
/*
** Set the Net Protection option
*/
if (strstr(string, "-MESSAGES")) {
Session.NetProtect = false;
continue;
}
/*
** Allow "attract" mode
*/
if (strstr(string, "-ATTRACT")) {
Session.Attract = true;
continue;
}
#ifdef WIN32
/*
** Set screen to 640x480 instead of 640x400
*/
if (strstr(string, "-480")) {
ScreenHeight = 480;
continue;
}
/*
** Check for spawn from WChat
*/
#ifndef FIXIT_VERSION_3 // WChat eliminated.
if (strstr(string,"-WCHAT")){
SpawnedFromWChat = true;
}
#endif
#endif
#ifdef CHEAT_KEYS
/*
** Specify the random number seed (for debugging)
*/
if (strstr(string, "-SEED")) {
CustomSeed = (unsigned short)(atoi(string + strlen("SEED")));
continue;
}
#ifndef WIN32
/*
** Don't install Page Fault Handler (MUST use this for debugger)
*/
if (stricmp(string, "-NOPFS") == 0) {
UsePageFaultHandler = 0;
continue;
}
#endif
#endif
#if(TEN)
/*
** Enable TEN
*/
if (strstr(string, "TEN")) {
#ifdef CHEAT_KEYS
Debug_Flag = true;
MonoClass::Enable();
#endif
Session.Type = GAME_TEN;
Special.IsFromInstall = false;
//
// Create the Ten network manager. This allows us to keep
// the packet queues clean even while we're initializing the game,
// so the queues don't fill up in case we're slow, or the user
// didn't insert a CD.
//
Ten = new TenConnManClass();
Ten->Init();
strcpy(Session.OptionsFile, "OPTIONS.INI");
Ten->Flush_All();
continue;
}
/*
** Set the game options filename
*/
if (strstr(string, "OPTIONS:")) {
strcpy(Session.OptionsFile, string + 8);
continue;
}
#endif // TEN
#if(MPATH)
/*
** Enable MPATH
*/
if (strstr(string, "MPATH")) {
#ifdef CHEAT_KEYS
Debug_Flag = true;
MonoClass::Enable();
#endif
Session.Type = GAME_MPATH;
Special.IsFromInstall = false;
//
// Create the MPath network manager. This allows us to keep
// the packet queues clean even while we're initializing the game,
// so the queues don't fill up in case we're slow, or the user
// didn't insert a CD.
//
MPath = new MPlayerManClass();
MPath->Init();
strcpy(Session.OptionsFile, "OPTIONS.INI");
MPath->Flush_All();
continue;
}
/*
** Set the game options filename
*/
if (strstr(string, "OPTIONS:")) {
strcpy(Session.OptionsFile, string + 8);
continue;
}
#endif // MPATH
#ifdef NEVER
/*
** Handle the prog init differently in this case.
*/
if (strstr(string, "-V")) {
continue;
}
#endif
/*
** look for passed-in video mode to default to
*/
#ifndef WIN32
if (strnicmp(string, "-V", strlen("-V")) == 0) {
Set_Video_Mode(MCGA_MODE); // do this to get around first_time variable...
Set_Original_Video_Mode(atoi(string+2));
continue;
}
#endif
#ifdef CHEAT_KEYS
if (strstr(string,"-NOMOVIES")){
bNoMovies = true;
}
#endif
/*
** Special command line control parsing.
*/
if (strnicmp(string, "-X", strlen("-O")) == 0) {
string += strlen("-X");
while (*string) {
char code = *string++;
switch (toupper(code)) {
#ifdef CHEAT_KEYS
/*
** Monochrome debug screen enable.
*/
case 'M':
MonoClass::Enable();
break;
/*
** Inert weapons -- no units take damage.
*/
case 'I':
Special.IsInert = true;
break;
/*
** Hussled recharge timer.
*/
case 'H':
Special.IsSpeedBuild = true;
break;
/*
** "Record" a multi-player game
*/
case 'X':
Session.Record = 1;
break;
/*
** "Play Back" a multi-player game
*/
case 'Y':
Session.Play = 1;
break;
/*
** Print lots of debug stuff about events & packets
*/
case 'P':
Debug_Print_Events = true;
break;
#endif
/*
** Quiet mode override control.
*/
case 'Q':
Debug_Quiet = true;
break;
default:
puts(TEXT_INVALID);
return(false);
}
}
continue;
}
}
return(true);
}
/***********************************************************************************************
* Obfuscate -- Sufficiently transform parameter to thwart casual hackers. *
* *
* This routine borrows from CRC and PGP technology to sufficiently alter the parameter *
* in order to make it difficult to reverse engineer the key phrase. This is designed to *
* be used for hidden game options that will be released at a later time over Westwood's *
* Web page or through magazine hint articles. *
* *
* This algorithm is cryptographically categorized as a "one way hash". *
* *
* Since this is a one way transformation, it becomes much more difficult to reverse *
* engineer the pass phrase even if the resultant pass code is known. This has an added *
* benefit of making this algorithm immune to traditional cryptographic attacks. *
* *
* The largest strength of this transformation algorithm lies in the restriction on the *
* source vector being legal ASCII uppercase characters. This restriction alone makes even *
* a simple CRC transformation practically impossible to reverse engineer. This algorithm *
* uses far more than a simple CRC transformation to achieve added strength from advanced *
* attack methods. *
* *
* INPUT: string -- Pointer to the key phrase that will be transformed into a code. *
* *
* OUTPUT: Returns with the code that the key phrase is translated into. *
* *
* WARNINGS: A zero length pass phrase results in a 0x00000000 result code. *
* *
* HISTORY: *
* 08/19/1995 JLB : Created. *
*=============================================================================================*/
long Obfuscate(char const * string)
{
char buffer[1024];
if (!string) return(0);
memset(buffer, '\xA5', sizeof(buffer));
/*
** Copy key phrase into a working buffer. This hides any transformation done
** to the string.
*/
strncpy(buffer, string, sizeof(buffer));
buffer[sizeof(buffer)-1] = '\0';
int length = strlen(buffer);
/*
** Only upper case letters are significant.
*/
strupr(buffer);
/*
** Ensure that only visible ASCII characters compose the key phrase. This
** discourages the direct forced illegal character input method of attack.
*/
int index;
for (index = 0; index < length; index++) {
if (!isgraph(buffer[index])) {
buffer[index] = 'A' + (index%26);
}
}
/*
** Increase the strength of even short pass phrases by extending the
** length to be at least a minimum number of characters. This helps prevent
** a weak pass phrase from compromising the obfuscation process. This
** process also forces the key phrase to be an even multiple of four.
** This is necessary to support the cypher process that occurs later.
*/
if (length < 16 || (length & 0x03)) {
int maxlen = 16;
if (((length+3) & 0x00FC) > maxlen) {
maxlen = ((length+3) & 0x00FC);
}
for (index = length; index < maxlen; index++) {
buffer[index] = 'A' + ((('?' ^ buffer[index-length]) + index) % 26);
}
length = index;
buffer[length] = '\0';
}
/*
** Transform the buffer into a number. This transformation is character
** order dependant.
*/
long code = Calculate_CRC(buffer, length);
/*
** Record a copy of this initial transformation to be used in a later
** self referential transformation.
*/
long copy = code;
/*
** Reverse the character string and combine with the previous transformation.
** This doubles the workload of trying to reverse engineer the CRC calculation.
*/
strrev(buffer);
code ^= Calculate_CRC(buffer, length);
/*
** Perform a self referential transformation. This makes a reverse engineering
** by using a cause and effect attack more difficult.
*/
code = code ^ copy;
/*
** Unroll and combine the code value into the pass phrase and then perform
** another self referential transformation. Although this is a trivial cypher
** process, it gives the sophisticated hacker false hope since the strong
** cypher process occurs later.
*/
strrev(buffer); // Restore original string order.
for (index = 0; index < length; index++) {
code ^= (unsigned char)buffer[index];
unsigned char temp = (unsigned char)code;
buffer[index] ^= temp;
code >>= 8;
code |= (((long)temp)<<24);
}
/*
** Introduce loss into the vector. This strengthens the key against traditional
** cryptographic attack engines. Since this also weakens the key against
** unconventional attacks, the loss is limited to less than 10%.
*/
for (index = 0; index < length; index++) {
static unsigned char _lossbits[] = {0x00,0x08,0x00,0x20,0x00,0x04,0x10,0x00};
static unsigned char _addbits[] = {0x10,0x00,0x00,0x80,0x40,0x00,0x00,0x04};
buffer[index] |= _addbits[index % (sizeof(_addbits)/sizeof(_addbits[0]))];
buffer[index] &= ~_lossbits[index % (sizeof(_lossbits)/sizeof(_lossbits[0]))];
}
/*
** Perform a general cypher transformation on the vector
** and use the vector itself as the cypher key. This is a variation on the
** cypher process used in PGP. It is a very strong cypher process with no known
** weaknesses. However, in this case, the cypher key is the vector itself and this
** opens up a weakness against attacks that have access to this transformation
** algorithm. The sheer workload of reversing this transformation should be enough
** to discourage even the most determined hackers.
*/
for (index = 0; index < length; index += 4) {
short key1 = buffer[index];
short key2 = buffer[index+1];
short key3 = buffer[index+2];
short key4 = buffer[index+3];
short val1 = key1;
short val2 = key2;
short val3 = key3;
short val4 = key4;
val1 *= key1;
val2 += key2;
val3 += key3;
val4 *= key4;
short s3 = val3;
val3 ^= val1;
val3 *= key1;
short s2 = val2;
val2 ^= val4;
val2 += val3;
val2 *= key3;
val3 += val2;
val1 ^= val2;
val4 ^= val3;
val2 ^= s3;
val3 ^= s2;
buffer[index] = val1;
buffer[index+1] = val2;
buffer[index+2] = val3;
buffer[index+3] = val4;
}
/*
** Convert this final vector into a cypher key code to be
** returned by this routine.
*/
code = Calculate_CRC(buffer, length);
/*
** Return the final code value.
*/
return(code);
}
/***********************************************************************************************
* Calculate_CRC -- Calculates a one-way hash from a data block. *
* *
* This routine is used to create a hash value from a data block. The algorithm is similar *
* to a CRC, but is faster. *
* *
* INPUT: buffer -- Pointer to a buffer of data to be 'hashed'. *
* *
* len -- The length of the buffer to compute the hash upon. *
* *
* OUTPUT: Returns with a 32bit hash value calculated from the specified buffer. *
* *
* WARNINGS: none *
* *
* HISTORY: *
* 03/02/1996 JLB : Created. *
*=============================================================================================*/
extern "C" {
long Calculate_CRC(void * buffer, long len)
{
return(CRCEngine()(buffer, len));
}
}
/***************************************************************************
* Init_Random -- Initializes the random-number generator *
* *
* INPUT: *
* none. *
* *
* OUTPUT: *
* none. *
* *
* WARNINGS: *
* none. *
* *
* HISTORY: *
* 12/04/1995 BRR : Created. *
*=========================================================================*/
void Init_Random(void)
{
#ifdef WIN32
/*
** Gather some "random" bits from the system timer. Actually, only the
** low order millisecond bits are secure. The other bits could be
** easily guessed from the system clock (most clocks are fairly accurate
** and thus predictable).
*/
SYSTEMTIME t;
GetSystemTime(&t);
CryptRandom.Seed_Byte(t.wMilliseconds);
CryptRandom.Seed_Bit(t.wSecond);
CryptRandom.Seed_Bit(t.wSecond>>1);
CryptRandom.Seed_Bit(t.wSecond>>2);
CryptRandom.Seed_Bit(t.wSecond>>3);
CryptRandom.Seed_Bit(t.wSecond>>4);
CryptRandom.Seed_Bit(t.wMinute);
CryptRandom.Seed_Bit(t.wMinute>>1);
CryptRandom.Seed_Bit(t.wMinute>>2);
CryptRandom.Seed_Bit(t.wMinute>>3);
CryptRandom.Seed_Bit(t.wMinute>>4);
CryptRandom.Seed_Bit(t.wHour);
CryptRandom.Seed_Bit(t.wDay);
CryptRandom.Seed_Bit(t.wDayOfWeek);
CryptRandom.Seed_Bit(t.wMonth);
CryptRandom.Seed_Bit(t.wYear);
#else
/*
** Gather some "random" bits from the DOS mode timer.
*/
struct timeb t;
ftime(&t);
CryptRandom.Seed_Byte(t.millitm);
CryptRandom.Seed_Byte(t.time);
#endif
#ifdef FIXIT_MULTI_SAVE
//
// If we've loaded a multiplayer save game, return now; the random #
// class is loaded along with ScenarioClass.
//
if (Session.LoadGame) {
return;
}
//
// If we're playing a recording, the Seed is loaded in
// Load_Recording_Values(). Just init the random # and return.
//
if (Session.Play) {
RandNumb = Seed;
Scen.RandomNumber = Seed;
return;
}
#else
/*
** Do nothing if we've loaded a multiplayer game, or we're playing back
** a recording; the random number generator is initialized by loading
** the game.
*/
if (Session.LoadGame || Session.Play) {
RandNumb = Seed;
Scen.RandomNumber = Seed;
return;
}
#endif // FIXIT_MULTI_SAVE
/*
** Initialize the random number Seed. For multiplayer, this will have been done
** in the connection dialogs. For single-player games, AND if we're not playing
** back a recording, init the Seed to a random value.
*/
if (Session.Type == GAME_NORMAL || Session.Type == GAME_SKIRMISH &&
!Session.Play) {
/*
** Set the optional user-specified seed
*/
if (CustomSeed != 0) {
Seed = CustomSeed;
} else {
srand(time(NULL));
Seed = rand();
}
}
/*
** Initialize the random-number generators
*/
Scen.RandomNumber = Seed;
RandNumb = Seed;
}
/***********************************************************************************************
* Load_Title_Page -- Load the background art for the title page. *
* *
* This routine will load the background art in a machine independent format. There is *
* different art required for the hi-res and lo-res versions of the game. *
* *
* INPUT: visible -- Should the title page art be copied to the visible page by this *
* routine? *
* *
* OUTPUT: none *
* *
* WARNINGS: Be sure the mouse is hidden if the image is to be copied to the visible page. *
* *
* HISTORY: *
* 06/03/1996 JLB : Created. *
*=============================================================================================*/
void Load_Title_Page(bool visible)
{
#ifdef WIN32
Load_Title_Screen("TITLE.PCX", &HidPage, (unsigned char*)CCPalette.Get_Data());
if (visible) {
HidPage.Blit(SeenPage);
}
#else
Load_Picture("TITLE.CPS", HidPage, HidPage, CCPalette, BM_DEFAULT);
if (visible) {
HidPage.Blit(SeenPage);
}
#endif
}
/***********************************************************************************************
* Init_Color_Remaps -- Initialize the text remap tables. *
* *
* There are various color scheme remap tables that are dependant upon the color remap *
* information embedded within the palette control file. This routine will fetch that *
* data and build the text remap tables as indicated. *
* *
* INPUT: none *
* *
* OUTPUT: none *
* *
* WARNINGS: none *
* *
* HISTORY: *
* 06/03/1996 JLB : Created. *
* 09/11/2019 ST : The default resolution doesn't have to match the size of the palette image*
*=============================================================================================*/
static void Init_Color_Remaps(void)
{
/*
** Setup the remap tables. PALETTE.CPS contains a special set of pixels in
** the upper-left corner. Each row of 16 pixels is one range of colors. The
** first row represents unity (the default color units are drawn in); rows
** after that are the remap colors.
*/
#ifdef WIN32
GraphicBufferClass temp_page(320, 200, (void*)NULL);
temp_page.Clear();
Load_Picture("PALETTE.CPS", temp_page, temp_page, NULL, BM_DEFAULT);
temp_page.Blit(HidPage);
#else
Load_Picture("PALETTE.CPS", HidPage, HidPage, NULL, BM_DEFAULT);
#endif
for (PlayerColorType pcolor = PCOLOR_FIRST; pcolor < PCOLOR_COUNT; pcolor++) {
unsigned char * ptr = ColorRemaps[pcolor].RemapTable;
for (int color = 0; color < 256; color++) {
ptr[color] = color;
}
int index;
for (index = 0; index < 16; index++) {
ptr[HidPage.Get_Pixel(index, 0)] = HidPage.Get_Pixel(index, pcolor);
}
for (index = 0; index < 6; index++) {
ColorRemaps[pcolor].FontRemap[10+index] = HidPage.Get_Pixel(2+index, pcolor);
}
ColorRemaps[pcolor].BrightColor = WHITE;
// ColorRemaps[pcolor].BrightColor = HidPage.Get_Pixel(1, pcolor);
ColorRemaps[pcolor].Color = HidPage.Get_Pixel(4, pcolor);
ColorRemaps[pcolor].Shadow = HidPage.Get_Pixel(10, pcolor);
ColorRemaps[pcolor].Background = HidPage.Get_Pixel(9, pcolor);
ColorRemaps[pcolor].Corners = HidPage.Get_Pixel(7, pcolor);
ColorRemaps[pcolor].Highlight = HidPage.Get_Pixel(4, pcolor);
ColorRemaps[pcolor].Bright = HidPage.Get_Pixel(0, pcolor);
ColorRemaps[pcolor].Underline = HidPage.Get_Pixel(0, pcolor);
ColorRemaps[pcolor].Bar = HidPage.Get_Pixel(6, pcolor);
/*
** This must grab from column 4 because the multiplayer color dialog palette counts
** on this to be true.
*/
ColorRemaps[pcolor].Box = HidPage.Get_Pixel(4, pcolor);
}
/* 12/9/2019 SKY - Swap Blue and Grey color remaps */
{
RemapControlType temp;
memcpy(&temp, &ColorRemaps[PCOLOR_BLUE], sizeof(RemapControlType));
memcpy(&ColorRemaps[PCOLOR_BLUE], &ColorRemaps[PCOLOR_GREY], sizeof(RemapControlType));
memcpy(&ColorRemaps[PCOLOR_GREY], &temp, sizeof(RemapControlType));
}
/*
** Now do the special dim grey scheme
*/
for (int color = 0; color < 256; color++) {
GreyScheme.RemapTable[color] = color;
}
for (int index = 0; index < 6; index++) {
GreyScheme.FontRemap[10+index] = HidPage.Get_Pixel(9+index, PCOLOR_GREY) & 0x00FF;
}
GreyScheme.BrightColor = HidPage.Get_Pixel(3, PCOLOR_GREY) & 0x00FF;
GreyScheme.Color = HidPage.Get_Pixel(7, PCOLOR_GREY) & 0x00FF;
GreyScheme.Shadow = ColorRemaps[PCOLOR_GREY].RemapTable[HidPage.Get_Pixel(15, PCOLOR_GREY) & 0x00FF];
GreyScheme.Background = ColorRemaps[PCOLOR_GREY].RemapTable[HidPage.Get_Pixel(14, PCOLOR_GREY) & 0x00FF];
GreyScheme.Corners = ColorRemaps[PCOLOR_GREY].RemapTable[HidPage.Get_Pixel(13, PCOLOR_GREY) & 0x00FF];
GreyScheme.Highlight = ColorRemaps[PCOLOR_GREY].RemapTable[HidPage.Get_Pixel(9, PCOLOR_GREY) & 0x00FF];
GreyScheme.Bright = ColorRemaps[PCOLOR_GREY].RemapTable[HidPage.Get_Pixel(5, PCOLOR_GREY) & 0x00FF];
GreyScheme.Underline = ColorRemaps[PCOLOR_GREY].RemapTable[HidPage.Get_Pixel(5, PCOLOR_GREY) & 0x00FF];
GreyScheme.Bar = ColorRemaps[PCOLOR_GREY].RemapTable[HidPage.Get_Pixel(11, PCOLOR_GREY) & 0x00FF];
GreyScheme.Box = ColorRemaps[PCOLOR_GREY].RemapTable[HidPage.Get_Pixel(11, PCOLOR_GREY) & 0x00FF];
/*
** Set up the metallic remap table for the font that prints over the tabs
*/
memset ((void*)&MetalScheme, 4, sizeof(MetalScheme));
for (int color_counter = 0; color_counter < 16; color_counter++) {
MetalScheme.FontRemap[color_counter] = color_counter;
}
MetalScheme.FontRemap[1] = 128;
MetalScheme.FontRemap[2] = 12;
MetalScheme.FontRemap[3] = 13;
MetalScheme.FontRemap[4] = 14;
MetalScheme.Color = 128;
MetalScheme.Background = 0;
MetalScheme.Underline = 128;
/*
** Set up the font remap table for the mission briefing font
*/
for (int colr = 0; colr < 16; colr++) {
ColorRemaps[PCOLOR_TYPE].FontRemap[colr] = HidPage.Get_Pixel(colr, PCOLOR_TYPE);
}
ColorRemaps[PCOLOR_TYPE].Shadow = 11;
ColorRemaps[PCOLOR_TYPE].Background = 10;
ColorRemaps[PCOLOR_TYPE].Corners = 10;
ColorRemaps[PCOLOR_TYPE].Highlight = 9;
ColorRemaps[PCOLOR_TYPE].Bright = 15;
ColorRemaps[PCOLOR_TYPE].Underline = 11;
ColorRemaps[PCOLOR_TYPE].Bar = 11;
ColorRemaps[PCOLOR_TYPE].Box = 10;
ColorRemaps[PCOLOR_TYPE].BrightColor = 15;
ColorRemaps[PCOLOR_TYPE].Color = 9;
GadgetClass::Set_Color_Scheme(&ColorRemaps[PCOLOR_DIALOG_BLUE]);
// GadgetClass::Set_Color_Scheme(&ColorRemaps[PCOLOR_BLUE]);
}
/***********************************************************************************************
* Init_Heaps -- Initialize the game heaps and buffers. *
* *
* This routine will allocate the game heaps and buffers. The rules file has already been *
* processed by the time that this routine is called. *
* *
* INPUT: none *
* *
* OUTPUT: none *
* *
* WARNINGS: none *
* *
* HISTORY: *
* 06/03/1996 JLB : Created. *
*=============================================================================================*/
static void Init_Heaps(void)
{
/*
** Initialize the game object heaps.
*/
Vessels.Set_Heap(Rule.VesselMax);
Units.Set_Heap(Rule.UnitMax);
Factories.Set_Heap(Rule.FactoryMax);
Terrains.Set_Heap(Rule.TerrainMax);
Templates.Set_Heap(Rule.TemplateMax);
Smudges.Set_Heap(Rule.SmudgeMax);
Overlays.Set_Heap(Rule.OverlayMax);
Infantry.Set_Heap(Rule.InfantryMax);
Bullets.Set_Heap(Rule.BulletMax);
Buildings.Set_Heap(Rule.BuildingMax);
Anims.Set_Heap(Rule.AnimMax);
Aircraft.Set_Heap(Rule.AircraftMax);
Triggers.Set_Heap(Rule.TriggerMax);
TeamTypes.Set_Heap(Rule.TeamTypeMax);
Teams.Set_Heap(Rule.TeamMax);
Houses.Set_Heap(HOUSE_MAX);
TriggerTypes.Set_Heap(Rule.TrigTypeMax);
// Weapons.Set_Heap(Rule.WeaponMax);
/*
** Speech holding tank buffer. Since speech does not mix, it can be placed
** into a custom holding tank only as large as the largest speech file to
** be played.
*/
for (int index = 0; index < ARRAY_SIZE(SpeechBuffer); index++) {
SpeechBuffer[index] = new char [SPEECH_BUFFER_SIZE];
SpeechRecord[index] = VOX_NONE;
assert(SpeechBuffer[index] != NULL);
}
/*
** Allocate the theater buffer block.
*/
TheaterBuffer = new Buffer(THEATER_BUFFER_SIZE);
assert(TheaterBuffer != NULL);
}
/***********************************************************************************************
* Init_Expansion_Files -- Fetch any override expansion mixfiles. *
* *
* This routine will search for and register/cache any override mixfiles found. *
* *
* INPUT: none *
* *
* OUTPUT: none *
* *
* WARNINGS: none *
* *
* HISTORY: *
* 06/03/1996 JLB : Created. *
*=============================================================================================*/
static void Init_Expansion_Files(void)
{
/*
** Need to search the search paths. ST - 3/15/2019 2:18PM
*/
const char *path = ".\\";
char search_path[_MAX_PATH];
char scan_path[_MAX_PATH];
for (int p = 0; p < 100; p++) {
strcpy(search_path, path);
if (search_path[strlen(search_path) - 1] != '\\') {
strcat(search_path, "\\");
}
strcpy(scan_path, search_path);
strcat(scan_path, "SC*.MIX");
WIN32_FIND_DATA find_data;
memset(&find_data, 0, sizeof(find_data));
HANDLE file_handle = FindFirstFile(scan_path, &find_data);
if (file_handle != INVALID_HANDLE_VALUE)
{
do
{
char *ptr = strdup(find_data.cFileName);
new MFCD(ptr, &FastKey);
} while (FindNextFile(file_handle, &find_data));
FindClose(file_handle);
}
memset(&find_data, 0, sizeof(find_data));
strcpy(scan_path, search_path);
strcat(scan_path, "Ss*.MIX");
file_handle = FindFirstFile(scan_path, &find_data);
if (file_handle != INVALID_HANDLE_VALUE)
{
do
{
char *ptr = strdup(find_data.cFileName);
new MFCD(ptr, &FastKey);
} while (FindNextFile(file_handle, &find_data));
FindClose(file_handle);
}
path = CDFileClass::Get_Search_Path(p);
if (path == NULL) {
break;
}
}
#if (0)
/*
** Before all else, cache any additional mixfiles.
*/
struct find_t ff; // for _dos_findfirst
if (!_dos_findfirst("SC*.MIX", _A_NORMAL, &ff)) {
char * ptr;
do {
ptr = strdup(ff.name);
new MFCD(ptr, &FastKey);
MFCD::Cache(ptr);
} while (!_dos_findnext(&ff));
}
if (!_dos_findfirst("SS*.MIX", _A_NORMAL, &ff)) {
char * ptr;
do {
ptr = strdup(ff.name);
new MFCD(ptr, &FastKey);
} while (!_dos_findnext(&ff));
}
#endif
}
/***********************************************************************************************
* Init_One_Time_Systems -- Initialize internal pointers to the bulk data. *
* *
* This performs the one-time processing required after the bulk data has been cached but *
* before the game actually starts. Typically, this routine extracts pointers to all the *
* embedded data sub-files within the main game data mixfile. This routine must be called *
* AFTER the bulk data has been cached. *
* *
* INPUT: none *
* *
* OUTPUT: none *
* *
* WARNINGS: Call this routine AFTER the bulk data has been cached. *
* *
* HISTORY: *
* 06/03/1996 JLB : Created. *
*=============================================================================================*/
static void Init_One_Time_Systems(void)
{
Call_Back();
Map.One_Time();
Logic.One_Time();
Options.One_Time();
Session.One_Time();
ObjectTypeClass::One_Time();
BuildingTypeClass::One_Time();
BulletTypeClass::One_Time();
HouseTypeClass::One_Time();
TemplateTypeClass::One_Time();
OverlayTypeClass::One_Time();
SmudgeTypeClass::One_Time();
TerrainTypeClass::One_Time();
UnitTypeClass::One_Time();
VesselTypeClass::One_Time();
InfantryTypeClass::One_Time();
AnimTypeClass::One_Time();
AircraftTypeClass::One_Time();
HouseClass::One_Time();
}
/***********************************************************************************************
* Init_Fonts -- Initialize all the game font pointers. *
* *
* This routine is used to fetch pointers to the game fonts. The mixfile containing these *
* fonts must have been previously cached. This routine is a necessary prerequisite to *
* displaying any dialogs or printing any text. *
* *
* INPUT: none *
* *
* OUTPUT: none *
* *
* WARNINGS: none *
* *
* HISTORY: *
* 06/03/1996 JLB : Created. *
*=============================================================================================*/
static void Init_Fonts(void)
{
Metal12FontPtr = MFCD::Retrieve("12METFNT.FNT");
MapFontPtr = MFCD::Retrieve("HELP.FNT");
Font6Ptr = MFCD::Retrieve("6POINT.FNT");
GradFont6Ptr = MFCD::Retrieve("GRAD6FNT.FNT");
EditorFont = MFCD::Retrieve("EDITFNT.FNT");
Font8Ptr = MFCD::Retrieve("8POINT.FNT");
FontPtr = (char *)Font8Ptr;
Set_Font(FontPtr);
Font3Ptr = MFCD::Retrieve("3POINT.FNT");
ScoreFontPtr = MFCD::Retrieve("SCOREFNT.FNT");
FontLEDPtr = MFCD::Retrieve("LED.FNT");
VCRFontPtr = MFCD::Retrieve("VCR.FNT");
TypeFontPtr = MFCD::Retrieve("8POINT.FNT"); //("TYPE.FNT"); //VG 10/17/96
}
/***********************************************************************************************
* Init_CDROM_Access -- Initialize the CD-ROM access handler. *
* *
* This routine is called to setup the CD-ROM access or emulation handler. It will ensure *
* that the appropriate CD-ROM is present (dependant on the RequiredCD global). *
* *
* INPUT: none *
* *
* OUTPUT: none *
* *
* WARNINGS: The fonts, palettes, and other bootstrap systems must have been initialized *
* prior to calling this routine since this routine will quite likely display *
* a dialog box requesting the appropriate CD be inserted. *
* *
* HISTORY: *
* 06/03/1996 JLB : Created. *
*=============================================================================================*/
static void Init_CDROM_Access(void)
{
VisiblePage.Clear();
HidPage.Clear();
#ifdef FIXIT_VERSION_3
// Determine if we're going to be running from a DVD.
// The entire session will either require a DVD, or the regular CDs. Never both.
// Call Using_DVD() to determine which case it is.
// Here we set the value that Using_DVD() returns.
Determine_If_Using_DVD();
// Force_CD_Available() is modified when Using_DVD() is true so that all requests become requests for the DVD.
#endif
/*
** Always try to look at the CD-ROM for data files.
*/
if (!CCFileClass::Is_There_Search_Drives()) {
/*
** This call is needed because of a side effect of this function. It will examine the
** CD-ROMs attached to this computer and set the appropriate status values. Without this
** call, the "?:\\" could not be filled in correctly.
*/
Force_CD_Available(-1);
/*
** If there are no search drives specified then we must be playing
** off cd, so read files from there.
*/
int error;
do {
error = CCFileClass::Set_Search_Drives("?:\\");
switch (error) {
case 1:
VisiblePage.Clear();
GamePalette.Set();
Show_Mouse();
WWMessageBox().Process(TXT_CD_ERROR1, TXT_OK);
Prog_End("Init_CDROM_Access - CD_ERROR1", true);
Emergency_Exit(EXIT_FAILURE);
case 2:
VisiblePage.Clear();
GamePalette.Set();
Show_Mouse();
if (WWMessageBox().Process(TXT_CD_DIALOG_1, TXT_OK, TXT_CANCEL) == 1) {
Prog_End("Init_CDROM_Access - CD_ERROR2", true);
Emergency_Exit(EXIT_FAILURE);
}
Hide_Mouse();
break;
default:
VisiblePage.Clear();
Show_Mouse();
if (!Force_CD_Available(RequiredCD)) {
Prog_End("Init_CDROM_Access - Force_CD_Available failed", true);
Emergency_Exit(EXIT_FAILURE);
}
Hide_Mouse();
break;
}
} while (error);
RequiredCD = -1;
} else {
/*
** If there are search drives specified then all files are to be
** considered local.
*/
RequiredCD = -2;
}
}
/***********************************************************************************************
* Init_Bootstrap_Mixfiles -- Registers and caches any mixfiles needed for bootstrapping. *
* *
* This routine will register the initial mixfiles that are required to display error *
* messages and get input from the player. *
* *
* INPUT: none *
* *
* OUTPUT: none *
* *
* WARNINGS: Be sure to call this routine before any dialogs would be displayed to the *
* player. *
* *
* HISTORY: *
* 06/03/1996 JLB : Created. *
*=============================================================================================*/
static void Init_Bootstrap_Mixfiles(void)
{
int temp = RequiredCD;
RequiredCD = -2;
#ifdef WOLAPI_INTEGRATION
CCFileClass fileWolapiMix( "WOLAPI.MIX" );
if( fileWolapiMix.Is_Available() )
{
new MFCD( "WOLAPI.MIX", &FastKey );
MFCD::Cache( "WOLAPI.MIX" );
}
#endif
#ifdef FIXIT_CSII // Ok. ajw
CCFileClass file2("EXPAND2.MIX");
if (file2.Is_Available()) {
new MFCD("EXPAND2.MIX", &FastKey);
bool ok = MFCD::Cache("EXPAND2.MIX");
assert(ok);
}
#endif
#ifdef FIXIT_CSII // Ok. ajw
bool ok1;
#if 0
new MFCD("HIRES1.MIX", &FastKey);
ok1 = MFCD::Cache("HIRES1.MIX");
assert(ok1);
#else
new MFCD("LORES1.MIX", &FastKey);
ok1 = MFCD::Cache("LORES1.MIX");
assert(ok1);
#endif
#endif
#ifdef FIXIT_ANTS // Ok. ajw
CCFileClass file("EXPAND.MIX");
if (file.Is_Available()) {
new MFCD("EXPAND.MIX", &FastKey);
bool ok = MFCD::Cache("EXPAND.MIX");
assert(ok);
}
#endif
new MFCD("REDALERT.MIX", &FastKey);
/*
** Bootstrap enough of the system so that the error dialog box can successfully
** be displayed.
*/
new MFCD("LOCAL.MIX", &FastKey); // Cached.
bool ok = MFCD::Cache("LOCAL.MIX");
assert(ok);
#if 0
new MFCD("HIRES.MIX", &FastKey);
ok = MFCD::Cache("HIRES.MIX");
assert(ok);
new MFCD("NCHIRES.MIX", &FastKey); //Non-cached hires stuff incl VQ palettes
#else
new MFCD("LORES.MIX", &FastKey);
ok = MFCD::Cache("LORES.MIX");
assert(ok);
#endif //WIN32
RequiredCD = temp;
}
/***********************************************************************************************
* Init_Secondary_Mixfiles -- Register and cache secondary mixfiles. *
* *
* This routine is used to register the mixfiles that are needed for main menu processing. *
* Call this routine before the main menu is display and processed. *
* *
* INPUT: none *
* *
* OUTPUT: none *
* *
* WARNINGS: none *
* *
* HISTORY: *
* 06/03/1996 JLB : Created. *
*=============================================================================================*/
//#define DENZIL_MIXEXTRACT
#ifdef DENZIL_MIXEXTRACT
void Extract(char* filename, char* outfile);
#endif
static void Init_Secondary_Mixfiles(void)
{
MainMix = new MFCD("MAIN.MIX", &FastKey);
assert(MainMix != NULL);
//Denzil extract mixfile
#ifdef DENZIL_MIXEXTRACT
#if(0)
Extract("CONQUER.MIX", "o:\\projects\\radvd\\data\\extract\\conquer.mix");
Extract("EDHI.MIX", "o:\\projects\\radvd\\data\\extract\\edhi.mix");
Extract("EDLO.MIX", "o:\\projects\\radvd\\data\\extract\\edlo.mix");
Extract("GENERAL.MIX", "o:\\projects\\radvd\\data\\extract\\general.mix");
Extract("INTERIOR.MIX", "o:\\projects\\radvd\\data\\extract\\interior.mix");
Extract("MOVIES1.MIX", "o:\\projects\\radvd\\data\\extract\\movies1.mix");
Extract("SCORES.MIX", "o:\\projects\\radvd\\data\\extract\\scores.mix");
Extract("SNOW.MIX", "o:\\projects\\radvd\\data\\extract\\snow.mix");
Extract("SOUNDS.MIX", "o:\\projects\\radvd\\data\\extract\\sounds.mix");
Extract("RUSSIAN.MIX", "o:\\projects\\radvd\\data\\extract\\russian.mix");
Extract("ALLIES.MIX", "o:\\projects\\radvd\\data\\extract\\allies.mix");
Extract("TEMPERAT.MIX", "o:\\projects\\radvd\\data\\extract\\temperat.mix");
#else
Extract("CONQUER.MIX", "o:\\projects\\radvd\\data\\extract\\conquer.mix");
Extract("EDHI.MIX", "o:\\projects\\radvd\\data\\extract\\edhi.mix");
Extract("EDLO.MIX", "o:\\projects\\radvd\\data\\extract\\edlo.mix");
Extract("GENERAL.MIX", "o:\\projects\\radvd\\data\\extract\\general.mix");
Extract("INTERIOR.MIX", "o:\\projects\\radvd\\data\\extract\\interior.mix");
Extract("MOVIES2.MIX", "o:\\projects\\radvd\\data\\extract\\movies2.mix");
Extract("SCORES.MIX", "o:\\projects\\radvd\\data\\extract\\scores.mix");
Extract("SNOW.MIX", "o:\\projects\\radvd\\data\\extract\\snow.mix");
Extract("SOUNDS.MIX", "o:\\projects\\radvd\\data\\extract\\sounds.mix");
Extract("RUSSIAN.MIX", "o:\\projects\\radvd\\data\\extract\\russian.mix");
Extract("ALLIES.MIX", "o:\\projects\\radvd\\data\\extract\\allies.mix");
Extract("TEMPERAT.MIX", "o:\\projects\\radvd\\data\\extract\\temperat.mix");
#endif
#endif
/*
** Inform the file system of the various MIX files.
*/
ConquerMix = new MFCD("CONQUER.MIX", &FastKey); // Cached.
// new MFCD("TRANSIT.MIX", &FastKey);
if (GeneralMix == NULL) GeneralMix = new MFCD("GENERAL.MIX", &FastKey); // Never cached.
if (CCFileClass("MOVIES1.MIX").Is_Available()) {
MoviesMix = new MFCD("MOVIES1.MIX", &FastKey); // Never cached.
} else {
MoviesMix = new MFCD("MOVIES2.MIX", &FastKey); // Never cached.
}
assert(MoviesMix != NULL);
/*
** Register the score mixfile.
*/
ScoresPresent = true;
ScoreMix = new MFCD("SCORES.MIX", &FastKey);
ThemeClass::Scan();
/*
** These are sound card specific, but the install program would have
** copied the correct versions to the hard drive.
*/
new MFCD("SPEECH.MIX", &FastKey); // Never cached.
new MFCD("SOUNDS.MIX", &FastKey); // Cached.
new MFCD("RUSSIAN.MIX", &FastKey); // Cached.
new MFCD("ALLIES.MIX", &FastKey); // Cached.
}
/***********************************************************************************************
* Bootstrap -- Perform the initial bootstrap procedure. *
* *
* This routine will load and initialize the game engine such that a dialog box could be *
* displayed. Because this is very critical, call this routine before any other game *
* initialization code. *
* *
* INPUT: none *
* *
* OUTPUT: none *
* *
* WARNINGS: none *
* *
* HISTORY: *
* 06/03/1996 JLB : Created. *
*=============================================================================================*/
static void Bootstrap(void)
{
BlackPalette.Set();
/*
** Be sure to short circuit the CD-ROM check if there is a CD-ROM override
** path.
*/
if (CCFileClass::Is_There_Search_Drives()) {
RequiredCD = -2;
}
/*
** Process the message loop until we are in focus. We need to be in focus to read pixels from
** the screen.
*/
#if (0) //PG
#ifdef WIN32
do {
Keyboard->Check();
} while (!GameInFocus);
AllSurfaces.SurfacesRestored = false;
#endif
/*
** Perform any special debug-only processing. This includes preparing the
** monochrome screen.
*/
Mono_Clear_Screen();
#endif
/*
** Register and make resident all local mixfiles with particular emphasis
** on the mixfiles that are necessary to display and error messages and
** process further initialization.
*/
Init_Bootstrap_Mixfiles();
/*
** Initialize the resident font pointers.
*/
Init_Fonts();
#ifndef WIN32
/*
** Install the hard error handler.
*/
_harderr(harderr_handler); // BG: Install hard error handler
/*
** Install a Page Fault handler
*/
if (UsePageFaultHandler) {
Install_Page_Fault_Handle();
}
#endif
/*
** Setup the keyboard processor in preparation for the game.
*/
#ifdef WIN32
Keyboard->Clear();
#else
Keyboard_Attributes_Off(BREAKON | SCROLLLOCKON | TRACKEXT | PAUSEON | CTRLSON | CTRLCON | FILTERONLY | TASKSWITCHABLE);
Keyboard_Attributes_On(PASSBREAKS);
Keyboard->Clear();
#endif
/*
** This is the shape staging buffer. It must always be available, so it is
** allocated here and never freed. The library sets the globals ShapeBuffer
** and ShapeBufferSize to these values, so it can be accessed for other
** purposes.
*/
Set_Shape_Buffer(new unsigned char[SHAPE_BUFFER_SIZE], SHAPE_BUFFER_SIZE);
/*
** Fetch the language text from the hard drive first. If it cannot be
** found on the hard drive, then look for it in the mixfile.
*/
#ifdef STEVES_LOAD_OVERRIDE
RawFileClass strings ("CONQUER.ENG");
if (strings.Is_Available()){
SystemStrings = new char [strings.Size()];
strings.Read((void*)SystemStrings, strings.Size());
}else{
SystemStrings = (char const *)MFCD::Retrieve(Language_Name("CONQUER"));
}
#else
SystemStrings = (char const *)MFCD::Retrieve(Language_Name("CONQUER"));
#endif
DebugStrings = (char const *)MFCD::Retrieve("DEBUG.ENG");
/*
** Default palette initialization.
*/
// PG_TO_FIX. This doesn't seem right. ST - 5/9/2019
//memmove((unsigned char *)&GamePalette[0], (void *)MFCD::Retrieve("TEMPERAT.PAL"), 768L);
//WhitePalette[0] = BlackPalette[0];
// GamePalette.Set();
/*
** Initialize expansion files (if present). Expansion files must be located
** in the current directory.
*/
Init_Expansion_Files();
SidebarScheme.Background = BLACK;
SidebarScheme.Corners = LTGREY;
SidebarScheme.Shadow = DKGREY;
SidebarScheme.Highlight = WHITE;
SidebarScheme.Color = LTGREY;
SidebarScheme.Bright = WHITE;
SidebarScheme.BrightColor = WHITE;
SidebarScheme.Box = LTGREY;
GadgetClass::Set_Color_Scheme(&SidebarScheme);
}
/***********************************************************************************************
* Init_Mouse -- Initialize the mouse system. *
* *
* This routine will ensure that a valid mouse driver is present and a working mouse *
* pointer can be displayed. The mouse is hidden when this routine exits. *
* *
* INPUT: none *
* *
* OUTPUT: none *
* *
* WARNINGS: none *
* *
* HISTORY: *
* 06/03/1996 JLB : Created. *
*=============================================================================================*/
static void Init_Mouse(void)
{
/*
** Since there is no mouse shape currently available we need
** to set one of our own.
*/
#ifdef WIN32
ShowCursor(false);
#endif
if (MouseInstalled) {
void const * temp_mouse_shapes = MFCD::Retrieve("MOUSE.SHP");
if (temp_mouse_shapes) {
Set_Mouse_Cursor(0, 0, Extract_Shape(temp_mouse_shapes, 0));
while (Get_Mouse_State() > 1) Show_Mouse();
}
} else {
char buffer[255];
GamePalette.Set();
GamePalette.Set();
sprintf(buffer, TEXT_NO_MOUSE);
VisiblePage.Clear();
WWMessageBox().Process(buffer, TXT_OK);
Prog_End("Init_Mouse", true);
Emergency_Exit(1);
}
Map.Set_Default_Mouse(MOUSE_NORMAL, false);
Show_Mouse();
while (Get_Mouse_State() > 1) Show_Mouse();
Call_Back();
Hide_Mouse();
}
#ifdef OBSOLETE
/***********************************************************************************************
* Init_Authorization -- Verifies that the player is authorized to play the game. *
* *
* This is a development routine that restricts access to the game by way of passwords. *
* *
* INPUT: none *
* *
* OUTPUT: none *
* *
* WARNINGS: none *
* *
* HISTORY: *
* 06/03/1996 JLB : Created. *
*=============================================================================================*/
static void Init_Authorization(void)
{
if (Special.IsFromInstall) return;
Load_Title_Page();
#ifdef WIN32
Wait_Vert_Blank();
#else //WIN32
Init_Delay();
Wait_Vert_Blank(VertBlank);
#endif //WIN32
CCPalette.Set();
// Set_Palette(Palette);
HidPage.Blit(SeenPage);
Show_Mouse();
/*
** Fetch the type of game to be played. This will be either
** C&C:Red Alert or C&C:Plus.
*/
//tryagain:
bool ok = Debug_Flag;
int counter = 3;
if (Debug_Flag) ok = true;
/*
** C&C:Red Alert requires a password for legal entry. Try (three times) to get a correct
** password. If not found, then try again.
*/
bool skipper = false;
#ifdef NEVER
while (!ok && counter) {
SmartPtr<char const> str = Fetch_Password(TXT_PASSWORD_CAPTION, TXT_PASSWORD_MESSAGE, TXT_OK);
SmartPtr<long const> lptr = &CheatCodes[0];
while (*lptr) {
if (Obfuscate(str) == *lptr++) {
ok = true;
break;
}
}
lptr = &EditorCodes[0];
while (*lptr) {
if (Obfuscate(str) == *lptr++) {
ok = true;
break;
}
}
lptr = &PlayCodes[0];
while (*lptr) {
if (Obfuscate(str) == *lptr++) {
ok = true;
skipper = true;
break;
}
}
if (ok) break;
Hide_Mouse();
Load_Title_Page();
HidPage.Blit(SeenPage);
Show_Mouse();
Delay(TIMER_SECOND*(4-counter)*1);
if (WWMessageBox().Process(TXT_PASSWORD_ERROR, TXT_TRY_AGAIN, TXT_CANCEL)) {
goto tryagain;
}
counter--;
if (counter == 0) goto tryagain;
}
#endif
if (!skipper) {
CCPalette.Set();
}
Hide_Mouse();
Load_Title_Page();
HidPage.Blit(SeenPage);
Show_Mouse();
Call_Back();
}
#endif
/***********************************************************************************************
* Init_Bulk_Data -- Initialize the time-consuming mixfile caching. *
* *
* This routine is called to handle the time consuming process of game initialization. *
* The title page will be displayed when this routine is called. *
* *
* INPUT: none *
* *
* OUTPUT: none *
* *
* WARNINGS: This routine will take a very long time. *
* *
* HISTORY: *
* 06/03/1996 JLB : Created. *
*=============================================================================================*/
static void Init_Bulk_Data(void)
{
/*
** Cache the main game data. This operation can take a very long time.
*/
MFCD::Cache("CONQUER.MIX");
if (SampleType != 0 && !Debug_Quiet) {
MFCD::Cache("SOUNDS.MIX");
MFCD::Cache("RUSSIAN.MIX");
MFCD::Cache("ALLIES.MIX");
}
Call_Back();
/*
** Fetch the tutorial message data.
*/
INIClass ini;
ini.Load(CCFileClass("TUTORIAL.INI"));
for (int index = 0; index < ARRAY_SIZE(TutorialText); index++) {
TutorialText[index] = NULL;
char buffer[128];
char num[10];
sprintf(num, "%d", index);
if (ini.Get_String("Tutorial", num, "", buffer, sizeof(buffer))) {
TutorialText[index] = strdup(buffer);
}
}
/*
** Perform one-time game system initializations.
*/
Init_One_Time_Systems();
}
/***********************************************************************************************
* Init_Keys -- Initialize the cryptographic keys. *
* *
* This routine will initialize the fast cryptographic key. It will also initialize the *
* slow one if this is a scenario editor version of the game. *
* *
* INPUT: none *
* *
* OUTPUT: none *
* *
* WARNINGS: none *
* *
* HISTORY: *
* 07/08/1996 JLB : Created. *
*=============================================================================================*/
static void Init_Keys(void)
{
RAMFileClass file((void*)Keys, strlen(Keys));
INIClass ini;
ini.Load(file);
FastKey = ini.Get_PKey(true);
#ifdef SCENARIO_EDITOR
SlowKey = ini.Get_PKey(false);
#endif
}
/***************************************************************************
* Save_Recording_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_Recording_Values(CCFileClass & file)
{
Session.Save(file);
file.Write(&BuildLevel, sizeof(BuildLevel));
file.Write(&Debug_Unshroud, sizeof(Debug_Unshroud));
file.Write(&Seed, sizeof(Seed));
file.Write(&Scen.Scenario, sizeof(Scen.Scenario));
file.Write(Scen.ScenarioName, sizeof(Scen.ScenarioName));
file.Write(&Whom, sizeof(Whom));
file.Write(&Special, sizeof(SpecialClass));
file.Write(&Options, sizeof(GameOptionsClass));
return (true);
}
/***************************************************************************
* Load_Recording_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_Recording_Values(CCFileClass & file)
{
Session.Load(file);
file.Read(&BuildLevel, sizeof(BuildLevel));
file.Read(&Debug_Unshroud, sizeof(Debug_Unshroud));
file.Read(&Seed, sizeof(Seed));
file.Read(&Scen.Scenario, sizeof(Scen.Scenario));
file.Read(Scen.ScenarioName, sizeof(Scen.ScenarioName));
file.Read(&Whom, sizeof(Whom));
file.Read(&Special, sizeof(SpecialClass));
file.Read(&Options, sizeof(GameOptionsClass));
return (true);
}
extern "C" {
void __PRO(void) {
// printf("_pro\n");
}
}
#ifdef DENZIL_MIXEXTRACT
void Extract(char* filename, char* outname)
{
CCFileClass inFile(filename);
CCFileClass outFile(outname);
inFile.Open();
outFile.Open(WRITE);
void* buffer = malloc(32768);
if (buffer)
{
unsigned long size = inFile.Size();
unsigned long bytes;
while (size > 0)
{
bytes = inFile.Read(buffer, 32768);
outFile.Write(buffer, bytes);
size -= bytes;
}
free(buffer);
}
}
#endif
#ifdef FIXIT_VERSION_3
bool bUsingDVD = false;
const char* Game_Registry_Key();
//***********************************************************************************************
bool Is_DVD_Installed()
{
bool bInstalled;
HKEY hKey;
if( RegOpenKeyEx( HKEY_LOCAL_MACHINE, Game_Registry_Key(), 0, KEY_READ, &hKey ) != ERROR_SUCCESS )
return false;
DWORD dwValue;
DWORD dwBufSize = sizeof( DWORD );
if( RegQueryValueEx( hKey, "DVD", 0, NULL, (LPBYTE)&dwValue, &dwBufSize ) != ERROR_SUCCESS )
bInstalled = false;
else
bInstalled = (bool)dwValue; // (Presumably true, if it's there...)
RegCloseKey( hKey );
return bInstalled;
}
//***********************************************************************************************
bool Determine_If_Using_DVD()
{
// Determines if the user has a DVD currently available. If they do, we'll use it throughout the
// session. Else we won't check for it again and will always ask for CDs.
if( Is_DVD_Installed() )
{
if( Force_CD_Available( 5 ) )
{
bUsingDVD = true;
}
else
{
// User hit cancel. Allow things to progress normally. They will be prompted for
// a Red Alert disk as usual.
bUsingDVD = false;
}
}
else
bUsingDVD = false;
return bUsingDVD;
}
//***********************************************************************************************
bool Using_DVD()
{
return bUsingDVD;
}
#endif
/***********************************************************************************************
* Free_Heaps -- Clear out the heaps before exit *
* *
* *
* INPUT: none *
* *
* OUTPUT: none *
* *
* WARNINGS: *
* *
* HISTORY: *
* 12/18/2019 11:59AM ST : Created. *
*=============================================================================================*/
void Free_Heaps(void)
{
HouseTypes.Clear();
BuildingTypes.Clear();
AircraftTypes.Clear();
InfantryTypes.Clear();
BulletTypes.Clear();
AnimTypes.Clear();
UnitTypes.Clear();
VesselTypes.Clear();
TemplateTypes.Clear();
TerrainTypes.Clear();
OverlayTypes.Clear();
SmudgeTypes.Clear();
#if (0)
HouseTypeClass::Init_Heap();
BuildingTypeClass::Init_Heap();
AircraftTypeClass::Init_Heap();
InfantryTypeClass::Init_Heap();
BulletTypeClass::Init_Heap();
AnimTypeClass::Init_Heap();
UnitTypeClass::Init_Heap();
VesselTypeClass::Init_Heap();
TemplateTypeClass::Init_Heap();
TerrainTypeClass::Init_Heap();
OverlayTypeClass::Init_Heap();
SmudgeTypeClass::Init_Heap();
// Heap init moved here from globals.cpp. ST - 5/20/2019
CCPtr<AircraftClass>::Set_Heap(&Aircraft);
CCPtr<AnimClass>::Set_Heap(&Anims);
CCPtr<BuildingClass>::Set_Heap(&Buildings);
CCPtr<BulletClass>::Set_Heap(&Bullets);
CCPtr<FactoryClass>::Set_Heap(&Factories);
CCPtr<HouseClass>::Set_Heap(&Houses);
CCPtr<InfantryClass>::Set_Heap(&Infantry);
CCPtr<OverlayClass>::Set_Heap(&Overlays);
CCPtr<SmudgeClass>::Set_Heap(&Smudges);
CCPtr<TeamClass>::Set_Heap(&Teams);
CCPtr<TeamTypeClass>::Set_Heap(&TeamTypes);
CCPtr<TemplateClass>::Set_Heap(&Templates);
CCPtr<TerrainClass>::Set_Heap(&Terrains);
CCPtr<TriggerClass>::Set_Heap(&Triggers);
CCPtr<TriggerTypeClass>::Set_Heap(&TriggerTypes);
CCPtr<HouseTypeClass>::Set_Heap(&HouseTypes);
CCPtr<BuildingTypeClass>::Set_Heap(&BuildingTypes);
CCPtr<AircraftTypeClass>::Set_Heap(&AircraftTypes);
CCPtr<InfantryTypeClass>::Set_Heap(&InfantryTypes);
CCPtr<BulletTypeClass>::Set_Heap(&BulletTypes);
CCPtr<AnimTypeClass>::Set_Heap(&AnimTypes);
CCPtr<UnitTypeClass>::Set_Heap(&UnitTypes);
CCPtr<VesselTypeClass>::Set_Heap(&VesselTypes);
CCPtr<TemplateTypeClass>::Set_Heap(&TemplateTypes);
CCPtr<TerrainTypeClass>::Set_Heap(&TerrainTypes);
CCPtr<OverlayTypeClass>::Set_Heap(&OverlayTypes);
CCPtr<SmudgeTypeClass>::Set_Heap(&SmudgeTypes);
#endif
Vessels.Clear();
Units.Clear();
Factories.Clear();
Terrains.Clear();
Templates.Clear();
Smudges.Clear();
Overlays.Clear();
Infantry.Clear();
Bullets.Clear();
Buildings.Clear();
Anims.Clear();
Aircraft.Clear();
Triggers.Clear();
TeamTypes.Clear();
Teams.Clear();
Houses.Clear();
TriggerTypes.Clear();
/*
** Speech holding tank buffer. Since speech does not mix, it can be placed
** into a custom holding tank only as large as the largest speech file to
** be played.
*/
for (int index = 0; index < ARRAY_SIZE(SpeechBuffer); index++) {
if (SpeechBuffer[index]) {
delete [] SpeechBuffer[index];
SpeechBuffer[index] = NULL;
}
}
/*
** Allocate the theater buffer block.
*/
if (TheaterBuffer) {
delete TheaterBuffer;
TheaterBuffer = NULL;
}
}