2012-11-15 00:39:56 +01:00
|
|
|
#include "stdafx.h"
|
2014-06-02 19:27:24 +02:00
|
|
|
#include <atomic>
|
|
|
|
|
|
2014-06-17 17:44:03 +02:00
|
|
|
#include "Utilities/Log.h"
|
2012-11-15 00:39:56 +01:00
|
|
|
#include "Memory.h"
|
2014-07-11 13:59:13 +02:00
|
|
|
#include "Emu/System.h"
|
|
|
|
|
#include "Ini.h"
|
2012-11-15 00:39:56 +01:00
|
|
|
|
|
|
|
|
MemoryBase Memory;
|
|
|
|
|
|
2014-07-08 16:26:49 +02:00
|
|
|
MemBlockInfo::MemBlockInfo(u64 _addr, u32 _size)
|
|
|
|
|
: MemInfo(_addr, PAGE_4K(_size))
|
|
|
|
|
{
|
|
|
|
|
void* real_addr = (void*)((u64)Memory.GetBaseAddr() + _addr);
|
|
|
|
|
#ifdef _WIN32
|
|
|
|
|
mem = VirtualAlloc(real_addr, size, MEM_COMMIT, PAGE_READWRITE);
|
|
|
|
|
#else
|
|
|
|
|
if (::mprotect(real_addr, size, PROT_READ | PROT_WRITE))
|
|
|
|
|
{
|
|
|
|
|
mem = nullptr;
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
mem = real_addr;
|
|
|
|
|
}
|
|
|
|
|
#endif
|
|
|
|
|
if (mem != real_addr)
|
|
|
|
|
{
|
|
|
|
|
LOG_ERROR(MEMORY, "Memory allocation failed (addr=0x%llx, size=0x%llx)", addr, size);
|
|
|
|
|
Emu.Pause();
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
Memory.RegisterPages(_addr, PAGE_4K(_size));
|
|
|
|
|
memset(mem, 0, size);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void MemBlockInfo::Free()
|
|
|
|
|
{
|
|
|
|
|
if (mem)
|
|
|
|
|
{
|
|
|
|
|
Memory.UnregisterPages(addr, size);
|
|
|
|
|
#ifdef _WIN32
|
|
|
|
|
if (!VirtualFree(mem, size, MEM_DECOMMIT))
|
|
|
|
|
#else
|
|
|
|
|
if (::mprotect(mem, size, PROT_NONE))
|
|
|
|
|
#endif
|
|
|
|
|
{
|
|
|
|
|
LOG_ERROR(MEMORY, "Memory deallocation failed (addr=0x%llx, size=0x%llx)", addr, size);
|
|
|
|
|
Emu.Pause();
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2012-11-15 00:39:56 +01:00
|
|
|
//MemoryBlock
|
2014-07-08 16:26:49 +02:00
|
|
|
MemoryBlock::MemoryBlock() : mem_inf(nullptr)
|
2012-11-15 00:39:56 +01:00
|
|
|
{
|
|
|
|
|
Init();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
MemoryBlock::~MemoryBlock()
|
|
|
|
|
{
|
|
|
|
|
Delete();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void MemoryBlock::Init()
|
|
|
|
|
{
|
|
|
|
|
range_start = 0;
|
|
|
|
|
range_size = 0;
|
|
|
|
|
|
2014-07-12 16:53:36 +02:00
|
|
|
mem = Memory.GetMemFromAddr(0);
|
2012-11-15 00:39:56 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void MemoryBlock::InitMemory()
|
|
|
|
|
{
|
2014-07-12 16:53:36 +02:00
|
|
|
if (!range_size)
|
|
|
|
|
{
|
|
|
|
|
mem = Memory.GetMemFromAddr(range_start);
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
Free();
|
|
|
|
|
mem_inf = new MemBlockInfo(range_start, range_size);
|
|
|
|
|
mem = (u8*)mem_inf->mem;
|
|
|
|
|
}
|
2014-07-08 16:26:49 +02:00
|
|
|
}
|
2013-06-30 10:46:29 +02:00
|
|
|
|
2014-07-08 16:26:49 +02:00
|
|
|
void MemoryBlock::Free()
|
|
|
|
|
{
|
|
|
|
|
if (mem_inf)
|
|
|
|
|
{
|
|
|
|
|
delete mem_inf;
|
|
|
|
|
mem_inf = nullptr;
|
|
|
|
|
}
|
2012-11-15 00:39:56 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void MemoryBlock::Delete()
|
|
|
|
|
{
|
2014-07-08 16:26:49 +02:00
|
|
|
Free();
|
2012-11-15 00:39:56 +01:00
|
|
|
Init();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
u64 MemoryBlock::FixAddr(const u64 addr) const
|
|
|
|
|
{
|
|
|
|
|
return addr - GetStartAddr();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
MemoryBlock* MemoryBlock::SetRange(const u64 start, const u32 size)
|
|
|
|
|
{
|
2014-07-12 16:53:36 +02:00
|
|
|
if (start + size > 0x100000000) return nullptr;
|
|
|
|
|
|
2012-11-15 00:39:56 +01:00
|
|
|
range_start = start;
|
|
|
|
|
range_size = size;
|
|
|
|
|
|
|
|
|
|
InitMemory();
|
|
|
|
|
return this;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
bool MemoryBlock::IsMyAddress(const u64 addr)
|
|
|
|
|
{
|
2014-06-01 11:38:37 +02:00
|
|
|
return mem && addr >= GetStartAddr() && addr < GetEndAddr();
|
2012-11-15 00:39:56 +01:00
|
|
|
}
|
|
|
|
|
|
2014-04-28 09:42:02 +02:00
|
|
|
template <typename T>
|
|
|
|
|
__forceinline const T MemoryBlock::FastRead(const u64 addr) const
|
2012-11-15 00:39:56 +01:00
|
|
|
{
|
2014-04-28 09:42:02 +02:00
|
|
|
volatile const T data = *(const T *)GetMem(addr);
|
2014-01-21 00:34:16 +01:00
|
|
|
return re(data);
|
2012-11-15 00:39:56 +01:00
|
|
|
}
|
|
|
|
|
|
2014-04-28 09:42:02 +02:00
|
|
|
template <>
|
|
|
|
|
__forceinline const u128 MemoryBlock::FastRead<u128>(const u64 addr) const
|
2012-11-15 00:39:56 +01:00
|
|
|
{
|
2014-04-28 09:42:02 +02:00
|
|
|
volatile const u128 data = *(const u128 *)GetMem(addr);
|
2012-11-15 00:39:56 +01:00
|
|
|
u128 ret;
|
2014-01-21 00:34:16 +01:00
|
|
|
ret.lo = re(data.hi);
|
|
|
|
|
ret.hi = re(data.lo);
|
2012-11-15 00:39:56 +01:00
|
|
|
return ret;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
bool MemoryBlock::Read8(const u64 addr, u8* value)
|
|
|
|
|
{
|
|
|
|
|
if(!IsMyAddress(addr))
|
|
|
|
|
{
|
|
|
|
|
*value = 0;
|
|
|
|
|
return false;
|
|
|
|
|
}
|
|
|
|
|
|
2014-04-28 09:42:02 +02:00
|
|
|
*value = FastRead<u8>(FixAddr(addr));
|
2012-11-15 00:39:56 +01:00
|
|
|
return true;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
bool MemoryBlock::Read16(const u64 addr, u16* value)
|
|
|
|
|
{
|
|
|
|
|
if(!IsMyAddress(addr))
|
|
|
|
|
{
|
|
|
|
|
*value = 0;
|
|
|
|
|
return false;
|
|
|
|
|
}
|
|
|
|
|
|
2014-04-28 09:42:02 +02:00
|
|
|
*value = FastRead<u16>(FixAddr(addr));
|
2012-11-15 00:39:56 +01:00
|
|
|
return true;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
bool MemoryBlock::Read32(const u64 addr, u32* value)
|
|
|
|
|
{
|
|
|
|
|
if(!IsMyAddress(addr))
|
|
|
|
|
{
|
|
|
|
|
*value = 0;
|
|
|
|
|
return false;
|
|
|
|
|
}
|
|
|
|
|
|
2014-04-28 09:42:02 +02:00
|
|
|
*value = FastRead<u32>(FixAddr(addr));
|
2012-11-15 00:39:56 +01:00
|
|
|
return true;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
bool MemoryBlock::Read64(const u64 addr, u64* value)
|
|
|
|
|
{
|
|
|
|
|
if(!IsMyAddress(addr))
|
|
|
|
|
{
|
|
|
|
|
*value = 0;
|
|
|
|
|
return false;
|
|
|
|
|
}
|
|
|
|
|
|
2014-04-28 09:42:02 +02:00
|
|
|
*value = FastRead<u64>(FixAddr(addr));
|
2012-11-15 00:39:56 +01:00
|
|
|
return true;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
bool MemoryBlock::Read128(const u64 addr, u128* value)
|
|
|
|
|
{
|
|
|
|
|
if(!IsMyAddress(addr))
|
|
|
|
|
{
|
|
|
|
|
*value = u128::From32(0);
|
|
|
|
|
return false;
|
|
|
|
|
}
|
|
|
|
|
|
2014-04-28 09:42:02 +02:00
|
|
|
*value = FastRead<u128>(FixAddr(addr));
|
2012-11-15 00:39:56 +01:00
|
|
|
return true;
|
|
|
|
|
}
|
|
|
|
|
|
2014-04-28 09:42:02 +02:00
|
|
|
template <typename T>
|
|
|
|
|
__forceinline void MemoryBlock::FastWrite(const u64 addr, const T value)
|
2012-11-15 00:39:56 +01:00
|
|
|
{
|
2014-04-28 09:42:02 +02:00
|
|
|
*(T *)GetMem(addr) = re(value);
|
2012-11-15 00:39:56 +01:00
|
|
|
}
|
|
|
|
|
|
2014-04-28 09:42:02 +02:00
|
|
|
template <>
|
|
|
|
|
__forceinline void MemoryBlock::FastWrite<u128>(const u64 addr, const u128 value)
|
2012-11-15 00:39:56 +01:00
|
|
|
{
|
2014-01-21 00:34:16 +01:00
|
|
|
u128 res;
|
|
|
|
|
res.lo = re(value.hi);
|
|
|
|
|
res.hi = re(value.lo);
|
|
|
|
|
*(u128*)GetMem(addr) = res;
|
2012-11-15 00:39:56 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
bool MemoryBlock::Write8(const u64 addr, const u8 value)
|
|
|
|
|
{
|
2013-12-07 21:56:42 +01:00
|
|
|
if(!IsMyAddress(addr) || IsLocked(addr)) return false;
|
2012-11-15 00:39:56 +01:00
|
|
|
|
2014-04-28 09:42:02 +02:00
|
|
|
FastWrite<u8>(FixAddr(addr), value);
|
2012-11-15 00:39:56 +01:00
|
|
|
return true;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
bool MemoryBlock::Write16(const u64 addr, const u16 value)
|
|
|
|
|
{
|
2013-12-07 21:56:42 +01:00
|
|
|
if(!IsMyAddress(addr) || IsLocked(addr)) return false;
|
2012-11-15 00:39:56 +01:00
|
|
|
|
2014-04-28 09:42:02 +02:00
|
|
|
FastWrite<u16>(FixAddr(addr), value);
|
2012-11-15 00:39:56 +01:00
|
|
|
return true;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
bool MemoryBlock::Write32(const u64 addr, const u32 value)
|
|
|
|
|
{
|
2013-12-07 21:56:42 +01:00
|
|
|
if(!IsMyAddress(addr) || IsLocked(addr)) return false;
|
2012-11-15 00:39:56 +01:00
|
|
|
|
2014-04-28 09:42:02 +02:00
|
|
|
FastWrite<u32>(FixAddr(addr), value);
|
2012-11-15 00:39:56 +01:00
|
|
|
return true;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
bool MemoryBlock::Write64(const u64 addr, const u64 value)
|
|
|
|
|
{
|
2013-12-07 21:56:42 +01:00
|
|
|
if(!IsMyAddress(addr) || IsLocked(addr)) return false;
|
2012-11-15 00:39:56 +01:00
|
|
|
|
2014-04-28 09:42:02 +02:00
|
|
|
FastWrite<u64>(FixAddr(addr), value);
|
2012-11-15 00:39:56 +01:00
|
|
|
return true;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
bool MemoryBlock::Write128(const u64 addr, const u128 value)
|
|
|
|
|
{
|
2013-12-07 21:56:42 +01:00
|
|
|
if(!IsMyAddress(addr) || IsLocked(addr)) return false;
|
2012-11-15 00:39:56 +01:00
|
|
|
|
2014-04-28 09:42:02 +02:00
|
|
|
FastWrite<u128>(FixAddr(addr), value);
|
2012-11-15 00:39:56 +01:00
|
|
|
return true;
|
|
|
|
|
}
|
|
|
|
|
|
2013-11-05 19:12:18 +01:00
|
|
|
bool MemoryBlockLE::Read8(const u64 addr, u8* value)
|
|
|
|
|
{
|
|
|
|
|
if(!IsMyAddress(addr)) return false;
|
|
|
|
|
|
|
|
|
|
*value = *(u8*)GetMem(FixAddr(addr));
|
|
|
|
|
return true;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
bool MemoryBlockLE::Read16(const u64 addr, u16* value)
|
|
|
|
|
{
|
|
|
|
|
if(!IsMyAddress(addr)) return false;
|
|
|
|
|
|
|
|
|
|
*value = *(u16*)GetMem(FixAddr(addr));
|
|
|
|
|
return true;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
bool MemoryBlockLE::Read32(const u64 addr, u32* value)
|
|
|
|
|
{
|
|
|
|
|
if(!IsMyAddress(addr)) return false;
|
|
|
|
|
|
|
|
|
|
*value = *(u32*)GetMem(FixAddr(addr));
|
|
|
|
|
return true;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
bool MemoryBlockLE::Read64(const u64 addr, u64* value)
|
|
|
|
|
{
|
|
|
|
|
if(!IsMyAddress(addr)) return false;
|
|
|
|
|
|
|
|
|
|
*value = *(u64*)GetMem(FixAddr(addr));
|
|
|
|
|
return true;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
bool MemoryBlockLE::Read128(const u64 addr, u128* value)
|
|
|
|
|
{
|
|
|
|
|
if(!IsMyAddress(addr)) return false;
|
|
|
|
|
|
|
|
|
|
*value = *(u128*)GetMem(FixAddr(addr));
|
|
|
|
|
return true;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
bool MemoryBlockLE::Write8(const u64 addr, const u8 value)
|
|
|
|
|
{
|
|
|
|
|
if(!IsMyAddress(addr)) return false;
|
|
|
|
|
|
|
|
|
|
*(u8*)GetMem(FixAddr(addr)) = value;
|
|
|
|
|
return true;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
bool MemoryBlockLE::Write16(const u64 addr, const u16 value)
|
|
|
|
|
{
|
|
|
|
|
if(!IsMyAddress(addr)) return false;
|
|
|
|
|
|
|
|
|
|
*(u16*)GetMem(FixAddr(addr)) = value;
|
|
|
|
|
return true;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
bool MemoryBlockLE::Write32(const u64 addr, const u32 value)
|
|
|
|
|
{
|
|
|
|
|
if(!IsMyAddress(addr)) return false;
|
|
|
|
|
|
|
|
|
|
*(u32*)GetMem(FixAddr(addr)) = value;
|
|
|
|
|
return true;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
bool MemoryBlockLE::Write64(const u64 addr, const u64 value)
|
|
|
|
|
{
|
|
|
|
|
if(!IsMyAddress(addr)) return false;
|
|
|
|
|
|
|
|
|
|
*(u64*)GetMem(FixAddr(addr)) = value;
|
|
|
|
|
return true;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
bool MemoryBlockLE::Write128(const u64 addr, const u128 value)
|
|
|
|
|
{
|
|
|
|
|
if(!IsMyAddress(addr)) return false;
|
|
|
|
|
|
|
|
|
|
*(u128*)GetMem(FixAddr(addr)) = value;
|
|
|
|
|
return true;
|
|
|
|
|
}
|
|
|
|
|
|
2014-07-14 21:15:30 +02:00
|
|
|
// MemoryBase
|
2013-11-19 11:30:58 +01:00
|
|
|
template<> __forceinline u64 MemoryBase::ReverseData<1>(u64 val) { return val; }
|
|
|
|
|
template<> __forceinline u64 MemoryBase::ReverseData<2>(u64 val) { return Reverse16(val); }
|
|
|
|
|
template<> __forceinline u64 MemoryBase::ReverseData<4>(u64 val) { return Reverse32(val); }
|
|
|
|
|
template<> __forceinline u64 MemoryBase::ReverseData<8>(u64 val) { return Reverse64(val); }
|
2014-01-17 17:56:03 +01:00
|
|
|
|
2014-03-28 03:50:30 +01:00
|
|
|
VirtualMemoryBlock::VirtualMemoryBlock() : MemoryBlock(), m_reserve_size(0)
|
2014-01-17 17:56:03 +01:00
|
|
|
{
|
|
|
|
|
}
|
|
|
|
|
|
2014-01-21 21:23:28 +01:00
|
|
|
MemoryBlock* VirtualMemoryBlock::SetRange(const u64 start, const u32 size)
|
|
|
|
|
{
|
|
|
|
|
range_start = start;
|
|
|
|
|
range_size = size;
|
|
|
|
|
|
|
|
|
|
return this;
|
|
|
|
|
}
|
|
|
|
|
|
2014-01-17 17:56:03 +01:00
|
|
|
bool VirtualMemoryBlock::IsInMyRange(const u64 addr)
|
|
|
|
|
{
|
2014-04-28 08:23:29 +02:00
|
|
|
return addr >= GetStartAddr() && addr < GetStartAddr() + GetSize() - GetReservedAmount();
|
2014-01-17 17:56:03 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
bool VirtualMemoryBlock::IsInMyRange(const u64 addr, const u32 size)
|
|
|
|
|
{
|
|
|
|
|
return IsInMyRange(addr) && IsInMyRange(addr + size - 1);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
bool VirtualMemoryBlock::IsMyAddress(const u64 addr)
|
|
|
|
|
{
|
2014-04-10 00:54:32 +02:00
|
|
|
for(u32 i=0; i<m_mapped_memory.size(); ++i)
|
2014-01-17 17:56:03 +01:00
|
|
|
{
|
|
|
|
|
if(addr >= m_mapped_memory[i].addr && addr < m_mapped_memory[i].addr + m_mapped_memory[i].size)
|
|
|
|
|
{
|
|
|
|
|
return true;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return false;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
u64 VirtualMemoryBlock::Map(u64 realaddr, u32 size, u64 addr)
|
|
|
|
|
{
|
|
|
|
|
if(addr)
|
|
|
|
|
{
|
|
|
|
|
if(!IsInMyRange(addr, size) && (IsMyAddress(addr) || IsMyAddress(addr + size - 1)))
|
|
|
|
|
return 0;
|
|
|
|
|
|
2014-04-10 00:54:32 +02:00
|
|
|
m_mapped_memory.emplace_back(addr, realaddr, size);
|
2014-01-17 17:56:03 +01:00
|
|
|
return addr;
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
2014-04-28 08:23:29 +02:00
|
|
|
for(u64 addr = GetStartAddr(); addr <= GetEndAddr() - GetReservedAmount() - size;)
|
2014-01-17 17:56:03 +01:00
|
|
|
{
|
|
|
|
|
bool is_good_addr = true;
|
|
|
|
|
|
2014-01-21 18:55:48 +01:00
|
|
|
// check if address is already mapped
|
2014-04-10 00:54:32 +02:00
|
|
|
for(u32 i=0; i<m_mapped_memory.size(); ++i)
|
2014-01-17 17:56:03 +01: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))
|
|
|
|
|
{
|
|
|
|
|
is_good_addr = false;
|
|
|
|
|
addr = m_mapped_memory[i].addr + m_mapped_memory[i].size;
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if(!is_good_addr) continue;
|
|
|
|
|
|
2014-04-10 00:54:32 +02:00
|
|
|
m_mapped_memory.emplace_back(addr, realaddr, size);
|
2014-01-17 17:56:03 +01:00
|
|
|
|
|
|
|
|
return addr;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2014-01-21 19:29:16 +01:00
|
|
|
u32 VirtualMemoryBlock::UnmapRealAddress(u64 realaddr)
|
2014-01-17 17:56:03 +01:00
|
|
|
{
|
2014-04-10 00:54:32 +02:00
|
|
|
for(u32 i=0; i<m_mapped_memory.size(); ++i)
|
2014-01-17 17:56:03 +01:00
|
|
|
{
|
2014-01-21 18:55:48 +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-01-21 19:29:16 +01:00
|
|
|
u32 size = m_mapped_memory[i].size;
|
2014-04-10 00:54:32 +02:00
|
|
|
m_mapped_memory.erase(m_mapped_memory.begin() + i);
|
2014-01-21 19:29:16 +01:00
|
|
|
return size;
|
2014-01-17 17:56:03 +01:00
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2014-01-21 19:29:16 +01:00
|
|
|
return 0;
|
2014-01-17 17:56:03 +01:00
|
|
|
}
|
|
|
|
|
|
2014-01-21 19:29:16 +01:00
|
|
|
u32 VirtualMemoryBlock::UnmapAddress(u64 addr)
|
2014-01-17 17:56:03 +01:00
|
|
|
{
|
2014-04-10 00:54:32 +02:00
|
|
|
for(u32 i=0; i<m_mapped_memory.size(); ++i)
|
2014-01-17 17:56:03 +01:00
|
|
|
{
|
2014-01-21 18:55:48 +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-01-21 19:29:16 +01:00
|
|
|
u32 size = m_mapped_memory[i].size;
|
2014-04-10 00:54:32 +02:00
|
|
|
m_mapped_memory.erase(m_mapped_memory.begin() + i);
|
2014-01-21 19:29:16 +01:00
|
|
|
return size;
|
2014-01-17 17:56:03 +01:00
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2014-01-21 19:29:16 +01:00
|
|
|
return 0;
|
2014-01-17 17:56:03 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
bool VirtualMemoryBlock::Read8(const u64 addr, u8* value)
|
|
|
|
|
{
|
|
|
|
|
u64 realAddr;
|
2014-05-19 08:12:28 +02:00
|
|
|
if(!getRealAddr(addr, realAddr))
|
|
|
|
|
return false;
|
|
|
|
|
*value = Memory.Read8(realAddr);
|
|
|
|
|
return true;
|
2014-01-17 17:56:03 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
bool VirtualMemoryBlock::Read16(const u64 addr, u16* value)
|
|
|
|
|
{
|
|
|
|
|
u64 realAddr;
|
2014-05-19 08:12:28 +02:00
|
|
|
if(!getRealAddr(addr, realAddr))
|
|
|
|
|
return false;
|
|
|
|
|
*value = Memory.Read16(realAddr);
|
|
|
|
|
return true;
|
2014-01-17 17:56:03 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
bool VirtualMemoryBlock::Read32(const u64 addr, u32* value)
|
|
|
|
|
{
|
2014-05-19 08:12:28 +02:00
|
|
|
u64 realAddr;
|
|
|
|
|
if (!getRealAddr(addr, realAddr))
|
2014-04-28 09:15:37 +02:00
|
|
|
return false;
|
|
|
|
|
*value = Memory.Read32(realAddr);
|
|
|
|
|
return true;
|
2014-01-17 17:56:03 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
bool VirtualMemoryBlock::Read64(const u64 addr, u64* value)
|
|
|
|
|
{
|
|
|
|
|
u64 realAddr;
|
2014-05-19 08:12:28 +02:00
|
|
|
if(!getRealAddr(addr, realAddr))
|
|
|
|
|
return false;
|
|
|
|
|
*value = Memory.Read64(realAddr);
|
|
|
|
|
return true;
|
2014-01-17 17:56:03 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
bool VirtualMemoryBlock::Read128(const u64 addr, u128* value)
|
|
|
|
|
{
|
|
|
|
|
u64 realAddr;
|
2014-05-19 08:12:28 +02:00
|
|
|
if(!getRealAddr(addr, realAddr))
|
|
|
|
|
return false;
|
|
|
|
|
*value = Memory.Read128(realAddr);
|
|
|
|
|
return true;
|
2014-01-17 17:56:03 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
bool VirtualMemoryBlock::Write8(const u64 addr, const u8 value)
|
|
|
|
|
{
|
|
|
|
|
u64 realAddr;
|
2014-05-19 08:12:28 +02:00
|
|
|
if(!getRealAddr(addr, realAddr))
|
|
|
|
|
return false;
|
|
|
|
|
Memory.Write8(realAddr, value);
|
|
|
|
|
return true;
|
2014-01-17 17:56:03 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
bool VirtualMemoryBlock::Write16(const u64 addr, const u16 value)
|
|
|
|
|
{
|
|
|
|
|
u64 realAddr;
|
2014-05-19 08:12:28 +02:00
|
|
|
if(!getRealAddr(addr, realAddr))
|
|
|
|
|
return false;
|
|
|
|
|
Memory.Write16(realAddr, value);
|
|
|
|
|
return true;
|
2014-01-17 17:56:03 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
bool VirtualMemoryBlock::Write32(const u64 addr, const u32 value)
|
|
|
|
|
{
|
|
|
|
|
u64 realAddr;
|
2014-05-19 08:12:28 +02:00
|
|
|
if(!getRealAddr(addr, realAddr))
|
|
|
|
|
return false;
|
|
|
|
|
Memory.Write32(realAddr, value);
|
|
|
|
|
return true;
|
2014-01-17 17:56:03 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
bool VirtualMemoryBlock::Write64(const u64 addr, const u64 value)
|
|
|
|
|
{
|
|
|
|
|
u64 realAddr;
|
2014-05-19 08:12:28 +02:00
|
|
|
if(!getRealAddr(addr, realAddr))
|
|
|
|
|
return false;
|
|
|
|
|
Memory.Write64(realAddr, value);
|
|
|
|
|
return true;
|
2014-01-17 17:56:03 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
bool VirtualMemoryBlock::Write128(const u64 addr, const u128 value)
|
|
|
|
|
{
|
|
|
|
|
u64 realAddr;
|
2014-05-19 08:12:28 +02:00
|
|
|
if(!getRealAddr(addr, realAddr))
|
|
|
|
|
return false;
|
|
|
|
|
Memory.Write128(realAddr, value);
|
|
|
|
|
return true;
|
2014-01-17 17:56:03 +01:00
|
|
|
}
|
|
|
|
|
|
2014-05-19 08:12:28 +02:00
|
|
|
bool VirtualMemoryBlock::getRealAddr(u64 addr, u64& result)
|
2014-01-17 17:56:03 +01:00
|
|
|
{
|
2014-04-10 00:54:32 +02:00
|
|
|
for(u32 i=0; i<m_mapped_memory.size(); ++i)
|
2014-01-17 17:56:03 +01:00
|
|
|
{
|
|
|
|
|
if(addr >= m_mapped_memory[i].addr && addr < m_mapped_memory[i].addr + m_mapped_memory[i].size)
|
|
|
|
|
{
|
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
|
|
|
}
|
|
|
|
|
|
2014-01-18 22:36:22 +01:00
|
|
|
u64 VirtualMemoryBlock::getMappedAddress(u64 realAddress)
|
|
|
|
|
{
|
2014-04-10 00:54:32 +02:00
|
|
|
for(u32 i=0; i<m_mapped_memory.size(); ++i)
|
2014-01-18 22:36:22 +01:00
|
|
|
{
|
|
|
|
|
if(realAddress >= m_mapped_memory[i].realAddress && realAddress < m_mapped_memory[i].realAddress + m_mapped_memory[i].size)
|
|
|
|
|
{
|
|
|
|
|
return m_mapped_memory[i].addr + (realAddress - m_mapped_memory[i].realAddress);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
|
}
|
|
|
|
|
|
2014-01-17 17:56:03 +01:00
|
|
|
void VirtualMemoryBlock::Delete()
|
|
|
|
|
{
|
2014-04-10 00:54:32 +02:00
|
|
|
m_mapped_memory.clear();
|
2014-01-17 17:56:03 +01:00
|
|
|
|
|
|
|
|
MemoryBlock::Delete();
|
2014-01-21 18:55:48 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
bool VirtualMemoryBlock::Reserve(u32 size)
|
|
|
|
|
{
|
2014-04-28 08:23:29 +02:00
|
|
|
if(size + GetReservedAmount() > GetEndAddr() - GetStartAddr())
|
2014-01-21 18:55:48 +01:00
|
|
|
return false;
|
|
|
|
|
|
|
|
|
|
m_reserve_size += size;
|
|
|
|
|
return true;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
bool VirtualMemoryBlock::Unreserve(u32 size)
|
|
|
|
|
{
|
2014-04-28 08:23:29 +02: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;
|
2014-04-10 00:54:32 +02:00
|
|
|
}
|