1
0

ThreadPoolSchedulerTest.cs 9.9 KB

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