705 lines
18 KiB
C++
705 lines
18 KiB
C++
// Copyright 2013 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/Common.h"
|
|
#include "Common/CommonPaths.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/NetPlayProto.h"
|
|
#include "Core/PatchEngine.h"
|
|
#include "Core/State.h"
|
|
#include "Core/VolumeHandler.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/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/PowerPC/PowerPC.h"
|
|
|
|
#ifdef USE_GDBSTUB
|
|
#include "Core/PowerPC/GDBStub.h"
|
|
#endif
|
|
|
|
#include "DiscIO/FileMonitor.h"
|
|
|
|
#include "VideoCommon/OnScreenDisplay.h"
|
|
#include "VideoCommon/VideoBackendBase.h"
|
|
|
|
// TODO: ugly, remove
|
|
bool g_aspect_wide;
|
|
|
|
namespace Core
|
|
{
|
|
|
|
// Declarations and definitions
|
|
static Common::Timer Timer;
|
|
static volatile u32 DrawnFrame = 0;
|
|
static u32 DrawnVideo = 0;
|
|
|
|
// Function forwarding
|
|
void Callback_WiimoteInterruptChannel(int _number, u16 _channelID, const void* _pData, u32 _Size);
|
|
|
|
// Function declarations
|
|
void EmuThread();
|
|
|
|
static bool g_bStopping = false;
|
|
static bool g_bHwInit = false;
|
|
static bool g_bStarted = false;
|
|
static void *g_pWindowHandle = nullptr;
|
|
static std::string g_stateFileName;
|
|
static std::thread g_EmuThread;
|
|
static StoppedCallbackFunc s_onStoppedCb = nullptr;
|
|
|
|
static std::thread g_cpu_thread;
|
|
static bool g_requestRefreshInfo = false;
|
|
static int g_pauseAndLockDepth = 0;
|
|
|
|
SCoreStartupParameter g_CoreStartupParameter;
|
|
static bool IsFramelimiterTempDisabled = false;
|
|
|
|
bool GetIsFramelimiterTempDisabled()
|
|
{
|
|
return IsFramelimiterTempDisabled;
|
|
}
|
|
|
|
void SetIsFramelimiterTempDisabled(bool disable)
|
|
{
|
|
IsFramelimiterTempDisabled = disable;
|
|
}
|
|
|
|
std::string GetStateFileName() { return g_stateFileName; }
|
|
void SetStateFileName(std::string val) { g_stateFileName = val; }
|
|
|
|
// Display messages and return values
|
|
|
|
// Formatted stop message
|
|
std::string StopMessage(bool bMainThread, std::string Message)
|
|
{
|
|
return StringFromFormat("Stop [%s %i]\t%s\t%s",
|
|
bMainThread ? "Main Thread" : "Video Thread", Common::CurrentThreadId(), MemUsage().c_str(), Message.c_str());
|
|
}
|
|
|
|
void DisplayMessage(const std::string& message, int time_in_ms)
|
|
{
|
|
// 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);
|
|
}
|
|
|
|
bool IsRunning()
|
|
{
|
|
return (GetState() != CORE_UNINITIALIZED) || g_bHwInit;
|
|
}
|
|
|
|
bool IsRunningAndStarted()
|
|
{
|
|
return g_bStarted && !g_bStopping;
|
|
}
|
|
|
|
bool IsRunningInCurrentThread()
|
|
{
|
|
return IsRunning() && IsCPUThread();
|
|
}
|
|
|
|
bool IsCPUThread()
|
|
{
|
|
return (g_cpu_thread.joinable() ? (g_cpu_thread.get_id() == std::this_thread::get_id()) : !g_bStarted);
|
|
}
|
|
|
|
bool IsGPUThread()
|
|
{
|
|
const SCoreStartupParameter& _CoreParameter =
|
|
SConfig::GetInstance().m_LocalCoreStartupParameter;
|
|
if (_CoreParameter.bCPUThread)
|
|
{
|
|
return (g_EmuThread.joinable() && (g_EmuThread.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 SCoreStartupParameter& _CoreParameter =
|
|
SConfig::GetInstance().m_LocalCoreStartupParameter;
|
|
|
|
if (g_EmuThread.joinable())
|
|
{
|
|
if (IsRunning())
|
|
{
|
|
PanicAlertT("Emu Thread already running");
|
|
return false;
|
|
}
|
|
|
|
// The Emu Thread was stopped, synchronize with it.
|
|
g_EmuThread.join();
|
|
}
|
|
|
|
g_CoreStartupParameter = _CoreParameter;
|
|
|
|
INFO_LOG(OSREPORT, "Starting core = %s mode",
|
|
g_CoreStartupParameter.bWii ? "Wii" : "GameCube");
|
|
INFO_LOG(OSREPORT, "CPU Thread separate = %s",
|
|
g_CoreStartupParameter.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"));
|
|
}
|
|
|
|
g_pWindowHandle = Host_GetRenderHandle();
|
|
|
|
// Start the emu thread
|
|
g_EmuThread = std::thread(EmuThread);
|
|
|
|
return true;
|
|
}
|
|
|
|
// Called from GUI thread
|
|
void Stop() // - Hammertime!
|
|
{
|
|
if (GetState() == CORE_STOPPING)
|
|
return;
|
|
|
|
const SCoreStartupParameter& _CoreParameter =
|
|
SConfig::GetInstance().m_LocalCoreStartupParameter;
|
|
|
|
g_bStopping = 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();
|
|
}
|
|
}
|
|
|
|
// Create the CPU thread, which is a CPU + Video thread in Single Core mode.
|
|
static void CpuThread()
|
|
{
|
|
const SCoreStartupParameter& _CoreParameter =
|
|
SConfig::GetInstance().m_LocalCoreStartupParameter;
|
|
|
|
if (_CoreParameter.bCPUThread)
|
|
{
|
|
Common::SetCurrentThreadName("CPU thread");
|
|
}
|
|
else
|
|
{
|
|
Common::SetCurrentThreadName("CPU-GPU thread");
|
|
g_video_backend->Video_Prepare();
|
|
}
|
|
|
|
#if _M_X86_64 || _M_ARM_32
|
|
if (_CoreParameter.bFastmem)
|
|
EMM::InstallExceptionHandler(); // Let's run under memory watch
|
|
#endif
|
|
|
|
if (!g_stateFileName.empty())
|
|
State::LoadAs(g_stateFileName);
|
|
|
|
g_bStarted = true;
|
|
|
|
|
|
#ifdef USE_GDBSTUB
|
|
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();
|
|
|
|
g_bStarted = false;
|
|
|
|
if (!_CoreParameter.bCPUThread)
|
|
g_video_backend->Video_Cleanup();
|
|
|
|
return;
|
|
}
|
|
|
|
static void FifoPlayerThread()
|
|
{
|
|
const SCoreStartupParameter& _CoreParameter = SConfig::GetInstance().m_LocalCoreStartupParameter;
|
|
|
|
if (_CoreParameter.bCPUThread)
|
|
{
|
|
Common::SetCurrentThreadName("FIFO player thread");
|
|
}
|
|
else
|
|
{
|
|
g_video_backend->Video_Prepare();
|
|
Common::SetCurrentThreadName("FIFO-GPU thread");
|
|
}
|
|
|
|
g_bStarted = true;
|
|
|
|
// Enter CPU run loop. When we leave it - we are done.
|
|
if (FifoPlayer::GetInstance().Open(_CoreParameter.m_strFilename))
|
|
{
|
|
FifoPlayer::GetInstance().Play();
|
|
FifoPlayer::GetInstance().Close();
|
|
}
|
|
|
|
g_bStarted = false;
|
|
|
|
if (!_CoreParameter.bCPUThread)
|
|
g_video_backend->Video_Cleanup();
|
|
|
|
return;
|
|
}
|
|
|
|
// Initalize and create emulation thread
|
|
// Call browser: Init():g_EmuThread().
|
|
// See the BootManager.cpp file description for a complete call schedule.
|
|
void EmuThread()
|
|
{
|
|
const SCoreStartupParameter& _CoreParameter =
|
|
SConfig::GetInstance().m_LocalCoreStartupParameter;
|
|
|
|
Common::SetCurrentThreadName("Emuthread - Starting");
|
|
|
|
DisplayMessage(cpu_info.brand_string, 8000);
|
|
DisplayMessage(cpu_info.Summarize(), 8000);
|
|
DisplayMessage(_CoreParameter.m_strFilename, 3000);
|
|
|
|
Movie::Init();
|
|
|
|
HW::Init();
|
|
|
|
if (!g_video_backend->Initialize(g_pWindowHandle))
|
|
{
|
|
PanicAlert("Failed to initialize video backend!");
|
|
Host_Message(WM_USER_STOP);
|
|
return;
|
|
}
|
|
|
|
OSD::AddMessage("Dolphin " + g_video_backend->GetName() + " Video Backend.", 5000);
|
|
|
|
if (!DSP::GetDSPEmulator()->Initialize(_CoreParameter.bWii, _CoreParameter.bDSPThread))
|
|
{
|
|
HW::Shutdown();
|
|
g_video_backend->Shutdown();
|
|
PanicAlert("Failed to initialize DSP emulator!");
|
|
Host_Message(WM_USER_STOP);
|
|
return;
|
|
}
|
|
|
|
Pad::Initialize(g_pWindowHandle);
|
|
// Load and Init Wiimotes - only if we are booting in wii mode
|
|
if (g_CoreStartupParameter.bWii)
|
|
{
|
|
Wiimote::Initialize(g_pWindowHandle, !g_stateFileName.empty());
|
|
|
|
// 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.
|
|
g_bHwInit = true;
|
|
|
|
// Boot to pause or not
|
|
Core::SetState(_CoreParameter.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();
|
|
|
|
// Setup our core, but can't use dynarec if we are compare server
|
|
if (_CoreParameter.iCPUCore && (!_CoreParameter.bRunCompareServer ||
|
|
_CoreParameter.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 (_CoreParameter.m_BootType == SCoreStartupParameter::BOOT_DFF)
|
|
cpuThreadFunc = FifoPlayerThread;
|
|
else
|
|
cpuThreadFunc = CpuThread;
|
|
|
|
// ENTER THE VIDEO THREAD LOOP
|
|
if (_CoreParameter.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
|
|
g_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 noone is pumping messages.
|
|
Common::SetCurrentThreadName("Emuthread - Idle");
|
|
|
|
// Spawn the CPU+GPU thread
|
|
g_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 g_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
|
|
|
|
g_cpu_thread.join();
|
|
|
|
INFO_LOG(CONSOLE, "%s", StopMessage(true, "CPU thread stopped.").c_str());
|
|
|
|
if (_CoreParameter.bCPUThread)
|
|
g_video_backend->Video_Cleanup();
|
|
|
|
VolumeHandler::EjectVolume();
|
|
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()
|
|
g_bHwInit = false;
|
|
INFO_LOG(CONSOLE, "%s", StopMessage(false, "Shutting down HW").c_str());
|
|
HW::Shutdown();
|
|
INFO_LOG(CONSOLE, "%s", StopMessage(false, "HW shutdown").c_str());
|
|
Pad::Shutdown();
|
|
Wiimote::Shutdown();
|
|
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 (_CoreParameter.bWii)
|
|
SConfig::GetInstance().m_SYSCONF->Reload();
|
|
|
|
INFO_LOG(CONSOLE, "Stop [Video Thread]\t\t---- Shutdown complete ----");
|
|
Movie::Shutdown();
|
|
PatchEngine::Shutdown();
|
|
|
|
g_bStopping = false;
|
|
|
|
if (s_onStoppedCb)
|
|
s_onStoppedCb();
|
|
}
|
|
|
|
// 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:
|
|
PanicAlertT("Invalid state");
|
|
break;
|
|
}
|
|
}
|
|
|
|
EState GetState()
|
|
{
|
|
if (g_bStopping)
|
|
return CORE_STOPPING;
|
|
|
|
if (g_bHwInit)
|
|
{
|
|
if (CCPU::IsStepping())
|
|
return CORE_PAUSE;
|
|
else
|
|
return CORE_RUN;
|
|
}
|
|
|
|
return CORE_UNINITIALIZED;
|
|
}
|
|
|
|
static std::string GenerateScreenshotName()
|
|
{
|
|
const std::string& gameId = SConfig::GetInstance().m_LocalCoreStartupParameter.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()
|
|
{
|
|
g_requestRefreshInfo = true;
|
|
}
|
|
|
|
bool PauseAndLock(bool doLock, bool unpauseOnUnlock)
|
|
{
|
|
// 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 ? g_pauseAndLockDepth++ : --g_pauseAndLockDepth)
|
|
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).
|
|
AudioCommon::PauseAndLock(doLock, unpauseOnUnlock);
|
|
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)Timer.GetTimeDifference();
|
|
if ((ElapseTime >= 1000 && DrawnVideo > 0) || g_requestRefreshInfo)
|
|
{
|
|
UpdateTitle();
|
|
|
|
// Reset counter
|
|
Timer.Update();
|
|
Common::AtomicStore(DrawnFrame, 0);
|
|
DrawnVideo = 0;
|
|
}
|
|
|
|
DrawnVideo++;
|
|
}
|
|
|
|
// 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(DrawnFrame);
|
|
const bool fps_slow = !(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(DrawnFrame);
|
|
Movie::FrameUpdate();
|
|
}
|
|
|
|
void UpdateTitle()
|
|
{
|
|
u32 ElapseTime = (u32)Timer.GetTimeDifference();
|
|
g_requestRefreshInfo = false;
|
|
SCoreStartupParameter& _CoreParameter = SConfig::GetInstance().m_LocalCoreStartupParameter;
|
|
|
|
if (ElapseTime == 0)
|
|
ElapseTime = 1;
|
|
|
|
float FPS = (float) (Common::AtomicLoad(DrawnFrame) * 1000.0 / ElapseTime);
|
|
float VPS = (float) (DrawnVideo * 1000.0 / ElapseTime);
|
|
float Speed = (float) (DrawnVideo * (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 (soundStream)
|
|
{
|
|
CMixer* pMixer = soundStream->GetMixer();
|
|
pMixer->UpdateSpeed((float)Speed / 100);
|
|
}
|
|
|
|
Host_UpdateTitle(SMessage);
|
|
}
|
|
|
|
void Shutdown()
|
|
{
|
|
if (g_EmuThread.joinable())
|
|
g_EmuThread.join();
|
|
}
|
|
|
|
void SetOnStoppedCallback(StoppedCallbackFunc callback)
|
|
{
|
|
s_onStoppedCb = callback;
|
|
}
|
|
|
|
} // Core
|