EE Cache: Invalid physical address caching and line locking

Hopefully the final fix required for the find my own way demo to work.
This commit is contained in:
Ty Lamontagne 2024-06-24 11:36:28 -04:00 committed by refractionpcsx2
parent a0b42f069f
commit d47cdfba2d
3 changed files with 115 additions and 49 deletions

View File

@ -19,7 +19,11 @@ namespace
struct CacheTag struct CacheTag
{ {
uptr rawValue; uptr rawValue;
// You are able to configure a TLB entry with non-existant physical address without causing a bus error.
// When this happens, the cache still fills with the data and when it gets evicted the data is lost.
// We don't emulate memory access on a logic level, so we need to ensure that we don't try to load/store to a non-existant physical address.
// This fixes the Find My Own Way demo.
bool validPFN = true;
// The lower parts of a cache tags structure is as follows: // The lower parts of a cache tags structure is as follows:
// 31 - 12: The physical address cache tag. // 31 - 12: The physical address cache tag.
// 11 - 7: Unused. // 11 - 7: Unused.
@ -99,6 +103,7 @@ namespace
uptr target = addr(); uptr target = addr();
CACHE_LOG("Write back at %zx", target); CACHE_LOG("Write back at %zx", target);
if (tag.validPFN)
*reinterpret_cast<CacheData*>(target) = data; *reinterpret_cast<CacheData*>(target) = data;
tag.clearDirty(); tag.clearDirty();
} }
@ -108,7 +113,16 @@ namespace
pxAssertMsg(!tag.isDirtyAndValid(), "Loaded a value into cache without writing back the old one!"); pxAssertMsg(!tag.isDirtyAndValid(), "Loaded a value into cache without writing back the old one!");
tag.setAddr(ppf); tag.setAddr(ppf);
if (!tag.validPFN)
{
// Reading from invalid physical addresses seems to return 0 on hardware
std::memset(&data, 0, sizeof(data));
}
else
{
std::memcpy(&data, reinterpret_cast<void*>(ppf & ~0x3FULL), sizeof(data)); std::memcpy(&data, reinterpret_cast<void*>(ppf & ~0x3FULL), sizeof(data));
}
tag.setValid(); tag.setValid();
tag.clearDirty(); tag.clearDirty();
} }
@ -163,29 +177,58 @@ static bool findInCache(const CacheSet& set, uptr ppf, int* way)
return check(0) || check(1); return check(0) || check(1);
} }
static int getFreeCache(u32 mem, int* way) static int getFreeCache(u32 mem, int* way, bool validPFN)
{ {
const int setIdx = cache.setIdxFor(mem); const int setIdx = cache.setIdxFor(mem);
CacheSet& set = cache.sets[setIdx]; CacheSet& set = cache.sets[setIdx];
VTLBVirtual vmv = vtlbdata.vmap[mem >> VTLB_PAGE_BITS]; VTLBVirtual vmv = vtlbdata.vmap[mem >> VTLB_PAGE_BITS];
*way = set.tags[0].lrf() ^ set.tags[1].lrf();
if (validPFN)
pxAssertMsg(!vmv.isHandler(mem), "Cache currently only supports non-handler addresses!"); pxAssertMsg(!vmv.isHandler(mem), "Cache currently only supports non-handler addresses!");
uptr ppf = vmv.assumePtr(mem); uptr ppf = vmv.assumePtr(mem);
[[unlikely]]
if ((cpuRegs.CP0.n.Config & 0x10000) == 0) if ((cpuRegs.CP0.n.Config & 0x10000) == 0)
CACHE_LOG("Cache off!"); CACHE_LOG("Cache off!");
if (findInCache(set, ppf, way)) if (findInCache(set, ppf, way))
{ {
[[unlikely]]
if (set.tags[*way].isLocked()) if (set.tags[*way].isLocked())
CACHE_LOG("Index %x Way %x Locked!!", setIdx, *way); {
// Check the other way
if (set.tags[*way ^ 1].isLocked())
{
Console.Error("CACHE: SECOND WAY IS LOCKED.", setIdx, *way);
}
else
{
// Force the unlocked way
*way ^= 1;
}
}
} }
else else
{ {
int newWay = set.tags[0].lrf() ^ set.tags[1].lrf(); int newWay = set.tags[0].lrf() ^ set.tags[1].lrf();
[[unlikely]]
if (set.tags[newWay].isLocked())
{
// If the new way is locked, we force the unlocked way, ignoring the lrf bits.
newWay = newWay ^ 1;
[[unlikely]]
if (set.tags[newWay].isLocked())
{
Console.Warning("CACHE: SECOND WAY IS LOCKED.", setIdx, *way);
}
}
*way = newWay; *way = newWay;
CacheLine line = cache.lineAt(setIdx, newWay);
CacheLine line = cache.lineAt(setIdx, newWay);
line.writeBackIfNeeded(); line.writeBackIfNeeded();
line.tag.validPFN = validPFN;
line.load(ppf); line.load(ppf);
line.tag.toggleLRF(); line.tag.toggleLRF();
} }
@ -194,10 +237,10 @@ static int getFreeCache(u32 mem, int* way)
} }
template <bool Write, int Bytes> template <bool Write, int Bytes>
void* prepareCacheAccess(u32 mem, int* way, int* idx) void* prepareCacheAccess(u32 mem, int* way, int* idx, bool validPFN = true)
{ {
*way = 0; *way = 0;
*idx = getFreeCache(mem, way); *idx = getFreeCache(mem, way, validPFN);
CacheLine line = cache.lineAt(*idx, *way); CacheLine line = cache.lineAt(*idx, *way);
if (Write) if (Write)
line.tag.setDirty(); line.tag.setDirty();
@ -206,49 +249,49 @@ void* prepareCacheAccess(u32 mem, int* way, int* idx)
} }
template <typename Int> template <typename Int>
void writeCache(u32 mem, Int value) void writeCache(u32 mem, Int value, bool validPFN)
{ {
int way, idx; int way, idx;
void* addr = prepareCacheAccess<true, sizeof(Int)>(mem, &way, &idx); void* addr = prepareCacheAccess<true, sizeof(Int)>(mem, &way, &idx, validPFN);
CACHE_LOG("writeCache%d %8.8x adding to %d, way %d, value %llx", 8 * sizeof(value), mem, idx, way, value); CACHE_LOG("writeCache%d %8.8x adding to %d, way %d, value %llx", 8 * sizeof(value), mem, idx, way, value);
*reinterpret_cast<Int*>(addr) = value; *reinterpret_cast<Int*>(addr) = value;
} }
void writeCache8(u32 mem, u8 value) void writeCache8(u32 mem, u8 value, bool validPFN)
{ {
writeCache<u8>(mem, value); writeCache<u8>(mem, value, validPFN);
} }
void writeCache16(u32 mem, u16 value) void writeCache16(u32 mem, u16 value, bool validPFN)
{ {
writeCache<u16>(mem, value); writeCache<u16>(mem, value, validPFN);
} }
void writeCache32(u32 mem, u32 value) void writeCache32(u32 mem, u32 value, bool validPFN)
{ {
writeCache<u32>(mem, value); writeCache<u32>(mem, value, validPFN);
} }
void writeCache64(u32 mem, const u64 value) void writeCache64(u32 mem, const u64 value, bool validPFN)
{ {
writeCache<u64>(mem, value); writeCache<u64>(mem, value, validPFN);
} }
void writeCache128(u32 mem, const mem128_t* value) void writeCache128(u32 mem, const mem128_t* value, bool validPFN)
{ {
int way, idx; int way, idx;
void* addr = prepareCacheAccess<true, sizeof(mem128_t)>(mem, &way, &idx); void* addr = prepareCacheAccess<true, sizeof(mem128_t)>(mem, &way, &idx, validPFN);
CACHE_LOG("writeCache128 %8.8x adding to %d, way %x, lo %llx, hi %llx", mem, idx, way, value->lo, value->hi); CACHE_LOG("writeCache128 %8.8x adding to %d, way %x, lo %llx, hi %llx", mem, idx, way, value->lo, value->hi);
*reinterpret_cast<mem128_t*>(addr) = *value; *reinterpret_cast<mem128_t*>(addr) = *value;
} }
template <typename Int> template <typename Int>
Int readCache(u32 mem) Int readCache(u32 mem, bool validPFN)
{ {
int way, idx; int way, idx;
void* addr = prepareCacheAccess<false, sizeof(Int)>(mem, &way, &idx); void* addr = prepareCacheAccess<false, sizeof(Int)>(mem, &way, &idx, validPFN);
Int value = *reinterpret_cast<Int*>(addr); Int value = *reinterpret_cast<Int*>(addr);
CACHE_LOG("readCache%d %8.8x from %d, way %d, value %llx", 8 * sizeof(value), mem, idx, way, value); CACHE_LOG("readCache%d %8.8x from %d, way %d, value %llx", 8 * sizeof(value), mem, idx, way, value);
@ -256,30 +299,30 @@ Int readCache(u32 mem)
} }
u8 readCache8(u32 mem) u8 readCache8(u32 mem, bool validPFN)
{ {
return readCache<u8>(mem); return readCache<u8>(mem, validPFN);
} }
u16 readCache16(u32 mem) u16 readCache16(u32 mem, bool validPFN)
{ {
return readCache<u16>(mem); return readCache<u16>(mem, validPFN);
} }
u32 readCache32(u32 mem) u32 readCache32(u32 mem, bool validPFN)
{ {
return readCache<u32>(mem); return readCache<u32>(mem, validPFN);
} }
u64 readCache64(u32 mem) u64 readCache64(u32 mem, bool validPFN)
{ {
return readCache<u64>(mem); return readCache<u64>(mem, validPFN);
} }
RETURNS_R128 readCache128(u32 mem) RETURNS_R128 readCache128(u32 mem, bool validPFN)
{ {
int way, idx; int way, idx;
void* addr = prepareCacheAccess<false, sizeof(mem128_t)>(mem, &way, &idx); void* addr = prepareCacheAccess<false, sizeof(mem128_t)>(mem, &way, &idx, validPFN);
r128 value = r128_load(addr); r128 value = r128_load(addr);
u64* vptr = reinterpret_cast<u64*>(&value); u64* vptr = reinterpret_cast<u64*>(&value);
CACHE_LOG("readCache128 %8.8x from %d, way %d, lo %llx, hi %llx", mem, idx, way, vptr[0], vptr[1]); CACHE_LOG("readCache128 %8.8x from %d, way %d, lo %llx, hi %llx", mem, idx, way, vptr[0], vptr[1]);
@ -402,11 +445,11 @@ void CACHE()
const int way = addr & 0x1; const int way = addr & 0x1;
CacheLine line = cache.lineAt(index, way); CacheLine line = cache.lineAt(index, way);
line.tag.setAddr(cpuRegs.CP0.n.TagLo);
line.tag.rawValue &= ~CacheTag::ALL_FLAGS; line.tag.rawValue &= ~CacheTag::ALL_FLAGS;
line.tag.rawValue |= (cpuRegs.CP0.n.TagLo & CacheTag::ALL_FLAGS); line.tag.rawValue |= (cpuRegs.CP0.n.TagLo & CacheTag::ALL_FLAGS);
CACHE_LOG("CACHE DXSTG addr %x, index %d, way %d, DATA %x OP %x", addr, index, way, cpuRegs.CP0.n.TagLo, cpuRegs.code); CACHE_LOG("CACHE DXSTG addr %x, index %d, way %d, DATA %x OP %x", addr, index, way, cpuRegs.CP0.n.TagLo, cpuRegs.code);
CACHE_LOG("WARNING: DXSTG emulation supports flags only, things will probably break");
break; break;
} }

View File

@ -8,13 +8,13 @@
#include "common/SingleRegisterTypes.h" #include "common/SingleRegisterTypes.h"
void resetCache(); void resetCache();
void writeCache8(u32 mem, u8 value); void writeCache8(u32 mem, u8 value, bool validPFN = true);
void writeCache16(u32 mem, u16 value); void writeCache16(u32 mem, u16 value, bool validPFN = true);
void writeCache32(u32 mem, u32 value); void writeCache32(u32 mem, u32 value, bool validPFN = true);
void writeCache64(u32 mem, const u64 value); void writeCache64(u32 mem, const u64 value, bool validPFN = true);
void writeCache128(u32 mem, const mem128_t* value); void writeCache128(u32 mem, const mem128_t* value, bool validPFN = true);
u8 readCache8(u32 mem); u8 readCache8(u32 mem, bool validPFN = true);
u16 readCache16(u32 mem); u16 readCache16(u32 mem, bool validPFN = true);
u32 readCache32(u32 mem); u32 readCache32(u32 mem, bool validPFN = true);
u64 readCache64(u32 mem); u64 readCache64(u32 mem, bool validPFN = true);
RETURNS_R128 readCache128(u32 mem); RETURNS_R128 readCache128(u32 mem, bool validPFN = true);

View File

@ -567,12 +567,35 @@ static void vtlbUnmappedVWriteSm(u32 addr, OperandType data) { vtlb_Miss(addr, 1
static void TAKES_R128 vtlbUnmappedVWriteLg(u32 addr, r128 data) { vtlb_Miss(addr, 1); } static void TAKES_R128 vtlbUnmappedVWriteLg(u32 addr, r128 data) { vtlb_Miss(addr, 1); }
template <typename OperandType> template <typename OperandType>
static OperandType vtlbUnmappedPReadSm(u32 addr) { vtlb_BusError(addr, 0); return 0; } static OperandType vtlbUnmappedPReadSm(u32 addr) {
static RETURNS_R128 vtlbUnmappedPReadLg(u32 addr) { vtlb_BusError(addr, 0); return r128_zero(); } vtlb_BusError(addr, 0);
if(!CHECK_EEREC && CHECK_CACHE && CheckCache(addr)){
switch (sizeof(OperandType)) {
case 1: return readCache8(addr, false);
case 2: return readCache16(addr, false);
case 4: return readCache32(addr, false);
case 8: return readCache64(addr, false);
default: pxFail("Invalid data size for unmapped physical cache load");
}
}
return 0;
}
static RETURNS_R128 vtlbUnmappedPReadLg(u32 addr) { vtlb_BusError(addr, 0); if(!CHECK_EEREC && CHECK_CACHE && CheckCache(addr)){ return readCache128(addr, false); } return r128_zero(); }
template <typename OperandType> template <typename OperandType>
static void vtlbUnmappedPWriteSm(u32 addr, OperandType data) { vtlb_BusError(addr, 1); } static void vtlbUnmappedPWriteSm(u32 addr, OperandType data) {
static void TAKES_R128 vtlbUnmappedPWriteLg(u32 addr, r128 data) { vtlb_BusError(addr, 1); } vtlb_BusError(addr, 1);
if (!CHECK_EEREC && CHECK_CACHE && CheckCache(addr)) {
switch (sizeof(OperandType)) {
case 1: writeCache8(addr, data, false); break;
case 2: writeCache16(addr, data, false); break;
case 4: writeCache32(addr, data, false); break;
case 8: writeCache64(addr, data, false); break;
default: pxFail("Invalid data size for unmapped physical cache store");
}
}
}
static void TAKES_R128 vtlbUnmappedPWriteLg(u32 addr, r128 data) { vtlb_BusError(addr, 1); if(!CHECK_EEREC && CHECK_CACHE && CheckCache(addr)) { writeCache128(addr, reinterpret_cast<mem128_t*>(&data) /*Safe??*/, false); }}
// clang-format on // clang-format on
// -------------------------------------------------------------------------------------- // --------------------------------------------------------------------------------------