GS/HW: Fix fixed/invalid TEX0 case in TimeSplitters 2

This commit is contained in:
Stenzek 2023-02-10 12:42:38 +10:00 committed by refractionpcsx2
parent bca49184e7
commit 1b0c03d892
2 changed files with 34 additions and 13 deletions

View File

@ -1959,9 +1959,13 @@ GSTextureCache::Source* GSTextureCache::CreateSource(const GIFRegTEX0& TEX0, con
const GSLocalMemory::psm_t& psm = GSLocalMemory::m_psm[TEX0.PSM]; const GSLocalMemory::psm_t& psm = GSLocalMemory::m_psm[TEX0.PSM];
Source* src = new Source(TEX0, TEXA); Source* src = new Source(TEX0, TEXA);
int tw = 1 << TEX0.TW; // Normally we wouldn't use the region with targets, but for the case where we're drawing UVs and the
int th = 1 << TEX0.TH; // clamp rectangle exceeds the TW/TH (which is now unused), we do need to use it. Timesplitters 2 does
//int tp = TEX0.TBW << 6; // its frame blending effect using a smaller TW/TH, *and* triangles instead of sprites just to be extra
// annoying.
int tw = region.IsFixedTEX0W(1 << TEX0.TW) ? region.GetWidth() : (1 << TEX0.TW);
int th = region.IsFixedTEX0H(1 << TEX0.TH) ? region.GetHeight() : (1 << TEX0.TH);
int tlevels = 1; int tlevels = 1;
if (lod) if (lod)
{ {
@ -2176,16 +2180,21 @@ GSTextureCache::Source* GSTextureCache::CreateSource(const GIFRegTEX0& TEX0, con
const GSDrawingContext* const context = g_gs_renderer->m_context; const GSDrawingContext* const context = g_gs_renderer->m_context;
if (context->FRAME.Block() == TEX0.TBP0 || context->ZBUF.Block() == TEX0.TBP0) if (context->FRAME.Block() == TEX0.TBP0 || context->ZBUF.Block() == TEX0.TBP0)
{ {
// if it looks like a texture shuffle, we might read up to +/- 8 pixels on either side. // For the TS2 case above, src_range is going to be incorrect, since TW/TH are incorrect.
GSVector4 adjusted_src_range(*src_range); // We can remove this check once we move it to tex-is-fb instead.
if (GSRendererHW::GetInstance()->IsPossibleTextureShuffle(dst, TEX0)) if (!region.IsFixedTEX0(1 << TEX0.TW, 1 << TEX0.TH))
adjusted_src_range += GSVector4(-8.0f, 0.0f, 8.0f, 0.0f); {
// if it looks like a texture shuffle, we might read up to +/- 8 pixels on either side.
GSVector4 adjusted_src_range(*src_range);
if (GSRendererHW::GetInstance()->IsPossibleTextureShuffle(dst, TEX0))
adjusted_src_range += GSVector4(-8.0f, 0.0f, 8.0f, 0.0f);
// don't forget to scale the copy range // don't forget to scale the copy range
adjusted_src_range = adjusted_src_range * GSVector4(scale).xyxy(); adjusted_src_range = adjusted_src_range * GSVector4(scale).xyxy();
sRect = sRect.rintersect(GSVector4i(adjusted_src_range)); sRect = sRect.rintersect(GSVector4i(adjusted_src_range));
destX = sRect.x; destX = sRect.x;
destY = sRect.y; destY = sRect.y;
}
// clean up immediately afterwards // clean up immediately afterwards
m_temporary_source = src; m_temporary_source = src;
@ -3748,7 +3757,17 @@ bool GSTextureCache::SurfaceOffsetKeyEqual::operator()(const GSTextureCache::Sur
bool GSTextureCache::SourceRegion::IsFixedTEX0(int tw, int th) const bool GSTextureCache::SourceRegion::IsFixedTEX0(int tw, int th) const
{ {
return (GetMinX() >= static_cast<u32>(tw) || GetMinY() >= static_cast<u32>(th)); return IsFixedTEX0W(tw) || IsFixedTEX0H(th);
}
bool GSTextureCache::SourceRegion::IsFixedTEX0W(int tw) const
{
return (GetMaxX() > static_cast<u32>(tw));
}
bool GSTextureCache::SourceRegion::IsFixedTEX0H(int th) const
{
return (GetMaxY() > static_cast<u32>(th));
} }
GSVector4i GSTextureCache::SourceRegion::GetRect(int tw, int th) const GSVector4i GSTextureCache::SourceRegion::GetRect(int tw, int th) const

View File

@ -65,6 +65,8 @@ public:
/// Returns true if the area of the region exceeds the TW/TH size (i.e. "fixed tex0"). /// Returns true if the area of the region exceeds the TW/TH size (i.e. "fixed tex0").
bool IsFixedTEX0(int tw, int th) const; bool IsFixedTEX0(int tw, int th) const;
bool IsFixedTEX0W(int tw) const;
bool IsFixedTEX0H(int th) const;
/// Returns the rectangle relative to the texture base pointer that the region occupies. /// Returns the rectangle relative to the texture base pointer that the region occupies.
GSVector4i GetRect(int tw, int th) const; GSVector4i GetRect(int tw, int th) const;