Skip to content

Using [LibraryImport] with a generic delegate parameter (e.g. Func<int, int>) causes an exception #113590

Open
@ascpixi

Description

@ascpixi

Description

When using a generic delegate, like Func<int, int, int> as a parameter to a function that is marked with LibraryImportAttribute, an ArgumentException: The specified Type must not be a generic type is thrown when the source-generated code attempts to invoke GetFunctionPointerForDelegate.

Reproduction Steps

Declare a function like this:

    [LibraryImport("user32.dll")]
    [return: MarshalAs(UnmanagedType.Bool)]
    public static partial bool EnumWindows(
        Func<nint, nint, bool> lpEnumFunc,
        nint lParam
    );

...then invoke it:

        User32.EnumWindows((hwnd, _) => {
            Console.WriteLine(hwnd);
            return true;
        }, default);

The invocation will cause a System.ArgumentException to be thrown, specifically from the last line of this source-generated snippet:

public static partial bool EnumWindows(global::System.Func<nint, nint, bool> lpEnumFunc, nint lParam)
{
    nint __lpEnumFunc_native;
    bool __retVal;
    int __retVal_native;
    // Marshal - Convert managed data to native data.
    __lpEnumFunc_native = lpEnumFunc != null ? global::System.Runtime.InteropServices.Marshal.GetFunctionPointerForDelegate(lpEnumFunc) : default; // <-- HERE

Expected behavior

The source generator should either recognize generic delegates like Func<...> or Action<...>, and include a special case when generating code to marshal parameters with these types, or it should throw a compile-time error to notify the developer that this is not supported.

Actual behavior

A runtime exception is thrown, at the time of the invocation of the imported method.

Regression?

No response

Known Workarounds

Declare a delegate type, like so:

public delegate bool EnumWindowsProc(IntPtr hWnd, IntPtr lParam);

...then use that non-generic delegate type instead of the generic delegate type. This isn't that much of an issue, but it does slightly impact code aesthetics. Declaring the function parameters inside the method declaration takes up less lines and displays the signature of the method when inspecting the [LibraryImport]-ed declaration.

Configuration

.NET 9.0.100, running under Windows x64.

Other information

Might be related to #32963 if we were to change the functionality of Marshal.GetFunctionPointerForDelegate(). I think an easier approach would be to change the source generator itself.

Metadata

Metadata

Assignees

No one assigned

    Type

    No type

    Projects

    • Status

      No status

    Milestone

    No milestone

    Relationships

    None yet

    Development

    No branches or pull requests

    Issue actions