SchedulerWrapper.cs 4.2 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126
  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. using System;
  5. #if !NO_WEAKTABLE
  6. using System.Runtime.CompilerServices;
  7. #endif
  8. namespace System.Reactive.Concurrency
  9. {
  10. internal abstract class SchedulerWrapper : IScheduler, IServiceProvider
  11. {
  12. protected readonly IScheduler _scheduler;
  13. public SchedulerWrapper(IScheduler scheduler)
  14. {
  15. _scheduler = scheduler;
  16. #if !NO_WEAKTABLE
  17. _cache = new ConditionalWeakTable<IScheduler, IScheduler>();
  18. #endif
  19. }
  20. public DateTimeOffset Now
  21. {
  22. get { return _scheduler.Now; }
  23. }
  24. public IDisposable Schedule<TState>(TState state, Func<IScheduler, TState, IDisposable> action)
  25. {
  26. if (action == null)
  27. throw new ArgumentNullException(nameof(action));
  28. return _scheduler.Schedule(state, Wrap(action));
  29. }
  30. public IDisposable Schedule<TState>(TState state, TimeSpan dueTime, Func<IScheduler, TState, IDisposable> action)
  31. {
  32. if (action == null)
  33. throw new ArgumentNullException(nameof(action));
  34. return _scheduler.Schedule(state, dueTime, Wrap(action));
  35. }
  36. public IDisposable Schedule<TState>(TState state, DateTimeOffset dueTime, Func<IScheduler, TState, IDisposable> action)
  37. {
  38. if (action == null)
  39. throw new ArgumentNullException(nameof(action));
  40. return _scheduler.Schedule(state, dueTime, Wrap(action));
  41. }
  42. protected virtual Func<IScheduler, TState, IDisposable> Wrap<TState>(Func<IScheduler, TState, IDisposable> action)
  43. {
  44. return (self, state) => action(GetRecursiveWrapper(self), state);
  45. }
  46. #if !NO_WEAKTABLE
  47. private readonly ConditionalWeakTable<IScheduler, IScheduler> _cache;
  48. public SchedulerWrapper(IScheduler scheduler, ConditionalWeakTable<IScheduler, IScheduler> cache)
  49. {
  50. _scheduler = scheduler;
  51. _cache = cache;
  52. }
  53. protected IScheduler GetRecursiveWrapper(IScheduler scheduler)
  54. {
  55. return _cache.GetValue(scheduler, s => Clone(s, _cache));
  56. }
  57. protected abstract SchedulerWrapper Clone(IScheduler scheduler, ConditionalWeakTable<IScheduler, IScheduler> cache);
  58. #else
  59. private readonly object _gate = new object();
  60. private IScheduler _recursiveOriginal;
  61. private IScheduler _recursiveWrapper;
  62. protected IScheduler GetRecursiveWrapper(IScheduler scheduler)
  63. {
  64. var recursiveWrapper = default(IScheduler);
  65. lock (_gate)
  66. {
  67. //
  68. // Chances are the recursive scheduler will remain the same. In practice, this
  69. // single-shot caching scheme works out quite well. Notice we propagate our
  70. // mini-cache to recursive raw scheduler wrappers too.
  71. //
  72. if (!object.ReferenceEquals(scheduler, _recursiveOriginal))
  73. {
  74. _recursiveOriginal = scheduler;
  75. var wrapper = Clone(scheduler);
  76. wrapper._recursiveOriginal = scheduler;
  77. wrapper._recursiveWrapper = wrapper;
  78. _recursiveWrapper = wrapper;
  79. }
  80. recursiveWrapper = _recursiveWrapper;
  81. }
  82. return recursiveWrapper;
  83. }
  84. protected abstract SchedulerWrapper Clone(IScheduler scheduler);
  85. #endif
  86. public object GetService(Type serviceType)
  87. {
  88. var serviceProvider = _scheduler as IServiceProvider;
  89. if (serviceProvider == null)
  90. return null;
  91. var result = default(object);
  92. if (TryGetService(serviceProvider, serviceType, out result))
  93. return result;
  94. return serviceProvider.GetService(serviceType);
  95. }
  96. protected abstract bool TryGetService(IServiceProvider provider, Type serviceType, out object service);
  97. }
  98. }