| David Tolnay | 5515a9e | 2020-11-25 19:07:54 -0800 | [diff] [blame] | 1 | use core::mem; |
| David Tolnay | 9bffb93 | 2021-01-02 02:15:21 -0800 | [diff] [blame] | 2 | use core::ptr::{self, NonNull}; |
| David Tolnay | 5515a9e | 2020-11-25 19:07:54 -0800 | [diff] [blame] | 3 | use core::slice; |
| 4 | |
| David Tolnay | 5515a9e | 2020-11-25 19:07:54 -0800 | [diff] [blame] | 5 | #[repr(C)] |
| David Tolnay | 5515a9e | 2020-11-25 19:07:54 -0800 | [diff] [blame] | 6 | pub struct RustSlice { |
| David Tolnay | 5577e7c | 2021-04-10 11:36:45 -0700 | [diff] [blame] | 7 | repr: NonNull<[()]>, |
| David Tolnay | 5515a9e | 2020-11-25 19:07:54 -0800 | [diff] [blame] | 8 | } |
| 9 | |
| 10 | impl RustSlice { |
| David Tolnay | 9bffb93 | 2021-01-02 02:15:21 -0800 | [diff] [blame] | 11 | pub fn from_ref<T>(slice: &[T]) -> Self { |
| David Tolnay | 5577e7c | 2021-04-10 11:36:45 -0700 | [diff] [blame] | 12 | let ptr = NonNull::from(slice).cast::<T>(); |
| 13 | let len = slice.len(); |
| 14 | Self::from_raw_parts(ptr, len) |
| David Tolnay | 5515a9e | 2020-11-25 19:07:54 -0800 | [diff] [blame] | 15 | } |
| 16 | |
| David Tolnay | 9bffb93 | 2021-01-02 02:15:21 -0800 | [diff] [blame] | 17 | pub fn from_mut<T>(slice: &mut [T]) -> Self { |
| David Tolnay | 5577e7c | 2021-04-10 11:36:45 -0700 | [diff] [blame] | 18 | let ptr = NonNull::from(&mut *slice).cast::<T>(); |
| 19 | let len = slice.len(); |
| 20 | Self::from_raw_parts(ptr, len) |
| 21 | } |
| 22 | |
| 23 | pub unsafe fn as_slice<'a, T>(self) -> &'a [T] { |
| 24 | let ptr = self.as_ptr(); |
| 25 | let len = self.len(); |
| 26 | slice::from_raw_parts(ptr.as_ptr(), len) |
| 27 | } |
| 28 | |
| 29 | pub unsafe fn as_mut_slice<'a, T>(self) -> &'a mut [T] { |
| 30 | let ptr = self.as_ptr(); |
| 31 | let len = self.len(); |
| 32 | slice::from_raw_parts_mut(ptr.as_ptr(), len) |
| 33 | } |
| 34 | |
| 35 | pub(crate) fn from_raw_parts<T>(ptr: NonNull<T>, len: usize) -> Self { |
| 36 | // TODO: use NonNull::from_raw_parts(ptr.cast(), len) when stable. |
| 37 | // https://doc.rust-lang.org/nightly/std/ptr/struct.NonNull.html#method.from_raw_parts |
| 38 | // https://github.com/rust-lang/rust/issues/81513 |
| 39 | let ptr = ptr::slice_from_raw_parts_mut(ptr.as_ptr().cast(), len); |
| David Tolnay | 5515a9e | 2020-11-25 19:07:54 -0800 | [diff] [blame] | 40 | RustSlice { |
| David Tolnay | 9bffb93 | 2021-01-02 02:15:21 -0800 | [diff] [blame] | 41 | repr: unsafe { NonNull::new_unchecked(ptr) }, |
| David Tolnay | 5515a9e | 2020-11-25 19:07:54 -0800 | [diff] [blame] | 42 | } |
| 43 | } |
| 44 | |
| David Tolnay | 5577e7c | 2021-04-10 11:36:45 -0700 | [diff] [blame] | 45 | pub(crate) fn as_ptr<T>(&self) -> NonNull<T> { |
| 46 | self.repr.cast() |
| David Tolnay | 5515a9e | 2020-11-25 19:07:54 -0800 | [diff] [blame] | 47 | } |
| 48 | |
| David Tolnay | 5577e7c | 2021-04-10 11:36:45 -0700 | [diff] [blame] | 49 | pub(crate) fn len(&self) -> usize { |
| 50 | // TODO: use self.repr.len() when stable. |
| 51 | // https://doc.rust-lang.org/nightly/std/ptr/struct.NonNull.html#method.len |
| 52 | // https://github.com/rust-lang/rust/issues/71146 |
| 53 | let slice = unsafe { self.repr.as_ref() }; |
| 54 | slice.len() |
| David Tolnay | 5515a9e | 2020-11-25 19:07:54 -0800 | [diff] [blame] | 55 | } |
| 56 | } |
| 57 | |
| 58 | const_assert_eq!( |
| 59 | mem::size_of::<Option<RustSlice>>(), |
| 60 | mem::size_of::<RustSlice>(), |
| 61 | ); |