123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558 |
- /*
- * Server-side handle management
- *
- * Copyright (C) 1998 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 "wine/port.h"
- #include <assert.h>
- #include <limits.h>
- #include <string.h>
- #include <stdarg.h>
- #include <stdio.h>
- #include <stdlib.h>
- #include "windef.h"
- #include "winbase.h"
- #include "handle.h"
- #include "process.h"
- #include "thread.h"
- #include "request.h"
- struct handle_entry
- {
- struct object *ptr; /* object */
- unsigned int access; /* access rights */
- int fd; /* file descriptor (in client process) */
- };
- struct handle_table
- {
- struct object obj; /* object header */
- struct process *process; /* process owning this table */
- int count; /* number of allocated entries */
- int last; /* last used entry */
- int free; /* first entry that may be free */
- struct handle_entry *entries; /* handle entries */
- };
- static struct handle_table *global_table;
- /* reserved handle access rights */
- #define RESERVED_SHIFT 25
- #define RESERVED_INHERIT (HANDLE_FLAG_INHERIT << RESERVED_SHIFT)
- #define RESERVED_CLOSE_PROTECT (HANDLE_FLAG_PROTECT_FROM_CLOSE << RESERVED_SHIFT)
- #define RESERVED_ALL (RESERVED_INHERIT | RESERVED_CLOSE_PROTECT)
- #define MIN_HANDLE_ENTRIES 32
- /* handle to table index conversion */
- /* handles are a multiple of 4 under NT; handle 0 is not used */
- inline static obj_handle_t index_to_handle( int index )
- {
- return (obj_handle_t)((index + 1) << 2);
- }
- inline static int handle_to_index( obj_handle_t handle )
- {
- return ((unsigned int)handle >> 2) - 1;
- }
- /* global handle conversion */
- #define HANDLE_OBFUSCATOR 0x544a4def
- inline static int handle_is_global( obj_handle_t handle)
- {
- return ((unsigned long)handle ^ HANDLE_OBFUSCATOR) < 0x10000;
- }
- inline static obj_handle_t handle_local_to_global( obj_handle_t handle )
- {
- if (!handle) return 0;
- return (obj_handle_t)((unsigned long)handle ^ HANDLE_OBFUSCATOR);
- }
- inline static obj_handle_t handle_global_to_local( obj_handle_t handle )
- {
- return (obj_handle_t)((unsigned long)handle ^ HANDLE_OBFUSCATOR);
- }
- static void handle_table_dump( struct object *obj, int verbose );
- static void handle_table_destroy( struct object *obj );
- static const struct object_ops handle_table_ops =
- {
- sizeof(struct handle_table), /* size */
- handle_table_dump, /* dump */
- no_add_queue, /* add_queue */
- NULL, /* remove_queue */
- NULL, /* signaled */
- NULL, /* satisfied */
- no_get_fd, /* get_fd */
- handle_table_destroy /* destroy */
- };
- /* dump a handle table */
- static void handle_table_dump( struct object *obj, int verbose )
- {
- int i;
- struct handle_table *table = (struct handle_table *)obj;
- struct handle_entry *entry = table->entries;
- assert( obj->ops == &handle_table_ops );
- fprintf( stderr, "Handle table last=%d count=%d process=%p\n",
- table->last, table->count, table->process );
- if (!verbose) return;
- entry = table->entries;
- for (i = 0; i <= table->last; i++, entry++)
- {
- if (!entry->ptr) continue;
- fprintf( stderr, "%9u: %p %08x ",
- (unsigned int)index_to_handle(i), entry->ptr, entry->access );
- entry->ptr->ops->dump( entry->ptr, 0 );
- }
- }
- /* destroy a handle table */
- static void handle_table_destroy( struct object *obj )
- {
- int i;
- struct handle_table *table = (struct handle_table *)obj;
- struct handle_entry *entry = table->entries;
- assert( obj->ops == &handle_table_ops );
- for (i = 0; i <= table->last; i++, entry++)
- {
- struct object *obj = entry->ptr;
- entry->ptr = NULL;
- if (obj) release_object( obj );
- }
- free( table->entries );
- }
- /* allocate a new handle table */
- struct handle_table *alloc_handle_table( struct process *process, int count )
- {
- struct handle_table *table;
- if (count < MIN_HANDLE_ENTRIES) count = MIN_HANDLE_ENTRIES;
- if (!(table = alloc_object( &handle_table_ops )))
- return NULL;
- table->process = process;
- table->count = count;
- table->last = -1;
- table->free = 0;
- if ((table->entries = mem_alloc( count * sizeof(*table->entries) ))) return table;
- release_object( table );
- return NULL;
- }
- /* grow a handle table */
- static int grow_handle_table( struct handle_table *table )
- {
- struct handle_entry *new_entries;
- int count = table->count;
- if (count >= INT_MAX / 2) return 0;
- count *= 2;
- if (!(new_entries = realloc( table->entries, count * sizeof(struct handle_entry) )))
- {
- set_error( STATUS_NO_MEMORY );
- return 0;
- }
- table->entries = new_entries;
- table->count = count;
- return 1;
- }
- /* allocate the first free entry in the handle table */
- static obj_handle_t alloc_entry( struct handle_table *table, void *obj, unsigned int access )
- {
- struct handle_entry *entry = table->entries + table->free;
- int i;
- for (i = table->free; i <= table->last; i++, entry++) if (!entry->ptr) goto found;
- if (i >= table->count)
- {
- if (!grow_handle_table( table )) return 0;
- entry = table->entries + i; /* the entries may have moved */
- }
- table->last = i;
- found:
- table->free = i + 1;
- entry->ptr = grab_object( obj );
- entry->access = access;
- entry->fd = -1;
- return index_to_handle(i);
- }
- /* allocate a handle for an object, incrementing its refcount */
- /* return the handle, or 0 on error */
- obj_handle_t alloc_handle( struct process *process, void *obj, unsigned int access, int inherit )
- {
- struct handle_table *table = process->handles;
- assert( table );
- assert( !(access & RESERVED_ALL) );
- if (inherit) access |= RESERVED_INHERIT;
- return alloc_entry( table, obj, access );
- }
- /* allocate a global handle for an object, incrementing its refcount */
- /* return the handle, or 0 on error */
- static obj_handle_t alloc_global_handle( void *obj, unsigned int access )
- {
- if (!global_table)
- {
- if (!(global_table = (struct handle_table *)alloc_handle_table( NULL, 0 )))
- return 0;
- }
- return handle_local_to_global( alloc_entry( global_table, obj, access ));
- }
- /* return a handle entry, or NULL if the handle is invalid */
- static struct handle_entry *get_handle( struct process *process, obj_handle_t handle )
- {
- struct handle_table *table = process->handles;
- struct handle_entry *entry;
- int index;
- if (handle_is_global(handle))
- {
- handle = handle_global_to_local(handle);
- table = global_table;
- }
- if (!table) goto error;
- index = handle_to_index( handle );
- if (index < 0) goto error;
- if (index > table->last) goto error;
- entry = table->entries + index;
- if (!entry->ptr) goto error;
- return entry;
- error:
- set_error( STATUS_INVALID_HANDLE );
- return NULL;
- }
- /* attempt to shrink a table */
- static void shrink_handle_table( struct handle_table *table )
- {
- struct handle_entry *entry = table->entries + table->last;
- struct handle_entry *new_entries;
- int count = table->count;
- while (table->last >= 0)
- {
- if (entry->ptr) break;
- table->last--;
- entry--;
- }
- if (table->last >= count / 4) return; /* no need to shrink */
- if (count < MIN_HANDLE_ENTRIES * 2) return; /* too small to shrink */
- count /= 2;
- if (!(new_entries = realloc( table->entries, count * sizeof(*new_entries) ))) return;
- table->count = count;
- table->entries = new_entries;
- }
- /* copy the handle table of the parent process */
- /* return 1 if OK, 0 on error */
- struct handle_table *copy_handle_table( struct process *process, struct process *parent )
- {
- struct handle_table *parent_table = parent->handles;
- struct handle_table *table;
- int i;
- assert( parent_table );
- assert( parent_table->obj.ops == &handle_table_ops );
- if (!(table = (struct handle_table *)alloc_handle_table( process, parent_table->count )))
- return NULL;
- if ((table->last = parent_table->last) >= 0)
- {
- struct handle_entry *ptr = table->entries;
- memcpy( ptr, parent_table->entries, (table->last + 1) * sizeof(struct handle_entry) );
- for (i = 0; i <= table->last; i++, ptr++)
- {
- if (!ptr->ptr) continue;
- ptr->fd = -1;
- if (ptr->access & RESERVED_INHERIT) grab_object( ptr->ptr );
- else ptr->ptr = NULL; /* don't inherit this entry */
- }
- }
- /* attempt to shrink the table */
- shrink_handle_table( table );
- return table;
- }
- /* close a handle and decrement the refcount of the associated object */
- /* return 1 if OK, 0 on error */
- int close_handle( struct process *process, obj_handle_t handle, int *fd )
- {
- struct handle_table *table;
- struct handle_entry *entry;
- struct object *obj;
- if (!(entry = get_handle( process, handle ))) return 0;
- if (entry->access & RESERVED_CLOSE_PROTECT)
- {
- set_error( STATUS_INVALID_HANDLE );
- return 0;
- }
- obj = entry->ptr;
- entry->ptr = NULL;
- if (fd) *fd = entry->fd;
- else if (entry->fd != -1) return 1; /* silently ignore close attempt if we cannot close the fd */
- entry->fd = -1;
- table = handle_is_global(handle) ? global_table : process->handles;
- if (entry < table->entries + table->free) table->free = entry - table->entries;
- if (entry == table->entries + table->last) shrink_handle_table( table );
- /* hack: windows seems to treat registry handles differently */
- registry_close_handle( obj, handle );
- release_object( obj );
- return 1;
- }
- /* close all the global handles */
- void close_global_handles(void)
- {
- if (global_table)
- {
- release_object( global_table );
- global_table = NULL;
- }
- }
- /* retrieve the object corresponding to one of the magic pseudo-handles */
- static inline struct object *get_magic_handle( obj_handle_t handle )
- {
- switch((unsigned long)handle)
- {
- case 0xfffffffe: /* current thread pseudo-handle */
- return ¤t->obj;
- case 0x7fffffff: /* current process pseudo-handle */
- case 0xffffffff: /* current process pseudo-handle */
- return (struct object *)current->process;
- default:
- return NULL;
- }
- }
- /* retrieve the object corresponding to a handle, incrementing its refcount */
- struct object *get_handle_obj( struct process *process, obj_handle_t handle,
- unsigned int access, const struct object_ops *ops )
- {
- struct handle_entry *entry;
- struct object *obj;
- if (!(obj = get_magic_handle( handle )))
- {
- if (!(entry = get_handle( process, handle ))) return NULL;
- if ((entry->access & access) != access)
- {
- set_error( STATUS_ACCESS_DENIED );
- return NULL;
- }
- obj = entry->ptr;
- }
- if (ops && (obj->ops != ops))
- {
- set_error( STATUS_OBJECT_TYPE_MISMATCH ); /* not the right type */
- return NULL;
- }
- return grab_object( obj );
- }
- /* retrieve the cached fd for a given handle */
- int get_handle_unix_fd( struct process *process, obj_handle_t handle, unsigned int access )
- {
- struct handle_entry *entry;
- if (!(entry = get_handle( process, handle ))) return -1;
- if ((entry->access & access) != access)
- {
- set_error( STATUS_ACCESS_DENIED );
- return -1;
- }
- return entry->fd;
- }
- /* remove the cached fd and return it */
- int flush_cached_fd( struct process *process, obj_handle_t handle )
- {
- struct handle_entry *entry = get_handle( process, handle );
- int fd = -1;
- if (entry)
- {
- fd = entry->fd;
- entry->fd = -1;
- }
- return fd;
- }
- /* find the first inherited handle of the given type */
- /* this is needed for window stations and desktops (don't ask...) */
- obj_handle_t find_inherited_handle( struct process *process, const struct object_ops *ops )
- {
- struct handle_table *table = process->handles;
- struct handle_entry *ptr;
- int i;
- if (!table) return 0;
- for (i = 0, ptr = table->entries; i <= table->last; i++, ptr++)
- {
- if (!ptr->ptr) continue;
- if (ptr->ptr->ops != ops) continue;
- if (ptr->access & RESERVED_INHERIT) return index_to_handle(i);
- }
- return 0;
- }
- /* get/set the handle reserved flags */
- /* return the old flags (or -1 on error) */
- int set_handle_info( struct process *process, obj_handle_t handle, int mask, int flags, int *fd )
- {
- struct handle_entry *entry;
- unsigned int old_access;
- if (get_magic_handle( handle ))
- {
- /* we can retrieve but not set info for magic handles */
- if (mask) set_error( STATUS_ACCESS_DENIED );
- return 0;
- }
- if (!(entry = get_handle( process, handle ))) return -1;
- old_access = entry->access;
- mask = (mask << RESERVED_SHIFT) & RESERVED_ALL;
- flags = (flags << RESERVED_SHIFT) & mask;
- entry->access = (entry->access & ~mask) | flags;
- /* if no current fd set it, otherwise return current fd */
- if (entry->fd == -1) entry->fd = *fd;
- *fd = entry->fd;
- return (old_access & RESERVED_ALL) >> RESERVED_SHIFT;
- }
- /* duplicate a handle */
- obj_handle_t duplicate_handle( struct process *src, obj_handle_t src_handle, struct process *dst,
- unsigned int access, int inherit, int options )
- {
- obj_handle_t res;
- struct object *obj = get_handle_obj( src, src_handle, 0, NULL );
- if (!obj) return 0;
- if (options & DUP_HANDLE_SAME_ACCESS)
- {
- struct handle_entry *entry = get_handle( src, src_handle );
- if (entry)
- access = entry->access;
- else /* pseudo-handle, give it full access */
- {
- access = STANDARD_RIGHTS_ALL | SPECIFIC_RIGHTS_ALL;
- clear_error();
- }
- }
- access &= ~RESERVED_ALL;
- if (options & DUP_HANDLE_MAKE_GLOBAL)
- res = alloc_global_handle( obj, access );
- else
- res = alloc_handle( dst, obj, access, inherit );
- release_object( obj );
- return res;
- }
- /* open a new handle to an existing object */
- obj_handle_t open_object( const struct namespace *namespace, const WCHAR *name, size_t len,
- const struct object_ops *ops, unsigned int access, int inherit )
- {
- obj_handle_t handle = 0;
- struct object *obj = find_object( namespace, name, len );
- if (obj)
- {
- if (ops && obj->ops != ops)
- set_error( STATUS_OBJECT_TYPE_MISMATCH );
- else
- handle = alloc_handle( current->process, obj, access, inherit );
- release_object( obj );
- }
- else
- set_error( STATUS_OBJECT_NAME_NOT_FOUND );
- return handle;
- }
- /* return the size of the handle table of a given process */
- unsigned int get_handle_table_count( struct process *process )
- {
- return process->handles->count;
- }
- /* close a handle */
- DECL_HANDLER(close_handle)
- {
- close_handle( current->process, req->handle, &reply->fd );
- }
- /* set a handle information */
- DECL_HANDLER(set_handle_info)
- {
- int fd = req->fd;
- if (handle_is_global(req->handle)) fd = -1; /* no fd cache for global handles */
- reply->old_flags = set_handle_info( current->process, req->handle,
- req->mask, req->flags, &fd );
- reply->cur_fd = fd;
- }
- /* duplicate a handle */
- DECL_HANDLER(dup_handle)
- {
- struct process *src, *dst;
- reply->handle = 0;
- reply->fd = -1;
- if ((src = get_process_from_handle( req->src_process, PROCESS_DUP_HANDLE )))
- {
- if (req->options & DUP_HANDLE_MAKE_GLOBAL)
- {
- reply->handle = duplicate_handle( src, req->src_handle, NULL,
- req->access, req->inherit, req->options );
- }
- else if ((dst = get_process_from_handle( req->dst_process, PROCESS_DUP_HANDLE )))
- {
- reply->handle = duplicate_handle( src, req->src_handle, dst,
- req->access, req->inherit, req->options );
- release_object( dst );
- }
- /* close the handle no matter what happened */
- if (req->options & DUP_HANDLE_CLOSE_SOURCE)
- {
- if (src == current->process) close_handle( src, req->src_handle, &reply->fd );
- else close_handle( src, req->src_handle, NULL );
- }
- release_object( src );
- }
- }
|