polyphony 0.73.1 → 0.77

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 (45) hide show
  1. checksums.yaml +4 -4
  2. data/.github/workflows/test.yml +13 -10
  3. data/.github/workflows/test_io_uring.yml +32 -0
  4. data/CHANGELOG.md +22 -0
  5. data/Gemfile.lock +10 -7
  6. data/bin/pdbg +0 -0
  7. data/bin/polyphony-debug +0 -0
  8. data/bin/stress.rb +0 -0
  9. data/bin/test +0 -0
  10. data/examples/core/trap1.rb +21 -0
  11. data/examples/core/trap2.rb +14 -0
  12. data/ext/polyphony/backend_common.c +84 -12
  13. data/ext/polyphony/backend_common.h +8 -0
  14. data/ext/polyphony/backend_io_uring.c +231 -107
  15. data/ext/polyphony/backend_io_uring_context.c +1 -0
  16. data/ext/polyphony/backend_io_uring_context.h +2 -1
  17. data/ext/polyphony/backend_libev.c +12 -9
  18. data/ext/polyphony/event.c +5 -2
  19. data/ext/polyphony/polyphony.c +11 -1
  20. data/ext/polyphony/polyphony.h +4 -1
  21. data/ext/polyphony/queue.c +10 -5
  22. data/ext/polyphony/runqueue_ring_buffer.c +3 -1
  23. data/ext/polyphony/socket_extensions.c +5 -2
  24. data/ext/test_eintr.c +50 -0
  25. data/lib/polyphony/extensions/fiber.rb +85 -5
  26. data/lib/polyphony/extensions/openssl.rb +5 -1
  27. data/lib/polyphony/extensions/socket.rb +12 -6
  28. data/lib/polyphony/extensions/thread.rb +9 -3
  29. data/lib/polyphony/version.rb +1 -1
  30. data/lib/polyphony.rb +4 -1
  31. data/test/helper.rb +2 -6
  32. data/test/stress.rb +1 -1
  33. data/test/test_backend.rb +3 -5
  34. data/test/test_fiber.rb +6 -4
  35. data/test/test_global_api.rb +10 -14
  36. data/test/test_io.rb +2 -2
  37. data/test/test_kernel.rb +2 -2
  38. data/test/test_signal.rb +57 -0
  39. data/test/test_socket.rb +35 -2
  40. data/test/test_thread.rb +1 -1
  41. data/test/test_thread_pool.rb +1 -1
  42. data/test/test_throttler.rb +3 -3
  43. data/test/test_timer.rb +2 -2
  44. data/test/test_trace.rb +7 -1
  45. metadata +11 -7
checksums.yaml CHANGED
@@ -1,7 +1,7 @@
1
1
  ---
2
2
  SHA256:
3
- metadata.gz: c1647a2b1735e71dd4adf19ff701a649198786c3055430a0941b93e0905cf9dc
4
- data.tar.gz: d00f4541d6bab4d7852ced8c0a8e15b6756305ab1c9eab3f7931bf7a18cfd6fc
3
+ metadata.gz: 10f049946dc02d9cdd984cc8658687d2319e0fd85f6f2cce74aa0a887bc714cb
4
+ data.tar.gz: 5118859d4640aebeddb9186c43d58617e9b455a63ea97c1dc8283432db954fd1
5
5
  SHA512:
6
- metadata.gz: 5ec65b272aad6089f9f5a0a208a1e2a537e306fd1fc5247bb5aa25df22ad135d37c771947287b4d4ce5e3e4d5c5d8ced4fef9bab2291a6b7300f7b22245dd70f
7
- data.tar.gz: 245c18039397202618c4a70538878396048ce409f36e51ece94919cd1720020cc47e13f213020c378c4d6caa9017df1c5082cd292e5f7987f11730486a0b0c0a
6
+ metadata.gz: e107e21d6c42f9cd9e91521e288dfee37a6f0da4b7d8a8d311924bf88ccf732d2cc94de834dbbaa3c7a974e5bb84ce1d8d20efd68870c9f2dac9b73155d28558
7
+ data.tar.gz: 8e4f7005c9cbc46cd084bbe66a4e8f7145f6a4a3062ce027a70eee07c0e648f79289515591ef4b48010580b7125c403ab09e45417781346ebf6da4b2d528ec45
@@ -7,26 +7,29 @@ jobs:
7
7
  strategy:
8
8
  fail-fast: false
9
9
  matrix:
10
- os: [ubuntu-latest, ubuntu-18.04, macos-10.15]
11
- ruby: [2.6, 2.7, 3.0]
10
+ os: [ubuntu-latest, macos-latest]
11
+ ruby: ['2.7', '3.0', '3.1', 'head']
12
12
 
13
13
  name: >-
14
14
  ${{matrix.os}}, ${{matrix.ruby}}
15
15
 
16
16
  runs-on: ${{matrix.os}}
17
+
18
+ env:
19
+ POLYPHONY_USE_LIBEV: "1"
20
+
17
21
  steps:
18
- - uses: actions/checkout@v1
19
- - uses: ruby/setup-ruby@v1
22
+ - name: Setup OS
23
+ uses: actions/checkout@v1
24
+ - name: Setup Ruby
25
+ uses: ruby/setup-ruby@v1
20
26
  with:
21
27
  ruby-version: ${{matrix.ruby}}
22
28
  bundler-cache: true # 'bundle install' and cache
23
- - name: Install dependencies
24
- run: |
25
- gem install bundler
26
- bundle install
27
- - name: Show Linux kernel version
29
+ cache-version: 2
30
+ - name: Kernel version
28
31
  run: uname -a
29
32
  - name: Compile C-extension
30
- run: POLYPHONY_USE_LIBEV=1 bundle exec rake compile
33
+ run: bundle exec rake compile
31
34
  - name: Run tests
32
35
  run: bundle exec rake test
@@ -0,0 +1,32 @@
1
+ name: Tests with io_uring
2
+
3
+ on: [push, pull_request]
4
+
5
+ jobs:
6
+ build:
7
+ strategy:
8
+ fail-fast: false
9
+ matrix:
10
+ os: [ubuntu-latest]
11
+ ruby: [2.6, 2.7, 3.0]
12
+
13
+ name: >-
14
+ ${{matrix.os}}, ${{matrix.ruby}}
15
+
16
+ runs-on: ${{matrix.os}}
17
+ steps:
18
+ - uses: actions/checkout@v1
19
+ - uses: ruby/setup-ruby@v1
20
+ with:
21
+ ruby-version: ${{matrix.ruby}}
22
+ bundler-cache: true # 'bundle install' and cache
23
+ - name: Install dependencies
24
+ run: |
25
+ gem install bundler
26
+ bundle install
27
+ - name: Show Linux kernel version
28
+ run: uname -a
29
+ - name: Compile C-extension
30
+ run: bundle exec rake compile
31
+ - name: Run tests
32
+ run: bundle exec rake test
data/CHANGELOG.md CHANGED
@@ -1,3 +1,25 @@
1
+ ## 0.77 2022-02-07
2
+
3
+ - Fix behaviour of signal traps (#71)
4
+
5
+ ## 0.76 2022-02-06
6
+
7
+ - Comment out `Backend_close` API (#70)
8
+
9
+ ## 0.75 2022-02-04
10
+
11
+ - Fix handling of MoveOn on main fiber of forked process
12
+ - Ensure SSLSocket underlying socket is in nonblocking mode
13
+ - Add `Polyphony.backend_verify_blocking_mode` API
14
+ - Fix address resolution for hostnames with IPv6 address
15
+ - Improve behaviour of OOB fiber
16
+ - Include caller in `fiber_switchpoint` trace
17
+
18
+ ## 0.74 2022-02-01
19
+
20
+ - Add support for IPv6 (#69)
21
+ - Override TCPSocket.open
22
+
1
23
  ## 0.73.1 2021-12-17
2
24
 
3
25
  - Fix Gemfile.lock
data/Gemfile.lock CHANGED
@@ -1,7 +1,7 @@
1
1
  PATH
2
2
  remote: .
3
3
  specs:
4
- polyphony (0.73.1)
4
+ polyphony (0.77)
5
5
 
6
6
  GEM
7
7
  remote: https://rubygems.org/
@@ -19,7 +19,7 @@ GEM
19
19
  method_source (1.0.0)
20
20
  mime-types (3.4.1)
21
21
  mime-types-data (~> 3.2015)
22
- mime-types-data (3.2021.1115)
22
+ mime-types-data (3.2022.0105)
23
23
  minitest (5.14.4)
24
24
  minitest-reporters (1.4.2)
25
25
  ansi
@@ -29,12 +29,12 @@ GEM
29
29
  msgpack (1.4.2)
30
30
  multi_xml (0.6.0)
31
31
  parallel (1.21.0)
32
- parser (3.0.3.2)
32
+ parser (3.1.0.0)
33
33
  ast (~> 2.4.1)
34
34
  pry (0.13.1)
35
35
  coderay (~> 1.1)
36
36
  method_source (~> 1.0)
37
- rainbow (3.0.0)
37
+ rainbow (3.1.1)
38
38
  rake (13.0.6)
39
39
  rake-compiler (1.1.1)
40
40
  rake
@@ -49,7 +49,7 @@ GEM
49
49
  rubocop-ast (>= 0.0.3)
50
50
  ruby-progressbar (~> 1.7)
51
51
  unicode-display_width (>= 1.4.0, < 2.0)
52
- rubocop-ast (1.15.0)
52
+ rubocop-ast (1.15.1)
53
53
  parser (>= 3.0.1.1)
54
54
  ruby-progressbar (1.11.0)
55
55
  simplecov (0.17.1)
@@ -60,7 +60,10 @@ GEM
60
60
  unicode-display_width (1.8.0)
61
61
 
62
62
  PLATFORMS
63
- ruby
63
+ universal-darwin
64
+ universal-freebsd
65
+ universal-linux
66
+ universal-unknown
64
67
 
65
68
  DEPENDENCIES
66
69
  httparty (= 0.17.1)
@@ -75,4 +78,4 @@ DEPENDENCIES
75
78
  simplecov (= 0.17.1)
76
79
 
77
80
  BUNDLED WITH
78
- 2.1.4
81
+ 2.3.3
data/bin/pdbg CHANGED
File without changes
data/bin/polyphony-debug CHANGED
File without changes
data/bin/stress.rb CHANGED
File without changes
data/bin/test CHANGED
File without changes
@@ -0,0 +1,21 @@
1
+ # frozen_string_literal: true
2
+
3
+ pid = Process.pid
4
+ fork do
5
+ sleep 1
6
+ Process.kill('SIGINT', pid)
7
+ # sleep 10
8
+ # Process.kill(-9, pid)
9
+ end
10
+
11
+ require 'bundler/setup'
12
+ require 'polyphony'
13
+
14
+ Thread.backend.trace_proc = proc { |*e| STDOUT.orig_write("#{e.inspect}\n") }
15
+ trap('SIGINT') { STDOUT.orig_write("* recv SIGINT\n") }
16
+ # trap('SIGCHLD') { STDOUT.orig_write("* recv SIGCHLD\n") }
17
+ STDOUT.orig_write("* pre gets\n")
18
+ # STDIN.wait_readable
19
+ s = gets
20
+ p s
21
+ STDOUT.orig_write("* post gets\n")
@@ -0,0 +1,14 @@
1
+ # frozen_string_literal: true
2
+
3
+ pid = Process.pid
4
+ fork do
5
+ sleep 1
6
+ Process.kill('SIGINT', pid)
7
+ # sleep 10
8
+ # Process.kill(-9, pid)
9
+ end
10
+
11
+ trap('SIGINT') {}
12
+ p :before
13
+ result = IO.select([STDIN])
14
+ p :after
@@ -64,7 +64,8 @@ VALUE backend_base_switch_fiber(VALUE backend, struct Backend_base *base) {
64
64
  unsigned int idle_tasks_run_count = 0;
65
65
 
66
66
  base->switch_count++;
67
- COND_TRACE(base, 2, SYM_fiber_switchpoint, current_fiber);
67
+ if (SHOULD_TRACE(base))
68
+ TRACE(base, 3, SYM_fiber_switchpoint, current_fiber, CALLER());
68
69
 
69
70
  while (1) {
70
71
  next = runqueue_shift(&base->runqueue);
@@ -106,14 +107,14 @@ VALUE backend_base_switch_fiber(VALUE backend, struct Backend_base *base) {
106
107
 
107
108
  void backend_base_schedule_fiber(VALUE thread, VALUE backend, struct Backend_base *base, VALUE fiber, VALUE value, int prioritize) {
108
109
  int already_runnable;
110
+ runqueue_t *runqueue;
109
111
 
110
112
  if (rb_fiber_alive_p(fiber) != Qtrue) return;
111
113
  already_runnable = rb_ivar_get(fiber, ID_ivar_runnable) != Qnil;
112
114
 
113
115
  COND_TRACE(base, 4, SYM_fiber_schedule, fiber, value, prioritize ? Qtrue : Qfalse);
114
116
 
115
- runqueue_t *runqueue = rb_ivar_get(fiber, ID_ivar_parked) == Qtrue ?
116
- &base->parked_runqueue : &base->runqueue;
117
+ runqueue = rb_ivar_get(fiber, ID_ivar_parked) == Qtrue ? &base->parked_runqueue : &base->runqueue;
117
118
 
118
119
  (prioritize ? runqueue_unshift : runqueue_push)(runqueue, fiber, value, already_runnable);
119
120
  if (!already_runnable) {
@@ -206,6 +207,36 @@ inline VALUE io_enc_str(VALUE str, rb_io_t *fptr) {
206
207
  return str;
207
208
  }
208
209
 
210
+ static inline void free_io_buffer(rb_io_buffer_t *buf)
211
+ {
212
+ if (buf->ptr) {
213
+ ruby_xfree(buf->ptr);
214
+ buf->ptr = NULL;
215
+ }
216
+ }
217
+
218
+ static inline void clear_codeconv(rb_io_t *fptr) {
219
+ if (fptr->readconv) {
220
+ rb_econv_close(fptr->readconv);
221
+ fptr->readconv = NULL;
222
+ }
223
+ free_io_buffer(&fptr->cbuf);
224
+
225
+ if (fptr->writeconv) {
226
+ rb_econv_close(fptr->writeconv);
227
+ fptr->writeconv = NULL;
228
+ }
229
+ fptr->writeconv_initialized = 0;
230
+ }
231
+
232
+ void fptr_finalize(rb_io_t *fptr) {
233
+ fptr->fd = -1;
234
+ fptr->stdio_file = 0;
235
+ free_io_buffer(&fptr->rbuf);
236
+ free_io_buffer(&fptr->wbuf);
237
+ clear_codeconv(fptr);
238
+ }
239
+
209
240
  //////////////////////////////////////////////////////////////////////
210
241
  //////////////////////////////////////////////////////////////////////
211
242
 
@@ -238,17 +269,22 @@ inline void rectify_io_file_pos(rb_io_t *fptr) {
238
269
 
239
270
  inline double current_time() {
240
271
  struct timespec ts;
272
+ double t;
273
+ uint64_t ns;
274
+
241
275
  clock_gettime(CLOCK_MONOTONIC, &ts);
242
- long long ns = ts.tv_sec;
276
+ ns = ts.tv_sec;
243
277
  ns = ns * 1e9 + ts.tv_nsec;
244
- double t = ns;
278
+ t = ns;
245
279
  return t / 1e9;
246
280
  }
247
281
 
248
282
  inline uint64_t current_time_ns() {
249
283
  struct timespec ts;
284
+ uint64_t ns;
285
+
250
286
  clock_gettime(CLOCK_MONOTONIC, &ts);
251
- uint64_t ns = ts.tv_sec;
287
+ ns = ts.tv_sec;
252
288
  return ns * 1e9 + ts.tv_nsec;
253
289
  }
254
290
 
@@ -276,6 +312,9 @@ VALUE Backend_timeout_ensure_safe(VALUE arg) {
276
312
  static VALUE empty_string = Qnil;
277
313
 
278
314
  VALUE Backend_sendv(VALUE self, VALUE io, VALUE ary, VALUE flags) {
315
+ VALUE joined;
316
+ VALUE result;
317
+
279
318
  switch (RARRAY_LEN(ary)) {
280
319
  case 0:
281
320
  return Qnil;
@@ -286,14 +325,16 @@ VALUE Backend_sendv(VALUE self, VALUE io, VALUE ary, VALUE flags) {
286
325
  empty_string = rb_str_new_literal("");
287
326
  rb_global_variable(&empty_string);
288
327
  }
289
- VALUE joined = rb_ary_join(ary, empty_string);
290
- VALUE result = Backend_send(self, io, joined, flags);
328
+ joined = rb_ary_join(ary, empty_string);
329
+ result = Backend_send(self, io, joined, flags);
291
330
  RB_GC_GUARD(joined);
292
331
  return result;
293
332
  }
294
333
  }
295
334
 
296
335
  inline void io_verify_blocking_mode(rb_io_t *fptr, VALUE io, VALUE blocking) {
336
+ int flags;
337
+ int is_nonblocking;
297
338
  VALUE blocking_mode = rb_ivar_get(io, ID_ivar_blocking_mode);
298
339
  if (blocking == blocking_mode) return;
299
340
 
@@ -303,9 +344,9 @@ inline void io_verify_blocking_mode(rb_io_t *fptr, VALUE io, VALUE blocking) {
303
344
  if (blocking != Qtrue)
304
345
  rb_w32_set_nonblock(fptr->fd);
305
346
  #elif defined(F_GETFL)
306
- int flags = fcntl(fptr->fd, F_GETFL);
347
+ flags = fcntl(fptr->fd, F_GETFL);
307
348
  if (flags == -1) return;
308
- int is_nonblocking = flags & O_NONBLOCK;
349
+ is_nonblocking = flags & O_NONBLOCK;
309
350
 
310
351
  if (blocking == Qtrue) {
311
352
  if (!is_nonblocking) return;
@@ -319,12 +360,14 @@ inline void io_verify_blocking_mode(rb_io_t *fptr, VALUE io, VALUE blocking) {
319
360
  }
320
361
 
321
362
  inline void backend_run_idle_tasks(struct Backend_base *base) {
363
+ double now;
364
+
322
365
  if (base->idle_proc != Qnil)
323
366
  rb_funcall(base->idle_proc, ID_call, 0);
324
367
 
325
368
  if (base->idle_gc_period == 0) return;
326
369
 
327
- double now = current_time();
370
+ now = current_time();
328
371
  if (now - base->idle_gc_last_time < base->idle_gc_period) return;
329
372
 
330
373
  base->idle_gc_last_time = now;
@@ -373,6 +416,13 @@ VALUE Backend_stats(VALUE self) {
373
416
  return stats;
374
417
  }
375
418
 
419
+ VALUE Backend_verify_blocking_mode(VALUE self, VALUE io, VALUE blocking) {
420
+ rb_io_t *fptr;
421
+ GetOpenFile(io, fptr);
422
+ io_verify_blocking_mode(fptr, io, blocking);
423
+ return self;
424
+ }
425
+
376
426
  void backend_setup_stats_symbols() {
377
427
  SYM_runqueue_size = ID2SYM(rb_intern("runqueue_size"));
378
428
  SYM_runqueue_length = ID2SYM(rb_intern("runqueue_length"));
@@ -389,4 +439,26 @@ void backend_setup_stats_symbols() {
389
439
  rb_global_variable(&SYM_switch_count);
390
440
  rb_global_variable(&SYM_poll_count);
391
441
  rb_global_variable(&SYM_pending_ops);
392
- }
442
+ }
443
+
444
+ int backend_getaddrinfo(VALUE host, VALUE port, struct sockaddr **ai_addr) {
445
+ VALUE port_string;
446
+ struct addrinfo hints;
447
+ struct addrinfo *addrinfo_result;
448
+ int ret;
449
+
450
+ memset(&hints, 0, sizeof(struct addrinfo));
451
+ hints.ai_family = AF_UNSPEC; /* allow IPv4 or IPv6 */
452
+ hints.ai_socktype = SOCK_STREAM;
453
+
454
+ port_string = rb_funcall(port, ID_to_s, 0);
455
+ ret = getaddrinfo(StringValueCStr(host), StringValueCStr(port_string), &hints, &addrinfo_result);
456
+ RB_GC_GUARD(port_string);
457
+ if (ret != 0) {
458
+ VALUE msg = rb_str_new2(gai_strerror(ret));
459
+ rb_funcall(rb_mKernel, ID_raise, 1, msg);
460
+ RB_GC_GUARD(msg);
461
+ }
462
+ *ai_addr = addrinfo_result->ai_addr;
463
+ return addrinfo_result->ai_addrlen;
464
+ }
@@ -1,6 +1,11 @@
1
1
  #ifndef BACKEND_COMMON_H
2
2
  #define BACKEND_COMMON_H
3
3
 
4
+ #include <sys/types.h>
5
+ #include <arpa/inet.h>
6
+ #include <netinet/in.h>
7
+ #include <netdb.h>
8
+
4
9
  #include "ruby.h"
5
10
  #include "ruby/io.h"
6
11
  #include "runqueue.h"
@@ -68,6 +73,7 @@ void io_shrink_read_string(VALUE str, long n);
68
73
  void io_set_read_length(VALUE str, long n, int shrinkable);
69
74
  rb_encoding* io_read_encoding(rb_io_t *fptr);
70
75
  VALUE io_enc_str(VALUE str, rb_io_t *fptr);
76
+ void fptr_finalize(rb_io_t *fptr);
71
77
 
72
78
  //////////////////////////////////////////////////////////////////////
73
79
  //////////////////////////////////////////////////////////////////////
@@ -106,8 +112,10 @@ VALUE Backend_timeout_ensure_safe(VALUE arg);
106
112
  VALUE Backend_timeout_ensure_safe(VALUE arg);
107
113
  VALUE Backend_sendv(VALUE self, VALUE io, VALUE ary, VALUE flags);
108
114
  VALUE Backend_stats(VALUE self);
115
+ VALUE Backend_verify_blocking_mode(VALUE self, VALUE io, VALUE blocking);
109
116
  void backend_run_idle_tasks(struct Backend_base *base);
110
117
  void io_verify_blocking_mode(rb_io_t *fptr, VALUE io, VALUE blocking);
111
118
  void backend_setup_stats_symbols();
119
+ int backend_getaddrinfo(VALUE host, VALUE port, struct sockaddr **ai_addr);
112
120
 
113
121
  #endif /* BACKEND_COMMON_H */