2007-05-27 12:01:53 -04:00
|
|
|
//
|
|
|
|
// SuperTuxKart - a fun racing game with go-kart
|
2015-03-29 20:31:42 -04:00
|
|
|
// Copyright (C) 2004-2015 Ingo Ruhnke <grumbel@gmx.de>
|
|
|
|
// Copyright (C) 2006-2015 SuperTuxKart-Team
|
2007-05-27 12:01:53 -04:00
|
|
|
//
|
|
|
|
// This program is free software; you can redistribute it and/or
|
|
|
|
// modify it under the terms of the GNU General Public License
|
2008-06-12 20:53:52 -04:00
|
|
|
// as published by the Free Software Foundation; either version 3
|
2007-05-27 12:01:53 -04:00
|
|
|
// of the License, or (at your option) any later version.
|
|
|
|
//
|
|
|
|
// This program is distributed in the hope that it will be useful,
|
|
|
|
// but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
|
|
// GNU General Public License for more details.
|
|
|
|
//
|
|
|
|
// You should have received a copy of the GNU General Public License
|
|
|
|
// along with this program; if not, write to the Free Software
|
|
|
|
// Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA.
|
|
|
|
|
2008-09-15 00:34:46 -04:00
|
|
|
#include "main_loop.hpp"
|
2007-05-27 12:01:53 -04:00
|
|
|
|
2014-10-15 17:13:55 -04:00
|
|
|
#include "audio/sfx_manager.hpp"
|
2009-06-11 06:00:43 -04:00
|
|
|
#include "config/user_config.hpp"
|
2018-08-03 06:36:35 -04:00
|
|
|
#include "graphics/central_settings.hpp"
|
2009-02-10 00:30:59 -05:00
|
|
|
#include "graphics/irr_driver.hpp"
|
2009-06-02 08:37:29 -04:00
|
|
|
#include "graphics/material_manager.hpp"
|
2018-08-03 06:36:35 -04:00
|
|
|
#include "graphics/sp/sp_texture_manager.hpp"
|
2009-07-18 13:48:36 -04:00
|
|
|
#include "guiengine/engine.hpp"
|
2018-02-18 01:06:17 -05:00
|
|
|
#include "guiengine/message_queue.hpp"
|
2018-04-02 00:15:07 -04:00
|
|
|
#include "guiengine/modaldialog.hpp"
|
2018-10-24 02:32:01 -04:00
|
|
|
#include "guiengine/screen_keyboard.hpp"
|
2009-06-02 08:37:29 -04:00
|
|
|
#include "input/input_manager.hpp"
|
2009-08-23 11:42:58 -04:00
|
|
|
#include "modes/profile_world.hpp"
|
2009-02-10 00:30:59 -05:00
|
|
|
#include "modes/world.hpp"
|
2015-11-11 17:53:12 -05:00
|
|
|
#include "network/network_config.hpp"
|
2019-04-16 07:16:55 -04:00
|
|
|
#include "network/network_timer_synchronizer.hpp"
|
2018-06-24 23:57:18 -04:00
|
|
|
#include "network/protocols/game_protocol.hpp"
|
2013-07-31 14:03:11 -04:00
|
|
|
#include "network/protocol_manager.hpp"
|
2016-02-10 17:03:51 -05:00
|
|
|
#include "network/race_event_manager.hpp"
|
2017-06-04 18:54:52 -04:00
|
|
|
#include "network/rewind_manager.hpp"
|
2015-11-01 03:45:28 -05:00
|
|
|
#include "network/stk_host.hpp"
|
2014-01-08 23:57:58 -05:00
|
|
|
#include "online/request_manager.hpp"
|
2016-09-22 17:56:21 -04:00
|
|
|
#include "race/history.hpp"
|
2009-06-02 21:36:48 -04:00
|
|
|
#include "race/race_manager.hpp"
|
2009-08-23 11:42:58 -04:00
|
|
|
#include "states_screens/state_manager.hpp"
|
2011-07-09 15:13:17 -04:00
|
|
|
#include "utils/profiler.hpp"
|
2019-06-15 01:52:28 -04:00
|
|
|
#include "utils/string_utils.hpp"
|
2018-08-27 01:49:11 -04:00
|
|
|
#include "utils/time.hpp"
|
2019-06-15 01:52:28 -04:00
|
|
|
#include "utils/translation.hpp"
|
2007-05-27 12:01:53 -04:00
|
|
|
|
2018-03-17 00:26:37 -04:00
|
|
|
#ifndef WIN32
|
|
|
|
#include <unistd.h>
|
|
|
|
#endif
|
|
|
|
|
2008-09-15 00:34:46 -04:00
|
|
|
MainLoop* main_loop = 0;
|
2007-05-27 12:01:53 -04:00
|
|
|
|
2018-07-01 16:56:56 -04:00
|
|
|
#ifdef WIN32
|
|
|
|
LRESULT CALLBACK separateProcessProc(_In_ HWND hwnd, _In_ UINT uMsg,
|
|
|
|
_In_ WPARAM wParam, _In_ LPARAM lParam)
|
|
|
|
{
|
|
|
|
if (uMsg == WM_DESTROY)
|
|
|
|
{
|
|
|
|
PostQuitMessage(0);
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
return DefWindowProc(hwnd, uMsg, wParam, lParam);
|
|
|
|
};
|
|
|
|
#endif
|
|
|
|
|
2018-03-19 18:42:26 -04:00
|
|
|
// ----------------------------------------------------------------------------
|
2018-03-02 00:25:14 -05:00
|
|
|
MainLoop::MainLoop(unsigned parent_pid)
|
2018-09-22 16:06:27 -04:00
|
|
|
: m_abort(false), m_request_abort(false), m_ticks_adjustment(0),
|
|
|
|
m_parent_pid(parent_pid)
|
2007-05-27 12:01:53 -04:00
|
|
|
{
|
2018-02-12 02:01:05 -05:00
|
|
|
m_curr_time = 0;
|
|
|
|
m_prev_time = 0;
|
|
|
|
m_throttle_fps = true;
|
2018-10-28 18:21:55 -04:00
|
|
|
m_allow_large_dt = false;
|
2018-04-23 13:01:47 -04:00
|
|
|
m_frame_before_loading_world = false;
|
2018-03-02 00:25:14 -05:00
|
|
|
#ifdef WIN32
|
|
|
|
if (parent_pid != 0)
|
|
|
|
{
|
|
|
|
std::string class_name = "separate_process";
|
|
|
|
class_name += StringUtils::toString(GetCurrentProcessId());
|
|
|
|
WNDCLASSEX wx = {};
|
|
|
|
wx.cbSize = sizeof(WNDCLASSEX);
|
2018-07-01 16:56:56 -04:00
|
|
|
wx.lpfnWndProc = separateProcessProc;
|
2018-03-02 00:25:14 -05:00
|
|
|
wx.hInstance = GetModuleHandle(0);
|
|
|
|
wx.lpszClassName = &class_name[0];
|
|
|
|
if (RegisterClassEx(&wx))
|
|
|
|
{
|
|
|
|
CreateWindowEx(0, &class_name[0], "stk_server_only",
|
|
|
|
0, 0, 0, 0, 0, HWND_MESSAGE, NULL, NULL, NULL);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
#endif
|
2008-09-15 00:34:46 -04:00
|
|
|
} // MainLoop
|
2007-05-27 12:01:53 -04:00
|
|
|
|
|
|
|
//-----------------------------------------------------------------------------
|
2008-09-15 00:34:46 -04:00
|
|
|
MainLoop::~MainLoop()
|
2007-05-27 12:01:53 -04:00
|
|
|
{
|
2008-09-15 00:34:46 -04:00
|
|
|
} // ~MainLoop
|
2007-05-27 12:01:53 -04:00
|
|
|
|
2009-07-12 07:54:21 -04:00
|
|
|
//-----------------------------------------------------------------------------
|
|
|
|
/** Returns the current dt, which guarantees a limited frame rate. If dt is
|
|
|
|
* too low (the frame rate too high), the process will sleep to reach the
|
2019-06-04 19:04:18 -04:00
|
|
|
* maximum frame rate.
|
2009-07-12 07:54:21 -04:00
|
|
|
*/
|
|
|
|
float MainLoop::getLimitedDt()
|
|
|
|
{
|
2018-02-11 17:54:46 -05:00
|
|
|
m_prev_time = m_curr_time;
|
2016-09-22 17:56:21 -04:00
|
|
|
float dt = 0;
|
|
|
|
|
2016-05-03 07:28:11 -04:00
|
|
|
// In profile mode without graphics, run with a fixed dt of 1/60
|
2016-05-05 04:25:54 -04:00
|
|
|
if ((ProfileWorld::isProfileMode() && ProfileWorld::isNoGraphics()) ||
|
|
|
|
UserConfigParams::m_arena_ai_stats)
|
2016-05-03 07:28:11 -04:00
|
|
|
{
|
|
|
|
return 1.0f/60.0f;
|
|
|
|
}
|
|
|
|
|
2009-07-12 07:54:21 -04:00
|
|
|
while( 1 )
|
|
|
|
{
|
2019-05-06 03:17:19 -04:00
|
|
|
m_curr_time = StkTime::getMonoTimeMs();
|
2019-04-16 07:16:55 -04:00
|
|
|
if (m_prev_time > m_curr_time)
|
|
|
|
{
|
|
|
|
m_prev_time = m_curr_time;
|
|
|
|
// If system time adjusted backwards, return fixed dt and
|
|
|
|
// resynchronize network timer if exists in client
|
|
|
|
if (STKHost::existHost())
|
|
|
|
{
|
|
|
|
#ifndef SERVER_ONLY
|
|
|
|
if (UserConfigParams::m_artist_debug_mode &&
|
|
|
|
!ProfileWorld::isNoGraphics())
|
|
|
|
{
|
|
|
|
core::stringw err = L"System clock running backwards in"
|
|
|
|
" networking game.";
|
|
|
|
MessageQueue::add(MessageQueue::MT_ERROR, err);
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
Log::error("MainLoop", "System clock running backwards in"
|
|
|
|
" networking game.");
|
|
|
|
if (STKHost::get()->getNetworkTimerSynchronizer())
|
|
|
|
{
|
|
|
|
STKHost::get()->getNetworkTimerSynchronizer()
|
|
|
|
->resynchroniseTimer();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2009-07-12 07:54:21 -04:00
|
|
|
dt = (float)(m_curr_time - m_prev_time);
|
2018-01-22 16:30:23 -05:00
|
|
|
// On a server (i.e. without graphics) the frame rate can be under
|
|
|
|
// 1 ms, i.e. dt = 0. Additionally, the resolution of a sleep
|
|
|
|
// statement is not that precise either: if the sleep statement
|
|
|
|
// would be consistent < 1ms, but the stk time would increase by
|
|
|
|
// 1 ms, the stk clock would be desynchronised from real time
|
|
|
|
// (it would go faster), resulting in synchronisation problems
|
|
|
|
// with clients (server time is supposed to be behind client time).
|
|
|
|
// So we play it safe by adding a loop to make sure at least 1ms
|
|
|
|
// (minimum time that can be handled by the integer timer) delay here.
|
2019-04-16 07:16:55 -04:00
|
|
|
while (dt == 0)
|
2017-08-11 03:22:25 -04:00
|
|
|
{
|
2018-01-22 16:30:23 -05:00
|
|
|
StkTime::sleep(1);
|
2019-05-06 03:17:19 -04:00
|
|
|
m_curr_time = StkTime::getMonoTimeMs();
|
2019-04-16 07:16:55 -04:00
|
|
|
if (m_prev_time > m_curr_time)
|
|
|
|
{
|
|
|
|
Log::error("MainLopp", "System clock keeps backwards!");
|
|
|
|
m_prev_time = m_curr_time;
|
|
|
|
}
|
2018-01-22 16:30:23 -05:00
|
|
|
dt = (float)(m_curr_time - m_prev_time);
|
2017-08-11 03:22:25 -04:00
|
|
|
}
|
2018-01-22 16:30:23 -05:00
|
|
|
|
2015-09-28 17:58:24 -04:00
|
|
|
const World* const world = World::getWorld();
|
|
|
|
if (UserConfigParams::m_fps_debug && world)
|
|
|
|
{
|
|
|
|
const LinearWorld *lw = dynamic_cast<const LinearWorld*>(world);
|
|
|
|
if (lw)
|
|
|
|
{
|
|
|
|
Log::verbose("fps", "time %f distance %f dt %f fps %f",
|
|
|
|
lw->getTime(),
|
2018-04-08 20:47:24 -04:00
|
|
|
lw->getDistanceDownTrackForKart(0, true),
|
2015-09-28 17:58:24 -04:00
|
|
|
dt*0.001f, 1000.0f / dt);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
Log::verbose("fps", "time %f dt %f fps %f",
|
|
|
|
world->getTime(), dt*0.001f, 1000.0f / dt);
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|
2009-07-12 07:54:21 -04:00
|
|
|
|
2016-03-29 18:02:55 -04:00
|
|
|
// Don't allow the game to run slower than a certain amount.
|
2009-07-12 07:54:21 -04:00
|
|
|
// when the computer can't keep it up, slow down the shown time instead
|
2016-03-29 18:02:55 -04:00
|
|
|
// But this can not be done in networking, otherwise the game time on
|
|
|
|
// client and server will not be in synch anymore
|
2018-10-28 18:21:55 -04:00
|
|
|
if ((!NetworkConfig::get()->isNetworking() || !World::getWorld()) &&
|
|
|
|
!m_allow_large_dt)
|
2016-03-29 18:02:55 -04:00
|
|
|
{
|
2017-01-31 02:25:30 -05:00
|
|
|
/* time 3 internal substeps take */
|
|
|
|
const float MAX_ELAPSED_TIME = 3.0f*1.0f / 60.0f*1000.0f;
|
|
|
|
if (dt > MAX_ELAPSED_TIME) dt = MAX_ELAPSED_TIME;
|
2016-03-29 18:02:55 -04:00
|
|
|
}
|
2017-01-31 02:25:30 -05:00
|
|
|
if (!m_throttle_fps || ProfileWorld::isProfileMode()) break;
|
2009-07-12 07:54:21 -04:00
|
|
|
|
2013-05-29 18:04:35 -04:00
|
|
|
// Throttle fps if more than maximum, which can reduce
|
2009-07-12 07:54:21 -04:00
|
|
|
// the noise the fan on a graphics card makes.
|
2017-08-08 19:50:46 -04:00
|
|
|
// When in menus, reduce FPS much, it's not necessary to push to the
|
|
|
|
// maximum for plain menus
|
2017-03-21 19:43:00 -04:00
|
|
|
const int max_fps = (irr_driver->isRecording() &&
|
2017-08-08 19:50:46 -04:00
|
|
|
UserConfigParams::m_limit_game_fps )
|
|
|
|
? UserConfigParams::m_record_fps
|
|
|
|
: ( StateManager::get()->throttleFPS()
|
|
|
|
? 60
|
|
|
|
: UserConfigParams::m_max_fps );
|
2017-01-31 02:25:30 -05:00
|
|
|
const int current_fps = (int)(1000.0f / dt);
|
2017-08-08 19:50:46 -04:00
|
|
|
if (!m_throttle_fps || current_fps <= max_fps ||
|
|
|
|
ProfileWorld::isProfileMode() ) break;
|
2017-01-31 02:25:30 -05:00
|
|
|
|
|
|
|
int wait_time = 1000 / max_fps - 1000 / current_fps;
|
|
|
|
if (wait_time < 1) wait_time = 1;
|
|
|
|
|
|
|
|
PROFILER_PUSH_CPU_MARKER("Throttle framerate", 0, 0, 0);
|
|
|
|
StkTime::sleep(wait_time);
|
|
|
|
PROFILER_POP_CPU_MARKER();
|
2017-08-08 19:50:46 -04:00
|
|
|
} // while(1)
|
2009-07-12 07:54:21 -04:00
|
|
|
|
|
|
|
dt *= 0.001f;
|
|
|
|
return dt;
|
2009-09-16 01:32:17 -04:00
|
|
|
} // getLimitedDt
|
2009-07-12 07:54:21 -04:00
|
|
|
|
|
|
|
//-----------------------------------------------------------------------------
|
|
|
|
/** Updates all race related objects.
|
2018-03-19 18:25:39 -04:00
|
|
|
* \param ticks Number of ticks (physics steps) to simulate - should be 1.
|
2019-02-01 04:22:41 -05:00
|
|
|
* \param fast_forward If true, then only rewinders in network will be
|
|
|
|
* updated, but not the physics.
|
2009-07-12 07:54:21 -04:00
|
|
|
*/
|
2019-02-01 04:22:41 -05:00
|
|
|
void MainLoop::updateRace(int ticks, bool fast_forward)
|
2009-07-12 07:54:21 -04:00
|
|
|
{
|
2017-01-31 02:25:30 -05:00
|
|
|
if (!World::getWorld()) return; // No race on atm - i.e. we are in menu
|
|
|
|
|
2016-02-10 17:03:51 -05:00
|
|
|
// The race event manager will update world in case of an online race
|
2016-12-16 07:26:34 -05:00
|
|
|
if ( RaceEventManager::getInstance() &&
|
|
|
|
RaceEventManager::getInstance()->isRunning() )
|
2019-02-01 04:22:41 -05:00
|
|
|
RaceEventManager::getInstance()->update(ticks, fast_forward);
|
2013-07-31 14:03:11 -04:00
|
|
|
else
|
2018-03-19 18:25:39 -04:00
|
|
|
World::getWorld()->updateWorld(ticks);
|
2009-07-12 07:54:21 -04:00
|
|
|
} // updateRace
|
|
|
|
|
2007-05-27 12:01:53 -04:00
|
|
|
//-----------------------------------------------------------------------------
|
2008-09-15 00:34:46 -04:00
|
|
|
/** Run the actual main loop.
|
2013-08-15 02:20:31 -04:00
|
|
|
* The sequnce in which various parts of STK are updated is:
|
|
|
|
* - Determine next time step size (`getLimitedDt`). This takes maximum fps
|
|
|
|
* into account (i.e. sleep if the fps would be too high), and will actually
|
|
|
|
* slow down the in-game clock if the fps are too low (if more than 3/60 of
|
|
|
|
* a second have passed, more than 3 physics time steps would be needed,
|
|
|
|
* and physics do at most 3 time steps).
|
|
|
|
* - if a race is taking place (i.e. not only a menu being shown), call
|
|
|
|
* `updateRace()`, which is a thin wrapper around a call to
|
|
|
|
* `World::updateWorld()`:
|
|
|
|
* - Update history manager (which will either set the kart position and/or
|
|
|
|
* controls when replaying, or store the current info for a replay).
|
|
|
|
* This is mostly for debugging only (though available even in release
|
|
|
|
* mode).
|
|
|
|
* - Updates Replays - either storing data when not replaying, or
|
|
|
|
* updating kart positions/control when replaying).
|
|
|
|
* - Calls `WorldStatus::update()`, which updates the race state (e.g.
|
|
|
|
* go from 'ready' to 'set' etc), and clock.
|
|
|
|
* - Updates the physics (`Physics::update()`). This will simulate all
|
|
|
|
* physical objects for the specified time with bullet.
|
|
|
|
* - Updates all karts (`Kart::update()`). Obviously the update function
|
|
|
|
* does a lot more than what is described here, this is only supposed to
|
|
|
|
* be a _very_ high level overview:
|
|
|
|
* - Updates its rewinder (to store potentially changed controls
|
|
|
|
* as events) in `KartRewinder::update()`.
|
|
|
|
* - Calls `Moveable::update()`, which takes the new position from
|
|
|
|
* the physics and saves it (and computes dependent values, like
|
|
|
|
* heading, local velocity).
|
|
|
|
* - Updates its controller. This is either:
|
|
|
|
* - an AI using `SkiddingController::update()` (which then will
|
|
|
|
* compute the new controls), or
|
|
|
|
* - a player controller using `PlayerController::update()`, which will
|
|
|
|
* handle smooth steering (in case of digital input devices steering
|
|
|
|
* is adjusted a bit over time to avoid an instant change from all
|
|
|
|
* left to all right). Input events will be handled when updating
|
|
|
|
* the irrlicht driver later at the end of the main loop.
|
|
|
|
* - Updates kart animation (like rescue, ...) if one is shown atm.
|
|
|
|
* - Update attachments.
|
|
|
|
* - update physics, i.e. taking the current steering and updating
|
|
|
|
* the bullet raycast vehicle with that data. The settings are actually
|
|
|
|
* only used in the next frame when the physics are updated.
|
|
|
|
* - Updates all cameras via `Camera::update()`. The camera position and
|
|
|
|
* rotation is adjusted according to the position etc of the kart (and
|
|
|
|
* special circumstances like rescue, falling).
|
|
|
|
* - Updates all projectiles using the projectile manager. Some of the
|
|
|
|
* projectiles are mostly handled by the physics (e.g. a cake will mainly
|
|
|
|
* check if it's out of bounds), others (like basket ball) do all
|
|
|
|
* their aiming and movement here.
|
|
|
|
* - Updates the rewind manager to store rewind states.
|
|
|
|
* - Updates the music manager.
|
|
|
|
* - Updates the input manager (which only updates internal time, actual
|
|
|
|
* input handling follows late)
|
|
|
|
* - Updates the wiimote manager. This will read the data of all wiimotes
|
|
|
|
* and feed the corresponding events to the irrlicht event system.
|
|
|
|
* - Updates the STK internal gui engine. This updates all widgets, and
|
|
|
|
* e.g. takes care of the rotation of the karts in the KartSelection
|
|
|
|
* screen using the ModelViewWidget.
|
|
|
|
* - Updates STK's irrlicht driver `IrrDriver::update()`:
|
|
|
|
* - Calls Irrlicht's `beginScene()` .
|
|
|
|
* - Renders the scene (several times with different viewport if
|
|
|
|
* split screen is being used)
|
|
|
|
* - Calls `GUIEngine::render()`, which renders all widgets with the
|
|
|
|
* help of Irrlicht's GUIEnvironment (`drawAll()`). This will also
|
|
|
|
* handle all events, i.e. all input is now handled (e.g. steering,
|
|
|
|
* firing etc are all set in the corresponding karts depending on
|
|
|
|
* user input).
|
|
|
|
* - Calls Irrlicht's `endScene()`
|
2008-09-15 00:34:46 -04:00
|
|
|
*/
|
|
|
|
void MainLoop::run()
|
2007-05-27 12:01:53 -04:00
|
|
|
{
|
2019-05-06 03:17:19 -04:00
|
|
|
m_curr_time = StkTime::getMonoTimeMs();
|
2018-02-11 17:54:46 -05:00
|
|
|
// DT keeps track of the leftover time, since the race update
|
|
|
|
// happens in fixed timesteps
|
|
|
|
float left_over_time = 0;
|
2018-03-02 00:25:14 -05:00
|
|
|
|
|
|
|
#ifdef WIN32
|
|
|
|
HANDLE parent = 0;
|
|
|
|
if (m_parent_pid != 0)
|
|
|
|
{
|
|
|
|
parent = OpenProcess(PROCESS_ALL_ACCESS, FALSE, m_parent_pid);
|
|
|
|
if (parent == 0 || parent == INVALID_HANDLE_VALUE)
|
|
|
|
{
|
|
|
|
Log::warn("MainLoop", "Cannot open parent handle, this child "
|
|
|
|
"may not be auto destroyed when parent is terminated");
|
|
|
|
}
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
2018-04-06 01:26:11 -04:00
|
|
|
while (!m_abort)
|
2007-05-27 12:01:53 -04:00
|
|
|
{
|
2018-03-02 00:25:14 -05:00
|
|
|
#ifdef WIN32
|
|
|
|
if (parent != 0 && parent != INVALID_HANDLE_VALUE)
|
|
|
|
{
|
|
|
|
MSG msg;
|
|
|
|
while (PeekMessage(&msg, NULL, 0, 0, PM_REMOVE))
|
|
|
|
{
|
|
|
|
TranslateMessage(&msg);
|
|
|
|
DispatchMessage(&msg);
|
|
|
|
if (msg.message == WM_QUIT)
|
2018-09-26 16:27:24 -04:00
|
|
|
{
|
|
|
|
m_request_abort = true;
|
|
|
|
}
|
2018-03-02 00:25:14 -05:00
|
|
|
}
|
|
|
|
// If parent is killed, abort the child main loop too
|
|
|
|
if (WaitForSingleObject(parent, 0) != WAIT_TIMEOUT)
|
2018-09-26 16:27:24 -04:00
|
|
|
{
|
|
|
|
m_request_abort = true;
|
|
|
|
}
|
2018-03-02 00:25:14 -05:00
|
|
|
}
|
2018-03-17 00:26:37 -04:00
|
|
|
#else
|
|
|
|
// POSIX equivalent
|
|
|
|
if (m_parent_pid != 0 && getppid() != (int)m_parent_pid)
|
2018-09-26 16:27:24 -04:00
|
|
|
{
|
|
|
|
m_request_abort = true;
|
|
|
|
}
|
2018-03-02 00:25:14 -05:00
|
|
|
#endif
|
2018-10-28 18:21:55 -04:00
|
|
|
|
2011-09-05 18:55:55 -04:00
|
|
|
PROFILER_PUSH_CPU_MARKER("Main loop", 0xFF, 0x00, 0xF7);
|
2013-05-29 18:04:35 -04:00
|
|
|
|
2018-02-11 17:54:46 -05:00
|
|
|
left_over_time += getLimitedDt();
|
2018-02-23 07:22:49 -05:00
|
|
|
int num_steps = stk_config->time2Ticks(left_over_time);
|
|
|
|
float dt = stk_config->ticks2Time(1);
|
2018-02-11 17:54:46 -05:00
|
|
|
left_over_time -= num_steps * dt ;
|
2016-08-31 02:31:59 -04:00
|
|
|
|
2018-04-06 01:26:11 -04:00
|
|
|
// Shutdown next frame if shutdown request is sent while loading the
|
|
|
|
// world
|
2018-09-26 16:27:24 -04:00
|
|
|
if ((STKHost::existHost() && STKHost::get()->requestedShutdown()) ||
|
|
|
|
m_request_abort)
|
2018-02-17 03:25:53 -05:00
|
|
|
{
|
2018-09-26 16:27:24 -04:00
|
|
|
bool exist_host = STKHost::existHost();
|
2018-03-11 23:38:37 -04:00
|
|
|
core::stringw msg = _("Server connection timed out.");
|
2018-10-24 02:32:01 -04:00
|
|
|
|
2018-09-26 16:27:24 -04:00
|
|
|
if (!m_request_abort)
|
2018-02-24 02:48:30 -05:00
|
|
|
{
|
2018-09-26 16:27:24 -04:00
|
|
|
if (!ProfileWorld::isNoGraphics())
|
2018-09-22 16:06:27 -04:00
|
|
|
{
|
2018-09-26 16:27:24 -04:00
|
|
|
SFXManager::get()->quickSound("anvil");
|
|
|
|
if (!STKHost::get()->getErrorMessage().empty())
|
|
|
|
{
|
|
|
|
msg = STKHost::get()->getErrorMessage();
|
|
|
|
}
|
2018-09-22 16:06:27 -04:00
|
|
|
}
|
2018-02-24 02:48:30 -05:00
|
|
|
}
|
2018-10-24 02:32:01 -04:00
|
|
|
|
2018-09-26 16:27:24 -04:00
|
|
|
if (exist_host == true)
|
|
|
|
{
|
|
|
|
STKHost::get()->shutdown();
|
|
|
|
}
|
2018-08-03 06:36:35 -04:00
|
|
|
|
|
|
|
#ifndef SERVER_ONLY
|
|
|
|
if (CVS->isGLSL())
|
|
|
|
{
|
|
|
|
// Flush all command before delete world, avoid later access
|
|
|
|
SP::SPTextureManager::get()
|
|
|
|
->checkForGLCommand(true/*before_scene*/);
|
|
|
|
// Reset screen in case the minimap was drawn
|
|
|
|
glViewport(0, 0, irr_driver->getActualScreenSize().Width,
|
|
|
|
irr_driver->getActualScreenSize().Height);
|
|
|
|
}
|
|
|
|
#endif
|
2018-10-24 02:32:01 -04:00
|
|
|
|
2018-09-26 16:27:24 -04:00
|
|
|
// In case the user opened a race pause dialog
|
|
|
|
GUIEngine::ModalDialog::dismiss();
|
2018-10-24 02:32:01 -04:00
|
|
|
GUIEngine::ScreenKeyboard::dismiss();
|
|
|
|
|
2018-02-18 01:06:17 -05:00
|
|
|
if (World::getWorld())
|
|
|
|
{
|
2018-04-21 04:00:14 -04:00
|
|
|
race_manager->clearNetworkGrandPrixResult();
|
2018-02-18 01:06:17 -05:00
|
|
|
race_manager->exitRace();
|
|
|
|
}
|
2018-10-24 02:32:01 -04:00
|
|
|
|
2018-09-26 16:27:24 -04:00
|
|
|
if (exist_host == true)
|
2018-02-18 01:06:17 -05:00
|
|
|
{
|
2018-09-26 16:27:24 -04:00
|
|
|
if (!ProfileWorld::isNoGraphics())
|
|
|
|
{
|
|
|
|
StateManager::get()->resetAndSetStack(
|
|
|
|
NetworkConfig::get()->getResetScreens().data());
|
|
|
|
MessageQueue::add(MessageQueue::MT_ERROR, msg);
|
|
|
|
}
|
|
|
|
|
|
|
|
NetworkConfig::get()->unsetNetworking();
|
|
|
|
}
|
2018-10-24 02:32:01 -04:00
|
|
|
|
2018-09-26 16:27:24 -04:00
|
|
|
if (m_request_abort)
|
|
|
|
{
|
|
|
|
m_abort = true;
|
2018-02-23 01:57:59 -05:00
|
|
|
}
|
2018-09-22 16:06:27 -04:00
|
|
|
}
|
2018-02-17 03:25:53 -05:00
|
|
|
|
2018-02-22 02:10:30 -05:00
|
|
|
if (!m_abort)
|
2016-10-17 21:55:30 -04:00
|
|
|
{
|
2018-02-11 17:54:46 -05:00
|
|
|
float frame_duration = num_steps * dt;
|
2018-02-22 02:10:30 -05:00
|
|
|
if (!ProfileWorld::isNoGraphics())
|
|
|
|
{
|
2018-03-19 18:58:34 -04:00
|
|
|
PROFILER_PUSH_CPU_MARKER("Update race", 0, 255, 255);
|
|
|
|
if (World::getWorld())
|
|
|
|
World::getWorld()->updateGraphics(frame_duration);
|
|
|
|
PROFILER_POP_CPU_MARKER();
|
|
|
|
|
2018-02-22 02:10:30 -05:00
|
|
|
// Render the previous frame, and also handle all user input.
|
|
|
|
PROFILER_PUSH_CPU_MARKER("IrrDriver update", 0x00, 0x00, 0x7F);
|
|
|
|
irr_driver->update(frame_duration);
|
|
|
|
PROFILER_POP_CPU_MARKER();
|
|
|
|
|
|
|
|
PROFILER_PUSH_CPU_MARKER("Input/GUI", 0x7F, 0x00, 0x00);
|
|
|
|
input_manager->update(frame_duration);
|
|
|
|
GUIEngine::update(frame_duration);
|
|
|
|
PROFILER_POP_CPU_MARKER();
|
|
|
|
PROFILER_PUSH_CPU_MARKER("Music", 0x7F, 0x00, 0x00);
|
|
|
|
SFXManager::get()->update();
|
|
|
|
PROFILER_POP_CPU_MARKER();
|
|
|
|
}
|
|
|
|
// Some protocols in network will use RequestManager
|
2013-08-18 17:08:27 -04:00
|
|
|
PROFILER_PUSH_CPU_MARKER("Database polling update", 0x00, 0x7F, 0x7F);
|
2018-02-11 17:54:46 -05:00
|
|
|
Online::RequestManager::get()->update(frame_duration);
|
2013-07-31 14:03:11 -04:00
|
|
|
PROFILER_POP_CPU_MARKER();
|
2013-05-29 18:04:35 -04:00
|
|
|
|
2018-09-26 16:27:24 -04:00
|
|
|
m_ticks_adjustment.lock();
|
|
|
|
if (m_ticks_adjustment.getData() != 0)
|
2018-07-01 02:31:42 -04:00
|
|
|
{
|
2018-09-26 16:27:24 -04:00
|
|
|
if (m_ticks_adjustment.getData() > 0)
|
2018-07-01 02:31:42 -04:00
|
|
|
{
|
2018-09-26 16:27:24 -04:00
|
|
|
num_steps += m_ticks_adjustment.getData();
|
|
|
|
m_ticks_adjustment.getData() = 0;
|
2018-07-01 02:31:42 -04:00
|
|
|
}
|
2018-09-26 16:27:24 -04:00
|
|
|
else if (m_ticks_adjustment.getData() < 0)
|
2018-07-01 02:31:42 -04:00
|
|
|
{
|
2018-09-26 16:27:24 -04:00
|
|
|
int new_steps = num_steps + m_ticks_adjustment.getData();
|
|
|
|
if (new_steps < 0)
|
|
|
|
{
|
|
|
|
num_steps = 0;
|
|
|
|
m_ticks_adjustment.getData() = new_steps;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
num_steps = new_steps;
|
|
|
|
m_ticks_adjustment.getData() = 0;
|
|
|
|
}
|
2018-07-01 02:31:42 -04:00
|
|
|
}
|
|
|
|
}
|
2018-09-26 16:27:24 -04:00
|
|
|
m_ticks_adjustment.unlock();
|
2019-02-01 04:22:41 -05:00
|
|
|
|
|
|
|
// Avoid hang when some function in world takes too long time or
|
|
|
|
// when leave / come back from android home button
|
|
|
|
bool fast_forward = NetworkConfig::get()->isNetworking() &&
|
|
|
|
NetworkConfig::get()->isClient() &&
|
|
|
|
num_steps > stk_config->time2Ticks(1.0f);
|
2018-09-26 16:27:24 -04:00
|
|
|
for (int i = 0; i < num_steps; i++)
|
2018-04-23 13:01:47 -04:00
|
|
|
{
|
2018-09-26 16:27:24 -04:00
|
|
|
if (World::getWorld() && history->replayHistory())
|
2018-04-23 13:01:47 -04:00
|
|
|
{
|
2018-09-26 16:27:24 -04:00
|
|
|
history->updateReplay(
|
|
|
|
World::getWorld()->getTicksSinceStart());
|
|
|
|
}
|
2019-02-01 04:22:41 -05:00
|
|
|
|
2018-09-26 16:27:24 -04:00
|
|
|
PROFILER_PUSH_CPU_MARKER("Protocol manager update",
|
|
|
|
0x7F, 0x00, 0x7F);
|
|
|
|
if (auto pm = ProtocolManager::lock())
|
|
|
|
{
|
|
|
|
pm->update(1);
|
|
|
|
}
|
|
|
|
PROFILER_POP_CPU_MARKER();
|
2019-02-01 04:22:41 -05:00
|
|
|
|
2018-09-26 16:27:24 -04:00
|
|
|
PROFILER_PUSH_CPU_MARKER("Update race", 0, 255, 255);
|
|
|
|
if (World::getWorld())
|
|
|
|
{
|
2019-02-01 04:22:41 -05:00
|
|
|
updateRace(1, fast_forward);
|
2018-09-26 16:27:24 -04:00
|
|
|
}
|
|
|
|
PROFILER_POP_CPU_MARKER();
|
2019-02-01 04:22:41 -05:00
|
|
|
|
2018-09-26 16:27:24 -04:00
|
|
|
// We need to check again because update_race may have requested
|
|
|
|
// the main loop to abort; and it's not a good idea to continue
|
|
|
|
// since the GUI engine is no more to be called then.
|
|
|
|
if (m_abort || m_request_abort)
|
|
|
|
break;
|
2019-02-01 04:22:41 -05:00
|
|
|
|
2018-09-26 16:27:24 -04:00
|
|
|
if (m_frame_before_loading_world)
|
|
|
|
{
|
2019-04-08 14:06:59 -04:00
|
|
|
// This will be called when changing introcutscene 1 and 2
|
|
|
|
// in CutsceneWorld::enterRaceOverState
|
|
|
|
// Reset the timer for correct time for cutscene
|
2018-09-26 16:27:24 -04:00
|
|
|
m_frame_before_loading_world = false;
|
2019-05-06 03:17:19 -04:00
|
|
|
m_curr_time = StkTime::getMonoTimeMs();
|
2019-04-08 14:06:59 -04:00
|
|
|
left_over_time = 0.0f;
|
2018-04-23 13:01:47 -04:00
|
|
|
break;
|
|
|
|
}
|
2019-02-01 04:22:41 -05:00
|
|
|
|
2018-09-26 16:27:24 -04:00
|
|
|
if (World::getWorld())
|
|
|
|
{
|
|
|
|
if (World::getWorld()->getPhase()==WorldStatus::SETUP_PHASE)
|
|
|
|
{
|
|
|
|
// Skip the large num steps contributed by loading time
|
|
|
|
World::getWorld()->updateTime(1);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
World::getWorld()->updateTime(1);
|
|
|
|
}
|
|
|
|
} // for i < num_steps
|
2018-08-18 00:49:31 -04:00
|
|
|
|
2018-09-26 16:27:24 -04:00
|
|
|
// Handle controller the last to avoid slow PC sending actions too
|
|
|
|
// late
|
2018-08-18 00:49:31 -04:00
|
|
|
if (!ProfileWorld::isNoGraphics())
|
|
|
|
{
|
|
|
|
// User aborted (e.g. closed window)
|
|
|
|
bool abort = !irr_driver->getDevice()->run();
|
2019-04-08 14:06:59 -04:00
|
|
|
|
|
|
|
if (m_frame_before_loading_world)
|
|
|
|
{
|
|
|
|
// irr_driver->getDevice()->run() loads the world
|
|
|
|
m_frame_before_loading_world = false;
|
2019-05-06 03:17:19 -04:00
|
|
|
m_curr_time = StkTime::getMonoTimeMs();
|
2019-04-08 14:06:59 -04:00
|
|
|
left_over_time = 0.0f;
|
|
|
|
}
|
|
|
|
|
2018-08-18 00:49:31 -04:00
|
|
|
if (abort)
|
2018-09-26 16:27:24 -04:00
|
|
|
{
|
|
|
|
m_request_abort = true;
|
|
|
|
}
|
2018-08-18 00:49:31 -04:00
|
|
|
}
|
2019-02-01 04:22:41 -05:00
|
|
|
|
2018-08-18 00:49:31 -04:00
|
|
|
if (auto gp = GameProtocol::lock())
|
2018-09-26 16:27:24 -04:00
|
|
|
{
|
2018-08-18 00:49:31 -04:00
|
|
|
gp->sendActions();
|
2018-09-26 16:27:24 -04:00
|
|
|
}
|
2018-08-18 00:49:31 -04:00
|
|
|
}
|
2018-02-12 16:41:49 -05:00
|
|
|
PROFILER_POP_CPU_MARKER(); // MainLoop pop
|
2014-09-04 19:45:16 -04:00
|
|
|
PROFILER_SYNC_FRAME();
|
2014-06-25 16:22:14 -04:00
|
|
|
} // while !m_abort
|
2009-07-12 07:54:21 -04:00
|
|
|
|
2018-03-02 00:25:14 -05:00
|
|
|
#ifdef WIN32
|
|
|
|
if (parent != 0 && parent != INVALID_HANDLE_VALUE)
|
|
|
|
CloseHandle(parent);
|
|
|
|
#endif
|
|
|
|
|
2007-05-27 12:01:53 -04:00
|
|
|
} // run
|
|
|
|
|
2018-12-04 17:51:41 -05:00
|
|
|
// ----------------------------------------------------------------------------
|
2018-12-05 17:58:29 -05:00
|
|
|
/** Renders the GUI. This function is used during loading a track to get a
|
|
|
|
* responsive GUI, and allow GUI animations (like a progress bar) to be
|
|
|
|
* shown.
|
|
|
|
* \param phase An integer indicated a phase. The maximum number of phases
|
2018-12-10 19:39:32 -05:00
|
|
|
* is used to show a progress bar. The values are between 0 and 8200.
|
2018-12-05 17:58:29 -05:00
|
|
|
* \param loop_index If the call is from a loop, the current loop index.
|
|
|
|
* \param loop_size The number of loop iterations. Used to smooth update
|
|
|
|
* e.g. a progress bar.
|
|
|
|
*/
|
|
|
|
void MainLoop::renderGUI(int phase, int loop_index, int loop_size)
|
2018-12-04 17:51:41 -05:00
|
|
|
{
|
2018-12-21 06:52:27 -05:00
|
|
|
return;
|
2018-12-09 08:29:14 -05:00
|
|
|
#ifdef SERVER_ONLY
|
|
|
|
return;
|
|
|
|
#else
|
|
|
|
if (NetworkConfig::get()->isNetworking() &&
|
|
|
|
NetworkConfig::get()->isServer() )
|
|
|
|
{
|
|
|
|
return;
|
|
|
|
}
|
2018-12-10 19:39:32 -05:00
|
|
|
// Rendering past phase 7000 causes the minimap to not work
|
2018-12-05 17:58:29 -05:00
|
|
|
// on higher graphical settings
|
2018-12-12 17:40:45 -05:00
|
|
|
if (phase > 7000)
|
|
|
|
{
|
|
|
|
m_request_abort = !irr_driver->getDevice()->run();
|
|
|
|
return;
|
|
|
|
}
|
2018-12-05 17:58:29 -05:00
|
|
|
|
2019-05-06 03:17:19 -04:00
|
|
|
uint64_t now = StkTime::getMonoTimeMs();
|
2018-12-04 17:51:41 -05:00
|
|
|
float dt = (now - m_curr_time)/1000.0f;
|
2018-12-10 19:39:32 -05:00
|
|
|
|
2018-12-12 17:40:45 -05:00
|
|
|
if (dt < 1.0 / 30.0f) return;
|
2018-12-05 17:58:29 -05:00
|
|
|
|
2018-12-04 17:51:41 -05:00
|
|
|
m_curr_time = now;
|
2018-12-21 06:40:52 -05:00
|
|
|
|
2018-12-05 17:58:29 -05:00
|
|
|
// TODO: remove debug output
|
2018-12-12 17:16:57 -05:00
|
|
|
//Log::verbose("mainloop", "Rendergui t %llu dt %f phase %d index %d / %d",
|
|
|
|
// now, dt, phase, loop_index, loop_size);
|
2018-12-04 17:51:41 -05:00
|
|
|
|
|
|
|
irr_driver->update(dt, /*is_loading*/true);
|
|
|
|
GUIEngine::update(dt);
|
2018-12-12 17:40:45 -05:00
|
|
|
m_request_abort = !irr_driver->getDevice()->run();
|
|
|
|
|
2018-12-05 17:58:29 -05:00
|
|
|
//TODO: remove debug output
|
2019-05-06 03:17:19 -04:00
|
|
|
// uint64_t now2 = StkTime::getMonoTimeMs();
|
2018-12-10 19:39:32 -05:00
|
|
|
// Log::verbose("mainloop", " duration t %llu dt %llu", now, now2-now);
|
2018-12-09 08:29:14 -05:00
|
|
|
#endif
|
2018-12-04 17:51:41 -05:00
|
|
|
} // renderGUI
|
2007-05-27 12:01:53 -04:00
|
|
|
/* EOF */
|