Mesen-SX/Core/MemoryAccessCounter.cpp
2020-12-19 23:30:09 +03:00

191 lines
4.5 KiB
C++

#include "stdafx.h"
#include "MemoryAccessCounter.h"
#include "MemoryManager.h"
#include "DebugBreakHelper.h"
#include "Debugger.h"
#include "MemoryDumper.h"
#include "Spc.h"
#include "Sa1.h"
#include "Gsu.h"
#include "Cx4.h"
#include "Gameboy.h"
#include "BaseCartridge.h"
MemoryAccessCounter::MemoryAccessCounter(Debugger* debugger, Console* console)
{
_debugger = debugger;
_memoryManager = console->GetMemoryManager().get();
_spc = console->GetSpc().get();
_sa1 = console->GetCartridge()->GetSa1();
_gsu = console->GetCartridge()->GetGsu();
_cx4 = console->GetCartridge()->GetCx4();
_gameboy = console->GetCartridge()->GetGameboy();
for (int i = (int)SnesMemoryType::PrgRom; i < (int)SnesMemoryType::Register; i++)
{
uint32_t memSize = _debugger->GetMemoryDumper()->GetMemorySize((SnesMemoryType)i);
_counters[i].reserve(memSize);
for (uint32_t j = 0; j < memSize; j++)
{
_counters[i].push_back({j});
}
}
}
bool MemoryAccessCounter::IsAddressUninitialized(AddressInfo& addressInfo)
{
if (!DebugUtilities::IsRomMemory(addressInfo.Type))
{
return _counters[(int)addressInfo.Type][addressInfo.Address].WriteCount == 0;
}
return false;
}
uint64_t MemoryAccessCounter::GetReadCount(AddressInfo& addressInfo)
{
return _counters[(int)addressInfo.Type][addressInfo.Address].ReadCount;
}
bool MemoryAccessCounter::ProcessMemoryRead(AddressInfo& addressInfo, uint64_t masterClock)
{
if (addressInfo.Address < 0)
{
return false;
}
AddressCounters& counts = _counters[(int)addressInfo.Type][addressInfo.Address];
counts.ReadCount++;
counts.ReadStamp = masterClock;
if (counts.WriteCount == 0 && IsAddressUninitialized(addressInfo))
{
//Mark address as read before being written to (if trying to read/execute)
counts.UninitRead = true;
return true;
}
return false;
}
void MemoryAccessCounter::ProcessMemoryWrite(AddressInfo& addressInfo, uint64_t masterClock)
{
if (addressInfo.Address < 0)
{
return;
}
AddressCounters& counts = _counters[(int)addressInfo.Type][addressInfo.Address];
counts.WriteCount++;
counts.WriteStamp = masterClock;
}
void MemoryAccessCounter::ProcessMemoryExec(AddressInfo& addressInfo, uint64_t masterClock)
{
if (addressInfo.Address < 0)
{
return;
}
AddressCounters& counts = _counters[(int)addressInfo.Type][addressInfo.Address];
counts.ExecCount++;
counts.ExecStamp = masterClock;
}
void MemoryAccessCounter::ResetCounts()
{
DebugBreakHelper helper(_debugger);
for (int i = 0; i < (int)SnesMemoryType::Register; i++)
{
for (uint32_t j = 0; j < _counters[i].size(); j++)
{
_counters[i][j] = {j};
}
}
}
void MemoryAccessCounter::GetAccessCounts(uint32_t offset, uint32_t length, SnesMemoryType memoryType,
AddressCounters counts[])
{
switch (memoryType)
{
case SnesMemoryType::CpuMemory:
for (uint32_t i = 0; i < length; i++)
{
AddressInfo info = _memoryManager->GetMemoryMappings()->GetAbsoluteAddress(offset + i);
if (info.Address >= 0)
{
counts[i] = _counters[(int)info.Type][info.Address];
}
}
break;
case SnesMemoryType::SpcMemory:
for (uint32_t i = 0; i < length; i++)
{
AddressInfo info = _spc->GetAbsoluteAddress(offset + i);
if (info.Address >= 0)
{
counts[i] = _counters[(int)info.Type][info.Address];
}
}
break;
case SnesMemoryType::Sa1Memory:
if (_sa1)
{
for (uint32_t i = 0; i < length; i++)
{
AddressInfo info = _sa1->GetMemoryMappings()->GetAbsoluteAddress(offset + i);
if (info.Address >= 0)
{
counts[i] = _counters[(int)info.Type][info.Address];
}
}
}
break;
case SnesMemoryType::GsuMemory:
if (_gsu)
{
for (uint32_t i = 0; i < length; i++)
{
AddressInfo info = _gsu->GetMemoryMappings()->GetAbsoluteAddress(offset + i);
if (info.Address >= 0)
{
counts[i] = _counters[(int)info.Type][info.Address];
}
}
}
break;
case SnesMemoryType::Cx4Memory:
if (_cx4)
{
for (uint32_t i = 0; i < length; i++)
{
AddressInfo info = _cx4->GetMemoryMappings()->GetAbsoluteAddress(offset + i);
if (info.Address >= 0)
{
counts[i] = _counters[(int)info.Type][info.Address];
}
}
}
break;
case SnesMemoryType::GameboyMemory:
if (_gameboy)
{
for (uint32_t i = 0; i < length; i++)
{
AddressInfo info = _gameboy->GetAbsoluteAddress(offset + i);
if (info.Address >= 0)
{
counts[i] = _counters[(int)info.Type][info.Address];
}
}
}
break;
default:
memcpy(counts, _counters[(int)memoryType].data() + offset, length * sizeof(AddressCounters));
break;
}
}