// Important! This file assumes Color.hlsl and ACES.hlsl has been already included. #ifndef HDROUTPUT_INCLUDED #define HDROUTPUT_INCLUDED #include "Packages/com.unity.render-pipelines.core/Runtime/PostProcessing/HDROutputDefines.cs.hlsl" #if defined(HDR_COLORSPACE_CONVERSION_AND_ENCODING) #define HDR_COLORSPACE_CONVERSION #define HDR_ENCODING #endif int _HDRColorspace; int _HDREncoding; // A bit of nomenclature that will be used in the file: // Gamut: It is the subset of colors that is possible to reproduce by using three specific primary colors. // Rec709 (ITU-R Recommendation BT709) is a HDTV standard, in our context, we mostly care about its color gamut (https://en.wikipedia.org/wiki/Rec._709). The Rec709 gamut is the same as BT1886 and sRGB. // Rec2020 (ITU-R Recommendation BT2020) is an UHDTV standard. As above, we mostly reference it w.r.t. the color gamut. (https://en.wikipedia.org/wiki/Rec._2020). Nice property is that all primaries are on the locus. // DCI-P3 (or just P3): is a gamut used in cinema grading and used by iPhone for example. // ACEScg: A gamut that is larger than Rec2020. // ACES2065-1: A gamut that covers the full XYZ space, part of the ACES specs. Mostly used for storage since it is harder to work with than ACEScg. // WCG: Wide color gamut. This is defined as a color gamut that is wider than the Rec709 one. // LMS: A color space represented by the response of the three cones of human eye (responsivity peaks Long, Medium, Short) // OETF (Optical Eelectro Transfer Function): This is a function to goes from optical (linear light) to electro (signal transmitted to the display). // EOTF (Eelectro Optical Transfer Function): The inverse of the OETF, used by the TV/Monitor. // EETF (Eelectro-Electro Transfer Function): This is generally just a remapping function, we use the BT2390 EETF to perform range reduction based on the actual display. // PQ (Perceptual Quantizer): the EOTF used for HDR10 TVs. It works in the range [0, 10000] nits. Important to keep in mind that this represents an absolute intensity and not relative as for SDR. Sometimes this can be referenced as ST2084. As OETF we'll use the inverse of the PQ curve. // scRGB: a wide color gamut that uses same color space and white point as sRGB, but with much wider coordinates. Used on windows when 16 bit depth is selected. Most of the color space is imaginary colors. Works differently than with PQ (encoding is linear). // G22 (Gamma 2.2): the EOTF used for exact gamma 2.2 curve. // -------------------------------- // Perceptual Quantizer (PQ) / ST 2084 // -------------------------------- #define MAX_PQ_VALUE 10000 // 10k nits is the maximum supported by the standard. #define PQ_N (2610.0f / 4096.0f / 4.0f) #define PQ_M (2523.0f / 4096.0f * 128.0f) #define PQ_C1 (3424.0f / 4096.0f) #define PQ_C2 (2413.0f / 4096.0f * 32.0f) #define PQ_C3 (2392.0f / 4096.0f * 32.0f) float LinearToPQ(float value, float maxPQValue) { value /= maxPQValue; float Ym1 = PositivePow(value, PQ_N); float n = (PQ_C1 + PQ_C2 * Ym1); float d = (1.0f + PQ_C3 * Ym1); return PositivePow(n / d, PQ_M); } float LinearToPQ(float value) { return LinearToPQ(value, MAX_PQ_VALUE); } float3 LinearToPQ(float3 value, float maxPQValue) { float3 outPQ; outPQ.x = LinearToPQ(value.x, maxPQValue); outPQ.y = LinearToPQ(value.y, maxPQValue); outPQ.z = LinearToPQ(value.z, maxPQValue); return outPQ; } float3 LinearToPQ(float3 value) { return LinearToPQ(value, MAX_PQ_VALUE); } float PQToLinear(float value) { float Em2 = PositivePow(value, 1.0f / PQ_M); float X = (max(0.0, Em2 - PQ_C1)) / (PQ_C2 - PQ_C3 * Em2); return PositivePow(X, 1.0f / PQ_N); } float PQToLinear(float value, float maxPQValue) { return PQToLinear(value) * maxPQValue; } float3 PQToLinear(float3 value, float maxPQValue) { float3 outLinear; outLinear.x = PQToLinear(value.x, maxPQValue); outLinear.y = PQToLinear(value.y, maxPQValue); outLinear.z = PQToLinear(value.z, maxPQValue); return outLinear; } float3 PQToLinear(float3 value) { float3 outLinear; outLinear.x = PQToLinear(value.x); outLinear.y = PQToLinear(value.y); outLinear.z = PQToLinear(value.z); return outLinear; } // -------------------------------------------------------------------------------------------- // -------------------------------- // Color Space transforms // -------------------------------- // As any other space transform, changing color space involves a change of basis and therefore a matrix multiplication. // Note that Rec2020 and Rec2100 share the same color space. // This section should be kept in sync with Runtime\Utilities\ColorSpaceUtils.cs float3 RotateRec709ToRec2020(float3 Rec709Input) { static const float3x3 Rec709ToRec2020Mat = float3x3( 0.627402, 0.329292, 0.043306, 0.069095, 0.919544, 0.011360, 0.016394, 0.088028, 0.895578 ); return mul(Rec709ToRec2020Mat, Rec709Input); } float3 RotateRec709ToP3D65(float3 Rec709Input) { static const float3x3 Rec709ToP3D65Mat = float3x3( 0.822462, 0.177538, 0.000000, 0.033194, 0.966806, 0.000000, 0.017083, 0.072397, 0.910520 ); return mul(Rec709ToP3D65Mat, Rec709Input); } float3 RotateRec2020ToRec709(float3 Rec2020Input) { static const float3x3 Rec2020ToRec709Mat = float3x3( 1.660496, -0.587656, -0.072840, -0.124547, 1.132895, -0.008348, -0.018154, -0.100597, 1.118751 ); return mul(Rec2020ToRec709Mat, Rec2020Input); } float3 RotateRec2020ToP3D65(float3 Rec2020Input) { static const float3x3 Rec2020ToP3D65Mat = float3x3( 1.343578, -0.28218, -0.0613986, -0.065298, 1.075788, -0.010491, 0.002822, -0.019599, 1.016777 ); return mul(Rec2020ToP3D65Mat, Rec2020Input); } float3 RotateP3D65ToRec2020(float3 P3D65Input) { static const float3x3 P3D65ToRec2020Mat = float3x3( 0.753833, 0.198597, 0.04757, 0.045744, 0.941777, 0.012479, -0.001210, 0.017602, 0.983609 ); return mul(P3D65ToRec2020Mat, P3D65Input); } float3 RotateRec709ToOutputSpace(float3 Rec709Input) { if (_HDRColorspace == HDRCOLORSPACE_REC2020) { return RotateRec709ToRec2020(Rec709Input); } else if (_HDRColorspace == HDRCOLORSPACE_P3D65) { return RotateRec709ToP3D65(Rec709Input); } else // HDRCOLORSPACE_REC709 { return Rec709Input; } } float3 RotateRec2020ToOutputSpace(float3 Rec2020Input) { if (_HDRColorspace == HDRCOLORSPACE_REC2020) { return Rec2020Input; } else if (_HDRColorspace == HDRCOLORSPACE_P3D65) { return RotateRec2020ToP3D65(Rec2020Input); } else // HDRCOLORSPACE_REC709 { return RotateRec2020ToRec709(Rec2020Input); } } float3 RotateRec2020ToLMS(float3 Rec2020Input) { static const float3x3 Rec2020ToLMSMat = { 0.412109375, 0.52392578125, 0.06396484375, 0.166748046875, 0.720458984375, 0.11279296875, 0.024169921875, 0.075439453125, 0.900390625 }; return mul(Rec2020ToLMSMat, Rec2020Input); } float3 Rotate709ToLMS(float3 Rec709Input) { static const float3x3 Rec709ToLMSMat = { 0.412109375, 0.52392578125, 0.06396484375, 0.166748046875, 0.720458984375, 0.11279296875, 0.024169921875, 0.075439453125, 0.900390625 }; return mul(Rec709ToLMSMat, Rec709Input); } // Ref: ICtCp Dolby white paper (https://www.dolby.com/us/en/technologies/dolby-vision/ictcp-white-paper.pdf) float3 RotatePQLMSToICtCp(float3 LMSInput) { static const float3x3 PQLMSToICtCpMat = float3x3( 0.5f, 0.5f, 0.0f, 1.613769f, -3.323486f, 1.709716f, 4.378174f, -4.245605f, -0.1325683f ); return mul(PQLMSToICtCpMat, LMSInput); } float3 RotateLMSToICtCp(float3 lms) { float3 PQLMS = LinearToPQ(max(0.0f, lms)); return RotatePQLMSToICtCp(PQLMS); } float3 RotateRec2020ToICtCp(float3 Rec2020) { float3 lms = RotateRec2020ToLMS(Rec2020); float3 PQLMS = LinearToPQ(max(0.0f, lms)); return RotatePQLMSToICtCp(PQLMS); } float3 RotateOutputSpaceToICtCp(float3 inputColor) { // TODO: Do the conversion directly from Rec709 (bake matrix Rec709 -> XYZ -> LMS) if (_HDRColorspace == HDRCOLORSPACE_REC709) { inputColor = RotateRec709ToRec2020(inputColor); } else if (_HDRColorspace == HDRCOLORSPACE_P3D65) { inputColor = RotateP3D65ToRec2020(inputColor); } return RotateRec2020ToICtCp(inputColor); } float3 RotateLMSToXYZ(float3 LMSInput) { static const float3x3 LMSToXYZMat = float3x3( 2.07018005669561320f, -1.32645687610302100f, 0.206616006847855170f, 0.36498825003265756f, 0.68046736285223520f, -0.045421753075853236f, -0.04959554223893212f, -0.04942116118675749f, 1.187995941732803400f ); return mul(LMSToXYZMat, LMSInput); } float3 RotateXYZToRec2020(float3 XYZ) { static const float3x3 XYZToRec2020Mat = float3x3( 1.71235168f, -0.35487896f, -0.25034135f, -0.66728621f, 1.61794055f, 0.01495380f, 0.01763985f, -0.04277060f, 0.94210320f ); return mul(XYZToRec2020Mat, XYZ); } float3 RotateXYZToRec709(float3 XYZ) { return mul(XYZ_2_REC709_MAT, XYZ); } float3 RotateXYZToP3D65(float3 XYZ) { return mul(XYZ_2_P3D65_MAT, XYZ); } float3 RotateXYZToOutputSpace(float3 xyz) { if (_HDRColorspace == HDRCOLORSPACE_REC2020) { return RotateXYZToRec2020(xyz); } else if (_HDRColorspace == HDRCOLORSPACE_P3D65) { return RotateXYZToP3D65(xyz); } else // HDRCOLORSPACE_REC709 { return RotateXYZToRec709(xyz); } } float3 RotateRec709ToXYZ(float3 rgb) { static const float3x3 Rec709ToXYZMat = float3x3( 0.412391f, 0.357584f, 0.180481, 0.212639, 0.715169, 0.0721923, 0.0193308, 0.119195, 0.950532 ); return mul(Rec709ToXYZMat, rgb); } float3 RotateRec2020ToXYZ(float3 rgb) { static const float3x3 Rec2020ToXYZMat = float3x3( 0.638574, 0.144617, 0.167265, 0.263367, 0.677998, 0.0586353, 0.0f, 0.0280727, 1.06099 ); return mul(Rec2020ToXYZMat, rgb); } float3 RotateP3D65ToXYZ(float3 rgb) { static const float3x3 P3D65ToXYZMat = float3x3( 0.486571, 0.265668, 0.198217, 0.228975, 0.691739, 0.079287, 0, 0.045113, 1.043944 ); return mul(P3D65ToXYZMat, rgb); } float3 RotateOutputSpaceToXYZ(float3 rgb) { if (_HDRColorspace == HDRCOLORSPACE_REC2020) { return RotateRec2020ToXYZ(rgb); } else if (_HDRColorspace == HDRCOLORSPACE_P3D65) { return RotateP3D65ToXYZ(rgb); } else // HDRCOLORSPACE_REC709 { return RotateRec709ToXYZ(rgb); } } float3 RotateICtCpToPQLMS(float3 ICtCp) { static const float3x3 ICtCpToPQLMSMat = float3x3( 1.0f, 0.0086051456939815f, 0.1110356044754732f, 1.0f, -0.0086051456939815f, -0.1110356044754732f, 1.0f, 0.5600488595626390f, -0.3206374702321221f ); return mul(ICtCpToPQLMSMat, ICtCp); } float3 RotateICtCpToXYZ(float3 ICtCp) { float3 PQLMS = RotateICtCpToPQLMS(ICtCp); float3 LMS = PQToLinear(PQLMS, MAX_PQ_VALUE); return RotateLMSToXYZ(LMS); } float3 RotateICtCpToRec2020(float3 ICtCp) { return RotateXYZToRec2020(RotateICtCpToXYZ(ICtCp)); } float3 RotateICtCpToRec709(float3 ICtCp) { return RotateXYZToRec709(RotateICtCpToXYZ(ICtCp)); } float3 RotateICtCpToP3D65(float3 ICtCp) { return RotateXYZToP3D65(RotateICtCpToXYZ(ICtCp)); } float3 RotateICtCpToOutputSpace(float3 ICtCp) { if (_HDRColorspace == HDRCOLORSPACE_REC2020) { return RotateICtCpToRec2020(ICtCp); } else if (_HDRColorspace == HDRCOLORSPACE_P3D65) { return RotateICtCpToP3D65(ICtCp); } else // HDRCOLORSPACE_REC709 { return RotateICtCpToRec709(ICtCp); } } // -------------------------------------------------------------------------------------------- // -------------------------------- // OETFs // -------------------------------- // The functions here are OETF, technically for applying the opposite of the PQ curve, we are mapping // from linear to PQ space as this is what the display expects. // See this desmos for comparisons https://www.desmos.com/calculator/5jdfc4pgtk #define PRECISE_PQ 0 #define ISS_APPROX_PQ 1 #define GTS_APPROX_PQ 2 #define OETF_CHOICE GTS_APPROX_PQ // What the platforms expects as SDR max brightness (different from paper white brightness) in linear encoding #if defined(SHADER_API_METAL) #define SDR_REF_WHITE 100 #else #define SDR_REF_WHITE 80 #endif // Ref: [Patry 2017] HDR Display Support in Infamous Second Son and Infamous First Light // Fastest option, but also the least accurate. Behaves well for values up to 1400 nits but then starts diverging. // IMPORTANT! It requires the input to be scaled from [0 ... 10000] to [0...100]! float3 PatryApproxLinToPQ(float3 x) { return (x * (x * (x * (x * (x * 533095.76 + 47438306.2) + 29063622.1) + 575216.76) + 383.09104) + 0.000487781) / (x * (x * (x * (x * 66391357.4 + 81884528.2) + 4182885.1) + 10668.404) + 1.0); } // Ref: [Uchimura and Suzuki 2018] Practical HDR and Wide Color Techniques in Gran Turismo Sport // Slower than Infamous approx, but more precise ( https://www.desmos.com/calculator/up4wwozghk ) in the full [0... 10 000] range, but still faster than reference // IMPORTANT! It requires the input to be scaled from [0 ... 10000] to [0...100]! float3 GTSApproxLinToPQ(float3 inputCol) { float3 k = PositivePow((inputCol * 0.01), PQ_N); return (3.61972*(1e-8) + k * (0.00102859 + k * (-0.101284 + 2.05784 * k))) / (0.0495245 + k * (0.135214 + k * (0.772669 + k))); } // IMPORTANT! This wants the input in [0...10000] range, if the method requires scaling, it is done inside this function. float3 OETF(float3 inputCol, float maxNits) { if (_HDREncoding == HDRENCODING_LINEAR) { // IMPORTANT! This assumes that the maximum nits is always higher or same as the reference white. Seems like a sensible choice, but revisit if we find weird use cases (just min with the the max nits). // We need to map the value 1 to [reference white] nits. return inputCol / SDR_REF_WHITE; } else if (_HDREncoding == HDRENCODING_PQ) { #if OETF_CHOICE == PRECISE_PQ return LinearToPQ(inputCol); #elif OETF_CHOICE == ISS_APPROX_PQ return PatryApproxLinToPQ(inputCol * 0.01f); #elif OETF_CHOICE == GTS_APPROX_PQ return GTSApproxLinToPQ(inputCol * 0.01f); #endif } else if (_HDREncoding == HDRENCODING_GAMMA22) { return LinearToGamma22(inputCol / (float3)maxNits); // Usually used to encode into UNORM output 0->1 where 1 is the max display brightness, this will be very device specific so use our maxNits. } else if (_HDREncoding == HDRENCODING_S_RGB) { return inputCol / (float3)maxNits; // Usually used to encode into UNORM output 0->1 where 1 is the max display brightness, this will be very device specific so use our maxNits. } else { return inputCol; } } #define LIN_TO_PQ_FOR_LUT GTS_APPROX_PQ // GTS is close enough https://www.desmos.com/calculator/up4wwozghk float3 LinearToPQForLUT(float3 inputCol) { #if LIN_TO_PQ_FOR_LUT == PRECISE_PQ return LinearToPQ(inputCol); #elif LIN_TO_PQ_FOR_LUT == ISS_APPROX_PQ return PatryApproxLinToPQ(inputCol * 0.01f); #elif LIN_TO_PQ_FOR_LUT == GTS_APPROX_PQ return GTSApproxLinToPQ(inputCol * 0.01f); #endif } // -------------------------------------------------------------------------------------------- // -------------------------------- // Inverse OETFs // -------------------------------- // The functions here are the inverse of our OETF, to allow reading in data we have processed through our OETF. // This is not the same as the EOTF as that is not always a direct inverse of the OETF so that an end to end transfer may not be linear. // We pass the HDR Encoding value manually into this function rather than reading from _HDREncoding as that is the currnet destination encoding // and we may be reading from a different encoding as part of translating between encodings. float3 InverseOETF(float3 inputCol, float maxNits, int hdrEncoding) { if (hdrEncoding == HDRENCODING_LINEAR) { // IMPORTANT! This assumes that the maximum nits is always higher or same as the reference white. Seems like a sensible choice, but revisit if we find weird use cases (just min with the the max nits). // We need to map the value 1 to [reference white] nits. return inputCol * SDR_REF_WHITE; } else if (hdrEncoding == HDRENCODING_PQ) { // For the time being always use the reference spec for computing the inverse. For the optimized version of LinearToPQ we would need to fit our own PQToLinear version. return PQToLinear(inputCol); } else if (hdrEncoding == HDRENCODING_GAMMA22) { return Gamma22ToLinear(inputCol) * maxNits; } else if (hdrEncoding == HDRENCODING_S_RGB) { return inputCol * maxNits; } else { return inputCol; } } // -------------------------------------------------------------------------------------------- // -------------------------------- // Range reduction // -------------------------------- // This section of the file concerns the way we map from full range to whatever range the display supports. // Also note, we always tonemap luminance component only, so we need to reach this point after we converted // to a format such as ICtCp or YCbCr // See https://www.desmos.com/calculator/pqc3raolms for plots #define RANGE_REDUCTION HDRRANGEREDUCTION_BT2390LUMA_ONLY // Note this takes x being in [0...10k nits] float ReinhardTonemap(float x, float peakValue) { float m = MAX_PQ_VALUE * peakValue / (MAX_PQ_VALUE - peakValue); return x * m / (x + m); } /// BT2390 EETF Helper functions float T(float A, float Ks) { return (A - Ks) / (1.0f - Ks); } float P(float B, float Ks, float L_max) { float TB2 = T(B, Ks) * T(B, Ks); float TB3 = TB2 * T(B, Ks); return lerp((TB3 - 2 * TB2 + T(B, Ks)), (2.0f * TB3 - 3.0f * TB2 + 1.0f), Ks) + (-2.0f * TB3 + 3.0f*TB2)*L_max; } // Ref: https://www.itu.int/dms_pub/itu-r/opb/rep/R-REP-BT.2390-4-2018-PDF-E.pdf page 21 // This takes values in [0...10k nits] and it outputs in the same space. PQ conversion outside. // If we chose this, it can be optimized (a few identity happen with moving between linear and PQ) float BT2390EETF(float x, float minLimit, float maxLimit) { float E_0 = LinearToPQ(x); // For the following formulas we are assuming L_B = 0 and L_W = 10000 -- see original paper for full formulation float E_1 = E_0; float L_min = LinearToPQ(minLimit); float L_max = LinearToPQ(maxLimit); float Ks = 1.5f * L_max - 0.5f; // Knee start float b = L_min; float E_2 = E_1 < Ks ? E_1 : P(E_1, Ks, L_max); float E3Part = (1.0f - E_2); float E3Part2 = E3Part * E3Part; float E_3 = E_2 + b * (E3Part2 * E3Part2); float E_4 = E_3; // Is like this because PQ(L_W)= 1 and PQ(L_B) = 0 return PQToLinear(E_4, MAX_PQ_VALUE); } float3 PerformRangeReduction(float3 input, float minNits, float maxNits) { float3 ICtCp = RotateOutputSpaceToICtCp(input); // This is in PQ space. float linearLuma = PQToLinear(ICtCp.x, MAX_PQ_VALUE); #if RANGE_REDUCTION == HDRRANGEREDUCTION_REINHARD_LUMA_ONLY linearLuma = ReinhardTonemap(linearLuma, maxNits); #elif RANGE_REDUCTION == HDRRANGEREDUCTION_BT2390LUMA_ONLY linearLuma = BT2390EETF(linearLuma, minNits, maxNits); #endif ICtCp.x = LinearToPQ(linearLuma); return RotateICtCpToOutputSpace(ICtCp); // This moves back to linear too! } // TODO: This is very ad-hoc and eyeballed on a limited set. Would be nice to find a standard. float3 DesaturateReducedICtCp(float3 ICtCp, float lumaPre, float maxNits) { float saturationAmount = min(1.0f, ICtCp.x / max(lumaPre, 1e-6f)); // BT2390, but only when getting darker. //saturationAmount = min(lumaPre / ICtCp.x, ICtCp.x / lumaPre); // Actual BT2390 suggestion saturationAmount *= saturationAmount; //saturationAmount = pow(smoothstep(1.0f, 0.4f, ICtCp.x), 0.9f); // A smoothstepp-y function. ICtCp.yz *= saturationAmount; return ICtCp; } float LumaRangeReduction(float input, float minNits, float maxNits, int mode) { float output = input; if (mode == HDRRANGEREDUCTION_REINHARD) { output = ReinhardTonemap(input, maxNits); } else if (mode == HDRRANGEREDUCTION_BT2390) { output = BT2390EETF(input, minNits, maxNits); } return output; } float3 HuePreservingRangeReduction(float3 input, float minNits, float maxNits, int mode) { float3 ICtCp = RotateOutputSpaceToICtCp(input); float lumaPreRed = ICtCp.x; float linearLuma = PQToLinear(ICtCp.x, MAX_PQ_VALUE); linearLuma = LumaRangeReduction(linearLuma, minNits, maxNits, mode); ICtCp.x = LinearToPQ(linearLuma); ICtCp = DesaturateReducedICtCp(ICtCp, lumaPreRed, maxNits); return RotateICtCpToOutputSpace(ICtCp); } float3 HueShiftingRangeReduction(float3 input, float minNits, float maxNits, int mode) { float3 hueShiftedResult = input; if (mode == HDRRANGEREDUCTION_REINHARD) { hueShiftedResult.x = ReinhardTonemap(input.x, maxNits); hueShiftedResult.y = ReinhardTonemap(input.y, maxNits); hueShiftedResult.z = ReinhardTonemap(input.z, maxNits); } else if(mode == HDRRANGEREDUCTION_BT2390) { hueShiftedResult.x = BT2390EETF(input.x, minNits, maxNits); hueShiftedResult.y = BT2390EETF(input.y, minNits, maxNits); hueShiftedResult.z = BT2390EETF(input.z, minNits, maxNits); } return hueShiftedResult; } // Ref "High Dynamic Range color grading and display in Frostbite" [Fry 2017] float3 FryHuePreserving(float3 input, float minNits, float maxNits, float hueShift, int mode) { float3 ictcp = RotateOutputSpaceToICtCp(input); // Hue-preserving range compression requires desaturation in order to achieve a natural look. We adaptively desaturate the input based on its luminance. float saturationAmount = pow(smoothstep(1.0, 0.3, ictcp.x), 1.3); float3 col = RotateICtCpToOutputSpace(ictcp * float3(1, saturationAmount.xx)); // Only compress luminance starting at a certain point. Dimmer inputs are passed through without modification. float linearSegmentEnd = 0.25f; // Hue-preserving mapping float maxCol = max(col.x, max(col.y, col.z)); float mappedMax = maxCol; if (maxCol > linearSegmentEnd) { mappedMax = LumaRangeReduction(maxCol, minNits, maxNits, mode); } float3 compressedHuePreserving = col * mappedMax / maxCol; // Non-hue preserving mapping float3 perChannelCompressed = 0; perChannelCompressed.x = col.x > linearSegmentEnd ? LumaRangeReduction(col.x, minNits, maxNits, mode) : col.x; perChannelCompressed.y = col.y > linearSegmentEnd ? LumaRangeReduction(col.y, minNits, maxNits, mode) : col.y; perChannelCompressed.z = col.z > linearSegmentEnd ? LumaRangeReduction(col.z, minNits, maxNits, mode) : col.z; // Combine hue-preserving and non-hue-preserving colors. Absolute hue preservation looks unnatural, as bright colors *appear* to have been hue shifted. // Actually doing some amount of hue shifting looks more pleasing col = lerp(perChannelCompressed, compressedHuePreserving, 1-hueShift); float3 ictcpMapped = RotateOutputSpaceToICtCp(col); // Smoothly ramp off saturation as brightness increases, but keep some even for very bright input float postCompressionSaturationBoost = 0.3 * smoothstep(1.0, 0.5, ictcp.x); // Re-introduce some hue from the pre-compression color. Something similar could be accomplished by delaying the luma-dependent desaturation before range compression. // Doing it here however does a better job of preserving perceptual luminance of highly saturated colors. Because in the hue-preserving path we only range-compress the max channel, // saturated colors lose luminance. By desaturating them more aggressively first, compressing, and then re-adding some saturation, we can preserve their brightness to a greater extent. ictcpMapped.yz = lerp(ictcpMapped.yz, ictcp.yz * ictcpMapped.x / max(1e-3, ictcp.x), postCompressionSaturationBoost); col = RotateICtCpToOutputSpace(ictcpMapped); return col; } float3 PerformRangeReduction(float3 input, float minNits, float maxNits, int mode, float hueShift) { float3 outputValue = input; bool reduceLuma = hueShift < 1.0f; bool needHueShiftVersion = hueShift > 0.0f; if (mode == HDRRANGEREDUCTION_NONE) { outputValue = input; } else { float3 huePreserving = reduceLuma ? HuePreservingRangeReduction(input, minNits, maxNits, mode) : 0; float3 hueShifted = needHueShiftVersion ? HueShiftingRangeReduction(input, minNits, maxNits, mode) : 0; if (reduceLuma && !needHueShiftVersion) { outputValue = huePreserving; } else if (!reduceLuma && needHueShiftVersion) { outputValue = hueShifted; } else { // We need to combine the two cases outputValue = lerp(huePreserving, hueShifted, hueShift); } } return outputValue; } // -------------------------------------------------------------------------------------------- // -------------------------------- // Public facing functions // -------------------------------- // These functions are aggregate of most of what we have above. You can think of this as the public API of the HDR Output library. // Note that throughout HDRP we are assuming that when it comes to the final pass adjustements, our tonemapper has *NOT* // performed range reduction and everything is assumed to be displayed on a reference 10k nits display and everything post-tonemapping // is in either the Rec 2020 or Rec709 color space. The Rec709 version just rotate to Rec2020 before going forward if required by the output device. float3 HDRMappingFromRec2020(float3 Rec2020Input, float paperWhite, float minNits, float maxNits, int reductionMode, float hueShift, bool skipOETF = false) { float3 outputSpaceInput = RotateRec2020ToOutputSpace(Rec2020Input); float3 reducedHDR = PerformRangeReduction(outputSpaceInput * paperWhite, minNits, maxNits, reductionMode, hueShift); if (skipOETF) return reducedHDR; return OETF(reducedHDR, maxNits); } float3 HDRMappingFromRec709(float3 Rec709Input, float paperWhite, float minNits, float maxNits, int reductionMode, float hueShift, bool skipOETF = false) { float3 outputSpaceInput = RotateRec709ToOutputSpace(Rec709Input); float3 reducedHDR = PerformRangeReduction(outputSpaceInput * paperWhite, minNits, maxNits, reductionMode, hueShift); if (skipOETF) return reducedHDR; return OETF(reducedHDR, maxNits); } float3 HDRMappingACES(float3 aces, float hdrBoost, float minNits, float maxNits, int reductionMode, bool skipOETF = false) { aces = (aces * hdrBoost * 0.01f); float3 oces = RRT(aces); float3 AP1ODT = 0; // This is a static branch. if (reductionMode == HDRRANGEREDUCTION_ACES1000NITS) { AP1ODT = ODT_1000nits_ToAP1(oces); } else if (reductionMode == HDRRANGEREDUCTION_ACES2000NITS) { AP1ODT = ODT_2000nits_ToAP1(oces); } else if (reductionMode == HDRRANGEREDUCTION_ACES4000NITS) { AP1ODT = ODT_4000nits_ToAP1(oces); } float3 linearODT = 0; if (_HDRColorspace == HDRCOLORSPACE_REC2020) { const float3x3 AP1_2_Rec2020 = mul(XYZ_2_REC2020_MAT, mul(D60_2_D65_CAT, AP1_2_XYZ_MAT)); linearODT = mul(AP1_2_Rec2020, AP1ODT); } else if (_HDRColorspace == HDRCOLORSPACE_P3D65) { const float3x3 API1_2_P3D65 = mul(XYZ_2_P3D65_MAT, mul(D60_2_D65_CAT, AP1_2_XYZ_MAT)); linearODT = mul(API1_2_P3D65, AP1ODT); } else // HDRCOLORSPACE_REC709 { const float3x3 AP1_2_Rec709 = mul(XYZ_2_REC709_MAT, mul(D60_2_D65_CAT, AP1_2_XYZ_MAT)); linearODT = mul(AP1_2_Rec709, AP1ODT); } if (skipOETF) return linearODT; return OETF(linearODT, maxNits); } // -------------------------------------------------------------------------------------------- // -------------------------------- // UI Related functions // -------------------------------- float3 ProcessUIForHDR(float3 uiSample, float paperWhite, float maxNits) { uiSample.rgb = RotateRec709ToOutputSpace(uiSample.rgb); uiSample.rgb *= paperWhite; return uiSample.rgb; } float3 SceneUIComposition(float4 uiSample, float3 sceneColor, float paperWhite, float maxNits) { // Undo the pre multiply. uiSample.rgb = uiSample.rgb / (uiSample.a == 0.0f ? 1.0 : uiSample.a); uiSample.rgb = ProcessUIForHDR(uiSample.rgb, paperWhite, maxNits); return uiSample.rgb * uiSample.a + sceneColor.rgb * (1.0f - uiSample.a); } // -------------------------------------------------------------------------------------------- #endif