| // Test that we correctly infer variance for region parameters in |
| // various self-contained types. |
| |
| #![feature(rustc_attrs)] |
| |
| // Regions that just appear in normal spots are contravariant: |
| |
| #[rustc_variance] |
| struct Test2<'a, 'b, 'c> { //~ ERROR ['a: +, 'b: +, 'c: +] |
| x: &'a isize, |
| y: &'b [isize], |
| c: &'c str |
| } |
| |
| // Those same annotations in function arguments become covariant: |
| |
| #[rustc_variance] |
| struct Test3<'a, 'b, 'c> { //~ ERROR ['a: -, 'b: -, 'c: -] |
| x: extern "Rust" fn(&'a isize), |
| y: extern "Rust" fn(&'b [isize]), |
| c: extern "Rust" fn(&'c str), |
| } |
| |
| // Mutability induces invariance: |
| |
| #[rustc_variance] |
| struct Test4<'a, 'b:'a> { //~ ERROR ['a: +, 'b: o] |
| x: &'a mut &'b isize, |
| } |
| |
| // Mutability induces invariance, even when in a |
| // contravariant context: |
| |
| #[rustc_variance] |
| struct Test5<'a, 'b:'a> { //~ ERROR ['a: -, 'b: o] |
| x: extern "Rust" fn(&'a mut &'b isize), |
| } |
| |
| // Invariance is a trap from which NO ONE CAN ESCAPE. |
| // In other words, even though the `&'b isize` occurs in |
| // an argument list (which is contravariant), that |
| // argument list occurs in an invariant context. |
| |
| #[rustc_variance] |
| struct Test6<'a, 'b:'a> { //~ ERROR ['a: +, 'b: o] |
| x: &'a mut extern "Rust" fn(&'b isize), |
| } |
| |
| // No uses at all is bivariant: |
| |
| #[rustc_variance] |
| struct Test7<'a> { //~ ERROR ['a: *] |
| //~^ ERROR: `'a` is never used |
| x: isize |
| } |
| |
| // Try enums too. |
| |
| #[rustc_variance] |
| enum Test8<'a, 'b, 'c:'b> { //~ ERROR ['a: -, 'b: +, 'c: o] |
| Test8A(extern "Rust" fn(&'a isize)), |
| Test8B(&'b [isize]), |
| Test8C(&'b mut &'c str), |
| } |
| |
| fn main() {} |