concurrent-ruby 1.0.0.pre1-java → 1.0.0.pre2-java

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (77) hide show
  1. checksums.yaml +4 -4
  2. data/CHANGELOG.md +14 -1
  3. data/README.md +16 -18
  4. data/lib/concurrent.rb +3 -3
  5. data/lib/concurrent/agent.rb +583 -0
  6. data/lib/concurrent/array.rb +1 -0
  7. data/lib/concurrent/async.rb +236 -111
  8. data/lib/concurrent/atom.rb +101 -46
  9. data/lib/concurrent/atomic/atomic_boolean.rb +2 -0
  10. data/lib/concurrent/atomic/atomic_fixnum.rb +2 -0
  11. data/lib/concurrent/atomic/cyclic_barrier.rb +1 -1
  12. data/lib/concurrent/atomic/event.rb +1 -1
  13. data/lib/concurrent/atomic/mutex_atomic_boolean.rb +1 -1
  14. data/lib/concurrent/atomic/mutex_atomic_fixnum.rb +1 -1
  15. data/lib/concurrent/atomic/mutex_count_down_latch.rb +1 -1
  16. data/lib/concurrent/atomic/mutex_semaphore.rb +2 -2
  17. data/lib/concurrent/atomic/read_write_lock.rb +5 -4
  18. data/lib/concurrent/atomic/reentrant_read_write_lock.rb +3 -1
  19. data/lib/concurrent/atomic/thread_local_var.rb +2 -0
  20. data/lib/concurrent/atomic_reference/mutex_atomic.rb +1 -1
  21. data/lib/concurrent/atomics.rb +6 -4
  22. data/lib/concurrent/collection/copy_on_notify_observer_set.rb +7 -15
  23. data/lib/concurrent/collection/copy_on_write_observer_set.rb +7 -15
  24. data/lib/concurrent/collection/map/atomic_reference_map_backend.rb +5 -0
  25. data/lib/concurrent/concern/observable.rb +38 -13
  26. data/lib/concurrent/configuration.rb +5 -4
  27. data/lib/concurrent/delay.rb +9 -8
  28. data/lib/concurrent/exchanger.rb +2 -0
  29. data/lib/concurrent/executor/abstract_executor_service.rb +2 -2
  30. data/lib/concurrent/executor/java_single_thread_executor.rb +0 -1
  31. data/lib/concurrent/executor/ruby_executor_service.rb +10 -4
  32. data/lib/concurrent/executor/ruby_single_thread_executor.rb +10 -68
  33. data/lib/concurrent/executor/safe_task_executor.rb +7 -8
  34. data/lib/concurrent/executor/serialized_execution.rb +4 -4
  35. data/lib/concurrent/executor/single_thread_executor.rb +20 -10
  36. data/lib/concurrent/executor/timer_set.rb +4 -2
  37. data/lib/concurrent/executors.rb +0 -1
  38. data/lib/concurrent/future.rb +3 -2
  39. data/lib/concurrent/hash.rb +1 -1
  40. data/lib/concurrent/immutable_struct.rb +5 -1
  41. data/lib/concurrent/ivar.rb +1 -1
  42. data/lib/concurrent/mutable_struct.rb +7 -6
  43. data/lib/concurrent/{executor/executor.rb → options.rb} +4 -3
  44. data/lib/concurrent/promise.rb +3 -2
  45. data/lib/concurrent/scheduled_task.rb +3 -2
  46. data/lib/concurrent/settable_struct.rb +5 -4
  47. data/lib/concurrent/synchronization.rb +11 -3
  48. data/lib/concurrent/synchronization/abstract_lockable_object.rb +117 -0
  49. data/lib/concurrent/synchronization/abstract_object.rb +16 -129
  50. data/lib/concurrent/synchronization/abstract_struct.rb +2 -3
  51. data/lib/concurrent/synchronization/condition.rb +6 -4
  52. data/lib/concurrent/synchronization/jruby_lockable_object.rb +13 -0
  53. data/lib/concurrent/synchronization/{java_object.rb → jruby_object.rb} +5 -3
  54. data/lib/concurrent/synchronization/lock.rb +3 -2
  55. data/lib/concurrent/synchronization/lockable_object.rb +59 -0
  56. data/lib/concurrent/synchronization/mri_lockable_object.rb +71 -0
  57. data/lib/concurrent/synchronization/mri_object.rb +35 -0
  58. data/lib/concurrent/synchronization/object.rb +111 -39
  59. data/lib/concurrent/synchronization/rbx_lockable_object.rb +64 -0
  60. data/lib/concurrent/synchronization/rbx_object.rb +17 -68
  61. data/lib/concurrent/thread_safe/util.rb +0 -9
  62. data/lib/concurrent/thread_safe/util/adder.rb +3 -0
  63. data/lib/concurrent/thread_safe/util/array_hash_rbx.rb +3 -1
  64. data/lib/concurrent/thread_safe/util/cheap_lockable.rb +3 -0
  65. data/lib/concurrent/thread_safe/util/power_of_two_tuple.rb +1 -0
  66. data/lib/concurrent/thread_safe/util/striped64.rb +6 -1
  67. data/lib/concurrent/thread_safe/util/volatile.rb +2 -0
  68. data/lib/concurrent/thread_safe/util/xor_shift_random.rb +2 -0
  69. data/lib/concurrent/tvar.rb +36 -0
  70. data/lib/concurrent/utility/at_exit.rb +1 -1
  71. data/lib/concurrent/utility/monotonic_time.rb +3 -4
  72. data/lib/concurrent/utility/native_extension_loader.rb +1 -1
  73. data/lib/concurrent/version.rb +2 -2
  74. data/lib/concurrent_ruby_ext.jar +0 -0
  75. metadata +11 -6
  76. data/lib/concurrent/synchronization/monitor_object.rb +0 -27
  77. data/lib/concurrent/synchronization/mutex_object.rb +0 -43
@@ -27,6 +27,7 @@ module Concurrent
27
27
 
28
28
  elsif Concurrent.on_rbx?
29
29
  require 'monitor'
30
+ require 'concurrent/thread_safe/util/array_hash_rbx'
30
31
 
31
32
  # @!macro concurrent_array
32
33
  class Array < ::Array
@@ -1,52 +1,214 @@
1
- require 'thread'
2
- require 'concurrent/configuration'
3
- require 'concurrent/delay'
4
- require 'concurrent/errors'
5
1
  require 'concurrent/ivar'
6
- require 'concurrent/executor/immediate_executor'
7
- require 'concurrent/executor/serialized_execution'
2
+ require 'concurrent/executor/single_thread_executor'
8
3
 
9
4
  module Concurrent
10
5
 
11
- # A mixin module that provides simple asynchronous behavior to any standard
12
- # class/object or object.
6
+ # A mixin module that provides simple asynchronous behavior to a class,
7
+ # turning it into a simple actor. Loosely based on Erlang's
8
+ # [gen_server](http://www.erlang.org/doc/man/gen_server.html), but without
9
+ # supervision or linking.
10
+ #
11
+ # A more feature-rich {Concurrent::Actor} is also available when the
12
+ # capabilities of `Async` are too limited.
13
13
  #
14
14
  # ```cucumber
15
15
  # Feature:
16
- # As a stateful, plain old Ruby class/object
16
+ # As a stateful, plain old Ruby class
17
17
  # I want safe, asynchronous behavior
18
18
  # So my long-running methods don't block the main thread
19
19
  # ```
20
20
  #
21
- # Stateful, mutable objects must be managed carefully when used asynchronously.
22
- # But Ruby is an object-oriented language so designing with objects and classes
23
- # plays to Ruby's strengths and is often more natural to many Ruby programmers.
24
- # The `Async` module is a way to mix simple yet powerful asynchronous capabilities
25
- # into any plain old Ruby object or class. These capabilities provide a reasonable
26
- # level of thread safe guarantees when used correctly.
21
+ # The `Async` module is a way to mix simple yet powerful asynchronous
22
+ # capabilities into any plain old Ruby object or class, turning each object
23
+ # into a simple Actor. Method calls are processed on a background thread. The
24
+ # caller is free to perform other actions while processing occurs in the
25
+ # background.
26
+ #
27
+ # Method calls to the asynchronous object are made via two proxy methods:
28
+ # `async` (alias `cast`) and `await` (alias `call`). These proxy methods post
29
+ # the method call to the object's background thread and return a "future"
30
+ # which will eventually contain the result of the method call.
31
+ #
32
+ # This behavior is loosely patterned after Erlang's `gen_server` behavior.
33
+ # When an Erlang module implements the `gen_server` behavior it becomes
34
+ # inherently asynchronous. The `start` or `start_link` function spawns a
35
+ # process (similar to a thread but much more lightweight and efficient) and
36
+ # reurns the ID of the process. Using the process ID, other processes can
37
+ # send messages to the `gen_server` via the `cast` and `call` methods. Unlike
38
+ # Erlang's `gen_server`, however, `Async` classes do not support linking or
39
+ # supervision trees.
27
40
  #
28
- # When this module is mixed into a class or object it provides to new methods:
29
- # `async` and `await`. These methods are thread safe with respect to the enclosing
30
- # object. The former method allows methods to be called asynchronously by posting
31
- # to the global thread pool. The latter allows a method to be called synchronously
32
- # on the current thread but does so safely with respect to any pending asynchronous
33
- # method calls. Both methods return an `IVar` which can be inspected for
34
- # the result of the method call. Calling a method with `async` will return a
41
+ # ## Basic Usage
42
+ #
43
+ # When this module is mixed into a class, objects of the class become inherently
44
+ # asynchronous. Each object gets its own background thread (specifically,
45
+ # `SingleThreadExecutor`) on which to post asynchronous method calls.
46
+ # Asynchronous method calls are executed in the background one at a time in
47
+ # the order they are received.
48
+ #
49
+ # To create an asynchronous class, simply mix in the `Concurrent::Async` module:
50
+ #
51
+ # ```
52
+ # class Hello
53
+ # include Concurrent::Async
54
+ #
55
+ # def hello(name)
56
+ # "Hello, #{name}!"
57
+ # end
58
+ # end
59
+ # ```
60
+ #
61
+ # When defining a constructor it is critica that the first line be a call to
62
+ # `super` with no arguments. The `super` method initializes the background
63
+ # thread and other asynchronous components.
64
+ #
65
+ # ```
66
+ # class BackgroundLogger
67
+ # include Concurrent::Async
68
+ #
69
+ # def initialize(level)
70
+ # super()
71
+ # @logger = Logger.new(STDOUT)
72
+ # @logger.level = level
73
+ # end
74
+ #
75
+ # def info(msg)
76
+ # @logger.info(msg)
77
+ # end
78
+ # end
79
+ # ```
80
+ #
81
+ # Mixing this module into a class provides each object two proxy methods:
82
+ # `async` and `await`. These methods are thread safe with respect to the
83
+ # enclosing object. The former proxy allows methods to be called
84
+ # asynchronously by posting to the object's internal thread. The latter proxy
85
+ # allows a method to be called synchronously but does so safely with respect
86
+ # to any pending asynchronous method calls and ensures proper ordering. Both
87
+ # methods return a {Concurrent::IVar} which can be inspected for the result
88
+ # of the proxied method call. Calling a method with `async` will return a
35
89
  # `:pending` `IVar` whereas `await` will return a `:complete` `IVar`.
90
+ #
91
+ # ```
92
+ # class Echo
93
+ # include Concurrent::Async
94
+ #
95
+ # def echo(msg)
96
+ # print "#{msg}\n"
97
+ # end
98
+ # end
99
+ #
100
+ # horn = Echo.new
101
+ # horn.echo('zero') # synchronous, not thread-safe
102
+ # # returns the actual return value of the method
103
+ #
104
+ # horn.async.echo('one') # asynchronous, non-blocking, thread-safe
105
+ # # returns an IVar in the :pending state
106
+ #
107
+ # horn.await.echo('two') # synchronous, blocking, thread-safe
108
+ # # returns an IVar in the :complete state
109
+ # ```
110
+ #
111
+ # ## Let It Fail
112
+ #
113
+ # The `async` and `await` proxy methods have built-in error protection based
114
+ # on Erlang's famous "let it fail" philosophy. Instance methods should not be
115
+ # programmed defensively. When an exception is raised by a delegated method
116
+ # the proxy will rescue the exception, expose it to the caller as the `reason`
117
+ # attribute of the returned future, then process the next method call.
118
+ #
119
+ # ## Calling Methods Internally
120
+ #
121
+ # External method calls should *always* use the `async` and `await` proxy
122
+ # methods. When one method calls another method, the `async` proxy should
123
+ # rarely be used and the `await` proxy should *never* be used.
124
+ #
125
+ # When an object calls one of its own methods using the `await` proxy the
126
+ # second call will be enqueued *behind* the currently running method call.
127
+ # Any attempt to wait on the result will fail as the second call will never
128
+ # run until after the current call completes.
129
+ #
130
+ # Calling a method using the `await` proxy from within a method that was
131
+ # itself called using `async` or `await` will irreversibly deadlock the
132
+ # object. Do *not* do this, ever.
133
+ #
134
+ # ## Instance Variables and Attribute Accessors
135
+ #
136
+ # Instance variables do not need to be thread-safe so long as they are private.
137
+ # Asynchronous method calls are processed in the order they are received and
138
+ # are processed one at a time. Therefore private instance variables can only
139
+ # be accessed by one thread at a time. This is inherently thread-safe.
140
+ #
141
+ # When using private instance variables within asynchronous methods, the best
142
+ # practice is to read the instance variable into a local variable at the start
143
+ # of the method then update the instance variable at the *end* of the method.
144
+ # This way, should an exception be raised during method execution the internal
145
+ # state of the boject will not have been changed.
146
+ #
147
+ # ### Reader Attributes
148
+ #
149
+ # The use of `attr_reader` is discouraged. Internal state exposed externally,
150
+ # when necessary, should be done through accessor methods. The instance
151
+ # variables exposed by these methods *must* be thread-safe, or they must be
152
+ # called using the `async` and `await` proxy methods. These two approaches are
153
+ # subtly different.
154
+ #
155
+ # When internal state is accessed via the `async` and `await` proxy methods,
156
+ # the returned value represents the object's sate *at the time the call is
157
+ # processed*, which may *not* be the state of the object at the time the call
158
+ # is made.
159
+ #
160
+ # To get the state *at the current* time, irrespective of an enqueued method
161
+ # calls, a reader method must be called directly. This is inherently unsafe
162
+ # unless the instance variable is itself thread-safe, preferrably using one
163
+ # of the thread-safe classes within this library. Because the thread-safe
164
+ # classes within this library are internally-locking or non-locking, they can
165
+ # be safely used from within asynchronous methods without causing deadlocks.
166
+ #
167
+ # Generally speaking, the best practice is to *not* expose internal state via
168
+ # reader methods. The best practice is to simply use the method's return value.
169
+ #
170
+ # ### Writer Attributes
171
+ #
172
+ # Writer attributes should never be used with asynchronous classes. Changing
173
+ # the state externally, even when done in the thread-safe way, is not logically
174
+ # consistent. Changes to state need to be timed with respect to all asynchronous
175
+ # method calls which my be in-process or enqueued. The only safe practice is to
176
+ # pass all necessary data to each method as arguments and let the method update
177
+ # the internal state as necessary.
178
+ #
179
+ # ## Class Constants, Variables, and Methods
180
+ #
181
+ # ### Class Constants
182
+ #
183
+ # Class constants do not need to be thread-safe. Since they are read-only and
184
+ # immutable they may be safely read both externally and from within
185
+ # asynchronous methods.
186
+ #
187
+ # ### Class Variables
188
+ #
189
+ # Class variables should be avoided. Class variables represent shared state.
190
+ # Shared state is anathema to concurrency. Should there be a need to share
191
+ # state using class variables they *must* be thread-safe, preferrably
192
+ # using the thread-safe classes within this library. When updating class
193
+ # variables, never assign a new value/object to the variable itself. Assignment
194
+ # is not thread-safe in Ruby. Instead, use the thread-safe update functions
195
+ # of the variable itself to change the value.
196
+ #
197
+ # The best practice is to *never* use class variables with `Async` classes.
198
+ #
199
+ # ### Class Methods
200
+ #
201
+ # Class methods which are pure functions are safe. Class methods which modify
202
+ # class variables should be avoided, for all the reasons listed above.
36
203
  #
37
- # Very loosely based on the `async` and `await` keywords in C#.
38
- #
39
- # ### An Important Note About Thread Safe Guarantees
204
+ # ## An Important Note About Thread Safe Guarantees
40
205
  #
41
206
  # > Thread safe guarantees can only be made when asynchronous method calls
42
- # > are not mixed with synchronous method calls. Use only synchronous calls
207
+ # > are not mixed with direct method calls. Use only direct method calls
43
208
  # > when the object is used exclusively on a single thread. Use only
44
209
  # > `async` and `await` when the object is shared between threads. Once you
45
- # > call a method using `async`, you should no longer call any methods
210
+ # > call a method using `async` or `await`, you should no longer call methods
46
211
  # > directly on the object. Use `async` and `await` exclusively from then on.
47
- # > With careful programming it is possible to switch back and forth but it's
48
- # > also very easy to create race conditions and break your application.
49
- # > Basically, it's "async all the way down."
50
212
  #
51
213
  # @example
52
214
  #
@@ -54,19 +216,25 @@ module Concurrent
54
216
  # include Concurrent::Async
55
217
  #
56
218
  # def echo(msg)
57
- # sleep(rand)
58
219
  # print "#{msg}\n"
59
- # nil
60
220
  # end
61
221
  # end
62
222
  #
63
223
  # horn = Echo.new
64
224
  # horn.echo('zero') # synchronous, not thread-safe
225
+ # # returns the actual return value of the method
65
226
  #
66
227
  # horn.async.echo('one') # asynchronous, non-blocking, thread-safe
228
+ # # returns an IVar in the :pending state
229
+ #
67
230
  # horn.await.echo('two') # synchronous, blocking, thread-safe
231
+ # # returns an IVar in the :complete state
68
232
  #
69
- # @see Concurrent::IVar
233
+ # @see Concurrent::Actor
234
+ # @see Concurrent::SingleThreadExecutor
235
+ # @see https://en.wikipedia.org/wiki/Actor_model "Actor Model" at Wikipedia
236
+ # @see http://www.erlang.org/doc/man/gen_server.html Erlang gen_server
237
+ # @see http://c2.com/cgi/wiki?LetItCrash "Let It Crash" at http://c2.com/
70
238
  module Async
71
239
 
72
240
  # @!method self.new(*args, &block)
@@ -76,7 +244,7 @@ module Concurrent
76
244
  #
77
245
  # @param [Array<Object>] args Zero or more arguments to be passed to the
78
246
  # object's initializer.
79
- # @param [Proc] bloc Optional block to pass to the object's initializer.
247
+ # @param [Proc] block Optional block to pass to the object's initializer.
80
248
  # @return [Object] A properly initialized object of the asynchronous class.
81
249
 
82
250
  # Check for the presence of a method on an object and determine if a given
@@ -138,13 +306,11 @@ module Concurrent
138
306
  # given executor. Block if necessary.
139
307
  #
140
308
  # @param [Object] delegate the object to wrap and delegate method calls to
141
- # @param [Concurrent::Delay] executor a `Delay` wrapping the executor on which to execute delegated method calls
142
- # @param [Concurrent::SerializedExecution] serializer the serializer to use when delegating method calls
309
+ # @param [Concurrent::ExecutorService] executor the executor on which to execute delegated method calls
143
310
  # @param [Boolean] blocking will block awaiting result when `true`
144
- def initialize(delegate, executor, serializer, blocking = false)
311
+ def initialize(delegate, executor, blocking)
145
312
  @delegate = delegate
146
313
  @executor = executor
147
- @serializer = serializer
148
314
  @blocking = blocking
149
315
  end
150
316
 
@@ -163,89 +329,61 @@ module Concurrent
163
329
  super unless @delegate.respond_to?(method)
164
330
  Async::validate_argc(@delegate, method, *args)
165
331
 
166
- self.define_singleton_method(method) do |*args2|
167
- Async::validate_argc(@delegate, method, *args2)
168
- ivar = Concurrent::IVar.new
169
- @serializer.post(@executor.value) do
170
- begin
171
- ivar.set(@delegate.send(method, *args2, &block))
172
- rescue => reason
173
- ivar.fail(reason)
174
- end
332
+ ivar = Concurrent::IVar.new
333
+ @executor.post(args) do |arguments|
334
+ begin
335
+ ivar.set(@delegate.send(method, *arguments, &block))
336
+ rescue => error
337
+ ivar.fail(error)
175
338
  end
176
- ivar.value if @blocking
177
- ivar
178
339
  end
179
-
180
- self.send(method, *args)
340
+ ivar.wait if @blocking
341
+ ivar
181
342
  end
182
343
  end
183
344
  private_constant :AsyncDelegator
184
345
 
185
346
  # Causes the chained method call to be performed asynchronously on the
186
- # global thread pool. The method called by this method will return a
187
- # future object in the `:pending` state and the method call will have
188
- # been scheduled on the global thread pool. The final disposition of the
189
- # method call can be obtained by inspecting the returned future.
190
- #
191
- # Before scheduling the method on the global thread pool a best-effort
192
- # attempt will be made to validate that the method exists on the object
193
- # and that the given arguments match the arity of the requested function.
194
- # Due to the dynamic nature of Ruby and limitations of its reflection
195
- # library, some edge cases will be missed. For more information see
196
- # the documentation for the `validate_argc` method.
347
+ # object's thread. The delegated method will return a future in the
348
+ # `:pending` state and the method call will have been scheduled on the
349
+ # object's thread. The final disposition of the method call can be obtained
350
+ # by inspecting the returned future.
197
351
  #
198
352
  # @!macro [attach] async_thread_safety_warning
199
353
  # @note The method call is guaranteed to be thread safe with respect to
200
354
  # all other method calls against the same object that are called with
201
355
  # either `async` or `await`. The mutable nature of Ruby references
202
356
  # (and object orientation in general) prevent any other thread safety
203
- # guarantees. Do NOT mix non-protected method calls with protected
204
- # method call. Use *only* protected method calls when sharing the object
205
- # between threads.
357
+ # guarantees. Do NOT mix direct method calls with delegated method calls.
358
+ # Use *only* delegated method calls when sharing the object between threads.
206
359
  #
207
360
  # @return [Concurrent::IVar] the pending result of the asynchronous operation
208
361
  #
209
- # @raise [NameError] the object does not respond to `method` method
210
- # @raise [ArgumentError] the given `args` do not match the arity of `method`
211
- #
212
- # @see Concurrent::IVar
362
+ # @raise [NameError] the object does not respond to the requested method
363
+ # @raise [ArgumentError] the given `args` do not match the arity of
364
+ # the requested method
213
365
  def async
214
- @__async_delegator__.value
366
+ @__async_delegator__
215
367
  end
368
+ alias_method :cast, :async
216
369
 
217
370
  # Causes the chained method call to be performed synchronously on the
218
- # current thread. The method called by this method will return an
219
- # `IVar` object in either the `:fulfilled` or `rejected` state and the
220
- # method call will have completed. The final disposition of the
221
- # method call can be obtained by inspecting the returned `IVar`.
222
- #
223
- # Before scheduling the method on the global thread pool a best-effort
224
- # attempt will be made to validate that the method exists on the object
225
- # and that the given arguments match the arity of the requested function.
226
- # Due to the dynamic nature of Ruby and limitations of its reflection
227
- # library, some edge cases will be missed. For more information see
228
- # the documentation for the `validate_argc` method.
371
+ # current thread. The delegated will return a future in either the
372
+ # `:fulfilled` or `:rejected` state and the delegated method will have
373
+ # completed. The final disposition of the delegated method can be obtained
374
+ # by inspecting the returned future.
229
375
  #
230
376
  # @!macro async_thread_safety_warning
231
377
  #
232
378
  # @return [Concurrent::IVar] the completed result of the synchronous operation
233
379
  #
234
- # @raise [NameError] the object does not respond to `method` method
235
- # @raise [ArgumentError] the given `args` do not match the arity of `method`
236
- #
237
- # @see Concurrent::IVar
380
+ # @raise [NameError] the object does not respond to the requested method
381
+ # @raise [ArgumentError] the given `args` do not match the arity of the
382
+ # requested method
238
383
  def await
239
- @__await_delegator__.value
240
- end
241
-
242
- # Set a new executor.
243
- #
244
- # @raise [ArgumentError] executor has already been set.
245
- def executor=(executor)
246
- @__async_executor__.reconfigure { executor } or
247
- raise ArgumentError.new('executor has already been set')
384
+ @__await_delegator__
248
385
  end
386
+ alias_method :call, :await
249
387
 
250
388
  private
251
389
 
@@ -254,27 +392,14 @@ module Concurrent
254
392
  # @note This method *must* be called immediately upon object construction.
255
393
  # This is the only way thread-safe initialization can be guaranteed.
256
394
  #
257
- # @raise [Concurrent::InitializationError] when called more than once
258
- #
259
395
  # @!visibility private
260
396
  def init_synchronization
261
397
  return self if @__async_initialized__
262
-
263
398
  @__async_initialized__ = true
264
- serializer = Concurrent::SerializedExecution.new
265
-
266
- @__async_executor__ = Delay.new {
267
- Concurrent.global_io_executor
268
- }
269
-
270
- @__await_delegator__ = Delay.new {
271
- AsyncDelegator.new(self, Delay.new{ Concurrent::ImmediateExecutor.new }, serializer, true)
272
- }
273
-
274
- @__async_delegator__ = Delay.new {
275
- AsyncDelegator.new(self, @__async_executor__, serializer, false)
276
- }
277
-
399
+ @__async_executor__ = Concurrent::SingleThreadExecutor.new(
400
+ fallback_policy: :caller_runs, auto_terminate: true)
401
+ @__await_delegator__ = AsyncDelegator.new(self, @__async_executor__, true)
402
+ @__async_delegator__ = AsyncDelegator.new(self, @__async_executor__, false)
278
403
  self
279
404
  end
280
405
  end