2 // UnixSignalTest.cs - NUnit Test Cases for Mono.Unix.UnixSignal
5 // Jonathan Pryor <jonpryor@vt.edu>
7 // (C) 2008 Jonathan Pryor
10 using NUnit.Framework;
12 using NUnit.Framework.SyntaxHelpers;
16 using System.Threading;
18 using Mono.Unix.Android;
19 using Mono.Unix.Native;
21 namespace NUnit.Framework.SyntaxHelpers { class Dummy {} }
23 namespace MonoTests.Mono.Unix {
26 public class UnixSignalTest {
28 // helper method to create a thread waiting on a UnixSignal
29 static Thread CreateWaitSignalThread (UnixSignal signal, int timeout)
31 Thread t1 = new Thread(delegate() {
32 DateTime start = DateTime.Now;
33 bool r = signal.WaitOne (timeout, false);
34 DateTime end = DateTime.Now;
35 Assert.AreEqual (signal.Count, 1);
36 Assert.AreEqual (r, true);
37 if ((end - start) > new TimeSpan (0, 0, timeout/1000))
38 throw new InvalidOperationException ("Signal slept too long");
43 // helper method to create a two-thread test
44 static void MultiThreadTest (UnixSignal signal, int timeout, ThreadStart tstart)
46 Thread t1 = CreateWaitSignalThread (signal, timeout);
47 Thread t2 = new Thread (tstart);
55 public void TestNestedInvocation()
57 UnixSignal s = new UnixSignal(Signum.SIGINT);
58 Thread a = new Thread(delegate() {
59 bool r = s.WaitOne (1000, false);
61 Thread b = new Thread(delegate() {
62 bool r = s.WaitOne (500, false);
71 public void TestWaitAnyFailsWithMore64Signals()
73 UnixSignal s1 = new UnixSignal(Signum.SIGINT);
74 UnixSignal[] signals = new UnixSignal[65];
75 for (int i=0; i<65; ++i)
78 Assert.That(UnixSignal.WaitAny(signals, new TimeSpan(0,0,1)), Is.EqualTo(-1));
82 public void TestConcurrentWaitOne()
84 UnixSignal s1 = new UnixSignal(Signum.SIGINT);
85 UnixSignal s2 = new UnixSignal(Signum.SIGINT);
86 Thread a = CreateWaitSignalThread(s1, 10000);
87 Thread b = CreateWaitSignalThread(s2, 5000);
88 Thread c = new Thread (delegate () {
90 Stdlib.raise (Signum.SIGINT);
98 Assert.That(s1.Count, Is.EqualTo(1), "Expected 1 signal raised");
99 Assert.That(s2.Count, Is.EqualTo(1), "Expected 1 signal raised");
103 public void TestConcurrentWaitOneSameInstance()
105 UnixSignal s1 = new UnixSignal(Signum.SIGINT);
106 Thread a = CreateWaitSignalThread(s1, 10000);
107 Thread b = CreateWaitSignalThread(s1, 10000);
108 Thread c = new Thread (delegate () {
110 Stdlib.raise (Signum.SIGINT);
121 [Category ("AndroidNotWorking")] // Crashes (silently) the runtime in similar fashion to real-time signals
122 public void TestSignumProperty ()
124 UnixSignal signal1 = new UnixSignal (Signum.SIGSEGV);
125 Assert.That (signal1.Signum, Is.EqualTo (Signum.SIGSEGV));
129 [Category ("NotOnMac")]
130 public void TestRealTimeCstor ()
132 if (!TestHelper.CanUseRealTimeSignals ())
134 RealTimeSignum rts = new RealTimeSignum (0);
135 using (UnixSignal s = new UnixSignal (rts))
137 Assert.That(s.IsRealTimeSignal);
138 Assert.That(s.RealTimeSignum, Is.EqualTo (rts));
144 [Category ("NotOnMac")]
145 public void TestSignumPropertyThrows ()
147 if (!TestHelper.CanUseRealTimeSignals ())
149 UnixSignal signal1 = new UnixSignal (new RealTimeSignum (0));
150 Signum s = signal1.Signum;
154 [Category ("NotOnMac")]
155 public void TestRealTimeSignumProperty ()
157 if (!TestHelper.CanUseRealTimeSignals ())
159 RealTimeSignum rts = new RealTimeSignum (0);
160 UnixSignal signal1 = new UnixSignal (rts);
161 Assert.That (signal1.RealTimeSignum, Is.EqualTo (rts));
166 [Category ("NotOnMac")]
167 public void TestRealTimePropertyThrows ()
169 if (!TestHelper.CanUseRealTimeSignals ())
171 UnixSignal signal1 = new UnixSignal (Signum.SIGSEGV);
172 RealTimeSignum s = signal1.RealTimeSignum;
176 [Category ("NotOnMac")]
177 public void TestRaiseRTMINSignal ()
179 if (!TestHelper.CanUseRealTimeSignals ())
181 RealTimeSignum rts = new RealTimeSignum (0);
182 using (UnixSignal signal = new UnixSignal (rts))
184 MultiThreadTest (signal, 5000, delegate() {
192 [Category ("NotOnMac")]
193 public void TestRaiseRTMINPlusOneSignal ()
195 if (!TestHelper.CanUseRealTimeSignals ())
197 /*this number is a guestimate, but it's ok*/
198 for (int i = 1; i < 10; ++i) {
199 RealTimeSignum rts = new RealTimeSignum (i);
202 signal = new UnixSignal (rts);
203 } catch (ArgumentException) { /*skip the ones that are unavailable*/
208 MultiThreadTest (signal, 5000, delegate() {
215 Assert.IsTrue (false, "#1 No available RT signal");
219 [Category ("NotOnMac")]
220 public void TestCanRegisterRTSignalMultipleTimes ()
222 if (!TestHelper.CanUseRealTimeSignals ())
224 /*this number is a guestimate, but it's ok*/
225 for (int i = 1; i < 10; ++i) {
226 RealTimeSignum rts = new RealTimeSignum (i);
229 signal = new UnixSignal (rts);
230 } catch (ArgumentException) { /*skip the ones that are unavailable*/
234 using (UnixSignal signal2 = new UnixSignal (rts))
239 } catch (ArgumentException) { /*skip the ones that are unavailable*/
240 Assert.IsTrue (false, "#1 Could not register second signal handler");
243 Assert.IsTrue (false, "#2 No available RT signal");
247 public void TestRaise ()
249 Thread t1 = new Thread (delegate () {
250 using (UnixSignal a = new UnixSignal (Signum.SIGINT)) {
251 DateTime start = DateTime.Now;
252 bool r = a.WaitOne (5000, false);
253 DateTime end = DateTime.Now;
254 Assert.AreEqual (a.Count, 1);
255 Assert.AreEqual (r, true);
256 if ((end - start) > new TimeSpan (0, 0, 5))
257 throw new InvalidOperationException ("Signal slept too long");
260 Thread t2 = new Thread (delegate () {
262 Stdlib.raise (Signum.SIGINT);
271 public void TestRaiseAny ()
273 Thread t1 = new Thread (delegate () {
274 using (UnixSignal a = new UnixSignal (Signum.SIGINT)) {
275 DateTime start = DateTime.Now;
276 int idx = UnixSignal.WaitAny (new UnixSignal[]{a}, 5000);
277 DateTime end = DateTime.Now;
278 Assert.AreEqual (idx, 0);
279 Assert.AreEqual (a.Count, 1);
280 if ((end - start) > new TimeSpan (0, 0, 5))
281 throw new InvalidOperationException ("Signal slept too long");
284 Thread t2 = new Thread (delegate () {
286 Stdlib.raise (Signum.SIGINT);
295 public void TestSeparation ()
297 Thread t1 = new Thread (delegate () {
298 using (UnixSignal a = new UnixSignal (Signum.SIGINT))
299 using (UnixSignal b = new UnixSignal (Signum.SIGTERM)) {
300 DateTime start = DateTime.Now;
301 int idx = UnixSignal.WaitAny (new UnixSignal[]{a, b}, 5000);
302 DateTime end = DateTime.Now;
303 Assert.AreEqual (idx, 1);
304 Assert.AreEqual (a.Count, 0);
305 Assert.AreEqual (b.Count, 1);
306 if ((end - start) > new TimeSpan (0, 0, 5))
307 throw new InvalidOperationException ("Signal slept too long");
310 Thread t2 = new Thread (delegate () {
312 Stdlib.raise (Signum.SIGTERM);
321 public void TestNoEmit ()
323 using (UnixSignal u = new UnixSignal (Signum.SIGINT)) {
324 DateTime start = DateTime.Now;
325 bool r = u.WaitOne (5100, false);
326 Assert.AreEqual (r, false);
327 DateTime end = DateTime.Now;
328 if ((end - start) < new TimeSpan (0, 0, 5))
329 throw new InvalidOperationException ("Signal didn't block for 5s; blocked for " + (end-start).ToString());
334 public void TestNoEmitAny ()
336 using (UnixSignal u = new UnixSignal (Signum.SIGINT)) {
337 int idx = UnixSignal.WaitAny (new UnixSignal[]{u}, 5100);
338 Assert.AreEqual (idx, 5100);
343 public void TestDispose1 ()
345 UnixSignal a = new UnixSignal (Signum.SIGINT);
346 UnixSignal b = new UnixSignal (Signum.SIGINT);
348 Stdlib.raise (Signum.SIGINT);
349 SleepUntilSignaled (a);
351 Assert.AreEqual (a.Count, 1);
352 Assert.AreEqual (b.Count, 1);
357 Stdlib.raise (Signum.SIGINT);
358 SleepUntilSignaled (b);
359 Assert.AreEqual (b.Count, 1);
364 static void SleepUntilSignaled (UnixSignal s)
366 for (int i = 0; i < 10; ++i) {
374 public void TestDispose2 ()
376 UnixSignal a = new UnixSignal (Signum.SIGINT);
377 UnixSignal b = new UnixSignal (Signum.SIGINT);
379 Stdlib.raise (Signum.SIGINT);
380 SleepUntilSignaled (a);
382 Assert.AreEqual (a.Count, 1);
383 Assert.AreEqual (b.Count, 1);
388 Stdlib.raise (Signum.SIGINT);
389 SleepUntilSignaled (a);
390 Assert.AreEqual (a.Count, 1);
396 [Category ("AndroidNotWorking")] // Android 4.4.4 doesn't have signal(2)
397 public void TestSignalActionInteraction ()
399 using (UnixSignal a = new UnixSignal (Signum.SIGINT)) {
400 Stdlib.SetSignalAction (Signum.SIGINT, SignalAction.Ignore);
401 Stdlib.raise (Signum.SIGINT);
402 Assert.AreEqual (a.Count, 0); // never invoked
406 static readonly Signum[] signals = new Signum[] {
407 Signum.SIGHUP, Signum.SIGINT, Signum.SIGTERM, Signum.SIGCONT,
410 const int StormCount = 100000;
413 [Category("NotOnMac")] // OSX signal storming will not deliver every one
414 public void TestRaiseStorm ()
416 UnixSignal[] usignals = CreateSignals (signals);
417 Thread[] threads = new Thread[]{
418 CreateRaiseStormThread (StormCount/4),
419 CreateRaiseStormThread (StormCount/4),
420 CreateRaiseStormThread (StormCount/4),
421 CreateRaiseStormThread (StormCount/4),
423 foreach (Thread t in threads)
425 foreach (Thread t in threads)
427 AssertCount (usignals);
428 CloseSignals (usignals);
431 static void AssertCount (UnixSignal[] usignals)
434 foreach (UnixSignal s in usignals)
436 Assert.AreEqual (sum, StormCount);
439 static UnixSignal[] CreateSignals (Signum[] signals)
441 UnixSignal[] s = new UnixSignal [signals.Length];
442 for (int i = 0; i < signals.Length; ++i)
443 s [i] = new UnixSignal (signals [i]);
447 static void CloseSignals (UnixSignal[] signals)
449 foreach (UnixSignal s in signals)
453 static Thread CreateRaiseStormThread (int max)
455 return new Thread (delegate () {
456 Random r = new Random (Environment.TickCount);
457 for (int i = 0; i < max; ++i) {
458 int n = r.Next (0, signals.Length);
459 Stdlib.raise (signals [n]);
465 [Category("NotOnMac")] // OSX signal storming will not deliver every one
466 public void TestAddRemove ()
468 UnixSignal[] usignals = CreateSignals (signals);
470 Thread[] threads = new Thread[]{
471 CreateRaiseStormThread (StormCount),
472 CreateSignalCreatorThread (),
475 foreach (Thread t in threads)
477 foreach (Thread t in threads)
480 AssertCount (usignals);
481 CloseSignals (usignals);
484 static Thread CreateSignalCreatorThread ()
486 return new Thread (delegate () {
487 Random r = new Random (Environment.TickCount << 4);
488 for (int i = 0; i < StormCount; ++i) {
489 int n = r.Next (0, signals.Length);
490 using (new UnixSignal (signals [n]))
491 using (new UnixSignal (signals [(n+1)%signals.Length]))
492 using (new UnixSignal (signals [(n+2)%signals.Length]))
493 using (new UnixSignal (signals [(n+3)%signals.Length])) {
500 [Category("NotOnMac")] // OSX signal storming will not deliver every one
501 public void TestWaitAny ()
503 UnixSignal[] usignals = CreateSignals (signals);
505 Thread[] threads = new Thread[]{
506 CreateRaiseStormThread (StormCount),
507 CreateSignalCreatorThread (),
508 CreateWaitAnyThread (usignals [0], usignals [2]),
509 CreateWaitAnyThread (usignals [1], usignals [3]),
510 CreateWaitAnyThread (usignals [1], usignals [2]),
513 foreach (Thread t in threads)
515 foreach (Thread t in threads)
518 AssertCount (usignals);
519 CloseSignals (usignals);
522 static Thread CreateWaitAnyThread (params UnixSignal[] usignals)
524 return new Thread (delegate () {
525 int idx = UnixSignal.WaitAny (usignals, 30000);
526 Assert.AreEqual (idx >= 0 && idx < usignals.Length, true);