Description
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
Labels
Type
Projects
Status
No status