Files
itgmania212121/src/GameLoop.cpp
T
Martin Natano b68ca517e6 Clean up math functions
- Remove checking for standard functions from the build system
- Prefix all invocations with std::
- Replace suffixed functions with unprefixed versions
- Include <cmath> in all files that use it and remove the global include

e.g. floorf(x) -> std::floor(x)
2023-04-19 19:31:40 +02:00

498 lines
13 KiB
C++

#include "global.h"
#include "GameLoop.h"
#include "RageLog.h"
#include "RageTextureManager.h"
#include "RageSoundManager.h"
#include "PrefsManager.h"
#include "RageDisplay.h"
#include "arch/ArchHooks/ArchHooks.h"
#include "GameSoundManager.h"
#include "ThemeManager.h"
#include "SongManager.h"
#include "GameState.h"
#include "MemoryCardManager.h"
#include "ScreenManager.h"
#include "InputFilter.h"
#include "InputMapper.h"
#include "RageFileManager.h"
#include "LightsManager.h"
#include "RageTimer.h"
#include "RageInput.h"
#include <cmath>
static RageTimer g_GameplayTimer;
static Preference<bool> g_bNeverBoostAppPriority( "NeverBoostAppPriority", false );
/* experimental: force a specific update rate. This prevents big animation
* jumps on frame skips. 0 to disable. */
static Preference<float> g_fConstantUpdateDeltaSeconds( "ConstantUpdateDeltaSeconds", 0 );
void HandleInputEvents( float fDeltaTime );
static float g_fUpdateRate = 1;
void GameLoop::SetUpdateRate( float fUpdateRate )
{
g_fUpdateRate = fUpdateRate;
}
static void CheckGameLoopTimerSkips( float fDeltaTime )
{
if( !PREFSMAN->m_bLogSkips )
return;
static int iLastFPS = 0;
int iThisFPS = DISPLAY->GetFPS();
/* If vsync is on, and we have a solid framerate (vsync == refresh and we've
* sustained this for at least one second), we expect the amount of time for
* the last frame to be 1/FPS. */
if( iThisFPS != DISPLAY->GetActualVideoModeParams().rate || iThisFPS != iLastFPS )
{
iLastFPS = iThisFPS;
return;
}
const float fExpectedTime = 1.0f / iThisFPS;
const float fDifference = fDeltaTime - fExpectedTime;
if( std::abs(fDifference) > 0.002f && std::abs(fDifference) < 0.100f )
LOG->Trace( "GameLoop timer skip: %i FPS, expected %.3f, got %.3f (%.3f difference)",
iThisFPS, fExpectedTime, fDeltaTime, fDifference );
}
static bool ChangeAppPri()
{
if( g_bNeverBoostAppPriority.Get() )
return false;
// if using NTPAD don't boost or else input is laggy
#if defined(_WINDOWS)
{
std::vector<InputDeviceInfo> vDevices;
// This can get called before INPUTMAN is constructed.
if( INPUTMAN )
{
INPUTMAN->GetDevicesAndDescriptions(vDevices);
if (std::any_of(vDevices.begin(), vDevices.end(), [](InputDeviceInfo const &d) {
return d.sDesc.find("NTPAD") != std::string::npos;
}))
{
LOG->Trace( "Using NTPAD. Don't boost priority." );
return false;
}
}
}
#endif
// If this is a debug build, don't. It makes the VC debugger sluggish.
#if defined(WIN32) && defined(DEBUG)
return false;
#else
return true;
#endif
}
static void CheckFocus()
{
if( !HOOKS->AppFocusChanged() )
return;
// If we lose focus, we may lose input events, especially key releases.
INPUTFILTER->Reset();
if( ChangeAppPri() )
{
if( HOOKS->AppHasFocus() )
HOOKS->BoostPriority();
else
HOOKS->UnBoostPriority();
}
}
// On the next update, change themes, and load sNewScreen.
static RString g_NewTheme;
static RString g_NewGame;
void GameLoop::ChangeTheme(const RString &sNewTheme)
{
g_NewTheme = sNewTheme;
}
void GameLoop::ChangeGame(const RString& new_game, const RString& new_theme)
{
g_NewGame= new_game;
g_NewTheme= new_theme;
}
#include "StepMania.h" // XXX
#include "GameManager.h"
#include "Game.h"
namespace
{
void DoChangeTheme()
{
SAFE_DELETE( SCREENMAN );
TEXTUREMAN->DoDelayedDelete();
// In case the previous theme overloaded class bindings, reinitialize them.
LUA->RegisterTypes();
// We always need to force the theme to reload because we cleared the lua
// state by calling RegisterTypes so the scripts in Scripts/ need to run.
THEME->SwitchThemeAndLanguage( g_NewTheme, THEME->GetCurLanguage(), PREFSMAN->m_bPseudoLocalize, true );
PREFSMAN->m_sTheme.Set( g_NewTheme );
// Apply the new window title, icon and aspect ratio.
StepMania::ApplyGraphicOptions();
SCREENMAN = new ScreenManager();
StepMania::ResetGame();
SCREENMAN->ThemeChanged();
// The previous system for changing the theme fetched the "NextScreen"
// metric from the current theme, then changed the theme, then tried to
// set the new screen to the name that had been fetched.
// If the new screen didn't exist in the new theme, there would be a
// crash.
// So now the correct thing to do is for a theme to specify its entry
// point after a theme change, ensuring that we are going to a valid
// screen and not crashing. -Kyz
RString new_screen= THEME->GetMetric("Common", "InitialScreen");
if(THEME->HasMetric("Common", "AfterThemeChangeScreen"))
{
RString after_screen= THEME->GetMetric("Common", "AfterThemeChangeScreen");
if(SCREENMAN->IsScreenNameValid(after_screen))
{
new_screen= after_screen;
}
}
if(!SCREENMAN->IsScreenNameValid(new_screen))
{
new_screen= "ScreenInitialScreenIsInvalid";
}
SCREENMAN->SetNewScreen(new_screen);
g_NewTheme = RString();
}
void DoChangeGame()
{
const Game* g= GAMEMAN->StringToGame(g_NewGame);
ASSERT(g != nullptr);
GAMESTATE->SetCurGame(g);
bool theme_changing= false;
// The prefs allow specifying a different default theme to use for each
// game type. So if a theme name isn't passed in, fetch from the prefs.
if(g_NewTheme.empty())
{
g_NewTheme= PREFSMAN->m_sTheme;
}
if(g_NewTheme != THEME->GetCurThemeName() && THEME->IsThemeSelectable(g_NewTheme))
{
theme_changing= true;
}
if(theme_changing)
{
SAFE_DELETE(SCREENMAN);
TEXTUREMAN->DoDelayedDelete();
LUA->RegisterTypes();
THEME->SwitchThemeAndLanguage(g_NewTheme, THEME->GetCurLanguage(),
PREFSMAN->m_bPseudoLocalize);
PREFSMAN->m_sTheme.Set(g_NewTheme);
StepMania::ApplyGraphicOptions();
SCREENMAN= new ScreenManager();
}
StepMania::ResetGame();
RString new_screen= THEME->GetMetric("Common", "InitialScreen");
RString after_screen;
if(theme_changing)
{
SCREENMAN->ThemeChanged();
if(THEME->HasMetric("Common", "AfterGameAndThemeChangeScreen"))
{
after_screen= THEME->GetMetric("Common", "AfterGameAndThemeChangeScreen");
}
}
else
{
if(THEME->HasMetric("Common", "AfterGameChangeScreen"))
{
after_screen= THEME->GetMetric("Common", "AfterGameChangeScreen");
}
}
if(SCREENMAN->IsScreenNameValid(after_screen))
{
new_screen= after_screen;
}
SCREENMAN->SetNewScreen(new_screen);
// Set the input scheme for the new game, and load keymaps.
if( INPUTMAPPER )
{
INPUTMAPPER->SetInputScheme(&g->m_InputScheme);
INPUTMAPPER->ReadMappingsFromDisk();
}
// aj's comment transplanted from ScreenOptionsMasterPrefs.cpp:GameSel. -Kyz
/* Reload metrics to force a refresh of CommonMetrics::DIFFICULTIES_TO_SHOW,
* mainly if we're not switching themes. I'm not sure if this was the
* case going from theme to theme, but if it was, it should be fixed
* now. There's probably be a better way to do it, but I'm not sure
* what it'd be. -aj */
THEME->UpdateLuaGlobals();
THEME->ReloadMetrics();
g_NewGame= RString();
g_NewTheme= RString();
}
}
static bool m_bUpdatedDuringVBLANK = false;
void GameLoop::UpdateAllButDraw(bool bRunningFromVBLANK)
{
//if we are running our once per frame routine and we were already run from VBLANK, we did the work already
if (!bRunningFromVBLANK && m_bUpdatedDuringVBLANK)
{
m_bUpdatedDuringVBLANK = false;
return; //would it kill us to run it again or do we want to draw asap?
}
//if vblank called us, we will tell the game loop we received an update for the frame it wants to process
if (bRunningFromVBLANK) m_bUpdatedDuringVBLANK = true;
else m_bUpdatedDuringVBLANK = false;
// Update our stuff
float fDeltaTime = g_GameplayTimer.GetDeltaTime();
if (g_fConstantUpdateDeltaSeconds > 0)
fDeltaTime = g_fConstantUpdateDeltaSeconds;
CheckGameLoopTimerSkips(fDeltaTime);
fDeltaTime *= g_fUpdateRate;
// Update SOUNDMAN early (before any RageSound::GetPosition calls), to flush position data.
SOUNDMAN->Update();
/* Update song beat information -before- calling update on all the classes that
* depend on it. If you don't do this first, the classes are all acting on old
* information and will lag. (but no longer fatally, due to timestamping -glenn) */
SOUND->Update(fDeltaTime);
TEXTUREMAN->Update(fDeltaTime);
GAMESTATE->Update(fDeltaTime);
SCREENMAN->Update(fDeltaTime);
MEMCARDMAN->Update();
/* Important: Process input AFTER updating game logic, or input will be
* acting on song beat from last frame */
HandleInputEvents(fDeltaTime);
//bandaid for low max audio sample counter
SOUNDMAN->low_sample_count_workaround();
LIGHTSMAN->Update(fDeltaTime);
}
void GameLoop::RunGameLoop()
{
/* People may want to do something else while songs are loading, so do
* this after loading songs. */
if( ChangeAppPri() )
HOOKS->BoostPriority();
while( !ArchHooks::UserQuit() )
{
if(!g_NewGame.empty())
{
DoChangeGame();
}
if(!g_NewTheme.empty())
{
DoChangeTheme();
}
CheckFocus();
UpdateAllButDraw(false);
if( INPUTMAN->DevicesChanged() )
{
INPUTFILTER->Reset(); // fix "buttons stuck" once per frame if button held while unplugged
INPUTMAN->LoadDrivers();
RString sMessage;
if( INPUTMAPPER->CheckForChangedInputDevicesAndRemap(sMessage) )
SCREENMAN->SystemMessage( sMessage );
}
SCREENMAN->Draw();
}
// If we ended mid-game, finish up.
GAMESTATE->SaveLocalData();
if( ChangeAppPri() )
HOOKS->UnBoostPriority();
}
class ConcurrentRenderer
{
public:
ConcurrentRenderer();
~ConcurrentRenderer();
void Start();
void Stop();
private:
RageThread m_Thread;
RageEvent m_Event;
bool m_bShutdown;
void RenderThread();
static int StartRenderThread( void *p );
enum State { RENDERING_IDLE, RENDERING_START, RENDERING_ACTIVE, RENDERING_END };
State m_State;
};
static ConcurrentRenderer *g_pConcurrentRenderer = nullptr;
ConcurrentRenderer::ConcurrentRenderer():
m_Event("ConcurrentRenderer")
{
m_bShutdown = false;
m_State = RENDERING_IDLE;
m_Thread.SetName( "ConcurrentRenderer" );
m_Thread.Create( StartRenderThread, this );
}
ConcurrentRenderer::~ConcurrentRenderer()
{
ASSERT( m_State == RENDERING_IDLE );
m_bShutdown = true;
m_Thread.Wait();
}
void ConcurrentRenderer::Start()
{
DISPLAY->BeginConcurrentRenderingMainThread();
m_Event.Lock();
ASSERT( m_State == RENDERING_IDLE );
m_State = RENDERING_START;
m_Event.Signal();
while( m_State != RENDERING_ACTIVE )
m_Event.Wait();
m_Event.Unlock();
}
void ConcurrentRenderer::Stop()
{
m_Event.Lock();
ASSERT( m_State == RENDERING_ACTIVE );
m_State = RENDERING_END;
m_Event.Signal();
while( m_State != RENDERING_IDLE )
m_Event.Wait();
m_Event.Unlock();
DISPLAY->EndConcurrentRenderingMainThread();
}
void ConcurrentRenderer::RenderThread()
{
ASSERT( SCREENMAN != nullptr );
while( !m_bShutdown )
{
m_Event.Lock();
while( m_State == RENDERING_IDLE && !m_bShutdown )
m_Event.Wait();
m_Event.Unlock();
if( m_State == RENDERING_START )
{
/* We're starting to render. Set up, and then kick the event to wake
* up the calling thread. */
DISPLAY->BeginConcurrentRendering();
HOOKS->SetupConcurrentRenderingThread();
LOG->Trace( "ConcurrentRenderer::RenderThread start" );
m_Event.Lock();
m_State = RENDERING_ACTIVE;
m_Event.Signal();
m_Event.Unlock();
}
/* This is started during Update(). The next thing the game loop
* will do is Draw, so shift operations around to put Draw at the
* top. This makes sure updates are seamless. */
if( m_State == RENDERING_ACTIVE )
{
SCREENMAN->Draw();
float fDeltaTime = g_GameplayTimer.GetDeltaTime();
SCREENMAN->Update( fDeltaTime );
}
if( m_State == RENDERING_END )
{
LOG->Trace( "ConcurrentRenderer::RenderThread done" );
DISPLAY->EndConcurrentRendering();
m_Event.Lock();
m_State = RENDERING_IDLE;
m_Event.Signal();
m_Event.Unlock();
}
}
}
int ConcurrentRenderer::StartRenderThread( void *p )
{
((ConcurrentRenderer *) p)->RenderThread();
return 0;
}
void GameLoop::StartConcurrentRendering()
{
if( g_pConcurrentRenderer == nullptr )
g_pConcurrentRenderer = new ConcurrentRenderer;
g_pConcurrentRenderer->Start();
}
void GameLoop::FinishConcurrentRendering()
{
g_pConcurrentRenderer->Stop();
}
/*
* (c) 2001-2005 Chris Danford, Glenn Maynard
* All rights reserved.
*
* Permission is hereby granted, free of charge, to any person obtaining a
* copy of this software and associated documentation files (the
* "Software"), to deal in the Software without restriction, including
* without limitation the rights to use, copy, modify, merge, publish,
* distribute, and/or sell copies of the Software, and to permit persons to
* whom the Software is furnished to do so, provided that the above
* copyright notice(s) and this permission notice appear in all copies of
* the Software and that both the above copyright notice(s) and this
* permission notice appear in supporting documentation.
*
* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS
* OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
* MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT OF
* THIRD PARTY RIGHTS. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR HOLDERS
* INCLUDED IN THIS NOTICE BE LIABLE FOR ANY CLAIM, OR ANY SPECIAL INDIRECT
* OR CONSEQUENTIAL DAMAGES, OR ANY DAMAGES WHATSOEVER RESULTING FROM LOSS
* OF USE, DATA OR PROFITS, WHETHER IN AN ACTION OF CONTRACT, NEGLIGENCE OR
* OTHER TORTIOUS ACTION, ARISING OUT OF OR IN CONNECTION WITH THE USE OR
* PERFORMANCE OF THIS SOFTWARE.
*/