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: Binding wrong associated type when lowering bounds like T: Trait<Assoc = U> #19182

Merged
merged 2 commits into from
Feb 22, 2025

Conversation

ShoyuVanilla
Copy link
Member

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

fn show_combobox<'a, T: Clone + ToString + PartialEq>(
    sel: &mut T,
    ui: &mut Ui,
    config: Option<&'a mut dyn Iterator<Item = T>>,
    id: impl Hash + Clone,
) -> Response {

When lowering trait object dyn Iterator<Item = T> on the following lines

fn lower_dyn_trait(&mut self, bounds: &[TypeBound]) -> Ty {
let self_ty = TyKind::BoundVar(BoundVar::new(DebruijnIndex::INNERMOST, 0)).intern(Interner);
// INVARIANT: The principal trait bound, if present, must come first. Others may be in any
// order but should be in the same order for the same set but possibly different order of
// bounds in the input.
// INVARIANT: If this function returns `DynTy`, there should be at least one trait bound.
// These invariants are utilized by `TyExt::dyn_trait()` and chalk.
let mut lifetime = None;
let bounds = self.with_shifted_in(DebruijnIndex::ONE, |ctx| {
let mut lowered_bounds = Vec::new();
for b in bounds {
ctx.lower_type_bound(b, self_ty.clone(), false).for_each(|b| {

on L704, the bound TraitObject: Iterator<Item = T> passed to the following lines
pub(crate) fn lower_type_bound<'b>(
&'b mut self,
bound: &'b TypeBound,
self_ty: Ty,
ignore_bindings: bool,
) -> impl Iterator<Item = QuantifiedWhereClause> + use<'b, 'a> {
let mut assoc_bounds = None;
let mut clause = None;
match bound {
&TypeBound::Path(path, TraitBoundModifier::None) | &TypeBound::ForLifetime(_, path) => {
// FIXME Don't silently drop the hrtb lifetimes here
if let Some((trait_ref, ctx)) = self.lower_trait_ref_from_path(path, self_ty) {
if !ignore_bindings {
assoc_bounds =
ctx.assoc_type_bindings_from_type_bound(bound, trait_ref.clone());
}
clause = Some(crate::wrap_empty_binders(WhereClause::Implemented(trait_ref)));

and then passed to the following function through L661.
pub(super) fn assoc_type_bindings_from_type_bound<'c>(
mut self,
bound: &'c TypeBound,
trait_ref: TraitRef,
) -> Option<impl Iterator<Item = QuantifiedWhereClause> + use<'a, 'b, 'c>> {

But in that function, while trying to find target_param_idx in the following lines,
(note that the problematic behaviour happens only when target is WherePredicateTypeTarget::TypeOrConstParam in L844, which means we have desugared impl Trait generic type as a function's parameter type)
let target_param_idx =
self.ctx.resolver.where_predicates_in_scope().find_map(|(p, _)| {
match p {
WherePredicate::TypeBound {
target: WherePredicateTypeTarget::TypeOrConstParam(idx),
bound: b,
} if b == bound => Some(idx),

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 the TypesMap that contains the actual TypeRef it is pointing out(the PathId is actually Idx<TypeRef>

#[derive(Clone, PartialEq, Eq, Hash, Debug)]
pub enum TypeBound {
Path(PathId, TraitBoundModifier),
ForLifetime(Box<[Name]>, PathId),
Lifetime(LifetimeRef),
Use(Box<[UseArgRef]>),
Error,
}

and if bound and b are from two different TypesMap but have the same Idx, our associated type in trait ref bound is bound to a wrong parameter, and this caused panic inside chalk.

Adding equality check for two TypeBounds' provenance TypesMap fixes this issue, but I'm so scared because similar "Local Idx comparison between different Arenas" might be happening somewhere in our codebase 😨

@rustbot rustbot added the S-waiting-on-review Status: Awaiting review from the assignee but also interested parties. label Feb 19, 2025
) {
}
"#,
|e| matches!(e, MirEvalError::MirLowerError(_, MirLowerError::GenericArgNotProvided(..))),
Copy link
Member Author

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

WherePredicate::TypeBound {
target: WherePredicateTypeTarget::TypeOrConstParam(idx),
bound: b,
} if b == bound && self.ctx.types_map == types_map => Some(idx),
Copy link
Contributor

@ChayimFriedman2 ChayimFriedman2 Feb 19, 2025

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?

Copy link
Contributor

@ChayimFriedman2 ChayimFriedman2 left a 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).

@ChayimFriedman2 ChayimFriedman2 added this pull request to the merge queue Feb 22, 2025
@ShoyuVanilla
Copy link
Member Author

I agree with you that comparing the type maps feels weird. I considered comparing the actual TyRefs inside the map instead, but that also seems problematic, as non-lowered TyRefs can semantically refer to different things. That said, I doubt this issue arises in the code I modified in this PR 😅.
Anyway, I chose to compare the maps to enforce a stricter "same origin" policy.

Merged via the queue into rust-lang:master with commit 9df88ff Feb 22, 2025
9 checks passed
@ShoyuVanilla ShoyuVanilla deleted the issue-19177 branch February 22, 2025 22:21
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
S-waiting-on-review Status: Awaiting review from the assignee but also interested parties.
Projects
None yet
Development

Successfully merging this pull request may close these issues.

Panic at chalk-ir-0.98.0/src/lib.rs:2732:10
3 participants