From f7d76ebf1d71cc827e780fe9e4bcfae6be332493 Mon Sep 17 00:00:00 2001 From: Stuart Kenny Date: Mon, 6 Jun 2022 14:33:28 +0100 Subject: [PATCH] GS: Add lottes crt to present shader. --- bin/resources/shaders/dx11/present.fx | 283 +++++++++++++++++- bin/resources/shaders/opengl/present.glsl | 314 ++++++++++++++++++-- bin/resources/shaders/vulkan/present.glsl | 281 +++++++++++++++++- pcsx2-qt/Settings/GraphicsSettingsWidget.ui | 5 + pcsx2/GS/GS.cpp | 1 + pcsx2/GS/Renderers/Common/GSDevice.cpp | 1 + pcsx2/GS/Renderers/Common/GSDevice.h | 1 + pcsx2/GS/Renderers/Common/GSRenderer.cpp | 4 +- pcsx2/GS/Renderers/Metal/present.metal | 257 ++++++++++++++++ 9 files changed, 1086 insertions(+), 61 deletions(-) diff --git a/bin/resources/shaders/dx11/present.fx b/bin/resources/shaders/dx11/present.fx index 2456fff3a2..043097d240 100644 --- a/bin/resources/shaders/dx11/present.fx +++ b/bin/resources/shaders/dx11/present.fx @@ -66,7 +66,7 @@ VS_OUTPUT vs_main(VS_INPUT input) PS_OUTPUT ps_copy(PS_INPUT input) { PS_OUTPUT output; - + output.c = sample_c(input.t); return output; @@ -74,24 +74,24 @@ PS_OUTPUT ps_copy(PS_INPUT input) float4 ps_crt(PS_INPUT input, int i) { - float4 mask[4] = - { - float4(1, 0, 0, 0), - float4(0, 1, 0, 0), - float4(0, 0, 1, 0), - float4(1, 1, 1, 0) - }; - + float4 mask[4] = + { + float4(1, 0, 0, 0), + float4(0, 1, 0, 0), + float4(0, 0, 1, 0), + float4(1, 1, 1, 0) + }; + return sample_c(input.t) * saturate(mask[i] + 0.5f); } float4 ps_scanlines(PS_INPUT input, int i) { float4 mask[2] = - { - float4(1, 1, 1, 0), - float4(0, 0, 0, 0) - }; + { + float4(1, 1, 1, 0), + float4(0, 0, 0, 0) + }; return sample_c(input.t) * saturate(mask[i] + 0.5f); } @@ -99,7 +99,7 @@ float4 ps_scanlines(PS_INPUT input, int i) PS_OUTPUT ps_filter_scanlines(PS_INPUT input) { PS_OUTPUT output; - + uint4 p = (uint4)input.p; output.c = ps_scanlines(input, p.y % 2); @@ -124,7 +124,7 @@ PS_OUTPUT ps_filter_triangular(PS_INPUT input) uint4 p = (uint4)input.p; - // output.c = ps_crt(input, ((p.x + (p.y & 1) * 3) >> 1) % 3); + // output.c = ps_crt(input, ((p.x + (p.y & 1) * 3) >> 1) % 3); output.c = ps_crt(input, ((p.x + ((p.y >> 1) & 1) * 3) >> 1) % 3); return output; @@ -143,4 +143,257 @@ PS_OUTPUT ps_filter_complex(PS_INPUT input) // triangular return output; } +//Lottes CRT +#define MaskingType 4 //[1|2|3|4] The type of CRT shadow masking used. 1: compressed TV style, 2: Aperture-grille, 3: Stretched VGA style, 4: VGA style. +#define ScanBrightness -8.00 //[-16.0 to 1.0] The overall brightness of the scanline effect. Lower for darker, higher for brighter. +#define FilterCRTAmount -1.00 //[-4.0 to 1.0] The amount of filtering used, to replicate the TV CRT look. Lower for less, higher for more. +#define HorizontalWarp 0.00 //[0.0 to 0.1] The distortion warping effect for the horizontal (x) axis of the screen. Use small increments. +#define VerticalWarp 0.00 //[0.0 to 0.1] The distortion warping effect for the verticle (y) axis of the screen. Use small increments. +#define MaskAmountDark 0.80 //[0.0 to 1.0] The value of the dark masking line effect used. Lower for darker lower end masking, higher for brighter. +#define MaskAmountLight 1.50 //[0.0 to 2.0] The value of the light masking line effect used. Lower for darker higher end masking, higher for brighter. +#define ResolutionScale 2.00 //[0.1 to 4.0] The scale of the image resolution. Lowering this can give off a nice retro TV look. Raising it can clear up the image. +#define MaskResolutionScale 0.80 //[0.1 to 2.0] The scale of the CRT mask resolution. Use this for balancing the scanline mask scale for difference resolution scaling. +#define UseShadowMask 1 //[0 or 1] Enables, or disables the use of the CRT shadow mask. 0 is disabled, 1 is enabled. + +float ToLinear1(float c) +{ + c = saturate(c); + return c <= 0.04045 ? c / 12.92 : pow((c + 0.055) / 1.055, 2.4); +} + +float3 ToLinear(float3 c) +{ + return float3(ToLinear1(c.r), ToLinear1(c.g), ToLinear1(c.b)); +} + +float ToSrgb1(float c) +{ + c = saturate(c); + return c < 0.0031308 ? c * 12.92 : 1.055 * pow(c, 0.41666) - 0.055; +} + +float3 ToSrgb(float3 c) +{ + return float3(ToSrgb1(c.r), ToSrgb1(c.g), ToSrgb1(c.b)); +} + +float3 Fetch(float2 pos, float2 off) +{ + float2 screenSize = u_source_resolution; + float2 res = (screenSize * ResolutionScale); + pos = round(pos * res + off) / res; + if (max(abs(pos.x - 0.5), abs(pos.y - 0.5)) > 0.5) + { + return float3(0.0, 0.0, 0.0); + } + else + { + return ToLinear(Texture.Sample(TextureSampler, pos.xy).rgb); + } +} + +float2 Dist(float2 pos) +{ + float2 crtRes = u_rcp_target_resolution; + float2 res = (crtRes * MaskResolutionScale); + pos = (pos * res); + + return -((pos - floor(pos)) - float2(0.5, 0.5)); +} + +float Gaus(float pos, float scale) +{ + return exp2(scale * pos * pos); +} + +float3 Horz3(float2 pos, float off) +{ + float3 b = Fetch(pos, float2(-1.0, off)); + float3 c = Fetch(pos, float2(0.0, off)); + float3 d = Fetch(pos, float2(1.0, off)); + float dst = Dist(pos).x; + + // Convert distance to weight. + float scale = FilterCRTAmount; + float wb = Gaus(dst - 1.0, scale); + float wc = Gaus(dst + 0.0, scale); + float wd = Gaus(dst + 1.0, scale); + + return (b * wb) + (c * wc) + (d * wd) / (wb + wc + wd); +} + +float3 Horz5(float2 pos, float off) +{ + float3 a = Fetch(pos, float2(-2.0, off)); + float3 b = Fetch(pos, float2(-1.0, off)); + float3 c = Fetch(pos, float2(0.0, off)); + float3 d = Fetch(pos, float2(1.0, off)); + float3 e = Fetch(pos, float2(2.0, off)); + float dst = Dist(pos).x; + + // Convert distance to weight. + float scale = FilterCRTAmount; + + float wa = Gaus(dst - 2.0, scale); + float wb = Gaus(dst - 1.0, scale); + float wc = Gaus(dst + 0.0, scale); + float wd = Gaus(dst + 1.0, scale); + float we = Gaus(dst + 2.0, scale); + + return (a * wa) + (b * wb) + (c * wc) + (d * wd) + (e * we) / (wa + wb + wc + wd + we); +} + +// Return scanline weight. +float Scan(float2 pos, float off) +{ + float dst = Dist(pos).y; + return Gaus(dst + off, ScanBrightness); +} + +float3 Tri(float2 pos) +{ + float3 a = Horz3(pos, -1.0); + float3 b = Horz5(pos, 0.0); + float3 c = Horz3(pos, 1.0); + + float wa = Scan(pos, -1.0); + float wb = Scan(pos, 0.0); + float wc = Scan(pos, 1.0); + + return (a * wa) + (b * wb) + (c * wc); +} + +float2 Warp(float2 pos) +{ + pos = pos * 2.0 - 1.0; + pos *= float2(1.0 + (pos.y * pos.y) * HorizontalWarp, 1.0 + (pos.x * pos.x) * VerticalWarp); + return pos * 0.5 + 0.5; +} + +float3 Mask(float2 pos) +{ +#if MaskingType == 1 + // Very compressed TV style shadow mask. + float lines = MaskAmountLight; + float odd = 0.0; + + if (frac(pos.x / 6.0) < 0.5) + { + odd = 1.0; + } + if (frac((pos.y + odd) / 2.0) < 0.5) + { + lines = MaskAmountDark; + } + pos.x = frac(pos.x / 3.0); + float3 mask = float3(MaskAmountDark, MaskAmountDark, MaskAmountDark); + + if (pos.x < 0.333) + { + mask.r = MaskAmountLight; + } + else if (pos.x < 0.666) + { + mask.g = MaskAmountLight; + } + else + { + mask.b = MaskAmountLight; + } + + mask *= lines; + + return mask; + +#elif MaskingType == 2 + // Aperture-grille. + pos.x = frac(pos.x / 3.0); + float3 mask = float3(MaskAmountDark, MaskAmountDark, MaskAmountDark); + + if (pos.x < 0.333) + { + mask.r = MaskAmountLight; + } + else if (pos.x < 0.666) + { + mask.g = MaskAmountLight; + } + else + { + mask.b = MaskAmountLight; + } + + return mask; + +#elif MaskingType == 3 + // Stretched VGA style shadow mask (same as prior shaders). + pos.x += pos.y * 3.0; + float3 mask = float3(MaskAmountDark, MaskAmountDark, MaskAmountDark); + pos.x = frac(pos.x / 6.0); + + if (pos.x < 0.333) + { + mask.r = MaskAmountLight; + } + else if (pos.x < 0.666) + { + mask.g = MaskAmountLight; + } + else + { + mask.b = MaskAmountLight; + } + + return mask; + +#else + // VGA style shadow mask. + pos.xy = floor(pos.xy * float2(1.0, 0.5)); + pos.x += pos.y * 3.0; + + float3 mask = float3(MaskAmountDark, MaskAmountDark, MaskAmountDark); + pos.x = frac(pos.x / 6.0); + + if (pos.x < 0.333) + { + mask.r = MaskAmountLight; + } + else if (pos.x < 0.666) + { + mask.g = MaskAmountLight; + } + else + { + mask.b = MaskAmountLight; + } + return mask; +#endif +} + +float4 LottesCRTPass(float4 fragcoord) +{ + fragcoord -= u_target_rect; + float2 inSize = u_target_resolution - (2 * u_target_rect.xy); + float4 color; + float2 pos = Warp(fragcoord.xy / inSize); + +#if UseShadowMask == 0 + color.rgb = Tri(pos); +#else + color.rgb = Tri(pos) * Mask(fragcoord.xy); +#endif + color.rgb = ToSrgb(color.rgb); + color.a = 1.0; + + return color; +} + +PS_OUTPUT ps_filter_lottes(PS_INPUT input) +{ + PS_OUTPUT output; + output.c = LottesCRTPass(input.p); + + return output; +} + #endif diff --git a/bin/resources/shaders/opengl/present.glsl b/bin/resources/shaders/opengl/present.glsl index 911dd01f4a..3e2b07ceec 100644 --- a/bin/resources/shaders/opengl/present.glsl +++ b/bin/resources/shaders/opengl/present.glsl @@ -21,10 +21,10 @@ out vec4 PSin_c; void vs_main() { - PSin_p = vec4(POSITION, 0.5f, 1.0f); - PSin_t = TEXCOORD0; - PSin_c = COLOR; - gl_Position = vec4(POSITION, 0.5f, 1.0f); // NOTE I don't know if it is possible to merge POSITION_OUT and gl_Position + PSin_p = vec4(POSITION, 0.5f, 1.0f); + PSin_t = TEXCOORD0; + PSin_c = COLOR; + gl_Position = vec4(POSITION, 0.5f, 1.0f); // NOTE I don't know if it is possible to merge POSITION_OUT and gl_Position } #endif @@ -50,69 +50,66 @@ layout(location = 0) out vec4 SV_Target0; vec4 sample_c() { - return texture(TextureSampler, PSin_t); + return texture(TextureSampler, PSin_t); } vec4 ps_crt(uint i) { - vec4 mask[4] = vec4[4] - ( - vec4(1, 0, 0, 0), - vec4(0, 1, 0, 0), - vec4(0, 0, 1, 0), - vec4(1, 1, 1, 0) - ); - return sample_c() * clamp((mask[i] + 0.5f), 0.0f, 1.0f); + vec4 mask[4] = vec4[4]( + vec4(1, 0, 0, 0), + vec4(0, 1, 0, 0), + vec4(0, 0, 1, 0), + vec4(1, 1, 1, 0)); + return sample_c() * clamp((mask[i] + 0.5f), 0.0f, 1.0f); } #ifdef ps_copy void ps_copy() { - SV_Target0 = sample_c(); + SV_Target0 = sample_c(); } #endif #ifdef ps_filter_scanlines vec4 ps_scanlines(uint i) { - vec4 mask[2] = - { - vec4(1, 1, 1, 0), - vec4(0, 0, 0, 0) - }; + vec4 mask[2] = + { + vec4(1, 1, 1, 0), + vec4(0, 0, 0, 0)}; - return sample_c() * clamp((mask[i] + 0.5f), 0.0f, 1.0f); + return sample_c() * clamp((mask[i] + 0.5f), 0.0f, 1.0f); } void ps_filter_scanlines() // scanlines { - highp uvec4 p = uvec4(gl_FragCoord); + highp uvec4 p = uvec4(gl_FragCoord); - vec4 c = ps_scanlines(p.y % 2u); + vec4 c = ps_scanlines(p.y % 2u); - SV_Target0 = c; + SV_Target0 = c; } #endif #ifdef ps_filter_diagonal void ps_filter_diagonal() // diagonal { - highp uvec4 p = uvec4(gl_FragCoord); + highp uvec4 p = uvec4(gl_FragCoord); - vec4 c = ps_crt((p.x + (p.y % 3u)) % 3u); + vec4 c = ps_crt((p.x + (p.y % 3u)) % 3u); - SV_Target0 = c; + SV_Target0 = c; } #endif #ifdef ps_filter_triangular void ps_filter_triangular() // triangular { - highp uvec4 p = uvec4(gl_FragCoord); + highp uvec4 p = uvec4(gl_FragCoord); - vec4 c = ps_crt(((p.x + ((p.y >> 1u) & 1u) * 3u) >> 1u) % 3u); + vec4 c = ps_crt(((p.x + ((p.y >> 1u) & 1u) * 3u) >> 1u) % 3u); - SV_Target0 = c; + SV_Target0 = c; } #endif @@ -124,8 +121,265 @@ void ps_filter_complex() float factor = (0.9f - 0.4f * cos(2.0f * PI * PSin_t.y * texdim.y)); vec4 c = factor * texture(TextureSampler, vec2(PSin_t.x, (floor(PSin_t.y * texdim.y) + 0.5f) / texdim.y)); - SV_Target0 = c; + SV_Target0 = c; } #endif +#ifdef ps_filter_lottes + +#define MaskingType 4 //[1|2|3|4] The type of CRT shadow masking used. 1: compressed TV style, 2: Aperture-grille, 3: Stretched VGA style, 4: VGA style. +#define ScanBrightness -8.00 //[-16.0 to 1.0] The overall brightness of the scanline effect. Lower for darker, higher for brighter. +#define FilterCRTAmount -1.00 //[-4.0 to 1.0] The amount of filtering used, to replicate the TV CRT look. Lower for less, higher for more. +#define HorizontalWarp 0.00 //[0.0 to 0.1] The distortion warping effect for the horizontal (x) axis of the screen. Use small increments. +#define VerticalWarp 0.00 //[0.0 to 0.1] The distortion warping effect for the verticle (y) axis of the screen. Use small increments. +#define MaskAmountDark 0.80 //[0.0 to 1.0] The value of the dark masking line effect used. Lower for darker lower end masking, higher for brighter. +#define MaskAmountLight 1.50 //[0.0 to 2.0] The value of the light masking line effect used. Lower for darker higher end masking, higher for brighter. +#define ResolutionScale 2.00 //[0.1 to 4.0] The scale of the image resolution. Lowering this can give off a nice retro TV look. Raising it can clear up the image. +#define MaskResolutionScale 0.80 //[0.1 to 2.0] The scale of the CRT mask resolution. Use this for balancing the scanline mask scale for difference resolution scaling. +#define UseShadowMask 1 //[0 or 1] Enables, or disables the use of the CRT shadow mask. 0 is disabled, 1 is enabled. + +#define saturate(x) clamp(x, 0.0, 1.0) + +float ToLinear1(float c) +{ + c = saturate(c); + return c <= 0.04045 ? c / 12.92 : pow((c + 0.055) / 1.055, 2.4); +} + +vec3 ToLinear(vec3 c) +{ + return vec3(ToLinear1(c.r), ToLinear1(c.g), ToLinear1(c.b)); +} + +float ToSrgb1(float c) +{ + c = saturate(c); + return c < 0.0031308 ? c * 12.92 : 1.055 * pow(c, 0.41666) - 0.055; +} + +vec3 ToSrgb(vec3 c) +{ + return vec3(ToSrgb1(c.r), ToSrgb1(c.g), ToSrgb1(c.b)); +} + +vec3 Fetch(vec2 pos, vec2 off) +{ + vec2 screenSize = u_source_resolution; + vec2 res = (screenSize * ResolutionScale); + pos = round(pos * res + off) / res; + if (max(abs(pos.x - 0.5), abs(pos.y - 0.5)) > 0.5) + { + return vec3(0.0, 0.0, 0.0); + } + else + { + return ToLinear(texture(TextureSampler, pos.xy).rgb); + } +} + +vec2 Dist(vec2 pos) +{ + vec2 crtRes = u_rcp_target_resolution; + vec2 res = (crtRes * MaskResolutionScale); + pos = (pos * res); + + return -((pos - floor(pos)) - vec2(0.5, 0.5)); +} + +float Gaus(float pos, float scale) +{ + return exp2(scale * pos * pos); +} + +vec3 Horz3(vec2 pos, float off) +{ + vec3 b = Fetch(pos, vec2(-1.0, off)); + vec3 c = Fetch(pos, vec2(0.0, off)); + vec3 d = Fetch(pos, vec2(1.0, off)); + float dst = Dist(pos).x; + + // Convert distance to weight. + float scale = FilterCRTAmount; + float wb = Gaus(dst - 1.0, scale); + float wc = Gaus(dst + 0.0, scale); + float wd = Gaus(dst + 1.0, scale); + + return (b * wb) + (c * wc) + (d * wd) / (wb + wc + wd); +} + +vec3 Horz5(vec2 pos, float off) +{ + vec3 a = Fetch(pos, vec2(-2.0, off)); + vec3 b = Fetch(pos, vec2(-1.0, off)); + vec3 c = Fetch(pos, vec2(0.0, off)); + vec3 d = Fetch(pos, vec2(1.0, off)); + vec3 e = Fetch(pos, vec2(2.0, off)); + float dst = Dist(pos).x; + + // Convert distance to weight. + float scale = FilterCRTAmount; + + float wa = Gaus(dst - 2.0, scale); + float wb = Gaus(dst - 1.0, scale); + float wc = Gaus(dst + 0.0, scale); + float wd = Gaus(dst + 1.0, scale); + float we = Gaus(dst + 2.0, scale); + + return (a * wa) + (b * wb) + (c * wc) + (d * wd) + (e * we) / (wa + wb + wc + wd + we); +} + +// Return scanline weight. +float Scan(vec2 pos, float off) +{ + float dst = Dist(pos).y; + return Gaus(dst + off, ScanBrightness); +} + +vec3 Tri(vec2 pos) +{ + vec3 a = Horz3(pos, -1.0); + vec3 b = Horz5(pos, 0.0); + vec3 c = Horz3(pos, 1.0); + + float wa = Scan(pos, -1.0); + float wb = Scan(pos, 0.0); + float wc = Scan(pos, 1.0); + + return (a * wa) + (b * wb) + (c * wc); +} + +vec2 Warp(vec2 pos) +{ + pos = pos * 2.0 - 1.0; + pos *= vec2(1.0 + (pos.y * pos.y) * HorizontalWarp, 1.0 + (pos.x * pos.x) * VerticalWarp); + return pos * 0.5 + 0.5; +} + +vec3 Mask(vec2 pos) +{ +#if MaskingType == 1 + // Very compressed TV style shadow mask. + float lines = MaskAmountLight; + float odd = 0.0; + + if (fract(pos.x / 6.0) < 0.5) + { + odd = 1.0; + } + if (fract((pos.y + odd) / 2.0) < 0.5) + { + lines = MaskAmountDark; + } + pos.x = fract(pos.x / 3.0); + vec3 mask = vec3(MaskAmountDark, MaskAmountDark, MaskAmountDark); + + if (pos.x < 0.333) + { + mask.r = MaskAmountLight; + } + else if (pos.x < 0.666) + { + mask.g = MaskAmountLight; + } + else + { + mask.b = MaskAmountLight; + } + + mask *= lines; + + return mask; + +#elif MaskingType == 2 + // Aperture-grille. + pos.x = fract(pos.x / 3.0); + vec3 mask = vec3(MaskAmountDark, MaskAmountDark, MaskAmountDark); + + if (pos.x < 0.333) + { + mask.r = MaskAmountLight; + } + else if (pos.x < 0.666) + { + mask.g = MaskAmountLight; + } + else + { + mask.b = MaskAmountLight; + } + + return mask; + +#elif MaskingType == 3 + // Stretched VGA style shadow mask (same as prior shaders). + pos.x += pos.y * 3.0; + vec3 mask = vec3(MaskAmountDark, MaskAmountDark, MaskAmountDark); + pos.x = fract(pos.x / 6.0); + + if (pos.x < 0.333) + { + mask.r = MaskAmountLight; + } + else if (pos.x < 0.666) + { + mask.g = MaskAmountLight; + } + else + { + mask.b = MaskAmountLight; + } + + return mask; + +#else + // VGA style shadow mask. + pos.xy = floor(pos.xy * vec2(1.0, 0.5)); + pos.x += pos.y * 3.0; + + vec3 mask = vec3(MaskAmountDark, MaskAmountDark, MaskAmountDark); + pos.x = fract(pos.x / 6.0); + + if (pos.x < 0.333) + { + mask.r = MaskAmountLight; + } + else if (pos.x < 0.666) + { + mask.g = MaskAmountLight; + } + else + { + mask.b = MaskAmountLight; + } + return mask; +#endif +} + +vec4 LottesCRTPass() +{ + //flipped y axis in opengl + vec2 fragcoord = vec2(gl_FragCoord.x, u_target_resolution.y - gl_FragCoord.y) - u_target_rect.xy; + vec4 color; + vec2 inSize = u_target_resolution - (2 * u_target_rect.xy); + + vec2 pos = Warp(fragcoord.xy / inSize); + +#if UseShadowMask == 0 + color.rgb = Tri(pos); +#else + color.rgb = Tri(pos) * Mask(fragcoord.xy); +#endif + color.rgb = ToSrgb(color.rgb); + color.a = 1.0; + + return color; +} + +void ps_filter_lottes() +{ + SV_Target0 = LottesCRTPass(); +} + +#endif + #endif diff --git a/bin/resources/shaders/vulkan/present.glsl b/bin/resources/shaders/vulkan/present.glsl index 68c1d438e5..33cd93d690 100644 --- a/bin/resources/shaders/vulkan/present.glsl +++ b/bin/resources/shaders/vulkan/present.glsl @@ -46,25 +46,22 @@ vec4 sample_c(vec2 uv) vec4 ps_crt(uint i) { - vec4 mask[4] = vec4[4] - ( - vec4(1, 0, 0, 0), - vec4(0, 1, 0, 0), - vec4(0, 0, 1, 0), - vec4(1, 1, 1, 0) - ); - return sample_c(v_tex) * clamp((mask[i] + 0.5f), 0.0f, 1.0f); + vec4 mask[4] = vec4[4]( + vec4(1, 0, 0, 0), + vec4(0, 1, 0, 0), + vec4(0, 0, 1, 0), + vec4(1, 1, 1, 0)); + return sample_c(v_tex) * clamp((mask[i] + 0.5f), 0.0f, 1.0f); } vec4 ps_scanlines(uint i) { - vec4 mask[2] = + vec4 mask[2] = { - vec4(1, 1, 1, 0), - vec4(0, 0, 0, 0) - }; + vec4(1, 1, 1, 0), + vec4(0, 0, 0, 0)}; - return sample_c(v_tex) * clamp((mask[i] + 0.5f), 0.0f, 1.0f); + return sample_c(v_tex) * clamp((mask[i] + 0.5f), 0.0f, 1.0f); } #ifdef ps_copy @@ -96,7 +93,7 @@ void ps_filter_triangular() // triangular { uvec4 p = uvec4(gl_FragCoord); - // output.c = ps_crt(input, ((p.x + (p.y & 1) * 3) >> 1) % 3); + // output.c = ps_crt(input, ((p.x + (p.y & 1) * 3) >> 1) % 3); o_col0 = ps_crt(((p.x + ((p.y >> 1) & 1) * 3) >> 1) % 3); } #endif @@ -111,4 +108,260 @@ void ps_filter_complex() // triangular } #endif +#ifdef ps_filter_lottes + +#define MaskingType 4 //[1|2|3|4] The type of CRT shadow masking used. 1: compressed TV style, 2: Aperture-grille, 3: Stretched VGA style, 4: VGA style. +#define ScanBrightness -8.00 //[-16.0 to 1.0] The overall brightness of the scanline effect. Lower for darker, higher for brighter. +#define FilterCRTAmount -1.00 //[-4.0 to 1.0] The amount of filtering used, to replicate the TV CRT look. Lower for less, higher for more. +#define HorizontalWarp 0.00 //[0.0 to 0.1] The distortion warping effect for the horizontal (x) axis of the screen. Use small increments. +#define VerticalWarp 0.00 //[0.0 to 0.1] The distortion warping effect for the verticle (y) axis of the screen. Use small increments. +#define MaskAmountDark 0.80 //[0.0 to 1.0] The value of the dark masking line effect used. Lower for darker lower end masking, higher for brighter. +#define MaskAmountLight 1.50 //[0.0 to 2.0] The value of the light masking line effect used. Lower for darker higher end masking, higher for brighter. +#define ResolutionScale 2.00 //[0.1 to 4.0] The scale of the image resolution. Lowering this can give off a nice retro TV look. Raising it can clear up the image. +#define MaskResolutionScale 0.80 //[0.1 to 2.0] The scale of the CRT mask resolution. Use this for balancing the scanline mask scale for difference resolution scaling. +#define UseShadowMask 1 //[0 or 1] Enables, or disables the use of the CRT shadow mask. 0 is disabled, 1 is enabled. + +#define saturate(x) clamp(x, 0.0, 1.0) + +float ToLinear1(float c) +{ + c = saturate(c); + return c <= 0.04045 ? c / 12.92 : pow((c + 0.055) / 1.055, 2.4); +} + +vec3 ToLinear(vec3 c) +{ + return vec3(ToLinear1(c.r), ToLinear1(c.g), ToLinear1(c.b)); +} + +float ToSrgb1(float c) +{ + c = saturate(c); + return c < 0.0031308 ? c * 12.92 : 1.055 * pow(c, 0.41666) - 0.055; +} + +vec3 ToSrgb(vec3 c) +{ + return vec3(ToSrgb1(c.r), ToSrgb1(c.g), ToSrgb1(c.b)); +} + +vec3 Fetch(vec2 pos, vec2 off) +{ + vec2 screenSize = u_source_resolution; + vec2 res = (screenSize * ResolutionScale); + pos = round(pos * res + off) / res; + if (max(abs(pos.x - 0.5), abs(pos.y - 0.5)) > 0.5) + { + return vec3(0.0, 0.0, 0.0); + } + else + { + return ToLinear(texture(samp0, pos.xy).rgb); + } +} + +vec2 Dist(vec2 pos) +{ + vec2 crtRes = u_rcp_target_resolution; + vec2 res = (crtRes * MaskResolutionScale); + pos = (pos * res); + + return -((pos - floor(pos)) - vec2(0.5, 0.5)); +} + +float Gaus(float pos, float scale) +{ + return exp2(scale * pos * pos); +} + +vec3 Horz3(vec2 pos, float off) +{ + vec3 b = Fetch(pos, vec2(-1.0, off)); + vec3 c = Fetch(pos, vec2(0.0, off)); + vec3 d = Fetch(pos, vec2(1.0, off)); + float dst = Dist(pos).x; + + // Convert distance to weight. + float scale = FilterCRTAmount; + float wb = Gaus(dst - 1.0, scale); + float wc = Gaus(dst + 0.0, scale); + float wd = Gaus(dst + 1.0, scale); + + return (b * wb) + (c * wc) + (d * wd) / (wb + wc + wd); +} + +vec3 Horz5(vec2 pos, float off) +{ + vec3 a = Fetch(pos, vec2(-2.0, off)); + vec3 b = Fetch(pos, vec2(-1.0, off)); + vec3 c = Fetch(pos, vec2(0.0, off)); + vec3 d = Fetch(pos, vec2(1.0, off)); + vec3 e = Fetch(pos, vec2(2.0, off)); + float dst = Dist(pos).x; + + // Convert distance to weight. + float scale = FilterCRTAmount; + + float wa = Gaus(dst - 2.0, scale); + float wb = Gaus(dst - 1.0, scale); + float wc = Gaus(dst + 0.0, scale); + float wd = Gaus(dst + 1.0, scale); + float we = Gaus(dst + 2.0, scale); + + return (a * wa) + (b * wb) + (c * wc) + (d * wd) + (e * we) / (wa + wb + wc + wd + we); +} + +// Return scanline weight. +float Scan(vec2 pos, float off) +{ + float dst = Dist(pos).y; + return Gaus(dst + off, ScanBrightness); +} + +vec3 Tri(vec2 pos) +{ + vec3 a = Horz3(pos, -1.0); + vec3 b = Horz5(pos, 0.0); + vec3 c = Horz3(pos, 1.0); + + float wa = Scan(pos, -1.0); + float wb = Scan(pos, 0.0); + float wc = Scan(pos, 1.0); + + return (a * wa) + (b * wb) + (c * wc); +} + +vec2 Warp(vec2 pos) +{ + pos = pos * 2.0 - 1.0; + pos *= vec2(1.0 + (pos.y * pos.y) * HorizontalWarp, 1.0 + (pos.x * pos.x) * VerticalWarp); + return pos * 0.5 + 0.5; +} + +vec3 Mask(vec2 pos) +{ +#if MaskingType == 1 + // Very compressed TV style shadow mask. + float lines = MaskAmountLight; + float odd = 0.0; + + if (fract(pos.x / 6.0) < 0.5) + { + odd = 1.0; + } + if (fract((pos.y + odd) / 2.0) < 0.5) + { + lines = MaskAmountDark; + } + pos.x = fract(pos.x / 3.0); + vec3 mask = vec3(MaskAmountDark, MaskAmountDark, MaskAmountDark); + + if (pos.x < 0.333) + { + mask.r = MaskAmountLight; + } + else if (pos.x < 0.666) + { + mask.g = MaskAmountLight; + } + else + { + mask.b = MaskAmountLight; + } + + mask *= lines; + + return mask; + +#elif MaskingType == 2 + // Aperture-grille. + pos.x = fract(pos.x / 3.0); + vec3 mask = vec3(MaskAmountDark, MaskAmountDark, MaskAmountDark); + + if (pos.x < 0.333) + { + mask.r = MaskAmountLight; + } + else if (pos.x < 0.666) + { + mask.g = MaskAmountLight; + } + else + { + mask.b = MaskAmountLight; + } + + return mask; + +#elif MaskingType == 3 + // Stretched VGA style shadow mask (same as prior shaders). + pos.x += pos.y * 3.0; + vec3 mask = vec3(MaskAmountDark, MaskAmountDark, MaskAmountDark); + pos.x = fract(pos.x / 6.0); + + if (pos.x < 0.333) + { + mask.r = MaskAmountLight; + } + else if (pos.x < 0.666) + { + mask.g = MaskAmountLight; + } + else + { + mask.b = MaskAmountLight; + } + + return mask; + +#else + // VGA style shadow mask. + pos.xy = floor(pos.xy * vec2(1.0, 0.5)); + pos.x += pos.y * 3.0; + + vec3 mask = vec3(MaskAmountDark, MaskAmountDark, MaskAmountDark); + pos.x = fract(pos.x / 6.0); + + if (pos.x < 0.333) + { + mask.r = MaskAmountLight; + } + else if (pos.x < 0.666) + { + mask.g = MaskAmountLight; + } + else + { + mask.b = MaskAmountLight; + } + return mask; +#endif +} + +vec4 LottesCRTPass() +{ + vec4 fragcoord = gl_FragCoord - u_target_rect; + vec4 color; + vec2 inSize = u_target_resolution - (2 * u_target_rect.xy); + + vec2 pos = Warp(fragcoord.xy / inSize); + +#if UseShadowMask == 0 + color.rgb = Tri(pos); +#else + color.rgb = Tri(pos) * Mask(fragcoord.xy); +#endif + color.rgb = ToSrgb(color.rgb); + color.a = 1.0; + + return color; +} + +void ps_filter_lottes() +{ + o_col0 = LottesCRTPass(); +} + +#endif + #endif \ No newline at end of file diff --git a/pcsx2-qt/Settings/GraphicsSettingsWidget.ui b/pcsx2-qt/Settings/GraphicsSettingsWidget.ui index 46aa3a5311..835c79a72b 100644 --- a/pcsx2-qt/Settings/GraphicsSettingsWidget.ui +++ b/pcsx2-qt/Settings/GraphicsSettingsWidget.ui @@ -1240,6 +1240,11 @@ Wave Filter + + + Lottes CRT + + diff --git a/pcsx2/GS/GS.cpp b/pcsx2/GS/GS.cpp index ddf1823a82..a4791cf772 100644 --- a/pcsx2/GS/GS.cpp +++ b/pcsx2/GS/GS.cpp @@ -1325,6 +1325,7 @@ void GSApp::Init() m_gs_tv_shaders.push_back(GSSetting(2, "Diagonal filter", "")); m_gs_tv_shaders.push_back(GSSetting(3, "Triangular filter", "")); m_gs_tv_shaders.push_back(GSSetting(4, "Wave filter", "")); + m_gs_tv_shaders.push_back(GSSetting(5, "Lottes CRT filter", "")); m_gs_dump_compression.push_back(GSSetting(static_cast(GSDumpCompressionMethod::Uncompressed), "Uncompressed", "")); m_gs_dump_compression.push_back(GSSetting(static_cast(GSDumpCompressionMethod::LZMA), "LZMA (xz)", "")); diff --git a/pcsx2/GS/Renderers/Common/GSDevice.cpp b/pcsx2/GS/Renderers/Common/GSDevice.cpp index 29d463da14..869217fc14 100644 --- a/pcsx2/GS/Renderers/Common/GSDevice.cpp +++ b/pcsx2/GS/Renderers/Common/GSDevice.cpp @@ -57,6 +57,7 @@ const char* shaderName(PresentShader value) case PresentShader::DIAGONAL_FILTER: return "ps_filter_diagonal"; case PresentShader::TRIANGULAR_FILTER: return "ps_filter_triangular"; case PresentShader::COMPLEX_FILTER: return "ps_filter_complex"; + case PresentShader::LOTTES_FILTER: return "ps_filter_lottes"; // clang-format on default: ASSERT(0); diff --git a/pcsx2/GS/Renderers/Common/GSDevice.h b/pcsx2/GS/Renderers/Common/GSDevice.h index 69cff9e49f..8cb97ffe2e 100644 --- a/pcsx2/GS/Renderers/Common/GSDevice.h +++ b/pcsx2/GS/Renderers/Common/GSDevice.h @@ -58,6 +58,7 @@ enum class PresentShader DIAGONAL_FILTER, TRIANGULAR_FILTER, COMPLEX_FILTER, + LOTTES_FILTER, Count }; diff --git a/pcsx2/GS/Renderers/Common/GSRenderer.cpp b/pcsx2/GS/Renderers/Common/GSRenderer.cpp index 9ff542c8ed..4018d9c759 100644 --- a/pcsx2/GS/Renderers/Common/GSRenderer.cpp +++ b/pcsx2/GS/Renderers/Common/GSRenderer.cpp @@ -56,10 +56,10 @@ static std::string GetDumpSerial() } #endif -static constexpr std::array s_tv_shader_indices = { +static constexpr std::array s_tv_shader_indices = { PresentShader::COPY, PresentShader::SCANLINE, PresentShader::DIAGONAL_FILTER, PresentShader::TRIANGULAR_FILTER, - PresentShader::COMPLEX_FILTER}; + PresentShader::COMPLEX_FILTER, PresentShader::LOTTES_FILTER}; std::unique_ptr g_gs_renderer; diff --git a/pcsx2/GS/Renderers/Metal/present.metal b/pcsx2/GS/Renderers/Metal/present.metal index 179b38f7f8..899a046508 100644 --- a/pcsx2/GS/Renderers/Metal/present.metal +++ b/pcsx2/GS/Renderers/Metal/present.metal @@ -71,3 +71,260 @@ fragment float4 ps_filter_complex(ConvertShaderData data [[stage_in]], ConvertPS return factor * res.sample(float2(data.t.x, ycoord)); } + +#define MaskingType 4 //[1|2|3|4] The type of CRT shadow masking used. 1: compressed TV style, 2: Aperture-grille, 3: Stretched VGA style, 4: VGA style. +#define ScanBrightness -8.00 //[-16.0 to 1.0] The overall brightness of the scanline effect. Lower for darker, higher for brighter. +#define FilterCRTAmount -1.00 //[-4.0 to 1.0] The amount of filtering used, to replicate the TV CRT look. Lower for less, higher for more. +#define HorizontalWarp 0.00 //[0.0 to 0.1] The distortion warping effect for the horizontal (x) axis of the screen. Use small increments. +#define VerticalWarp 0.00 //[0.0 to 0.1] The distortion warping effect for the verticle (y) axis of the screen. Use small increments. +#define MaskAmountDark 0.80 //[0.0 to 1.0] The value of the dark masking line effect used. Lower for darker lower end masking, higher for brighter. +#define MaskAmountLight 1.50 //[0.0 to 2.0] The value of the light masking line effect used. Lower for darker higher end masking, higher for brighter. +#define ResolutionScale 2.00 //[0.1 to 4.0] The scale of the image resolution. Lowering this can give off a nice retro TV look. Raising it can clear up the image. +#define MaskResolutionScale 0.80 //[0.1 to 2.0] The scale of the CRT mask resolution. Use this for balancing the scanline mask scale for difference resolution scaling. +#define UseShadowMask 1 //[0 or 1] Enables, or disables the use of the CRT shadow mask. 0 is disabled, 1 is enabled. + +struct LottesCRTPass +{ + thread ConvertPSRes& res; + constant GSMTLPresentPSUniform& uniform; + LottesCRTPass(thread ConvertPSRes& res, constant GSMTLPresentPSUniform& uniform): res(res), uniform(uniform) {} + + float ToLinear1(float c) + { + c = saturate(c); + return c <= 0.04045 ? c / 12.92 : pow((c + 0.055) / 1.055, 2.4); + } + + float3 ToLinear(float3 c) + { + return float3(ToLinear1(c.r), ToLinear1(c.g), ToLinear1(c.b)); + } + + float ToSrgb1(float c) + { + c = saturate(c); + return c < 0.0031308 ? c * 12.92 : 1.055 * pow(c, 0.41666) - 0.055; + } + + float3 ToSrgb(float3 c) + { + return float3(ToSrgb1(c.r), ToSrgb1(c.g), ToSrgb1(c.b)); + } + + float3 Fetch(float2 pos, float2 off) + { + float2 screenSize = uniform.source_resolution; + float2 scaledRes = (screenSize * ResolutionScale); + pos = round(pos * scaledRes + off) / scaledRes; + if (max(abs(pos.x - 0.5), abs(pos.y - 0.5)) > 0.5) + { + return float3(0.0, 0.0, 0.0); + } + else + { + return ToLinear(res.sample(pos.xy).rgb); + } + } + + float2 Dist(float2 pos) + { + float2 crtRes = uniform.rcp_target_resolution; + float2 res = (crtRes * MaskResolutionScale); + pos = (pos * res); + + return -((pos - floor(pos)) - float2(0.5, 0.5)); + } + + float Gaus(float pos, float scale) + { + return exp2(scale * pos * pos); + } + + float3 Horz3(float2 pos, float off) + { + float3 b = Fetch(pos, float2(-1.0, off)); + float3 c = Fetch(pos, float2(0.0, off)); + float3 d = Fetch(pos, float2(1.0, off)); + float dst = Dist(pos).x; + + // Convert distance to weight. + float scale = FilterCRTAmount; + float wb = Gaus(dst - 1.0, scale); + float wc = Gaus(dst + 0.0, scale); + float wd = Gaus(dst + 1.0, scale); + + return (b * wb) + (c * wc) + (d * wd) / (wb + wc + wd); + } + + float3 Horz5(float2 pos, float off) + { + float3 a = Fetch(pos, float2(-2.0, off)); + float3 b = Fetch(pos, float2(-1.0, off)); + float3 c = Fetch(pos, float2(0.0, off)); + float3 d = Fetch(pos, float2(1.0, off)); + float3 e = Fetch(pos, float2(2.0, off)); + float dst = Dist(pos).x; + + // Convert distance to weight. + float scale = FilterCRTAmount; + + float wa = Gaus(dst - 2.0, scale); + float wb = Gaus(dst - 1.0, scale); + float wc = Gaus(dst + 0.0, scale); + float wd = Gaus(dst + 1.0, scale); + float we = Gaus(dst + 2.0, scale); + + return (a * wa) + (b * wb) + (c * wc) + (d * wd) + (e * we) / (wa + wb + wc + wd + we); + } + + // Return scanline weight. + float Scan(float2 pos, float off) + { + float dst = Dist(pos).y; + return Gaus(dst + off, ScanBrightness); + } + + float3 Tri(float2 pos) + { + float3 a = Horz3(pos, -1.0); + float3 b = Horz5(pos, 0.0); + float3 c = Horz3(pos, 1.0); + + float wa = Scan(pos, -1.0); + float wb = Scan(pos, 0.0); + float wc = Scan(pos, 1.0); + + return (a * wa) + (b * wb) + (c * wc); + } + + float2 Warp(float2 pos) + { + pos = pos * 2.0 - 1.0; + pos *= float2(1.0 + (pos.y * pos.y) * HorizontalWarp, 1.0 + (pos.x * pos.x) * VerticalWarp); + return pos * 0.5 + 0.5; + } + + float3 Mask(float2 pos) + { +#if MaskingType == 1 + // Very compressed TV style shadow mask. + float lines = MaskAmountLight; + float odd = 0.0; + + if (fract(pos.x / 6.0) < 0.5) + { + odd = 1.0; + } + if (fract((pos.y + odd) / 2.0) < 0.5) + { + lines = MaskAmountDark; + } + pos.x = fract(pos.x / 3.0); + float3 mask = float3(MaskAmountDark, MaskAmountDark, MaskAmountDark); + + if (pos.x < 0.333) + { + mask.r = MaskAmountLight; + } + else if (pos.x < 0.666) + { + mask.g = MaskAmountLight; + } + else + { + mask.b = MaskAmountLight; + } + + mask *= lines; + + return mask; + +#elif MaskingType == 2 + // Aperture-grille. + pos.x = fract(pos.x / 3.0); + float3 mask = float3(MaskAmountDark, MaskAmountDark, MaskAmountDark); + + if (pos.x < 0.333) + { + mask.r = MaskAmountLight; + } + else if (pos.x < 0.666) + { + mask.g = MaskAmountLight; + } + else + { + mask.b = MaskAmountLight; + } + + return mask; + +#elif MaskingType == 3 + // Stretched VGA style shadow mask (same as prior shaders). + pos.x += pos.y * 3.0; + float3 mask = float3(MaskAmountDark, MaskAmountDark, MaskAmountDark); + pos.x = fract(pos.x / 6.0); + + if (pos.x < 0.333) + { + mask.r = MaskAmountLight; + } + else if (pos.x < 0.666) + { + mask.g = MaskAmountLight; + } + else + { + mask.b = MaskAmountLight; + } + + return mask; + +#else + // VGA style shadow mask. + pos.xy = floor(pos.xy * float2(1.0, 0.5)); + pos.x += pos.y * 3.0; + + float3 mask = float3(MaskAmountDark, MaskAmountDark, MaskAmountDark); + pos.x = fract(pos.x / 6.0); + + if (pos.x < 0.333) + { + mask.r = MaskAmountLight; + } + else if (pos.x < 0.666) + { + mask.g = MaskAmountLight; + } + else + { + mask.b = MaskAmountLight; + } + return mask; +#endif + } + + float4 Run(float4 fragcoord) + { + fragcoord -= uniform.target_rect; + float2 inSize = uniform.target_resolution - (2 * uniform.target_rect.xy); + float4 color; + float2 pos = Warp(fragcoord.xy / inSize); + +#if UseShadowMask == 0 + color.rgb = Tri(pos); +#else + color.rgb = Tri(pos) * Mask(fragcoord.xy); +#endif + color.rgb = ToSrgb(color.rgb); + color.a = 1.0; + + return color; + } +}; + +fragment float4 ps_filter_lottes(ConvertShaderData data [[stage_in]], ConvertPSRes res, + constant GSMTLPresentPSUniform& uniform [[buffer(GSMTLBufferIndexUniforms)]]) +{ + return LottesCRTPass(res, uniform).Run(data.p); +}