2012-11-15 00:39:56 +01:00
|
|
|
#include "stdafx.h"
|
2014-06-17 17:44:03 +02:00
|
|
|
#include "Utilities/Log.h"
|
2014-08-28 18:29:05 +02:00
|
|
|
#include "Memory.h"
|
2014-08-22 23:15:02 +02:00
|
|
|
|
2015-07-11 22:44:53 +02:00
|
|
|
VirtualMemoryBlock RSXIOMem;
|
2012-11-15 00:39:56 +01:00
|
|
|
|
2015-07-11 22:44:53 +02:00
|
|
|
VirtualMemoryBlock* VirtualMemoryBlock::SetRange(const u32 start, const u32 size)
|
2014-08-22 16:21:55 +02:00
|
|
|
{
|
2015-07-11 22:44:53 +02:00
|
|
|
m_range_start = start;
|
|
|
|
|
m_range_size = size;
|
2014-01-21 21:23:28 +01:00
|
|
|
|
|
|
|
|
return this;
|
|
|
|
|
}
|
|
|
|
|
|
2015-02-12 21:10:25 +01:00
|
|
|
bool VirtualMemoryBlock::IsInMyRange(const u32 addr, const u32 size)
|
2014-01-17 17:56:03 +01:00
|
|
|
{
|
2015-07-11 22:44:53 +02:00
|
|
|
return addr >= m_range_start && addr + size - 1 <= m_range_start + m_range_size - 1 - GetReservedAmount();
|
2014-01-17 17:56:03 +01:00
|
|
|
}
|
|
|
|
|
|
2015-02-12 21:10:25 +01:00
|
|
|
u32 VirtualMemoryBlock::Map(u32 realaddr, u32 size)
|
2014-01-17 17:56:03 +01:00
|
|
|
{
|
2015-02-12 21:10:25 +01:00
|
|
|
assert(size);
|
2014-01-17 17:56:03 +01:00
|
|
|
|
2015-07-11 22:44:53 +02:00
|
|
|
for (u32 addr = m_range_start; addr <= m_range_start + m_range_size - 1 - GetReservedAmount() - size;)
|
2014-01-17 17:56:03 +01:00
|
|
|
{
|
2014-10-04 16:43:46 +02:00
|
|
|
bool is_good_addr = true;
|
2014-01-17 17:56:03 +01:00
|
|
|
|
2014-10-04 16:43:46 +02:00
|
|
|
// check if address is already mapped
|
|
|
|
|
for (u32 i = 0; i<m_mapped_memory.size(); ++i)
|
2014-01-17 17:56:03 +01:00
|
|
|
{
|
2014-10-04 16:43:46 +02:00
|
|
|
if ((addr >= m_mapped_memory[i].addr && addr < m_mapped_memory[i].addr + m_mapped_memory[i].size) ||
|
|
|
|
|
(m_mapped_memory[i].addr >= addr && m_mapped_memory[i].addr < addr + size))
|
2014-01-17 17:56:03 +01:00
|
|
|
{
|
2014-10-04 16:43:46 +02:00
|
|
|
is_good_addr = false;
|
|
|
|
|
addr = m_mapped_memory[i].addr + m_mapped_memory[i].size;
|
|
|
|
|
break;
|
2014-01-17 17:56:03 +01:00
|
|
|
}
|
2014-10-04 16:43:46 +02:00
|
|
|
}
|
2014-01-17 17:56:03 +01:00
|
|
|
|
2014-10-04 16:43:46 +02:00
|
|
|
if (!is_good_addr) continue;
|
2014-01-17 17:56:03 +01:00
|
|
|
|
2014-10-04 16:43:46 +02:00
|
|
|
m_mapped_memory.emplace_back(addr, realaddr, size);
|
2014-01-17 17:56:03 +01:00
|
|
|
|
2014-10-04 16:43:46 +02:00
|
|
|
return addr;
|
2014-01-17 17:56:03 +01:00
|
|
|
}
|
2014-10-04 16:43:46 +02:00
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
|
}
|
|
|
|
|
|
2015-02-12 21:10:25 +01:00
|
|
|
bool VirtualMemoryBlock::Map(u32 realaddr, u32 size, u32 addr)
|
2014-10-04 16:43:46 +02:00
|
|
|
{
|
2015-02-12 21:10:25 +01:00
|
|
|
assert(size);
|
|
|
|
|
|
|
|
|
|
if (!IsInMyRange(addr, size))
|
|
|
|
|
{
|
2014-10-04 16:43:46 +02:00
|
|
|
return false;
|
2015-02-12 21:10:25 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
for (u32 i = 0; i<m_mapped_memory.size(); ++i)
|
|
|
|
|
{
|
|
|
|
|
if (addr >= m_mapped_memory[i].addr && addr + size - 1 <= m_mapped_memory[i].addr + m_mapped_memory[i].size - 1)
|
|
|
|
|
{
|
|
|
|
|
return false;
|
|
|
|
|
}
|
|
|
|
|
}
|
2014-10-04 16:43:46 +02:00
|
|
|
|
|
|
|
|
m_mapped_memory.emplace_back(addr, realaddr, size);
|
|
|
|
|
return true;
|
2014-01-17 17:56:03 +01:00
|
|
|
}
|
|
|
|
|
|
2015-02-12 21:10:25 +01:00
|
|
|
bool VirtualMemoryBlock::UnmapRealAddress(u32 realaddr, u32& size)
|
2014-01-17 17:56:03 +01:00
|
|
|
{
|
2014-11-19 15:16:30 +01:00
|
|
|
for (u32 i = 0; i<m_mapped_memory.size(); ++i)
|
2014-01-17 17:56:03 +01:00
|
|
|
{
|
2014-11-19 15:16:30 +01:00
|
|
|
if (m_mapped_memory[i].realAddress == realaddr && IsInMyRange(m_mapped_memory[i].addr, m_mapped_memory[i].size))
|
2014-01-17 17:56:03 +01:00
|
|
|
{
|
2014-10-04 16:43:46 +02:00
|
|
|
size = m_mapped_memory[i].size;
|
2014-04-10 00:54:32 +02:00
|
|
|
m_mapped_memory.erase(m_mapped_memory.begin() + i);
|
2014-10-04 16:43:46 +02:00
|
|
|
return true;
|
2014-01-17 17:56:03 +01:00
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2014-10-04 16:43:46 +02:00
|
|
|
return false;
|
2014-01-17 17:56:03 +01:00
|
|
|
}
|
|
|
|
|
|
2015-02-12 21:10:25 +01:00
|
|
|
bool VirtualMemoryBlock::UnmapAddress(u32 addr, u32& size)
|
2014-01-17 17:56:03 +01:00
|
|
|
{
|
2014-11-19 15:16:30 +01:00
|
|
|
for (u32 i = 0; i<m_mapped_memory.size(); ++i)
|
2014-01-17 17:56:03 +01:00
|
|
|
{
|
2014-11-19 15:16:30 +01:00
|
|
|
if (m_mapped_memory[i].addr == addr && IsInMyRange(m_mapped_memory[i].addr, m_mapped_memory[i].size))
|
2014-01-17 17:56:03 +01:00
|
|
|
{
|
2014-10-04 16:43:46 +02:00
|
|
|
size = m_mapped_memory[i].size;
|
2014-04-10 00:54:32 +02:00
|
|
|
m_mapped_memory.erase(m_mapped_memory.begin() + i);
|
2014-10-04 16:43:46 +02:00
|
|
|
return true;
|
2014-01-17 17:56:03 +01:00
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2014-10-04 16:43:46 +02:00
|
|
|
return false;
|
2014-01-17 17:56:03 +01:00
|
|
|
}
|
|
|
|
|
|
2015-02-12 21:10:25 +01:00
|
|
|
bool VirtualMemoryBlock::Read32(const u32 addr, u32* value)
|
2014-01-17 17:56:03 +01:00
|
|
|
{
|
2015-02-12 21:10:25 +01:00
|
|
|
u32 realAddr;
|
2014-05-19 08:12:28 +02:00
|
|
|
if (!getRealAddr(addr, realAddr))
|
2014-04-28 09:15:37 +02:00
|
|
|
return false;
|
2015-06-19 17:49:38 +02:00
|
|
|
*value = vm::ps3::read32(realAddr);
|
2014-04-28 09:15:37 +02:00
|
|
|
return true;
|
2014-01-17 17:56:03 +01:00
|
|
|
}
|
|
|
|
|
|
2015-02-12 21:10:25 +01:00
|
|
|
bool VirtualMemoryBlock::Write32(const u32 addr, const u32 value)
|
2014-01-17 17:56:03 +01:00
|
|
|
{
|
2015-02-12 21:10:25 +01:00
|
|
|
u32 realAddr;
|
2014-11-19 15:16:30 +01:00
|
|
|
if (!getRealAddr(addr, realAddr))
|
2014-05-19 08:12:28 +02:00
|
|
|
return false;
|
2015-06-19 17:49:38 +02:00
|
|
|
vm::ps3::write32(realAddr, value);
|
2014-05-19 08:12:28 +02:00
|
|
|
return true;
|
2014-01-17 17:56:03 +01:00
|
|
|
}
|
|
|
|
|
|
2015-02-12 21:10:25 +01:00
|
|
|
bool VirtualMemoryBlock::getRealAddr(u32 addr, u32& result)
|
2014-01-17 17:56:03 +01:00
|
|
|
{
|
2014-11-19 15:16:30 +01:00
|
|
|
for (u32 i = 0; i<m_mapped_memory.size(); ++i)
|
2014-01-17 17:56:03 +01:00
|
|
|
{
|
2014-11-19 15:16:30 +01:00
|
|
|
if (addr >= m_mapped_memory[i].addr && addr < m_mapped_memory[i].addr + m_mapped_memory[i].size)
|
2014-01-17 17:56:03 +01:00
|
|
|
{
|
2014-05-19 08:12:28 +02:00
|
|
|
result = m_mapped_memory[i].realAddress + (addr - m_mapped_memory[i].addr);
|
|
|
|
|
return true;
|
2014-01-17 17:56:03 +01:00
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2014-05-19 08:12:28 +02:00
|
|
|
return false;
|
2014-01-17 17:56:03 +01:00
|
|
|
}
|
|
|
|
|
|
2015-02-12 21:10:25 +01:00
|
|
|
u32 VirtualMemoryBlock::getMappedAddress(u32 realAddress)
|
2014-01-18 22:36:22 +01:00
|
|
|
{
|
2014-11-19 15:16:30 +01:00
|
|
|
for (u32 i = 0; i<m_mapped_memory.size(); ++i)
|
2014-01-18 22:36:22 +01:00
|
|
|
{
|
2014-11-19 15:16:30 +01:00
|
|
|
if (realAddress >= m_mapped_memory[i].realAddress && realAddress < m_mapped_memory[i].realAddress + m_mapped_memory[i].size)
|
2014-01-18 22:36:22 +01:00
|
|
|
{
|
|
|
|
|
return m_mapped_memory[i].addr + (realAddress - m_mapped_memory[i].realAddress);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
|
}
|
|
|
|
|
|
2014-01-21 18:55:48 +01:00
|
|
|
bool VirtualMemoryBlock::Reserve(u32 size)
|
|
|
|
|
{
|
2015-07-11 22:44:53 +02:00
|
|
|
if (size + GetReservedAmount() > m_range_size)
|
2014-01-21 18:55:48 +01:00
|
|
|
return false;
|
|
|
|
|
|
|
|
|
|
m_reserve_size += size;
|
|
|
|
|
return true;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
bool VirtualMemoryBlock::Unreserve(u32 size)
|
|
|
|
|
{
|
2014-11-19 15:16:30 +01:00
|
|
|
if (size > GetReservedAmount())
|
2014-01-21 18:55:48 +01:00
|
|
|
return false;
|
|
|
|
|
|
|
|
|
|
m_reserve_size -= size;
|
|
|
|
|
return true;
|
|
|
|
|
}
|
|
|
|
|
|
2014-04-28 08:23:29 +02:00
|
|
|
u32 VirtualMemoryBlock::GetReservedAmount()
|
2014-01-21 18:55:48 +01:00
|
|
|
{
|
|
|
|
|
return m_reserve_size;
|
2015-02-07 00:39:51 +01:00
|
|
|
}
|