mirror of
https://github.com/dolphin-emu/dolphin.git
synced 2025-07-24 14:49:42 -06:00
BPFunctions: Move GX viewport conversion to VideoCommon
This commit is contained in:
@ -12,6 +12,7 @@
|
||||
#include "VideoCommon/VertexManagerBase.h"
|
||||
#include "VideoCommon/VideoCommon.h"
|
||||
#include "VideoCommon/VideoConfig.h"
|
||||
#include "VideoCommon/XFMemory.h"
|
||||
|
||||
namespace BPFunctions
|
||||
{
|
||||
@ -68,6 +69,76 @@ void SetScissor()
|
||||
g_renderer->SetScissorRect(rc);
|
||||
}
|
||||
|
||||
void SetViewport()
|
||||
{
|
||||
int scissor_x_off = bpmem.scissorOffset.x * 2;
|
||||
int scissor_y_off = bpmem.scissorOffset.y * 2;
|
||||
float x = g_renderer->EFBToScaledXf(xfmem.viewport.xOrig - xfmem.viewport.wd - scissor_x_off);
|
||||
float y = g_renderer->EFBToScaledYf(xfmem.viewport.yOrig + xfmem.viewport.ht - scissor_y_off);
|
||||
|
||||
float width = g_renderer->EFBToScaledXf(2.0f * xfmem.viewport.wd);
|
||||
float height = g_renderer->EFBToScaledYf(-2.0f * xfmem.viewport.ht);
|
||||
float min_depth = (xfmem.viewport.farZ - xfmem.viewport.zRange) / 16777216.0f;
|
||||
float max_depth = xfmem.viewport.farZ / 16777216.0f;
|
||||
if (width < 0.f)
|
||||
{
|
||||
x += width;
|
||||
width *= -1;
|
||||
}
|
||||
if (height < 0.f)
|
||||
{
|
||||
y += height;
|
||||
height *= -1;
|
||||
}
|
||||
|
||||
// The maximum depth that is written to the depth buffer should never exceed this value.
|
||||
// This is necessary because we use a 2^24 divisor for all our depth values to prevent
|
||||
// floating-point round-trip errors. However the console GPU doesn't ever write a value
|
||||
// to the depth buffer that exceeds 2^24 - 1.
|
||||
constexpr float GX_MAX_DEPTH = 16777215.0f / 16777216.0f;
|
||||
if (!g_ActiveConfig.backend_info.bSupportsDepthClamp)
|
||||
{
|
||||
// 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);
|
||||
}
|
||||
|
||||
if (g_renderer->UseVertexDepthRange())
|
||||
{
|
||||
// We need to ensure depth values are clamped the maximum value supported by the console GPU.
|
||||
// Taking into account whether the depth range is inverted or not.
|
||||
if (xfmem.viewport.zRange < 0.0f && g_ActiveConfig.backend_info.bSupportsReversedDepthRange)
|
||||
{
|
||||
min_depth = GX_MAX_DEPTH;
|
||||
max_depth = 0.0f;
|
||||
}
|
||||
else
|
||||
{
|
||||
min_depth = 0.0f;
|
||||
max_depth = GX_MAX_DEPTH;
|
||||
}
|
||||
}
|
||||
|
||||
float near_depth, far_depth;
|
||||
if (g_ActiveConfig.backend_info.bSupportsReversedDepthRange)
|
||||
{
|
||||
// Set the reversed depth range.
|
||||
near_depth = max_depth;
|
||||
far_depth = min_depth;
|
||||
}
|
||||
else
|
||||
{
|
||||
// We use an inverted depth range here to apply the Reverse Z trick.
|
||||
// This trick makes sure we match the precision provided by the 1:0
|
||||
// clipping depth range on the hardware.
|
||||
near_depth = 1.0f - max_depth;
|
||||
far_depth = 1.0f - min_depth;
|
||||
}
|
||||
|
||||
g_renderer->SetViewport(x, y, width, height, near_depth, far_depth);
|
||||
}
|
||||
|
||||
void SetDepthMode()
|
||||
{
|
||||
DepthState state = {};
|
||||
|
@ -17,6 +17,7 @@ namespace BPFunctions
|
||||
void FlushPipeline();
|
||||
void SetGenerationMode();
|
||||
void SetScissor();
|
||||
void SetViewport();
|
||||
void SetDepthMode();
|
||||
void SetBlendMode();
|
||||
void ClearScreen(const EFBRectangle& rc);
|
||||
|
@ -130,6 +130,7 @@ static void BPWritten(const BPCmd& bp)
|
||||
case BPMEM_SCISSORBR: // Scissor Rectable Bottom, Right
|
||||
case BPMEM_SCISSOROFFSET: // Scissor Offset
|
||||
SetScissor();
|
||||
SetViewport();
|
||||
VertexShaderManager::SetViewportChanged();
|
||||
GeometryShaderManager::SetViewportChanged();
|
||||
return;
|
||||
@ -1415,6 +1416,7 @@ void BPReload()
|
||||
// note that PixelShaderManager is already covered since it has its own DoState.
|
||||
SetGenerationMode();
|
||||
SetScissor();
|
||||
SetViewport();
|
||||
SetDepthMode();
|
||||
SetBlendMode();
|
||||
OnPixelFormatChange();
|
||||
|
@ -71,12 +71,6 @@ static int OSDTime;
|
||||
|
||||
std::unique_ptr<Renderer> g_renderer;
|
||||
|
||||
// The maximum depth that is written to the depth buffer should never exceed this value.
|
||||
// This is necessary because we use a 2^24 divisor for all our depth values to prevent
|
||||
// floating-point round-trip errors. However the console GPU doesn't ever write a value
|
||||
// to the depth buffer that exceeds 2^24 - 1.
|
||||
const float Renderer::GX_MAX_DEPTH = 16777215.0f / 16777216.0f;
|
||||
|
||||
static float AspectToWidescreen(float aspect)
|
||||
{
|
||||
return aspect * ((16.0f / 9.0f) / (4.0f / 3.0f));
|
||||
|
@ -77,7 +77,10 @@ public:
|
||||
virtual void SetSamplerState(u32 index, const SamplerState& state) {}
|
||||
virtual void UnbindTexture(const AbstractTexture* texture) {}
|
||||
virtual void SetInterlacingMode() {}
|
||||
virtual void SetViewport() {}
|
||||
virtual void SetViewport(float x, float y, float width, float height, float near_depth,
|
||||
float far_depth)
|
||||
{
|
||||
}
|
||||
virtual void SetFullscreen(bool enable_fullscreen) {}
|
||||
virtual bool IsFullscreen() const { return false; }
|
||||
virtual void ApplyState() {}
|
||||
@ -184,8 +187,6 @@ protected:
|
||||
|
||||
std::unique_ptr<PostProcessingShaderImplementation> m_post_processor;
|
||||
|
||||
static const float GX_MAX_DEPTH;
|
||||
|
||||
void* m_surface_handle = nullptr;
|
||||
void* m_new_surface_handle = nullptr;
|
||||
Common::Flag m_surface_needs_change;
|
||||
|
@ -16,6 +16,7 @@
|
||||
#include "Common/MathUtil.h"
|
||||
#include "Core/ConfigManager.h"
|
||||
#include "Core/Core.h"
|
||||
#include "VideoCommon/BPFunctions.h"
|
||||
#include "VideoCommon/BPMemory.h"
|
||||
#include "VideoCommon/CPMemory.h"
|
||||
#include "VideoCommon/RenderBase.h"
|
||||
@ -420,8 +421,7 @@ void VertexShaderManager::SetConstants()
|
||||
}
|
||||
|
||||
dirty = true;
|
||||
// This is so implementation-dependent that we can't have it here.
|
||||
g_renderer->SetViewport();
|
||||
BPFunctions::SetViewport();
|
||||
|
||||
// Update projection if the viewport isn't 1:1 useable
|
||||
if (!g_ActiveConfig.backend_info.bSupportsOversizedViewports)
|
||||
|
Reference in New Issue
Block a user