From 65e131ef5f3c8e25d8b4033a6d95ba5fd7f3d8f7 Mon Sep 17 00:00:00 2001 From: Lioncash Date: Tue, 31 Aug 2021 09:33:23 -0400 Subject: [PATCH 1/9] MMU: Convert PTE unions over to Common::BitField --- Source/Core/Core/PowerPC/MMU.cpp | 49 ++++++++++++++++---------------- 1 file changed, 24 insertions(+), 25 deletions(-) diff --git a/Source/Core/Core/PowerPC/MMU.cpp b/Source/Core/Core/PowerPC/MMU.cpp index 87cd6102d7..8d7993a8d3 100644 --- a/Source/Core/Core/PowerPC/MMU.cpp +++ b/Source/Core/Core/PowerPC/MMU.cpp @@ -1171,29 +1171,31 @@ TranslateResult JitCache_TranslateAddress(u32 address) // Hey! these duplicate a structure in Gekko.h union UPTE1 { - struct - { - u32 API : 6; - u32 H : 1; - u32 VSID : 24; - u32 V : 1; - }; - u32 Hex; + BitField<0, 6, u32> API; + BitField<6, 1, u32> H; + BitField<7, 24, u32> VSID; + BitField<31, 1, u32> V; + + u32 Hex = 0; + + UPTE1() = default; + explicit UPTE1(u32 hex_) : Hex{hex_} {} }; union UPTE2 { - struct - { - u32 PP : 2; - u32 : 1; - u32 WIMG : 4; - u32 C : 1; - u32 R : 1; - u32 : 3; - u32 RPN : 20; - }; - u32 Hex; + BitField<0, 2, u32> PP; + BitField<2, 1, u32> reserved_1; + BitField<3, 4, u32> WIMG; + BitField<7, 1, u32> C; + BitField<8, 1, u32> R; + BitField<9, 3, u32> reserved_2; + BitField<12, 20, u32> RPN; + + u32 Hex = 0; + + UPTE2() = default; + explicit UPTE2(u32 hex_) : Hex{hex_} {} }; static void GenerateDSIException(u32 effective_address, bool write) @@ -1258,8 +1260,7 @@ static TLBLookupResult LookupTLBPageAddress(const XCheckTLBFlag flag, const u32 if (tlbe.tag[0] == tag) { - UPTE2 PTE2; - PTE2.Hex = tlbe.pte[0]; + UPTE2 PTE2(tlbe.pte[0]); // Check if C bit requires updating if (flag == XCheckTLBFlag::Write) @@ -1282,8 +1283,7 @@ static TLBLookupResult LookupTLBPageAddress(const XCheckTLBFlag flag, const u32 } if (tlbe.tag[1] == tag) { - UPTE2 PTE2; - PTE2.Hex = tlbe.pte[1]; + UPTE2 PTE2(tlbe.pte[1]); // Check if C bit requires updating if (flag == XCheckTLBFlag::Write) @@ -1388,8 +1388,7 @@ static TranslateAddressResult TranslatePageAddress(const u32 address, const XChe if (pte1 == pteg) { - UPTE2 PTE2; - PTE2.Hex = Memory::Read_U32(pteg_addr + 4); + UPTE2 PTE2(Memory::Read_U32(pteg_addr + 4)); // set the access bits switch (flag) From c008b1c335ada812395fc2a9ba854ea45bd0e9c5 Mon Sep 17 00:00:00 2001 From: Lioncash Date: Tue, 31 Aug 2021 09:43:37 -0400 Subject: [PATCH 2/9] MMU: Move UPTE1/UPTE2 unions into Gekko.h Keeps the register definitions in one spot and lets us remove a comment about them. --- Source/Core/Core/PowerPC/Gekko.h | 29 +++++++++++++ Source/Core/Core/PowerPC/MMU.cpp | 74 ++++++++++---------------------- 2 files changed, 51 insertions(+), 52 deletions(-) diff --git a/Source/Core/Core/PowerPC/Gekko.h b/Source/Core/Core/PowerPC/Gekko.h index 97ead7f37a..6da4a91d7b 100644 --- a/Source/Core/Core/PowerPC/Gekko.h +++ b/Source/Core/Core/PowerPC/Gekko.h @@ -764,6 +764,35 @@ union UReg_PTE u32 Hex32[2]; }; +union UPTE_Lo +{ + BitField<0, 6, u32> API; + BitField<6, 1, u32> H; + BitField<7, 24, u32> VSID; + BitField<31, 1, u32> V; + + u32 Hex = 0; + + UPTE_Lo() = default; + explicit UPTE_Lo(u32 hex_) : Hex{hex_} {} +}; + +union UPTE_Hi +{ + BitField<0, 2, u32> PP; + BitField<2, 1, u32> reserved_1; + BitField<3, 4, u32> WIMG; + BitField<7, 1, u32> C; + BitField<8, 1, u32> R; + BitField<9, 3, u32> reserved_2; + BitField<12, 20, u32> RPN; + + u32 Hex = 0; + + UPTE_Hi() = default; + explicit UPTE_Hi(u32 hex_) : Hex{hex_} {} +}; + // // --- Gekko Types and Defs --- // diff --git a/Source/Core/Core/PowerPC/MMU.cpp b/Source/Core/Core/PowerPC/MMU.cpp index 8d7993a8d3..c4da0bef40 100644 --- a/Source/Core/Core/PowerPC/MMU.cpp +++ b/Source/Core/Core/PowerPC/MMU.cpp @@ -1168,36 +1168,6 @@ TranslateResult JitCache_TranslateAddress(u32 address) #define PTE2_WIMG(v) (((v) >> 3) & 0xf) #define PTE2_PP(v) ((v)&3) -// Hey! these duplicate a structure in Gekko.h -union UPTE1 -{ - BitField<0, 6, u32> API; - BitField<6, 1, u32> H; - BitField<7, 24, u32> VSID; - BitField<31, 1, u32> V; - - u32 Hex = 0; - - UPTE1() = default; - explicit UPTE1(u32 hex_) : Hex{hex_} {} -}; - -union UPTE2 -{ - BitField<0, 2, u32> PP; - BitField<2, 1, u32> reserved_1; - BitField<3, 4, u32> WIMG; - BitField<7, 1, u32> C; - BitField<8, 1, u32> R; - BitField<9, 3, u32> reserved_2; - BitField<12, 20, u32> RPN; - - u32 Hex = 0; - - UPTE2() = default; - explicit UPTE2(u32 hex_) : Hex{hex_} {} -}; - static void GenerateDSIException(u32 effective_address, bool write) { // DSI exceptions are only supported in MMU mode. @@ -1260,15 +1230,15 @@ static TLBLookupResult LookupTLBPageAddress(const XCheckTLBFlag flag, const u32 if (tlbe.tag[0] == tag) { - UPTE2 PTE2(tlbe.pte[0]); + UPTE_Hi pte2(tlbe.pte[0]); // Check if C bit requires updating if (flag == XCheckTLBFlag::Write) { - if (PTE2.C == 0) + if (pte2.C == 0) { - PTE2.C = 1; - tlbe.pte[0] = PTE2.Hex; + pte2.C = 1; + tlbe.pte[0] = pte2.Hex; return TLBLookupResult::UpdateC; } } @@ -1277,21 +1247,21 @@ static TLBLookupResult LookupTLBPageAddress(const XCheckTLBFlag flag, const u32 tlbe.recent = 0; *paddr = tlbe.paddr[0] | (vpa & 0xfff); - *wi = (PTE2.WIMG & 0b1100) != 0; + *wi = (pte2.WIMG & 0b1100) != 0; return TLBLookupResult::Found; } if (tlbe.tag[1] == tag) { - UPTE2 PTE2(tlbe.pte[1]); + UPTE_Hi pte2(tlbe.pte[1]); // Check if C bit requires updating if (flag == XCheckTLBFlag::Write) { - if (PTE2.C == 0) + if (pte2.C == 0) { - PTE2.C = 1; - tlbe.pte[1] = PTE2.Hex; + pte2.C = 1; + tlbe.pte[1] = pte2.Hex; return TLBLookupResult::UpdateC; } } @@ -1300,14 +1270,14 @@ static TLBLookupResult LookupTLBPageAddress(const XCheckTLBFlag flag, const u32 tlbe.recent = 1; *paddr = tlbe.paddr[1] | (vpa & 0xfff); - *wi = (PTE2.WIMG & 0b1100) != 0; + *wi = (pte2.WIMG & 0b1100) != 0; return TLBLookupResult::Found; } return TLBLookupResult::NotFound; } -static void UpdateTLBEntry(const XCheckTLBFlag flag, UPTE2 PTE2, const u32 address) +static void UpdateTLBEntry(const XCheckTLBFlag flag, UPTE_Hi pte2, const u32 address) { if (IsNoExceptionFlag(flag)) return; @@ -1316,8 +1286,8 @@ static void UpdateTLBEntry(const XCheckTLBFlag flag, UPTE2 PTE2, const u32 addre TLBEntry& tlbe = ppcState.tlb[IsOpcodeFlag(flag)][tag & HW_PAGE_INDEX_MASK]; const int index = tlbe.recent == 0 && tlbe.tag[0] != TLBEntry::INVALID_TAG; tlbe.recent = index; - tlbe.paddr[index] = PTE2.RPN << HW_PAGE_INDEX_SHIFT; - tlbe.pte[index] = PTE2.Hex; + tlbe.paddr[index] = pte2.RPN << HW_PAGE_INDEX_SHIFT; + tlbe.pte[index] = pte2.Hex; tlbe.tag[index] = tag; } @@ -1388,7 +1358,7 @@ static TranslateAddressResult TranslatePageAddress(const u32 address, const XChe if (pte1 == pteg) { - UPTE2 PTE2(Memory::Read_U32(pteg_addr + 4)); + UPTE_Hi pte2(Memory::Read_U32(pteg_addr + 4)); // set the access bits switch (flag) @@ -1397,30 +1367,30 @@ static TranslateAddressResult TranslatePageAddress(const u32 address, const XChe case XCheckTLBFlag::OpcodeNoException: break; case XCheckTLBFlag::Read: - PTE2.R = 1; + pte2.R = 1; break; case XCheckTLBFlag::Write: - PTE2.R = 1; - PTE2.C = 1; + pte2.R = 1; + pte2.C = 1; break; case XCheckTLBFlag::Opcode: - PTE2.R = 1; + pte2.R = 1; break; } if (!IsNoExceptionFlag(flag)) { - Memory::Write_U32(PTE2.Hex, pteg_addr + 4); + Memory::Write_U32(pte2.Hex, pteg_addr + 4); } // We already updated the TLB entry if this was caused by a C bit. if (res != TLBLookupResult::UpdateC) - UpdateTLBEntry(flag, PTE2, address); + UpdateTLBEntry(flag, pte2, address); - *wi = (PTE2.WIMG & 0b1100) != 0; + *wi = (pte2.WIMG & 0b1100) != 0; return TranslateAddressResult{TranslateAddressResultEnum::PAGE_TABLE_TRANSLATED, - (PTE2.RPN << 12) | offset}; + (pte2.RPN << 12) | offset}; } } } From 3216040bfe5646f68c34edf5ea05ed3d194396ce Mon Sep 17 00:00:00 2001 From: Lioncash Date: Tue, 31 Aug 2021 09:55:13 -0400 Subject: [PATCH 3/9] MMU: Remove implicit sign conversions and truncation in UpdateTLBEntry These are trivial to resolve. Converting the structure member into a u32 results in no increase in structure size, as it's making use of the three extra padding bits in the structure. --- Source/Core/Core/PowerPC/MMU.cpp | 4 ++-- Source/Core/Core/PowerPC/PowerPC.h | 2 +- 2 files changed, 3 insertions(+), 3 deletions(-) diff --git a/Source/Core/Core/PowerPC/MMU.cpp b/Source/Core/Core/PowerPC/MMU.cpp index c4da0bef40..abdcbf67c7 100644 --- a/Source/Core/Core/PowerPC/MMU.cpp +++ b/Source/Core/Core/PowerPC/MMU.cpp @@ -1282,9 +1282,9 @@ static void UpdateTLBEntry(const XCheckTLBFlag flag, UPTE_Hi pte2, const u32 add if (IsNoExceptionFlag(flag)) return; - const int tag = address >> HW_PAGE_INDEX_SHIFT; + const u32 tag = address >> HW_PAGE_INDEX_SHIFT; TLBEntry& tlbe = ppcState.tlb[IsOpcodeFlag(flag)][tag & HW_PAGE_INDEX_MASK]; - const int index = tlbe.recent == 0 && tlbe.tag[0] != TLBEntry::INVALID_TAG; + const u32 index = tlbe.recent == 0 && tlbe.tag[0] != TLBEntry::INVALID_TAG; tlbe.recent = index; tlbe.paddr[index] = pte2.RPN << HW_PAGE_INDEX_SHIFT; tlbe.pte[index] = pte2.Hex; diff --git a/Source/Core/Core/PowerPC/PowerPC.h b/Source/Core/Core/PowerPC/PowerPC.h index b0d5d18b46..db1fc04de5 100644 --- a/Source/Core/Core/PowerPC/PowerPC.h +++ b/Source/Core/Core/PowerPC/PowerPC.h @@ -54,7 +54,7 @@ struct TLBEntry u32 tag[TLB_WAYS] = {INVALID_TAG, INVALID_TAG}; u32 paddr[TLB_WAYS] = {}; u32 pte[TLB_WAYS] = {}; - u8 recent = 0; + u32 recent = 0; }; struct PairedSingle From 1c776d8c1a9a181793aec245f4446fa2e49804f2 Mon Sep 17 00:00:00 2001 From: Lioncash Date: Tue, 31 Aug 2021 10:04:42 -0400 Subject: [PATCH 4/9] MMU: Move invalidation logic into the TLBEntry struct Puts the invalidation logic in one place and lets us tidy up InvalidateTLBEntry a little. --- Source/Core/Core/PowerPC/MMU.cpp | 9 ++------- Source/Core/Core/PowerPC/PowerPC.h | 11 ++++++++--- 2 files changed, 10 insertions(+), 10 deletions(-) diff --git a/Source/Core/Core/PowerPC/MMU.cpp b/Source/Core/Core/PowerPC/MMU.cpp index abdcbf67c7..399f90b2a1 100644 --- a/Source/Core/Core/PowerPC/MMU.cpp +++ b/Source/Core/Core/PowerPC/MMU.cpp @@ -1295,13 +1295,8 @@ void InvalidateTLBEntry(u32 address) { const u32 entry_index = (address >> HW_PAGE_INDEX_SHIFT) & HW_PAGE_INDEX_MASK; - TLBEntry& tlbe = ppcState.tlb[0][entry_index]; - tlbe.tag[0] = TLBEntry::INVALID_TAG; - tlbe.tag[1] = TLBEntry::INVALID_TAG; - - TLBEntry& tlbe_i = ppcState.tlb[1][entry_index]; - tlbe_i.tag[0] = TLBEntry::INVALID_TAG; - tlbe_i.tag[1] = TLBEntry::INVALID_TAG; + ppcState.tlb[0][entry_index].Invalidate(); + ppcState.tlb[1][entry_index].Invalidate(); } // Page Address Translation diff --git a/Source/Core/Core/PowerPC/PowerPC.h b/Source/Core/Core/PowerPC/PowerPC.h index db1fc04de5..33566908b9 100644 --- a/Source/Core/Core/PowerPC/PowerPC.h +++ b/Source/Core/Core/PowerPC/PowerPC.h @@ -3,6 +3,7 @@ #pragma once +#include #include #include #include @@ -49,12 +50,16 @@ constexpr size_t TLB_WAYS = 2; struct TLBEntry { + using WayArray = std::array; + static constexpr u32 INVALID_TAG = 0xffffffff; - u32 tag[TLB_WAYS] = {INVALID_TAG, INVALID_TAG}; - u32 paddr[TLB_WAYS] = {}; - u32 pte[TLB_WAYS] = {}; + WayArray tag{INVALID_TAG, INVALID_TAG}; + WayArray paddr{}; + WayArray pte{}; u32 recent = 0; + + void Invalidate() { tag.fill(INVALID_TAG); } }; struct PairedSingle From 75840f62ff10d53ac6fcacec91774c9e16d3c346 Mon Sep 17 00:00:00 2001 From: Lioncash Date: Tue, 31 Aug 2021 10:20:54 -0400 Subject: [PATCH 5/9] MMU: Make use of UReg_SDR1 in SDRUpdated() Lets us simplify SDRUpdated() a little bit. This also fixes the layout of UReg_SDR1. Turns out this struct has been incorrect (from a little-endian perspective) the entire time and went unnoticed, since the union was never used. --- Source/Core/Core/PowerPC/Gekko.h | 9 ++++++--- Source/Core/Core/PowerPC/MMU.cpp | 12 +++++++----- 2 files changed, 13 insertions(+), 8 deletions(-) diff --git a/Source/Core/Core/PowerPC/Gekko.h b/Source/Core/Core/PowerPC/Gekko.h index 6da4a91d7b..f2613df73c 100644 --- a/Source/Core/Core/PowerPC/Gekko.h +++ b/Source/Core/Core/PowerPC/Gekko.h @@ -611,11 +611,14 @@ union UReg_HID4 // SDR1 - Page Table format union UReg_SDR1 { - BitField<0, 16, u32> htaborg; - BitField<16, 7, u32> reserved; - BitField<23, 9, u32> htabmask; + BitField<0, 9, u32> htabmask; + BitField<9, 7, u32> reserved; + BitField<16, 16, u32> htaborg; u32 Hex = 0; + + UReg_SDR1() = default; + explicit UReg_SDR1(u32 hex_) : Hex{hex_} {} }; // MMCR0 - Monitor Mode Control Register 0 format diff --git a/Source/Core/Core/PowerPC/MMU.cpp b/Source/Core/Core/PowerPC/MMU.cpp index 399f90b2a1..960e67147f 100644 --- a/Source/Core/Core/PowerPC/MMU.cpp +++ b/Source/Core/Core/PowerPC/MMU.cpp @@ -1199,7 +1199,9 @@ static void GenerateISIException(u32 effective_address) void SDRUpdated() { - u32 htabmask = SDR1_HTABMASK(PowerPC::ppcState.spr[SPR_SDR]); + const auto sdr = UReg_SDR1{ppcState.spr[SPR_SDR]}; + const u32 htabmask = sdr.htabmask; + if (!Common::IsValidLowMask(htabmask)) WARN_LOG_FMT(POWERPC, "Invalid HTABMASK: 0b{:032b}", htabmask); @@ -1207,12 +1209,12 @@ void SDRUpdated() // must be equal to the number of trailing ones in the mask (i.e. HTABORG must be // properly aligned), this is actually not a hard requirement. Real hardware will just OR // the base address anyway. Ignoring SDR changes would lead to incorrect emulation. - u32 htaborg = SDR1_HTABORG(PowerPC::ppcState.spr[SPR_SDR]); - if (htaborg & htabmask) + const u32 htaborg = sdr.htaborg; + if ((htaborg & htabmask) != 0) WARN_LOG_FMT(POWERPC, "Invalid HTABORG: htaborg=0x{:08x} htabmask=0x{:08x}", htaborg, htabmask); - PowerPC::ppcState.pagetable_base = htaborg << 16; - PowerPC::ppcState.pagetable_hashmask = ((htabmask << 10) | 0x3ff); + ppcState.pagetable_base = htaborg << 16; + ppcState.pagetable_hashmask = ((htabmask << 10) | 0x3ff); } enum class TLBLookupResult From e687569e0217e92c9a3a057eb95d0c44d5a49d2c Mon Sep 17 00:00:00 2001 From: Lioncash Date: Tue, 31 Aug 2021 10:38:08 -0400 Subject: [PATCH 6/9] Gekko: Add union for segment registers Will be used in the following commit. --- Source/Core/Core/PowerPC/Gekko.h | 21 +++++++++++++++++++++ 1 file changed, 21 insertions(+) diff --git a/Source/Core/Core/PowerPC/Gekko.h b/Source/Core/Core/PowerPC/Gekko.h index f2613df73c..f06e16ef7a 100644 --- a/Source/Core/Core/PowerPC/Gekko.h +++ b/Source/Core/Core/PowerPC/Gekko.h @@ -721,6 +721,27 @@ union UReg_BAT_Lo explicit UReg_BAT_Lo(u32 hex_) : Hex{hex_} {} }; +// Segment register +union UReg_SR +{ + BitField<0, 24, u32> VSID; // Virtual segment ID + BitField<24, 4, u32> reserved; // Reserved + BitField<28, 1, u32> N; // No-execute protection + BitField<29, 1, u32> Kp; // User-state protection + BitField<30, 1, u32> Ks; // Supervisor-state protection + BitField<31, 1, u32> T; // Segment register format selector + + // These override other fields if T = 1 + + BitField<0, 20, u32> CNTLR_SPEC; // Device-specific data for I/O controller + BitField<20, 9, u32> BUID; // Bus unit ID + + u32 Hex = 0; + + UReg_SR() = default; + explicit UReg_SR(u32 hex_) : Hex{hex_} {} +}; + union UReg_THRM12 { BitField<0, 1, u32> V; // Valid From ee40b640d3fa0d722c7ffc00c086a8a3070de4eb Mon Sep 17 00:00:00 2001 From: Lioncash Date: Tue, 31 Aug 2021 10:52:49 -0400 Subject: [PATCH 7/9] MMU: Make use of UPTE_Lo and UReg_SR in TranslatePageAddress() Allows us to get rid of a bit of masking in exchange for stating the bits being accessed or written to directly. --- Source/Core/Core/PowerPC/MMU.cpp | 33 ++++++++++++++++++-------------- 1 file changed, 19 insertions(+), 14 deletions(-) diff --git a/Source/Core/Core/PowerPC/MMU.cpp b/Source/Core/Core/PowerPC/MMU.cpp index 960e67147f..b6888a73bd 100644 --- a/Source/Core/Core/PowerPC/MMU.cpp +++ b/Source/Core/Core/PowerPC/MMU.cpp @@ -1308,34 +1308,39 @@ static TranslateAddressResult TranslatePageAddress(const u32 address, const XChe // 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; - TLBLookupResult res = LookupTLBPageAddress(flag, address, &translatedAddress, wi); + u32 translated_address = 0; + const TLBLookupResult res = LookupTLBPageAddress(flag, address, &translated_address, wi); if (res == TLBLookupResult::Found) + { return TranslateAddressResult{TranslateAddressResultEnum::PAGE_TABLE_TRANSLATED, - translatedAddress}; + translated_address}; + } - u32 sr = PowerPC::ppcState.sr[EA_SR(address)]; + const auto sr = UReg_SR{ppcState.sr[EA_SR(address)]}; - if (sr & 0x80000000) + if (sr.T != 0) return TranslateAddressResult{TranslateAddressResultEnum::DIRECT_STORE_SEGMENT, 0}; // TODO: Handle KS/KP segment register flags. // No-execute segment register flag. - if ((flag == XCheckTLBFlag::Opcode || flag == XCheckTLBFlag::OpcodeNoException) && - (sr & 0x10000000)) + if ((flag == XCheckTLBFlag::Opcode || flag == XCheckTLBFlag::OpcodeNoException) && sr.N != 0) { return TranslateAddressResult{TranslateAddressResultEnum::PAGE_FAULT, 0}; } - u32 offset = EA_Offset(address); // 12 bit - u32 page_index = EA_PageIndex(address); // 16 bit - u32 VSID = SR_VSID(sr); // 24 bit - u32 api = EA_API(address); // 6 bit (part of page_index) + const u32 offset = EA_Offset(address); // 12 bit + const u32 page_index = EA_PageIndex(address); // 16 bit + const u32 VSID = sr.VSID; // 24 bit + const u32 api = EA_API(address); // 6 bit (part of page_index) // hash function no 1 "xor" .360 u32 hash = (VSID ^ page_index); - u32 pte1 = (VSID << 7) | api | PTE1_V; + + UPTE_Lo pte1; + pte1.VSID = VSID; + pte1.API = api; + pte1.V = 1; for (int hash_func = 0; hash_func < 2; hash_func++) { @@ -1343,7 +1348,7 @@ static TranslateAddressResult TranslatePageAddress(const u32 address, const XChe if (hash_func == 1) { hash = ~hash; - pte1 |= PTE1_H; + pte1.H = 1; } u32 pteg_addr = @@ -1353,7 +1358,7 @@ static TranslateAddressResult TranslatePageAddress(const u32 address, const XChe { const u32 pteg = Memory::Read_U32(pteg_addr); - if (pte1 == pteg) + if (pte1.Hex == pteg) { UPTE_Hi pte2(Memory::Read_U32(pteg_addr + 4)); From 029aff074161311e13517700bec7d2c855b32bbf Mon Sep 17 00:00:00 2001 From: Lioncash Date: Tue, 31 Aug 2021 11:13:42 -0400 Subject: [PATCH 8/9] MMU: Remove now-unused defines Now that we're using register unions where applicable, we can remove these defines, since they're now unused. --- Source/Core/Core/PowerPC/MMU.cpp | 80 +++++++++++++------------------- 1 file changed, 31 insertions(+), 49 deletions(-) diff --git a/Source/Core/Core/PowerPC/MMU.cpp b/Source/Core/Core/PowerPC/MMU.cpp index b6888a73bd..d25e05c143 100644 --- a/Source/Core/Core/PowerPC/MMU.cpp +++ b/Source/Core/Core/PowerPC/MMU.cpp @@ -1134,40 +1134,6 @@ TranslateResult JitCache_TranslateAddress(u32 address) * Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA. */ -#define PPC_EXC_DSISR_PAGE (1 << 30) -#define PPC_EXC_DSISR_PROT (1 << 27) -#define PPC_EXC_DSISR_STORE (1 << 25) - -#define SDR1_HTABORG(v) (((v) >> 16) & 0xffff) -#define SDR1_HTABMASK(v) ((v)&0x1ff) -#define SDR1_PAGETABLE_BASE(v) ((v)&0xffff) -#define SR_T (1 << 31) -#define SR_Ks (1 << 30) -#define SR_Kp (1 << 29) -#define SR_N (1 << 28) -#define SR_VSID(v) ((v)&0xffffff) -#define SR_BUID(v) (((v) >> 20) & 0x1ff) -#define SR_CNTRL_SPEC(v) ((v)&0xfffff) - -#define EA_SR(v) (((v) >> 28) & 0xf) -#define EA_PageIndex(v) (((v) >> 12) & 0xffff) -#define EA_Offset(v) ((v)&0xfff) -#define EA_API(v) (((v) >> 22) & 0x3f) - -#define PA_RPN(v) (((v) >> 12) & 0xfffff) -#define PA_Offset(v) ((v)&0xfff) - -#define PTE1_V (1 << 31) -#define PTE1_VSID(v) (((v) >> 7) & 0xffffff) -#define PTE1_H (1 << 6) -#define PTE1_API(v) ((v)&0x3f) - -#define PTE2_RPN(v) ((v)&0xfffff000) -#define PTE2_R (1 << 8) -#define PTE2_C (1 << 7) -#define PTE2_WIMG(v) (((v) >> 3) & 0xf) -#define PTE2_PP(v) ((v)&3) - static void GenerateDSIException(u32 effective_address, bool write) { // DSI exceptions are only supported in MMU mode. @@ -1178,14 +1144,17 @@ static void GenerateDSIException(u32 effective_address, bool write) return; } - if (effective_address) - PowerPC::ppcState.spr[SPR_DSISR] = PPC_EXC_DSISR_PAGE | PPC_EXC_DSISR_STORE; + constexpr u32 dsisr_page = 1U << 30; + constexpr u32 dsisr_store = 1U << 25; + + if (effective_address != 0) + ppcState.spr[SPR_DSISR] = dsisr_page | dsisr_store; else - PowerPC::ppcState.spr[SPR_DSISR] = PPC_EXC_DSISR_PAGE; + ppcState.spr[SPR_DSISR] = dsisr_page; - PowerPC::ppcState.spr[SPR_DAR] = effective_address; + ppcState.spr[SPR_DAR] = effective_address; - PowerPC::ppcState.Exceptions |= EXCEPTION_DSI; + ppcState.Exceptions |= EXCEPTION_DSI; } static void GenerateISIException(u32 effective_address) @@ -1301,22 +1270,35 @@ void InvalidateTLBEntry(u32 address) ppcState.tlb[1][entry_index].Invalidate(); } +union EffectiveAddress +{ + BitField<0, 12, u32> offset; + BitField<12, 16, u32> page_index; + BitField<22, 6, u32> API; + BitField<28, 4, u32> SR; + + u32 Hex = 0; + + EffectiveAddress() = default; + explicit EffectiveAddress(u32 address) : Hex{address} {} +}; + // Page Address Translation -static TranslateAddressResult TranslatePageAddress(const u32 address, const XCheckTLBFlag flag, - bool* wi) +static TranslateAddressResult TranslatePageAddress(const EffectiveAddress address, + const XCheckTLBFlag flag, bool* wi) { // TLB cache // This catches 99%+ of lookups in practice, so the actual page table entry code below doesn't // benefit much from optimization. u32 translated_address = 0; - const TLBLookupResult res = LookupTLBPageAddress(flag, address, &translated_address, wi); + const TLBLookupResult res = LookupTLBPageAddress(flag, address.Hex, &translated_address, wi); if (res == TLBLookupResult::Found) { return TranslateAddressResult{TranslateAddressResultEnum::PAGE_TABLE_TRANSLATED, translated_address}; } - const auto sr = UReg_SR{ppcState.sr[EA_SR(address)]}; + const auto sr = UReg_SR{ppcState.sr[address.SR]}; if (sr.T != 0) return TranslateAddressResult{TranslateAddressResultEnum::DIRECT_STORE_SEGMENT, 0}; @@ -1329,10 +1311,10 @@ static TranslateAddressResult TranslatePageAddress(const u32 address, const XChe return TranslateAddressResult{TranslateAddressResultEnum::PAGE_FAULT, 0}; } - const u32 offset = EA_Offset(address); // 12 bit - const u32 page_index = EA_PageIndex(address); // 16 bit - const u32 VSID = sr.VSID; // 24 bit - const u32 api = EA_API(address); // 6 bit (part of page_index) + const u32 offset = address.offset; // 12 bit + const u32 page_index = address.page_index; // 16 bit + const u32 VSID = sr.VSID; // 24 bit + const u32 api = address.API; // 6 bit (part of page_index) // hash function no 1 "xor" .360 u32 hash = (VSID ^ page_index); @@ -1387,7 +1369,7 @@ static TranslateAddressResult TranslatePageAddress(const u32 address, const XChe // We already updated the TLB entry if this was caused by a C bit. if (res != TLBLookupResult::UpdateC) - UpdateTLBEntry(flag, pte2, address); + UpdateTLBEntry(flag, pte2, address.Hex); *wi = (pte2.WIMG & 0b1100) != 0; @@ -1558,7 +1540,7 @@ static TranslateAddressResult TranslateAddress(u32 address) if (TranslateBatAddess(IsOpcodeFlag(flag) ? ibat_table : dbat_table, &address, &wi)) return TranslateAddressResult{TranslateAddressResultEnum::BAT_TRANSLATED, address, wi}; - return TranslatePageAddress(address, flag, &wi); + return TranslatePageAddress(EffectiveAddress{address}, flag, &wi); } std::optional GetTranslatedAddress(u32 address) From ed0aa5928f37436e2889fb0f53488c8ce527bebc Mon Sep 17 00:00:00 2001 From: Lioncash Date: Tue, 31 Aug 2021 12:04:07 -0400 Subject: [PATCH 9/9] Gekko: Remove unused UReg_PTE PTEs are only ever operated on in terms of their halves in our code currently, so this can be removed. --- Source/Core/Core/PowerPC/Gekko.h | 18 ------------------ 1 file changed, 18 deletions(-) diff --git a/Source/Core/Core/PowerPC/Gekko.h b/Source/Core/Core/PowerPC/Gekko.h index f06e16ef7a..9f89004552 100644 --- a/Source/Core/Core/PowerPC/Gekko.h +++ b/Source/Core/Core/PowerPC/Gekko.h @@ -770,24 +770,6 @@ union UReg_THRM3 explicit UReg_THRM3(u32 hex_) : Hex{hex_} {} }; -union UReg_PTE -{ - BitField<0, 6, u64> API; - BitField<6, 1, u64> H; - BitField<7, 24, u64> VSID; - BitField<31, 1, u64> V; - BitField<32, 2, u64> PP; - BitField<34, 1, u64> reserved_1; - BitField<35, 4, u64> WIMG; - BitField<39, 1, u64> C; - BitField<40, 1, u64> R; - BitField<41, 3, u64> reserved_2; - BitField<44, 20, u64> RPN; - - u64 Hex = 0; - u32 Hex32[2]; -}; - union UPTE_Lo { BitField<0, 6, u32> API;