diff --git a/crates/cust/src/memory/device/device_slice.rs b/crates/cust/src/memory/device/device_slice.rs index e460c8c2..32d90d93 100644 --- a/crates/cust/src/memory/device/device_slice.rs +++ b/crates/cust/src/memory/device/device_slice.rs @@ -556,6 +556,79 @@ impl DeviceSlice { } } +impl DeviceSlice { + /// Thw Asynchronously should now copy data from the `source` into this slice and then synchronizes the stream. + /// + /// This will be a safe wrapper around the unsafe asynchronous copy function. It will first launch the + /// copy on the provided stream and then synchronizes the stream so that the copy is complete + /// before returning it. This eliminates the need for the caller to deal with unsafe code and + /// explicit synchronization. + /// + /// # Panics + /// + /// Panics will occur if the length of `source.as_ref()` does not match this slice’s length. + /// + /// # So Example will be + /// + /// ```rust + /// # let _context = cust::quick_init().unwrap(); + /// use cust::{memory::*, stream::Stream, stream::StreamFlags}; + /// let stream = Stream::new(StreamFlags::NON_BLOCKING, None).unwrap(); + /// let start = [0u64, 1, 2, 3, 4, 5]; + /// let mut buf = DeviceBuffer::from_slice(&[0u64; 6]).unwrap(); + /// // Now this will safely perform the async copy and then wait for it to complete: + /// buf.as_slice().copy_from_async_sync(&start, &stream).unwrap(); + /// let mut host = [0u64; 6]; + /// buf.copy_to(&mut host).unwrap(); + /// assert_eq!(start, host); + /// ``` + pub fn copy_from_async_sync(&mut self, source: &I, stream: &Stream) -> crate::error::CudaResult<()> + where + I: AsRef<[T]> + AsMut<[T]> + ?Sized, + { + //I Call the unsafe async copy and then synchronize the stream. + unsafe { + >::async_copy_from(self, source, stream)? + }; + stream.synchronize()?; + Ok(()) + } + + /// This Asynchronously copies data from this slice into `dest` and then synchronizes the stream. + /// + /// This is a safe wrapper around the unsafe asynchronous copy function i propose. this launches the copy on + /// the provided stream and then synchronizes, ensuring that the copy is complete before returning without causing error or panics. + /// + /// # Panics + /// + /// Thus Panics if the length of `dest.as_mut()` does not match this slice’s length. + /// + /// # Example + /// + /// ```rust + /// # let _context = cust::quick_init().unwrap(); + /// use cust::{memory::*, stream::Stream, stream::StreamFlags}; + /// let stream = Stream::new(StreamFlags::NON_BLOCKING, None).unwrap(); + /// let start = [0u64, 1, 2, 3, 4, 5]; + /// let buf = DeviceBuffer::from_slice(&start).unwrap(); + /// let mut host = [0u64; 6]; + /// buf.as_slice().copy_to_async_sync(&mut host, &stream).unwrap(); + /// assert_eq!(start, host); + /// ``` + pub fn copy_to_async_sync(&self, dest: &mut I, stream: &Stream) -> crate::error::CudaResult<()> + where + I: AsRef<[T]> + AsMut<[T]> + ?Sized, + { + unsafe { + >::async_copy_to(self, dest, stream)? + }; + stream.synchronize()?; + Ok(()) + } +} + + + impl crate::private::Sealed for DeviceSlice {} impl + AsMut<[T]> + ?Sized> CopyDestination for DeviceSlice { fn copy_from(&mut self, val: &I) -> CudaResult<()> {