labview-interop 0.4.2

Types and wrappers for interperating with LabVIEW when called as a library
Documentation
use std::fmt::Debug;
use std::marker::PhantomData;
use std::ops::{Deref, DerefMut};

use super::{LVCopy, UHandle};
use crate::errors::{InternalError, Result};
use crate::labview::memory_api;

/// A value allocated in the LabVIEW memory managed by Rust.
///
/// This will manage the lifetime and free the handle on drop.
///
/// This is a semantic difference from handle and is transparent with the handle data.
///
/// This means it can be used in structs in place of a handle.
///
/// # Example In Struct ([LStrOwned](crate::types::LStrOwned) is equivalent of [`OwnedUHandle<LStr>`]).
/// ```no_run
///# use labview_interop::labview_layout;
///# use labview_interop::types::LStrOwned;
///labview_layout!(
///pub struct UserEventCluster {
///    eventno: i32,
///    id: LStrOwned,
///}
///);
///```
/// # From a Handle
///
/// You can convert a handle to an owned value using the [`UHandle::try_to_owned`] method.
///
/// The type must be [`LVCopy`] to ensure it is safe to copy the internal data.
///
/// ## Example
/// ```no_run
/// use labview_interop::types::{LStrHandle, LStrOwned};
///
/// fn handle_to_owned(handle: LStrHandle) -> LStrOwned {
///    handle.try_to_owned().unwrap()
/// }
/// ```
///
/// # Clone
///
/// Clone is implemented but can panic if there is not enough memory to create the new handle.
///
/// There is also a [`OwnedUHandle::try_clone`] method which will return a Result.
///
/// The type must be [`LVCopy`] to ensure it is safe to clone.
///
///
#[repr(transparent)]
pub struct OwnedUHandle<T: ?Sized + 'static>(UHandle<'static, T>);

impl<T: Copy + 'static> OwnedUHandle<T> {
    /// Create a new handle to a sized value of `T`.
    ///
    /// It will copy the data from the provided value.
    pub fn new(value: &T) -> Result<Self> {
        let handle = unsafe { memory_api()?.new_handle(std::mem::size_of::<T>()) } as *mut *mut T;

        if handle.is_null() {
            Err(InternalError::HandleCreationFailed.into())
        } else {
            // Copy the value into the handle.
            // # Safety - these pointers have just been created by the memory manager and we checked null.
            unsafe {
                **handle = *value;
            }
            Ok(Self(UHandle(handle, PhantomData)))
        }
    }
}

impl<T: ?Sized> OwnedUHandle<T> {
    /// Create a new handle to the type `T`. It will create an empty handle
    /// which you must initialise with the `init_routine`.
    /// This is useful for unsized types.
    ///
    /// # Safety
    ///
    /// * This will create a handle to un-initialized memory. The provided initialisation
    ///   routine must prepare the memory.
    pub(crate) unsafe fn new_unsized(
        init_routine: impl FnOnce(&mut UHandle<'static, T>) -> Result<()>,
    ) -> Result<Self> {
        let handle = memory_api()?.new_handle(0);
        if handle.is_null() {
            Err(InternalError::HandleCreationFailed.into())
        } else {
            let mut new_value = UHandle(handle as *mut *mut T, PhantomData);
            init_routine(&mut new_value)?;
            Ok(Self(new_value))
        }
    }

    /// Return a new handle to the inner value.
    ///
    /// This takes a mutable borrow on the owned value as you can use the handle
    /// to modify the inner value.
    ///
    /// Note: This is only if you need a true handle to put into a structure that is expecting this.
    /// Better options are :
    /// * If you can define the type, just define it with the owned value. An owned value can take the place of a handle.
    /// * If you just need access to the data then use the Deref methods to access the handle.
    ///
    /// # Safety
    ///
    /// * This needs to take a mutable reference to self and lifetime annotation on UHandle,
    ///   to avoid creating multiple UHandles.
    ///
    /// # Examples
    ///
    /// ## Use Handle in Struct
    ///
    /// ```no_run
    /// use labview_interop::types::{LStrHandle, LStrOwned};
    /// use labview_interop::labview_layout;
    ///
    /// // This must have a handle due to other uses.
    /// labview_layout! {
    ///   struct ClusterWithString<'a> {
    ///     string_handle: LStrHandle<'a>
    ///   }
    /// }
    ///
    /// // Mutable is required since once you have a handle you can mutate the data.
    /// let mut owned_string = LStrOwned::from_data(b"Hello World!").unwrap();
    /// let handle = owned_string.handle_to_inner();
    /// let cluster = ClusterWithString {
    ///   string_handle: handle
    /// };
    /// // Do something with the cluster.
    ///
    /// ```
    ///
    /// ## Lifetime Guarantees - Single Handle
    /// ```compile_fail,E0515
    /// use labview_interop::memory::LvOwned;
    ///
    /// let mut owned = LvOwned::<f64>::new().unwrap();
    /// let mut handle = owned.handle_to_inner();
    /// // Cannot get a second handle due to lifetime.
    /// // This fails to compile.
    /// let handle2 = owned.handle_to_inner();
    ///
    /// *handle = 1.0;
    ///
    /// ```
    ///
    /// ## Lifetime Guarantees - Owned Outlives Handle
    ///
    /// ```compile_fail,E0515
    /// use labview_interop::memory::LvOwned;
    ///
    /// let mut owned = LvOwned::<f64>::new().unwrap();
    /// let mut handle = owned.handle_to_inner();
    /// // Cannot drop owned because we have a handle.
    /// // This fails to compile.
    /// drop(owned);
    ///
    /// *handle = 1.0;
    ///
    /// ```
    pub fn handle_to_inner(&mut self) -> UHandle<'_, T> {
        UHandle(self.0 .0, PhantomData)
    }
}

impl<T: ?Sized> Deref for OwnedUHandle<T> {
    type Target = UHandle<'static, T>;

    fn deref(&self) -> &Self::Target {
        &self.0
    }
}

impl<T: ?Sized> DerefMut for OwnedUHandle<T> {
    fn deref_mut(&mut self) -> &mut Self::Target {
        &mut self.0
    }
}

impl<T: ?Sized> Drop for OwnedUHandle<T> {
    fn drop(&mut self) {
        let result = memory_api().map(|api| unsafe {
            api.dispose_handle(self.0 .0 as usize)
                .to_specific_result(())
        });
        if let Err(e) | Ok(Err(e)) = result {
            println!("Error freeing handle from LV: {e}");
        }
    }
}

impl<T: Debug + ?Sized> Debug for OwnedUHandle<T> {
    fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result {
        super::fmt_handle("LvOwned", &self.0, f)
    }
}

impl<T: ?Sized + LVCopy + 'static> OwnedUHandle<T> {
    /// Try to clone the handle.
    ///
    /// This will create a new handle to the same data.
    ///
    /// # Errors
    ///
    /// * If there is not enough memory to create the handle this may error.
    pub fn try_clone(&self) -> Result<Self> {
        // Safety - The handle should be valid because it is an owned handle.
        // Safety - The initialisation function will initialise the data or error.
        unsafe {
            OwnedUHandle::new_unsized(|handle| {
                self.clone_into_pointer(handle as *mut UHandle<'static, T>)
            })
        }
    }
}

impl<T: ?Sized + LVCopy + 'static> UHandle<'_, T> {
    /// Try to create an owned handle from the current handle.
    ///
    /// The owned handle will have its own handle to the data and
    /// will be responsible for freeing it.
    ///
    /// # Errors
    ///
    /// * If there is not enough memory to create the handle this may error.
    /// * If the source handle is not valid this will error.
    pub fn try_to_owned(&self) -> Result<OwnedUHandle<T>> {
        // Safety - The clone_into_pointer will check the handle is valid.
        // Safety - The initialisation function will initialise the data or error.
        unsafe {
            OwnedUHandle::new_unsized(|handle| {
                self.clone_into_pointer(handle as *mut UHandle<'static, T>)
            })
        }
    }
}

impl<T: ?Sized + LVCopy + 'static> Clone for OwnedUHandle<T> {
    fn clone(&self) -> Self {
        self.try_clone().unwrap()
    }
}

/// # Safety
///
/// * LvOwned memory is access through UHandle which is managed by the Labview Memory Manager, which is thread safe
unsafe impl<T: ?Sized> Send for OwnedUHandle<T> {}
unsafe impl<T: ?Sized> Sync for OwnedUHandle<T> {}

#[cfg(test)]
mod tests {
    use super::*;

    #[test]
    fn test_lvowned_debug() {
        let mut value = 42;
        let mut value_ptr = std::ptr::addr_of_mut!(value);
        let handle = UHandle(std::ptr::addr_of_mut!(value_ptr), std::marker::PhantomData);
        let owned = OwnedUHandle(handle);
        assert_eq!(format!("{:?}", owned), "LvOwned(42)");
    }
}