-
Notifications
You must be signed in to change notification settings - Fork 972
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] PendingArraySize::Expression
breaks compaction
#6788
Comments
Should we be rolling back #6654? I don't know the git commands off the top of my head |
@kentslaney I don't think we need to roll it back. This isn't a show-stopper, it's just something we should deal with. |
ok, I'll look into this |
Yeah this sounds like the way to go. I tried making things better without understanding the full requirements today (for #6787) and ultimately my attempt didn't work. |
I wonder if we can get away with saying that no expression can refer to an override-sized array and check this in the validator. |
I filed #6793 for this validation issue. |
This would require not just checking the But one of the principles of the validator is that handle validation runs first, so that the rest of the validator can just assume that whatever handles it runs into are okay. We used to sort of check handles in the midst of everything else, and of course we forgot to do it all the time. Being cycle-free, well, I'm not positive where this guarantee is taken advantage of. But assuming the absence of cycles where they might actually be present is an absolutely classic kind of bug, in every variety of programming, and I think this shows that people sort of generally don't anticipate cycles, whether they should know better or not. So I think it's good for handle validation to just settle the question once and for all, to allow rest of the validator to be naive and common-sensey. And I think that means that we shouldn't be depending on type validation to establish the absence of cycles. |
Thanks for your patience, I took yesterday mostly off. Just to read back my current understanding: This is resolved from a practical standpoint by the two compacting passes already in the code base. In the first one, the override-sized array type gets removed for not being used and in the second the expression gets removed for not being used. I don't think there's anything that's nested more than that. The issue is that the current compacting setup orders the tracing based on a one-directional dependency assumption that's no longer true. Ideally, the example given should be solved in a single pass. The other issue is caused by the same problem, but has different implications. Practically, we know the variants don't have any bidirectional dependancies since types can't be used in array size override expressions. That being said, we should validate that they're acyclic anyway since this is the most obvious time something has changed and it's a convenient guarantee. |
Oh, no problem - it's great to have someone contributing to this area, I think most people are scared off. 😀
Right. In #6800 I changed handle validation to check (Calling something "challenging" makes it sound like a good thing; "error-prone" might be more honest.) |
I've only skimmed your #6800 solution, but you're right, it sounds similar to my current thinking for this issue. My current plan for this one, using the guarantee of cycle-free arenas to avoid stack overflow:
We can simplify this if we assume that the expressions are well ordered since they're not merged like types are (ie. expression A referring to a type referring to expression B has A > B). A similar assumption is made in
E: not sure why the box drawing characters breaks monospace now; it lined up when I posted it originally. |
@jimblandy #6806 is ready for review. |
In compaction, correctly identify unused types and global expressions, rather than treating all global expressions referred to by `PendingArraySize::Expression` array lengths as used even if the array type itself is not. This is tested by checking that going from expression to type and back, along with the opposite, is correctly marked as used. It also checks that adding an unused type using an expression gets compacted out. See the comments on `ModuleTracer::type_expression_tandem` for details. Fixes gfx-rs#6788.
In compaction, correctly identify unused types and global expressions, rather than treating all global expressions referred to by `PendingArraySize::Expression` array lengths as used even if the array type itself is not. This is tested by checking that going from expression to type and back, along with the opposite, is correctly marked as used. It also checks that adding an unused type using an expression gets compacted out. See the comments on `ModuleTracer::type_expression_tandem` for details. Fixes gfx-rs#6788.
The introduction of
PendingArraySize
in #6654 didn't adequately update the compaction algorithm, making it possible for unused expressions and types to leak. I'm not sure this is a problem in practice, but the algorithm is definitely wrong now.PendingArraySize
was introduced to correctly model WGSL's rules around array type equivalence. WGSL treats arrays differently depending on whether their lengths are const expressions, override expressions consisting of a single identifier, or more general override expressions. In light of WGSL's requirements,PendingArraySize
seems like the right way to represent modules.However, the introduction of
PendingArraySize::Expression
was the first time that types could refer to expressions. Expressions can refer to types in a few ways, too. This means that it is possible to have cycles betweenModule::types
andModule::global_expressions
. The compaction algorithm previously assumed that such cycles were impossible, allowing one to first decide which expressions were used, and then retain the types that those expressions needed. But once cycles are possible, expression and type liveness need to be considered simultaneously.The way to see that things have definitely gone awry is to look at the following code added to
compact::compact
(link):This essentially treats all
PendingArraySize::Expression
handles as live by assumption, meaning that they'll be retained even if the override-sized array type itself is unused. This is a leak.Any fix will somehow entail traversing global expressions and types together, which is a pretty big change, unfortunately.
This probably has implications for handle validation as well: the
check_dep
code wants to ensure our arena contents are acyclic.cc @teoxoy @kentslaney
The text was updated successfully, but these errors were encountered: