Files
UnrealEngineUWP/Engine/Source/Runtime/Renderer/Private/PostProcess/PostProcessLocalExposure.cpp
guillaume abadie 59a7373ea7 Fixes a D3D11 validation failure binding a Texture2DArray into a RWTexture2D with TSR, sg.AntiAliasingQuality=0, sg.MotionBlurQuality=0
TSR's UpdateHistory now always render half or quarter res in a Texture2DArray of size 1, to not introduce shader permutation or dynamic branch on SceneColorOutputMip1 that could create dependent image_store on some hardware.

#jira UE-179496

[CL 30605439 by guillaume abadie in ue5-main branch]
2024-01-12 17:20:16 -05:00

254 lines
11 KiB
C++

// Copyright Epic Games, Inc. All Rights Reserved.
/*=============================================================================
PostProcessLocalExposure.cpp: Post processing local exposure implementation.
=============================================================================*/
#include "PostProcess/PostProcessLocalExposure.h"
#include "PostProcess/PostProcessEyeAdaptation.h"
#include "PostProcess/PostProcessWeightedSampleSum.h"
#include "Curves/CurveFloat.h"
#include "SceneRendering.h"
#include "ShaderCompilerCore.h"
#include "DataDrivenShaderPlatformInfo.h"
namespace
{
class FSetupLogLuminanceCS : public FGlobalShader
{
public:
// Changing these numbers requires LocalExposure.usf to be recompiled
static const uint32 ThreadGroupSizeX = 8;
static const uint32 ThreadGroupSizeY = 8;
DECLARE_GLOBAL_SHADER(FSetupLogLuminanceCS);
SHADER_USE_PARAMETER_STRUCT(FSetupLogLuminanceCS, FGlobalShader);
BEGIN_SHADER_PARAMETER_STRUCT(FParameters, )
SHADER_PARAMETER_STRUCT_REF(FViewUniformShaderParameters, View)
SHADER_PARAMETER_STRUCT(FEyeAdaptationParameters, EyeAdaptation)
SHADER_PARAMETER_STRUCT(FScreenPassTextureViewportParameters, Input)
SHADER_PARAMETER_RDG_TEXTURE_SRV(Texture2D, InputTexture)
SHADER_PARAMETER_RDG_TEXTURE_UAV(RWTexture2D<float>, OutputFloat)
END_SHADER_PARAMETER_STRUCT()
static bool ShouldCompilePermutation(const FGlobalShaderPermutationParameters& Parameters)
{
return IsFeatureLevelSupported(Parameters.Platform, ERHIFeatureLevel::SM5);
}
static void ModifyCompilationEnvironment(const FGlobalShaderPermutationParameters& Parameters, FShaderCompilerEnvironment& OutEnvironment)
{
FGlobalShader::ModifyCompilationEnvironment(Parameters, OutEnvironment);
OutEnvironment.SetDefine(TEXT("THREADGROUP_SIZEX"), ThreadGroupSizeX);
OutEnvironment.SetDefine(TEXT("THREADGROUP_SIZEY"), ThreadGroupSizeY);
}
};
IMPLEMENT_GLOBAL_SHADER(FSetupLogLuminanceCS, "/Engine/Private/PostProcessLocalExposure.usf", "SetupLogLuminanceCS", SF_Compute);
class FApplyLocalExposureCS : public FGlobalShader
{
public:
// Changing these numbers requires LocalExposure.usf to be recompiled
static const uint32 ThreadGroupSizeX = 8;
static const uint32 ThreadGroupSizeY = 8;
DECLARE_GLOBAL_SHADER(FApplyLocalExposureCS);
SHADER_USE_PARAMETER_STRUCT(FApplyLocalExposureCS, FGlobalShader);
BEGIN_SHADER_PARAMETER_STRUCT(FParameters, )
SHADER_PARAMETER_STRUCT_REF(FViewUniformShaderParameters, View)
SHADER_PARAMETER_STRUCT(FScreenPassTextureViewportParameters, Input)
SHADER_PARAMETER_STRUCT(FScreenPassTextureViewportParameters, Output)
SHADER_PARAMETER_RDG_TEXTURE_SRV(Texture2D, InputTexture)
SHADER_PARAMETER_RDG_TEXTURE_UAV(RWTexture2D<float4>, OutputFloat4)
SHADER_PARAMETER_STRUCT(FEyeAdaptationParameters, EyeAdaptation)
SHADER_PARAMETER_RDG_BUFFER_SRV(StructuredBuffer<float4>, EyeAdaptationBuffer)
SHADER_PARAMETER_STRUCT(FLocalExposureParameters, LocalExposure)
SHADER_PARAMETER_RDG_TEXTURE(Texture3D, LumBilateralGrid)
SHADER_PARAMETER_RDG_TEXTURE(Texture2D, BlurredLogLum)
SHADER_PARAMETER_SAMPLER(SamplerState, TextureSampler)
END_SHADER_PARAMETER_STRUCT()
static bool ShouldCompilePermutation(const FGlobalShaderPermutationParameters& Parameters)
{
return IsFeatureLevelSupported(Parameters.Platform, ERHIFeatureLevel::SM5);
}
static void ModifyCompilationEnvironment(const FGlobalShaderPermutationParameters& Parameters, FShaderCompilerEnvironment& OutEnvironment)
{
FGlobalShader::ModifyCompilationEnvironment(Parameters, OutEnvironment);
OutEnvironment.SetDefine(TEXT("THREADGROUP_SIZEX"), ThreadGroupSizeX);
OutEnvironment.SetDefine(TEXT("THREADGROUP_SIZEY"), ThreadGroupSizeY);
}
};
IMPLEMENT_GLOBAL_SHADER(FApplyLocalExposureCS, "/Engine/Private/PostProcessLocalExposure.usf", "ApplyLocalExposureCS", SF_Compute);
} //! namespace
FVector2f GetLocalExposureBilateralGridUVScale(const FIntPoint ViewRectSize);
FLocalExposureParameters GetLocalExposureParameters(const FViewInfo& View, FIntPoint ViewRectSize, const FEyeAdaptationParameters& EyeAdaptationParameters)
{
const FPostProcessSettings& Settings = View.FinalPostProcessSettings;
const EAutoExposureMethod AutoExposureMethod = GetAutoExposureMethod(View);
float LocalExposureMiddleGreyExposureCompensation = FMath::Pow(2.0f, View.FinalPostProcessSettings.LocalExposureMiddleGreyBias);
if (AutoExposureMethod == EAutoExposureMethod::AEM_Manual)
{
// when using manual exposure cancel exposure compensation setting and curve from middle grey used by local exposure.
LocalExposureMiddleGreyExposureCompensation /= (EyeAdaptationParameters.ExposureCompensationSettings * EyeAdaptationParameters.ExposureCompensationCurve);
}
const FVector2f LocalExposureBilateralGridUVScale = GetLocalExposureBilateralGridUVScale(ViewRectSize);
float HighlightContrast = Settings.LocalExposureHighlightContrastScale;
float ShadowContrast = Settings.LocalExposureShadowContrastScale;
const float AverageSceneLuminance = View.GetLastAverageSceneLuminance();
if (AverageSceneLuminance > 0)
{
const float LuminanceMax = LuminanceMaxFromLensAttenuation();
// We need the Log2(0.18) to convert from average luminance to saturation luminance
const float LuminanceEV100 = LuminanceToEV100(LuminanceMax, AverageSceneLuminance) + FMath::Log2(1.0f / 0.18f);
if (Settings.LocalExposureHighlightContrastCurve)
{
HighlightContrast *= Settings.LocalExposureHighlightContrastCurve->GetFloatValue(LuminanceEV100);
}
if (Settings.LocalExposureShadowContrastCurve)
{
ShadowContrast *= Settings.LocalExposureShadowContrastCurve->GetFloatValue(LuminanceEV100);
}
}
FLocalExposureParameters Parameters;
Parameters.HighlightContrastScale = HighlightContrast;
Parameters.ShadowContrastScale = ShadowContrast;
Parameters.DetailStrength = Settings.LocalExposureDetailStrength;
Parameters.BlurredLuminanceBlend = Settings.LocalExposureBlurredLuminanceBlend;
Parameters.MiddleGreyExposureCompensation = LocalExposureMiddleGreyExposureCompensation;
Parameters.BilateralGridUVScale = LocalExposureBilateralGridUVScale;
Parameters.HighlightThreshold = Settings.LocalExposureHighlightThreshold;
Parameters.ShadowThreshold = Settings.LocalExposureShadowThreshold;
return Parameters;
}
FRDGTextureRef AddLocalExposureBlurredLogLuminancePass(
FRDGBuilder& GraphBuilder,
const FViewInfo& View,
const FEyeAdaptationParameters& EyeAdaptationParameters,
FScreenPassTextureSlice InputTexture)
{
check(InputTexture.IsValid());
RDG_EVENT_SCOPE(GraphBuilder, "LocalExposure - Blurred Luminance");
FRDGTextureRef GaussianLumSetupTexture;
// Copy log luminance to temporary texture
{
const FRDGTextureDesc TextureDesc = FRDGTextureDesc::Create2D(
InputTexture.ViewRect.Size(),
PF_R16F,
FClearValueBinding::None,
TexCreate_UAV | TexCreate_ShaderResource);
GaussianLumSetupTexture = GraphBuilder.CreateTexture(TextureDesc, TEXT("GaussianLumSetupTexture"));
auto* PassParameters = GraphBuilder.AllocParameters<FSetupLogLuminanceCS::FParameters>();
PassParameters->View = View.ViewUniformBuffer;
PassParameters->EyeAdaptation = EyeAdaptationParameters;
PassParameters->Input = GetScreenPassTextureViewportParameters(FScreenPassTextureViewport(InputTexture));
PassParameters->InputTexture = InputTexture.TextureSRV;
PassParameters->OutputFloat = GraphBuilder.CreateUAV(GaussianLumSetupTexture);
FComputeShaderUtils::AddPass(
GraphBuilder,
RDG_EVENT_NAME("SetupLogLuminance %dx%d", GaussianLumSetupTexture->Desc.Extent.X, GaussianLumSetupTexture->Desc.Extent.Y),
ERDGPassFlags::Compute,
View.ShaderMap->GetShader<FSetupLogLuminanceCS>(),
PassParameters,
FComputeShaderUtils::GetGroupCount(GaussianLumSetupTexture->Desc.Extent, FIntPoint(FSetupLogLuminanceCS::ThreadGroupSizeX, FSetupLogLuminanceCS::ThreadGroupSizeY)));
}
FRDGTextureRef GaussianTexture;
{
FGaussianBlurInputs GaussianBlurInputs;
GaussianBlurInputs.NameX = TEXT("LocalExposureGaussianX");
GaussianBlurInputs.NameY = TEXT("LocalExposureGaussianY");
GaussianBlurInputs.Filter = FScreenPassTextureSlice::CreateFromScreenPassTexture(GraphBuilder, FScreenPassTexture(GaussianLumSetupTexture));
GaussianBlurInputs.TintColor = FLinearColor::White;
GaussianBlurInputs.CrossCenterWeight = FVector2f::ZeroVector;
GaussianBlurInputs.KernelSizePercent = View.FinalPostProcessSettings.LocalExposureBlurredLuminanceKernelSizePercent;
GaussianBlurInputs.UseMirrorAddressMode = true;
GaussianTexture = AddGaussianBlurPass(GraphBuilder, View, GaussianBlurInputs).Texture;
}
return GaussianTexture;
}
void AddApplyLocalExposurePass(
FRDGBuilder& GraphBuilder,
const FViewInfo& View,
const FEyeAdaptationParameters& EyeAdaptationParameters,
FRDGBufferRef EyeAdaptationBuffer,
const FLocalExposureParameters& LocalExposureParamaters,
FRDGTextureRef LocalExposureTexture,
FRDGTextureRef BlurredLogLuminanceTexture,
FScreenPassTextureSlice Input,
FScreenPassTextureSlice Output,
ERDGPassFlags PassFlags)
{
check(Input.IsValid() && Output.IsValid());
check(PassFlags == ERDGPassFlags::Compute || PassFlags == ERDGPassFlags::AsyncCompute);
RDG_EVENT_SCOPE(GraphBuilder, "LocalExposure - Apply");
auto* PassParameters = GraphBuilder.AllocParameters<FApplyLocalExposureCS::FParameters>();
PassParameters->View = View.ViewUniformBuffer;
PassParameters->Input = GetScreenPassTextureViewportParameters(FScreenPassTextureViewport(Input));
PassParameters->Output = GetScreenPassTextureViewportParameters(FScreenPassTextureViewport(Output));
PassParameters->InputTexture = Input.TextureSRV;
{
FRDGTextureUAVDesc OutputDesc(Output.TextureSRV->Desc.Texture);
if (Output.TextureSRV->Desc.Texture->Desc.IsTextureArray())
{
OutputDesc.DimensionOverride = ETextureDimension::Texture2D;
OutputDesc.FirstArraySlice = Output.TextureSRV->Desc.FirstArraySlice;
OutputDesc.NumArraySlices = 1;
}
PassParameters->OutputFloat4 = GraphBuilder.CreateUAV(OutputDesc);
}
PassParameters->EyeAdaptation = EyeAdaptationParameters;
PassParameters->EyeAdaptationBuffer = GraphBuilder.CreateSRV(EyeAdaptationBuffer);
PassParameters->LocalExposure = LocalExposureParamaters;
PassParameters->LumBilateralGrid = LocalExposureTexture;
PassParameters->BlurredLogLum = BlurredLogLuminanceTexture;
PassParameters->TextureSampler = TStaticSamplerState<SF_Bilinear, AM_Clamp, AM_Clamp, AM_Clamp>::GetRHI();;
FComputeShaderUtils::AddPass(
GraphBuilder,
RDG_EVENT_NAME("ApplyLocalExposure %dx%d", Output.ViewRect.Width(), Output.ViewRect.Height()),
PassFlags,
View.ShaderMap->GetShader<FApplyLocalExposureCS>(),
PassParameters,
FComputeShaderUtils::GetGroupCount(Output.ViewRect.Size(), FIntPoint(FApplyLocalExposureCS::ThreadGroupSizeX, FApplyLocalExposureCS::ThreadGroupSizeY)));
}