完成端口取消已投递的IO操作的问题

lsgt 2009-10-20 10:38:54
先描述,后代码。
先投递一个ConnectEx(不带数据),在PENDING时,如果在投递操作的线程里closesocket,能够立刻返回。如果在GQCS的线程里closesocket,GQCS不会返回,知道20秒之后返回,错误码121.
在ConnectEx连接上后,在GQCS里投递一个WSARecv(服务器不返回任何数据),则不论在GQCS线程或者单独线程里closesocket,GQCS都不返回。
以上情况加不加shutdown都一样。
我还试过不用closesocket,用PostQueuedCompletionStatus,GQCS是返回了,但随后用closesocket释放资源,后面GQCS还会再返回一次。
连接上之后用国SO_UPDATE_CONNECT_CONTEXT也一样。

下面是代码:

#include <winsock2.h>
#include <mswsock.h>
#include <process.h>
#include <stdio.h>
#pragma comment(lib,"ws2_32.lib")
#pragma comment(lib,"mswsock.lib")
struct IODATA
{
OVERLAPPED ol;
DWORD op;
SOCKET s;
HANDLE hIocp;
};
int main(int argc, char* argv[])
{
int ret=0;
DWORD bytes=0;
WSADATA data={0};
static LPFN_CONNECTEX lpfnConnectEx=NULL;
int iRet=WSAStartup(0x0202,&data);

HANDLE hIocp=CreateIoCompletionPort(INVALID_HANDLE_VALUE,NULL,0,0);
IODATA io={0};
io.s=socket(AF_INET,SOCK_STREAM,0);
io.op=FD_CONNECT;
io.hIocp=hIocp;
CreateIoCompletionPort((HANDLE)io.s,hIocp,(ULONG_PTR)&io,0);

SOCKADDR_IN lAddr,rAddr;
lAddr.sin_family=AF_INET;
lAddr.sin_port=INADDR_ANY;
DWORD dwAddr=inet_addr("172.18.12.166");
memcpy((char*)&(lAddr.sin_addr),&dwAddr,sizeof(dwAddr));

rAddr.sin_family=AF_INET;
rAddr.sin_port=htons(80);
dwAddr=inet_addr("172.19.50.5");
memcpy((char*)&(rAddr.sin_addr),&dwAddr,sizeof(dwAddr));

ret=bind(io.s,(SOCKADDR*)&lAddr,sizeof(lAddr));

if(NULL==lpfnConnectEx)
{
GUID guid=WSAID_CONNECTEX;
ret=WSAIoctl(io.s,SIO_GET_EXTENSION_FUNCTION_POINTER,&guid,sizeof(guid),&lpfnConnectEx,sizeof(lpfnConnectEx),&bytes,NULL,NULL);
}

// _beginthreadex(NULL,0,_IoHandle,&io,0,NULL);

printf("connecting...\n");
ret=lpfnConnectEx(io.s,(SOCKADDR*)&rAddr,sizeof(rAddr),NULL,0,&bytes,&io.ol);

DWORD dwTick=GetTickCount();
while(1)
{
IODATA* pio=NULL;
LPOVERLAPPED pol=NULL;
ret=GetQueuedCompletionStatus(hIocp,&bytes,(PULONG_PTR)&pio,&pol,1000);
printf("GQCS returned --- ");
if(!pio)
{
printf("GQCS timeout.closesocket[socket=%d] -%d\n",io.s,GetTickCount()-dwTick);
if(ret==0)
{
shutdown(io.s,SD_BOTH);
closesocket(io.s); }
}
else
{
if(pio->op==FD_CONNECT)
{
if(ret)
{
printf("connect OK, recving[socket=%d]...\n",pio->s);
WSABUF wb;
wb.buf=new char[4096];
wb.len=4096;
pio->op=FD_READ;
ret=WSARecv(pio->s,&wb,1,&bytes,0,&pio->ol,NULL);
continue;
}
else
{
printf("%s[%d] failed %d\n",pio->op==FD_CONNECT ? "Connect" : "Recv",pio->s,WSAGetLastError());
break;
}
}
if(pio->op=FD_READ)
{
printf("printf returned[%d]\n",bytes);
continue;
}
}
}

system("pause");
return 0;
}

另外还有个帖子也是问这个的,解决了一起给分。
http://topic.csdn.net/u/20091019/16/524e5415-1712-4000-8157-5210fc88bcb3.html?seed=1000414584&r=60529111#r_60529111
...全文
316 5 打赏 收藏 转发到动态 举报
写回复
用AI写文章
5 条回复
切换为时间正序
请发表友善的回复…
发表回复
lsgt 2009-10-30
  • 打赏
  • 举报
回复
不行啊。ConnectEx还没连接成功,没法调用DisconnectEx。各位能不能帮我测试一下》
zealstone 2009-10-27
  • 打赏
  • 举报
回复
关闭连接不应该直接调用closesocket吧,应该调用DisconnectEx(),然后在GQCS里面处理断开事件。
lsgt 2009-10-26
  • 打赏
  • 举报
回复
我这个是客户端。但我需要发起上千个连接。GQCS写的不规范没关系,我是用来测试如何取消COnnectEx的。
zealstone 2009-10-23
  • 打赏
  • 举报
回复
帮楼主顶一下。
你这个是服务端还是客户端,怎么又有iocp又有ConnectEx。GetQueuedCompletionStatus()这个函数返回值的处理也不标准吧。依据msdn是不是应该先判断ret?
lsgt 2009-10-21
  • 打赏
  • 举报
回复
不是吧?怎么没人理呢?

4,356

社区成员

发帖
与我相关
我的任务
社区描述
通信技术相关讨论
社区管理员
  • 网络通信
加入社区
  • 近7日
  • 近30日
  • 至今
社区公告
暂无公告

试试用AI创作助手写篇文章吧