mirror of
https://github.com/dolphin-emu/dolphin.git
synced 2025-01-18 20:11:16 +01:00
842 lines
22 KiB
C++
842 lines
22 KiB
C++
// Copyright 2008 Dolphin Emulator Project
|
|
// Licensed under GPLv2+
|
|
// Refer to the license.txt file included.
|
|
|
|
#include <cctype>
|
|
|
|
#ifdef _WIN32
|
|
#include <windows.h>
|
|
#endif
|
|
|
|
#include "AudioCommon/AudioCommon.h"
|
|
|
|
#include "Common/Atomic.h"
|
|
#include "Common/CommonPaths.h"
|
|
#include "Common/CommonTypes.h"
|
|
#include "Common/CPUDetect.h"
|
|
#include "Common/MathUtil.h"
|
|
#include "Common/MemoryUtil.h"
|
|
#include "Common/StringUtil.h"
|
|
#include "Common/Thread.h"
|
|
#include "Common/Timer.h"
|
|
#include "Common/Logging/LogManager.h"
|
|
|
|
#include "Core/ConfigManager.h"
|
|
#include "Core/Core.h"
|
|
#include "Core/CoreTiming.h"
|
|
#include "Core/DSPEmulator.h"
|
|
#include "Core/Host.h"
|
|
#include "Core/MemTools.h"
|
|
#include "Core/Movie.h"
|
|
#include "Core/NetPlayClient.h"
|
|
#include "Core/NetPlayProto.h"
|
|
#include "Core/PatchEngine.h"
|
|
#include "Core/State.h"
|
|
#include "Core/Boot/Boot.h"
|
|
#include "Core/FifoPlayer/FifoPlayer.h"
|
|
#include "Core/HW/AudioInterface.h"
|
|
#include "Core/HW/CPU.h"
|
|
#include "Core/HW/DSP.h"
|
|
#include "Core/HW/EXI.h"
|
|
#include "Core/HW/GCKeyboard.h"
|
|
#include "Core/HW/GCPad.h"
|
|
#include "Core/HW/GPFifo.h"
|
|
#include "Core/HW/HW.h"
|
|
#include "Core/HW/Memmap.h"
|
|
#include "Core/HW/ProcessorInterface.h"
|
|
#include "Core/HW/SystemTimers.h"
|
|
#include "Core/HW/VideoInterface.h"
|
|
#include "Core/HW/Wiimote.h"
|
|
#include "Core/IPC_HLE/WII_IPC_HLE_Device_usb.h"
|
|
#include "Core/IPC_HLE/WII_IPC_HLE_WiiMote.h"
|
|
#include "Core/IPC_HLE/WII_Socket.h"
|
|
#include "Core/PowerPC/JitInterface.h"
|
|
#include "Core/PowerPC/PowerPC.h"
|
|
|
|
#ifdef USE_GDBSTUB
|
|
#include "Core/PowerPC/GDBStub.h"
|
|
#endif
|
|
|
|
#include "DiscIO/FileMonitor.h"
|
|
#include "InputCommon/ControllerInterface/ControllerInterface.h"
|
|
#include "VideoCommon/OnScreenDisplay.h"
|
|
#include "VideoCommon/VideoBackendBase.h"
|
|
|
|
// This can mostly be removed when we move to VS2015
|
|
// to use the thread_local keyword
|
|
#ifdef _MSC_VER
|
|
#define ThreadLocalStorage __declspec(thread)
|
|
#elif defined __ANDROID__ || defined __APPLE__
|
|
// This will most likely have to stay, to support android
|
|
#include <pthread.h>
|
|
#else // Everything besides VS and Android
|
|
#define ThreadLocalStorage __thread
|
|
#endif
|
|
|
|
// TODO: ugly, remove
|
|
bool g_aspect_wide;
|
|
|
|
namespace Core
|
|
{
|
|
|
|
bool g_want_determinism;
|
|
|
|
// Declarations and definitions
|
|
static Common::Timer s_timer;
|
|
static volatile u32 s_drawn_frame = 0;
|
|
static u32 s_drawn_video = 0;
|
|
|
|
// Function forwarding
|
|
void Callback_WiimoteInterruptChannel(int _number, u16 _channelID, const void* _pData, u32 _Size);
|
|
|
|
// Function declarations
|
|
void EmuThread();
|
|
|
|
static bool s_is_stopping = false;
|
|
static bool s_hardware_initialized = false;
|
|
static bool s_is_started = false;
|
|
static void* s_window_handle = nullptr;
|
|
static std::string s_state_filename;
|
|
static std::thread s_emu_thread;
|
|
static StoppedCallbackFunc s_on_stopped_callback = nullptr;
|
|
|
|
static std::thread s_cpu_thread;
|
|
static bool s_request_refresh_info = false;
|
|
static int s_pause_and_lock_depth = 0;
|
|
static bool s_is_framelimiter_temp_disabled = false;
|
|
|
|
#ifdef ThreadLocalStorage
|
|
static ThreadLocalStorage bool tls_is_cpu_thread = false;
|
|
#else
|
|
static pthread_key_t s_tls_is_cpu_key;
|
|
static pthread_once_t s_cpu_key_is_init = PTHREAD_ONCE_INIT;
|
|
static void InitIsCPUKey()
|
|
{
|
|
pthread_key_create(&s_tls_is_cpu_key, nullptr);
|
|
}
|
|
#endif
|
|
|
|
bool GetIsFramelimiterTempDisabled()
|
|
{
|
|
return s_is_framelimiter_temp_disabled;
|
|
}
|
|
|
|
void SetIsFramelimiterTempDisabled(bool disable)
|
|
{
|
|
s_is_framelimiter_temp_disabled = disable;
|
|
}
|
|
|
|
std::string GetStateFileName() { return s_state_filename; }
|
|
void SetStateFileName(const std::string& val) { s_state_filename = val; }
|
|
|
|
void FrameUpdateOnCPUThread()
|
|
{
|
|
if (NetPlay::IsNetPlayRunning())
|
|
NetPlayClient::SendTimeBase();
|
|
}
|
|
|
|
// Display messages and return values
|
|
|
|
// Formatted stop message
|
|
std::string StopMessage(bool main_thread, const std::string& message)
|
|
{
|
|
return StringFromFormat("Stop [%s %i]\t%s\t%s",
|
|
main_thread ? "Main Thread" : "Video Thread", Common::CurrentThreadId(), MemUsage().c_str(), message.c_str());
|
|
}
|
|
|
|
void DisplayMessage(const std::string& message, int time_in_ms)
|
|
{
|
|
if (!IsRunning())
|
|
return;
|
|
|
|
// Actually displaying non-ASCII could cause things to go pear-shaped
|
|
for (const char& c : message)
|
|
{
|
|
if (!std::isprint(c))
|
|
return;
|
|
}
|
|
|
|
g_video_backend->Video_AddMessage(message, time_in_ms);
|
|
Host_UpdateTitle(message);
|
|
}
|
|
|
|
bool IsRunning()
|
|
{
|
|
return (GetState() != CORE_UNINITIALIZED) || s_hardware_initialized;
|
|
}
|
|
|
|
bool IsRunningAndStarted()
|
|
{
|
|
return s_is_started && !s_is_stopping;
|
|
}
|
|
|
|
bool IsRunningInCurrentThread()
|
|
{
|
|
return IsRunning() && IsCPUThread();
|
|
}
|
|
|
|
bool IsCPUThread()
|
|
{
|
|
#ifdef ThreadLocalStorage
|
|
return tls_is_cpu_thread;
|
|
#else
|
|
// Use pthread implementation for Android and Mac
|
|
// Make sure that s_tls_is_cpu_key is initialized
|
|
pthread_once(&s_cpu_key_is_init, InitIsCPUKey);
|
|
return pthread_getspecific(s_tls_is_cpu_key);
|
|
#endif
|
|
}
|
|
|
|
bool IsGPUThread()
|
|
{
|
|
const SConfig& _CoreParameter = SConfig::GetInstance();
|
|
if (_CoreParameter.bCPUThread)
|
|
{
|
|
return (s_emu_thread.joinable() && (s_emu_thread.get_id() == std::this_thread::get_id()));
|
|
}
|
|
else
|
|
{
|
|
return IsCPUThread();
|
|
}
|
|
}
|
|
|
|
// This is called from the GUI thread. See the booting call schedule in
|
|
// BootManager.cpp
|
|
bool Init()
|
|
{
|
|
const SConfig& _CoreParameter = SConfig::GetInstance();
|
|
|
|
if (s_emu_thread.joinable())
|
|
{
|
|
if (IsRunning())
|
|
{
|
|
PanicAlertT("Emu Thread already running");
|
|
return false;
|
|
}
|
|
|
|
// The Emu Thread was stopped, synchronize with it.
|
|
s_emu_thread.join();
|
|
}
|
|
|
|
Core::UpdateWantDeterminism(/*initial*/ true);
|
|
|
|
INFO_LOG(OSREPORT, "Starting core = %s mode",
|
|
_CoreParameter.bWii ? "Wii" : "GameCube");
|
|
INFO_LOG(OSREPORT, "CPU Thread separate = %s",
|
|
_CoreParameter.bCPUThread ? "Yes" : "No");
|
|
|
|
Host_UpdateMainFrame(); // Disable any menus or buttons at boot
|
|
|
|
g_aspect_wide = _CoreParameter.bWii;
|
|
if (g_aspect_wide)
|
|
{
|
|
IniFile gameIni = _CoreParameter.LoadGameIni();
|
|
gameIni.GetOrCreateSection("Wii")->Get("Widescreen", &g_aspect_wide,
|
|
!!SConfig::GetInstance().m_SYSCONF->GetData<u8>("IPL.AR"));
|
|
}
|
|
|
|
s_window_handle = Host_GetRenderHandle();
|
|
|
|
// Start the emu thread
|
|
s_emu_thread = std::thread(EmuThread);
|
|
|
|
return true;
|
|
}
|
|
|
|
// Called from GUI thread
|
|
void Stop() // - Hammertime!
|
|
{
|
|
if (GetState() == CORE_STOPPING)
|
|
return;
|
|
|
|
const SConfig& _CoreParameter = SConfig::GetInstance();
|
|
|
|
s_is_stopping = true;
|
|
|
|
g_video_backend->EmuStateChange(EMUSTATE_CHANGE_STOP);
|
|
|
|
INFO_LOG(CONSOLE, "Stop [Main Thread]\t\t---- Shutting down ----");
|
|
|
|
// Stop the CPU
|
|
INFO_LOG(CONSOLE, "%s", StopMessage(true, "Stop CPU").c_str());
|
|
PowerPC::Stop();
|
|
|
|
// Kick it if it's waiting (code stepping wait loop)
|
|
CCPU::StepOpcode();
|
|
|
|
if (_CoreParameter.bCPUThread)
|
|
{
|
|
// Video_EnterLoop() should now exit so that EmuThread()
|
|
// will continue concurrently with the rest of the commands
|
|
// in this function. We no longer rely on Postmessage.
|
|
INFO_LOG(CONSOLE, "%s", StopMessage(true, "Wait for Video Loop to exit ...").c_str());
|
|
|
|
g_video_backend->Video_ExitLoop();
|
|
}
|
|
if (s_emu_thread.joinable())
|
|
s_emu_thread.join();
|
|
}
|
|
|
|
static void DeclareAsCPUThread()
|
|
{
|
|
#ifdef ThreadLocalStorage
|
|
tls_is_cpu_thread = true;
|
|
#else
|
|
// Use pthread implementation for Android and Mac
|
|
// Make sure that s_tls_is_cpu_key is initialized
|
|
pthread_once(&s_cpu_key_is_init, InitIsCPUKey);
|
|
pthread_setspecific(s_tls_is_cpu_key, (void*)true);
|
|
#endif
|
|
}
|
|
|
|
static void UndeclareAsCPUThread()
|
|
{
|
|
#ifdef ThreadLocalStorage
|
|
tls_is_cpu_thread = false;
|
|
#else
|
|
// Use pthread implementation for Android and Mac
|
|
// Make sure that s_tls_is_cpu_key is initialized
|
|
pthread_once(&s_cpu_key_is_init, InitIsCPUKey);
|
|
pthread_setspecific(s_tls_is_cpu_key, (void*)false);
|
|
#endif
|
|
}
|
|
|
|
// Create the CPU thread, which is a CPU + Video thread in Single Core mode.
|
|
static void CpuThread()
|
|
{
|
|
DeclareAsCPUThread();
|
|
|
|
const SConfig& _CoreParameter = SConfig::GetInstance();
|
|
|
|
if (_CoreParameter.bCPUThread)
|
|
{
|
|
Common::SetCurrentThreadName("CPU thread");
|
|
}
|
|
else
|
|
{
|
|
Common::SetCurrentThreadName("CPU-GPU thread");
|
|
g_video_backend->Video_Prepare();
|
|
}
|
|
|
|
if (_CoreParameter.bFastmem)
|
|
EMM::InstallExceptionHandler(); // Let's run under memory watch
|
|
|
|
if (!s_state_filename.empty())
|
|
State::LoadAs(s_state_filename);
|
|
|
|
s_is_started = true;
|
|
|
|
|
|
#ifdef USE_GDBSTUB
|
|
#ifndef _WIN32
|
|
if (!_CoreParameter.gdb_socket.empty())
|
|
{
|
|
gdb_init_local(_CoreParameter.gdb_socket.data());
|
|
gdb_break();
|
|
}
|
|
else
|
|
#endif
|
|
if (_CoreParameter.iGDBPort > 0)
|
|
{
|
|
gdb_init(_CoreParameter.iGDBPort);
|
|
// break at next instruction (the first instruction)
|
|
gdb_break();
|
|
}
|
|
#endif
|
|
|
|
// Enter CPU run loop. When we leave it - we are done.
|
|
CCPU::Run();
|
|
|
|
s_is_started = false;
|
|
|
|
if (!_CoreParameter.bCPUThread)
|
|
g_video_backend->Video_Cleanup();
|
|
|
|
if (_CoreParameter.bFastmem)
|
|
EMM::UninstallExceptionHandler();
|
|
|
|
return;
|
|
}
|
|
|
|
static void FifoPlayerThread()
|
|
{
|
|
const SConfig& _CoreParameter = SConfig::GetInstance();
|
|
|
|
if (_CoreParameter.bCPUThread)
|
|
{
|
|
Common::SetCurrentThreadName("FIFO player thread");
|
|
}
|
|
else
|
|
{
|
|
g_video_backend->Video_Prepare();
|
|
Common::SetCurrentThreadName("FIFO-GPU thread");
|
|
}
|
|
|
|
s_is_started = true;
|
|
DeclareAsCPUThread();
|
|
|
|
// Enter CPU run loop. When we leave it - we are done.
|
|
if (FifoPlayer::GetInstance().Open(_CoreParameter.m_strFilename))
|
|
{
|
|
FifoPlayer::GetInstance().Play();
|
|
FifoPlayer::GetInstance().Close();
|
|
}
|
|
|
|
UndeclareAsCPUThread();
|
|
s_is_started = false;
|
|
|
|
if (!_CoreParameter.bCPUThread)
|
|
g_video_backend->Video_Cleanup();
|
|
|
|
return;
|
|
}
|
|
|
|
// Initialize and create emulation thread
|
|
// Call browser: Init():s_emu_thread().
|
|
// See the BootManager.cpp file description for a complete call schedule.
|
|
void EmuThread()
|
|
{
|
|
const SConfig& core_parameter = SConfig::GetInstance();
|
|
|
|
Common::SetCurrentThreadName("Emuthread - Starting");
|
|
|
|
if (SConfig::GetInstance().m_OCEnable)
|
|
DisplayMessage("WARNING: running at non-native CPU clock! Game may not be stable.", 8000);
|
|
DisplayMessage(cpu_info.brand_string, 8000);
|
|
DisplayMessage(cpu_info.Summarize(), 8000);
|
|
DisplayMessage(core_parameter.m_strFilename, 3000);
|
|
|
|
// For a time this acts as the CPU thread...
|
|
DeclareAsCPUThread();
|
|
|
|
Movie::Init();
|
|
|
|
HW::Init();
|
|
|
|
if (!g_video_backend->Initialize(s_window_handle))
|
|
{
|
|
PanicAlert("Failed to initialize video backend!");
|
|
Host_Message(WM_USER_STOP);
|
|
return;
|
|
}
|
|
|
|
OSD::AddMessage("Dolphin " + g_video_backend->GetName() + " Video Backend.", 5000);
|
|
|
|
if (cpu_info.HTT)
|
|
SConfig::GetInstance().bDSPThread = cpu_info.num_cores > 4;
|
|
else
|
|
SConfig::GetInstance().bDSPThread = cpu_info.num_cores > 2;
|
|
|
|
if (!DSP::GetDSPEmulator()->Initialize(core_parameter.bWii, core_parameter.bDSPThread))
|
|
{
|
|
HW::Shutdown();
|
|
g_video_backend->Shutdown();
|
|
PanicAlert("Failed to initialize DSP emulation!");
|
|
Host_Message(WM_USER_STOP);
|
|
return;
|
|
}
|
|
|
|
bool init_controllers = false;
|
|
if (!g_controller_interface.IsInit())
|
|
{
|
|
Pad::Initialize(s_window_handle);
|
|
Keyboard::Initialize(s_window_handle);
|
|
init_controllers = true;
|
|
}
|
|
else
|
|
{
|
|
// Update references in case controllers were refreshed
|
|
Pad::LoadConfig();
|
|
Keyboard::LoadConfig();
|
|
}
|
|
|
|
// Load and Init Wiimotes - only if we are booting in Wii mode
|
|
if (core_parameter.bWii)
|
|
{
|
|
if (init_controllers)
|
|
Wiimote::Initialize(s_window_handle, !s_state_filename.empty());
|
|
else
|
|
Wiimote::LoadConfig();
|
|
|
|
// Activate Wiimotes which don't have source set to "None"
|
|
for (unsigned int i = 0; i != MAX_BBMOTES; ++i)
|
|
if (g_wiimote_sources[i])
|
|
GetUsbPointer()->AccessWiiMote(i | 0x100)->Activate(true);
|
|
|
|
}
|
|
|
|
AudioCommon::InitSoundStream();
|
|
|
|
// The hardware is initialized.
|
|
s_hardware_initialized = true;
|
|
|
|
// Boot to pause or not
|
|
Core::SetState(core_parameter.bBootToPause ? Core::CORE_PAUSE : Core::CORE_RUN);
|
|
|
|
// Load GCM/DOL/ELF whatever ... we boot with the interpreter core
|
|
PowerPC::SetMode(PowerPC::MODE_INTERPRETER);
|
|
|
|
CBoot::BootUp();
|
|
|
|
// Thread is no longer acting as CPU Thread
|
|
UndeclareAsCPUThread();
|
|
|
|
// Setup our core, but can't use dynarec if we are compare server
|
|
if (core_parameter.iCPUCore != PowerPC::CORE_INTERPRETER
|
|
&& (!core_parameter.bRunCompareServer || core_parameter.bRunCompareClient))
|
|
{
|
|
PowerPC::SetMode(PowerPC::MODE_JIT);
|
|
}
|
|
else
|
|
{
|
|
PowerPC::SetMode(PowerPC::MODE_INTERPRETER);
|
|
}
|
|
|
|
// Update the window again because all stuff is initialized
|
|
Host_UpdateDisasmDialog();
|
|
Host_UpdateMainFrame();
|
|
|
|
// Determine the CPU thread function
|
|
void (*cpuThreadFunc)(void);
|
|
if (core_parameter.m_BootType == SConfig::BOOT_DFF)
|
|
cpuThreadFunc = FifoPlayerThread;
|
|
else
|
|
cpuThreadFunc = CpuThread;
|
|
|
|
// ENTER THE VIDEO THREAD LOOP
|
|
if (core_parameter.bCPUThread)
|
|
{
|
|
// This thread, after creating the EmuWindow, spawns a CPU
|
|
// thread, and then takes over and becomes the video thread
|
|
Common::SetCurrentThreadName("Video thread");
|
|
|
|
g_video_backend->Video_Prepare();
|
|
|
|
// Spawn the CPU thread
|
|
s_cpu_thread = std::thread(cpuThreadFunc);
|
|
|
|
// become the GPU thread
|
|
g_video_backend->Video_EnterLoop();
|
|
|
|
// We have now exited the Video Loop
|
|
INFO_LOG(CONSOLE, "%s", StopMessage(false, "Video Loop Ended").c_str());
|
|
}
|
|
else // SingleCore mode
|
|
{
|
|
// The spawned CPU Thread also does the graphics.
|
|
// The EmuThread is thus an idle thread, which sleeps while
|
|
// waiting for the program to terminate. Without this extra
|
|
// thread, the video backend window hangs in single core mode
|
|
// because no one is pumping messages.
|
|
Common::SetCurrentThreadName("Emuthread - Idle");
|
|
|
|
// Spawn the CPU+GPU thread
|
|
s_cpu_thread = std::thread(cpuThreadFunc);
|
|
|
|
while (PowerPC::GetState() != PowerPC::CPU_POWERDOWN)
|
|
{
|
|
g_video_backend->PeekMessages();
|
|
Common::SleepCurrentThread(20);
|
|
}
|
|
}
|
|
|
|
INFO_LOG(CONSOLE, "%s", StopMessage(true, "Stopping Emu thread ...").c_str());
|
|
|
|
// Wait for s_cpu_thread to exit
|
|
INFO_LOG(CONSOLE, "%s", StopMessage(true, "Stopping CPU-GPU thread ...").c_str());
|
|
|
|
#ifdef USE_GDBSTUB
|
|
INFO_LOG(CONSOLE, "%s", StopMessage(true, "Stopping GDB ...").c_str());
|
|
gdb_deinit();
|
|
INFO_LOG(CONSOLE, "%s", StopMessage(true, "GDB stopped.").c_str());
|
|
#endif
|
|
|
|
s_cpu_thread.join();
|
|
|
|
INFO_LOG(CONSOLE, "%s", StopMessage(true, "CPU thread stopped.").c_str());
|
|
|
|
if (core_parameter.bCPUThread)
|
|
g_video_backend->Video_Cleanup();
|
|
|
|
FileMon::Close();
|
|
|
|
// Stop audio thread - Actually this does nothing when using HLE
|
|
// emulation, but stops the DSP Interpreter when using LLE emulation.
|
|
DSP::GetDSPEmulator()->DSP_StopSoundStream();
|
|
|
|
// We must set up this flag before executing HW::Shutdown()
|
|
s_hardware_initialized = false;
|
|
INFO_LOG(CONSOLE, "%s", StopMessage(false, "Shutting down HW").c_str());
|
|
HW::Shutdown();
|
|
INFO_LOG(CONSOLE, "%s", StopMessage(false, "HW shutdown").c_str());
|
|
|
|
if (init_controllers)
|
|
{
|
|
Wiimote::Shutdown();
|
|
Keyboard::Shutdown();
|
|
Pad::Shutdown();
|
|
init_controllers = false;
|
|
}
|
|
|
|
g_video_backend->Shutdown();
|
|
AudioCommon::ShutdownSoundStream();
|
|
|
|
INFO_LOG(CONSOLE, "%s", StopMessage(true, "Main Emu thread stopped").c_str());
|
|
|
|
// Clear on screen messages that haven't expired
|
|
g_video_backend->Video_ClearMessages();
|
|
|
|
// Reload sysconf file in order to see changes committed during emulation
|
|
if (core_parameter.bWii)
|
|
SConfig::GetInstance().m_SYSCONF->Reload();
|
|
|
|
INFO_LOG(CONSOLE, "Stop [Video Thread]\t\t---- Shutdown complete ----");
|
|
Movie::Shutdown();
|
|
PatchEngine::Shutdown();
|
|
|
|
s_is_stopping = false;
|
|
|
|
if (s_on_stopped_callback)
|
|
s_on_stopped_callback();
|
|
}
|
|
|
|
// Set or get the running state
|
|
|
|
void SetState(EState _State)
|
|
{
|
|
switch (_State)
|
|
{
|
|
case CORE_PAUSE:
|
|
CCPU::EnableStepping(true); // Break
|
|
Wiimote::Pause();
|
|
break;
|
|
case CORE_RUN:
|
|
CCPU::EnableStepping(false);
|
|
Wiimote::Resume();
|
|
break;
|
|
default:
|
|
PanicAlert("Invalid state");
|
|
break;
|
|
}
|
|
}
|
|
|
|
EState GetState()
|
|
{
|
|
if (s_is_stopping)
|
|
return CORE_STOPPING;
|
|
|
|
if (s_hardware_initialized)
|
|
{
|
|
if (CCPU::IsStepping())
|
|
return CORE_PAUSE;
|
|
else
|
|
return CORE_RUN;
|
|
}
|
|
|
|
return CORE_UNINITIALIZED;
|
|
}
|
|
|
|
static std::string GenerateScreenshotName()
|
|
{
|
|
const std::string& gameId = SConfig::GetInstance().GetUniqueID();
|
|
std::string path = File::GetUserPath(D_SCREENSHOTS_IDX) + gameId + DIR_SEP_CHR;
|
|
|
|
if (!File::CreateFullPath(path))
|
|
{
|
|
// fallback to old-style screenshots, without folder.
|
|
path = File::GetUserPath(D_SCREENSHOTS_IDX);
|
|
}
|
|
|
|
//append gameId, path only contains the folder here.
|
|
path += gameId;
|
|
|
|
std::string name;
|
|
for (int i = 1; File::Exists(name = StringFromFormat("%s-%d.png", path.c_str(), i)); ++i)
|
|
{
|
|
// TODO?
|
|
}
|
|
|
|
return name;
|
|
}
|
|
|
|
void SaveScreenShot()
|
|
{
|
|
const bool bPaused = (GetState() == CORE_PAUSE);
|
|
|
|
SetState(CORE_PAUSE);
|
|
|
|
g_video_backend->Video_Screenshot(GenerateScreenshotName());
|
|
|
|
if (!bPaused)
|
|
SetState(CORE_RUN);
|
|
}
|
|
|
|
void RequestRefreshInfo()
|
|
{
|
|
s_request_refresh_info = true;
|
|
}
|
|
|
|
bool PauseAndLock(bool doLock, bool unpauseOnUnlock)
|
|
{
|
|
if (!IsRunning())
|
|
return true;
|
|
|
|
// let's support recursive locking to simplify things on the caller's side,
|
|
// and let's do it at this outer level in case the individual systems don't support it.
|
|
if (doLock ? s_pause_and_lock_depth++ : --s_pause_and_lock_depth)
|
|
return true;
|
|
|
|
// first pause or unpause the CPU
|
|
bool wasUnpaused = CCPU::PauseAndLock(doLock, unpauseOnUnlock);
|
|
ExpansionInterface::PauseAndLock(doLock, unpauseOnUnlock);
|
|
|
|
// audio has to come after CPU, because CPU thread can wait for audio thread (m_throttle).
|
|
DSP::GetDSPEmulator()->PauseAndLock(doLock, unpauseOnUnlock);
|
|
|
|
// video has to come after CPU, because CPU thread can wait for video thread (s_efbAccessRequested).
|
|
g_video_backend->PauseAndLock(doLock, unpauseOnUnlock);
|
|
return wasUnpaused;
|
|
}
|
|
|
|
// Apply Frame Limit and Display FPS info
|
|
// This should only be called from VI
|
|
void VideoThrottle()
|
|
{
|
|
// Update info per second
|
|
u32 ElapseTime = (u32)s_timer.GetTimeDifference();
|
|
if ((ElapseTime >= 1000 && s_drawn_video > 0) || s_request_refresh_info)
|
|
{
|
|
UpdateTitle();
|
|
|
|
// Reset counter
|
|
s_timer.Update();
|
|
Common::AtomicStore(s_drawn_frame, 0);
|
|
s_drawn_video = 0;
|
|
}
|
|
|
|
s_drawn_video++;
|
|
}
|
|
|
|
// Executed from GPU thread
|
|
// reports if a frame should be skipped or not
|
|
// depending on the framelimit set
|
|
bool ShouldSkipFrame(int skipped)
|
|
{
|
|
const u32 TargetFPS = (SConfig::GetInstance().m_Framelimit > 1)
|
|
? (SConfig::GetInstance().m_Framelimit - 1) * 5
|
|
: VideoInterface::TargetRefreshRate;
|
|
const u32 frames = Common::AtomicLoad(s_drawn_frame);
|
|
const bool fps_slow = !(s_timer.GetTimeDifference() < (frames + skipped) * 1000 / TargetFPS);
|
|
|
|
return fps_slow;
|
|
}
|
|
|
|
// --- Callbacks for backends / engine ---
|
|
|
|
// Should be called from GPU thread when a frame is drawn
|
|
void Callback_VideoCopiedToXFB(bool video_update)
|
|
{
|
|
if (video_update)
|
|
Common::AtomicIncrement(s_drawn_frame);
|
|
Movie::FrameUpdate();
|
|
}
|
|
|
|
void UpdateTitle()
|
|
{
|
|
u32 ElapseTime = (u32)s_timer.GetTimeDifference();
|
|
s_request_refresh_info = false;
|
|
SConfig& _CoreParameter = SConfig::GetInstance();
|
|
|
|
if (ElapseTime == 0)
|
|
ElapseTime = 1;
|
|
|
|
float FPS = (float) (Common::AtomicLoad(s_drawn_frame) * 1000.0 / ElapseTime);
|
|
float VPS = (float) (s_drawn_video * 1000.0 / ElapseTime);
|
|
float Speed = (float) (s_drawn_video * (100 * 1000.0) / (VideoInterface::TargetRefreshRate * ElapseTime));
|
|
|
|
// Settings are shown the same for both extended and summary info
|
|
std::string SSettings = StringFromFormat("%s %s | %s | %s", cpu_core_base->GetName(), _CoreParameter.bCPUThread ? "DC" : "SC",
|
|
g_video_backend->GetDisplayName().c_str(), _CoreParameter.bDSPHLE ? "HLE" : "LLE");
|
|
|
|
std::string SFPS;
|
|
|
|
if (Movie::IsPlayingInput())
|
|
SFPS = StringFromFormat("VI: %u/%u - Input: %u/%u - FPS: %.0f - VPS: %.0f - %.0f%%", (u32)Movie::g_currentFrame, (u32)Movie::g_totalFrames, (u32)Movie::g_currentInputCount, (u32)Movie::g_totalInputCount, FPS, VPS, Speed);
|
|
else if (Movie::IsRecordingInput())
|
|
SFPS = StringFromFormat("VI: %u - Input: %u - FPS: %.0f - VPS: %.0f - %.0f%%", (u32)Movie::g_currentFrame, (u32)Movie::g_currentInputCount, FPS, VPS, Speed);
|
|
else
|
|
{
|
|
SFPS = StringFromFormat("FPS: %.0f - VPS: %.0f - %.0f%%", FPS, VPS, Speed);
|
|
if (SConfig::GetInstance().m_InterfaceExtendedFPSInfo)
|
|
{
|
|
// Use extended or summary information. The summary information does not print the ticks data,
|
|
// that's more of a debugging interest, it can always be optional of course if someone is interested.
|
|
static u64 ticks = 0;
|
|
static u64 idleTicks = 0;
|
|
u64 newTicks = CoreTiming::GetTicks();
|
|
u64 newIdleTicks = CoreTiming::GetIdleTicks();
|
|
|
|
u64 diff = (newTicks - ticks) / 1000000;
|
|
u64 idleDiff = (newIdleTicks - idleTicks) / 1000000;
|
|
|
|
ticks = newTicks;
|
|
idleTicks = newIdleTicks;
|
|
|
|
float TicksPercentage = (float)diff / (float)(SystemTimers::GetTicksPerSecond() / 1000000) * 100;
|
|
|
|
SFPS += StringFromFormat(" | CPU: %s%i MHz [Real: %i + IdleSkip: %i] / %i MHz (%s%3.0f%%)",
|
|
_CoreParameter.bSkipIdle ? "~" : "",
|
|
(int)(diff),
|
|
(int)(diff - idleDiff),
|
|
(int)(idleDiff),
|
|
SystemTimers::GetTicksPerSecond() / 1000000,
|
|
_CoreParameter.bSkipIdle ? "~" : "",
|
|
TicksPercentage);
|
|
}
|
|
}
|
|
// This is our final "frame counter" string
|
|
std::string SMessage = StringFromFormat("%s | %s", SSettings.c_str(), SFPS.c_str());
|
|
|
|
// Update the audio timestretcher with the current speed
|
|
if (g_sound_stream)
|
|
{
|
|
CMixer* pMixer = g_sound_stream->GetMixer();
|
|
pMixer->UpdateSpeed((float)Speed / 100);
|
|
}
|
|
|
|
Host_UpdateTitle(SMessage);
|
|
}
|
|
|
|
void SetOnStoppedCallback(StoppedCallbackFunc callback)
|
|
{
|
|
s_on_stopped_callback = callback;
|
|
}
|
|
|
|
void UpdateWantDeterminism(bool initial)
|
|
{
|
|
// For now, this value is not itself configurable. Instead, individual
|
|
// settings that depend on it, such as GPU determinism mode. should have
|
|
// override options for testing,
|
|
bool new_want_determinism =
|
|
Movie::IsPlayingInput() ||
|
|
Movie::IsRecordingInput() ||
|
|
NetPlay::IsNetPlayRunning();
|
|
if (new_want_determinism != g_want_determinism || initial)
|
|
{
|
|
WARN_LOG(COMMON, "Want determinism <- %s", new_want_determinism ? "true" : "false");
|
|
|
|
bool was_unpaused = Core::PauseAndLock(true);
|
|
|
|
g_want_determinism = new_want_determinism;
|
|
WiiSockMan::GetInstance().UpdateWantDeterminism(new_want_determinism);
|
|
g_video_backend->UpdateWantDeterminism(new_want_determinism);
|
|
// We need to clear the cache because some parts of the JIT depend on want_determinism, e.g. use of FMA.
|
|
JitInterface::ClearCache();
|
|
Common::InitializeWiiRoot(g_want_determinism);
|
|
|
|
Core::PauseAndLock(false, was_unpaused);
|
|
}
|
|
}
|
|
|
|
} // Core
|