blob: 81d4ba70ec2ccc26eb1e63a9a3ee60b900722898 [file] [log] [blame]
Alexandre Julliardd30dfd21998-09-27 18:28:36 +00001/*
2 * Server-side event management
3 *
4 * Copyright (C) 1998 Alexandre Julliard
5 */
6
7#include <assert.h>
8#include <stdio.h>
9#include <stdlib.h>
10
11#include "winerror.h"
12#include "winnt.h"
Alexandre Julliard43c190e1999-05-15 10:48:19 +000013
14#include "handle.h"
15#include "thread.h"
Alexandre Julliard5bc78081999-06-22 17:26:53 +000016#include "request.h"
Alexandre Julliardd30dfd21998-09-27 18:28:36 +000017
18struct event
19{
20 struct object obj; /* object header */
21 int manual_reset; /* is it a manual reset event? */
22 int signaled; /* event has been signaled */
23};
24
Alexandre Julliard338e7571998-12-27 15:28:54 +000025static void event_dump( struct object *obj, int verbose );
Alexandre Julliardd30dfd21998-09-27 18:28:36 +000026static int event_signaled( struct object *obj, struct thread *thread );
27static int event_satisfied( struct object *obj, struct thread *thread );
Alexandre Julliardd30dfd21998-09-27 18:28:36 +000028
29static const struct object_ops event_ops =
30{
Alexandre Julliard5bc78081999-06-22 17:26:53 +000031 sizeof(struct event),
Alexandre Julliard338e7571998-12-27 15:28:54 +000032 event_dump,
Alexandre Julliardc6e45ed1998-12-27 08:35:39 +000033 add_queue,
34 remove_queue,
Alexandre Julliardd30dfd21998-09-27 18:28:36 +000035 event_signaled,
36 event_satisfied,
Alexandre Julliardaa0ebd01998-12-30 12:06:45 +000037 no_read_fd,
38 no_write_fd,
39 no_flush,
Alexandre Julliard05625391999-01-03 11:55:56 +000040 no_get_file_info,
Alexandre Julliard5bc78081999-06-22 17:26:53 +000041 no_destroy
Alexandre Julliardd30dfd21998-09-27 18:28:36 +000042};
43
44
Alexandre Julliard5bc78081999-06-22 17:26:53 +000045static struct event *create_event( const char *name, size_t len,
46 int manual_reset, int initial_state )
Alexandre Julliardd30dfd21998-09-27 18:28:36 +000047{
48 struct event *event;
49
Alexandre Julliard5bc78081999-06-22 17:26:53 +000050 if ((event = create_named_object( &event_ops, name, len )))
Alexandre Julliardd30dfd21998-09-27 18:28:36 +000051 {
Alexandre Julliard5bc78081999-06-22 17:26:53 +000052 if (get_error() != ERROR_ALREADY_EXISTS)
53 {
54 /* initialize it if it didn't already exist */
55 event->manual_reset = manual_reset;
56 event->signaled = initial_state;
57 }
Alexandre Julliardd30dfd21998-09-27 18:28:36 +000058 }
Alexandre Julliard5bc78081999-06-22 17:26:53 +000059 return event;
Alexandre Julliardd30dfd21998-09-27 18:28:36 +000060}
61
Alexandre Julliard43c190e1999-05-15 10:48:19 +000062static int pulse_event( int handle )
Alexandre Julliardd30dfd21998-09-27 18:28:36 +000063{
64 struct event *event;
65
66 if (!(event = (struct event *)get_handle_obj( current->process, handle,
67 EVENT_MODIFY_STATE, &event_ops )))
68 return 0;
69 event->signaled = 1;
70 /* wake up all waiters if manual reset, a single one otherwise */
71 wake_up( &event->obj, !event->manual_reset );
72 event->signaled = 0;
73 release_object( event );
74 return 1;
75}
76
Alexandre Julliard43c190e1999-05-15 10:48:19 +000077static int set_event( int handle )
Alexandre Julliardd30dfd21998-09-27 18:28:36 +000078{
79 struct event *event;
80
81 if (!(event = (struct event *)get_handle_obj( current->process, handle,
82 EVENT_MODIFY_STATE, &event_ops )))
83 return 0;
84 event->signaled = 1;
85 /* wake up all waiters if manual reset, a single one otherwise */
86 wake_up( &event->obj, !event->manual_reset );
87 release_object( event );
88 return 1;
89}
90
Alexandre Julliard43c190e1999-05-15 10:48:19 +000091static int reset_event( int handle )
Alexandre Julliardd30dfd21998-09-27 18:28:36 +000092{
93 struct event *event;
94
95 if (!(event = (struct event *)get_handle_obj( current->process, handle,
96 EVENT_MODIFY_STATE, &event_ops )))
97 return 0;
98 event->signaled = 0;
99 release_object( event );
100 return 1;
101}
102
Alexandre Julliard338e7571998-12-27 15:28:54 +0000103static void event_dump( struct object *obj, int verbose )
Alexandre Julliardd30dfd21998-09-27 18:28:36 +0000104{
105 struct event *event = (struct event *)obj;
106 assert( obj->ops == &event_ops );
Alexandre Julliard05625391999-01-03 11:55:56 +0000107 fprintf( stderr, "Event manual=%d signaled=%d name='%s'\n",
108 event->manual_reset, event->signaled,
109 get_object_name( &event->obj ) );
Alexandre Julliardd30dfd21998-09-27 18:28:36 +0000110}
111
112static int event_signaled( struct object *obj, struct thread *thread )
113{
114 struct event *event = (struct event *)obj;
115 assert( obj->ops == &event_ops );
116 return event->signaled;
117}
118
119static int event_satisfied( struct object *obj, struct thread *thread )
120{
121 struct event *event = (struct event *)obj;
122 assert( obj->ops == &event_ops );
123 /* Reset if it's an auto-reset event */
124 if (!event->manual_reset) event->signaled = 0;
125 return 0; /* Not abandoned */
126}
127
Alexandre Julliard43c190e1999-05-15 10:48:19 +0000128/* create an event */
129DECL_HANDLER(create_event)
130{
Alexandre Julliardebe29ef1999-06-26 08:43:26 +0000131 size_t len = get_req_strlen( req->name );
Alexandre Julliard5bc78081999-06-22 17:26:53 +0000132 struct event *event;
Alexandre Julliard43c190e1999-05-15 10:48:19 +0000133
Alexandre Julliardebe29ef1999-06-26 08:43:26 +0000134 req->handle = -1;
135 if ((event = create_event( req->name, len, req->manual_reset, req->initial_state )))
Alexandre Julliard43c190e1999-05-15 10:48:19 +0000136 {
Alexandre Julliardebe29ef1999-06-26 08:43:26 +0000137 req->handle = alloc_handle( current->process, event, EVENT_ALL_ACCESS, req->inherit );
Alexandre Julliard5bc78081999-06-22 17:26:53 +0000138 release_object( event );
Alexandre Julliard43c190e1999-05-15 10:48:19 +0000139 }
Alexandre Julliard43c190e1999-05-15 10:48:19 +0000140}
141
142/* open a handle to an event */
143DECL_HANDLER(open_event)
144{
Alexandre Julliardebe29ef1999-06-26 08:43:26 +0000145 size_t len = get_req_strlen( req->name );
146 req->handle = open_object( req->name, len, &event_ops, req->access, req->inherit );
Alexandre Julliard43c190e1999-05-15 10:48:19 +0000147}
148
149/* do an event operation */
150DECL_HANDLER(event_op)
151{
152 switch(req->op)
153 {
154 case PULSE_EVENT:
155 pulse_event( req->handle );
156 break;
157 case SET_EVENT:
158 set_event( req->handle );
159 break;
160 case RESET_EVENT:
161 reset_event( req->handle );
162 break;
163 default:
Alexandre Julliardebe29ef1999-06-26 08:43:26 +0000164 fatal_protocol_error( current, "event_op: invalid operation %d\n", req->op );
Alexandre Julliard43c190e1999-05-15 10:48:19 +0000165 }
Alexandre Julliard43c190e1999-05-15 10:48:19 +0000166}