mirror of
https://github.com/dolphin-emu/dolphin.git
synced 2025-04-04 19:20:20 +00:00
VideoBackends: Use vertex shader depth range if ztexture is used.
This commit is contained in:
parent
bde8126913
commit
a15555fe03
Source/Core
VideoBackends
VideoCommon
@ -576,8 +576,7 @@ void Renderer::SetViewport()
|
|||||||
|
|
||||||
// If an inverted or oversized depth range is used, we need to calculate the depth range in the
|
// If an inverted or oversized depth range is used, we need to calculate the depth range in the
|
||||||
// vertex shader.
|
// vertex shader.
|
||||||
if (xfmem.viewport.zRange < 0.0f || fabs(xfmem.viewport.zRange) > 16777215.0f ||
|
if (UseVertexDepthRange())
|
||||||
fabs(xfmem.viewport.farZ) > 16777215.0f)
|
|
||||||
{
|
{
|
||||||
// We need to ensure depth values are clamped the maximum value supported by the console GPU.
|
// We need to ensure depth values are clamped the maximum value supported by the console GPU.
|
||||||
min_depth = 0.0f;
|
min_depth = 0.0f;
|
||||||
|
@ -481,8 +481,7 @@ void Renderer::SetViewport()
|
|||||||
|
|
||||||
// If an inverted or oversized depth range is used, we need to calculate the depth range in the
|
// If an inverted or oversized depth range is used, we need to calculate the depth range in the
|
||||||
// vertex shader.
|
// vertex shader.
|
||||||
if (xfmem.viewport.zRange < 0.0f || fabs(xfmem.viewport.zRange) > 16777215.0f ||
|
if (UseVertexDepthRange())
|
||||||
fabs(xfmem.viewport.farZ) > 16777215.0f)
|
|
||||||
{
|
{
|
||||||
// We need to ensure depth values are clamped the maximum value supported by the console GPU.
|
// We need to ensure depth values are clamped the maximum value supported by the console GPU.
|
||||||
min_depth = 0.0f;
|
min_depth = 0.0f;
|
||||||
|
@ -1135,41 +1135,36 @@ void Renderer::SetViewport()
|
|||||||
glViewport(iceilf(X), iceilf(Y), iceilf(Width), iceilf(Height));
|
glViewport(iceilf(X), iceilf(Y), iceilf(Width), iceilf(Height));
|
||||||
}
|
}
|
||||||
|
|
||||||
// Set the reversed depth range.
|
if (!g_ActiveConfig.backend_info.bSupportsOversizedDepthRanges &&
|
||||||
if (g_ActiveConfig.backend_info.bSupportsOversizedDepthRanges)
|
!g_ActiveConfig.backend_info.bSupportsDepthClamp)
|
||||||
{
|
{
|
||||||
glDepthRangedNV(max_depth, min_depth);
|
// There's no way to support oversized depth ranges in this situation. Let's just clamp the
|
||||||
|
// range to the maximum value supported by the console GPU and hope for the best.
|
||||||
|
min_depth = MathUtil::Clamp(min_depth, 0.0f, GX_MAX_DEPTH);
|
||||||
|
max_depth = MathUtil::Clamp(max_depth, 0.0f, GX_MAX_DEPTH);
|
||||||
}
|
}
|
||||||
else
|
|
||||||
|
if (UseVertexDepthRange())
|
||||||
{
|
{
|
||||||
// If an oversized depth range is used, we need to calculate the depth range in the
|
// We need to ensure depth values are clamped the maximum value supported by the console GPU.
|
||||||
// vertex shader.
|
// Taking into account whether the depth range is inverted or not.
|
||||||
if (g_ActiveConfig.backend_info.bSupportsDepthClamp &&
|
if (xfmem.viewport.zRange < 0.0f)
|
||||||
(fabs(xfmem.viewport.zRange) > 16777215.0f || fabs(xfmem.viewport.farZ) > 16777215.0f))
|
|
||||||
{
|
{
|
||||||
// We need to ensure depth values are clamped the maximum value supported by the console GPU.
|
min_depth = GX_MAX_DEPTH;
|
||||||
// Taking into account whether the depth range is inverted or not.
|
max_depth = 0.0f;
|
||||||
if (xfmem.viewport.zRange < 0.0f)
|
|
||||||
{
|
|
||||||
min_depth = GX_MAX_DEPTH;
|
|
||||||
max_depth = 0.0f;
|
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
|
||||||
min_depth = 0.0f;
|
|
||||||
max_depth = GX_MAX_DEPTH;
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
// There's no way to support oversized depth ranges in this situation. Let's just clamp the
|
min_depth = 0.0f;
|
||||||
// range to the maximum value supported by the console GPU and hope for the best.
|
max_depth = GX_MAX_DEPTH;
|
||||||
min_depth = MathUtil::Clamp(min_depth, 0.0f, GX_MAX_DEPTH);
|
|
||||||
max_depth = MathUtil::Clamp(max_depth, 0.0f, GX_MAX_DEPTH);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
glDepthRangef(max_depth, min_depth);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// Set the reversed depth range.
|
||||||
|
if (g_ActiveConfig.backend_info.bSupportsOversizedDepthRanges)
|
||||||
|
glDepthRangedNV(max_depth, min_depth);
|
||||||
|
else
|
||||||
|
glDepthRangef(max_depth, min_depth);
|
||||||
}
|
}
|
||||||
|
|
||||||
void Renderer::ClearScreen(const EFBRectangle& rc, bool colorEnable, bool alphaEnable, bool zEnable,
|
void Renderer::ClearScreen(const EFBRectangle& rc, bool colorEnable, bool alphaEnable, bool zEnable,
|
||||||
|
@ -4,7 +4,6 @@
|
|||||||
|
|
||||||
#include "VideoBackends/Vulkan/Renderer.h"
|
#include "VideoBackends/Vulkan/Renderer.h"
|
||||||
|
|
||||||
#include <cmath>
|
|
||||||
#include <cstddef>
|
#include <cstddef>
|
||||||
#include <cstdio>
|
#include <cstdio>
|
||||||
#include <limits>
|
#include <limits>
|
||||||
@ -1657,8 +1656,7 @@ void Renderer::SetViewport()
|
|||||||
// If an oversized or inverted depth range is used, we need to calculate the depth range in the
|
// If an oversized or inverted depth range is used, we need to calculate the depth range in the
|
||||||
// vertex shader.
|
// vertex shader.
|
||||||
// TODO: Inverted depth ranges are bugged in all drivers, which should be added to DriverDetails.
|
// TODO: Inverted depth ranges are bugged in all drivers, which should be added to DriverDetails.
|
||||||
if (xfmem.viewport.zRange < 0.0f || fabs(xfmem.viewport.zRange) > 16777215.0f ||
|
if (UseVertexDepthRange())
|
||||||
fabs(xfmem.viewport.farZ) > 16777215.0f)
|
|
||||||
{
|
{
|
||||||
// We need to ensure depth values are clamped the maximum value supported by the console GPU.
|
// We need to ensure depth values are clamped the maximum value supported by the console GPU.
|
||||||
min_depth = 0.0f;
|
min_depth = 0.0f;
|
||||||
|
@ -332,6 +332,8 @@ static void BPWritten(const BPCmd& bp)
|
|||||||
{
|
{
|
||||||
if (bp.changes & 3)
|
if (bp.changes & 3)
|
||||||
PixelShaderManager::SetZTextureTypeChanged();
|
PixelShaderManager::SetZTextureTypeChanged();
|
||||||
|
if (bp.changes & 12)
|
||||||
|
VertexShaderManager::SetViewportChanged();
|
||||||
#if defined(_DEBUG) || defined(DEBUGFAST)
|
#if defined(_DEBUG) || defined(DEBUGFAST)
|
||||||
const char* pzop[] = {"DISABLE", "ADD", "REPLACE", "?"};
|
const char* pzop[] = {"DISABLE", "ADD", "REPLACE", "?"};
|
||||||
const char* pztype[] = {"Z8", "Z16", "Z24", "?"};
|
const char* pztype[] = {"Z8", "Z16", "Z24", "?"};
|
||||||
|
@ -940,3 +940,30 @@ void Renderer::DumpFrameToImage(const FrameDumpConfig& config)
|
|||||||
TextureToPng(config.data, config.stride, filename, config.width, config.height, false);
|
TextureToPng(config.data, config.stride, filename, config.width, config.height, false);
|
||||||
m_frame_dump_image_counter++;
|
m_frame_dump_image_counter++;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
bool Renderer::UseVertexDepthRange() const
|
||||||
|
{
|
||||||
|
// We can't compute the depth range in the vertex shader if we don't support depth clamp.
|
||||||
|
if (!g_ActiveConfig.backend_info.bSupportsDepthClamp)
|
||||||
|
return false;
|
||||||
|
|
||||||
|
if (g_ActiveConfig.backend_info.bSupportsOversizedDepthRanges)
|
||||||
|
{
|
||||||
|
// We support oversized depth ranges, but we need a full depth range if a ztexture is used.
|
||||||
|
return bpmem.ztex2.type != ZTEXTURE_DISABLE;
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
// We need a full depth range if a ztexture is used.
|
||||||
|
if (bpmem.ztex2.type != ZTEXTURE_DISABLE)
|
||||||
|
return true;
|
||||||
|
|
||||||
|
// If an inverted depth range is unsupported, we also need to check if the range is inverted.
|
||||||
|
if (!g_ActiveConfig.backend_info.bSupportsReversedDepthRange && xfmem.viewport.zRange < 0.0f)
|
||||||
|
return true;
|
||||||
|
|
||||||
|
// If an oversized depth range or a ztexture is used, we need to calculate the depth range
|
||||||
|
// in the vertex shader.
|
||||||
|
return fabs(xfmem.viewport.zRange) > 16777215.0f || fabs(xfmem.viewport.farZ) > 16777215.0f;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
@ -145,6 +145,8 @@ public:
|
|||||||
// Final surface changing
|
// Final surface changing
|
||||||
// This is called when the surface is resized (WX) or the window changes (Android).
|
// This is called when the surface is resized (WX) or the window changes (Android).
|
||||||
virtual void ChangeSurface(void* new_surface_handle) {}
|
virtual void ChangeSurface(void* new_surface_handle) {}
|
||||||
|
bool UseVertexDepthRange() const;
|
||||||
|
|
||||||
protected:
|
protected:
|
||||||
static void CalculateTargetScale(int x, int y, int* scaledX, int* scaledY);
|
static void CalculateTargetScale(int x, int y, int* scaledX, int* scaledY);
|
||||||
bool CalculateTargetSize();
|
bool CalculateTargetSize();
|
||||||
|
@ -391,40 +391,27 @@ void VertexShaderManager::SetConstants()
|
|||||||
constants.pixelcentercorrection[2] = 1.0f;
|
constants.pixelcentercorrection[2] = 1.0f;
|
||||||
constants.pixelcentercorrection[3] = 0.0f;
|
constants.pixelcentercorrection[3] = 0.0f;
|
||||||
|
|
||||||
if (g_ActiveConfig.backend_info.bSupportsDepthClamp &&
|
if (g_renderer->UseVertexDepthRange())
|
||||||
!g_ActiveConfig.backend_info.bSupportsOversizedDepthRanges)
|
|
||||||
{
|
{
|
||||||
// Oversized depth ranges are handled in the vertex shader. We need to reverse
|
// Oversized depth ranges are handled in the vertex shader. We need to reverse
|
||||||
// the far value to get a reversed depth range mapping. This is necessary
|
// the far value to use the reversed-Z trick.
|
||||||
// because the standard depth range equation pushes all depth values towards
|
|
||||||
// the back of the depth buffer where conventionally depth buffers have the
|
|
||||||
// least precision.
|
|
||||||
if (g_ActiveConfig.backend_info.bSupportsReversedDepthRange)
|
if (g_ActiveConfig.backend_info.bSupportsReversedDepthRange)
|
||||||
{
|
{
|
||||||
if (fabs(xfmem.viewport.zRange) > 16777215.0f || fabs(xfmem.viewport.farZ) > 16777215.0f)
|
// Sometimes the console also tries to use the reversed-Z trick. We can only do
|
||||||
{
|
// that with the expected accuracy if the backend can reverse the depth range.
|
||||||
// For backends that support reversing the depth range we also support cases
|
constants.pixelcentercorrection[2] = fabs(xfmem.viewport.zRange) / 16777215.0f;
|
||||||
// where the console also uses reversed depth with the same accuracy. We need
|
if (xfmem.viewport.zRange < 0.0f)
|
||||||
// to make sure the depth range is positive here and then reverse the depth in
|
constants.pixelcentercorrection[3] = xfmem.viewport.farZ / 16777215.0f;
|
||||||
// the backend viewport.
|
else
|
||||||
constants.pixelcentercorrection[2] = fabs(xfmem.viewport.zRange) / 16777215.0f;
|
constants.pixelcentercorrection[3] = 1.0f - xfmem.viewport.farZ / 16777215.0f;
|
||||||
if (xfmem.viewport.zRange < 0.0f)
|
|
||||||
constants.pixelcentercorrection[3] = xfmem.viewport.farZ / 16777215.0f;
|
|
||||||
else
|
|
||||||
constants.pixelcentercorrection[3] = 1.0f - xfmem.viewport.farZ / 16777215.0f;
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
if (xfmem.viewport.zRange < 0.0f || xfmem.viewport.zRange > 16777215.0f ||
|
// For backends that don't support reversing the depth range we can still render
|
||||||
fabs(xfmem.viewport.farZ) > 16777215.0f)
|
// cases where the console uses the reversed-Z trick. But we simply can't provide
|
||||||
{
|
// the expected accuracy, which might result in z-fighting.
|
||||||
// For backends that don't support reversing the depth range we can still render
|
constants.pixelcentercorrection[2] = xfmem.viewport.zRange / 16777215.0f;
|
||||||
// cases where the console uses reversed depth correctly. But we simply can't
|
constants.pixelcentercorrection[3] = 1.0f - xfmem.viewport.farZ / 16777215.0f;
|
||||||
// provide the same accuracy as the console.
|
|
||||||
constants.pixelcentercorrection[2] = xfmem.viewport.zRange / 16777215.0f;
|
|
||||||
constants.pixelcentercorrection[3] = 1.0f - xfmem.viewport.farZ / 16777215.0f;
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
Loading…
x
Reference in New Issue
Block a user