#![no_std] #![warn(unsafe_code)] #![warn(rust_2018_idioms, single_use_lifetimes)] #![allow(dead_code)] use core::{marker::PhantomPinned, pin::Pin}; use pin_project_lite::pin_project; #[test] fn test_pin_project() { pin_project! { struct Foo { #[pin] field1: T, field2: U, } } let mut foo = Foo { field1: 1, field2: 2 }; let mut foo_orig = Pin::new(&mut foo); let foo = foo_orig.as_mut().project(); let x: Pin<&mut i32> = foo.field1; assert_eq!(*x, 1); let y: &mut i32 = foo.field2; assert_eq!(*y, 2); assert_eq!(foo_orig.as_ref().field1, 1); assert_eq!(foo_orig.as_ref().field2, 2); let mut foo = Foo { field1: 1, field2: 2 }; let foo = Pin::new(&mut foo).project(); let field1 = foo.field1; let field2 = foo.field2; let _: Pin<&mut i32> = field1; let _: &mut i32 = field2; } #[test] fn where_clause_and_associated_type_fields() { pin_project! { struct Struct1 where I: Iterator, { #[pin] field1: I, field2: I::Item, } } pin_project! { struct Struct2 where I: Iterator, { #[pin] field1: I, field2: J, } } // TODO(#7): where clause does not support yet. // pin_project! { // pub struct Struct3 // where // T: 'static, // { // field: T, // } // } // trait Static: 'static {} // impl Static for Struct3 {} } // #[allow(explicit_outlives_requirements)] // https://github.com/rust-lang/rust/issues/60993 // #[test] // fn unsized_in_where_clause() { // pin_project! { // struct Struct // where // I: ?Sized, // { // #[pin] // field: I, // } // } // } #[test] fn derive_copy() { pin_project! { #[derive(Clone, Copy)] struct Struct { val: T, } } fn is_copy() {} is_copy::>(); } #[test] fn move_out() { struct NotCopy; pin_project! { struct Struct { val: NotCopy, } } let foo = Struct { val: NotCopy }; let _val: NotCopy = foo.val; } #[test] fn trait_bounds_on_type_generics() { pin_project! { pub struct Struct1<'a, T: ?Sized> { field: &'a mut T, } } pin_project! { pub struct Struct2<'a, T: ::core::fmt::Debug> { field: &'a mut T, } } pin_project! { pub struct Struct3<'a, T: core::fmt::Debug> { field: &'a mut T, } } // pin_project! { // pub struct Struct4<'a, T: core::fmt::Debug + core::fmt::Display> { // field: &'a mut T, // } // } // pin_project! { // pub struct Struct5<'a, T: core::fmt::Debug + ?Sized> { // field: &'a mut T, // } // } pin_project! { pub struct Struct6<'a, T: core::fmt::Debug = [u8; 16]> { field: &'a mut T, } } let _: Struct6<'_> = Struct6 { field: &mut [0u8; 16] }; pin_project! { pub struct Struct7 { field: T, } } trait Static: 'static {} impl Static for Struct7 {} pin_project! { pub struct Struct8<'a, 'b: 'a> { field1: &'a u8, field2: &'b u8, } } } #[test] fn private_type_in_public_type() { pin_project! { pub struct PublicStruct { #[pin] inner: PrivateStruct, } } struct PrivateStruct(T); } #[test] fn lifetime_project() { pin_project! { struct Struct1 { #[pin] pinned: T, unpinned: U, } } pin_project! { struct Struct2<'a, T, U> { #[pin] pinned: &'a mut T, unpinned: U, } } impl Struct1 { fn get_pin_ref<'a>(self: Pin<&'a Self>) -> Pin<&'a T> { self.project_ref().pinned } fn get_pin_mut<'a>(self: Pin<&'a mut Self>) -> Pin<&'a mut T> { self.project().pinned } } impl<'b, T, U> Struct2<'b, T, U> { fn get_pin_ref<'a>(self: Pin<&'a Self>) -> Pin<&'a &'b mut T> { self.project_ref().pinned } fn get_pin_mut<'a>(self: Pin<&'a mut Self>) -> Pin<&'a mut &'b mut T> { self.project().pinned } } } #[test] fn lifetime_project_elided() { pin_project! { struct Struct1 { #[pin] pinned: T, unpinned: U, } } pin_project! { struct Struct2<'a, T, U> { #[pin] pinned: &'a mut T, unpinned: U, } } impl Struct1 { fn get_pin_ref(self: Pin<&Self>) -> Pin<&T> { self.project_ref().pinned } fn get_pin_mut(self: Pin<&mut Self>) -> Pin<&mut T> { self.project().pinned } } impl<'b, T, U> Struct2<'b, T, U> { fn get_pin_ref(self: Pin<&Self>) -> Pin<&&'b mut T> { self.project_ref().pinned } fn get_pin_mut(self: Pin<&mut Self>) -> Pin<&mut &'b mut T> { self.project().pinned } } } mod visibility { use pin_project_lite::pin_project; pin_project! { pub(crate) struct A { pub b: u8, } } } #[test] fn visibility() { let mut x = visibility::A { b: 0 }; let x = Pin::new(&mut x); let y = x.as_ref().project_ref(); let _: &u8 = y.b; let y = x.project(); let _: &mut u8 = y.b; } #[test] fn trivial_bounds() { pin_project! { pub struct NoGenerics { #[pin] field: PhantomPinned, } } } #[test] fn dst() { pin_project! { pub struct A { x: T, } } let _: &mut A = &mut A { x: 0u8 } as _; pin_project! { pub struct B { #[pin] x: T, } } } #[test] fn dyn_type() { pin_project! { struct Struct1 { a: i32, f: dyn core::fmt::Debug, } } pin_project! { struct Struct2 { a: i32, #[pin] f: dyn core::fmt::Debug, } } pin_project! { struct Struct3 { a: i32, f: dyn core::fmt::Debug + Send, } } pin_project! { struct Struct4 { a: i32, #[pin] f: dyn core::fmt::Debug + Send, } } }