]> git.pond.sub.org Git - empserver/blobdiff - src/lib/lwp/sel.c
Update copyright notice.
[empserver] / src / lib / lwp / sel.c
index 2668ec9a1dfd151faf350187c83c3a66737e6e05..1d1b12adabdbd507b20bc287a11b78536822d5d6 100644 (file)
@@ -1,7 +1,8 @@
 /*
  *  Empire - A multi-player, client/server Internet based war game.
- *  Copyright (C) 1986-2005, Dave Pare, Jeff Bailey, Thomas Ruschak,
+ *  Copyright (C) 1994-2007, Dave Pare, Jeff Bailey, Thomas Ruschak,
  *                           Ken Stevens, Steve McClure
+ *  Copyright (C) 1991-3 Stephen Crane
  *
  *  This program is free software; you can redistribute it and/or modify
  *  it under the terms of the GNU General Public License as published by
@@ -19,9 +20,9 @@
  *
  *  ---
  *
- *  See the "LEGAL", "LICENSE", "CREDITS" and "README" files for all the
- *  related information and legal notices. It is expected that any future
- *  projects/authors will amend these files as needed.
+ *  See files README, COPYING and CREDITS in the root of the source
+ *  tree for related information and legal notices.  It is expected
+ *  that future projects/authors will amend these files as needed.
  *
  *  ---
  *
@@ -33,9 +34,7 @@
 
 #include <config.h>
 
-#include <stdlib.h>
 #include <errno.h>
-#include <sys/types.h>
 #include <sys/uio.h>
 #include <sys/file.h>
 #include <sys/time.h>
 #include <unistd.h>
 #include "lwp.h"
 #include "lwpint.h"
+#include "prototypes.h"
 
-#if defined(_EMPTH_LWP)
+/* Largest fd in LwpReadfds, LwpWritefds */
+static int LwpMaxfd;
 
-struct lwpSelect {
-    int maxfd;
-    int nfds;
-    fd_set readmask;
-    fd_set writemask;
-    struct lwpProc **wait;
-    struct lwpQueue delayq;
-    struct lwpProc *proc;
-};
+/* Number of file descriptors in LwpReadfds, LwpWritefds */
+static int LwpNfds;
+
+/* File descriptors waited for in lwpSleepFd() */
+static fd_set LwpReadfds, LwpWritefds;
 
-struct lwpSelect LwpSelect;
+/* Map file descriptor to thread sleeping in lwpSleepFd() */
+static struct lwpProc **LwpFdwait;
+
+/* Threads sleeping in lwpSleepUntil(), in no particular order */
+static struct lwpQueue LwpDelayq;
+
+/* The thread executing lwpSelect() */
+static struct lwpProc *LwpSelProc;     
 
 void
 lwpInitSelect(struct lwpProc *proc)
 {
-    LwpSelect.maxfd = 0;
-    LwpSelect.nfds = 0;
-    FD_ZERO(&LwpSelect.readmask);
-    FD_ZERO(&LwpSelect.writemask);
-    LwpSelect.wait = calloc(FD_SETSIZE, sizeof(struct lwpProc *));
-    LwpSelect.delayq.head = 0;
-    LwpSelect.delayq.tail = 0;
-    LwpSelect.proc = proc;
+    LwpMaxfd = 0;
+    LwpNfds = 0;
+    FD_ZERO(&LwpReadfds);
+    FD_ZERO(&LwpWritefds);
+    LwpFdwait = calloc(FD_SETSIZE, sizeof(struct lwpProc *));
+    LwpDelayq.head = 0;
+    LwpDelayq.tail = 0;
+    LwpSelProc = proc;
 }
 
 void
@@ -78,27 +82,27 @@ lwpSleepFd(int fd, int mask)
 
     if (CANT_HAPPEN(fd > FD_SETSIZE))
        return;
-    if (LwpSelect.wait[fd] != 0) {
+    if (LwpFdwait[fd] != 0) {
        lwpStatus(LwpCurrent,
                  "multiple sleeps attempted on file descriptor %d", fd);
        return;
     }
     if (mask & LWP_FD_READ)
-       FD_SET(fd, &LwpSelect.readmask);
+       FD_SET(fd, &LwpReadfds);
     if (mask & LWP_FD_WRITE)
-       FD_SET(fd, &LwpSelect.writemask);
+       FD_SET(fd, &LwpWritefds);
 
-    LwpSelect.nfds++;
+    LwpNfds++;
 
-    if (LwpSelect.maxfd == 0 && LwpSelect.delayq.head == 0) {
+    if (LwpMaxfd == 0 && LwpDelayq.head == 0) {
        /* select process is sleeping until first waiter arrives */
        lwpStatus(LwpCurrent, "going to resched fd %d", fd);
-       lwpReady(LwpSelect.proc);
+       lwpReady(LwpSelProc);
     }
     lwpStatus(LwpCurrent, "going to wait on fd %d", fd);
-    if (fd > LwpSelect.maxfd)
-       LwpSelect.maxfd = fd;
-    LwpSelect.wait[fd] = LwpCurrent;
+    if (fd > LwpMaxfd)
+       LwpMaxfd = fd;
+    LwpFdwait[fd] = LwpCurrent;
     LwpCurrent->fd = fd;
     lwpReschedule();
 }
@@ -110,24 +114,24 @@ lwpWakeupFd(struct lwpProc *proc)
        return;
 
     lwpStatus(proc, "awakening; was sleeping on fd %d", proc->fd);
-    FD_CLR(proc->fd, &LwpSelect.readmask);
-    FD_CLR(proc->fd, &LwpSelect.writemask);
-    LwpSelect.nfds--;
-    LwpSelect.wait[proc->fd] = 0;
+    FD_CLR(proc->fd, &LwpReadfds);
+    FD_CLR(proc->fd, &LwpWritefds);
+    LwpNfds--;
+    LwpFdwait[proc->fd] = 0;
     proc->fd = -1;
     lwpReady(proc);
 }
 
 void
-lwpSleepUntil(long int until)
+lwpSleepUntil(long until)
 {
     lwpStatus(LwpCurrent, "sleeping for %ld sec", until - time(0));
     LwpCurrent->runtime = until;
-    if (LwpSelect.maxfd == 0 && LwpSelect.delayq.head == 0) {
+    if (LwpMaxfd == 0 && LwpDelayq.head == 0) {
        /* select process is sleeping until first waiter arrives */
-       lwpReady(LwpSelect.proc);
+       lwpReady(LwpSelProc);
     }
-    lwpAddTail(&LwpSelect.delayq, LwpCurrent);
+    lwpAddTail(&LwpDelayq, LwpCurrent);
     lwpReschedule();
 }
 
@@ -151,20 +155,20 @@ lwpSelect(void *arg)
     FD_ZERO(&writemask);
     while (1) {
        while (1) {
-           if (LwpSelect.nfds)
+           if (LwpNfds)
                break;
-           if (LwpSelect.delayq.head)
+           if (LwpDelayq.head)
                break;
            /* wait for someone to lwpSleepFd or lwpSleepUntil */
-           LwpSelect.maxfd = 0;
+           LwpMaxfd = 0;
            lwpStatus(us, "no fds or sleepers, waiting");
            lwpReschedule();
        }
        tv.tv_sec = 1000000;
        tv.tv_usec = 0;
-       if (LwpSelect.delayq.head) {
+       if (LwpDelayq.head) {
            time(&now);
-           proc = LwpSelect.delayq.head;
+           proc = LwpDelayq.head;
            for (; proc != 0; proc = proc->next) {
                delta = proc->runtime - now;
                if (delta < tv.tv_sec)
@@ -175,27 +179,25 @@ lwpSelect(void *arg)
        }
        lwpStatus(us, "selecting; sleep %ld secs", tv.tv_sec);
 
-       memcpy(&readmask, &LwpSelect.readmask, sizeof(fd_set));
-       memcpy(&writemask, &LwpSelect.writemask, sizeof(fd_set));
-       n = select(LwpSelect.maxfd + 1, &readmask, &writemask,
-                  (fd_set *) 0, &tv);
-
+       memcpy(&readmask, &LwpReadfds, sizeof(fd_set));
+       memcpy(&writemask, &LwpWritefds, sizeof(fd_set));
+       n = select(LwpMaxfd + 1, &readmask, &writemask, NULL, &tv);
        if (n < 0) {
-           if (errno == EINTR) {
-               /* go handle the signal */
-               lwpReady(us);
-               lwpReschedule();
-               continue;
+           if (errno != EINTR) {
+               logerror("select failed (%s)", strerror(errno));
+               exit(1);
            }
-           lwpStatus(us, "select failed (bad file descriptor?)");
-           exit(-1);
+           /* go handle the signal */
+           lwpReady(us);
+           lwpReschedule();
+           continue;
        }
 
-       if (LwpSelect.delayq.head) {
+       if (LwpDelayq.head) {
            /* sleeping proecss activity */
            time(&now);
            save.tail = save.head = 0;
-           while (NULL != (proc = lwpGetFirst(&LwpSelect.delayq))) {
+           while (NULL != (proc = lwpGetFirst(&LwpDelayq))) {
                if (now >= proc->runtime) {
                    lwpStatus(proc, "sleep done");
                    lwpReady(proc);
@@ -203,21 +205,21 @@ lwpSelect(void *arg)
                    lwpAddTail(&save, proc);
                }
            }
-           LwpSelect.delayq = save;
+           LwpDelayq = save;
        }
        if (n > 0) {
            /* file descriptor activity */
-           for (fd = 0; fd <= LwpSelect.maxfd; fd++) {
-               if (LwpSelect.wait[fd] == 0)
+           for (fd = 0; fd <= LwpMaxfd; fd++) {
+               if (LwpFdwait[fd] == 0)
                    continue;
                if (FD_ISSET(fd, &readmask)) {
-                   lwpStatus(LwpSelect.wait[fd], "input ready");
-                   lwpWakeupFd(LwpSelect.wait[fd]);
+                   lwpStatus(LwpFdwait[fd], "input ready");
+                   lwpWakeupFd(LwpFdwait[fd]);
                    continue;
                }
                if (FD_ISSET(fd, &writemask)) {
-                   lwpStatus(LwpSelect.wait[fd], "output ready");
-                   lwpWakeupFd(LwpSelect.wait[fd]);
+                   lwpStatus(LwpFdwait[fd], "output ready");
+                   lwpWakeupFd(LwpFdwait[fd]);
                    continue;
                }
            }
@@ -228,4 +230,3 @@ lwpSelect(void *arg)
     }
     /*NOTREACHED*/
 }
-#endif