Commit 64dbc6bb authored by Joel Martin's avatar Joel Martin

wswrapper: timeout select.

The select call needs to timeout if a WebSocket socket keeps reporting
ready but actually isn't ready. To prevent it hanging forever in that
condition, the timeout value is now adjusted now for each call.

Move the DO_DEBUG and DO_TRACE settings to wswrapper.c.
parent 6b900d25
...@@ -13,6 +13,9 @@ ...@@ -13,6 +13,9 @@
* - programs using ppoll or epoll will not work correctly * - programs using ppoll or epoll will not work correctly
*/ */
#define DO_DEBUG 1
#define DO_TRACE 1
#include <stdio.h> #include <stdio.h>
#include <stdlib.h> #include <stdlib.h>
...@@ -24,13 +27,13 @@ ...@@ -24,13 +27,13 @@
#include <errno.h> #include <errno.h>
#include <string.h> #include <string.h>
#include <resolv.h> /* base64 encode/decode */ #include <resolv.h> /* base64 encode/decode */
#include <sys/time.h>
#include "md5.h" #include "md5.h"
#include "wswrapper.h" #include "wswrapper.h"
/* /*
* If WSWRAP_PORT environment variable is set then listen to the bind fd that * If WSWRAP_PORT environment variable is set then listen to the bind fd that
* matches WSWRAP_PORT, otherwise listen to the first socket fd that bind is * matches WSWRAP_PORT
* called on.
*/ */
int _WS_listen_fd = -1; int _WS_listen_fd = -1;
int _WS_nfds = 0; int _WS_nfds = 0;
...@@ -38,6 +41,42 @@ int _WS_fds[WS_MAX_FDS]; ...@@ -38,6 +41,42 @@ int _WS_fds[WS_MAX_FDS];
_WS_connection *_WS_connections[65536]; _WS_connection *_WS_connections[65536];
/*
* Utillity routines
*/
/*
* Subtract the `struct timeval' values X and Y, storing the
* result in RESULT. If TS is set then RESULT and X are really
* type-cast `struct timespec` so scale them appropriately.
* Return 1 if the difference is negative, otherwise 0.
*/
int _WS_subtract_time (result, x, y, ts)
struct timeval *result, *x, *y;
{
int scale = ts ? 1000 : 1;
/* Perform the carry for the later subtraction by updating y. */
if ((x->tv_usec / scale) < y->tv_usec) {
int sec = (y->tv_usec - (x->tv_usec / scale)) / 1000000 + 1;
y->tv_usec -= 1000000 * sec;
y->tv_sec += sec;
}
if ((x->tv_usec / scale) - y->tv_usec > 1000000) {
int sec = ((x->tv_usec / scale) - y->tv_usec) / 1000000;
y->tv_usec += 1000000 * sec;
y->tv_sec -= sec;
}
/* Compute the time remaining to wait.
* tv_usec is certainly positive. */
result->tv_sec = x->tv_sec - y->tv_sec;
result->tv_usec = x->tv_usec - (y->tv_usec * scale);
/* Return 1 if result is negative. */
return x->tv_sec < y->tv_sec;
}
/* /*
* WebSocket handshake routines * WebSocket handshake routines
*/ */
...@@ -209,9 +248,9 @@ int _WS_handshake(int sockfd) ...@@ -209,9 +248,9 @@ int _WS_handshake(int sockfd)
} }
/* /*
* Check WebSockets socket and return a positive value if there is enough data * Strip empty WebSockets frames and return a positive value if there is
* to base64 decode (a 4 byte chunk). If nonblock is not set then it will * enough data to base64 decode (a 4 byte chunk). If nonblock is not set then
* block until there is enough data (or until an error occurs). * it will block until there is enough data (or until an error occurs).
*/ */
ssize_t _WS_ready(int sockfd, int nonblock) ssize_t _WS_ready(int sockfd, int nonblock)
{ {
...@@ -402,7 +441,7 @@ ssize_t _WS_recv(int recvf, int sockfd, const void *buf, ...@@ -402,7 +441,7 @@ ssize_t _WS_recv(int recvf, int sockfd, const void *buf,
deccount = fend - fstart; deccount = fend - fstart;
} }
/* Now consume what was processed */ /* Now consume what was processed (if not MSG_PEEK) */
if (flags & MSG_PEEK) { if (flags & MSG_PEEK) {
DEBUG("_WS_recv(%d, _, %d) MSG_PEEK, not consuming\n", sockfd, len); DEBUG("_WS_recv(%d, _, %d) MSG_PEEK, not consuming\n", sockfd, len);
} else { } else {
...@@ -547,28 +586,28 @@ int _WS_select(int mode, int nfds, fd_set *readfds, ...@@ -547,28 +586,28 @@ int _WS_select(int mode, int nfds, fd_set *readfds,
{ {
_WS_connection *ws; _WS_connection *ws;
fd_set carryfds, savefds; fd_set carryfds, savefds;
struct timeval savetv; /* Assumes timeptr is two longs whether timeval or timespec */
struct timespec savets; struct timeval savetv, starttv, nowtv, difftv;
int carrycnt = 0; int carrycnt = 0;
int ret, i, ready, fd; int ret, less, i, ready, fd;
static void * (*func)(), * (*func2)(); static void * (*func0)(), * (*func1)();
if (!func) func = (void *(*)()) dlsym(RTLD_NEXT, "select"); if (!func0) func0 = (void *(*)()) dlsym(RTLD_NEXT, "select");
if (!func2) func2 = (void *(*)()) dlsym(RTLD_NEXT, "pselect"); if (!func1) func1 = (void *(*)()) dlsym(RTLD_NEXT, "pselect");
if ((_WS_listen_fd == -1) || (_WS_nfds == 0)) { if ((_WS_listen_fd == -1) || (_WS_nfds == 0)) {
if (mode == 0) { if (mode == 0) {
ret = (int) func(nfds, readfds, writefds, exceptfds, ret = (int) func0(nfds, readfds, writefds, exceptfds,
(struct timeval *)timeptr); (struct timeval *)timeptr);
} else if (mode == 1) { } else if (mode == 1) {
ret = (int) func2(nfds, readfds, writefds, exceptfds, ret = (int) func1(nfds, readfds, writefds, exceptfds,
(struct timespec *)timeptr, sigmask); (struct timespec *)timeptr, sigmask);
} }
return ret; return ret;
} }
TRACE(">> _WS_select(%d, %d, _, _, _, _) called\n", mode, nfds); TRACE(">> _WS_select(%d, %d, _, _, _, _) called\n", mode, nfds);
memcpy(&savetv, timeptr, sizeof(savetv)); memcpy(&savetv, timeptr, sizeof(savetv));
memcpy(&savets, timeptr, sizeof(savets)); gettimeofday(&starttv, NULL);
/* If we have carry-over return it right away */ /* If we have carry-over return it right away */
FD_ZERO(&carryfds); FD_ZERO(&carryfds);
...@@ -597,11 +636,14 @@ int _WS_select(int mode, int nfds, fd_set *readfds, ...@@ -597,11 +636,14 @@ int _WS_select(int mode, int nfds, fd_set *readfds,
} }
do { do {
TRACE(" _WS_select(%d, %d, _, _, _, _) tv/ts: %ld:%ld\n", mode, nfds,
((struct timeval *) timeptr)->tv_sec,
((struct timeval *) timeptr)->tv_usec);
if (mode == 0) { if (mode == 0) {
ret = (int) func(nfds, readfds, writefds, exceptfds, ret = (int) func0(nfds, readfds, writefds, exceptfds,
(struct timeval *)timeptr); (struct timeval *)timeptr);
} else if (mode == 1) { } else if (mode == 1) {
ret = (int) func2(nfds, readfds, writefds, exceptfds, ret = (int) func1(nfds, readfds, writefds, exceptfds,
(struct timespec *)timeptr, sigmask); (struct timespec *)timeptr, sigmask);
} }
if (! readfds) { if (! readfds) {
...@@ -630,17 +672,31 @@ int _WS_select(int mode, int nfds, fd_set *readfds, ...@@ -630,17 +672,31 @@ int _WS_select(int mode, int nfds, fd_set *readfds,
/* /*
* If all the ready readfds were WebSockets, but none of * If all the ready readfds were WebSockets, but none of
* them were really ready (empty frames) then repeat. * them were really ready (empty frames) then we select again.
*/ */
if (ret == 0) { if (ret == 0) {
/* Restore original values*/ /* Restore original values less passage of time */
/* TODO: fix timeptr for repeat */
memcpy(timeptr, &savetv, sizeof(savetv));
memcpy(timeptr, &savets, sizeof(savets));
memcpy(readfds, &savefds, sizeof(savefds)); memcpy(readfds, &savefds, sizeof(savefds));
gettimeofday(&nowtv, NULL);
/* Amount of time that has passed */
_WS_subtract_time(&difftv, &nowtv, &starttv, mode);
/* Subtract from original timout */
less = _WS_subtract_time((struct timeval *) timeptr,
&savetv, &difftv, mode);
if (less) {
/* Timer has expired */
TRACE(" _WS_select expired timer\n", mode, nfds);
break;
}
} }
} while (ret == 0); } while (ret == 0);
/* Restore original time value for pselect glibc does */
if (mode == 1) {
memcpy(timeptr, &savetv, sizeof(savetv));
}
TRACE("<< _WS_select(%d, %d, _, _, _, _) ret %d, errno %d\n", TRACE("<< _WS_select(%d, %d, _, _, _, _) ret %d, errno %d\n",
mode, nfds, ret, errno); mode, nfds, ret, errno);
return ret; return ret;
......
...@@ -7,9 +7,6 @@ ...@@ -7,9 +7,6 @@
* wswrapper.so. * wswrapper.so.
*/ */
//#define DO_DEBUG 1
//#define DO_TRACE 1
#ifdef DO_DEBUG #ifdef DO_DEBUG
#define DEBUG(...) \ #define DEBUG(...) \
if (DO_DEBUG) { \ if (DO_DEBUG) { \
......
Markdown is supported
0% or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment