summaryrefslogtreecommitdiff
diff options
context:
space:
mode:
-rw-r--r--pflocal/sock.c126
1 files changed, 86 insertions, 40 deletions
diff --git a/pflocal/sock.c b/pflocal/sock.c
index 1ffac4c9..925f8187 100644
--- a/pflocal/sock.c
+++ b/pflocal/sock.c
@@ -20,14 +20,68 @@
#include <string.h> /* For bzero() */
-#include "pflocal.h"
+#include <cthreads.h>
+
#include "sock.h"
#include "pipe.h"
+
+/* ---------------------------------------------------------------- */
-/* We hold this lock before we lock two sockets at once, to prevent someone
- else trying to lock the same two sockets in the reverse order, resulting
- in a deadlock. */
-static struct mutex socket_pair_lock;
+/* A port bucket to handle SOCK_USERs and ADDRs. */
+static struct port_bucket *sock_port_bucket;
+
+/* True if there are threads servicing sock requests. */
+static int sock_server_active = 0;
+static spin_lock_t sock_server_active_lock = SPIN_LOCK_INITIALIZER;
+
+/* A demuxer for socket operations. */
+static int
+sock_demuxer (mach_msg_header_t *inp, mach_msg_header_t *outp)
+{
+ extern int socket_server (mach_msg_header_t *inp, mach_msg_header_t *outp);
+ extern int io_server (mach_msg_header_t *inp, mach_msg_header_t *outp);
+ return socket_server (inp, outp) || io_server (inp, outp);
+}
+
+/* Handle socket requests while there are sockets around. */
+static void
+handle_sock_requests ()
+{
+ 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,
+ 1, MACH_PORT_NULL);
+ }
+
+ /* The last service thread is about to exist; make this known. */
+ spin_lock (&sock_server_active_lock);
+ sock_server_active = 0;
+ spin_unlock (&sock_server_active_lock);
+
+ /* Let the whole joke start once again. */
+ ports_enable_bucket (sock_port_bucket);
+}
+
+/* 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. */
+static void
+ensure_sock_server ()
+{
+ spin_lock (&sock_server_active_lock);
+ if (sock_server_active)
+ spin_unlock (&sock_server_active_lock);
+ else
+ {
+ sock_server_active = 1;
+ spin_unlock (&sock_server_active_lock);
+ cthread_detach (cthread_fork ((cthread_fn_t)handle_sock_requests,
+ (any_t)0));
+ }
+}
/* ---------------------------------------------------------------- */
@@ -109,6 +163,7 @@ sock_create (struct pipe_class *pipe_class, struct sock **sock)
if (! (pipe_class->flags & PIPE_CLASS_CONNECTIONLESS))
/* No data source yet. */
new->read_pipe->flags |= PIPE_BROKEN;
+ new->read_pipe->refs++;
new->refs = 0;
new->flags = 0;
@@ -155,7 +210,7 @@ sock_clone (struct sock *template, struct sock **sock)
/* ---------------------------------------------------------------- */
-struct port_class *sock_user_port_class = NULL;
+struct port_class *sock_user_port_class;
/* Get rid of a user reference to a socket. */
static void
@@ -169,18 +224,15 @@ clean_sock_user (void *vuser)
error_t
sock_create_port (struct sock *sock, mach_port_t *port)
{
- struct sock_user *user;
-
- if (sock_user_port_class == NULL)
- sock_user_port_class = ports_create_class (NULL, clean_sock_user);
-
- user =
- ports_allocate_port (pflocal_port_bucket,
+ struct sock_user *user =
+ ports_allocate_port (sock_port_bucket,
sizeof (struct sock_user), sock_user_port_class);
if (!user)
return ENOMEM;
+ ensure_sock_server ();
+
mutex_lock (&sock->lock);
sock->refs++;
mutex_unlock (&sock->lock);
@@ -202,7 +254,7 @@ struct addr
struct mutex lock;
};
-struct port_class *addr_port_class = NULL;
+struct port_class *addr_port_class;
/* Get rid of ADDR's socket's reference to it, in preparation for ADDR going
away. */
@@ -242,15 +294,14 @@ clean_addr (void *vaddr)
inline error_t
addr_create (struct addr **addr)
{
- if (addr_port_class == NULL)
- addr_port_class = ports_create_class (unbind_addr, clean_addr);
-
*addr =
- ports_allocate_port (pflocal_port_bucket,
+ ports_allocate_port (sock_port_bucket,
sizeof (struct addr), addr_port_class);
if (! *addr)
return ENOMEM;
+ ensure_sock_server ();
+
(*addr)->sock = NULL;
return 0;
}
@@ -347,26 +398,14 @@ sock_get_addr (struct sock *sock, struct addr **addr)
return err; /* XXX */
}
-
-/* If SOCK is a connected socket, returns a send right to SOCK's peer's
- address in ADDR_PORT. */
-error_t
-sock_get_write_addr_port (struct sock *sock, mach_port_t *addr_port)
-{
- error_t err = 0;
-
- mutex_lock (&sock->lock);
- if (sock->write_addr)
- *addr_port = ports_get_right (sock->write_addr);
- else
- err = ENOTCONN;
- mutex_unlock (&sock->lock);
-
- return err;
-}
/* ---------------------------------------------------------------- */
+/* We hold this lock before we lock two sockets at once, to prevent someone
+ else trying to lock the same two sockets in the reverse order, resulting
+ in a deadlock. */
+static struct mutex socket_pair_lock;
+
/* Connect SOCK1 and SOCK2. */
error_t
sock_connect (struct sock *sock1, struct sock *sock2)
@@ -382,15 +421,13 @@ sock_connect (struct sock *sock1, struct sock *sock2)
void connect (struct sock *wr, struct sock *rd)
{
- if ((wr->flags & SOCK_SHUTDOWN_WRITE)
- || (rd->flags & SOCK_SHUTDOWN_READ))
+ if (!( (wr->flags & SOCK_SHUTDOWN_WRITE)
+ || (rd->flags & SOCK_SHUTDOWN_READ)))
{
struct pipe *pipe = rd->read_pipe;
pipe_aquire (pipe);
pipe->flags &= ~PIPE_BROKEN; /* Not yet... */
wr->write_pipe = pipe;
- ensure_addr (rd, &wr->write_addr); /* XXXXXXXXXXX */
- ports_port_ref (wr->write_addr);
mutex_unlock (&pipe->lock);
}
}
@@ -479,9 +516,18 @@ sock_shutdown (struct sock *sock, unsigned flags)
/* ---------------------------------------------------------------- */
+error_t
+sock_global_init ()
+{
+ sock_port_bucket = ports_create_bucket ();
+ sock_user_port_class = ports_create_class (NULL, clean_sock_user);
+ addr_port_class = ports_create_class (unbind_addr, clean_addr);
+ return 0;
+}
+
/* Try to shutdown any active sockets, returning EBUSY if we can't. */
error_t
-sock_goaway (int flags)
+sock_global_shutdown (int flags)
{
/* XXX */
return 0;