2017-02-08 16:35:41 +00:00
|
|
|
// Copyright (c) 2012- 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 <d3d11.h>
|
2017-02-18 00:10:46 +00:00
|
|
|
#include <d3d11_1.h>
|
2017-02-08 16:35:41 +00:00
|
|
|
|
2020-09-14 22:34:45 +00:00
|
|
|
#include <algorithm>
|
|
|
|
|
2020-10-03 22:25:21 +00:00
|
|
|
#include "Common/Data/Convert/SmallDataConvert.h"
|
2017-02-08 17:07:34 +00:00
|
|
|
|
2017-02-08 16:35:41 +00:00
|
|
|
#include "GPU/Math3D.h"
|
|
|
|
#include "GPU/GPUState.h"
|
|
|
|
#include "GPU/ge_constants.h"
|
|
|
|
#include "GPU/Common/GPUStateUtils.h"
|
|
|
|
#include "Core/System.h"
|
|
|
|
#include "Core/Config.h"
|
|
|
|
#include "Core/Reporting.h"
|
|
|
|
|
2020-08-03 21:17:22 +00:00
|
|
|
#include "GPU/Common/FramebufferManagerCommon.h"
|
2017-02-08 17:07:34 +00:00
|
|
|
#include "GPU/D3D11/DrawEngineD3D11.h"
|
2017-02-12 17:29:58 +00:00
|
|
|
#include "GPU/D3D11/StateMappingD3D11.h"
|
2017-02-08 17:07:34 +00:00
|
|
|
#include "GPU/D3D11/FramebufferManagerD3D11.h"
|
2017-02-09 23:30:42 +00:00
|
|
|
#include "GPU/D3D11/TextureCacheD3D11.h"
|
2017-02-08 16:35:41 +00:00
|
|
|
|
|
|
|
// These tables all fit into u8s.
|
|
|
|
static const D3D11_BLEND d3d11BlendFactorLookup[(size_t)BlendFactor::COUNT] = {
|
|
|
|
D3D11_BLEND_ZERO,
|
|
|
|
D3D11_BLEND_ONE,
|
|
|
|
D3D11_BLEND_SRC_COLOR,
|
|
|
|
D3D11_BLEND_INV_SRC_COLOR,
|
|
|
|
D3D11_BLEND_DEST_COLOR,
|
|
|
|
D3D11_BLEND_INV_DEST_COLOR,
|
|
|
|
D3D11_BLEND_SRC_ALPHA,
|
|
|
|
D3D11_BLEND_INV_SRC_ALPHA,
|
|
|
|
D3D11_BLEND_DEST_ALPHA,
|
|
|
|
D3D11_BLEND_INV_DEST_ALPHA,
|
|
|
|
D3D11_BLEND_BLEND_FACTOR,
|
|
|
|
D3D11_BLEND_INV_BLEND_FACTOR,
|
|
|
|
D3D11_BLEND_BLEND_FACTOR,
|
|
|
|
D3D11_BLEND_INV_BLEND_FACTOR,
|
|
|
|
D3D11_BLEND_SRC1_COLOR,
|
|
|
|
D3D11_BLEND_INV_SRC1_COLOR,
|
|
|
|
D3D11_BLEND_SRC1_ALPHA,
|
|
|
|
D3D11_BLEND_INV_SRC1_ALPHA,
|
|
|
|
};
|
|
|
|
|
|
|
|
static const D3D11_BLEND_OP d3d11BlendEqLookup[(size_t)BlendEq::COUNT] = {
|
|
|
|
D3D11_BLEND_OP_ADD,
|
|
|
|
D3D11_BLEND_OP_SUBTRACT,
|
|
|
|
D3D11_BLEND_OP_REV_SUBTRACT,
|
|
|
|
D3D11_BLEND_OP_MIN,
|
|
|
|
D3D11_BLEND_OP_MAX,
|
|
|
|
};
|
|
|
|
|
|
|
|
static const D3D11_CULL_MODE cullingMode[] = {
|
|
|
|
D3D11_CULL_BACK,
|
|
|
|
D3D11_CULL_FRONT,
|
|
|
|
};
|
|
|
|
|
|
|
|
static const D3D11_COMPARISON_FUNC compareOps[] = {
|
|
|
|
D3D11_COMPARISON_NEVER,
|
|
|
|
D3D11_COMPARISON_ALWAYS,
|
|
|
|
D3D11_COMPARISON_EQUAL,
|
|
|
|
D3D11_COMPARISON_NOT_EQUAL,
|
|
|
|
D3D11_COMPARISON_LESS,
|
|
|
|
D3D11_COMPARISON_LESS_EQUAL,
|
|
|
|
D3D11_COMPARISON_GREATER,
|
|
|
|
D3D11_COMPARISON_GREATER_EQUAL,
|
|
|
|
};
|
|
|
|
|
|
|
|
static const D3D11_STENCIL_OP stencilOps[] = {
|
|
|
|
D3D11_STENCIL_OP_KEEP,
|
|
|
|
D3D11_STENCIL_OP_ZERO,
|
|
|
|
D3D11_STENCIL_OP_REPLACE,
|
|
|
|
D3D11_STENCIL_OP_INVERT,
|
|
|
|
D3D11_STENCIL_OP_INCR_SAT,
|
|
|
|
D3D11_STENCIL_OP_DECR_SAT,
|
|
|
|
D3D11_STENCIL_OP_KEEP, // reserved
|
|
|
|
D3D11_STENCIL_OP_KEEP, // reserved
|
|
|
|
};
|
|
|
|
|
|
|
|
static const D3D11_PRIMITIVE_TOPOLOGY primToD3D11[8] = {
|
2021-10-31 20:06:06 +00:00
|
|
|
D3D11_PRIMITIVE_TOPOLOGY_TRIANGLELIST, // Points are expanded to triangles.
|
2021-10-31 21:46:46 +00:00
|
|
|
D3D11_PRIMITIVE_TOPOLOGY_TRIANGLELIST, // Lines are expanded to triangles too.
|
|
|
|
D3D11_PRIMITIVE_TOPOLOGY_TRIANGLELIST, // Lines are expanded to triangles too.
|
2017-02-08 16:35:41 +00:00
|
|
|
D3D11_PRIMITIVE_TOPOLOGY_TRIANGLELIST,
|
|
|
|
D3D11_PRIMITIVE_TOPOLOGY_TRIANGLESTRIP,
|
|
|
|
D3D11_PRIMITIVE_TOPOLOGY_UNDEFINED, // D3D11 doesn't do triangle fans.
|
2021-10-31 20:06:06 +00:00
|
|
|
D3D11_PRIMITIVE_TOPOLOGY_TRIANGLELIST, // Rectangles are expanded to triangles.
|
2017-02-08 16:35:41 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
static const D3D11_LOGIC_OP logicOps[] = {
|
|
|
|
D3D11_LOGIC_OP_CLEAR,
|
|
|
|
D3D11_LOGIC_OP_AND,
|
|
|
|
D3D11_LOGIC_OP_AND_REVERSE,
|
|
|
|
D3D11_LOGIC_OP_COPY,
|
|
|
|
D3D11_LOGIC_OP_AND_INVERTED,
|
2017-02-18 00:10:46 +00:00
|
|
|
D3D11_LOGIC_OP_NOOP,
|
2017-02-08 16:35:41 +00:00
|
|
|
D3D11_LOGIC_OP_XOR,
|
|
|
|
D3D11_LOGIC_OP_OR,
|
|
|
|
D3D11_LOGIC_OP_NOR,
|
2017-02-18 00:10:46 +00:00
|
|
|
D3D11_LOGIC_OP_EQUIV,
|
2017-02-08 16:35:41 +00:00
|
|
|
D3D11_LOGIC_OP_INVERT,
|
|
|
|
D3D11_LOGIC_OP_OR_REVERSE,
|
|
|
|
D3D11_LOGIC_OP_COPY_INVERTED,
|
|
|
|
D3D11_LOGIC_OP_OR_INVERTED,
|
|
|
|
D3D11_LOGIC_OP_NAND,
|
|
|
|
D3D11_LOGIC_OP_SET,
|
|
|
|
};
|
|
|
|
|
|
|
|
class FramebufferManagerD3D11;
|
|
|
|
class ShaderManagerD3D11;
|
|
|
|
|
2017-02-09 23:37:56 +00:00
|
|
|
void DrawEngineD3D11::ApplyDrawState(int prim) {
|
2017-04-03 19:40:30 +00:00
|
|
|
dynState_.topology = primToD3D11[prim];
|
|
|
|
|
2017-08-15 14:01:50 +00:00
|
|
|
if (!gstate_c.IsDirty(DIRTY_BLEND_STATE | DIRTY_TEXTURE_IMAGE | DIRTY_TEXTURE_PARAMS | DIRTY_VIEWPORTSCISSOR_STATE | DIRTY_RASTER_STATE | DIRTY_DEPTHSTENCIL_STATE)) {
|
2017-04-03 19:40:30 +00:00
|
|
|
// nothing to do
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2020-04-04 17:51:47 +00:00
|
|
|
bool useBufferedRendering = framebufferManager_->UseBufferedRendering();
|
2017-01-30 10:56:14 +00:00
|
|
|
// Blend
|
2017-01-30 10:56:14 +00:00
|
|
|
if (gstate_c.IsDirty(DIRTY_BLEND_STATE)) {
|
2017-01-30 10:56:14 +00:00
|
|
|
if (gstate.isModeClear()) {
|
2017-04-03 14:30:00 +00:00
|
|
|
keys_.blend.value = 0; // full wipe
|
2017-01-30 10:56:14 +00:00
|
|
|
keys_.blend.blendEnable = false;
|
2017-04-03 15:20:31 +00:00
|
|
|
dynState_.useBlendColor = false;
|
2017-01-30 10:56:14 +00:00
|
|
|
// Color Test
|
|
|
|
bool alphaMask = gstate.isClearModeAlphaMask();
|
|
|
|
bool colorMask = gstate.isClearModeColorMask();
|
|
|
|
keys_.blend.colorWriteMask = (colorMask ? (1 | 2 | 4) : 0) | (alphaMask ? 8 : 0);
|
2017-01-30 11:02:14 +00:00
|
|
|
} else {
|
2017-04-03 14:30:00 +00:00
|
|
|
keys_.blend.value = 0;
|
2020-11-08 22:17:06 +00:00
|
|
|
|
2022-09-03 21:28:03 +00:00
|
|
|
pipelineState_.Convert(draw_->GetDeviceCaps().fragmentShaderInt32Supported);
|
2022-09-02 19:07:29 +00:00
|
|
|
GenericMaskState &maskState = pipelineState_.maskState;
|
|
|
|
GenericBlendState &blendState = pipelineState_.blendState;
|
2022-04-24 22:06:21 +00:00
|
|
|
|
2020-11-10 21:07:20 +00:00
|
|
|
if (blendState.applyFramebufferRead || maskState.applyFramebufferRead) {
|
2022-09-03 12:47:47 +00:00
|
|
|
bool fboTexNeedsBind = false;
|
|
|
|
ApplyFramebufferRead(&fboTexNeedsBind);
|
2022-05-01 10:49:19 +00:00
|
|
|
// The shader takes over the responsibility for blending, so recompute.
|
|
|
|
ApplyStencilReplaceAndLogicOpIgnoreBlend(blendState.replaceAlphaWithStencil, blendState);
|
2022-08-05 19:11:33 +00:00
|
|
|
|
2022-09-03 12:47:47 +00:00
|
|
|
if (fboTexNeedsBind) {
|
2022-08-05 19:11:33 +00:00
|
|
|
framebufferManager_->BindFramebufferAsColorTexture(1, framebufferManager_->GetCurrentRenderVFB(), BINDFBCOLOR_MAY_COPY);
|
|
|
|
// No sampler required, we do a plain Load in the pixel shader.
|
|
|
|
fboTexBound_ = true;
|
|
|
|
|
|
|
|
framebufferManager_->RebindFramebuffer("RebindFramebuffer - ApplyDrawState");
|
|
|
|
// Must dirty blend state here so we re-copy next time. Example: Lunar's spell effects.
|
2022-08-28 15:34:48 +00:00
|
|
|
dirtyRequiresRecheck_ |= DIRTY_BLEND_STATE;
|
2022-08-05 19:11:33 +00:00
|
|
|
gstate_c.Dirty(DIRTY_BLEND_STATE);
|
|
|
|
}
|
|
|
|
|
2022-08-28 15:34:48 +00:00
|
|
|
dirtyRequiresRecheck_ |= DIRTY_FRAGMENTSHADER_STATE;
|
2020-11-08 13:34:04 +00:00
|
|
|
gstate_c.Dirty(DIRTY_FRAGMENTSHADER_STATE);
|
2022-09-03 12:47:47 +00:00
|
|
|
} else {
|
|
|
|
if (fboTexBound_) {
|
|
|
|
fboTexBound_ = false;
|
|
|
|
dirtyRequiresRecheck_ |= DIRTY_FRAGMENTSHADER_STATE;
|
|
|
|
gstate_c.Dirty(DIRTY_FRAGMENTSHADER_STATE);
|
|
|
|
}
|
2017-01-30 10:56:14 +00:00
|
|
|
}
|
2017-02-08 16:35:41 +00:00
|
|
|
|
2022-09-02 19:07:29 +00:00
|
|
|
if (blendState.blendEnabled) {
|
2017-01-30 10:56:14 +00:00
|
|
|
keys_.blend.blendEnable = true;
|
|
|
|
keys_.blend.logicOpEnable = false;
|
|
|
|
keys_.blend.blendOpColor = d3d11BlendEqLookup[(size_t)blendState.eqColor];
|
|
|
|
keys_.blend.blendOpAlpha = d3d11BlendEqLookup[(size_t)blendState.eqAlpha];
|
|
|
|
keys_.blend.srcColor = d3d11BlendFactorLookup[(size_t)blendState.srcColor];
|
|
|
|
keys_.blend.srcAlpha = d3d11BlendFactorLookup[(size_t)blendState.srcAlpha];
|
|
|
|
keys_.blend.destColor = d3d11BlendFactorLookup[(size_t)blendState.dstColor];
|
|
|
|
keys_.blend.destAlpha = d3d11BlendFactorLookup[(size_t)blendState.dstAlpha];
|
2020-11-08 11:49:06 +00:00
|
|
|
if (blendState.dirtyShaderBlendFixValues) {
|
2022-08-28 15:34:48 +00:00
|
|
|
dirtyRequiresRecheck_ |= DIRTY_SHADERBLEND;
|
2017-01-30 10:56:14 +00:00
|
|
|
gstate_c.Dirty(DIRTY_SHADERBLEND);
|
|
|
|
}
|
|
|
|
dynState_.useBlendColor = blendState.useBlendColor;
|
|
|
|
if (blendState.useBlendColor) {
|
|
|
|
dynState_.blendColor = blendState.blendColor;
|
|
|
|
}
|
2017-01-30 11:02:14 +00:00
|
|
|
} else {
|
2017-01-30 10:56:14 +00:00
|
|
|
keys_.blend.blendEnable = false;
|
|
|
|
dynState_.useBlendColor = false;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (gstate_c.Supports(GPU_SUPPORTS_LOGIC_OP)) {
|
|
|
|
// Logic Ops
|
|
|
|
if (gstate.isLogicOpEnabled() && gstate.getLogicOp() != GE_LOGIC_COPY) {
|
|
|
|
keys_.blend.blendEnable = false; // Can't have both blend & logic op - although I think the PSP can!
|
|
|
|
keys_.blend.logicOpEnable = true;
|
|
|
|
keys_.blend.logicOp = logicOps[gstate.getLogicOp()];
|
2017-01-30 11:02:14 +00:00
|
|
|
} else {
|
2017-01-30 10:56:14 +00:00
|
|
|
keys_.blend.logicOpEnable = false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-09-03 20:51:50 +00:00
|
|
|
keys_.blend.colorWriteMask = maskState.channelMask;
|
2017-02-08 16:35:41 +00:00
|
|
|
}
|
2017-02-12 15:50:45 +00:00
|
|
|
}
|
2017-02-08 16:35:41 +00:00
|
|
|
|
2017-01-30 11:07:26 +00:00
|
|
|
if (gstate_c.IsDirty(DIRTY_RASTER_STATE)) {
|
2017-04-03 14:45:58 +00:00
|
|
|
keys_.raster.value = 0;
|
2015-12-14 01:17:21 +00:00
|
|
|
bool wantCull = !gstate.isModeClear() && prim != GE_PRIM_RECTANGLES && prim > GE_PRIM_LINE_STRIP && gstate.isCullEnabled();
|
2020-09-08 22:52:01 +00:00
|
|
|
keys_.raster.cullMode = wantCull ? (gstate.getCullMode() ? D3D11_CULL_FRONT : D3D11_CULL_BACK) : D3D11_CULL_NONE;
|
|
|
|
|
2018-09-19 04:49:17 +00:00
|
|
|
if (gstate.isModeClear() || gstate.isModeThrough()) {
|
|
|
|
// TODO: Might happen in clear mode if not through...
|
|
|
|
keys_.raster.depthClipEnable = 1;
|
2017-04-03 14:19:10 +00:00
|
|
|
} else {
|
2018-09-19 04:49:17 +00:00
|
|
|
if (gstate.getDepthRangeMin() == 0 || gstate.getDepthRangeMax() == 65535) {
|
|
|
|
// TODO: Still has a bug where we clamp to depth range if one is not the full range.
|
|
|
|
// But the alternate is not clamping in either direction...
|
|
|
|
keys_.raster.depthClipEnable = !gstate.isDepthClampEnabled() || !gstate_c.Supports(GPU_SUPPORTS_DEPTH_CLAMP);
|
|
|
|
} else {
|
|
|
|
// We just want to clip in this case, the clamp would be clipped anyway.
|
|
|
|
keys_.raster.depthClipEnable = 1;
|
|
|
|
}
|
2017-04-03 14:19:10 +00:00
|
|
|
}
|
|
|
|
}
|
2017-02-08 16:35:41 +00:00
|
|
|
|
2017-01-30 11:02:14 +00:00
|
|
|
if (gstate_c.IsDirty(DIRTY_DEPTHSTENCIL_STATE)) {
|
2021-01-24 23:54:13 +00:00
|
|
|
GenericStencilFuncState stencilState;
|
|
|
|
ConvertStencilFuncState(stencilState);
|
|
|
|
|
2022-08-17 09:31:02 +00:00
|
|
|
if (gstate.isModeClear()) {
|
2017-04-03 14:45:58 +00:00
|
|
|
keys_.depthStencil.value = 0;
|
2017-02-12 17:29:58 +00:00
|
|
|
keys_.depthStencil.depthTestEnable = true;
|
2017-01-30 11:02:14 +00:00
|
|
|
keys_.depthStencil.depthCompareOp = D3D11_COMPARISON_ALWAYS;
|
|
|
|
keys_.depthStencil.depthWriteEnable = gstate.isClearModeDepthMask();
|
2017-02-08 16:35:41 +00:00
|
|
|
|
2017-01-30 11:02:14 +00:00
|
|
|
// Stencil Test
|
|
|
|
bool alphaMask = gstate.isClearModeAlphaMask();
|
|
|
|
if (alphaMask) {
|
|
|
|
keys_.depthStencil.stencilTestEnable = true;
|
|
|
|
keys_.depthStencil.stencilCompareFunc = D3D11_COMPARISON_ALWAYS;
|
|
|
|
keys_.depthStencil.stencilPassOp = D3D11_STENCIL_OP_REPLACE;
|
|
|
|
keys_.depthStencil.stencilFailOp = D3D11_STENCIL_OP_REPLACE;
|
|
|
|
keys_.depthStencil.stencilDepthFailOp = D3D11_STENCIL_OP_REPLACE;
|
|
|
|
dynState_.useStencil = true;
|
|
|
|
// In clear mode, the stencil value is set to the alpha value of the vertex.
|
|
|
|
// A normal clear will be 2 points, the second point has the color.
|
|
|
|
// We override this value in the pipeline from software transform for clear rectangles.
|
|
|
|
dynState_.stencilRef = 0xFF;
|
2019-03-17 02:42:18 +00:00
|
|
|
// But we still apply the stencil write mask.
|
2021-01-24 23:54:13 +00:00
|
|
|
keys_.depthStencil.stencilWriteMask = stencilState.writeMask;
|
2017-01-30 11:02:14 +00:00
|
|
|
} else {
|
|
|
|
keys_.depthStencil.stencilTestEnable = false;
|
|
|
|
dynState_.useStencil = false;
|
|
|
|
}
|
|
|
|
|
2017-02-13 00:22:55 +00:00
|
|
|
} else {
|
2017-04-03 14:45:58 +00:00
|
|
|
keys_.depthStencil.value = 0;
|
2017-01-30 11:02:14 +00:00
|
|
|
// Depth Test
|
|
|
|
if (gstate.isDepthTestEnabled()) {
|
|
|
|
keys_.depthStencil.depthTestEnable = true;
|
|
|
|
keys_.depthStencil.depthCompareOp = compareOps[gstate.getDepthTestFunction()];
|
|
|
|
keys_.depthStencil.depthWriteEnable = gstate.isDepthWriteEnabled();
|
|
|
|
} else {
|
|
|
|
keys_.depthStencil.depthTestEnable = false;
|
|
|
|
keys_.depthStencil.depthWriteEnable = false;
|
|
|
|
keys_.depthStencil.depthCompareOp = D3D11_COMPARISON_ALWAYS;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Stencil Test
|
|
|
|
if (stencilState.enabled) {
|
|
|
|
keys_.depthStencil.stencilTestEnable = true;
|
|
|
|
keys_.depthStencil.stencilCompareFunc = compareOps[stencilState.testFunc];
|
|
|
|
keys_.depthStencil.stencilPassOp = stencilOps[stencilState.zPass];
|
|
|
|
keys_.depthStencil.stencilFailOp = stencilOps[stencilState.sFail];
|
|
|
|
keys_.depthStencil.stencilDepthFailOp = stencilOps[stencilState.zFail];
|
|
|
|
keys_.depthStencil.stencilCompareMask = stencilState.testMask;
|
|
|
|
keys_.depthStencil.stencilWriteMask = stencilState.writeMask;
|
|
|
|
dynState_.useStencil = true;
|
|
|
|
dynState_.stencilRef = stencilState.testRef;
|
2022-08-27 08:05:57 +00:00
|
|
|
|
|
|
|
// Nasty special case for Spongebob and similar where it tries to write zeros to alpha/stencil during
|
|
|
|
// depth-fail. We can't write to alpha then because the pixel is killed. However, we can invert the depth
|
|
|
|
// test and modify the alpha function...
|
2022-08-28 05:11:00 +00:00
|
|
|
if (SpongebobDepthInverseConditions(stencilState)) {
|
2022-08-27 08:05:57 +00:00
|
|
|
keys_.blend.blendEnable = true;
|
|
|
|
keys_.blend.blendOpAlpha = D3D11_BLEND_OP_ADD;
|
|
|
|
keys_.blend.blendOpColor = D3D11_BLEND_OP_ADD;
|
|
|
|
keys_.blend.srcColor = D3D11_BLEND_ZERO;
|
|
|
|
keys_.blend.destColor = D3D11_BLEND_ZERO;
|
|
|
|
keys_.blend.logicOpEnable = false;
|
|
|
|
keys_.blend.srcAlpha = D3D11_BLEND_ZERO;
|
|
|
|
keys_.blend.destAlpha = D3D11_BLEND_ZERO;
|
|
|
|
keys_.blend.colorWriteMask = D3D11_COLOR_WRITE_ENABLE_ALPHA;
|
|
|
|
|
|
|
|
keys_.depthStencil.depthCompareOp = D3D11_COMPARISON_LESS; // Inverse of GREATER_EQUAL
|
|
|
|
keys_.depthStencil.stencilCompareFunc = D3D11_COMPARISON_ALWAYS;
|
|
|
|
// Invert
|
|
|
|
keys_.depthStencil.stencilPassOp = D3D11_STENCIL_OP_ZERO;
|
|
|
|
keys_.depthStencil.stencilFailOp = D3D11_STENCIL_OP_ZERO;
|
|
|
|
keys_.depthStencil.stencilDepthFailOp = D3D11_STENCIL_OP_KEEP;
|
|
|
|
|
2022-08-28 15:39:05 +00:00
|
|
|
dirtyRequiresRecheck_ |= DIRTY_BLEND_STATE | DIRTY_DEPTHSTENCIL_STATE;
|
|
|
|
gstate_c.Dirty(DIRTY_BLEND_STATE | DIRTY_DEPTHSTENCIL_STATE);
|
2022-08-27 08:05:57 +00:00
|
|
|
}
|
2017-01-30 11:02:14 +00:00
|
|
|
} else {
|
|
|
|
keys_.depthStencil.stencilTestEnable = false;
|
|
|
|
dynState_.useStencil = false;
|
|
|
|
}
|
2017-02-08 16:35:41 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-01-30 11:22:35 +00:00
|
|
|
if (gstate_c.IsDirty(DIRTY_VIEWPORTSCISSOR_STATE)) {
|
2017-01-30 11:02:14 +00:00
|
|
|
ViewportAndScissor vpAndScissor;
|
|
|
|
ConvertViewportAndScissor(useBufferedRendering,
|
|
|
|
framebufferManager_->GetRenderWidth(), framebufferManager_->GetRenderHeight(),
|
|
|
|
framebufferManager_->GetTargetBufferWidth(), framebufferManager_->GetTargetBufferHeight(),
|
|
|
|
vpAndScissor);
|
2022-08-20 21:16:55 +00:00
|
|
|
UpdateCachedViewportState(vpAndScissor);
|
2017-01-30 11:02:14 +00:00
|
|
|
|
|
|
|
float depthMin = vpAndScissor.depthRangeMin;
|
|
|
|
float depthMax = vpAndScissor.depthRangeMax;
|
|
|
|
|
|
|
|
if (depthMin < 0.0f) depthMin = 0.0f;
|
|
|
|
if (depthMax > 1.0f) depthMax = 1.0f;
|
2017-02-12 10:20:55 +00:00
|
|
|
|
2017-01-30 11:02:14 +00:00
|
|
|
Draw::Viewport &vp = dynState_.viewport;
|
|
|
|
vp.TopLeftX = vpAndScissor.viewportX;
|
|
|
|
vp.TopLeftY = vpAndScissor.viewportY;
|
|
|
|
vp.Width = vpAndScissor.viewportW;
|
|
|
|
vp.Height = vpAndScissor.viewportH;
|
|
|
|
vp.MinDepth = depthMin;
|
|
|
|
vp.MaxDepth = depthMax;
|
2017-08-15 08:38:20 +00:00
|
|
|
|
2017-01-30 11:02:14 +00:00
|
|
|
D3D11_RECT &scissor = dynState_.scissor;
|
2022-07-24 19:04:54 +00:00
|
|
|
scissor.left = vpAndScissor.scissorX;
|
|
|
|
scissor.top = vpAndScissor.scissorY;
|
|
|
|
scissor.right = vpAndScissor.scissorX + std::max(0, vpAndScissor.scissorW);
|
|
|
|
scissor.bottom = vpAndScissor.scissorY + std::max(0, vpAndScissor.scissorH);
|
2017-02-08 16:35:41 +00:00
|
|
|
}
|
|
|
|
|
2022-08-27 08:05:18 +00:00
|
|
|
// Actually create/set the state objects only after we're done mapping all the state.
|
|
|
|
// There might have been interactions between depth and blend above.
|
|
|
|
if (gstate_c.IsDirty(DIRTY_BLEND_STATE)) {
|
|
|
|
if (!device1_) {
|
|
|
|
ID3D11BlendState *bs = blendCache_.Get(keys_.blend.value);
|
|
|
|
if (bs == nullptr) {
|
|
|
|
D3D11_BLEND_DESC desc{};
|
|
|
|
D3D11_RENDER_TARGET_BLEND_DESC &rt = desc.RenderTarget[0];
|
|
|
|
rt.BlendEnable = keys_.blend.blendEnable;
|
|
|
|
rt.BlendOp = (D3D11_BLEND_OP)keys_.blend.blendOpColor;
|
|
|
|
rt.BlendOpAlpha = (D3D11_BLEND_OP)keys_.blend.blendOpAlpha;
|
|
|
|
rt.SrcBlend = (D3D11_BLEND)keys_.blend.srcColor;
|
|
|
|
rt.DestBlend = (D3D11_BLEND)keys_.blend.destColor;
|
|
|
|
rt.SrcBlendAlpha = (D3D11_BLEND)keys_.blend.srcAlpha;
|
|
|
|
rt.DestBlendAlpha = (D3D11_BLEND)keys_.blend.destAlpha;
|
|
|
|
rt.RenderTargetWriteMask = keys_.blend.colorWriteMask;
|
|
|
|
ASSERT_SUCCESS(device_->CreateBlendState(&desc, &bs));
|
|
|
|
blendCache_.Insert(keys_.blend.value, bs);
|
|
|
|
}
|
|
|
|
blendState_ = bs;
|
|
|
|
} else {
|
|
|
|
ID3D11BlendState1 *bs1 = blendCache1_.Get(keys_.blend.value);
|
|
|
|
if (bs1 == nullptr) {
|
|
|
|
D3D11_BLEND_DESC1 desc1{};
|
|
|
|
D3D11_RENDER_TARGET_BLEND_DESC1 &rt = desc1.RenderTarget[0];
|
|
|
|
rt.BlendEnable = keys_.blend.blendEnable;
|
|
|
|
rt.BlendOp = (D3D11_BLEND_OP)keys_.blend.blendOpColor;
|
|
|
|
rt.BlendOpAlpha = (D3D11_BLEND_OP)keys_.blend.blendOpAlpha;
|
|
|
|
rt.SrcBlend = (D3D11_BLEND)keys_.blend.srcColor;
|
|
|
|
rt.DestBlend = (D3D11_BLEND)keys_.blend.destColor;
|
|
|
|
rt.SrcBlendAlpha = (D3D11_BLEND)keys_.blend.srcAlpha;
|
|
|
|
rt.DestBlendAlpha = (D3D11_BLEND)keys_.blend.destAlpha;
|
|
|
|
rt.RenderTargetWriteMask = keys_.blend.colorWriteMask;
|
|
|
|
rt.LogicOpEnable = keys_.blend.logicOpEnable;
|
|
|
|
rt.LogicOp = (D3D11_LOGIC_OP)keys_.blend.logicOp;
|
|
|
|
ASSERT_SUCCESS(device1_->CreateBlendState1(&desc1, &bs1));
|
|
|
|
blendCache1_.Insert(keys_.blend.value, bs1);
|
|
|
|
}
|
|
|
|
blendState1_ = bs1;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (gstate_c.IsDirty(DIRTY_RASTER_STATE)) {
|
|
|
|
ID3D11RasterizerState *rs = rasterCache_.Get(keys_.raster.value);
|
|
|
|
if (rs == nullptr) {
|
|
|
|
D3D11_RASTERIZER_DESC desc{};
|
|
|
|
desc.CullMode = (D3D11_CULL_MODE)(keys_.raster.cullMode);
|
|
|
|
desc.FillMode = D3D11_FILL_SOLID;
|
|
|
|
desc.ScissorEnable = TRUE;
|
|
|
|
desc.FrontCounterClockwise = TRUE;
|
|
|
|
desc.DepthClipEnable = keys_.raster.depthClipEnable;
|
|
|
|
ASSERT_SUCCESS(device_->CreateRasterizerState(&desc, &rs));
|
|
|
|
rasterCache_.Insert(keys_.raster.value, rs);
|
|
|
|
}
|
|
|
|
rasterState_ = rs;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (gstate_c.IsDirty(DIRTY_DEPTHSTENCIL_STATE)) {
|
|
|
|
ID3D11DepthStencilState *ds = depthStencilCache_.Get(keys_.depthStencil.value);
|
|
|
|
if (ds == nullptr) {
|
|
|
|
D3D11_DEPTH_STENCIL_DESC desc{};
|
|
|
|
desc.DepthEnable = keys_.depthStencil.depthTestEnable;
|
|
|
|
desc.DepthWriteMask = keys_.depthStencil.depthWriteEnable ? D3D11_DEPTH_WRITE_MASK_ALL : D3D11_DEPTH_WRITE_MASK_ZERO;
|
|
|
|
desc.DepthFunc = (D3D11_COMPARISON_FUNC)keys_.depthStencil.depthCompareOp;
|
|
|
|
desc.StencilEnable = keys_.depthStencil.stencilTestEnable;
|
|
|
|
desc.StencilReadMask = keys_.depthStencil.stencilCompareMask;
|
|
|
|
desc.StencilWriteMask = keys_.depthStencil.stencilWriteMask;
|
|
|
|
desc.FrontFace.StencilFailOp = (D3D11_STENCIL_OP)keys_.depthStencil.stencilFailOp;
|
|
|
|
desc.FrontFace.StencilPassOp = (D3D11_STENCIL_OP)keys_.depthStencil.stencilPassOp;
|
|
|
|
desc.FrontFace.StencilDepthFailOp = (D3D11_STENCIL_OP)keys_.depthStencil.stencilDepthFailOp;
|
|
|
|
desc.FrontFace.StencilFunc = (D3D11_COMPARISON_FUNC)keys_.depthStencil.stencilCompareFunc;
|
|
|
|
desc.BackFace = desc.FrontFace;
|
|
|
|
ASSERT_SUCCESS(device_->CreateDepthStencilState(&desc, &ds));
|
|
|
|
depthStencilCache_.Insert(keys_.depthStencil.value, ds);
|
|
|
|
}
|
|
|
|
depthStencilState_ = ds;
|
|
|
|
}
|
|
|
|
|
2017-02-09 23:30:42 +00:00
|
|
|
if (gstate_c.IsDirty(DIRTY_TEXTURE_IMAGE | DIRTY_TEXTURE_PARAMS) && !gstate.isModeClear() && gstate.isTextureMapEnabled()) {
|
|
|
|
textureCache_->SetTexture();
|
|
|
|
gstate_c.Clean(DIRTY_TEXTURE_IMAGE | DIRTY_TEXTURE_PARAMS);
|
2018-04-22 18:16:33 +00:00
|
|
|
} else if (gstate.getTextureAddress(0) == ((gstate.getFrameBufRawAddress() | 0x04000000) & 0x3FFFFFFF)) {
|
|
|
|
// This catches the case of clearing a texture.
|
|
|
|
gstate_c.Dirty(DIRTY_TEXTURE_IMAGE);
|
2017-02-09 23:30:42 +00:00
|
|
|
}
|
2017-02-08 17:07:34 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void DrawEngineD3D11::ApplyDrawStateLate(bool applyStencilRef, uint8_t stencilRef) {
|
2017-03-07 12:23:35 +00:00
|
|
|
// we go through Draw here because it automatically handles screen rotation, as needed in UWP on mobiles.
|
2017-04-03 19:40:30 +00:00
|
|
|
if (gstate_c.IsDirty(DIRTY_VIEWPORTSCISSOR_STATE)) {
|
|
|
|
draw_->SetViewports(1, &dynState_.viewport);
|
|
|
|
draw_->SetScissorRect(dynState_.scissor.left, dynState_.scissor.top, dynState_.scissor.right - dynState_.scissor.left, dynState_.scissor.bottom - dynState_.scissor.top);
|
|
|
|
}
|
|
|
|
if (gstate_c.IsDirty(DIRTY_RASTER_STATE)) {
|
|
|
|
context_->RSSetState(rasterState_);
|
|
|
|
}
|
|
|
|
if (gstate_c.IsDirty(DIRTY_BLEND_STATE)) {
|
2022-08-05 19:11:33 +00:00
|
|
|
// Need to do this AFTER ApplyTexture because the process of depalettization can ruin the blend state.
|
2017-08-14 12:35:26 +00:00
|
|
|
float blendColor[4];
|
|
|
|
Uint8x4ToFloat4(blendColor, dynState_.blendColor);
|
2017-04-03 19:40:30 +00:00
|
|
|
if (device1_) {
|
|
|
|
context1_->OMSetBlendState(blendState1_, blendColor, 0xFFFFFFFF);
|
|
|
|
} else {
|
|
|
|
context_->OMSetBlendState(blendState_, blendColor, 0xFFFFFFFF);
|
|
|
|
}
|
|
|
|
}
|
2017-08-14 12:35:26 +00:00
|
|
|
if (gstate_c.IsDirty(DIRTY_DEPTHSTENCIL_STATE) || applyStencilRef) {
|
2017-04-03 19:40:30 +00:00
|
|
|
context_->OMSetDepthStencilState(depthStencilState_, applyStencilRef ? stencilRef : dynState_.stencilRef);
|
2017-02-18 01:14:10 +00:00
|
|
|
}
|
2017-04-03 21:51:54 +00:00
|
|
|
gstate_c.Clean(DIRTY_VIEWPORTSCISSOR_STATE | DIRTY_DEPTHSTENCIL_STATE | DIRTY_RASTER_STATE | DIRTY_BLEND_STATE);
|
2022-08-28 15:34:48 +00:00
|
|
|
gstate_c.Dirty(dirtyRequiresRecheck_);
|
|
|
|
dirtyRequiresRecheck_ = 0;
|
2017-04-03 21:51:54 +00:00
|
|
|
}
|