Passing a C# callback function through Interop/pinvoke

佐手、 提交于 2019-12-25 09:03:37

问题


I am writing a C# application which uses Interop services to access functions in a native C++ DLL. I am already using about 10 different functions which are working.

Now I am not sure how to handle passing a callback as a parameter so that the DLL can call my code.

Here is the function prototype of the DLL:

typedef void (WINAPI * lpfnFunc)(const char *arg1, const char *arg2)

And the function that allows me to pass the above type:

int WINAPI SetFunc(lpfnFunc f)

Here is my C# code for the delegate and function definitions:

public delegate void Func(string arg1, string arg2);

public static void MyFunc(string arg1, string arg2)

Here is my C# code for the SetFunc Interop function:

[DllImport("lib.dll", CharSet = CharSet.Ansi)]
public static extern int SetFunc(Func lpfn);

And finally here is the code where I call the SetFunc function and pass it my callback:

SetFunc(new Func(MyFunc));

Unfortunately my function is not being called when it should be. The return value of the SetFunc function is returning the error code for a Success, so either it's not calling my function or it's not working because my code is wrong.


回答1:


This works for me:

Calc.h (Calc.dll, C++):

extern "C" __declspec(dllexport) double Calc(double x, double y, double __stdcall p(double, double));

Calc.cpp (Calc.dll, C++):

#include "calc.h"

__declspec(dllimport) double Calc(double x, double y, double __stdcall p(double, double))
{
    double s = p(x*x, y*y);
    return x * y + s;
}

Program.cs (Sample.exe, C#):

class Program
{
    delegate double MyCallback(double x, double y);
    [DllImport("Calc.dll", CallingConvention = CallingConvention.Cdecl)]
    static extern double Calc(double x, double y, [MarshalAs(UnmanagedType.FunctionPtr)]MyCallback func);

    static void Main(string[] args)
    {
        double z = Calc(1, 2, (x, y) => 45);
    }
}



回答2:


Can you try changing the Func delegate to

    public delegate void Func([In, MarshalAs(UnmanagedType.LPStr)] string arg1, [In, MarshalAs(UnmanagedType.LPStr)] string arg2);

And the SetFunc method to

[DllImport("lib", CharSet = CharSet.Ansi, CallingConvention = CallingConvention.Winapi)]
public static extern int SetFunc(Func lpfn);


来源:https://stackoverflow.com/questions/9056048/how-to-call-c-sharp-code-from-c-code

易学教程内所有资源均来自网络或用户发布的内容,如有违反法律规定的内容欢迎反馈
该文章没有解决你所遇到的问题?点击提问,说说你的问题,让更多的人一起探讨吧!