我正在为C ++ / CLI编写一个C ++库的包装器(静态.lib,没有可用的源代码)来控制工业计算机上的数字I / O引脚。
我的目标是使用.NET CLR处理现有C#应用程序上的DIO引脚事件。
我能想到的唯一可行的选择是在C ++ / CLI中使用委托来在引脚状态发生变化时触发事件(由现有的lib通知),然后在C#部分处理这些事件。我在这里使用更简单的模拟对象尝试了基本功能:Firing events in C++ and handling them in C#
我的问题是,当引脚更改时,库用于注册回调的函数需要一个(非托管)指针才能运行,而且我不知道如何粘合所有这些。此函数的签名(来自库附带的头文件)如下:
typedef void (__stdcall* COS_INT_CALLBACK)(COS_INT_CALLBACK_ARG* arg);
BOOL RegisterCallbackDICOS(COS_INT_CALLBACK callback);
我尝试从非托管代码调用EventHandler委托,但当然这不起作用:
#include "stdafx.h"
#include <WDT_DIO.H>
#include <string.h>
#include <stdio.h>
using namespace System;
using namespace System::Runtime::InteropServices;
public ref class PinStateChangedEventArgs : public EventArgs
{
public:
property int Direction;
property DateTime TimeReached;
};
public ref class CppDIOHandler
{
private:
public:
CppDIOHandler() {
};
void StartDIO() {
//Step 1, initialize DIO library by invoking InitDIO()
if (!InitDIO())
{
Console::WriteLine("InitDIO --> FAILED");
return;
}
Console::WriteLine("InitDIO --> PASSED");
//Step 2, setup Change-of-State Interrupt mask and level/edge mode
COS_INT_SETUP setup;
memset(&setup, 0, sizeof(setup));
setup.portMask = 0x0f; // 00001111b, enable ch.0~3
setup.edgeMode = 0x00; // generate interrupt on level change
setup.edgeType = 0x00; // rising/falling edge, only effective when edgeMode = 1
if (!SetupDICOS(&setup, sizeof(setup)))
{
Console::WriteLine("SetupDICOS --> FAILED");
return;
}
Console::WriteLine("SetupDICOS --> PASSED");
//Step 3, register the callback function
if (!RegisterCallbackDICOS(callback_function))
{
Console::WriteLine("RegisterCallbackDICOS --> FAILED");
return;
}
Console::WriteLine("RegisterCallbackDICOS --> PASSED");
//Step 4, start the DI Change-of-State Interrupt
if (!StartDICOS())
{
Console::WriteLine("StartDICOS --> FAILED");
return;
}
Console::WriteLine("StartDICOS --> PASSED");
}
void TriggerEvent(int direction)
{
PinStateChangedEventArgs^ args = gcnew PinStateChangedEventArgs();
args->Direction = direction;
args->TimeReached = DateTime::Now;
OnPinStateChanged(args);
}
event EventHandler<PinStateChangedEventArgs^>^ PinStateChanged;
protected:
virtual void OnPinStateChanged(PinStateChangedEventArgs^ e)
{
PinStateChanged(this, e);
}
};
#pragma unmanaged
//Step 0, define a Change-of-State Interrupt callback function
void __stdcall callback_function(COS_INT_CALLBACK_ARG* arg)
{
printf("data=0x%02x, flag=0x%02x, seq=%02d\n",
arg->portData, arg->intrFlag, arg->intrSeq);
//Here should go some code to trigger an event with the managed delegate
//The following can't be used on unmanaged code
CppDIOHandler^ dh = gcnew CppDIOHandler();
dh->TriggerEvent(1);
}
#pragma managed
void main()
{
return;
}
有没有正确的方法呢?要么是先前代码的旁路工作,要么是使用我的&#34; PinStateChanged&#34;委托作为回调函数对我有效。
答案 0 :(得分:0)
使用Hans Passant提供的信息和链接,我得到了它的工作。代码结束如下(可能需要一些清理):
#include "stdafx.h"
#include <WDT_DIO.H>
#include <string.h>
#include <stdio.h>
using namespace System;
using namespace System::Runtime::InteropServices;
public ref class PinStateChangedEventArgs : public EventArgs
{
public:
property int Direction;
property DateTime TimeReached;
};
public ref class CppDIOHandler
{
public:
CppDIOHandler() {}
delegate void CallbackDelegate(COS_INT_CALLBACK_ARG*);
void __stdcall callback_function(COS_INT_CALLBACK_ARG* arg)
{
printf("data=0x%02x, flag=0x%02x, seq=%02d\n",
arg->portData, arg->intrFlag, arg->intrSeq);
TriggerEvent(1);
}
static void StartDIO() {
//Step 1, initialize DIO library by invoking InitDIO()
if (!InitDIO())
{
Console::WriteLine("InitDIO --> FAILED");
return;
}
Console::WriteLine("InitDIO --> PASSED");
//Step 2, setup Change-of-State Interrupt mask and level/edge mode
COS_INT_SETUP setup;
memset(&setup, 0, sizeof(setup));
setup.portMask = 0x0f; // 00001111b, enable ch.0~3
setup.edgeMode = 0x00; // generate interrupt on level change
setup.edgeType = 0x00; // rising/falling edge, only effective when edgeMode = 1
if (!SetupDICOS(&setup, sizeof(setup)))
{
Console::WriteLine("SetupDICOS --> FAILED");
return;
}
Console::WriteLine("SetupDICOS --> PASSED");
CppDIOHandler^ cdh = gcnew CppDIOHandler();
CallbackDelegate^ callback = gcnew CallbackDelegate(cdh, &callback_function);
IntPtr stubPointer = Marshal::GetFunctionPointerForDelegate(callback);
COS_INT_CALLBACK functionPointer = static_cast<COS_INT_CALLBACK>(stubPointer.ToPointer());
GC::KeepAlive(callback);
//Step 3, register the callback function
if (!RegisterCallbackDICOS(functionPointer))
{
Console::WriteLine("RegisterCallbackDICOS --> FAILED");
return;
}
Console::WriteLine("RegisterCallbackDICOS --> PASSED");
//Step 4, start the DI Change-of-State Interrupt
if (!StartDICOS())
{
Console::WriteLine("StartDICOS --> FAILED");
return;
}
Console::WriteLine("StartDICOS --> PASSED");
}
void TriggerEvent(int direction)
{
PinStateChangedEventArgs^ args = gcnew PinStateChangedEventArgs();
args->Direction = direction;
args->TimeReached = DateTime::Now;
OnPinStateChanged(args);
}
event EventHandler<PinStateChangedEventArgs^>^ PinStateChanged;
virtual void OnPinStateChanged(PinStateChangedEventArgs^ e)
{
PinStateChanged(this, e);
}
};
void main()
{
return;
}