blob: 94fe31fde5eadfdb785bf84a0cad6b7d3ef722d6 [file] [log] [blame]
Alexandre Julliardd30dfd21998-09-27 18:28:36 +00001/*
2 * Server-side semaphore management
3 *
4 * Copyright (C) 1998 Alexandre Julliard
5 */
6
7#include <assert.h>
8#include <stdio.h>
9#include <stdlib.h>
10
Alexandre Julliardd30dfd21998-09-27 18:28:36 +000011#include "winnt.h"
Alexandre Julliard43c190e1999-05-15 10:48:19 +000012
13#include "handle.h"
14#include "thread.h"
Alexandre Julliard5bc78081999-06-22 17:26:53 +000015#include "request.h"
Alexandre Julliardd30dfd21998-09-27 18:28:36 +000016
17struct semaphore
18{
19 struct object obj; /* object header */
20 unsigned int count; /* current count */
21 unsigned int max; /* maximum possible count */
22};
23
Alexandre Julliard338e7571998-12-27 15:28:54 +000024static void semaphore_dump( struct object *obj, int verbose );
Alexandre Julliardd30dfd21998-09-27 18:28:36 +000025static int semaphore_signaled( struct object *obj, struct thread *thread );
26static int semaphore_satisfied( struct object *obj, struct thread *thread );
Alexandre Julliardd30dfd21998-09-27 18:28:36 +000027
28static const struct object_ops semaphore_ops =
29{
Alexandre Julliard1dca5e22000-01-01 00:56:27 +000030 sizeof(struct semaphore), /* size */
31 semaphore_dump, /* dump */
32 add_queue, /* add_queue */
33 remove_queue, /* remove_queue */
34 semaphore_signaled, /* signaled */
35 semaphore_satisfied, /* satisfied */
36 NULL, /* get_poll_events */
37 NULL, /* poll_event */
38 no_read_fd, /* get_read_fd */
39 no_write_fd, /* get_write_fd */
40 no_flush, /* flush */
41 no_get_file_info, /* get_file_info */
42 no_destroy /* destroy */
Alexandre Julliardd30dfd21998-09-27 18:28:36 +000043};
44
45
Alexandre Julliardd16319c1999-11-25 21:30:24 +000046static struct semaphore *create_semaphore( const WCHAR *name, size_t len,
Alexandre Julliard5bc78081999-06-22 17:26:53 +000047 unsigned int initial, unsigned int max )
Alexandre Julliardd30dfd21998-09-27 18:28:36 +000048{
49 struct semaphore *sem;
50
51 if (!max || (initial > max))
52 {
Alexandre Julliardcb1fc732000-01-24 21:58:06 +000053 set_error( STATUS_INVALID_PARAMETER );
Alexandre Julliardd30dfd21998-09-27 18:28:36 +000054 return NULL;
55 }
Alexandre Julliard5bc78081999-06-22 17:26:53 +000056 if ((sem = create_named_object( &semaphore_ops, name, len )))
Alexandre Julliardd30dfd21998-09-27 18:28:36 +000057 {
Alexandre Julliardcb1fc732000-01-24 21:58:06 +000058 if (get_error() != STATUS_OBJECT_NAME_COLLISION)
Alexandre Julliard5bc78081999-06-22 17:26:53 +000059 {
60 /* initialize it if it didn't already exist */
61 sem->count = initial;
62 sem->max = max;
63 }
Alexandre Julliardd30dfd21998-09-27 18:28:36 +000064 }
Alexandre Julliard5bc78081999-06-22 17:26:53 +000065 return sem;
Alexandre Julliardd30dfd21998-09-27 18:28:36 +000066}
67
Alexandre Julliardebe29ef1999-06-26 08:43:26 +000068static unsigned int release_semaphore( int handle, unsigned int count )
Alexandre Julliardd30dfd21998-09-27 18:28:36 +000069{
70 struct semaphore *sem;
Alexandre Julliardebe29ef1999-06-26 08:43:26 +000071 unsigned int prev = 0;
Alexandre Julliardd30dfd21998-09-27 18:28:36 +000072
Alexandre Julliardebe29ef1999-06-26 08:43:26 +000073 if ((sem = (struct semaphore *)get_handle_obj( current->process, handle,
74 SEMAPHORE_MODIFY_STATE, &semaphore_ops )))
Alexandre Julliardd30dfd21998-09-27 18:28:36 +000075 {
Alexandre Julliardebe29ef1999-06-26 08:43:26 +000076 prev = sem->count;
77 if (sem->count + count < sem->count || sem->count + count > sem->max)
78 {
Alexandre Julliardcb1fc732000-01-24 21:58:06 +000079 set_error( STATUS_SEMAPHORE_LIMIT_EXCEEDED );
Alexandre Julliardebe29ef1999-06-26 08:43:26 +000080 }
81 else if (sem->count)
82 {
83 /* there cannot be any thread waiting if the count is != 0 */
84 assert( !sem->obj.head );
85 sem->count += count;
86 }
87 else
88 {
89 sem->count = count;
90 wake_up( &sem->obj, count );
91 }
92 release_object( sem );
Alexandre Julliardd30dfd21998-09-27 18:28:36 +000093 }
Alexandre Julliardebe29ef1999-06-26 08:43:26 +000094 return prev;
Alexandre Julliardd30dfd21998-09-27 18:28:36 +000095}
96
Alexandre Julliard338e7571998-12-27 15:28:54 +000097static void semaphore_dump( struct object *obj, int verbose )
Alexandre Julliardd30dfd21998-09-27 18:28:36 +000098{
99 struct semaphore *sem = (struct semaphore *)obj;
100 assert( obj->ops == &semaphore_ops );
Alexandre Julliardd16319c1999-11-25 21:30:24 +0000101 fprintf( stderr, "Semaphore count=%d max=%d ", sem->count, sem->max );
102 dump_object_name( &sem->obj );
103 fputc( '\n', stderr );
Alexandre Julliardd30dfd21998-09-27 18:28:36 +0000104}
105
106static int semaphore_signaled( struct object *obj, struct thread *thread )
107{
108 struct semaphore *sem = (struct semaphore *)obj;
109 assert( obj->ops == &semaphore_ops );
110 return (sem->count > 0);
111}
112
113static int semaphore_satisfied( struct object *obj, struct thread *thread )
114{
115 struct semaphore *sem = (struct semaphore *)obj;
116 assert( obj->ops == &semaphore_ops );
117 assert( sem->count );
118 sem->count--;
119 return 0; /* not abandoned */
120}
121
Alexandre Julliard43c190e1999-05-15 10:48:19 +0000122/* create a semaphore */
123DECL_HANDLER(create_semaphore)
124{
Alexandre Julliardef886372000-04-04 19:33:27 +0000125 size_t len = get_req_strlenW( req, req->name );
Alexandre Julliard5bc78081999-06-22 17:26:53 +0000126 struct semaphore *sem;
Alexandre Julliard43c190e1999-05-15 10:48:19 +0000127
Alexandre Julliardebe29ef1999-06-26 08:43:26 +0000128 req->handle = -1;
129 if ((sem = create_semaphore( req->name, len, req->initial, req->max )))
Alexandre Julliard43c190e1999-05-15 10:48:19 +0000130 {
Alexandre Julliardebe29ef1999-06-26 08:43:26 +0000131 req->handle = alloc_handle( current->process, sem, SEMAPHORE_ALL_ACCESS, req->inherit );
Alexandre Julliard5bc78081999-06-22 17:26:53 +0000132 release_object( sem );
Alexandre Julliard43c190e1999-05-15 10:48:19 +0000133 }
Alexandre Julliard43c190e1999-05-15 10:48:19 +0000134}
135
136/* open a handle to a semaphore */
137DECL_HANDLER(open_semaphore)
138{
Alexandre Julliardef886372000-04-04 19:33:27 +0000139 size_t len = get_req_strlenW( req, req->name );
Alexandre Julliardebe29ef1999-06-26 08:43:26 +0000140 req->handle = open_object( req->name, len, &semaphore_ops, req->access, req->inherit );
Alexandre Julliard43c190e1999-05-15 10:48:19 +0000141}
142
143/* release a semaphore */
144DECL_HANDLER(release_semaphore)
145{
Alexandre Julliardebe29ef1999-06-26 08:43:26 +0000146 req->prev_count = release_semaphore( req->handle, req->count );
Alexandre Julliard43c190e1999-05-15 10:48:19 +0000147}