rpcsx/rpcs3/Emu/Memory/Memory.h

413 lines
8.2 KiB
C
Raw Normal View History

#pragma once
2014-07-08 19:18:12 +02:00
#include "MemoryBlock.h"
2014-07-11 13:59:13 +02:00
#include "Emu/SysCalls/Callback.h"
using std::nullptr_t;
#define safe_delete(x) do {delete (x);(x)=nullptr;} while(0)
#define safe_free(x) do {free(x);(x)=nullptr;} while(0)
2014-08-31 11:54:12 +02:00
extern void* const m_base_addr;
enum MemoryType
{
Memory_PS3,
Memory_PSV,
Memory_PSP,
};
2014-07-07 19:22:36 +02:00
enum : u64
{
RAW_SPU_OFFSET = 0x0000000000100000,
RAW_SPU_BASE_ADDR = 0x00000000E0000000,
RAW_SPU_LS_OFFSET = 0x0000000000000000,
RAW_SPU_PROB_OFFSET = 0x0000000000040000,
};
class MemoryBase
{
2014-07-07 19:22:36 +02:00
std::vector<MemoryBlock*> MemoryBlocks;
u32 m_pages[0x100000000 / 4096]; // information about every page
2014-07-09 22:02:21 +02:00
std::recursive_mutex m_mutex;
public:
MemoryBlock* UserMemory;
DynamicMemoryBlock MainMem;
DynamicMemoryBlock PRXMem;
DynamicMemoryBlock RSXCMDMem;
DynamicMemoryBlock MmaperMem;
DynamicMemoryBlock RSXFBMem;
DynamicMemoryBlock StackMem;
MemoryBlock* RawSPUMem[(0x100000000 - RAW_SPU_BASE_ADDR) / RAW_SPU_OFFSET];
VirtualMemoryBlock RSXIOMem;
2014-07-13 20:55:14 +02:00
struct Wrapper32LE
{
2014-07-13 20:55:14 +02:00
void Write8(const u32 addr, const u8 data) { *(u8*)((u8*)m_base_addr + addr) = data; }
void Write16(const u32 addr, const u16 data) { *(u16*)((u8*)m_base_addr + addr) = data; }
void Write32(const u32 addr, const u32 data) { *(u32*)((u8*)m_base_addr + addr) = data; }
void Write64(const u32 addr, const u64 data) { *(u64*)((u8*)m_base_addr + addr) = data; }
void Write128(const u32 addr, const u128 data) { *(u128*)((u8*)m_base_addr + addr) = data; }
2014-07-13 20:55:14 +02:00
u8 Read8(const u32 addr) { return *(u8*)((u8*)m_base_addr + addr); }
u16 Read16(const u32 addr) { return *(u16*)((u8*)m_base_addr + addr); }
u32 Read32(const u32 addr) { return *(u32*)((u8*)m_base_addr + addr); }
u64 Read64(const u32 addr) { return *(u64*)((u8*)m_base_addr + addr); }
u128 Read128(const u32 addr) { return *(u128*)((u8*)m_base_addr + addr); }
};
2014-07-13 20:55:14 +02:00
struct : Wrapper32LE
{
DynamicMemoryBlockLE RAM;
DynamicMemoryBlockLE Userspace;
} PSV;
2014-07-13 20:55:14 +02:00
struct : Wrapper32LE
{
DynamicMemoryBlockLE Scratchpad;
DynamicMemoryBlockLE VRAM;
DynamicMemoryBlockLE RAM;
DynamicMemoryBlockLE Kernel;
DynamicMemoryBlockLE Userspace;
} PSP;
bool m_inited;
MemoryBase()
{
m_inited = false;
}
~MemoryBase()
{
Close();
}
2014-08-31 11:54:12 +02:00
static void* const GetBaseAddr()
2014-07-07 01:36:07 +02:00
{
return m_base_addr;
}
2014-08-22 16:21:55 +02:00
__noinline void InvalidAddress(const char* func, const u64 addr);
2014-08-22 16:21:55 +02:00
void RegisterPages(u64 addr, u32 size);
2014-07-09 01:04:36 +02:00
2014-08-22 16:21:55 +02:00
void UnregisterPages(u64 addr, u32 size);
template<typename T> u8* GetMemFromAddr(const T addr)
{
if ((u32)addr == addr)
{
return (u8*)GetBaseAddr() + addr;
}
else
{
2014-07-14 21:15:30 +02:00
InvalidAddress(__FUNCTION__, addr);
return (u8*)GetBaseAddr();
}
}
template<typename T> void* VirtualToRealAddr(const T vaddr)
{
return GetMemFromAddr<T>(vaddr);
}
u32 RealToVirtualAddr(const void* addr)
{
2014-07-07 19:22:36 +02:00
const u64 res = (u64)addr - (u64)GetBaseAddr();
2014-08-30 19:51:00 +02:00
if ((u32)res == res)
{
2014-08-30 19:51:00 +02:00
return (u32)res;
}
2014-07-07 19:22:36 +02:00
else
{
assert(!addr);
2014-07-07 19:22:36 +02:00
return 0;
}
}
2014-08-22 16:21:55 +02:00
u32 InitRawSPU(MemoryBlock* raw_spu);
2014-08-22 16:21:55 +02:00
void CloseRawSPU(MemoryBlock* raw_spu, const u32 num);
2014-08-22 16:21:55 +02:00
void Init(MemoryType type);
2014-07-14 21:15:30 +02:00
template<typename T> bool IsGoodAddr(const T addr)
{
2014-07-14 21:15:30 +02:00
if ((u32)addr != addr || !m_pages[addr / 4096]) // TODO: define page parameters
{
return false;
2014-07-07 19:22:36 +02:00
}
else
2014-07-07 19:22:36 +02:00
{
return true;
}
}
2014-07-14 21:15:30 +02:00
template<typename T> bool IsGoodAddr(const T addr, const u32 size)
{
2014-07-14 21:15:30 +02:00
if ((u32)addr != addr || (u64)addr + (u64)size > 0x100000000ull)
2014-07-07 19:22:36 +02:00
{
return false;
2014-07-07 19:22:36 +02:00
}
else
{
2014-08-30 22:41:01 +02:00
for (u32 i = (u32)addr / 4096; i <= ((u32)addr + size - 1) / 4096; i++)
{
if (!m_pages[i]) return false; // TODO: define page parameters
}
return true;
}
}
2014-08-22 16:21:55 +02:00
void Close();
2014-07-14 21:15:30 +02:00
//MemoryBase
template<typename T> void Write8(T addr, const u8 data)
{
if ((u32)addr == addr)
{
*(u8*)((u8*)GetBaseAddr() + addr) = data;
}
else
{
InvalidAddress(__FUNCTION__, addr);
*(u8*)GetBaseAddr() = data;
}
}
2014-07-14 21:15:30 +02:00
template<typename T> void Write16(T addr, const u16 data)
{
if ((u32)addr == addr)
{
*(u16*)((u8*)GetBaseAddr() + addr) = re16(data);
}
else
{
InvalidAddress(__FUNCTION__, addr);
*(u16*)GetBaseAddr() = data;
}
}
__noinline void WriteMMIO32(u32 addr, const u32 data);
2014-07-14 21:15:30 +02:00
template<typename T> void Write32(T addr, const u32 data)
{
if ((u32)addr == addr)
{
if (addr < RAW_SPU_BASE_ADDR || (addr % RAW_SPU_OFFSET) < RAW_SPU_PROB_OFFSET)
2014-07-14 21:15:30 +02:00
{
*(u32*)((u8*)GetBaseAddr() + addr) = re32(data);
}
else
{
2014-08-30 19:51:00 +02:00
WriteMMIO32((u32)addr, data);
2014-07-14 21:15:30 +02:00
}
}
else
{
InvalidAddress(__FUNCTION__, addr);
*(u32*)GetBaseAddr() = data;
}
}
template<typename T> void Write64(T addr, const u64 data)
{
if ((u32)addr == addr)
{
*(u64*)((u8*)GetBaseAddr() + addr) = re64(data);
}
else
{
InvalidAddress(__FUNCTION__, addr);
*(u64*)GetBaseAddr() = data;
}
}
template<typename T> void Write128(T addr, const u128 data)
{
if ((u32)addr == addr)
{
*(u128*)((u8*)GetBaseAddr() + addr) = re128(data);
}
else
{
InvalidAddress(__FUNCTION__, addr);
*(u128*)GetBaseAddr() = data;
}
}
2014-07-14 21:15:30 +02:00
template<typename T> u8 Read8(T addr)
{
if ((u32)addr == addr)
{
return *(u8*)((u8*)GetBaseAddr() + addr);
}
else
{
InvalidAddress(__FUNCTION__, addr);
return *(u8*)GetBaseAddr();
}
}
2014-07-14 21:15:30 +02:00
template<typename T> u16 Read16(T addr)
{
if ((u32)addr == addr)
{
return re16(*(u16*)((u8*)GetBaseAddr() + addr));
}
else
{
InvalidAddress(__FUNCTION__, addr);
return *(u16*)GetBaseAddr();
}
}
__noinline u32 ReadMMIO32(u32 addr);
2014-07-14 21:15:30 +02:00
template<typename T> u32 Read32(T addr)
{
if ((u32)addr == addr)
{
if (addr < RAW_SPU_BASE_ADDR || (addr % RAW_SPU_OFFSET) < RAW_SPU_PROB_OFFSET)
2014-07-14 21:15:30 +02:00
{
return re32(*(u32*)((u8*)GetBaseAddr() + addr));
}
else
{
2014-08-30 19:51:00 +02:00
return ReadMMIO32((u32)addr);
2014-07-14 21:15:30 +02:00
}
}
else
{
InvalidAddress(__FUNCTION__, addr);
return *(u32*)GetBaseAddr();
}
}
template<typename T> u64 Read64(T addr)
{
if ((u32)addr == addr)
{
return re64(*(u64*)((u8*)GetBaseAddr() + addr));
}
else
{
InvalidAddress(__FUNCTION__, addr);
return *(u64*)GetBaseAddr();
}
}
template<typename T> u128 Read128(T addr)
{
if ((u32)addr == addr)
{
return re128(*(u128*)((u8*)GetBaseAddr() + addr));
}
else
{
InvalidAddress(__FUNCTION__, addr);
return *(u128*)GetBaseAddr();
}
}
void ReadLeft(u8* dst, const u64 addr, const u32 size)
{
2014-07-07 19:22:36 +02:00
for (u32 i = 0; i < size; ++i) dst[size - 1 - i] = Read8(addr + i);
}
void WriteLeft(const u64 addr, const u32 size, const u8* src)
{
2014-07-07 19:22:36 +02:00
for (u32 i = 0; i < size; ++i) Write8(addr + i, src[size - 1 - i]);
}
void ReadRight(u8* dst, const u64 addr, const u32 size)
{
2014-07-07 19:22:36 +02:00
for (u32 i = 0; i < size; ++i) dst[i] = Read8(addr + (size - 1 - i));
}
void WriteRight(const u64 addr, const u32 size, const u8* src)
{
2014-07-07 19:22:36 +02:00
for (u32 i = 0; i < size; ++i) Write8(addr + (size - 1 - i), src[i]);
}
2014-07-18 18:55:26 +02:00
template<typename T, typename Td> void WriteData(const T addr, const Td* data)
{
2014-07-18 18:55:26 +02:00
memcpy(GetMemFromAddr<T>(addr), data, sizeof(Td));
}
2014-07-18 18:55:26 +02:00
template<typename T, typename Td> void WriteData(const T addr, const Td data)
{
2014-07-18 18:55:26 +02:00
*(Td*)GetMemFromAddr<T>(addr) = data;
}
template<typename T> std::string ReadString(const T addr, const u64 len)
{
std::string ret((const char *)GetMemFromAddr<T>(addr), len);
return ret;
}
template<typename T> std::string ReadString(const T addr)
{
return std::string((const char*)GetMemFromAddr<T>(addr));
}
template<typename T> void WriteString(const T addr, const std::string& str)
{
2014-08-31 21:38:03 +02:00
strcpy((char*)GetMemFromAddr<T>(addr), str.c_str());
}
u32 GetUserMemTotalSize()
{
return UserMemory->GetSize();
}
u32 GetUserMemAvailSize()
{
return UserMemory->GetSize() - UserMemory->GetUsedSize();
}
u64 Alloc(const u32 size, const u32 align)
{
return UserMemory->AllocAlign(size, align);
}
bool Free(const u64 addr)
{
return UserMemory->Free(addr);
}
bool Lock(const u64 addr, const u32 size)
{
return UserMemory->Lock(addr, size);
}
bool Unlock(const u64 addr, const u32 size)
{
return UserMemory->Unlock(addr, size);
}
2014-08-22 16:21:55 +02:00
bool Map(const u64 dst_addr, const u64 src_addr, const u32 size);
2014-07-07 19:22:36 +02:00
2014-08-22 16:21:55 +02:00
bool Unmap(const u64 addr);
template<typename T> void* operator + (const T vaddr)
{
return GetMemFromAddr<T>(vaddr);
}
template<typename T> u8& operator[] (const T vaddr)
{
return *GetMemFromAddr<T>(vaddr);
}
};
extern MemoryBase Memory;
#include "vm.h"