CnC_Remastered_Collection

Command and Conquer: Red Alert
Log | Files | Refs | README | LICENSE

LOGIC.CPP (14884B)


      1 //
      2 // Copyright 2020 Electronic Arts Inc.
      3 //
      4 // TiberianDawn.DLL and RedAlert.dll and corresponding source code is free 
      5 // software: you can redistribute it and/or modify it under the terms of 
      6 // the GNU General Public License as published by the Free Software Foundation, 
      7 // either version 3 of the License, or (at your option) any later version.
      8 
      9 // TiberianDawn.DLL and RedAlert.dll and corresponding source code is distributed 
     10 // in the hope that it will be useful, but with permitted additional restrictions 
     11 // under Section 7 of the GPL. See the GNU General Public License in LICENSE.TXT 
     12 // distributed with this program. You should have received a copy of the 
     13 // GNU General Public License along with permitted additional restrictions 
     14 // with this program. If not, see https://github.com/electronicarts/CnC_Remastered_Collection
     15 
     16 /* $Header:   F:\projects\c&c\vcs\code\logic.cpv   2.17   16 Oct 1995 16:50:52   JOE_BOSTIC  $ */
     17 /***********************************************************************************************
     18  ***             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               ***
     19  ***********************************************************************************************
     20  *                                                                                             *
     21  *                 Project Name : Command & Conquer                                            *
     22  *                                                                                             *
     23  *                    File Name : LOGIC.CPP                                                    *
     24  *                                                                                             *
     25  *                   Programmer : Joe L. Bostic                                                *
     26  *                                                                                             *
     27  *                   Start Date : September 27, 1993                                           *
     28  *                                                                                             *
     29  *                  Last Update : December 23, 1994 [JLB]                                      *
     30  *                                                                                             *
     31  *---------------------------------------------------------------------------------------------*
     32  * Functions:                                                                                  *
     33  *   LogicClass::AI -- Handles AI logic processing for game objects.                           *
     34  *   LogicClass::Debug_Dump -- Displays logic class status to the mono screen.                 *
     35  * - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - */
     36 
     37 #include	"function.h"
     38 #include	"logic.h"
     39 
     40 static unsigned FramesPerSecond=0;
     41 
     42 #ifdef CHEAT_KEYS
     43 
     44 static unsigned TotalFrames;
     45 static unsigned FPSDivider = 1;
     46 static unsigned AverageFramesPerSecond;
     47 
     48 /***********************************************************************************************
     49  * LogicClass::Debug_Dump -- Displays logic class status to the mono screen.                   *
     50  *                                                                                             *
     51  *    This is a debugging support routine. It displays the current state of the logic class    *
     52  *    to the monochrome monitor. It assumes that it is being called once per second.           *
     53  *                                                                                             *
     54  * INPUT:   none                                                                               *
     55  *                                                                                             *
     56  * OUTPUT:  none                                                                               *
     57  *                                                                                             *
     58  * WARNINGS:   Call this routine only once per second.                                         *
     59  *                                                                                             *
     60  * HISTORY:                                                                                    *
     61  *   05/31/1994 JLB : Created.                                                                 *
     62  *=============================================================================================*/
     63 void LogicClass::Debug_Dump(MonoClass *mono) const
     64 {
     65 	#define RECORDCOUNT	40
     66 	#define RECORDHEIGHT 21
     67 	static struct {
     68 		int Graphic;
     69 	} _record[RECORDCOUNT];
     70 	static int _framecounter = 0;
     71 
     72 	TotalFrames+= FramesPerSecond;
     73 	AverageFramesPerSecond = TotalFrames/FPSDivider++;
     74 
     75 	mono->Set_Cursor(21, 9);
     76 	mono->Print(
     77 		"��������������������������������������������������������Ŀ\r"
     78 		"�Units.....�   �Frame Rate:      Avg:      Frame:        �\r"
     79 		"�Infantry..�   �����������������������������������������Ĵ\r"
     80 		"�Aircraft..�   �                                         �\r"
     81 		"�Buildings.�   �                                         �\r"
     82 		"�Terrain...�   �                                         �\r"
     83 		"�Bullets...�   �                                         �\r"
     84 		"�Anims.....�   �                                         �\r"
     85 		"�Teams.....�   �                                        Ĵ\r"
     86 		"�Triggers..�   �                                         �\r"
     87 		"�Factories.�   �                                         �\r"
     88 		"�          �   �                                         �\r"
     89 		"�          �   �                                         �\r"
     90 		"����������������������������ĴSpare CPU Time��������������\r");
     91 
     92 	_framecounter++;
     93 	mono->Set_Cursor(70, 10);mono->Printf("%ld", Frame);
     94 	if (ScenarioInit) {
     95 		mono->Set_Cursor(21, 9);mono->Printf("%d", ScenarioInit);
     96 	}
     97 
     98 	mono->Set_Cursor(33, 10);mono->Printf("%3d", Units.Count());
     99 	mono->Set_Cursor(33, 11);mono->Printf("%3d", Infantry.Count());
    100 	mono->Set_Cursor(33, 12);mono->Printf("%3d", Aircraft.Count());
    101 	mono->Set_Cursor(33, 13);mono->Printf("%3d", Buildings.Count());
    102 	mono->Set_Cursor(33, 14);mono->Printf("%3d", Terrains.Count());
    103 	mono->Set_Cursor(33, 15);mono->Printf("%3d", Bullets.Count());
    104 	mono->Set_Cursor(33, 16);mono->Printf("%3d", Anims.Count());
    105 	mono->Set_Cursor(33, 17);mono->Printf("%3d", Teams.Count());
    106 	mono->Set_Cursor(33, 18);mono->Printf("%3d", Triggers.Count());
    107 	mono->Set_Cursor(33, 19);mono->Printf("%3d", Factories.Count());
    108 
    109 	mono->Set_Cursor(48, 10);mono->Printf("%d", FramesPerSecond);
    110 	mono->Set_Cursor(58, 10);mono->Printf("%d", AverageFramesPerSecond);
    111 
    112 	/*
    113 	**	Advance to the next recorded performance record. If the record buffer
    114 	**	is full then throw out the oldest record.
    115 	*/
    116 	memcpy(&_record[0], &_record[1], sizeof(_record[0])*(RECORDCOUNT-1));
    117 
    118 	/*
    119 	**	Fill in the data for the current frame's performance record.
    120 	*/
    121 	SpareTicks = MIN((long)SpareTicks, (long)TIMER_SECOND);
    122 	_record[RECORDCOUNT-1].Graphic = Fixed_To_Cardinal(RECORDHEIGHT, Cardinal_To_Fixed(TIMER_SECOND, SpareTicks));
    123 
    124 	/*
    125 	**	Draw the bars across the performance record screen.
    126 	*/
    127 	for (int column = 0; column < RECORDCOUNT; column++) {
    128 		for (int row = 1; row < RECORDHEIGHT; row += 2) {
    129 			static unsigned char _barchar[4] = {' ', 220, 0, 219};
    130 			char str[2];
    131 			int index = 0;
    132 
    133 			index |= (_record[column].Graphic >= row) ? 0x01 : 0x00;
    134 			index |= (_record[column].Graphic >= row+1) ? 0x02: 0x00;
    135 
    136 			str[1] = '\0';
    137 			str[0] = _barchar[index];
    138 			mono->Text_Print(str, 37+column, 21-(row/2));
    139 		}
    140 	}
    141 
    142 	SpareTicks = 0;
    143 	FramesPerSecond = 0;
    144 }
    145 #endif
    146 
    147 
    148 /***********************************************************************************************
    149  * LogicClass::AI -- Handles AI logic processing for game objects.                             *
    150  *                                                                                             *
    151  *    This routine is used to perform the AI processing for all game objects. This includes    *
    152  *    all houses, factories, objects, and teams.                                               *
    153  *                                                                                             *
    154  * INPUT:   none                                                                               *
    155  *                                                                                             *
    156  * OUTPUT:  none                                                                               *
    157  *                                                                                             *
    158  * WARNINGS:   none                                                                            *
    159  *                                                                                             *
    160  * HISTORY:                                                                                    *
    161  *   05/29/1994 JLB : Created.                                                                 *
    162  *   12/17/1994 JLB : Must perform one complete pass rather than bailing early.                *
    163  *   12/23/1994 JLB : Esures that no object gets skipped if it was deleted.                    *
    164  *=============================================================================================*/
    165 void LogicClass::AI(void)
    166 {
    167 	int index;
    168 
    169 	FramesPerSecond++;
    170 
    171 	/*
    172 	**	Crate regeneration is handled here.
    173 	*/
    174 	if (GameToPlay != GAME_NORMAL && CrateMaker && CrateTimer.Expired()) {
    175 		Map.Place_Random_Crate();
    176 		CrateTimer = TICKS_PER_MINUTE * Random_Pick(7, 15);
    177 	}
    178 
    179 	/*
    180 	**	Team AI is processed.
    181 	*/
    182 	for (index = 0; index < Teams.Count(); index++) {
    183 		Teams.Ptr(index)->AI();
    184 	}
    185 
    186 //	Heap_Dump_Check( "After Team AI" );
    187 
    188 	/*
    189 	**	AI for all sentient objects is processed.
    190 	*/
    191 	for (index = 0; index < Count(); index++) {
    192 		ObjectClass * obj = (*this)[index];
    193 		int count = Count();
    194 
    195 		obj->AI();
    196 
    197 		/*
    198 		**	If the object was destroyed in the process of performing its AI, then
    199 		**	adjust the index so that no object gets skipped.
    200 		*/
    201 		int count_diff = Count() - count;
    202 		if (count_diff < 0) {
    203 			index += count_diff;
    204 		}
    205 	}
    206 
    207 //	Heap_Dump_Check( "After Object AI" );
    208 
    209 	/*
    210 	**	A second pass through the sentient objects is required so that the appropriate scan
    211 	**	bits will be set for the owner house.
    212 	*/
    213 	for (index = 0; index < Units.Count(); index++) {
    214 		UnitClass const * unit = Units.Ptr(index);
    215 		if (unit->IsLocked && (GameToPlay != GAME_NORMAL || !unit->House->IsHuman || unit->IsDiscoveredByPlayer)) {
    216 			unit->House->NewUScan |= (1L << unit->Class->Type);
    217 			if (!unit->IsInLimbo) unit->House->NewActiveUScan |= (1L << unit->Class->Type);
    218 		}
    219 	}
    220 	for (index = 0; index < Infantry.Count(); index++) {
    221 		InfantryClass const * infantry = Infantry.Ptr(index);
    222 		if (infantry->IsLocked && (GameToPlay != GAME_NORMAL || !infantry->House->IsHuman || infantry->IsDiscoveredByPlayer)) {
    223 			infantry->House->NewIScan |= (1L << infantry->Class->Type);
    224 			if (!infantry->IsInLimbo) infantry->House->NewActiveIScan |= (1L << infantry->Class->Type);
    225 		}
    226 	}
    227 	for (index = 0; index < Aircraft.Count(); index++) {
    228 		AircraftClass const * aircraft = Aircraft.Ptr(index);
    229 		if (aircraft->IsLocked && (GameToPlay != GAME_NORMAL || !aircraft->House->IsHuman || aircraft->IsDiscoveredByPlayer)) {
    230 			aircraft->House->NewAScan |= (1L << aircraft->Class->Type);
    231 			if (!aircraft->IsInLimbo) aircraft->House->NewActiveAScan |= (1L << aircraft->Class->Type);
    232 		}
    233 	}
    234 	for (index = 0; index < Buildings.Count(); index++) {
    235 		BuildingClass const * building = Buildings.Ptr(index);
    236 		if (building->IsLocked && (GameToPlay != GAME_NORMAL || !building->House->IsHuman || building->IsDiscoveredByPlayer)) {
    237 			building->House->NewBScan |= (1L << building->Class->Type);
    238 			if (!building->IsInLimbo) building->House->NewActiveBScan |= (1L << building->Class->Type);
    239 		}
    240 	}
    241 
    242 //	Heap_Dump_Check( "After Object AI 2" );
    243 
    244 #ifdef USE_RA_AI
    245 	//
    246 	// Added for RA AI in TD. ST - 7/26/2019 10:56AM
    247 	//
    248 	HouseClass::Recalc_Attributes();
    249 #endif // USE_RA_AI
    250 
    251 	/*
    252 	**	Map related logic is performed.
    253 	*/
    254 	Map.Logic();
    255 
    256 //	Heap_Dump_Check( "After Map.Logic" );
    257 
    258 	/*
    259 	**	Factory processing is performed.
    260 	*/
    261 	for (index = 0; index < Factories.Count(); index++) {
    262 		Factories.Ptr(index)->AI();
    263 	}
    264 
    265 //	Heap_Dump_Check( "After Factory AI" );
    266 
    267 #if (1)
    268 	/*
    269 	** Changed integrated from RA to only call AI on the houses that need it. Without this change, AI houses immediately
    270 	** become paranoid at the start of a multiplayer match
    271 	** ST - 10/30/2019 11:15AM
    272 	*/
    273 	if (GameToPlay != GAME_NORMAL) {
    274 		for (HousesType house = HOUSE_MULTI1; house < HOUSE_COUNT; house++) {
    275 			HouseClass * hptr = HouseClass::As_Pointer(house);
    276 			if (hptr && hptr->IsActive) {
    277 				hptr->AI();
    278 			}
    279 		}
    280 
    281 		HouseClass* neutral_house = HouseClass::As_Pointer(HOUSE_NEUTRAL);
    282 		if (neutral_house && neutral_house->IsActive) {
    283 			neutral_house->AI();
    284 		}
    285 
    286 		HouseClass* jp_house = HouseClass::As_Pointer(HOUSE_JP);
    287 		if (jp_house && jp_house->IsActive) {
    288 			jp_house->AI();
    289 		}
    290 
    291 	} else {
    292 		
    293 		for (HousesType house = HOUSE_FIRST; house < HOUSE_COUNT; house++) {
    294 			HouseClass * hptr = HouseClass::As_Pointer(house);
    295 			if (hptr && hptr->IsActive) {
    296 				hptr->AI();
    297 			}
    298 		}
    299 	}
    300 
    301 
    302 #else
    303 	/*
    304 	**	House processing is performed.
    305 	*/
    306 	for (HousesType house = HOUSE_FIRST; house < HOUSE_COUNT; house++) {
    307 		HouseClass * hptr = HouseClass::As_Pointer(house);
    308 		if (hptr && hptr->IsActive) {
    309 			hptr->AI();
    310 		}
    311 	}
    312 #endif
    313 
    314 //	Heap_Dump_Check( "After House AI" );
    315 }
    316 
    317 
    318 
    319 
    320 
    321 
    322 /***********************************************************************************************
    323  * LogicClass::Clear_Recently_Created_Bits -- Clear out the indicators that objects were       *
    324  *                                            recently created                                 *
    325  *                                                                                             *
    326  * INPUT:   none                                                                               *
    327  *                                                                                             *
    328  * OUTPUT:  none                                                                               *
    329  *                                                                                             *
    330  * WARNINGS:   none                                                                            *
    331  *                                                                                             *
    332  * HISTORY:                                                                                    *
    333  *   8/19/2019 5:47PM ST : Created.                                                            *
    334  *=============================================================================================*/
    335 void LogicClass::Clear_Recently_Created_Bits(void)
    336 {
    337 	for (int index = 0; index < Count(); index++) {
    338 		ObjectClass * obj = (*this)[index];
    339 		obj->IsRecentlyCreated = false;
    340 	}
    341 }
    342 
    343