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