FD_ACCEPT检测到,但然后调用'接受'失败,10038 - WSAENOTSOCK

问题描述:

基本上,就是这样。同时,客户的连接呼叫是成功的。这怎么可能?我没有添加任何代码,因为我不知道错误在哪里。FD_ACCEPT检测到,但然后调用'接受'失败,10038 - WSAENOTSOCK

服务器:检测FD_ACCEPT。呼叫接受()失败。

客户端:呼叫连接()成功。然后它检测到FD_CONNECT。以下send()成功。之后的send()失败(10053 - WSAECONNABORTED)。

void Server::get_addressinfo() { 
    // Resolve the local address and port to be used by the server 
    const char * p = port.c_str(); 
    int iResult = getaddrinfo(NULL, p, &hints, &result); 
    if (iResult != 0) { 
     throw MyException("getaddrinfo failed."); 
    } 
} 

void Server::create_socket() { 
    ListenSocket = socket(result->ai_family, result->ai_socktype, result->ai_protocol); 
    if (ListenSocket == INVALID_SOCKET) { 
     throw MyException("socket creation failed."); 
    } 
} 

void Server::bind_socket() { 
    int iResult = bind(ListenSocket, result->ai_addr, (int) result->ai_addrlen); 
    if (iResult == SOCKET_ERROR) { 
     closesocket(ListenSocket); 
     throw MyException("bind failed."); 
    } 
} 

void Server::listen_for_connection() { 
    int iResult = listen(ListenSocket, SOMAXCONN); 
    if (iResult == SOCKET_ERROR) { 
     closesocket(ListenSocket); 
     throw MyException("listen failed."); 
    } 
} 

void Server::goLive() { 
    get_addressinfo(); 
    create_socket(); 
    bind_socket(); 
    listen_for_connection(); 
    wait_for(FD_ACCEPT); 
    accept_connection(); 
} 

void Server::wait_for(u_int event_type) { 
    WSAEVENT event = WSACreateEvent(); 
    WSAEventSelect(ListenSocket, event, event_type); 
    WSAEVENT lphEvents[1] = {event}; 
    WSANETWORKEVENTS NetworkEvents = {0}; 
    int nReturnCode = WSAWaitForMultipleEvents(1, &lphEvents[0], false, WSA_INFINITE, false); 
    if (nReturnCode==WSA_WAIT_FAILED) 
     throw MyException("WSA__WAIT_FAILED.\n"); 
    if (WSAEnumNetworkEvents(ListenSocket, lphEvents[0], &NetworkEvents) == SOCKET_ERROR) 
     throw MyException("WSAEnumNetworkEvents => SOCKET_ERROR.\n"); 
    WSACloseEvent(event); ***// THIS WAS THE BUG !!!*** 
} 

void Server::accept_connection() { 
    ClientSocket = accept(ListenSocket, NULL, NULL); 
    if (ClientSocket == INVALID_SOCKET) { 
     closesocket(ListenSocket); 
     throw MyException("accept failed."); 
    } 
} 
+1

错误信息几乎不可能更清晰,'错误在哪里'显然是在调用'accept()'。但如果你不想要答案,不要发布代码。随你便。 – EJP

+0

添加了代码。 – Earnie

+1

如果接受失败,为什么要关闭侦听套接字?这是否应该只接受一个连接或者是处理多个连接的方案的一部分? –

我在代码上做了一些改进,但是bug仍然存在。现在我发现了问题行:

WSACloseEvent(event); 

注释掉后,服务器接受连接。

从MSDN WSACloseEvent函数文档:

的WSACloseEvent功能关闭句柄到一个事件对象并释放与该事件对象相关联的资源。

我想这意味着它释放了socket =>因此10038 - WSAENOTSOCK。我落入了与INVALID_SOCKET不相等的套接字值。所以我认为套接字是有效的。它不是......

+0

你应该删除整个事件的事情。你不需要它。 accept()方法已经被阻塞。如果您需要计时,请使用SO_RCVTIMEO。 – EJP

+0

我可以重现错误。这很奇怪,因为'WSACreateEvent()'简单地调用'CreateEvent()','WSAWaitForMultipleEvents()'简单地调用'WaitForMultipleObjectsEx()',而'WSACloseEvent()'简单地调用'CloseHandle()'。所以你实际上关闭了一个标准的Win32事件,这与套接字无关。另一方面,我猜'WSAEventSelect()'是以某种方式将它们连接在一起的,所以关闭一个关闭另一个。 –

+0

@EJP:或者,你可以在调用accept()之前使用'select()'。 –