2019-02-12 22:13:09 -05:00
|
|
|
#include "stdafx.h"
|
|
|
|
#include "Console.h"
|
|
|
|
#include "Cpu.h"
|
|
|
|
#include "MemoryManager.h"
|
|
|
|
#include "Debugger.h"
|
2019-02-15 21:33:13 -05:00
|
|
|
#include "NotificationManager.h"
|
2019-02-13 23:02:43 -05:00
|
|
|
#include "VideoDecoder.h"
|
|
|
|
#include "VideoRenderer.h"
|
|
|
|
#include "DebugHud.h"
|
2019-02-12 22:13:09 -05:00
|
|
|
#include "../Utilities/Timer.h"
|
|
|
|
#include "../Utilities/VirtualFile.h"
|
|
|
|
|
2019-02-13 23:02:43 -05:00
|
|
|
void Console::Initialize()
|
|
|
|
{
|
2019-02-15 21:33:13 -05:00
|
|
|
_notificationManager.reset(new NotificationManager());
|
2019-02-13 23:02:43 -05:00
|
|
|
_videoDecoder.reset(new VideoDecoder(shared_from_this()));
|
|
|
|
_videoRenderer.reset(new VideoRenderer(shared_from_this()));
|
|
|
|
_debugHud.reset(new DebugHud());
|
|
|
|
|
|
|
|
_videoDecoder->StartThread();
|
|
|
|
_videoRenderer->StartThread();
|
|
|
|
}
|
|
|
|
|
2019-02-16 01:22:31 -05:00
|
|
|
void Console::Release()
|
|
|
|
{
|
|
|
|
Stop();
|
|
|
|
|
|
|
|
_videoDecoder->StopThread();
|
|
|
|
_videoRenderer->StopThread();
|
|
|
|
|
|
|
|
_videoDecoder.reset();
|
|
|
|
_videoRenderer.reset();
|
|
|
|
_debugHud.reset();
|
|
|
|
_notificationManager.reset();
|
|
|
|
}
|
|
|
|
|
2019-02-12 22:13:09 -05:00
|
|
|
void Console::Run()
|
|
|
|
{
|
|
|
|
if(!_cpu) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
_stopFlag = false;
|
|
|
|
|
|
|
|
auto lock = _runLock.AcquireSafe();
|
|
|
|
while(!_stopFlag) {
|
|
|
|
_cpu->Exec();
|
|
|
|
}
|
|
|
|
|
|
|
|
//Timer timer;
|
|
|
|
/*std::cout << "Time: " << std::to_string(timer.GetElapsedMS()) << std::endl;
|
|
|
|
std::cout << "OP Count: " << std::to_string(_cpu->opCount) << std::endl;
|
|
|
|
std::cout << "OP/sec: " << std::to_string(_cpu->opCount * 1000 / timer.GetElapsedMS()) << std::endl;
|
|
|
|
while(true);*/
|
|
|
|
}
|
|
|
|
|
|
|
|
void Console::Stop()
|
|
|
|
{
|
|
|
|
_stopFlag = true;
|
2019-02-16 08:10:08 -05:00
|
|
|
|
|
|
|
shared_ptr<Debugger> debugger = _debugger;
|
|
|
|
if(debugger) {
|
|
|
|
debugger->Run();
|
|
|
|
}
|
|
|
|
|
2019-02-12 22:13:09 -05:00
|
|
|
_runLock.WaitForRelease();
|
|
|
|
|
|
|
|
_cpu.reset();
|
2019-02-13 13:32:21 -05:00
|
|
|
_ppu.reset();
|
2019-02-16 01:16:57 -05:00
|
|
|
_cart.reset();
|
2019-02-12 22:13:09 -05:00
|
|
|
_memoryManager.reset();
|
|
|
|
}
|
|
|
|
|
|
|
|
void Console::LoadRom(VirtualFile romFile, VirtualFile patchFile)
|
|
|
|
{
|
|
|
|
Stop();
|
|
|
|
|
|
|
|
shared_ptr<BaseCartridge> cart = BaseCartridge::CreateCartridge(romFile, patchFile);
|
|
|
|
if(cart) {
|
2019-02-13 18:44:39 -05:00
|
|
|
_ppu.reset(new Ppu(shared_from_this()));
|
2019-02-15 21:33:13 -05:00
|
|
|
_cart = cart;
|
2019-02-15 00:08:50 -05:00
|
|
|
_memoryManager.reset(new MemoryManager());
|
2019-02-16 01:16:57 -05:00
|
|
|
_memoryManager->Initialize(shared_from_this());
|
2019-02-15 00:08:50 -05:00
|
|
|
|
2019-02-12 22:13:09 -05:00
|
|
|
_cpu.reset(new Cpu(_memoryManager));
|
2019-02-16 00:47:53 -05:00
|
|
|
|
2019-02-16 08:10:08 -05:00
|
|
|
//if(_debugger) {
|
2019-02-16 00:47:53 -05:00
|
|
|
//Reset debugger if it was running before
|
|
|
|
auto lock = _debuggerLock.AcquireSafe();
|
|
|
|
_debugger.reset();
|
|
|
|
GetDebugger();
|
2019-02-16 08:10:08 -05:00
|
|
|
//}
|
2019-02-12 22:13:09 -05:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-02-13 23:02:43 -05:00
|
|
|
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;
|
|
|
|
}
|
|
|
|
|
2019-02-13 23:02:43 -05:00
|
|
|
shared_ptr<DebugHud> Console::GetDebugHud()
|
|
|
|
{
|
|
|
|
return _debugHud;
|
|
|
|
}
|
|
|
|
|
2019-02-13 18:44:39 -05:00
|
|
|
shared_ptr<Cpu> Console::GetCpu()
|
|
|
|
{
|
|
|
|
return _cpu;
|
|
|
|
}
|
|
|
|
|
2019-02-13 13:32:21 -05:00
|
|
|
shared_ptr<Ppu> Console::GetPpu()
|
|
|
|
{
|
|
|
|
return _ppu;
|
|
|
|
}
|
|
|
|
|
2019-02-15 21:33:13 -05:00
|
|
|
shared_ptr<BaseCartridge> Console::GetCartridge()
|
|
|
|
{
|
|
|
|
return _cart;
|
|
|
|
}
|
|
|
|
|
2019-02-13 23:02:43 -05:00
|
|
|
shared_ptr<MemoryManager> Console::GetMemoryManager()
|
|
|
|
{
|
|
|
|
return _memoryManager;
|
|
|
|
}
|
|
|
|
|
2019-02-16 00:47:53 -05:00
|
|
|
shared_ptr<Debugger> Console::GetDebugger(bool autoStart)
|
2019-02-12 22:13:09 -05:00
|
|
|
{
|
2019-02-16 00:47:53 -05:00
|
|
|
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-12 22:13:09 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
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);
|
|
|
|
}
|
|
|
|
}
|