Welcome to mirror list, hosted at ThFree Co, Russian Federation.

github.com/mono/rx.git - Unnamed repository; edit this file 'description' to name the repository.
summaryrefslogtreecommitdiff
diff options
context:
space:
mode:
Diffstat (limited to 'Rx/NET/Source/Tests.System.Reactive/Tests/Concurrency/ThreadPoolSchedulerTest.cs')
-rw-r--r--Rx/NET/Source/Tests.System.Reactive/Tests/Concurrency/ThreadPoolSchedulerTest.cs322
1 files changed, 322 insertions, 0 deletions
diff --git a/Rx/NET/Source/Tests.System.Reactive/Tests/Concurrency/ThreadPoolSchedulerTest.cs b/Rx/NET/Source/Tests.System.Reactive/Tests/Concurrency/ThreadPoolSchedulerTest.cs
new file mode 100644
index 0000000..3cd87f3
--- /dev/null
+++ b/Rx/NET/Source/Tests.System.Reactive/Tests/Concurrency/ThreadPoolSchedulerTest.cs
@@ -0,0 +1,322 @@
+// Copyright (c) Microsoft Open Technologies, Inc. All rights reserved. See License.txt in the project root for license information.
+
+#if !NO_THREAD
+using System;
+using System.Collections.Generic;
+using System.Linq;
+using System.Reactive.Concurrency;
+using System.Reactive.Disposables;
+using System.Threading;
+using Microsoft.Reactive.Testing;
+using Microsoft.VisualStudio.TestTools.UnitTesting;
+
+namespace ReactiveTests.Tests
+{
+ [TestClass]
+ public class ThreadPoolSchedulerTest
+ {
+ [TestMethod]
+ public void Schedule_ArgumentChecking()
+ {
+ ReactiveAssert.Throws<ArgumentNullException>(() => ThreadPoolScheduler.Instance.Schedule<int>(42, default(Func<IScheduler, int, IDisposable>)));
+ ReactiveAssert.Throws<ArgumentNullException>(() => ThreadPoolScheduler.Instance.Schedule<int>(42, DateTimeOffset.Now, default(Func<IScheduler, int, IDisposable>)));
+ ReactiveAssert.Throws<ArgumentNullException>(() => ThreadPoolScheduler.Instance.Schedule<int>(42, TimeSpan.Zero, default(Func<IScheduler, int, IDisposable>)));
+ }
+
+ [TestMethod]
+ public void Get_Now()
+ {
+ var res = ThreadPoolScheduler.Instance.Now - DateTime.Now;
+ Assert.IsTrue(res.Seconds < 1);
+ }
+
+ [TestMethod]
+ public void ScheduleAction()
+ {
+ var id = Thread.CurrentThread.ManagedThreadId;
+ var nt = ThreadPoolScheduler.Instance;
+ var evt = new ManualResetEvent(false);
+ nt.Schedule(() => { Assert.AreNotEqual(id, Thread.CurrentThread.ManagedThreadId); evt.Set(); });
+ evt.WaitOne();
+ }
+
+#if !NO_CDS
+ [TestMethod]
+ public void ProperRooting_NoGC_SingleShot()
+ {
+ var cts = new CancellationTokenSource();
+
+ new Thread(() =>
+ {
+ while (!cts.IsCancellationRequested)
+ {
+ Thread.Sleep(50);
+ GC.Collect();
+ GC.WaitForPendingFinalizers();
+ }
+ }).Start();
+
+ var tp = ThreadPoolScheduler.Instance;
+ var N = 100;
+ var cd = new CountdownEvent(N);
+ for (int i = 0; i < N; i++)
+ {
+ tp.Schedule(TimeSpan.FromMilliseconds(100 + i), () => { cd.Signal(); });
+ }
+
+ Assert.IsTrue(cd.Wait(TimeSpan.FromMinutes(1)));
+ cts.Cancel();
+ }
+
+ [TestMethod]
+ public void ProperRooting_NoGC_Periodic()
+ {
+ var cts = new CancellationTokenSource();
+
+ new Thread(() =>
+ {
+ while (!cts.IsCancellationRequested)
+ {
+ Thread.Sleep(50);
+ GC.Collect();
+ GC.WaitForPendingFinalizers();
+ }
+ }).Start();
+
+ var tp = ThreadPoolScheduler.Instance;
+ var N = 5;
+ var e = new ManualResetEvent(false);
+ var n = 0;
+ var d = tp.SchedulePeriodic(TimeSpan.FromMilliseconds(80), () => { if (Interlocked.Increment(ref n) == N) e.Set(); });
+
+ Assert.IsTrue(e.WaitOne(TimeSpan.FromMinutes(1)));
+
+ d.Dispose();
+
+ cts.Cancel();
+ }
+#endif
+
+#if !SILVERLIGHT
+ [TestMethod]
+ public void ScheduleActionDueRelative()
+ {
+ var id = Thread.CurrentThread.ManagedThreadId;
+ var nt = ThreadPoolScheduler.Instance;
+ var evt = new ManualResetEvent(false);
+ nt.Schedule(TimeSpan.FromSeconds(0.2), () => { Assert.AreNotEqual(id, Thread.CurrentThread.ManagedThreadId); evt.Set(); });
+ evt.WaitOne();
+ }
+
+ [TestMethod]
+ public void ScheduleActionDue0()
+ {
+ var id = Thread.CurrentThread.ManagedThreadId;
+ var nt = ThreadPoolScheduler.Instance;
+ var evt = new ManualResetEvent(false);
+ nt.Schedule(TimeSpan.FromTicks(0), () => { Assert.AreNotEqual(id, Thread.CurrentThread.ManagedThreadId); evt.Set(); });
+ evt.WaitOne();
+ }
+
+ [TestMethod]
+ public void ScheduleActionDueAbsolute()
+ {
+ var id = Thread.CurrentThread.ManagedThreadId;
+ var nt = ThreadPoolScheduler.Instance;
+ var evt = new ManualResetEvent(false);
+ nt.Schedule(DateTimeOffset.UtcNow + TimeSpan.FromSeconds(0.2), () => { Assert.AreNotEqual(id, Thread.CurrentThread.ManagedThreadId); evt.Set(); });
+ evt.WaitOne();
+ }
+#endif
+
+ [TestMethod]
+ public void ScheduleActionCancel()
+ {
+ var id = Thread.CurrentThread.ManagedThreadId;
+ var nt = ThreadPoolScheduler.Instance;
+ var set = false;
+ var d = nt.Schedule(TimeSpan.FromSeconds(0.2), () => { Assert.Fail(); set = true; });
+ d.Dispose();
+ Thread.Sleep(400);
+ Assert.IsFalse(set);
+ }
+
+#if !NO_PERF
+
+ [TestMethod]
+ public void ScheduleLongRunning_ArgumentChecking()
+ {
+ ReactiveAssert.Throws<ArgumentNullException>(() => ThreadPoolScheduler.Instance.ScheduleLongRunning<int>(42, default(Action<int, ICancelable>)));
+ }
+
+ [TestMethod]
+ public void ScheduleLongRunning()
+ {
+ var id = Thread.CurrentThread.ManagedThreadId;
+ var nt = ThreadPoolScheduler.Instance;
+ var evt = new ManualResetEvent(false);
+ nt.ScheduleLongRunning(42, (x, cancel) => { Assert.AreNotEqual(id, Thread.CurrentThread.ManagedThreadId); evt.Set(); });
+ evt.WaitOne();
+ }
+
+ [TestMethod]
+ public void ScheduleLongRunningCancel()
+ {
+ var nt = ThreadPoolScheduler.Instance;
+
+ var started = new ManualResetEvent(false);
+ var stopped = new ManualResetEvent(false);
+
+ var n = 0;
+
+ var d = nt.ScheduleLongRunning(42, (x, cancel) =>
+ {
+ for (n = 0; !cancel.IsDisposed; n++)
+ {
+ if (n == 10)
+ started.Set();
+ }
+
+ stopped.Set();
+ });
+
+ started.WaitOne();
+ d.Dispose();
+
+ stopped.WaitOne();
+ Assert.IsTrue(n >= 10);
+ }
+
+#if !NO_STOPWATCH
+
+ [TestMethod]
+ public void Stopwatch()
+ {
+ var nt = ThreadPoolScheduler.Instance;
+
+ var sw = nt.StartStopwatch();
+
+ var s0 = sw.Elapsed.Ticks;
+ Thread.Sleep(10);
+ var s1 = sw.Elapsed.Ticks;
+
+ Assert.IsTrue(s1 > s0);
+ }
+
+#endif
+
+ [TestMethod]
+ public void Periodic_ArgumentChecking()
+ {
+ ReactiveAssert.Throws<ArgumentNullException>(() => ThreadPoolScheduler.Instance.SchedulePeriodic(0, TimeSpan.FromSeconds(1), null));
+ ReactiveAssert.Throws<ArgumentOutOfRangeException>(() => ThreadPoolScheduler.Instance.SchedulePeriodic(0, TimeSpan.Zero, _ => _));
+ ReactiveAssert.Throws<ArgumentOutOfRangeException>(() => ThreadPoolScheduler.Instance.SchedulePeriodic(0, TimeSpan.FromSeconds(-1), _ => _));
+ }
+
+ [TestMethod]
+ public void Periodic_Regular()
+ {
+ var gate = new object();
+ var n = 0;
+ var e = new ManualResetEvent(false);
+
+ var lst = new List<int>();
+
+ var d = ThreadPoolScheduler.Instance.SchedulePeriodic(0, TimeSpan.FromMilliseconds(25), x =>
+ {
+ lock (gate)
+ {
+ if (n++ == 10)
+ e.Set();
+ }
+
+ lst.Add(x);
+ return x + 1;
+ });
+
+ e.WaitOne();
+ d.Dispose();
+
+ var m = default(int);
+ var k = default(int);
+
+ var i = 0;
+ do
+ {
+ lock (gate)
+ m = n;
+
+ Thread.Sleep(50);
+
+ lock (gate)
+ k = n;
+ } while (m != k && i++ < 10); // Wait for Dispose to reach the timer; should be almost instantaneous due to nop'ing out of the action.
+
+ Assert.AreNotEqual(10, i);
+
+ var res = lst.ToArray();
+ Assert.IsTrue(res.Length >= 10);
+ Assert.IsTrue(res.Take(10).SequenceEqual(Enumerable.Range(0, 10)));
+ }
+
+ [TestMethod]
+ public void Periodic_NonReentrant()
+ {
+ var n = 0;
+ var fail = false;
+
+ var d = ThreadPoolScheduler.Instance.SchedulePeriodic(0, TimeSpan.FromMilliseconds(50), x =>
+ {
+ try
+ {
+ if (Interlocked.Increment(ref n) > 1) // Without an AsyncLock this would fail.
+ fail = true;
+
+ Thread.Sleep(100);
+
+ return x + 1;
+ }
+ finally
+ {
+ Interlocked.Decrement(ref n);
+ }
+ });
+
+ Thread.Sleep(500);
+ d.Dispose();
+
+ Assert.IsFalse(fail);
+ }
+
+#endif
+
+#if DESKTOPCLR
+ [TestMethod]
+ public void No_ThreadPool_Starvation_Dispose()
+ {
+ var bwt = default(int);
+ var bio = default(int);
+ ThreadPool.GetAvailableThreads(out bwt, out bio);
+
+ var N = Environment.ProcessorCount * 2;
+
+ for (int i = 0; i < N; i++)
+ {
+ var e = new ManualResetEvent(false);
+ var f = new ManualResetEvent(false);
+ var d = ThreadPoolScheduler.Instance.Schedule(TimeSpan.FromMilliseconds(1), () => { e.Set(); f.WaitOne(); });
+ e.WaitOne();
+ d.Dispose();
+ f.Set();
+ }
+
+ var ewt = default(int);
+ var eio = default(int);
+ ThreadPool.GetAvailableThreads(out ewt, out eio);
+
+ Assert.IsFalse(bwt - ewt >= N);
+ }
+#endif
+ }
+}
+#endif \ No newline at end of file