gpt4 book ai didi

c# - 如何使用显式对齐(非托管 dll)将结构从 C 编码到 C#

转载 作者:太空宇宙 更新时间:2023-11-04 03:17:03 24 4
gpt4 key购买 nike

我必须在我的 C# 应用程序中使用非托管 dll(用 C 编写)。在这个 dll 中我有一个结构:

typedef struct {
void (*Func1)(void*, int*);
void (*Func2)(void*, int*);
} myStructure;

并且,我有一个使用这个结构作为 [in/out] 参数的函数:

void functionInterface(myStructure* pToStruct);

在 C# 中,我将该结构翻译为:

[StructLayout(LayoutKind.Sequential)]
public unsafe struct myStructure
{
//defining delegate instances
public func1 Func1;
public func2 Func2;
...
//defining delegate types
public delegate void Func1(void* l,int* data);
public delegate void Func2(void* l,int* data);
...
}

和 C# 中的函数:

[DllImport("lib.dll", CallingConvention = CallingConvention.Cdecl)]
public static extern void functionInterface(ref myStructure pToStruct);

当调用这个 functionInterface() 时,在运行时,异常会上升。比如“在...地址...写入的访问冲突”或“在……地址……阅读的访问冲突”

我找不到原因,我认为问题可能在于翻译结构中成员的正确对齐,或堆栈对齐,但是,我不知道如何正确地做到这一点。或者问题可能不同,性质不同,我无法检测到,希望你们能帮助我。

最佳答案

这非常有效...正如我喜欢说的:你打破它,你修复它...

C# 端:

[StructLayout(LayoutKind.Sequential)]
public struct myStructure
{
public Fn1 Func1;
public Fn2 Func2;

[UnmanagedFunctionPointer(CallingConvention.Cdecl)]
public delegate void Fn1(IntPtr p1, ref int p2);

[UnmanagedFunctionPointer(CallingConvention.Cdecl)]
public delegate void Fn2(IntPtr p1, ref int p2);
}

[DllImport("lib.dll", CallingConvention = CallingConvention.Cdecl)]
public static extern void functionInterface(out myStructure pToStruct);

然后:

myStructure myStructure;
functionInterface(out myStructure);

int num1 = 100, num2 = 200;
myStructure.Func1((IntPtr)0x1000, ref num1);
Console.WriteLine("C#-side: out Func1: {0}", num1);
myStructure.Func2((IntPtr)0x2000, ref num2);
Console.WriteLine("C#-side: out Func2: {0}", num2);

C端:(我把所有的代码都放在一个.c/.cpp文件里,没有header)

#ifdef __cplusplus
extern "C"
{
#endif
typedef struct
{
void (*Func1)(void*, int*);
void (*Func2)(void*, int*);
} myStructure;

void Fn1(void* p1, int* p2)
{
printf("C-side: Fn1: %p, %d - ", p1, *p2);
*p2 += 1;
}

void Fn2(void* p1, int* p2)
{
printf("C-side: Fn2: %p, %d - ", p1, *p2);
*p2 += 1;
}

__declspec(dllexport) void functionInterface(myStructure* pToStruct)
{
pToStruct->Func1 = Fn1;
pToStruct->Func2 = Fn2;
}
#ifdef __cplusplus
}
#endif

关于c# - 如何使用显式对齐(非托管 dll)将结构从 C 编码到 C#,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/50471810/

24 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com