-
Notifications
You must be signed in to change notification settings - Fork 1.7k
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: Binding wrong associated type when lowering bounds like T: Trait<Assoc = U>
#19182
Conversation
) { | ||
} | ||
"#, | ||
|e| matches!(e, MirEvalError::MirLowerError(_, MirLowerError::GenericArgNotProvided(..))), |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Since the mir evaluation does mir lowering and then mir monomorphizing, this makes panics without the fix in this PR, in the process of solving "x is Copy
?" goals while borrowck
crates/hir-ty/src/lower/path.rs
Outdated
WherePredicate::TypeBound { | ||
target: WherePredicateTypeTarget::TypeOrConstParam(idx), | ||
bound: b, | ||
} if b == bound && self.ctx.types_map == types_map => Some(idx), |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
This seems like it would do a deep comparison of the type map, which is expensive. Is it possible to make a pointer comparison?
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I also fear there are other instances of this pattern, but I can't think of an idea to solve it generally, nor can I think how to solve this more elegantly (comparing the types map is a bit weird).
I agree with you that comparing the type maps feels weird. I considered comparing the actual |
Fixes #19177
The panic happens while borrow checking the following function in the mentioned crate
https://github.com/PingPongun/egui_struct/blob/b9549e51491e02a6c471e1cf7a6bd4b77bd87203/src/lib.rs#L774-L779
When lowering trait object
dyn Iterator<Item = T>
on the following linesrust-analyzer/crates/hir-ty/src/lower.rs
Lines 693 to 704 in 9691da7
on
L704
, the boundTraitObject: Iterator<Item = T>
passed to the following linesrust-analyzer/crates/hir-ty/src/lower.rs
Lines 647 to 663 in 9691da7
and then passed to the following function through
L661
.rust-analyzer/crates/hir-ty/src/lower/path.rs
Lines 787 to 791 in 9691da7
But in that function, while trying to find
target_param_idx
in the following lines,(note that the problematic behaviour happens only when
target
isWherePredicateTypeTarget::TypeOrConstParam
inL844
, which means we have desugaredimpl Trait
generic type as a function's parameter type)rust-analyzer/crates/hir-ty/src/lower/path.rs
Lines 840 to 846 in 9691da7
we check whether the two type bounds are equal(
b == bound
) with<TypeBound as PartialEq>::Eq
However, the
TypeBounds
is somewhat like a local pointer bound to theTypesMap
that contains the actualTypeRef
it is pointing out(thePathId
is actuallyIdx<TypeRef>
rust-analyzer/crates/hir-def/src/hir/type_ref.rs
Lines 253 to 260 in 9691da7
and if
bound
andb
are from two differentTypesMap
but have the sameIdx
, our associated type in trait ref bound is bound to a wrong parameter, and this caused panic inside chalk.Adding equality check for two
TypeBound
s' provenanceTypesMap
fixes this issue, but I'm so scared because similar "LocalIdx
comparison between differentArena
s" might be happening somewhere in our codebase 😨