Play-/Source/PS2VM.cpp
2018-01-15 14:14:59 -05:00

825 lines
21 KiB
C++

#include <stdio.h>
#include <exception>
#include <boost/filesystem.hpp>
#include <memory>
#include <fenv.h>
#include "make_unique.h"
#include "string_format.h"
#include "PS2VM.h"
#include "PS2VM_Preferences.h"
#include "ee/PS2OS.h"
#include "Ps2Const.h"
#include "iop/Iop_SifManPs2.h"
#include "StdStream.h"
#include "StdStreamUtils.h"
#include "GZipStream.h"
#include "MemoryStateFile.h"
#include "zip/ZipArchiveWriter.h"
#include "zip/ZipArchiveReader.h"
#include "xml/Node.h"
#include "xml/Writer.h"
#include "xml/Parser.h"
#include "AppConfig.h"
#include "PathUtils.h"
#include "iop/IopBios.h"
#include "iop/DirectoryDevice.h"
#include "iop/OpticalMediaDevice.h"
#include "Log.h"
#include "ISO9660/BlockProvider.h"
#include "DiskUtils.h"
#define LOG_NAME ("ps2vm")
#define PREF_PS2_HOST_DIRECTORY_DEFAULT ("vfs/host")
#define PREF_PS2_MC0_DIRECTORY_DEFAULT ("vfs/mc0")
#define PREF_PS2_MC1_DIRECTORY_DEFAULT ("vfs/mc1")
#define FRAME_TICKS (PS2::EE_CLOCK_FREQ / 60)
#define ONSCREEN_TICKS (FRAME_TICKS * 9 / 10)
#define VBLANK_TICKS (FRAME_TICKS / 10)
#define SPU_UPDATE_TICKS (PS2::IOP_CLOCK_OVER_FREQ / 1000)
namespace filesystem = boost::filesystem;
CPS2VM::CPS2VM()
: m_nStatus(PAUSED)
, m_nEnd(false)
, m_pad(NULL)
, m_singleStepEe(false)
, m_singleStepIop(false)
, m_singleStepVu0(false)
, m_singleStepVu1(false)
, m_vblankTicks(0)
, m_inVblank(false)
, m_eeExecutionTicks(0)
, m_iopExecutionTicks(0)
, m_spuUpdateTicks(SPU_UPDATE_TICKS)
, m_eeProfilerZone(CProfiler::GetInstance().RegisterZone("EE"))
, m_iopProfilerZone(CProfiler::GetInstance().RegisterZone("IOP"))
, m_spuProfilerZone(CProfiler::GetInstance().RegisterZone("SPU"))
, m_gsSyncProfilerZone(CProfiler::GetInstance().RegisterZone("GSSYNC"))
, m_otherProfilerZone(CProfiler::GetInstance().RegisterZone("OTHER"))
{
static const char* basicDirectorySettings[] =
{
PREF_PS2_HOST_DIRECTORY, PREF_PS2_HOST_DIRECTORY_DEFAULT,
PREF_PS2_MC0_DIRECTORY, PREF_PS2_MC0_DIRECTORY_DEFAULT,
PREF_PS2_MC1_DIRECTORY, PREF_PS2_MC1_DIRECTORY_DEFAULT,
NULL, NULL
};
for(unsigned int i = 0; basicDirectorySettings[i] != NULL; i += 2)
{
const char* setting = basicDirectorySettings[i + 0];
const char* path = basicDirectorySettings[i + 1];
Framework::CConfig::PathType absolutePath = CAppConfig::GetBasePath() / path;
Framework::PathUtils::EnsurePathExists(absolutePath);
//TODO: We ought to add a function to write a "path" in the settings. Since it can be wchar_t or char.
CAppConfig::GetInstance().RegisterPreferenceString(setting, absolutePath.string().c_str());
}
CAppConfig::GetInstance().RegisterPreferencePath(PREF_PS2_CDROM0_PATH, "");
m_iop = std::make_unique<Iop::CSubSystem>(true);
m_iopOs = std::make_shared<CIopBios>(m_iop->m_cpu, m_iop->m_executor, m_iop->m_ram, PS2::IOP_RAM_SIZE, m_iop->m_scratchPad);
m_ee = std::make_unique<Ee::CSubSystem>(m_iop->m_ram, *m_iopOs);
m_ee->m_os->OnRequestLoadExecutable.connect(boost::bind(&CPS2VM::ReloadExecutable, this, _1, _2));
}
CPS2VM::~CPS2VM()
{
{
//Big hack to force deletion of the IopBios
m_iop->SetBios(Iop::BiosBasePtr());
m_iopOs.reset();
}
}
//////////////////////////////////////////////////
//Various Message Functions
//////////////////////////////////////////////////
void CPS2VM::CreateGSHandler(const CGSHandler::FactoryFunction& factoryFunction)
{
if(m_ee->m_gs != nullptr) return;
m_mailBox.SendCall([this, factoryFunction] () { CreateGsHandlerImpl(factoryFunction); }, true);
}
CGSHandler* CPS2VM::GetGSHandler()
{
return m_ee->m_gs;
}
void CPS2VM::DestroyGSHandler()
{
if(m_ee->m_gs == nullptr) return;
m_mailBox.SendCall([this] () { DestroyGsHandlerImpl(); }, true);
}
void CPS2VM::CreatePadHandler(const CPadHandler::FactoryFunction& factoryFunction)
{
if(m_pad != nullptr) return;
m_mailBox.SendCall([this, factoryFunction] () { CreatePadHandlerImpl(factoryFunction); }, true);
}
CPadHandler* CPS2VM::GetPadHandler()
{
return m_pad;
}
void CPS2VM::DestroyPadHandler()
{
if(m_pad == nullptr) return;
m_mailBox.SendCall([this] () { DestroyPadHandlerImpl(); }, true);
}
void CPS2VM::CreateSoundHandler(const CSoundHandler::FactoryFunction& factoryFunction)
{
if(m_soundHandler != nullptr) return;
m_mailBox.SendCall([this, factoryFunction] () { CreateSoundHandlerImpl(factoryFunction); }, true);
}
void CPS2VM::DestroySoundHandler()
{
if(m_soundHandler == nullptr) return;
m_mailBox.SendCall([this] () { DestroySoundHandlerImpl(); }, true);
}
CVirtualMachine::STATUS CPS2VM::GetStatus() const
{
return m_nStatus;
}
void CPS2VM::StepEe()
{
if(GetStatus() == RUNNING) return;
m_singleStepEe = true;
m_mailBox.SendCall(std::bind(&CPS2VM::ResumeImpl, this), true);
}
void CPS2VM::StepIop()
{
if(GetStatus() == RUNNING) return;
m_singleStepIop = true;
m_mailBox.SendCall(std::bind(&CPS2VM::ResumeImpl, this), true);
}
void CPS2VM::StepVu0()
{
if(GetStatus() == RUNNING) return;
m_singleStepVu0 = true;
m_mailBox.SendCall(std::bind(&CPS2VM::ResumeImpl, this), true);
}
void CPS2VM::StepVu1()
{
if(GetStatus() == RUNNING) return;
m_singleStepVu1 = true;
m_mailBox.SendCall(std::bind(&CPS2VM::ResumeImpl, this), true);
}
void CPS2VM::Resume()
{
if(m_nStatus == RUNNING) return;
m_mailBox.SendCall(std::bind(&CPS2VM::ResumeImpl, this), true);
OnRunningStateChange();
}
void CPS2VM::Pause()
{
if(m_nStatus == PAUSED) return;
m_mailBox.SendCall(std::bind(&CPS2VM::PauseImpl, this), true);
OnMachineStateChange();
OnRunningStateChange();
}
void CPS2VM::Reset()
{
assert(m_nStatus == PAUSED);
ResetVM();
}
void CPS2VM::DumpEEIntcHandlers()
{
// if(m_pOS == NULL) return;
if(m_nStatus != PAUSED) return;
m_ee->m_os->DumpIntcHandlers();
}
void CPS2VM::DumpEEDmacHandlers()
{
// if(m_pOS == NULL) return;
if(m_nStatus != PAUSED) return;
m_ee->m_os->DumpDmacHandlers();
}
void CPS2VM::Initialize()
{
CreateVM();
m_nEnd = false;
m_thread = std::thread([&] () { EmuThread(); });
}
void CPS2VM::Destroy()
{
m_mailBox.SendCall(std::bind(&CPS2VM::DestroyImpl, this));
m_thread.join();
DestroyVM();
}
boost::filesystem::path CPS2VM::GetStateDirectoryPath()
{
return CAppConfig::GetBasePath() / boost::filesystem::path("states/");
}
boost::filesystem::path CPS2VM::GenerateStatePath(unsigned int slot) const
{
auto stateFileName = string_format("%s.st%d.zip", m_ee->m_os->GetExecutableName(), slot);
return GetStateDirectoryPath() / boost::filesystem::path(stateFileName);
}
std::future<bool> CPS2VM::SaveState(const filesystem::path& statePath)
{
auto promise = std::make_shared<std::promise<bool>>();
auto future = promise->get_future();
m_mailBox.SendCall(
[this, promise, statePath] ()
{
auto result = SaveVMState(statePath);
promise->set_value(result);
}
);
return future;
}
std::future<bool> CPS2VM::LoadState(const filesystem::path& statePath)
{
auto promise = std::make_shared<std::promise<bool>>();
auto future = promise->get_future();
m_mailBox.SendCall(
[this, promise, statePath] ()
{
auto result = LoadVMState(statePath);
promise->set_value(result);
}
);
return future;
}
void CPS2VM::TriggerFrameDump(const FrameDumpCallback& frameDumpCallback)
{
m_mailBox.SendCall(
[=] ()
{
std::unique_lock<std::mutex> frameDumpCallbackMutexLock(m_frameDumpCallbackMutex);
if(m_frameDumpCallback) return;
m_frameDumpCallback = frameDumpCallback;
},
false
);
}
CPS2VM::CPU_UTILISATION_INFO CPS2VM::GetCpuUtilisationInfo() const
{
return m_cpuUtilisation;
}
#ifdef DEBUGGER_INCLUDED
#define TAGS_SECTION_TAGS ("tags")
#define TAGS_SECTION_EE_FUNCTIONS ("ee_functions")
#define TAGS_SECTION_EE_COMMENTS ("ee_comments")
#define TAGS_SECTION_VU1_FUNCTIONS ("vu1_functions")
#define TAGS_SECTION_VU1_COMMENTS ("vu1_comments")
#define TAGS_SECTION_IOP ("iop")
#define TAGS_SECTION_IOP_FUNCTIONS ("functions")
#define TAGS_SECTION_IOP_COMMENTS ("comments")
#define TAGS_PATH ("tags/")
std::string CPS2VM::MakeDebugTagsPackagePath(const char* packageName)
{
auto tagsPath = CAppConfig::GetBasePath() / boost::filesystem::path(TAGS_PATH);
Framework::PathUtils::EnsurePathExists(tagsPath);
auto tagsPackagePath = tagsPath / (std::string(packageName) + std::string(".tags.xml"));
return tagsPackagePath.string();
}
void CPS2VM::LoadDebugTags(const char* packageName)
{
try
{
std::string packagePath = MakeDebugTagsPackagePath(packageName);
Framework::CStdStream stream(packagePath.c_str(), "rb");
boost::scoped_ptr<Framework::Xml::CNode> document(Framework::Xml::CParser::ParseDocument(stream));
Framework::Xml::CNode* tagsNode = document->Select(TAGS_SECTION_TAGS);
if(!tagsNode) return;
m_ee->m_EE.m_Functions.Unserialize(tagsNode, TAGS_SECTION_EE_FUNCTIONS);
m_ee->m_EE.m_Comments.Unserialize(tagsNode, TAGS_SECTION_EE_COMMENTS);
m_ee->m_VU1.m_Functions.Unserialize(tagsNode, TAGS_SECTION_VU1_FUNCTIONS);
m_ee->m_VU1.m_Comments.Unserialize(tagsNode, TAGS_SECTION_VU1_COMMENTS);
{
Framework::Xml::CNode* sectionNode = tagsNode->Select(TAGS_SECTION_IOP);
if(sectionNode)
{
m_iop->m_cpu.m_Functions.Unserialize(sectionNode, TAGS_SECTION_IOP_FUNCTIONS);
m_iop->m_cpu.m_Comments.Unserialize(sectionNode, TAGS_SECTION_IOP_COMMENTS);
m_iopOs->LoadDebugTags(sectionNode);
}
}
}
catch(...)
{
}
}
void CPS2VM::SaveDebugTags(const char* packageName)
{
try
{
std::string packagePath = MakeDebugTagsPackagePath(packageName);
Framework::CStdStream stream(packagePath.c_str(), "wb");
boost::scoped_ptr<Framework::Xml::CNode> document(new Framework::Xml::CNode(TAGS_SECTION_TAGS, true));
m_ee->m_EE.m_Functions.Serialize(document.get(), TAGS_SECTION_EE_FUNCTIONS);
m_ee->m_EE.m_Comments.Serialize(document.get(), TAGS_SECTION_EE_COMMENTS);
m_ee->m_VU1.m_Functions.Serialize(document.get(), TAGS_SECTION_VU1_FUNCTIONS);
m_ee->m_VU1.m_Comments.Serialize(document.get(), TAGS_SECTION_VU1_COMMENTS);
{
Framework::Xml::CNode* iopNode = new Framework::Xml::CNode(TAGS_SECTION_IOP, true);
m_iop->m_cpu.m_Functions.Serialize(iopNode, TAGS_SECTION_IOP_FUNCTIONS);
m_iop->m_cpu.m_Comments.Serialize(iopNode, TAGS_SECTION_IOP_COMMENTS);
m_iopOs->SaveDebugTags(iopNode);
document->InsertNode(iopNode);
}
Framework::Xml::CWriter::WriteDocument(stream, document.get());
}
catch(...)
{
}
}
#endif
//////////////////////////////////////////////////
//Non extern callable methods
//////////////////////////////////////////////////
void CPS2VM::CreateVM()
{
ResetVM();
}
void CPS2VM::ResetVM()
{
m_ee->Reset();
m_iop->Reset();
m_iop->SetBios(m_iopOs);
//LoadBIOS();
if(m_ee->m_gs != NULL)
{
m_ee->m_gs->Reset();
}
m_iopOs->Reset(std::make_shared<Iop::CSifManPs2>(m_ee->m_sif, m_ee->m_ram, m_iop->m_ram));
CDROM0_SyncPath();
m_iopOs->GetIoman()->RegisterDevice("host", Iop::CIoman::DevicePtr(new Iop::Ioman::CDirectoryDevice(PREF_PS2_HOST_DIRECTORY)));
m_iopOs->GetIoman()->RegisterDevice("mc0", Iop::CIoman::DevicePtr(new Iop::Ioman::CDirectoryDevice(PREF_PS2_MC0_DIRECTORY)));
m_iopOs->GetIoman()->RegisterDevice("mc1", Iop::CIoman::DevicePtr(new Iop::Ioman::CDirectoryDevice(PREF_PS2_MC1_DIRECTORY)));
m_iopOs->GetIoman()->RegisterDevice("cdrom", Iop::CIoman::DevicePtr(new Iop::Ioman::COpticalMediaDevice(m_cdrom0)));
m_iopOs->GetIoman()->RegisterDevice("cdrom0", Iop::CIoman::DevicePtr(new Iop::Ioman::COpticalMediaDevice(m_cdrom0)));
m_iopOs->GetLoadcore()->SetLoadExecutableHandler(std::bind(&CPS2OS::LoadExecutable, m_ee->m_os, std::placeholders::_1, std::placeholders::_2));
m_vblankTicks = ONSCREEN_TICKS;
m_inVblank = false;
m_eeExecutionTicks = 0;
m_iopExecutionTicks = 0;
m_spuUpdateTicks = SPU_UPDATE_TICKS;
m_currentSpuBlock = 0;
RegisterModulesInPadHandler();
}
void CPS2VM::DestroyVM()
{
CDROM0_Reset();
}
bool CPS2VM::SaveVMState(const filesystem::path& statePath)
{
if(m_ee->m_gs == NULL)
{
printf("PS2VM: GS Handler was not instancied. Cannot save state.\r\n");
return false;
}
try
{
auto stateStream = Framework::CreateOutputStdStream(statePath.native());
Framework::CZipArchiveWriter archive;
m_ee->SaveState(archive);
m_iop->SaveState(archive);
m_ee->m_gs->SaveState(archive);
archive.Write(stateStream);
}
catch(...)
{
return false;
}
return true;
}
bool CPS2VM::LoadVMState(const filesystem::path& statePath)
{
if(m_ee->m_gs == NULL)
{
printf("PS2VM: GS Handler was not instancied. Cannot load state.\r\n");
return false;
}
try
{
auto stateStream = Framework::CreateInputStdStream(statePath.native());
Framework::CZipArchiveReader archive(stateStream);
try
{
m_ee->LoadState(archive);
m_iop->LoadState(archive);
m_ee->m_gs->LoadState(archive);
}
catch(...)
{
//Any error that occurs in the previous block is critical
PauseImpl();
throw;
}
}
catch(...)
{
return false;
}
OnMachineStateChange();
return true;
}
void CPS2VM::PauseImpl()
{
m_nStatus = PAUSED;
}
void CPS2VM::ResumeImpl()
{
#ifdef DEBUGGER_INCLUDED
m_ee->m_executor.DisableBreakpointsOnce();
m_iop->m_executor.DisableBreakpointsOnce();
m_ee->m_vpu1->DisableBreakpointsOnce();
#endif
m_nStatus = RUNNING;
}
void CPS2VM::DestroyImpl()
{
DestroyGsHandlerImpl();
DestroyPadHandlerImpl();
DestroySoundHandlerImpl();
m_nEnd = true;
}
void CPS2VM::CreateGsHandlerImpl(const CGSHandler::FactoryFunction& factoryFunction)
{
m_ee->m_gs = factoryFunction();
m_ee->m_gs->Initialize();
m_ee->m_gs->OnNewFrame.connect(boost::bind(&CPS2VM::OnGsNewFrame, this));
}
void CPS2VM::DestroyGsHandlerImpl()
{
if(m_ee->m_gs == nullptr) return;
m_ee->m_gs->Release();
delete m_ee->m_gs;
m_ee->m_gs = nullptr;
}
void CPS2VM::CreatePadHandlerImpl(const CPadHandler::FactoryFunction& factoryFunction)
{
m_pad = factoryFunction();
RegisterModulesInPadHandler();
}
void CPS2VM::DestroyPadHandlerImpl()
{
if(m_pad == nullptr) return;
delete m_pad;
m_pad = nullptr;
}
void CPS2VM::CreateSoundHandlerImpl(const CSoundHandler::FactoryFunction& factoryFunction)
{
m_soundHandler = factoryFunction();
}
void CPS2VM::DestroySoundHandlerImpl()
{
if(m_soundHandler == nullptr) return;
delete m_soundHandler;
m_soundHandler = nullptr;
}
void CPS2VM::OnGsNewFrame()
{
#ifdef DEBUGGER_INCLUDED
std::unique_lock<std::mutex> dumpFrameCallbackMutexLock(m_frameDumpCallbackMutex);
if(m_dumpingFrame && !m_frameDump.GetPackets().empty())
{
m_ee->m_gs->SetFrameDump(nullptr);
m_frameDumpCallback(m_frameDump);
m_dumpingFrame = false;
m_frameDumpCallback = FrameDumpCallback();
}
else if(m_frameDumpCallback)
{
m_frameDump.Reset();
memcpy(m_frameDump.GetInitialGsRam(), m_ee->m_gs->GetRam(), CGSHandler::RAMSIZE);
memcpy(m_frameDump.GetInitialGsRegisters(), m_ee->m_gs->GetRegisters(), CGSHandler::REGISTER_MAX * sizeof(uint64));
m_frameDump.SetInitialSMODE2(m_ee->m_gs->GetSMODE2());
m_ee->m_gs->SetFrameDump(&m_frameDump);
m_dumpingFrame = true;
}
#endif
}
void CPS2VM::UpdateEe()
{
#ifdef PROFILE
CProfilerZone profilerZone(m_eeProfilerZone);
#endif
while(m_eeExecutionTicks > 0)
{
int executed = m_ee->ExecuteCpu(m_singleStepEe ? 1 : m_eeExecutionTicks);
if(m_ee->IsCpuIdle())
{
#ifdef PROFILE
m_cpuUtilisation.eeIdleTicks += (m_eeExecutionTicks - executed);
#endif
executed = m_eeExecutionTicks;
}
#ifdef PROFILE
m_cpuUtilisation.eeTotalTicks += executed;
#endif
m_ee->m_vpu0->Execute(m_singleStepVu0 ? 1 : executed);
m_ee->m_vpu1->Execute(m_singleStepVu1 ? 1 : executed);
m_eeExecutionTicks -= executed;
m_ee->CountTicks(executed);
m_vblankTicks -= executed;
#ifdef DEBUGGER_INCLUDED
if(m_singleStepEe) break;
if(m_ee->m_executor.MustBreak()) break;
#endif
}
}
void CPS2VM::UpdateIop()
{
#ifdef PROFILE
CProfilerZone profilerZone(m_iopProfilerZone);
#endif
while(m_iopExecutionTicks > 0)
{
int executed = m_iop->ExecuteCpu(m_singleStepIop ? 1 : m_iopExecutionTicks);
if(m_iop->IsCpuIdle())
{
#ifdef PROFILE
m_cpuUtilisation.iopIdleTicks += (m_iopExecutionTicks - executed);
#endif
executed = m_iopExecutionTicks;
}
#ifdef PROFILE
m_cpuUtilisation.iopTotalTicks += executed;
#endif
m_iopExecutionTicks -= executed;
m_spuUpdateTicks -= executed;
m_iop->CountTicks(executed);
#ifdef DEBUGGER_INCLUDED
if(m_singleStepIop) break;
if(m_iop->m_executor.MustBreak()) break;
#endif
}
}
void CPS2VM::UpdateSpu()
{
#ifdef PROFILE
CProfilerZone profilerZone(m_spuProfilerZone);
#endif
unsigned int blockOffset = (BLOCK_SIZE * m_currentSpuBlock);
int16* samplesSpu0 = m_samples + blockOffset;
m_iop->m_spuCore0.Render(samplesSpu0, BLOCK_SIZE, 44100);
if(m_iop->m_spuCore1.IsEnabled())
{
int16 samplesSpu1[BLOCK_SIZE];
m_iop->m_spuCore1.Render(samplesSpu1, BLOCK_SIZE, 44100);
for(unsigned int i = 0; i < BLOCK_SIZE; i++)
{
int32 resultSample = static_cast<int32>(samplesSpu0[i]) + static_cast<int32>(samplesSpu1[i]);
resultSample = std::max<int32>(resultSample, SHRT_MIN);
resultSample = std::min<int32>(resultSample, SHRT_MAX);
samplesSpu0[i] = static_cast<int16>(resultSample);
}
}
m_currentSpuBlock++;
if(m_currentSpuBlock == BLOCK_COUNT)
{
if(m_soundHandler)
{
if(m_soundHandler->HasFreeBuffers())
{
m_soundHandler->RecycleBuffers();
}
m_soundHandler->Write(m_samples, BLOCK_SIZE * BLOCK_COUNT, 44100);
}
m_currentSpuBlock = 0;
}
}
void CPS2VM::CDROM0_SyncPath()
{
//TODO: Check if there's an m_cdrom0 already
//TODO: Check if files are linked to this m_cdrom0 too and do something with them
CDROM0_Reset();
auto path = CAppConfig::GetInstance().GetPreferencePath(PREF_PS2_CDROM0_PATH);
if(!path.empty())
{
try
{
m_cdrom0 = DiskUtils::CreateOpticalMediaFromPath(path);
SetIopOpticalMedia(m_cdrom0.get());
}
catch(const std::exception& Exception)
{
printf("PS2VM: Error mounting cdrom0 device: %s\r\n", Exception.what());
}
}
}
void CPS2VM::CDROM0_Reset()
{
SetIopOpticalMedia(nullptr);
m_cdrom0.reset();
}
void CPS2VM::SetIopOpticalMedia(COpticalMedia* opticalMedia)
{
m_iopOs->GetCdvdfsv()->SetOpticalMedia(opticalMedia);
m_iopOs->GetCdvdman()->SetOpticalMedia(opticalMedia);
}
void CPS2VM::RegisterModulesInPadHandler()
{
if(m_pad == nullptr) return;
m_pad->RemoveAllListeners();
m_pad->InsertListener(m_iopOs->GetPadman());
m_pad->InsertListener(&m_iop->m_sio2);
}
void CPS2VM::ReloadExecutable(const char* executablePath, const CPS2OS::ArgumentList& arguments)
{
ResetVM();
m_ee->m_os->BootFromVirtualPath(executablePath, arguments);
}
void CPS2VM::EmuThread()
{
fesetround(FE_TOWARDZERO);
CProfiler::GetInstance().SetWorkThread();
#ifdef PROFILE
CProfilerZone profilerZone(m_otherProfilerZone);
#endif
m_ee->m_executor.AddExceptionHandler();
while(1)
{
while(m_mailBox.IsPending())
{
m_mailBox.ReceiveCall();
}
if(m_nEnd) break;
if(m_nStatus == PAUSED)
{
std::this_thread::sleep_for(std::chrono::milliseconds(100));
}
if(m_nStatus == RUNNING)
{
if(m_spuUpdateTicks <= 0)
{
UpdateSpu();
m_spuUpdateTicks += SPU_UPDATE_TICKS;
}
//EE execution
{
//Check vblank stuff
if(m_vblankTicks <= 0)
{
m_inVblank = !m_inVblank;
if(m_inVblank)
{
m_vblankTicks += VBLANK_TICKS;
m_ee->NotifyVBlankStart();
m_iop->NotifyVBlankStart();
if(m_ee->m_gs != NULL)
{
#ifdef PROFILE
CProfilerZone profilerZone(m_gsSyncProfilerZone);
#endif
m_ee->m_gs->SetVBlank();
}
if(m_pad != NULL)
{
m_pad->Update(m_ee->m_ram);
}
#ifdef PROFILE
{
CProfiler::GetInstance().CountCurrentZone();
auto stats = CProfiler::GetInstance().GetStats();
ProfileFrameDone(stats);
CProfiler::GetInstance().Reset();
}
m_cpuUtilisation = CPU_UTILISATION_INFO();
#endif
}
else
{
m_vblankTicks += ONSCREEN_TICKS;
m_ee->NotifyVBlankEnd();
m_iop->NotifyVBlankEnd();
if(m_ee->m_gs != NULL)
{
m_ee->m_gs->ResetVBlank();
}
}
}
//EE CPU is 8 times faster than the IOP CPU
static const int tickStep = 4800;
m_eeExecutionTicks += tickStep;
m_iopExecutionTicks += tickStep / 8;
UpdateEe();
UpdateIop();
}
#ifdef DEBUGGER_INCLUDED
if(
m_ee->m_executor.MustBreak() ||
m_iop->m_executor.MustBreak() ||
m_ee->m_vpu1->MustBreak() ||
m_singleStepEe || m_singleStepIop || m_singleStepVu0 || m_singleStepVu1)
{
m_nStatus = PAUSED;
m_singleStepEe = false;
m_singleStepIop = false;
m_singleStepVu0 = false;
m_singleStepVu1 = false;
OnRunningStateChange();
OnMachineStateChange();
}
#endif
}
}
m_ee->m_executor.RemoveExceptionHandler();
}