polyphony 0.77 → 0.80

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 (52) hide show
  1. checksums.yaml +4 -4
  2. data/.github/workflows/test.yml +1 -1
  3. data/CHANGELOG.md +18 -0
  4. data/Gemfile.lock +2 -1
  5. data/examples/core/pingpong.rb +7 -4
  6. data/examples/core/zlib_stream.rb +15 -0
  7. data/ext/polyphony/backend_common.c +16 -8
  8. data/ext/polyphony/backend_common.h +8 -3
  9. data/ext/polyphony/backend_io_uring.c +19 -3
  10. data/ext/polyphony/backend_libev.c +33 -17
  11. data/ext/polyphony/fiber.c +28 -28
  12. data/ext/polyphony/polyphony.c +1 -8
  13. data/ext/polyphony/polyphony.h +11 -8
  14. data/ext/polyphony/queue.c +82 -6
  15. data/ext/polyphony/thread.c +6 -2
  16. data/lib/polyphony/adapters/fs.rb +4 -0
  17. data/lib/polyphony/adapters/process.rb +14 -1
  18. data/lib/polyphony/adapters/redis.rb +28 -0
  19. data/lib/polyphony/adapters/sequel.rb +19 -1
  20. data/lib/polyphony/core/debug.rb +203 -0
  21. data/lib/polyphony/core/exceptions.rb +21 -6
  22. data/lib/polyphony/core/global_api.rb +228 -73
  23. data/lib/polyphony/core/resource_pool.rb +65 -20
  24. data/lib/polyphony/core/sync.rb +57 -12
  25. data/lib/polyphony/core/thread_pool.rb +42 -5
  26. data/lib/polyphony/core/throttler.rb +21 -5
  27. data/lib/polyphony/core/timer.rb +125 -1
  28. data/lib/polyphony/extensions/exception.rb +36 -6
  29. data/lib/polyphony/extensions/fiber.rb +244 -61
  30. data/lib/polyphony/extensions/io.rb +4 -2
  31. data/lib/polyphony/extensions/kernel.rb +9 -4
  32. data/lib/polyphony/extensions/object.rb +8 -0
  33. data/lib/polyphony/extensions/openssl.rb +3 -1
  34. data/lib/polyphony/extensions/socket.rb +458 -39
  35. data/lib/polyphony/extensions/thread.rb +108 -43
  36. data/lib/polyphony/extensions/timeout.rb +12 -1
  37. data/lib/polyphony/extensions.rb +1 -0
  38. data/lib/polyphony/net.rb +59 -0
  39. data/lib/polyphony/version.rb +1 -1
  40. data/lib/polyphony.rb +0 -2
  41. data/test/test_backend.rb +6 -2
  42. data/test/test_global_api.rb +0 -23
  43. data/test/test_io.rb +7 -7
  44. data/test/test_queue.rb +103 -1
  45. data/test/test_resource_pool.rb +1 -1
  46. data/test/test_signal.rb +15 -15
  47. data/test/test_supervise.rb +27 -0
  48. data/test/test_thread.rb +1 -1
  49. data/test/test_throttler.rb +0 -6
  50. data/test/test_trace.rb +189 -24
  51. metadata +9 -8
  52. data/lib/polyphony/core/channel.rb +0 -15
checksums.yaml CHANGED
@@ -1,7 +1,7 @@
1
1
  ---
2
2
  SHA256:
3
- metadata.gz: 10f049946dc02d9cdd984cc8658687d2319e0fd85f6f2cce74aa0a887bc714cb
4
- data.tar.gz: 5118859d4640aebeddb9186c43d58617e9b455a63ea97c1dc8283432db954fd1
3
+ metadata.gz: cf3658689c8bb7614624ad75492b16fc31401c5c8ee37cd510bb4aab1c9d1449
4
+ data.tar.gz: 5e2520d758db10e9dfe8022d6a85df61e1702f2e1e6560a95968e37b6f4f6e82
5
5
  SHA512:
6
- metadata.gz: e107e21d6c42f9cd9e91521e288dfee37a6f0da4b7d8a8d311924bf88ccf732d2cc94de834dbbaa3c7a974e5bb84ce1d8d20efd68870c9f2dac9b73155d28558
7
- data.tar.gz: 8e4f7005c9cbc46cd084bbe66a4e8f7145f6a4a3062ce027a70eee07c0e648f79289515591ef4b48010580b7125c403ab09e45417781346ebf6da4b2d528ec45
6
+ metadata.gz: 7bd42d8fed28064941281c0e010f86a7ff7fff56e28c8e8190c8bc5c68eebd26a8d568e03ce20120db64684a036afb8d9b8586668c59f39fadebdaba2624ba75
7
+ data.tar.gz: cd91d0394a0642fbb43d59cc3ba6a3100377be955f1cabeebc051f8f6159d6fb0e78fa7deedabc6bc4fabc20b12453c963b85ed9ce76002211d3423b05f51542
@@ -19,7 +19,7 @@ jobs:
19
19
  POLYPHONY_USE_LIBEV: "1"
20
20
 
21
21
  steps:
22
- - name: Setup OS
22
+ - name: Setup machine
23
23
  uses: actions/checkout@v1
24
24
  - name: Setup Ruby
25
25
  uses: ruby/setup-ruby@v1
data/CHANGELOG.md CHANGED
@@ -1,3 +1,21 @@
1
+ ## 0.80 2022-02-28
2
+
3
+ - Prevent reentry into `trace_proc`
4
+ - Rename `__parser_read_method__` to `__read_method__`
5
+ - Rename `ResourcePool#preheat!` to `#fill`.
6
+ - Remove ability to use `#cancel_after` or `#move_on` without a block
7
+ - Add #move_on alias to `Fiber#interrupt`
8
+ - Allow specifying exception in `Fiber#cancel`
9
+ - Remove deprecated `Polyphony::Channel` class
10
+
11
+ ## 0.79 2022-02-19
12
+
13
+ - Overhaul trace events system (#73)
14
+
15
+ ## 0.78 2022-02-16
16
+
17
+ - Fix Polyphony::Queue API compatibility (#72)
18
+
1
19
  ## 0.77 2022-02-07
2
20
 
3
21
  - Fix behaviour of signal traps (#71)
data/Gemfile.lock CHANGED
@@ -1,7 +1,7 @@
1
1
  PATH
2
2
  remote: .
3
3
  specs:
4
- polyphony (0.77)
4
+ polyphony (0.80)
5
5
 
6
6
  GEM
7
7
  remote: https://rubygems.org/
@@ -60,6 +60,7 @@ GEM
60
60
  unicode-display_width (1.8.0)
61
61
 
62
62
  PLATFORMS
63
+ ruby
63
64
  universal-darwin
64
65
  universal-freebsd
65
66
  universal-linux
@@ -3,18 +3,21 @@
3
3
  require 'bundler/setup'
4
4
  require 'polyphony'
5
5
 
6
- pong = spin_loop do
6
+ require 'polyphony/core/debug'
7
+ Polyphony::Trace.start_event_firehose(STDOUT)
8
+
9
+ pong = spin_loop(:pong) do
7
10
  msg, ping = receive
8
11
  puts msg
9
12
  ping << 'pong'
10
13
  end
11
14
 
12
- ping = spin do
13
- 3.times do
15
+ ping = spin(:ping) do
16
+ 1.times do
14
17
  pong << ['ping', Fiber.current]
15
18
  msg = receive
16
19
  puts msg
17
20
  end
18
21
  end
19
22
 
20
- ping.await
23
+ ping.await
@@ -0,0 +1,15 @@
1
+ # frozen_string_literal: true
2
+
3
+ require 'bundler/setup'
4
+ require 'polyphony'
5
+ require 'zlib'
6
+
7
+ r, w = IO.pipe
8
+ writer = Zlib::GzipWriter.new(w)
9
+
10
+ writer << 'chunk'
11
+ writer.flush
12
+ p pos: writer.pos
13
+ w.close
14
+
15
+ p r.read
@@ -17,6 +17,7 @@ inline void backend_base_initialize(struct Backend_base *base) {
17
17
  base->idle_gc_last_time = 0;
18
18
  base->idle_proc = Qnil;
19
19
  base->trace_proc = Qnil;
20
+ base->in_trace_proc = 0;
20
21
  }
21
22
 
22
23
  inline void backend_base_finalize(struct Backend_base *base) {
@@ -65,7 +66,7 @@ VALUE backend_base_switch_fiber(VALUE backend, struct Backend_base *base) {
65
66
 
66
67
  base->switch_count++;
67
68
  if (SHOULD_TRACE(base))
68
- TRACE(base, 3, SYM_fiber_switchpoint, current_fiber, CALLER());
69
+ TRACE(base, 3, SYM_block, current_fiber, CALLER());
69
70
 
70
71
  while (1) {
71
72
  next = runqueue_shift(&base->runqueue);
@@ -96,8 +97,6 @@ VALUE backend_base_switch_fiber(VALUE backend, struct Backend_base *base) {
96
97
  if (next.fiber == Qnil) return Qnil;
97
98
 
98
99
  // run next fiber
99
- COND_TRACE(base, 3, SYM_fiber_run, next.fiber, next.value);
100
-
101
100
  rb_ivar_set(next.fiber, ID_ivar_runnable, Qnil);
102
101
  RB_GC_GUARD(next.fiber);
103
102
  RB_GC_GUARD(next.value);
@@ -112,7 +111,7 @@ void backend_base_schedule_fiber(VALUE thread, VALUE backend, struct Backend_bas
112
111
  if (rb_fiber_alive_p(fiber) != Qtrue) return;
113
112
  already_runnable = rb_ivar_get(fiber, ID_ivar_runnable) != Qnil;
114
113
 
115
- COND_TRACE(base, 4, SYM_fiber_schedule, fiber, value, prioritize ? Qtrue : Qfalse);
114
+ COND_TRACE(base, 5, SYM_schedule, fiber, value, prioritize ? Qtrue : Qfalse, CALLER());
116
115
 
117
116
  runqueue = rb_ivar_get(fiber, ID_ivar_parked) == Qtrue ? &base->parked_runqueue : &base->runqueue;
118
117
 
@@ -139,7 +138,7 @@ inline void backend_base_unpark_fiber(struct Backend_base *base, VALUE fiber) {
139
138
  }
140
139
 
141
140
  inline void backend_trace(struct Backend_base *base, int argc, VALUE *argv) {
142
- if (base->trace_proc == Qnil) return;
141
+ if (base->trace_proc == Qnil || base->in_trace_proc) return;
143
142
 
144
143
  rb_funcallv(base->trace_proc, ID_call, argc, argv);
145
144
  }
@@ -244,15 +243,24 @@ inline VALUE backend_await(struct Backend_base *backend) {
244
243
  VALUE ret;
245
244
  backend->pending_count++;
246
245
  ret = Thread_switch_fiber(rb_thread_current());
246
+
247
+ // run next fiber
248
+ COND_TRACE(backend, 4, SYM_unblock, rb_fiber_current(), ret, CALLER());
249
+
247
250
  backend->pending_count--;
248
251
  RB_GC_GUARD(ret);
249
252
  return ret;
250
253
  }
251
254
 
252
- inline VALUE backend_snooze() {
255
+ inline VALUE backend_snooze(struct Backend_base *backend) {
253
256
  VALUE ret;
254
- Fiber_make_runnable(rb_fiber_current(), Qnil);
255
- ret = Thread_switch_fiber(rb_thread_current());
257
+ VALUE fiber = rb_fiber_current();
258
+ VALUE thread = rb_thread_current();
259
+ Fiber_make_runnable(fiber, Qnil);
260
+ ret = Thread_switch_fiber(thread);
261
+
262
+ COND_TRACE(backend, 4, SYM_unblock, fiber, ret, CALLER());
263
+
256
264
  return ret;
257
265
  }
258
266
 
@@ -32,6 +32,7 @@ struct Backend_base {
32
32
  double idle_gc_last_time;
33
33
  VALUE idle_proc;
34
34
  VALUE trace_proc;
35
+ unsigned int in_trace_proc;
35
36
  };
36
37
 
37
38
  void backend_base_initialize(struct Backend_base *base);
@@ -46,8 +47,12 @@ void backend_trace(struct Backend_base *base, int argc, VALUE *argv);
46
47
  struct backend_stats backend_base_stats(struct Backend_base *base);
47
48
 
48
49
  // tracing
49
- #define SHOULD_TRACE(base) ((base)->trace_proc != Qnil)
50
- #define TRACE(base, ...) rb_funcall((base)->trace_proc, ID_call, __VA_ARGS__)
50
+ #define SHOULD_TRACE(base) ((base)->trace_proc != Qnil && !(base)->in_trace_proc)
51
+ #define TRACE(base, ...) { \
52
+ (base)->in_trace_proc = 1; \
53
+ rb_funcall((base)->trace_proc, ID_call, __VA_ARGS__); \
54
+ (base)->in_trace_proc = 0; \
55
+ }
51
56
  #define COND_TRACE(base, ...) if (SHOULD_TRACE(base)) { TRACE(base, __VA_ARGS__); }
52
57
 
53
58
 
@@ -80,7 +85,7 @@ void fptr_finalize(rb_io_t *fptr);
80
85
 
81
86
  struct backend_stats backend_get_stats(VALUE self);
82
87
  VALUE backend_await(struct Backend_base *backend);
83
- VALUE backend_snooze();
88
+ VALUE backend_snooze(struct Backend_base *backend);
84
89
 
85
90
  // macros for doing read loops
86
91
  #define READ_LOOP_PREPARE_STR() { \
@@ -216,12 +216,12 @@ inline VALUE Backend_poll(VALUE self, VALUE blocking) {
216
216
  io_uring_submit(&backend->ring);
217
217
  }
218
218
 
219
- COND_TRACE(&backend->base, 2, SYM_fiber_event_poll_enter, rb_fiber_current());
219
+ COND_TRACE(&backend->base, 2, SYM_enter_poll, rb_fiber_current());
220
220
 
221
221
  if (is_blocking) io_uring_backend_poll(backend);
222
222
  io_uring_backend_handle_ready_cqes(backend);
223
223
 
224
- COND_TRACE(&backend->base, 2, SYM_fiber_event_poll_leave, rb_fiber_current());
224
+ COND_TRACE(&backend->base, 2, SYM_leave_poll, rb_fiber_current());
225
225
 
226
226
  return self;
227
227
  }
@@ -1108,7 +1108,7 @@ VALUE Backend_timer_loop(VALUE self, VALUE interval) {
1108
1108
  if (!completed) return resume_value;
1109
1109
  }
1110
1110
  else {
1111
- resume_value = backend_snooze();
1111
+ resume_value = backend_snooze(&backend->base);
1112
1112
  RAISE_IF_EXCEPTION(resume_value);
1113
1113
  }
1114
1114
 
@@ -1591,6 +1591,22 @@ VALUE Backend_trace_proc_set(VALUE self, VALUE block) {
1591
1591
  return self;
1592
1592
  }
1593
1593
 
1594
+ VALUE Backend_snooze(VALUE self) {
1595
+ VALUE ret;
1596
+ VALUE fiber = rb_fiber_current();
1597
+ Backend_t *backend;
1598
+ GetBackend(self, backend);
1599
+
1600
+ Fiber_make_runnable(fiber, Qnil);
1601
+ ret = Thread_switch_fiber(rb_thread_current());
1602
+
1603
+ COND_TRACE(&backend->base, 4, SYM_unblock, rb_fiber_current(), ret, CALLER());
1604
+
1605
+ RAISE_IF_EXCEPTION(ret);
1606
+ RB_GC_GUARD(ret);
1607
+ return ret;
1608
+ }
1609
+
1594
1610
  void Backend_park_fiber(VALUE self, VALUE fiber) {
1595
1611
  Backend_t *backend;
1596
1612
  GetBackend(self, backend);
@@ -168,7 +168,7 @@ inline VALUE Backend_poll(VALUE self, VALUE blocking) {
168
168
 
169
169
  backend->base.poll_count++;
170
170
 
171
- COND_TRACE(&backend->base, 2, SYM_fiber_event_poll_enter, rb_fiber_current());
171
+ COND_TRACE(&backend->base, 2, SYM_enter_poll, rb_fiber_current());
172
172
 
173
173
  ev_run:
174
174
  backend->base.currently_polling = 1;
@@ -177,7 +177,7 @@ ev_run:
177
177
  backend->base.currently_polling = 0;
178
178
  if (errno == EINTR && runqueue_empty_p(&backend->base.runqueue)) goto ev_run;
179
179
 
180
- COND_TRACE(&backend->base, 2, SYM_fiber_event_poll_leave, rb_fiber_current());
180
+ COND_TRACE(&backend->base, 2, SYM_leave_poll, rb_fiber_current());
181
181
 
182
182
  return self;
183
183
  }
@@ -305,7 +305,7 @@ VALUE Backend_read(VALUE self, VALUE io, VALUE str, VALUE length, VALUE to_eof,
305
305
  if (TEST_EXCEPTION(switchpoint_result)) goto error;
306
306
  }
307
307
  else {
308
- switchpoint_result = backend_snooze();
308
+ switchpoint_result = backend_snooze(&backend->base);
309
309
  if (TEST_EXCEPTION(switchpoint_result)) goto error;
310
310
 
311
311
  if (n == 0) break; // EOF
@@ -375,7 +375,7 @@ VALUE Backend_read_loop(VALUE self, VALUE io, VALUE maxlen) {
375
375
  if (TEST_EXCEPTION(switchpoint_result)) goto error;
376
376
  }
377
377
  else {
378
- switchpoint_result = backend_snooze();
378
+ switchpoint_result = backend_snooze(&backend->base);
379
379
 
380
380
  if (TEST_EXCEPTION(switchpoint_result)) goto error;
381
381
 
@@ -428,7 +428,7 @@ VALUE Backend_feed_loop(VALUE self, VALUE io, VALUE receiver, VALUE method) {
428
428
  if (TEST_EXCEPTION(switchpoint_result)) goto error;
429
429
  }
430
430
  else {
431
- switchpoint_result = backend_snooze();
431
+ switchpoint_result = backend_snooze(&backend->base);
432
432
 
433
433
  if (TEST_EXCEPTION(switchpoint_result)) goto error;
434
434
 
@@ -483,7 +483,7 @@ VALUE Backend_write(VALUE self, VALUE io, VALUE str) {
483
483
  }
484
484
 
485
485
  if (watcher.fiber == Qnil) {
486
- switchpoint_result = backend_snooze();
486
+ switchpoint_result = backend_snooze(&backend->base);
487
487
 
488
488
  if (TEST_EXCEPTION(switchpoint_result)) goto error;
489
489
  }
@@ -558,7 +558,7 @@ VALUE Backend_writev(VALUE self, VALUE io, int argc, VALUE *argv) {
558
558
  }
559
559
  }
560
560
  if (watcher.fiber == Qnil) {
561
- switchpoint_result = backend_snooze();
561
+ switchpoint_result = backend_snooze(&backend->base);
562
562
  if (TEST_EXCEPTION(switchpoint_result)) goto error;
563
563
  }
564
564
 
@@ -611,7 +611,7 @@ VALUE Backend_accept(VALUE self, VALUE server_socket, VALUE socket_class) {
611
611
  else {
612
612
  VALUE socket;
613
613
  rb_io_t *fp;
614
- switchpoint_result = backend_snooze();
614
+ switchpoint_result = backend_snooze(&backend->base);
615
615
 
616
616
  if (TEST_EXCEPTION(switchpoint_result)) {
617
617
  close(fd); // close fd since we're raising an exception
@@ -669,7 +669,7 @@ VALUE Backend_accept_loop(VALUE self, VALUE server_socket, VALUE socket_class) {
669
669
  }
670
670
  else {
671
671
  rb_io_t *fp;
672
- switchpoint_result = backend_snooze();
672
+ switchpoint_result = backend_snooze(&backend->base);
673
673
 
674
674
  if (TEST_EXCEPTION(switchpoint_result)) {
675
675
  close(fd); // close fd since we're raising an exception
@@ -727,7 +727,7 @@ VALUE Backend_connect(VALUE self, VALUE sock, VALUE host, VALUE port) {
727
727
  if (TEST_EXCEPTION(switchpoint_result)) goto error;
728
728
  }
729
729
  else {
730
- switchpoint_result = backend_snooze();
730
+ switchpoint_result = backend_snooze(&backend->base);
731
731
 
732
732
  if (TEST_EXCEPTION(switchpoint_result)) goto error;
733
733
  }
@@ -774,7 +774,7 @@ VALUE Backend_send(VALUE self, VALUE io, VALUE str, VALUE flags) {
774
774
  }
775
775
 
776
776
  if (watcher.fiber == Qnil) {
777
- switchpoint_result = backend_snooze();
777
+ switchpoint_result = backend_snooze(&backend->base);
778
778
 
779
779
  if (TEST_EXCEPTION(switchpoint_result)) goto error;
780
780
  }
@@ -880,7 +880,7 @@ VALUE Backend_splice(VALUE self, VALUE src, VALUE dest, VALUE maxlen) {
880
880
  }
881
881
 
882
882
  if (watcher.ctx.fiber == Qnil) {
883
- switchpoint_result = backend_snooze();
883
+ switchpoint_result = backend_snooze(&backend->base);
884
884
  if (TEST_EXCEPTION(switchpoint_result)) goto error;
885
885
  }
886
886
 
@@ -935,7 +935,7 @@ VALUE Backend_splice_to_eof(VALUE self, VALUE src, VALUE dest, VALUE maxlen) {
935
935
  }
936
936
 
937
937
  if (watcher.ctx.fiber == Qnil) {
938
- switchpoint_result = backend_snooze();
938
+ switchpoint_result = backend_snooze(&backend->base);
939
939
  if (TEST_EXCEPTION(switchpoint_result)) goto error;
940
940
  }
941
941
 
@@ -1009,7 +1009,7 @@ VALUE Backend_splice(VALUE self, VALUE src, VALUE dest, VALUE maxlen) {
1009
1009
  }
1010
1010
 
1011
1011
  if (watcher.fiber == Qnil) {
1012
- switchpoint_result = backend_snooze();
1012
+ switchpoint_result = backend_snooze(&backend->base);
1013
1013
  if (TEST_EXCEPTION(switchpoint_result)) goto error;
1014
1014
  }
1015
1015
 
@@ -1089,7 +1089,7 @@ VALUE Backend_splice_to_eof(VALUE self, VALUE src, VALUE dest, VALUE maxlen) {
1089
1089
 
1090
1090
  done:
1091
1091
  if (watcher.fiber == Qnil) {
1092
- switchpoint_result = backend_snooze();
1092
+ switchpoint_result = backend_snooze(&backend->base);
1093
1093
  if (TEST_EXCEPTION(switchpoint_result)) goto error;
1094
1094
  }
1095
1095
 
@@ -1171,7 +1171,7 @@ noreturn VALUE Backend_timer_loop(VALUE self, VALUE interval) {
1171
1171
  RAISE_IF_EXCEPTION(resume_value);
1172
1172
  }
1173
1173
  else {
1174
- resume_value = backend_snooze();
1174
+ resume_value = backend_snooze(&backend->base);
1175
1175
  RAISE_IF_EXCEPTION(resume_value);
1176
1176
  }
1177
1177
 
@@ -1530,7 +1530,7 @@ VALUE Backend_splice_chunks(VALUE self, VALUE src, VALUE dest, VALUE prefix, VAL
1530
1530
  }
1531
1531
 
1532
1532
  if (watcher.ctx.fiber == Qnil) {
1533
- result = backend_snooze();
1533
+ result = backend_snooze(&backend->base);
1534
1534
  if (TEST_EXCEPTION(result)) goto error;
1535
1535
  }
1536
1536
  RB_GC_GUARD(str);
@@ -1564,6 +1564,22 @@ VALUE Backend_trace_proc_set(VALUE self, VALUE block) {
1564
1564
  return self;
1565
1565
  }
1566
1566
 
1567
+ VALUE Backend_snooze(VALUE self) {
1568
+ VALUE ret;
1569
+ VALUE fiber = rb_fiber_current();
1570
+ Backend_t *backend;
1571
+ GetBackend(self, backend);
1572
+
1573
+ Fiber_make_runnable(fiber, Qnil);
1574
+ ret = Thread_switch_fiber(rb_thread_current());
1575
+
1576
+ COND_TRACE(&backend->base, 4, SYM_unblock, rb_fiber_current(), ret, CALLER());
1577
+
1578
+ RAISE_IF_EXCEPTION(ret);
1579
+ RB_GC_GUARD(ret);
1580
+ return ret;
1581
+ }
1582
+
1567
1583
  void Backend_park_fiber(VALUE self, VALUE fiber) {
1568
1584
  Backend_t *backend;
1569
1585
  GetBackend(self, backend);
@@ -10,13 +10,13 @@ VALUE SYM_running;
10
10
  VALUE SYM_runnable;
11
11
  VALUE SYM_waiting;
12
12
 
13
- VALUE SYM_fiber_create;
14
- VALUE SYM_fiber_event_poll_enter;
15
- VALUE SYM_fiber_event_poll_leave;
16
- VALUE SYM_fiber_run;
17
- VALUE SYM_fiber_schedule;
18
- VALUE SYM_fiber_switchpoint;
19
- VALUE SYM_fiber_terminate;
13
+ VALUE SYM_spin;
14
+ VALUE SYM_enter_poll;
15
+ VALUE SYM_leave_poll;
16
+ VALUE SYM_unblock;
17
+ VALUE SYM_schedule;
18
+ VALUE SYM_block;
19
+ VALUE SYM_terminate;
20
20
 
21
21
  static VALUE Fiber_safe_transfer(int argc, VALUE *argv, VALUE self) {
22
22
  VALUE arg = (argc == 0) ? Qnil : argv[0];
@@ -97,7 +97,7 @@ VALUE Fiber_receive(VALUE self) {
97
97
  mailbox = rb_funcall(cQueue, ID_new, 0);
98
98
  rb_ivar_set(self, ID_ivar_mailbox, mailbox);
99
99
  }
100
- return Queue_shift(mailbox);
100
+ return Queue_shift(0, 0, mailbox);
101
101
  }
102
102
 
103
103
  VALUE Fiber_mailbox(VALUE self) {
@@ -157,24 +157,24 @@ void Init_Fiber() {
157
157
  rb_global_variable(&SYM_runnable);
158
158
  rb_global_variable(&SYM_waiting);
159
159
 
160
- ID_ivar_auto_watcher = rb_intern("@auto_watcher");
161
- ID_ivar_mailbox = rb_intern("@mailbox");
162
- ID_ivar_result = rb_intern("@result");
163
- ID_ivar_waiting_fibers = rb_intern("@waiting_fibers");
164
-
165
- SYM_fiber_create = ID2SYM(rb_intern("fiber_create"));
166
- SYM_fiber_event_poll_enter = ID2SYM(rb_intern("fiber_event_poll_enter"));
167
- SYM_fiber_event_poll_leave = ID2SYM(rb_intern("fiber_event_poll_leave"));
168
- SYM_fiber_run = ID2SYM(rb_intern("fiber_run"));
169
- SYM_fiber_schedule = ID2SYM(rb_intern("fiber_schedule"));
170
- SYM_fiber_switchpoint = ID2SYM(rb_intern("fiber_switchpoint"));
171
- SYM_fiber_terminate = ID2SYM(rb_intern("fiber_terminate"));
172
-
173
- rb_global_variable(&SYM_fiber_create);
174
- rb_global_variable(&SYM_fiber_event_poll_enter);
175
- rb_global_variable(&SYM_fiber_event_poll_leave);
176
- rb_global_variable(&SYM_fiber_run);
177
- rb_global_variable(&SYM_fiber_schedule);
178
- rb_global_variable(&SYM_fiber_switchpoint);
179
- rb_global_variable(&SYM_fiber_terminate);
160
+ ID_ivar_auto_watcher = rb_intern("@auto_watcher");
161
+ ID_ivar_mailbox = rb_intern("@mailbox");
162
+ ID_ivar_result = rb_intern("@result");
163
+ ID_ivar_waiting_fibers = rb_intern("@waiting_fibers");
164
+
165
+ SYM_spin = ID2SYM(rb_intern("spin"));
166
+ SYM_enter_poll = ID2SYM(rb_intern("enter_poll"));
167
+ SYM_leave_poll = ID2SYM(rb_intern("leave_poll"));
168
+ SYM_unblock = ID2SYM(rb_intern("unblock"));
169
+ SYM_schedule = ID2SYM(rb_intern("schedule"));
170
+ SYM_block = ID2SYM(rb_intern("block"));
171
+ SYM_terminate = ID2SYM(rb_intern("terminate"));
172
+
173
+ rb_global_variable(&SYM_spin);
174
+ rb_global_variable(&SYM_enter_poll);
175
+ rb_global_variable(&SYM_leave_poll);
176
+ rb_global_variable(&SYM_unblock);
177
+ rb_global_variable(&SYM_schedule);
178
+ rb_global_variable(&SYM_block);
179
+ rb_global_variable(&SYM_terminate);
180
180
  }
@@ -27,14 +27,7 @@ ID ID_W;
27
27
  ID ID_RW;
28
28
 
29
29
  VALUE Polyphony_snooze(VALUE self) {
30
- VALUE ret;
31
- VALUE fiber = rb_fiber_current();
32
-
33
- Fiber_make_runnable(fiber, Qnil);
34
- ret = Thread_switch_fiber(rb_thread_current());
35
- RAISE_IF_EXCEPTION(ret);
36
- RB_GC_GUARD(ret);
37
- return ret;
30
+ return Backend_snooze(BACKEND());
38
31
  }
39
32
 
40
33
  static VALUE Polyphony_suspend(VALUE self) {
@@ -61,20 +61,20 @@ extern ID ID_switch_fiber;
61
61
  extern ID ID_to_s;
62
62
  extern ID ID_transfer;
63
63
 
64
- extern VALUE SYM_fiber_create;
65
- extern VALUE SYM_fiber_event_poll_enter;
66
- extern VALUE SYM_fiber_event_poll_leave;
67
- extern VALUE SYM_fiber_run;
68
- extern VALUE SYM_fiber_schedule;
69
- extern VALUE SYM_fiber_switchpoint;
70
- extern VALUE SYM_fiber_terminate;
64
+ extern VALUE SYM_spin;
65
+ extern VALUE SYM_enter_poll;
66
+ extern VALUE SYM_leave_poll;
67
+ extern VALUE SYM_unblock;
68
+ extern VALUE SYM_schedule;
69
+ extern VALUE SYM_block;
70
+ extern VALUE SYM_terminate;
71
71
 
72
72
  VALUE Fiber_auto_watcher(VALUE self);
73
73
  void Fiber_make_runnable(VALUE fiber, VALUE value);
74
74
 
75
75
  VALUE Queue_push(VALUE self, VALUE value);
76
76
  VALUE Queue_unshift(VALUE self, VALUE value);
77
- VALUE Queue_shift(VALUE self);
77
+ VALUE Queue_shift(int argc,VALUE *argv, VALUE self);
78
78
  VALUE Queue_shift_all(VALUE self);
79
79
 
80
80
  void Runqueue_push(VALUE self, VALUE fiber, VALUE value, int reschedule);
@@ -121,11 +121,14 @@ VALUE Backend_wait_event(VALUE self, VALUE raise_on_exception);
121
121
  VALUE Backend_wakeup(VALUE self);
122
122
  VALUE Backend_run_idle_tasks(VALUE self);
123
123
  VALUE Backend_switch_fiber(VALUE self);
124
+
124
125
  void Backend_schedule_fiber(VALUE thread, VALUE self, VALUE fiber, VALUE value, int prioritize);
125
126
  void Backend_unschedule_fiber(VALUE self, VALUE fiber);
126
127
  void Backend_park_fiber(VALUE self, VALUE fiber);
127
128
  void Backend_unpark_fiber(VALUE self, VALUE fiber);
128
129
 
130
+ VALUE Backend_snooze(VALUE self);
131
+
129
132
  void Thread_schedule_fiber(VALUE thread, VALUE fiber, VALUE value);
130
133
  void Thread_schedule_fiber_with_priority(VALUE thread, VALUE fiber, VALUE value);
131
134
  VALUE Thread_switch_fiber(VALUE thread);