GPU: Refactor display presentation workflow
Adds the ability to set overlays, and fixes postfx with prerotation.
This commit is contained in:
parent
01db85fa62
commit
0e49ce71da
|
@ -5,6 +5,34 @@
|
||||||
|
|
||||||
#include <cmath>
|
#include <cmath>
|
||||||
|
|
||||||
|
GSVector4i GSVector4i::rfit(const GSVector4i& fit_rect, const GSVector2i& image_size)
|
||||||
|
{
|
||||||
|
const GSVector2 ffit_size = GSVector2(fit_rect.rsize());
|
||||||
|
const GSVector2 fimage_size = GSVector2(image_size);
|
||||||
|
const float fit_ar = ffit_size.x / ffit_size.y;
|
||||||
|
const float image_ar = fimage_size.x / fimage_size.y;
|
||||||
|
|
||||||
|
GSVector4i ret;
|
||||||
|
if (fit_ar > image_ar)
|
||||||
|
{
|
||||||
|
// center horizontally
|
||||||
|
const float width = ffit_size.y * image_ar;
|
||||||
|
const float offset = (ffit_size.x - width) / 2.0f;
|
||||||
|
const float height = ffit_size.y;
|
||||||
|
ret = GSVector4i(GSVector4(offset, 0.0f, offset + width, height));
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
// center vertically
|
||||||
|
const float height = ffit_size.x / image_ar;
|
||||||
|
const float offset = (ffit_size.y - height) / 2.0f;
|
||||||
|
const float width = ffit_size.x;
|
||||||
|
ret = GSVector4i(GSVector4(0.0f, offset, width, offset + height));
|
||||||
|
}
|
||||||
|
|
||||||
|
return ret;
|
||||||
|
}
|
||||||
|
|
||||||
GSMatrix2x2::GSMatrix2x2(float e00, float e01, float e10, float e11)
|
GSMatrix2x2::GSMatrix2x2(float e00, float e01, float e10, float e11)
|
||||||
{
|
{
|
||||||
E[0][0] = e00;
|
E[0][0] = e00;
|
||||||
|
|
|
@ -2345,6 +2345,8 @@ public:
|
||||||
|
|
||||||
ALWAYS_INLINE static GSVector4i xyxy(const GSVector2i& xyzw) { return GSVector4i(vcombine_s32(xyzw.v2s, xyzw.v2s)); }
|
ALWAYS_INLINE static GSVector4i xyxy(const GSVector2i& xyzw) { return GSVector4i(vcombine_s32(xyzw.v2s, xyzw.v2s)); }
|
||||||
|
|
||||||
|
static GSVector4i rfit(const GSVector4i& fit_rect, const GSVector2i& image_size);
|
||||||
|
|
||||||
ALWAYS_INLINE GSVector2i xy() const { return GSVector2i(vget_low_s32(v4s)); }
|
ALWAYS_INLINE GSVector2i xy() const { return GSVector2i(vget_low_s32(v4s)); }
|
||||||
|
|
||||||
ALWAYS_INLINE GSVector2i zw() const { return GSVector2i(vget_high_s32(v4s)); }
|
ALWAYS_INLINE GSVector2i zw() const { return GSVector2i(vget_high_s32(v4s)); }
|
||||||
|
|
|
@ -1565,7 +1565,6 @@ public:
|
||||||
return ret;
|
return ret;
|
||||||
}
|
}
|
||||||
|
|
||||||
template<bool aligned>
|
|
||||||
ALWAYS_INLINE static GSVector4i loadh(const GSVector2i& v)
|
ALWAYS_INLINE static GSVector4i loadh(const GSVector2i& v)
|
||||||
{
|
{
|
||||||
return loadh<true>(&v);
|
return loadh<true>(&v);
|
||||||
|
@ -1668,6 +1667,8 @@ public:
|
||||||
|
|
||||||
ALWAYS_INLINE static GSVector4i xyxy(const GSVector2i& xyzw) { return GSVector4i(xyzw.x, xyzw.y, xyzw.x, xyzw.y); }
|
ALWAYS_INLINE static GSVector4i xyxy(const GSVector2i& xyzw) { return GSVector4i(xyzw.x, xyzw.y, xyzw.x, xyzw.y); }
|
||||||
|
|
||||||
|
static GSVector4i rfit(const GSVector4i& fit_rect, const GSVector2i& image_size);
|
||||||
|
|
||||||
ALWAYS_INLINE GSVector2i xy() const { return GSVector2i(x, y); }
|
ALWAYS_INLINE GSVector2i xy() const { return GSVector2i(x, y); }
|
||||||
ALWAYS_INLINE GSVector2i zw() const { return GSVector2i(z, w); }
|
ALWAYS_INLINE GSVector2i zw() const { return GSVector2i(z, w); }
|
||||||
|
|
||||||
|
|
|
@ -1745,7 +1745,6 @@ public:
|
||||||
return GSVector4i(_mm_castps_si128(_mm_loadh_pi(_mm_setzero_ps(), static_cast<const __m64*>(p))));
|
return GSVector4i(_mm_castps_si128(_mm_loadh_pi(_mm_setzero_ps(), static_cast<const __m64*>(p))));
|
||||||
}
|
}
|
||||||
|
|
||||||
template<bool aligned>
|
|
||||||
ALWAYS_INLINE static GSVector4i loadh(const GSVector2i& v)
|
ALWAYS_INLINE static GSVector4i loadh(const GSVector2i& v)
|
||||||
{
|
{
|
||||||
return GSVector4i(_mm_unpacklo_epi64(_mm_setzero_si128(), v.m));
|
return GSVector4i(_mm_unpacklo_epi64(_mm_setzero_si128(), v.m));
|
||||||
|
@ -1842,6 +1841,8 @@ public:
|
||||||
return GSVector4i(_mm_unpacklo_epi64(xy.m, zw.m));
|
return GSVector4i(_mm_unpacklo_epi64(xy.m, zw.m));
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static GSVector4i rfit(const GSVector4i& fit_rect, const GSVector2i& image_size);
|
||||||
|
|
||||||
ALWAYS_INLINE GSVector2i xy() const { return GSVector2i(m); }
|
ALWAYS_INLINE GSVector2i xy() const { return GSVector2i(m); }
|
||||||
|
|
||||||
ALWAYS_INLINE GSVector2i zw() const { return GSVector2i(_mm_shuffle_epi32(m, _MM_SHUFFLE(3, 2, 3, 2))); }
|
ALWAYS_INLINE GSVector2i zw() const { return GSVector2i(_mm_shuffle_epi32(m, _MM_SHUFFLE(3, 2, 3, 2))); }
|
||||||
|
|
|
@ -29,6 +29,7 @@
|
||||||
#include "common/gsvector_formatter.h"
|
#include "common/gsvector_formatter.h"
|
||||||
#include "common/log.h"
|
#include "common/log.h"
|
||||||
#include "common/path.h"
|
#include "common/path.h"
|
||||||
|
#include "common/settings_interface.h"
|
||||||
#include "common/small_string.h"
|
#include "common/small_string.h"
|
||||||
#include "common/string_util.h"
|
#include "common/string_util.h"
|
||||||
#include "common/threading.h"
|
#include "common/threading.h"
|
||||||
|
@ -38,6 +39,8 @@
|
||||||
|
|
||||||
LOG_CHANNEL(GPU);
|
LOG_CHANNEL(GPU);
|
||||||
|
|
||||||
|
#include "common/ryml_helpers.h"
|
||||||
|
|
||||||
static constexpr GPUTexture::Format DISPLAY_INTERNAL_POSTFX_FORMAT = GPUTexture::Format::RGBA8;
|
static constexpr GPUTexture::Format DISPLAY_INTERNAL_POSTFX_FORMAT = GPUTexture::Format::RGBA8;
|
||||||
|
|
||||||
GPUPresenter::GPUPresenter() = default;
|
GPUPresenter::GPUPresenter() = default;
|
||||||
|
@ -50,6 +53,15 @@ GPUPresenter::~GPUPresenter()
|
||||||
|
|
||||||
bool GPUPresenter::Initialize(Error* error)
|
bool GPUPresenter::Initialize(Error* error)
|
||||||
{
|
{
|
||||||
|
// we can't change the format after compiling shaders
|
||||||
|
m_present_format =
|
||||||
|
g_gpu_device->HasMainSwapChain() ? g_gpu_device->GetMainSwapChain()->GetFormat() : GPUTexture::Format::RGBA8;
|
||||||
|
VERBOSE_LOG("Presentation format is {}", GPUTexture::GetFormatName(m_present_format));
|
||||||
|
|
||||||
|
// overlay has to come first, because it sets the alpha blending on the display pipeline
|
||||||
|
if (LoadOverlaySettings())
|
||||||
|
LoadOverlayTexture();
|
||||||
|
|
||||||
if (!CompileDisplayPipelines(true, true, g_gpu_settings.display_24bit_chroma_smoothing, error))
|
if (!CompileDisplayPipelines(true, true, g_gpu_settings.display_24bit_chroma_smoothing, error))
|
||||||
return false;
|
return false;
|
||||||
|
|
||||||
|
@ -79,6 +91,22 @@ bool GPUPresenter::UpdateSettings(const GPUSettings& old_settings, Error* error)
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
bool GPUPresenter::UpdatePostProcessingSettings(Error* error)
|
||||||
|
{
|
||||||
|
if (LoadOverlaySettings())
|
||||||
|
{
|
||||||
|
// something changed, need to recompile pipelines
|
||||||
|
if (LoadOverlayTexture() && m_border_overlay_alpha_blend &&
|
||||||
|
(!m_present_copy_blend_pipeline || !m_display_blend_pipeline) &&
|
||||||
|
!CompileDisplayPipelines(true, false, false, error))
|
||||||
|
{
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
bool GPUPresenter::CompileDisplayPipelines(bool display, bool deinterlace, bool chroma_smoothing, Error* error)
|
bool GPUPresenter::CompileDisplayPipelines(bool display, bool deinterlace, bool chroma_smoothing, Error* error)
|
||||||
{
|
{
|
||||||
const GPUShaderGen shadergen(g_gpu_device->GetRenderAPI(), g_gpu_device->GetFeatures().dual_source_blend,
|
const GPUShaderGen shadergen(g_gpu_device->GetRenderAPI(), g_gpu_device->GetFeatures().dual_source_blend,
|
||||||
|
@ -100,10 +128,14 @@ bool GPUPresenter::CompileDisplayPipelines(bool display, bool deinterlace, bool
|
||||||
GPUBackend::SetScreenQuadInputLayout(plconfig);
|
GPUBackend::SetScreenQuadInputLayout(plconfig);
|
||||||
|
|
||||||
plconfig.layout = GPUPipeline::Layout::SingleTextureAndPushConstants;
|
plconfig.layout = GPUPipeline::Layout::SingleTextureAndPushConstants;
|
||||||
plconfig.SetTargetFormats(g_gpu_device->HasMainSwapChain() ? g_gpu_device->GetMainSwapChain()->GetFormat() :
|
plconfig.SetTargetFormats(m_present_format);
|
||||||
GPUTexture::Format::RGBA8);
|
|
||||||
|
std::unique_ptr<GPUShader> vso = g_gpu_device->CreateShader(GPUShaderStage::Vertex, shadergen.GetLanguage(),
|
||||||
|
shadergen.GenerateDisplayVertexShader(), error);
|
||||||
|
if (!vso)
|
||||||
|
return false;
|
||||||
|
GL_OBJECT_NAME(vso, "Display Vertex Shader");
|
||||||
|
|
||||||
std::string vs = shadergen.GenerateDisplayVertexShader();
|
|
||||||
std::string fs;
|
std::string fs;
|
||||||
switch (g_gpu_settings.display_scaling)
|
switch (g_gpu_settings.display_scaling)
|
||||||
{
|
{
|
||||||
|
@ -123,25 +155,59 @@ bool GPUPresenter::CompileDisplayPipelines(bool display, bool deinterlace, bool
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
|
|
||||||
std::unique_ptr<GPUShader> vso =
|
|
||||||
g_gpu_device->CreateShader(GPUShaderStage::Vertex, shadergen.GetLanguage(), vs, error);
|
|
||||||
std::unique_ptr<GPUShader> fso =
|
std::unique_ptr<GPUShader> fso =
|
||||||
g_gpu_device->CreateShader(GPUShaderStage::Fragment, shadergen.GetLanguage(), fs, error);
|
g_gpu_device->CreateShader(GPUShaderStage::Fragment, shadergen.GetLanguage(), fs, error);
|
||||||
if (!vso || !fso)
|
if (!fso)
|
||||||
return false;
|
return false;
|
||||||
GL_OBJECT_NAME(vso, "Display Vertex Shader");
|
|
||||||
GL_OBJECT_NAME_FMT(fso, "Display Fragment Shader [{}]",
|
GL_OBJECT_NAME_FMT(fso, "Display Fragment Shader [{}]",
|
||||||
Settings::GetDisplayScalingName(g_gpu_settings.display_scaling));
|
Settings::GetDisplayScalingName(g_gpu_settings.display_scaling));
|
||||||
|
|
||||||
plconfig.vertex_shader = vso.get();
|
plconfig.vertex_shader = vso.get();
|
||||||
plconfig.fragment_shader = fso.get();
|
plconfig.fragment_shader = fso.get();
|
||||||
if (!(m_display_pipeline = g_gpu_device->CreatePipeline(plconfig, error)))
|
if (!(m_display_pipeline = g_gpu_device->CreatePipeline(plconfig, error)))
|
||||||
return false;
|
return false;
|
||||||
GL_OBJECT_NAME_FMT(m_display_pipeline, "Display Pipeline [{}]",
|
GL_OBJECT_NAME_FMT(m_display_pipeline, "Display Pipeline [{}]",
|
||||||
Settings::GetDisplayScalingName(g_gpu_settings.display_scaling));
|
Settings::GetDisplayScalingName(g_gpu_settings.display_scaling));
|
||||||
|
|
||||||
|
std::unique_ptr<GPUShader> rotate_copy_fso = g_gpu_device->CreateShader(
|
||||||
|
GPUShaderStage::Fragment, shadergen.GetLanguage(), shadergen.GenerateCopyFragmentShader(false), error);
|
||||||
|
if (!rotate_copy_fso)
|
||||||
|
return false;
|
||||||
|
GL_OBJECT_NAME(rotate_copy_fso, "Display Rotate/Copy Fragment Shader");
|
||||||
|
|
||||||
|
plconfig.fragment_shader = rotate_copy_fso.get();
|
||||||
|
if (!(m_present_copy_pipeline = g_gpu_device->CreatePipeline(plconfig, error)))
|
||||||
|
return false;
|
||||||
|
GL_OBJECT_NAME(m_present_copy_pipeline, "Display Rotate/Copy Pipeline");
|
||||||
|
|
||||||
|
// blended variants
|
||||||
|
if (m_border_overlay_texture && m_border_overlay_alpha_blend)
|
||||||
|
{
|
||||||
|
// destination blend the main present, not source
|
||||||
|
plconfig.blend.enable = true;
|
||||||
|
plconfig.blend.src_blend = GPUPipeline::BlendFunc::InvDstAlpha;
|
||||||
|
plconfig.blend.blend_op = GPUPipeline::BlendOp::Add;
|
||||||
|
plconfig.blend.dst_blend = GPUPipeline::BlendFunc::One;
|
||||||
|
plconfig.blend.src_alpha_blend = GPUPipeline::BlendFunc::One;
|
||||||
|
plconfig.blend.alpha_blend_op = GPUPipeline::BlendOp::Add;
|
||||||
|
plconfig.blend.dst_alpha_blend = GPUPipeline::BlendFunc::Zero;
|
||||||
|
|
||||||
|
plconfig.fragment_shader = fso.get();
|
||||||
|
if (!(m_display_blend_pipeline = g_gpu_device->CreatePipeline(plconfig, error)))
|
||||||
|
return false;
|
||||||
|
GL_OBJECT_NAME_FMT(m_display_blend_pipeline, "Display Pipeline [Blended, {}]",
|
||||||
|
Settings::GetDisplayScalingName(g_gpu_settings.display_scaling));
|
||||||
|
|
||||||
|
plconfig.fragment_shader = rotate_copy_fso.get();
|
||||||
|
if (!(m_present_copy_blend_pipeline = g_gpu_device->CreatePipeline(plconfig, error)))
|
||||||
|
return false;
|
||||||
|
GL_OBJECT_NAME(m_present_copy_blend_pipeline, "Display Rotate/Copy Pipeline [Blended]");
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
plconfig.input_layout = {};
|
plconfig.input_layout = {};
|
||||||
plconfig.primitive = GPUPipeline::Primitive::Triangles;
|
plconfig.primitive = GPUPipeline::Primitive::Triangles;
|
||||||
|
plconfig.blend = GPUPipeline::BlendState::GetNoBlendingState();
|
||||||
|
|
||||||
if (deinterlace)
|
if (deinterlace)
|
||||||
{
|
{
|
||||||
|
@ -304,18 +370,38 @@ void GPUPresenter::SetDisplayTexture(GPUTexture* texture, GPUTexture* depth_buff
|
||||||
|
|
||||||
GPUDevice::PresentResult GPUPresenter::PresentDisplay()
|
GPUDevice::PresentResult GPUPresenter::PresentDisplay()
|
||||||
{
|
{
|
||||||
if (!g_gpu_device->HasMainSwapChain())
|
DebugAssert(g_gpu_device->HasMainSwapChain());
|
||||||
return GPUDevice::PresentResult::SkipPresent;
|
|
||||||
|
u32 display_area_width = g_gpu_device->GetMainSwapChain()->GetWidth();
|
||||||
|
u32 display_area_height = g_gpu_device->GetMainSwapChain()->GetHeight();
|
||||||
|
GSVector4i overlay_display_rect = GSVector4i::zero();
|
||||||
|
GSVector4i overlay_rect = GSVector4i::zero();
|
||||||
|
if (m_border_overlay_texture)
|
||||||
|
{
|
||||||
|
overlay_rect = GSVector4i::rfit(GSVector4i(0, 0, display_area_width, display_area_height),
|
||||||
|
m_border_overlay_texture->GetSizeVec());
|
||||||
|
|
||||||
|
const GSVector2 scale = GSVector2(overlay_rect.rsize()) / GSVector2(m_border_overlay_texture->GetSizeVec());
|
||||||
|
overlay_display_rect =
|
||||||
|
GSVector4i(GSVector4(m_border_overlay_display_rect) * GSVector4::xyxy(scale)).add32(overlay_rect.xyxy());
|
||||||
|
display_area_width = overlay_display_rect.width();
|
||||||
|
display_area_height = overlay_display_rect.height();
|
||||||
|
}
|
||||||
|
|
||||||
GSVector4i display_rect;
|
GSVector4i display_rect;
|
||||||
GSVector4i draw_rect;
|
GSVector4i draw_rect;
|
||||||
CalculateDrawRect(g_gpu_device->GetMainSwapChain()->GetWidth(), g_gpu_device->GetMainSwapChain()->GetHeight(),
|
CalculateDrawRect(display_area_width, display_area_height, !g_gpu_settings.gpu_show_vram, true, &display_rect,
|
||||||
!g_gpu_settings.gpu_show_vram, true, &display_rect, &draw_rect);
|
&draw_rect);
|
||||||
return RenderDisplay(nullptr, display_rect, draw_rect, !g_gpu_settings.gpu_show_vram);
|
|
||||||
|
display_rect = display_rect.add32(overlay_display_rect.xyxy());
|
||||||
|
draw_rect = draw_rect.add32(overlay_display_rect.xyxy());
|
||||||
|
|
||||||
|
return RenderDisplay(nullptr, overlay_rect, display_rect, draw_rect, !g_gpu_settings.gpu_show_vram);
|
||||||
}
|
}
|
||||||
|
|
||||||
GPUDevice::PresentResult GPUPresenter::RenderDisplay(GPUTexture* target, const GSVector4i display_rect,
|
GPUDevice::PresentResult GPUPresenter::RenderDisplay(GPUTexture* target, const GSVector4i overlay_rect,
|
||||||
const GSVector4i draw_rect, bool postfx)
|
const GSVector4i display_rect, const GSVector4i draw_rect,
|
||||||
|
bool postfx)
|
||||||
{
|
{
|
||||||
GL_SCOPE_FMT("RenderDisplay: {}", draw_rect);
|
GL_SCOPE_FMT("RenderDisplay: {}", draw_rect);
|
||||||
|
|
||||||
|
@ -332,10 +418,6 @@ GPUDevice::PresentResult GPUPresenter::RenderDisplay(GPUTexture* target, const G
|
||||||
PostProcessing::InternalChain.CheckTargets(DISPLAY_INTERNAL_POSTFX_FORMAT, display_texture_view_width,
|
PostProcessing::InternalChain.CheckTargets(DISPLAY_INTERNAL_POSTFX_FORMAT, display_texture_view_width,
|
||||||
display_texture_view_height))
|
display_texture_view_height))
|
||||||
{
|
{
|
||||||
DebugAssert(display_texture_view_x == 0 && display_texture_view_y == 0 &&
|
|
||||||
static_cast<s32>(display_texture->GetWidth()) == display_texture_view_width &&
|
|
||||||
static_cast<s32>(display_texture->GetHeight()) == display_texture_view_height);
|
|
||||||
|
|
||||||
// Now we can apply the post chain.
|
// Now we can apply the post chain.
|
||||||
GPUTexture* post_output_texture = PostProcessing::InternalChain.GetOutputTexture();
|
GPUTexture* post_output_texture = PostProcessing::InternalChain.GetOutputTexture();
|
||||||
if (PostProcessing::InternalChain.Apply(display_texture, m_display_depth_buffer, post_output_texture,
|
if (PostProcessing::InternalChain.Apply(display_texture, m_display_depth_buffer, post_output_texture,
|
||||||
|
@ -350,39 +432,121 @@ GPUDevice::PresentResult GPUPresenter::RenderDisplay(GPUTexture* target, const G
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
const GPUTexture::Format hdformat = target ? target->GetFormat() : g_gpu_device->GetMainSwapChain()->GetFormat();
|
// There's a bunch of scenarios where we need to use intermediate buffers.
|
||||||
const u32 target_width = target ? target->GetWidth() : g_gpu_device->GetMainSwapChain()->GetWidth();
|
// If we have post-processing and overlays enabled, postfx needs to happen on an intermediate buffer first.
|
||||||
const u32 target_height = target ? target->GetHeight() : g_gpu_device->GetMainSwapChain()->GetHeight();
|
// If pre-rotation is enabled with post-processing, we need to draw to an intermediate buffer, and apply the
|
||||||
const bool really_postfx = (postfx && PostProcessing::DisplayChain.IsActive() && g_gpu_device->HasMainSwapChain() &&
|
// rotation at the end.
|
||||||
hdformat != GPUTexture::Format::Unknown && target_width > 0 && target_height > 0 &&
|
GPUSwapChain* const swap_chain = g_gpu_device->GetMainSwapChain();
|
||||||
PostProcessing::DisplayChain.CheckTargets(hdformat, target_width, target_height));
|
const WindowInfo::PreRotation prerotation = target ? WindowInfo::PreRotation::Identity : swap_chain->GetPreRotation();
|
||||||
const u32 real_target_width =
|
const bool have_overlay = static_cast<bool>(m_border_overlay_texture);
|
||||||
(target || really_postfx) ? target_width : g_gpu_device->GetMainSwapChain()->GetPostRotatedWidth();
|
const bool have_prerotation = (prerotation != WindowInfo::PreRotation::Identity);
|
||||||
const u32 real_target_height =
|
const GSVector2i target_size = target ? target->GetSizeVec() : swap_chain->GetSizeVec();
|
||||||
(target || really_postfx) ? target_height : g_gpu_device->GetMainSwapChain()->GetPostRotatedHeight();
|
GL_INS(have_overlay ? "Overlay is ENABLED" : "Overlay is disabled");
|
||||||
GSVector4i real_draw_rect =
|
GL_INS_FMT("Prerotation: {}", static_cast<u32>(prerotation));
|
||||||
(target || really_postfx) ? draw_rect : g_gpu_device->GetMainSwapChain()->PreRotateClipRect(draw_rect);
|
GL_INS_FMT("Final target size: {}x{}", target_size.x, target_size.y);
|
||||||
if (really_postfx)
|
|
||||||
{
|
// Postfx active?
|
||||||
g_gpu_device->ClearRenderTarget(PostProcessing::DisplayChain.GetInputTexture(), GPUDevice::DEFAULT_CLEAR_COLOR);
|
const GSVector2i postfx_size = have_overlay ? display_rect.rsize() : target_size;
|
||||||
g_gpu_device->SetRenderTarget(PostProcessing::DisplayChain.GetInputTexture());
|
const bool really_postfx =
|
||||||
}
|
(postfx && PostProcessing::DisplayChain.IsActive() &&
|
||||||
else
|
PostProcessing::DisplayChain.CheckTargets(m_present_format, postfx_size.x, postfx_size.y));
|
||||||
{
|
GL_INS(really_postfx ? "Post-processing is ENABLED" : "Post-processing is disabled");
|
||||||
|
GL_INS_FMT("Post-processing render target size: {}x{}", postfx_size.x, postfx_size.y);
|
||||||
|
|
||||||
|
// Helper to bind swap chain/final target.
|
||||||
|
const auto bind_final_target = [target, swap_chain](bool clear) {
|
||||||
if (target)
|
if (target)
|
||||||
{
|
{
|
||||||
|
if (clear)
|
||||||
|
g_gpu_device->ClearRenderTarget(target, GPUDevice::DEFAULT_CLEAR_COLOR);
|
||||||
|
else
|
||||||
|
g_gpu_device->InvalidateRenderTarget(target);
|
||||||
g_gpu_device->SetRenderTarget(target);
|
g_gpu_device->SetRenderTarget(target);
|
||||||
|
return GPUDevice::PresentResult::OK;
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
const GPUDevice::PresentResult pres = g_gpu_device->BeginPresent(g_gpu_device->GetMainSwapChain());
|
return g_gpu_device->BeginPresent(swap_chain);
|
||||||
if (pres != GPUDevice::PresentResult::OK)
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
// If postfx is enabled, we need to draw to an intermediate buffer first.
|
||||||
|
if (really_postfx)
|
||||||
|
{
|
||||||
|
// Remove draw offset if we're using an overlay.
|
||||||
|
const GSVector4i real_draw_rect = have_overlay ? draw_rect.sub32(display_rect.xyxy()) : draw_rect;
|
||||||
|
|
||||||
|
// Display is always drawn to the postfx input.
|
||||||
|
GPUTexture* const postfx_input = PostProcessing::DisplayChain.GetInputTexture();
|
||||||
|
g_gpu_device->ClearRenderTarget(postfx_input, GPUDevice::DEFAULT_CLEAR_COLOR);
|
||||||
|
g_gpu_device->SetRenderTarget(postfx_input);
|
||||||
|
if (display_texture)
|
||||||
|
{
|
||||||
|
DrawDisplay(postfx_size, display_texture, display_texture_view_x, display_texture_view_y,
|
||||||
|
display_texture_view_width, display_texture_view_height, real_draw_rect, false,
|
||||||
|
g_gpu_settings.display_rotation, WindowInfo::PreRotation::Identity);
|
||||||
|
}
|
||||||
|
postfx_input->MakeReadyForSampling();
|
||||||
|
|
||||||
|
// Apply postprocessing to an intermediate texture if we're prerotating or have an overlay.
|
||||||
|
if (have_prerotation || have_overlay)
|
||||||
|
{
|
||||||
|
GPUTexture* const postfx_output = PostProcessing::DisplayChain.GetTextureUnusedAtEndOfChain();
|
||||||
|
const GSVector4i real_display_rect = have_overlay ? display_rect.sub32(display_rect.xyxy()) : display_rect;
|
||||||
|
ApplyDisplayPostProcess(postfx_output, postfx_input, real_display_rect);
|
||||||
|
postfx_output->MakeReadyForSampling();
|
||||||
|
|
||||||
|
// Start draw to final buffer.
|
||||||
|
if (const GPUDevice::PresentResult pres = bind_final_target(have_overlay); pres != GPUDevice::PresentResult::OK)
|
||||||
return pres;
|
return pres;
|
||||||
|
|
||||||
|
// If we have an overlay, draw it, and then copy the postprocessed framebuffer in.
|
||||||
|
if (have_overlay)
|
||||||
|
{
|
||||||
|
DrawTextureCopy(target_size, overlay_rect, m_border_overlay_texture.get(), false, true, prerotation);
|
||||||
|
DrawTextureCopy(target_size, draw_rect, postfx_output, m_border_overlay_alpha_blend, false, prerotation);
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
// Ohterwise, just copy the framebuffer.
|
||||||
|
DrawTextureCopy(target_size, draw_rect, postfx_output, false, false, prerotation);
|
||||||
|
}
|
||||||
|
|
||||||
|
// All done
|
||||||
|
return GPUDevice::PresentResult::OK;
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
// Otherwise apply postprocessing directly to swap chain.
|
||||||
|
return ApplyDisplayPostProcess(target, postfx_input, display_rect);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
// The non-postprocessing cases are much simpler. We always optionally draw the overlay, then draw the display.
|
||||||
|
// The only tricky bit is we have to combine the display rotation and prerotation for the latter.
|
||||||
|
if (const GPUDevice::PresentResult pres = bind_final_target(true); pres != GPUDevice::PresentResult::OK)
|
||||||
|
return pres;
|
||||||
|
|
||||||
|
if (have_overlay)
|
||||||
|
DrawTextureCopy(target_size, overlay_rect, m_border_overlay_texture.get(), false, true, prerotation);
|
||||||
|
|
||||||
if (display_texture)
|
if (display_texture)
|
||||||
{
|
{
|
||||||
|
DrawDisplay(target_size, display_texture, display_texture_view_x, display_texture_view_y,
|
||||||
|
display_texture_view_width, display_texture_view_height, display_rect, m_border_overlay_alpha_blend,
|
||||||
|
g_gpu_settings.display_rotation, prerotation);
|
||||||
|
}
|
||||||
|
|
||||||
|
return GPUDevice::PresentResult::OK;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
void GPUPresenter::DrawDisplay(const GSVector2i target_size, GPUTexture* display_texture, s32 display_texture_view_x,
|
||||||
|
s32 display_texture_view_y, s32 display_texture_view_width,
|
||||||
|
s32 display_texture_view_height, const GSVector4i display_rect, bool dst_alpha_blend,
|
||||||
|
DisplayRotation rotation, WindowInfo::PreRotation prerotation)
|
||||||
|
{
|
||||||
bool texture_filter_linear = false;
|
bool texture_filter_linear = false;
|
||||||
|
|
||||||
struct alignas(16) Uniforms
|
struct alignas(16) Uniforms
|
||||||
|
@ -408,9 +572,10 @@ GPUDevice::PresentResult GPUPresenter::RenderDisplay(GPUTexture* target, const G
|
||||||
{
|
{
|
||||||
texture_filter_linear = true;
|
texture_filter_linear = true;
|
||||||
uniforms.params[0] = std::max(
|
uniforms.params[0] = std::max(
|
||||||
std::floor(static_cast<float>(draw_rect.width()) / static_cast<float>(m_display_texture_view_width)), 1.0f);
|
std::floor(static_cast<float>(display_rect.width()) / static_cast<float>(m_display_texture_view_width)), 1.0f);
|
||||||
uniforms.params[1] = std::max(
|
uniforms.params[1] = std::max(
|
||||||
std::floor(static_cast<float>(draw_rect.height()) / static_cast<float>(m_display_texture_view_height)), 1.0f);
|
std::floor(static_cast<float>(display_rect.height()) / static_cast<float>(m_display_texture_view_height)),
|
||||||
|
1.0f);
|
||||||
uniforms.params[2] = 0.5f - 0.5f / uniforms.params[0];
|
uniforms.params[2] = 0.5f - 0.5f / uniforms.params[0];
|
||||||
uniforms.params[3] = 0.5f - 0.5f / uniforms.params[1];
|
uniforms.params[3] = 0.5f - 0.5f / uniforms.params[1];
|
||||||
}
|
}
|
||||||
|
@ -421,7 +586,7 @@ GPUDevice::PresentResult GPUPresenter::RenderDisplay(GPUTexture* target, const G
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
|
|
||||||
g_gpu_device->SetPipeline(m_display_pipeline.get());
|
g_gpu_device->SetPipeline(dst_alpha_blend ? m_display_blend_pipeline.get() : m_display_pipeline.get());
|
||||||
g_gpu_device->SetTextureSampler(
|
g_gpu_device->SetTextureSampler(
|
||||||
0, display_texture, texture_filter_linear ? g_gpu_device->GetLinearSampler() : g_gpu_device->GetNearestSampler());
|
0, display_texture, texture_filter_linear ? g_gpu_device->GetLinearSampler() : g_gpu_device->GetNearestSampler());
|
||||||
|
|
||||||
|
@ -443,11 +608,16 @@ GPUDevice::PresentResult GPUPresenter::RenderDisplay(GPUTexture* target, const G
|
||||||
GSVector4::xyxy(display_texture_size, GSVector2::cxpr(1.0f) / display_texture_size));
|
GSVector4::xyxy(display_texture_size, GSVector2::cxpr(1.0f) / display_texture_size));
|
||||||
|
|
||||||
g_gpu_device->PushUniformBuffer(&uniforms, sizeof(uniforms));
|
g_gpu_device->PushUniformBuffer(&uniforms, sizeof(uniforms));
|
||||||
|
DrawScreenQuad(display_rect, uv_rect, target_size, rotation, prerotation);
|
||||||
|
}
|
||||||
|
|
||||||
g_gpu_device->SetViewport(0, 0, real_target_width, real_target_height);
|
void GPUPresenter::DrawScreenQuad(const GSVector4i rect, const GSVector4 uv_rect, const GSVector2i target_size,
|
||||||
g_gpu_device->SetScissor(g_gpu_device->UsesLowerLeftOrigin() ?
|
DisplayRotation rotation, WindowInfo::PreRotation prerotation)
|
||||||
GPUDevice::FlipToLowerLeft(real_draw_rect, real_target_height) :
|
{
|
||||||
real_draw_rect);
|
const GSVector4i real_rect = GPUSwapChain::PreRotateClipRect(prerotation, target_size, rect);
|
||||||
|
g_gpu_device->SetViewport(GSVector4i::loadh(target_size));
|
||||||
|
g_gpu_device->SetScissor(g_gpu_device->UsesLowerLeftOrigin() ? GPUDevice::FlipToLowerLeft(real_rect, target_size.y) :
|
||||||
|
real_rect);
|
||||||
|
|
||||||
GPUBackend::ScreenVertex* vertices;
|
GPUBackend::ScreenVertex* vertices;
|
||||||
u32 space;
|
u32 space;
|
||||||
|
@ -455,17 +625,12 @@ GPUDevice::PresentResult GPUPresenter::RenderDisplay(GPUTexture* target, const G
|
||||||
g_gpu_device->MapVertexBuffer(sizeof(GPUBackend::ScreenVertex), 4, reinterpret_cast<void**>(&vertices), &space,
|
g_gpu_device->MapVertexBuffer(sizeof(GPUBackend::ScreenVertex), 4, reinterpret_cast<void**>(&vertices), &space,
|
||||||
&base_vertex);
|
&base_vertex);
|
||||||
|
|
||||||
const WindowInfo::PreRotation surface_prerotation = (target || really_postfx) ?
|
const GSVector4 xy = GPUBackend::GetScreenQuadClipSpaceCoordinates(real_rect, target_size);
|
||||||
WindowInfo::PreRotation::Identity :
|
|
||||||
g_gpu_device->GetMainSwapChain()->GetPreRotation();
|
|
||||||
|
|
||||||
const DisplayRotation uv_rotation = static_cast<DisplayRotation>(
|
// Combine display rotation and prerotation together, since the rectangle has already been adjusted.
|
||||||
(static_cast<u32>(g_gpu_settings.display_rotation) + static_cast<u32>(surface_prerotation)) %
|
const DisplayRotation effective_rotation = static_cast<DisplayRotation>(
|
||||||
static_cast<u32>(DisplayRotation::Count));
|
(static_cast<u32>(rotation) + static_cast<u32>(prerotation)) % static_cast<u32>(DisplayRotation::Count));
|
||||||
|
switch (effective_rotation)
|
||||||
const GSVector4 xy =
|
|
||||||
GPUBackend::GetScreenQuadClipSpaceCoordinates(real_draw_rect, GSVector2i(real_target_width, real_target_height));
|
|
||||||
switch (uv_rotation)
|
|
||||||
{
|
{
|
||||||
case DisplayRotation::Normal:
|
case DisplayRotation::Normal:
|
||||||
vertices[0].Set(xy.xy(), uv_rect.xy());
|
vertices[0].Set(xy.xy(), uv_rect.xy());
|
||||||
|
@ -473,18 +638,21 @@ GPUDevice::PresentResult GPUPresenter::RenderDisplay(GPUTexture* target, const G
|
||||||
vertices[2].Set(xy.xwzw().xy(), uv_rect.xwzw().xy());
|
vertices[2].Set(xy.xwzw().xy(), uv_rect.xwzw().xy());
|
||||||
vertices[3].Set(xy.zw(), uv_rect.zw());
|
vertices[3].Set(xy.zw(), uv_rect.zw());
|
||||||
break;
|
break;
|
||||||
|
|
||||||
case DisplayRotation::Rotate90:
|
case DisplayRotation::Rotate90:
|
||||||
vertices[0].Set(xy.xy(), uv_rect.xwzw().xy());
|
vertices[0].Set(xy.xy(), uv_rect.xwzw().xy());
|
||||||
vertices[1].Set(xy.zyzw().xy(), uv_rect.xy());
|
vertices[1].Set(xy.zyzw().xy(), uv_rect.xy());
|
||||||
vertices[2].Set(xy.xwzw().xy(), uv_rect.zw());
|
vertices[2].Set(xy.xwzw().xy(), uv_rect.zw());
|
||||||
vertices[3].Set(xy.zw(), uv_rect.zyzw().xy());
|
vertices[3].Set(xy.zw(), uv_rect.zyzw().xy());
|
||||||
break;
|
break;
|
||||||
|
|
||||||
case DisplayRotation::Rotate180:
|
case DisplayRotation::Rotate180:
|
||||||
vertices[0].Set(xy.xy(), uv_rect.xwzw().xy());
|
vertices[0].Set(xy.xy(), uv_rect.xwzw().xy());
|
||||||
vertices[1].Set(xy.zyzw().xy(), uv_rect.zw());
|
vertices[1].Set(xy.zyzw().xy(), uv_rect.zw());
|
||||||
vertices[2].Set(xy.xwzw().xy(), uv_rect.xy());
|
vertices[2].Set(xy.xwzw().xy(), uv_rect.xy());
|
||||||
vertices[3].Set(xy.zw(), uv_rect.zyzw().xy());
|
vertices[3].Set(xy.zw(), uv_rect.zyzw().xy());
|
||||||
break;
|
break;
|
||||||
|
|
||||||
case DisplayRotation::Rotate270:
|
case DisplayRotation::Rotate270:
|
||||||
vertices[0].Set(xy.xy(), uv_rect.zyzw().xy());
|
vertices[0].Set(xy.xy(), uv_rect.zyzw().xy());
|
||||||
vertices[1].Set(xy.zyzw().xy(), uv_rect.zw());
|
vertices[1].Set(xy.zyzw().xy(), uv_rect.zw());
|
||||||
|
@ -497,29 +665,37 @@ GPUDevice::PresentResult GPUPresenter::RenderDisplay(GPUTexture* target, const G
|
||||||
|
|
||||||
g_gpu_device->UnmapVertexBuffer(sizeof(GPUBackend::ScreenVertex), 4);
|
g_gpu_device->UnmapVertexBuffer(sizeof(GPUBackend::ScreenVertex), 4);
|
||||||
g_gpu_device->Draw(4, base_vertex);
|
g_gpu_device->Draw(4, base_vertex);
|
||||||
}
|
}
|
||||||
|
|
||||||
if (really_postfx)
|
GPUDevice::PresentResult GPUPresenter::ApplyDisplayPostProcess(GPUTexture* target, GPUTexture* input,
|
||||||
{
|
const GSVector4i display_rect)
|
||||||
|
{
|
||||||
DebugAssert(!g_gpu_settings.gpu_show_vram);
|
DebugAssert(!g_gpu_settings.gpu_show_vram);
|
||||||
|
|
||||||
// "original size" in postfx includes padding.
|
// "original size" in postfx includes padding.
|
||||||
const float upscale_x =
|
const float upscale_x =
|
||||||
m_display_texture ? static_cast<float>(m_display_texture_view_width) / static_cast<float>(m_display_vram_width) :
|
m_display_texture ? static_cast<float>(m_display_texture_view_width) / static_cast<float>(m_display_vram_width) :
|
||||||
1.0f;
|
1.0f;
|
||||||
const float upscale_y = m_display_texture ? static_cast<float>(m_display_texture_view_height) /
|
const float upscale_y =
|
||||||
static_cast<float>(m_display_vram_height) :
|
m_display_texture ? static_cast<float>(m_display_texture_view_height) / static_cast<float>(m_display_vram_height) :
|
||||||
1.0f;
|
1.0f;
|
||||||
const s32 orig_width = static_cast<s32>(std::ceil(static_cast<float>(m_display_width) * upscale_x));
|
const s32 orig_width = static_cast<s32>(std::ceil(static_cast<float>(m_display_width) * upscale_x));
|
||||||
const s32 orig_height = static_cast<s32>(std::ceil(static_cast<float>(m_display_height) * upscale_y));
|
const s32 orig_height = static_cast<s32>(std::ceil(static_cast<float>(m_display_height) * upscale_y));
|
||||||
|
|
||||||
return PostProcessing::DisplayChain.Apply(PostProcessing::DisplayChain.GetInputTexture(), nullptr, target,
|
return PostProcessing::DisplayChain.Apply(PostProcessing::DisplayChain.GetInputTexture(), nullptr, target,
|
||||||
display_rect, orig_width, orig_height, m_display_width, m_display_height);
|
display_rect, orig_width, orig_height, m_display_width, m_display_height);
|
||||||
}
|
}
|
||||||
else
|
|
||||||
{
|
void GPUPresenter::DrawTextureCopy(const GSVector2i target_size, const GSVector4i draw_rect, GPUTexture* input,
|
||||||
return GPUDevice::PresentResult::OK;
|
bool dst_alpha_blend, bool linear, WindowInfo::PreRotation prerotation)
|
||||||
}
|
{
|
||||||
|
GL_SCOPE_FMT("DrawTextureCopy({}, blend={}, linear={}, prerotation={})", draw_rect, dst_alpha_blend, draw_rect,
|
||||||
|
static_cast<u32>(prerotation));
|
||||||
|
|
||||||
|
g_gpu_device->SetPipeline(dst_alpha_blend ? m_present_copy_blend_pipeline.get() : m_present_copy_pipeline.get());
|
||||||
|
g_gpu_device->SetTextureSampler(0, input, g_gpu_device->GetNearestSampler());
|
||||||
|
|
||||||
|
DrawScreenQuad(draw_rect, GSVector4::cxpr(0.0f, 0.0f, 1.0f, 1.0f), target_size, DisplayRotation::Normal, prerotation);
|
||||||
}
|
}
|
||||||
|
|
||||||
void GPUPresenter::SendDisplayToMediaCapture(MediaCapture* cap)
|
void GPUPresenter::SendDisplayToMediaCapture(MediaCapture* cap)
|
||||||
|
@ -542,7 +718,7 @@ void GPUPresenter::SendDisplayToMediaCapture(MediaCapture* cap)
|
||||||
// Not cleared by RenderDisplay().
|
// Not cleared by RenderDisplay().
|
||||||
g_gpu_device->ClearRenderTarget(target, GPUDevice::DEFAULT_CLEAR_COLOR);
|
g_gpu_device->ClearRenderTarget(target, GPUDevice::DEFAULT_CLEAR_COLOR);
|
||||||
|
|
||||||
if (RenderDisplay(target, display_rect, draw_rect, postfx) != GPUDevice::PresentResult::OK ||
|
if (RenderDisplay(target, GSVector4i::zero(), display_rect, draw_rect, postfx) != GPUDevice::PresentResult::OK ||
|
||||||
!cap->DeliverVideoFrame(target)) [[unlikely]]
|
!cap->DeliverVideoFrame(target)) [[unlikely]]
|
||||||
{
|
{
|
||||||
WARNING_LOG("Failed to render/deliver video capture frame.");
|
WARNING_LOG("Failed to render/deliver video capture frame.");
|
||||||
|
@ -857,21 +1033,19 @@ void GPUPresenter::SleepUntilPresentTime(u64 present_time)
|
||||||
bool GPUPresenter::RenderScreenshotToBuffer(u32 width, u32 height, const GSVector4i display_rect,
|
bool GPUPresenter::RenderScreenshotToBuffer(u32 width, u32 height, const GSVector4i display_rect,
|
||||||
const GSVector4i draw_rect, bool postfx, Image* out_image)
|
const GSVector4i draw_rect, bool postfx, Image* out_image)
|
||||||
{
|
{
|
||||||
const GPUTexture::Format hdformat =
|
const ImageFormat image_format = GPUTexture::GetImageFormatForTextureFormat(m_present_format);
|
||||||
g_gpu_device->HasMainSwapChain() ? g_gpu_device->GetMainSwapChain()->GetFormat() : GPUTexture::Format::RGBA8;
|
|
||||||
const ImageFormat image_format = GPUTexture::GetImageFormatForTextureFormat(hdformat);
|
|
||||||
if (image_format == ImageFormat::None)
|
if (image_format == ImageFormat::None)
|
||||||
return false;
|
return false;
|
||||||
|
|
||||||
auto render_texture = g_gpu_device->FetchAutoRecycleTexture(width, height, 1, 1, 1, GPUTexture::Type::RenderTarget,
|
auto render_texture = g_gpu_device->FetchAutoRecycleTexture(width, height, 1, 1, 1, GPUTexture::Type::RenderTarget,
|
||||||
hdformat, GPUTexture::Flags::None);
|
m_present_format, GPUTexture::Flags::None);
|
||||||
if (!render_texture)
|
if (!render_texture)
|
||||||
return false;
|
return false;
|
||||||
|
|
||||||
g_gpu_device->ClearRenderTarget(render_texture.get(), GPUDevice::DEFAULT_CLEAR_COLOR);
|
g_gpu_device->ClearRenderTarget(render_texture.get(), GPUDevice::DEFAULT_CLEAR_COLOR);
|
||||||
|
|
||||||
// TODO: this should use copy shader instead.
|
// TODO: this should use copy shader instead.
|
||||||
RenderDisplay(render_texture.get(), display_rect, draw_rect, postfx);
|
RenderDisplay(render_texture.get(), GSVector4i::zero(), display_rect, draw_rect, postfx);
|
||||||
|
|
||||||
Image image(width, height, image_format);
|
Image image(width, height, image_format);
|
||||||
|
|
||||||
|
@ -879,12 +1053,12 @@ bool GPUPresenter::RenderScreenshotToBuffer(u32 width, u32 height, const GSVecto
|
||||||
std::unique_ptr<GPUDownloadTexture> dltex;
|
std::unique_ptr<GPUDownloadTexture> dltex;
|
||||||
if (g_gpu_device->GetFeatures().memory_import)
|
if (g_gpu_device->GetFeatures().memory_import)
|
||||||
{
|
{
|
||||||
dltex = g_gpu_device->CreateDownloadTexture(width, height, hdformat, image.GetPixels(), image.GetStorageSize(),
|
dltex = g_gpu_device->CreateDownloadTexture(width, height, m_present_format, image.GetPixels(),
|
||||||
image.GetPitch(), &error);
|
image.GetStorageSize(), image.GetPitch(), &error);
|
||||||
}
|
}
|
||||||
if (!dltex)
|
if (!dltex)
|
||||||
{
|
{
|
||||||
if (!(dltex = g_gpu_device->CreateDownloadTexture(width, height, hdformat, &error)))
|
if (!(dltex = g_gpu_device->CreateDownloadTexture(width, height, m_present_format, &error)))
|
||||||
{
|
{
|
||||||
ERROR_LOG("Failed to create {}x{} download texture: {}", width, height, error.GetDescription());
|
ERROR_LOG("Failed to create {}x{} download texture: {}", width, height, error.GetDescription());
|
||||||
return false;
|
return false;
|
||||||
|
@ -945,3 +1119,165 @@ void GPUPresenter::CalculateScreenshotSize(DisplayScreenshotMode mode, u32* widt
|
||||||
CalculateDrawRect(*width, *height, true, !g_settings.gpu_show_vram, display_rect, draw_rect);
|
CalculateDrawRect(*width, *height, true, !g_settings.gpu_show_vram, display_rect, draw_rect);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
bool GPUPresenter::LoadOverlaySettings()
|
||||||
|
{
|
||||||
|
std::string preset_name;
|
||||||
|
std::string image_path;
|
||||||
|
GSVector4i display_rect = m_border_overlay_display_rect;
|
||||||
|
bool alpha_blend = m_border_overlay_alpha_blend;
|
||||||
|
{
|
||||||
|
const auto lock = Host::GetSettingsLock();
|
||||||
|
const SettingsInterface* si = Host::GetSettingsInterface();
|
||||||
|
|
||||||
|
preset_name = si->GetStringValue("BorderOverlay", "PresetName");
|
||||||
|
if (preset_name == "Custom")
|
||||||
|
{
|
||||||
|
image_path = si->GetStringValue("BorderOverlay", "ImagePath");
|
||||||
|
display_rect = GSVector4i(
|
||||||
|
si->GetIntValue("BorderOverlay", "DisplayStartX", 0), si->GetIntValue("BorderOverlay", "DisplayStartY", 0),
|
||||||
|
si->GetIntValue("BorderOverlay", "DisplayEndX", 0), si->GetIntValue("BorderOverlay", "DisplayEndY", 0));
|
||||||
|
alpha_blend = si->GetBoolValue("BorderOverlay", "AlphaBlend", false);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// check rect validity.. ignore everything if it's bogus
|
||||||
|
if (!image_path.empty() && display_rect.rempty())
|
||||||
|
{
|
||||||
|
ERROR_LOG("Border overlay rectangle {} is invalid.", display_rect);
|
||||||
|
image_path = {};
|
||||||
|
}
|
||||||
|
if (image_path.empty())
|
||||||
|
{
|
||||||
|
// using preset?
|
||||||
|
if (!preset_name.empty())
|
||||||
|
{
|
||||||
|
// don't worry about the other settings, the loader will fix them up
|
||||||
|
if (m_border_overlay_image_path == preset_name)
|
||||||
|
return false;
|
||||||
|
|
||||||
|
image_path = std::move(preset_name);
|
||||||
|
}
|
||||||
|
|
||||||
|
display_rect = GSVector4i::zero();
|
||||||
|
alpha_blend = false;
|
||||||
|
}
|
||||||
|
|
||||||
|
// display rect can be updated without issue
|
||||||
|
m_border_overlay_display_rect = display_rect;
|
||||||
|
|
||||||
|
// but images and alphablend require pipeline/texture changes
|
||||||
|
if (m_border_overlay_image_path == image_path && (image_path.empty() || alpha_blend == m_border_overlay_alpha_blend))
|
||||||
|
{
|
||||||
|
m_border_overlay_alpha_blend = alpha_blend;
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
m_border_overlay_image_path = std::move(image_path);
|
||||||
|
m_border_overlay_alpha_blend = alpha_blend;
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
bool GPUPresenter::LoadOverlayTexture()
|
||||||
|
{
|
||||||
|
g_gpu_device->RecycleTexture(std::move(m_border_overlay_texture));
|
||||||
|
if (m_border_overlay_image_path.empty())
|
||||||
|
{
|
||||||
|
m_border_overlay_display_rect = GSVector4i::zero();
|
||||||
|
m_border_overlay_image_path = {};
|
||||||
|
m_border_overlay_alpha_blend = false;
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
Image image;
|
||||||
|
Error error;
|
||||||
|
|
||||||
|
bool image_load_result;
|
||||||
|
if (Path::IsAbsolute(m_border_overlay_image_path))
|
||||||
|
image_load_result = image.LoadFromFile(m_border_overlay_image_path.c_str(), &error);
|
||||||
|
else
|
||||||
|
image_load_result = LoadOverlayPreset(&error, &image);
|
||||||
|
if (!image_load_result ||
|
||||||
|
!(m_border_overlay_texture = g_gpu_device->FetchAndUploadTextureImage(image, GPUTexture::Flags::None, &error)))
|
||||||
|
{
|
||||||
|
ERROR_LOG("Failed to load overlay '{}': {}", Path::GetFileName(m_border_overlay_image_path),
|
||||||
|
error.GetDescription());
|
||||||
|
m_border_overlay_display_rect = GSVector4i::zero();
|
||||||
|
m_border_overlay_image_path = {};
|
||||||
|
m_border_overlay_alpha_blend = false;
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
INFO_LOG("Loaded overlay image {}: {}x{}", Path::GetFileName(m_border_overlay_image_path),
|
||||||
|
m_border_overlay_texture->GetWidth(), m_border_overlay_texture->GetHeight());
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
std::vector<std::string> GPUPresenter::EnumerateBorderOverlayPresets()
|
||||||
|
{
|
||||||
|
static constexpr const char* pattern = "*.yml";
|
||||||
|
|
||||||
|
std::vector<std::string> ret;
|
||||||
|
|
||||||
|
FileSystem::FindResultsArray files;
|
||||||
|
FileSystem::FindFiles(Path::Combine(EmuFolders::Resources, "overlays").c_str(), pattern,
|
||||||
|
FILESYSTEM_FIND_RELATIVE_PATHS | FILESYSTEM_FIND_FILES, &files);
|
||||||
|
FileSystem::FindFiles(Path::Combine(EmuFolders::UserResources, "overlays").c_str(), pattern,
|
||||||
|
FILESYSTEM_FIND_RELATIVE_PATHS | FILESYSTEM_FIND_FILES | FILESYSTEM_FIND_KEEP_ARRAY, &files);
|
||||||
|
|
||||||
|
ret.reserve(files.size());
|
||||||
|
for (FILESYSTEM_FIND_DATA& fd : files)
|
||||||
|
{
|
||||||
|
const std::string_view name = Path::GetFileTitle(fd.FileName);
|
||||||
|
if (StringUtil::IsInStringList(ret, name))
|
||||||
|
continue;
|
||||||
|
|
||||||
|
ret.emplace_back(name);
|
||||||
|
}
|
||||||
|
|
||||||
|
std::sort(ret.begin(), ret.end());
|
||||||
|
return ret;
|
||||||
|
}
|
||||||
|
|
||||||
|
bool GPUPresenter::LoadOverlayPreset(Error* error, Image* image)
|
||||||
|
{
|
||||||
|
SmallString path = SmallString::from_format("overlays/{}.yml", m_border_overlay_image_path);
|
||||||
|
std::optional<std::string> yaml_data = Host::ReadResourceFileToString(path, true, error);
|
||||||
|
if (!yaml_data.has_value())
|
||||||
|
return false;
|
||||||
|
|
||||||
|
// TODO: FIXME
|
||||||
|
ryml::reset_callbacks();
|
||||||
|
|
||||||
|
const ryml::Tree yaml =
|
||||||
|
ryml::parse_in_place(to_csubstr(path), c4::substr(reinterpret_cast<char*>(yaml_data->data()), yaml_data->size()));
|
||||||
|
const ryml::ConstNodeRef root = yaml.rootref();
|
||||||
|
if (root.empty())
|
||||||
|
{
|
||||||
|
Error::SetStringView(error, "Configuration is empty.");
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
std::string_view image_filename;
|
||||||
|
GSVector4i display_area = GSVector4i::zero();
|
||||||
|
bool display_alpha_blend = false;
|
||||||
|
if (!GetStringFromObject(root, "image", &image_filename) ||
|
||||||
|
!GetUIntFromObject(root, "displayStartX", &display_area.x) ||
|
||||||
|
!GetUIntFromObject(root, "displayStartY", &display_area.y) ||
|
||||||
|
!GetUIntFromObject(root, "displayEndX", &display_area.z) ||
|
||||||
|
!GetUIntFromObject(root, "displayEndY", &display_area.w) ||
|
||||||
|
!GetUIntFromObject(root, "alphaBlend", &display_alpha_blend))
|
||||||
|
{
|
||||||
|
Error::SetStringView(error, "One or more parameters is missing.");
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
path.format("overlays/{}", image_filename);
|
||||||
|
std::optional<DynamicHeapArray<u8>> image_data = Host::ReadResourceFile(path, true, error);
|
||||||
|
if (!image_data.has_value() || !image->LoadFromBuffer(image_filename, image_data.value(), error))
|
||||||
|
return false;
|
||||||
|
|
||||||
|
m_border_overlay_display_rect = display_area;
|
||||||
|
m_border_overlay_alpha_blend = display_alpha_blend;
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
|
@ -3,9 +3,13 @@
|
||||||
|
|
||||||
#pragma once
|
#pragma once
|
||||||
|
|
||||||
|
#include "types.h"
|
||||||
|
|
||||||
#include "util/gpu_device.h"
|
#include "util/gpu_device.h"
|
||||||
|
|
||||||
#include <memory>
|
#include <memory>
|
||||||
|
#include <string>
|
||||||
|
#include <vector>
|
||||||
|
|
||||||
class Error;
|
class Error;
|
||||||
class Image;
|
class Image;
|
||||||
|
@ -28,6 +32,9 @@ public:
|
||||||
/// Main frame presenter - used both when a game is and is not running.
|
/// Main frame presenter - used both when a game is and is not running.
|
||||||
static bool PresentFrame(GPUPresenter* presenter, GPUBackend* backend, bool allow_skip_present, u64 present_time);
|
static bool PresentFrame(GPUPresenter* presenter, GPUBackend* backend, bool allow_skip_present, u64 present_time);
|
||||||
|
|
||||||
|
/// Returns a list of border overlay presets.
|
||||||
|
static std::vector<std::string> EnumerateBorderOverlayPresets();
|
||||||
|
|
||||||
ALWAYS_INLINE s32 GetDisplayWidth() const { return m_display_width; }
|
ALWAYS_INLINE s32 GetDisplayWidth() const { return m_display_width; }
|
||||||
ALWAYS_INLINE s32 GetDisplayHeight() const { return m_display_height; }
|
ALWAYS_INLINE s32 GetDisplayHeight() const { return m_display_height; }
|
||||||
ALWAYS_INLINE s32 GetDisplayVRAMWidth() const { return m_display_vram_width; }
|
ALWAYS_INLINE s32 GetDisplayVRAMWidth() const { return m_display_vram_width; }
|
||||||
|
@ -44,6 +51,8 @@ public:
|
||||||
|
|
||||||
bool UpdateSettings(const GPUSettings& old_settings, Error* error);
|
bool UpdateSettings(const GPUSettings& old_settings, Error* error);
|
||||||
|
|
||||||
|
bool UpdatePostProcessingSettings(Error* error);
|
||||||
|
|
||||||
void ClearDisplay();
|
void ClearDisplay();
|
||||||
void ClearDisplayTexture();
|
void ClearDisplayTexture();
|
||||||
void SetDisplayParameters(u16 display_width, u16 display_height, u16 display_origin_left, u16 display_origin_top,
|
void SetDisplayParameters(u16 display_width, u16 display_height, u16 display_origin_left, u16 display_origin_top,
|
||||||
|
@ -82,12 +91,28 @@ private:
|
||||||
|
|
||||||
bool CompileDisplayPipelines(bool display, bool deinterlace, bool chroma_smoothing, Error* error);
|
bool CompileDisplayPipelines(bool display, bool deinterlace, bool chroma_smoothing, Error* error);
|
||||||
|
|
||||||
GPUDevice::PresentResult RenderDisplay(GPUTexture* target, const GSVector4i display_rect, const GSVector4i draw_rect,
|
GPUDevice::PresentResult RenderDisplay(GPUTexture* target, const GSVector4i overlay_rect,
|
||||||
bool postfx);
|
const GSVector4i display_rect, const GSVector4i draw_rect, bool postfx);
|
||||||
|
|
||||||
|
void DrawDisplay(const GSVector2i target_size, GPUTexture* display_texture, s32 display_texture_view_x,
|
||||||
|
s32 display_texture_view_y, s32 display_texture_view_width, s32 display_texture_view_height,
|
||||||
|
const GSVector4i display_rect, bool dst_alpha_blend, DisplayRotation rotation,
|
||||||
|
WindowInfo::PreRotation prerotation);
|
||||||
|
GPUDevice::PresentResult ApplyDisplayPostProcess(GPUTexture* target, GPUTexture* input,
|
||||||
|
const GSVector4i display_rect);
|
||||||
|
void DrawTextureCopy(const GSVector2i target_size, const GSVector4i draw_rect, GPUTexture* input,
|
||||||
|
bool dst_alpha_blend, bool linear, WindowInfo::PreRotation prerotation);
|
||||||
|
void DrawScreenQuad(const GSVector4i rect, const GSVector4 uv_rect, const GSVector2i target_size,
|
||||||
|
DisplayRotation uv_rotation, WindowInfo::PreRotation prerotation);
|
||||||
|
|
||||||
bool DeinterlaceSetTargetSize(u32 width, u32 height, bool preserve);
|
bool DeinterlaceSetTargetSize(u32 width, u32 height, bool preserve);
|
||||||
void DestroyDeinterlaceTextures();
|
void DestroyDeinterlaceTextures();
|
||||||
|
|
||||||
|
/// Returns true if the image path or alpha blend option has changed.
|
||||||
|
bool LoadOverlaySettings();
|
||||||
|
bool LoadOverlayTexture();
|
||||||
|
bool LoadOverlayPreset(Error* error, Image* image);
|
||||||
|
|
||||||
s32 m_display_width = 0;
|
s32 m_display_width = 0;
|
||||||
s32 m_display_height = 0;
|
s32 m_display_height = 0;
|
||||||
|
|
||||||
|
@ -114,6 +139,20 @@ private:
|
||||||
s32 m_display_texture_view_height = 0;
|
s32 m_display_texture_view_height = 0;
|
||||||
|
|
||||||
u32 m_skipped_present_count = 0;
|
u32 m_skipped_present_count = 0;
|
||||||
|
GPUTexture::Format m_present_format = GPUTexture::Format::Unknown;
|
||||||
|
|
||||||
|
std::unique_ptr<GPUPipeline> m_present_copy_pipeline;
|
||||||
|
|
||||||
|
// blended variants of pipelines, used when overlays are enabled
|
||||||
|
std::unique_ptr<GPUPipeline> m_display_blend_pipeline;
|
||||||
|
std::unique_ptr<GPUPipeline> m_present_copy_blend_pipeline;
|
||||||
|
|
||||||
|
std::unique_ptr<GPUTexture> m_border_overlay_texture;
|
||||||
|
GSVector4i m_border_overlay_display_rect = GSVector4i::zero();
|
||||||
|
|
||||||
|
// Low-traffic variables down here.
|
||||||
|
std::string m_border_overlay_image_path;
|
||||||
|
bool m_border_overlay_alpha_blend = false;
|
||||||
};
|
};
|
||||||
|
|
||||||
namespace Host {
|
namespace Host {
|
||||||
|
|
|
@ -973,7 +973,8 @@ void GPUThread::UpdateSettingsOnThread(const GPUSettings& old_settings)
|
||||||
PostProcessing::UpdateSettings();
|
PostProcessing::UpdateSettings();
|
||||||
|
|
||||||
Error error;
|
Error error;
|
||||||
if (!s_state.gpu_presenter->UpdateSettings(old_settings, &error) ||
|
if (!s_state.gpu_presenter->UpdatePostProcessingSettings(&error) ||
|
||||||
|
!s_state.gpu_presenter->UpdateSettings(old_settings, &error) ||
|
||||||
!s_state.gpu_backend->UpdateSettings(old_settings, &error)) [[unlikely]]
|
!s_state.gpu_backend->UpdateSettings(old_settings, &error)) [[unlikely]]
|
||||||
{
|
{
|
||||||
ReportFatalErrorAndShutdown(fmt::format("Failed to update settings: {}", error.GetDescription()));
|
ReportFatalErrorAndShutdown(fmt::format("Failed to update settings: {}", error.GetDescription()));
|
||||||
|
@ -1071,6 +1072,14 @@ void GPUThread::UpdateSettings(bool gpu_settings_changed, bool device_settings_c
|
||||||
if (s_state.gpu_backend)
|
if (s_state.gpu_backend)
|
||||||
{
|
{
|
||||||
PostProcessing::UpdateSettings();
|
PostProcessing::UpdateSettings();
|
||||||
|
|
||||||
|
Error error;
|
||||||
|
if (!s_state.gpu_presenter->UpdatePostProcessingSettings(&error))
|
||||||
|
{
|
||||||
|
ReportFatalErrorAndShutdown(fmt::format("Failed to update settings: {}", error.GetDescription()));
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
if (ImGuiManager::UpdateDebugWindowConfig() || (PostProcessing::DisplayChain.IsActive() && !IsSystemPaused()))
|
if (ImGuiManager::UpdateDebugWindowConfig() || (PostProcessing::DisplayChain.IsActive() && !IsSystemPaused()))
|
||||||
PresentFrameAndRestoreContext();
|
PresentFrameAndRestoreContext();
|
||||||
}
|
}
|
||||||
|
|
|
@ -205,6 +205,9 @@ struct GPUSettings
|
||||||
bool operator!=(const TextureReplacementSettings& rhs) const;
|
bool operator!=(const TextureReplacementSettings& rhs) const;
|
||||||
} texture_replacements;
|
} texture_replacements;
|
||||||
|
|
||||||
|
std::string overlay_image_path;
|
||||||
|
s16 mingus2[4];
|
||||||
|
|
||||||
float GetDisplayAspectRatioValue() const;
|
float GetDisplayAspectRatioValue() const;
|
||||||
|
|
||||||
ALWAYS_INLINE bool IsUsingSoftwareRenderer() const { return (gpu_renderer == GPURenderer::Software); }
|
ALWAYS_INLINE bool IsUsingSoftwareRenderer() const { return (gpu_renderer == GPURenderer::Software); }
|
||||||
|
@ -256,7 +259,7 @@ struct Settings : public GPUSettings
|
||||||
{
|
{
|
||||||
Settings();
|
Settings();
|
||||||
|
|
||||||
u32 cpu_overclock_numerator = 1;
|
ALIGN_TO_CACHE_LINE u32 cpu_overclock_numerator = 1;
|
||||||
u32 cpu_overclock_denominator = 1;
|
u32 cpu_overclock_denominator = 1;
|
||||||
|
|
||||||
TickCount dma_max_slice_ticks = DEFAULT_DMA_MAX_SLICE_TICKS;
|
TickCount dma_max_slice_ticks = DEFAULT_DMA_MAX_SLICE_TICKS;
|
||||||
|
|
|
@ -360,6 +360,9 @@
|
||||||
<QtUi Include="isobrowserwindow.ui">
|
<QtUi Include="isobrowserwindow.ui">
|
||||||
<FileType>Document</FileType>
|
<FileType>Document</FileType>
|
||||||
</QtUi>
|
</QtUi>
|
||||||
|
<QtUi Include="postprocessingoverlayconfigwidget.ui">
|
||||||
|
<FileType>Document</FileType>
|
||||||
|
</QtUi>
|
||||||
<None Include="translations\duckstation-qt_es-es.ts" />
|
<None Include="translations\duckstation-qt_es-es.ts" />
|
||||||
<None Include="translations\duckstation-qt_sv.ts" />
|
<None Include="translations\duckstation-qt_sv.ts" />
|
||||||
<None Include="translations\duckstation-qt_tr.ts" />
|
<None Include="translations\duckstation-qt_tr.ts" />
|
||||||
|
|
|
@ -295,6 +295,7 @@
|
||||||
<QtUi Include="gamecheatcodechoiceeditordialog.ui" />
|
<QtUi Include="gamecheatcodechoiceeditordialog.ui" />
|
||||||
<QtUi Include="memorycardrenamefiledialog.ui" />
|
<QtUi Include="memorycardrenamefiledialog.ui" />
|
||||||
<QtUi Include="isobrowserwindow.ui" />
|
<QtUi Include="isobrowserwindow.ui" />
|
||||||
|
<QtUi Include="postprocessingoverlayconfigwidget.ui" />
|
||||||
</ItemGroup>
|
</ItemGroup>
|
||||||
<ItemGroup>
|
<ItemGroup>
|
||||||
<ResourceCompile Include="duckstation-qt.rc" />
|
<ResourceCompile Include="duckstation-qt.rc" />
|
||||||
|
|
|
@ -5,13 +5,17 @@
|
||||||
#include "qthost.h"
|
#include "qthost.h"
|
||||||
#include "settingwidgetbinder.h"
|
#include "settingwidgetbinder.h"
|
||||||
|
|
||||||
|
#include "core/gpu_presenter.h"
|
||||||
|
|
||||||
#include "util/postprocessing.h"
|
#include "util/postprocessing.h"
|
||||||
|
|
||||||
#include "common/error.h"
|
#include "common/error.h"
|
||||||
|
|
||||||
|
#include <QtCore/QDir>
|
||||||
#include <QtWidgets/QCheckBox>
|
#include <QtWidgets/QCheckBox>
|
||||||
#include <QtWidgets/QDialogButtonBox>
|
#include <QtWidgets/QDialogButtonBox>
|
||||||
#include <QtWidgets/QGridLayout>
|
#include <QtWidgets/QGridLayout>
|
||||||
|
#include <QtWidgets/QInputDialog>
|
||||||
#include <QtWidgets/QLabel>
|
#include <QtWidgets/QLabel>
|
||||||
#include <QtWidgets/QMessageBox>
|
#include <QtWidgets/QMessageBox>
|
||||||
#include <QtWidgets/QSlider>
|
#include <QtWidgets/QSlider>
|
||||||
|
@ -22,6 +26,7 @@ PostProcessingSettingsWidget::PostProcessingSettingsWidget(SettingsWindow* dialo
|
||||||
tr("Display"));
|
tr("Display"));
|
||||||
addTab(new PostProcessingChainConfigWidget(dialog, this, PostProcessing::Config::INTERNAL_CHAIN_SECTION),
|
addTab(new PostProcessingChainConfigWidget(dialog, this, PostProcessing::Config::INTERNAL_CHAIN_SECTION),
|
||||||
tr("Internal"));
|
tr("Internal"));
|
||||||
|
addTab(new PostProcessingOverlayConfigWidget(dialog, this), tr("Border Overlay"));
|
||||||
setDocumentMode(true);
|
setDocumentMode(true);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -477,3 +482,53 @@ void PostProcessingShaderConfigWidget::createUi()
|
||||||
row++;
|
row++;
|
||||||
m_layout->addItem(new QSpacerItem(1, 1, QSizePolicy::Minimum, QSizePolicy::Expanding), row, 0, 1, 3);
|
m_layout->addItem(new QSpacerItem(1, 1, QSizePolicy::Minimum, QSizePolicy::Expanding), row, 0, 1, 3);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
PostProcessingOverlayConfigWidget::PostProcessingOverlayConfigWidget(SettingsWindow* dialog, QWidget* parent)
|
||||||
|
: QWidget(parent), m_dialog(dialog)
|
||||||
|
{
|
||||||
|
SettingsInterface* sif = dialog->getSettingsInterface();
|
||||||
|
|
||||||
|
m_ui.setupUi(this);
|
||||||
|
|
||||||
|
m_ui.overlayName->addItem(tr("None"), QString());
|
||||||
|
m_ui.overlayName->addItem(tr("Custom..."), QStringLiteral("Custom"));
|
||||||
|
for (const std::string& name : GPUPresenter::EnumerateBorderOverlayPresets())
|
||||||
|
{
|
||||||
|
const QString qname = QString::fromStdString(name);
|
||||||
|
m_ui.overlayName->addItem(qname, qname);
|
||||||
|
}
|
||||||
|
|
||||||
|
SettingWidgetBinder::BindWidgetToStringSetting(sif, m_ui.overlayName, "BorderOverlay", "PresetName");
|
||||||
|
SettingWidgetBinder::BindWidgetToStringSetting(sif, m_ui.imagePath, "BorderOverlay", "ImagePath");
|
||||||
|
SettingWidgetBinder::BindWidgetToIntSetting(sif, m_ui.displayStartX, "BorderOverlay", "DisplayStartX", 0);
|
||||||
|
SettingWidgetBinder::BindWidgetToIntSetting(sif, m_ui.displayStartY, "BorderOverlay", "DisplayStartY", 0);
|
||||||
|
SettingWidgetBinder::BindWidgetToIntSetting(sif, m_ui.displayEndX, "BorderOverlay", "DisplayEndX", 0);
|
||||||
|
SettingWidgetBinder::BindWidgetToIntSetting(sif, m_ui.displayEndY, "BorderOverlay", "DisplayEndY", 0);
|
||||||
|
SettingWidgetBinder::BindWidgetToBoolSetting(sif, m_ui.alphaBlend, "BorderOverlay", "AlphaBlend", false);
|
||||||
|
|
||||||
|
connect(m_ui.overlayName, &QComboBox::currentIndexChanged, this,
|
||||||
|
&PostProcessingOverlayConfigWidget::onOverlayNameCurrentIndexChanged);
|
||||||
|
connect(m_ui.imagePathBrowse, &QPushButton::clicked, this,
|
||||||
|
&PostProcessingOverlayConfigWidget::onImagePathBrowseClicked);
|
||||||
|
|
||||||
|
onOverlayNameCurrentIndexChanged(m_ui.overlayName->currentIndex());
|
||||||
|
}
|
||||||
|
|
||||||
|
PostProcessingOverlayConfigWidget::~PostProcessingOverlayConfigWidget() = default;
|
||||||
|
|
||||||
|
void PostProcessingOverlayConfigWidget::onOverlayNameCurrentIndexChanged(int index)
|
||||||
|
{
|
||||||
|
const int custom_idx = m_dialog->isPerGameSettings() ? 2 : 1;
|
||||||
|
const bool enable_custom = (index == custom_idx);
|
||||||
|
m_ui.customConfiguration->setEnabled(enable_custom);
|
||||||
|
}
|
||||||
|
|
||||||
|
void PostProcessingOverlayConfigWidget::onImagePathBrowseClicked()
|
||||||
|
{
|
||||||
|
const QString path = QFileDialog::getOpenFileName(QtUtils::GetRootWidget(this), tr("Select Image"), QString(),
|
||||||
|
tr("All Cover Image Types (*.jpg *.jpeg *.png *.webp)"));
|
||||||
|
if (path.isEmpty())
|
||||||
|
return;
|
||||||
|
|
||||||
|
m_ui.imagePath->setText(QDir::toNativeSeparators(path));
|
||||||
|
}
|
||||||
|
|
|
@ -1,9 +1,10 @@
|
||||||
// SPDX-FileCopyrightText: 2019-2024 Connor McLaughlin <stenzek@gmail.com>
|
// SPDX-FileCopyrightText: 2019-2025 Connor McLaughlin <stenzek@gmail.com>
|
||||||
// SPDX-License-Identifier: CC-BY-NC-ND-4.0
|
// SPDX-License-Identifier: CC-BY-NC-ND-4.0
|
||||||
|
|
||||||
#pragma once
|
#pragma once
|
||||||
|
|
||||||
#include "ui_postprocessingchainconfigwidget.h"
|
#include "ui_postprocessingchainconfigwidget.h"
|
||||||
|
#include "ui_postprocessingoverlayconfigwidget.h"
|
||||||
|
|
||||||
#include "util/postprocessing.h"
|
#include "util/postprocessing.h"
|
||||||
|
|
||||||
|
@ -13,7 +14,7 @@
|
||||||
class SettingsWindow;
|
class SettingsWindow;
|
||||||
class PostProcessingShaderConfigWidget;
|
class PostProcessingShaderConfigWidget;
|
||||||
|
|
||||||
class PostProcessingSettingsWidget : public QTabWidget
|
class PostProcessingSettingsWidget final : public QTabWidget
|
||||||
{
|
{
|
||||||
Q_OBJECT
|
Q_OBJECT
|
||||||
|
|
||||||
|
@ -61,7 +62,7 @@ private:
|
||||||
PostProcessingShaderConfigWidget* m_shader_config = nullptr;
|
PostProcessingShaderConfigWidget* m_shader_config = nullptr;
|
||||||
};
|
};
|
||||||
|
|
||||||
class PostProcessingShaderConfigWidget : public QWidget
|
class PostProcessingShaderConfigWidget final : public QWidget
|
||||||
{
|
{
|
||||||
Q_OBJECT
|
Q_OBJECT
|
||||||
|
|
||||||
|
@ -73,7 +74,7 @@ public:
|
||||||
private Q_SLOTS:
|
private Q_SLOTS:
|
||||||
void onResetDefaultsClicked();
|
void onResetDefaultsClicked();
|
||||||
|
|
||||||
protected:
|
private:
|
||||||
void createUi();
|
void createUi();
|
||||||
void updateConfigForOption(const PostProcessing::ShaderOption& option);
|
void updateConfigForOption(const PostProcessing::ShaderOption& option);
|
||||||
|
|
||||||
|
@ -86,3 +87,20 @@ protected:
|
||||||
u32 m_stage_index;
|
u32 m_stage_index;
|
||||||
std::vector<PostProcessing::ShaderOption> m_options;
|
std::vector<PostProcessing::ShaderOption> m_options;
|
||||||
};
|
};
|
||||||
|
|
||||||
|
class PostProcessingOverlayConfigWidget final : public QWidget
|
||||||
|
{
|
||||||
|
Q_OBJECT
|
||||||
|
|
||||||
|
public:
|
||||||
|
PostProcessingOverlayConfigWidget(SettingsWindow* dialog, QWidget* parent);
|
||||||
|
~PostProcessingOverlayConfigWidget();
|
||||||
|
|
||||||
|
private Q_SLOTS:
|
||||||
|
void onOverlayNameCurrentIndexChanged(int index);
|
||||||
|
void onImagePathBrowseClicked();
|
||||||
|
|
||||||
|
private:
|
||||||
|
Ui::PostProcessingOverlayConfigWidget m_ui;
|
||||||
|
SettingsWindow* m_dialog;
|
||||||
|
};
|
||||||
|
|
|
@ -233,10 +233,11 @@ GPUSwapChain::GPUSwapChain(const WindowInfo& wi, GPUVSyncMode vsync_mode, bool a
|
||||||
|
|
||||||
GPUSwapChain::~GPUSwapChain() = default;
|
GPUSwapChain::~GPUSwapChain() = default;
|
||||||
|
|
||||||
GSVector4i GPUSwapChain::PreRotateClipRect(const GSVector4i& v)
|
GSVector4i GPUSwapChain::PreRotateClipRect(WindowInfo::PreRotation prerotation, const GSVector2i surface_size,
|
||||||
|
const GSVector4i& v)
|
||||||
{
|
{
|
||||||
GSVector4i new_clip;
|
GSVector4i new_clip;
|
||||||
switch (m_window_info.surface_prerotation)
|
switch (prerotation)
|
||||||
{
|
{
|
||||||
case WindowInfo::PreRotation::Identity:
|
case WindowInfo::PreRotation::Identity:
|
||||||
new_clip = v;
|
new_clip = v;
|
||||||
|
@ -245,7 +246,7 @@ GSVector4i GPUSwapChain::PreRotateClipRect(const GSVector4i& v)
|
||||||
case WindowInfo::PreRotation::Rotate90Clockwise:
|
case WindowInfo::PreRotation::Rotate90Clockwise:
|
||||||
{
|
{
|
||||||
const s32 height = (v.w - v.y);
|
const s32 height = (v.w - v.y);
|
||||||
const s32 y = m_window_info.surface_height - v.y - height;
|
const s32 y = surface_size.y - v.y - height;
|
||||||
new_clip = GSVector4i(y, v.x, y + height, v.z);
|
new_clip = GSVector4i(y, v.x, y + height, v.z);
|
||||||
}
|
}
|
||||||
break;
|
break;
|
||||||
|
@ -254,8 +255,8 @@ GSVector4i GPUSwapChain::PreRotateClipRect(const GSVector4i& v)
|
||||||
{
|
{
|
||||||
const s32 width = (v.z - v.x);
|
const s32 width = (v.z - v.x);
|
||||||
const s32 height = (v.w - v.y);
|
const s32 height = (v.w - v.y);
|
||||||
const s32 x = m_window_info.surface_width - v.x - width;
|
const s32 x = surface_size.x - v.x - width;
|
||||||
const s32 y = m_window_info.surface_height - v.y - height;
|
const s32 y = surface_size.y - v.y - height;
|
||||||
new_clip = GSVector4i(x, y, x + width, y + height);
|
new_clip = GSVector4i(x, y, x + width, y + height);
|
||||||
}
|
}
|
||||||
break;
|
break;
|
||||||
|
@ -263,7 +264,7 @@ GSVector4i GPUSwapChain::PreRotateClipRect(const GSVector4i& v)
|
||||||
case WindowInfo::PreRotation::Rotate270Clockwise:
|
case WindowInfo::PreRotation::Rotate270Clockwise:
|
||||||
{
|
{
|
||||||
const s32 width = (v.z - v.x);
|
const s32 width = (v.z - v.x);
|
||||||
const s32 x = m_window_info.surface_width - v.x - width;
|
const s32 x = surface_size.x - v.x - width;
|
||||||
new_clip = GSVector4i(v.y, x, v.w, x + width);
|
new_clip = GSVector4i(v.y, x, v.w, x + width);
|
||||||
}
|
}
|
||||||
break;
|
break;
|
||||||
|
@ -736,12 +737,11 @@ void GPUDevice::RenderImGui(GPUSwapChain* swap_chain)
|
||||||
SetPipeline(m_imgui_pipeline.get());
|
SetPipeline(m_imgui_pipeline.get());
|
||||||
SetViewport(0, 0, swap_chain->GetPostRotatedWidth(), post_rotated_height);
|
SetViewport(0, 0, swap_chain->GetPostRotatedWidth(), post_rotated_height);
|
||||||
|
|
||||||
|
const bool prerotated = (swap_chain->GetPreRotation() != WindowInfo::PreRotation::Identity);
|
||||||
GSMatrix4x4 mproj = GSMatrix4x4::OffCenterOrthographicProjection(
|
GSMatrix4x4 mproj = GSMatrix4x4::OffCenterOrthographicProjection(
|
||||||
0.0f, 0.0f, static_cast<float>(swap_chain->GetWidth()), static_cast<float>(swap_chain->GetHeight()), 0.0f, 1.0f);
|
0.0f, 0.0f, static_cast<float>(swap_chain->GetWidth()), static_cast<float>(swap_chain->GetHeight()), 0.0f, 1.0f);
|
||||||
if (swap_chain->GetPreRotation() != WindowInfo::PreRotation::Identity)
|
if (prerotated)
|
||||||
{
|
|
||||||
mproj = GSMatrix4x4::RotationZ(WindowInfo::GetZRotationForPreRotation(swap_chain->GetPreRotation())) * mproj;
|
mproj = GSMatrix4x4::RotationZ(WindowInfo::GetZRotationForPreRotation(swap_chain->GetPreRotation())) * mproj;
|
||||||
}
|
|
||||||
PushUniformBuffer(&mproj, sizeof(mproj));
|
PushUniformBuffer(&mproj, sizeof(mproj));
|
||||||
|
|
||||||
// Render command lists
|
// Render command lists
|
||||||
|
@ -766,7 +766,9 @@ void GPUDevice::RenderImGui(GPUSwapChain* swap_chain)
|
||||||
}
|
}
|
||||||
|
|
||||||
GSVector4i clip = GSVector4i(GSVector4::load<false>(&pcmd->ClipRect.x));
|
GSVector4i clip = GSVector4i(GSVector4::load<false>(&pcmd->ClipRect.x));
|
||||||
clip = swap_chain->PreRotateClipRect(clip);
|
|
||||||
|
if (prerotated)
|
||||||
|
clip = GPUSwapChain::PreRotateClipRect(swap_chain->GetPreRotation(), swap_chain->GetSizeVec(), clip);
|
||||||
if (flip)
|
if (flip)
|
||||||
clip = FlipToLowerLeft(clip, post_rotated_height);
|
clip = FlipToLowerLeft(clip, post_rotated_height);
|
||||||
|
|
||||||
|
|
|
@ -516,6 +516,10 @@ public:
|
||||||
{
|
{
|
||||||
return GSVector2i(m_window_info.surface_width, m_window_info.surface_height);
|
return GSVector2i(m_window_info.surface_width, m_window_info.surface_height);
|
||||||
}
|
}
|
||||||
|
ALWAYS_INLINE GSVector2i GetPostRotatedSizeVec() const
|
||||||
|
{
|
||||||
|
return GSVector2i(m_window_info.GetPostRotatedWidth(), m_window_info.GetPostRotatedHeight());
|
||||||
|
}
|
||||||
|
|
||||||
ALWAYS_INLINE GPUVSyncMode GetVSyncMode() const { return m_vsync_mode; }
|
ALWAYS_INLINE GPUVSyncMode GetVSyncMode() const { return m_vsync_mode; }
|
||||||
ALWAYS_INLINE bool IsVSyncModeBlocking() const { return (m_vsync_mode == GPUVSyncMode::FIFO); }
|
ALWAYS_INLINE bool IsVSyncModeBlocking() const { return (m_vsync_mode == GPUVSyncMode::FIFO); }
|
||||||
|
@ -524,11 +528,12 @@ public:
|
||||||
virtual bool ResizeBuffers(u32 new_width, u32 new_height, float new_scale, Error* error) = 0;
|
virtual bool ResizeBuffers(u32 new_width, u32 new_height, float new_scale, Error* error) = 0;
|
||||||
virtual bool SetVSyncMode(GPUVSyncMode mode, bool allow_present_throttle, Error* error) = 0;
|
virtual bool SetVSyncMode(GPUVSyncMode mode, bool allow_present_throttle, Error* error) = 0;
|
||||||
|
|
||||||
GSVector4i PreRotateClipRect(const GSVector4i& v);
|
|
||||||
|
|
||||||
bool ShouldSkipPresentingFrame();
|
bool ShouldSkipPresentingFrame();
|
||||||
void ThrottlePresentation();
|
void ThrottlePresentation();
|
||||||
|
|
||||||
|
static GSVector4i PreRotateClipRect(WindowInfo::PreRotation prerotation, const GSVector2i surface_size,
|
||||||
|
const GSVector4i& v);
|
||||||
|
|
||||||
protected:
|
protected:
|
||||||
// TODO: Merge WindowInfo into this struct...
|
// TODO: Merge WindowInfo into this struct...
|
||||||
WindowInfo m_window_info;
|
WindowInfo m_window_info;
|
||||||
|
|
|
@ -577,45 +577,6 @@ bool PostProcessing::Chain::CheckTargets(GPUTexture::Format target_format, u32 t
|
||||||
|
|
||||||
Error error;
|
Error error;
|
||||||
|
|
||||||
if (!IsInternalChain() && (!m_rotated_copy_pipeline || m_target_format != target_format))
|
|
||||||
{
|
|
||||||
const RenderAPI rapi = g_gpu_device->GetRenderAPI();
|
|
||||||
const ShaderGen shadergen(rapi, ShaderGen::GetShaderLanguageForAPI(rapi), false, false);
|
|
||||||
const std::unique_ptr<GPUShader> vso = g_gpu_device->CreateShader(GPUShaderStage::Vertex, shadergen.GetLanguage(),
|
|
||||||
shadergen.GenerateRotateVertexShader(), &error);
|
|
||||||
const std::unique_ptr<GPUShader> fso = g_gpu_device->CreateShader(GPUShaderStage::Fragment, shadergen.GetLanguage(),
|
|
||||||
shadergen.GenerateRotateFragmentShader(), &error);
|
|
||||||
if (!vso || !fso)
|
|
||||||
{
|
|
||||||
ERROR_LOG("Failed to compile post-processing rotate shaders: {}", error.GetDescription());
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
GL_OBJECT_NAME(vso, "Post-processing rotate blit VS");
|
|
||||||
GL_OBJECT_NAME(vso, "Post-processing rotate blit FS");
|
|
||||||
|
|
||||||
const GPUPipeline::GraphicsConfig config = {.layout = GPUPipeline::Layout::SingleTextureAndPushConstants,
|
|
||||||
.primitive = GPUPipeline::Primitive::Triangles,
|
|
||||||
.input_layout = {},
|
|
||||||
.rasterization = GPUPipeline::RasterizationState::GetNoCullState(),
|
|
||||||
.depth = GPUPipeline::DepthState::GetNoTestsState(),
|
|
||||||
.blend = GPUPipeline::BlendState::GetNoBlendingState(),
|
|
||||||
.vertex_shader = vso.get(),
|
|
||||||
.geometry_shader = nullptr,
|
|
||||||
.fragment_shader = fso.get(),
|
|
||||||
.color_formats = {target_format},
|
|
||||||
.depth_format = GPUTexture::Format::Unknown,
|
|
||||||
.samples = 1,
|
|
||||||
.per_sample_shading = false,
|
|
||||||
.render_pass_flags = GPUPipeline::NoRenderPassFlags};
|
|
||||||
m_rotated_copy_pipeline = g_gpu_device->CreatePipeline(config, &error);
|
|
||||||
if (!m_rotated_copy_pipeline)
|
|
||||||
{
|
|
||||||
ERROR_LOG("Failed to compile post-processing rotate pipeline: {}", error.GetDescription());
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
GL_OBJECT_NAME(m_rotated_copy_pipeline, "Post-processing rotate pipeline");
|
|
||||||
}
|
|
||||||
|
|
||||||
// In case any allocs fail.
|
// In case any allocs fail.
|
||||||
DestroyTextures();
|
DestroyTextures();
|
||||||
|
|
||||||
|
@ -677,11 +638,6 @@ void PostProcessing::Chain::DestroyTextures()
|
||||||
g_gpu_device->RecycleTexture(std::move(m_input_texture));
|
g_gpu_device->RecycleTexture(std::move(m_input_texture));
|
||||||
}
|
}
|
||||||
|
|
||||||
void PostProcessing::Chain::DestroyPipelines()
|
|
||||||
{
|
|
||||||
m_rotated_copy_pipeline.reset();
|
|
||||||
}
|
|
||||||
|
|
||||||
GPUDevice::PresentResult PostProcessing::Chain::Apply(GPUTexture* input_color, GPUTexture* input_depth,
|
GPUDevice::PresentResult PostProcessing::Chain::Apply(GPUTexture* input_color, GPUTexture* input_depth,
|
||||||
GPUTexture* final_target, GSVector4i final_rect, s32 orig_width,
|
GPUTexture* final_target, GSVector4i final_rect, s32 orig_width,
|
||||||
s32 orig_height, s32 native_width, s32 native_height)
|
s32 orig_height, s32 native_width, s32 native_height)
|
||||||
|
@ -693,25 +649,14 @@ GPUDevice::PresentResult PostProcessing::Chain::Apply(GPUTexture* input_color, G
|
||||||
if (input_depth)
|
if (input_depth)
|
||||||
input_depth->MakeReadyForSampling();
|
input_depth->MakeReadyForSampling();
|
||||||
|
|
||||||
GPUTexture* draw_final_target = final_target;
|
|
||||||
const WindowInfo::PreRotation prerotation =
|
|
||||||
final_target ? WindowInfo::PreRotation::Identity : g_gpu_device->GetMainSwapChain()->GetPreRotation();
|
|
||||||
if (prerotation != WindowInfo::PreRotation::Identity)
|
|
||||||
{
|
|
||||||
// We have prerotation and post processing. This is messy, since we need to run the shader on the "real" size,
|
|
||||||
// then copy it across to the rotated image. We can use the input or output texture from the chain, whichever
|
|
||||||
// was not the last that was drawn to.
|
|
||||||
draw_final_target = GetTextureUnusedAtEndOfChain();
|
|
||||||
}
|
|
||||||
|
|
||||||
const float time = static_cast<float>(Timer::ConvertValueToSeconds(Timer::GetCurrentValue() - s_start_time));
|
const float time = static_cast<float>(Timer::ConvertValueToSeconds(Timer::GetCurrentValue() - s_start_time));
|
||||||
for (const std::unique_ptr<Shader>& stage : m_stages)
|
for (const std::unique_ptr<Shader>& stage : m_stages)
|
||||||
{
|
{
|
||||||
const bool is_final = (stage.get() == m_stages.back().get());
|
const bool is_final = (stage.get() == m_stages.back().get());
|
||||||
|
|
||||||
if (const GPUDevice::PresentResult pres =
|
if (const GPUDevice::PresentResult pres =
|
||||||
stage->Apply(input_color, input_depth, is_final ? draw_final_target : output, final_rect, orig_width,
|
stage->Apply(input_color, input_depth, is_final ? final_target : output, final_rect, orig_width, orig_height,
|
||||||
orig_height, native_width, native_height, m_target_width, m_target_height, time);
|
native_width, native_height, m_target_width, m_target_height, time);
|
||||||
pres != GPUDevice::PresentResult::OK)
|
pres != GPUDevice::PresentResult::OK)
|
||||||
{
|
{
|
||||||
return pres;
|
return pres;
|
||||||
|
@ -725,30 +670,6 @@ GPUDevice::PresentResult PostProcessing::Chain::Apply(GPUTexture* input_color, G
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
if (prerotation != WindowInfo::PreRotation::Identity)
|
|
||||||
{
|
|
||||||
draw_final_target->MakeReadyForSampling();
|
|
||||||
|
|
||||||
// Rotate and blit to final swap chain.
|
|
||||||
GPUSwapChain* const swap_chain = g_gpu_device->GetMainSwapChain();
|
|
||||||
if (const GPUDevice::PresentResult pres = g_gpu_device->BeginPresent(swap_chain);
|
|
||||||
pres != GPUDevice::PresentResult::OK)
|
|
||||||
{
|
|
||||||
return pres;
|
|
||||||
}
|
|
||||||
|
|
||||||
GL_PUSH_FMT("Apply swap chain pre-rotation");
|
|
||||||
|
|
||||||
const GSMatrix2x2 rotmat = GSMatrix2x2::Rotation(WindowInfo::GetZRotationForPreRotation(prerotation));
|
|
||||||
g_gpu_device->SetPipeline(m_rotated_copy_pipeline.get());
|
|
||||||
g_gpu_device->PushUniformBuffer(&rotmat, sizeof(rotmat));
|
|
||||||
g_gpu_device->SetTextureSampler(0, draw_final_target, g_gpu_device->GetNearestSampler());
|
|
||||||
g_gpu_device->SetViewportAndScissor(0, 0, swap_chain->GetPostRotatedWidth(), swap_chain->GetPostRotatedHeight());
|
|
||||||
g_gpu_device->Draw(3, 0);
|
|
||||||
|
|
||||||
GL_POP();
|
|
||||||
}
|
|
||||||
|
|
||||||
return GPUDevice::PresentResult::OK;
|
return GPUDevice::PresentResult::OK;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -771,7 +692,6 @@ void PostProcessing::Shutdown()
|
||||||
s_samplers.clear();
|
s_samplers.clear();
|
||||||
ForAllChains([](Chain& chain) {
|
ForAllChains([](Chain& chain) {
|
||||||
chain.ClearStages();
|
chain.ClearStages();
|
||||||
chain.DestroyPipelines();
|
|
||||||
chain.DestroyTextures();
|
chain.DestroyTextures();
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
@ -788,7 +708,6 @@ bool PostProcessing::ReloadShaders()
|
||||||
|
|
||||||
ForAllChains([](Chain& chain) {
|
ForAllChains([](Chain& chain) {
|
||||||
chain.ClearStages();
|
chain.ClearStages();
|
||||||
chain.DestroyPipelines();
|
|
||||||
chain.DestroyTextures();
|
chain.DestroyTextures();
|
||||||
chain.LoadStages();
|
chain.LoadStages();
|
||||||
});
|
});
|
||||||
|
|
|
@ -107,7 +107,7 @@ void UnsetStageOption(SettingsInterface& si, const char* section, u32 index, con
|
||||||
void ClearStages(SettingsInterface& si, const char* section);
|
void ClearStages(SettingsInterface& si, const char* section);
|
||||||
} // namespace Config
|
} // namespace Config
|
||||||
|
|
||||||
class Chain
|
class Chain final
|
||||||
{
|
{
|
||||||
public:
|
public:
|
||||||
Chain(const char* section);
|
Chain(const char* section);
|
||||||
|
@ -129,7 +129,6 @@ public:
|
||||||
void LoadStages();
|
void LoadStages();
|
||||||
void ClearStages();
|
void ClearStages();
|
||||||
void DestroyTextures();
|
void DestroyTextures();
|
||||||
void DestroyPipelines();
|
|
||||||
|
|
||||||
/// Temporarily toggles post-processing on/off.
|
/// Temporarily toggles post-processing on/off.
|
||||||
void Toggle();
|
void Toggle();
|
||||||
|
@ -156,7 +155,6 @@ private:
|
||||||
std::vector<std::unique_ptr<PostProcessing::Shader>> m_stages;
|
std::vector<std::unique_ptr<PostProcessing::Shader>> m_stages;
|
||||||
std::unique_ptr<GPUTexture> m_input_texture;
|
std::unique_ptr<GPUTexture> m_input_texture;
|
||||||
std::unique_ptr<GPUTexture> m_output_texture;
|
std::unique_ptr<GPUTexture> m_output_texture;
|
||||||
std::unique_ptr<GPUPipeline> m_rotated_copy_pipeline;
|
|
||||||
};
|
};
|
||||||
|
|
||||||
// [display_name, filename]
|
// [display_name, filename]
|
||||||
|
|
|
@ -793,40 +793,6 @@ void ShaderGen::DeclareFragmentEntryPoint(
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
std::string ShaderGen::GenerateRotateVertexShader() const
|
|
||||||
{
|
|
||||||
std::stringstream ss;
|
|
||||||
WriteHeader(ss);
|
|
||||||
DeclareUniformBuffer(ss, { "float2 u_rotation_matrix0", "float2 u_rotation_matrix1" }, true);
|
|
||||||
DeclareVertexEntryPoint(ss, {}, 0, 1, {}, true);
|
|
||||||
ss << "{\n";
|
|
||||||
ss << " v_tex0 = float2(float((v_id << 1) & 2u), float(v_id & 2u));\n";
|
|
||||||
ss << " v_pos = float4(v_tex0 * float2(2.0f, -2.0f) + float2(-1.0f, 1.0f), 0.0f, 1.0f);\n";
|
|
||||||
ss << " v_pos.xy = float2(dot(u_rotation_matrix0, v_pos.xy), dot(u_rotation_matrix1, v_pos.xy));\n";
|
|
||||||
ss << " #if API_OPENGL || API_OPENGL_ES || API_VULKAN\n";
|
|
||||||
ss << " v_pos.y = -v_pos.y;\n";
|
|
||||||
ss << " #endif\n";
|
|
||||||
ss << "}\n";
|
|
||||||
|
|
||||||
return ss.str();
|
|
||||||
}
|
|
||||||
|
|
||||||
std::string ShaderGen::GenerateRotateFragmentShader() const
|
|
||||||
{
|
|
||||||
std::stringstream ss;
|
|
||||||
WriteHeader(ss);
|
|
||||||
DeclareTexture(ss, "samp0", 0);
|
|
||||||
DeclareFragmentEntryPoint(ss, 0, 1);
|
|
||||||
|
|
||||||
ss << R"(
|
|
||||||
{
|
|
||||||
o_col0 = SAMPLE_TEXTURE(samp0, v_tex0);
|
|
||||||
}
|
|
||||||
)";
|
|
||||||
|
|
||||||
return ss.str();
|
|
||||||
}
|
|
||||||
|
|
||||||
std::string ShaderGen::GenerateScreenQuadVertexShader(float z /* = 0.0f */) const
|
std::string ShaderGen::GenerateScreenQuadVertexShader(float z /* = 0.0f */) const
|
||||||
{
|
{
|
||||||
std::stringstream ss;
|
std::stringstream ss;
|
||||||
|
@ -879,20 +845,33 @@ std::string ShaderGen::GenerateFillFragmentShader() const
|
||||||
return ss.str();
|
return ss.str();
|
||||||
}
|
}
|
||||||
|
|
||||||
std::string ShaderGen::GenerateCopyFragmentShader() const
|
std::string ShaderGen::GenerateCopyFragmentShader(bool offset) const
|
||||||
{
|
{
|
||||||
std::stringstream ss;
|
std::stringstream ss;
|
||||||
WriteHeader(ss);
|
WriteHeader(ss);
|
||||||
|
if (offset)
|
||||||
DeclareUniformBuffer(ss, {"float4 u_src_rect"}, true);
|
DeclareUniformBuffer(ss, {"float4 u_src_rect"}, true);
|
||||||
|
|
||||||
DeclareTexture(ss, "samp0", 0);
|
DeclareTexture(ss, "samp0", 0);
|
||||||
DeclareFragmentEntryPoint(ss, 0, 1);
|
DeclareFragmentEntryPoint(ss, 0, 1);
|
||||||
|
|
||||||
|
if (offset)
|
||||||
|
{
|
||||||
ss << R"(
|
ss << R"(
|
||||||
{
|
{
|
||||||
float2 coords = u_src_rect.xy + v_tex0 * u_src_rect.zw;
|
float2 coords = u_src_rect.xy + v_tex0 * u_src_rect.zw;
|
||||||
o_col0 = SAMPLE_TEXTURE(samp0, coords);
|
o_col0 = SAMPLE_TEXTURE(samp0, coords);
|
||||||
}
|
}
|
||||||
)";
|
)";
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
ss << R"(
|
||||||
|
{
|
||||||
|
o_col0 = SAMPLE_TEXTURE(samp0, v_tex0);
|
||||||
|
}
|
||||||
|
)";
|
||||||
|
}
|
||||||
|
|
||||||
return ss.str();
|
return ss.str();
|
||||||
}
|
}
|
||||||
|
|
|
@ -27,13 +27,10 @@ public:
|
||||||
ALWAYS_INLINE bool IsVulkan() const { return (m_render_api == RenderAPI::Vulkan); }
|
ALWAYS_INLINE bool IsVulkan() const { return (m_render_api == RenderAPI::Vulkan); }
|
||||||
ALWAYS_INLINE bool IsMetal() const { return (m_render_api == RenderAPI::Metal); }
|
ALWAYS_INLINE bool IsMetal() const { return (m_render_api == RenderAPI::Metal); }
|
||||||
|
|
||||||
std::string GenerateRotateVertexShader() const;
|
|
||||||
std::string GenerateRotateFragmentShader() const;
|
|
||||||
|
|
||||||
std::string GenerateScreenQuadVertexShader(float z = 0.0f) const;
|
std::string GenerateScreenQuadVertexShader(float z = 0.0f) const;
|
||||||
std::string GenerateUVQuadVertexShader() const;
|
std::string GenerateUVQuadVertexShader() const;
|
||||||
std::string GenerateFillFragmentShader() const;
|
std::string GenerateFillFragmentShader() const;
|
||||||
std::string GenerateCopyFragmentShader() const;
|
std::string GenerateCopyFragmentShader(bool offset = true) const;
|
||||||
|
|
||||||
std::string GenerateImGuiVertexShader() const;
|
std::string GenerateImGuiVertexShader() const;
|
||||||
std::string GenerateImGuiFragmentShader() const;
|
std::string GenerateImGuiFragmentShader() const;
|
||||||
|
|
Loading…
Reference in New Issue