using System;
using System.Diagnostics;
using System.Runtime.InteropServices;
using Unity.Collections.LowLevel.Unsafe;
namespace Unity.Burst
{
///
/// Base interface for a function pointer.
///
public interface IFunctionPointer
{
///
/// Converts a pointer to a function pointer.
///
/// The native pointer.
/// An instance of this interface.
[Obsolete("This method will be removed in a future version of Burst")]
IFunctionPointer FromIntPtr(IntPtr ptr);
}
///
/// A function pointer that can be used from a Burst Job or from regular C#.
/// It needs to be compiled through
///
/// Type of the delegate of this function pointer
public readonly struct FunctionPointer : IFunctionPointer
{
// DOTSPLAYER's shim package relies on Burst for it's jobs code
// so Burst does not see the DOTSPLAYER definition of this attribute
[NativeDisableUnsafePtrRestriction]
private readonly IntPtr _ptr;
///
/// Creates a new instance of this function pointer with the following native pointer.
///
/// Native Pointer
public FunctionPointer(IntPtr ptr)
{
_ptr = ptr;
}
///
/// Gets the underlying pointer.
///
public IntPtr Value => _ptr;
[Conditional("ENABLE_UNITY_COLLECTIONS_CHECKS")]
private void CheckIsCreated()
{
if (!IsCreated)
{
throw new NullReferenceException("Object reference not set to an instance of an object");
}
}
///
/// Gets the delegate associated to this function pointer in order to call the function pointer.
/// This delegate can be called from a Burst Job or from regular C#.
/// If calling from regular C#, it is recommended to cache the returned delegate of this property
/// instead of using this property every time you need to call the delegate.
///
public T Invoke
{
get
{
CheckIsCreated();
return Marshal.GetDelegateForFunctionPointer(_ptr);
}
}
///
/// Whether the function pointer is valid.
///
public bool IsCreated => _ptr != IntPtr.Zero;
///
/// Converts a pointer to a function pointer.
///
/// The native pointer.
/// An instance of this interface.
IFunctionPointer IFunctionPointer.FromIntPtr(IntPtr ptr) => new FunctionPointer(ptr);
}
}