Skip to content

Commit 1de894f

Browse files
More tests and tweak comments
1 parent af24d0b commit 1de894f

11 files changed

+298
-11
lines changed

compiler/rustc_hir_analysis/src/collect/resolve_bound_vars.rs

+15-4
Original file line numberDiff line numberDiff line change
@@ -1640,9 +1640,9 @@ impl<'a, 'tcx> BoundVarContext<'a, 'tcx> {
16401640
// }
16411641
// ```
16421642
// and a bound that looks like:
1643-
// `for<'a> T::Trait<'a, x(): for<'b> Other<'b>>`
1643+
// `for<'a> T::Trait<'a, x(..): for<'b> Other<'b>>`
16441644
// this is going to expand to something like:
1645-
// `for<'a> for<'r, T> <T as Trait<'a>>::x::<'r, T>::{opaque#0}: for<'b> Other<'b>`.
1645+
// `for<'a> for<'r> <T as Trait<'a>>::x::<'r, T>::{opaque#0}: for<'b> Other<'b>`.
16461646
if constraint.gen_args.parenthesized == hir::GenericArgsParentheses::ReturnTypeNotation
16471647
{
16481648
let bound_vars = if let Some(type_def_id) = type_def_id
@@ -1853,13 +1853,13 @@ impl<'a, 'tcx> BoundVarContext<'a, 'tcx> {
18531853
// For example, given
18541854
// ```
18551855
// trait Foo {
1856-
// async fn x<'r, T>();
1856+
// async fn x<'r>();
18571857
// }
18581858
// ```
18591859
// and a bound that looks like:
18601860
// `for<'a, 'b> <T as Trait<'a>>::x(): Other<'b>`
18611861
// this is going to expand to something like:
1862-
// `for<'a, 'b, 'r, T> <T as Trait<'a>>::x::<'r, T>::{opaque#0}: Other<'b>`.
1862+
// `for<'a, 'b, 'r> <T as Trait<'a>>::x::<'r, T>::{opaque#0}: Other<'b>`.
18631863
//
18641864
// We handle this similarly for associated-type-bound style return-type-notation
18651865
// in `visit_segment_args`.
@@ -2000,6 +2000,17 @@ impl<'a, 'tcx> BoundVarContext<'a, 'tcx> {
20002000
// the new bound vars. We do this because we need to know how many bound vars
20012001
// are present on the binder explicitly (i.e. not return-type-notation vars)
20022002
// to do bound var shifting correctly in HIR lowering.
2003+
//
2004+
// For example, in `where for<'a> <T as Trait<'a>>::method(..): Other`,
2005+
// the `late_bound_vars` of the where clause predicate (i.e. this HIR ty's
2006+
// parent) will include `'a` AND all the early- and late-bound vars of the
2007+
// method. But when lowering the RTN type, we just want the list of vars
2008+
// we used to resolve the trait ref. We explicitly stored those back onto
2009+
// the item segment, since there's no other good place to put them.
2010+
//
2011+
// See where these vars are used in `HirTyLowerer::lower_ty_maybe_return_type_notation`.
2012+
// And this is exercised in:
2013+
// `tests/ui/associated-type-bounds/return-type-notation/higher-ranked-bound-works.rs`.
20032014
let existing_bound_vars = self.map.late_bound_vars.get_mut(&hir_id).unwrap();
20042015
let existing_bound_vars_saved = existing_bound_vars.clone();
20052016
existing_bound_vars.extend(bound_vars);

tests/ui/associated-type-bounds/return-type-notation/bad-inputs-and-output.rs

+18
Original file line numberDiff line numberDiff line change
@@ -16,4 +16,22 @@ fn bar<T: Trait<method() -> (): Send>>() {}
1616
fn baz<T: Trait<method(): Send>>() {}
1717
//~^ ERROR return type notation arguments must be elided with `..`
1818

19+
fn foo_path<T: Trait>() where T::method(i32): Send {}
20+
//~^ ERROR argument types not allowed with return type notation
21+
22+
fn bar_path<T: Trait>() where T::method() -> (): Send {}
23+
//~^ ERROR return type not allowed with return type notation
24+
25+
fn baz_path<T: Trait>() where T::method(): Send {}
26+
//~^ ERROR return type notation arguments must be elided with `..`
27+
28+
fn foo_qualified<T: Trait>() where <T as Trait>::method(i32): Send {}
29+
//~^ ERROR expected associated type
30+
31+
fn bar_qualified<T: Trait>() where <T as Trait>::method() -> (): Send {}
32+
//~^ ERROR expected associated type
33+
34+
fn baz_qualified<T: Trait>() where <T as Trait>::method(): Send {}
35+
//~^ ERROR expected associated type
36+
1937
fn main() {}

tests/ui/associated-type-bounds/return-type-notation/bad-inputs-and-output.stderr

+38-1
Original file line numberDiff line numberDiff line change
@@ -1,3 +1,21 @@
1+
error[E0575]: expected associated type, found associated function `Trait::method`
2+
--> $DIR/bad-inputs-and-output.rs:28:36
3+
|
4+
LL | fn foo_qualified<T: Trait>() where <T as Trait>::method(i32): Send {}
5+
| ^^^^^^^^^^^^^^^^^^^^^^^^^ not a associated type
6+
7+
error[E0575]: expected associated type, found associated function `Trait::method`
8+
--> $DIR/bad-inputs-and-output.rs:31:36
9+
|
10+
LL | fn bar_qualified<T: Trait>() where <T as Trait>::method() -> (): Send {}
11+
| ^^^^^^^^^^^^^^^^^^^^^^^^^^^^ not a associated type
12+
13+
error[E0575]: expected associated type, found associated function `Trait::method`
14+
--> $DIR/bad-inputs-and-output.rs:34:36
15+
|
16+
LL | fn baz_qualified<T: Trait>() where <T as Trait>::method(): Send {}
17+
| ^^^^^^^^^^^^^^^^^^^^^^ not a associated type
18+
119
warning: the feature `return_type_notation` is incomplete and may not be safe to use and/or cause compiler crashes
220
--> $DIR/bad-inputs-and-output.rs:3:12
321
|
@@ -25,5 +43,24 @@ error: return type notation arguments must be elided with `..`
2543
LL | fn baz<T: Trait<method(): Send>>() {}
2644
| ^^ help: add `..`: `(..)`
2745

28-
error: aborting due to 3 previous errors; 1 warning emitted
46+
error: argument types not allowed with return type notation
47+
--> $DIR/bad-inputs-and-output.rs:19:40
48+
|
49+
LL | fn foo_path<T: Trait>() where T::method(i32): Send {}
50+
| ^^^^^ help: remove the input types: `()`
51+
52+
error: return type not allowed with return type notation
53+
--> $DIR/bad-inputs-and-output.rs:22:42
54+
|
55+
LL | fn bar_path<T: Trait>() where T::method() -> (): Send {}
56+
| ^^^^^^ help: remove the return type
57+
58+
error: return type notation arguments must be elided with `..`
59+
--> $DIR/bad-inputs-and-output.rs:25:40
60+
|
61+
LL | fn baz_path<T: Trait>() where T::method(): Send {}
62+
| ^^ help: add `..`: `(..)`
63+
64+
error: aborting due to 9 previous errors; 1 warning emitted
2965

66+
For more information about this error, try `rustc --explain E0575`.
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,52 @@
1+
//@ check-pass
2+
3+
#![feature(return_type_notation)]
4+
//~^ WARN the feature `return_type_notation` is incomplete
5+
6+
trait Trait<'a> {
7+
fn late<'b>(&'b self, _: &'a ()) -> impl Sized;
8+
fn early<'b: 'b>(&'b self, _: &'a ()) -> impl Sized;
9+
}
10+
11+
#[allow(refining_impl_trait_internal)]
12+
impl<'a> Trait<'a> for () {
13+
fn late<'b>(&'b self, _: &'a ()) -> i32 { 1 }
14+
fn early<'b: 'b>(&'b self, _: &'a ()) -> i32 { 1 }
15+
}
16+
17+
trait Other<'c> {}
18+
impl Other<'_> for i32 {}
19+
20+
fn test<T>(t: &T)
21+
where
22+
T: for<'a, 'c> Trait<'a, late(..): Other<'c>>,
23+
// which is basically:
24+
// for<'a, 'c> Trait<'a, for<'b> method<'b>: Other<'c>>,
25+
T: for<'a, 'c> Trait<'a, early(..): Other<'c>>,
26+
// which is basically:
27+
// for<'a, 'c> Trait<'a, for<'b> method<'b>: Other<'c>>,
28+
{
29+
is_other_impl(t.late(&()));
30+
is_other_impl(t.early(&()));
31+
}
32+
33+
fn test_path<T>(t: &T)
34+
where
35+
T: for<'a> Trait<'a>,
36+
for<'a, 'c> <T as Trait<'a>>::late(..): Other<'c>,
37+
// which is basically:
38+
// for<'a, 'b, 'c> <T as Trait<'a>>::method::<'b>: Other<'c>
39+
for<'a, 'c> <T as Trait<'a>>::early(..): Other<'c>,
40+
// which is basically:
41+
// for<'a, 'b, 'c> <T as Trait<'a>>::method::<'b>: Other<'c>
42+
{
43+
is_other_impl(t.late(&()));
44+
is_other_impl(t.early(&()));
45+
}
46+
47+
fn is_other_impl(_: impl for<'c> Other<'c>) {}
48+
49+
fn main() {
50+
test(&());
51+
test(&());
52+
}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,11 @@
1+
warning: the feature `return_type_notation` is incomplete and may not be safe to use and/or cause compiler crashes
2+
--> $DIR/higher-ranked-bound-works.rs:3:12
3+
|
4+
LL | #![feature(return_type_notation)]
5+
| ^^^^^^^^^^^^^^^^^^^^
6+
|
7+
= note: see issue #109417 <https://github.com/rust-lang/rust/issues/109417> for more information
8+
= note: `#[warn(incomplete_features)]` on by default
9+
10+
warning: 1 warning emitted
11+
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,42 @@
1+
//@ check-pass
2+
3+
#![allow(non_camel_case_types)]
4+
#![feature(return_type_notation)]
5+
//~^ WARN the feature `return_type_notation` is incomplete
6+
7+
trait Foo {
8+
type test;
9+
10+
fn test() -> impl Bar;
11+
}
12+
13+
fn call_path<T: Foo>()
14+
where
15+
T::test(..): Bar,
16+
{
17+
}
18+
19+
fn call_bound<T: Foo<test(..): Bar>>() {}
20+
21+
trait Bar {}
22+
struct NotBar;
23+
struct YesBar;
24+
impl Bar for YesBar {}
25+
26+
impl Foo for () {
27+
type test = NotBar;
28+
29+
// Use refinement here so we can observe `YesBar: Bar`.
30+
#[allow(refining_impl_trait_internal)]
31+
fn test() -> YesBar {
32+
YesBar
33+
}
34+
}
35+
36+
fn main() {
37+
// If `T::test(..)` resolved to the GAT (erroneously), then this would be
38+
// an error since `<() as Foo>::bar` -- the associated type -- does not
39+
// implement `Bar`, but the return type of the method does.
40+
call_path::<()>();
41+
call_bound::<()>();
42+
}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,11 @@
1+
warning: the feature `return_type_notation` is incomplete and may not be safe to use and/or cause compiler crashes
2+
--> $DIR/namespace-conflict.rs:4:12
3+
|
4+
LL | #![feature(return_type_notation)]
5+
| ^^^^^^^^^^^^^^^^^^^^
6+
|
7+
= note: see issue #109417 <https://github.com/rust-lang/rust/issues/109417> for more information
8+
= note: `#[warn(incomplete_features)]` on by default
9+
10+
warning: 1 warning emitted
11+

tests/ui/associated-type-bounds/return-type-notation/non-rpitit.rs

+5-2
Original file line numberDiff line numberDiff line change
@@ -5,7 +5,10 @@ trait Trait {
55
fn method() {}
66
}
77

8-
fn test<T: Trait<method(..): Send>>() {}
9-
//~^ ERROR return type notation used on function that is not `async` and does not return `impl Trait`
8+
fn bound<T: Trait<method(..): Send>>() {}
9+
//~^ ERROR return type notation used on function that is not `async` and does not return `impl Trait`
10+
11+
fn path<T>() where T: Trait, T::method(..): Send {}
12+
//~^ ERROR return type notation used on function that is not `async` and does not return `impl Trait`
1013

1114
fn main() {}

tests/ui/associated-type-bounds/return-type-notation/non-rpitit.stderr

+15-4
Original file line numberDiff line numberDiff line change
@@ -8,15 +8,26 @@ LL | #![feature(return_type_notation)]
88
= note: `#[warn(incomplete_features)]` on by default
99

1010
error: return type notation used on function that is not `async` and does not return `impl Trait`
11-
--> $DIR/non-rpitit.rs:8:18
11+
--> $DIR/non-rpitit.rs:8:19
1212
|
1313
LL | fn method() {}
1414
| ----------- this function must be `async` or return `impl Trait`
1515
...
16-
LL | fn test<T: Trait<method(..): Send>>() {}
17-
| ^^^^^^^^^^^^^^^^
16+
LL | fn bound<T: Trait<method(..): Send>>() {}
17+
| ^^^^^^^^^^^^^^^^
1818
|
1919
= note: function returns `()`, which is not compatible with associated type return bounds
2020

21-
error: aborting due to 1 previous error; 1 warning emitted
21+
error: return type notation used on function that is not `async` and does not return `impl Trait`
22+
--> $DIR/non-rpitit.rs:11:30
23+
|
24+
LL | fn method() {}
25+
| ----------- this function must be `async` or return `impl Trait`
26+
...
27+
LL | fn path<T>() where T: Trait, T::method(..): Send {}
28+
| ^^^^^^^^^^^^^
29+
|
30+
= note: function returns `()`, which is not compatible with associated type return bounds
31+
32+
error: aborting due to 2 previous errors; 1 warning emitted
2233

Original file line numberDiff line numberDiff line change
@@ -0,0 +1,42 @@
1+
#![feature(return_type_notation)]
2+
//~^ WARN the feature `return_type_notation` is incomplete
3+
4+
fn function() {}
5+
6+
fn not_a_method()
7+
where
8+
function(..): Send,
9+
//~^ ERROR expected function, found function `function`
10+
//~| ERROR return type notation not allowed in this position yet
11+
{
12+
}
13+
14+
fn not_a_method_and_typoed()
15+
where
16+
function(): Send,
17+
//~^ ERROR expected type, found function `function`
18+
{
19+
}
20+
21+
trait Tr {
22+
fn method();
23+
}
24+
25+
// Forgot the `T::`
26+
fn maybe_method_overlaps<T: Tr>()
27+
where
28+
method(..): Send,
29+
//~^ ERROR cannot find function `method` in this scope
30+
//~| ERROR return type notation not allowed in this position yet
31+
{
32+
}
33+
34+
// Forgot the `T::`, AND typoed `(..)` to `()`
35+
fn maybe_method_overlaps_and_typoed<T: Tr>()
36+
where
37+
method(): Send,
38+
//~^ ERROR cannot find type `method` in this scope
39+
{
40+
}
41+
42+
fn main() {}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,49 @@
1+
error[E0575]: expected function, found function `function`
2+
--> $DIR/not-a-method.rs:8:5
3+
|
4+
LL | function(..): Send,
5+
| ^^^^^^^^^^^^ not a function
6+
7+
error[E0573]: expected type, found function `function`
8+
--> $DIR/not-a-method.rs:16:5
9+
|
10+
LL | function(): Send,
11+
| ^^^^^^^^^^ not a type
12+
13+
error[E0576]: cannot find function `method` in this scope
14+
--> $DIR/not-a-method.rs:28:5
15+
|
16+
LL | method(..): Send,
17+
| ^^^^^^ not found in this scope
18+
19+
error[E0412]: cannot find type `method` in this scope
20+
--> $DIR/not-a-method.rs:37:5
21+
|
22+
LL | method(): Send,
23+
| ^^^^^^ not found in this scope
24+
25+
warning: the feature `return_type_notation` is incomplete and may not be safe to use and/or cause compiler crashes
26+
--> $DIR/not-a-method.rs:1:12
27+
|
28+
LL | #![feature(return_type_notation)]
29+
| ^^^^^^^^^^^^^^^^^^^^
30+
|
31+
= note: see issue #109417 <https://github.com/rust-lang/rust/issues/109417> for more information
32+
= note: `#[warn(incomplete_features)]` on by default
33+
34+
error: return type notation not allowed in this position yet
35+
--> $DIR/not-a-method.rs:8:5
36+
|
37+
LL | function(..): Send,
38+
| ^^^^^^^^^^^^
39+
40+
error: return type notation not allowed in this position yet
41+
--> $DIR/not-a-method.rs:28:5
42+
|
43+
LL | method(..): Send,
44+
| ^^^^^^^^^^
45+
46+
error: aborting due to 6 previous errors; 1 warning emitted
47+
48+
Some errors have detailed explanations: E0412, E0573, E0575, E0576.
49+
For more information about an error, try `rustc --explain E0412`.

0 commit comments

Comments
 (0)