-
-
Notifications
You must be signed in to change notification settings - Fork 3.7k
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
[Merged by Bors] - Revamp Bloom #6677
Changes from 126 commits
425c7ae
1dd39be
7704e12
bb465b5
82e0f2f
fa4fccb
d7a1ae7
9a94a40
bb06589
892fb5e
eb27945
2e3f428
dabf32a
5fd4116
38df5ad
7de822b
cff860e
e58aeac
af760a7
1719571
d6b642a
3f4ecb8
4398515
d172215
5fbdc01
4b63ea3
aea919a
0e481ff
47f2e3d
d4f491c
e64900e
193932f
8972fd3
ec53b37
4daff1b
5fb4806
2ab6225
bf8e301
a574388
85af8f9
1dc0d41
774f008
2f35c92
896ee46
9f1c6cc
79bc5b9
ec3c569
9bef787
f849c4e
9298bc6
3452b32
c74f301
3d0a912
9d5c544
dc257d4
df53631
ad1f11c
3f51432
8971a88
8a0d1d1
dac7680
82a553d
4ff46bf
0214e39
1cbe98a
ebe2cff
3e8d9fa
c0a7f2f
cc57711
9377e14
12e4b10
75f59f8
a495344
38d7c45
4a2587b
c515fc5
c7faf67
f87bf01
895d626
b2dba3a
1536709
8ae71e6
7a321f3
677b758
d4b61ac
f315d78
aa2a597
a297ce9
c2955b3
09ae245
a96ad6e
870435c
a187c4b
95ad4d7
c041374
6dd011b
e1485f1
48fedbf
1c11c9d
c296a45
860effc
f99a027
6d4ee2b
e8fb506
3339655
fe5aa43
70b398e
62c885c
7b7db8a
ddfe45e
6cb8b0c
adedc79
8981638
13d131c
93d9042
86372b1
80b7ce2
b7e0951
9b1cdad
98060e8
24af9c6
26a503b
63630e8
c624b06
61391d4
129f334
7a02aae
e047f41
0e2d92f
f480488
c7d722a
c28ad59
5c40029
bcc6994
6cb4c99
e72f201
File filter
Filter by extension
Conversations
Jump to
Diff view
Diff view
There are no files selected for viewing
Original file line number | Diff line number | Diff line change | ||||||||||||||||||||||||||||||||||||||||||||||||
---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|
@@ -1,138 +1,148 @@ | ||||||||||||||||||||||||||||||||||||||||||||||||||
// Bloom works by creating an intermediate texture with a bunch of mip levels, each half the size of the previous. | ||||||||||||||||||||||||||||||||||||||||||||||||||
// You then downsample each mip (starting with the original texture) to the lower resolution mip under it, going in order. | ||||||||||||||||||||||||||||||||||||||||||||||||||
// You then upsample each mip (starting from the smallest mip) and blend with the higher resolution mip above it (ending on the original texture). | ||||||||||||||||||||||||||||||||||||||||||||||||||
// | ||||||||||||||||||||||||||||||||||||||||||||||||||
// References: | ||||||||||||||||||||||||||||||||||||||||||||||||||
// * [COD] - Next Generation Post Processing in Call of Duty - http://www.iryoku.com/next-generation-post-processing-in-call-of-duty-advanced-warfare | ||||||||||||||||||||||||||||||||||||||||||||||||||
// * [PBB] - Physically Based Bloom - https://learnopengl.com/Guest-Articles/2022/Phys.-Based-Bloom | ||||||||||||||||||||||||||||||||||||||||||||||||||
|
||||||||||||||||||||||||||||||||||||||||||||||||||
#import bevy_core_pipeline::fullscreen_vertex_shader | ||||||||||||||||||||||||||||||||||||||||||||||||||
#import bevy_core_pipeline::tonemapping | ||||||||||||||||||||||||||||||||||||||||||||||||||
|
||||||||||||||||||||||||||||||||||||||||||||||||||
struct BloomUniforms { | ||||||||||||||||||||||||||||||||||||||||||||||||||
threshold: f32, | ||||||||||||||||||||||||||||||||||||||||||||||||||
knee: f32, | ||||||||||||||||||||||||||||||||||||||||||||||||||
scale: f32, | ||||||||||||||||||||||||||||||||||||||||||||||||||
intensity: f32, | ||||||||||||||||||||||||||||||||||||||||||||||||||
struct BloomDownsamplingUniforms { | ||||||||||||||||||||||||||||||||||||||||||||||||||
threshold_precomputations: vec4<f32>, | ||||||||||||||||||||||||||||||||||||||||||||||||||
viewport: vec4<f32>, | ||||||||||||||||||||||||||||||||||||||||||||||||||
}; | ||||||||||||||||||||||||||||||||||||||||||||||||||
|
||||||||||||||||||||||||||||||||||||||||||||||||||
@group(0) @binding(0) | ||||||||||||||||||||||||||||||||||||||||||||||||||
var original: texture_2d<f32>; | ||||||||||||||||||||||||||||||||||||||||||||||||||
var input_texture: texture_2d<f32>; | ||||||||||||||||||||||||||||||||||||||||||||||||||
@group(0) @binding(1) | ||||||||||||||||||||||||||||||||||||||||||||||||||
var original_sampler: sampler; | ||||||||||||||||||||||||||||||||||||||||||||||||||
@group(0) @binding(2) | ||||||||||||||||||||||||||||||||||||||||||||||||||
var<uniform> uniforms: BloomUniforms; | ||||||||||||||||||||||||||||||||||||||||||||||||||
@group(0) @binding(3) | ||||||||||||||||||||||||||||||||||||||||||||||||||
var up: texture_2d<f32>; | ||||||||||||||||||||||||||||||||||||||||||||||||||
var s: sampler; | ||||||||||||||||||||||||||||||||||||||||||||||||||
|
||||||||||||||||||||||||||||||||||||||||||||||||||
fn quadratic_threshold(color: vec4<f32>, threshold: f32, curve: vec3<f32>) -> vec4<f32> { | ||||||||||||||||||||||||||||||||||||||||||||||||||
let br = max(max(color.r, color.g), color.b); | ||||||||||||||||||||||||||||||||||||||||||||||||||
|
||||||||||||||||||||||||||||||||||||||||||||||||||
var rq: f32 = clamp(br - curve.x, 0.0, curve.y); | ||||||||||||||||||||||||||||||||||||||||||||||||||
rq = curve.z * rq * rq; | ||||||||||||||||||||||||||||||||||||||||||||||||||
|
||||||||||||||||||||||||||||||||||||||||||||||||||
return color * max(rq, br - threshold) / max(br, 0.0001); | ||||||||||||||||||||||||||||||||||||||||||||||||||
#ifdef FIRST_DOWNSAMPLE | ||||||||||||||||||||||||||||||||||||||||||||||||||
@group(0) @binding(2) | ||||||||||||||||||||||||||||||||||||||||||||||||||
var<uniform> uniforms: BloomDownsamplingUniforms; | ||||||||||||||||||||||||||||||||||||||||||||||||||
|
||||||||||||||||||||||||||||||||||||||||||||||||||
// https://catlikecoding.com/unity/tutorials/advanced-rendering/bloom/#3.4 | ||||||||||||||||||||||||||||||||||||||||||||||||||
fn soft_threshold(color: vec3<f32>) -> vec3<f32> { | ||||||||||||||||||||||||||||||||||||||||||||||||||
let brightness = max(color.r, max(color.g, color.b)); | ||||||||||||||||||||||||||||||||||||||||||||||||||
var softness = brightness - uniforms.threshold_precomputations.y; | ||||||||||||||||||||||||||||||||||||||||||||||||||
softness = clamp(softness, 0.0, uniforms.threshold_precomputations.z); | ||||||||||||||||||||||||||||||||||||||||||||||||||
softness = softness * softness * uniforms.threshold_precomputations.w; | ||||||||||||||||||||||||||||||||||||||||||||||||||
var contribution = max(brightness - uniforms.threshold_precomputations.x, softness); | ||||||||||||||||||||||||||||||||||||||||||||||||||
contribution /= max(brightness, 0.00001); // Prevent division by 0 | ||||||||||||||||||||||||||||||||||||||||||||||||||
return color * contribution; | ||||||||||||||||||||||||||||||||||||||||||||||||||
} | ||||||||||||||||||||||||||||||||||||||||||||||||||
#endif | ||||||||||||||||||||||||||||||||||||||||||||||||||
|
||||||||||||||||||||||||||||||||||||||||||||||||||
// http://graphicrants.blogspot.com/2013/12/tone-mapping.html | ||||||||||||||||||||||||||||||||||||||||||||||||||
fn karis_average(color: vec3<f32>) -> f32 { | ||||||||||||||||||||||||||||||||||||||||||||||||||
// Luminance calculated by gamma-correcting linear RGB to non-linear sRGB using pow(color, 1.0 / 2.2) | ||||||||||||||||||||||||||||||||||||||||||||||||||
// and then calculating luminance based on Rec. 709 color primaries. | ||||||||||||||||||||||||||||||||||||||||||||||||||
let luma = tonemapping_luminance(rgb_to_srgb_simple(color)) / 4.0; | ||||||||||||||||||||||||||||||||||||||||||||||||||
JMS55 marked this conversation as resolved.
Show resolved
Hide resolved
|
||||||||||||||||||||||||||||||||||||||||||||||||||
return 1.0 / (1.0 + luma); | ||||||||||||||||||||||||||||||||||||||||||||||||||
} | ||||||||||||||||||||||||||||||||||||||||||||||||||
|
||||||||||||||||||||||||||||||||||||||||||||||||||
// Samples original around the supplied uv using a filter. | ||||||||||||||||||||||||||||||||||||||||||||||||||
// | ||||||||||||||||||||||||||||||||||||||||||||||||||
// o o o | ||||||||||||||||||||||||||||||||||||||||||||||||||
// o o | ||||||||||||||||||||||||||||||||||||||||||||||||||
// o o o | ||||||||||||||||||||||||||||||||||||||||||||||||||
// o o | ||||||||||||||||||||||||||||||||||||||||||||||||||
// o o o | ||||||||||||||||||||||||||||||||||||||||||||||||||
// | ||||||||||||||||||||||||||||||||||||||||||||||||||
// This is used because it has a number of advantages that | ||||||||||||||||||||||||||||||||||||||||||||||||||
// outweigh the cost of 13 samples that basically boil down | ||||||||||||||||||||||||||||||||||||||||||||||||||
// to it looking better. | ||||||||||||||||||||||||||||||||||||||||||||||||||
// | ||||||||||||||||||||||||||||||||||||||||||||||||||
// These advantages are outlined in a youtube video by the Cherno: | ||||||||||||||||||||||||||||||||||||||||||||||||||
// https://www.youtube.com/watch?v=tI70-HIc5ro | ||||||||||||||||||||||||||||||||||||||||||||||||||
fn sample_13_tap(uv: vec2<f32>, scale: vec2<f32>) -> vec4<f32> { | ||||||||||||||||||||||||||||||||||||||||||||||||||
let a = textureSample(original, original_sampler, uv + vec2<f32>(-1.0, -1.0) * scale); | ||||||||||||||||||||||||||||||||||||||||||||||||||
let b = textureSample(original, original_sampler, uv + vec2<f32>(0.0, -1.0) * scale); | ||||||||||||||||||||||||||||||||||||||||||||||||||
let c = textureSample(original, original_sampler, uv + vec2<f32>(1.0, -1.0) * scale); | ||||||||||||||||||||||||||||||||||||||||||||||||||
let d = textureSample(original, original_sampler, uv + vec2<f32>(-0.5, -0.5) * scale); | ||||||||||||||||||||||||||||||||||||||||||||||||||
let e = textureSample(original, original_sampler, uv + vec2<f32>(0.5, -0.5) * scale); | ||||||||||||||||||||||||||||||||||||||||||||||||||
let f = textureSample(original, original_sampler, uv + vec2<f32>(-1.0, 0.0) * scale); | ||||||||||||||||||||||||||||||||||||||||||||||||||
let g = textureSample(original, original_sampler, uv + vec2<f32>(0.0, 0.0) * scale); | ||||||||||||||||||||||||||||||||||||||||||||||||||
let h = textureSample(original, original_sampler, uv + vec2<f32>(1.0, 0.0) * scale); | ||||||||||||||||||||||||||||||||||||||||||||||||||
let i = textureSample(original, original_sampler, uv + vec2<f32>(-0.5, 0.5) * scale); | ||||||||||||||||||||||||||||||||||||||||||||||||||
let j = textureSample(original, original_sampler, uv + vec2<f32>(0.5, 0.5) * scale); | ||||||||||||||||||||||||||||||||||||||||||||||||||
let k = textureSample(original, original_sampler, uv + vec2<f32>(-1.0, 1.0) * scale); | ||||||||||||||||||||||||||||||||||||||||||||||||||
let l = textureSample(original, original_sampler, uv + vec2<f32>(0.0, 1.0) * scale); | ||||||||||||||||||||||||||||||||||||||||||||||||||
let m = textureSample(original, original_sampler, uv + vec2<f32>(1.0, 1.0) * scale); | ||||||||||||||||||||||||||||||||||||||||||||||||||
|
||||||||||||||||||||||||||||||||||||||||||||||||||
let div = (1.0 / 4.0) * vec2<f32>(0.5, 0.125); | ||||||||||||||||||||||||||||||||||||||||||||||||||
|
||||||||||||||||||||||||||||||||||||||||||||||||||
var o: vec4<f32> = (d + e + i + j) * div.x; | ||||||||||||||||||||||||||||||||||||||||||||||||||
o = o + (a + b + g + f) * div.y; | ||||||||||||||||||||||||||||||||||||||||||||||||||
o = o + (b + c + h + g) * div.y; | ||||||||||||||||||||||||||||||||||||||||||||||||||
o = o + (f + g + l + k) * div.y; | ||||||||||||||||||||||||||||||||||||||||||||||||||
o = o + (g + h + m + l) * div.y; | ||||||||||||||||||||||||||||||||||||||||||||||||||
|
||||||||||||||||||||||||||||||||||||||||||||||||||
return o; | ||||||||||||||||||||||||||||||||||||||||||||||||||
// [COD] slide 153 | ||||||||||||||||||||||||||||||||||||||||||||||||||
fn sample_input_13_tap(uv: vec2<f32>) -> vec3<f32> { | ||||||||||||||||||||||||||||||||||||||||||||||||||
let texel_size = 1.0 / vec2<f32>(textureDimensions(input_texture)); | ||||||||||||||||||||||||||||||||||||||||||||||||||
let x = texel_size.x; | ||||||||||||||||||||||||||||||||||||||||||||||||||
let y = texel_size.y; | ||||||||||||||||||||||||||||||||||||||||||||||||||
|
||||||||||||||||||||||||||||||||||||||||||||||||||
let a = textureSample(input_texture, s, vec2<f32>(uv.x - 2.0 * x, uv.y + 2.0 * y)).rgb; | ||||||||||||||||||||||||||||||||||||||||||||||||||
JMS55 marked this conversation as resolved.
Show resolved
Hide resolved
|
||||||||||||||||||||||||||||||||||||||||||||||||||
let b = textureSample(input_texture, s, vec2<f32>(uv.x, uv.y + 2.0 * y)).rgb; | ||||||||||||||||||||||||||||||||||||||||||||||||||
let c = textureSample(input_texture, s, vec2<f32>(uv.x + 2.0 * x, uv.y + 2.0 * y)).rgb; | ||||||||||||||||||||||||||||||||||||||||||||||||||
|
||||||||||||||||||||||||||||||||||||||||||||||||||
let d = textureSample(input_texture, s, vec2<f32>(uv.x - 2.0 * x, uv.y)).rgb; | ||||||||||||||||||||||||||||||||||||||||||||||||||
let e = textureSample(input_texture, s, vec2<f32>(uv.x, uv.y)).rgb; | ||||||||||||||||||||||||||||||||||||||||||||||||||
let f = textureSample(input_texture, s, vec2<f32>(uv.x + 2.0 * x, uv.y)).rgb; | ||||||||||||||||||||||||||||||||||||||||||||||||||
|
||||||||||||||||||||||||||||||||||||||||||||||||||
let g = textureSample(input_texture, s, vec2<f32>(uv.x - 2.0 * x, uv.y - 2.0 * y)).rgb; | ||||||||||||||||||||||||||||||||||||||||||||||||||
let h = textureSample(input_texture, s, vec2<f32>(uv.x, uv.y - 2.0 * y)).rgb; | ||||||||||||||||||||||||||||||||||||||||||||||||||
let i = textureSample(input_texture, s, vec2<f32>(uv.x + 2.0 * x, uv.y - 2.0 * y)).rgb; | ||||||||||||||||||||||||||||||||||||||||||||||||||
|
||||||||||||||||||||||||||||||||||||||||||||||||||
let j = textureSample(input_texture, s, vec2<f32>(uv.x - x, uv.y + y)).rgb; | ||||||||||||||||||||||||||||||||||||||||||||||||||
let k = textureSample(input_texture, s, vec2<f32>(uv.x + x, uv.y + y)).rgb; | ||||||||||||||||||||||||||||||||||||||||||||||||||
let l = textureSample(input_texture, s, vec2<f32>(uv.x - x, uv.y - y)).rgb; | ||||||||||||||||||||||||||||||||||||||||||||||||||
let m = textureSample(input_texture, s, vec2<f32>(uv.x + x, uv.y - y)).rgb; | ||||||||||||||||||||||||||||||||||||||||||||||||||
|
||||||||||||||||||||||||||||||||||||||||||||||||||
#ifdef FIRST_DOWNSAMPLE | ||||||||||||||||||||||||||||||||||||||||||||||||||
// [COD] slide 168 | ||||||||||||||||||||||||||||||||||||||||||||||||||
// | ||||||||||||||||||||||||||||||||||||||||||||||||||
// The first downsample pass reads from the rendered frame which may exhibit | ||||||||||||||||||||||||||||||||||||||||||||||||||
// 'fireflies' (individual very bright pixels) that should not cause the bloom effect. | ||||||||||||||||||||||||||||||||||||||||||||||||||
// | ||||||||||||||||||||||||||||||||||||||||||||||||||
// The first downsample uses a firefly-reduction method proposed by Brian Karis | ||||||||||||||||||||||||||||||||||||||||||||||||||
// which takes a weighted-average of the samples to limit their luma range to [0, 1]. | ||||||||||||||||||||||||||||||||||||||||||||||||||
// This implementation matches the LearnOpenGL article [PBB]. | ||||||||||||||||||||||||||||||||||||||||||||||||||
var group0 = (a + b + d + e) * (0.125f / 4.0f); | ||||||||||||||||||||||||||||||||||||||||||||||||||
var group1 = (b + c + e + f) * (0.125f / 4.0f); | ||||||||||||||||||||||||||||||||||||||||||||||||||
var group2 = (d + e + g + h) * (0.125f / 4.0f); | ||||||||||||||||||||||||||||||||||||||||||||||||||
var group3 = (e + f + h + i) * (0.125f / 4.0f); | ||||||||||||||||||||||||||||||||||||||||||||||||||
var group4 = (j + k + l + m) * (0.5f / 4.0f); | ||||||||||||||||||||||||||||||||||||||||||||||||||
group0 *= karis_average(group0); | ||||||||||||||||||||||||||||||||||||||||||||||||||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. The article says:
I am not convinced the implementation in this PR is a valid simplification of that. For one, I don't think calculating the weight of a sum of samples will be the same as summing the weights of samples, when the formula for the weight is I would say that I think to be correct, the firefly reduction would need to be applied to each sample individually. But then I realised that each sample is a bilinear interpolation of samples around it, so it is already in some sense a weighted-average value. So, what is the goal here? To reduce fireflies. The PR considers each group, which are groups of four samples in various configurations at least 2 pixels apart, and then tries to eliminate fireflies in the average of those four. It seems incorrect regardless. I'm not yet sure what a correct and/or acceptable solution is. There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. It seems I missed part of the slides that discuss this approach. It was intentional to apply the firefly reduction to the averaged groups of 4 samples, so this is OK. We should add a reference for it though. Had it been there I'd have looked at it. :) There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. The firefly reduction method used in this PR is a well tested one that is extensively described and, as far as I see, correctly implemented one. There is another article (Bonus material 2: Karis average) that shows code very similar to what we have There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. Thanks for that reference. We should probably add a link to it too. There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. I see now there is a link at the top of the file. There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. Clarity: I'm happy to consider this resolved. |
||||||||||||||||||||||||||||||||||||||||||||||||||
group1 *= karis_average(group1); | ||||||||||||||||||||||||||||||||||||||||||||||||||
group2 *= karis_average(group2); | ||||||||||||||||||||||||||||||||||||||||||||||||||
group3 *= karis_average(group3); | ||||||||||||||||||||||||||||||||||||||||||||||||||
group4 *= karis_average(group4); | ||||||||||||||||||||||||||||||||||||||||||||||||||
return group0 + group1 + group2 + group3 + group4; | ||||||||||||||||||||||||||||||||||||||||||||||||||
Comment on lines
+81
to
+91
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. I feel like the full filter weights should be applied after the luminance averaging. So each 4x4 box filter just needs the 4 samples averaging (i.e. / 4 or * 0.25), then the luminance adjustment is applied, then the 13-tap weighting is applied.
Suggested change
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. Hmm, this thinks otherwise: https://learnopengl.com/Guest-Articles/2022/Phys.-Based-Bloom But maybe they did it wrong? There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. Asking Jimenez: https://twitter.com/swainrob/status/1624767458958516225 There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. Clarity: given this matches the learnopengl source material, I'm happy to consider this resolved, pending feedback from Jimenez. |
||||||||||||||||||||||||||||||||||||||||||||||||||
#else | ||||||||||||||||||||||||||||||||||||||||||||||||||
var sample = (a + c + g + i) * 0.03125; | ||||||||||||||||||||||||||||||||||||||||||||||||||
sample += (b + d + f + h) * 0.0625; | ||||||||||||||||||||||||||||||||||||||||||||||||||
sample += (e + j + k + l + m) * 0.125; | ||||||||||||||||||||||||||||||||||||||||||||||||||
return sample; | ||||||||||||||||||||||||||||||||||||||||||||||||||
#endif | ||||||||||||||||||||||||||||||||||||||||||||||||||
} | ||||||||||||||||||||||||||||||||||||||||||||||||||
|
||||||||||||||||||||||||||||||||||||||||||||||||||
// Samples original using a 3x3 tent filter. | ||||||||||||||||||||||||||||||||||||||||||||||||||
// | ||||||||||||||||||||||||||||||||||||||||||||||||||
// NOTE: Use a 2x2 filter for better perf, but 3x3 looks better. | ||||||||||||||||||||||||||||||||||||||||||||||||||
fn sample_original_3x3_tent(uv: vec2<f32>, scale: vec2<f32>) -> vec4<f32> { | ||||||||||||||||||||||||||||||||||||||||||||||||||
let d = vec4<f32>(1.0, 1.0, -1.0, 0.0); | ||||||||||||||||||||||||||||||||||||||||||||||||||
// [COD] slide 162 | ||||||||||||||||||||||||||||||||||||||||||||||||||
fn sample_input_3x3_tent(uv: vec2<f32>) -> vec3<f32> { | ||||||||||||||||||||||||||||||||||||||||||||||||||
// Radius. Empirically chosen by and tweaked from the LearnOpenGL article. | ||||||||||||||||||||||||||||||||||||||||||||||||||
let x = 0.004; | ||||||||||||||||||||||||||||||||||||||||||||||||||
let y = 0.004; | ||||||||||||||||||||||||||||||||||||||||||||||||||
JMS55 marked this conversation as resolved.
Show resolved
Hide resolved
JMS55 marked this conversation as resolved.
Show resolved
Hide resolved
|
||||||||||||||||||||||||||||||||||||||||||||||||||
|
||||||||||||||||||||||||||||||||||||||||||||||||||
let a = textureSample(input_texture, s, vec2<f32>(uv.x - x, uv.y + y)).rgb; | ||||||||||||||||||||||||||||||||||||||||||||||||||
let b = textureSample(input_texture, s, vec2<f32>(uv.x, uv.y + y)).rgb; | ||||||||||||||||||||||||||||||||||||||||||||||||||
let c = textureSample(input_texture, s, vec2<f32>(uv.x + x, uv.y + y)).rgb; | ||||||||||||||||||||||||||||||||||||||||||||||||||
|
||||||||||||||||||||||||||||||||||||||||||||||||||
var s: vec4<f32> = textureSample(original, original_sampler, uv - d.xy * scale); | ||||||||||||||||||||||||||||||||||||||||||||||||||
s = s + textureSample(original, original_sampler, uv - d.wy * scale) * 2.0; | ||||||||||||||||||||||||||||||||||||||||||||||||||
s = s + textureSample(original, original_sampler, uv - d.zy * scale); | ||||||||||||||||||||||||||||||||||||||||||||||||||
let d = textureSample(input_texture, s, vec2<f32>(uv.x - x, uv.y)).rgb; | ||||||||||||||||||||||||||||||||||||||||||||||||||
let e = textureSample(input_texture, s, vec2<f32>(uv.x, uv.y)).rgb; | ||||||||||||||||||||||||||||||||||||||||||||||||||
let f = textureSample(input_texture, s, vec2<f32>(uv.x + x, uv.y)).rgb; | ||||||||||||||||||||||||||||||||||||||||||||||||||
|
||||||||||||||||||||||||||||||||||||||||||||||||||
s = s + textureSample(original, original_sampler, uv + d.zw * scale) * 2.0; | ||||||||||||||||||||||||||||||||||||||||||||||||||
s = s + textureSample(original, original_sampler, uv) * 4.0; | ||||||||||||||||||||||||||||||||||||||||||||||||||
s = s + textureSample(original, original_sampler, uv + d.xw * scale) * 2.0; | ||||||||||||||||||||||||||||||||||||||||||||||||||
let g = textureSample(input_texture, s, vec2<f32>(uv.x - x, uv.y - y)).rgb; | ||||||||||||||||||||||||||||||||||||||||||||||||||
let h = textureSample(input_texture, s, vec2<f32>(uv.x, uv.y - y)).rgb; | ||||||||||||||||||||||||||||||||||||||||||||||||||
let i = textureSample(input_texture, s, vec2<f32>(uv.x + x, uv.y - y)).rgb; | ||||||||||||||||||||||||||||||||||||||||||||||||||
|
||||||||||||||||||||||||||||||||||||||||||||||||||
s = s + textureSample(original, original_sampler, uv + d.zy * scale); | ||||||||||||||||||||||||||||||||||||||||||||||||||
s = s + textureSample(original, original_sampler, uv + d.wy * scale) * 2.0; | ||||||||||||||||||||||||||||||||||||||||||||||||||
s = s + textureSample(original, original_sampler, uv + d.xy * scale); | ||||||||||||||||||||||||||||||||||||||||||||||||||
var sample = e * 0.25; | ||||||||||||||||||||||||||||||||||||||||||||||||||
sample += (b + d + f + h) * 0.125; | ||||||||||||||||||||||||||||||||||||||||||||||||||
sample += (a + c + g + i) * 0.0625; | ||||||||||||||||||||||||||||||||||||||||||||||||||
|
||||||||||||||||||||||||||||||||||||||||||||||||||
return s / 16.0; | ||||||||||||||||||||||||||||||||||||||||||||||||||
return sample; | ||||||||||||||||||||||||||||||||||||||||||||||||||
} | ||||||||||||||||||||||||||||||||||||||||||||||||||
|
||||||||||||||||||||||||||||||||||||||||||||||||||
#ifdef FIRST_DOWNSAMPLE | ||||||||||||||||||||||||||||||||||||||||||||||||||
@fragment | ||||||||||||||||||||||||||||||||||||||||||||||||||
fn downsample_prefilter(@location(0) output_uv: vec2<f32>) -> @location(0) vec4<f32> { | ||||||||||||||||||||||||||||||||||||||||||||||||||
fn downsample_first(@location(0) output_uv: vec2<f32>) -> @location(0) vec4<f32> { | ||||||||||||||||||||||||||||||||||||||||||||||||||
let sample_uv = uniforms.viewport.xy + output_uv * uniforms.viewport.zw; | ||||||||||||||||||||||||||||||||||||||||||||||||||
let texel_size = 1.0 / vec2<f32>(textureDimensions(original)); | ||||||||||||||||||||||||||||||||||||||||||||||||||
var sample = sample_input_13_tap(sample_uv); | ||||||||||||||||||||||||||||||||||||||||||||||||||
// Lower bound of 0.0001 is to avoid propagating multiplying by 0.0 through the | ||||||||||||||||||||||||||||||||||||||||||||||||||
// downscaling and upscaling which would result in black boxes. | ||||||||||||||||||||||||||||||||||||||||||||||||||
// The upper bound is to prevent NaNs. | ||||||||||||||||||||||||||||||||||||||||||||||||||
sample = clamp(sample, vec3<f32>(0.0001), vec3<f32>(3.40282347E+38)); | ||||||||||||||||||||||||||||||||||||||||||||||||||
|
||||||||||||||||||||||||||||||||||||||||||||||||||
let scale = texel_size; | ||||||||||||||||||||||||||||||||||||||||||||||||||
#ifdef USE_THRESHOLD | ||||||||||||||||||||||||||||||||||||||||||||||||||
sample = soft_threshold(sample); | ||||||||||||||||||||||||||||||||||||||||||||||||||
#endif | ||||||||||||||||||||||||||||||||||||||||||||||||||
|
||||||||||||||||||||||||||||||||||||||||||||||||||
let curve = vec3<f32>( | ||||||||||||||||||||||||||||||||||||||||||||||||||
uniforms.threshold - uniforms.knee, | ||||||||||||||||||||||||||||||||||||||||||||||||||
uniforms.knee * 2.0, | ||||||||||||||||||||||||||||||||||||||||||||||||||
0.25 / uniforms.knee, | ||||||||||||||||||||||||||||||||||||||||||||||||||
); | ||||||||||||||||||||||||||||||||||||||||||||||||||
|
||||||||||||||||||||||||||||||||||||||||||||||||||
var o: vec4<f32> = sample_13_tap(sample_uv, scale); | ||||||||||||||||||||||||||||||||||||||||||||||||||
|
||||||||||||||||||||||||||||||||||||||||||||||||||
o = quadratic_threshold(o, uniforms.threshold, curve); | ||||||||||||||||||||||||||||||||||||||||||||||||||
o = max(o, vec4<f32>(0.00001)); | ||||||||||||||||||||||||||||||||||||||||||||||||||
|
||||||||||||||||||||||||||||||||||||||||||||||||||
return o; | ||||||||||||||||||||||||||||||||||||||||||||||||||
return vec4<f32>(sample, 1.0); | ||||||||||||||||||||||||||||||||||||||||||||||||||
} | ||||||||||||||||||||||||||||||||||||||||||||||||||
#endif | ||||||||||||||||||||||||||||||||||||||||||||||||||
|
||||||||||||||||||||||||||||||||||||||||||||||||||
@fragment | ||||||||||||||||||||||||||||||||||||||||||||||||||
fn downsample(@location(0) uv: vec2<f32>) -> @location(0) vec4<f32> { | ||||||||||||||||||||||||||||||||||||||||||||||||||
let texel_size = 1.0 / vec2<f32>(textureDimensions(original)); | ||||||||||||||||||||||||||||||||||||||||||||||||||
|
||||||||||||||||||||||||||||||||||||||||||||||||||
let scale = texel_size; | ||||||||||||||||||||||||||||||||||||||||||||||||||
|
||||||||||||||||||||||||||||||||||||||||||||||||||
return sample_13_tap(uv, scale); | ||||||||||||||||||||||||||||||||||||||||||||||||||
return vec4<f32>(sample_input_13_tap(uv), 1.0); | ||||||||||||||||||||||||||||||||||||||||||||||||||
} | ||||||||||||||||||||||||||||||||||||||||||||||||||
|
||||||||||||||||||||||||||||||||||||||||||||||||||
@fragment | ||||||||||||||||||||||||||||||||||||||||||||||||||
fn upsample(@location(0) uv: vec2<f32>) -> @location(0) vec4<f32> { | ||||||||||||||||||||||||||||||||||||||||||||||||||
let texel_size = 1.0 / vec2<f32>(textureDimensions(original)); | ||||||||||||||||||||||||||||||||||||||||||||||||||
|
||||||||||||||||||||||||||||||||||||||||||||||||||
let upsample = sample_original_3x3_tent(uv, texel_size * uniforms.scale); | ||||||||||||||||||||||||||||||||||||||||||||||||||
var color: vec4<f32> = textureSample(up, original_sampler, uv); | ||||||||||||||||||||||||||||||||||||||||||||||||||
color = vec4<f32>(color.rgb + upsample.rgb, upsample.a); | ||||||||||||||||||||||||||||||||||||||||||||||||||
|
||||||||||||||||||||||||||||||||||||||||||||||||||
return color; | ||||||||||||||||||||||||||||||||||||||||||||||||||
} | ||||||||||||||||||||||||||||||||||||||||||||||||||
|
||||||||||||||||||||||||||||||||||||||||||||||||||
@fragment | ||||||||||||||||||||||||||||||||||||||||||||||||||
fn upsample_final(@location(0) uv: vec2<f32>) -> @location(0) vec4<f32> { | ||||||||||||||||||||||||||||||||||||||||||||||||||
let texel_size = 1.0 / vec2<f32>(textureDimensions(original)); | ||||||||||||||||||||||||||||||||||||||||||||||||||
|
||||||||||||||||||||||||||||||||||||||||||||||||||
let upsample = sample_original_3x3_tent(uv, texel_size * uniforms.scale); | ||||||||||||||||||||||||||||||||||||||||||||||||||
|
||||||||||||||||||||||||||||||||||||||||||||||||||
return vec4<f32>(upsample.rgb * uniforms.intensity, upsample.a); | ||||||||||||||||||||||||||||||||||||||||||||||||||
return vec4<f32>(sample_input_3x3_tent(uv), 1.0); | ||||||||||||||||||||||||||||||||||||||||||||||||||
} |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I'm not sure what luminance mapping to use here, but this is based on Rec.709. Apparently the values in
tonemapping_luminance
are based on the Rec.709 colour primaries. The article says that if RGB need to be mapped into the range, then the max colour component should be used. I'm not sure this is correct.There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I'm not sure either.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
The implementation is faithful to the source material. Input signal is in linearized sRGB and luma is a weighted sum or sRGB components. We convert our image to sRGB and perform the luminance function (which is the same as luma but with gamma-corrected input). I don't know why this works considering we are trying to process physical light and avoid any human-based measurements like luminance and, even more so, luma, but this function seems to do the trick somehow. I am pretty positive this will lead to artifacts if the input color space changes (for example to ACES 2065-1) but at the moment it works. I think the approach the original developers took here was somewhat similar to the one described in this video (15:47 - 16:15)
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
:)
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
For clarity, and because I can't resolve these discussions, I consider this discussion resolved well enough for now.