MMU: small simplification of TLB structure
We only need one "recent" per set, not NUM_WAYS recents. Slightly faster. Breaks savestate compatibility.
This commit is contained in:
parent
a0e5c76a1f
commit
dde8b24d00
|
@ -667,54 +667,48 @@ void SDRUpdated()
|
||||||
static __forceinline u32 LookupTLBPageAddress(const XCheckTLBFlag _Flag, const u32 vpa, u32 *paddr)
|
static __forceinline u32 LookupTLBPageAddress(const XCheckTLBFlag _Flag, const u32 vpa, u32 *paddr)
|
||||||
{
|
{
|
||||||
int tag = vpa >> HW_PAGE_INDEX_SHIFT;
|
int tag = vpa >> HW_PAGE_INDEX_SHIFT;
|
||||||
PowerPC::tlb_entry *tlbe = PowerPC::ppcState.tlb[_Flag == FLAG_OPCODE][tag & HW_PAGE_INDEX_MASK];
|
PowerPC::tlb_entry *tlbe = &PowerPC::ppcState.tlb[_Flag == FLAG_OPCODE][tag & HW_PAGE_INDEX_MASK];
|
||||||
if (tlbe[0].tag == tag && !(tlbe[0].flags & TLB_FLAG_INVALID))
|
if (tlbe->tag[0] == tag)
|
||||||
{
|
{
|
||||||
// Check if C bit requires updating
|
// Check if C bit requires updating
|
||||||
if (_Flag == FLAG_WRITE)
|
if (_Flag == FLAG_WRITE)
|
||||||
{
|
{
|
||||||
UPTE2 PTE2;
|
UPTE2 PTE2;
|
||||||
PTE2.Hex = tlbe[0].pte;
|
PTE2.Hex = tlbe->pte[0];
|
||||||
if (PTE2.C == 0)
|
if (PTE2.C == 0)
|
||||||
{
|
{
|
||||||
PTE2.C = 1;
|
PTE2.C = 1;
|
||||||
tlbe[0].pte = PTE2.Hex;
|
tlbe->pte[0] = PTE2.Hex;
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
if (_Flag != FLAG_NO_EXCEPTION)
|
if (_Flag != FLAG_NO_EXCEPTION)
|
||||||
{
|
tlbe->recent = 0;
|
||||||
tlbe[0].flags |= TLB_FLAG_MOST_RECENT;
|
|
||||||
tlbe[1].flags &= ~TLB_FLAG_MOST_RECENT;
|
|
||||||
}
|
|
||||||
|
|
||||||
*paddr = tlbe[0].paddr | (vpa & 0xfff);
|
*paddr = tlbe->paddr[0] | (vpa & 0xfff);
|
||||||
|
|
||||||
return 1;
|
return 1;
|
||||||
}
|
}
|
||||||
if (tlbe[1].tag == tag && !(tlbe[1].flags & TLB_FLAG_INVALID))
|
if (tlbe->tag[1] == tag)
|
||||||
{
|
{
|
||||||
// Check if C bit requires updating
|
// Check if C bit requires updating
|
||||||
if (_Flag == FLAG_WRITE)
|
if (_Flag == FLAG_WRITE)
|
||||||
{
|
{
|
||||||
UPTE2 PTE2;
|
UPTE2 PTE2;
|
||||||
PTE2.Hex = tlbe[1].pte;
|
PTE2.Hex = tlbe->pte[1];
|
||||||
if (PTE2.C == 0)
|
if (PTE2.C == 0)
|
||||||
{
|
{
|
||||||
PTE2.C = 1;
|
PTE2.C = 1;
|
||||||
tlbe[1].pte = PTE2.Hex;
|
tlbe->pte[1] = PTE2.Hex;
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
if (_Flag != FLAG_NO_EXCEPTION)
|
if (_Flag != FLAG_NO_EXCEPTION)
|
||||||
{
|
tlbe->recent = 1;
|
||||||
tlbe[1].flags |= TLB_FLAG_MOST_RECENT;
|
|
||||||
tlbe[0].flags &= ~TLB_FLAG_MOST_RECENT;
|
|
||||||
}
|
|
||||||
|
|
||||||
*paddr = tlbe[1].paddr | (vpa & 0xfff);
|
*paddr = tlbe->paddr[1] | (vpa & 0xfff);
|
||||||
|
|
||||||
return 1;
|
return 1;
|
||||||
}
|
}
|
||||||
|
@ -726,39 +720,31 @@ static __forceinline void UpdateTLBEntry(const XCheckTLBFlag _Flag, UPTE2 PTE2,
|
||||||
if (_Flag == FLAG_NO_EXCEPTION)
|
if (_Flag == FLAG_NO_EXCEPTION)
|
||||||
return;
|
return;
|
||||||
|
|
||||||
PowerPC::tlb_entry *tlbe = PowerPC::ppcState.tlb[_Flag == FLAG_OPCODE][(vpa >> HW_PAGE_INDEX_SHIFT) & HW_PAGE_INDEX_MASK];
|
int tag = vpa >> HW_PAGE_INDEX_SHIFT;
|
||||||
if ((tlbe[0].flags & TLB_FLAG_MOST_RECENT) == 0 || (tlbe[0].flags & TLB_FLAG_INVALID))
|
PowerPC::tlb_entry *tlbe = &PowerPC::ppcState.tlb[_Flag == FLAG_OPCODE][tag & HW_PAGE_INDEX_MASK];
|
||||||
{
|
int index = tlbe->recent == 0 && tlbe->tag[0] != TLB_TAG_INVALID;
|
||||||
tlbe[0].flags = TLB_FLAG_MOST_RECENT;
|
tlbe->recent = index;
|
||||||
tlbe[1].flags &= ~TLB_FLAG_MOST_RECENT;
|
tlbe->paddr[index] = PTE2.RPN << HW_PAGE_INDEX_SHIFT;
|
||||||
tlbe[0].paddr = PTE2.RPN << HW_PAGE_INDEX_SHIFT;
|
tlbe->pte[index] = PTE2.Hex;
|
||||||
tlbe[0].pte = PTE2.Hex;
|
tlbe->tag[index] = tag;
|
||||||
tlbe[0].tag = vpa >> HW_PAGE_INDEX_SHIFT;
|
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
|
||||||
tlbe[1].flags = TLB_FLAG_MOST_RECENT;
|
|
||||||
tlbe[0].flags &= ~TLB_FLAG_MOST_RECENT;
|
|
||||||
tlbe[1].paddr = PTE2.RPN << HW_PAGE_INDEX_SHIFT;
|
|
||||||
tlbe[1].pte = PTE2.Hex;
|
|
||||||
tlbe[1].tag = vpa >> HW_PAGE_INDEX_SHIFT;
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
void InvalidateTLBEntry(u32 vpa)
|
void InvalidateTLBEntry(u32 vpa)
|
||||||
{
|
{
|
||||||
PowerPC::tlb_entry *tlbe = PowerPC::ppcState.tlb[0][(vpa >> HW_PAGE_INDEX_SHIFT) & HW_PAGE_INDEX_MASK];
|
PowerPC::tlb_entry *tlbe = &PowerPC::ppcState.tlb[0][(vpa >> HW_PAGE_INDEX_SHIFT) & HW_PAGE_INDEX_MASK];
|
||||||
tlbe[0].flags |= TLB_FLAG_INVALID;
|
tlbe->tag[0] = TLB_TAG_INVALID;
|
||||||
tlbe[1].flags |= TLB_FLAG_INVALID;
|
tlbe->tag[1] = TLB_TAG_INVALID;
|
||||||
PowerPC::tlb_entry *tlbe_i = PowerPC::ppcState.tlb[1][(vpa >> HW_PAGE_INDEX_SHIFT) & HW_PAGE_INDEX_MASK];
|
PowerPC::tlb_entry *tlbe_i = &PowerPC::ppcState.tlb[1][(vpa >> HW_PAGE_INDEX_SHIFT) & HW_PAGE_INDEX_MASK];
|
||||||
tlbe_i[0].flags |= TLB_FLAG_INVALID;
|
tlbe_i->tag[0] = TLB_TAG_INVALID;
|
||||||
tlbe_i[1].flags |= TLB_FLAG_INVALID;
|
tlbe_i->tag[1] = TLB_TAG_INVALID;
|
||||||
}
|
}
|
||||||
|
|
||||||
// Page Address Translation
|
// Page Address Translation
|
||||||
static __forceinline u32 TranslatePageAddress(const u32 _Address, const XCheckTLBFlag _Flag)
|
static __forceinline u32 TranslatePageAddress(const u32 _Address, const XCheckTLBFlag _Flag)
|
||||||
{
|
{
|
||||||
// TLB cache
|
// TLB cache
|
||||||
|
// This catches 99%+ of lookups in practice, so the actual page table entry code below doesn't benefit
|
||||||
|
// much from optimization.
|
||||||
u32 translatedAddress = 0;
|
u32 translatedAddress = 0;
|
||||||
if (LookupTLBPageAddress(_Flag, _Address, &translatedAddress))
|
if (LookupTLBPageAddress(_Flag, _Address, &translatedAddress))
|
||||||
return translatedAddress;
|
return translatedAddress;
|
||||||
|
|
|
@ -125,12 +125,12 @@ void Init(int cpu_core)
|
||||||
{
|
{
|
||||||
for (int set = 0; set < 64; set++)
|
for (int set = 0; set < 64; set++)
|
||||||
{
|
{
|
||||||
|
ppcState.tlb[tlb][set].recent = 0;
|
||||||
for (int way = 0; way < 2; way++)
|
for (int way = 0; way < 2; way++)
|
||||||
{
|
{
|
||||||
ppcState.tlb[tlb][set][way].flags = TLB_FLAG_INVALID;
|
ppcState.tlb[tlb][set].paddr[way] = 0;
|
||||||
ppcState.tlb[tlb][set][way].paddr = 0;
|
ppcState.tlb[tlb][set].pte[way] = 0;
|
||||||
ppcState.tlb[tlb][set][way].pte = 0;
|
ppcState.tlb[tlb][set].tag[way] = TLB_TAG_INVALID;
|
||||||
ppcState.tlb[tlb][set][way].tag = 0;
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -29,22 +29,21 @@ enum CoreMode
|
||||||
|
|
||||||
// TLB cache
|
// TLB cache
|
||||||
#define TLB_SIZE 128
|
#define TLB_SIZE 128
|
||||||
#define TLB_WAYS 2
|
|
||||||
#define NUM_TLBS 2
|
#define NUM_TLBS 2
|
||||||
|
#define TLB_WAYS 2
|
||||||
|
|
||||||
#define HW_PAGE_INDEX_SHIFT 12
|
#define HW_PAGE_INDEX_SHIFT 12
|
||||||
#define HW_PAGE_INDEX_MASK 0x3f
|
#define HW_PAGE_INDEX_MASK 0x3f
|
||||||
#define HW_PAGE_TAG_SHIFT 18
|
#define HW_PAGE_TAG_SHIFT 18
|
||||||
|
|
||||||
#define TLB_FLAG_MOST_RECENT 0x01
|
#define TLB_TAG_INVALID 0xffffffff
|
||||||
#define TLB_FLAG_INVALID 0x02
|
|
||||||
|
|
||||||
struct tlb_entry
|
struct tlb_entry
|
||||||
{
|
{
|
||||||
u32 tag;
|
u32 tag[TLB_WAYS];
|
||||||
u32 paddr;
|
u32 paddr[TLB_WAYS];
|
||||||
u32 pte;
|
u32 pte[TLB_WAYS];
|
||||||
u8 flags;
|
u8 recent;
|
||||||
};
|
};
|
||||||
|
|
||||||
// This contains the entire state of the emulated PowerPC "Gekko" CPU.
|
// This contains the entire state of the emulated PowerPC "Gekko" CPU.
|
||||||
|
@ -107,7 +106,7 @@ struct GC_ALIGNED64(PowerPCState)
|
||||||
// also for power management, but we don't care about that.
|
// also for power management, but we don't care about that.
|
||||||
u32 spr[1024];
|
u32 spr[1024];
|
||||||
|
|
||||||
tlb_entry tlb[NUM_TLBS][TLB_SIZE / TLB_WAYS][TLB_WAYS];
|
tlb_entry tlb[NUM_TLBS][TLB_SIZE / TLB_WAYS];
|
||||||
|
|
||||||
u32 pagetable_base;
|
u32 pagetable_base;
|
||||||
u32 pagetable_hashmask;
|
u32 pagetable_hashmask;
|
||||||
|
|
|
@ -64,7 +64,7 @@ static Common::Event g_compressAndDumpStateSyncEvent;
|
||||||
static std::thread g_save_thread;
|
static std::thread g_save_thread;
|
||||||
|
|
||||||
// Don't forget to increase this after doing changes on the savestate system
|
// Don't forget to increase this after doing changes on the savestate system
|
||||||
static const u32 STATE_VERSION = 37;
|
static const u32 STATE_VERSION = 38;
|
||||||
|
|
||||||
enum
|
enum
|
||||||
{
|
{
|
||||||
|
|
Loading…
Reference in New Issue