aboutsummaryrefslogtreecommitdiff
path: root/pflocal/sserver.c
diff options
context:
space:
mode:
Diffstat (limited to 'pflocal/sserver.c')
-rw-r--r--pflocal/sserver.c105
1 files changed, 105 insertions, 0 deletions
diff --git a/pflocal/sserver.c b/pflocal/sserver.c
new file mode 100644
index 00000000..7df69a41
--- /dev/null
+++ b/pflocal/sserver.c
@@ -0,0 +1,105 @@
+/* Server for socket ops
+
+ Copyright (C) 1995, 1997, 2013 Free Software Foundation, Inc.
+
+ Written by Miles Bader <miles@gnu.ai.mit.edu>
+
+ 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 the Free Software Foundation; either version 2, or (at
+ your option) any later version.
+
+ This program is distributed in the hope that it will be useful, but
+ WITHOUT ANY WARRANTY; without even the implied warranty of
+ MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
+ General Public License for more details.
+
+ You should have received a copy of the GNU General Public License
+ along with this program; if not, write to the Free Software
+ Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA. */
+
+#include <pthread.h>
+#include <stdio.h>
+
+#include <hurd/ports.h>
+
+/* A port bucket to handle SOCK_USERs and ADDRs. */
+struct port_bucket *sock_port_bucket;
+
+/* ---------------------------------------------------------------- */
+
+/* True if there are threads servicing sock requests. */
+static int sock_server_active = 0;
+static pthread_spinlock_t sock_server_active_lock = PTHREAD_SPINLOCK_INITIALIZER;
+
+#include "io_S.h"
+#include "socket_S.h"
+#include "../libports/interrupt_S.h"
+#include "../libports/notify_S.h"
+
+/* A demuxer for socket operations. */
+static int
+sock_demuxer (mach_msg_header_t *inp, mach_msg_header_t *outp)
+{
+ mig_routine_t routine;
+ if ((routine = io_server_routine (inp)) ||
+ (routine = socket_server_routine (inp)) ||
+ (routine = ports_interrupt_server_routine (inp)) ||
+ (routine = ports_notify_server_routine (inp)))
+ {
+ (*routine) (inp, outp);
+ return TRUE;
+ }
+ else
+ return FALSE;
+}
+
+/* Handle socket requests while there are sockets around. */
+static void *
+handle_sock_requests (void *unused)
+{
+ while (ports_count_bucket (sock_port_bucket) > 0)
+ {
+ ports_enable_bucket (sock_port_bucket);
+ ports_manage_port_operations_multithread (sock_port_bucket, sock_demuxer,
+ 30*1000, 2*60*1000, 0);
+ }
+
+ /* The last service thread is about to exist; make this known. */
+ pthread_spin_lock (&sock_server_active_lock);
+ sock_server_active = 0;
+ pthread_spin_unlock (&sock_server_active_lock);
+
+ /* Let the whole joke start once again. */
+ ports_enable_bucket (sock_port_bucket);
+
+ return NULL;
+}
+
+/* Makes sure there are some request threads for sock operations, and starts
+ a server if necessary. This routine should be called *after* creating the
+ port(s) which need server, as the server routine only operates while there
+ are any ports. */
+void
+ensure_sock_server ()
+{
+ pthread_t thread;
+ error_t err;
+
+ pthread_spin_lock (&sock_server_active_lock);
+ if (sock_server_active)
+ pthread_spin_unlock (&sock_server_active_lock);
+ else
+ {
+ sock_server_active = 1;
+ pthread_spin_unlock (&sock_server_active_lock);
+ err = pthread_create (&thread, NULL, handle_sock_requests, NULL);
+ if (!err)
+ pthread_detach (thread);
+ else
+ {
+ errno = err;
+ perror ("pthread_create");
+ }
+ }
+}