223 lines
7.0 KiB
Plaintext
223 lines
7.0 KiB
Plaintext
Shader "Hidden/PostProcessing/Lut2DBaker"
|
|
{
|
|
HLSLINCLUDE
|
|
|
|
#pragma target 3.0
|
|
#include "../StdLib.hlsl"
|
|
#include "../Colors.hlsl"
|
|
#include "../ACES.hlsl"
|
|
|
|
TEXTURE2D_SAMPLER2D(_MainTex, sampler_MainTex);
|
|
float4 _Lut2D_Params;
|
|
float4 _UserLut2D_Params;
|
|
|
|
float3 _ColorBalance;
|
|
float3 _ColorFilter;
|
|
float3 _HueSatCon;
|
|
float _Brightness; // LDR only
|
|
|
|
float3 _ChannelMixerRed;
|
|
float3 _ChannelMixerGreen;
|
|
float3 _ChannelMixerBlue;
|
|
|
|
float3 _Lift;
|
|
float3 _InvGamma;
|
|
float3 _Gain;
|
|
|
|
TEXTURE2D_SAMPLER2D(_Curves, sampler_Curves);
|
|
|
|
float4 _CustomToneCurve;
|
|
float4 _ToeSegmentA;
|
|
float4 _ToeSegmentB;
|
|
float4 _MidSegmentA;
|
|
float4 _MidSegmentB;
|
|
float4 _ShoSegmentA;
|
|
float4 _ShoSegmentB;
|
|
|
|
float3 ApplyCommonGradingSteps(float3 colorLinear)
|
|
{
|
|
colorLinear = WhiteBalance(colorLinear, _ColorBalance);
|
|
colorLinear *= _ColorFilter;
|
|
colorLinear = ChannelMixer(colorLinear, _ChannelMixerRed, _ChannelMixerGreen, _ChannelMixerBlue);
|
|
colorLinear = LiftGammaGainHDR(colorLinear, _Lift, _InvGamma, _Gain);
|
|
|
|
// Do NOT feed negative values to RgbToHsv or they'll wrap around
|
|
colorLinear = max((float3)0.0, colorLinear);
|
|
|
|
float3 hsv = RgbToHsv(colorLinear);
|
|
|
|
// Hue Vs Sat
|
|
float satMult;
|
|
satMult = saturate(SAMPLE_TEXTURE2D_LOD(_Curves, sampler_Curves, float2(hsv.x, 0.25), 0).y) * 2.0;
|
|
|
|
// Sat Vs Sat
|
|
satMult *= saturate(SAMPLE_TEXTURE2D_LOD(_Curves, sampler_Curves, float2(hsv.y, 0.25), 0).z) * 2.0;
|
|
|
|
// Lum Vs Sat
|
|
satMult *= saturate(SAMPLE_TEXTURE2D_LOD(_Curves, sampler_Curves, float2(Luminance(colorLinear), 0.25), 0).w) * 2.0;
|
|
|
|
// Hue Vs Hue
|
|
float hue = hsv.x + _HueSatCon.x;
|
|
float offset = saturate(SAMPLE_TEXTURE2D_LOD(_Curves, sampler_Curves, float2(hue, 0.25), 0).x) - 0.5;
|
|
hue += offset;
|
|
hsv.x = RotateHue(hue, 0.0, 1.0);
|
|
|
|
colorLinear = HsvToRgb(hsv);
|
|
colorLinear = Saturation(colorLinear, _HueSatCon.y * satMult);
|
|
|
|
return colorLinear;
|
|
}
|
|
|
|
//
|
|
// LDR Grading process
|
|
//
|
|
float3 ColorGradeLDR(float3 colorLinear)
|
|
{
|
|
// Brightness is a simple linear multiplier. Works better in LDR than using e.v.
|
|
colorLinear *= _Brightness;
|
|
|
|
// Contrast is done in linear, switching to log for that in LDR is pointless and doesn't
|
|
// feel as good to tweak
|
|
const float kMidGrey = pow(0.5, 2.2);
|
|
colorLinear = Contrast(colorLinear, kMidGrey, _HueSatCon.z);
|
|
|
|
colorLinear = ApplyCommonGradingSteps(colorLinear);
|
|
|
|
// YRGB only works in LDR for now as we don't do any curve range remapping
|
|
colorLinear = YrgbCurve(saturate(colorLinear), TEXTURE2D_PARAM(_Curves, sampler_Curves));
|
|
|
|
return saturate(colorLinear);
|
|
}
|
|
|
|
float4 FragLDRFromScratch(VaryingsDefault i) : SV_Target
|
|
{
|
|
float3 colorLinear = GetLutStripValue(i.texcoordStereo, _Lut2D_Params);
|
|
float3 graded = ColorGradeLDR(colorLinear);
|
|
return float4(graded, 1.0);
|
|
}
|
|
|
|
float4 FragLDR(VaryingsDefault i) : SV_Target
|
|
{
|
|
// Note: user luts may not have the same size as the internal one
|
|
float3 neutralColorLinear = GetLutStripValue(i.texcoordStereo, _Lut2D_Params);
|
|
float3 lookup = ApplyLut2D(TEXTURE2D_PARAM(_MainTex, sampler_MainTex), neutralColorLinear, _UserLut2D_Params.xyz);
|
|
float3 colorLinear = lerp(neutralColorLinear, lookup, _UserLut2D_Params.w);
|
|
float3 graded = ColorGradeLDR(colorLinear);
|
|
return float4(graded, 1.0);
|
|
}
|
|
|
|
//
|
|
// HDR Grading process
|
|
//
|
|
float3 LogGradeHDR(float3 colorLog)
|
|
{
|
|
// HDR contrast feels a lot more natural when done in log rather than doing it in linear
|
|
colorLog = Contrast(colorLog, ACEScc_MIDGRAY, _HueSatCon.z);
|
|
return colorLog;
|
|
}
|
|
|
|
float3 LinearGradeHDR(float3 colorLinear)
|
|
{
|
|
colorLinear = ApplyCommonGradingSteps(colorLinear);
|
|
return colorLinear;
|
|
}
|
|
|
|
float3 ColorGradeHDR(float3 colorLutSpace)
|
|
{
|
|
#if TONEMAPPING_ACES
|
|
{
|
|
float3 colorLinear = LUT_SPACE_DECODE(colorLutSpace);
|
|
float3 aces = unity_to_ACES(colorLinear);
|
|
|
|
// ACEScc (log) space
|
|
float3 acescc = ACES_to_ACEScc(aces);
|
|
acescc = LogGradeHDR(acescc);
|
|
aces = ACEScc_to_ACES(acescc);
|
|
|
|
// ACEScg (linear) space
|
|
float3 acescg = ACES_to_ACEScg(aces);
|
|
acescg = LinearGradeHDR(acescg);
|
|
|
|
// Tonemap ODT(RRT(aces))
|
|
aces = ACEScg_to_ACES(acescg);
|
|
colorLinear = AcesTonemap(aces);
|
|
|
|
return colorLinear;
|
|
}
|
|
#else
|
|
{
|
|
// colorLutSpace is already in log space
|
|
colorLutSpace = LogGradeHDR(colorLutSpace);
|
|
|
|
// Switch back to linear
|
|
float3 colorLinear = LUT_SPACE_DECODE(colorLutSpace);
|
|
colorLinear = LinearGradeHDR(colorLinear);
|
|
colorLinear = max(0.0, colorLinear);
|
|
|
|
// Tonemap
|
|
#if TONEMAPPING_NEUTRAL
|
|
{
|
|
colorLinear = NeutralTonemap(colorLinear);
|
|
}
|
|
#elif TONEMAPPING_CUSTOM
|
|
{
|
|
colorLinear = CustomTonemap(
|
|
colorLinear, _CustomToneCurve.xyz,
|
|
_ToeSegmentA, _ToeSegmentB.xy,
|
|
_MidSegmentA, _MidSegmentB.xy,
|
|
_ShoSegmentA, _ShoSegmentB.xy
|
|
);
|
|
}
|
|
#endif
|
|
|
|
return colorLinear;
|
|
}
|
|
#endif
|
|
}
|
|
|
|
float4 FragHDR(VaryingsDefault i) : SV_Target
|
|
{
|
|
float3 colorLutSpace = GetLutStripValue(i.texcoord, _Lut2D_Params);
|
|
float3 graded = ColorGradeHDR(colorLutSpace);
|
|
return float4(max(graded, 0.0), 1.0);
|
|
}
|
|
|
|
ENDHLSL
|
|
|
|
SubShader
|
|
{
|
|
Cull Off ZWrite Off ZTest Always
|
|
|
|
Pass
|
|
{
|
|
HLSLPROGRAM
|
|
|
|
#pragma vertex VertDefault
|
|
#pragma fragment FragLDRFromScratch
|
|
|
|
ENDHLSL
|
|
}
|
|
|
|
Pass
|
|
{
|
|
HLSLPROGRAM
|
|
|
|
#pragma vertex VertDefault
|
|
#pragma fragment FragLDR
|
|
|
|
ENDHLSL
|
|
}
|
|
|
|
Pass
|
|
{
|
|
HLSLPROGRAM
|
|
|
|
#pragma vertex VertDefault
|
|
#pragma fragment FragHDR
|
|
#pragma multi_compile __ TONEMAPPING_ACES TONEMAPPING_NEUTRAL TONEMAPPING_CUSTOM
|
|
|
|
ENDHLSL
|
|
}
|
|
}
|
|
}
|