2017-02-27 21:57:46 +01:00
# include <mutex>
2017-12-19 15:20:24 +01:00
# include <atomic>
# include <thread>
2017-02-27 21:57:46 +01:00
2020-10-04 10:30:18 +02:00
# include "Common/System/NativeApp.h"
# include "Common/System/System.h"
2023-03-24 17:40:03 +01:00
# include "Common/System/Request.h"
2020-10-01 13:05:04 +02:00
# include "Common/Data/Text/I18n.h"
2020-10-01 09:36:43 +02:00
# include "Common/Input/InputState.h"
2020-10-01 13:05:04 +02:00
# include "Common/Data/Encoding/Utf8.h"
2013-08-13 08:09:36 -07:00
# include "Common/Log.h"
# include "Common/StringUtils.h"
2017-12-15 12:40:38 +01:00
# include "Common/GraphicsContext.h"
2020-08-15 20:53:08 +02:00
# include "Common/TimeUtil.h"
2020-10-04 10:30:18 +02:00
# include "Common/Thread/ThreadUtil.h"
2013-08-13 08:09:36 -07:00
# include "Windows/EmuThread.h"
2014-09-26 21:16:56 -07:00
# include "Windows/W32Util/Misc.h"
2015-09-19 13:14:05 +02:00
# include "Windows/MainWindow.h"
2013-08-13 08:09:36 -07:00
# include "Windows/resource.h"
2017-05-05 06:53:48 -07:00
# include "Windows/WindowsHost.h"
2013-08-13 08:09:36 -07:00
# include "Core/Reporting.h"
# include "Core/MemMap.h"
# include "Core/Core.h"
# include "Core/System.h"
# include "Core/Config.h"
2018-06-16 18:42:31 -07:00
# include "Core/ConfigValues.h"
2012-11-01 16:19:01 +01:00
2023-03-21 12:36:57 +01:00
# if PPSSPP_API(ANY_GL)
# include "Windows/GPU/WindowsGLContext.h"
# endif
# include "Windows/GPU/WindowsVulkanContext.h"
# include "Windows/GPU/D3D9Context.h"
# include "Windows/GPU/D3D11Context.h"
2018-02-07 15:52:19 +01:00
enum class EmuThreadState {
DISABLED ,
START_REQUESTED ,
RUNNING ,
QUIT_REQUESTED ,
STOPPED ,
} ;
2013-06-08 08:32:07 +08:00
2018-02-07 15:52:19 +01:00
static std : : thread emuThread ;
static std : : atomic < int > emuThreadState ( ( int ) EmuThreadState : : DISABLED ) ;
2017-12-15 12:40:38 +01:00
2018-02-07 15:52:19 +01:00
static std : : thread mainThread ;
static bool useEmuThread ;
2018-01-16 14:16:56 +01:00
static std : : string g_error_message ;
2018-02-07 15:52:19 +01:00
static bool g_inLoop ;
2017-12-15 12:40:38 +01:00
2014-08-31 03:05:59 -04:00
extern std : : vector < std : : wstring > GetWideCmdLine ( ) ;
2017-12-15 12:40:38 +01:00
class GraphicsContext ;
static GraphicsContext * g_graphicsContext ;
2018-02-07 15:52:19 +01:00
void MainThreadFunc ( ) ;
2012-11-01 16:19:01 +01:00
2018-02-07 15:52:19 +01:00
// On most other platforms, we let the "main" thread become the render thread and
2018-02-07 13:11:43 +01:00
// start a separate emu thread from that, if needed. Should probably switch to that
// to make it the same on all platforms.
2018-02-07 15:52:19 +01:00
void MainThread_Start ( bool separateEmuThread ) {
useEmuThread = separateEmuThread ;
mainThread = std : : thread ( & MainThreadFunc ) ;
2012-11-01 16:19:01 +01:00
}
2018-02-07 15:52:19 +01:00
void MainThread_Stop ( ) {
2013-08-13 08:03:13 -07:00
// Already stopped?
2014-06-22 09:38:46 +02:00
UpdateUIState ( UISTATE_EXIT ) ;
2018-02-11 15:19:16 -08:00
Core_Stop ( ) ;
2018-02-07 15:52:19 +01:00
mainThread . join ( ) ;
2012-11-01 16:19:01 +01:00
}
2018-02-07 15:52:19 +01:00
bool MainThread_Ready ( ) {
return g_inLoop ;
2013-06-08 08:32:07 +08:00
}
2018-02-07 15:52:19 +01:00
static void EmuThreadFunc ( GraphicsContext * graphicsContext ) {
2020-12-01 00:46:26 +01:00
SetCurrentThreadName ( " Emu " ) ;
2018-01-16 14:16:56 +01:00
2018-02-07 15:52:19 +01:00
// There's no real requirement that NativeInit happen on this thread.
// We just call the update/render loop here.
emuThreadState = ( int ) EmuThreadState : : RUNNING ;
NativeInitGraphics ( graphicsContext ) ;
2018-01-16 14:16:56 +01:00
2018-02-07 15:52:19 +01:00
while ( emuThreadState ! = ( int ) EmuThreadState : : QUIT_REQUESTED ) {
// We're here again, so the game quit. Restart Core_Run() which controls the UI.
// This way they can load a new game.
if ( ! Core_IsActive ( ) )
UpdateUIState ( UISTATE_MENU ) ;
2022-11-08 21:59:08 +01:00
if ( ! Core_Run ( g_graphicsContext ) ) {
emuThreadState = ( int ) EmuThreadState : : QUIT_REQUESTED ;
}
2018-01-16 18:13:31 +01:00
}
2018-02-07 15:52:19 +01:00
emuThreadState = ( int ) EmuThreadState : : STOPPED ;
NativeShutdownGraphics ( ) ;
2018-02-10 16:12:33 -08:00
// Ask the main thread to stop. This prevents a hang on a race condition.
graphicsContext - > StopThread ( ) ;
2017-12-15 12:40:38 +01:00
}
2018-02-07 15:52:19 +01:00
static void EmuThreadStart ( GraphicsContext * graphicsContext ) {
emuThreadState = ( int ) EmuThreadState : : START_REQUESTED ;
emuThread = std : : thread ( & EmuThreadFunc , graphicsContext ) ;
}
2013-06-08 08:32:07 +08:00
2018-02-07 15:52:19 +01:00
static void EmuThreadStop ( ) {
2022-11-08 21:59:08 +01:00
if ( emuThreadState ! = ( int ) EmuThreadState : : QUIT_REQUESTED & &
emuThreadState ! = ( int ) EmuThreadState : : STOPPED ) {
emuThreadState = ( int ) EmuThreadState : : QUIT_REQUESTED ;
}
2018-02-07 15:52:19 +01:00
}
static void EmuThreadJoin ( ) {
emuThread . join ( ) ;
2020-08-15 16:13:24 +02:00
INFO_LOG ( SYSTEM , " EmuThreadJoin - joined " ) ;
2018-02-07 15:52:19 +01:00
}
2023-03-21 12:36:57 +01:00
bool CreateGraphicsBackend ( std : : string * error_message , GraphicsContext * * ctx ) {
WindowsGraphicsContext * graphicsContext = nullptr ;
switch ( g_Config . iGPUBackend ) {
# if PPSSPP_API(ANY_GL)
case ( int ) GPUBackend : : OPENGL :
graphicsContext = new WindowsGLContext ( ) ;
break ;
# endif
case ( int ) GPUBackend : : DIRECT3D9 :
graphicsContext = new D3D9Context ( ) ;
break ;
case ( int ) GPUBackend : : DIRECT3D11 :
graphicsContext = new D3D11Context ( ) ;
break ;
case ( int ) GPUBackend : : VULKAN :
graphicsContext = new WindowsVulkanContext ( ) ;
break ;
default :
return false ;
}
if ( graphicsContext - > Init ( MainWindow : : GetHInstance ( ) , MainWindow : : GetDisplayHWND ( ) , error_message ) ) {
* ctx = graphicsContext ;
return true ;
} else {
delete graphicsContext ;
* ctx = nullptr ;
return false ;
}
}
2018-02-07 15:52:19 +01:00
void MainThreadFunc ( ) {
2023-03-24 17:52:17 +01:00
// We'll start up a separate thread we'll call Emu
SetCurrentThreadName ( useEmuThread ? " Render " : " Emu " ) ;
2012-11-01 16:19:01 +01:00
2023-03-25 10:43:00 +01:00
SetConsolePosition ( ) ;
2023-03-24 19:57:24 +01:00
2023-03-24 17:40:03 +01:00
System_SetWindowTitle ( " " ) ;
2017-05-05 06:53:48 -07:00
2017-12-19 15:20:24 +01:00
// We convert command line arguments to UTF-8 immediately.
2014-08-31 01:17:25 -04:00
std : : vector < std : : wstring > wideArgs = GetWideCmdLine ( ) ;
std : : vector < std : : string > argsUTF8 ;
2014-08-31 04:16:22 -04:00
for ( auto & string : wideArgs ) {
2014-08-31 03:05:59 -04:00
argsUTF8 . push_back ( ConvertWStringToUTF8 ( string ) ) ;
2014-08-31 01:17:25 -04:00
}
std : : vector < const char * > args ;
2014-08-31 04:16:22 -04:00
for ( auto & string : argsUTF8 ) {
2014-08-31 01:17:25 -04:00
args . push_back ( string . c_str ( ) ) ;
}
2017-04-15 16:33:30 -07:00
bool performingRestart = NativeIsRestarting ( ) ;
2021-02-28 15:09:36 -08:00
NativeInit ( static_cast < int > ( args . size ( ) ) , & args [ 0 ] , " " , " " , nullptr ) ;
2014-08-31 01:17:25 -04:00
2020-03-28 06:19:11 -07:00
if ( g_Config . iGPUBackend = = ( int ) GPUBackend : : OPENGL ) {
if ( ! useEmuThread ) {
// Okay, we must've switched to OpenGL. Let's flip the emu thread on.
useEmuThread = true ;
2020-12-01 00:46:26 +01:00
SetCurrentThreadName ( " Render " ) ;
2020-03-28 06:19:11 -07:00
}
} else if ( useEmuThread ) {
// We must've failed over from OpenGL, flip the emu thread off.
useEmuThread = false ;
2020-12-01 00:46:26 +01:00
SetCurrentThreadName ( " Emu " ) ;
2020-03-28 06:19:11 -07:00
}
2018-09-01 13:57:20 -07:00
if ( g_Config . sFailedGPUBackends . find ( " ALL " ) ! = std : : string : : npos ) {
Reporting : : ReportMessage ( " Graphics init error: %s " , " ALL " ) ;
2023-04-06 00:34:50 +02:00
auto err = GetI18NCategory ( I18NCat : : ERRORS ) ;
2020-05-08 23:20:37 -07:00
const char * defaultErrorAll = " PPSSPP failed to startup with any graphics backend. Try upgrading your graphics and other drivers. " ;
const char * genericError = err - > T ( " GenericAllStartupError " , defaultErrorAll ) ;
2018-09-01 13:57:20 -07:00
std : : wstring title = ConvertUTF8ToWString ( err - > T ( " GenericGraphicsError " , " Graphics Error " ) ) ;
MessageBox ( 0 , ConvertUTF8ToWString ( genericError ) . c_str ( ) , title . c_str ( ) , MB_OK ) ;
// Let's continue (and probably crash) just so they have a way to keep trying.
}
2023-03-21 11:21:19 +01:00
System_Notify ( SystemNotification : : UI ) ;
2014-01-27 07:36:39 +08:00
2013-03-10 23:08:57 +01:00
std : : string error_string ;
2023-03-21 12:36:57 +01:00
bool success = CreateGraphicsBackend ( & error_string , & g_graphicsContext ) ;
2018-01-16 14:16:56 +01:00
if ( success ) {
2018-02-07 15:52:19 +01:00
// Main thread is the render thread.
success = g_graphicsContext - > InitFromRenderThread ( & error_string ) ;
2018-01-16 14:16:56 +01:00
}
if ( ! success ) {
2017-04-15 16:33:30 -07:00
// Before anything: are we restarting right now?
if ( performingRestart ) {
// Okay, switching graphics didn't work out. Probably a driver bug - fallback to restart.
// This happens on NVIDIA when switching OpenGL -> Vulkan.
2019-02-23 10:49:49 +01:00
g_Config . Save ( " switch_graphics_failed " ) ;
2017-04-15 16:33:30 -07:00
W32Util : : ExitAndRestart ( ) ;
}
2023-04-06 00:34:50 +02:00
auto err = GetI18NCategory ( I18NCat : : ERRORS ) ;
2014-08-17 16:07:14 +02:00
Reporting : : ReportMessage ( " Graphics init error: %s " , error_string . c_str ( ) ) ;
2014-09-26 21:16:56 -07:00
2016-03-13 09:33:39 -07:00
const char * defaultErrorVulkan = " Failed initializing graphics. Try upgrading your graphics drivers. \n \n Would you like to try switching to OpenGL? \n \n Error message: " ;
2014-09-26 21:16:56 -07:00
const char * defaultErrorOpenGL = " Failed initializing graphics. Try upgrading your graphics drivers. \n \n Would you like to try switching to DirectX 9? \n \n Error message: " ;
2014-09-28 15:55:16 +08:00
const char * defaultErrorDirect3D9 = " Failed initializing graphics. Try upgrading your graphics drivers and directx 9 runtime. \n \n Would you like to try switching to OpenGL? \n \n Error message: " ;
2014-09-26 21:16:56 -07:00
const char * genericError ;
2017-12-26 15:55:24 -08:00
GPUBackend nextBackend = GPUBackend : : DIRECT3D9 ;
2014-09-26 21:16:56 -07:00
switch ( g_Config . iGPUBackend ) {
2017-12-26 15:55:24 -08:00
case ( int ) GPUBackend : : DIRECT3D9 :
nextBackend = GPUBackend : : OPENGL ;
2014-09-26 21:16:56 -07:00
genericError = err - > T ( " GenericDirect3D9Error " , defaultErrorDirect3D9 ) ;
break ;
2017-12-26 15:55:24 -08:00
case ( int ) GPUBackend : : VULKAN :
nextBackend = GPUBackend : : OPENGL ;
2016-03-24 17:03:04 -07:00
genericError = err - > T ( " GenericVulkanError " , defaultErrorVulkan ) ;
2016-03-13 09:33:39 -07:00
break ;
2017-12-26 15:55:24 -08:00
case ( int ) GPUBackend : : OPENGL :
2014-09-26 21:16:56 -07:00
default :
2017-12-26 15:55:24 -08:00
nextBackend = GPUBackend : : DIRECT3D9 ;
2014-09-26 21:16:56 -07:00
genericError = err - > T ( " GenericOpenGLError " , defaultErrorOpenGL ) ;
break ;
}
std : : string full_error = StringFromFormat ( " %s \n \n %s " , genericError , error_string . c_str ( ) ) ;
std : : wstring title = ConvertUTF8ToWString ( err - > T ( " GenericGraphicsError " , " Graphics Error " ) ) ;
bool yes = IDYES = = MessageBox ( 0 , ConvertUTF8ToWString ( full_error ) . c_str ( ) , title . c_str ( ) , MB_ICONERROR | MB_YESNO ) ;
2021-02-14 09:59:04 -08:00
ERROR_LOG ( BOOT , " %s " , full_error . c_str ( ) ) ;
2013-09-11 21:35:04 +02:00
2014-09-26 21:16:56 -07:00
if ( yes ) {
// Change the config to the alternative and restart.
2017-12-26 15:55:24 -08:00
g_Config . iGPUBackend = ( int ) nextBackend ;
2018-09-01 13:57:20 -07:00
// Clear this to ensure we try their selection.
g_Config . sFailedGPUBackends . clear ( ) ;
2019-02-23 10:49:49 +01:00
g_Config . Save ( " save_graphics_fallback " ) ;
2014-09-26 21:16:56 -07:00
W32Util : : ExitAndRestart ( ) ;
2018-03-25 00:46:11 +01:00
} else {
if ( g_Config . iGPUBackend = = ( int ) GPUBackend : : DIRECT3D9 ) {
// Allow the user to download the DX9 runtime.
2023-03-21 10:42:23 +01:00
System_LaunchUrl ( LaunchUrlType : : BROWSER_URL , " https://www.microsoft.com/en-us/download/details.aspx?id=34429 " ) ;
2018-03-25 00:46:11 +01:00
}
2014-09-26 21:16:56 -07:00
}
// No safe way out without graphics.
2018-03-25 00:46:11 +01:00
ExitProcess ( 1 ) ;
2013-03-10 23:08:57 +01:00
}
2012-11-01 16:19:01 +01:00
2018-02-07 15:52:19 +01:00
GraphicsContext * graphicsContext = g_graphicsContext ;
if ( ! useEmuThread ) {
NativeInitGraphics ( graphicsContext ) ;
NativeResized ( ) ;
}
2012-11-01 16:19:01 +01:00
2021-08-28 09:14:53 -07:00
DEBUG_LOG ( BOOT , " Done. " ) ;
2012-11-01 16:19:01 +01:00
2013-06-08 08:32:07 +08:00
if ( coreState = = CORE_POWERDOWN ) {
INFO_LOG ( BOOT , " Exit before core loop. " ) ;
goto shutdown ;
}
2018-02-07 15:52:19 +01:00
g_inLoop = true ;
if ( useEmuThread ) {
EmuThreadStart ( graphicsContext ) ;
}
graphicsContext - > ThreadStart ( ) ;
2013-06-08 08:32:07 +08:00
if ( g_Config . bBrowse )
PostMessage ( MainWindow : : GetHWND ( ) , WM_COMMAND , ID_FILE_LOAD , 0 ) ;
2018-02-07 13:11:43 +01:00
Core_EnableStepping ( false ) ;
2013-06-08 08:32:07 +08:00
2018-02-07 15:52:19 +01:00
if ( useEmuThread ) {
while ( emuThreadState ! = ( int ) EmuThreadState : : DISABLED ) {
graphicsContext - > ThreadFrame ( ) ;
if ( GetUIState ( ) = = UISTATE_EXIT ) {
break ;
}
}
} else {
while ( GetUIState ( ) ! = UISTATE_EXIT ) {
// We're here again, so the game quit. Restart Core_Run() which controls the UI.
// This way they can load a new game.
if ( ! Core_IsActive ( ) )
UpdateUIState ( UISTATE_MENU ) ;
Core_Run ( g_graphicsContext ) ;
2021-03-28 19:44:17 -07:00
if ( coreState = = CORE_BOOT_ERROR ) {
break ;
}
2018-02-07 15:52:19 +01:00
}
}
Core_Stop ( ) ;
2018-02-10 16:23:34 -08:00
if ( ! useEmuThread ) {
// Process the shutdown. Without this, non-GL delays 800ms on shutdown.
Core_Run ( g_graphicsContext ) ;
}
2018-02-07 15:52:19 +01:00
Core_WaitInactive ( 800 ) ;
g_inLoop = false ;
if ( useEmuThread ) {
EmuThreadStop ( ) ;
2018-02-11 11:40:11 -08:00
while ( graphicsContext - > ThreadFrame ( ) ) {
2018-02-07 15:52:19 +01:00
// Need to keep eating frames to allow the EmuThread to exit correctly.
2018-02-11 11:40:11 -08:00
continue ;
2018-02-07 15:52:19 +01:00
}
EmuThreadJoin ( ) ;
2013-06-18 00:31:15 -07:00
}
2012-11-01 16:19:01 +01:00
2012-11-05 13:42:33 +01:00
shutdown :
2013-08-16 00:57:04 +02:00
2018-02-07 15:52:19 +01:00
if ( ! useEmuThread ) {
NativeShutdownGraphics ( ) ;
}
g_graphicsContext - > ThreadEnd ( ) ;
g_graphicsContext - > ShutdownFromRenderThread ( ) ;
2017-05-07 10:40:11 +02:00
2023-03-21 12:36:57 +01:00
g_graphicsContext - > Shutdown ( ) ;
2023-03-25 17:09:03 -07:00
UpdateConsolePosition ( ) ;
2013-03-29 18:50:08 +01:00
NativeShutdown ( ) ;
2012-11-01 16:19:01 +01:00
2018-02-07 15:52:19 +01:00
PostMessage ( MainWindow : : GetHWND ( ) , MainWindow : : WM_USER_UPDATE_UI , 0 , 0 ) ;
}