2007-05-01 Dick Porter <dick@ximian.com>
[mono.git] / mcs / class / corlib / System.Threading / RegisteredWaitHandle.cs
1 //
2 // System.Threading.RegisteredWaitHandle.cs
3 //
4 // Author:
5 //   Dick Porter (dick@ximian.com)
6 //   Lluis Sanchez Gual (lluis@ideary.com)
7 //
8 // (C) Ximian, Inc.  http://www.ximian.com
9 //
10
11 //
12 // Copyright (C) 2004, 2005 Novell, Inc (http://www.novell.com)
13 //
14 // Permission is hereby granted, free of charge, to any person obtaining
15 // a copy of this software and associated documentation files (the
16 // "Software"), to deal in the Software without restriction, including
17 // without limitation the rights to use, copy, modify, merge, publish,
18 // distribute, sublicense, and/or sell copies of the Software, and to
19 // permit persons to whom the Software is furnished to do so, subject to
20 // the following conditions:
21 // 
22 // The above copyright notice and this permission notice shall be
23 // included in all copies or substantial portions of the Software.
24 // 
25 // THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
26 // EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
27 // MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
28 // NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE
29 // LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION
30 // OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION
31 // WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
32 //
33
34 #if NET_2_0
35 using System.Runtime.InteropServices;
36 #endif
37
38 namespace System.Threading
39 {
40 #if NET_2_0
41         [ComVisible (true)]
42 #endif
43         public sealed class RegisteredWaitHandle : MarshalByRefObject
44         {
45                 WaitHandle _waitObject;
46                 WaitOrTimerCallback _callback;
47                 TimeSpan _timeout;
48                 object _state;
49                 bool _executeOnlyOnce;
50                 WaitHandle _finalEvent;
51                 ManualResetEvent _cancelEvent;
52                 int _callsInProcess;
53                 bool _unregistered;
54
55                 internal RegisteredWaitHandle (WaitHandle waitObject, WaitOrTimerCallback callback, object state, TimeSpan timeout, bool executeOnlyOnce)
56                 {
57                         _waitObject = waitObject;
58                         _callback = callback;
59                         _state = state;
60                         _timeout = timeout;
61                         _executeOnlyOnce = executeOnlyOnce;
62                         _finalEvent = null;
63                         _cancelEvent = new ManualResetEvent (false);
64                         _callsInProcess = 0;
65                         _unregistered = false;
66                 }
67
68                 internal void Wait (object state)
69                 {
70                         try
71                         {
72                                 WaitHandle[] waits = new WaitHandle[] {_waitObject, _cancelEvent};
73                                 do 
74                                 {
75                                         int signal = WaitHandle.WaitAny (waits, _timeout, false);
76                                         if (!_unregistered)
77                                         {
78                                                 lock (this) { _callsInProcess++; }
79                                                 ThreadPool.QueueUserWorkItem (new WaitCallback (DoCallBack), (signal == WaitHandle.WaitTimeout));
80                                         }
81                                 } 
82                                 while (!_unregistered && !_executeOnlyOnce);
83                         }
84                         catch {}
85
86                         lock (this) {
87                                 _unregistered = true;
88                                 if (_callsInProcess == 0 && _finalEvent != null)
89                                         NativeEventCalls.SetEvent_internal (_finalEvent.Handle);
90                         }
91                 }
92
93                 private void DoCallBack (object timedOut)
94                 {
95                         if (_callback != null)
96                                 _callback (_state, (bool)timedOut); 
97
98                         lock (this) 
99                         {
100                                 _callsInProcess--;
101                                 if (_unregistered && _callsInProcess == 0 && _finalEvent != null)
102                                         NativeEventCalls.SetEvent_internal (_finalEvent.Handle);
103                         }
104                 }
105
106 #if NET_2_0
107                 [ComVisible (true)]
108 #endif
109                 public bool Unregister(WaitHandle waitObject) 
110                 {
111                         lock (this) 
112                         {
113                                 if (_unregistered) return false;
114                                 _finalEvent = waitObject;
115                                 _unregistered = true;
116                                 _cancelEvent.Set();
117                                 return true;
118                         }
119                 }
120
121 #if ONLY_1_1
122                 [MonoTODO]
123                 ~RegisteredWaitHandle() {
124                         // FIXME
125                 }
126 #endif
127         }
128 }