We're writing a client and a server to do (what I thought was) pretty simple network communications. Mulitple clients connect to the server which then is supposed to send the data back to all other clients.
The server just sits in a blocking select
loop waiting for traffic, and when it comes, sends the data to the other clients. This seems to work just fine.
The problem is the client. In response to a read, it will sometimes want to do a write.
However, I've found that if I use:
rv = select(fdmax + 1, &master_list, NULL, NULL, NULL);
My code will block until there is new data to read. But sometimes (asynchronously, from another thread) I'll have new data to write on the network communication thread. So, I want my select to wake up periodically and let me check if there are data to write, like:
if (select(....) != -1)
{
if (FD_SET(sockfd, &master_list))
// handle data or disconnect
else
// look for data to write and write() / send() those.
}
I tried setting the select to poll mode (or ridiculously short timeouts) with:
// master list contains the sockfd from the getaddrinfo/socket/connect seq
struct timeval t;
memset(&t, 0, sizeof t);
rv = select(fdmax + 1, &master_list, NULL, NULL, &t);
but have found that then then client never gets any incoming data.
I also tried setting the socket fd to be non-blocking, like:
fcntl(sockfd, F_SETFL, O_NONBLOCK);
but that doesn't solve the problem:
- if my client
select()
has nostruct timeval
, reading data works, but it never unblocks to let me look for writable data. - if my client
select()
has atimeval
to get it to poll, then it never signals that there are incoming data to read, and my app freezes thinking there is no network connection made (despite the fact that all other function calls have succeeded)
Any pointers at all as to what I could be doing wrong? Is it not possible to do read-write on the same socket (I can't believe that to be true).
(EDIT: The correct answer, and thing that I remembered on the server but not on the client, is to have a second fd_set, and copy master_list before each call to select():
// declare and FD_ZERO read_fds:
// put sockfd in master_list
while (1)
{
read_fds = master_list;
select(...);
if (FD_ISSET(read_fds))
....
else
// sleep or otherwise don't hog cpu resources
}
)