Skip to content
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

Introduce dithering to reduce banding #4497

Merged
merged 8 commits into from
Jul 8, 2024
Merged

Introduce dithering to reduce banding #4497

merged 8 commits into from
Jul 8, 2024

Conversation

jwagner
Copy link
Contributor

@jwagner jwagner commented May 14, 2024

This PR introduces dithering in the egui_glow and egui_wgpu backends to reduce banding artifacts.

It's based on the approach mentioned in #4493 with the small difference that the amount of noise is scaled down slightly to avoid dithering colors that can be represented exactly. This keeps flat surfaces clean.

Exaggerated dithering to show what is happening:
Screenshot from 2024-05-14 19-09-48

Subtle dithering as commited.
Screenshot from 2024-05-14 19-13-40

Closes #4493

Copy link
Contributor

@murl-digital murl-digital left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

LGTM, I tested this with the baseview windowing backend, with is glow based. Taking a screenshot of a gradient and zooming all the way in shows me the dithering's there:

image

@@ -87,5 +114,6 @@ fn fs_main_gamma_framebuffer(in: VertexOutput) -> @location(0) vec4<f32> {
let tex_linear = textureSample(r_tex_color, r_tex_sampler, in.tex_coord);
let tex_gamma = gamma_from_linear_rgba(tex_linear);
let out_color_gamma = in.color * tex_gamma;
return out_color_gamma;
let out_color_dithered = vec4<f32>(dither_interleaved(out_color_gamma.rgb, 256.0, in.position * r_locals.pixels_per_point), out_color_gamma.a);
Copy link
Owner

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Should have the same comments as the code above, and probably a linebreak

let out_color_linear = linear_from_gamma_rgb(out_color_gamma.rgb);
// Dither the float color down to eight bits to reduce banding.
// This step is optional for egui backends.
let out_color_dithered = dither_interleaved(out_color_linear, 256.0, in.position);
Copy link
Owner

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Are you sure we should apply it to the linear color and not the gamma color? 🤔

Copy link
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I agree, this should be done on the gamma color. It's important to know the context of when this fragment shader is used:

"fs_main_linear_framebuffer"
it is enabled in the case of using an srgb converting target. That means whatever float value we emit here will be upon write converted to an srgb value. I.e. as-if the output of this method was wrapped in another call of gamma_from_linear_rgba.

Therefore, we have to make sure that all the meaningful value changes happen on the gamma color.

Note that it would be an entirely different story if we're outputing anything that isn't ending up on a wgpu::Surface that is targeting a srgb interpreting monitor/window. Meaning that all of this breaks down for HDR.

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I assumed that the output buffer was linear in that case dithering the linear color is what is wanted. If it's converted back to srgb before storage, it isn't optimal. I'll need to double check that.

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Just saw the comment be @Wumpf bellow, so you are right, this is off.

crates/egui-wgpu/src/renderer.rs Show resolved Hide resolved
@emilk emilk requested a review from Wumpf May 27, 2024 14:28
Copy link
Collaborator

@Wumpf Wumpf left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

great addition, however I think there should be an easy way to disable dithering:

  • this is srgb dithering and doesn't make sense if someone wants to use an HDR target (e.g. WebGPU can also support P3 instead of srgb)
  • there may be corner cases where a user doesn't actually want to dither everything drawn with egui indiscriminately as the added noise may destroy otherwise sharp corners

let out_color_linear = linear_from_gamma_rgb(out_color_gamma.rgb);
// Dither the float color down to eight bits to reduce banding.
// This step is optional for egui backends.
let out_color_dithered = dither_interleaved(out_color_linear, 256.0, in.position);
Copy link
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I agree, this should be done on the gamma color. It's important to know the context of when this fragment shader is used:

"fs_main_linear_framebuffer"
it is enabled in the case of using an srgb converting target. That means whatever float value we emit here will be upon write converted to an srgb value. I.e. as-if the output of this method was wrapped in another call of gamma_from_linear_rgba.

Therefore, we have to make sure that all the meaningful value changes happen on the gamma color.

Note that it would be an entirely different story if we're outputing anything that isn't ending up on a wgpu::Surface that is targeting a srgb interpreting monitor/window. Meaning that all of this breaks down for HDR.

crates/egui-wgpu/src/egui.wgsl Outdated Show resolved Hide resolved
crates/egui-wgpu/src/renderer.rs Outdated Show resolved Hide resolved
@jwagner
Copy link
Contributor Author

jwagner commented May 27, 2024

@Wumpf @emilk for disabling dithering, would a feature on the crate do the job or should this be a runtime option?

@emilk
Copy link
Owner

emilk commented May 27, 2024

I think a runtime feature flag is likely superior, as it is simpler, and lets the same user use egui_wgpu with different options depending on their render target, which may depend on the adapter capabilities.

Checking a bool in the shader should be fine

@jwagner
Copy link
Contributor Author

jwagner commented May 27, 2024

I'll do a revision of the PR adding an option and fixing the coordinates in wgpu. I'm not certain where to add it yet. Adding it to native / web options in eframe, and then an implementation specific option in the specific backend (WgpuConfiguration for wgpu , EguiGlow::new for glow) seems about right.

If you'd like it to be added another way, just let me know.

@murl-digital
Copy link
Contributor

Adding it to native / web options in eframe, and then an implementation specific option in the specific backend (WgpuConfiguration for wgpu , EguiGlow::new for glow) seems about right.

That works for me, and that's probably the best way to do it, unless you want to mess with feature flags to reduce a tiny amount of runtime overhead

@jwagner jwagner marked this pull request as draft May 30, 2024 18:54
@jwagner
Copy link
Contributor Author

jwagner commented May 30, 2024

I got a first draft of configurable dithering to work but I'm running out of time for today. I changed to state of the PR back to draft for now.

@jwagner
Copy link
Contributor Author

jwagner commented Jun 1, 2024

I updated and cleaned up the code and will mark the PR as 'Ready for review' again.

Testing performed

I tested the demo app using both wgpu and glow, with dithering enabled and disabled, on two different systems (Linux and Mac), as well as on the web using Chrome and Firefox. The dithering works as expected in all cases. While the results from wgpu and glow are not identical, they are very close.

There were some minute differences with dithering disabled as well. So I guess producing identical output between different backends isn't a requirement (and likely not a reasonable goal across systems anyways).

In addition to the demo app, I tested the proposed changes with one of my own applications.

Potential Optimizations

egui_glow::Painter::new, RenderState::create and Renderer::new, now all take quite a few arguments. If desired I can change them to take a struct (with a default) instead.

The implementation of the condition is different for wgpu and glow.

The glow implementation makes use of the glsl preprocessor using a define and if, similar to the existing code for SRGB_TEXTURES. This should result in zero shading cost when dithering is disabled.

The wgpu implementation uses a uniform instead. At least as far as I know wgsl doesn't have a preprocessor and I wasn't to keen to introduce some code to process the shader source. Instead it's using a uniform and a condition in the shader as suggested by @emilk. I'm not sure about the optimizations applied in wgpu, let alone it's various backends but this is likely very cheap (coherent branching) but not zero cost in either case.

Lastly it would be possible to make the amount if dithering configurable, for instance for using egui in an application with a bit depth lower than 8 bit. I don't think this is a very common use case though.

@jwagner jwagner marked this pull request as ready for review June 1, 2024 16:40
@Wumpf
Copy link
Collaborator

Wumpf commented Jun 2, 2024

At least as far as I know wgsl doesn't have a preprocessor and I wasn't to keen to introduce some code to process the shader source.

there's no preprocessor, but this is a great usecase for a pipeline-overridable constant! :) There's an example in the wgpu 0.20.0 release notes.
Unfortunately, they got only added in wgpu 0.20 and egui had to back out of 0.20 for the moment. So would be good to leave a todo note towards (and subsequent patch in) #4560
(not a big deal anyways, this kind of static condition is extremely cheap)

@jwagner
Copy link
Contributor Author

jwagner commented Jun 3, 2024

there's no preprocessor, but this is a great usecase for a pipeline-overridable constant! :) There's an example in the wgpu 0.20.0 release notes. Unfortunately, they got only added in wgpu 0.20 and egui had to back out of 0.20 for the moment. So would be good to leave a todo note towards (and subsequent patch in) #4560 (not a big deal anyways, this kind of static condition is extremely cheap)

That sounds like a great solution. I'll need to do a bit more digging to figure out what it does in practice. If it's ok, I'll do a follow up on this PR once it is merged.

@emilk emilk requested a review from Wumpf June 5, 2024 07:26
Copy link
Collaborator

@Wumpf Wumpf left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

lgtm except docs!

crates/eframe/src/epi.rs Outdated Show resolved Hide resolved
crates/eframe/src/epi.rs Outdated Show resolved Hide resolved
@emilk emilk added egui_glow Relates to running egui_glow on native egui-wgpu labels Jun 19, 2024
@jwagner
Copy link
Contributor Author

jwagner commented Jul 7, 2024

Sorry for the delayed response. Life happened.

I rebased the branch and applied the suggestions. I'm not sure if the more detailed descriptions should maybe go into the lower level crates (egui_glow, egui-wgpu) rather than eframe since users of those crates are more likely to care about the specifics.

Another consideration is api compatibility. I guess we don't want to promise that dithering true will always dither to 8 bit. So maybe _Currently_ dithering assumes an sRGB output and thus will apply noise to any input value that lies between would be a slightly softer comment.

Both are minor concerns though so I'd be happy to get the change merged as is.

Thank you for your patience @Wumpf @emilk :)

@Wumpf Wumpf self-requested a review July 8, 2024 07:46
Copy link
Collaborator

@Wumpf Wumpf left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

lgtm!

Still torn about having it opt-out instead of opt-in, but I wrote earlier to leave it be, so let's just do that (:

now that wgpu 0.20 is in we could use pipeline constants, but that constant buffer value won't hurt anyone 🤷

@Wumpf
Copy link
Collaborator

Wumpf commented Jul 8, 2024

Re api compatibility: I think for the renderers to handle hdr output correctly some other stuff needs to happen so not too concerned, whoever looks into that can also do something about that. Worst case they have to break the variable name to make it more obvious

@Wumpf Wumpf merged commit b283b8a into emilk:master Jul 8, 2024
19 checks passed
486c pushed a commit to 486c/egui that referenced this pull request Oct 9, 2024
This PR introduces dithering in the egui_glow and egui_wgpu backends to
reduce banding artifacts.

It's based on the approach mentioned in emilk#4493 with the small difference
that the amount of noise is scaled down slightly to avoid dithering
colors that can be represented exactly. This keeps flat surfaces clean.

Exaggerated dithering to show what is happening:
![Screenshot from 2024-05-14
19-09-48](https://github.com/emilk/egui/assets/293536/75782b83-9023-4cb2-99f7-a24e15fdefcc)

Subtle dithering as commited.
![Screenshot from 2024-05-14
19-13-40](https://github.com/emilk/egui/assets/293536/eb904698-a6ec-494a-952b-447e9a49bfda)

Closes emilk#4493
hacknus pushed a commit to hacknus/egui that referenced this pull request Oct 30, 2024
This PR introduces dithering in the egui_glow and egui_wgpu backends to
reduce banding artifacts.

It's based on the approach mentioned in emilk#4493 with the small difference
that the amount of noise is scaled down slightly to avoid dithering
colors that can be represented exactly. This keeps flat surfaces clean.

Exaggerated dithering to show what is happening:
![Screenshot from 2024-05-14
19-09-48](https://github.com/emilk/egui/assets/293536/75782b83-9023-4cb2-99f7-a24e15fdefcc)

Subtle dithering as commited.
![Screenshot from 2024-05-14
19-13-40](https://github.com/emilk/egui/assets/293536/eb904698-a6ec-494a-952b-447e9a49bfda)

Closes emilk#4493
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
egui_glow Relates to running egui_glow on native egui-wgpu
Projects
None yet
Development

Successfully merging this pull request may close these issues.

Dithering in Fragment Shader
4 participants