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

fix: avoid norm16 for textures with linear filtering #3194

Merged
merged 1 commit into from
Jan 2, 2025

Conversation

slak44
Copy link
Contributor

@slak44 slak44 commented Dec 19, 2024

Context

On certain hardware, combining linear texture filtering with the normalized fixed point types from EXT_texture_norm16 results in black pixels and a WebGL warning from the browser.

[.WebGL-0x7202996900]RENDER WARNING: texture bound to texture unit 0 is not renderable. It might be non-power-of-2 or have incompatible texture filtering (maybe)?

According to this proposed webgpu extension, there are a lot of mobile devices that don't support linear filtering for norm16 (eg Adreno 5XX, 7XX devices). This includes recent flagships like the Samsung Galaxy 24 Ultra, where I found this issue.

Unlike floats which have the OES_texture_float_linear extension, it doesn't seem possible to check using webgl if linear filtering is present for norm16 types. It would have been much better to limit the change only when it really is a problem. Not sure if I'm missing anything here, but this seems like a gap in the standard... I might open an issue there to ask
I asked: KhronosGroup/WebGL#3706

Also, here's two recent bugs from cornerstone that I think are a manifestation of this problem. Same error message, same black canvas (except turned gray due to window levels)

The phones mentioned there are Huawei P50, Huawei P60, Huawei Mate 9. These have Adreno/Mali GPUs, matching what the webgpu proposal said.

Results

Texture now acts as if EXT_texture_norm16 is unsupported when LINEAR filtering is chosen. It should fall back to some float type.

Changes

  • disabled norm16 + linear filtering by default when a runtime check determines it won't work
  • added override API if someone wants to force it enableLinearNorm16Ext(useLinearNorm16: boolean): void
  • Documentation and TypeScript definitions were updated to match those changes

PR and Code Checklist

  • semantic-release commit messages
  • Run npm run reformat to have correctly formatted code

Testing

Unfortunately you need a phone that has a problematic GPU. Like the previous texture filtering issue, I tested this with a modified vtk.js directly in OHIF.

If you do have a phone, I found a useful jsfiddle that reproduces the bug. It directly renders a texture with webgl: https://jsfiddle.net/sedghi/hxLegk5y/
This is how it should look, on a desktop GPU where everything is fine:
image

This is how it looks on a phone that has the problem:
Screenshot_20241219_170531_Chrome

And identical code, but replacing LINEAR with NEAREST fixes the black image:
image

@sedghi
Copy link
Contributor

sedghi commented Dec 19, 2024

Can we render a simple offscreen canvas (16x16) using the sample pattern (in your examples) to check if it is black entirely? This will help us decide whether to use the norm16 usage with linear filtering. I don't see any other viable options.

The funny thing is WebKit in iOS also doesn't have support for float linear, which makes me wonder if this approach (rendering something as a template and then deciding on the right path—norm16 or not, float or not, filterable or not) is the maintainable way forward.

@slak44 slak44 force-pushed the fix/norm16-linear-filtering branch from 806ad25 to dd7e000 Compare December 20, 2024 09:00
@slak44
Copy link
Contributor Author

slak44 commented Dec 20, 2024

I added the runtime check on a 4x4 canvas. It should be enough to check one pixel, if it's not black then linear filtering works.

It's probably better for compatibility to do these checks at runtime, but there might be some performance impact. A 4x4 canvas is small so it's fine, but doing the same test on a 512x512 canvas gave me a warning from chrome, about a GPU stall when using readPixels.

OpenGL and its adjacent APIs like WebGL have always had quirks, on top of multiple versions and extensions... I suppose we've been spoiled by how uniformly supported most web/JS stuff has become after IE died.

This only happens on certain devices due to a driver bug, see
KhronosGroup/WebGL#3706
Copy link
Collaborator

@floryst floryst left a comment

Choose a reason for hiding this comment

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

Code-wise this looks good.

@sankhesh
Copy link
Collaborator

LGTM.

On the question of runtime testing of extension functionality - the OpenGL/webGL specs are clear that if the extension is present, it is implemented. Occasionally, we come across such issues where the driver implementers left something out. IMO, testing all the required functionality can be an added step but should not be part of the render step. I can imagine a testing suite runs through and spits out all the supported functionality providing the flags that need to be set for a working vtk-js environment on the current system. These flags can then form a configuration file loaded in each time for future vtk-js app runs.

@floryst floryst added this pull request to the merge queue Jan 2, 2025
Merged via the queue into Kitware:master with commit 3c01212 Jan 2, 2025
3 checks passed
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
None yet
Projects
None yet
Development

Successfully merging this pull request may close these issues.

4 participants