ThreadPoolSchedulerTest.cs 9.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330
  1. // Licensed to the .NET Foundation under one or more agreements.
  2. // The .NET Foundation licenses this file to you under the Apache 2.0 License.
  3. // See the LICENSE file in the project root for more information.
  4. #if !NO_THREAD
  5. using System;
  6. using System.Collections.Generic;
  7. using System.Linq;
  8. using System.Reactive.Concurrency;
  9. using System.Reactive.Disposables;
  10. using System.Threading;
  11. using Microsoft.Reactive.Testing;
  12. using Xunit;
  13. namespace ReactiveTests.Tests
  14. {
  15. public class ThreadPoolSchedulerTest
  16. {
  17. [Fact]
  18. public void Schedule_ArgumentChecking()
  19. {
  20. ReactiveAssert.Throws<ArgumentNullException>(() => ThreadPoolScheduler.Instance.Schedule<int>(42, default(Func<IScheduler, int, IDisposable>)));
  21. ReactiveAssert.Throws<ArgumentNullException>(() => ThreadPoolScheduler.Instance.Schedule<int>(42, DateTimeOffset.Now, default(Func<IScheduler, int, IDisposable>)));
  22. ReactiveAssert.Throws<ArgumentNullException>(() => ThreadPoolScheduler.Instance.Schedule<int>(42, TimeSpan.Zero, default(Func<IScheduler, int, IDisposable>)));
  23. }
  24. [Fact]
  25. public void Get_Now()
  26. {
  27. var res = ThreadPoolScheduler.Instance.Now - DateTime.Now;
  28. Assert.True(res.Seconds < 1);
  29. }
  30. [Fact]
  31. public void ScheduleAction()
  32. {
  33. var id = Thread.CurrentThread.ManagedThreadId;
  34. var nt = ThreadPoolScheduler.Instance;
  35. var evt = new ManualResetEvent(false);
  36. nt.Schedule(() => { Assert.NotEqual(id, Thread.CurrentThread.ManagedThreadId); evt.Set(); });
  37. evt.WaitOne();
  38. }
  39. #if !NO_CDS
  40. [Fact]
  41. public void ProperRooting_NoGC_SingleShot()
  42. {
  43. var cts = new CancellationTokenSource();
  44. new Thread(() =>
  45. {
  46. while (!cts.IsCancellationRequested)
  47. {
  48. Thread.Sleep(50);
  49. GC.Collect();
  50. GC.WaitForPendingFinalizers();
  51. }
  52. }).Start();
  53. var tp = ThreadPoolScheduler.Instance;
  54. var N = 100;
  55. var cd = new CountdownEvent(N);
  56. for (int i = 0; i < N; i++)
  57. {
  58. tp.Schedule(TimeSpan.FromMilliseconds(100 + i), () => { cd.Signal(); });
  59. }
  60. Assert.True(cd.Wait(TimeSpan.FromMinutes(1)));
  61. cts.Cancel();
  62. }
  63. [Fact]
  64. public void ProperRooting_NoGC_Periodic()
  65. {
  66. var cts = new CancellationTokenSource();
  67. new Thread(() =>
  68. {
  69. while (!cts.IsCancellationRequested)
  70. {
  71. Thread.Sleep(50);
  72. GC.Collect();
  73. GC.WaitForPendingFinalizers();
  74. }
  75. }).Start();
  76. var tp = ThreadPoolScheduler.Instance;
  77. var N = 5;
  78. var e = new ManualResetEvent(false);
  79. var n = 0;
  80. var d = tp.SchedulePeriodic(TimeSpan.FromMilliseconds(80), () => { if (Interlocked.Increment(ref n) == N) e.Set(); });
  81. Assert.True(e.WaitOne(TimeSpan.FromMinutes(1)));
  82. d.Dispose();
  83. cts.Cancel();
  84. }
  85. #endif
  86. #if !SILVERLIGHT
  87. [Fact]
  88. public void ScheduleActionDueRelative()
  89. {
  90. var id = Thread.CurrentThread.ManagedThreadId;
  91. var nt = ThreadPoolScheduler.Instance;
  92. var evt = new ManualResetEvent(false);
  93. nt.Schedule(TimeSpan.FromSeconds(0.2), () => { Assert.NotEqual(id, Thread.CurrentThread.ManagedThreadId); evt.Set(); });
  94. evt.WaitOne();
  95. }
  96. [Fact]
  97. public void ScheduleActionDue0()
  98. {
  99. var id = Thread.CurrentThread.ManagedThreadId;
  100. var nt = ThreadPoolScheduler.Instance;
  101. var evt = new ManualResetEvent(false);
  102. nt.Schedule(TimeSpan.FromTicks(0), () => { Assert.NotEqual(id, Thread.CurrentThread.ManagedThreadId); evt.Set(); });
  103. evt.WaitOne();
  104. }
  105. [Fact]
  106. public void ScheduleActionDueAbsolute()
  107. {
  108. var id = Thread.CurrentThread.ManagedThreadId;
  109. var nt = ThreadPoolScheduler.Instance;
  110. var evt = new ManualResetEvent(false);
  111. nt.Schedule(DateTimeOffset.UtcNow + TimeSpan.FromSeconds(0.2), () => { Assert.NotEqual(id, Thread.CurrentThread.ManagedThreadId); evt.Set(); });
  112. evt.WaitOne();
  113. }
  114. #endif
  115. [Fact]
  116. public void ScheduleActionCancel()
  117. {
  118. var id = Thread.CurrentThread.ManagedThreadId;
  119. var nt = ThreadPoolScheduler.Instance;
  120. var set = false;
  121. var d = nt.Schedule(TimeSpan.FromSeconds(0.2), () => { Assert.True(false); set = true; });
  122. d.Dispose();
  123. Thread.Sleep(400);
  124. Assert.False(set);
  125. }
  126. #if !NO_PERF
  127. [Fact]
  128. public void ScheduleLongRunning_ArgumentChecking()
  129. {
  130. ReactiveAssert.Throws<ArgumentNullException>(() => ThreadPoolScheduler.Instance.ScheduleLongRunning<int>(42, default(Action<int, ICancelable>)));
  131. }
  132. [Fact]
  133. public void ScheduleLongRunning()
  134. {
  135. var id = Thread.CurrentThread.ManagedThreadId;
  136. var nt = ThreadPoolScheduler.Instance;
  137. var evt = new ManualResetEvent(false);
  138. nt.ScheduleLongRunning(42, (x, cancel) => { Assert.NotEqual(id, Thread.CurrentThread.ManagedThreadId); evt.Set(); });
  139. evt.WaitOne();
  140. }
  141. [Fact]
  142. public void ScheduleLongRunningCancel()
  143. {
  144. var nt = ThreadPoolScheduler.Instance;
  145. var started = new ManualResetEvent(false);
  146. var stopped = new ManualResetEvent(false);
  147. var n = 0;
  148. var d = nt.ScheduleLongRunning(42, (x, cancel) =>
  149. {
  150. for (n = 0; !cancel.IsDisposed; n++)
  151. {
  152. if (n == 10)
  153. started.Set();
  154. }
  155. stopped.Set();
  156. });
  157. started.WaitOne();
  158. d.Dispose();
  159. stopped.WaitOne();
  160. Assert.True(n >= 10);
  161. }
  162. [Fact]
  163. public void Stopwatch()
  164. {
  165. var nt = ThreadPoolScheduler.Instance;
  166. var sw = nt.StartStopwatch();
  167. var s0 = sw.Elapsed.Ticks;
  168. Thread.Sleep(10);
  169. var s1 = sw.Elapsed.Ticks;
  170. Assert.True(s1 > s0);
  171. }
  172. [Fact]
  173. public void Periodic_ArgumentChecking()
  174. {
  175. ReactiveAssert.Throws<ArgumentNullException>(() => ThreadPoolScheduler.Instance.SchedulePeriodic(0, TimeSpan.FromSeconds(1), null));
  176. ReactiveAssert.Throws<ArgumentOutOfRangeException>(() => ThreadPoolScheduler.Instance.SchedulePeriodic(0, TimeSpan.FromSeconds(-1), _ => _));
  177. }
  178. [Fact]
  179. public void Periodic_Regular()
  180. {
  181. Periodic_Impl(TimeSpan.FromMilliseconds(25));
  182. }
  183. [Fact]
  184. public void Periodic_Zero()
  185. {
  186. Periodic_Impl(TimeSpan.Zero);
  187. }
  188. private void Periodic_Impl(TimeSpan period)
  189. {
  190. var gate = new object();
  191. var n = 0;
  192. var e = new ManualResetEvent(false);
  193. var lst = new List<int>();
  194. var d = ThreadPoolScheduler.Instance.SchedulePeriodic(0, period, x =>
  195. {
  196. lock (gate)
  197. {
  198. if (n++ == 10)
  199. e.Set();
  200. }
  201. lst.Add(x);
  202. return x + 1;
  203. });
  204. e.WaitOne();
  205. d.Dispose();
  206. var m = default(int);
  207. var k = default(int);
  208. var i = 0;
  209. do
  210. {
  211. lock (gate)
  212. m = n;
  213. Thread.Sleep(50);
  214. lock (gate)
  215. k = n;
  216. } while (m != k && i++ < 10); // Wait for Dispose to reach the timer; should be almost instantaneous due to nop'ing out of the action.
  217. Assert.NotEqual(10, i);
  218. var res = lst.ToArray();
  219. Assert.True(res.Length >= 10);
  220. Assert.True(res.Take(10).SequenceEqual(Enumerable.Range(0, 10)));
  221. }
  222. [Fact]
  223. public void Periodic_NonReentrant()
  224. {
  225. var n = 0;
  226. var fail = false;
  227. var d = ThreadPoolScheduler.Instance.SchedulePeriodic(0, TimeSpan.FromMilliseconds(50), x =>
  228. {
  229. try
  230. {
  231. if (Interlocked.Increment(ref n) > 1) // Without an AsyncLock this would fail.
  232. fail = true;
  233. Thread.Sleep(100);
  234. return x + 1;
  235. }
  236. finally
  237. {
  238. Interlocked.Decrement(ref n);
  239. }
  240. });
  241. Thread.Sleep(500);
  242. d.Dispose();
  243. Assert.False(fail);
  244. }
  245. #endif
  246. #if DESKTOPCLR
  247. [Fact]
  248. public void No_ThreadPool_Starvation_Dispose()
  249. {
  250. var bwt = default(int);
  251. var bio = default(int);
  252. ThreadPool.GetAvailableThreads(out bwt, out bio);
  253. var N = Environment.ProcessorCount * 2;
  254. for (int i = 0; i < N; i++)
  255. {
  256. var e = new ManualResetEvent(false);
  257. var f = new ManualResetEvent(false);
  258. var d = ThreadPoolScheduler.Instance.Schedule(TimeSpan.FromMilliseconds(1), () => { e.Set(); f.WaitOne(); });
  259. e.WaitOne();
  260. d.Dispose();
  261. f.Set();
  262. }
  263. var ewt = default(int);
  264. var eio = default(int);
  265. ThreadPool.GetAvailableThreads(out ewt, out eio);
  266. Assert.False(bwt - ewt >= N);
  267. }
  268. #endif
  269. }
  270. }
  271. #endif