| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240 |
- //
- // System.Threading.Timer.cs
- //
- // Authors:
- // Dick Porter ([email protected])
- // Gonzalo Paniagua Javier ([email protected])
- //
- // (C) 2001, 2002 Ximian, Inc. http://www.ximian.com
- //
- namespace System.Threading
- {
- public sealed class Timer : MarshalByRefObject, IDisposable
- {
- sealed class Runner : MarshalByRefObject, IDisposable
- {
- ManualResetEvent wait;
- AutoResetEvent start_event;
- TimerCallback callback;
- object state;
- int dueTime;
- int period;
- bool disposed;
- bool aborted;
- public Runner (TimerCallback callback, object state, AutoResetEvent start_event)
- {
- this.callback = callback;
- this.state = state;
- this.start_event = start_event;
- this.wait = new ManualResetEvent (false);
- }
- public int DueTime {
- get { return dueTime; }
- set { dueTime = value; }
- }
- public int Period {
- get { return period; }
- set { period = value == 0 ? Timeout.Infinite : value; }
- }
- bool WaitForDueTime ()
- {
- if (dueTime > 0) {
- bool signaled;
- do {
- wait.Reset ();
- signaled = wait.WaitOne (dueTime, false);
- } while (signaled == true && !disposed && !aborted);
- if (!signaled)
- callback (state);
- if (disposed)
- return false;
- }
- return true;
- }
- public void Abort ()
- {
- lock (this) {
- aborted = true;
- wait.Set ();
- }
- }
- public void Start ()
- {
- while (start_event.WaitOne ()) {
- aborted = false;
- if (dueTime == Timeout.Infinite)
- continue;
- if (!WaitForDueTime ())
- return;
- if (aborted || (period == Timeout.Infinite))
- continue;
- bool signaled = false;
- while (true) {
- wait.Reset ();
- signaled = wait.WaitOne (period, false);
- if (disposed)
- return;
- if (aborted)
- break;
- if (!signaled) {
- callback (state);
- } else if (!WaitForDueTime ()) {
- return;
- }
- }
- }
- }
- public void Dispose ()
- {
- Dispose (true);
- }
- void Dispose (bool disposing)
- {
- disposed = true;
- if (wait != null) {
- wait.Set ();
- Thread.Sleep (100);
- ((IDisposable) wait).Dispose ();
- wait = null;
- }
- if (disposing)
- GC.SuppressFinalize (this);
- }
- ~Runner ()
- {
- Dispose (false);
- }
- }
- Runner runner;
- AutoResetEvent start_event;
- public Timer (TimerCallback callback, object state, int dueTime, int period)
- {
- if (dueTime < -1)
- throw new ArgumentOutOfRangeException ("dueTime");
- if (period < -1)
- throw new ArgumentOutOfRangeException ("period");
- Init (callback, state, dueTime, period);
- }
- public Timer (TimerCallback callback, object state, long dueTime, long period)
- {
- if (dueTime < -1)
- throw new ArgumentOutOfRangeException ("dueTime");
- if (period < -1)
- throw new ArgumentOutOfRangeException ("period");
- Init (callback, state, (int) dueTime, (int) period);
- }
- public Timer (TimerCallback callback, object state, TimeSpan dueTime, TimeSpan period)
- : this (callback, state, Convert.ToInt32(dueTime.TotalMilliseconds), Convert.ToInt32(period.TotalMilliseconds))
- {
- }
- [CLSCompliant(false)]
- public Timer (TimerCallback callback, object state, uint dueTime, uint period)
- : this (callback, state, (long) dueTime, (long) period)
- {
- }
- void Init (TimerCallback callback, object state, int dueTime, int period)
- {
- start_event = new AutoResetEvent (false);
- runner = new Runner (callback, state, start_event);
- Change (dueTime, period);
- Thread t = new Thread (new ThreadStart (runner.Start));
- t.IsBackground = true;
- t.Start ();
- }
- [MonoTODO("false return?")]
- public bool Change (int dueTime, int period)
- {
- if (dueTime < -1)
- throw new ArgumentOutOfRangeException ("dueTime");
- if (period < -1)
- throw new ArgumentOutOfRangeException ("period");
- runner.DueTime = dueTime;
- runner.Period = period;
- runner.Abort ();
- start_event.Set ();
- return true;
- }
- public bool Change (long dueTime, long period)
- {
- if(dueTime > 4294967294)
- throw new NotSupportedException ("Due time too large");
- if(period > 4294967294)
- throw new NotSupportedException ("Period too large");
- return Change ((int) dueTime, (int) period);
- }
- public bool Change (TimeSpan dueTime, TimeSpan period)
- {
- return Change (Convert.ToInt32(dueTime.TotalMilliseconds), Convert.ToInt32(period.TotalMilliseconds));
- }
- [CLSCompliant(false)]
- public bool Change (uint dueTime, uint period)
- {
- if (dueTime > Int32.MaxValue)
- throw new NotSupportedException ("Due time too large");
- if (period > Int32.MaxValue)
- throw new NotSupportedException ("Period too large");
- return Change ((int) dueTime, (int) period);
- }
- public void Dispose ()
- {
- runner.Dispose ();
- runner = null;
- GC.SuppressFinalize (this);
- }
- [MonoTODO("How do we signal the handler?")]
- public bool Dispose (WaitHandle notifyObject)
- {
- Dispose ();
- return true; //FIXME
- }
- ~Timer ()
- {
- runner = null;
- }
- }
- }
|