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

因此,我试图以64位模式运行.NET应用程序。我对可以为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.cpp和buffer.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(反之亦然),则该int是4字节还是8字节?关键字int会因平台的特定大小而改变,还是必须指定int64来获取8字节整数?这适用于上述较大问题的C ++源代码部分。

参考方案

在p / invoke调用中,请勿使用string作为返回值。这将假定内存是使用CoTaskMemAlloc分配的,因此将导致封送处理程序在返回的指针上调用CoTaskMemFree。这会导致运行时错误。

一种选择是,您可以更改本机代码,以将CoTaskMemAlloc用于其分配给C#的内存。如果这样做,则可以在p / invoke返回值中继续使用string

另一种选择是像现在一样使用malloc,在C#端使用IntPtr,然后从您的本机代码中导出一个解除分配器。

要将IntPtr转换为字符串,请使用Marshal.PtrToStringAnsi()。完成后,可以将指针传递回本机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

当回复有时是一个对象有时是一个数组时,如何在使用改造时解析JSON回复? - java

我正在使用Retrofit来获取JSON答复。这是我实施的一部分-@GET("/api/report/list") Observable<Bills> listBill(@Query("employee_id") String employeeID); 而条例草案类是-public static class…

Mongo汇总 - javascript

我的收藏中有以下文件{ "_id": ObjectId("54490b8104f7142f22ecc97f"), "title": "Sample1", "slug": "samplenews", "cat": …

快速解析来自http://bit.ly链接的方法 - java

给定来自流的一系列URL,其中可能有数百万个bit.ly,google或tinyurl缩短了链接,那么解析这些URL以获得最终URL的最可扩展的方法是什么?一个多线程爬虫在每个短链接上执行HEAD请求,同时缓存已经解决的请求?是否已经有提供此服务的服务?还要考虑到不会被url缩短服务阻止。假设规模是每天缩短2000万个网址。 java大神给出的解决方案 Go…

jQuery DataTable TableTool在IE和Firefox中不起作用 - c#

我在MVC4 ASP.NET Web应用程序中使用Jquery DataTable TableTool。导出到Excel和PDF可以与Chrome完美配合。但是不能在IE和FireFox中使用。我的代码如下 dom: 'T<"clear">lfrtip', tableTools: { "sSwfP…

如何通过Ajax将数据发送到不同的元素? - php

script.js $(document).on("click", ".send", function (event) { $.ajax({ url: "update.php", data: { id: id, }, type: "POST", success: function…