Skip to content

Commit 22932f1

Browse files
authored
Unrolled build for rust-lang#133519
Rollup merge of rust-lang#133519 - compiler-errors:xform-ret-wf, r=lcnr Check `xform_ret_ty` for WF in the new solver to improve method winnowing This is a bit interesting. Method probing in the old solver is stronger than the new solver because eagerly normalizing types causes us to check their corresponding trait goals. This is important because we don't end up checking all of the where clauses of a method when method probing; just the where clauses of the impl. i.e., for: ``` impl Foo where WC1, { fn method() where WC2, {} } ``` We only check WC1 and not WC2. This is because at this point in probing the method is instantiated w/ infer vars, and checking the where clauses in WC2 will lead to cycles if we were to check them (at least that's my understanding; I could investigate changing that in general, incl. in the old solver, but I don't have much confidence that it won't lead to really bad overflows.) This PR chooses to emulate the old solver by just checking that the return type is WF. This is theoretically stronger, but I'm not too worried about it. I think we alternatively have several approaches we can take here, though this one seems the simplest. Thoughts? r? lcnr
2 parents eddb717 + 4120fdb commit 22932f1

File tree

2 files changed

+69
-0
lines changed

2 files changed

+69
-0
lines changed

compiler/rustc_hir_typeck/src/method/probe.rs

+22
Original file line numberDiff line numberDiff line change
@@ -1642,6 +1642,28 @@ impl<'a, 'tcx> ProbeContext<'a, 'tcx> {
16421642
}
16431643
}
16441644

1645+
// FIXME(-Znext-solver): See the linked issue below.
1646+
// <https://github.com/rust-lang/trait-system-refactor-initiative/issues/134>
1647+
//
1648+
// In the new solver, check the well-formedness of the return type.
1649+
// This emulates, in a way, the predicates that fall out of
1650+
// normalizing the return type in the old solver.
1651+
//
1652+
// We alternatively could check the predicates of the method itself hold,
1653+
// but we intentionally do not do this in the old solver b/c of cycles,
1654+
// and doing it in the new solver would be stronger. This should be fixed
1655+
// in the future, since it likely leads to much better method winnowing.
1656+
if let Some(xform_ret_ty) = xform_ret_ty
1657+
&& self.infcx.next_trait_solver()
1658+
{
1659+
ocx.register_obligation(traits::Obligation::new(
1660+
self.tcx,
1661+
cause.clone(),
1662+
self.param_env,
1663+
ty::ClauseKind::WellFormed(xform_ret_ty.into()),
1664+
));
1665+
}
1666+
16451667
// Evaluate those obligations to see if they might possibly hold.
16461668
for error in ocx.select_where_possible() {
16471669
result = ProbeResult::NoMatch;
+47
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,47 @@
1+
//@ check-pass
2+
//@ compile-flags: -Znext-solver
3+
4+
use std::ops::Deref;
5+
6+
pub struct List<T> {
7+
skel: [T],
8+
}
9+
10+
impl<'a, T: Copy> IntoIterator for &'a List<T> {
11+
type Item = T;
12+
type IntoIter = std::iter::Copied<<&'a [T] as IntoIterator>::IntoIter>;
13+
14+
fn into_iter(self) -> Self::IntoIter {
15+
todo!()
16+
}
17+
}
18+
19+
impl<T> Deref for List<T> {
20+
type Target = [T];
21+
22+
fn deref(&self) -> &[T] {
23+
todo!()
24+
}
25+
}
26+
27+
impl<T> List<T> {
28+
fn iter(&self) -> <&Self as IntoIterator>::IntoIter
29+
where
30+
T: Copy,
31+
{
32+
todo!()
33+
}
34+
}
35+
36+
fn test<Q>(t: &List<Q>) {
37+
// Checking that `<&List<Q> as IntoIterator>::IntoIter` is WF
38+
// will disqualify the inherent method, since normalizing it
39+
// requires `Q: Copy` which does not hold. and allow us to fall
40+
// through to the deref'd `<[Q]>::iter` method which works.
41+
//
42+
// In the old solver, the same behavior is achieved by just
43+
// eagerly normalizing the return type.
44+
t.iter();
45+
}
46+
47+
fn main() {}

0 commit comments

Comments
 (0)