1 use expect_test::expect;
3 use super::{check_infer, check_infer_with_mismatches, check_no_mismatches, check_types};
6 fn infer_block_expr_type_mismatch() {
10 let a: i32 = { 1i64 };
14 10..40 '{ ...4 }; }': ()
16 29..37 '{ 1i64 }': i64
26 //- minicore: coerce_unsized
29 fn f<T>(_: &[T]) -> T { loop {} }
30 fn g<T>(_: S<&[T]>) -> T { loop {} }
32 fn gen<T>() -> *mut [T; 2] { loop {} }
33 fn test1<U>() -> *mut [U] {
38 let arr: &[u8; 1] = &[1];
42 let c: &[_] = { arr };
43 let d = g(S { a: arr });
44 let e: [&[_]; 1] = [arr];
45 let f: [&[_]; 2] = [arr; 2];
46 let g: (&[_], &[_]) = (arr, arr);
51 44..55 '{ loop {} }': T
55 81..92 '{ loop {} }': T
58 121..132 '{ loop {} }': *mut [T; 2]
61 159..172 '{ gen() }': *mut [U]
62 165..168 'gen': fn gen<U>() -> *mut [U; 2]
63 165..170 'gen()': *mut [U; 2]
64 185..419 '{ ...rr); }': ()
65 195..198 'arr': &[u8; 1]
66 211..215 '&[1]': &[u8; 1]
67 212..215 '[1]': [u8; 1]
70 236..239 'arr': &[u8; 1]
72 253..254 'f': fn f<u8>(&[u8]) -> u8
74 255..258 'arr': &[u8; 1]
76 279..286 '{ arr }': &[u8]
77 281..284 'arr': &[u8; 1]
79 300..301 'g': fn g<u8>(S<&[u8]>) -> u8
80 300..315 'g(S { a: arr })': u8
81 302..314 'S { a: arr }': S<&[u8]>
82 309..312 'arr': &[u8; 1]
83 325..326 'e': [&[u8]; 1]
84 340..345 '[arr]': [&[u8]; 1]
85 341..344 'arr': &[u8; 1]
86 355..356 'f': [&[u8]; 2]
87 370..378 '[arr; 2]': [&[u8]; 2]
88 371..374 'arr': &[u8; 1]
90 388..389 'g': (&[u8], &[u8])
91 406..416 '(arr, arr)': (&[u8], &[u8])
92 407..410 'arr': &[u8; 1]
93 412..415 'arr': &[u8; 1]
99 fn infer_let_stmt_coerce() {
103 let x: &[isize] = &[1];
104 let x: *const [isize] = &[1];
108 10..75 '{ ...[1]; }': ()
110 34..38 '&[1]': &[isize; 1]
111 35..38 '[1]': [isize; 1]
113 48..49 'x': *const [isize]
114 68..72 '&[1]': &[isize; 1]
115 69..72 '[1]': [isize; 1]
122 fn infer_custom_coerce_unsized() {
125 //- minicore: coerce_unsized
126 use core::{marker::Unsize, ops::CoerceUnsized};
128 struct A<T: ?Sized>(*const T);
129 struct B<T: ?Sized>(*const T);
130 struct C<T: ?Sized> { inner: *const T }
132 impl<T: ?Sized + Unsize<U>, U: ?Sized> CoerceUnsized<B<U>> for B<T> {}
133 impl<T: ?Sized + Unsize<U>, U: ?Sized> CoerceUnsized<C<U>> for C<T> {}
135 fn foo1<T>(x: A<[T]>) -> A<[T]> { x }
136 fn foo2<T>(x: B<[T]>) -> B<[T]> { x }
137 fn foo3<T>(x: C<[T]>) -> C<[T]> { x }
139 fn test(a: A<[u8; 2]>, b: B<[u8; 2]>, c: C<[u8; 2]>) {
147 327..332 '{ x }': A<[T]>
150 365..370 '{ x }': B<[T]>
153 403..408 '{ x }': C<[T]>
155 418..419 'a': A<[u8; 2]>
156 433..434 'b': B<[u8; 2]>
157 448..449 'c': C<[u8; 2]>
158 463..529 '{ ...(c); }': ()
159 473..474 'd': A<[{unknown}]>
160 477..481 'foo1': fn foo1<{unknown}>(A<[{unknown}]>) -> A<[{unknown}]>
161 477..484 'foo1(a)': A<[{unknown}]>
162 482..483 'a': A<[u8; 2]>
163 494..495 'e': B<[u8]>
164 498..502 'foo2': fn foo2<u8>(B<[u8]>) -> B<[u8]>
165 498..505 'foo2(b)': B<[u8]>
166 503..504 'b': B<[u8; 2]>
167 515..516 'f': C<[u8]>
168 519..523 'foo3': fn foo3<u8>(C<[u8]>) -> C<[u8]>
169 519..526 'foo3(c)': C<[u8]>
170 524..525 'c': C<[u8; 2]>
176 fn infer_if_coerce() {
180 fn foo<T>(x: &[T]) -> &[T] { loop {} }
191 27..38 '{ loop {} }': &[T]
194 49..125 '{ ... }; }': ()
196 63..122 'if tru... }': &[i32]
198 71..96 '{ ... }': &[i32]
199 81..84 'foo': fn foo<i32>(&[i32]) -> &[i32]
200 81..90 'foo(&[1])': &[i32]
201 85..89 '&[1]': &[i32; 1]
202 86..89 '[1]': [i32; 1]
204 102..122 '{ ... }': &[i32; 1]
205 112..116 '&[1]': &[i32; 1]
206 113..116 '[1]': [i32; 1]
213 fn infer_if_else_coerce() {
216 //- minicore: coerce_unsized
217 fn foo<T>(x: &[T]) -> &[T] { loop {} }
228 27..38 '{ loop {} }': &[T]
231 49..125 '{ ... }; }': ()
233 63..122 'if tru... }': &[i32]
235 71..91 '{ ... }': &[i32; 1]
236 81..85 '&[1]': &[i32; 1]
237 82..85 '[1]': [i32; 1]
239 97..122 '{ ... }': &[i32]
240 107..110 'foo': fn foo<i32>(&[i32]) -> &[i32]
241 107..116 'foo(&[1])': &[i32]
242 111..115 '&[1]': &[i32; 1]
243 112..115 '[1]': [i32; 1]
250 fn infer_match_first_coerce() {
254 fn foo<T>(x: &[T]) -> &[T] { loop {} }
265 27..38 '{ loop {} }': &[T]
269 55..149 '{ ... }; }': ()
271 69..146 'match ... }': &[i32]
275 92..95 'foo': fn foo<i32>(&[i32]) -> &[i32]
276 92..101 'foo(&[2])': &[i32]
277 96..100 '&[2]': &[i32; 1]
278 97..100 '[2]': [i32; 1]
282 116..120 '&[1]': &[i32; 1]
283 117..120 '[1]': [i32; 1]
286 135..139 '&[3]': &[i32; 1]
287 136..139 '[3]': [i32; 1]
294 fn infer_match_second_coerce() {
297 //- minicore: coerce_unsized
298 fn foo<T>(x: &[T]) -> &[T] { loop {} }
309 27..38 '{ loop {} }': &[T]
313 55..149 '{ ... }; }': ()
315 69..146 'match ... }': &[i32]
319 92..96 '&[1]': &[i32; 1]
320 93..96 '[1]': [i32; 1]
324 111..114 'foo': fn foo<i32>(&[i32]) -> &[i32]
325 111..120 'foo(&[2])': &[i32]
326 115..119 '&[2]': &[i32; 1]
327 116..119 '[2]': [i32; 1]
330 135..139 '&[3]': &[i32; 1]
331 136..139 '[3]': [i32; 1]
338 fn coerce_merge_one_by_one1() {
339 cov_mark::check!(coerce_merge_fail_fallback);
348 _ => t as *const i32,
353 10..144 '{ ... }; }': ()
355 24..30 '&mut 1': &mut i32
357 40..41 'x': *const i32
358 44..141 'match ... }': *const i32
363 67..80 't as *mut i32': *mut i32
367 95..104 't as &i32': &i32
369 119..120 't': &mut i32
370 119..134 't as *const i32': *const i32
376 fn return_coerce_unknown() {
377 check_infer_with_mismatches(
384 16..39 '{ ...own; }': u32
385 22..36 'return unknown': !
386 29..36 'unknown': u32
392 fn coerce_autoderef() {
393 check_infer_with_mismatches(
396 fn takes_ref_foo(x: &Foo) {}
399 takes_ref_foo(&&Foo);
400 takes_ref_foo(&&&Foo);
406 51..132 '{ ...oo); }': ()
407 57..70 'takes_ref_foo': fn takes_ref_foo(&Foo)
408 57..76 'takes_...(&Foo)': ()
411 82..95 'takes_ref_foo': fn takes_ref_foo(&Foo)
412 82..102 'takes_...&&Foo)': ()
413 96..101 '&&Foo': &&Foo
416 108..121 'takes_ref_foo': fn takes_ref_foo(&Foo)
417 108..129 'takes_...&&Foo)': ()
418 122..128 '&&&Foo': &&&Foo
419 123..128 '&&Foo': &&Foo
420 124..128 '&Foo': &Foo
427 fn coerce_autoderef_generic() {
428 check_infer_with_mismatches(
431 fn takes_ref<T>(x: &T) -> T { *x }
443 57..126 '{ ...oo); }': ()
444 63..72 'takes_ref': fn takes_ref<Foo>(&Foo) -> Foo
445 63..78 'takes_ref(&Foo)': Foo
448 84..93 'takes_ref': fn takes_ref<&Foo>(&&Foo) -> &Foo
449 84..100 'takes_...&&Foo)': &Foo
450 94..99 '&&Foo': &&Foo
453 106..115 'takes_ref': fn takes_ref<&&Foo>(&&&Foo) -> &&Foo
454 106..123 'takes_...&&Foo)': &&Foo
455 116..122 '&&&Foo': &&&Foo
456 117..122 '&&Foo': &&Foo
457 118..122 '&Foo': &Foo
464 fn coerce_autoderef_block() {
465 check_infer_with_mismatches(
469 trait Deref { type Target; }
470 impl Deref for String { type Target = str; }
471 fn takes_ref_str(x: &str) {}
472 fn returns_string() -> String { loop {} }
474 takes_ref_str(&{ returns_string() });
480 168..179 '{ loop {} }': String
481 170..177 'loop {}': !
483 190..235 '{ ... }); }': ()
484 196..209 'takes_ref_str': fn takes_ref_str(&str)
485 196..232 'takes_...g() })': ()
486 210..231 '&{ ret...ng() }': &String
487 211..231 '{ retu...ng() }': String
488 213..227 'returns_string': fn returns_string() -> String
489 213..229 'return...ring()': String
495 fn closure_return_coerce() {
496 check_infer_with_mismatches(
508 9..105 '{ ... }; }': ()
509 19..20 'x': || -> &u32
510 23..102 '|| { ... }': || -> &u32
511 26..102 '{ ... }': &u32
512 36..81 'if tru... }': ()
515 58..70 'return &1u32': !
518 90..96 '&&1u32': &&u32
526 fn coerce_fn_item_to_fn_ptr() {
527 check_infer_with_mismatches(
529 fn foo(x: u32) -> isize { 1 }
531 let f: fn(u32) -> isize = foo;
536 24..29 '{ 1 }': isize
538 40..78 '{ ...foo; }': ()
539 50..51 'f': fn(u32) -> isize
540 72..75 'foo': fn foo(u32) -> isize
546 fn coerce_fn_items_in_match_arms() {
547 cov_mark::check!(coerce_fn_reification);
549 check_infer_with_mismatches(
551 fn foo1(x: u32) -> isize { 1 }
552 fn foo2(x: u32) -> isize { 2 }
553 fn foo3(x: u32) -> isize { 3 }
564 25..30 '{ 1 }': isize
567 56..61 '{ 2 }': isize
570 87..92 '{ 3 }': isize
572 103..192 '{ ... }; }': ()
573 113..114 'x': fn(u32) -> isize
574 117..189 'match ... }': fn(u32) -> isize
578 140..144 'foo1': fn foo1(u32) -> isize
581 159..163 'foo2': fn foo2(u32) -> isize
583 178..182 'foo3': fn foo3(u32) -> isize
589 fn coerce_closure_to_fn_ptr() {
590 check_infer_with_mismatches(
593 let f: fn(u32) -> isize = |x| { 1 };
597 10..54 '{ ...1 }; }': ()
598 20..21 'f': fn(u32) -> isize
599 42..51 '|x| { 1 }': |u32| -> isize
601 46..51 '{ 1 }': isize
608 fn coerce_placeholder_ref() {
609 // placeholders should unify, even behind references
610 check_infer_with_mismatches(
614 fn get(&self) -> &TT {
620 50..54 'self': &S<TT>
621 63..86 '{ ... }': &TT
622 73..80 '&self.t': &TT
623 74..78 'self': &S<TT>
630 fn coerce_unsize_array() {
631 check_infer_with_mismatches(
633 //- minicore: coerce_unsized
635 let f: &[usize] = &[1, 2, 3];
639 10..47 '{ ... 3]; }': ()
641 34..44 '&[1, 2, 3]': &[usize; 3]
642 35..44 '[1, 2, 3]': [usize; 3]
651 fn coerce_unsize_trait_object_simple() {
652 check_infer_with_mismatches(
654 //- minicore: coerce_unsized
656 trait Bar<U, T, X>: Foo<T, U> {}
657 trait Baz<T, X>: Bar<usize, T, X> {}
660 impl<T, X> Foo<T, usize> for S<T, X> {}
661 impl<T, X> Bar<usize, T, X> for S<T, X> {}
662 impl<T, X> Baz<T, X> for S<T, X> {}
665 let obj: &dyn Baz<i8, i16> = &S;
666 let obj: &dyn Bar<_, i8, i16> = &S;
667 let obj: &dyn Foo<i8, _> = &S;
671 236..351 '{ ... &S; }': ()
672 246..249 'obj': &dyn Baz<i8, i16>
673 271..273 '&S': &S<i8, i16>
674 272..273 'S': S<i8, i16>
675 283..286 'obj': &dyn Bar<usize, i8, i16>
676 311..313 '&S': &S<i8, i16>
677 312..313 'S': S<i8, i16>
678 323..326 'obj': &dyn Foo<i8, usize>
679 346..348 '&S': &S<i8, {unknown}>
680 347..348 'S': S<i8, {unknown}>
686 fn coerce_unsize_trait_object_to_trait_object() {
687 // FIXME: The rust reference says this should be possible, but rustc doesn't
688 // implement it. We used to support it, but Chalk doesn't. Here's the
691 // 424..609 '{ ...bj2; }': ()
692 // 434..437 'obj': &dyn Baz<i8, i16>
693 // 459..461 '&S': &S<i8, i16>
694 // 460..461 'S': S<i8, i16>
695 // 471..474 'obj': &dyn Bar<usize, i8, i16>
696 // 496..499 'obj': &dyn Baz<i8, i16>
697 // 509..512 'obj': &dyn Foo<i8, usize>
698 // 531..534 'obj': &dyn Bar<usize, i8, i16>
699 // 544..548 'obj2': &dyn Baz<i8, i16>
700 // 570..572 '&S': &S<i8, i16>
701 // 571..572 'S': S<i8, i16>
702 // 582..583 '_': &dyn Foo<i8, usize>
703 // 602..606 'obj2': &dyn Baz<i8, i16>
704 check_infer_with_mismatches(
706 //- minicore: coerce_unsized
708 trait Bar<U, T, X>: Foo<T, U> {}
709 trait Baz<T, X>: Bar<usize, T, X> {}
712 impl<T, X> Foo<T, usize> for S<T, X> {}
713 impl<T, X> Bar<usize, T, X> for S<T, X> {}
714 impl<T, X> Baz<T, X> for S<T, X> {}
717 let obj: &dyn Baz<i8, i16> = &S;
718 let obj: &dyn Bar<_, _, _> = obj;
719 let obj: &dyn Foo<_, _> = obj;
720 let obj2: &dyn Baz<i8, i16> = &S;
721 let _: &dyn Foo<_, _> = obj2;
725 236..421 '{ ...bj2; }': ()
726 246..249 'obj': &dyn Baz<i8, i16>
727 271..273 '&S': &S<i8, i16>
728 272..273 'S': S<i8, i16>
729 283..286 'obj': &dyn Bar<{unknown}, {unknown}, {unknown}>
730 308..311 'obj': &dyn Baz<i8, i16>
731 321..324 'obj': &dyn Foo<{unknown}, {unknown}>
732 343..346 'obj': &dyn Bar<{unknown}, {unknown}, {unknown}>
733 356..360 'obj2': &dyn Baz<i8, i16>
734 382..384 '&S': &S<i8, i16>
735 383..384 'S': S<i8, i16>
736 394..395 '_': &dyn Foo<{unknown}, {unknown}>
737 414..418 'obj2': &dyn Baz<i8, i16>
738 308..311: expected &dyn Bar<{unknown}, {unknown}, {unknown}>, got &dyn Baz<i8, i16>
739 343..346: expected &dyn Foo<{unknown}, {unknown}>, got &dyn Bar<{unknown}, {unknown}, {unknown}>
740 414..418: expected &dyn Foo<{unknown}, {unknown}>, got &dyn Baz<i8, i16>
746 fn coerce_unsize_super_trait_cycle() {
747 check_infer_with_mismatches(
749 //- minicore: coerce_unsized
762 let obj: &dyn D = &S;
763 let obj: &dyn A = &S;
767 140..195 '{ ... &S; }': ()
768 150..153 'obj': &dyn D
771 176..179 'obj': &dyn A
779 fn coerce_unsize_generic() {
780 // FIXME: fix the type mismatches here
781 check_infer_with_mismatches(
783 //- minicore: coerce_unsized
784 struct Foo<T> { t: T };
785 struct Bar<T>(Foo<T>);
788 let _: &Foo<[usize]> = &Foo { t: [1, 2, 3] };
789 let _: &Bar<[usize]> = &Bar(Foo { t: [1, 2, 3] });
793 58..166 '{ ... }); }': ()
794 68..69 '_': &Foo<[usize]>
795 87..108 '&Foo {..., 3] }': &Foo<[usize]>
796 88..108 'Foo { ..., 3] }': Foo<[usize]>
797 97..106 '[1, 2, 3]': [usize; 3]
801 118..119 '_': &Bar<[usize]>
802 137..163 '&Bar(F... 3] })': &Bar<[i32; 3]>
803 138..141 'Bar': Bar<[i32; 3]>(Foo<[i32; 3]>) -> Bar<[i32; 3]>
804 138..163 'Bar(Fo... 3] })': Bar<[i32; 3]>
805 142..162 'Foo { ..., 3] }': Foo<[i32; 3]>
806 151..160 '[1, 2, 3]': [i32; 3]
810 97..106: expected [usize], got [usize; 3]
811 137..163: expected &Bar<[usize]>, got &Bar<[i32; 3]>
817 fn coerce_unsize_apit() {
819 check_infer_with_mismatches(
821 //- minicore: coerce_unsized
824 fn test(f: impl Foo) {
825 let _: &dyn Foo = &f;
830 35..64 '{ ... &f; }': ()
832 59..61 '&f': &impl Foo
834 59..61: expected &dyn Foo, got &impl Foo
840 fn infer_two_closures_lub() {
844 let add = |a: i32, b: i32| a + b;
845 let sub = |a, b| a - b;
846 //^ |i32, i32| -> i32
847 if c > 42 { add } else { sub };
848 //^ fn(i32, i32) -> i32
855 fn infer_match_diverging_branch_1() {
858 enum Result<T> { Ok(T), Err }
859 fn parse<T>() -> T { loop {} }
862 let a = match parse() {
874 fn infer_match_diverging_branch_2() {
875 // same as 1 except for order of branches
878 enum Result<T> { Ok(T), Err }
879 fn parse<T>() -> T { loop {} }
882 let a = match parse() {
899 pub macro panic_2015 {
901 $crate::panicking::panic()
907 pub fn panic() -> ! { loop {} }
910 #[rustc_builtin_macro = "core_panic"]
912 // Expands to either `$crate::panic::panic_2015` or `$crate::panic::panic_2021`
913 // depending on the edition of the caller.
915 /* compiler built-in */
927 fn coerce_unsize_expected_type() {
930 //- minicore: coerce_unsized
932 let foo: &[u32] = &[1, 2];
933 let foo: &[u32] = match true {
937 let foo: &[u32] = if true {