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

[naga hlsl-out msl-out spv-out] Avoid undefined behaviour when casting floats to integers #7422

Merged
merged 1 commit into from
Mar 27, 2025

Conversation

jamienicol
Copy link
Contributor

@jamienicol jamienicol commented Mar 25, 2025

Connections
Depends on #7421 (not functionally, but they touch adjacent code so easier to land that first)
Depends on #7424
Fixes #7386
Fixes #7387

Description
We currently hit undefined behaviour when casting from floats to integers if the value is out of range for the target integer type. This avoids that. Additionally we must ensure we adhere to the spec's requirements:

If X is exactly representable in the target type T, then the result is that value.

Otherwise, the result is the value in T closest to truncate(X) and also exactly representable in the original floating point type.

This can be achieved by clamp()ing to the minimum and maximum floating point values that are also representable in the target type, before casting.

Even though F64, I64, and U64 are not part of the WGSL spec we use the same behaviour when casting to/from those types, as it seems the less surprising choice.

Testing
Added new snapshot tests to ensure we handle this correctly during const eval.

I tested by hand with a compute shader that runtime values were as expected (on Windows/DXC, Linux/Vulkan, and Macos/Metal) but there is nothing in the test suite.

Squash or Rebase?

Either

Checklist

  • Run cargo fmt.
  • Run cargo clippy --tests. If applicable, add:
  • Run cargo xtask test to run tests.
  • If this contains user-facing changes, add a CHANGELOG.md entry.

@jamienicol jamienicol marked this pull request as draft March 25, 2025 17:00
@jamienicol
Copy link
Contributor Author

Failing test due to a newly discovered bug. So also depends on #7424

@jamienicol jamienicol force-pushed the cast-float-to-int-ub branch 4 times, most recently from e7815de to 7a3923d Compare March 26, 2025 14:13
…g floats to integers

Currently we generate code to convert floating point values to integers
using constructor-style casts in HLSL, static_cast in MSL, and
OpConvertFToS/OpConvertFToU instructions in SPV. Unfortunately the
behaviour of these operations is undefined when the original value is
outside of the range of the target type.

This patch avoids undefined behaviour by first clamping the value to
be inside the target type's range, then performing the cast.
Additionally, we specifically clamp to the minimum and maximum values
that are exactly representable in both the original and the target
type, as per the WGSL spec[1]. Note that these may not be the same as
the minimum and maximum values of the target type.

We additionally must ensure we clamp in the same manner for
conversions during const evaluation. Lastly, although not part of the
WGSL spec, we do the same for casting from F64 and/or to I64 or U64.

[1] https://www.w3.org/TR/WGSL/#floating-point-conversion
Copy link
Member

@teoxoy teoxoy left a comment

Choose a reason for hiding this comment

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

Looks great!

@teoxoy teoxoy merged commit 4791731 into gfx-rs:trunk Mar 27, 2025
37 checks passed
@jamienicol jamienicol deleted the cast-float-to-int-ub branch March 27, 2025 16:02
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
None yet
Projects
None yet
2 participants