2015-08-12 11:16:09 +00:00
|
|
|
#![feature(plugin)]
|
|
|
|
#![plugin(clippy)]
|
|
|
|
|
2016-01-14 18:27:24 +00:00
|
|
|
#![deny(needless_lifetimes, unused_lifetimes)]
|
2017-02-20 03:50:31 +00:00
|
|
|
#![allow(dead_code, needless_pass_by_value)]
|
2016-01-29 21:18:14 +00:00
|
|
|
|
2015-08-13 05:51:24 +00:00
|
|
|
fn distinct_lifetimes<'a, 'b>(_x: &'a u8, _y: &'b u8, _z: u8) { }
|
2017-02-08 13:58:07 +00:00
|
|
|
|
2015-08-12 11:16:09 +00:00
|
|
|
|
2015-08-13 05:51:24 +00:00
|
|
|
fn distinct_and_static<'a, 'b>(_x: &'a u8, _y: &'b u8, _z: &'static u8) { }
|
2017-02-08 13:58:07 +00:00
|
|
|
|
2015-08-12 11:16:09 +00:00
|
|
|
|
|
|
|
fn same_lifetime_on_input<'a>(_x: &'a u8, _y: &'a u8) { } // no error, same lifetime on two params
|
|
|
|
|
|
|
|
fn only_static_on_input(_x: &u8, _y: &u8, _z: &'static u8) { } // no error, static involved
|
|
|
|
|
2015-08-15 07:36:02 +00:00
|
|
|
fn mut_and_static_input(_x: &mut u8, _y: &'static str) { }
|
|
|
|
|
2015-08-13 05:51:24 +00:00
|
|
|
fn in_and_out<'a>(x: &'a u8, _y: u8) -> &'a u8 { x }
|
2017-02-08 13:58:07 +00:00
|
|
|
|
2015-08-12 11:16:09 +00:00
|
|
|
|
|
|
|
fn multiple_in_and_out_1<'a>(x: &'a u8, _y: &'a u8) -> &'a u8 { x } // no error, multiple input refs
|
|
|
|
|
|
|
|
fn multiple_in_and_out_2<'a, 'b>(x: &'a u8, _y: &'b u8) -> &'a u8 { x } // no error, multiple input refs
|
|
|
|
|
|
|
|
fn in_static_and_out<'a>(x: &'a u8, _y: &'static u8) -> &'a u8 { x } // no error, static involved
|
|
|
|
|
|
|
|
fn deep_reference_1<'a, 'b>(x: &'a u8, _y: &'b u8) -> Result<&'a u8, ()> { Ok(x) } // no error
|
|
|
|
|
|
|
|
fn deep_reference_2<'a>(x: Result<&'a u8, &'a u8>) -> &'a u8 { x.unwrap() } // no error, two input refs
|
|
|
|
|
2015-08-13 05:51:24 +00:00
|
|
|
fn deep_reference_3<'a>(x: &'a u8, _y: u8) -> Result<&'a u8, ()> { Ok(x) }
|
2017-02-08 13:58:07 +00:00
|
|
|
|
2015-08-12 11:16:09 +00:00
|
|
|
|
2015-08-30 07:57:52 +00:00
|
|
|
// where clause, but without lifetimes
|
|
|
|
fn where_clause_without_lt<'a, T>(x: &'a u8, _y: u8) -> Result<&'a u8, ()> where T: Copy { Ok(x) }
|
2017-02-08 13:58:07 +00:00
|
|
|
|
2015-08-30 07:57:52 +00:00
|
|
|
|
2015-08-13 04:43:25 +00:00
|
|
|
type Ref<'r> = &'r u8;
|
|
|
|
|
2015-08-13 15:24:47 +00:00
|
|
|
fn lifetime_param_1<'a>(_x: Ref<'a>, _y: &'a u8) { } // no error, same lifetime on two params
|
2015-08-13 04:43:25 +00:00
|
|
|
|
2015-08-13 15:24:47 +00:00
|
|
|
fn lifetime_param_2<'a, 'b>(_x: Ref<'a>, _y: &'b u8) { }
|
2017-02-08 13:58:07 +00:00
|
|
|
|
2015-08-13 04:43:25 +00:00
|
|
|
|
2015-08-13 15:24:47 +00:00
|
|
|
fn lifetime_param_3<'a, 'b: 'a>(_x: Ref<'a>, _y: &'b u8) { } // no error, bounded lifetime
|
|
|
|
|
2015-08-30 07:57:52 +00:00
|
|
|
fn lifetime_param_4<'a, 'b>(_x: Ref<'a>, _y: &'b u8) where 'b: 'a { } // no error, bounded lifetime
|
|
|
|
|
2015-08-31 09:11:51 +00:00
|
|
|
struct Lt<'a, I: 'static> {
|
|
|
|
x: &'a I
|
|
|
|
}
|
|
|
|
|
|
|
|
fn fn_bound<'a, F, I>(_m: Lt<'a, I>, _f: F) -> Lt<'a, I>
|
|
|
|
where F: Fn(Lt<'a, I>) -> Lt<'a, I> // no error, fn bound references 'a
|
|
|
|
{ unreachable!() }
|
|
|
|
|
2017-02-08 13:58:07 +00:00
|
|
|
fn fn_bound_2<'a, F, I>(_m: Lt<'a, I>, _f: F) -> Lt<'a, I>
|
2015-08-31 09:11:51 +00:00
|
|
|
where for<'x> F: Fn(Lt<'x, I>) -> Lt<'x, I>
|
|
|
|
{ unreachable!() }
|
|
|
|
|
2015-08-12 11:16:09 +00:00
|
|
|
struct X {
|
|
|
|
x: u8,
|
|
|
|
}
|
|
|
|
|
|
|
|
impl X {
|
2015-08-13 05:51:24 +00:00
|
|
|
fn self_and_out<'s>(&'s self) -> &'s u8 { &self.x }
|
2017-02-08 13:58:07 +00:00
|
|
|
|
2015-08-12 11:16:09 +00:00
|
|
|
|
|
|
|
fn self_and_in_out<'s, 't>(&'s self, _x: &'t u8) -> &'s u8 { &self.x } // no error, multiple input refs
|
|
|
|
|
2015-08-13 05:51:24 +00:00
|
|
|
fn distinct_self_and_in<'s, 't>(&'s self, _x: &'t u8) { }
|
2017-02-08 13:58:07 +00:00
|
|
|
|
2015-08-12 11:16:09 +00:00
|
|
|
|
|
|
|
fn self_and_same_in<'s>(&'s self, _x: &'s u8) { } // no error, same lifetimes on two params
|
|
|
|
}
|
|
|
|
|
2015-08-13 15:40:45 +00:00
|
|
|
struct Foo<'a>(&'a u8);
|
|
|
|
|
|
|
|
impl<'a> Foo<'a> {
|
|
|
|
fn self_shared_lifetime(&self, _: &'a u8) {} // no error, lifetime 'a not defined in method
|
|
|
|
fn self_bound_lifetime<'b: 'a>(&self, _: &'b u8) {} // no error, bounds exist
|
|
|
|
}
|
2015-08-12 11:16:09 +00:00
|
|
|
|
2015-09-30 14:40:54 +00:00
|
|
|
fn already_elided<'a>(_: &u8, _: &'a u8) -> &'a u8 {
|
|
|
|
unimplemented!()
|
|
|
|
}
|
|
|
|
|
2017-02-08 13:58:07 +00:00
|
|
|
fn struct_with_lt<'a>(_foo: Foo<'a>) -> &'a str { unimplemented!() }
|
2015-11-10 23:26:22 +00:00
|
|
|
|
|
|
|
// no warning, two input lifetimes (named on the reference, anonymous on Foo)
|
|
|
|
fn struct_with_lt2<'a>(_foo: &'a Foo) -> &'a str { unimplemented!() }
|
|
|
|
|
|
|
|
// no warning, two input lifetimes (anonymous on the reference, named on Foo)
|
|
|
|
fn struct_with_lt3<'a>(_foo: &Foo<'a> ) -> &'a str { unimplemented!() }
|
|
|
|
|
|
|
|
// no warning, two input lifetimes
|
|
|
|
fn struct_with_lt4<'a, 'b>(_foo: &'a Foo<'b> ) -> &'a str { unimplemented!() }
|
|
|
|
|
|
|
|
trait WithLifetime<'a> {}
|
2016-01-29 21:18:14 +00:00
|
|
|
|
2015-11-10 23:26:22 +00:00
|
|
|
type WithLifetimeAlias<'a> = WithLifetime<'a>;
|
|
|
|
|
|
|
|
// should not warn because it won't build without the lifetime
|
|
|
|
fn trait_obj_elided<'a>(_arg: &'a WithLifetime) -> &'a str { unimplemented!() }
|
|
|
|
|
|
|
|
// this should warn because there is no lifetime on Drop, so this would be
|
|
|
|
// unambiguous if we elided the lifetime
|
2017-02-08 13:58:07 +00:00
|
|
|
fn trait_obj_elided2<'a>(_arg: &'a Drop) -> &'a str { unimplemented!() }
|
2015-11-10 23:26:22 +00:00
|
|
|
|
2015-12-04 14:56:25 +00:00
|
|
|
type FooAlias<'a> = Foo<'a>;
|
|
|
|
|
2017-02-08 13:58:07 +00:00
|
|
|
fn alias_with_lt<'a>(_foo: FooAlias<'a>) -> &'a str { unimplemented!() }
|
2015-12-04 14:56:25 +00:00
|
|
|
|
|
|
|
// no warning, two input lifetimes (named on the reference, anonymous on Foo)
|
|
|
|
fn alias_with_lt2<'a>(_foo: &'a FooAlias) -> &'a str { unimplemented!() }
|
|
|
|
|
|
|
|
// no warning, two input lifetimes (anonymous on the reference, named on Foo)
|
|
|
|
fn alias_with_lt3<'a>(_foo: &FooAlias<'a> ) -> &'a str { unimplemented!() }
|
|
|
|
|
|
|
|
// no warning, two input lifetimes
|
|
|
|
fn alias_with_lt4<'a, 'b>(_foo: &'a FooAlias<'b> ) -> &'a str { unimplemented!() }
|
|
|
|
|
2017-02-08 13:58:07 +00:00
|
|
|
fn named_input_elided_output<'a>(_arg: &'a str) -> &str { unimplemented!() }
|
2016-01-17 16:53:41 +00:00
|
|
|
|
|
|
|
fn elided_input_named_output<'a>(_arg: &str) -> &'a str { unimplemented!() }
|
|
|
|
|
2017-02-08 13:58:07 +00:00
|
|
|
fn trait_bound_ok<'a, T: WithLifetime<'static>>(_: &'a u8, _: T) { unimplemented!() }
|
2016-01-29 21:18:14 +00:00
|
|
|
fn trait_bound<'a, T: WithLifetime<'a>>(_: &'a u8, _: T) { unimplemented!() }
|
|
|
|
|
2015-08-12 11:16:09 +00:00
|
|
|
fn main() {
|
|
|
|
}
|