mirror of https://github.com/PCSX2/pcsx2.git
Did some general cleanups to vtlb's memory protection and recompiled block integrity checking systems.
git-svn-id: http://pcsx2.googlecode.com/svn/trunk@1208 96395faa-99c1-11dd-bbfe-3dabce05a288
This commit is contained in:
parent
e1b38852b6
commit
f6ce237b8e
|
@ -69,7 +69,7 @@ void MapTLB(int i)
|
||||||
for (addr=saddr; addr<eaddr; addr++) {
|
for (addr=saddr; addr<eaddr; addr++) {
|
||||||
if ((addr & mask) == ((tlb[i].VPN2 >> 12) & mask)) { //match
|
if ((addr & mask) == ((tlb[i].VPN2 >> 12) & mask)) { //match
|
||||||
memSetPageAddr(addr << 12, tlb[i].PFN0 + ((addr - saddr) << 12));
|
memSetPageAddr(addr << 12, tlb[i].PFN0 + ((addr - saddr) << 12));
|
||||||
Cpu->Clear(addr << 12, 1);
|
Cpu->Clear(addr << 12, 0x400);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -82,7 +82,7 @@ void MapTLB(int i)
|
||||||
for (addr=saddr; addr<eaddr; addr++) {
|
for (addr=saddr; addr<eaddr; addr++) {
|
||||||
if ((addr & mask) == ((tlb[i].VPN2 >> 12) & mask)) { //match
|
if ((addr & mask) == ((tlb[i].VPN2 >> 12) & mask)) { //match
|
||||||
memSetPageAddr(addr << 12, tlb[i].PFN1 + ((addr - saddr) << 12));
|
memSetPageAddr(addr << 12, tlb[i].PFN1 + ((addr - saddr) << 12));
|
||||||
Cpu->Clear(addr << 12, 1);
|
Cpu->Clear(addr << 12, 0x400);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -109,7 +109,7 @@ void UnmapTLB(int i)
|
||||||
for (addr=saddr; addr<eaddr; addr++) {
|
for (addr=saddr; addr<eaddr; addr++) {
|
||||||
if ((addr & mask) == ((tlb[i].VPN2 >> 12) & mask)) { //match
|
if ((addr & mask) == ((tlb[i].VPN2 >> 12) & mask)) { //match
|
||||||
memClearPageAddr(addr << 12);
|
memClearPageAddr(addr << 12);
|
||||||
Cpu->Clear(addr << 12, 1);
|
Cpu->Clear(addr << 12, 0x400);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -122,7 +122,7 @@ void UnmapTLB(int i)
|
||||||
for (addr=saddr; addr<eaddr; addr++) {
|
for (addr=saddr; addr<eaddr; addr++) {
|
||||||
if ((addr & mask) == ((tlb[i].VPN2 >> 12) & mask)) { //match
|
if ((addr & mask) == ((tlb[i].VPN2 >> 12) & mask)) { //match
|
||||||
memClearPageAddr(addr << 12);
|
memClearPageAddr(addr << 12);
|
||||||
Cpu->Clear(addr << 12, 1);
|
Cpu->Clear(addr << 12, 0x400);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -398,11 +398,14 @@ namespace HostSys
|
||||||
void MemProtect( void* baseaddr, size_t size, PageProtectionMode mode, bool allowExecution )
|
void MemProtect( void* baseaddr, size_t size, PageProtectionMode mode, bool allowExecution )
|
||||||
{
|
{
|
||||||
// Breakpoint this to trap potentially inappropriate use of page protection, which would
|
// Breakpoint this to trap potentially inappropriate use of page protection, which would
|
||||||
// be caused by failed aligned directives on global vars (dunno if it's happening or not yet)
|
// be caused by failed aligned directives on global vars.
|
||||||
/*Console::Error(
|
/*if( (baseaddr & m_pagemask) != 0 )
|
||||||
"*PCSX2/Linux Warning* Inappropriate use of page protection detected.\n"
|
{
|
||||||
"\tbaseaddr not page aligned: 0x%08X", params (uptr)baseaddr
|
Console::Error(
|
||||||
);*/
|
"*PCSX2/Linux Warning* Inappropriate use of page protection detected.\n"
|
||||||
|
"\tbaseaddr not page aligned: 0x%08X", params (uptr)baseaddr
|
||||||
|
);
|
||||||
|
}*/
|
||||||
|
|
||||||
int lnxmode = 0;
|
int lnxmode = 0;
|
||||||
|
|
||||||
|
|
146
pcsx2/Memory.cpp
146
pcsx2/Memory.cpp
|
@ -117,8 +117,6 @@ void loadBiosRom( const char *ext, u8 *dest, long maxSize )
|
||||||
fclose(fp);
|
fclose(fp);
|
||||||
}
|
}
|
||||||
|
|
||||||
static u32 psMPWC[(Ps2MemSize::Base/32)>>12];
|
|
||||||
static std::vector<u32> psMPWVA[Ps2MemSize::Base>>12];
|
|
||||||
|
|
||||||
u8 *psM = NULL; //32mb Main Ram
|
u8 *psM = NULL; //32mb Main Ram
|
||||||
u8 *psR = NULL; //4mb rom area
|
u8 *psR = NULL; //4mb rom area
|
||||||
|
@ -828,52 +826,122 @@ void memReset()
|
||||||
loadBiosRom("erom", PS2MEM_EROM, Ps2MemSize::ERom);
|
loadBiosRom("erom", PS2MEM_EROM, Ps2MemSize::ERom);
|
||||||
}
|
}
|
||||||
|
|
||||||
int mmap_GetRamPageInfo(void* ptr)
|
//////////////////////////////////////////////////////////////////////////////////////////
|
||||||
|
// Memory Protection and Block Checking, vtlb Style!
|
||||||
|
//
|
||||||
|
// For the first time code is recompiled (executed), the PS2 ram page for that code is
|
||||||
|
// protected using Virtual Memory (mprotect). If the game modifies its own code then this
|
||||||
|
// protection causes an *exception* to be raised (signal in Linux), which is handled by
|
||||||
|
// unprotecting the page and switching the recompiled block to "manual" protection.
|
||||||
|
//
|
||||||
|
// Manual protection uses a simple brute-force memcmp of the recompiled code to the code
|
||||||
|
// currently in RAM for *each time* the block is executed. Fool-proof, but slow, which
|
||||||
|
// is why we default to using the exception-based protection scheme described above.
|
||||||
|
//
|
||||||
|
// Why manual blocks? Because many games contain code and data in the same 4k page, so
|
||||||
|
// we *cannot* automatically recompile and reprotect pages, lest we end up recompiling and
|
||||||
|
// reprotecting them constantly (Which would be very slow). As a counter, the R5900 side
|
||||||
|
// of the block checking code does try to periodically re-protect blocks [going from manual
|
||||||
|
// back to protected], so that blocks which underwent a single invalidation don't need to
|
||||||
|
// incur a permanent performance penalty.
|
||||||
|
//
|
||||||
|
// Page Granularity:
|
||||||
|
// Fortunately for us MIPS and x86 use the same page granularity for TLB and memory
|
||||||
|
// protection, so we can use a 1:1 correspondence when protecting pages. Page granularity
|
||||||
|
// is 4096 (4k), which is why you'll see a lot of 0xfff's, >><< 12's, and 0x1000's in the
|
||||||
|
// code below.
|
||||||
|
//
|
||||||
|
|
||||||
|
enum vtlb_ProtectionMode
|
||||||
{
|
{
|
||||||
u32 offset=((u8*)ptr-psM);
|
ProtMode_None = 0, // page is 'unaccounted' -- neither protected nor unprotected
|
||||||
if (offset>=Ps2MemSize::Base)
|
ProtMode_Write, // page is under write protection (exception handler)
|
||||||
|
ProtMode_Manual // page is under manual protection (self-checked at execution)
|
||||||
|
};
|
||||||
|
|
||||||
|
struct vtlb_PageProtectionInfo
|
||||||
|
{
|
||||||
|
// Ram De-mapping -- used to convert fully translated/mapped offsets into psM back
|
||||||
|
// into their originating ps2 physical ram address. Values are assigned when pages
|
||||||
|
// are marked for protection.
|
||||||
|
u32 ReverseRamMap;
|
||||||
|
|
||||||
|
vtlb_ProtectionMode Mode;
|
||||||
|
};
|
||||||
|
|
||||||
|
PCSX2_ALIGNED16( static vtlb_PageProtectionInfo m_PageProtectInfo[Ps2MemSize::Base >> 12] );
|
||||||
|
|
||||||
|
|
||||||
|
// returns:
|
||||||
|
// -1 - unchecked block (resides in ROM, thus is integrity is constant)
|
||||||
|
// 0 - page is using Write protection
|
||||||
|
// 1 - page is using manual protection (recompiler must include execution-time
|
||||||
|
// self-checking of block integrity)
|
||||||
|
//
|
||||||
|
int mmap_GetRamPageInfo( u32 paddr )
|
||||||
|
{
|
||||||
|
paddr &= ~0xfff;
|
||||||
|
|
||||||
|
uptr ptr = (uptr)PSM( paddr );
|
||||||
|
uptr rampage = ptr - (uptr)psM;
|
||||||
|
|
||||||
|
if (rampage >= Ps2MemSize::Base)
|
||||||
return -1; //not in ram, no tracking done ...
|
return -1; //not in ram, no tracking done ...
|
||||||
offset>>=12;
|
|
||||||
return (psMPWC[(offset/32)]&(1<<(offset&31)))?1:0;
|
rampage >>= 12;
|
||||||
|
return ( m_PageProtectInfo[rampage].Mode == ProtMode_Manual ) ? 1 : 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
void mmap_MarkCountedRamPage(void* ptr,u32 vaddr)
|
// paddr - physically mapped address
|
||||||
|
void mmap_MarkCountedRamPage( u32 paddr )
|
||||||
{
|
{
|
||||||
HostSys::MemProtect( ptr, 1, Protect_ReadOnly );
|
paddr &= ~0xfff;
|
||||||
|
|
||||||
u32 offset=((u8*)ptr-psM);
|
uptr ptr = (uptr)PSM( paddr );
|
||||||
offset>>=12;
|
int rampage = (ptr - (uptr)psM) >> 12;
|
||||||
psMPWC[(offset/32)] &= ~(1<<(offset&31));
|
|
||||||
|
|
||||||
for (u32 i=0;i<psMPWVA[offset].size();i++)
|
// Important: reassign paddr here, since TLB changes could alter the paddr->psM mapping
|
||||||
{
|
// (and clear blocks accordingly), but don't necessarily clear the protection status.
|
||||||
if (psMPWVA[offset][i]==vaddr)
|
m_PageProtectInfo[rampage].ReverseRamMap = paddr;
|
||||||
return;
|
|
||||||
}
|
if( m_PageProtectInfo[rampage].Mode == ProtMode_Write )
|
||||||
psMPWVA[offset].push_back(vaddr);
|
return; // skip town if we're already protected.
|
||||||
}
|
|
||||||
|
if( m_PageProtectInfo[rampage].Mode == ProtMode_Manual )
|
||||||
void mmap_ResetBlockTracking()
|
DbgCon::WriteLn( "dyna_page_reset @ 0x%05x", params paddr>>12 );
|
||||||
{
|
else
|
||||||
DevCon::WriteLn("vtlb/mmap: Block Tracking reset...");
|
DbgCon::WriteLn( "Write-protected page @ 0x%05x", params paddr>>12 );
|
||||||
memzero_obj(psMPWC);
|
|
||||||
for(u32 i=0;i<(Ps2MemSize::Base>>12);i++)
|
m_PageProtectInfo[rampage].Mode = ProtMode_Write;
|
||||||
{
|
HostSys::MemProtect( &psM[rampage<<12], 1, Protect_ReadOnly );
|
||||||
psMPWVA[i].clear();
|
|
||||||
}
|
|
||||||
HostSys::MemProtect( psM, Ps2MemSize::Base, Protect_ReadWrite );
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// offset - offset of address relative to psM. The exception handler for the platform/host
|
||||||
|
// OS should ensure that only addresses within psM address space are passed. Anything else
|
||||||
|
// will produce undefined results (ie, crashes).
|
||||||
void mmap_ClearCpuBlock( uint offset )
|
void mmap_ClearCpuBlock( uint offset )
|
||||||
{
|
{
|
||||||
HostSys::MemProtect( &psM[offset], 1, Protect_ReadWrite );
|
int rampage = offset >> 12;
|
||||||
|
|
||||||
offset>>=12;
|
|
||||||
psMPWC[(offset/32)]|=(1<<(offset&31));
|
|
||||||
|
|
||||||
for (u32 i=0;i<psMPWVA[offset].size();i++)
|
// Assertion: This function should never be run on a block that's already under
|
||||||
{
|
// manual protection. Indicates a logic error in the recompiler or protection code.
|
||||||
Cpu->Clear(psMPWVA[offset][i],0x400);
|
jASSUME( m_PageProtectInfo[rampage].Mode != ProtMode_Manual );
|
||||||
}
|
|
||||||
psMPWVA[offset].clear();
|
#ifndef __LINUX__ // this function is called from the signal handler
|
||||||
}
|
DbgCon::WriteLn( "Manual page @ 0x%05x", params m_PageProtectInfo[rampage].ReverseRamMap>>12 );
|
||||||
|
#endif
|
||||||
|
|
||||||
|
HostSys::MemProtect( &psM[rampage<<12], 1, Protect_ReadWrite );
|
||||||
|
m_PageProtectInfo[rampage].Mode = ProtMode_Manual;
|
||||||
|
Cpu->Clear( m_PageProtectInfo[rampage].ReverseRamMap, 0x400 );
|
||||||
|
}
|
||||||
|
|
||||||
|
// Clears all block tracking statuses, manual protection flags, and write protection.
|
||||||
|
// This does not clear any recompiler blocks. IT is assumed (and necessary) for the caller
|
||||||
|
// to ensure the EErec is also reset in conjunction with calling this function.
|
||||||
|
void mmap_ResetBlockTracking()
|
||||||
|
{
|
||||||
|
DevCon::WriteLn( "vtlb/mmap: Block Tracking reset..." );
|
||||||
|
memzero_obj( m_PageProtectInfo );
|
||||||
|
HostSys::MemProtect( psM, Ps2MemSize::Base, Protect_ReadWrite );
|
||||||
|
}
|
||||||
|
|
|
@ -141,8 +141,8 @@ extern void memClearPageAddr(u32 vaddr);
|
||||||
|
|
||||||
extern void memMapVUmicro();
|
extern void memMapVUmicro();
|
||||||
|
|
||||||
extern int mmap_GetRamPageInfo(void* ptr);
|
extern int mmap_GetRamPageInfo( u32 paddr );
|
||||||
extern void mmap_MarkCountedRamPage(void* ptr,u32 vaddr);
|
extern void mmap_MarkCountedRamPage( u32 paddr );
|
||||||
extern void mmap_ResetBlockTracking();
|
extern void mmap_ResetBlockTracking();
|
||||||
extern void mmap_ClearCpuBlock( uint offset );
|
extern void mmap_ClearCpuBlock( uint offset );
|
||||||
|
|
||||||
|
|
|
@ -209,7 +209,8 @@ __forceinline void SIF0Dma()
|
||||||
|
|
||||||
SIF0read((u32*)ptag, readSize << 2);
|
SIF0read((u32*)ptag, readSize << 2);
|
||||||
|
|
||||||
Cpu->Clear(sif0dma->madr, readSize*4);
|
// Clearing handled by vtlb memory protection and manual blocks.
|
||||||
|
//Cpu->Clear(sif0dma->madr, readSize*4);
|
||||||
|
|
||||||
cycles += readSize * BIAS; // fixme : BIAS is factored in below
|
cycles += readSize * BIAS; // fixme : BIAS is factored in below
|
||||||
sif0dma->qwc -= readSize;
|
sif0dma->qwc -= readSize;
|
||||||
|
|
|
@ -1215,19 +1215,20 @@ void badespfn() {
|
||||||
assert(0);
|
assert(0);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// Called when a block under manual protection fails it's pre-execution integrity check.
|
||||||
void __fastcall dyna_block_discard(u32 start,u32 sz)
|
void __fastcall dyna_block_discard(u32 start,u32 sz)
|
||||||
{
|
{
|
||||||
DevCon::WriteLn("dyna_block_discard .. start: %08X count=%d", params start,sz);
|
DevCon::WriteLn("dyna_block_discard .. start=0x%08X size=%d", params start, sz*4);
|
||||||
recClear(start, sz);
|
recClear(start, sz);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// called when a block under manual protection has been run enough times to be a
|
||||||
|
// candidate for being reset under the faster vtlb write protection.
|
||||||
void __fastcall dyna_page_reset(u32 start,u32 sz)
|
void __fastcall dyna_page_reset(u32 start,u32 sz)
|
||||||
{
|
{
|
||||||
DevCon::WriteLn("dyna_page_reset .. start=%08X size=%d", params start,sz*4);
|
|
||||||
recClear(start & ~0xfffUL, 0x400);
|
recClear(start & ~0xfffUL, 0x400);
|
||||||
manual_counter[start >> 12]++;
|
manual_counter[start >> 12]++;
|
||||||
mmap_MarkCountedRamPage(PSM(start), start & ~0xfffUL);
|
mmap_MarkCountedRamPage( start );
|
||||||
}
|
}
|
||||||
|
|
||||||
void recRecompile( const u32 startpc )
|
void recRecompile( const u32 startpc )
|
||||||
|
@ -1492,15 +1493,14 @@ StartRecomp:
|
||||||
|
|
||||||
// note: blocks are guaranteed to reside within the confines of a single page.
|
// note: blocks are guaranteed to reside within the confines of a single page.
|
||||||
|
|
||||||
const int PageType = mmap_GetRamPageInfo((u32*)PSM(inpage_ptr));
|
const int PageType = mmap_GetRamPageInfo( inpage_ptr );
|
||||||
const u32 inpage_offs = inpage_ptr & 0xFFF;
|
|
||||||
//const u32 pgsz = std::min(0x1000 - inpage_offs, inpage_sz);
|
//const u32 pgsz = std::min(0x1000 - inpage_offs, inpage_sz);
|
||||||
const u32 pgsz = inpage_sz;
|
const u32 pgsz = inpage_sz;
|
||||||
|
|
||||||
if(PageType!=-1)
|
if(PageType!=-1)
|
||||||
{
|
{
|
||||||
if (PageType==0) {
|
if (PageType==0) {
|
||||||
mmap_MarkCountedRamPage(PSM(inpage_ptr),inpage_ptr&~0xFFF);
|
mmap_MarkCountedRamPage( inpage_ptr );
|
||||||
manual_page[inpage_ptr >> 12] = 0;
|
manual_page[inpage_ptr >> 12] = 0;
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
|
@ -1550,13 +1550,13 @@ StartRecomp:
|
||||||
xJC( dyna_page_reset );
|
xJC( dyna_page_reset );
|
||||||
|
|
||||||
// note: clearcnt is measured per-page, not per-block!
|
// note: clearcnt is measured per-page, not per-block!
|
||||||
DbgCon::WriteLn( "Manual block @ %08X : size=%3d page/offs=%05X/%03X inpgsz=%d clearcnt=%d",
|
//DbgCon::WriteLn( "Manual block @ %08X : size=%3d page/offs=%05X/%03X inpgsz=%d clearcnt=%d",
|
||||||
params startpc, sz, inpage_ptr>>12, inpage_offs, inpage_sz, manual_counter[inpage_ptr >> 12] );
|
// params startpc, sz, inpage_ptr>>12, inpage_ptr&0xfff, inpage_sz, manual_counter[inpage_ptr >> 12] );
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
DbgCon::Notice( "Uncounted Manual block @ %08X : size=%3d page/offs=%05X/%03X inpgsz=%d",
|
DbgCon::Notice( "Uncounted Manual block @ %08X : size=%3d page/offs=%05X/%03X inpgsz=%d",
|
||||||
params startpc, sz, inpage_ptr>>12, inpage_offs, pgsz, inpage_sz );
|
params startpc, sz, inpage_ptr>>12, inpage_ptr&0xfff, pgsz, inpage_sz );
|
||||||
}
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
Loading…
Reference in New Issue