2012-11-15 00:39:56 +01:00
|
|
|
#include "stdafx.h"
|
2014-06-17 17:44:03 +02:00
|
|
|
#include "Utilities/Log.h"
|
2014-07-11 13:59:13 +02:00
|
|
|
#include "Emu/System.h"
|
2014-08-28 18:29:05 +02:00
|
|
|
#include "Memory.h"
|
2014-08-30 19:51:00 +02:00
|
|
|
#include "Emu/Cell/RawSPUThread.h"
|
2014-08-22 23:15:02 +02:00
|
|
|
|
2012-11-15 00:39:56 +01:00
|
|
|
MemoryBase Memory;
|
|
|
|
|
|
2014-08-22 16:21:55 +02:00
|
|
|
u32 MemoryBase::InitRawSPU(MemoryBlock* raw_spu)
|
|
|
|
|
{
|
2014-09-05 01:23:36 +02:00
|
|
|
LV2_LOCK(0);
|
2014-08-22 16:21:55 +02:00
|
|
|
|
|
|
|
|
u32 index;
|
|
|
|
|
for (index = 0; index < sizeof(RawSPUMem) / sizeof(RawSPUMem[0]); index++)
|
|
|
|
|
{
|
|
|
|
|
if (!RawSPUMem[index])
|
|
|
|
|
{
|
|
|
|
|
RawSPUMem[index] = raw_spu;
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
MemoryBlocks.push_back(raw_spu->SetRange(RAW_SPU_BASE_ADDR + RAW_SPU_OFFSET * index, RAW_SPU_PROB_OFFSET));
|
|
|
|
|
return index;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void MemoryBase::CloseRawSPU(MemoryBlock* raw_spu, const u32 num)
|
|
|
|
|
{
|
2014-09-05 01:23:36 +02:00
|
|
|
LV2_LOCK(0);
|
2014-08-22 16:21:55 +02:00
|
|
|
|
|
|
|
|
for (int i = 0; i < MemoryBlocks.size(); ++i)
|
|
|
|
|
{
|
|
|
|
|
if (MemoryBlocks[i] == raw_spu)
|
|
|
|
|
{
|
|
|
|
|
MemoryBlocks.erase(MemoryBlocks.begin() + i);
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
if (num < sizeof(RawSPUMem) / sizeof(RawSPUMem[0])) RawSPUMem[num] = nullptr;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void MemoryBase::Init(MemoryType type)
|
|
|
|
|
{
|
2014-09-05 01:23:36 +02:00
|
|
|
LV2_LOCK(0);
|
2014-08-22 16:21:55 +02:00
|
|
|
|
|
|
|
|
if (m_inited) return;
|
|
|
|
|
m_inited = true;
|
|
|
|
|
|
|
|
|
|
memset(RawSPUMem, 0, sizeof(RawSPUMem));
|
|
|
|
|
|
2015-02-13 16:26:42 +01:00
|
|
|
LOG_NOTICE(MEMORY, "Initializing memory: g_base_addr = 0x%llx, g_priv_addr = 0x%llx", (u64)vm::g_base_addr, (u64)vm::g_priv_addr);
|
2015-02-07 16:35:54 +01:00
|
|
|
|
2014-08-22 16:21:55 +02:00
|
|
|
#ifdef _WIN32
|
2015-02-07 16:35:54 +01:00
|
|
|
if (!vm::g_base_addr || !vm::g_priv_addr)
|
2014-08-22 16:21:55 +02:00
|
|
|
#else
|
2015-02-07 16:35:54 +01:00
|
|
|
if ((s64)vm::g_base_addr == (s64)-1 || (s64)vm::g_priv_addr == (s64)-1)
|
2014-08-22 16:21:55 +02:00
|
|
|
#endif
|
|
|
|
|
{
|
|
|
|
|
LOG_ERROR(MEMORY, "Initializing memory failed");
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
switch (type)
|
|
|
|
|
{
|
|
|
|
|
case Memory_PS3:
|
2015-02-13 15:04:03 +01:00
|
|
|
MemoryBlocks.push_back(MainMem.SetRange(0x00010000, 0x1FFF0000));
|
|
|
|
|
MemoryBlocks.push_back(UserMemory = Userspace.SetRange(0x20000000, 0x10000000));
|
2014-08-22 16:21:55 +02:00
|
|
|
MemoryBlocks.push_back(RSXFBMem.SetRange(0xC0000000, 0x10000000));
|
|
|
|
|
MemoryBlocks.push_back(StackMem.SetRange(0xD0000000, 0x10000000));
|
|
|
|
|
break;
|
|
|
|
|
|
|
|
|
|
case Memory_PSV:
|
|
|
|
|
MemoryBlocks.push_back(PSV.RAM.SetRange(0x81000000, 0x10000000));
|
2015-02-01 08:09:24 +01:00
|
|
|
MemoryBlocks.push_back(UserMemory = PSV.Userspace.SetRange(0x91000000, 0x2F000000));
|
2014-08-22 16:21:55 +02:00
|
|
|
break;
|
|
|
|
|
|
|
|
|
|
case Memory_PSP:
|
|
|
|
|
MemoryBlocks.push_back(PSP.Scratchpad.SetRange(0x00010000, 0x00004000));
|
|
|
|
|
MemoryBlocks.push_back(PSP.VRAM.SetRange(0x04000000, 0x00200000));
|
|
|
|
|
MemoryBlocks.push_back(PSP.RAM.SetRange(0x08000000, 0x02000000));
|
|
|
|
|
MemoryBlocks.push_back(PSP.Kernel.SetRange(0x88000000, 0x00800000));
|
|
|
|
|
MemoryBlocks.push_back(UserMemory = PSP.Userspace.SetRange(0x08800000, 0x01800000));
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
LOG_NOTICE(MEMORY, "Memory initialized.");
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void MemoryBase::Close()
|
|
|
|
|
{
|
2014-09-05 01:23:36 +02:00
|
|
|
LV2_LOCK(0);
|
2014-08-22 16:21:55 +02:00
|
|
|
|
|
|
|
|
if (!m_inited) return;
|
|
|
|
|
m_inited = false;
|
|
|
|
|
|
|
|
|
|
LOG_NOTICE(MEMORY, "Closing memory...");
|
|
|
|
|
|
|
|
|
|
for (auto block : MemoryBlocks)
|
|
|
|
|
{
|
|
|
|
|
block->Delete();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
RSXIOMem.Delete();
|
|
|
|
|
|
|
|
|
|
MemoryBlocks.clear();
|
|
|
|
|
}
|
|
|
|
|
|
2015-02-15 18:13:06 +01:00
|
|
|
bool MemoryBase::WriteMMIO32(u32 addr, const u32 data)
|
2014-08-25 16:56:13 +02:00
|
|
|
{
|
2015-01-02 00:41:29 +01:00
|
|
|
LV2_LOCK(0);
|
2014-08-25 16:56:13 +02:00
|
|
|
|
2015-01-02 00:41:29 +01:00
|
|
|
if (RawSPUMem[(addr - RAW_SPU_BASE_ADDR) / RAW_SPU_OFFSET] && ((RawSPUThread*)RawSPUMem[(addr - RAW_SPU_BASE_ADDR) / RAW_SPU_OFFSET])->Write32(addr, data))
|
|
|
|
|
{
|
2015-02-15 18:13:06 +01:00
|
|
|
return true;
|
2014-08-25 16:56:13 +02:00
|
|
|
}
|
|
|
|
|
|
2015-02-15 18:13:06 +01:00
|
|
|
return false;
|
2014-08-25 16:56:13 +02:00
|
|
|
}
|
|
|
|
|
|
2015-02-15 18:13:06 +01:00
|
|
|
bool MemoryBase::ReadMMIO32(u32 addr, u32& result)
|
2014-08-25 16:56:13 +02:00
|
|
|
{
|
2015-01-02 00:41:29 +01:00
|
|
|
LV2_LOCK(0);
|
2014-08-25 16:56:13 +02:00
|
|
|
|
2015-02-15 18:13:06 +01:00
|
|
|
if (RawSPUMem[(addr - RAW_SPU_BASE_ADDR) / RAW_SPU_OFFSET] && ((RawSPUThread*)RawSPUMem[(addr - RAW_SPU_BASE_ADDR) / RAW_SPU_OFFSET])->Read32(addr, &result))
|
2015-01-02 00:41:29 +01:00
|
|
|
{
|
2015-02-15 18:13:06 +01:00
|
|
|
return true;
|
2014-08-25 16:56:13 +02:00
|
|
|
}
|
|
|
|
|
|
2015-02-15 18:13:06 +01:00
|
|
|
return false;
|
2014-08-25 16:56:13 +02:00
|
|
|
}
|
|
|
|
|
|
2015-02-12 21:10:25 +01:00
|
|
|
bool MemoryBase::Map(const u32 addr, const u32 size)
|
2014-08-22 16:21:55 +02:00
|
|
|
{
|
2015-02-12 21:10:25 +01:00
|
|
|
assert(size && (size | addr) % 4096 == 0);
|
|
|
|
|
|
2014-09-05 01:23:36 +02:00
|
|
|
LV2_LOCK(0);
|
2014-08-22 16:21:55 +02:00
|
|
|
|
2015-02-12 21:10:25 +01:00
|
|
|
for (u32 i = addr / 4096; i < addr / 4096 + size / 4096; i++)
|
2014-08-22 16:21:55 +02:00
|
|
|
{
|
2015-02-13 15:04:03 +01:00
|
|
|
if (vm::check_addr(i * 4096, 4096))
|
2014-09-02 23:48:44 +02:00
|
|
|
{
|
2015-02-12 21:10:25 +01:00
|
|
|
return false;
|
2014-09-02 23:48:44 +02:00
|
|
|
}
|
|
|
|
|
}
|
2014-08-22 16:21:55 +02:00
|
|
|
|
2014-09-02 23:48:44 +02:00
|
|
|
MemoryBlocks.push_back((new MemoryBlock())->SetRange(addr, size));
|
2015-02-07 16:35:54 +01:00
|
|
|
|
2015-02-12 21:10:25 +01:00
|
|
|
LOG_WARNING(MEMORY, "Memory mapped at 0x%x: size=0x%x", addr, size);
|
2014-08-22 16:21:55 +02:00
|
|
|
return true;
|
|
|
|
|
}
|
|
|
|
|
|
2015-02-12 21:10:25 +01:00
|
|
|
bool MemoryBase::Unmap(const u32 addr)
|
2014-08-22 16:21:55 +02:00
|
|
|
{
|
2014-09-05 01:23:36 +02:00
|
|
|
LV2_LOCK(0);
|
2014-08-22 16:21:55 +02:00
|
|
|
|
2014-09-02 23:48:44 +02:00
|
|
|
for (u32 i = 0; i < MemoryBlocks.size(); i++)
|
2014-08-22 16:21:55 +02:00
|
|
|
{
|
2014-09-02 23:48:44 +02:00
|
|
|
if (MemoryBlocks[i]->GetStartAddr() == addr)
|
2014-08-22 16:21:55 +02:00
|
|
|
{
|
2014-09-02 23:48:44 +02:00
|
|
|
delete MemoryBlocks[i];
|
|
|
|
|
MemoryBlocks.erase(MemoryBlocks.begin() + i);
|
|
|
|
|
return true;
|
2014-08-22 16:21:55 +02:00
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
return false;
|
|
|
|
|
}
|
|
|
|
|
|
2015-02-12 21:10:25 +01:00
|
|
|
MemBlockInfo::MemBlockInfo(u32 addr, u32 size)
|
|
|
|
|
: MemInfo(addr, size)
|
2014-07-08 16:26:49 +02:00
|
|
|
{
|
2015-02-13 15:04:03 +01:00
|
|
|
vm::page_map(addr, size, vm::page_readable | vm::page_writable | vm::page_executable);
|
2014-07-08 16:26:49 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void MemBlockInfo::Free()
|
|
|
|
|
{
|
2015-02-13 15:04:03 +01:00
|
|
|
if (addr && size)
|
2014-07-08 16:26:49 +02:00
|
|
|
{
|
2015-02-13 15:04:03 +01:00
|
|
|
vm::page_unmap(addr, size);
|
|
|
|
|
addr = size = 0;
|
2014-07-08 16:26:49 +02:00
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
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;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void MemoryBlock::InitMemory()
|
|
|
|
|
{
|
2015-02-13 15:04:03 +01:00
|
|
|
if (range_size)
|
2014-07-12 16:53:36 +02:00
|
|
|
{
|
|
|
|
|
Free();
|
|
|
|
|
mem_inf = new MemBlockInfo(range_start, range_size);
|
|
|
|
|
}
|
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();
|
|
|
|
|
}
|
|
|
|
|
|
2015-02-12 21:10:25 +01:00
|
|
|
MemoryBlock* MemoryBlock::SetRange(const u32 start, const u32 size)
|
2012-11-15 00:39:56 +01:00
|
|
|
{
|
|
|
|
|
range_start = start;
|
|
|
|
|
range_size = size;
|
|
|
|
|
|
|
|
|
|
InitMemory();
|
|
|
|
|
return this;
|
|
|
|
|
}
|
|
|
|
|
|
2014-09-02 23:48:44 +02:00
|
|
|
DynamicMemoryBlockBase::DynamicMemoryBlockBase()
|
|
|
|
|
: MemoryBlock()
|
|
|
|
|
, m_max_size(0)
|
|
|
|
|
{
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
const u32 DynamicMemoryBlockBase::GetUsedSize() const
|
|
|
|
|
{
|
2014-09-05 01:23:36 +02:00
|
|
|
LV2_LOCK(0);
|
2014-09-02 23:48:44 +02:00
|
|
|
|
|
|
|
|
u32 size = 0;
|
|
|
|
|
|
|
|
|
|
for (u32 i = 0; i<m_allocated.size(); ++i)
|
|
|
|
|
{
|
|
|
|
|
size += m_allocated[i].size;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return size;
|
|
|
|
|
}
|
|
|
|
|
|
2015-02-12 21:10:25 +01:00
|
|
|
bool DynamicMemoryBlockBase::IsInMyRange(const u32 addr, const u32 size)
|
2014-09-02 23:48:44 +02:00
|
|
|
{
|
2015-02-12 21:10:25 +01:00
|
|
|
return addr >= MemoryBlock::GetStartAddr() && addr + size - 1 <= MemoryBlock::GetEndAddr();
|
2014-09-02 23:48:44 +02:00
|
|
|
}
|
|
|
|
|
|
2015-02-12 21:10:25 +01:00
|
|
|
MemoryBlock* DynamicMemoryBlockBase::SetRange(const u32 start, const u32 size)
|
2014-09-02 23:48:44 +02:00
|
|
|
{
|
2014-09-05 01:23:36 +02:00
|
|
|
LV2_LOCK(0);
|
2014-09-02 23:48:44 +02:00
|
|
|
|
|
|
|
|
m_max_size = PAGE_4K(size);
|
|
|
|
|
if (!MemoryBlock::SetRange(start, 0))
|
|
|
|
|
{
|
|
|
|
|
assert(0);
|
|
|
|
|
return nullptr;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return this;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void DynamicMemoryBlockBase::Delete()
|
|
|
|
|
{
|
2014-09-05 01:23:36 +02:00
|
|
|
LV2_LOCK(0);
|
2014-09-02 23:48:44 +02:00
|
|
|
|
|
|
|
|
m_allocated.clear();
|
|
|
|
|
m_max_size = 0;
|
|
|
|
|
|
|
|
|
|
MemoryBlock::Delete();
|
|
|
|
|
}
|
|
|
|
|
|
2015-02-12 21:10:25 +01:00
|
|
|
bool DynamicMemoryBlockBase::AllocFixed(u32 addr, u32 size)
|
2014-09-02 23:48:44 +02:00
|
|
|
{
|
2015-02-12 21:10:25 +01:00
|
|
|
assert(size);
|
|
|
|
|
|
2014-09-02 23:48:44 +02:00
|
|
|
size = PAGE_4K(size + (addr & 4095)); // align size
|
|
|
|
|
|
|
|
|
|
addr &= ~4095; // align start address
|
|
|
|
|
|
|
|
|
|
if (!IsInMyRange(addr, size))
|
|
|
|
|
{
|
|
|
|
|
assert(0);
|
|
|
|
|
return false;
|
|
|
|
|
}
|
|
|
|
|
|
2014-09-05 01:23:36 +02:00
|
|
|
LV2_LOCK(0);
|
2014-09-02 23:48:44 +02:00
|
|
|
|
|
|
|
|
for (u32 i = 0; i<m_allocated.size(); ++i)
|
|
|
|
|
{
|
|
|
|
|
if (addr >= m_allocated[i].addr && addr < m_allocated[i].addr + m_allocated[i].size) return false;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
AppendMem(addr, size);
|
|
|
|
|
|
|
|
|
|
return true;
|
|
|
|
|
}
|
|
|
|
|
|
2015-02-12 21:10:25 +01:00
|
|
|
void DynamicMemoryBlockBase::AppendMem(u32 addr, u32 size) /* private */
|
2014-09-02 23:48:44 +02:00
|
|
|
{
|
|
|
|
|
m_allocated.emplace_back(addr, size);
|
|
|
|
|
}
|
|
|
|
|
|
2015-02-12 21:10:25 +01:00
|
|
|
u32 DynamicMemoryBlockBase::AllocAlign(u32 size, u32 align)
|
2014-09-02 23:48:44 +02:00
|
|
|
{
|
2015-02-12 21:10:25 +01:00
|
|
|
assert(size && align);
|
|
|
|
|
|
2014-11-02 00:19:14 +01:00
|
|
|
if (!MemoryBlock::GetStartAddr())
|
|
|
|
|
{
|
|
|
|
|
LOG_ERROR(MEMORY, "DynamicMemoryBlockBase::AllocAlign(size=0x%x, align=0x%x): memory block not initialized", size, align);
|
|
|
|
|
return 0;
|
|
|
|
|
}
|
2015-02-07 16:35:54 +01:00
|
|
|
|
2014-09-02 23:48:44 +02:00
|
|
|
size = PAGE_4K(size);
|
|
|
|
|
u32 exsize;
|
|
|
|
|
|
|
|
|
|
if (align <= 4096)
|
|
|
|
|
{
|
|
|
|
|
align = 0;
|
|
|
|
|
exsize = size;
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
align &= ~4095;
|
|
|
|
|
exsize = size + align - 1;
|
|
|
|
|
}
|
|
|
|
|
|
2014-09-05 01:23:36 +02:00
|
|
|
LV2_LOCK(0);
|
2014-09-02 23:48:44 +02:00
|
|
|
|
2015-02-12 21:10:25 +01:00
|
|
|
for (u32 addr = MemoryBlock::GetStartAddr(); addr <= MemoryBlock::GetEndAddr() - exsize;)
|
2014-09-02 23:48:44 +02:00
|
|
|
{
|
|
|
|
|
bool is_good_addr = true;
|
|
|
|
|
|
|
|
|
|
for (u32 i = 0; i<m_allocated.size(); ++i)
|
|
|
|
|
{
|
|
|
|
|
if ((addr >= m_allocated[i].addr && addr < m_allocated[i].addr + m_allocated[i].size) ||
|
|
|
|
|
(m_allocated[i].addr >= addr && m_allocated[i].addr < addr + exsize))
|
|
|
|
|
{
|
|
|
|
|
is_good_addr = false;
|
|
|
|
|
addr = m_allocated[i].addr + m_allocated[i].size;
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (!is_good_addr) continue;
|
|
|
|
|
|
|
|
|
|
if (align)
|
|
|
|
|
{
|
|
|
|
|
addr = (addr + (align - 1)) & ~(align - 1);
|
|
|
|
|
}
|
|
|
|
|
|
2015-02-13 22:45:36 +01:00
|
|
|
//LOG_NOTICE(MEMORY, "AllocAlign(size=0x%x) -> 0x%x", size, addr);
|
2014-09-02 23:48:44 +02:00
|
|
|
|
|
|
|
|
AppendMem(addr, size);
|
|
|
|
|
|
|
|
|
|
return addr;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
bool DynamicMemoryBlockBase::Alloc()
|
|
|
|
|
{
|
|
|
|
|
return AllocAlign(GetSize() - GetUsedSize()) != 0;
|
|
|
|
|
}
|
|
|
|
|
|
2015-02-12 21:10:25 +01:00
|
|
|
bool DynamicMemoryBlockBase::Free(u32 addr)
|
2014-09-02 23:48:44 +02:00
|
|
|
{
|
2014-09-05 01:23:36 +02:00
|
|
|
LV2_LOCK(0);
|
2014-09-02 23:48:44 +02:00
|
|
|
|
|
|
|
|
for (u32 num = 0; num < m_allocated.size(); num++)
|
|
|
|
|
{
|
|
|
|
|
if (addr == m_allocated[num].addr)
|
|
|
|
|
{
|
2015-02-13 22:45:36 +01:00
|
|
|
//LOG_NOTICE(MEMORY, "Free(0x%x)", addr);
|
2014-09-02 23:48:44 +02:00
|
|
|
|
|
|
|
|
m_allocated.erase(m_allocated.begin() + num);
|
|
|
|
|
return true;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2015-02-12 21:10:25 +01:00
|
|
|
LOG_ERROR(MEMORY, "DynamicMemoryBlock::Free(addr=0x%x): failed", addr);
|
2014-09-02 23:48:44 +02:00
|
|
|
for (u32 i = 0; i < m_allocated.size(); i++)
|
|
|
|
|
{
|
2015-02-12 21:10:25 +01:00
|
|
|
LOG_NOTICE(MEMORY, "*** Memory Block: addr = 0x%x, size = 0x%x", m_allocated[i].addr, m_allocated[i].size);
|
2014-09-02 23:48:44 +02:00
|
|
|
}
|
|
|
|
|
return false;
|
|
|
|
|
}
|
|
|
|
|
|
2014-03-28 03:50:30 +01:00
|
|
|
VirtualMemoryBlock::VirtualMemoryBlock() : MemoryBlock(), m_reserve_size(0)
|
2014-01-17 17:56:03 +01:00
|
|
|
{
|
|
|
|
|
}
|
|
|
|
|
|
2015-02-12 21:10:25 +01:00
|
|
|
MemoryBlock* VirtualMemoryBlock::SetRange(const u32 start, const u32 size)
|
2014-01-21 21:23:28 +01:00
|
|
|
{
|
|
|
|
|
range_start = start;
|
|
|
|
|
range_size = size;
|
|
|
|
|
|
|
|
|
|
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-02-12 21:10:25 +01:00
|
|
|
return addr >= GetStartAddr() && addr + size - 1 <= GetEndAddr() - 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-02-12 21:10:25 +01:00
|
|
|
for (u32 addr = GetStartAddr(); addr <= GetEndAddr() - 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;
|
2014-09-06 15:33:01 +02:00
|
|
|
*value = vm::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;
|
2014-09-06 15:33:01 +02:00
|
|
|
vm::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-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-11-19 15:16:30 +01: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-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
|
|
|
}
|