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

report_similar_impl_candidates seems to not put things in a canonical order #53302

Closed
scottmcm opened this issue Aug 13, 2018 · 3 comments
Closed
Labels
A-diagnostics Area: Messages for errors, warnings, and lints E-easy Call for participation: Easy difficulty. Experience needed to fix: Not much. Good first issue.

Comments

@scottmcm
Copy link
Member

scottmcm commented Aug 13, 2018

I hit the following UI test failure in travis today:

 	   |         ^^^^^^^^ the trait `std::convert::From<&str>` is not implemented for `i32`
 	   |
 	   = help: the following implementations were found:
+	             <i32 as std::convert::From<u8>>
 	             <i32 as std::convert::From<i8>>
-	             <i32 as std::convert::From<bool>>
 	             <i32 as std::convert::From<u16>>
 	             <i32 as std::convert::From<i16>>
-	             <i32 as std::convert::From<u8>>
+	             <i32 as std::convert::From<bool>>
 	   = note: required by `std::convert::From::from`

https://travis-ci.org/rust-lang/rust/jobs/415272826#L1942-L1958

Despite being fully-up-to-date against master and that passing locally.

Perhaps the order is deterministic normally, but different for me because I built --incremental?

I figured it was worth filing, at least, as I can't see any obvious sort in the relevant code:

fn find_similar_impl_candidates(&self,
trait_ref: ty::PolyTraitRef<'tcx>)
-> Vec<ty::TraitRef<'tcx>>
{
let simp = fast_reject::simplify_type(self.tcx,
trait_ref.skip_binder().self_ty(),
true);
let mut impl_candidates = Vec::new();
match simp {
Some(simp) => self.tcx.for_each_impl(trait_ref.def_id(), |def_id| {
let imp = self.tcx.impl_trait_ref(def_id).unwrap();
let imp_simp = fast_reject::simplify_type(self.tcx,
imp.self_ty(),
true);
if let Some(imp_simp) = imp_simp {
if simp != imp_simp {
return;
}
}
impl_candidates.push(imp);
}),
None => self.tcx.for_each_impl(trait_ref.def_id(), |def_id| {
impl_candidates.push(
self.tcx.impl_trait_ref(def_id).unwrap());
})
};
impl_candidates
}
fn report_similar_impl_candidates(&self,
impl_candidates: Vec<ty::TraitRef<'tcx>>,
err: &mut DiagnosticBuilder)
{
if impl_candidates.is_empty() {
return;
}
let end = if impl_candidates.len() <= 5 {
impl_candidates.len()
} else {
4
};
let normalize = |candidate| self.tcx.global_tcx().infer_ctxt().enter(|ref infcx| {
let normalized = infcx
.at(&ObligationCause::dummy(), ty::ParamEnv::empty())
.normalize(candidate)
.ok();
match normalized {
Some(normalized) => format!("\n {:?}", normalized.value),
None => format!("\n {:?}", candidate),
}
});
err.help(&format!("the following implementations were found:{}{}",
&impl_candidates[0..end].iter().map(normalize).collect::<String>(),
if impl_candidates.len() > 5 {
format!("\nand {} others", impl_candidates.len() - 4)
} else {
"".to_owned()
}
));
}

@scottmcm scottmcm added the A-diagnostics Area: Messages for errors, warnings, and lints label Aug 13, 2018
@estebank estebank added the E-easy Call for participation: Easy difficulty. Experience needed to fix: Not much. Good first issue. label Aug 13, 2018
@estebank
Copy link
Contributor

In the code linked above, modify impl_candidates to be mut and sort it first thing in the method report_similar_impl_candidates. You might have to sort the normalized values though, as TraitRef might 1) not be sortable and 2) might sort differently in different targets. Because of this it would make sense to sort on the display value for the fields. This will be less efficient, but we're already reporting an error at this point, so it's ok.

@BurntPizza
Copy link
Contributor

I can take this.

@BurntPizza
Copy link
Contributor

This should be fixed now by #53196.

Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
A-diagnostics Area: Messages for errors, warnings, and lints E-easy Call for participation: Easy difficulty. Experience needed to fix: Not much. Good first issue.
Projects
None yet
Development

No branches or pull requests

3 participants