Sending to a disconnected peer

There are three basic ways a TCP connection can fail. They are as follows:

A network outage prevents data from reaching your peer. In this case, TCP tries to retransmit data. If connectivity is re-established, TCP simply picks back up where it left off. Otherwise, the connection eventually times out. This timeout can be on the order of 10 minutes.

The second way a TCP connection can fail is if the connected peer application crashes. In this case, the peer's operating system sends a FIN message. This case is indistinguishable from the peer calling close() on their end. If your application continues to send data after having received the FIN message, the peer's system will send an RST message to indicate an error.

Finally, a connection could fail because the peer's whole system has crashed. In this case, it won't be able to send a FIN message. This case looks the same as a network outage, and the TCP connection would eventually timeout. However, consider what happens if the crashed system reboots before the connection times out. In that case, the rebooted system will eventually receive a TCP message from the original connection. The rebooted system will not recognize the TCP connection and will send an RST message in response to indicate an error condition.

To reiterate, if you use send() on a socket that your peer thinks is closed, that peer will respond with an RST message. This state is easily detected by the return value of recv().

A more serious issue to consider is what happens when send() is called on a socket that has already received an RST message from its peer. On Unix-based systems, the default is to send a SIGPIPE signal to your program. If you don't handle this signal, the operating system will terminate your program.

It is therefore essential for TCP servers to either handle or disable the SIGPIPE signal. Failure to handle this case means that a rude client could kill your server.

Signals are complicated. If you're already using signals in your program, you may want to handle SIGPIPE. Otherwise, I recommend you just disable it by setting the SIGPIPE handler to SIG_IGN.

The following code disables SIGPIPE on Unix-based systems:

#if !defined(_WIN32)
#include <signal.h>
#endif

#if !defined(_WIN32)
signal(SIGPIPE, SIG_IGN);
#endif

As an alternative, you can use MSG_NOSIGNAL with send() as shown in the following code:

send(my_socket, buffer, sizeof(buffer), MSG_NOSIGNAL);

If the signal is ignored or MSG_NOSIGNAL is used, send() returns -1 and setsĀ errno to EPIPE.

On Windows, attempting to call send() on a closed socket generally results in WSAGetLastError() returning WSAECONNRESET.

An example program, server_crash.c, is included in this chapter's code repository. This program accepts TCP connections on port 8080. It then waits for the client to disconnect, and then attempts two sends to that disconnected client. This program is useful as a tool to explore the return values, error codes, and function behavior in different scenarios.