Skip to content

[API Proposal]: System.Runtime.InteropServices.Marshalling.DelegateMarshaller for delegate #108738

@Gaoyifei1011

Description

@Gaoyifei1011

Background and motivation

System.Runtime.InteropServices.Marshalling.DelegateMarshaller for delegate

The System.Runtime.InteropServices.Marshalling namespace already contains a number of XXXMarshaller classes for P/Invoke source generation and COM source generation, which play an important role. However, regarding the delegate delegate type, we need to manually get the pointer of the delegate type or convert the pointer to the delegate type (method: Marshal. GetFunctionPointerForDelegate() and Marshal. GetDelegateForFunctionPointer())。 So hopefully the DOTNET team will provide a DelegateMarshaller type that can automatically handle delegate-type marshalling, similar to UTF16StringMarshaller That's it.

#----------------------------------------------

目前在 System.Runtime.InteropServices.Marshalling 命名空间中已经包含了大量的 XXXMarshaller 类用于 P/Invoke 源生成和 COM 源生成,这些类起到了重要作用。然而有关 delegate 委托类型,需要我们手动获取 delegate 类型的指针或者将指针转换为委托类型(方法:Marshal.GetFunctionPointerForDelegate()Marshal.GetDelegateForFunctionPointer())。所以希望 DOTNET 团队能提供一个 DelegateMarshaller 类型,能自动处理 delegate 类型封送,类似 UTF16StringMarshaller 那样。

API Proposal

namespace System.Runtime.InteropServices.Marshalling
{
    [CLSCompliant(false)]
    [CustomMarshaller(typeof(Delegate), MarshalMode.Default, typeof(DelegateMarshaller))]
    public static unsafe class DelegateMarshaller
    {
        /// <summary>
        /// Converts a delegate to an unmanaged version.
        /// </summary>
        /// <param name="managed">The managed delegate to convert.</param>
        /// <returns>An unmanaged delegate.</returns>
        public static void* ConvertToUnmanaged(Delegate? managed);

        /// <summary>
        /// Converts an unmanaged delegate to a managed version.
        /// </summary>
        /// <param name="unmanaged">The unmanaged delegate to convert.</param>
        /// <returns>A managed delegate.</returns>
        public static Delegate? ConvertToManaged(void* unmanaged);
    }

    [CLSCompliant(false)]
    [CustomMarshaller(typeof(CustomMarshallerAttribute.GenericPlaceholder), MarshalMode.Default, typeof(DelegateMarshaller<>))]
    public static unsafe class DelegateMarshaller<TDelegate> where TDelegate : notnull
    {
        /// <summary>
        /// Converts a delegate to an unmanaged version.
        /// </summary>
        /// <param name="managed">The managed delegate to convert.</param>
        /// <returns>An unmanaged delegate.</returns>
        public static void* ConvertToUnmanaged(TDelegate? managed);

        /// <summary>
        /// Converts an unmanaged delegate to a managed version.
        /// </summary>
        /// <param name="unmanaged">The unmanaged delegate to convert.</param>
        /// <returns>A managed delegate.</returns>
        public static TDelegate? ConvertToManaged(void* unmanaged);
    }
}

API Usage

Use for LibraryImport (P/Invoke source generation) and COMWrappers (COM source generation)

Alternative Designs

None

Risks

None

Metadata

Metadata

Assignees

No one assigned

    Projects

    • Status

      No status

    Relationships

    None yet

    Development

    No branches or pull requests

    Issue actions