2017-10-26 02:17:54 +03:00
|
|
|
/*
|
2023-01-14 21:05:12 +03:00
|
|
|
* Copyright 2011-2023 Branimir Karadzic. All rights reserved.
|
2022-01-15 22:59:06 +03:00
|
|
|
* License: https://github.com/bkaradzic/bgfx/blob/master/LICENSE
|
2017-10-26 02:17:54 +03:00
|
|
|
*/
|
|
|
|
|
|
|
|
#include <bgfx_shader.sh>
|
|
|
|
|
2018-12-30 10:10:28 +03:00
|
|
|
uniform vec4 u_params0;
|
|
|
|
#define u_textureLod u_params0.x
|
|
|
|
#define u_textureLayer u_params0.y
|
|
|
|
#define u_inLinear u_params0.z
|
|
|
|
#define u_ev u_params0.w
|
|
|
|
|
|
|
|
uniform vec4 u_params1;
|
|
|
|
#define u_outputFormat u_params1.x
|
|
|
|
#define u_sdrWhiteNits u_params1.y
|
2017-10-26 02:17:54 +03:00
|
|
|
|
2017-12-13 04:31:22 +03:00
|
|
|
vec3 toLinear(vec3 _rgb)
|
|
|
|
{
|
|
|
|
return pow(abs(_rgb), vec3_splat(2.2) );
|
|
|
|
}
|
|
|
|
|
|
|
|
vec3 toGamma(vec3 _rgb)
|
|
|
|
{
|
|
|
|
return pow(abs(_rgb), vec3_splat(1.0/2.2) );
|
|
|
|
}
|
|
|
|
|
2018-12-30 10:10:28 +03:00
|
|
|
vec3 applyExposure(vec3 _rgb)
|
|
|
|
{
|
|
|
|
vec3 rgb = mix(toLinear(_rgb.xyz), _rgb.xyz, u_inLinear);
|
|
|
|
return (rgb * pow(2.0, u_ev) );
|
|
|
|
}
|
|
|
|
|
2017-10-26 02:17:54 +03:00
|
|
|
vec4 toEv(vec4 _color)
|
|
|
|
{
|
2018-12-30 10:10:28 +03:00
|
|
|
return vec4(toGamma(applyExposure(_color.xyz) ), _color.w);
|
|
|
|
}
|
|
|
|
|
|
|
|
float toSrgbGamma(float _val)
|
|
|
|
{
|
|
|
|
if (_val <= 0.0031308)
|
|
|
|
{
|
|
|
|
return 12.92 * _val;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
return 1.055 * pow(_val, (1.0/2.4) ) - 0.055;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
vec3 toSrgbGamma(vec3 _rgb)
|
|
|
|
{
|
|
|
|
_rgb.x = toSrgbGamma(_rgb.x);
|
|
|
|
_rgb.y = toSrgbGamma(_rgb.y);
|
|
|
|
_rgb.z = toSrgbGamma(_rgb.z);
|
|
|
|
return _rgb;
|
|
|
|
}
|
|
|
|
|
|
|
|
vec3 toXyzFromSrgb(vec3 _rgb)
|
|
|
|
{
|
|
|
|
mat3 toXYZ = mat3(
|
|
|
|
0.4125564, 0.3575761, 0.1804375,
|
|
|
|
0.2126729, 0.7151522, 0.0721750,
|
|
|
|
0.0193339, 0.1191920, 0.9503041
|
|
|
|
);
|
|
|
|
return mul(toXYZ, _rgb);
|
|
|
|
}
|
|
|
|
|
|
|
|
vec3 toRec2020FromXyz(vec3 _xyz)
|
|
|
|
{
|
|
|
|
mat3 toRec2020 = mat3(
|
|
|
|
1.7166512, -0.3556708, -0.2533663,
|
|
|
|
-0.6666844, 1.6164812, 0.0157685,
|
|
|
|
0.0176399, -0.0427706, 0.9421031
|
|
|
|
);
|
|
|
|
return mul(toRec2020, _xyz);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
vec3 toPqOetf(vec3 _color)
|
|
|
|
{
|
|
|
|
// reference PQ OETF will yield reference OOTF when
|
|
|
|
// displayed on a reference monitor employing EOTF
|
|
|
|
|
|
|
|
float m1 = 0.1593017578125;
|
|
|
|
float m2 = 78.84375;
|
|
|
|
float c1 = 0.8359375;
|
|
|
|
float c2 = 18.8515625;
|
|
|
|
float c3 = 18.6875;
|
|
|
|
|
|
|
|
vec3 Ym1 = pow(_color.xyz * (1.0/10000.0), vec3_splat(m1) );
|
|
|
|
_color = pow((c1 + c2*Ym1) / (vec3_splat(1.0) + c3*Ym1), vec3_splat(m2) );
|
|
|
|
|
|
|
|
return _color;
|
|
|
|
}
|
|
|
|
|
|
|
|
vec4 toOutput(vec4 _color, float _outputFormat, float _sdrWhiteNits)
|
|
|
|
{
|
|
|
|
// assumed that _color is linear with sRGB/rec709 primaries
|
|
|
|
// and 1.0 is SDR white point
|
|
|
|
|
|
|
|
vec3 outColor = vec3_splat(0.0);
|
|
|
|
|
|
|
|
if (_outputFormat < 0.5)
|
|
|
|
{
|
|
|
|
// output == 0 -> sRGB/rec709, apply gamma
|
|
|
|
// values over 1.0 will saturate
|
|
|
|
outColor = toSrgbGamma(saturate(_color.xyz));
|
|
|
|
}
|
|
|
|
else if (_outputFormat < 1.5)
|
|
|
|
{
|
|
|
|
// output == 1 -> scRGB, remains linear.
|
|
|
|
// values over 1.0 will appear as HDR
|
|
|
|
outColor = _color.xyz;
|
|
|
|
}
|
|
|
|
else if (_outputFormat < 2.5)
|
|
|
|
{
|
|
|
|
// output == 2 -> PQ
|
|
|
|
|
|
|
|
// change primaries from sRGB/rec709 to rec2020
|
|
|
|
vec3 _xyz = toXyzFromSrgb(_color.xyz);
|
|
|
|
outColor = toRec2020FromXyz(_xyz);
|
|
|
|
|
|
|
|
// if 1.0 is SDR white, should map to 80 nits
|
|
|
|
// but that could lead to dim results as SDR
|
|
|
|
// monitors tend to be brighter than standard
|
|
|
|
outColor = toPqOetf(outColor * _sdrWhiteNits);
|
|
|
|
}
|
|
|
|
|
|
|
|
return vec4(outColor, _color.w);
|
2017-10-26 02:17:54 +03:00
|
|
|
}
|