summaryrefslogtreecommitdiffstats
path: root/src/test/ui/associated-types/defaults-cyclic-pass-2.rs
blob: 69315a022100b89aa3ac4a377924ca4726737f8e (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
// check-pass

#![feature(associated_type_defaults)]

// Having a cycle in assoc. type defaults is okay, as long as there's no impl
// that retains it.
trait Tr {
    type A = Vec<Self::B>;
    type B = Box<Self::A>;

    fn f();
}

// An impl has to break the cycle to be accepted.
impl Tr for u8 {
    type A = u8;

    fn f() {
        // Check that the type propagates as expected (seen from inside the impl)
        let _: Self::A = 0u8;
        let _: Self::B = Box::new(0u8);
    }
}

impl Tr for String {
    type B = ();

    fn f() {
        // Check that the type propagates as expected (seen from inside the impl)
        let _: Self::A = Vec::<()>::new();
        let _: Self::B = ();
    }
}

impl Tr for () {
    type A = Vec<()>;
    type B = u8;

    fn f() {
        // Check that the type propagates as expected (seen from inside the impl)
        let _: Self::A = Vec::<()>::new();
        let _: Self::B = 0u8;
    }
}

fn main() {
    // Check that both impls now have the right types (seen from outside the impls)
    let _: <u8 as Tr>::A = 0u8;
    let _: <u8 as Tr>::B = Box::new(0u8);

    let _: <String as Tr>::A = Vec::<()>::new();
    let _: <String as Tr>::B = ();

    let _: <() as Tr>::A = Vec::<()>::new();
    let _: <() as Tr>::B = 0u8;
}