2010-06-12 02:02:06 +00:00
|
|
|
#include "stdafx.h"
|
2016-01-13 15:21:58 +00:00
|
|
|
|
2022-10-10 00:22:17 +00:00
|
|
|
#include <Common\MemoryManagement.h>
|
|
|
|
#include <Project64-core\Debugger.h>
|
|
|
|
#include <Project64-core\N64System\Mips\Disk.h>
|
2022-01-04 05:41:52 +00:00
|
|
|
#include <Project64-core\N64System\Mips\MemoryVirtualMem.h>
|
|
|
|
#include <Project64-core\N64System\N64Rom.h>
|
|
|
|
#include <Project64-core\N64System\N64System.h>
|
2022-10-10 00:22:17 +00:00
|
|
|
#include <Project64-core\N64System\SystemGlobals.h>
|
2016-01-13 15:01:40 +00:00
|
|
|
#include <stdio.h>
|
2010-06-12 02:02:06 +00:00
|
|
|
|
2016-06-29 13:40:36 +00:00
|
|
|
uint32_t CMipsMemoryVM::RegModValue;
|
2015-11-08 18:04:32 +00:00
|
|
|
|
2022-10-10 00:22:17 +00:00
|
|
|
#pragma warning(disable : 4355) // Disable 'this' : used in base member initializer list
|
2015-11-08 18:04:32 +00:00
|
|
|
|
2022-03-21 00:29:02 +00:00
|
|
|
CMipsMemoryVM::CMipsMemoryVM(CN64System & System, bool SavesReadOnly) :
|
2022-05-01 22:06:50 +00:00
|
|
|
m_System(System),
|
2022-03-21 00:29:02 +00:00
|
|
|
m_Reg(System.m_Reg),
|
2023-12-14 01:39:24 +00:00
|
|
|
m_TLB(System.m_TLB),
|
2022-03-21 00:29:02 +00:00
|
|
|
m_AudioInterfaceHandler(System, System.m_Reg),
|
2022-08-01 00:45:56 +00:00
|
|
|
m_CartridgeDomain1Address1Handler(System.m_Reg, g_DDRom),
|
2022-03-21 10:27:57 +00:00
|
|
|
m_CartridgeDomain2Address1Handler(System.m_Reg),
|
2022-04-25 07:42:07 +00:00
|
|
|
m_CartridgeDomain2Address2Handler(System, System.m_Reg, *this, SavesReadOnly),
|
2022-03-21 00:29:02 +00:00
|
|
|
m_RDRAMRegistersHandler(System.m_Reg),
|
|
|
|
m_DPCommandRegistersHandler(System, System.GetPlugins(), System.m_Reg),
|
2022-11-08 00:24:01 +00:00
|
|
|
m_ISViewerHandler(System, m_RomMemoryHandler, *g_Rom),
|
2022-03-21 00:29:02 +00:00
|
|
|
m_MIPSInterfaceHandler(System.m_Reg),
|
2022-07-04 07:44:27 +00:00
|
|
|
m_PeripheralInterfaceHandler(System, *this, System.m_Reg, m_CartridgeDomain2Address2Handler),
|
2022-10-16 21:57:52 +00:00
|
|
|
m_PifRamHandler(System, SavesReadOnly),
|
2022-03-21 00:29:02 +00:00
|
|
|
m_RDRAMInterfaceHandler(System.m_Reg),
|
2022-04-18 11:27:59 +00:00
|
|
|
m_RomMemoryHandler(System, System.m_Reg, *g_Rom),
|
2022-03-21 04:34:59 +00:00
|
|
|
m_SerialInterfaceHandler(*this, System.m_Reg),
|
2022-03-21 00:29:02 +00:00
|
|
|
m_SPRegistersHandler(System, *this, System.m_Reg),
|
|
|
|
m_VideoInterfaceHandler(System, *this, System.m_Reg),
|
2022-04-04 01:00:27 +00:00
|
|
|
m_MemoryReadMap(nullptr),
|
|
|
|
m_MemoryWriteMap(nullptr),
|
2021-04-12 11:35:39 +00:00
|
|
|
m_TLB_ReadMap(nullptr),
|
|
|
|
m_TLB_WriteMap(nullptr),
|
|
|
|
m_RDRAM(nullptr),
|
2022-05-16 01:30:20 +00:00
|
|
|
m_Rom(*g_Rom)
|
2015-11-08 18:04:32 +00:00
|
|
|
{
|
|
|
|
g_Settings->RegisterChangeCB(Game_RDRamSize, this, (CSettings::SettingChangedFunc)RdramChanged);
|
2010-06-12 02:02:06 +00:00
|
|
|
}
|
|
|
|
|
2015-04-28 22:19:02 +00:00
|
|
|
CMipsMemoryVM::~CMipsMemoryVM()
|
2010-06-12 02:02:06 +00:00
|
|
|
{
|
2015-11-08 18:04:32 +00:00
|
|
|
g_Settings->UnregisterChangeCB(Game_RDRamSize, this, (CSettings::SettingChangedFunc)RdramChanged);
|
|
|
|
FreeMemory();
|
2010-06-12 02:02:06 +00:00
|
|
|
}
|
|
|
|
|
2015-11-08 18:04:32 +00:00
|
|
|
void CMipsMemoryVM::Reset(bool /*EraseMemory*/)
|
2010-06-14 21:14:58 +00:00
|
|
|
{
|
2022-04-04 01:00:27 +00:00
|
|
|
if (m_MemoryReadMap != nullptr && m_MemoryWriteMap != nullptr)
|
|
|
|
{
|
2022-05-22 20:54:56 +00:00
|
|
|
memset(m_MemoryReadMap, -1, 0x100000 * sizeof(m_MemoryReadMap[0]));
|
|
|
|
memset(m_MemoryWriteMap, -1, 0x100000 * sizeof(m_MemoryWriteMap[0]));
|
2022-04-04 01:00:27 +00:00
|
|
|
for (uint32_t i = 0; i < 2; i++)
|
|
|
|
{
|
|
|
|
uint32_t BaseAddress = i == 0 ? 0x80000000 : 0xA0000000;
|
|
|
|
for (size_t Address = BaseAddress; Address < BaseAddress + m_AllocatedRdramSize; Address += 0x1000)
|
|
|
|
{
|
|
|
|
m_MemoryReadMap[Address >> 12] = (size_t)((m_RDRAM + (Address & 0x1FFFFFFF)) - Address);
|
|
|
|
m_MemoryWriteMap[Address >> 12] = (size_t)((m_RDRAM + (Address & 0x1FFFFFFF)) - Address);
|
|
|
|
}
|
|
|
|
for (size_t Address = BaseAddress + 0x04000000; Address < (BaseAddress + 0x04001000); Address += 0x1000)
|
|
|
|
{
|
2022-10-17 06:59:05 +00:00
|
|
|
m_MemoryReadMap[Address >> 12] = (size_t)(Dmem() - Address);
|
2022-04-04 01:00:27 +00:00
|
|
|
}
|
|
|
|
for (size_t Address = BaseAddress + 0x04001000; Address < (BaseAddress + 0x04002000); Address += 0x1000)
|
|
|
|
{
|
2022-10-17 06:59:05 +00:00
|
|
|
m_MemoryReadMap[Address >> 12] = (size_t)(Imem() - Address);
|
2022-04-04 01:00:27 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2015-11-08 18:04:32 +00:00
|
|
|
if (m_TLB_ReadMap)
|
|
|
|
{
|
2022-05-22 20:54:56 +00:00
|
|
|
memset(m_TLB_ReadMap, -1, 0xFFFFF * sizeof(m_TLB_ReadMap[0]));
|
|
|
|
memset(m_TLB_WriteMap, -1, 0xFFFFF * sizeof(m_TLB_WriteMap[0]));
|
|
|
|
for (uint32_t Address = 0x80000000; Address < 0xC0000000; Address += 0x1000)
|
2015-11-08 18:04:32 +00:00
|
|
|
{
|
2022-05-02 09:40:35 +00:00
|
|
|
m_TLB_ReadMap[Address >> 12] = (Address & 0x1FFFFFFF) - Address;
|
|
|
|
m_TLB_WriteMap[Address >> 12] = (Address & 0x1FFFFFFF) - Address;
|
2015-11-08 18:04:32 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
if (g_Settings->LoadDword(Rdb_TLB_VAddrStart) != 0)
|
|
|
|
{
|
2022-04-04 01:00:27 +00:00
|
|
|
uint32_t Start = g_Settings->LoadDword(Rdb_TLB_VAddrStart); //0x7F000000;
|
2022-10-10 00:22:17 +00:00
|
|
|
uint32_t Len = g_Settings->LoadDword(Rdb_TLB_VAddrLen); //0x01000000;
|
2022-04-04 01:00:27 +00:00
|
|
|
uint32_t PAddr = g_Settings->LoadDword(Rdb_TLB_PAddrStart); //0x10034b30;
|
|
|
|
uint32_t End = Start + Len;
|
|
|
|
for (uint32_t Address = Start; Address < End; Address += 0x1000)
|
2015-11-08 18:04:32 +00:00
|
|
|
{
|
2022-04-04 01:00:27 +00:00
|
|
|
uint32_t TargetAddress = (Address - Start + PAddr);
|
|
|
|
if (TargetAddress < m_AllocatedRdramSize)
|
|
|
|
{
|
|
|
|
m_MemoryReadMap[Address >> 12] = ((size_t)m_RDRAM + TargetAddress) - Address;
|
|
|
|
m_MemoryWriteMap[Address >> 12] = ((size_t)m_RDRAM + TargetAddress) - Address;
|
|
|
|
}
|
2022-04-18 11:27:59 +00:00
|
|
|
if (TargetAddress >= 0x10000000 && TargetAddress < (0x10000000 + g_Rom->GetRomSize()))
|
2022-04-04 01:00:27 +00:00
|
|
|
{
|
2022-04-18 11:27:59 +00:00
|
|
|
m_MemoryReadMap[Address >> 12] = ((size_t)g_Rom->GetRomAddress() + (TargetAddress - 0x10000000)) - Address;
|
2022-04-04 01:00:27 +00:00
|
|
|
}
|
2022-05-02 09:40:35 +00:00
|
|
|
m_TLB_ReadMap[Address >> 12] = TargetAddress - Address;
|
|
|
|
m_TLB_WriteMap[Address >> 12] = TargetAddress - Address;
|
2015-11-08 18:04:32 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2010-06-14 21:14:58 +00:00
|
|
|
}
|
|
|
|
|
2024-06-13 02:20:06 +00:00
|
|
|
bool CMipsMemoryVM::Initialize(void)
|
2010-06-12 02:02:06 +00:00
|
|
|
{
|
2021-04-12 11:35:39 +00:00
|
|
|
if (m_RDRAM != nullptr)
|
2015-11-08 18:04:32 +00:00
|
|
|
{
|
|
|
|
return true;
|
|
|
|
}
|
2024-06-13 02:20:06 +00:00
|
|
|
m_RDRAM = (uint8_t *)AllocateAddressSpace(0x02000000, (void *)g_Settings->LoadDword(Setting_FixedRdramAddress));
|
2021-04-12 11:35:39 +00:00
|
|
|
if (m_RDRAM == nullptr)
|
2015-11-08 18:04:32 +00:00
|
|
|
{
|
2024-06-13 02:20:06 +00:00
|
|
|
WriteTrace(TraceN64System, TraceError, "Failed to reserve RDRAM (Size: 0x%X)", 0x02000000);
|
2015-11-08 18:04:32 +00:00
|
|
|
FreeMemory();
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2023-01-22 22:00:13 +00:00
|
|
|
bool RdbRamSet = g_Settings->LoadDword(Rdb_RDRamSize, m_AllocatedRdramSize);
|
|
|
|
if (!g_Settings->LoadDword(Game_RDRamSize, m_AllocatedRdramSize) && !RdbRamSet)
|
2023-01-16 10:23:48 +00:00
|
|
|
{
|
|
|
|
m_AllocatedRdramSize = g_Settings->LoadDword(g_Settings->LoadBool(Game_Known) ? Default_RDRamSizeKnown : Default_RDRamSizeUnknown);
|
|
|
|
}
|
2021-04-12 11:35:39 +00:00
|
|
|
if (CommitMemory(m_RDRAM, m_AllocatedRdramSize, MEM_READWRITE) == nullptr)
|
2015-11-08 18:04:32 +00:00
|
|
|
{
|
2021-05-18 11:51:36 +00:00
|
|
|
WriteTrace(TraceN64System, TraceError, "Failed to allocate RDRAM (Size: 0x%X)", m_AllocatedRdramSize);
|
2015-11-08 18:04:32 +00:00
|
|
|
FreeMemory();
|
|
|
|
return false;
|
|
|
|
}
|
2023-09-27 21:59:11 +00:00
|
|
|
g_Settings->SaveDword(Setting_AllocatedRdramSize, m_AllocatedRdramSize);
|
2015-11-08 18:04:32 +00:00
|
|
|
|
2022-10-10 00:22:17 +00:00
|
|
|
m_MemoryReadMap = new size_t[0x100000];
|
2022-04-04 01:00:27 +00:00
|
|
|
if (m_MemoryReadMap == nullptr)
|
|
|
|
{
|
|
|
|
WriteTrace(TraceN64System, TraceError, "Failed to allocate m_MemoryReadMap (Size: 0x%X)", 0x100000 * sizeof(size_t));
|
|
|
|
FreeMemory();
|
|
|
|
return false;
|
|
|
|
}
|
2022-10-10 00:22:17 +00:00
|
|
|
m_MemoryWriteMap = new size_t[0x100000];
|
2022-04-04 01:00:27 +00:00
|
|
|
if (m_MemoryWriteMap == nullptr)
|
|
|
|
{
|
|
|
|
WriteTrace(TraceN64System, TraceError, "Failed to allocate m_MemoryWriteMap (Size: 0x%X)", 0x100000 * sizeof(size_t));
|
|
|
|
FreeMemory();
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2022-05-02 09:40:35 +00:00
|
|
|
m_TLB_ReadMap = new uint32_t[0x100000];
|
2021-04-12 11:35:39 +00:00
|
|
|
if (m_TLB_ReadMap == nullptr)
|
2015-11-08 18:04:32 +00:00
|
|
|
{
|
2021-05-18 11:51:36 +00:00
|
|
|
WriteTrace(TraceN64System, TraceError, "Failed to allocate m_TLB_ReadMap (Size: 0x%X)", 0x100000 * sizeof(size_t));
|
2015-11-08 18:04:32 +00:00
|
|
|
FreeMemory();
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2022-05-02 09:40:35 +00:00
|
|
|
m_TLB_WriteMap = new uint32_t[0x100000];
|
2021-04-12 11:35:39 +00:00
|
|
|
if (m_TLB_WriteMap == nullptr)
|
2015-11-08 18:04:32 +00:00
|
|
|
{
|
2021-05-18 11:51:36 +00:00
|
|
|
WriteTrace(TraceN64System, TraceError, "Failed to allocate m_TLB_WriteMap (Size: 0x%X)", 0xFFFFF * sizeof(size_t));
|
2015-11-08 18:04:32 +00:00
|
|
|
FreeMemory();
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
Reset(false);
|
|
|
|
return true;
|
2010-06-12 02:02:06 +00:00
|
|
|
}
|
|
|
|
|
2015-04-28 22:19:02 +00:00
|
|
|
void CMipsMemoryVM::FreeMemory()
|
2010-06-12 02:02:06 +00:00
|
|
|
{
|
2015-11-08 18:04:32 +00:00
|
|
|
if (m_RDRAM)
|
|
|
|
{
|
2024-06-13 02:20:06 +00:00
|
|
|
DecommitMemory(m_RDRAM, 0x02000000);
|
|
|
|
FreeAddressSpace(m_RDRAM, 0x02000000);
|
2021-04-12 11:35:39 +00:00
|
|
|
m_RDRAM = nullptr;
|
2015-11-08 18:04:32 +00:00
|
|
|
}
|
|
|
|
if (m_TLB_ReadMap)
|
|
|
|
{
|
2016-01-13 11:52:21 +00:00
|
|
|
delete[] m_TLB_ReadMap;
|
2021-04-12 11:35:39 +00:00
|
|
|
m_TLB_ReadMap = nullptr;
|
2015-11-08 18:04:32 +00:00
|
|
|
}
|
|
|
|
if (m_TLB_WriteMap)
|
|
|
|
{
|
2016-01-13 11:52:21 +00:00
|
|
|
delete[] m_TLB_WriteMap;
|
2021-04-12 11:35:39 +00:00
|
|
|
m_TLB_WriteMap = nullptr;
|
2015-11-08 18:04:32 +00:00
|
|
|
}
|
2022-04-04 01:00:27 +00:00
|
|
|
if (m_MemoryReadMap)
|
|
|
|
{
|
|
|
|
delete[] m_MemoryReadMap;
|
|
|
|
m_MemoryReadMap = nullptr;
|
|
|
|
}
|
|
|
|
if (m_MemoryWriteMap)
|
|
|
|
{
|
|
|
|
delete[] m_MemoryWriteMap;
|
|
|
|
m_MemoryWriteMap = nullptr;
|
|
|
|
}
|
2010-06-12 02:02:06 +00:00
|
|
|
}
|
|
|
|
|
2022-06-13 01:54:36 +00:00
|
|
|
uint8_t * CMipsMemoryVM::MemoryPtr(uint32_t VAddr, uint32_t Size, bool Read)
|
2022-05-02 10:52:31 +00:00
|
|
|
{
|
|
|
|
if (m_TLB_ReadMap[VAddr >> 12] == -1)
|
|
|
|
{
|
2022-05-16 05:56:20 +00:00
|
|
|
return nullptr;
|
2022-05-02 10:52:31 +00:00
|
|
|
}
|
2022-05-16 05:56:20 +00:00
|
|
|
uint32_t PAddr = m_TLB_ReadMap[VAddr >> 12] + VAddr;
|
2022-06-13 01:54:36 +00:00
|
|
|
if ((PAddr + Size) < m_AllocatedRdramSize)
|
2022-05-02 10:52:31 +00:00
|
|
|
{
|
2022-10-10 00:22:17 +00:00
|
|
|
return (uint8_t *)(m_RDRAM + PAddr);
|
2022-05-02 10:52:31 +00:00
|
|
|
}
|
|
|
|
|
2022-10-17 06:59:05 +00:00
|
|
|
if (PAddr >= 0x04000000 && (PAddr + Size) <= 0x04001000)
|
2022-05-02 10:52:31 +00:00
|
|
|
{
|
2022-10-17 06:59:05 +00:00
|
|
|
return (uint8_t *)&m_SPRegistersHandler.Dmem()[PAddr - 0x04000000];
|
2022-05-02 10:52:31 +00:00
|
|
|
}
|
2022-10-17 06:59:05 +00:00
|
|
|
if (PAddr >= 0x04001000 && (PAddr + Size) <= 0x04002000)
|
2022-05-02 10:52:31 +00:00
|
|
|
{
|
2022-10-17 06:59:05 +00:00
|
|
|
return (uint8_t *)&m_SPRegistersHandler.Imem()[PAddr - 0x04001000];
|
2022-05-16 05:56:20 +00:00
|
|
|
}
|
2022-06-13 01:54:36 +00:00
|
|
|
if (Read && PAddr >= 0x10000000 && (PAddr + Size) < (0x10000000 + m_Rom.GetRomSize()))
|
2022-05-16 05:56:20 +00:00
|
|
|
{
|
|
|
|
return (uint8_t *)&m_Rom.GetRomAddress()[PAddr - 0x10000000];
|
2022-05-02 10:52:31 +00:00
|
|
|
}
|
2022-05-16 05:56:20 +00:00
|
|
|
|
2022-05-02 10:52:31 +00:00
|
|
|
g_Notify->BreakPoint(__FILE__, __LINE__);
|
2022-05-16 05:56:20 +00:00
|
|
|
return nullptr;
|
2022-05-02 10:52:31 +00:00
|
|
|
}
|
|
|
|
|
2022-05-16 05:56:20 +00:00
|
|
|
bool CMipsMemoryVM::MemoryValue8(uint32_t VAddr, uint8_t & Value)
|
2022-05-02 10:52:31 +00:00
|
|
|
{
|
2022-05-16 05:56:20 +00:00
|
|
|
uint8_t * ptr = MemoryPtr(VAddr ^ 3, 1, true);
|
|
|
|
if (ptr == nullptr)
|
2022-05-02 10:52:31 +00:00
|
|
|
{
|
|
|
|
return false;
|
|
|
|
}
|
2022-05-16 05:56:20 +00:00
|
|
|
Value = *ptr;
|
|
|
|
return true;
|
|
|
|
}
|
2022-05-02 10:52:31 +00:00
|
|
|
|
2022-05-20 01:02:15 +00:00
|
|
|
bool CMipsMemoryVM::MemoryValue16(uint32_t VAddr, uint16_t & Value)
|
2022-05-16 05:56:20 +00:00
|
|
|
{
|
|
|
|
uint8_t * ptr = MemoryPtr(VAddr ^ 2, 2, true);
|
|
|
|
if (ptr == nullptr)
|
2022-05-02 10:52:31 +00:00
|
|
|
{
|
2022-05-16 05:56:20 +00:00
|
|
|
return false;
|
2022-05-02 10:52:31 +00:00
|
|
|
}
|
2022-10-10 00:22:17 +00:00
|
|
|
Value = *(uint16_t *)(ptr);
|
2022-05-16 05:56:20 +00:00
|
|
|
return true;
|
2022-05-02 10:52:31 +00:00
|
|
|
}
|
|
|
|
|
2022-05-20 01:02:15 +00:00
|
|
|
bool CMipsMemoryVM::MemoryValue32(uint32_t VAddr, uint32_t & Value)
|
2022-05-02 10:52:31 +00:00
|
|
|
{
|
2022-05-16 05:56:20 +00:00
|
|
|
uint8_t * ptr = MemoryPtr(VAddr, 4, true);
|
|
|
|
if (ptr == nullptr)
|
2022-05-02 10:52:31 +00:00
|
|
|
{
|
|
|
|
return false;
|
|
|
|
}
|
2022-10-10 00:22:17 +00:00
|
|
|
Value = *(uint32_t *)(ptr);
|
2022-05-16 05:56:20 +00:00
|
|
|
return true;
|
2022-05-02 10:52:31 +00:00
|
|
|
}
|
|
|
|
|
2022-06-13 01:54:36 +00:00
|
|
|
bool CMipsMemoryVM::MemoryValue64(uint32_t VAddr, uint64_t & Value)
|
|
|
|
{
|
|
|
|
uint8_t * ptr = MemoryPtr(VAddr, 8, true);
|
|
|
|
if (ptr == nullptr)
|
|
|
|
{
|
|
|
|
return false;
|
|
|
|
}
|
2022-10-10 00:22:17 +00:00
|
|
|
*((uint32_t *)(&Value) + 1) = *(uint32_t *)(ptr);
|
|
|
|
*((uint32_t *)(&Value) + 0) = *(uint32_t *)(ptr + 4);
|
2022-06-13 01:54:36 +00:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2022-05-02 10:52:31 +00:00
|
|
|
bool CMipsMemoryVM::UpdateMemoryValue8(uint32_t VAddr, uint8_t Value)
|
|
|
|
{
|
2022-05-16 05:56:20 +00:00
|
|
|
uint8_t * ptr = MemoryPtr(VAddr ^ 3, 1, false);
|
|
|
|
if (ptr == nullptr)
|
2022-05-02 10:52:31 +00:00
|
|
|
{
|
|
|
|
return false;
|
|
|
|
}
|
2022-05-16 05:56:20 +00:00
|
|
|
*ptr = Value;
|
|
|
|
return true;
|
2022-05-02 10:52:31 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
bool CMipsMemoryVM::UpdateMemoryValue16(uint32_t VAddr, uint16_t Value)
|
|
|
|
{
|
2022-05-16 05:56:20 +00:00
|
|
|
uint8_t * ptr = MemoryPtr(VAddr ^ 2, 2, false);
|
|
|
|
if (ptr == nullptr)
|
2022-05-02 10:52:31 +00:00
|
|
|
{
|
|
|
|
return false;
|
|
|
|
}
|
2022-10-10 00:22:17 +00:00
|
|
|
*(uint16_t *)(ptr) = Value;
|
2022-05-16 05:56:20 +00:00
|
|
|
return true;
|
2022-05-02 10:52:31 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
bool CMipsMemoryVM::UpdateMemoryValue32(uint32_t VAddr, uint32_t Value)
|
|
|
|
{
|
2022-05-16 05:56:20 +00:00
|
|
|
uint8_t * ptr = MemoryPtr(VAddr, 4, false);
|
|
|
|
if (ptr == nullptr)
|
2022-05-02 10:52:31 +00:00
|
|
|
{
|
|
|
|
return false;
|
|
|
|
}
|
2022-10-10 00:22:17 +00:00
|
|
|
*(uint32_t *)(ptr) = Value;
|
2022-05-16 05:56:20 +00:00
|
|
|
return true;
|
2022-05-02 10:52:31 +00:00
|
|
|
}
|
|
|
|
|
2022-10-10 00:22:17 +00:00
|
|
|
bool CMipsMemoryVM::LB_Memory(uint64_t VAddr, uint8_t & Value)
|
2010-06-12 02:02:06 +00:00
|
|
|
{
|
2023-10-05 03:58:32 +00:00
|
|
|
if ((uint64_t)((int32_t)VAddr) != VAddr)
|
2022-10-03 08:05:50 +00:00
|
|
|
{
|
2023-10-05 03:58:32 +00:00
|
|
|
uint32_t PAddr;
|
|
|
|
bool MemoryUnused;
|
2023-12-14 01:39:24 +00:00
|
|
|
if (!m_TLB.VAddrToPAddr(VAddr, PAddr, MemoryUnused))
|
2023-10-05 03:58:32 +00:00
|
|
|
{
|
|
|
|
m_Reg.TriggerAddressException(VAddr, MemoryUnused ? EXC_RADE : EXC_RMISS);
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
return LB_PhysicalAddress(PAddr, Value);
|
2022-10-03 08:05:50 +00:00
|
|
|
}
|
2022-09-19 12:06:36 +00:00
|
|
|
uint32_t VAddr32 = (uint32_t)VAddr;
|
|
|
|
if (HaveReadBP() && g_Debugger->ReadBP8(VAddr32) && MemoryBreakpoint())
|
2022-06-13 01:54:36 +00:00
|
|
|
{
|
|
|
|
return false;
|
|
|
|
}
|
2022-10-10 00:22:17 +00:00
|
|
|
uint8_t * MemoryPtr = (uint8_t *)m_MemoryReadMap[VAddr32 >> 12];
|
|
|
|
if (MemoryPtr != (uint8_t *)-1)
|
2022-04-04 01:00:27 +00:00
|
|
|
{
|
2022-10-10 00:22:17 +00:00
|
|
|
Value = *(uint8_t *)(MemoryPtr + (VAddr32 ^ 3));
|
2022-04-04 01:00:27 +00:00
|
|
|
return true;
|
|
|
|
}
|
2023-10-04 23:02:45 +00:00
|
|
|
return LB_VAddr32(VAddr32, Value);
|
2010-06-12 02:02:06 +00:00
|
|
|
}
|
|
|
|
|
2022-09-19 12:06:36 +00:00
|
|
|
bool CMipsMemoryVM::LH_Memory(uint64_t VAddr, uint16_t & Value)
|
2010-06-12 02:02:06 +00:00
|
|
|
{
|
2023-10-05 03:58:32 +00:00
|
|
|
if ((VAddr & 1) != 0)
|
2022-10-03 08:05:50 +00:00
|
|
|
{
|
2023-10-05 03:58:32 +00:00
|
|
|
m_Reg.TriggerAddressException(VAddr, EXC_RADE);
|
2022-10-03 08:05:50 +00:00
|
|
|
return false;
|
|
|
|
}
|
2023-10-05 03:58:32 +00:00
|
|
|
if ((uint64_t)((int32_t)VAddr) != VAddr)
|
2022-06-13 01:54:36 +00:00
|
|
|
{
|
2023-10-05 03:58:32 +00:00
|
|
|
uint32_t PAddr;
|
|
|
|
bool MemoryUnused;
|
2023-12-14 01:39:24 +00:00
|
|
|
if (!m_TLB.VAddrToPAddr(VAddr, PAddr, MemoryUnused))
|
2023-10-05 03:58:32 +00:00
|
|
|
{
|
|
|
|
m_Reg.TriggerAddressException(VAddr, MemoryUnused ? EXC_RADE : EXC_RMISS);
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
return LH_PhysicalAddress(PAddr, Value);
|
2022-06-13 01:54:36 +00:00
|
|
|
}
|
2023-10-05 03:58:32 +00:00
|
|
|
uint32_t VAddr32 = (uint32_t)VAddr;
|
2022-09-19 12:06:36 +00:00
|
|
|
if (HaveReadBP() && g_Debugger->ReadBP16(VAddr32) && MemoryBreakpoint())
|
2022-06-13 01:54:36 +00:00
|
|
|
{
|
|
|
|
return false;
|
|
|
|
}
|
2022-10-10 00:22:17 +00:00
|
|
|
uint8_t * MemoryPtr = (uint8_t *)m_MemoryReadMap[VAddr32 >> 12];
|
|
|
|
if (MemoryPtr != (uint8_t *)-1)
|
2022-04-04 01:00:27 +00:00
|
|
|
{
|
2022-10-10 00:22:17 +00:00
|
|
|
Value = *(uint16_t *)(MemoryPtr + (VAddr32 ^ 2));
|
2022-04-04 01:00:27 +00:00
|
|
|
return true;
|
|
|
|
}
|
2023-10-04 23:02:45 +00:00
|
|
|
return LH_VAddr32(VAddr32, Value);
|
2010-06-12 02:02:06 +00:00
|
|
|
}
|
|
|
|
|
2022-09-19 12:06:36 +00:00
|
|
|
bool CMipsMemoryVM::LW_Memory(uint64_t VAddr, uint32_t & Value)
|
2010-06-12 02:02:06 +00:00
|
|
|
{
|
2023-10-05 03:58:32 +00:00
|
|
|
if ((VAddr & 3) != 0)
|
2022-10-03 08:05:50 +00:00
|
|
|
{
|
2023-10-05 03:58:32 +00:00
|
|
|
m_Reg.TriggerAddressException(VAddr, EXC_RADE);
|
2022-10-03 08:05:50 +00:00
|
|
|
return false;
|
|
|
|
}
|
2023-10-05 03:58:32 +00:00
|
|
|
if ((uint64_t)((int32_t)VAddr) != VAddr)
|
2022-06-13 01:54:36 +00:00
|
|
|
{
|
2023-10-05 03:58:32 +00:00
|
|
|
uint32_t PAddr;
|
|
|
|
bool MemoryUnused;
|
2023-12-14 01:39:24 +00:00
|
|
|
if (!m_TLB.VAddrToPAddr(VAddr, PAddr, MemoryUnused))
|
2023-10-05 03:58:32 +00:00
|
|
|
{
|
|
|
|
m_Reg.TriggerAddressException(VAddr, MemoryUnused ? EXC_RADE : EXC_RMISS);
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
return LW_PhysicalAddress(PAddr, Value);
|
2022-06-13 01:54:36 +00:00
|
|
|
}
|
2023-10-05 03:58:32 +00:00
|
|
|
uint32_t VAddr32 = (uint32_t)VAddr;
|
2022-09-19 12:06:36 +00:00
|
|
|
if (HaveReadBP() && g_Debugger->ReadBP32(VAddr32) && MemoryBreakpoint())
|
2022-06-13 01:54:36 +00:00
|
|
|
{
|
|
|
|
return false;
|
|
|
|
}
|
2022-10-10 00:22:17 +00:00
|
|
|
uint8_t * MemoryPtr = (uint8_t *)m_MemoryReadMap[VAddr32 >> 12];
|
2022-04-04 01:00:27 +00:00
|
|
|
if (MemoryPtr != (uint8_t *)-1)
|
|
|
|
{
|
2022-10-10 00:22:17 +00:00
|
|
|
Value = *(uint32_t *)(MemoryPtr + VAddr32);
|
2022-04-04 01:00:27 +00:00
|
|
|
return true;
|
|
|
|
}
|
2023-10-04 23:02:45 +00:00
|
|
|
return LW_VAddr32(VAddr32, Value);
|
2010-06-12 02:02:06 +00:00
|
|
|
}
|
|
|
|
|
2022-10-10 00:22:17 +00:00
|
|
|
bool CMipsMemoryVM::LD_Memory(uint64_t VAddr, uint64_t & Value)
|
2010-06-12 02:02:06 +00:00
|
|
|
{
|
2023-10-05 03:58:32 +00:00
|
|
|
if ((VAddr & 7) != 0)
|
2022-10-03 08:05:50 +00:00
|
|
|
{
|
2023-10-05 03:58:32 +00:00
|
|
|
m_Reg.TriggerAddressException(VAddr, EXC_RADE);
|
2022-10-03 08:05:50 +00:00
|
|
|
return false;
|
|
|
|
}
|
2023-10-05 03:58:32 +00:00
|
|
|
if ((uint64_t)((int32_t)VAddr) != VAddr)
|
2022-06-13 01:54:36 +00:00
|
|
|
{
|
2023-10-05 03:58:32 +00:00
|
|
|
uint32_t PAddr;
|
|
|
|
bool MemoryUnused;
|
2023-12-14 01:39:24 +00:00
|
|
|
if (!m_TLB.VAddrToPAddr(VAddr, PAddr, MemoryUnused))
|
2023-10-05 03:58:32 +00:00
|
|
|
{
|
|
|
|
m_Reg.TriggerAddressException(VAddr, MemoryUnused ? EXC_RADE : EXC_RMISS);
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
return LD_PhysicalAddress(PAddr, Value);
|
2022-06-13 01:54:36 +00:00
|
|
|
}
|
2023-10-05 03:58:32 +00:00
|
|
|
uint32_t VAddr32 = (uint32_t)VAddr;
|
2022-09-19 12:06:36 +00:00
|
|
|
if (HaveReadBP() && g_Debugger->ReadBP64(VAddr32) && MemoryBreakpoint())
|
2022-06-13 01:54:36 +00:00
|
|
|
{
|
|
|
|
return false;
|
|
|
|
}
|
2022-10-10 00:22:17 +00:00
|
|
|
uint8_t * MemoryPtr = (uint8_t *)m_MemoryReadMap[VAddr32 >> 12];
|
2022-04-04 01:00:27 +00:00
|
|
|
if (MemoryPtr != (uint8_t *)-1)
|
|
|
|
{
|
2022-10-10 00:22:17 +00:00
|
|
|
*((uint32_t *)(&Value) + 1) = *(uint32_t *)(MemoryPtr + VAddr32);
|
|
|
|
*((uint32_t *)(&Value) + 0) = *(uint32_t *)(MemoryPtr + VAddr32 + 4);
|
2022-04-04 01:00:27 +00:00
|
|
|
return true;
|
|
|
|
}
|
2023-10-04 23:02:45 +00:00
|
|
|
return LD_VAddr32(VAddr32, Value);
|
2010-06-12 02:02:06 +00:00
|
|
|
}
|
|
|
|
|
2022-09-19 12:06:36 +00:00
|
|
|
bool CMipsMemoryVM::SB_Memory(uint64_t VAddr, uint32_t Value)
|
2015-01-22 05:50:20 +00:00
|
|
|
{
|
2023-10-05 03:58:32 +00:00
|
|
|
if ((uint64_t)((int32_t)VAddr) != VAddr)
|
2022-10-03 08:05:50 +00:00
|
|
|
{
|
2023-10-05 03:58:32 +00:00
|
|
|
uint32_t PAddr;
|
|
|
|
bool MemoryUnused;
|
2023-12-14 01:39:24 +00:00
|
|
|
if (!m_TLB.VAddrToPAddr(VAddr, PAddr, MemoryUnused))
|
2023-10-05 03:58:32 +00:00
|
|
|
{
|
|
|
|
m_Reg.TriggerAddressException(VAddr, MemoryUnused ? EXC_WADE : EXC_WMISS);
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
return SB_PhysicalAddress(PAddr, Value);
|
2022-10-03 08:05:50 +00:00
|
|
|
}
|
2022-09-19 12:06:36 +00:00
|
|
|
uint32_t VAddr32 = (uint32_t)VAddr;
|
|
|
|
if (HaveWriteBP() && g_Debugger->WriteBP8(VAddr32) && MemoryBreakpoint())
|
2022-06-13 01:54:36 +00:00
|
|
|
{
|
|
|
|
return false;
|
|
|
|
}
|
2022-10-10 00:22:17 +00:00
|
|
|
uint8_t * MemoryPtr = (uint8_t *)m_MemoryWriteMap[VAddr32 >> 12];
|
2022-04-04 01:00:27 +00:00
|
|
|
if (MemoryPtr != (uint8_t *)-1)
|
2015-11-08 18:04:32 +00:00
|
|
|
{
|
2022-10-10 00:22:17 +00:00
|
|
|
*(uint8_t *)(MemoryPtr + (VAddr32 ^ 3)) = (uint8_t)Value;
|
2015-11-08 18:04:32 +00:00
|
|
|
return true;
|
|
|
|
}
|
2023-10-04 23:02:45 +00:00
|
|
|
return SB_VAddr32(VAddr32, Value);
|
2010-06-12 02:02:06 +00:00
|
|
|
}
|
|
|
|
|
2022-09-19 12:06:36 +00:00
|
|
|
bool CMipsMemoryVM::SH_Memory(uint64_t VAddr, uint32_t Value)
|
2010-06-12 02:02:06 +00:00
|
|
|
{
|
2023-10-05 03:58:32 +00:00
|
|
|
if ((VAddr & 1) != 0)
|
2022-10-03 08:05:50 +00:00
|
|
|
{
|
2023-10-05 03:58:32 +00:00
|
|
|
m_Reg.TriggerAddressException(VAddr, EXC_WADE);
|
2022-10-03 08:05:50 +00:00
|
|
|
return false;
|
|
|
|
}
|
2023-10-05 03:58:32 +00:00
|
|
|
if ((uint64_t)((int32_t)VAddr) != VAddr)
|
2022-06-13 01:54:36 +00:00
|
|
|
{
|
2023-10-05 03:58:32 +00:00
|
|
|
uint32_t PAddr;
|
|
|
|
bool MemoryUnused;
|
2023-12-14 01:39:24 +00:00
|
|
|
if (!m_TLB.VAddrToPAddr(VAddr, PAddr, MemoryUnused))
|
2023-10-05 03:58:32 +00:00
|
|
|
{
|
|
|
|
m_Reg.TriggerAddressException(VAddr, MemoryUnused ? EXC_WADE : EXC_WMISS);
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
return SH_PhysicalAddress(PAddr, Value);
|
2022-06-13 01:54:36 +00:00
|
|
|
}
|
2023-10-05 03:58:32 +00:00
|
|
|
uint32_t VAddr32 = (uint32_t)VAddr;
|
2022-09-19 12:06:36 +00:00
|
|
|
if (HaveWriteBP() && g_Debugger->WriteBP16(VAddr32) && MemoryBreakpoint())
|
2022-06-13 01:54:36 +00:00
|
|
|
{
|
|
|
|
return false;
|
|
|
|
}
|
2022-10-10 00:22:17 +00:00
|
|
|
uint8_t * MemoryPtr = (uint8_t *)m_MemoryWriteMap[VAddr32 >> 12];
|
2022-04-04 01:00:27 +00:00
|
|
|
if (MemoryPtr != (uint8_t *)-1)
|
|
|
|
{
|
2022-10-10 00:22:17 +00:00
|
|
|
*(uint16_t *)(MemoryPtr + (VAddr32 ^ 2)) = (uint16_t)Value;
|
2022-04-04 01:00:27 +00:00
|
|
|
return true;
|
|
|
|
}
|
2023-10-04 23:02:45 +00:00
|
|
|
return SH_VAddr32(VAddr32, Value);
|
2010-06-12 02:02:06 +00:00
|
|
|
}
|
|
|
|
|
2022-09-19 12:06:36 +00:00
|
|
|
bool CMipsMemoryVM::SW_Memory(uint64_t VAddr, uint32_t Value)
|
2010-06-12 02:02:06 +00:00
|
|
|
{
|
2023-10-05 03:58:32 +00:00
|
|
|
if ((VAddr & 3) != 0)
|
2022-10-03 08:05:50 +00:00
|
|
|
{
|
2023-10-05 03:58:32 +00:00
|
|
|
m_Reg.TriggerAddressException(VAddr, EXC_WADE);
|
2022-10-03 08:05:50 +00:00
|
|
|
return false;
|
|
|
|
}
|
2023-10-05 03:58:32 +00:00
|
|
|
if ((uint64_t)((int32_t)VAddr) != VAddr)
|
2022-06-13 01:54:36 +00:00
|
|
|
{
|
2023-10-05 03:58:32 +00:00
|
|
|
uint32_t PAddr;
|
|
|
|
bool MemoryUnused;
|
2023-12-14 01:39:24 +00:00
|
|
|
if (!m_TLB.VAddrToPAddr(VAddr, PAddr, MemoryUnused))
|
2023-10-05 03:58:32 +00:00
|
|
|
{
|
|
|
|
m_Reg.TriggerAddressException(VAddr, MemoryUnused ? EXC_WADE : EXC_WMISS);
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
return SW_PhysicalAddress(PAddr, Value);
|
2022-06-13 01:54:36 +00:00
|
|
|
}
|
2023-10-05 03:58:32 +00:00
|
|
|
uint32_t VAddr32 = (uint32_t)VAddr;
|
2022-09-19 12:06:36 +00:00
|
|
|
if (HaveWriteBP() && g_Debugger->WriteBP32(VAddr32) && MemoryBreakpoint())
|
2022-06-13 01:54:36 +00:00
|
|
|
{
|
|
|
|
return false;
|
|
|
|
}
|
2022-10-10 00:22:17 +00:00
|
|
|
uint8_t * MemoryPtr = (uint8_t *)m_MemoryWriteMap[VAddr32 >> 12];
|
2022-04-04 01:00:27 +00:00
|
|
|
if (MemoryPtr != (uint8_t *)-1)
|
|
|
|
{
|
2022-10-10 00:22:17 +00:00
|
|
|
*(uint32_t *)(MemoryPtr + VAddr32) = Value;
|
2022-04-04 01:00:27 +00:00
|
|
|
return true;
|
|
|
|
}
|
2023-10-04 23:02:45 +00:00
|
|
|
return SW_VAddr32(VAddr32, Value);
|
2010-06-12 02:02:06 +00:00
|
|
|
}
|
|
|
|
|
2022-09-19 12:06:36 +00:00
|
|
|
bool CMipsMemoryVM::SD_Memory(uint64_t VAddr, uint64_t Value)
|
2010-06-12 02:02:06 +00:00
|
|
|
{
|
2023-10-05 03:58:32 +00:00
|
|
|
if ((VAddr & 7) != 0)
|
2022-10-03 08:05:50 +00:00
|
|
|
{
|
2023-10-05 03:58:32 +00:00
|
|
|
m_Reg.TriggerAddressException(VAddr, EXC_WADE);
|
2022-10-03 08:05:50 +00:00
|
|
|
return false;
|
|
|
|
}
|
2023-10-05 03:58:32 +00:00
|
|
|
if ((uint64_t)((int32_t)VAddr) != VAddr)
|
2022-06-13 01:54:36 +00:00
|
|
|
{
|
2023-10-05 03:58:32 +00:00
|
|
|
uint32_t PAddr;
|
|
|
|
bool MemoryUnused;
|
2023-12-14 01:39:24 +00:00
|
|
|
if (!m_TLB.VAddrToPAddr(VAddr, PAddr, MemoryUnused))
|
2023-10-05 03:58:32 +00:00
|
|
|
{
|
|
|
|
m_Reg.TriggerAddressException(VAddr, MemoryUnused ? EXC_WADE : EXC_WMISS);
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
return SD_PhysicalAddress(PAddr, Value);
|
2022-06-13 01:54:36 +00:00
|
|
|
}
|
2023-10-05 03:58:32 +00:00
|
|
|
uint32_t VAddr32 = (uint32_t)VAddr;
|
2022-09-19 12:06:36 +00:00
|
|
|
if (HaveWriteBP() && g_Debugger->WriteBP64(VAddr32) && MemoryBreakpoint())
|
2022-06-13 01:54:36 +00:00
|
|
|
{
|
|
|
|
return false;
|
|
|
|
}
|
2022-10-10 00:22:17 +00:00
|
|
|
uint8_t * MemoryPtr = (uint8_t *)m_MemoryWriteMap[VAddr32 >> 12];
|
2022-04-04 01:00:27 +00:00
|
|
|
if (MemoryPtr != (uint8_t *)-1)
|
|
|
|
{
|
2022-10-10 00:22:17 +00:00
|
|
|
*(uint32_t *)(MemoryPtr + VAddr32 + 0) = *((uint32_t *)(&Value) + 1);
|
|
|
|
*(uint32_t *)(MemoryPtr + VAddr32 + 4) = *((uint32_t *)(&Value));
|
2022-04-04 01:00:27 +00:00
|
|
|
return true;
|
|
|
|
}
|
2023-10-04 23:02:45 +00:00
|
|
|
return SD_VAddr32(VAddr32, Value);
|
2010-06-12 02:02:06 +00:00
|
|
|
}
|
|
|
|
|
2015-11-08 18:04:32 +00:00
|
|
|
bool CMipsMemoryVM::ValidVaddr(uint32_t VAddr) const
|
2010-06-12 02:02:06 +00:00
|
|
|
{
|
2022-05-02 09:40:35 +00:00
|
|
|
return m_TLB_ReadMap[VAddr >> 12] != -1;
|
2010-06-12 02:02:06 +00:00
|
|
|
}
|
|
|
|
|
2022-10-10 00:22:17 +00:00
|
|
|
bool CMipsMemoryVM::VAddrToPAddr(uint32_t VAddr, uint32_t & PAddr) const
|
2010-06-12 02:02:06 +00:00
|
|
|
{
|
2022-05-02 09:40:35 +00:00
|
|
|
if (m_TLB_ReadMap[VAddr >> 12] == -1)
|
2015-11-08 18:04:32 +00:00
|
|
|
{
|
|
|
|
return false;
|
|
|
|
}
|
2022-05-02 10:52:31 +00:00
|
|
|
PAddr = m_TLB_ReadMap[VAddr >> 12] + VAddr;
|
2015-11-08 18:04:32 +00:00
|
|
|
return true;
|
2010-06-12 02:02:06 +00:00
|
|
|
}
|
|
|
|
|
2023-10-18 23:58:25 +00:00
|
|
|
bool CMipsMemoryVM::MemoryBreakpoint()
|
|
|
|
{
|
|
|
|
if (g_Settings->LoadBool(Debugger_SteppingOps))
|
|
|
|
{
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
g_Settings->SaveBool(Debugger_SteppingOps, true);
|
|
|
|
g_Debugger->WaitForStep();
|
|
|
|
if (SkipOp())
|
|
|
|
{
|
|
|
|
// Skip command if instructed by the debugger
|
|
|
|
g_Settings->SaveBool(Debugger_SkipOp, false);
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2023-10-04 23:02:45 +00:00
|
|
|
bool CMipsMemoryVM::LB_VAddr32(uint32_t VAddr, uint8_t & Value)
|
2015-03-29 17:19:28 +00:00
|
|
|
{
|
2022-05-09 00:36:10 +00:00
|
|
|
uint32_t BaseAddress = m_TLB_ReadMap[VAddr >> 12];
|
|
|
|
if (BaseAddress == -1)
|
|
|
|
{
|
2023-11-16 06:41:05 +00:00
|
|
|
m_Reg.TriggerAddressException((int32_t)VAddr, EXC_RMISS);
|
2022-05-09 00:36:10 +00:00
|
|
|
return false;
|
|
|
|
}
|
2023-10-04 23:02:45 +00:00
|
|
|
return LB_PhysicalAddress(BaseAddress + VAddr, Value);
|
|
|
|
}
|
2022-10-17 01:01:54 +00:00
|
|
|
|
2023-10-04 23:02:45 +00:00
|
|
|
bool CMipsMemoryVM::LH_VAddr32(uint32_t VAddr, uint16_t & Value)
|
|
|
|
{
|
|
|
|
uint32_t BaseAddress = m_TLB_ReadMap[VAddr >> 12];
|
|
|
|
if (BaseAddress == -1)
|
|
|
|
{
|
2023-11-16 06:41:05 +00:00
|
|
|
m_Reg.TriggerAddressException((int32_t)VAddr, EXC_RMISS);
|
2023-10-04 23:02:45 +00:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
return LH_PhysicalAddress(BaseAddress + VAddr, Value);
|
|
|
|
}
|
|
|
|
|
|
|
|
bool CMipsMemoryVM::LW_VAddr32(uint32_t VAddr, uint32_t & Value)
|
|
|
|
{
|
|
|
|
uint32_t BaseAddress = m_TLB_ReadMap[VAddr >> 12];
|
|
|
|
if (BaseAddress == -1)
|
|
|
|
{
|
2023-11-16 06:41:05 +00:00
|
|
|
m_Reg.TriggerAddressException((int32_t)VAddr, EXC_RMISS);
|
2023-10-04 23:02:45 +00:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
return LW_PhysicalAddress(BaseAddress + VAddr, Value);
|
|
|
|
}
|
|
|
|
|
|
|
|
bool CMipsMemoryVM::LD_VAddr32(uint32_t VAddr, uint64_t & Value)
|
|
|
|
{
|
|
|
|
uint32_t BaseAddress = m_TLB_ReadMap[VAddr >> 12];
|
|
|
|
if (BaseAddress == -1)
|
|
|
|
{
|
2023-11-16 06:41:05 +00:00
|
|
|
m_Reg.TriggerAddressException((int32_t)VAddr, EXC_RMISS);
|
2023-10-04 23:02:45 +00:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
return LD_PhysicalAddress(BaseAddress + VAddr, Value);
|
|
|
|
}
|
|
|
|
|
|
|
|
bool CMipsMemoryVM::LB_PhysicalAddress(uint32_t PAddr, uint8_t & Value)
|
|
|
|
{
|
2022-10-17 01:01:54 +00:00
|
|
|
uint32_t ReadAddress = PAddr & ~3;
|
|
|
|
uint32_t Value32;
|
|
|
|
switch (PAddr & 0xFFF00000)
|
2022-04-18 11:27:59 +00:00
|
|
|
{
|
2022-10-17 01:01:54 +00:00
|
|
|
case 0x1FC00000: m_PifRamHandler.Read32(ReadAddress, Value32); break;
|
|
|
|
default:
|
2023-10-05 03:58:32 +00:00
|
|
|
if (PAddr < RdramSize())
|
|
|
|
{
|
|
|
|
Value = *(uint8_t *)(m_RDRAM + (PAddr ^ 3));
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
else if (PAddr >= 0x10000000 && PAddr < 0x20000000)
|
2022-04-18 11:27:59 +00:00
|
|
|
{
|
2022-10-17 01:01:54 +00:00
|
|
|
if (!m_RomMemoryHandler.Read32(PAddr, Value32))
|
|
|
|
{
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
Value = ((Value32 >> (((PAddr & 1) ^ 3) << 3)) & 0xff);
|
|
|
|
return true;
|
2022-04-18 11:27:59 +00:00
|
|
|
}
|
2022-10-17 01:01:54 +00:00
|
|
|
else
|
2022-08-01 00:30:07 +00:00
|
|
|
{
|
2022-10-17 01:01:54 +00:00
|
|
|
if (BreakOnUnhandledMemory())
|
|
|
|
{
|
|
|
|
g_Notify->BreakPoint(__FILE__, __LINE__);
|
|
|
|
}
|
|
|
|
Value32 = 0;
|
2022-08-01 00:30:07 +00:00
|
|
|
}
|
2015-11-08 18:04:32 +00:00
|
|
|
}
|
2022-10-17 01:01:54 +00:00
|
|
|
Value = ((Value32 >> (((PAddr & 3) ^ 3) << 3)) & 0xff);
|
2016-06-29 13:40:36 +00:00
|
|
|
return true;
|
2017-08-22 08:08:54 +00:00
|
|
|
}
|
2015-11-08 18:04:32 +00:00
|
|
|
|
2023-10-04 23:02:45 +00:00
|
|
|
bool CMipsMemoryVM::LH_PhysicalAddress(uint32_t PAddr, uint16_t & Value)
|
2016-06-29 13:40:36 +00:00
|
|
|
{
|
2022-10-17 01:01:54 +00:00
|
|
|
uint32_t ReadAddress = PAddr & ~1;
|
|
|
|
uint32_t Value32;
|
|
|
|
switch (PAddr & 0xFFF00000)
|
2022-07-04 07:44:27 +00:00
|
|
|
{
|
2022-10-17 01:01:54 +00:00
|
|
|
case 0x1FC00000: m_PifRamHandler.Read32(ReadAddress, Value32); break;
|
|
|
|
default:
|
2023-10-05 03:58:32 +00:00
|
|
|
if (PAddr < RdramSize())
|
|
|
|
{
|
|
|
|
Value = *(uint16_t *)(m_RDRAM + (PAddr ^ 2));
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
else if (PAddr >= 0x10000000 && PAddr < 0x20000000)
|
2022-07-04 07:44:27 +00:00
|
|
|
{
|
2022-10-17 01:01:54 +00:00
|
|
|
if (!m_RomMemoryHandler.Read32(PAddr, Value32))
|
|
|
|
{
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
Value = ((Value32 >> 16) & 0xffff);
|
|
|
|
return true;
|
2022-07-04 07:44:27 +00:00
|
|
|
}
|
2022-10-17 01:01:54 +00:00
|
|
|
else
|
2022-09-05 01:30:15 +00:00
|
|
|
{
|
2022-10-17 01:01:54 +00:00
|
|
|
if (BreakOnUnhandledMemory())
|
|
|
|
{
|
|
|
|
g_Notify->BreakPoint(__FILE__, __LINE__);
|
|
|
|
}
|
|
|
|
Value32 = 0;
|
2022-09-05 01:30:15 +00:00
|
|
|
}
|
2022-07-04 07:44:27 +00:00
|
|
|
}
|
2022-10-17 01:01:54 +00:00
|
|
|
Value = ((Value32 >> (((PAddr ^ 1) & 1) << 4)) & 0xffff);
|
2022-07-04 07:44:27 +00:00
|
|
|
return true;
|
2010-06-12 02:02:06 +00:00
|
|
|
}
|
|
|
|
|
2023-10-04 23:02:45 +00:00
|
|
|
bool CMipsMemoryVM::LW_PhysicalAddress(uint32_t PAddr, uint32_t & Value)
|
2015-03-29 17:19:28 +00:00
|
|
|
{
|
2022-04-18 11:27:59 +00:00
|
|
|
switch (PAddr & 0xFFF00000)
|
2015-11-08 18:04:32 +00:00
|
|
|
{
|
2022-05-09 00:36:10 +00:00
|
|
|
case 0x03F00000: m_RDRAMRegistersHandler.Read32(PAddr, Value); break;
|
|
|
|
case 0x04000000: m_SPRegistersHandler.Read32(PAddr, Value); break;
|
|
|
|
case 0x04100000: m_DPCommandRegistersHandler.Read32(PAddr, Value); break;
|
|
|
|
case 0x04300000: m_MIPSInterfaceHandler.Read32(PAddr, Value); break;
|
|
|
|
case 0x04400000: m_VideoInterfaceHandler.Read32(PAddr, Value); break;
|
|
|
|
case 0x04500000: m_AudioInterfaceHandler.Read32(PAddr, Value); break;
|
|
|
|
case 0x04600000: m_PeripheralInterfaceHandler.Read32(PAddr, Value); break;
|
|
|
|
case 0x04700000: m_RDRAMInterfaceHandler.Read32(PAddr, Value); break;
|
|
|
|
case 0x04800000: m_SerialInterfaceHandler.Read32(PAddr, Value); break;
|
|
|
|
case 0x05000000: m_CartridgeDomain2Address1Handler.Read32(PAddr, Value); break;
|
|
|
|
case 0x06000000: m_CartridgeDomain1Address1Handler.Read32(PAddr, Value); break;
|
|
|
|
case 0x08000000: m_CartridgeDomain2Address2Handler.Read32(PAddr, Value); break;
|
2022-10-16 22:29:26 +00:00
|
|
|
case 0x13F00000: m_ISViewerHandler.Read32(PAddr, Value); break;
|
2022-05-09 00:36:10 +00:00
|
|
|
case 0x1FC00000: m_PifRamHandler.Read32(PAddr, Value); break;
|
|
|
|
case 0x1FF00000: m_CartridgeDomain1Address3Handler.Read32(PAddr, Value); break;
|
2022-04-18 11:27:59 +00:00
|
|
|
default:
|
2023-10-05 03:58:32 +00:00
|
|
|
if (PAddr < RdramSize())
|
|
|
|
{
|
|
|
|
Value = *(uint32_t *)(m_RDRAM + PAddr);
|
|
|
|
}
|
|
|
|
else if (PAddr >= 0x10000000 && PAddr < 0x20000000)
|
2022-04-18 11:27:59 +00:00
|
|
|
{
|
2022-05-09 00:36:10 +00:00
|
|
|
m_RomMemoryHandler.Read32(PAddr, Value);
|
2022-04-18 11:27:59 +00:00
|
|
|
}
|
2022-10-16 22:29:26 +00:00
|
|
|
else if (BreakOnUnhandledMemory())
|
|
|
|
{
|
|
|
|
g_Notify->BreakPoint(__FILE__, __LINE__);
|
|
|
|
Value = ((PAddr & 0xFFFF) << 16) | (PAddr & 0xFFFF);
|
|
|
|
}
|
2022-04-18 11:27:59 +00:00
|
|
|
else
|
2015-11-08 18:04:32 +00:00
|
|
|
{
|
2022-05-09 00:36:10 +00:00
|
|
|
Value = ((PAddr & 0xFFFF) << 16) | (PAddr & 0xFFFF);
|
2015-11-08 18:04:32 +00:00
|
|
|
}
|
|
|
|
}
|
2016-06-29 13:40:36 +00:00
|
|
|
return true;
|
2017-08-22 08:08:54 +00:00
|
|
|
}
|
2010-06-12 02:02:06 +00:00
|
|
|
|
2023-10-04 23:02:45 +00:00
|
|
|
bool CMipsMemoryVM::LD_PhysicalAddress(uint32_t PAddr, uint64_t & Value)
|
2022-05-09 00:36:10 +00:00
|
|
|
{
|
2023-10-05 03:58:32 +00:00
|
|
|
if (PAddr < RdramSize())
|
|
|
|
{
|
|
|
|
*((uint32_t *)(&Value) + 1) = *(uint32_t *)(m_RDRAM + PAddr);
|
|
|
|
*((uint32_t *)(&Value) + 0) = *(uint32_t *)(m_RDRAM + PAddr + 4);
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
else if (PAddr < 0x800000)
|
2022-05-09 00:36:10 +00:00
|
|
|
{
|
|
|
|
Value = 0;
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
g_Notify->BreakPoint(__FILE__, __LINE__);
|
|
|
|
Value = 0;
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2023-10-04 23:02:45 +00:00
|
|
|
bool CMipsMemoryVM::SB_VAddr32(uint32_t VAddr, uint32_t Value)
|
|
|
|
{
|
|
|
|
uint32_t BaseAddress = m_TLB_WriteMap[VAddr >> 12];
|
|
|
|
if (BaseAddress == -1)
|
|
|
|
{
|
2023-11-16 06:41:05 +00:00
|
|
|
m_Reg.TriggerAddressException((int32_t)VAddr, EXC_WMISS);
|
2023-10-04 23:02:45 +00:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
return SB_PhysicalAddress(BaseAddress + VAddr, Value);
|
|
|
|
}
|
|
|
|
|
|
|
|
bool CMipsMemoryVM::SH_VAddr32(uint32_t VAddr, uint32_t Value)
|
|
|
|
{
|
|
|
|
uint32_t BaseAddress = m_TLB_WriteMap[VAddr >> 12];
|
|
|
|
if (BaseAddress == -1)
|
|
|
|
{
|
2023-11-16 06:41:05 +00:00
|
|
|
m_Reg.TriggerAddressException((int32_t)VAddr, EXC_WMISS);
|
2023-10-04 23:02:45 +00:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
return SH_PhysicalAddress(BaseAddress + VAddr, Value);
|
|
|
|
}
|
|
|
|
|
|
|
|
bool CMipsMemoryVM::SW_VAddr32(uint32_t VAddr, uint32_t Value)
|
|
|
|
{
|
|
|
|
uint32_t BaseAddress = m_TLB_WriteMap[VAddr >> 12];
|
|
|
|
if (BaseAddress == -1)
|
|
|
|
{
|
2023-11-16 06:41:05 +00:00
|
|
|
m_Reg.TriggerAddressException((int32_t)VAddr, EXC_WMISS);
|
2023-10-04 23:02:45 +00:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
return SW_PhysicalAddress(BaseAddress + VAddr, Value);
|
|
|
|
}
|
|
|
|
|
|
|
|
bool CMipsMemoryVM::SD_VAddr32(uint32_t VAddr, uint64_t Value)
|
2015-03-29 17:19:28 +00:00
|
|
|
{
|
2023-08-30 02:05:53 +00:00
|
|
|
uint32_t BaseAddress = m_TLB_WriteMap[VAddr >> 12];
|
2022-05-09 00:36:10 +00:00
|
|
|
if (BaseAddress == -1)
|
|
|
|
{
|
2023-11-16 06:41:05 +00:00
|
|
|
m_Reg.TriggerAddressException((int32_t)VAddr, EXC_WMISS);
|
2022-05-09 00:36:10 +00:00
|
|
|
return false;
|
|
|
|
}
|
2023-10-04 23:02:45 +00:00
|
|
|
return SD_PhysicalAddress(BaseAddress + VAddr, Value);
|
|
|
|
}
|
|
|
|
|
|
|
|
bool CMipsMemoryVM::SB_PhysicalAddress(uint32_t PAddr, uint32_t Value)
|
|
|
|
{
|
2015-11-08 18:04:32 +00:00
|
|
|
switch (PAddr & 0xFFF00000)
|
|
|
|
{
|
|
|
|
case 0x00000000:
|
|
|
|
case 0x00100000:
|
|
|
|
case 0x00200000:
|
|
|
|
case 0x00300000:
|
|
|
|
case 0x00400000:
|
|
|
|
case 0x00500000:
|
|
|
|
case 0x00600000:
|
|
|
|
case 0x00700000:
|
2016-06-29 13:40:36 +00:00
|
|
|
if (PAddr < RdramSize())
|
2015-11-08 18:04:32 +00:00
|
|
|
{
|
2016-06-29 13:40:36 +00:00
|
|
|
g_Recompiler->ClearRecompCode_Phys(PAddr & ~0xFFF, 0xFFC, CRecompiler::Remove_ProtectedMem);
|
2022-10-10 00:22:17 +00:00
|
|
|
*(uint8_t *)(m_RDRAM + (PAddr ^ 3)) = (uint8_t)Value;
|
2015-11-08 18:04:32 +00:00
|
|
|
}
|
2016-06-29 13:40:36 +00:00
|
|
|
break;
|
2022-10-17 06:59:05 +00:00
|
|
|
case 0x04000000: m_SPRegistersHandler.Write32(PAddr & ~3, Value << ((3 - (PAddr & 3)) * 8), 0xFFFFFFFF); break;
|
2022-10-17 01:01:54 +00:00
|
|
|
case 0x1FC00000: m_PifRamHandler.Write32(PAddr & ~3, Value << ((3 - (PAddr & 3)) * 8), 0xFFFFFFFF); break;
|
2016-06-29 13:40:36 +00:00
|
|
|
default:
|
2022-08-08 10:03:16 +00:00
|
|
|
if (PAddr >= 0x10000000 && PAddr < 0x20000000)
|
|
|
|
{
|
|
|
|
m_RomMemoryHandler.Write32(PAddr, Value << ((3 - (PAddr & 3)) * 8), 0xFFFFFFFF);
|
|
|
|
}
|
|
|
|
else if (BreakOnUnhandledMemory())
|
2022-08-01 00:30:07 +00:00
|
|
|
{
|
|
|
|
g_Notify->BreakPoint(__FILE__, __LINE__);
|
|
|
|
}
|
2015-11-08 18:04:32 +00:00
|
|
|
}
|
2016-06-29 13:40:36 +00:00
|
|
|
return true;
|
2010-06-12 02:02:06 +00:00
|
|
|
}
|
|
|
|
|
2023-10-04 23:02:45 +00:00
|
|
|
bool CMipsMemoryVM::SH_PhysicalAddress(uint32_t PAddr, uint32_t Value)
|
2010-06-16 07:31:47 +00:00
|
|
|
{
|
2016-06-29 13:40:36 +00:00
|
|
|
switch (PAddr & 0xFFF00000)
|
2015-11-08 18:04:32 +00:00
|
|
|
{
|
2016-06-29 13:40:36 +00:00
|
|
|
case 0x00000000:
|
|
|
|
case 0x00100000:
|
|
|
|
case 0x00200000:
|
|
|
|
case 0x00300000:
|
|
|
|
case 0x00400000:
|
|
|
|
case 0x00500000:
|
|
|
|
case 0x00600000:
|
|
|
|
case 0x00700000:
|
|
|
|
if (PAddr < RdramSize())
|
|
|
|
{
|
2024-05-09 08:26:28 +00:00
|
|
|
if (CGameSettings::bSMM_StoreInstruc())
|
2022-06-06 02:11:09 +00:00
|
|
|
{
|
|
|
|
g_Recompiler->ClearRecompCode_Phys(PAddr & ~0xFFF, 0x1000, CRecompiler::Remove_ProtectedMem);
|
2024-05-09 08:26:28 +00:00
|
|
|
m_TLB_WriteMap[(0x80000000 + PAddr) >> 12] = PAddr - (0x80000000 + PAddr);
|
|
|
|
m_TLB_WriteMap[(0xA0000000 + PAddr) >> 12] = PAddr - (0xA0000000 + PAddr);
|
2022-08-08 10:03:16 +00:00
|
|
|
*(uint16_t *)(m_RDRAM + (PAddr ^ 2)) = (uint16_t)Value;
|
2022-06-06 02:11:09 +00:00
|
|
|
}
|
2016-06-29 13:40:36 +00:00
|
|
|
}
|
|
|
|
break;
|
2022-10-17 06:59:05 +00:00
|
|
|
case 0x04000000: m_SPRegistersHandler.Write32(PAddr & ~3, Value << ((2 - (PAddr & 2)) * 8), 0xFFFFFFFF); break;
|
2022-10-17 01:01:54 +00:00
|
|
|
case 0x1FC00000: m_PifRamHandler.Write32(PAddr & ~3, Value << ((2 - (PAddr & 2)) * 8), 0xFFFFFFFF); break;
|
2016-06-29 13:40:36 +00:00
|
|
|
default:
|
2022-08-08 10:03:16 +00:00
|
|
|
if (PAddr >= 0x10000000 && PAddr < 0x20000000)
|
|
|
|
{
|
|
|
|
m_RomMemoryHandler.Write32(PAddr, Value << ((2 - (PAddr & 2)) * 8), 0xFFFFFFFF);
|
|
|
|
}
|
|
|
|
else if (BreakOnUnhandledMemory())
|
2022-08-01 00:30:07 +00:00
|
|
|
{
|
|
|
|
g_Notify->BreakPoint(__FILE__, __LINE__);
|
|
|
|
}
|
2016-06-29 13:40:36 +00:00
|
|
|
return false;
|
2015-11-08 18:04:32 +00:00
|
|
|
}
|
|
|
|
|
2016-06-29 13:40:36 +00:00
|
|
|
return true;
|
2010-06-16 07:31:47 +00:00
|
|
|
}
|
|
|
|
|
2023-10-04 23:02:45 +00:00
|
|
|
bool CMipsMemoryVM::SW_PhysicalAddress(uint32_t PAddr, uint32_t Value)
|
2010-06-12 02:02:06 +00:00
|
|
|
{
|
2016-06-29 13:40:36 +00:00
|
|
|
switch (PAddr & 0xFFF00000)
|
2015-11-08 18:04:32 +00:00
|
|
|
{
|
2016-06-29 13:40:36 +00:00
|
|
|
case 0x00000000:
|
|
|
|
case 0x00100000:
|
|
|
|
case 0x00200000:
|
|
|
|
case 0x00300000:
|
|
|
|
case 0x00400000:
|
|
|
|
case 0x00500000:
|
|
|
|
case 0x00600000:
|
|
|
|
case 0x00700000:
|
2022-08-01 00:30:07 +00:00
|
|
|
case 0x00800000:
|
2016-06-29 13:40:36 +00:00
|
|
|
if (PAddr < RdramSize())
|
2015-11-08 18:04:32 +00:00
|
|
|
{
|
2024-05-09 08:26:28 +00:00
|
|
|
if (CGameSettings::bSMM_StoreInstruc())
|
2022-06-06 02:11:09 +00:00
|
|
|
{
|
|
|
|
g_Recompiler->ClearRecompCode_Phys(PAddr & ~0xFFF, 0x1000, CRecompiler::Remove_ProtectedMem);
|
2024-05-09 08:26:28 +00:00
|
|
|
m_TLB_WriteMap[(0x80000000 + PAddr) >> 12] = PAddr - (0x80000000 + PAddr);
|
|
|
|
m_TLB_WriteMap[(0xA0000000 + PAddr) >> 12] = PAddr - (0xA0000000 + PAddr);
|
2022-06-06 02:11:09 +00:00
|
|
|
*(uint32_t *)(m_RDRAM + PAddr) = Value;
|
|
|
|
}
|
2015-11-08 18:04:32 +00:00
|
|
|
}
|
2016-06-29 13:40:36 +00:00
|
|
|
break;
|
2022-02-21 09:17:14 +00:00
|
|
|
case 0x03F00000: m_RDRAMRegistersHandler.Write32(PAddr, Value, 0xFFFFFFFF); break;
|
2022-10-17 06:59:05 +00:00
|
|
|
case 0x04000000: m_SPRegistersHandler.Write32(PAddr, Value, 0xFFFFFFFF); break;
|
2022-02-21 11:26:25 +00:00
|
|
|
case 0x04100000: m_DPCommandRegistersHandler.Write32(PAddr, Value, 0xFFFFFFFF); break;
|
2022-03-04 12:23:30 +00:00
|
|
|
case 0x04300000: m_MIPSInterfaceHandler.Write32(PAddr, Value, 0xFFFFFFFF); break;
|
2022-03-07 23:48:56 +00:00
|
|
|
case 0x04400000: m_VideoInterfaceHandler.Write32(PAddr, Value, 0xFFFFFFFF); break;
|
2022-03-21 00:29:02 +00:00
|
|
|
case 0x04500000: m_AudioInterfaceHandler.Write32(PAddr, Value, 0xFFFFFFFF); break;
|
2022-01-04 21:44:03 +00:00
|
|
|
case 0x04600000: m_PeripheralInterfaceHandler.Write32(PAddr, Value, 0xFFFFFFFF); break;
|
2022-01-04 05:41:52 +00:00
|
|
|
case 0x04700000: m_RDRAMInterfaceHandler.Write32(PAddr, Value, 0xFFFFFFFF); break;
|
2022-03-21 04:52:42 +00:00
|
|
|
case 0x04800000: m_SerialInterfaceHandler.Write32(PAddr, Value, 0xFFFFFFFF); break;
|
2022-03-21 10:27:57 +00:00
|
|
|
case 0x05000000: m_CartridgeDomain2Address1Handler.Write32(PAddr, Value, 0xFFFFFFFF); break;
|
2022-04-25 07:42:07 +00:00
|
|
|
case 0x06000000: m_CartridgeDomain1Address1Handler.Write32(PAddr, Value, 0xFFFFFFFF); break;
|
2022-07-25 12:30:41 +00:00
|
|
|
case 0x08000000:
|
|
|
|
case 0x0fe00000:
|
2022-10-10 00:22:17 +00:00
|
|
|
m_CartridgeDomain2Address2Handler.Write32(PAddr, Value, 0xFFFFFFFF);
|
2022-07-25 12:30:41 +00:00
|
|
|
break;
|
2022-07-18 09:36:34 +00:00
|
|
|
case 0x13F00000: m_ISViewerHandler.Write32(PAddr, Value, 0xFFFFFFFF); break;
|
2022-04-19 00:07:57 +00:00
|
|
|
case 0x1FC00000: m_PifRamHandler.Write32(PAddr, Value, 0xFFFFFFFF); break;
|
2022-04-25 07:42:07 +00:00
|
|
|
case 0x1FF00000: m_CartridgeDomain1Address3Handler.Write32(PAddr, Value, 0xFFFFFFFF); break;
|
2016-06-29 13:40:36 +00:00
|
|
|
default:
|
2022-08-01 00:30:07 +00:00
|
|
|
if (PAddr >= 0x10000000 && PAddr < 0x20000000)
|
2022-04-18 11:27:59 +00:00
|
|
|
{
|
|
|
|
m_RomMemoryHandler.Write32(PAddr, Value, 0xFFFFFFFF);
|
|
|
|
}
|
2022-08-01 00:30:07 +00:00
|
|
|
else if (BreakOnUnhandledMemory())
|
2022-07-18 09:36:34 +00:00
|
|
|
{
|
|
|
|
g_Notify->BreakPoint(__FILE__, __LINE__);
|
|
|
|
}
|
2016-06-29 13:40:36 +00:00
|
|
|
break;
|
2015-11-08 18:04:32 +00:00
|
|
|
}
|
2022-05-09 00:36:10 +00:00
|
|
|
return true;
|
|
|
|
}
|
2016-06-29 13:40:36 +00:00
|
|
|
|
2023-10-04 23:02:45 +00:00
|
|
|
bool CMipsMemoryVM::SD_PhysicalAddress(uint32_t PAddr, uint64_t Value)
|
2022-05-09 00:36:10 +00:00
|
|
|
{
|
2022-05-30 10:50:25 +00:00
|
|
|
switch (PAddr & 0xFFF00000)
|
|
|
|
{
|
|
|
|
case 0x00000000:
|
|
|
|
case 0x00100000:
|
|
|
|
case 0x00200000:
|
|
|
|
case 0x00300000:
|
|
|
|
case 0x00400000:
|
|
|
|
case 0x00500000:
|
|
|
|
case 0x00600000:
|
|
|
|
case 0x00700000:
|
|
|
|
if (PAddr < RdramSize())
|
|
|
|
{
|
|
|
|
g_Recompiler->ClearRecompCode_Phys(PAddr & ~0xFFF, 0xFFC, CRecompiler::Remove_ProtectedMem);
|
|
|
|
*(uint64_t *)(m_RDRAM + PAddr) = Value;
|
|
|
|
}
|
|
|
|
break;
|
2022-10-17 06:59:05 +00:00
|
|
|
case 0x04000000: m_SPRegistersHandler.Write32(PAddr, (int32_t)(Value >> 32), 0xFFFFFFFF); break;
|
2022-05-30 10:50:25 +00:00
|
|
|
default:
|
2022-08-08 10:03:16 +00:00
|
|
|
if (PAddr >= 0x10000000 && PAddr < 0x20000000)
|
|
|
|
{
|
|
|
|
m_RomMemoryHandler.Write32(PAddr, (int32_t)(Value >> 32), 0xFFFFFFFF);
|
|
|
|
}
|
|
|
|
else if (BreakOnUnhandledMemory())
|
2022-08-01 00:30:07 +00:00
|
|
|
{
|
|
|
|
g_Notify->BreakPoint(__FILE__, __LINE__);
|
|
|
|
}
|
2022-05-30 10:50:25 +00:00
|
|
|
return false;
|
|
|
|
}
|
2016-06-29 13:40:36 +00:00
|
|
|
return true;
|
2017-08-22 08:08:54 +00:00
|
|
|
}
|
2010-06-12 02:02:06 +00:00
|
|
|
|
2022-06-06 02:11:09 +00:00
|
|
|
void CMipsMemoryVM::ClearMemoryWriteMap(uint32_t VAddr, uint32_t Length)
|
|
|
|
{
|
2023-08-30 02:05:53 +00:00
|
|
|
uint32_t BaseAddress = m_TLB_WriteMap[VAddr >> 12];
|
2022-06-06 02:11:09 +00:00
|
|
|
if (BaseAddress == -1)
|
|
|
|
{
|
|
|
|
return;
|
|
|
|
}
|
2023-08-30 02:05:53 +00:00
|
|
|
uint32_t PAddr = m_TLB_WriteMap[VAddr >> 12] + VAddr;
|
2023-01-09 03:56:35 +00:00
|
|
|
for (uint32_t i = PAddr, n = (PAddr + (Length & ~0xFFF)) + 0x1000; i < n; i += 0x1000)
|
2022-06-06 02:11:09 +00:00
|
|
|
{
|
|
|
|
m_MemoryWriteMap[(i + 0x80000000) >> 12] = (size_t)-1;
|
|
|
|
m_MemoryWriteMap[(i + 0xA0000000) >> 12] = (size_t)-1;
|
|
|
|
}
|
|
|
|
}
|
2022-05-09 00:36:10 +00:00
|
|
|
|
2016-01-13 11:52:21 +00:00
|
|
|
const char * CMipsMemoryVM::LabelName(uint32_t Address) const
|
2010-06-12 02:02:06 +00:00
|
|
|
{
|
2015-11-08 18:04:32 +00:00
|
|
|
sprintf(m_strLabelName, "0x%08X", Address);
|
|
|
|
return m_strLabelName;
|
2010-06-12 02:02:06 +00:00
|
|
|
}
|
|
|
|
|
2023-11-16 06:41:05 +00:00
|
|
|
void CMipsMemoryVM::TLB_Mapped(uint32_t VAddr, uint32_t Len, uint32_t PAddr, bool bReadOnly)
|
2010-06-12 02:02:06 +00:00
|
|
|
{
|
2023-10-05 03:12:31 +00:00
|
|
|
uint64_t VEnd = VAddr + Len;
|
|
|
|
for (uint64_t Address = VAddr; Address < VEnd; Address += 0x1000)
|
2015-11-08 18:04:32 +00:00
|
|
|
{
|
2023-10-05 03:12:31 +00:00
|
|
|
size_t Index = (size_t)(Address >> 12);
|
2022-11-14 10:50:28 +00:00
|
|
|
if ((Address - VAddr + PAddr) < m_AllocatedRdramSize)
|
|
|
|
{
|
|
|
|
m_MemoryReadMap[Index] = (size_t)((m_RDRAM + (Address - VAddr + PAddr)) - Address);
|
|
|
|
}
|
2023-10-05 03:12:31 +00:00
|
|
|
m_TLB_ReadMap[Index] = (size_t)((Address - VAddr + PAddr) - Address);
|
2015-11-08 18:04:32 +00:00
|
|
|
if (!bReadOnly)
|
|
|
|
{
|
2022-11-14 10:50:28 +00:00
|
|
|
if ((Address - VAddr + PAddr) < m_AllocatedRdramSize)
|
|
|
|
{
|
|
|
|
m_MemoryWriteMap[Index] = (size_t)((m_RDRAM + (Address - VAddr + PAddr)) - Address);
|
|
|
|
}
|
2023-10-05 03:12:31 +00:00
|
|
|
m_TLB_WriteMap[Index] = (size_t)((Address - VAddr + PAddr) - Address);
|
2015-11-08 18:04:32 +00:00
|
|
|
}
|
|
|
|
}
|
2010-06-12 02:02:06 +00:00
|
|
|
}
|
|
|
|
|
2023-11-16 06:41:05 +00:00
|
|
|
void CMipsMemoryVM::TLB_Unmaped(uint32_t Vaddr, uint32_t Len)
|
2010-06-12 02:02:06 +00:00
|
|
|
{
|
2023-10-05 03:12:31 +00:00
|
|
|
uint64_t End = Vaddr + Len;
|
|
|
|
for (uint64_t Address = Vaddr; Address < End && Address >= Vaddr; Address += 0x1000)
|
2015-11-08 18:04:32 +00:00
|
|
|
{
|
2023-10-05 03:12:31 +00:00
|
|
|
size_t Index = (size_t)(Address >> 12);
|
2023-11-16 07:44:15 +00:00
|
|
|
if (Address >= 0x80000000 && Address < 0x80000000 + m_AllocatedRdramSize)
|
|
|
|
{
|
|
|
|
m_MemoryReadMap[Address >> 12] = (size_t)((m_RDRAM + (Address & 0x1FFFFFFF)) - Address);
|
|
|
|
m_MemoryWriteMap[Address >> 12] = (size_t)((m_RDRAM + (Address & 0x1FFFFFFF)) - Address);
|
|
|
|
m_TLB_ReadMap[Address >> 12] = (uint32_t)((Address & 0x1FFFFFFF) - Address);
|
|
|
|
m_TLB_WriteMap[Address >> 12] = (uint32_t)((Address & 0x1FFFFFFF) - Address);
|
|
|
|
}
|
|
|
|
else if (Address >= 0xA0000000 && Address < 0xA0000000 + m_AllocatedRdramSize)
|
|
|
|
{
|
|
|
|
m_MemoryReadMap[Address >> 12] = (size_t)((m_RDRAM + (Address & 0x1FFFFFFF)) - Address);
|
|
|
|
m_MemoryWriteMap[Address >> 12] = (size_t)((m_RDRAM + (Address & 0x1FFFFFFF)) - Address);
|
|
|
|
m_TLB_ReadMap[Address >> 12] = (uint32_t)((Address & 0x1FFFFFFF) - Address);
|
|
|
|
m_TLB_WriteMap[Address >> 12] = (uint32_t)((Address & 0x1FFFFFFF) - Address);
|
|
|
|
}
|
|
|
|
else if (Address >= 0x80000000 && Address < 0xC0000000)
|
|
|
|
{
|
|
|
|
m_MemoryReadMap[Index] = (size_t)-1;
|
|
|
|
m_MemoryWriteMap[Index] = (size_t)-1;
|
|
|
|
m_TLB_ReadMap[Address >> 12] = (uint32_t)((Address & 0x1FFFFFFF) - Address);
|
|
|
|
m_TLB_WriteMap[Address >> 12] = (uint32_t)((Address & 0x1FFFFFFF) - Address);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
m_MemoryReadMap[Index] = (size_t)-1;
|
|
|
|
m_MemoryWriteMap[Index] = (size_t)-1;
|
|
|
|
m_TLB_ReadMap[Index] = (uint32_t)-1;
|
|
|
|
m_TLB_WriteMap[Index] = (uint32_t)-1;
|
|
|
|
}
|
2015-11-08 18:04:32 +00:00
|
|
|
}
|
2010-06-12 02:02:06 +00:00
|
|
|
}
|
|
|
|
|
2015-11-08 18:04:32 +00:00
|
|
|
void CMipsMemoryVM::RdramChanged(CMipsMemoryVM * _this)
|
2010-06-12 02:02:06 +00:00
|
|
|
{
|
2015-11-08 18:04:32 +00:00
|
|
|
const size_t new_size = g_Settings->LoadDword(Game_RDRamSize);
|
|
|
|
const size_t old_size = _this->m_AllocatedRdramSize;
|
|
|
|
|
|
|
|
if (old_size == new_size)
|
|
|
|
{
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
if (old_size > new_size)
|
|
|
|
{
|
2016-01-13 11:52:21 +00:00
|
|
|
DecommitMemory(_this->m_RDRAM + new_size, old_size - new_size);
|
2015-11-08 18:04:32 +00:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2016-01-13 11:52:21 +00:00
|
|
|
void * result = CommitMemory(_this->m_RDRAM + old_size, new_size - old_size, MEM_READWRITE);
|
2021-04-12 11:35:39 +00:00
|
|
|
if (result == nullptr)
|
2015-11-08 18:04:32 +00:00
|
|
|
{
|
2021-05-18 11:51:36 +00:00
|
|
|
WriteTrace(TraceN64System, TraceError, "Failed to allocate extended memory");
|
2015-11-08 18:04:32 +00:00
|
|
|
g_Notify->FatalError(GS(MSG_MEM_ALLOC_ERROR));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (new_size > 0xFFFFFFFFul)
|
2021-05-18 11:51:36 +00:00
|
|
|
{ // Should be unreachable because: size_t new_size = g_Settings->(uint32_t)
|
2015-12-09 11:37:58 +00:00
|
|
|
g_Notify->BreakPoint(__FILE__, __LINE__);
|
2021-05-18 11:51:36 +00:00
|
|
|
} // However, FFFFFFFF also is a limit to RCP addressing, so we care
|
2015-11-08 18:04:32 +00:00
|
|
|
_this->m_AllocatedRdramSize = (uint32_t)new_size;
|
2010-06-12 02:02:06 +00:00
|
|
|
}
|
|
|
|
|
2015-04-28 22:19:02 +00:00
|
|
|
void CMipsMemoryVM::ChangeMiIntrMask()
|
2015-03-29 17:19:28 +00:00
|
|
|
{
|
2015-11-08 18:04:32 +00:00
|
|
|
if ((RegModValue & MI_INTR_MASK_CLR_SP) != 0)
|
|
|
|
{
|
|
|
|
g_Reg->MI_INTR_MASK_REG &= ~MI_INTR_MASK_SP;
|
|
|
|
}
|
|
|
|
if ((RegModValue & MI_INTR_MASK_SET_SP) != 0)
|
|
|
|
{
|
|
|
|
g_Reg->MI_INTR_MASK_REG |= MI_INTR_MASK_SP;
|
|
|
|
}
|
|
|
|
if ((RegModValue & MI_INTR_MASK_CLR_SI) != 0)
|
|
|
|
{
|
|
|
|
g_Reg->MI_INTR_MASK_REG &= ~MI_INTR_MASK_SI;
|
|
|
|
}
|
|
|
|
if ((RegModValue & MI_INTR_MASK_SET_SI) != 0)
|
|
|
|
{
|
|
|
|
g_Reg->MI_INTR_MASK_REG |= MI_INTR_MASK_SI;
|
|
|
|
}
|
|
|
|
if ((RegModValue & MI_INTR_MASK_CLR_AI) != 0)
|
|
|
|
{
|
|
|
|
g_Reg->MI_INTR_MASK_REG &= ~MI_INTR_MASK_AI;
|
|
|
|
}
|
|
|
|
if ((RegModValue & MI_INTR_MASK_SET_AI) != 0)
|
|
|
|
{
|
|
|
|
g_Reg->MI_INTR_MASK_REG |= MI_INTR_MASK_AI;
|
|
|
|
}
|
|
|
|
if ((RegModValue & MI_INTR_MASK_CLR_VI) != 0)
|
|
|
|
{
|
|
|
|
g_Reg->MI_INTR_MASK_REG &= ~MI_INTR_MASK_VI;
|
|
|
|
}
|
|
|
|
if ((RegModValue & MI_INTR_MASK_SET_VI) != 0)
|
|
|
|
{
|
|
|
|
g_Reg->MI_INTR_MASK_REG |= MI_INTR_MASK_VI;
|
|
|
|
}
|
|
|
|
if ((RegModValue & MI_INTR_MASK_CLR_PI) != 0)
|
|
|
|
{
|
|
|
|
g_Reg->MI_INTR_MASK_REG &= ~MI_INTR_MASK_PI;
|
|
|
|
}
|
|
|
|
if ((RegModValue & MI_INTR_MASK_SET_PI) != 0)
|
|
|
|
{
|
|
|
|
g_Reg->MI_INTR_MASK_REG |= MI_INTR_MASK_PI;
|
|
|
|
}
|
|
|
|
if ((RegModValue & MI_INTR_MASK_CLR_DP) != 0)
|
|
|
|
{
|
|
|
|
g_Reg->MI_INTR_MASK_REG &= ~MI_INTR_MASK_DP;
|
|
|
|
}
|
|
|
|
if ((RegModValue & MI_INTR_MASK_SET_DP) != 0)
|
|
|
|
{
|
|
|
|
g_Reg->MI_INTR_MASK_REG |= MI_INTR_MASK_DP;
|
|
|
|
}
|
2017-05-04 08:19:05 +00:00
|
|
|
}
|