题
我已经为WinCC(西门子)(SCADA)编写了一个用户控件。现在我想传递一个指向控件的指针。执行此操作的唯一方法是:将指针写入属性。
WinCC 只有这个方法来设置属性
- SetPropBOOL
- 设置属性字符
- 设置PropDouble
- 设置属性词
控件的属性具有 UInt 作为数据类型,我使用 SetPropDouble 方法来设置对象的地址。
WinCC 全局脚本 (ANSI-C)
//autoDB is an ADODB.Connection object
//object* autoDB = __object_create("ADODB.Connection");
extern __object* autoDB;
//SetPropDouble( "PictureName", "ControlName", "PropertyName", (DWORD)(&autoDB) );
SetPropDouble( "PictureName", "ControlName", "PropertyName", (DWORD)autoDB );
我已经调试了我的控件(挂钩 WinCC 进程),并且我看到属性集被分配了一个地址值,例如0x03041080。
现在的问题是:我如何获取地址上的 C# (.Net) 对象?
我的尝试抛出异常: 执行引擎异常
private ADODB.Connection _database;
private IntPtr _ptr = IntPtr.Zero;
public uint DataBase{
get{
return (uint)_ptr;
}
set{
if( value != 0 ){
_ptr = (IntPtr)value;
GCHandle gH = GCHandle.FromIntPtr(_ptr); // THIS LINE THROW THE EXCEPTION
_database = gH.Target;
}
}
}
好的:我已经更改了我的代码以使用 STRING
温控中心
extern __object* autoDB;
DWORD addr = (DWORD)autoDB;
char sAddr[11];
sprintf( sAddr, "%d\0", addr );
SetPropChar( "PictureName", "ControlName", "DataBaseAddr", sAddr );
现在是c#
private string _lpszDataBaseAddr = "";
public string DataBaseAddr{
get{
return _lpszDataBaseAddr;
}
set{
uint addr;
bool ret = uint.TryParse( value, out addr );
if( ! ret ){
return;
}
IntPtr ptr = (IntPtr)addr;
GCHandle gH = GCHandle.FromIntPtr( ptr ); // THE SAME ERROR!
}
}
其他发现!
ADO 对象的地址不在调用我的控件的进程内存中(使用 ollydbg 进行调试)。WinCC 有两个程序:用于可视化的 PDLRT.exe(这正在调用我的控件)和用于运行 GLOBAL-SCRIPT (Ansi-C) 的 SCRIPT.exe。
从 PDLRT,我可以访问 ADO 对象的指针地址。通过在 C# 中调用 ADO-object-address 的 GCHandle,会引发异常。(执行引擎异常)
解决方案 2
好的,
很久以前,我已经请求了支持 Siemens
.
西门子:加载的 Dll、控件等加载到单独的内存中,而不是加载到应用程序(主)内存中。Dll、控件之间的内存地址共享...不管用。都有各自独立的记忆。
极好的。唯一办法:管道或其他通信实现(TCP/IP,...)。
其他提示
我不知道c#是否可以通过指针进入c ++。
无论如何,这个:(DWORD)(&autoDB)
是错误的,它将指针的地址作为属性的值,这是无意义的。
需要指针的值,即(DWORD) autoDB
。
SetPropDouble()
接受类型生成的值,即浮点数。这不会是共享指针的非常好方法,这是一个(大)整数。尝试一些不同的表示,如果您无法访问足够大的整数,则字符串可能会起作用。 不隶属于 StackOverflow