mirror of
https://github.com/hrydgard/ppsspp.git
synced 2024-11-23 13:30:02 +00:00
602 lines
21 KiB
C++
602 lines
21 KiB
C++
// Copyright (c) 2021- PPSSPP Project.
|
|
|
|
// This program is free software: you can redistribute it and/or modify
|
|
// it under the terms of the GNU General Public License as published by
|
|
// the Free Software Foundation, version 2.0 or later versions.
|
|
|
|
// This program is distributed in the hope that it will be useful,
|
|
// but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
// GNU General Public License 2.0 for more details.
|
|
|
|
// A copy of the GPL 2.0 should have been included with the program.
|
|
// If not, see http://www.gnu.org/licenses/
|
|
|
|
// Official git repository and contact information can be found at
|
|
// https://github.com/hrydgard/ppsspp and http://www.ppsspp.org/.
|
|
|
|
#include "Common/Data/Convert/ColorConv.h"
|
|
#include "Common/StringUtils.h"
|
|
#include "Core/MemMap.h"
|
|
#include "GPU/Common/TextureDecoder.h"
|
|
#include "GPU/GPUState.h"
|
|
#include "GPU/Software/FuncId.h"
|
|
|
|
static_assert(sizeof(SamplerID) == sizeof(SamplerID::fullKey) + sizeof(SamplerID::cached) + sizeof(SamplerID::pad), "Bad sampler ID size");
|
|
static_assert(sizeof(PixelFuncID) == sizeof(PixelFuncID::fullKey) + sizeof(PixelFuncID::cached), "Bad pixel func ID size");
|
|
|
|
static inline GEComparison OptimizeRefByteCompare(GEComparison func, u8 ref) {
|
|
// Not equal tests are easier.
|
|
if (ref == 0 && func == GE_COMP_GREATER)
|
|
return GE_COMP_NOTEQUAL;
|
|
if (ref == 0xFF && func == GE_COMP_LESS)
|
|
return GE_COMP_NOTEQUAL;
|
|
|
|
// Sometimes games pointlessly use tests like these.
|
|
if (ref == 0 && func == GE_COMP_GEQUAL)
|
|
return GE_COMP_ALWAYS;
|
|
if (ref == 0xFF && func == GE_COMP_LEQUAL)
|
|
return GE_COMP_ALWAYS;
|
|
return func;
|
|
}
|
|
|
|
static inline PixelBlendFactor OptimizeAlphaFactor(uint32_t color) {
|
|
if (color == 0x00000000)
|
|
return PixelBlendFactor::ZERO;
|
|
if (color == 0x00FFFFFF)
|
|
return PixelBlendFactor::ONE;
|
|
return PixelBlendFactor::FIX;
|
|
}
|
|
|
|
void ComputePixelFuncID(PixelFuncID *id) {
|
|
id->fullKey = 0;
|
|
|
|
// TODO: Could this be minz > 0x0000 || maxz < 0xFFFF? Maybe unsafe, depending on verts...
|
|
id->applyDepthRange = !gstate.isModeThrough();
|
|
// Dither happens even in clear mode.
|
|
id->dithering = gstate.isDitherEnabled();
|
|
id->fbFormat = gstate.FrameBufFormat();
|
|
id->useStandardStride = gstate.FrameBufStride() == 512;
|
|
id->applyColorWriteMask = gstate.getColorMask() != 0;
|
|
|
|
id->clearMode = gstate.isModeClear();
|
|
if (id->clearMode) {
|
|
id->colorTest = gstate.isClearModeColorMask();
|
|
id->stencilTest = gstate.isClearModeAlphaMask() && gstate.FrameBufFormat() != GE_FORMAT_565;
|
|
id->depthWrite = gstate.isClearModeDepthMask();
|
|
id->depthTestFunc = GE_COMP_ALWAYS;
|
|
id->alphaTestFunc = GE_COMP_ALWAYS;
|
|
} else {
|
|
id->colorTest = gstate.isColorTestEnabled() && gstate.getColorTestFunction() != GE_COMP_ALWAYS;
|
|
if (gstate.isStencilTestEnabled() && gstate.getStencilTestFunction() == GE_COMP_ALWAYS) {
|
|
// If stencil always passes, force off when we won't write any stencil bits.
|
|
bool stencilWrite = (gstate.pmska & 0xFF) != 0xFF && gstate.FrameBufFormat() != GE_FORMAT_565;
|
|
if (gstate.isDepthTestEnabled() && gstate.getDepthTestFunction() != GE_COMP_ALWAYS)
|
|
id->stencilTest = stencilWrite && (gstate.getStencilOpZPass() != GE_STENCILOP_KEEP || gstate.getStencilOpZFail() != GE_STENCILOP_KEEP);
|
|
else
|
|
id->stencilTest = stencilWrite && gstate.getStencilOpZPass() != GE_STENCILOP_KEEP;
|
|
} else {
|
|
id->stencilTest = gstate.isStencilTestEnabled();
|
|
}
|
|
id->depthWrite = gstate.isDepthTestEnabled() && gstate.isDepthWriteEnabled();
|
|
id->depthTestFunc = gstate.isDepthTestEnabled() ? gstate.getDepthTestFunction() : GE_COMP_ALWAYS;
|
|
|
|
if (id->stencilTest) {
|
|
id->stencilTestRef = gstate.getStencilTestRef() & gstate.getStencilTestMask();
|
|
id->stencilTestFunc = OptimizeRefByteCompare(gstate.getStencilTestFunction(), id->stencilTestRef);
|
|
id->hasStencilTestMask = gstate.getStencilTestMask() != 0xFF && gstate.FrameBufFormat() != GE_FORMAT_565;
|
|
|
|
// Stencil can't be written on 565, and any invalid op acts like KEEP, which is 0.
|
|
if (gstate.FrameBufFormat() != GE_FORMAT_565 && gstate.getStencilOpSFail() <= GE_STENCILOP_DECR)
|
|
id->sFail = gstate.getStencilOpSFail();
|
|
if (gstate.FrameBufFormat() != GE_FORMAT_565 && gstate.getStencilOpZFail() <= GE_STENCILOP_DECR)
|
|
id->zFail = gstate.isDepthTestEnabled() ? gstate.getStencilOpZFail() : GE_STENCILOP_KEEP;
|
|
if (gstate.FrameBufFormat() != GE_FORMAT_565 && gstate.getStencilOpZPass() <= GE_STENCILOP_DECR)
|
|
id->zPass = gstate.getStencilOpZPass();
|
|
|
|
// Normalize REPLACE 00 to ZERO, especially if using a mask.
|
|
if (gstate.getStencilTestRef() == 0) {
|
|
if (id->SFail() == GE_STENCILOP_REPLACE)
|
|
id->sFail = GE_STENCILOP_ZERO;
|
|
if (id->ZFail() == GE_STENCILOP_REPLACE)
|
|
id->zFail = GE_STENCILOP_ZERO;
|
|
if (id->ZPass() == GE_STENCILOP_REPLACE)
|
|
id->zPass = GE_STENCILOP_ZERO;
|
|
}
|
|
|
|
// For 5551, DECR is also the same as ZERO.
|
|
if (id->FBFormat() == GE_FORMAT_5551) {
|
|
if (id->SFail() == GE_STENCILOP_DECR)
|
|
id->sFail = GE_STENCILOP_ZERO;
|
|
if (id->ZFail() == GE_STENCILOP_DECR)
|
|
id->zFail = GE_STENCILOP_ZERO;
|
|
if (id->ZPass() == GE_STENCILOP_DECR)
|
|
id->zPass = GE_STENCILOP_ZERO;
|
|
}
|
|
|
|
// And same for sFail if there's no stencil test. Prefer KEEP, though.
|
|
if (id->StencilTestFunc() == GE_COMP_ALWAYS) {
|
|
if (id->DepthTestFunc() == GE_COMP_ALWAYS)
|
|
id->zFail = GE_STENCILOP_KEEP;
|
|
id->sFail = GE_STENCILOP_KEEP;
|
|
// Always doesn't need a mask.
|
|
id->stencilTestRef = gstate.getStencilTestRef();
|
|
id->hasStencilTestMask = false;
|
|
|
|
// Turn off stencil testing if it's doing nothing.
|
|
if (id->SFail() == GE_STENCILOP_KEEP && id->ZFail() == GE_STENCILOP_KEEP && id->ZPass() == GE_STENCILOP_KEEP) {
|
|
id->stencilTest = false;
|
|
id->stencilTestFunc = 0;
|
|
id->stencilTestRef = 0;
|
|
}
|
|
} else if (id->DepthTestFunc() == GE_COMP_ALWAYS) {
|
|
// Always treat zPass/zFail the same if there's no depth test.
|
|
id->zFail = id->zPass;
|
|
}
|
|
}
|
|
|
|
id->alphaTestFunc = gstate.isAlphaTestEnabled() ? gstate.getAlphaTestFunction() : GE_COMP_ALWAYS;
|
|
if (id->AlphaTestFunc() != GE_COMP_ALWAYS) {
|
|
id->alphaTestRef = gstate.getAlphaTestRef() & gstate.getAlphaTestMask();
|
|
id->hasAlphaTestMask = gstate.getAlphaTestMask() != 0xFF;
|
|
// Try to pick a more optimal variant.
|
|
id->alphaTestFunc = OptimizeRefByteCompare(id->AlphaTestFunc(), id->alphaTestRef);
|
|
if (id->alphaTestFunc == GE_COMP_ALWAYS) {
|
|
id->alphaTestRef = 0;
|
|
id->hasAlphaTestMask = false;
|
|
}
|
|
}
|
|
|
|
// If invalid (6 or 7), doesn't do any blending, so force off.
|
|
id->alphaBlend = gstate.isAlphaBlendEnabled() && gstate.getBlendEq() <= 5;
|
|
// Force it off if the factors are constant and don't blend. Some games use this...
|
|
if (id->alphaBlend && gstate.getBlendEq() == GE_BLENDMODE_MUL_AND_ADD) {
|
|
bool srcFixedOne = gstate.getBlendFuncA() == GE_SRCBLEND_FIXA && gstate.getFixA() == 0x00FFFFFF;
|
|
bool dstFixedZero = gstate.getBlendFuncB() == GE_DSTBLEND_FIXB && gstate.getFixB() == 0x00000000;
|
|
if (srcFixedOne && dstFixedZero)
|
|
id->alphaBlend = false;
|
|
}
|
|
if (id->alphaBlend)
|
|
id->alphaBlendEq = gstate.getBlendEq();
|
|
if (id->alphaBlend && id->alphaBlendEq <= GE_BLENDMODE_MUL_AND_SUBTRACT_REVERSE) {
|
|
id->alphaBlendSrc = gstate.getBlendFuncA();
|
|
id->alphaBlendDst = gstate.getBlendFuncB();
|
|
// Special values.
|
|
if (id->alphaBlendSrc >= GE_SRCBLEND_FIXA)
|
|
id->alphaBlendSrc = (uint8_t)OptimizeAlphaFactor(gstate.getFixA());
|
|
if (id->alphaBlendDst >= GE_DSTBLEND_FIXB)
|
|
id->alphaBlendDst = (uint8_t)OptimizeAlphaFactor(gstate.getFixB());
|
|
}
|
|
|
|
if (id->colorTest && gstate.getColorTestFunction() == GE_COMP_NOTEQUAL && gstate.getColorTestRef() == 0 && gstate.getColorTestMask() == 0xFFFFFF) {
|
|
if (!id->depthWrite && !id->stencilTest && id->alphaBlend && id->AlphaBlendEq() == GE_BLENDMODE_MUL_AND_ADD) {
|
|
// Might be a pointless color test (seen in Ridge Racer, for example.)
|
|
if (id->AlphaBlendDst() == PixelBlendFactor::ONE)
|
|
id->colorTest = false;
|
|
}
|
|
}
|
|
|
|
id->applyLogicOp = gstate.isLogicOpEnabled() && gstate.getLogicOp() != GE_LOGIC_COPY;
|
|
id->applyFog = gstate.isFogEnabled() && !gstate.isModeThrough();
|
|
|
|
id->earlyZChecks = id->DepthTestFunc() != GE_COMP_ALWAYS;
|
|
if (id->stencilTest && id->earlyZChecks) {
|
|
// Can't do them early if stencil might need to write.
|
|
if (id->SFail() != GE_STENCILOP_KEEP || id->ZFail() != GE_STENCILOP_KEEP)
|
|
id->earlyZChecks = false;
|
|
}
|
|
}
|
|
|
|
if (id->useStandardStride && (id->depthTestFunc != GE_COMP_ALWAYS || id->depthWrite))
|
|
id->useStandardStride = gstate.DepthBufStride() == 512;
|
|
|
|
// Cache some values for later convenience.
|
|
if (id->dithering) {
|
|
for (int y = 0; y < 4; ++y) {
|
|
for (int x = 0; x < 4; ++x)
|
|
id->cached.ditherMatrix[y * 4 + x] = gstate.getDitherValue(x, y);
|
|
}
|
|
}
|
|
if (id->applyColorWriteMask) {
|
|
uint32_t mask = gstate.getColorMask();
|
|
// This flag means stencil clear or stencil test, basically whether writing to stencil.
|
|
if (!id->stencilTest)
|
|
mask |= 0xFF000000;
|
|
|
|
switch (id->fbFormat) {
|
|
case GE_FORMAT_565:
|
|
id->cached.colorWriteMask = RGBA8888ToRGB565(mask);
|
|
break;
|
|
|
|
case GE_FORMAT_5551:
|
|
id->cached.colorWriteMask = RGBA8888ToRGBA5551(mask);
|
|
break;
|
|
|
|
case GE_FORMAT_4444:
|
|
id->cached.colorWriteMask = RGBA8888ToRGBA4444(mask);
|
|
break;
|
|
|
|
case GE_FORMAT_8888:
|
|
id->cached.colorWriteMask = mask;
|
|
break;
|
|
}
|
|
}
|
|
if (id->applyFog) {
|
|
id->cached.fogColor = gstate.fogcolor & 0x00FFFFFF;
|
|
}
|
|
if (id->applyLogicOp)
|
|
id->cached.logicOp = gstate.getLogicOp();
|
|
id->cached.minz = gstate.getDepthRangeMin();
|
|
id->cached.maxz = gstate.getDepthRangeMax();
|
|
id->cached.framebufStride = gstate.FrameBufStride();
|
|
id->cached.depthbufStride = gstate.DepthBufStride();
|
|
|
|
if (id->hasStencilTestMask) {
|
|
// Without the mask applied, unlike the one in the key.
|
|
id->cached.stencilRef = gstate.getStencilTestRef();
|
|
id->cached.stencilTestMask = gstate.getStencilTestMask();
|
|
}
|
|
if (id->hasAlphaTestMask)
|
|
id->cached.alphaTestMask = gstate.getAlphaTestMask();
|
|
if (!id->clearMode && id->colorTest) {
|
|
id->cached.colorTestFunc = gstate.getColorTestFunction();
|
|
id->cached.colorTestMask = gstate.getColorTestMask();
|
|
id->cached.colorTestRef = gstate.getColorTestRef() & id->cached.colorTestMask;
|
|
}
|
|
if (id->alphaBlendSrc == GE_SRCBLEND_FIXA)
|
|
id->cached.alphaBlendSrc = gstate.getFixA();
|
|
if (id->alphaBlendDst == GE_DSTBLEND_FIXB)
|
|
id->cached.alphaBlendDst = gstate.getFixB();
|
|
}
|
|
|
|
std::string DescribePixelFuncID(const PixelFuncID &id) {
|
|
std::string desc;
|
|
if (id.clearMode) {
|
|
desc = "Clear";
|
|
if (id.ColorClear())
|
|
desc += "C";
|
|
if (id.StencilClear())
|
|
desc += "S";
|
|
if (id.DepthClear())
|
|
desc += "D";
|
|
desc += ":";
|
|
}
|
|
if (id.applyDepthRange)
|
|
desc += "DepthR:";
|
|
if (id.useStandardStride)
|
|
desc += "Str512:";
|
|
if (id.dithering)
|
|
desc += "Dith:";
|
|
if (id.applyColorWriteMask)
|
|
desc += "Msk:";
|
|
|
|
switch (id.fbFormat) {
|
|
case GE_FORMAT_565: desc += "5650:"; break;
|
|
case GE_FORMAT_5551: desc += "5551:"; break;
|
|
case GE_FORMAT_4444: desc += "4444:"; break;
|
|
case GE_FORMAT_8888: desc += "8888:"; break;
|
|
}
|
|
|
|
if (id.AlphaTestFunc() != GE_COMP_ALWAYS) {
|
|
if (id.clearMode)
|
|
desc = "INVALID:" + desc;
|
|
switch (id.AlphaTestFunc()) {
|
|
case GE_COMP_NEVER: desc += "ANever"; break;
|
|
case GE_COMP_ALWAYS: break;
|
|
case GE_COMP_EQUAL: desc += "AEQ"; break;
|
|
case GE_COMP_NOTEQUAL: desc += "ANE"; break;
|
|
case GE_COMP_LESS: desc += "ALT"; break;
|
|
case GE_COMP_LEQUAL: desc += "ALE"; break;
|
|
case GE_COMP_GREATER: desc += "AGT"; break;
|
|
case GE_COMP_GEQUAL: desc += "AGE"; break;
|
|
}
|
|
if (id.hasAlphaTestMask)
|
|
desc += "Msk";
|
|
desc += StringFromFormat("%02X:", id.alphaTestRef);
|
|
} else if (id.hasAlphaTestMask || id.alphaTestRef != 0) {
|
|
desc = "INVALID:" + desc;
|
|
}
|
|
|
|
if (id.earlyZChecks)
|
|
desc += "ZEarly:";
|
|
if (id.DepthTestFunc() != GE_COMP_ALWAYS) {
|
|
if (id.clearMode)
|
|
desc = "INVALID:" + desc;
|
|
switch (id.DepthTestFunc()) {
|
|
case GE_COMP_NEVER: desc += "ZNever:"; break;
|
|
case GE_COMP_ALWAYS: break;
|
|
case GE_COMP_EQUAL: desc += "ZEQ:"; break;
|
|
case GE_COMP_NOTEQUAL: desc += "ZNE:"; break;
|
|
case GE_COMP_LESS: desc += "ZLT:"; break;
|
|
case GE_COMP_LEQUAL: desc += "ZLE:"; break;
|
|
case GE_COMP_GREATER: desc += "ZGT:"; break;
|
|
case GE_COMP_GEQUAL: desc += "ZGE:"; break;
|
|
}
|
|
}
|
|
if (id.depthWrite && !id.clearMode)
|
|
desc += "ZWr:";
|
|
|
|
if (id.colorTest && !id.clearMode)
|
|
desc += "CTest:";
|
|
|
|
if (id.stencilTest && !id.clearMode) {
|
|
switch (id.StencilTestFunc()) {
|
|
case GE_COMP_NEVER: desc += "SNever"; break;
|
|
case GE_COMP_ALWAYS: desc += "SAlways"; break;
|
|
case GE_COMP_EQUAL: desc += "SEQ"; break;
|
|
case GE_COMP_NOTEQUAL: desc += "SNE"; break;
|
|
case GE_COMP_LESS: desc += "SLT"; break;
|
|
case GE_COMP_LEQUAL: desc += "SLE"; break;
|
|
case GE_COMP_GREATER: desc += "SGT"; break;
|
|
case GE_COMP_GEQUAL: desc += "SGE"; break;
|
|
}
|
|
if (id.hasStencilTestMask)
|
|
desc += "Msk";
|
|
if (id.StencilTestFunc() != GE_COMP_ALWAYS || id.DepthTestFunc() != GE_COMP_ALWAYS)
|
|
desc += StringFromFormat("%02X:", id.stencilTestRef);
|
|
} else if (id.hasStencilTestMask || id.stencilTestRef != 0 || id.stencilTestFunc != 0) {
|
|
desc = "INVALID:" + desc;
|
|
}
|
|
|
|
switch (id.SFail()) {
|
|
case GE_STENCILOP_KEEP: break;
|
|
case GE_STENCILOP_ZERO: desc += "STstF0:"; break;
|
|
case GE_STENCILOP_REPLACE: desc += "STstFRpl:"; break;
|
|
case GE_STENCILOP_INVERT: desc += "STstFXor:"; break;
|
|
case GE_STENCILOP_INCR: desc += "STstFInc:"; break;
|
|
case GE_STENCILOP_DECR: desc += "STstFDec:"; break;
|
|
}
|
|
switch (id.ZFail()) {
|
|
case GE_STENCILOP_KEEP: break;
|
|
case GE_STENCILOP_ZERO: desc += "ZTstF0:"; break;
|
|
case GE_STENCILOP_REPLACE: desc += "ZTstFRpl:"; break;
|
|
case GE_STENCILOP_INVERT: desc += "ZTstFXor:"; break;
|
|
case GE_STENCILOP_INCR: desc += "ZTstFInc:"; break;
|
|
case GE_STENCILOP_DECR: desc += "ZTstFDec:"; break;
|
|
}
|
|
if (id.StencilTestFunc() == GE_COMP_ALWAYS && id.DepthTestFunc() == GE_COMP_ALWAYS) {
|
|
switch (id.ZPass()) {
|
|
case GE_STENCILOP_KEEP: break;
|
|
case GE_STENCILOP_ZERO: desc += "Zero:"; break;
|
|
case GE_STENCILOP_REPLACE: desc += StringFromFormat("Rpl%02X:", id.stencilTestRef); break;
|
|
case GE_STENCILOP_INVERT: desc += "Xor:"; break;
|
|
case GE_STENCILOP_INCR: desc += "Inc:"; break;
|
|
case GE_STENCILOP_DECR: desc += "Dec:"; break;
|
|
}
|
|
} else {
|
|
switch (id.ZPass()) {
|
|
case GE_STENCILOP_KEEP: break;
|
|
case GE_STENCILOP_ZERO: desc += "ZTstT0:"; break;
|
|
case GE_STENCILOP_REPLACE: desc += "ZTstTRpl:"; break;
|
|
case GE_STENCILOP_INVERT: desc += "ZTstTXor:"; break;
|
|
case GE_STENCILOP_INCR: desc += "ZTstTInc:"; break;
|
|
case GE_STENCILOP_DECR: desc += "ZTstTDec:"; break;
|
|
}
|
|
}
|
|
if (!id.stencilTest || id.clearMode) {
|
|
if (id.sFail != 0 || id.zFail != 0 || id.zPass != 0)
|
|
desc = "INVALID:" + desc;
|
|
}
|
|
|
|
if (id.alphaBlend) {
|
|
if (id.clearMode)
|
|
desc = "INVALID:" + desc;
|
|
switch (id.AlphaBlendEq()) {
|
|
case GE_BLENDMODE_MUL_AND_ADD: desc += "BlendAdd<"; break;
|
|
case GE_BLENDMODE_MUL_AND_SUBTRACT: desc += "BlendSub<"; break;
|
|
case GE_BLENDMODE_MUL_AND_SUBTRACT_REVERSE: desc += "BlendRSub<"; break;
|
|
case GE_BLENDMODE_MIN: desc += "BlendMin<"; break;
|
|
case GE_BLENDMODE_MAX: desc += "BlendMax<"; break;
|
|
case GE_BLENDMODE_ABSDIFF: desc += "BlendDiff<"; break;
|
|
}
|
|
switch (id.AlphaBlendSrc()) {
|
|
case PixelBlendFactor::OTHERCOLOR: desc += "DstRGB,"; break;
|
|
case PixelBlendFactor::INVOTHERCOLOR: desc += "1-DstRGB,"; break;
|
|
case PixelBlendFactor::SRCALPHA: desc += "SrcA,"; break;
|
|
case PixelBlendFactor::INVSRCALPHA: desc += "1-SrcA,"; break;
|
|
case PixelBlendFactor::DSTALPHA: desc += "DstA,"; break;
|
|
case PixelBlendFactor::INVDSTALPHA: desc += "1-DstA,"; break;
|
|
case PixelBlendFactor::DOUBLESRCALPHA: desc += "2*SrcA,"; break;
|
|
case PixelBlendFactor::DOUBLEINVSRCALPHA: desc += "1-2*SrcA,"; break;
|
|
case PixelBlendFactor::DOUBLEDSTALPHA: desc += "2*DstA,"; break;
|
|
case PixelBlendFactor::DOUBLEINVDSTALPHA: desc += "1-2*DstA,"; break;
|
|
case PixelBlendFactor::FIX: desc += "Fix,"; break;
|
|
case PixelBlendFactor::ZERO: desc += "0,"; break;
|
|
case PixelBlendFactor::ONE: desc += "1,"; break;
|
|
}
|
|
switch (id.AlphaBlendDst()) {
|
|
case PixelBlendFactor::OTHERCOLOR: desc += "SrcRGB>:"; break;
|
|
case PixelBlendFactor::INVOTHERCOLOR: desc += "1-SrcRGB>:"; break;
|
|
case PixelBlendFactor::SRCALPHA: desc += "SrcA>:"; break;
|
|
case PixelBlendFactor::INVSRCALPHA: desc += "1-SrcA>:"; break;
|
|
case PixelBlendFactor::DSTALPHA: desc += "DstA>:"; break;
|
|
case PixelBlendFactor::INVDSTALPHA: desc += "1-DstA>:"; break;
|
|
case PixelBlendFactor::DOUBLESRCALPHA: desc += "2*SrcA>:"; break;
|
|
case PixelBlendFactor::DOUBLEINVSRCALPHA: desc += "1-2*SrcA>:"; break;
|
|
case PixelBlendFactor::DOUBLEDSTALPHA: desc += "2*DstA>:"; break;
|
|
case PixelBlendFactor::DOUBLEINVDSTALPHA: desc += "1-2*DstA>:"; break;
|
|
case PixelBlendFactor::FIX: desc += "Fix>:"; break;
|
|
case PixelBlendFactor::ZERO: desc += "0>:"; break;
|
|
case PixelBlendFactor::ONE: desc += "1>:"; break;
|
|
}
|
|
} else if (id.alphaBlendEq != 0 || id.alphaBlendSrc != 0 || id.alphaBlendDst != 0) {
|
|
desc = "INVALID:" + desc;
|
|
}
|
|
|
|
if (id.applyLogicOp)
|
|
desc += "Logic:";
|
|
else if (id.clearMode)
|
|
desc = "INVALID:" + desc;
|
|
if (id.applyFog)
|
|
desc += "Fog:";
|
|
else if (id.clearMode)
|
|
desc = "INVALID:" + desc;
|
|
|
|
if (desc.empty())
|
|
return "INVALID";
|
|
desc.resize(desc.size() - 1);
|
|
return desc;
|
|
}
|
|
|
|
void ComputeSamplerID(SamplerID *id_out) {
|
|
SamplerID id{};
|
|
|
|
id.useStandardBufw = true;
|
|
id.overReadSafe = true;
|
|
int maxLevel = gstate.isMipmapEnabled() ? gstate.getTextureMaxLevel() : 0;
|
|
GETextureFormat fmt = gstate.getTextureFormat();
|
|
for (int i = 0; i <= maxLevel; ++i) {
|
|
uint32_t addr = gstate.getTextureAddress(i);
|
|
if (!Memory::IsValidAddress(addr))
|
|
id.hasInvalidPtr = true;
|
|
|
|
int bufw = GetTextureBufw(i, addr, fmt);
|
|
int bitspp = textureBitsPerPixel[fmt];
|
|
// We use a 16 byte minimum for all small bufws, so allow those as standard.
|
|
int w = gstate.getTextureWidth(i);
|
|
if (bitspp == 0 || std::max(w, 128 / bitspp) != bufw)
|
|
id.useStandardBufw = false;
|
|
// TODO: Verify 16 bit bufw align handling in DXT.
|
|
if (fmt >= GE_TFMT_DXT1 && w != bufw)
|
|
id.useStandardBufw = false;
|
|
|
|
int h = gstate.getTextureHeight(i);
|
|
int bytes = h * (bufw * bitspp) / 8;
|
|
if (bitspp < 32 && !Memory::IsValidAddress(addr + bytes + (32 - bitspp) / 8))
|
|
id.overReadSafe = false;
|
|
|
|
id.cached.sizes[i].w = w;
|
|
id.cached.sizes[i].h = h;
|
|
}
|
|
// TODO: What specifically happens if these are above 11?
|
|
id.width0Shift = gstate.texsize[0] & 0xF;
|
|
id.height0Shift = (gstate.texsize[0] >> 8) & 0xF;
|
|
id.hasAnyMips = maxLevel != 0;
|
|
|
|
id.texfmt = fmt;
|
|
id.swizzle = gstate.isTextureSwizzled();
|
|
// Only CLUT4 can use separate CLUTs per mimap.
|
|
id.useSharedClut = fmt != GE_TFMT_CLUT4 || maxLevel == 0 || !gstate.isMipmapEnabled() || gstate.isClutSharedForMipmaps();
|
|
if (gstate.isTextureFormatIndexed()) {
|
|
id.clutfmt = gstate.getClutPaletteFormat();
|
|
id.hasClutMask = gstate.getClutIndexMask() != 0xFF;
|
|
id.hasClutShift = gstate.getClutIndexShift() != 0;
|
|
id.hasClutOffset = gstate.getClutIndexStartPos() != 0;
|
|
id.cached.clutFormat = gstate.clutformat;
|
|
}
|
|
|
|
id.clampS = gstate.isTexCoordClampedS();
|
|
id.clampT = gstate.isTexCoordClampedT();
|
|
|
|
id.useTextureAlpha = gstate.isTextureAlphaUsed();
|
|
id.useColorDoubling = gstate.isColorDoublingEnabled();
|
|
id.texFunc = gstate.getTextureFunction();
|
|
if (id.texFunc > GE_TEXFUNC_ADD)
|
|
id.texFunc = GE_TEXFUNC_ADD;
|
|
|
|
if (id.texFunc == GE_TEXFUNC_BLEND)
|
|
id.cached.texBlendColor = gstate.getTextureEnvColRGB();
|
|
|
|
*id_out = id;
|
|
}
|
|
|
|
std::string DescribeSamplerID(const SamplerID &id) {
|
|
std::string name;
|
|
switch (id.TexFmt()) {
|
|
case GE_TFMT_5650: name = "5650"; break;
|
|
case GE_TFMT_5551: name = "5551"; break;
|
|
case GE_TFMT_4444: name = "4444"; break;
|
|
case GE_TFMT_8888: name = "8888"; break;
|
|
case GE_TFMT_CLUT4: name = "CLUT4"; break;
|
|
case GE_TFMT_CLUT8: name = "CLUT8"; break;
|
|
case GE_TFMT_CLUT16: name = "CLUT16"; break;
|
|
case GE_TFMT_CLUT32: name = "CLUT32"; break;
|
|
case GE_TFMT_DXT1: name = "DXT1"; break;
|
|
case GE_TFMT_DXT3: name = "DXT3"; break;
|
|
case GE_TFMT_DXT5: name = "DXT5"; break;
|
|
default: name = "INVALID"; break;
|
|
}
|
|
switch (id.ClutFmt()) {
|
|
case GE_CMODE_16BIT_BGR5650:
|
|
switch (id.TexFmt()) {
|
|
case GE_TFMT_CLUT4:
|
|
case GE_TFMT_CLUT8:
|
|
case GE_TFMT_CLUT16:
|
|
case GE_TFMT_CLUT32:
|
|
name += ":C5650";
|
|
break;
|
|
default:
|
|
// Ignore 0 clutfmt when no clut.
|
|
break;
|
|
}
|
|
break;
|
|
case GE_CMODE_16BIT_ABGR5551: name += ":C5551"; break;
|
|
case GE_CMODE_16BIT_ABGR4444: name += ":C4444"; break;
|
|
case GE_CMODE_32BIT_ABGR8888: name += ":C8888"; break;
|
|
}
|
|
if (id.swizzle) {
|
|
name += ":SWZ";
|
|
}
|
|
if (!id.useSharedClut) {
|
|
name += ":CMIP";
|
|
}
|
|
if (id.hasInvalidPtr) {
|
|
name += ":INV";
|
|
}
|
|
if (id.hasClutMask) {
|
|
name += ":CMASK";
|
|
}
|
|
if (id.hasClutShift) {
|
|
name += ":CSHF";
|
|
}
|
|
if (id.hasClutOffset) {
|
|
name += ":COFF";
|
|
}
|
|
if (id.clampS || id.clampT) {
|
|
name += std::string(":CL") + (id.clampS ? "S" : "") + (id.clampT ? "T" : "");
|
|
}
|
|
if (!id.useStandardBufw) {
|
|
name += ":BUFW";
|
|
}
|
|
if (!id.overReadSafe) {
|
|
name += ":XRD";
|
|
}
|
|
if (id.hasAnyMips) {
|
|
name += ":MIP";
|
|
}
|
|
if (id.linear) {
|
|
name += ":LERP";
|
|
}
|
|
if (id.fetch) {
|
|
name += ":FETCH";
|
|
}
|
|
if (id.useTextureAlpha) {
|
|
name += ":A";
|
|
}
|
|
if (id.useColorDoubling) {
|
|
name += ":DBL";
|
|
}
|
|
switch (id.texFunc) {
|
|
case GE_TEXFUNC_MODULATE:
|
|
name += ":MOD";
|
|
break;
|
|
case GE_TEXFUNC_DECAL:
|
|
name += ":DECAL";
|
|
break;
|
|
case GE_TEXFUNC_BLEND:
|
|
name += ":BLEND";
|
|
break;
|
|
case GE_TEXFUNC_REPLACE:
|
|
break;
|
|
case GE_TEXFUNC_ADD:
|
|
name += ":ADD";
|
|
default:
|
|
break;
|
|
}
|
|
name += StringFromFormat(":W%dH%d", 1 << id.width0Shift, 1 << id.height0Shift);
|
|
if (id.width0Shift > 10 || id.height0Shift > 10)
|
|
name = "INVALID:" + name;
|
|
|
|
return name;
|
|
}
|