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