2015-10-24 21:49:05 +00:00
|
|
|
#pragma once
|
|
|
|
|
2020-11-08 22:17:06 +00:00
|
|
|
#include <cstdint>
|
2017-02-24 19:26:38 +00:00
|
|
|
#include "Common/CommonTypes.h"
|
|
|
|
|
2015-11-04 08:35:55 +00:00
|
|
|
#include "GPU/ge_constants.h"
|
2022-08-28 05:11:00 +00:00
|
|
|
#include "GPU/GPUState.h"
|
2015-11-04 08:35:55 +00:00
|
|
|
|
2016-12-26 22:51:17 +00:00
|
|
|
// TODO: Replace enums and structs with same from thin3d.h, for convenient mapping.
|
|
|
|
|
2015-10-24 21:49:05 +00:00
|
|
|
enum StencilValueType {
|
|
|
|
STENCIL_VALUE_UNIFORM,
|
|
|
|
STENCIL_VALUE_ZERO,
|
|
|
|
STENCIL_VALUE_ONE,
|
|
|
|
STENCIL_VALUE_KEEP,
|
|
|
|
STENCIL_VALUE_INVERT,
|
|
|
|
STENCIL_VALUE_INCR_4,
|
|
|
|
STENCIL_VALUE_INCR_8,
|
|
|
|
STENCIL_VALUE_DECR_4,
|
|
|
|
STENCIL_VALUE_DECR_8,
|
2015-03-16 02:25:34 +00:00
|
|
|
};
|
|
|
|
|
2015-10-24 21:49:05 +00:00
|
|
|
enum ReplaceAlphaType {
|
|
|
|
REPLACE_ALPHA_NO = 0,
|
|
|
|
REPLACE_ALPHA_YES = 1,
|
|
|
|
REPLACE_ALPHA_DUALSOURCE = 2,
|
|
|
|
};
|
2015-03-16 02:25:34 +00:00
|
|
|
|
2015-10-24 21:49:05 +00:00
|
|
|
enum ReplaceBlendType {
|
2020-11-08 22:17:06 +00:00
|
|
|
REPLACE_BLEND_NO, // Blend function handled directly with blend states.
|
|
|
|
|
2015-10-24 21:49:05 +00:00
|
|
|
REPLACE_BLEND_STANDARD,
|
2020-11-08 22:17:06 +00:00
|
|
|
|
|
|
|
// SRC part of blend function handled in-shader.
|
2015-10-24 21:49:05 +00:00
|
|
|
REPLACE_BLEND_PRE_SRC,
|
|
|
|
REPLACE_BLEND_PRE_SRC_2X_ALPHA,
|
|
|
|
REPLACE_BLEND_2X_ALPHA,
|
|
|
|
REPLACE_BLEND_2X_SRC,
|
2020-11-08 22:17:06 +00:00
|
|
|
|
|
|
|
// Full blend equation runs in shader.
|
|
|
|
// We might have to make a copy of the framebuffer target to read from.
|
2015-10-24 21:49:05 +00:00
|
|
|
REPLACE_BLEND_COPY_FBO,
|
2022-04-24 18:53:09 +00:00
|
|
|
|
|
|
|
// Color blend mode and color gets copied to alpha blend mode.
|
|
|
|
REPLACE_BLEND_BLUE_TO_ALPHA,
|
2015-10-24 21:49:05 +00:00
|
|
|
};
|
2015-03-16 02:25:34 +00:00
|
|
|
|
2015-10-24 21:49:05 +00:00
|
|
|
enum LogicOpReplaceType {
|
|
|
|
LOGICOPTYPE_NORMAL,
|
|
|
|
LOGICOPTYPE_ONE,
|
|
|
|
LOGICOPTYPE_INVERT,
|
|
|
|
};
|
2015-03-16 02:25:34 +00:00
|
|
|
|
2015-10-24 21:49:05 +00:00
|
|
|
bool IsAlphaTestTriviallyTrue();
|
|
|
|
bool IsColorTestAgainstZero();
|
|
|
|
bool IsColorTestTriviallyTrue();
|
|
|
|
bool IsAlphaTestAgainstZero();
|
2018-07-28 03:04:36 +00:00
|
|
|
bool NeedsTestDiscard();
|
2018-12-01 22:05:29 +00:00
|
|
|
bool IsStencilTestOutputDisabled();
|
2015-03-16 02:25:34 +00:00
|
|
|
|
2015-10-24 21:49:05 +00:00
|
|
|
StencilValueType ReplaceAlphaWithStencilType();
|
|
|
|
ReplaceAlphaType ReplaceAlphaWithStencil(ReplaceBlendType replaceBlend);
|
2015-11-04 08:35:55 +00:00
|
|
|
ReplaceBlendType ReplaceBlendWithShader(bool allowShaderBlend, GEBufferFormat bufferFormat);
|
2015-03-16 02:25:34 +00:00
|
|
|
|
2015-10-24 21:49:05 +00:00
|
|
|
LogicOpReplaceType ReplaceLogicOpType();
|
2015-11-08 17:13:34 +00:00
|
|
|
|
|
|
|
// Common representation, should be able to set this directly with any modern API.
|
|
|
|
struct ViewportAndScissor {
|
|
|
|
int scissorX;
|
|
|
|
int scissorY;
|
|
|
|
int scissorW;
|
|
|
|
int scissorH;
|
|
|
|
float viewportX;
|
|
|
|
float viewportY;
|
|
|
|
float viewportW;
|
|
|
|
float viewportH;
|
|
|
|
float depthRangeMin;
|
|
|
|
float depthRangeMax;
|
2022-08-20 21:16:55 +00:00
|
|
|
float widthScale;
|
|
|
|
float heightScale;
|
|
|
|
float depthScale;
|
|
|
|
float xOffset;
|
|
|
|
float yOffset;
|
|
|
|
float zOffset;
|
|
|
|
bool throughMode;
|
2015-11-08 17:13:34 +00:00
|
|
|
};
|
|
|
|
void ConvertViewportAndScissor(bool useBufferedRendering, float renderWidth, float renderHeight, int bufferWidth, int bufferHeight, ViewportAndScissor &out);
|
2022-08-20 21:16:55 +00:00
|
|
|
void UpdateCachedViewportState(const ViewportAndScissor &vpAndScissor);
|
2019-02-10 17:58:01 +00:00
|
|
|
float ToScaledDepthFromIntegerScale(float z);
|
2020-08-07 09:19:48 +00:00
|
|
|
|
|
|
|
struct DepthScaleFactors {
|
|
|
|
float offset;
|
|
|
|
float scale;
|
|
|
|
|
|
|
|
float Apply(float z) const {
|
|
|
|
return (z - offset) * scale;
|
|
|
|
}
|
2022-08-01 09:57:22 +00:00
|
|
|
|
|
|
|
float ApplyInverse(float z) const {
|
|
|
|
return (z / scale) + offset;
|
|
|
|
}
|
2020-08-07 09:19:48 +00:00
|
|
|
};
|
|
|
|
DepthScaleFactors GetDepthScaleFactors();
|
|
|
|
|
2016-01-20 04:19:50 +00:00
|
|
|
float DepthSliceFactor();
|
2015-11-08 19:09:30 +00:00
|
|
|
|
|
|
|
// These are common to all modern APIs and can be easily converted with a lookup table.
|
|
|
|
enum class BlendFactor : uint8_t {
|
|
|
|
ZERO,
|
|
|
|
ONE,
|
|
|
|
SRC_COLOR,
|
|
|
|
ONE_MINUS_SRC_COLOR,
|
|
|
|
DST_COLOR,
|
|
|
|
ONE_MINUS_DST_COLOR,
|
|
|
|
SRC_ALPHA,
|
|
|
|
ONE_MINUS_SRC_ALPHA,
|
|
|
|
DST_ALPHA,
|
|
|
|
ONE_MINUS_DST_ALPHA,
|
|
|
|
CONSTANT_COLOR,
|
|
|
|
ONE_MINUS_CONSTANT_COLOR,
|
|
|
|
CONSTANT_ALPHA,
|
|
|
|
ONE_MINUS_CONSTANT_ALPHA,
|
2016-12-26 22:51:17 +00:00
|
|
|
SRC1_COLOR,
|
|
|
|
ONE_MINUS_SRC1_COLOR,
|
2015-11-08 19:09:30 +00:00
|
|
|
SRC1_ALPHA,
|
|
|
|
ONE_MINUS_SRC1_ALPHA,
|
|
|
|
INVALID,
|
|
|
|
COUNT,
|
|
|
|
};
|
|
|
|
|
|
|
|
enum class BlendEq : uint8_t {
|
|
|
|
ADD,
|
|
|
|
SUBTRACT,
|
|
|
|
REVERSE_SUBTRACT,
|
|
|
|
MIN,
|
|
|
|
MAX,
|
|
|
|
COUNT
|
|
|
|
};
|
|
|
|
|
|
|
|
struct GenericBlendState {
|
|
|
|
bool enabled;
|
2020-11-08 11:49:06 +00:00
|
|
|
bool resetFramebufferRead;
|
|
|
|
bool applyFramebufferRead;
|
|
|
|
bool dirtyShaderBlendFixValues;
|
2015-11-08 20:47:08 +00:00
|
|
|
ReplaceAlphaType replaceAlphaWithStencil;
|
2015-11-08 19:09:30 +00:00
|
|
|
|
|
|
|
BlendFactor srcColor;
|
|
|
|
BlendFactor dstColor;
|
|
|
|
BlendFactor srcAlpha;
|
|
|
|
BlendFactor dstAlpha;
|
|
|
|
|
|
|
|
BlendEq eqColor;
|
|
|
|
BlendEq eqAlpha;
|
|
|
|
|
2015-11-08 21:29:43 +00:00
|
|
|
bool useBlendColor;
|
|
|
|
u32 blendColor;
|
|
|
|
|
2015-11-08 19:09:30 +00:00
|
|
|
void setFactors(BlendFactor srcC, BlendFactor dstC, BlendFactor srcA, BlendFactor dstA) {
|
|
|
|
srcColor = srcC;
|
|
|
|
dstColor = dstC;
|
|
|
|
srcAlpha = srcA;
|
|
|
|
dstAlpha = dstA;
|
|
|
|
}
|
|
|
|
void setEquation(BlendEq eqC, BlendEq eqA) {
|
|
|
|
eqColor = eqC;
|
|
|
|
eqAlpha = eqA;
|
|
|
|
}
|
2015-11-08 21:29:43 +00:00
|
|
|
void setBlendColor(uint32_t color, uint8_t alpha) {
|
2015-11-08 21:50:16 +00:00
|
|
|
blendColor = color | ((uint32_t)alpha << 24);
|
2015-11-08 21:29:43 +00:00
|
|
|
useBlendColor = true;
|
|
|
|
}
|
|
|
|
void defaultBlendColor(uint8_t alpha) {
|
2015-11-08 21:50:16 +00:00
|
|
|
blendColor = 0xFFFFFF | ((uint32_t)alpha << 24);
|
2015-11-08 21:29:43 +00:00
|
|
|
useBlendColor = true;
|
|
|
|
}
|
2015-11-08 19:09:30 +00:00
|
|
|
};
|
2015-11-08 21:32:48 +00:00
|
|
|
|
2022-04-24 22:06:21 +00:00
|
|
|
void ConvertBlendState(GenericBlendState &blendState, bool allowShaderBlend, bool forceReplaceBlend);
|
2020-11-08 13:34:04 +00:00
|
|
|
void ApplyStencilReplaceAndLogicOpIgnoreBlend(ReplaceAlphaType replaceAlphaWithStencil, GenericBlendState &blendState);
|
2015-12-31 09:50:18 +00:00
|
|
|
|
2020-11-08 22:17:06 +00:00
|
|
|
struct GenericMaskState {
|
|
|
|
bool applyFramebufferRead;
|
|
|
|
uint32_t uniformMask; // For each bit, opposite to the PSP.
|
|
|
|
bool rgba[4]; // true = draw, false = don't draw this channel
|
|
|
|
};
|
|
|
|
|
|
|
|
void ConvertMaskState(GenericMaskState &maskState, bool allowFramebufferRead);
|
|
|
|
bool IsColorWriteMaskComplex(bool allowFramebufferRead);
|
|
|
|
|
2015-12-31 09:50:18 +00:00
|
|
|
struct GenericStencilFuncState {
|
|
|
|
bool enabled;
|
|
|
|
GEComparison testFunc;
|
|
|
|
u8 testRef;
|
|
|
|
u8 testMask;
|
|
|
|
u8 writeMask;
|
|
|
|
GEStencilOp sFail;
|
|
|
|
GEStencilOp zFail;
|
|
|
|
GEStencilOp zPass;
|
|
|
|
};
|
|
|
|
|
|
|
|
void ConvertStencilFuncState(GenericStencilFuncState &stencilFuncState);
|
2022-08-28 05:11:00 +00:00
|
|
|
|
|
|
|
// See issue #15898
|
|
|
|
inline bool SpongebobDepthInverseConditions(const GenericStencilFuncState &stencilState) {
|
2022-08-28 05:25:08 +00:00
|
|
|
// Check that the depth/stencil state matches the conditions exactly
|
|
|
|
return gstate.isDepthTestEnabled() && !gstate.isDepthWriteEnabled() &&
|
|
|
|
gstate.getDepthTestFunction() == GE_COMP_GEQUAL &&
|
|
|
|
stencilState.zFail == GE_STENCILOP_ZERO && stencilState.sFail == GE_STENCILOP_KEEP && stencilState.zPass == GE_STENCILOP_KEEP &&
|
2022-08-28 05:11:00 +00:00
|
|
|
stencilState.testFunc == GE_COMP_ALWAYS &&
|
|
|
|
stencilState.writeMask == 0xFF && stencilState.testMask == 0xFF && stencilState.testRef == 0xFF &&
|
2022-08-28 05:25:08 +00:00
|
|
|
// And also verify no color is written. The game does this through simple alpha blending with a constant zero alpha.
|
|
|
|
// We also check for color mask, since it's more natural, in case another game does it.
|
|
|
|
(gstate.isAlphaBlendEnabled() &&
|
|
|
|
gstate.getBlendFuncA() == GE_SRCBLEND_SRCALPHA &&
|
|
|
|
gstate.getBlendFuncB() == GE_DSTBLEND_INVSRCALPHA &&
|
|
|
|
gstate.getMaterialAmbientA() == 0x0 && // our accessor is kinda misnamed here, but material diffuse A is both used as default color and as ambient alpha
|
|
|
|
!gstate.isTextureMapEnabled()
|
|
|
|
) || gstate.getColorMask() == 0xFFFFFF00; // note that PSP masks are "inverted"
|
2022-08-28 05:11:00 +00:00
|
|
|
}
|