polyphony 0.78 → 0.81

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 (48) hide show
  1. checksums.yaml +4 -4
  2. data/CHANGELOG.md +19 -0
  3. data/Gemfile.lock +2 -1
  4. data/examples/core/pingpong.rb +7 -4
  5. data/examples/core/zlib_stream.rb +15 -0
  6. data/ext/polyphony/backend_common.c +16 -8
  7. data/ext/polyphony/backend_common.h +9 -3
  8. data/ext/polyphony/backend_io_uring.c +85 -31
  9. data/ext/polyphony/backend_libev.c +33 -17
  10. data/ext/polyphony/fiber.c +27 -27
  11. data/ext/polyphony/polyphony.c +9 -8
  12. data/ext/polyphony/polyphony.h +21 -7
  13. data/ext/polyphony/thread.c +6 -2
  14. data/lib/polyphony/adapters/fs.rb +4 -0
  15. data/lib/polyphony/adapters/process.rb +14 -1
  16. data/lib/polyphony/adapters/redis.rb +28 -0
  17. data/lib/polyphony/adapters/sequel.rb +19 -1
  18. data/lib/polyphony/core/debug.rb +201 -0
  19. data/lib/polyphony/core/exceptions.rb +21 -6
  20. data/lib/polyphony/core/global_api.rb +228 -73
  21. data/lib/polyphony/core/resource_pool.rb +65 -20
  22. data/lib/polyphony/core/sync.rb +57 -12
  23. data/lib/polyphony/core/thread_pool.rb +42 -5
  24. data/lib/polyphony/core/throttler.rb +21 -5
  25. data/lib/polyphony/core/timer.rb +125 -1
  26. data/lib/polyphony/extensions/exception.rb +36 -6
  27. data/lib/polyphony/extensions/fiber.rb +244 -61
  28. data/lib/polyphony/extensions/io.rb +4 -2
  29. data/lib/polyphony/extensions/kernel.rb +9 -4
  30. data/lib/polyphony/extensions/object.rb +8 -0
  31. data/lib/polyphony/extensions/openssl.rb +3 -1
  32. data/lib/polyphony/extensions/socket.rb +458 -39
  33. data/lib/polyphony/extensions/thread.rb +108 -43
  34. data/lib/polyphony/extensions/timeout.rb +12 -1
  35. data/lib/polyphony/extensions.rb +1 -0
  36. data/lib/polyphony/net.rb +66 -7
  37. data/lib/polyphony/version.rb +1 -1
  38. data/lib/polyphony.rb +0 -2
  39. data/test/test_backend.rb +6 -2
  40. data/test/test_global_api.rb +0 -23
  41. data/test/test_io.rb +7 -7
  42. data/test/test_resource_pool.rb +1 -1
  43. data/test/test_signal.rb +15 -15
  44. data/test/test_thread.rb +1 -1
  45. data/test/test_throttler.rb +0 -6
  46. data/test/test_trace.rb +189 -24
  47. metadata +9 -8
  48. data/lib/polyphony/core/channel.rb +0 -15
checksums.yaml CHANGED
@@ -1,7 +1,7 @@
1
1
  ---
2
2
  SHA256:
3
- metadata.gz: 51eecc20956cb4f1a35a7f6510dea2730652700e99d29c7d42ade201a119ea2e
4
- data.tar.gz: bc2b57dc2cc5a8918bba1d8bf526cdbe23f44d134f9bdca5018b9e9e66930f91
3
+ metadata.gz: aef5f8ee7585e1ae6a6632099329ba35b7dc6959dba5041f4cdcbaf92caded72
4
+ data.tar.gz: b168b56f080029e4167e2528e66b83186a635ea4bda8dc74882dfe70639421fc
5
5
  SHA512:
6
- metadata.gz: 21c446f9a6fa032577b245e6de5155832c8e6932d6c17e58824a55923c82be014453b64b29ccbb66fe5d01cde8622d3275664acda403637954f1e7e6851e93c7
7
- data.tar.gz: a57831fe861ac51fa43692b66175291a457077822cdb5f795a093fca1e44bf95fb2b6a312cfe8a6c2b3c3e873319ffd03436e6236874a6acdd45843a21bc4981
6
+ metadata.gz: aa7bd86706d5879c884b4f5bdc7a10556cbe28e6737ff89e5107589e1630028098dd815424ef357e51879a7e689cd5127ea30105eb1c36a174da7385b8f6f433
7
+ data.tar.gz: 3c4b35eed6900d55c5295c366803b944a6e423775ca5278f4b2600a309cb547cda12a2167ae40a4ef8e0001a1b6a8b8ea936129246f24d87b0eec59c9d31a3b4
data/CHANGELOG.md CHANGED
@@ -1,3 +1,22 @@
1
+ ## 0.81 2022-03-03
2
+
3
+ - Restore public visibility for `Polyphony::Process.kill_process`
4
+ - Restore public visibility for `Polyphony::Net.setup_alpn`
5
+
6
+ ## 0.80 2022-02-28
7
+
8
+ - Prevent reentry into `trace_proc`
9
+ - Rename `__parser_read_method__` to `__read_method__`
10
+ - Rename `ResourcePool#preheat!` to `#fill`.
11
+ - Remove ability to use `#cancel_after` or `#move_on` without a block
12
+ - Add #move_on alias to `Fiber#interrupt`
13
+ - Allow specifying exception in `Fiber#cancel`
14
+ - Remove deprecated `Polyphony::Channel` class
15
+
16
+ ## 0.79 2022-02-19
17
+
18
+ - Overhaul trace events system (#73)
19
+
1
20
  ## 0.78 2022-02-16
2
21
 
3
22
  - Fix Polyphony::Queue API compatibility (#72)
data/Gemfile.lock CHANGED
@@ -1,7 +1,7 @@
1
1
  PATH
2
2
  remote: .
3
3
  specs:
4
- polyphony (0.78)
4
+ polyphony (0.81)
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
 
@@ -9,6 +9,7 @@
9
9
  #include "ruby.h"
10
10
  #include "ruby/io.h"
11
11
  #include "runqueue.h"
12
+ #include "polyphony.h"
12
13
 
13
14
  struct backend_stats {
14
15
  unsigned int runqueue_size;
@@ -32,6 +33,7 @@ struct Backend_base {
32
33
  double idle_gc_last_time;
33
34
  VALUE idle_proc;
34
35
  VALUE trace_proc;
36
+ unsigned int in_trace_proc;
35
37
  };
36
38
 
37
39
  void backend_base_initialize(struct Backend_base *base);
@@ -46,8 +48,12 @@ void backend_trace(struct Backend_base *base, int argc, VALUE *argv);
46
48
  struct backend_stats backend_base_stats(struct Backend_base *base);
47
49
 
48
50
  // tracing
49
- #define SHOULD_TRACE(base) ((base)->trace_proc != Qnil)
50
- #define TRACE(base, ...) rb_funcall((base)->trace_proc, ID_call, __VA_ARGS__)
51
+ #define SHOULD_TRACE(base) ((base)->trace_proc != Qnil && !(base)->in_trace_proc)
52
+ #define TRACE(base, ...) { \
53
+ (base)->in_trace_proc = 1; \
54
+ rb_funcall((base)->trace_proc, ID_call, __VA_ARGS__); \
55
+ (base)->in_trace_proc = 0; \
56
+ }
51
57
  #define COND_TRACE(base, ...) if (SHOULD_TRACE(base)) { TRACE(base, __VA_ARGS__); }
52
58
 
53
59
 
@@ -80,7 +86,7 @@ void fptr_finalize(rb_io_t *fptr);
80
86
 
81
87
  struct backend_stats backend_get_stats(VALUE self);
82
88
  VALUE backend_await(struct Backend_base *backend);
83
- VALUE backend_snooze();
89
+ VALUE backend_snooze(struct Backend_base *backend);
84
90
 
85
91
  // macros for doing read loops
86
92
  #define READ_LOOP_PREPARE_STR() { \
@@ -102,6 +102,24 @@ static VALUE Backend_initialize(VALUE self) {
102
102
  return self;
103
103
  }
104
104
 
105
+ static inline struct io_buffer get_io_buffer(VALUE in) {
106
+ if (FIXNUM_P(in)) {
107
+ struct raw_buffer *raw = (struct raw_buffer *)(FIX2LONG(in));
108
+ return (struct io_buffer){ raw->base, raw->size, 1 };
109
+ }
110
+ return (struct io_buffer){ RSTRING_PTR(in), RSTRING_LEN(in), 0 };
111
+ }
112
+
113
+ static inline VALUE coerce_io_string_or_buffer(VALUE buf) {
114
+ switch (TYPE(buf)) {
115
+ case T_STRING:
116
+ case T_FIXNUM:
117
+ return buf;
118
+ default:
119
+ return StringValue(buf);
120
+ }
121
+ }
122
+
105
123
  VALUE Backend_finalize(VALUE self) {
106
124
  Backend_t *backend;
107
125
  GetBackend(self, backend);
@@ -216,12 +234,12 @@ inline VALUE Backend_poll(VALUE self, VALUE blocking) {
216
234
  io_uring_submit(&backend->ring);
217
235
  }
218
236
 
219
- COND_TRACE(&backend->base, 2, SYM_fiber_event_poll_enter, rb_fiber_current());
237
+ COND_TRACE(&backend->base, 2, SYM_enter_poll, rb_fiber_current());
220
238
 
221
239
  if (is_blocking) io_uring_backend_poll(backend);
222
240
  io_uring_backend_handle_ready_cqes(backend);
223
241
 
224
- COND_TRACE(&backend->base, 2, SYM_fiber_event_poll_leave, rb_fiber_current());
242
+ COND_TRACE(&backend->base, 2, SYM_leave_poll, rb_fiber_current());
225
243
 
226
244
  return self;
227
245
  }
@@ -332,23 +350,37 @@ VALUE io_uring_backend_wait_fd(Backend_t *backend, int fd, int write) {
332
350
  VALUE Backend_read(VALUE self, VALUE io, VALUE str, VALUE length, VALUE to_eof, VALUE pos) {
333
351
  Backend_t *backend;
334
352
  rb_io_t *fptr;
335
- long dynamic_len = length == Qnil;
336
- long buffer_size = dynamic_len ? 4096 : NUM2INT(length);
353
+ struct io_buffer buffer = get_io_buffer(str);
337
354
  long buf_pos = NUM2INT(pos);
338
- int shrinkable;
339
- char *buf;
355
+ int shrinkable_string = 0;
356
+ int expandable_buffer = 0;
340
357
  long total = 0;
341
358
  int read_to_eof = RTEST(to_eof);
342
359
  VALUE underlying_io = rb_ivar_get(io, ID_ivar_io);
343
360
 
344
-
345
- if (str != Qnil) {
346
- int current_len = RSTRING_LEN(str);
347
- if (buf_pos < 0 || buf_pos > current_len) buf_pos = current_len;
361
+ if (buffer.raw) {
362
+ if (buf_pos < 0 || buf_pos > buffer.size) buf_pos = buffer.size;
363
+ buffer.base += buf_pos;
364
+ buffer.size -= buf_pos;
365
+ }
366
+ else {
367
+ expandable_buffer = length == Qnil;
368
+ long expected_read_length = expandable_buffer ? 4096 : FIX2INT(length);
369
+ long string_cap = rb_str_capacity(str);
370
+ if (buf_pos < 0 || buf_pos > buffer.size) buf_pos = buffer.size;
371
+
372
+ if (string_cap < expected_read_length + buf_pos) {
373
+ shrinkable_string = io_setstrbuf(&str, expected_read_length + buf_pos);
374
+ buffer.base = RSTRING_PTR(str) + buf_pos;
375
+ buffer.size = expected_read_length;
376
+ }
377
+ else {
378
+ buffer.base += buf_pos;
379
+ buffer.size = string_cap - buf_pos;
380
+ if (buffer.size > expected_read_length)
381
+ buffer.size = expected_read_length;
382
+ }
348
383
  }
349
- else buf_pos = 0;
350
- shrinkable = io_setstrbuf(&str, buf_pos + buffer_size);
351
- buf = RSTRING_PTR(str) + buf_pos;
352
384
 
353
385
  GetBackend(self, backend);
354
386
  if (underlying_io != Qnil) io = underlying_io;
@@ -364,7 +396,7 @@ VALUE Backend_read(VALUE self, VALUE io, VALUE str, VALUE length, VALUE to_eof,
364
396
  int result;
365
397
  int completed;
366
398
 
367
- io_uring_prep_read(sqe, fptr->fd, buf, buffer_size - total, -1);
399
+ io_uring_prep_read(sqe, fptr->fd, buffer.base, buffer.size - total, -1);
368
400
 
369
401
  result = io_uring_backend_defer_submit_and_await(backend, sqe, ctx, &resume_value);
370
402
  completed = context_store_release(&backend->store, ctx);
@@ -383,22 +415,28 @@ VALUE Backend_read(VALUE self, VALUE io, VALUE str, VALUE length, VALUE to_eof,
383
415
  total += result;
384
416
  if (!read_to_eof) break;
385
417
 
386
- if (total == buffer_size) {
387
- if (!dynamic_len) break;
418
+ if (total == buffer.size) {
419
+ if (!expandable_buffer) break;
388
420
 
389
421
  // resize buffer
390
- rb_str_resize(str, buf_pos + total);
391
- rb_str_modify_expand(str, buffer_size);
392
- buf = RSTRING_PTR(str) + buf_pos + total;
393
- shrinkable = 0;
394
- buffer_size += buffer_size;
422
+ rb_str_resize(str, total + buf_pos);
423
+ rb_str_modify_expand(str, buffer.size);
424
+ shrinkable_string = 0;
425
+ buffer.base = RSTRING_PTR(str) + total + buf_pos;
426
+ buffer.size = buffer.size;
427
+ }
428
+ else {
429
+ buffer.base += result;
430
+ buffer.size -= result;
431
+ if (!buffer.size) break;
395
432
  }
396
- else buf += result;
397
433
  }
398
434
  }
399
435
 
400
- io_set_read_length(str, buf_pos + total, shrinkable);
401
- io_enc_str(str, fptr);
436
+ if (!buffer.raw) {
437
+ io_set_read_length(str, buf_pos + total, shrinkable_string);
438
+ io_enc_str(str, fptr);
439
+ }
402
440
 
403
441
  if (!total) return Qnil;
404
442
 
@@ -514,9 +552,9 @@ VALUE Backend_write(VALUE self, VALUE io, VALUE str) {
514
552
  Backend_t *backend;
515
553
  rb_io_t *fptr;
516
554
  VALUE underlying_io;
517
- char *buf = StringValuePtr(str);
518
- long len = RSTRING_LEN(str);
519
- long left = len;
555
+
556
+ struct io_buffer buffer = get_io_buffer(str);
557
+ long left = buffer.size;
520
558
 
521
559
  underlying_io = rb_ivar_get(io, ID_ivar_io);
522
560
  if (underlying_io != Qnil) io = underlying_io;
@@ -532,7 +570,7 @@ VALUE Backend_write(VALUE self, VALUE io, VALUE str) {
532
570
  int result;
533
571
  int completed;
534
572
 
535
- io_uring_prep_write(sqe, fptr->fd, buf, left, 0);
573
+ io_uring_prep_write(sqe, fptr->fd, buffer.base, left, 0);
536
574
 
537
575
  result = io_uring_backend_defer_submit_and_await(backend, sqe, ctx, &resume_value);
538
576
  completed = context_store_release(&backend->store, ctx);
@@ -546,12 +584,12 @@ VALUE Backend_write(VALUE self, VALUE io, VALUE str) {
546
584
  if (result < 0)
547
585
  rb_syserr_fail(-result, strerror(-result));
548
586
  else {
549
- buf += result;
587
+ buffer.base += result;
550
588
  left -= result;
551
589
  }
552
590
  }
553
591
 
554
- return INT2NUM(len);
592
+ return INT2NUM(buffer.size);
555
593
  }
556
594
 
557
595
  VALUE Backend_writev(VALUE self, VALUE io, int argc, VALUE *argv) {
@@ -1108,7 +1146,7 @@ VALUE Backend_timer_loop(VALUE self, VALUE interval) {
1108
1146
  if (!completed) return resume_value;
1109
1147
  }
1110
1148
  else {
1111
- resume_value = backend_snooze();
1149
+ resume_value = backend_snooze(&backend->base);
1112
1150
  RAISE_IF_EXCEPTION(resume_value);
1113
1151
  }
1114
1152
 
@@ -1591,6 +1629,22 @@ VALUE Backend_trace_proc_set(VALUE self, VALUE block) {
1591
1629
  return self;
1592
1630
  }
1593
1631
 
1632
+ VALUE Backend_snooze(VALUE self) {
1633
+ VALUE ret;
1634
+ VALUE fiber = rb_fiber_current();
1635
+ Backend_t *backend;
1636
+ GetBackend(self, backend);
1637
+
1638
+ Fiber_make_runnable(fiber, Qnil);
1639
+ ret = Thread_switch_fiber(rb_thread_current());
1640
+
1641
+ COND_TRACE(&backend->base, 4, SYM_unblock, rb_fiber_current(), ret, CALLER());
1642
+
1643
+ RAISE_IF_EXCEPTION(ret);
1644
+ RB_GC_GUARD(ret);
1645
+ return ret;
1646
+ }
1647
+
1594
1648
  void Backend_park_fiber(VALUE self, VALUE fiber) {
1595
1649
  Backend_t *backend;
1596
1650
  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];
@@ -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) {
@@ -125,6 +118,12 @@ VALUE Polyphony_backend_write(int argc, VALUE *argv, VALUE self) {
125
118
  return Backend_write_m(argc, argv, BACKEND());
126
119
  }
127
120
 
121
+ VALUE Polyphony_backend_test(VALUE self, VALUE io, VALUE str) {
122
+ struct raw_buffer buffer = { RSTRING_PTR(str), RSTRING_LEN(str) };
123
+ VALUE args[2] = { io, LONG2FIX((long)&buffer) };
124
+ return Polyphony_backend_write(2, args, self);
125
+ }
126
+
128
127
  // VALUE Polyphony_backend_close(VALUE self, VALUE io) {
129
128
  // return Backend_close(BACKEND(), io);
130
129
  // }
@@ -156,6 +155,8 @@ void Init_Polyphony() {
156
155
  // rb_define_singleton_method(mPolyphony, "backend_close", Polyphony_backend_close, 1);
157
156
  rb_define_singleton_method(mPolyphony, "backend_verify_blocking_mode", Backend_verify_blocking_mode, 2);
158
157
 
158
+ rb_define_singleton_method(mPolyphony, "backend_test", Polyphony_backend_test, 2);
159
+
159
160
  rb_define_global_function("snooze", Polyphony_snooze, 0);
160
161
  rb_define_global_function("suspend", Polyphony_suspend, 0);
161
162