bool stageTwoInitialise(void) { int i; debug(LOG_WZ, "== stageTwoInitalise =="); // make sure we clear on loading; this a bad hack to fix a bug when // loading a savegame where we are building a lassat for (i = 0; i < MAX_PLAYERS; i++) { setLasSatExists(false, i); } if(bMultiPlayer) { if (!multiTemplateSetup()) { return false; } } if (!dispInitialise()) /* Initialise the display system */ { return false; } if(!initMiscImds()) /* Set up the explosions */ { iV_ShutDown(); debug( LOG_FATAL, "Can't find all the explosions graphics?" ); abort(); return false; } if (!cmdDroidInit()) { return false; } /* Shift the interface initialisation here temporarily so that it can pick up the stats after they have been loaded */ if (!intInitialise()) { return false; } if (!initMessage()) /* Initialise the message heaps */ { return false; } if (!gwInitialise()) { return false; } // keymappings keyClearMappings(); keyInitMappings(false); // Set the default uncoloured cursor here, since it looks slightly // better for menus and such. wzSetCursor(CURSOR_DEFAULT); SetFormAudioIDs(ID_SOUND_WINDOWOPEN,ID_SOUND_WINDOWCLOSE); // Setup game queues. // Don't ask why this doesn't go in stage three. In fact, don't even ask me what stage one/two/three is supposed to mean, it seems about as descriptive as stage doStuff, stage doMoreStuff and stage doEvenMoreStuff... debug(LOG_MAIN, "Init game queues, I am %d.", selectedPlayer); sendQueuedDroidInfo(); // Discard any pending orders which could later get flushed into the game queue. for (i = 0; i < MAX_PLAYERS; ++i) { NETinitQueue(NETgameQueue(i)); if (!myResponsibility(i)) { NETsetNoSendOverNetwork(NETgameQueue(i)); } } debug(LOG_MAIN, "stageTwoInitialise: done"); return true; }
static void gameStateUpdate() { syncDebug("map = \"%s\", pseudorandom 32-bit integer = 0x%08X, allocated = %d %d %d %d %d %d %d %d %d %d, position = %d %d %d %d %d %d %d %d %d %d", game.map, gameRandU32(), NetPlay.players[0].allocated, NetPlay.players[1].allocated, NetPlay.players[2].allocated, NetPlay.players[3].allocated, NetPlay.players[4].allocated, NetPlay.players[5].allocated, NetPlay.players[6].allocated, NetPlay.players[7].allocated, NetPlay.players[8].allocated, NetPlay.players[9].allocated, NetPlay.players[0].position, NetPlay.players[1].position, NetPlay.players[2].position, NetPlay.players[3].position, NetPlay.players[4].position, NetPlay.players[5].position, NetPlay.players[6].position, NetPlay.players[7].position, NetPlay.players[8].position, NetPlay.players[9].position ); for (unsigned n = 0; n < MAX_PLAYERS; ++n) { syncDebug("Player %d = \"%s\"", n, NetPlay.players[n].name); } // Add version string to desynch logs. Different version strings will not trigger a desynch dump per se, due to the syncDebug{Get, Set}Crc guard. auto crc = syncDebugGetCrc(); syncDebug("My client version = %s", version_getVersionString()); syncDebugSetCrc(crc); // Actually send pending droid orders. sendQueuedDroidInfo(); sendPlayerGameTime(); NETflush(); // Make sure the game time tick message is really sent over the network. if (!paused && !scriptPaused()) { /* Update the event system */ if (!bInTutorial) { eventProcessTriggers(gameTime / SCR_TICKRATE); } else { eventProcessTriggers(realTime / SCR_TICKRATE); } updateScripts(); } // Update abandoned structures handleAbandonedStructures(); // Update the visibility change stuff visUpdateLevel(); // Put all droids/structures/features into the grid. gridReset(); // Check which objects are visible. processVisibility(); // Update the map. mapUpdate(); //update the findpath system fpathUpdate(); // update the command droids cmdDroidUpdate(); fireWaitingCallbacks(); //Now is the good time to fire waiting callbacks (since interpreter is off now) for (unsigned i = 0; i < MAX_PLAYERS; i++) { //update the current power available for a player updatePlayerPower(i); DROID *psNext; for (DROID *psCurr = apsDroidLists[i]; psCurr != nullptr; psCurr = psNext) { // Copy the next pointer - not 100% sure if the droid could get destroyed but this covers us anyway psNext = psCurr->psNext; droidUpdate(psCurr); } for (DROID *psCurr = mission.apsDroidLists[i]; psCurr != nullptr; psCurr = psNext) { /* Copy the next pointer - not 100% sure if the droid could get destroyed but this covers us anyway */ psNext = psCurr->psNext; missionDroidUpdate(psCurr); } // FIXME: These for-loops are code duplicationo STRUCTURE *psNBuilding; for (STRUCTURE *psCBuilding = apsStructLists[i]; psCBuilding != nullptr; psCBuilding = psNBuilding) { /* Copy the next pointer - not 100% sure if the structure could get destroyed but this covers us anyway */ psNBuilding = psCBuilding->psNext; structureUpdate(psCBuilding, false); } for (STRUCTURE *psCBuilding = mission.apsStructLists[i]; psCBuilding != nullptr; psCBuilding = psNBuilding) { /* Copy the next pointer - not 100% sure if the structure could get destroyed but this covers us anyway. It shouldn't do since its not even on the map!*/ psNBuilding = psCBuilding->psNext; structureUpdate(psCBuilding, true); // update for mission } } missionTimerUpdate(); proj_UpdateAll(); FEATURE *psNFeat; for (FEATURE *psCFeat = apsFeatureLists[0]; psCFeat; psCFeat = psNFeat) { psNFeat = psCFeat->psNext; featureUpdate(psCFeat); } // Clean up dead droid pointers in UI. hciUpdate(); // Free dead droid memory. objmemUpdate(); // Must end update, since we may or may not have ticked, and some message queue processing code may vary depending on whether it's in an update. gameTimeUpdateEnd(); // Must be at the beginning or end of each tick, since countUpdate is also called randomly (unsynchronised) between ticks. countUpdate(true); static int i = 0; if (i++ % 10 == 0) // trigger every second { jsDebugUpdate(); } }
static void gameStateUpdate() { // Can't dump isHumanPlayer, since it causes spurious desynch dumps when players leave. // TODO isHumanPlayer should probably be synchronised, since the game state seems to depend on it, so there might also be a risk of real desynchs when players leave. //syncDebug("map = \"%s\", humanPlayers = %d %d %d %d %d %d %d %d", game.map, isHumanPlayer(0), isHumanPlayer(1), isHumanPlayer(2), isHumanPlayer(3), isHumanPlayer(4), isHumanPlayer(5), isHumanPlayer(6), isHumanPlayer(7)); syncDebug("map = \"%s\"", game.map); // Actually send pending droid orders. sendQueuedDroidInfo(); sendPlayerGameTime(); gameSRand(gameTime); // Brute force way of synchronising the random number generator, which can't go out of synch. if (!paused && !scriptPaused() && !editPaused()) { /* Update the event system */ if (!bInTutorial) { eventProcessTriggers(gameTime/SCR_TICKRATE); } else { eventProcessTriggers(realTime/SCR_TICKRATE); } updateScripts(); } // Update abandoned structures handleAbandonedStructures(); // Update the visibility change stuff visUpdateLevel(); // Put all droids/structures/features into the grid. gridReset(); // Check which objects are visible. processVisibility(); // Update the map. mapUpdate(); //update the findpath system fpathUpdate(); // update the cluster system clusterUpdate(); // update the command droids cmdDroidUpdate(); if(getDrivingStatus()) { driveUpdate(); } fireWaitingCallbacks(); //Now is the good time to fire waiting callbacks (since interpreter is off now) for (unsigned i = 0; i < MAX_PLAYERS; i++) { //update the current power available for a player updatePlayerPower(i); //set the flag for each player setHQExists(false, i); setSatUplinkExists(false, i); numCommandDroids[i] = 0; numConstructorDroids[i] = 0; numDroids[i]=0; numTransporterDroids[i]=0; DROID *psNext; for (DROID *psCurr = apsDroidLists[i]; psCurr != NULL; psCurr = psNext) { // Copy the next pointer - not 100% sure if the droid could get destroyed but this covers us anyway psNext = psCurr->psNext; droidUpdate(psCurr); // update the droid counts numDroids[i]++; switch (psCurr->droidType) { case DROID_COMMAND: numCommandDroids[i] += 1; break; case DROID_CONSTRUCT: case DROID_CYBORG_CONSTRUCT: numConstructorDroids[i] += 1; break; case DROID_TRANSPORTER: if( (psCurr->psGroup != NULL) ) { DROID *psDroid = NULL; numTransporterDroids[i] += psCurr->psGroup->refCount-1; // and count the units inside it... for (psDroid = psCurr->psGroup->psList; psDroid != NULL && psDroid != psCurr; psDroid = psDroid->psGrpNext) { if (psDroid->droidType == DROID_CYBORG_CONSTRUCT || psDroid->droidType == DROID_CONSTRUCT) { numConstructorDroids[i] += 1; } if (psDroid->droidType == DROID_COMMAND) { numCommandDroids[i] += 1; } } } break; default: break; } } numMissionDroids[i]=0; for (DROID *psCurr = mission.apsDroidLists[i]; psCurr != NULL; psCurr = psNext) { /* Copy the next pointer - not 100% sure if the droid could get destroyed but this covers us anyway */ psNext = psCurr->psNext; missionDroidUpdate(psCurr); numMissionDroids[i]++; switch (psCurr->droidType) { case DROID_COMMAND: numCommandDroids[i] += 1; break; case DROID_CONSTRUCT: case DROID_CYBORG_CONSTRUCT: numConstructorDroids[i] += 1; break; case DROID_TRANSPORTER: if( (psCurr->psGroup != NULL) ) { numTransporterDroids[i] += psCurr->psGroup->refCount-1; } break; default: break; } } for (DROID *psCurr = apsLimboDroids[i]; psCurr != NULL; psCurr = psNext) { /* Copy the next pointer - not 100% sure if the droid could get destroyed but this covers us anyway */ psNext = psCurr->psNext; // count the type of units switch (psCurr->droidType) { case DROID_COMMAND: numCommandDroids[i] += 1; break; case DROID_CONSTRUCT: case DROID_CYBORG_CONSTRUCT: numConstructorDroids[i] += 1; break; default: break; } } // FIXME: These for-loops are code duplicationo /*set this up AFTER droidUpdate so that if trying to building a new one, we know whether one exists already*/ setLasSatExists(false, i); STRUCTURE *psNBuilding; for (STRUCTURE *psCBuilding = apsStructLists[i]; psCBuilding != NULL; psCBuilding = psNBuilding) { /* Copy the next pointer - not 100% sure if the structure could get destroyed but this covers us anyway */ psNBuilding = psCBuilding->psNext; structureUpdate(psCBuilding, false); //set animation flag if (psCBuilding->pStructureType->type == REF_HQ && psCBuilding->status == SS_BUILT) { setHQExists(true, i); } if (psCBuilding->pStructureType->type == REF_SAT_UPLINK && psCBuilding->status == SS_BUILT) { setSatUplinkExists(true, i); } //don't wait for the Las Sat to be built - can't build another if one is partially built if (asWeaponStats[psCBuilding->asWeaps[0].nStat]. weaponSubClass == WSC_LAS_SAT) { setLasSatExists(true, i); } } for (STRUCTURE *psCBuilding = mission.apsStructLists[i]; psCBuilding != NULL; psCBuilding = psNBuilding) { /* Copy the next pointer - not 100% sure if the structure could get destroyed but this covers us anyway. It shouldn't do since its not even on the map!*/ psNBuilding = psCBuilding->psNext; structureUpdate(psCBuilding, true); // update for mission if (psCBuilding->pStructureType->type == REF_HQ && psCBuilding->status == SS_BUILT) { setHQExists(true, i); } if (psCBuilding->pStructureType->type == REF_SAT_UPLINK && psCBuilding->status == SS_BUILT) { setSatUplinkExists(true, i); } //don't wait for the Las Sat to be built - can't build another if one is partially built if (asWeaponStats[psCBuilding->asWeaps[0].nStat]. weaponSubClass == WSC_LAS_SAT) { setLasSatExists(true, i); } } } missionTimerUpdate(); proj_UpdateAll(); FEATURE *psNFeat; for (FEATURE *psCFeat = apsFeatureLists[0]; psCFeat; psCFeat = psNFeat) { psNFeat = psCFeat->psNext; featureUpdate(psCFeat); } objmemUpdate(); // Do completely useless stuff. if (!isInSync()) { sendCheck(); // send some pointless checking info if we're doomed anyway } // Must end update, since we may or may not have ticked, and some message queue processing code may vary depending on whether it's in an update. gameTimeUpdateEnd(); }
static GAMECODE renderLoop() { if (bMultiPlayer && !NetPlay.isHostAlive && NetPlay.bComms && !NetPlay.isHost) { intAddInGamePopup(); } audio_Update(); wzShowMouse(true); INT_RETVAL intRetVal = INT_NONE; CURSOR cursor = CURSOR_DEFAULT; if (!paused) { /* Run the in game interface and see if it grabbed any mouse clicks */ if (!rotActive && getWidgetsStatus() && dragBox3D.status != DRAG_DRAGGING && wallDrag.status != DRAG_DRAGGING) { intRetVal = intRunWidgets(); // Send droid orders, if any. (Should do between intRunWidgets() calls, to avoid droid orders getting mixed up, in the case of multiple orders given while the game freezes due to net lag.) sendQueuedDroidInfo(); } //don't process the object lists if paused or about to quit to the front end if (!gameUpdatePaused() && intRetVal != INT_QUIT) { if (dragBox3D.status != DRAG_DRAGGING && wallDrag.status != DRAG_DRAGGING && (intRetVal == INT_INTERCEPT || (radarOnScreen && CoordInRadar(mouseX(), mouseY()) && radarPermitted))) { // Using software cursors (when on) for these menus due to a bug in SDL's SDL_ShowCursor() wzSetCursor(CURSOR_DEFAULT); } #ifdef DEBUG // check all flag positions for duplicate delivery points checkFactoryFlags(); #endif //handles callbacks for positioning of DP's process3DBuilding(); processDeliveryRepos(); //ajl. get the incoming netgame messages and process them. // FIXME Previous comment is deprecated. multiPlayerLoop does some other weird stuff, but not that anymore. if (bMultiPlayer) { multiPlayerLoop(); } for (unsigned i = 0; i < MAX_PLAYERS; i++) { for (DROID *psCurr = apsDroidLists[i]; psCurr; psCurr = psCurr->psNext) { // Don't copy the next pointer - if droids somehow get destroyed in the graphics rendering loop, who cares if we crash. calcDroidIllumination(psCurr); } } } if (!consolePaused()) { /* Process all the console messages */ updateConsoleMessages(); } if (!scrollPaused() && dragBox3D.status != DRAG_DRAGGING && intMode != INT_INGAMEOP) { cursor = scroll(); zoom(); } } else // paused { // Using software cursors (when on) for these menus due to a bug in SDL's SDL_ShowCursor() wzSetCursor(CURSOR_DEFAULT); if (dragBox3D.status != DRAG_DRAGGING) { cursor = scroll(); zoom(); } if (InGameOpUp || isInGamePopupUp) // ingame options menu up, run it! { WidgetTriggers const &triggers = widgRunScreen(psWScreen); unsigned widgval = triggers.empty() ? 0 : triggers.front().widget->id; // Just use first click here, since the next click could be on another menu. intProcessInGameOptions(widgval); if (widgval == INTINGAMEOP_QUIT_CONFIRM || widgval == INTINGAMEOP_POPUP_QUIT) { if (gamePaused()) { kf_TogglePauseMode(); } intRetVal = INT_QUIT; } } if (bLoadSaveUp && runLoadSave(true) && strlen(sRequestResult)) { debug(LOG_NEVER, "Returned %s", sRequestResult); if (bRequestLoad) { loopMissionState = LMS_LOADGAME; NET_InitPlayers(); // otherwise alliances were not cleared sstrcpy(saveGameName, sRequestResult); } else { char msgbuffer[256] = {'\0'}; if (saveInMissionRes()) { if (saveGame(sRequestResult, GTYPE_SAVE_START)) { sstrcpy(msgbuffer, _("GAME SAVED: ")); sstrcat(msgbuffer, sRequestResult); addConsoleMessage(msgbuffer, LEFT_JUSTIFY, NOTIFY_MESSAGE); } else { ASSERT(false, "Mission Results: saveGame Failed"); sstrcpy(msgbuffer, _("Could not save game!")); addConsoleMessage(msgbuffer, LEFT_JUSTIFY, NOTIFY_MESSAGE); deleteSaveGame(sRequestResult); } } else if (bMultiPlayer || saveMidMission()) { if (saveGame(sRequestResult, GTYPE_SAVE_MIDMISSION))//mid mission from [esc] menu { sstrcpy(msgbuffer, _("GAME SAVED: ")); sstrcat(msgbuffer, sRequestResult); addConsoleMessage(msgbuffer, LEFT_JUSTIFY, NOTIFY_MESSAGE); } else { ASSERT(!"saveGame(sRequestResult, GTYPE_SAVE_MIDMISSION) failed", "Mid Mission: saveGame Failed"); sstrcpy(msgbuffer, _("Could not save game!")); addConsoleMessage(msgbuffer, LEFT_JUSTIFY, NOTIFY_MESSAGE); deleteSaveGame(sRequestResult); } } else { ASSERT(false, "Attempt to save game with incorrect load/save mode"); } } } } /* Check for quit */ bool quitting = false; if (intRetVal == INT_QUIT) { if (!loop_GetVideoStatus()) { //quitting from the game to the front end //so get a new backdrop quitting = true; pie_LoadBackDrop(SCREEN_RANDOMBDROP); } } if (!loop_GetVideoStatus() && !quitting) { if (!gameUpdatePaused()) { if (dragBox3D.status != DRAG_DRAGGING && wallDrag.status != DRAG_DRAGGING && intRetVal != INT_INTERCEPT) { ProcessRadarInput(); } processInput(); //no key clicks or in Intelligence Screen if (!isMouseOverRadar() && intRetVal == INT_NONE && !InGameOpUp && !isInGamePopupUp) { CURSOR cursor2 = processMouseClickInput(); cursor = cursor2 == CURSOR_DEFAULT? cursor : cursor2; } displayWorld(); } wzPerfBegin(PERF_GUI, "User interface"); /* Display the in game interface */ pie_SetDepthBufferStatus(DEPTH_CMP_ALWAYS_WRT_ON); pie_SetFogStatus(false); if (bMultiPlayer && bDisplayMultiJoiningStatus) { intDisplayMultiJoiningStatus(bDisplayMultiJoiningStatus); setWidgetsStatus(false); } if (getWidgetsStatus()) { intDisplayWidgets(); } pie_SetDepthBufferStatus(DEPTH_CMP_LEQ_WRT_ON); pie_SetFogStatus(true); wzPerfEnd(PERF_GUI); } wzSetCursor(cursor); pie_GetResetCounts(&loopPieCount, &loopPolyCount); if (!quitting) { /* Check for toggling display mode */ if ((keyDown(KEY_LALT) || keyDown(KEY_RALT)) && keyPressed(KEY_RETURN)) { war_setFullscreen(!war_getFullscreen()); wzToggleFullscreen(); } } // deal with the mission state switch (loopMissionState) { case LMS_CLEAROBJECTS: missionDestroyObjects(); setScriptPause(true); loopMissionState = LMS_SETUPMISSION; break; case LMS_NORMAL: // default break; case LMS_SETUPMISSION: setScriptPause(false); if (!setUpMission(nextMissionType)) { return GAMECODE_QUITGAME; } break; case LMS_SAVECONTINUE: // just wait for this to be changed when the new mission starts break; case LMS_NEWLEVEL: //nextMissionType = MISSION_NONE; nextMissionType = LDS_NONE; return GAMECODE_NEWLEVEL; break; case LMS_LOADGAME: return GAMECODE_LOADGAME; break; default: ASSERT(false, "unknown loopMissionState"); break; } int clearMode = 0; if (getDrawShadows()) { clearMode |= CLEAR_SHADOW; } if (quitting || loopMissionState == LMS_SAVECONTINUE) { pie_SetFogStatus(false); clearMode = CLEAR_BLACK; } pie_ScreenFlip(clearMode);//gameloopflip if (quitting) { /* Check for toggling display mode */ if ((keyDown(KEY_LALT) || keyDown(KEY_RALT)) && keyPressed(KEY_RETURN)) { war_setFullscreen(!war_getFullscreen()); wzToggleFullscreen(); } return GAMECODE_QUITGAME; } else if (loop_GetVideoStatus()) { audio_StopAll(); return GAMECODE_PLAYVIDEO; } return GAMECODE_CONTINUE; }