gpt4 book ai didi

c# - 使用 p/Invoke 的 x64 版本的 .net 应用程序崩溃

转载 作者:行者123 更新时间:2023-12-02 02:59:13 24 4
gpt4 key购买 nike

因此,我尝试在 64 位模式下运行我的 .NET 应用程序。我依赖于一个 dll,我可以为 x64 编译该 dll,并且它可以正确加载。我可以调用一些调用,但是当我尝试调用特定调用时,应用程序会完全崩溃。没有错误或任何东西...

错误发生在两个地方之一;要么在这里:

#include "socket.h"
int SenderAddrSize = sizeof(SOCKADDR_IN);
SOCKADDR_IN CSocket::SenderAddr;

...

char* CSocket::tcpip()
{
if(sockid<0)return NULL;
if(getpeername(sockid, (SOCKADDR *)&SenderAddr, &SenderAddrSize) == SOCKET_ERROR)return NULL;
return inet_ntoa(SenderAddr.sin_addr);
}

...

或这里:

#include "buffer.h"
char CBuffer::retval[20001];
CBuffer::CBuffer()
{
BuffSize = 30;
data = (char*)malloc(BuffSize);
count = 0;
readpos = 0;
writepos = 0;
}

...

您可以在我的 GitHub HERE 找到完整代码在 socket.cppbuffer.cpp分别。 代码的入口点和调用上述方法的对象都位于main.cpp

p/invoke 调用如下:

对于套接字:

[DllImport("SockLib", CallingConvention = CallingConvention.Cdecl, EntryPoint = "tcpip")]
public static extern String tcpip(Double socket);

对于缓冲区:

 [DllImport("SockLib", CallingConvention = CallingConvention.Cdecl, EntryPoint = "createbuffer")]
public static extern Double createbuffer();

关于代码崩溃的任何想法。这在 x86 中就像一个魅力,但在 x64 中则不然......

另外,作为一个附带问题...如果我将一个 int 写入 x86 中运行的客户端上的缓冲区并在 x64 服务器上接收它(反之亦然),那么该 int 是 4 个字节还是 8 个字节?关键字 int 是否会根据平台特定大小进行更改,或者我是否必须指定 int64 才能获得 8 字节整数?这适用于上面较大问题的 C++ 源代码部分。

最佳答案

不要使用string作为p/invoke调用中的返回值。这将假设内存是使用 CoTaskMemAlloc 分配的,因此将导致编码器在返回的指针上调用 CoTaskMemFree 。这会导致运行时错误。

一种选择是您可以更改 native 代码以使用 CoTaskMemAlloc 来分配要返回给 C# 的内存。如果您这样做了,那么您可以继续在 p/invoke 返回值中使用 string

另一种方法是像现在一样使用 malloc,在 C# 端使用 IntPtr,并从 native 代码中导出释放器。

要从 IntPtr 转换为字符串,请使用 Marshal.PtrToStringAnsi()。完成此操作后,您可以将指针传递回 native DLL,以便它可以对其调用 free()

看起来像这样:

[DllImport("SockLib", CallingConvention=CallingConvention.Cdecl)]
public static extern IntPtr tcpip(double socket);

你这样调用它:

IntPtr strPtr = tcpip(socket);
string str = Marshal.PtrToStringAnsi(strPtr);

您需要对每个返回字符串的 p/invoke 调用执行类似的操作。

是否需要释放这些指针并不是 100% 清楚的。这取决于您是否在 C++ 代码中分配了内存。例如,tcpip 映射到对 inet_ntoa 的调用。这会返回一个指向套接字库拥有的缓冲区的指针。因此您的代码不得尝试释放它。但是,如果您返回使用 mallocnew[] 分配的内存,那么您也必须释放它。

<小时/>

其他一些观察结果:

  • 您返回的套接字在 C++ 代码中声明为 int,但在 C 包装器中将它们键入为 double。不要那样做。始终使用 int
  • C# 类型 int 为 32 字节。对于 64 位整数,请使用 long。无符号版本为 uintulong。对于机器字大小的类型,请使用 IntPtrUIntPtr

关于c# - 使用 p/Invoke 的 x64 版本的 .net 应用程序崩溃,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/12945273/

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