Move named pipe objects into directory name space.
Change tests accordingly.
Add small test for WaitNamedPipe.
diff --git a/server/named_pipe.c b/server/named_pipe.c
index 0aa5733..ac6aaab 100644
--- a/server/named_pipe.c
+++ b/server/named_pipe.c
@@ -99,6 +99,12 @@
struct list waiters; /* list of clients waiting to connect */
};
+struct named_pipe_device
+{
+ struct object obj; /* object header */
+ struct namespace *pipes; /* named pipe namespace */
+};
+
static void named_pipe_dump( struct object *obj, int verbose );
static void named_pipe_destroy( struct object *obj );
@@ -181,6 +187,26 @@
default_fd_cancel_async /* cancel_async */
};
+static void named_pipe_device_dump( struct object *obj, int verbose );
+static struct object *named_pipe_device_lookup_name( struct object *obj,
+ struct unicode_str *name, unsigned int attr );
+static void named_pipe_device_destroy( struct object *obj );
+
+static const struct object_ops named_pipe_device_ops =
+{
+ sizeof(struct named_pipe_device), /* size */
+ named_pipe_device_dump, /* dump */
+ no_add_queue, /* add_queue */
+ NULL, /* remove_queue */
+ NULL, /* signaled */
+ no_satisfied, /* satisfied */
+ no_signal, /* signal */
+ no_get_fd, /* get_fd */
+ named_pipe_device_lookup_name, /* lookup_name */
+ no_close_handle, /* close_handle */
+ named_pipe_device_destroy /* destroy */
+};
+
static void named_pipe_dump( struct object *obj, int verbose )
{
struct named_pipe *pipe = (struct named_pipe *) obj;
@@ -338,6 +364,58 @@
assert( !client->fd );
}
+static void named_pipe_device_dump( struct object *obj, int verbose )
+{
+ assert( obj->ops == &named_pipe_device_ops );
+ fprintf( stderr, "Named pipe device\n" );
+}
+
+static struct object *named_pipe_device_lookup_name( struct object *obj, struct unicode_str *name,
+ unsigned int attr )
+{
+ struct named_pipe_device *device = (struct named_pipe_device*)obj;
+ struct object *found;
+
+ assert( obj->ops == &named_pipe_device_ops );
+ assert( device->pipes );
+
+ if (!name->len) return NULL;
+
+ if ((found = find_object( device->pipes, name, attr | OBJ_CASE_INSENSITIVE )))
+ name->len = 0;
+
+ return found;
+}
+
+static void named_pipe_device_destroy( struct object *obj )
+{
+ struct named_pipe_device *device = (struct named_pipe_device*)obj;
+ assert( obj->ops == &named_pipe_device_ops );
+ free( device->pipes );
+}
+
+/* this will be deleted as soon an we fix wait_named_pipe */
+static struct named_pipe_device *named_pipe_device;
+
+struct named_pipe_device *create_named_pipe_device( void )
+{
+ static const WCHAR pipeW[] = {'\\','?','?','\\','P','I','P','E'};
+ static struct unicode_str pipe = {pipeW, sizeof(pipeW)};
+ struct named_pipe_device *dev;
+
+ if ((dev = create_named_object_dir( NULL, &pipe, 0, &named_pipe_device_ops )) &&
+ get_error() != STATUS_OBJECT_NAME_EXISTS)
+ {
+ if (!(dev->pipes = create_namespace( 7 )))
+ {
+ release_object( dev );
+ dev = NULL;
+ }
+ }
+ named_pipe_device = dev;
+ return dev;
+}
+
static int pipe_data_remaining( struct pipe_server *server )
{
struct pollfd pfd;
@@ -445,37 +523,43 @@
return flags;
}
-static struct named_pipe *create_named_pipe( const struct unicode_str *name, unsigned int attr )
-{
- struct named_pipe *pipe;
-
- pipe = create_named_object( sync_namespace, &named_pipe_ops, name, attr | OBJ_OPENIF );
- if (pipe)
- {
- if (get_error() != STATUS_OBJECT_NAME_EXISTS)
- {
- /* initialize it if it didn't already exist */
- pipe->instances = 0;
- list_init( &pipe->servers );
- list_init( &pipe->waiters );
- }
- }
- return pipe;
-}
-
-static struct named_pipe *open_named_pipe( const struct unicode_str *name, unsigned int attr )
+static struct named_pipe *create_named_pipe( struct directory *root, const struct unicode_str *name,
+ unsigned int attr )
{
struct object *obj;
+ struct named_pipe *pipe = NULL;
+ struct unicode_str new_name;
- if ((obj = find_object( sync_namespace, name, attr )))
+ if (!name || !name->len) return alloc_object( &named_pipe_ops );
+
+ if (!(obj = find_object_dir( root, name, attr, &new_name ))) return NULL;
+ if (!new_name.len)
{
- if (obj->ops == &named_pipe_ops) return (struct named_pipe *)obj;
- release_object( obj );
- set_error( STATUS_OBJECT_TYPE_MISMATCH );
+ if (attr & OBJ_OPENIF && obj->ops == &named_pipe_ops)
+ set_error( STATUS_OBJECT_NAME_EXISTS );
+ else
+ {
+ release_object( obj );
+ obj = NULL;
+ if (attr & OBJ_OPENIF)
+ set_error( STATUS_OBJECT_TYPE_MISMATCH );
+ else
+ set_error( STATUS_OBJECT_NAME_COLLISION );
+ }
+ return (struct named_pipe *)obj;
}
- else set_error( STATUS_OBJECT_NAME_NOT_FOUND );
- return NULL;
+ if (obj->ops != &named_pipe_device_ops)
+ set_error( STATUS_OBJECT_TYPE_MISMATCH );
+ else
+ {
+ struct named_pipe_device *dev = (struct named_pipe_device *)obj;
+ if ((pipe = create_object( dev->pipes, &named_pipe_ops, &new_name, NULL )))
+ clear_error();
+ }
+
+ release_object( obj );
+ return pipe;
}
static struct pipe_server *get_pipe_server_obj( struct process *process,
@@ -552,13 +636,24 @@
struct named_pipe *pipe;
struct pipe_server *server;
struct unicode_str name;
+ struct directory *root = NULL;
reply->handle = 0;
get_req_unicode_str( &name );
- if (!(pipe = create_named_pipe( &name, req->attributes ))) return;
+ if (req->rootdir && !(root = get_directory_obj( current->process, req->rootdir, 0 )))
+ return;
+
+ pipe = create_named_pipe( root, &name, req->attributes | OBJ_OPENIF );
+
+ if (root) release_object( root );
+ if (!pipe) return;
if (get_error() != STATUS_OBJECT_NAME_EXISTS)
{
+ /* initialize it if it didn't already exist */
+ pipe->instances = 0;
+ list_init( &pipe->servers );
+ list_init( &pipe->waiters );
pipe->insize = req->insize;
pipe->outsize = req->outsize;
pipe->maxinstances = req->maxinstances;
@@ -601,11 +696,18 @@
struct pipe_server *server;
struct pipe_client *client;
struct unicode_str name;
+ struct directory *root = NULL;
struct named_pipe *pipe;
int fds[2];
get_req_unicode_str( &name );
- if (!(pipe = open_named_pipe( &name, req->attributes ))) return;
+ if (req->rootdir && !(root = get_directory_obj( current->process, req->rootdir, 0 )))
+ return;
+
+ pipe = open_object_dir( root, &name, req->attributes, &named_pipe_ops );
+
+ if (root) release_object( root );
+ if (!pipe) return;
server = find_server2( pipe, ps_idle_server, ps_wait_open );
release_object( pipe );
@@ -701,7 +803,8 @@
struct unicode_str name;
get_req_unicode_str( &name );
- if (!(pipe = open_named_pipe( &name, OBJ_CASE_INSENSITIVE )))
+ pipe = (struct named_pipe *)find_object( named_pipe_device->pipes, &name, OBJ_CASE_INSENSITIVE );
+ if (!pipe)
{
set_error( STATUS_PIPE_NOT_AVAILABLE );
return;