2021-06-20 14:37:50 +00:00
use cov_mark ::check ;
2020-08-21 11:19:31 +00:00
use expect_test ::expect ;
2019-12-15 20:06:08 +00:00
2021-06-20 14:37:50 +00:00
use super ::{ check , check_infer , check_infer_with_mismatches , check_types } ;
2020-05-20 10:59:20 +00:00
2019-12-03 12:38:54 +00:00
#[ test ]
fn infer_await ( ) {
2020-06-29 15:22:47 +00:00
check_types (
2019-12-03 12:38:54 +00:00
r #"
2021-06-15 19:49:00 +00:00
//- minicore: future
2019-12-03 12:38:54 +00:00
struct IntFuture ;
2021-06-15 19:49:00 +00:00
impl core ::future ::Future for IntFuture {
2019-12-03 12:38:54 +00:00
type Output = u64 ;
}
fn test ( ) {
let r = IntFuture ;
let v = r . await ;
2020-06-29 15:22:47 +00:00
v ;
} //^ u64
2019-12-03 12:38:54 +00:00
" #,
) ;
}
2019-12-24 11:45:28 +00:00
#[ test ]
fn infer_async ( ) {
2020-06-29 15:22:47 +00:00
check_types (
2019-12-24 11:45:28 +00:00
r #"
2021-06-15 19:59:51 +00:00
//- minicore: future
async fn foo ( ) -> u64 { 128 }
2019-12-24 11:45:28 +00:00
fn test ( ) {
let r = foo ( ) ;
let v = r . await ;
2020-06-29 15:22:47 +00:00
v ;
} //^ u64
2019-12-24 11:45:28 +00:00
" #,
) ;
}
#[ test ]
fn infer_desugar_async ( ) {
2020-06-29 15:22:47 +00:00
check_types (
2019-12-24 11:45:28 +00:00
r #"
2021-06-15 22:23:04 +00:00
//- minicore: future, sized
2021-06-15 19:59:51 +00:00
async fn foo ( ) -> u64 { 128 }
2019-12-24 11:45:28 +00:00
fn test ( ) {
let r = foo ( ) ;
2020-06-29 15:22:47 +00:00
r ;
} //^ impl Future<Output = u64>
2019-12-24 11:45:28 +00:00
" #,
) ;
}
2020-09-10 12:01:23 +00:00
#[ test ]
fn infer_async_block ( ) {
check_types (
r #"
2021-06-15 19:59:51 +00:00
//- minicore: future, option
2020-09-10 12:01:23 +00:00
async fn test ( ) {
let a = async { 42 } ;
a ;
// ^ impl Future<Output = i32>
let x = a . await ;
x ;
// ^ i32
let b = async { } . await ;
b ;
// ^ ()
let c = async {
2021-06-15 19:59:51 +00:00
let y = None ;
2020-09-10 12:01:23 +00:00
y
// ^ Option<u64>
} ;
let _ : Option < u64 > = c . await ;
c ;
// ^ impl Future<Output = Option<u64>>
}
" #,
) ;
}
2019-12-03 12:38:54 +00:00
#[ test ]
fn infer_try ( ) {
2020-06-29 15:22:47 +00:00
check_types (
2019-12-03 12:38:54 +00:00
r #"
2020-06-11 14:22:31 +00:00
//- /main.rs crate:main deps:core
2019-12-03 12:38:54 +00:00
fn test ( ) {
let r : Result < i32 , u64 > = Result ::Ok ( 1 ) ;
let v = r ? ;
2020-06-29 15:22:47 +00:00
v ;
} //^ i32
2019-12-03 12:38:54 +00:00
2020-06-11 14:22:31 +00:00
//- /core.rs crate:core
2021-06-01 11:39:19 +00:00
pub mod ops {
pub trait Try {
2019-12-03 12:38:54 +00:00
type Ok ;
type Error ;
}
}
2021-06-01 11:39:19 +00:00
pub mod result {
pub enum Result < O , E > {
2019-12-03 12:38:54 +00:00
Ok ( O ) ,
Err ( E )
}
impl < O , E > crate ::ops ::Try for Result < O , E > {
type Ok = O ;
type Error = E ;
}
}
2021-06-01 11:39:19 +00:00
pub mod prelude {
pub mod rust_2018 {
pub use crate ::{ result ::* , ops ::* } ;
}
}
2019-12-03 12:38:54 +00:00
" #,
) ;
}
2021-05-25 12:59:54 +00:00
#[ test ]
2021-05-30 16:37:02 +00:00
fn infer_try_trait_v2 ( ) {
2021-05-25 12:59:54 +00:00
check_types (
r #"
//- /main.rs crate:main deps:core
fn test ( ) {
let r : Result < i32 , u64 > = Result ::Ok ( 1 ) ;
let v = r ? ;
v ;
} //^ i32
//- /core.rs crate:core
mod ops {
2021-05-30 16:37:02 +00:00
mod try_trait {
pub trait Try : FromResidual {
type Output ;
type Residual ;
}
pub trait FromResidual < R = < Self as Try > ::Residual > { }
2021-05-25 12:59:54 +00:00
}
2021-05-30 16:37:02 +00:00
pub use self ::try_trait ::FromResidual ;
pub use self ::try_trait ::Try ;
}
2021-07-31 14:17:08 +00:00
mod convert {
2021-05-30 16:37:02 +00:00
pub trait From < T > { }
impl < T > From < T > for T { }
2021-05-25 12:59:54 +00:00
}
2021-06-01 11:39:19 +00:00
pub mod result {
2021-05-30 16:37:02 +00:00
use crate ::convert ::From ;
use crate ::ops ::{ Try , FromResidual } ;
pub enum Infallible { }
pub enum Result < O , E > {
2021-05-25 12:59:54 +00:00
Ok ( O ) ,
Err ( E )
}
2021-05-30 16:37:02 +00:00
impl < O , E > Try for Result < O , E > {
2021-05-25 12:59:54 +00:00
type Output = O ;
type Error = Result < Infallible , E > ;
}
2021-05-30 16:37:02 +00:00
impl < T , E , F : From < E > > FromResidual < Result < Infallible , E > > for Result < T , F > { }
2021-05-25 12:59:54 +00:00
}
2021-06-01 11:39:19 +00:00
pub mod prelude {
pub mod rust_2018 {
pub use crate ::result ::* ;
}
}
2021-05-25 12:59:54 +00:00
" #,
) ;
}
2019-12-03 12:38:54 +00:00
#[ test ]
fn infer_for_loop ( ) {
2020-06-29 15:22:47 +00:00
check_types (
2019-12-03 12:38:54 +00:00
r #"
2020-06-11 14:22:31 +00:00
//- /main.rs crate:main deps:core,alloc
2021-06-01 11:39:19 +00:00
#![ no_std ]
2020-06-11 14:22:31 +00:00
use alloc ::collections ::Vec ;
2019-12-03 12:38:54 +00:00
fn test ( ) {
let v = Vec ::new ( ) ;
v . push ( " foo " ) ;
for x in v {
2020-06-29 15:22:47 +00:00
x ;
} //^ &str
2019-12-03 12:38:54 +00:00
}
2020-06-11 14:22:31 +00:00
//- /core.rs crate:core
2021-06-01 11:39:19 +00:00
pub mod iter {
pub trait IntoIterator {
2019-12-03 12:38:54 +00:00
type Item ;
}
}
2021-06-01 11:39:19 +00:00
pub mod prelude {
pub mod rust_2018 {
pub use crate ::iter ::* ;
}
}
2019-12-03 12:38:54 +00:00
2020-06-11 14:22:31 +00:00
//- /alloc.rs crate:alloc deps:core
2021-06-01 11:39:19 +00:00
#![ no_std ]
2021-07-21 15:51:56 +00:00
pub mod collections {
pub struct Vec < T > { }
2019-12-03 12:38:54 +00:00
impl < T > Vec < T > {
2021-03-20 16:12:49 +00:00
pub fn new ( ) -> Self { Vec { } }
pub fn push ( & mut self , t : T ) { }
2019-12-03 12:38:54 +00:00
}
2020-06-11 14:22:31 +00:00
impl < T > IntoIterator for Vec < T > {
2019-12-03 12:38:54 +00:00
type Item = T ;
}
}
" #,
) ;
}
2019-12-13 11:44:42 +00:00
#[ test ]
fn infer_ops_neg ( ) {
2020-06-29 15:22:47 +00:00
check_types (
2019-12-13 11:44:42 +00:00
r #"
//- /main.rs crate:main deps:std
struct Bar ;
struct Foo ;
impl std ::ops ::Neg for Bar {
type Output = Foo ;
}
fn test ( ) {
let a = Bar ;
let b = - a ;
2020-06-29 15:22:47 +00:00
b ;
} //^ Foo
2019-12-13 11:44:42 +00:00
//- /std.rs crate:std
#[ prelude_import ] use ops ::* ;
mod ops {
2019-12-29 16:39:31 +00:00
#[ lang = " neg " ]
2019-12-13 11:44:42 +00:00
pub trait Neg {
type Output ;
}
}
" #,
) ;
}
#[ test ]
fn infer_ops_not ( ) {
2020-06-29 15:22:47 +00:00
check_types (
2019-12-13 11:44:42 +00:00
r #"
//- /main.rs crate:main deps:std
struct Bar ;
struct Foo ;
impl std ::ops ::Not for Bar {
type Output = Foo ;
}
fn test ( ) {
let a = Bar ;
let b = ! a ;
2020-06-29 15:22:47 +00:00
b ;
} //^ Foo
2019-12-13 11:44:42 +00:00
//- /std.rs crate:std
#[ prelude_import ] use ops ::* ;
mod ops {
2019-12-29 16:39:31 +00:00
#[ lang = " not " ]
2019-12-13 11:44:42 +00:00
pub trait Not {
type Output ;
}
}
" #,
) ;
}
2019-12-03 12:38:54 +00:00
#[ test ]
fn infer_from_bound_1 ( ) {
2021-06-20 14:37:50 +00:00
check_types (
2020-07-21 10:08:55 +00:00
r #"
2021-04-10 16:03:27 +00:00
trait Trait < T > { }
struct S < T > ( T ) ;
impl < U > Trait < U > for S < U > { }
fn foo < T : Trait < u32 > > ( t : T ) { }
fn test ( ) {
let s = S ( unknown ) ;
2021-06-20 14:37:50 +00:00
// ^^^^^^^ u32
2021-04-10 16:03:27 +00:00
foo ( s ) ;
} " #,
2019-12-03 12:38:54 +00:00
) ;
}
#[ test ]
fn infer_from_bound_2 ( ) {
2021-06-20 14:37:50 +00:00
check_types (
2020-07-21 10:08:55 +00:00
r #"
2021-04-10 16:03:27 +00:00
trait Trait < T > { }
struct S < T > ( T ) ;
impl < U > Trait < U > for S < U > { }
2021-06-20 14:37:50 +00:00
fn foo < U , T : Trait < U > > ( t : T ) -> U { loop { } }
2021-04-10 16:03:27 +00:00
fn test ( ) {
let s = S ( unknown ) ;
2021-06-20 14:37:50 +00:00
// ^^^^^^^ u32
2021-04-10 16:03:27 +00:00
let x : u32 = foo ( s ) ;
} " #,
2019-12-03 12:38:54 +00:00
) ;
2020-02-14 18:16:42 +00:00
}
#[ test ]
fn trait_default_method_self_bound_implements_trait ( ) {
2021-03-08 20:19:44 +00:00
cov_mark ::check! ( trait_self_implements_self ) ;
2021-06-20 14:37:50 +00:00
check (
2020-07-21 10:08:55 +00:00
r #"
2021-04-10 16:03:27 +00:00
trait Trait {
fn foo ( & self ) -> i64 ;
2021-06-20 14:37:50 +00:00
fn bar ( & self ) -> ( ) {
self . foo ( ) ;
// ^^^^^^^^^^ type: i64
2021-04-10 16:03:27 +00:00
}
} " #,
2020-02-14 18:16:42 +00:00
) ;
}
#[ test ]
fn trait_default_method_self_bound_implements_super_trait ( ) {
2021-06-20 14:37:50 +00:00
check (
2020-07-21 10:08:55 +00:00
r #"
2021-04-10 16:03:27 +00:00
trait SuperTrait {
fn foo ( & self ) -> i64 ;
}
trait Trait : SuperTrait {
2021-06-20 14:37:50 +00:00
fn bar ( & self ) -> ( ) {
self . foo ( ) ;
// ^^^^^^^^^^ type: i64
2021-04-10 16:03:27 +00:00
}
} " #,
2020-02-14 18:16:42 +00:00
) ;
2019-12-03 12:38:54 +00:00
}
#[ test ]
fn infer_project_associated_type ( ) {
2021-06-20 14:37:50 +00:00
check_types (
2020-07-21 10:08:55 +00:00
r #"
2021-04-10 16:03:27 +00:00
trait Iterable {
type Item ;
}
struct S ;
impl Iterable for S { type Item = u32 ; }
fn test < T : Iterable > ( ) {
let x : < S as Iterable > ::Item = 1 ;
2021-06-20 14:37:50 +00:00
// ^ u32
let y : < T as Iterable > ::Item = u ;
// ^ Iterable::Item<T>
let z : T ::Item = u ;
// ^ Iterable::Item<T>
let a : < T > ::Item = u ;
// ^ Iterable::Item<T>
2021-04-10 16:03:27 +00:00
} " #,
2019-12-03 12:38:54 +00:00
) ;
}
#[ test ]
fn infer_return_associated_type ( ) {
2021-06-20 14:37:50 +00:00
check_types (
2020-07-21 10:08:55 +00:00
r #"
2021-04-10 16:03:27 +00:00
trait Iterable {
type Item ;
}
struct S ;
impl Iterable for S { type Item = u32 ; }
2021-06-20 14:37:50 +00:00
fn foo1 < T : Iterable > ( t : T ) -> T ::Item { loop { } }
fn foo2 < T : Iterable > ( t : T ) -> < T as Iterable > ::Item { loop { } }
fn foo3 < T : Iterable > ( t : T ) -> < T > ::Item { loop { } }
2021-04-10 16:03:27 +00:00
fn test ( ) {
2021-06-20 14:37:50 +00:00
foo1 ( S ) ;
// ^^^^^^^ u32
foo2 ( S ) ;
// ^^^^^^^ u32
foo3 ( S ) ;
// ^^^^^^^ u32
2021-04-10 16:03:27 +00:00
} " #,
2019-12-03 12:38:54 +00:00
) ;
}
#[ test ]
fn infer_associated_type_bound ( ) {
2021-06-20 14:37:50 +00:00
check_types (
2020-07-21 10:08:55 +00:00
r #"
2021-04-10 16:03:27 +00:00
trait Iterable {
type Item ;
}
fn test < T : Iterable < Item = u32 > > ( ) {
let y : T ::Item = unknown ;
2021-06-20 14:37:50 +00:00
// ^^^^^^^ u32
2021-04-10 16:03:27 +00:00
} " #,
2019-12-03 12:38:54 +00:00
) ;
}
#[ test ]
fn infer_const_body ( ) {
2021-06-20 14:37:50 +00:00
// FIXME make check_types work with other bodies
2020-07-21 10:08:55 +00:00
check_infer (
r #"
2021-04-10 16:03:27 +00:00
const A : u32 = 1 + 1 ;
2021-06-20 14:37:50 +00:00
static B : u64 = { let x = 1 ; x } ;
" #,
2020-07-21 10:08:55 +00:00
expect! [ [ r #"
15 .. 16 '1' : u32
15 .. 20 ' 1 + 1 ' : u32
19 .. 20 '1' : u32
38 .. 54 ' { let .. . 1 ; x } ' : u64
44 .. 45 'x' : u64
48 .. 49 '1' : u64
51 .. 52 'x' : u64
" #]],
2019-12-03 12:38:54 +00:00
) ;
}
#[ test ]
fn tuple_struct_fields ( ) {
2020-07-21 10:08:55 +00:00
check_infer (
r #"
2021-04-10 16:03:27 +00:00
struct S ( i32 , u64 ) ;
fn test ( ) -> u64 {
let a = S ( 4 , 6 ) ;
let b = a . 0 ;
a . 1
} " #,
2020-07-21 10:08:55 +00:00
expect! [ [ r #"
37 .. 86 ' { .. . a . 1 } ' : u64
47 .. 48 'a' : S
51 .. 52 'S' : S ( i32 , u64 ) -> S
51 .. 58 ' S ( 4 , 6 ) ' : S
53 .. 54 '4' : i32
56 .. 57 '6' : u64
68 .. 69 'b' : i32
72 .. 73 'a' : S
72 .. 75 ' a . 0 ' : i32
81 .. 82 'a' : S
81 .. 84 ' a . 1 ' : u64
" #]],
2019-12-03 12:38:54 +00:00
) ;
}
#[ test ]
fn tuple_struct_with_fn ( ) {
2020-07-21 10:08:55 +00:00
check_infer (
r #"
2021-04-10 16:03:27 +00:00
struct S ( fn ( u32 ) -> u64 ) ;
fn test ( ) -> u64 {
let a = S ( | i | 2 * i ) ;
let b = a . 0 ( 4 ) ;
a . 0 ( 2 )
} " #,
2020-07-21 10:08:55 +00:00
expect! [ [ r #"
43 .. 101 ' { .. . 0 ( 2 ) } ' : u64
53 .. 54 'a' : S
57 .. 58 'S' : S ( fn ( u32 ) -> u64 ) -> S
57 .. 67 ' S ( | i | 2 * i ) ' : S
59 .. 66 ' | i | 2 * i ' : | u32 | -> u64
60 .. 61 'i' : u32
63 .. 64 '2' : u32
63 .. 66 ' 2 * i ' : u32
65 .. 66 'i' : u32
77 .. 78 'b' : u64
81 .. 82 'a' : S
81 .. 84 ' a . 0 ' : fn ( u32 ) -> u64
81 .. 87 ' a . 0 ( 4 ) ' : u64
85 .. 86 '4' : u32
93 .. 94 'a' : S
93 .. 96 ' a . 0 ' : fn ( u32 ) -> u64
93 .. 99 ' a . 0 ( 2 ) ' : u64
97 .. 98 '2' : u32
" #]],
2019-12-03 12:38:54 +00:00
) ;
}
#[ test ]
fn indexing_arrays ( ) {
2020-07-21 10:08:55 +00:00
check_infer (
" fn main() { &mut [9][2]; } " ,
expect! [ [ r #"
10 .. 26 ' { & mut .. . [ 2 ] ; } ' : ( )
12 .. 23 ' & mut [ 9 ] [ 2 ] ' : & mut { unknown }
2021-05-11 12:06:33 +00:00
17 .. 20 ' [ 9 ] ' : [ i32 ; 1 ]
2020-07-21 10:08:55 +00:00
17 .. 23 ' [ 9 ] [ 2 ] ' : { unknown }
18 .. 19 '9' : i32
21 .. 22 '2' : i32
" #]],
2019-12-03 12:38:54 +00:00
)
}
2019-12-19 04:45:07 +00:00
#[ test ]
fn infer_ops_index ( ) {
2020-06-29 15:22:47 +00:00
check_types (
2019-12-19 04:45:07 +00:00
r #"
2021-06-18 19:37:03 +00:00
//- minicore: index
2019-12-19 04:45:07 +00:00
struct Bar ;
struct Foo ;
2021-06-18 19:37:03 +00:00
impl core ::ops ::Index < u32 > for Bar {
2019-12-19 04:45:07 +00:00
type Output = Foo ;
}
fn test ( ) {
let a = Bar ;
2020-04-10 15:44:43 +00:00
let b = a [ 1 u32 ] ;
2020-06-29 15:22:47 +00:00
b ;
} //^ Foo
2019-12-19 04:45:07 +00:00
" #,
) ;
}
2020-07-03 15:39:06 +00:00
#[ test ]
fn infer_ops_index_int ( ) {
check_types (
r #"
2021-06-18 19:37:03 +00:00
//- minicore: index
2020-07-03 15:39:06 +00:00
struct Bar ;
struct Foo ;
2021-06-18 19:37:03 +00:00
impl core ::ops ::Index < u32 > for Bar {
2020-07-03 15:39:06 +00:00
type Output = Foo ;
}
struct Range ;
2021-06-18 19:37:03 +00:00
impl core ::ops ::Index < Range > for Bar {
2020-07-03 15:39:06 +00:00
type Output = Bar ;
}
fn test ( ) {
let a = Bar ;
let b = a [ 1 ] ;
b ;
//^ Foo
}
" #,
) ;
}
2020-02-29 21:48:23 +00:00
#[ test ]
fn infer_ops_index_autoderef ( ) {
2020-06-29 15:22:47 +00:00
check_types (
2020-02-29 21:48:23 +00:00
r #"
2021-06-18 19:37:03 +00:00
//- minicore: index, slice
2020-02-29 21:48:23 +00:00
fn test ( ) {
let a = & [ 1 u32 , 2 , 3 ] ;
2021-06-18 19:37:03 +00:00
let b = a [ 1 ] ;
2020-06-29 15:22:47 +00:00
b ;
} //^ u32
2020-02-29 21:48:23 +00:00
" #,
) ;
}
2019-12-03 12:38:54 +00:00
#[ test ]
fn deref_trait ( ) {
2020-06-29 15:22:47 +00:00
check_types (
2019-12-03 12:38:54 +00:00
r #"
2021-06-15 18:57:56 +00:00
//- minicore: deref
2021-07-31 14:17:08 +00:00
struct Arc < T : ? Sized > ;
impl < T : ? Sized > core ::ops ::Deref for Arc < T > {
2019-12-03 12:38:54 +00:00
type Target = T ;
}
struct S ;
impl S {
2021-06-20 14:37:50 +00:00
fn foo ( & self ) -> u128 { 0 }
2019-12-03 12:38:54 +00:00
}
fn test ( s : Arc < S > ) {
2020-06-29 15:22:47 +00:00
( * s , s . foo ( ) ) ;
2021-06-20 14:37:50 +00:00
} //^^^^^^^^^^^^^ (S, u128)
2019-12-03 12:38:54 +00:00
" #,
) ;
}
#[ test ]
fn deref_trait_with_inference_var ( ) {
2020-06-29 15:22:47 +00:00
check_types (
2019-12-03 12:38:54 +00:00
r #"
2021-06-15 18:57:56 +00:00
//- minicore: deref
2021-07-31 14:17:08 +00:00
struct Arc < T : ? Sized > ;
fn new_arc < T : ? Sized > ( ) -> Arc < T > { Arc }
impl < T : ? Sized > core ::ops ::Deref for Arc < T > {
2019-12-03 12:38:54 +00:00
type Target = T ;
}
struct S ;
fn foo ( a : Arc < S > ) { }
fn test ( ) {
let a = new_arc ( ) ;
2021-06-20 14:37:50 +00:00
let b = * a ;
//^^ S
2019-12-03 12:38:54 +00:00
foo ( a ) ;
}
" #,
) ;
}
#[ test ]
fn deref_trait_infinite_recursion ( ) {
2020-06-29 15:22:47 +00:00
check_types (
2019-12-03 12:38:54 +00:00
r #"
2021-06-15 18:57:56 +00:00
//- minicore: deref
2019-12-03 12:38:54 +00:00
struct S ;
2021-06-15 18:57:56 +00:00
impl core ::ops ::Deref for S {
2019-12-03 12:38:54 +00:00
type Target = S ;
}
fn test ( s : S ) {
2020-06-29 15:22:47 +00:00
s . foo ( ) ;
2021-06-20 14:37:50 +00:00
} //^^^^^^^ {unknown}
2019-12-03 12:38:54 +00:00
" #,
) ;
}
#[ test ]
fn deref_trait_with_question_mark_size ( ) {
2020-06-29 15:22:47 +00:00
check_types (
2019-12-03 12:38:54 +00:00
r #"
2021-06-15 18:57:56 +00:00
//- minicore: deref
2021-07-31 14:17:08 +00:00
struct Arc < T : ? Sized > ;
2021-06-15 18:57:56 +00:00
impl < T : ? Sized > core ::ops ::Deref for Arc < T > {
2019-12-03 12:38:54 +00:00
type Target = T ;
}
struct S ;
impl S {
2021-06-20 14:37:50 +00:00
fn foo ( & self ) -> u128 { 0 }
2019-12-03 12:38:54 +00:00
}
fn test ( s : Arc < S > ) {
2020-06-29 15:22:47 +00:00
( * s , s . foo ( ) ) ;
2021-06-20 14:37:50 +00:00
} //^^^^^^^^^^^^^ (S, u128)
2019-12-03 12:38:54 +00:00
" #,
) ;
}
2021-08-09 09:30:05 +00:00
#[ test ]
fn deref_trait_with_implicit_sized_requirement_on_inference_var ( ) {
check_types (
r #"
//- minicore: deref
struct Foo < T > ;
impl < T > core ::ops ::Deref for Foo < T > {
type Target = ( ) ;
}
fn test ( ) {
let foo = Foo ;
* foo ;
//^^^^ ()
let _ : Foo < u8 > = foo ;
}
" #,
)
}
2019-12-03 12:38:54 +00:00
#[ test ]
fn obligation_from_function_clause ( ) {
2020-06-29 15:22:47 +00:00
check_types (
2019-12-03 12:38:54 +00:00
r #"
struct S ;
trait Trait < T > { }
impl Trait < u32 > for S { }
2021-06-20 14:37:50 +00:00
fn foo < T : Trait < U > , U > ( t : T ) -> U { loop { } }
2019-12-03 12:38:54 +00:00
fn test ( s : S ) {
2021-06-20 14:37:50 +00:00
foo ( s ) ;
} //^^^^^^ u32
2019-12-03 12:38:54 +00:00
" #,
) ;
}
#[ test ]
fn obligation_from_method_clause ( ) {
2020-06-29 15:22:47 +00:00
check_types (
2019-12-03 12:38:54 +00:00
r #"
//- /main.rs
struct S ;
trait Trait < T > { }
impl Trait < isize > for S { }
struct O ;
impl O {
2021-06-20 14:37:50 +00:00
fn foo < T : Trait < U > , U > ( & self , t : T ) -> U { loop { } }
2019-12-03 12:38:54 +00:00
}
fn test ( ) {
2020-06-29 15:22:47 +00:00
O . foo ( S ) ;
2021-06-20 14:37:50 +00:00
} //^^^^^^^^ isize
2019-12-03 12:38:54 +00:00
" #,
) ;
}
#[ test ]
fn obligation_from_self_method_clause ( ) {
2020-06-29 15:22:47 +00:00
check_types (
2019-12-03 12:38:54 +00:00
r #"
struct S ;
trait Trait < T > { }
impl Trait < i64 > for S { }
impl S {
2021-06-20 14:37:50 +00:00
fn foo < U > ( & self ) -> U where Self : Trait < U > { loop { } }
2019-12-03 12:38:54 +00:00
}
fn test ( ) {
2020-06-29 15:22:47 +00:00
S . foo ( ) ;
2021-06-20 14:37:50 +00:00
} //^^^^^^^ i64
2019-12-03 12:38:54 +00:00
" #,
) ;
}
#[ test ]
fn obligation_from_impl_clause ( ) {
2020-06-29 15:22:47 +00:00
check_types (
2019-12-03 12:38:54 +00:00
r #"
struct S ;
trait Trait < T > { }
impl Trait < & str > for S { }
struct O < T > ;
impl < U , T : Trait < U > > O < T > {
2021-06-20 14:37:50 +00:00
fn foo ( & self ) -> U { loop { } }
2019-12-03 12:38:54 +00:00
}
fn test ( o : O < S > ) {
2020-06-29 15:22:47 +00:00
o . foo ( ) ;
2021-06-20 14:37:50 +00:00
} //^^^^^^^ &str
2019-12-03 12:38:54 +00:00
" #,
) ;
}
#[ test ]
fn generic_param_env_1 ( ) {
2020-06-29 15:22:47 +00:00
check_types (
2019-12-03 12:38:54 +00:00
r #"
trait Clone { }
trait Trait { fn foo ( self ) -> u128 ; }
struct S ;
impl Clone for S { }
impl < T > Trait for T where T : Clone { }
2020-06-29 15:22:47 +00:00
fn test < T : Clone > ( t : T ) { t . foo ( ) ; }
2021-06-20 14:37:50 +00:00
//^^^^^^^ u128
2019-12-03 12:38:54 +00:00
" #,
) ;
}
#[ test ]
fn generic_param_env_1_not_met ( ) {
2020-06-29 15:22:47 +00:00
check_types (
2019-12-03 12:38:54 +00:00
r #"
//- /main.rs
trait Clone { }
trait Trait { fn foo ( self ) -> u128 ; }
struct S ;
impl Clone for S { }
impl < T > Trait for T where T : Clone { }
2020-06-29 15:22:47 +00:00
fn test < T > ( t : T ) { t . foo ( ) ; }
2021-06-20 14:37:50 +00:00
//^^^^^^^ {unknown}
2019-12-03 12:38:54 +00:00
" #,
) ;
}
#[ test ]
fn generic_param_env_2 ( ) {
2020-06-29 15:22:47 +00:00
check_types (
2019-12-03 12:38:54 +00:00
r #"
trait Trait { fn foo ( self ) -> u128 ; }
struct S ;
impl Trait for S { }
2020-06-29 15:22:47 +00:00
fn test < T : Trait > ( t : T ) { t . foo ( ) ; }
2021-06-20 14:37:50 +00:00
//^^^^^^^ u128
2019-12-03 12:38:54 +00:00
" #,
) ;
}
#[ test ]
fn generic_param_env_2_not_met ( ) {
2020-06-29 15:22:47 +00:00
check_types (
2019-12-03 12:38:54 +00:00
r #"
trait Trait { fn foo ( self ) -> u128 ; }
struct S ;
impl Trait for S { }
2020-06-29 15:22:47 +00:00
fn test < T > ( t : T ) { t . foo ( ) ; }
2021-06-20 14:37:50 +00:00
//^^^^^^^ {unknown}
2019-12-03 12:38:54 +00:00
" #,
) ;
}
#[ test ]
fn generic_param_env_deref ( ) {
2020-06-29 15:22:47 +00:00
check_types (
2019-12-03 12:38:54 +00:00
r #"
2021-06-18 19:47:02 +00:00
//- minicore: deref
2019-12-03 12:38:54 +00:00
trait Trait { }
2021-06-18 19:47:02 +00:00
impl < T > core ::ops ::Deref for T where T : Trait {
2019-12-03 12:38:54 +00:00
type Target = i128 ;
}
2021-06-20 14:37:50 +00:00
fn test < T : Trait > ( t : T ) { * t ; }
//^^ i128
2019-12-03 12:38:54 +00:00
" #,
) ;
}
#[ test ]
fn associated_type_placeholder ( ) {
2020-06-29 15:22:47 +00:00
// inside the generic function, the associated type gets normalized to a placeholder `ApplL::Out<T>` [https://rust-lang.github.io/rustc-guide/traits/associated-types.html#placeholder-associated-types].
check_types (
2019-12-03 12:38:54 +00:00
r #"
pub trait ApplyL {
type Out ;
}
pub struct RefMutL < T > ;
impl < T > ApplyL for RefMutL < T > {
type Out = < T as ApplyL > ::Out ;
}
fn test < T : ApplyL > ( ) {
let y : < RefMutL < T > as ApplyL > ::Out = no_matter ;
2020-06-29 15:22:47 +00:00
y ;
} //^ ApplyL::Out<T>
2019-12-03 12:38:54 +00:00
" #,
) ;
}
#[ test ]
fn associated_type_placeholder_2 ( ) {
2020-06-29 15:22:47 +00:00
check_types (
2019-12-03 12:38:54 +00:00
r #"
pub trait ApplyL {
type Out ;
}
fn foo < T : ApplyL > ( t : T ) -> < T as ApplyL > ::Out ;
fn test < T : ApplyL > ( t : T ) {
let y = foo ( t ) ;
2020-06-29 15:22:47 +00:00
y ;
} //^ ApplyL::Out<T>
2019-12-03 12:38:54 +00:00
" #,
) ;
}
2020-01-31 15:05:58 +00:00
#[ test ]
fn argument_impl_trait ( ) {
2020-07-21 10:08:55 +00:00
check_infer_with_mismatches (
r #"
2021-06-15 22:23:04 +00:00
//- minicore: sized
2021-04-10 16:03:27 +00:00
trait Trait < T > {
fn foo ( & self ) -> T ;
fn foo2 ( & self ) -> i64 ;
}
fn bar ( x : impl Trait < u16 > ) { }
struct S < T > ( T ) ;
impl < T > Trait < T > for S < T > { }
fn test ( x : impl Trait < u64 > , y : & impl Trait < u32 > ) {
x ;
y ;
let z = S ( 1 ) ;
bar ( z ) ;
x . foo ( ) ;
y . foo ( ) ;
z . foo ( ) ;
x . foo2 ( ) ;
y . foo2 ( ) ;
z . foo2 ( ) ;
} " #,
2020-07-21 10:08:55 +00:00
expect! [ [ r #"
2021-06-15 22:23:04 +00:00
29 .. 33 ' self ' : & Self
54 .. 58 ' self ' : & Self
77 .. 78 'x' : impl Trait < u16 >
97 .. 99 ' { } ' : ( )
154 .. 155 'x' : impl Trait < u64 >
174 .. 175 'y' : & impl Trait < u32 >
195 .. 323 ' { .. . 2 ( ) ; } ' : ( )
201 .. 202 'x' : impl Trait < u64 >
208 .. 209 'y' : & impl Trait < u32 >
219 .. 220 'z' : S < u16 >
223 .. 224 'S' : S < u16 > ( u16 ) -> S < u16 >
223 .. 227 ' S ( 1 ) ' : S < u16 >
225 .. 226 '1' : u16
233 .. 236 ' bar ' : fn bar ( S < u16 > )
233 .. 239 ' bar ( z ) ' : ( )
237 .. 238 'z' : S < u16 >
245 .. 246 'x' : impl Trait < u64 >
245 .. 252 ' x . foo ( ) ' : u64
258 .. 259 'y' : & impl Trait < u32 >
258 .. 265 ' y . foo ( ) ' : u32
271 .. 272 'z' : S < u16 >
271 .. 278 ' z . foo ( ) ' : u16
284 .. 285 'x' : impl Trait < u64 >
284 .. 292 ' x . foo2 ( ) ' : i64
298 .. 299 'y' : & impl Trait < u32 >
298 .. 306 ' y . foo2 ( ) ' : i64
312 .. 313 'z' : S < u16 >
312 .. 320 ' z . foo2 ( ) ' : i64
2020-07-21 10:08:55 +00:00
" #]],
2020-01-31 15:05:58 +00:00
) ;
}
2020-02-07 15:24:09 +00:00
#[ test ]
fn argument_impl_trait_type_args_1 ( ) {
2020-07-21 10:08:55 +00:00
check_infer_with_mismatches (
r #"
2021-06-15 22:23:04 +00:00
//- minicore: sized
2021-04-10 16:03:27 +00:00
trait Trait { }
trait Foo {
// this function has an implicit Self param, an explicit type param,
// and an implicit impl Trait param!
fn bar < T > ( x : impl Trait ) -> T { loop { } }
}
fn foo < T > ( x : impl Trait ) -> T { loop { } }
struct S ;
impl Trait for S { }
struct F ;
impl Foo for F { }
fn test ( ) {
Foo ::bar ( S ) ;
< F as Foo > ::bar ( S ) ;
F ::bar ( S ) ;
Foo ::bar ::< u32 > ( S ) ;
< F as Foo > ::bar ::< u32 > ( S ) ;
foo ( S ) ;
foo ::< u32 > ( S ) ;
foo ::< u32 , i32 > ( S ) ; // we should ignore the extraneous i32
} " #,
2020-07-21 10:08:55 +00:00
expect! [ [ r #"
2021-06-15 22:23:04 +00:00
155 .. 156 'x' : impl Trait
175 .. 186 ' { loop { } } ' : T
177 .. 184 ' loop { } ' : !
182 .. 184 ' { } ' : ( )
199 .. 200 'x' : impl Trait
219 .. 230 ' { loop { } } ' : T
221 .. 228 ' loop { } ' : !
226 .. 228 ' { } ' : ( )
300 .. 509 ' { .. . i32 } ' : ( )
306 .. 314 ' Foo ::bar ' : fn bar < { unknown } , { unknown } > ( S ) -> { unknown }
306 .. 317 ' Foo ::bar ( S ) ' : { unknown }
315 .. 316 'S' : S
323 .. 338 ' < F as Foo > ::bar ' : fn bar < F , { unknown } > ( S ) -> { unknown }
323 .. 341 ' < F as .. . bar ( S ) ' : { unknown }
339 .. 340 'S' : S
347 .. 353 ' F ::bar ' : fn bar < F , { unknown } > ( S ) -> { unknown }
347 .. 356 ' F ::bar ( S ) ' : { unknown }
354 .. 355 'S' : S
362 .. 377 ' Foo ::bar ::< u32 > ' : fn bar < { unknown } , u32 > ( S ) -> u32
362 .. 380 ' Foo ::b .. . 32 > ( S ) ' : u32
378 .. 379 'S' : S
386 .. 408 ' < F as .. . :< u32 > ' : fn bar < F , u32 > ( S ) -> u32
386 .. 411 ' < F as .. . 32 > ( S ) ' : u32
409 .. 410 'S' : S
418 .. 421 ' foo ' : fn foo < { unknown } > ( S ) -> { unknown }
418 .. 424 ' foo ( S ) ' : { unknown }
422 .. 423 'S' : S
430 .. 440 ' foo ::< u32 > ' : fn foo < u32 > ( S ) -> u32
430 .. 443 ' foo ::< u32 > ( S ) ' : u32
441 .. 442 'S' : S
449 .. 464 ' foo ::< u32 , i32 > ' : fn foo < u32 > ( S ) -> u32
449 .. 467 ' foo ::< .. . 32 > ( S ) ' : u32
465 .. 466 'S' : S
2020-07-21 10:08:55 +00:00
" #]],
2020-02-07 15:24:09 +00:00
) ;
}
#[ test ]
fn argument_impl_trait_type_args_2 ( ) {
2020-07-21 10:08:55 +00:00
check_infer_with_mismatches (
r #"
2021-06-15 22:23:04 +00:00
//- minicore: sized
2021-04-10 16:03:27 +00:00
trait Trait { }
struct S ;
impl Trait for S { }
struct F < T > ;
impl < T > F < T > {
fn foo < U > ( self , x : impl Trait ) -> ( T , U ) { loop { } }
}
2020-02-07 15:24:09 +00:00
2021-04-10 16:03:27 +00:00
fn test ( ) {
F . foo ( S ) ;
F ::< u32 > . foo ( S ) ;
F ::< u32 > . foo ::< i32 > ( S ) ;
F ::< u32 > . foo ::< i32 , u32 > ( S ) ; // extraneous argument should be ignored
} " #,
2020-07-21 10:08:55 +00:00
expect! [ [ r #"
2021-06-15 22:23:04 +00:00
87 .. 91 ' self ' : F < T >
93 .. 94 'x' : impl Trait
118 .. 129 ' { loop { } } ' : ( T , U )
120 .. 127 ' loop { } ' : !
125 .. 127 ' { } ' : ( )
143 .. 283 ' { .. . ored } ' : ( )
149 .. 150 'F' : F < { unknown } >
149 .. 157 ' F . foo ( S ) ' : ( { unknown } , { unknown } )
155 .. 156 'S' : S
163 .. 171 ' F ::< u32 > ' : F < u32 >
163 .. 178 ' F ::< u32 > . foo ( S ) ' : ( u32 , { unknown } )
176 .. 177 'S' : S
184 .. 192 ' F ::< u32 > ' : F < u32 >
184 .. 206 ' F ::< u3 .. . 32 > ( S ) ' : ( u32 , i32 )
204 .. 205 'S' : S
212 .. 220 ' F ::< u32 > ' : F < u32 >
212 .. 239 ' F ::< u3 .. . 32 > ( S ) ' : ( u32 , i32 )
2020-07-21 10:08:55 +00:00
237 .. 238 'S' : S
" #]],
2020-02-07 15:24:09 +00:00
) ;
}
2020-02-07 17:27:54 +00:00
#[ test ]
fn argument_impl_trait_to_fn_pointer ( ) {
2020-07-21 10:08:55 +00:00
check_infer_with_mismatches (
r #"
2021-06-15 22:23:04 +00:00
//- minicore: sized
2021-04-10 16:03:27 +00:00
trait Trait { }
fn foo ( x : impl Trait ) { loop { } }
struct S ;
impl Trait for S { }
2020-02-07 17:27:54 +00:00
2021-04-10 16:03:27 +00:00
fn test ( ) {
let f : fn ( S ) -> ( ) = foo ;
} " #,
2020-07-21 10:08:55 +00:00
expect! [ [ r #"
2021-06-15 22:23:04 +00:00
22 .. 23 'x' : impl Trait
37 .. 48 ' { loop { } } ' : ( )
39 .. 46 ' loop { } ' : !
44 .. 46 ' { } ' : ( )
90 .. 123 ' { .. . foo ; } ' : ( )
100 .. 101 'f' : fn ( S )
117 .. 120 ' foo ' : fn foo ( S )
2020-07-21 10:08:55 +00:00
" #]],
2020-02-07 17:27:54 +00:00
) ;
}
2019-12-03 12:38:54 +00:00
#[ test ]
fn impl_trait ( ) {
2020-07-21 10:08:55 +00:00
check_infer (
r #"
2021-06-15 22:23:04 +00:00
//- minicore: sized
2021-04-10 16:03:27 +00:00
trait Trait < T > {
fn foo ( & self ) -> T ;
fn foo2 ( & self ) -> i64 ;
}
fn bar ( ) -> impl Trait < u64 > { }
2020-07-21 10:08:55 +00:00
2021-04-10 16:03:27 +00:00
fn test ( x : impl Trait < u64 > , y : & impl Trait < u64 > ) {
x ;
y ;
let z = bar ( ) ;
x . foo ( ) ;
y . foo ( ) ;
z . foo ( ) ;
x . foo2 ( ) ;
y . foo2 ( ) ;
z . foo2 ( ) ;
} " #,
2020-07-21 10:08:55 +00:00
expect! [ [ r #"
2021-06-15 22:23:04 +00:00
29 .. 33 ' self ' : & Self
54 .. 58 ' self ' : & Self
98 .. 100 ' { } ' : ( )
110 .. 111 'x' : impl Trait < u64 >
130 .. 131 'y' : & impl Trait < u64 >
151 .. 268 ' { .. . 2 ( ) ; } ' : ( )
157 .. 158 'x' : impl Trait < u64 >
164 .. 165 'y' : & impl Trait < u64 >
175 .. 176 'z' : impl Trait < u64 >
179 .. 182 ' bar ' : fn bar ( ) -> impl Trait < u64 >
179 .. 184 ' bar ( ) ' : impl Trait < u64 >
2020-07-21 10:08:55 +00:00
190 .. 191 'x' : impl Trait < u64 >
2021-06-15 22:23:04 +00:00
190 .. 197 ' x . foo ( ) ' : u64
203 .. 204 'y' : & impl Trait < u64 >
203 .. 210 ' y . foo ( ) ' : u64
216 .. 217 'z' : impl Trait < u64 >
216 .. 223 ' z . foo ( ) ' : u64
229 .. 230 'x' : impl Trait < u64 >
229 .. 237 ' x . foo2 ( ) ' : i64
243 .. 244 'y' : & impl Trait < u64 >
243 .. 251 ' y . foo2 ( ) ' : i64
257 .. 258 'z' : impl Trait < u64 >
257 .. 265 ' z . foo2 ( ) ' : i64
2020-07-21 10:08:55 +00:00
" #]],
2019-12-03 12:38:54 +00:00
) ;
}
2020-03-04 22:00:44 +00:00
#[ test ]
2020-06-05 15:41:58 +00:00
fn simple_return_pos_impl_trait ( ) {
2021-03-08 20:19:44 +00:00
cov_mark ::check! ( lower_rpit ) ;
2020-07-21 10:08:55 +00:00
check_infer (
r #"
2021-06-15 22:23:04 +00:00
//- minicore: sized
2021-04-10 16:03:27 +00:00
trait Trait < T > {
fn foo ( & self ) -> T ;
}
fn bar ( ) -> impl Trait < u64 > { loop { } }
2020-06-05 15:41:58 +00:00
2021-04-10 16:03:27 +00:00
fn test ( ) {
let a = bar ( ) ;
a . foo ( ) ;
} " #,
2020-07-21 10:08:55 +00:00
expect! [ [ r #"
2021-06-15 22:23:04 +00:00
29 .. 33 ' self ' : & Self
71 .. 82 ' { loop { } } ' : !
73 .. 80 ' loop { } ' : !
78 .. 80 ' { } ' : ( )
94 .. 129 ' { .. . o ( ) ; } ' : ( )
104 .. 105 'a' : impl Trait < u64 >
108 .. 111 ' bar ' : fn bar ( ) -> impl Trait < u64 >
108 .. 113 ' bar ( ) ' : impl Trait < u64 >
119 .. 120 'a' : impl Trait < u64 >
119 .. 126 ' a . foo ( ) ' : u64
2020-07-21 10:08:55 +00:00
" #]],
2020-06-05 15:41:58 +00:00
) ;
}
#[ test ]
fn more_return_pos_impl_trait ( ) {
2020-07-21 10:08:55 +00:00
check_infer (
r #"
2021-06-15 22:23:04 +00:00
//- minicore: sized
2021-04-10 16:03:27 +00:00
trait Iterator {
type Item ;
fn next ( & mut self ) -> Self ::Item ;
}
trait Trait < T > {
fn foo ( & self ) -> T ;
}
fn bar ( ) -> ( impl Iterator < Item = impl Trait < u32 > > , impl Trait < u64 > ) { loop { } }
fn baz < T > ( t : T ) -> ( impl Iterator < Item = impl Trait < T > > , impl Trait < T > ) { loop { } }
fn test ( ) {
let ( a , b ) = bar ( ) ;
a . next ( ) . foo ( ) ;
b . foo ( ) ;
let ( c , d ) = baz ( 1 u128 ) ;
c . next ( ) . foo ( ) ;
d . foo ( ) ;
} " #,
2020-07-21 10:08:55 +00:00
expect! [ [ r #"
2021-06-15 22:23:04 +00:00
49 .. 53 ' self ' : & mut Self
101 .. 105 ' self ' : & Self
184 .. 195 ' { loop { } } ' : ( { unknown } , { unknown } )
186 .. 193 ' loop { } ' : !
191 .. 193 ' { } ' : ( )
206 .. 207 't' : T
268 .. 279 ' { loop { } } ' : ( { unknown } , { unknown } )
270 .. 277 ' loop { } ' : !
275 .. 277 ' { } ' : ( )
291 .. 413 ' { .. . o ( ) ; } ' : ( )
301 .. 307 ' ( a , b ) ' : ( impl Iterator < Item = impl Trait < u32 > > , impl Trait < u64 > )
302 .. 303 'a' : impl Iterator < Item = impl Trait < u32 > >
305 .. 306 'b' : impl Trait < u64 >
310 .. 313 ' bar ' : fn bar ( ) -> ( impl Iterator < Item = impl Trait < u32 > > , impl Trait < u64 > )
310 .. 315 ' bar ( ) ' : ( impl Iterator < Item = impl Trait < u32 > > , impl Trait < u64 > )
321 .. 322 'a' : impl Iterator < Item = impl Trait < u32 > >
321 .. 329 ' a . next ( ) ' : impl Trait < u32 >
321 .. 335 ' a . next ( ) . foo ( ) ' : u32
341 .. 342 'b' : impl Trait < u64 >
341 .. 348 ' b . foo ( ) ' : u64
358 .. 364 ' ( c , d ) ' : ( impl Iterator < Item = impl Trait < u128 > > , impl Trait < u128 > )
359 .. 360 'c' : impl Iterator < Item = impl Trait < u128 > >
362 .. 363 'd' : impl Trait < u128 >
367 .. 370 ' baz ' : fn baz < u128 > ( u128 ) -> ( impl Iterator < Item = impl Trait < u128 > > , impl Trait < u128 > )
367 .. 377 ' baz ( 1 u128 ) ' : ( impl Iterator < Item = impl Trait < u128 > > , impl Trait < u128 > )
371 .. 376 ' 1 u128 ' : u128
383 .. 384 'c' : impl Iterator < Item = impl Trait < u128 > >
383 .. 391 ' c . next ( ) ' : impl Trait < u128 >
383 .. 397 ' c . next ( ) . foo ( ) ' : u128
403 .. 404 'd' : impl Trait < u128 >
403 .. 410 ' d . foo ( ) ' : u128
2020-07-21 10:08:55 +00:00
" #]],
2020-03-04 22:00:44 +00:00
) ;
}
2019-12-03 12:38:54 +00:00
#[ test ]
fn dyn_trait ( ) {
2020-07-21 10:08:55 +00:00
check_infer (
r #"
2021-06-15 22:23:04 +00:00
//- minicore: sized
2021-04-10 16:03:27 +00:00
trait Trait < T > {
fn foo ( & self ) -> T ;
fn foo2 ( & self ) -> i64 ;
}
fn bar ( ) -> dyn Trait < u64 > { }
2020-07-21 10:08:55 +00:00
2021-04-10 16:03:27 +00:00
fn test ( x : dyn Trait < u64 > , y : & dyn Trait < u64 > ) {
x ;
y ;
let z = bar ( ) ;
x . foo ( ) ;
y . foo ( ) ;
z . foo ( ) ;
x . foo2 ( ) ;
y . foo2 ( ) ;
z . foo2 ( ) ;
} " #,
2020-07-21 10:08:55 +00:00
expect! [ [ r #"
2021-06-15 22:23:04 +00:00
29 .. 33 ' self ' : & Self
54 .. 58 ' self ' : & Self
97 .. 99 ' { } ' : ( )
109 .. 110 'x' : dyn Trait < u64 >
128 .. 129 'y' : & dyn Trait < u64 >
148 .. 265 ' { .. . 2 ( ) ; } ' : ( )
154 .. 155 'x' : dyn Trait < u64 >
2020-07-21 10:08:55 +00:00
161 .. 162 'y' : & dyn Trait < u64 >
2021-06-15 22:23:04 +00:00
172 .. 173 'z' : dyn Trait < u64 >
176 .. 179 ' bar ' : fn bar ( ) -> dyn Trait < u64 >
176 .. 181 ' bar ( ) ' : dyn Trait < u64 >
2020-07-21 10:08:55 +00:00
187 .. 188 'x' : dyn Trait < u64 >
2021-06-15 22:23:04 +00:00
187 .. 194 ' x . foo ( ) ' : u64
200 .. 201 'y' : & dyn Trait < u64 >
200 .. 207 ' y . foo ( ) ' : u64
213 .. 214 'z' : dyn Trait < u64 >
213 .. 220 ' z . foo ( ) ' : u64
226 .. 227 'x' : dyn Trait < u64 >
226 .. 234 ' x . foo2 ( ) ' : i64
240 .. 241 'y' : & dyn Trait < u64 >
240 .. 248 ' y . foo2 ( ) ' : i64
254 .. 255 'z' : dyn Trait < u64 >
254 .. 262 ' z . foo2 ( ) ' : i64
2020-07-21 10:08:55 +00:00
" #]],
2019-12-03 12:38:54 +00:00
) ;
}
2020-04-17 20:48:29 +00:00
#[ test ]
fn dyn_trait_in_impl ( ) {
2020-07-21 10:08:55 +00:00
check_infer (
r #"
2021-06-15 22:23:04 +00:00
//- minicore: sized
2021-04-10 16:03:27 +00:00
trait Trait < T , U > {
fn foo ( & self ) -> ( T , U ) ;
}
struct S < T , U > { }
impl < T , U > S < T , U > {
fn bar ( & self ) -> & dyn Trait < T , U > { loop { } }
}
trait Trait2 < T , U > {
fn baz ( & self ) -> ( T , U ) ;
}
impl < T , U > Trait2 < T , U > for dyn Trait < T , U > { }
2020-04-17 20:48:29 +00:00
2021-04-10 16:03:27 +00:00
fn test ( s : S < u32 , i32 > ) {
s . bar ( ) . baz ( ) ;
} " #,
2020-07-21 10:08:55 +00:00
expect! [ [ r #"
2021-06-15 22:23:04 +00:00
32 .. 36 ' self ' : & Self
102 .. 106 ' self ' : & S < T , U >
128 .. 139 ' { loop { } } ' : & dyn Trait < T , U >
130 .. 137 ' loop { } ' : !
135 .. 137 ' { } ' : ( )
175 .. 179 ' self ' : & Self
251 .. 252 's' : S < u32 , i32 >
267 .. 289 ' { .. . z ( ) ; } ' : ( )
273 .. 274 's' : S < u32 , i32 >
273 .. 280 ' s . bar ( ) ' : & dyn Trait < u32 , i32 >
273 .. 286 ' s . bar ( ) . baz ( ) ' : ( u32 , i32 )
2020-07-21 10:08:55 +00:00
" #]],
2020-04-17 20:48:29 +00:00
) ;
}
2019-12-03 12:38:54 +00:00
#[ test ]
fn dyn_trait_bare ( ) {
2020-07-21 10:08:55 +00:00
check_infer (
r #"
2021-06-15 22:23:04 +00:00
//- minicore: sized
2021-04-10 16:03:27 +00:00
trait Trait {
fn foo ( & self ) -> u64 ;
}
fn bar ( ) -> Trait { }
2020-07-21 10:08:55 +00:00
2021-04-10 16:03:27 +00:00
fn test ( x : Trait , y : & Trait ) -> u64 {
x ;
y ;
let z = bar ( ) ;
x . foo ( ) ;
y . foo ( ) ;
z . foo ( ) ;
} " #,
2020-07-21 10:08:55 +00:00
expect! [ [ r #"
2021-06-15 22:23:04 +00:00
26 .. 30 ' self ' : & Self
60 .. 62 ' { } ' : ( )
72 .. 73 'x' : dyn Trait
82 .. 83 'y' : & dyn Trait
100 .. 175 ' { .. . o ( ) ; } ' : ( )
106 .. 107 'x' : dyn Trait
113 .. 114 'y' : & dyn Trait
124 .. 125 'z' : dyn Trait
128 .. 131 ' bar ' : fn bar ( ) -> dyn Trait
128 .. 133 ' bar ( ) ' : dyn Trait
2020-07-21 10:08:55 +00:00
139 .. 140 'x' : dyn Trait
2021-06-15 22:23:04 +00:00
139 .. 146 ' x . foo ( ) ' : u64
152 .. 153 'y' : & dyn Trait
152 .. 159 ' y . foo ( ) ' : u64
165 .. 166 'z' : dyn Trait
165 .. 172 ' z . foo ( ) ' : u64
2020-07-21 10:08:55 +00:00
" #]],
2019-12-03 12:38:54 +00:00
) ;
}
#[ test ]
fn weird_bounds ( ) {
2020-07-21 10:08:55 +00:00
check_infer (
r #"
2021-06-15 22:23:04 +00:00
//- minicore: sized
2021-04-10 16:03:27 +00:00
trait Trait { }
fn test (
a : impl Trait + ' lifetime ,
b : impl ' lifetime ,
c : impl ( Trait ) ,
d : impl ( ' lifetime ) ,
e : impl ? Sized ,
f : impl Trait + ? Sized
) { }
" #,
2020-07-21 10:08:55 +00:00
expect! [ [ r #"
2021-06-15 22:23:04 +00:00
28 .. 29 'a' : impl Trait
59 .. 60 'b' : impl Sized
82 .. 83 'c' : impl Trait
103 .. 104 'd' : impl Sized
128 .. 129 'e' : impl ? Sized
148 .. 149 'f' : impl Trait + ? Sized
173 .. 175 ' { } ' : ( )
2020-07-21 10:08:55 +00:00
" #]],
2019-12-03 12:38:54 +00:00
) ;
}
2019-12-21 18:15:06 +00:00
#[ test ]
fn error_bound_chalk ( ) {
2020-06-29 15:22:47 +00:00
check_types (
2019-12-21 18:15:06 +00:00
r #"
trait Trait {
2021-06-20 14:37:50 +00:00
fn foo ( & self ) -> u32 { 0 }
2019-12-21 18:15:06 +00:00
}
fn test ( x : ( impl Trait + UnknownTrait ) ) {
2020-06-29 15:22:47 +00:00
x . foo ( ) ;
2021-06-20 14:37:50 +00:00
} //^^^^^^^ u32
2019-12-21 18:15:06 +00:00
" #,
) ;
}
2019-12-03 12:38:54 +00:00
#[ test ]
fn assoc_type_bindings ( ) {
2020-07-21 10:08:55 +00:00
check_infer (
r #"
2021-06-15 22:23:04 +00:00
//- minicore: sized
2021-04-10 16:03:27 +00:00
trait Trait {
type Type ;
}
2019-12-03 12:38:54 +00:00
2021-04-10 16:03:27 +00:00
fn get < T : Trait > ( t : T ) -> < T as Trait > ::Type { }
fn get2 < U , T : Trait < Type = U > > ( t : T ) -> U { }
fn set < T : Trait < Type = u64 > > ( t : T ) -> T { t }
struct S < T > ;
impl < T > Trait for S < T > { type Type = T ; }
fn test < T : Trait < Type = u32 > > ( x : T , y : impl Trait < Type = i64 > ) {
get ( x ) ;
get2 ( x ) ;
get ( y ) ;
get2 ( y ) ;
get ( set ( S ) ) ;
get2 ( set ( S ) ) ;
get2 ( S ::< str > ) ;
} " #,
2020-07-21 10:08:55 +00:00
expect! [ [ r #"
2021-06-15 22:23:04 +00:00
49 .. 50 't' : T
77 .. 79 ' { } ' : ( )
111 .. 112 't' : T
122 .. 124 ' { } ' : ( )
154 .. 155 't' : T
165 .. 168 ' { t } ' : T
166 .. 167 't' : T
256 .. 257 'x' : T
262 .. 263 'y' : impl Trait < Type = i64 >
289 .. 397 ' { .. . r > ) ; } ' : ( )
295 .. 298 ' get ' : fn get < T > ( T ) -> < T as Trait > ::Type
295 .. 301 ' get ( x ) ' : u32
299 .. 300 'x' : T
307 .. 311 ' get2 ' : fn get2 < u32 , T > ( T ) -> u32
307 .. 314 ' get2 ( x ) ' : u32
312 .. 313 'x' : T
320 .. 323 ' get ' : fn get < impl Trait < Type = i64 > > ( impl Trait < Type = i64 > ) -> < impl Trait < Type = i64 > as Trait > ::Type
320 .. 326 ' get ( y ) ' : i64
324 .. 325 'y' : impl Trait < Type = i64 >
332 .. 336 ' get2 ' : fn get2 < i64 , impl Trait < Type = i64 > > ( impl Trait < Type = i64 > ) -> i64
332 .. 339 ' get2 ( y ) ' : i64
337 .. 338 'y' : impl Trait < Type = i64 >
345 .. 348 ' get ' : fn get < S < u64 > > ( S < u64 > ) -> < S < u64 > as Trait > ::Type
345 .. 356 ' get ( set ( S ) ) ' : u64
349 .. 352 ' set ' : fn set < S < u64 > > ( S < u64 > ) -> S < u64 >
349 .. 355 ' set ( S ) ' : S < u64 >
353 .. 354 'S' : S < u64 >
362 .. 366 ' get2 ' : fn get2 < u64 , S < u64 > > ( S < u64 > ) -> u64
362 .. 374 ' get2 ( set ( S ) ) ' : u64
367 .. 370 ' set ' : fn set < S < u64 > > ( S < u64 > ) -> S < u64 >
367 .. 373 ' set ( S ) ' : S < u64 >
371 .. 372 'S' : S < u64 >
380 .. 384 ' get2 ' : fn get2 < str , S < str > > ( S < str > ) -> str
380 .. 394 ' get2 ( S ::< str > ) ' : str
385 .. 393 ' S ::< str > ' : S < str >
2020-07-21 10:08:55 +00:00
" #]],
2019-12-03 12:38:54 +00:00
) ;
}
#[ test ]
fn impl_trait_assoc_binding_projection_bug ( ) {
2020-06-29 15:22:47 +00:00
check_types (
2019-12-03 12:38:54 +00:00
r #"
2021-06-16 19:54:57 +00:00
//- minicore: iterator
2019-12-03 12:38:54 +00:00
pub trait Language {
type Kind ;
}
pub enum RustLanguage { }
impl Language for RustLanguage {
type Kind = SyntaxKind ;
}
struct SyntaxNode < L > { }
fn foo ( ) -> impl Iterator < Item = SyntaxNode < RustLanguage > > { }
trait Clone {
fn clone ( & self ) -> Self ;
}
fn api_walkthrough ( ) {
for node in foo ( ) {
2020-06-29 15:22:47 +00:00
node . clone ( ) ;
2021-06-20 14:37:50 +00:00
} //^^^^^^^^^^^^ {unknown}
2019-12-03 12:38:54 +00:00
}
" #,
) ;
}
#[ test ]
fn projection_eq_within_chalk ( ) {
2020-07-21 10:08:55 +00:00
check_infer (
r #"
2021-04-10 16:03:27 +00:00
trait Trait1 {
type Type ;
}
trait Trait2 < T > {
fn foo ( self ) -> T ;
}
impl < T , U > Trait2 < T > for U where U : Trait1 < Type = T > { }
2019-12-03 12:38:54 +00:00
2021-04-10 16:03:27 +00:00
fn test < T : Trait1 < Type = u32 > > ( x : T ) {
x . foo ( ) ;
} " #,
2020-07-21 10:08:55 +00:00
expect! [ [ r #"
61 .. 65 ' self ' : Self
163 .. 164 'x' : T
169 .. 185 ' { .. . o ( ) ; } ' : ( )
175 .. 176 'x' : T
175 .. 182 ' x . foo ( ) ' : u32
" #]],
2019-12-03 12:38:54 +00:00
) ;
}
#[ test ]
fn where_clause_trait_in_scope_for_method_resolution ( ) {
2020-06-29 15:22:47 +00:00
check_types (
2019-12-03 12:38:54 +00:00
r #"
mod foo {
trait Trait {
2021-06-20 14:37:50 +00:00
fn foo ( & self ) -> u32 { 0 }
2019-12-03 12:38:54 +00:00
}
}
fn test < T : foo ::Trait > ( x : T ) {
2020-06-29 15:22:47 +00:00
x . foo ( ) ;
2021-06-20 14:37:50 +00:00
} //^^^^^^^ u32
2019-12-03 12:38:54 +00:00
" #,
) ;
}
#[ test ]
fn super_trait_method_resolution ( ) {
2020-07-21 10:08:55 +00:00
check_infer (
r #"
2021-04-10 16:03:27 +00:00
mod foo {
trait SuperTrait {
fn foo ( & self ) -> u32 { }
}
}
trait Trait1 : foo ::SuperTrait { }
trait Trait2 where Self : foo ::SuperTrait { }
2019-12-03 12:38:54 +00:00
2021-04-10 16:03:27 +00:00
fn test < T : Trait1 , U : Trait2 > ( x : T , y : U ) {
x . foo ( ) ;
y . foo ( ) ;
} " #,
2020-07-21 10:08:55 +00:00
expect! [ [ r #"
49 .. 53 ' self ' : & Self
62 .. 64 ' { } ' : ( )
181 .. 182 'x' : T
187 .. 188 'y' : U
193 .. 222 ' { .. . o ( ) ; } ' : ( )
199 .. 200 'x' : T
199 .. 206 ' x . foo ( ) ' : u32
212 .. 213 'y' : U
212 .. 219 ' y . foo ( ) ' : u32
" #]],
2019-12-03 12:38:54 +00:00
) ;
}
2020-02-07 17:27:54 +00:00
#[ test ]
fn super_trait_impl_trait_method_resolution ( ) {
2020-07-21 10:08:55 +00:00
check_infer (
r #"
2021-06-15 22:23:04 +00:00
//- minicore: sized
2021-04-10 16:03:27 +00:00
mod foo {
trait SuperTrait {
fn foo ( & self ) -> u32 { }
}
}
trait Trait1 : foo ::SuperTrait { }
2020-02-07 17:27:54 +00:00
2021-04-10 16:03:27 +00:00
fn test ( x : & impl Trait1 ) {
x . foo ( ) ;
} " #,
2020-07-21 10:08:55 +00:00
expect! [ [ r #"
2021-06-15 22:23:04 +00:00
49 .. 53 ' self ' : & Self
62 .. 64 ' { } ' : ( )
115 .. 116 'x' : & impl Trait1
132 .. 148 ' { .. . o ( ) ; } ' : ( )
138 .. 139 'x' : & impl Trait1
138 .. 145 ' x . foo ( ) ' : u32
2020-07-21 10:08:55 +00:00
" #]],
2020-02-07 17:27:54 +00:00
) ;
}
2019-12-03 12:38:54 +00:00
#[ test ]
fn super_trait_cycle ( ) {
// This just needs to not crash
2020-07-21 10:08:55 +00:00
check_infer (
r #"
trait A : B { }
trait B : A { }
2019-12-03 12:38:54 +00:00
2020-07-21 10:08:55 +00:00
fn test < T : A > ( x : T ) {
x . foo ( ) ;
}
" #,
expect! [ [ r #"
43 .. 44 'x' : T
49 .. 65 ' { .. . o ( ) ; } ' : ( )
55 .. 56 'x' : T
55 .. 62 ' x . foo ( ) ' : { unknown }
" #]],
2019-12-03 12:38:54 +00:00
) ;
}
#[ test ]
fn super_trait_assoc_type_bounds ( ) {
2020-07-21 10:08:55 +00:00
check_infer (
r #"
2021-04-10 16:03:27 +00:00
trait SuperTrait { type Type ; }
trait Trait where Self : SuperTrait { }
2019-12-03 12:38:54 +00:00
2021-04-10 16:03:27 +00:00
fn get2 < U , T : Trait < Type = U > > ( t : T ) -> U { }
fn set < T : Trait < Type = u64 > > ( t : T ) -> T { t }
2019-12-03 12:38:54 +00:00
2021-04-10 16:03:27 +00:00
struct S < T > ;
impl < T > SuperTrait for S < T > { type Type = T ; }
impl < T > Trait for S < T > { }
2019-12-03 12:38:54 +00:00
2021-04-10 16:03:27 +00:00
fn test ( ) {
get2 ( set ( S ) ) ;
} " #,
2020-07-21 10:08:55 +00:00
expect! [ [ r #"
102 .. 103 't' : T
113 .. 115 ' { } ' : ( )
145 .. 146 't' : T
156 .. 159 ' { t } ' : T
157 .. 158 't' : T
258 .. 279 ' { .. . S ) ) ; } ' : ( )
264 .. 268 ' get2 ' : fn get2 < u64 , S < u64 > > ( S < u64 > ) -> u64
264 .. 276 ' get2 ( set ( S ) ) ' : u64
269 .. 272 ' set ' : fn set < S < u64 > > ( S < u64 > ) -> S < u64 >
269 .. 275 ' set ( S ) ' : S < u64 >
273 .. 274 'S' : S < u64 >
" #]],
2019-12-03 12:38:54 +00:00
) ;
}
#[ test ]
fn fn_trait ( ) {
2021-03-15 17:22:25 +00:00
check_infer_with_mismatches (
2020-07-21 10:08:55 +00:00
r #"
2021-04-10 16:03:27 +00:00
trait FnOnce < Args > {
type Output ;
2019-12-03 12:38:54 +00:00
2021-04-10 16:03:27 +00:00
fn call_once ( self , args : Args ) -> < Self as FnOnce < Args > > ::Output ;
}
2019-12-03 12:38:54 +00:00
2021-04-10 16:03:27 +00:00
fn test < F : FnOnce ( u32 , u64 ) -> u128 > ( f : F ) {
f . call_once ( ( 1 , 2 ) ) ;
} " #,
2020-07-21 10:08:55 +00:00
expect! [ [ r #"
56 .. 60 ' self ' : Self
62 .. 66 ' args ' : Args
149 .. 150 'f' : F
155 .. 183 ' { .. . 2 ) ) ; } ' : ( )
161 .. 162 'f' : F
161 .. 180 ' f . call .. . 1 , 2 ) ) ' : u128
173 .. 179 ' ( 1 , 2 ) ' : ( u32 , u64 )
174 .. 175 '1' : u32
177 .. 178 '2' : u64
" #]],
2019-12-03 12:38:54 +00:00
) ;
}
2020-05-18 06:07:31 +00:00
#[ test ]
fn fn_ptr_and_item ( ) {
2021-03-15 17:22:25 +00:00
check_infer_with_mismatches (
2020-07-21 10:08:55 +00:00
r #"
2021-04-10 16:03:27 +00:00
#[ lang= " fn_once " ]
trait FnOnce < Args > {
type Output ;
2020-05-18 06:07:31 +00:00
2021-04-10 16:03:27 +00:00
fn call_once ( self , args : Args ) -> Self ::Output ;
}
2020-05-18 06:07:31 +00:00
2021-04-10 16:03:27 +00:00
trait Foo < T > {
fn foo ( & self ) -> T ;
}
2020-05-18 06:07:31 +00:00
2021-04-10 16:03:27 +00:00
struct Bar < T > ( T ) ;
2020-05-18 06:07:31 +00:00
2021-04-10 16:03:27 +00:00
impl < A1 , R , F : FnOnce ( A1 ) -> R > Foo < ( A1 , R ) > for Bar < F > {
fn foo ( & self ) -> ( A1 , R ) { loop { } }
}
2020-05-18 06:07:31 +00:00
2021-04-10 16:03:27 +00:00
enum Opt < T > { None , Some ( T ) }
impl < T > Opt < T > {
fn map < U , F : FnOnce ( T ) -> U > ( self , f : F ) -> Opt < U > { loop { } }
}
2020-05-18 06:07:31 +00:00
2021-04-10 16:03:27 +00:00
fn test ( ) {
let bar : Bar < fn ( u8 ) -> u32 > ;
bar . foo ( ) ;
2020-07-21 10:08:55 +00:00
2021-04-10 16:03:27 +00:00
let opt : Opt < u8 > ;
let f : fn ( u8 ) -> u32 ;
opt . map ( f ) ;
} " #,
2020-07-21 10:08:55 +00:00
expect! [ [ r #"
74 .. 78 ' self ' : Self
80 .. 84 ' args ' : Args
139 .. 143 ' self ' : & Self
243 .. 247 ' self ' : & Bar < F >
2021-03-15 17:22:25 +00:00
260 .. 271 ' { loop { } } ' : ( A1 , R )
262 .. 269 ' loop { } ' : !
267 .. 269 ' { } ' : ( )
355 .. 359 ' self ' : Opt < T >
361 .. 362 'f' : F
377 .. 388 ' { loop { } } ' : Opt < U >
379 .. 386 ' loop { } ' : !
384 .. 386 ' { } ' : ( )
402 .. 518 ' { .. . ( f ) ; } ' : ( )
412 .. 415 ' bar ' : Bar < fn ( u8 ) -> u32 >
441 .. 444 ' bar ' : Bar < fn ( u8 ) -> u32 >
441 .. 450 ' bar . foo ( ) ' : ( u8 , u32 )
461 .. 464 ' opt ' : Opt < u8 >
483 .. 484 'f' : fn ( u8 ) -> u32
505 .. 508 ' opt ' : Opt < u8 >
505 .. 515 ' opt . map ( f ) ' : Opt < u32 >
513 .. 514 'f' : fn ( u8 ) -> u32
2020-07-21 10:08:55 +00:00
" #]],
2020-05-18 06:07:31 +00:00
) ;
}
#[ test ]
fn fn_trait_deref_with_ty_default ( ) {
2020-07-21 10:08:55 +00:00
check_infer (
r #"
2021-06-18 19:47:02 +00:00
//- minicore: deref, fn
2021-04-10 16:03:27 +00:00
struct Foo ;
2020-05-18 06:07:31 +00:00
2021-04-10 16:03:27 +00:00
impl Foo {
fn foo ( & self ) -> usize { }
}
2020-05-18 06:07:31 +00:00
2021-04-10 16:03:27 +00:00
struct Lazy < T , F = fn ( ) -> T > ( F ) ;
2020-05-18 06:07:31 +00:00
2021-04-10 16:03:27 +00:00
impl < T , F > Lazy < T , F > {
pub fn new ( f : F ) -> Lazy < T , F > { }
}
2020-05-18 06:07:31 +00:00
2021-06-18 19:47:02 +00:00
impl < T , F : FnOnce ( ) -> T > core ::ops ::Deref for Lazy < T , F > {
2021-04-10 16:03:27 +00:00
type Target = T ;
}
2020-05-18 06:07:31 +00:00
2021-04-10 16:03:27 +00:00
fn test ( ) {
let lazy1 : Lazy < Foo , _ > = Lazy ::new ( | | Foo ) ;
let r1 = lazy1 . foo ( ) ;
2020-07-21 10:08:55 +00:00
2021-04-10 16:03:27 +00:00
fn make_foo_fn ( ) -> Foo { }
let make_foo_fn_ptr : fn ( ) -> Foo = make_foo_fn ;
let lazy2 : Lazy < Foo , _ > = Lazy ::new ( make_foo_fn_ptr ) ;
let r2 = lazy2 . foo ( ) ;
} " #,
2020-07-21 10:08:55 +00:00
expect! [ [ r #"
2021-06-18 19:47:02 +00:00
36 .. 40 ' self ' : & Foo
51 .. 53 ' { } ' : ( )
131 .. 132 'f' : F
151 .. 153 ' { } ' : ( )
251 .. 497 ' { .. . o ( ) ; } ' : ( )
261 .. 266 ' lazy1 ' : Lazy < Foo , | | -> Foo >
283 .. 292 ' Lazy ::new ' : fn new < Foo , | | -> Foo > ( | | -> Foo ) -> Lazy < Foo , | | -> Foo >
283 .. 300 ' Lazy ::.. . | Foo ) ' : Lazy < Foo , | | -> Foo >
293 .. 299 ' | | Foo ' : | | -> Foo
296 .. 299 ' Foo ' : Foo
310 .. 312 ' r1 ' : usize
315 .. 320 ' lazy1 ' : Lazy < Foo , | | -> Foo >
315 .. 326 ' lazy1 . foo ( ) ' : usize
368 .. 383 ' make_foo_fn_ptr ' : fn ( ) -> Foo
399 .. 410 ' make_foo_fn ' : fn make_foo_fn ( ) -> Foo
420 .. 425 ' lazy2 ' : Lazy < Foo , fn ( ) -> Foo >
442 .. 451 ' Lazy ::new ' : fn new < Foo , fn ( ) -> Foo > ( fn ( ) -> Foo ) -> Lazy < Foo , fn ( ) -> Foo >
442 .. 468 ' Lazy ::.. . n_ptr ) ' : Lazy < Foo , fn ( ) -> Foo >
452 .. 467 ' make_foo_fn_ptr ' : fn ( ) -> Foo
478 .. 480 ' r2 ' : usize
483 .. 488 ' lazy2 ' : Lazy < Foo , fn ( ) -> Foo >
483 .. 494 ' lazy2 . foo ( ) ' : usize
357 .. 359 ' { } ' : ( )
2020-07-21 10:08:55 +00:00
" #]],
2020-05-18 06:07:31 +00:00
) ;
}
2019-12-03 12:38:54 +00:00
#[ test ]
fn closure_1 ( ) {
2021-03-15 17:22:25 +00:00
check_infer_with_mismatches (
2020-07-21 10:08:55 +00:00
r #"
2021-06-16 07:48:07 +00:00
//- minicore: fn
2021-04-10 16:03:27 +00:00
enum Option < T > { Some ( T ) , None }
impl < T > Option < T > {
fn map < U , F : FnOnce ( T ) -> U > ( self , f : F ) -> Option < U > { loop { } }
}
2019-12-03 12:38:54 +00:00
2021-04-10 16:03:27 +00:00
fn test ( ) {
let x = Option ::Some ( 1 u32 ) ;
x . map ( | v | v + 1 ) ;
x . map ( | _v | 1 u64 ) ;
let y : Option < i64 > = x . map ( | _v | 1 ) ;
} " #,
2020-07-21 10:08:55 +00:00
expect! [ [ r #"
2021-06-16 07:48:07 +00:00
86 .. 90 ' self ' : Option < T >
92 .. 93 'f' : F
111 .. 122 ' { loop { } } ' : Option < U >
113 .. 120 ' loop { } ' : !
118 .. 120 ' { } ' : ( )
136 .. 255 ' { .. . 1 ) ; } ' : ( )
146 .. 147 'x' : Option < u32 >
150 .. 162 ' Option ::Some ' : Some < u32 > ( u32 ) -> Option < u32 >
150 .. 168 ' Option .. . ( 1 u32 ) ' : Option < u32 >
163 .. 167 ' 1 u32 ' : u32
174 .. 175 'x' : Option < u32 >
174 .. 190 ' x . map ( .. . v + 1 ) ' : Option < u32 >
180 .. 189 ' | v | v + 1 ' : | u32 | -> u32
181 .. 182 'v' : u32
184 .. 185 'v' : u32
184 .. 189 ' v + 1 ' : u32
188 .. 189 '1' : u32
196 .. 197 'x' : Option < u32 >
196 .. 212 ' x . map ( .. . 1 u64 ) ' : Option < u64 >
202 .. 211 ' | _v | 1 u64 ' : | u32 | -> u64
203 .. 205 '_ v ' : u32
207 .. 211 ' 1 u64 ' : u64
222 .. 223 'y' : Option < i64 >
239 .. 240 'x' : Option < u32 >
239 .. 252 ' x . map ( | _v | 1 ) ' : Option < i64 >
245 .. 251 ' | _v | 1 ' : | u32 | -> i64
246 .. 248 '_ v ' : u32
250 .. 251 '1' : i64
2020-07-21 10:08:55 +00:00
" #]],
2019-12-03 12:38:54 +00:00
) ;
}
#[ test ]
fn closure_2 ( ) {
2021-03-15 17:22:25 +00:00
check_infer_with_mismatches (
2020-07-21 10:08:55 +00:00
r #"
2021-04-10 16:03:27 +00:00
#[ lang = " add " ]
pub trait Add < Rhs = Self > {
type Output ;
fn add ( self , rhs : Rhs ) -> Self ::Output ;
}
2021-04-10 15:16:35 +00:00
2021-04-10 16:03:27 +00:00
trait FnOnce < Args > {
type Output ;
}
2019-12-03 12:38:54 +00:00
2021-04-10 16:03:27 +00:00
impl Add for u64 {
type Output = Self ;
fn add ( self , rhs : u64 ) -> Self ::Output { 0 }
}
2021-04-10 15:16:35 +00:00
2021-04-10 16:03:27 +00:00
impl Add for u128 {
type Output = Self ;
fn add ( self , rhs : u128 ) -> Self ::Output { 0 }
}
2021-04-10 15:16:35 +00:00
2021-04-10 16:03:27 +00:00
fn test < F : FnOnce ( u32 ) -> u64 > ( f : F ) {
f ( 1 ) ;
let g = | v | v + 1 ;
g ( 1 u64 ) ;
let h = | v | 1 u128 + v ;
} " #,
2020-07-21 10:08:55 +00:00
expect! [ [ r #"
2021-04-10 15:16:35 +00:00
72 .. 76 ' self ' : Self
78 .. 81 ' rhs ' : Rhs
203 .. 207 ' self ' : u64
209 .. 212 ' rhs ' : u64
235 .. 238 ' { 0 } ' : u64
236 .. 237 '0' : u64
297 .. 301 ' self ' : u128
303 .. 306 ' rhs ' : u128
330 .. 333 ' { 0 } ' : u128
331 .. 332 '0' : u128
368 .. 369 'f' : F
374 .. 450 ' { .. . + v ; } ' : ( )
380 .. 381 'f' : F
380 .. 384 ' f ( 1 ) ' : { unknown }
382 .. 383 '1' : i32
394 .. 395 'g' : | u64 | -> u64
398 .. 407 ' | v | v + 1 ' : | u64 | -> u64
399 .. 400 'v' : u64
402 .. 403 'v' : u64
402 .. 407 ' v + 1 ' : u64
406 .. 407 '1' : u64
413 .. 414 'g' : | u64 | -> u64
413 .. 420 ' g ( 1 u64 ) ' : u64
415 .. 419 ' 1 u64 ' : u64
430 .. 431 'h' : | u128 | -> u128
434 .. 447 ' | v | 1 u128 + v ' : | u128 | -> u128
435 .. 436 'v' : u128
438 .. 443 ' 1 u128 ' : u128
438 .. 447 ' 1 u128 + v ' : u128
446 .. 447 'v' : u128
2020-07-21 10:08:55 +00:00
" #]],
2019-12-03 12:38:54 +00:00
) ;
}
#[ test ]
fn closure_as_argument_inference_order ( ) {
2021-03-15 17:22:25 +00:00
check_infer_with_mismatches (
2020-07-21 10:08:55 +00:00
r #"
2021-06-16 07:48:07 +00:00
//- minicore: fn
2021-04-10 16:03:27 +00:00
fn foo1 < T , U , F : FnOnce ( T ) -> U > ( x : T , f : F ) -> U { loop { } }
fn foo2 < T , U , F : FnOnce ( T ) -> U > ( f : F , x : T ) -> U { loop { } }
2019-12-03 12:38:54 +00:00
2021-04-10 16:03:27 +00:00
struct S ;
impl S {
fn method ( self ) -> u64 ;
2019-12-03 12:38:54 +00:00
2021-04-10 16:03:27 +00:00
fn foo1 < T , U , F : FnOnce ( T ) -> U > ( self , x : T , f : F ) -> U { loop { } }
fn foo2 < T , U , F : FnOnce ( T ) -> U > ( self , f : F , x : T ) -> U { loop { } }
}
2019-12-03 12:38:54 +00:00
2021-04-10 16:03:27 +00:00
fn test ( ) {
let x1 = foo1 ( S , | s | s . method ( ) ) ;
let x2 = foo2 ( | s | s . method ( ) , S ) ;
let x3 = S . foo1 ( S , | s | s . method ( ) ) ;
let x4 = S . foo2 ( | s | s . method ( ) , S ) ;
} " #,
2020-07-21 10:08:55 +00:00
expect! [ [ r #"
2021-06-16 07:48:07 +00:00
33 .. 34 'x' : T
39 .. 40 'f' : F
50 .. 61 ' { loop { } } ' : U
52 .. 59 ' loop { } ' : !
57 .. 59 ' { } ' : ( )
95 .. 96 'f' : F
101 .. 102 'x' : T
112 .. 123 ' { loop { } } ' : U
114 .. 121 ' loop { } ' : !
119 .. 121 ' { } ' : ( )
158 .. 162 ' self ' : S
210 .. 214 ' self ' : S
216 .. 217 'x' : T
222 .. 223 'f' : F
233 .. 244 ' { loop { } } ' : U
235 .. 242 ' loop { } ' : !
240 .. 242 ' { } ' : ( )
282 .. 286 ' self ' : S
288 .. 289 'f' : F
294 .. 295 'x' : T
305 .. 316 ' { loop { } } ' : U
307 .. 314 ' loop { } ' : !
312 .. 314 ' { } ' : ( )
330 .. 489 ' { .. . S ) ; } ' : ( )
340 .. 342 ' x1 ' : u64
345 .. 349 ' foo1 ' : fn foo1 < S , u64 , | S | -> u64 > ( S , | S | -> u64 ) -> u64
345 .. 368 ' foo1 ( S .. . hod ( ) ) ' : u64
350 .. 351 'S' : S
353 .. 367 ' | s | s . method ( ) ' : | S | -> u64
354 .. 355 's' : S
357 .. 358 's' : S
357 .. 367 ' s . method ( ) ' : u64
378 .. 380 ' x2 ' : u64
383 .. 387 ' foo2 ' : fn foo2 < S , u64 , | S | -> u64 > ( | S | -> u64 , S ) -> u64
383 .. 406 ' foo2 ( | .. . ( ) , S ) ' : u64
388 .. 402 ' | s | s . method ( ) ' : | S | -> u64
389 .. 390 's' : S
392 .. 393 's' : S
392 .. 402 ' s . method ( ) ' : u64
404 .. 405 'S' : S
416 .. 418 ' x3 ' : u64
421 .. 422 'S' : S
421 .. 446 ' S . foo1 .. . hod ( ) ) ' : u64
428 .. 429 'S' : S
431 .. 445 ' | s | s . method ( ) ' : | S | -> u64
432 .. 433 's' : S
435 .. 436 's' : S
435 .. 445 ' s . method ( ) ' : u64
456 .. 458 ' x4 ' : u64
461 .. 462 'S' : S
461 .. 486 ' S . foo2 .. . ( ) , S ) ' : u64
468 .. 482 ' | s | s . method ( ) ' : | S | -> u64
469 .. 470 's' : S
472 .. 473 's' : S
472 .. 482 ' s . method ( ) ' : u64
484 .. 485 'S' : S
2020-07-21 10:08:55 +00:00
" #]],
2019-12-03 12:38:54 +00:00
) ;
}
2020-07-11 17:55:11 +00:00
#[ test ]
fn fn_item_fn_trait ( ) {
check_types (
r #"
2021-06-16 07:48:07 +00:00
//- minicore: fn
2020-07-11 17:55:11 +00:00
struct S ;
2021-06-20 14:37:50 +00:00
fn foo ( ) -> S { S }
2020-07-11 17:55:11 +00:00
fn takes_closure < U , F : FnOnce ( ) -> U > ( f : F ) -> U { f ( ) }
fn test ( ) {
takes_closure ( foo ) ;
} //^^^^^^^^^^^^^^^^^^ S
" #,
) ;
}
2019-12-03 12:38:54 +00:00
#[ test ]
fn unselected_projection_in_trait_env_1 ( ) {
2020-06-29 15:22:47 +00:00
check_types (
2019-12-03 12:38:54 +00:00
r #"
//- /main.rs
trait Trait {
type Item ;
}
trait Trait2 {
fn foo ( & self ) -> u32 ;
}
fn test < T : Trait > ( ) where T ::Item : Trait2 {
let x : T ::Item = no_matter ;
2020-06-29 15:22:47 +00:00
x . foo ( ) ;
2021-06-20 14:37:50 +00:00
} //^^^^^^^ u32
2019-12-03 12:38:54 +00:00
" #,
) ;
}
#[ test ]
fn unselected_projection_in_trait_env_2 ( ) {
2020-06-29 15:22:47 +00:00
check_types (
2019-12-03 12:38:54 +00:00
r #"
trait Trait < T > {
type Item ;
}
trait Trait2 {
fn foo ( & self ) -> u32 ;
}
fn test < T , U > ( ) where T ::Item : Trait2 , T : Trait < U ::Item > , U : Trait < ( ) > {
let x : T ::Item = no_matter ;
2020-06-29 15:22:47 +00:00
x . foo ( ) ;
2021-06-20 14:37:50 +00:00
} //^^^^^^^ u32
2019-12-03 12:38:54 +00:00
" #,
) ;
}
2020-03-06 17:08:10 +00:00
#[ test ]
2020-04-12 10:29:03 +00:00
fn unselected_projection_on_impl_self ( ) {
2020-07-21 10:08:55 +00:00
check_infer (
2020-03-06 17:08:10 +00:00
r #"
2021-04-10 16:03:27 +00:00
//- /main.rs
trait Trait {
type Item ;
2020-03-06 17:08:10 +00:00
2021-04-10 16:03:27 +00:00
fn f ( & self , x : Self ::Item ) ;
}
2020-03-06 17:08:10 +00:00
2021-04-10 16:03:27 +00:00
struct S ;
2020-03-06 17:08:10 +00:00
2021-04-10 16:03:27 +00:00
impl Trait for S {
type Item = u32 ;
fn f ( & self , x : Self ::Item ) { let y = x ; }
}
2020-03-06 17:08:10 +00:00
2021-04-10 16:03:27 +00:00
struct S2 ;
2020-03-06 17:08:10 +00:00
2021-04-10 16:03:27 +00:00
impl Trait for S2 {
type Item = i32 ;
fn f ( & self , x : < Self > ::Item ) { let y = x ; }
} " #,
2020-07-21 10:08:55 +00:00
expect! [ [ r #"
40 .. 44 ' self ' : & Self
46 .. 47 'x' : Trait ::Item < Self >
126 .. 130 ' self ' : & S
132 .. 133 'x' : u32
147 .. 161 ' { let y = x ; } ' : ( )
153 .. 154 'y' : u32
157 .. 158 'x' : u32
228 .. 232 ' self ' : & S2
234 .. 235 'x' : i32
251 .. 265 ' { let y = x ; } ' : ( )
257 .. 258 'y' : i32
261 .. 262 'x' : i32
" #]],
) ;
2020-03-06 17:08:10 +00:00
}
2020-04-12 10:29:03 +00:00
#[ test ]
fn unselected_projection_on_trait_self ( ) {
2020-06-29 15:22:47 +00:00
check_types (
2020-04-12 10:29:03 +00:00
r #"
trait Trait {
type Item ;
fn f ( & self ) -> Self ::Item { loop { } }
}
struct S ;
impl Trait for S {
type Item = u32 ;
}
fn test ( ) {
2020-06-29 15:22:47 +00:00
S . f ( ) ;
2021-06-20 14:37:50 +00:00
} //^^^^^ u32
2020-04-12 10:29:03 +00:00
" #,
) ;
}
2020-04-26 14:56:25 +00:00
#[ test ]
fn unselected_projection_chalk_fold ( ) {
2020-06-29 15:22:47 +00:00
check_types (
2020-04-26 14:56:25 +00:00
r #"
trait Interner { }
trait Fold < I : Interner , TI = I > {
type Result ;
}
struct Ty < I : Interner > { }
impl < I : Interner , TI : Interner > Fold < I , TI > for Ty < I > {
type Result = Ty < TI > ;
}
fn fold < I : Interner , T > ( interner : & I , t : T ) -> T ::Result
where
T : Fold < I , I > ,
{
loop { }
}
fn foo < I : Interner > ( interner : & I , t : Ty < I > ) {
2020-06-29 15:22:47 +00:00
fold ( interner , t ) ;
2021-06-20 14:37:50 +00:00
} //^^^^^^^^^^^^^^^^^ Ty<I>
2020-04-26 14:56:25 +00:00
" #,
) ;
}
2019-12-03 12:38:54 +00:00
#[ test ]
fn trait_impl_self_ty ( ) {
2020-06-29 15:22:47 +00:00
check_types (
2019-12-03 12:38:54 +00:00
r #"
trait Trait < T > {
fn foo ( & self ) ;
}
struct S ;
impl Trait < Self > for S { }
fn test ( ) {
2020-06-29 15:22:47 +00:00
S . foo ( ) ;
2021-06-20 14:37:50 +00:00
} //^^^^^^^ ()
2019-12-03 12:38:54 +00:00
" #,
) ;
}
#[ test ]
fn trait_impl_self_ty_cycle ( ) {
2020-06-29 15:22:47 +00:00
check_types (
2019-12-03 12:38:54 +00:00
r #"
trait Trait {
fn foo ( & self ) ;
}
struct S < T > ;
impl Trait for S < Self > { }
fn test ( ) {
2020-06-29 15:22:47 +00:00
S . foo ( ) ;
2021-06-20 14:37:50 +00:00
} //^^^^^^^ {unknown}
2019-12-03 12:38:54 +00:00
" #,
) ;
}
#[ test ]
fn unselected_projection_in_trait_env_cycle_1 ( ) {
2020-06-29 15:22:47 +00:00
// this is a legitimate cycle
check_types (
2019-12-03 12:38:54 +00:00
r #"
trait Trait {
type Item ;
}
trait Trait2 < T > { }
fn test < T : Trait > ( ) where T : Trait2 < T ::Item > {
2020-06-29 15:22:47 +00:00
let x : T ::Item = no_matter ;
2021-06-20 14:37:50 +00:00
} //^^^^^^^^^ {unknown}
2019-12-03 12:38:54 +00:00
" #,
) ;
}
#[ test ]
fn unselected_projection_in_trait_env_cycle_2 ( ) {
2020-06-29 15:22:47 +00:00
// this is a legitimate cycle
check_types (
2019-12-03 12:38:54 +00:00
r #"
//- /main.rs
trait Trait < T > {
type Item ;
}
fn test < T , U > ( ) where T : Trait < U ::Item > , U : Trait < T ::Item > {
2020-06-29 15:22:47 +00:00
let x : T ::Item = no_matter ;
2021-06-20 14:37:50 +00:00
} //^^^^^^^^^ {unknown}
2019-12-03 12:38:54 +00:00
" #,
) ;
}
2019-12-06 11:45:00 +00:00
2021-03-20 14:26:42 +00:00
#[ test ]
fn unselected_projection_in_trait_env_cycle_3 ( ) {
2021-03-20 19:07:36 +00:00
// this is a cycle for rustc; we currently accept it
2021-03-20 14:26:42 +00:00
check_types (
r #"
//- /main.rs
trait Trait {
type Item ;
type OtherItem ;
}
fn test < T > ( ) where T : Trait < OtherItem = T ::Item > {
let x : T ::Item = no_matter ;
2021-06-20 14:37:50 +00:00
} //^^^^^^^^^ Trait::Item<T>
2021-03-20 14:26:42 +00:00
" #,
) ;
}
#[ test ]
fn unselected_projection_in_trait_env_no_cycle ( ) {
// this is not a cycle
check_types (
r #"
//- /main.rs
trait Index {
type Output ;
}
type Key < S : UnificationStoreBase > = < S as UnificationStoreBase > ::Key ;
pub trait UnificationStoreBase : Index < Output = Key < Self > > {
type Key ;
fn len ( & self ) -> usize ;
}
pub trait UnificationStoreMut : UnificationStoreBase {
fn push ( & mut self , value : Self ::Key ) ;
}
fn test < T > ( t : T ) where T : UnificationStoreMut {
let x ;
t . push ( x ) ;
let y : Key < T > ;
( x , y ) ;
2021-06-20 14:37:50 +00:00
} //^^^^^^ (UnificationStoreBase::Key<T>, UnificationStoreBase::Key<T>)
2021-03-20 14:26:42 +00:00
" #,
) ;
}
2020-04-10 20:05:46 +00:00
#[ test ]
fn inline_assoc_type_bounds_1 ( ) {
2020-06-29 15:22:47 +00:00
check_types (
2020-04-10 20:05:46 +00:00
r #"
trait Iterator {
type Item ;
}
trait OtherTrait < T > {
fn foo ( & self ) -> T ;
}
// workaround for Chalk assoc type normalization problems
pub struct S < T > ;
impl < T : Iterator > Iterator for S < T > {
type Item = < T as Iterator > ::Item ;
}
fn test < I : Iterator < Item : OtherTrait < u32 > > > ( ) {
let x : < S < I > as Iterator > ::Item ;
2020-06-29 15:22:47 +00:00
x . foo ( ) ;
2021-06-20 14:37:50 +00:00
} //^^^^^^^ u32
2020-04-10 20:05:46 +00:00
" #,
) ;
}
#[ test ]
fn inline_assoc_type_bounds_2 ( ) {
2020-06-29 15:22:47 +00:00
check_types (
2020-04-10 20:05:46 +00:00
r #"
trait Iterator {
type Item ;
}
fn test < I : Iterator < Item : Iterator < Item = u32 > > > ( ) {
let x : < < I as Iterator > ::Item as Iterator > ::Item ;
2020-06-29 15:22:47 +00:00
x ;
} //^ u32
2020-04-10 20:05:46 +00:00
" #,
) ;
}
2020-04-13 09:55:34 +00:00
#[ test ]
fn proc_macro_server_types ( ) {
2020-07-21 10:08:55 +00:00
check_infer (
r #"
2021-04-10 16:03:27 +00:00
macro_rules ! with_api {
( $S :ident , $self :ident , $m :ident ) = > {
$m ! {
TokenStream {
fn new ( ) -> $S ::TokenStream ;
} ,
Group {
} ,
2020-07-21 10:08:55 +00:00
}
2021-04-10 16:03:27 +00:00
} ;
}
macro_rules ! associated_item {
( type TokenStream ) = >
( type TokenStream : 'static ; ) ;
( type Group ) = >
( type Group : 'static ; ) ;
( $( $item :tt ) * ) = > ( $( $item ) * ; )
}
macro_rules ! declare_server_traits {
( $( $name :ident {
$( fn $method :ident ( $( $arg :ident : $arg_ty :ty ) , * $(, ) ? ) $( -> $ret_ty :ty ) ? ; ) *
} ) , * $(, ) ? ) = > {
pub trait Types {
$( associated_item! ( type $name ) ; ) *
2020-07-21 10:08:55 +00:00
}
2021-04-10 16:03:27 +00:00
$( pub trait $name : Types {
$( associated_item! ( fn $method ( $( $arg : $arg_ty ) , * ) $( -> $ret_ty ) ? ) ; ) *
} ) *
2020-07-21 10:08:55 +00:00
2021-04-10 16:03:27 +00:00
pub trait Server : Types $( + $name ) * { }
impl < S : Types $( + $name ) * > Server for S { }
}
}
with_api! ( Self , self_ , declare_server_traits ) ;
struct G { }
struct T { }
struct Rustc ;
impl Types for Rustc {
type TokenStream = T ;
type Group = G ;
}
fn make < T > ( ) -> T { loop { } }
impl TokenStream for Rustc {
fn new ( ) -> Self ::TokenStream {
let group : Self ::Group = make ( ) ;
make ( )
}
} " #,
2020-07-21 10:08:55 +00:00
expect! [ [ r #"
1061 .. 1072 ' { loop { } } ' : T
1063 .. 1070 ' loop { } ' : !
1068 .. 1070 ' { } ' : ( )
1136 .. 1199 ' { .. . } ' : T
1150 .. 1155 ' group ' : G
1171 .. 1175 ' make ' : fn make < G > ( ) -> G
1171 .. 1177 ' make ( ) ' : G
1187 .. 1191 ' make ' : fn make < T > ( ) -> T
1187 .. 1193 ' make ( ) ' : T
" #]],
2020-04-13 09:55:34 +00:00
) ;
}
2019-12-06 11:45:00 +00:00
#[ test ]
fn unify_impl_trait ( ) {
2020-07-21 10:08:55 +00:00
check_infer_with_mismatches (
r #"
2021-06-15 22:23:04 +00:00
//- minicore: sized
2021-04-10 16:03:27 +00:00
trait Trait < T > { }
2019-12-06 11:45:00 +00:00
2021-04-10 16:03:27 +00:00
fn foo ( x : impl Trait < u32 > ) { loop { } }
fn bar < T > ( x : impl Trait < T > ) -> T { loop { } }
2020-07-21 10:08:55 +00:00
2021-04-10 16:03:27 +00:00
struct S < T > ( T ) ;
impl < T > Trait < T > for S < T > { }
2020-07-21 10:08:55 +00:00
2021-04-10 16:03:27 +00:00
fn default < T > ( ) -> T { loop { } }
2020-07-21 10:08:55 +00:00
2021-04-10 16:03:27 +00:00
fn test ( ) -> impl Trait < i32 > {
let s1 = S ( default ( ) ) ;
foo ( s1 ) ;
let x : i32 = bar ( S ( default ( ) ) ) ;
S ( default ( ) )
} " #,
2020-07-21 10:08:55 +00:00
expect! [ [ r #"
2021-06-15 22:23:04 +00:00
26 .. 27 'x' : impl Trait < u32 >
46 .. 57 ' { loop { } } ' : ( )
48 .. 55 ' loop { } ' : !
53 .. 55 ' { } ' : ( )
68 .. 69 'x' : impl Trait < T >
91 .. 102 ' { loop { } } ' : T
93 .. 100 ' loop { } ' : !
98 .. 100 ' { } ' : ( )
171 .. 182 ' { loop { } } ' : T
173 .. 180 ' loop { } ' : !
178 .. 180 ' { } ' : ( )
213 .. 309 ' { .. . t ( ) ) } ' : S < { unknown } >
223 .. 225 ' s1 ' : S < u32 >
228 .. 229 'S' : S < u32 > ( u32 ) -> S < u32 >
228 .. 240 ' S ( default ( ) ) ' : S < u32 >
230 .. 237 ' default ' : fn default < u32 > ( ) -> u32
230 .. 239 ' default ( ) ' : u32
246 .. 249 ' foo ' : fn foo ( S < u32 > )
246 .. 253 ' foo ( s1 ) ' : ( )
250 .. 252 ' s1 ' : S < u32 >
263 .. 264 'x' : i32
272 .. 275 ' bar ' : fn bar < i32 > ( S < i32 > ) -> i32
272 .. 289 ' bar ( S ( .. . lt ( ) ) ) ' : i32
276 .. 277 'S' : S < i32 > ( i32 ) -> S < i32 >
276 .. 288 ' S ( default ( ) ) ' : S < i32 >
278 .. 285 ' default ' : fn default < i32 > ( ) -> i32
278 .. 287 ' default ( ) ' : i32
295 .. 296 'S' : S < { unknown } > ( { unknown } ) -> S < { unknown } >
295 .. 307 ' S ( default ( ) ) ' : S < { unknown } >
297 .. 304 ' default ' : fn default < { unknown } > ( ) -> { unknown }
297 .. 306 ' default ( ) ' : { unknown }
2020-07-21 10:08:55 +00:00
" #]],
2019-12-06 11:45:00 +00:00
) ;
}
2020-02-21 12:47:49 +00:00
#[ test ]
fn assoc_types_from_bounds ( ) {
2020-07-21 10:08:55 +00:00
check_infer (
r #"
2021-06-16 07:48:07 +00:00
//- minicore: fn
2021-04-10 16:03:27 +00:00
trait T {
type O ;
}
2020-02-21 12:47:49 +00:00
2021-04-10 16:03:27 +00:00
impl T for ( ) {
type O = ( ) ;
}
2020-02-21 12:47:49 +00:00
2021-04-10 16:03:27 +00:00
fn f < X , F > ( _v : F )
where
X : T ,
F : FnOnce ( & X ::O ) ,
{ }
2020-02-21 12:47:49 +00:00
2021-04-10 16:03:27 +00:00
fn main ( ) {
f ::< ( ) , _ > ( | z | { z ; } ) ;
} " #,
2020-07-21 10:08:55 +00:00
expect! [ [ r #"
2021-06-16 07:48:07 +00:00
72 .. 74 '_ v ' : F
117 .. 120 ' { } ' : ( )
132 .. 163 ' { .. . } ) ; } ' : ( )
138 .. 148 ' f ::< ( ) , _ > ' : fn f < ( ) , | & ( ) | -> ( ) > ( | & ( ) | -> ( ) )
138 .. 160 ' f ::< ( ) .. . z ; } ) ' : ( )
149 .. 159 ' | z | { z ; } ' : | & ( ) | -> ( )
150 .. 151 'z' : & ( )
153 .. 159 ' { z ; } ' : ( )
155 .. 156 'z' : & ( )
2020-07-21 10:08:55 +00:00
" #]],
2020-02-21 12:47:49 +00:00
) ;
}
2020-04-06 15:24:08 +00:00
2020-04-12 10:28:24 +00:00
#[ test ]
fn associated_type_bound ( ) {
2020-06-29 15:22:47 +00:00
check_types (
2020-04-12 10:28:24 +00:00
r #"
pub trait Trait {
type Item : OtherTrait < u32 > ;
}
pub trait OtherTrait < T > {
fn foo ( & self ) -> T ;
}
// this is just a workaround for chalk#234
pub struct S < T > ;
impl < T : Trait > Trait for S < T > {
type Item = < T as Trait > ::Item ;
}
fn test < T : Trait > ( ) {
let y : < S < T > as Trait > ::Item = no_matter ;
2020-06-29 15:22:47 +00:00
y . foo ( ) ;
2021-06-20 14:37:50 +00:00
} //^^^^^^^ u32
2020-04-12 10:28:24 +00:00
" #,
) ;
}
2020-04-06 15:24:08 +00:00
#[ test ]
fn dyn_trait_through_chalk ( ) {
2020-06-29 15:22:47 +00:00
check_types (
2020-04-06 15:24:08 +00:00
r #"
2021-06-15 19:02:29 +00:00
//- minicore: deref
2021-07-31 14:17:08 +00:00
struct Box < T : ? Sized > { }
impl < T : ? Sized > core ::ops ::Deref for Box < T > {
2020-04-06 15:24:08 +00:00
type Target = T ;
}
trait Trait {
fn foo ( & self ) ;
}
fn test ( x : Box < dyn Trait > ) {
2020-06-29 15:22:47 +00:00
x . foo ( ) ;
2021-06-20 14:37:50 +00:00
} //^^^^^^^ ()
2020-04-06 15:24:08 +00:00
" #,
) ;
}
2020-04-17 16:27:20 +00:00
#[ test ]
fn string_to_owned ( ) {
2020-06-29 15:22:47 +00:00
check_types (
2020-04-17 16:27:20 +00:00
r #"
struct String { }
pub trait ToOwned {
type Owned ;
fn to_owned ( & self ) -> Self ::Owned ;
}
impl ToOwned for str {
type Owned = String ;
}
fn test ( ) {
2020-06-29 15:22:47 +00:00
" foo " . to_owned ( ) ;
2021-06-20 14:37:50 +00:00
} //^^^^^^^^^^^^^^^^ String
2020-04-17 16:27:20 +00:00
" #,
) ;
}
#[ test ]
fn iterator_chain ( ) {
2021-03-15 17:22:25 +00:00
check_infer_with_mismatches (
2020-07-21 10:08:55 +00:00
r #"
2021-06-16 07:48:07 +00:00
//- minicore: fn, option
2021-04-10 16:03:27 +00:00
pub trait Iterator {
type Item ;
2020-04-17 16:27:20 +00:00
2021-04-10 16:03:27 +00:00
fn filter_map < B , F > ( self , f : F ) -> FilterMap < Self , F >
where
F : FnMut ( Self ::Item ) -> Option < B > ,
{ loop { } }
2020-04-17 16:27:20 +00:00
2021-04-10 16:03:27 +00:00
fn for_each < F > ( self , f : F )
where
F : FnMut ( Self ::Item ) ,
{ loop { } }
}
2020-04-17 16:27:20 +00:00
2021-04-10 16:03:27 +00:00
pub trait IntoIterator {
type Item ;
type IntoIter : Iterator < Item = Self ::Item > ;
fn into_iter ( self ) -> Self ::IntoIter ;
}
2020-04-17 16:27:20 +00:00
2021-04-10 16:03:27 +00:00
pub struct FilterMap < I , F > { }
impl < B , I : Iterator , F > Iterator for FilterMap < I , F >
where
F : FnMut ( I ::Item ) -> Option < B > ,
{
type Item = B ;
}
2020-04-17 16:27:20 +00:00
2021-04-10 16:03:27 +00:00
#[ stable(feature = " rust1 " , since = " 1.0.0 " ) ]
impl < I : Iterator > IntoIterator for I {
type Item = I ::Item ;
type IntoIter = I ;
2020-04-17 16:27:20 +00:00
2021-04-10 16:03:27 +00:00
fn into_iter ( self ) -> I {
self
}
}
2020-04-17 16:27:20 +00:00
2021-04-10 16:03:27 +00:00
struct Vec < T > { }
impl < T > Vec < T > {
fn new ( ) -> Self { loop { } }
}
2020-04-17 16:27:20 +00:00
2021-04-10 16:03:27 +00:00
impl < T > IntoIterator for Vec < T > {
type Item = T ;
type IntoIter = IntoIter < T > ;
}
2020-04-17 16:27:20 +00:00
2021-04-10 16:03:27 +00:00
pub struct IntoIter < T > { }
impl < T > Iterator for IntoIter < T > {
type Item = T ;
}
2020-04-17 16:27:20 +00:00
2021-04-10 16:03:27 +00:00
fn main ( ) {
Vec ::< i32 > ::new ( ) . into_iter ( )
. filter_map ( | x | if x > 0 { Some ( x as u32 ) } else { None } )
. for_each ( | y | { y ; } ) ;
} " #,
2020-07-21 10:08:55 +00:00
expect! [ [ r #"
2021-06-16 07:48:07 +00:00
61 .. 65 ' self ' : Self
67 .. 68 'f' : F
152 .. 163 ' { loop { } } ' : FilterMap < Self , F >
154 .. 161 ' loop { } ' : !
159 .. 161 ' { } ' : ( )
184 .. 188 ' self ' : Self
190 .. 191 'f' : F
240 .. 251 ' { loop { } } ' : ( )
242 .. 249 ' loop { } ' : !
247 .. 249 ' { } ' : ( )
360 .. 364 ' self ' : Self
689 .. 693 ' self ' : I
700 .. 720 ' { .. . } ' : I
710 .. 714 ' self ' : I
779 .. 790 ' { loop { } } ' : Vec < T >
781 .. 788 ' loop { } ' : !
786 .. 788 ' { } ' : ( )
977 .. 1104 ' { .. . } ) ; } ' : ( )
983 .. 998 ' Vec ::< i32 > ::new ' : fn new < i32 > ( ) -> Vec < i32 >
983 .. 1000 ' Vec ::< .. . :new ( ) ' : Vec < i32 >
983 .. 1012 ' Vec ::< .. . iter ( ) ' : IntoIter < i32 >
983 .. 1075 ' Vec ::< .. . one } ) ' : FilterMap < IntoIter < i32 > , | i32 | -> Option < u32 > >
983 .. 1101 ' Vec ::< .. . y ; } ) ' : ( )
1029 .. 1074 ' | x | if .. . None } ' : | i32 | -> Option < u32 >
1030 .. 1031 'x' : i32
1033 .. 1074 ' if x > .. . None } ' : Option < u32 >
1036 .. 1037 'x' : i32
1036 .. 1041 ' x > 0 ' : bool
1040 .. 1041 '0' : i32
1042 .. 1060 ' { Some .. . u32 ) } ' : Option < u32 >
1044 .. 1048 ' Some ' : Some < u32 > ( u32 ) -> Option < u32 >
1044 .. 1058 ' Some ( x as u32 ) ' : Option < u32 >
1049 .. 1050 'x' : i32
1049 .. 1057 ' x as u32 ' : u32
1066 .. 1074 ' { None } ' : Option < u32 >
1068 .. 1072 ' None ' : Option < u32 >
1090 .. 1100 ' | y | { y ; } ' : | u32 | -> ( )
1091 .. 1092 'y' : u32
1094 .. 1100 ' { y ; } ' : ( )
1096 .. 1097 'y' : u32
2020-07-21 10:08:55 +00:00
" #]],
2020-04-17 16:27:20 +00:00
) ;
}
#[ test ]
fn nested_assoc ( ) {
2020-06-29 15:22:47 +00:00
check_types (
2020-04-17 16:27:20 +00:00
r #"
struct Bar ;
struct Foo ;
trait A {
type OutputA ;
}
impl A for Bar {
type OutputA = Foo ;
}
trait B {
type Output ;
fn foo ( ) -> Self ::Output ;
}
impl < T :A > B for T {
type Output = T ::OutputA ;
fn foo ( ) -> Self ::Output { loop { } }
}
fn main ( ) {
2020-06-29 15:22:47 +00:00
Bar ::foo ( ) ;
2021-06-20 14:37:50 +00:00
} //^^^^^^^^^^ Foo
2020-04-17 16:27:20 +00:00
" #,
) ;
}
2020-04-17 17:41:37 +00:00
#[ test ]
fn trait_object_no_coercion ( ) {
2020-07-21 10:08:55 +00:00
check_infer_with_mismatches (
r #"
2021-04-10 16:03:27 +00:00
trait Foo { }
2020-04-17 17:41:37 +00:00
2021-04-10 16:03:27 +00:00
fn foo ( x : & dyn Foo ) { }
2020-04-17 17:41:37 +00:00
2021-04-10 16:03:27 +00:00
fn test ( x : & dyn Foo ) {
foo ( x ) ;
} " #,
2020-07-21 10:08:55 +00:00
expect! [ [ r #"
21 .. 22 'x' : & dyn Foo
34 .. 36 ' { } ' : ( )
46 .. 47 'x' : & dyn Foo
59 .. 74 ' { foo ( x ) ; } ' : ( )
65 .. 68 ' foo ' : fn foo ( & dyn Foo )
65 .. 71 ' foo ( x ) ' : ( )
69 .. 70 'x' : & dyn Foo
" #]],
2020-04-17 17:41:37 +00:00
) ;
}
2020-05-22 16:52:07 +00:00
#[ test ]
fn builtin_copy ( ) {
2020-07-21 10:08:55 +00:00
check_infer_with_mismatches (
r #"
2021-06-18 19:24:48 +00:00
//- minicore: copy
2021-04-10 16:03:27 +00:00
struct IsCopy ;
impl Copy for IsCopy { }
struct NotCopy ;
2020-05-22 16:52:07 +00:00
2021-04-10 16:03:27 +00:00
trait Test { fn test ( & self ) -> bool ; }
impl < T : Copy > Test for T { }
2020-05-22 16:52:07 +00:00
2021-04-10 16:03:27 +00:00
fn test ( ) {
IsCopy . test ( ) ;
NotCopy . test ( ) ;
( IsCopy , IsCopy ) . test ( ) ;
( IsCopy , NotCopy ) . test ( ) ;
} " #,
2020-07-21 10:08:55 +00:00
expect! [ [ r #"
2021-06-18 19:24:48 +00:00
78 .. 82 ' self ' : & Self
134 .. 235 ' { .. . t ( ) ; } ' : ( )
140 .. 146 ' IsCopy ' : IsCopy
140 .. 153 ' IsCopy . test ( ) ' : bool
159 .. 166 ' NotCopy ' : NotCopy
159 .. 173 ' NotCopy . test ( ) ' : { unknown }
179 .. 195 ' ( IsCop .. . sCopy ) ' : ( IsCopy , IsCopy )
179 .. 202 ' ( IsCop .. . test ( ) ' : bool
180 .. 186 ' IsCopy ' : IsCopy
188 .. 194 ' IsCopy ' : IsCopy
208 .. 225 ' ( IsCop .. . tCopy ) ' : ( IsCopy , NotCopy )
208 .. 232 ' ( IsCop .. . test ( ) ' : { unknown }
209 .. 215 ' IsCopy ' : IsCopy
217 .. 224 ' NotCopy ' : NotCopy
2020-07-21 10:08:55 +00:00
" #]],
2020-05-22 16:52:07 +00:00
) ;
}
2020-05-22 16:15:53 +00:00
#[ test ]
fn builtin_fn_def_copy ( ) {
2020-07-21 10:08:55 +00:00
check_infer_with_mismatches (
r #"
2021-06-18 19:24:48 +00:00
//- minicore: copy
2021-04-10 16:03:27 +00:00
fn foo ( ) { }
fn bar < T : Copy > ( T ) -> T { }
struct Struct ( usize ) ;
enum Enum { Variant ( usize ) }
2020-07-21 10:08:55 +00:00
2021-04-10 16:03:27 +00:00
trait Test { fn test ( & self ) -> bool ; }
impl < T : Copy > Test for T { }
2020-07-21 10:08:55 +00:00
2021-04-10 16:03:27 +00:00
fn test ( ) {
foo . test ( ) ;
bar . test ( ) ;
Struct . test ( ) ;
Enum ::Variant . test ( ) ;
} " #,
2020-07-21 10:08:55 +00:00
expect! [ [ r #"
2021-06-18 19:24:48 +00:00
9 .. 11 ' { } ' : ( )
28 .. 29 'T' : { unknown }
36 .. 38 ' { } ' : ( )
36 .. 38 : expected T , got ( )
113 .. 117 ' self ' : & Self
169 .. 249 ' { .. . t ( ) ; } ' : ( )
175 .. 178 ' foo ' : fn foo ( )
175 .. 185 ' foo . test ( ) ' : bool
191 .. 194 ' bar ' : fn bar < { unknown } > ( { unknown } ) -> { unknown }
191 .. 201 ' bar . test ( ) ' : bool
207 .. 213 ' Struct ' : Struct ( usize ) -> Struct
207 .. 220 ' Struct . test ( ) ' : bool
226 .. 239 ' Enum ::Variant ' : Variant ( usize ) -> Enum
226 .. 246 ' Enum ::.. . test ( ) ' : bool
2020-07-21 10:08:55 +00:00
" #]],
2020-05-22 16:15:53 +00:00
) ;
}
2020-05-22 17:13:17 +00:00
#[ test ]
fn builtin_fn_ptr_copy ( ) {
2020-07-21 10:08:55 +00:00
check_infer_with_mismatches (
r #"
2021-06-18 19:24:48 +00:00
//- minicore: copy
2021-04-10 16:03:27 +00:00
trait Test { fn test ( & self ) -> bool ; }
impl < T : Copy > Test for T { }
2020-05-22 17:13:17 +00:00
2021-04-10 16:03:27 +00:00
fn test ( f1 : fn ( ) , f2 : fn ( usize ) -> u8 , f3 : fn ( u8 , u8 ) -> & u8 ) {
f1 . test ( ) ;
f2 . test ( ) ;
f3 . test ( ) ;
} " #,
2020-07-21 10:08:55 +00:00
expect! [ [ r #"
2021-06-18 19:24:48 +00:00
22 .. 26 ' self ' : & Self
76 .. 78 ' f1 ' : fn ( )
86 .. 88 ' f2 ' : fn ( usize ) -> u8
107 .. 109 ' f3 ' : fn ( u8 , u8 ) -> & u8
130 .. 178 ' { .. . t ( ) ; } ' : ( )
136 .. 138 ' f1 ' : fn ( )
136 .. 145 ' f1 . test ( ) ' : bool
151 .. 153 ' f2 ' : fn ( usize ) -> u8
151 .. 160 ' f2 . test ( ) ' : bool
166 .. 168 ' f3 ' : fn ( u8 , u8 ) -> & u8
166 .. 175 ' f3 . test ( ) ' : bool
2020-07-21 10:08:55 +00:00
" #]],
2020-05-22 17:13:17 +00:00
) ;
}
2020-05-22 16:52:07 +00:00
#[ test ]
fn builtin_sized ( ) {
2020-07-21 10:08:55 +00:00
check_infer_with_mismatches (
r #"
2021-06-18 19:24:48 +00:00
//- minicore: sized
2021-04-10 16:03:27 +00:00
trait Test { fn test ( & self ) -> bool ; }
impl < T : Sized > Test for T { }
2020-05-22 16:52:07 +00:00
2021-04-10 16:03:27 +00:00
fn test ( ) {
1 u8 . test ( ) ;
( * " foo " ) . test ( ) ; // not Sized
( 1 u8 , 1 u8 ) . test ( ) ;
( 1 u8 , * " foo " ) . test ( ) ; // not Sized
} " #,
2020-07-21 10:08:55 +00:00
expect! [ [ r #"
2021-06-18 19:24:48 +00:00
22 .. 26 ' self ' : & Self
79 .. 194 ' { .. . ized } ' : ( )
85 .. 88 ' 1 u8 ' : u8
85 .. 95 ' 1 u8 . test ( ) ' : bool
101 .. 116 ' ( * " foo " ) . test ( ) ' : { unknown }
102 .. 108 ' * " foo " ' : str
103 .. 108 ' " foo " ' : & str
135 .. 145 ' ( 1 u8 , 1 u8 ) ' : ( u8 , u8 )
135 .. 152 ' ( 1 u8 , .. . test ( ) ' : bool
136 .. 139 ' 1 u8 ' : u8
141 .. 144 ' 1 u8 ' : u8
158 .. 171 ' ( 1 u8 , * " foo " ) ' : ( u8 , str )
158 .. 178 ' ( 1 u8 , .. . test ( ) ' : { unknown }
159 .. 162 ' 1 u8 ' : u8
164 .. 170 ' * " foo " ' : str
165 .. 170 ' " foo " ' : & str
2020-07-21 10:08:55 +00:00
" #]],
2020-05-22 16:52:07 +00:00
) ;
}
2020-05-29 14:49:52 +00:00
#[ test ]
fn integer_range_iterate ( ) {
2020-06-29 15:22:47 +00:00
check_types (
2020-05-29 14:49:52 +00:00
r #"
2020-06-11 14:22:31 +00:00
//- /main.rs crate:main deps:core
2020-05-29 14:49:52 +00:00
fn test ( ) {
2020-06-29 15:22:47 +00:00
for x in 0 .. 100 { x ; }
} //^ i32
2020-05-29 14:49:52 +00:00
2020-06-11 14:22:31 +00:00
//- /core.rs crate:core
2020-05-29 14:49:52 +00:00
pub mod ops {
pub struct Range < Idx > {
pub start : Idx ,
pub end : Idx ,
}
}
pub mod iter {
pub trait Iterator {
type Item ;
}
pub trait IntoIterator {
type Item ;
type IntoIter : Iterator < Item = Self ::Item > ;
}
impl < T > IntoIterator for T where T : Iterator {
type Item = < T as Iterator > ::Item ;
type IntoIter = Self ;
}
}
trait Step { }
impl Step for i32 { }
impl Step for i64 { }
impl < A : Step > iter ::Iterator for ops ::Range < A > {
type Item = A ;
}
" #,
) ;
}
2020-06-20 09:32:01 +00:00
#[ test ]
fn infer_closure_arg ( ) {
2020-07-21 10:08:55 +00:00
check_infer (
r #"
2021-04-10 16:03:27 +00:00
//- /lib.rs
2020-06-20 09:32:01 +00:00
2021-04-10 16:03:27 +00:00
enum Option < T > {
None ,
Some ( T )
}
2020-07-21 10:08:55 +00:00
2021-04-10 16:03:27 +00:00
fn foo ( ) {
let s = Option ::None ;
let f = | x : Option < i32 > | { } ;
( & f ) ( s )
} " #,
2020-07-21 10:08:55 +00:00
expect! [ [ r #"
52 .. 126 ' { .. . ) ( s ) } ' : ( )
62 .. 63 's' : Option < i32 >
66 .. 78 ' Option ::None ' : Option < i32 >
88 .. 89 'f' : | Option < i32 > | -> ( )
92 .. 111 ' | x : Op .. . 2 > | { } ' : | Option < i32 > | -> ( )
93 .. 94 'x' : Option < i32 >
109 .. 111 ' { } ' : ( )
117 .. 124 ' ( & f ) ( s ) ' : ( )
118 .. 120 ' & f ' : & | Option < i32 > | -> ( )
119 .. 120 'f' : | Option < i32 > | -> ( )
122 .. 123 's' : Option < i32 >
" #]],
2020-06-20 09:32:01 +00:00
) ;
}
#[ test ]
2021-06-29 15:35:37 +00:00
fn dyn_fn_param_informs_call_site_closure_signature ( ) {
cov_mark ::check! ( dyn_fn_param_informs_call_site_closure_signature ) ;
check_types (
r #"
//- minicore: fn, coerce_unsized
struct S ;
impl S {
fn inherent ( & self ) -> u8 { 0 }
}
fn take_dyn_fn ( f : & dyn Fn ( S ) ) { }
fn f ( ) {
take_dyn_fn ( & | x | { x . inherent ( ) ; } ) ;
//^^^^^^^^^^^^ u8
}
" #,
) ;
}
#[ test ]
2020-06-20 09:32:01 +00:00
fn infer_fn_trait_arg ( ) {
2021-03-15 17:22:25 +00:00
check_infer_with_mismatches (
2020-07-21 10:08:55 +00:00
r #"
2021-06-16 07:48:07 +00:00
//- minicore: fn, option
fn foo < F , T > ( f : F ) -> T
where
F : Fn ( Option < i32 > ) -> T ,
{
let s = None ;
f ( s )
}
" #,
2020-07-21 10:08:55 +00:00
expect! [ [ r #"
2021-06-16 07:48:07 +00:00
13 .. 14 'f' : F
59 .. 89 ' { .. . f ( s ) } ' : T
69 .. 70 's' : Option < i32 >
73 .. 77 ' None ' : Option < i32 >
83 .. 84 'f' : F
83 .. 87 ' f ( s ) ' : T
85 .. 86 's' : Option < i32 >
2020-07-21 10:08:55 +00:00
" #]],
2020-06-20 09:32:01 +00:00
) ;
}
#[ test ]
fn infer_box_fn_arg ( ) {
2021-05-25 12:24:08 +00:00
// The type mismatch is because we don't define Unsize and CoerceUnsized
2021-03-15 17:22:25 +00:00
check_infer_with_mismatches (
2020-07-21 10:08:55 +00:00
r #"
2021-06-18 19:47:02 +00:00
//- minicore: fn, deref, option
2021-04-10 16:03:27 +00:00
#[ lang = " owned_box " ]
pub struct Box < T : ? Sized > {
inner : * mut T ,
}
2020-06-20 09:32:01 +00:00
2021-06-18 19:47:02 +00:00
impl < T : ? Sized > core ::ops ::Deref for Box < T > {
2021-04-10 16:03:27 +00:00
type Target = T ;
2020-06-20 09:32:01 +00:00
2021-04-10 16:03:27 +00:00
fn deref ( & self ) -> & T {
& self . inner
}
}
2020-06-20 09:32:01 +00:00
2021-04-10 16:03:27 +00:00
fn foo ( ) {
2021-06-18 19:47:02 +00:00
let s = None ;
2021-04-10 16:03:27 +00:00
let f : Box < dyn FnOnce ( & Option < i32 > ) > = box ( | ps | { } ) ;
f ( & s ) ;
} " #,
2020-07-21 10:08:55 +00:00
expect! [ [ r #"
2021-06-18 19:47:02 +00:00
154 .. 158 ' self ' : & Box < T >
166 .. 193 ' { .. . } ' : & T
176 .. 187 ' & self . inner ' : & * mut T
177 .. 181 ' self ' : & Box < T >
177 .. 187 ' self . inner ' : * mut T
206 .. 296 ' { .. . & s ) ; } ' : ( )
216 .. 217 's' : Option < i32 >
220 .. 224 ' None ' : Option < i32 >
234 .. 235 'f' : Box < dyn FnOnce ( & Option < i32 > ) >
269 .. 282 ' box ( | ps | { } ) ' : Box < | { unknown } | -> ( ) >
274 .. 281 ' | ps | { } ' : | { unknown } | -> ( )
275 .. 277 ' ps ' : { unknown }
279 .. 281 ' { } ' : ( )
288 .. 289 'f' : Box < dyn FnOnce ( & Option < i32 > ) >
288 .. 293 ' f ( & s ) ' : ( )
290 .. 292 ' & s ' : & Option < i32 >
291 .. 292 's' : Option < i32 >
269 .. 282 : expected Box < dyn FnOnce ( & Option < i32 > ) > , got Box < | { unknown } | -> ( ) >
2020-07-21 10:08:55 +00:00
" #]],
2020-06-20 09:32:01 +00:00
) ;
}
2020-06-20 09:43:40 +00:00
#[ test ]
fn infer_dyn_fn_output ( ) {
2020-07-10 16:30:32 +00:00
check_types (
r #"
2021-06-16 07:48:07 +00:00
//- minicore: fn
2020-07-10 16:30:32 +00:00
fn foo ( ) {
let f : & dyn Fn ( ) -> i32 ;
f ( ) ;
//^^^ i32
} " #,
) ;
}
2020-06-20 09:43:40 +00:00
2020-07-10 16:30:32 +00:00
#[ test ]
fn infer_dyn_fn_once_output ( ) {
check_types (
r #"
2021-06-16 07:48:07 +00:00
//- minicore: fn
2020-07-10 16:30:32 +00:00
fn foo ( ) {
let f : dyn FnOnce ( ) -> i32 ;
f ( ) ;
//^^^ i32
} " #,
2020-06-20 09:43:40 +00:00
) ;
}
2020-06-28 19:17:27 +00:00
#[ test ]
2020-07-11 17:12:10 +00:00
fn variable_kinds_1 ( ) {
2020-06-28 19:17:27 +00:00
check_types (
r #"
trait Trait < T > { fn get ( self , t : T ) -> T ; }
struct S ;
impl Trait < u128 > for S { }
impl Trait < f32 > for S { }
fn test ( ) {
S . get ( 1 ) ;
//^^^^^^^^ u128
S . get ( 1. ) ;
2021-06-20 14:37:50 +00:00
//^^^^^^^^^ f32
2020-06-28 19:17:27 +00:00
}
" #,
) ;
}
2020-07-11 17:12:10 +00:00
#[ test ]
fn variable_kinds_2 ( ) {
check_types (
r #"
trait Trait { fn get ( self ) -> Self ; }
impl Trait for u128 { }
impl Trait for f32 { }
fn test ( ) {
1. get ( ) ;
//^^^^^^^ u128
( 1. ) . get ( ) ;
//^^^^^^^^^^ f32
}
" #,
) ;
}
2020-07-21 15:52:43 +00:00
#[ test ]
fn underscore_import ( ) {
check_types (
r #"
mod tr {
pub trait Tr {
fn method ( & self ) -> u8 { 0 }
}
}
struct Tr ;
impl crate ::tr ::Tr for Tr { }
use crate ::tr ::Tr as _ ;
fn test ( ) {
Tr . method ( ) ;
//^^^^^^^^^^^ u8
}
" #,
) ;
}
2021-02-10 14:54:21 +00:00
#[ test ]
fn inner_use ( ) {
check_types (
r #"
mod m {
pub trait Tr {
fn method ( & self ) -> u8 { 0 }
}
impl Tr for ( ) { }
}
fn f ( ) {
use m ::Tr ;
( ) . method ( ) ;
//^^^^^^^^^^^ u8
}
" #,
) ;
}
2021-03-09 17:18:35 +00:00
#[ test ]
fn trait_in_scope_with_inner_item ( ) {
check_infer (
r #"
mod m {
pub trait Tr {
fn method ( & self ) -> u8 { 0 }
}
impl Tr for ( ) { }
}
use m ::Tr ;
fn f ( ) {
fn inner ( ) {
( ) . method ( ) ;
//^^^^^^^^^^^ u8
}
2021-04-10 16:03:27 +00:00
} " #,
2021-03-09 17:18:35 +00:00
expect! [ [ r #"
46 .. 50 ' self ' : & Self
58 .. 63 ' { 0 } ' : u8
60 .. 61 '0' : u8
115 .. 185 ' { .. . } } ' : ( )
132 .. 183 ' { .. . } ' : ( )
142 .. 144 ' ( ) ' : ( )
142 .. 153 ' ( ) . method ( ) ' : u8
" #]],
) ;
}
2021-02-10 14:54:21 +00:00
#[ test ]
fn inner_use_in_block ( ) {
check_types (
r #"
mod m {
pub trait Tr {
fn method ( & self ) -> u8 { 0 }
}
impl Tr for ( ) { }
}
fn f ( ) {
{
use m ::Tr ;
( ) . method ( ) ;
//^^^^^^^^^^^ u8
}
{
( ) . method ( ) ;
//^^^^^^^^^^^ {unknown}
}
2021-02-10 16:37:15 +00:00
( ) . method ( ) ;
//^^^^^^^^^^^ {unknown}
2021-02-10 14:54:21 +00:00
}
" #,
) ;
}
2021-03-17 21:30:09 +00:00
#[ test ]
fn nested_inner_function_calling_self ( ) {
check_infer (
r #"
struct S ;
fn f ( ) {
fn inner ( ) -> S {
let s = inner ( ) ;
}
2021-04-10 16:03:27 +00:00
} " #,
2021-03-17 21:30:09 +00:00
expect! [ [ r #"
17 .. 73 ' { .. . } } ' : ( )
39 .. 71 ' { .. . } ' : ( )
53 .. 54 's' : S
57 .. 62 ' inner ' : fn inner ( ) -> S
57 .. 64 ' inner ( ) ' : S
" #]],
)
}
2021-03-22 18:13:43 +00:00
#[ test ]
fn infer_default_trait_type_parameter ( ) {
check_infer (
r #"
struct A ;
trait Op < RHS = Self > {
type Output ;
fn do_op ( self , rhs : RHS ) -> Self ::Output ;
}
impl Op for A {
type Output = bool ;
fn do_op ( self , rhs : Self ) -> Self ::Output {
true
}
}
fn test ( ) {
let x = A ;
let y = A ;
let r = x . do_op ( y ) ;
2021-04-10 16:03:27 +00:00
} " #,
2021-03-22 18:13:43 +00:00
expect! [ [ r #"
63 .. 67 ' self ' : Self
69 .. 72 ' rhs ' : RHS
153 .. 157 ' self ' : A
159 .. 162 ' rhs ' : A
186 .. 206 ' { .. . } ' : bool
196 .. 200 ' true ' : bool
220 .. 277 ' { .. . ( y ) ; } ' : ( )
230 .. 231 'x' : A
234 .. 235 'A' : A
245 .. 246 'y' : A
249 .. 250 'A' : A
260 .. 261 'r' : bool
264 .. 265 'x' : A
264 .. 274 ' x . do_op ( y ) ' : bool
272 .. 273 'y' : A
" #]],
)
}
2021-03-30 19:43:23 +00:00
#[ test ]
fn qualified_path_as_qualified_trait ( ) {
check_infer (
r #"
mod foo {
pub trait Foo {
type Target ;
}
pub trait Bar {
type Output ;
fn boo ( ) -> Self ::Output {
loop { }
}
}
}
struct F ;
impl foo ::Foo for F {
type Target = ( ) ;
}
impl foo ::Bar for F {
type Output = < F as foo ::Foo > ::Target ;
}
fn foo ( ) {
use foo ::Bar ;
let x = < F as Bar > ::boo ( ) ;
2021-04-10 16:03:27 +00:00
} " #,
2021-03-30 19:43:23 +00:00
expect! [ [ r #"
132 .. 163 ' { .. . } ' : Bar ::Output < Self >
146 .. 153 ' loop { } ' : !
151 .. 153 ' { } ' : ( )
306 .. 358 ' { .. . o ( ) ; } ' : ( )
334 .. 335 'x' : ( )
338 .. 353 ' < F as Bar > ::boo ' : fn boo < F > ( ) -> < F as Bar > ::Output
338 .. 355 ' < F as .. . :boo ( ) ' : ( )
" #]],
) ;
}
2021-04-09 13:25:12 +00:00
#[ test ]
fn renamed_extern_crate_in_block ( ) {
check_types (
r #"
//- /lib.rs crate:lib deps:serde
use serde ::Deserialize ;
struct Foo { }
const _ : ( ) = {
extern crate serde as _serde ;
impl _serde ::Deserialize for Foo {
fn deserialize ( ) -> u8 { 0 }
}
} ;
fn foo ( ) {
Foo ::deserialize ( ) ;
//^^^^^^^^^^^^^^^^^^ u8
}
//- /serde.rs crate:serde
pub trait Deserialize {
fn deserialize ( ) -> u8 ;
2021-04-10 16:03:27 +00:00
} " #,
2021-04-09 13:25:12 +00:00
) ;
}
2021-04-10 15:52:24 +00:00
#[ test ]
fn bin_op_adt_with_rhs_primitive ( ) {
check_infer_with_mismatches (
r #"
#[ lang = " add " ]
pub trait Add < Rhs = Self > {
type Output ;
fn add ( self , rhs : Rhs ) -> Self ::Output ;
}
struct Wrapper ( u32 ) ;
impl Add < u32 > for Wrapper {
type Output = Self ;
fn add ( self , rhs : u32 ) -> Wrapper {
Wrapper ( rhs )
}
}
fn main ( ) {
let wrapped = Wrapper ( 10 ) ;
let num : u32 = 2 ;
let res = wrapped + num ;
} " #,
expect! [ [ r #"
72 .. 76 ' self ' : Self
78 .. 81 ' rhs ' : Rhs
192 .. 196 ' self ' : Wrapper
198 .. 201 ' rhs ' : u32
219 .. 247 ' { .. . } ' : Wrapper
229 .. 236 ' Wrapper ' : Wrapper ( u32 ) -> Wrapper
229 .. 241 ' Wrapper ( rhs ) ' : Wrapper
237 .. 240 ' rhs ' : u32
259 .. 345 ' { .. . um ; } ' : ( )
269 .. 276 ' wrapped ' : Wrapper
279 .. 286 ' Wrapper ' : Wrapper ( u32 ) -> Wrapper
279 .. 290 ' Wrapper ( 10 ) ' : Wrapper
287 .. 289 ' 10 ' : u32
300 .. 303 ' num ' : u32
311 .. 312 '2' : u32
322 .. 325 ' res ' : Wrapper
328 .. 335 ' wrapped ' : Wrapper
328 .. 341 ' wrapped + num ' : Wrapper
338 .. 341 ' num ' : u32
" #]],
)
}
2021-05-14 07:59:30 +00:00
#[ test ]
fn array_length ( ) {
check_infer (
r #"
trait T {
type Output ;
fn do_thing ( & self ) -> Self ::Output ;
}
impl T for [ u8 ; 4 ] {
type Output = usize ;
fn do_thing ( & self ) -> Self ::Output {
2
}
}
impl T for [ u8 ; 2 ] {
type Output = u8 ;
fn do_thing ( & self ) -> Self ::Output {
2
}
}
fn main ( ) {
let v = [ 0 u8 ; 2 ] ;
let v2 = v . do_thing ( ) ;
let v3 = [ 0 u8 ; 4 ] ;
let v4 = v3 . do_thing ( ) ;
}
" #,
expect! [ [ r #"
44 .. 48 ' self ' : & Self
133 .. 137 ' self ' : & [ u8 ; 4 ]
155 .. 172 ' { .. . } ' : usize
165 .. 166 '2' : usize
236 .. 240 ' self ' : & [ u8 ; 2 ]
258 .. 275 ' { .. . } ' : u8
268 .. 269 '2' : u8
289 .. 392 ' { .. . g ( ) ; } ' : ( )
299 .. 300 'v' : [ u8 ; 2 ]
303 .. 311 ' [ 0 u8 ; 2 ] ' : [ u8 ; 2 ]
304 .. 307 ' 0 u8 ' : u8
309 .. 310 '2' : usize
321 .. 323 ' v2 ' : u8
326 .. 327 'v' : [ u8 ; 2 ]
326 .. 338 ' v . do_thing ( ) ' : u8
348 .. 350 ' v3 ' : [ u8 ; 4 ]
353 .. 361 ' [ 0 u8 ; 4 ] ' : [ u8 ; 4 ]
354 .. 357 ' 0 u8 ' : u8
359 .. 360 '4' : usize
371 .. 373 ' v4 ' : usize
376 .. 378 ' v3 ' : [ u8 ; 4 ]
376 .. 389 ' v3 . do_thing ( ) ' : usize
" #]],
)
}
// FIXME: We should infer the length of the returned array :)
#[ test ]
fn const_generics ( ) {
check_infer (
r #"
trait T {
type Output ;
fn do_thing ( & self ) -> Self ::Output ;
}
impl < const L : usize > T for [ u8 ; L ] {
type Output = [ u8 ; L ] ;
fn do_thing ( & self ) -> Self ::Output {
* self
}
}
fn main ( ) {
let v = [ 0 u8 ; 2 ] ;
let v2 = v . do_thing ( ) ;
}
" #,
expect! [ [ r #"
44 .. 48 ' self ' : & Self
151 .. 155 ' self ' : & [ u8 ; _ ]
173 .. 194 ' { .. . } ' : [ u8 ; _ ]
183 .. 188 ' * self ' : [ u8 ; _ ]
184 .. 188 ' self ' : & [ u8 ; _ ]
208 .. 260 ' { .. . g ( ) ; } ' : ( )
218 .. 219 'v' : [ u8 ; 2 ]
222 .. 230 ' [ 0 u8 ; 2 ] ' : [ u8 ; 2 ]
223 .. 226 ' 0 u8 ' : u8
228 .. 229 '2' : usize
240 .. 242 ' v2 ' : [ u8 ; _ ]
245 .. 246 'v' : [ u8 ; 2 ]
245 .. 257 ' v . do_thing ( ) ' : [ u8 ; _ ]
" #]],
)
}
2021-05-25 12:24:08 +00:00
#[ test ]
fn fn_returning_unit ( ) {
check_infer_with_mismatches (
r #"
2021-06-16 07:48:07 +00:00
//- minicore: fn
2021-05-25 12:24:08 +00:00
fn test < F : FnOnce ( ) > ( f : F ) {
let _ : ( ) = f ( ) ;
} " #,
expect! [ [ r #"
2021-06-16 07:48:07 +00:00
21 .. 22 'f' : F
27 .. 51 ' { .. . f ( ) ; } ' : ( )
37 .. 38 '_' : ( )
45 .. 46 'f' : F
45 .. 48 ' f ( ) ' : ( )
2021-05-25 12:24:08 +00:00
" #]],
) ;
}
2021-05-25 14:08:18 +00:00
#[ test ]
fn trait_in_scope_of_trait_impl ( ) {
check_infer (
r #"
mod foo {
pub trait Foo {
fn foo ( self ) ;
fn bar ( self ) -> usize { 0 }
}
}
impl foo ::Foo for u32 {
fn foo ( self ) {
let _x = self . bar ( ) ;
}
}
" #,
expect! [ [ r #"
45 .. 49 ' self ' : Self
67 .. 71 ' self ' : Self
82 .. 87 ' { 0 } ' : usize
84 .. 85 '0' : usize
131 .. 135 ' self ' : u32
137 .. 173 ' { .. . } ' : ( )
151 .. 153 '_ x ' : usize
156 .. 160 ' self ' : u32
156 .. 166 ' self . bar ( ) ' : usize
" #]],
) ;
}
2021-05-29 16:16:20 +00:00
#[ test ]
fn infer_async_ret_type ( ) {
check_types (
r #"
2021-06-15 20:07:25 +00:00
//- minicore: future, result
2021-05-29 16:16:20 +00:00
struct Fooey ;
impl Fooey {
fn collect < B : Convert > ( self ) -> B {
B ::new ( )
}
}
trait Convert {
fn new ( ) -> Self ;
}
impl Convert for u32 {
2021-06-20 14:37:50 +00:00
fn new ( ) -> Self { 0 }
2021-05-29 16:16:20 +00:00
}
async fn get_accounts ( ) -> Result < u32 , ( ) > {
let ret = Fooey . collect ( ) ;
2021-06-20 14:37:50 +00:00
// ^^^^^^^^^^^^^^^ u32
2021-05-29 16:16:20 +00:00
Ok ( ret )
}
" #,
) ;
}
2021-06-13 11:00:34 +00:00
#[ test ]
fn local_impl_1 ( ) {
2021-06-20 14:37:50 +00:00
check! ( block_local_impls ) ;
2021-06-13 11:00:34 +00:00
check_types (
r #"
trait Trait < T > {
fn foo ( & self ) -> T ;
}
fn test ( ) {
struct S ;
impl Trait < u32 > for S {
2021-06-20 14:37:50 +00:00
fn foo ( & self ) -> u32 { 0 }
2021-06-13 11:00:34 +00:00
}
S . foo ( ) ;
// ^^^^^^^ u32
}
" #,
) ;
}
#[ test ]
fn local_impl_2 ( ) {
2021-06-20 14:37:50 +00:00
check! ( block_local_impls ) ;
2021-06-13 11:00:34 +00:00
check_types (
r #"
struct S ;
fn test ( ) {
trait Trait < T > {
fn foo ( & self ) -> T ;
}
impl Trait < u32 > for S {
2021-06-20 14:37:50 +00:00
fn foo ( & self ) -> u32 { 0 }
2021-06-13 11:00:34 +00:00
}
S . foo ( ) ;
// ^^^^^^^ u32
}
" #,
) ;
}
#[ test ]
fn local_impl_3 ( ) {
2021-06-20 14:37:50 +00:00
check! ( block_local_impls ) ;
2021-06-13 11:00:34 +00:00
check_types (
r #"
trait Trait < T > {
fn foo ( & self ) -> T ;
}
fn test ( ) {
struct S1 ;
{
struct S2 ;
impl Trait < S1 > for S2 {
2021-06-20 14:37:50 +00:00
fn foo ( & self ) -> S1 { S1 }
2021-06-13 11:00:34 +00:00
}
S2 . foo ( ) ;
// ^^^^^^^^ S1
}
}
" #,
) ;
}
2021-06-12 16:47:07 +00:00
#[ test ]
fn associated_type_sized_bounds ( ) {
check_infer (
r #"
2021-08-03 12:01:00 +00:00
//- minicore: sized
2021-06-12 16:47:07 +00:00
struct Yes ;
trait IsSized { const IS_SIZED : Yes ; }
impl < T : Sized > IsSized for T { const IS_SIZED : Yes = Yes ; }
trait Foo {
type Explicit : Sized ;
type Implicit ;
type Relaxed : ? Sized ;
}
fn f < F : Foo > ( ) {
F ::Explicit ::IS_SIZED ;
F ::Implicit ::IS_SIZED ;
F ::Relaxed ::IS_SIZED ;
}
" #,
expect! [ [ r #"
2021-08-03 12:01:00 +00:00
104 .. 107 ' Yes ' : Yes
212 .. 295 ' { .. . ZED ; } ' : ( )
218 .. 239 ' F ::Exp .. . _SIZED ' : Yes
245 .. 266 ' F ::Imp .. . _SIZED ' : Yes
272 .. 292 ' F ::Rel .. . _SIZED ' : { unknown }
2021-06-12 16:47:07 +00:00
" #]],
) ;
}