Started moving functions that deal with Unix file descriptors to a
separate fd object. This will be needed for file locking.
diff --git a/server/fd.c b/server/fd.c
new file mode 100644
index 0000000..1dff6ef
--- /dev/null
+++ b/server/fd.c
@@ -0,0 +1,244 @@
+/*
+ * Server-side file descriptor management
+ *
+ * Copyright (C) 2003 Alexandre Julliard
+ *
+ * This library is free software; you can redistribute it and/or
+ * modify it under the terms of the GNU Lesser General Public
+ * License as published by the Free Software Foundation; either
+ * version 2.1 of the License, or (at your option) any later version.
+ *
+ * This library 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
+ * Lesser General Public License for more details.
+ *
+ * You should have received a copy of the GNU Lesser General Public
+ * License along with this library; if not, write to the Free Software
+ * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
+ */
+
+
+#include "config.h"
+
+#include <assert.h>
+#include <stdio.h>
+#include <string.h>
+#include <stdlib.h>
+#include <unistd.h>
+
+#include "object.h"
+#include "file.h"
+#include "handle.h"
+#include "process.h"
+#include "request.h"
+
+struct fd
+{
+ struct object obj; /* object header */
+ const struct fd_ops *fd_ops; /* file descriptor operations */
+ struct object *user; /* object using this file descriptor */
+ int unix_fd; /* unix file descriptor */
+ int mode; /* file protection mode */
+};
+
+static void fd_dump( struct object *obj, int verbose );
+static void fd_destroy( struct object *obj );
+
+static const struct object_ops fd_ops =
+{
+ sizeof(struct fd), /* size */
+ fd_dump, /* dump */
+ no_add_queue, /* add_queue */
+ NULL, /* remove_queue */
+ NULL, /* signaled */
+ NULL, /* satisfied */
+ default_get_fd, /* get_fd */
+ no_get_file_info, /* get_file_info */
+ fd_destroy /* destroy */
+};
+
+
+static void fd_dump( struct object *obj, int verbose )
+{
+ struct fd *fd = (struct fd *)obj;
+ fprintf( stderr, "Fd unix_fd=%d mode=%06o user=%p\n", fd->unix_fd, fd->mode, fd->user );
+}
+
+static void fd_destroy( struct object *obj )
+{
+#if 0
+ struct fd *fd = (struct fd *)obj;
+ close( fd->unix_fd );
+#endif
+}
+
+/* allocate an object that has an associated fd */
+void *alloc_fd_object( const struct object_ops *ops,
+ const struct fd_ops *fd_user_ops, int unix_fd )
+{
+ struct object *user;
+ struct fd *fd = alloc_object( &fd_ops, -1 );
+
+ if (!fd) return NULL;
+ if (!(user = alloc_object( ops, unix_fd )))
+ {
+ release_object( fd );
+ return NULL;
+ }
+ fd->fd_ops = fd_user_ops;
+ fd->user = user;
+ fd->unix_fd = unix_fd;
+ fd->mode = 0;
+
+ user->fd_obj = fd;
+ return user;
+}
+
+/* retrieve the unix fd for an object */
+int get_unix_fd( struct object *obj )
+{
+ struct fd *fd = obj->ops->get_fd( obj );
+ int unix_fd = -1;
+
+ if (fd)
+ {
+ unix_fd = fd->unix_fd;
+ release_object( fd );
+ }
+ return unix_fd;
+}
+
+/* set the unix fd for an object; can only be done once */
+void set_unix_fd( struct object *obj, int unix_fd )
+{
+ struct fd *fd = obj->fd_obj;
+
+ assert( fd );
+ assert( fd->unix_fd == -1 );
+
+ fd->unix_fd = unix_fd;
+ obj->fd = unix_fd;
+}
+
+/* close a file descriptor */
+void close_fd( struct fd *fd )
+{
+ release_object( fd );
+}
+
+/* callback for event happening in the main poll() loop */
+void fd_poll_event( struct object *obj, int event )
+{
+ struct fd *fd = obj->fd_obj;
+ return fd->fd_ops->poll_event( fd->user, event );
+}
+
+/* default add_queue() routine for objects that poll() on an fd */
+int default_fd_add_queue( struct object *obj, struct wait_queue_entry *entry )
+{
+ struct fd *fd = obj->fd_obj;
+
+ if (!obj->head) /* first on the queue */
+ set_select_events( obj, fd->fd_ops->get_poll_events( fd->user ) );
+ add_queue( obj, entry );
+ return 1;
+}
+
+/* default remove_queue() routine for objects that poll() on an fd */
+void default_fd_remove_queue( struct object *obj, struct wait_queue_entry *entry )
+{
+ grab_object( obj );
+ remove_queue( obj, entry );
+ if (!obj->head) /* last on the queue is gone */
+ set_select_events( obj, 0 );
+ release_object( obj );
+}
+
+/* default signaled() routine for objects that poll() on an fd */
+int default_fd_signaled( struct object *obj, struct thread *thread )
+{
+ struct fd *fd = obj->fd_obj;
+ int events = fd->fd_ops->get_poll_events( obj );
+
+ if (check_select_events( fd->unix_fd, events ))
+ {
+ /* stop waiting on select() if we are signaled */
+ set_select_events( obj, 0 );
+ return 1;
+ }
+ /* restart waiting on select() if we are no longer signaled */
+ if (obj->head) set_select_events( obj, events );
+ return 0;
+}
+
+/* default flush() routine */
+int no_flush( struct object *obj )
+{
+ set_error( STATUS_OBJECT_TYPE_MISMATCH );
+ return 0;
+}
+
+/* default queue_async() routine */
+void no_queue_async( struct object *obj, void* ptr, unsigned int status, int type, int count )
+{
+ set_error( STATUS_OBJECT_TYPE_MISMATCH );
+}
+
+/* same as get_handle_obj but retrieve the struct fd associated to the object */
+static struct fd *get_handle_fd_obj( struct process *process, obj_handle_t handle,
+ unsigned int access )
+{
+ struct fd *fd = NULL;
+ struct object *obj;
+
+ if ((obj = get_handle_obj( process, handle, 0, NULL )))
+ {
+ if (obj->fd_obj) fd = (struct fd *)grab_object( obj->fd_obj );
+ else set_error( STATUS_OBJECT_TYPE_MISMATCH );
+ release_object( obj );
+ }
+ return fd;
+}
+
+
+/* flush a file buffers */
+DECL_HANDLER(flush_file)
+{
+ struct fd *fd = get_handle_fd_obj( current->process, req->handle, 0 );
+
+ if (fd)
+ {
+ fd->fd_ops->flush( fd->user );
+ release_object( fd );
+ }
+}
+
+/* create / reschedule an async I/O */
+DECL_HANDLER(register_async)
+{
+ struct fd *fd = get_handle_fd_obj( current->process, req->handle, 0 );
+
+/*
+ * The queue_async method must do the following:
+ *
+ * 1. Get the async_queue for the request of given type.
+ * 2. Call find_async() to look for the specific client request in the queue (=> NULL if not found).
+ * 3. If status is STATUS_PENDING:
+ * a) If no async request found in step 2 (new request): call create_async() to initialize one.
+ * b) Set request's status to STATUS_PENDING.
+ * c) If the "queue" field of the async request is NULL: call async_insert() to put it into the queue.
+ * Otherwise:
+ * If the async request was found in step 2, destroy it by calling destroy_async().
+ * 4. Carry out any operations necessary to adjust the object's poll events
+ * Usually: set_elect_events (obj, obj->ops->get_poll_events()).
+ *
+ * See also the implementations in file.c, serial.c, and sock.c.
+*/
+
+ if (fd)
+ {
+ fd->fd_ops->queue_async( fd->user, req->overlapped, req->status, req->type, req->count );
+ release_object( fd );
+ }
+}