6
votes

I have a very annoying problem that I found several times on other forums but I can't find a proper solution. The problem is recv() returns 0 on the last few bytes of a connection. Here are some background information.

  • Both (client / server) applications run on the same machine.
  • Both (client / server) sockets are non-blocking
  • The transfered data size is 53 bytes.
  • Both (client / server) call shutdown and closesocket when the last send()/recv() was executed.
  • I also tried with SO_LINGER and 10 seconds, no success either

I call send() several times (small chunks) and from the client side 53 bytes are transfered. The server calls recv() several times (4 byte requests) and read 49 bytes and then returns 0 (54 Bytes - 49 Bytes, so 4 bytes are missing).

MSDN and some forums write for non-blocking sockets:

  • recv() definitely returns < 0 on error and errno / WSAGetLastError is set
  • recv() definitely returns = 0 when the other side closed the connection
  • recv() definitely returns > 0 when data was read

MSDN also says:

Using the closesocket or shutdown functions with SD_SEND or SD_BOTH results in a RELEASE signal being sent out on the control channel. Due to ATM's use of separate signal and data channels, it is possible that a RELEASE signal could reach the remote end before the last of the data reaches its destination, resulting in a loss of that data. One possible solutions is programming a sufficient delay between the last data sent and the closesocket or shutdown function calls for an ATM socket.

This is regarded in the example of recv() and send(): http://msdn.microsoft.com/en-us/library/windows/desktop/ms740121(v=vs.85).aspx

But still no success, I still get some interrupts in 10% of all connections after the 49 Byte is received, 90% of the connections succeed. Any ideas? Thx.

3
I suspect you are making one of several classic socket programming mistakes. I have a few ideas, but I'd rather see your code first before writing something up.selbie
SO_LINGER with a positive timeout in non-blocking mode doesn't make sense. It won't block or timeout.user207421

3 Answers

15
votes

recv() returns 0 only when you request a 0-byte buffer or the other peer has gracefully disconnected. If you are not receiving all of the data you are expecting, then you are not reading the data correctly to begin with. Please update your question with your actual code.

-1
votes

My guess is that you're not really sending all the data you think your are sending. Check out:

The Ultimate SO_LINGER page

-1
votes
  • recv() definitely returns = 0 when the other side closed the connection

This is not completely true, in the following code using non-blocking winsock2 tcp, when no data is available, select returns 1 and recv returns 0, as does WSAGetLastError().

fd_set test = {1, socket};
const timeval timeout = {0, 0};
if (!::select(0, &test, nullptr, nullptr, &timeout)) return 0;
int done = ::recv(socket, buffer, 1, 0);

This continues even after the other end has called:

::shutdown(socket, SD_BOTH);
::closesocket(socket);

and then ended. Communication works as expected, it is just ::recv that seems to be "broken".