Applying C - Interrupts & Polling |
Written by Harry Fairhead | ||||||
Monday, 26 December 2022 | ||||||
Page 2 of 2
An Event Driven Socket ServerAs an example of using poll let’s return to the socket server described in the previous chapter. If this isn't the case we need to make the call to accept non-blocking. The simplest way of doing this is to OR SOCK_NONBLOCK in the call to socket: int sockfd = socket(server→ai_family, server->ai_socktype| SOCK_NONBLOCK, server->ai_protocol); This change makes the attempt to read from the socket: int client_fd = accept(sockfd, return immediately but with -1 if there is no data to read. Our only solution is to repeatedly attempt to read the data using accept: for(;;){ int client_fd = accept(sockfd, (struct sockaddr *) &client_addr, &addr_size); if(client_fd>0){ process the data; } You can see that this keeps the thread busy in a tight polling loop. A much better solution is to use poll to suspend the thread until there is data to read: struct pollfd fdset[1]; fdset[0].fd = sockfd; fdset[0].events = POLLIN; fdset[0].revents = 0; poll(fdset, 1, 10000); The call now waits until there is data read to read from the socket or timeout occurs after 10 seconds. This means we can follow the call with code to read the data, but only after checking that the event occurred and it wasn’t a timeout or a signal: if (fdset[0].revents & POLLIN) { int client_fd = accept(sockfd, Of course this only allows one client to use the server. We still need to put the poll and the processing in a loop: struct pollfd fdset[1]; for (;;) { fdset[0].fd = sockfd; fdset[0].events = POLLIN; fdset[0].revents = 0; poll(fdset, 1, 10000); if (fdset[0].revents & POLLIN) { int client_fd = accept(sockfd, Notice that you have to initialize the struct for each call to poll and you can also add a check for a timeout. You can make this much more sophisticated but this is the basic idea. It is common for example to pass the request on to a separate thread so that the loop can go back to waiting for the next client. The complete program is: #define _GNU_SOURCE #include <stdio.h> #include <stdlib.h> #include <unistd.h> #include <sys/socket.h> #include <string.h> #include <sys/types.h> #include <netinet/in.h> #include <netdb.h> #include <poll.h> int main(int argc, char** argv) { struct addrinfo hints, *server; memset(&hints, 0, sizeof hints); hints.ai_family = AF_INET; hints.ai_socktype = SOCK_STREAM; hints.ai_flags = AI_PASSIVE; getaddrinfo(NULL, "1024", &hints, &server); int sockfd = socket(server→ai_family, In book but not included in this extract
Summary
Now available as a paperback or ebook from Amazon.Applying C For The IoT With Linux
Also see the companion book: Fundamental C <ASIN:1871962609> <ASIN:1871962617> To be informed about new articles on I Programmer, sign up for our weekly newsletter, subscribe to the RSS feed and follow us on Twitter, Facebook or Linkedin.
Comments
or email your comment to: comments@i-programmer.info |
||||||
Last Updated ( Wednesday, 28 December 2022 ) |