ScalarScheduleBenchmark.cs 5.0 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179
  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.Reactive.Concurrency;
  6. using System.Reactive.Linq;
  7. using System.Threading;
  8. using BenchmarkDotNet.Attributes;
  9. namespace Benchmarks.System.Reactive
  10. {
  11. [MemoryDiagnoser]
  12. public class ScalarScheduleBenchmark
  13. {
  14. private int _store;
  15. private Exception _exceptionStore;
  16. private IScheduler _eventLoop;
  17. private Exception _exception;
  18. [GlobalSetup]
  19. public void Setup()
  20. {
  21. _eventLoop = new EventLoopScheduler();
  22. _exception = new Exception();
  23. }
  24. private void BlockingConsume(IObservable<int> source)
  25. {
  26. var cde = new CountdownEvent(1);
  27. source.Subscribe(v => Volatile.Write(ref _store, v),
  28. e =>
  29. {
  30. Volatile.Write(ref _exceptionStore, e);
  31. cde.Signal();
  32. },
  33. () => cde.Signal()
  34. );
  35. // spin-wait will result in faster completion detection
  36. // because it takes 5 microseconds to resume a blocked thread
  37. // for me on Windows
  38. while (cde.CurrentCount != 0) ;
  39. }
  40. private void ConsumeSync(IObservable<int> source)
  41. {
  42. source.Subscribe(v => Volatile.Write(ref _store, v), e => Volatile.Write(ref _exceptionStore, e));
  43. }
  44. [Benchmark]
  45. public void Return_Immediate()
  46. {
  47. ConsumeSync(Observable.Return(1, ImmediateScheduler.Instance));
  48. }
  49. [Benchmark]
  50. public void Return_CurrentThread()
  51. {
  52. ConsumeSync(Observable.Return(1, CurrentThreadScheduler.Instance));
  53. }
  54. [Benchmark]
  55. public void Return_EventLoop()
  56. {
  57. BlockingConsume(Observable.Return(1, _eventLoop));
  58. }
  59. [Benchmark]
  60. public void Return_TaskPool()
  61. {
  62. BlockingConsume(Observable.Return(1, TaskPoolScheduler.Default));
  63. }
  64. [Benchmark]
  65. public void Return_ThreadPool()
  66. {
  67. BlockingConsume(Observable.Return(1, ThreadPoolScheduler.Instance));
  68. }
  69. [Benchmark]
  70. public void Throw_Immediate()
  71. {
  72. ConsumeSync(Observable.Throw<int>(_exception, ImmediateScheduler.Instance));
  73. }
  74. [Benchmark]
  75. public void Throw_CurrentThread()
  76. {
  77. ConsumeSync(Observable.Throw<int>(_exception, CurrentThreadScheduler.Instance));
  78. }
  79. [Benchmark]
  80. public void Throw_EventLoop()
  81. {
  82. BlockingConsume(Observable.Throw<int>(_exception, _eventLoop));
  83. }
  84. [Benchmark]
  85. public void Throw_TaskPool()
  86. {
  87. BlockingConsume(Observable.Throw<int>(_exception, TaskPoolScheduler.Default));
  88. }
  89. [Benchmark]
  90. public void Throw_ThreadPool()
  91. {
  92. BlockingConsume(Observable.Throw<int>(_exception, ThreadPoolScheduler.Instance));
  93. }
  94. #if CURRENT
  95. [Benchmark]
  96. public void Prepend_Immediate()
  97. {
  98. ConsumeSync(Observable.Return(1, ImmediateScheduler.Instance).Prepend(0, ImmediateScheduler.Instance));
  99. }
  100. [Benchmark]
  101. public void Prepend_CurrentThread()
  102. {
  103. ConsumeSync(Observable.Return(1, CurrentThreadScheduler.Instance).Prepend(0, CurrentThreadScheduler.Instance));
  104. }
  105. [Benchmark]
  106. public void Prepend_EventLoop()
  107. {
  108. BlockingConsume(Observable.Return(1, _eventLoop).Prepend(0, _eventLoop));
  109. }
  110. [Benchmark]
  111. public void Prepend_TaskPool()
  112. {
  113. BlockingConsume(Observable.Return(1, TaskPoolScheduler.Default).Prepend(0, TaskPoolScheduler.Default));
  114. }
  115. [Benchmark]
  116. public void Prepend_ThreadPool()
  117. {
  118. BlockingConsume(Observable.Return(1, ThreadPoolScheduler.Instance).Prepend(0, ThreadPoolScheduler.Instance));
  119. }
  120. [Benchmark]
  121. public void Append_Immediate()
  122. {
  123. ConsumeSync(Observable.Return(1, ImmediateScheduler.Instance).Append(0, ImmediateScheduler.Instance));
  124. }
  125. [Benchmark]
  126. public void Append_CurrentThread()
  127. {
  128. ConsumeSync(Observable.Return(1, CurrentThreadScheduler.Instance).Append(0, CurrentThreadScheduler.Instance));
  129. }
  130. [Benchmark]
  131. public void Append_EventLoop()
  132. {
  133. BlockingConsume(Observable.Return(1, _eventLoop).Append(0, _eventLoop));
  134. }
  135. [Benchmark]
  136. public void Append_TaskPool()
  137. {
  138. BlockingConsume(Observable.Return(1, TaskPoolScheduler.Default).Append(0, TaskPoolScheduler.Default));
  139. }
  140. [Benchmark]
  141. public void Append_ThreadPool()
  142. {
  143. BlockingConsume(Observable.Return(1, ThreadPoolScheduler.Instance).Append(0, ThreadPoolScheduler.Instance));
  144. }
  145. #endif
  146. }
  147. }