Merge pull request #248 from ItzWarty/master

virtual memory block reads/writes properly fail if address is out of bounds
This commit is contained in:
B1ackDaemon 2014-05-21 18:22:01 +03:00
commit e569de73d6
5 changed files with 73 additions and 44 deletions

View file

@ -11,7 +11,10 @@ u32 GetAddress(u32 offset, u8 location)
switch(location) switch(location)
{ {
case CELL_GCM_LOCATION_LOCAL: return Memory.RSXFBMem.GetStartAddr() + offset; case CELL_GCM_LOCATION_LOCAL: return Memory.RSXFBMem.GetStartAddr() + offset;
case CELL_GCM_LOCATION_MAIN: return Memory.RSXIOMem.getRealAddr(Memory.RSXIOMem.GetStartAddr() + offset); case CELL_GCM_LOCATION_MAIN:
u64 realAddr;
Memory.RSXIOMem.getRealAddr(Memory.RSXIOMem.GetStartAddr() + offset, realAddr); // TODO: Error Check?
return realAddr;
} }
ConLog.Error("GetAddress(offset=0x%x, location=0x%x)", location); ConLog.Error("GetAddress(offset=0x%x, location=0x%x)", location);

View file

@ -614,21 +614,25 @@ u32 VirtualMemoryBlock::UnmapAddress(u64 addr)
bool VirtualMemoryBlock::Read8(const u64 addr, u8* value) bool VirtualMemoryBlock::Read8(const u64 addr, u8* value)
{ {
u64 realAddr; u64 realAddr;
*value = Memory.Read8(realAddr = getRealAddr(addr)); if(!getRealAddr(addr, realAddr))
return realAddr != 0; return false;
*value = Memory.Read8(realAddr);
return true;
} }
bool VirtualMemoryBlock::Read16(const u64 addr, u16* value) bool VirtualMemoryBlock::Read16(const u64 addr, u16* value)
{ {
u64 realAddr; u64 realAddr;
*value = Memory.Read16(realAddr = getRealAddr(addr)); if(!getRealAddr(addr, realAddr))
return realAddr != 0; return false;
*value = Memory.Read16(realAddr);
return true;
} }
bool VirtualMemoryBlock::Read32(const u64 addr, u32* value) bool VirtualMemoryBlock::Read32(const u64 addr, u32* value)
{ {
u64 realAddr = getRealAddr(addr); u64 realAddr;
if (realAddr == 0) if (!getRealAddr(addr, realAddr))
return false; return false;
*value = Memory.Read32(realAddr); *value = Memory.Read32(realAddr);
return true; return true;
@ -637,63 +641,78 @@ bool VirtualMemoryBlock::Read32(const u64 addr, u32* value)
bool VirtualMemoryBlock::Read64(const u64 addr, u64* value) bool VirtualMemoryBlock::Read64(const u64 addr, u64* value)
{ {
u64 realAddr; u64 realAddr;
*value = Memory.Read64(realAddr = getRealAddr(addr)); if(!getRealAddr(addr, realAddr))
return realAddr != 0; return false;
*value = Memory.Read64(realAddr);
return true;
} }
bool VirtualMemoryBlock::Read128(const u64 addr, u128* value) bool VirtualMemoryBlock::Read128(const u64 addr, u128* value)
{ {
u64 realAddr; u64 realAddr;
*value = Memory.Read128(realAddr = getRealAddr(addr)); if(!getRealAddr(addr, realAddr))
return realAddr != 0; return false;
*value = Memory.Read128(realAddr);
return true;
} }
bool VirtualMemoryBlock::Write8(const u64 addr, const u8 value) bool VirtualMemoryBlock::Write8(const u64 addr, const u8 value)
{ {
u64 realAddr; u64 realAddr;
Memory.Write8(realAddr = getRealAddr(addr), value); if(!getRealAddr(addr, realAddr))
return realAddr != 0; return false;
Memory.Write8(realAddr, value);
return true;
} }
bool VirtualMemoryBlock::Write16(const u64 addr, const u16 value) bool VirtualMemoryBlock::Write16(const u64 addr, const u16 value)
{ {
u64 realAddr; u64 realAddr;
Memory.Write16(realAddr = getRealAddr(addr), value); if(!getRealAddr(addr, realAddr))
return realAddr != 0; return false;
Memory.Write16(realAddr, value);
return true;
} }
bool VirtualMemoryBlock::Write32(const u64 addr, const u32 value) bool VirtualMemoryBlock::Write32(const u64 addr, const u32 value)
{ {
u64 realAddr; u64 realAddr;
Memory.Write32(realAddr = getRealAddr(addr), value); if(!getRealAddr(addr, realAddr))
return realAddr != 0; return false;
Memory.Write32(realAddr, value);
return true;
} }
bool VirtualMemoryBlock::Write64(const u64 addr, const u64 value) bool VirtualMemoryBlock::Write64(const u64 addr, const u64 value)
{ {
u64 realAddr; u64 realAddr;
Memory.Write64(realAddr = getRealAddr(addr), value); if(!getRealAddr(addr, realAddr))
return realAddr != 0; return false;
Memory.Write64(realAddr, value);
return true;
} }
bool VirtualMemoryBlock::Write128(const u64 addr, const u128 value) bool VirtualMemoryBlock::Write128(const u64 addr, const u128 value)
{ {
u64 realAddr; u64 realAddr;
Memory.Write128(realAddr = getRealAddr(addr), value); if(!getRealAddr(addr, realAddr))
return realAddr != 0; return false;
Memory.Write128(realAddr, value);
return true;
} }
u64 VirtualMemoryBlock::getRealAddr(u64 addr) bool VirtualMemoryBlock::getRealAddr(u64 addr, u64& result)
{ {
for(u32 i=0; i<m_mapped_memory.size(); ++i) for(u32 i=0; i<m_mapped_memory.size(); ++i)
{ {
if(addr >= m_mapped_memory[i].addr && addr < m_mapped_memory[i].addr + m_mapped_memory[i].size) if(addr >= m_mapped_memory[i].addr && addr < m_mapped_memory[i].addr + m_mapped_memory[i].size)
{ {
return m_mapped_memory[i].realAddress + (addr - m_mapped_memory[i].addr); result = m_mapped_memory[i].realAddress + (addr - m_mapped_memory[i].addr);
return true;
} }
} }
return 0; return false;
} }
u64 VirtualMemoryBlock::getMappedAddress(u64 realAddress) u64 VirtualMemoryBlock::getMappedAddress(u64 realAddress)

View file

@ -281,8 +281,9 @@ public:
virtual bool Write64(const u64 addr, const u64 value); virtual bool Write64(const u64 addr, const u64 value);
virtual bool Write128(const u64 addr, const u128 value); virtual bool Write128(const u64 addr, const u128 value);
// return the real address given a mapped address, if not mapped return 0 // try to get the real address given a mapped address
u64 getRealAddr(u64 addr); // return true for success
bool getRealAddr(u64 addr, u64& result);
// return the mapped address given a real address, if not mapped return 0 // return the mapped address given a real address, if not mapped return 0
u64 getMappedAddress(u64 realAddress); u64 getMappedAddress(u64 realAddress);

View file

@ -575,9 +575,7 @@ int32_t cellGcmIoOffsetToAddress(u32 ioOffset, u64 address)
{ {
u64 realAddr; u64 realAddr;
realAddr = Memory.RSXIOMem.getRealAddr(Memory.RSXIOMem.GetStartAddr() + ioOffset); if (!Memory.RSXIOMem.getRealAddr(Memory.RSXIOMem.GetStartAddr() + ioOffset, realAddr))
if (!realAddr)
return CELL_GCM_ERROR_FAILURE; return CELL_GCM_ERROR_FAILURE;
Memory.Write64(address, realAddr); Memory.Write64(address, realAddr);

View file

@ -322,20 +322,28 @@ void RSXDebugger::GoToGet(wxCommandEvent& event)
{ {
if (!RSXReady()) return; if (!RSXReady()) return;
CellGcmControl* ctrl = (CellGcmControl*)&Memory[Emu.GetGSManager().GetRender().m_ctrlAddress]; CellGcmControl* ctrl = (CellGcmControl*)&Memory[Emu.GetGSManager().GetRender().m_ctrlAddress];
m_addr = Memory.RSXIOMem.getRealAddr(Memory.RSXIOMem.GetStartAddr() + ctrl->get); u64 realAddr;
if (Memory.RSXIOMem.getRealAddr(Memory.RSXIOMem.GetStartAddr() + ctrl->get, realAddr)) {
m_addr = realAddr; // WARNING: Potential Truncation? Cast from u64 to u32
t_addr->SetValue(wxString::Format("%08x", m_addr)); t_addr->SetValue(wxString::Format("%08x", m_addr));
UpdateInformation(); UpdateInformation();
event.Skip(); event.Skip();
}
// TODO: We should probably throw something?
} }
void RSXDebugger::GoToPut(wxCommandEvent& event) void RSXDebugger::GoToPut(wxCommandEvent& event)
{ {
if (!RSXReady()) return; if (!RSXReady()) return;
CellGcmControl* ctrl = (CellGcmControl*)&Memory[Emu.GetGSManager().GetRender().m_ctrlAddress]; CellGcmControl* ctrl = (CellGcmControl*)&Memory[Emu.GetGSManager().GetRender().m_ctrlAddress];
m_addr = Memory.RSXIOMem.getRealAddr(Memory.RSXIOMem.GetStartAddr() + ctrl->put); u64 realAddr;
if (Memory.RSXIOMem.getRealAddr(Memory.RSXIOMem.GetStartAddr() + ctrl->put, realAddr)) {
m_addr = realAddr; // WARNING: Potential Truncation? Cast from u64 to u32
t_addr->SetValue(wxString::Format("%08x", m_addr)); t_addr->SetValue(wxString::Format("%08x", m_addr));
UpdateInformation(); UpdateInformation();
event.Skip(); event.Skip();
}
// TODO: We should probably throw something?
} }
void RSXDebugger::UpdateInformation() void RSXDebugger::UpdateInformation()