-
Notifications
You must be signed in to change notification settings - Fork 963
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 front ends do not properly short-circuit &&
and ||
#4394
Comments
I can confirm glsl short circuits |
According to https://devblogs.microsoft.com/directx/announcing-hlsl-2021/#logical-operator-short-circuiting, hlsl only short-circuits in the 2021 version. |
We could store this information in Naga IR's // naga/src/lib.rs
pub enum BinaryOperator {
// ...
LogicalAnd { short_circuits: bool },
LogicalOr { short_circuits: bool },
} For reference, here's the relevant code in the SPIR-V backend: |
I started looking in to this as it seemed like an interesting issue for learning Naga. After doing some reading, I'm starting to think the issue lies with the front ends. When parsing
I'm working on a PR to introduce an IR branch and emit
I think this logic needs to be present in the wgsl and glsl frontends, and only in the hlsl frontend when parsing the 2021 edition. Does this seem like the right approach? Are there performance issues with introducing an if statement and local variable for each operator? |
Yes, I think this does need to be handled in the front end. In Naga IR, function calls are not expressions, they're statements. This is explained in the top-level comments in lib.rs, but roughly speaking, expressions are never supposed to have side effects, and function calls can have side effects, so they get lifted out into independent statements. So the translation of WGSL
This means that there's no reasonable way for back ends to apply the short circuiting when they see So your conclusion - that the front end needs to turn a How would you translate longer chains like |
Makes sense. I hadn't taken a good look at the IR yet by the time I made my suggestion; my bad. I agree with this approach. |
@jimblandy I thought the IR was supposed to follow wgsl semantics and the backends would do the necessary transformations. If that's the case then this should be handled in the backends because wgsl defines short circuiting for the operators, also both glsl and msl define short circuiting in it's semantics, so only 2/5 backends need this transform and I think it doesn't make sense to add a possible performance penalty to the other ones. |
I'm not sure what "follows wgsl semantics" means exactly, but this is how short-circuiting operators get lowered into any IR that lacks expression side effects. All frontends of languages with side effects in their expressions will need this transform, although we could consider omitting the branch in specific cases if there are no possible side effects on the RHS. My thought is that GPU compilers probably do the same transform in their IRs anyway and then optimize out the branch later, but I'm not sure of this and if the compilers do care about if statements vs short-circuiting operators, it wouldn't be too hard to make this slightly smarter. It could also be worthwhile to improve compilation time. |
The IR is supposed to mirror wgsl, including the semantics of it's operations as defined in the wgsl specification.
We don't know the internal IR of the graphics device so we must design around the interfaces provided to us, and in this case only 2 of the supported backends don't support short-circuiting, so I think it's better to make the transformations on those and let the others properly encode the operation.
We can do it in the backends :)
naga doesn't optimize
It's not our job to guess what the compiler does, that in itself would be impossible because there are many and they target many different architectures, we should just encode the program as closely as possible to souce.
I doubt any compiler spends a significant amount on this transform, and even if some does another could not have the transform and optimizing the branches to a short-circuit take more time. This is all speculation, so that's why we should just encode as closely to source as possible and let compilers do their job. |
How can you do this in the backend without changing the semantics of the shader? None of the current backends "support short circuiting" because our IR itself does not support short circuiting. |
The IR implicitly assumes short circuiting, uniformity analysis might not be correct but that's another issue, and both glsl, MSL, and wgsl have short circuiting |
I'm new to the code here, but as far as I can tell the IR explicitly does not allow side effects inside expressions, and encodes all control flow and evaluation order constraints through statements. This means that whether or not the logical operators short circuit is irrelevant, because there is no difference either way. This is a good IR design IMO, and very typical for these sorts of things. How do you think this should be solved? Should we allow expressions to have side effects in the IR? Here's the comments I'm referring to: https://github.com/gfx-rs/naga/blob/master/src/lib.rs#L27 |
Oh yeah I see the problem, sorry totally forgot that the IR pushes side effects to statements. In that case, yeah, I think we should go with your solution for now and we can then later if we see the need change it to something more purpose built (like a short circuiting statement is something like that) |
Note that HLSL also doesn't short-circuit until Language Versions 2021. Also checked MSL which always does. |
&&
and ||
&&
and ||
&&
and ||
&&
and ||
NOTE: description edited 2024-4-16, so much of the following discussion may look irrelevant
The WGSL specification says that
a && b
only evaluatesb
ifa
is true, but Naga generally will always evaluate both.The following input:
produces the following output WGSL:
The call to
g
is hoisted out to a statement and made unconditional, which is incorrect.This means that even though most backends turn Naga IR's
BinaryOperator::LogicalAnd
andBinaryOperator::LogicalOr
into short-circuiting operations in the target language, it doesn't help because the front end has already hoisted the right-hand side, which ought to be conditional, out into its own unconditional statement.The text was updated successfully, but these errors were encountered: