Mesen-SX/Core/Console.cpp

242 lines
4.9 KiB
C++
Raw Normal View History

#include "stdafx.h"
#include "Console.h"
#include "Cpu.h"
#include "Ppu.h"
#include "Spc.h"
#include "InternalRegisters.h"
2019-02-17 19:54:29 -05:00
#include "ControlManager.h"
#include "MemoryManager.h"
#include "DmaController.h"
#include "Debugger.h"
2019-02-15 21:33:13 -05:00
#include "NotificationManager.h"
#include "SoundMixer.h"
#include "VideoDecoder.h"
#include "VideoRenderer.h"
#include "DebugHud.h"
2019-02-21 17:18:56 -05:00
#include "FrameLimiter.h"
#include "MessageManager.h"
#include "KeyManager.h"
#include "../Utilities/Timer.h"
#include "../Utilities/VirtualFile.h"
2019-02-21 17:18:56 -05:00
#include "../Utilities/PlatformUtilities.h"
2019-02-24 23:53:14 -05:00
#include "../Utilities/FolderUtilities.h"
void Console::Initialize()
{
2019-02-15 21:33:13 -05:00
_notificationManager.reset(new NotificationManager());
_videoDecoder.reset(new VideoDecoder(shared_from_this()));
_videoRenderer.reset(new VideoRenderer(shared_from_this()));
_soundMixer.reset(new SoundMixer());
_debugHud.reset(new DebugHud());
_videoDecoder->StartThread();
_videoRenderer->StartThread();
}
void Console::Release()
{
Stop();
_videoDecoder->StopThread();
_videoRenderer->StopThread();
_videoDecoder.reset();
_videoRenderer.reset();
_debugHud.reset();
_notificationManager.reset();
}
void Console::Run()
{
if(!_cpu) {
return;
}
_stopFlag = false;
2019-02-21 17:18:56 -05:00
uint32_t previousFrameCount = 0;
FrameLimiter frameLimiter(16.63926405550947);
PlatformUtilities::EnableHighResolutionTimer();
uint32_t keyCode = KeyManager::GetKeyCode("Tab");
auto lock = _runLock.AcquireSafe();
while(!_stopFlag) {
_cpu->Exec();
2019-02-21 17:18:56 -05:00
if(previousFrameCount != _ppu->GetFrameCount()) {
if(!KeyManager::IsKeyPressed(keyCode)) {
frameLimiter.ProcessFrame();
frameLimiter.WaitForNextFrame();
}
2019-02-21 17:18:56 -05:00
previousFrameCount = _ppu->GetFrameCount();
}
}
2019-02-21 17:18:56 -05:00
PlatformUtilities::RestoreTimerResolution();
}
void Console::Stop()
{
_stopFlag = true;
shared_ptr<Debugger> debugger = _debugger;
if(debugger) {
debugger->Run();
}
2019-02-24 19:57:34 -05:00
_debugger.reset();
_runLock.WaitForRelease();
_cpu.reset();
_ppu.reset();
_spc.reset();
2019-02-16 01:16:57 -05:00
_cart.reset();
_internalRegisters.reset();
2019-02-17 19:54:29 -05:00
_controlManager.reset();
_memoryManager.reset();
_dmaController.reset();
_soundMixer->StopAudio();
}
void Console::LoadRom(VirtualFile romFile, VirtualFile patchFile)
{
Stop();
shared_ptr<BaseCartridge> cart = BaseCartridge::CreateCartridge(romFile, patchFile);
if(cart) {
MessageManager::ClearLog();
vector<uint8_t> spcRomData;
2019-02-24 23:53:14 -05:00
VirtualFile spcBios(FolderUtilities::CombinePath(FolderUtilities::GetHomeFolder(), "spc700.rom"));
if(spcBios.IsValid()) {
spcBios.ReadFile(spcRomData);
} else {
MessageManager::Log("[SPC] spc700.rom not found, cannot launch game.");
return;
}
2019-02-17 19:54:29 -05:00
_internalRegisters.reset(new InternalRegisters(shared_from_this()));
2019-02-13 18:44:39 -05:00
_ppu.reset(new Ppu(shared_from_this()));
_spc.reset(new Spc(shared_from_this(), spcRomData));
2019-02-15 21:33:13 -05:00
_cart = cart;
2019-02-17 19:54:29 -05:00
_controlManager.reset(new ControlManager(shared_from_this()));
2019-02-15 00:08:50 -05:00
_memoryManager.reset(new MemoryManager());
2019-02-24 19:57:34 -05:00
_dmaController.reset(new DmaController(_memoryManager.get()));
2019-02-16 01:16:57 -05:00
_memoryManager->Initialize(shared_from_this());
2019-02-15 00:08:50 -05:00
_cpu.reset(new Cpu(_memoryManager));
_memoryManager->IncrementMasterClockValue<160>();
//if(_debugger) {
//Reset debugger if it was running before
//auto lock = _debuggerLock.AcquireSafe();
//_debugger.reset();
//GetDebugger();
//}
}
}
shared_ptr<SoundMixer> Console::GetSoundMixer()
{
return _soundMixer;
}
shared_ptr<VideoRenderer> Console::GetVideoRenderer()
{
return _videoRenderer;
}
shared_ptr<VideoDecoder> Console::GetVideoDecoder()
{
return _videoDecoder;
}
2019-02-15 21:33:13 -05:00
shared_ptr<NotificationManager> Console::GetNotificationManager()
{
return _notificationManager;
}
shared_ptr<DebugHud> Console::GetDebugHud()
{
return _debugHud;
}
2019-02-13 18:44:39 -05:00
shared_ptr<Cpu> Console::GetCpu()
{
return _cpu;
}
shared_ptr<Ppu> Console::GetPpu()
{
return _ppu;
}
shared_ptr<Spc> Console::GetSpc()
{
return _spc;
}
2019-02-15 21:33:13 -05:00
shared_ptr<BaseCartridge> Console::GetCartridge()
{
return _cart;
}
shared_ptr<MemoryManager> Console::GetMemoryManager()
{
return _memoryManager;
}
shared_ptr<InternalRegisters> Console::GetInternalRegisters()
{
return _internalRegisters;
}
2019-02-17 19:54:29 -05:00
shared_ptr<ControlManager> Console::GetControlManager()
{
return _controlManager;
}
shared_ptr<DmaController> Console::GetDmaController()
{
return _dmaController;
}
shared_ptr<Debugger> Console::GetDebugger(bool autoStart)
{
shared_ptr<Debugger> debugger = _debugger;
if(!debugger && autoStart) {
//Lock to make sure we don't try to start debuggers in 2 separate threads at once
auto lock = _debuggerLock.AcquireSafe();
debugger = _debugger;
if(!debugger) {
debugger.reset(new Debugger(shared_from_this()));
_debugger = debugger;
}
}
return debugger;
}
2019-02-17 15:02:33 -05:00
bool Console::IsRunning()
{
return _cpu != nullptr;
}
void Console::ProcessCpuRead(uint32_t addr, uint8_t value, MemoryOperationType type)
{
if(_debugger) {
_debugger->ProcessCpuRead(addr, value, type);
}
}
void Console::ProcessCpuWrite(uint32_t addr, uint8_t value, MemoryOperationType type)
{
if(_debugger) {
_debugger->ProcessCpuWrite(addr, value, type);
}
}