Multiplexing with a large number of sockets

We've used select() in this book to multiplex between open sockets. The select() function is great because it is available on many platforms. However, if you have a large number of open sockets, you can quickly run into the limitations of select().

There is a maximum number of sockets you can pass to select(). This number is available through the FD_SETSIZE macro.

This chapter's code repository includes a program, setsize.c, which prints the value of FD_SETSIZE.

The following screenshot shows this program being compiled and run on Windows 10:

The preceding screenshot shows FD_SETSIZE is 64 on this system. Although Windows's default size for FD_SETSIZE is quite low, it is common to see higher values on other systems. The default value of FD_SETSIZE on Linux is 1024.

On Windows, it is possible to increase FD_SETSIZE easily. You only need to define FD_SETSIZE yourself before including the winsock2.h header. For example, the following code increases FD_SETSIZE to 1024 on Windows:

#ifndef FD_SETSIZE
#define FD_SETSIZE 1024
#endif
#include <winsock2.h>

This works because Winsock uses FD_SETSIZE to build the fd_set type.

On Linux, this trick does not work. Linux defines fd_set as a bitmask, and it is not possible to increase its size without recompiling the kernel.

There are possible workarounds to effectively cheat select() into accepting socket descriptors larger than 1023 on Linux. One trick that usually works is to allocate an array of fd_set variables. Setting a socket is then done like this:

FD_SET(s % FD_SETSIZE, &set_array[s / FD_SETSIZE])

However, if you have to resort to a hack such as the preceding code, you may be better off avoiding select() and using a different multiplexing technique. The poll() function, for example, provides the functionality of select() without a limit on the number of file descriptors it can handle.