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.
- checksums.yaml +4 -4
- data/CHANGELOG.md +19 -0
- data/Gemfile.lock +2 -1
- data/examples/core/pingpong.rb +7 -4
- data/examples/core/zlib_stream.rb +15 -0
- data/ext/polyphony/backend_common.c +16 -8
- data/ext/polyphony/backend_common.h +9 -3
- data/ext/polyphony/backend_io_uring.c +85 -31
- data/ext/polyphony/backend_libev.c +33 -17
- data/ext/polyphony/fiber.c +27 -27
- data/ext/polyphony/polyphony.c +9 -8
- data/ext/polyphony/polyphony.h +21 -7
- data/ext/polyphony/thread.c +6 -2
- data/lib/polyphony/adapters/fs.rb +4 -0
- data/lib/polyphony/adapters/process.rb +14 -1
- data/lib/polyphony/adapters/redis.rb +28 -0
- data/lib/polyphony/adapters/sequel.rb +19 -1
- data/lib/polyphony/core/debug.rb +201 -0
- data/lib/polyphony/core/exceptions.rb +21 -6
- data/lib/polyphony/core/global_api.rb +228 -73
- data/lib/polyphony/core/resource_pool.rb +65 -20
- data/lib/polyphony/core/sync.rb +57 -12
- data/lib/polyphony/core/thread_pool.rb +42 -5
- data/lib/polyphony/core/throttler.rb +21 -5
- data/lib/polyphony/core/timer.rb +125 -1
- data/lib/polyphony/extensions/exception.rb +36 -6
- data/lib/polyphony/extensions/fiber.rb +244 -61
- data/lib/polyphony/extensions/io.rb +4 -2
- data/lib/polyphony/extensions/kernel.rb +9 -4
- data/lib/polyphony/extensions/object.rb +8 -0
- data/lib/polyphony/extensions/openssl.rb +3 -1
- data/lib/polyphony/extensions/socket.rb +458 -39
- data/lib/polyphony/extensions/thread.rb +108 -43
- data/lib/polyphony/extensions/timeout.rb +12 -1
- data/lib/polyphony/extensions.rb +1 -0
- data/lib/polyphony/net.rb +66 -7
- data/lib/polyphony/version.rb +1 -1
- data/lib/polyphony.rb +0 -2
- data/test/test_backend.rb +6 -2
- data/test/test_global_api.rb +0 -23
- data/test/test_io.rb +7 -7
- data/test/test_resource_pool.rb +1 -1
- data/test/test_signal.rb +15 -15
- data/test/test_thread.rb +1 -1
- data/test/test_throttler.rb +0 -6
- data/test/test_trace.rb +189 -24
- metadata +9 -8
- data/lib/polyphony/core/channel.rb +0 -15
checksums.yaml
CHANGED
@@ -1,7 +1,7 @@
|
|
1
1
|
---
|
2
2
|
SHA256:
|
3
|
-
metadata.gz:
|
4
|
-
data.tar.gz:
|
3
|
+
metadata.gz: aef5f8ee7585e1ae6a6632099329ba35b7dc6959dba5041f4cdcbaf92caded72
|
4
|
+
data.tar.gz: b168b56f080029e4167e2528e66b83186a635ea4bda8dc74882dfe70639421fc
|
5
5
|
SHA512:
|
6
|
-
metadata.gz:
|
7
|
-
data.tar.gz:
|
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
data/examples/core/pingpong.rb
CHANGED
@@ -3,18 +3,21 @@
|
|
3
3
|
require 'bundler/setup'
|
4
4
|
require 'polyphony'
|
5
5
|
|
6
|
-
|
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
|
-
|
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
|
@@ -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,
|
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,
|
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
|
-
|
255
|
-
|
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, ...)
|
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,
|
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,
|
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
|
-
|
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
|
339
|
-
|
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
|
-
|
346
|
-
|
347
|
-
|
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,
|
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 ==
|
387
|
-
if (!
|
418
|
+
if (total == buffer.size) {
|
419
|
+
if (!expandable_buffer) break;
|
388
420
|
|
389
421
|
// resize buffer
|
390
|
-
rb_str_resize(str,
|
391
|
-
rb_str_modify_expand(str,
|
392
|
-
|
393
|
-
|
394
|
-
|
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
|
-
|
401
|
-
|
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
|
-
|
518
|
-
|
519
|
-
long left =
|
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,
|
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
|
-
|
587
|
+
buffer.base += result;
|
550
588
|
left -= result;
|
551
589
|
}
|
552
590
|
}
|
553
591
|
|
554
|
-
return INT2NUM(
|
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,
|
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,
|
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);
|
data/ext/polyphony/fiber.c
CHANGED
@@ -10,13 +10,13 @@ VALUE SYM_running;
|
|
10
10
|
VALUE SYM_runnable;
|
11
11
|
VALUE SYM_waiting;
|
12
12
|
|
13
|
-
VALUE
|
14
|
-
VALUE
|
15
|
-
VALUE
|
16
|
-
VALUE
|
17
|
-
VALUE
|
18
|
-
VALUE
|
19
|
-
VALUE
|
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
|
161
|
-
ID_ivar_mailbox
|
162
|
-
ID_ivar_result
|
163
|
-
ID_ivar_waiting_fibers
|
164
|
-
|
165
|
-
|
166
|
-
|
167
|
-
|
168
|
-
|
169
|
-
|
170
|
-
|
171
|
-
|
172
|
-
|
173
|
-
rb_global_variable(&
|
174
|
-
rb_global_variable(&
|
175
|
-
rb_global_variable(&
|
176
|
-
rb_global_variable(&
|
177
|
-
rb_global_variable(&
|
178
|
-
rb_global_variable(&
|
179
|
-
rb_global_variable(&
|
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
|
}
|
data/ext/polyphony/polyphony.c
CHANGED
@@ -27,14 +27,7 @@ ID ID_W;
|
|
27
27
|
ID ID_RW;
|
28
28
|
|
29
29
|
VALUE Polyphony_snooze(VALUE self) {
|
30
|
-
|
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
|
|