Blocking on send()

When we call send() with an amount of data, send() first copies this data into an outgoing buffer provided by the operating system. If we call send() when its outgoing buffer is already full, it blocks until its buffer has emptied enough to accept more of our data.

In some cases where send() would block, it instead returns without copying all of the data as requested. In this case, the return value of send() indicates how many bytes were actually copied. One example of this is if your program is blocking on send() and then receives a signal from the operating system. In these cases, it is up to the caller to try again with any remaining data.

In this chapter's TCP server code section, we ignored the possibility that send() could block or be interrupted. In a fully robust application, what we need to do is compare the return value from send() with the number of bytes that we tried to send. If the number of bytes actually sent is less than requested, we should use select() to determine when the socket is ready to accept new data, and then call send() with the remaining data. As you can imagine, this can become a bit complicated when keeping track of multiple sockets.

As the operating system usually provides a large enough outgoing buffer, we were able to avoid this possibility with our earlier server code. If we know that our server may try to send large amounts of data, we should certainly check for the return value from send().

The following code example assumes that buffer contains buffer_len bytes of data to send over a socket called peer_socket. This code blocks until we've sent all of buffer or an error (such as the peer disconnecting) occurs:

int begin = 0;
while (begin < buffer_len) {
int sent = send(peer_socket, buffer + begin, buffer_len - begin, 0);
if (sent == -1) {
//Handle error
}
begin += sent;
}

If we are managing multiple sockets and don't want to block, then we should put all sockets with pending send() into an fd_set and pass it as the third parameter to select(). When select() signals on these sockets, then we know that they are ready to send more data.

Chapter 13, Socket Programming Tips and Pitfalls, addresses concerns regarding the send() function's blocking behavior in more detail.