polyphony 0.42 → 0.43.4

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 CHANGED
@@ -1,7 +1,7 @@
1
1
  ---
2
2
  SHA256:
3
- metadata.gz: e284cae2a5eb2332bc8671090412bbe0cf7d57dbc163d68ba387f4d9635a8adf
4
- data.tar.gz: 58eba5a2607df83c7471b1a6b72cc350aa468e9a562e3d1e8f7451eaae12d068
3
+ metadata.gz: 9481b252526ebc3f8e0f5a0121eebb68492bc0c1502cfed447da7b0648e6095b
4
+ data.tar.gz: dfd2d5d0415833f14f9ebe2cc6336e87d0951ce86f07ebcdec0ad892f71fb74c
5
5
  SHA512:
6
- metadata.gz: 10b7bcf9240ea8f7355c75d025bd6c5e1bd00c86e7985f71440091cc149c629279b16eaf1c863d85de29e278739926b499625c77238db8eb803aa235395771a4
7
- data.tar.gz: 9001df441fdc058a3234bca05f6cca55d3a27edcf0b5b0d126a9c371d37bd0ffecc4d13d8f13c614d9b7182630f94d54481035f6d83fb044aebdf3238f83b03a
6
+ metadata.gz: 652633b1fc27623edb2e87b5657322ba4826ec176e8eb5131a43249e08340f793ba426743be343bfe0fce02fc60b73cb790afef816a4153585d68bdfeaa9c55d
7
+ data.tar.gz: 946abb7526324cae5f1d6b62e25c8410129b07ca75c82a6b7285d6269ed1911e161a6030a394ef29ba238a80da6f96787dd94fdbd5c05bd2a643379252ce80d0
@@ -1,3 +1,30 @@
1
+ ## 0.43.4 2020-07-09
2
+
3
+ * Reimplement Kernel#trap
4
+ * Dynamically allocate read buffer if length not given (#23)
5
+ * Prevent CPU saturation on infinite sleep (#24)
6
+
7
+ ## 0.43.3 2020-07-08
8
+
9
+ * Fix behaviour after call to `Process.daemon` (#8)
10
+ * Replace core `Queue` class with `Polyphony::Queue` (#22)
11
+ * Make `ResourcePool` reentrant (#1)
12
+ * Accept `:with_exception` argument in `cancel_after` (#16)
13
+
14
+ ## 0.43.2 2020-07-07
15
+
16
+ * Fix sending Redis commands with array arguments (#21)
17
+
18
+ ## 0.43.1 2020-06
19
+
20
+ * Fix compiling C-extension on MacOS (#20)
21
+
22
+ ## 0.43 2020-07-05
23
+
24
+ * Add IO#read_loop
25
+ * Fix OpenSSL extension
26
+ * More work on docs
27
+
1
28
  ## 0.42 2020-07-03
2
29
 
3
30
  * Improve documentation
@@ -1,7 +1,7 @@
1
1
  PATH
2
2
  remote: .
3
3
  specs:
4
- polyphony (0.42)
4
+ polyphony (0.43.4)
5
5
 
6
6
  GEM
7
7
  remote: https://rubygems.org/
@@ -12,7 +12,7 @@ GEM
12
12
  ast (2.4.0)
13
13
  builder (3.2.4)
14
14
  colorator (1.1.0)
15
- concurrent-ruby (1.1.5)
15
+ concurrent-ruby (1.1.6)
16
16
  docile (1.3.2)
17
17
  em-websocket (0.5.1)
18
18
  eventmachine (>= 0.12.9)
data/README.md CHANGED
@@ -1,6 +1,5 @@
1
1
  # Polyphony - Fine-Grained Concurrency for Ruby
2
2
 
3
-
4
3
  [![Gem Version](https://badge.fury.io/rb/polyphony.svg)](http://rubygems.org/gems/polyphony)
5
4
  [![Modulation Test](https://github.com/digital-fabric/polyphony/workflows/Tests/badge.svg)](https://github.com/digital-fabric/polyphony/actions?query=workflow%3ATests)
6
5
  [![MIT licensed](https://img.shields.io/badge/license-MIT-blue.svg)](https://github.com/digital-fabric/polyphony/blob/master/LICENSE)
data/TODO.md CHANGED
@@ -1,11 +1,10 @@
1
- ## 0.42
1
+ ## 0.43
2
2
 
3
3
  - Reimplement ResourcePool, Channel, Mutex using LibevQueue
4
4
  -- Add `Fiber#schedule_with_priority` method, aliased by `Fiber#wakeup`
5
5
  - Implement agent interface is virtual function table
6
6
  - Implement proxy agent for plugging in a user-provided agent class
7
7
 
8
- ## 0.43
9
8
  - Debugging
10
9
  - Eat your own dogfood: need a good tool to check what's going on when some
11
10
  test fails
@@ -46,13 +46,13 @@ heading_anchors: true
46
46
  back_to_top: true
47
47
  back_to_top_text: "Back to top"
48
48
 
49
- footer_content: "Copyright &copy; 2018-2020 Sharon Rosner. Distributed by an <a href=\"https://github.com/digital-fabric/polyphony/tree/master/LICENSE.txt\">MIT license.</a>"
49
+ footer_content: "Copyright &copy; 2020 Sharon Rosner. Distributed by an <a href=\"https://github.com/digital-fabric/polyphony/tree/master/LICENSE\">MIT license.</a>"
50
50
 
51
51
  # Footer "Edit this page on GitHub" link text
52
52
  gh_edit_link: true # show or hide edit this page link
53
53
  gh_edit_link_text: "Edit this page on GitHub"
54
54
  gh_edit_repository: "https://github.com/digital-fabric/polyphony" # the github URL for your repo
55
- gh_edit_branch: "master" # the branch that your docs is served from
55
+ gh_edit_branch: "master/docs" # the branch that your docs is served from
56
56
  gh_edit_view_mode: "tree" # "tree" or "edit" if you want the user to jump into the editor immediately
57
57
 
58
58
  compress_html:
@@ -0,0 +1,10 @@
1
+
2
+ h1.logo-title {
3
+ font-size: 42px !important;
4
+ font-weight: bold;
5
+ }
6
+
7
+ h2.logo-title {
8
+ margin-top: 0.25em;
9
+ margin-bottom: 1em;
10
+ }
Binary file
@@ -279,7 +279,7 @@ def chat_user_handler(user_name, connection)
279
279
  while command = connection.gets
280
280
  case command
281
281
  when /^connect (.+)/
282
- room&.send [:unsubscribe, message_subscriber]
282
+ room&.send [:subscribe, message_subscriber]
283
283
  room = CHAT_ROOMS[$1]
284
284
  when "disconnect"
285
285
  room&.send [:unsubscribe, message_subscriber]
@@ -416,28 +416,7 @@ request handler after all request headers have been received. This allows the
416
416
  application to better deal with slow client attacks, big file uploads, and also
417
417
  to minimize costly memory allocation and GC'ing.
418
418
 
419
- Preliminary benchmarks show the Polyphony web server to be about 3X as fast as
420
- Puma and 20X as fast as Unicorn. With SSL termination and using HTTP 2, the
421
- Polyphony web server is about 2X as fast as Falcon.
422
-
423
- ### HTTP/1
424
-
425
- |Server|requests/sec|average latency|max latency|
426
- |------|-----------:|--------------:|----------:|
427
- |Puma|
428
- |Unicorn|
429
- |Agoo|
430
- |Polyphony|
431
-
432
- ### HTTP/2 with SSL Termination
433
-
434
- |Server|requests/sec|average latency|max latency|
435
- |------|-----------:|--------------:|----------:|
436
- |Falcon|
437
- |Polyphony|
438
-
439
- (*Non-official benchmark with a basic "Hello world" Rack application. The usual
440
- caveats regarding benchmarks should be applied here.*)
419
+ Benchmarks will be included here at a later time.
441
420
 
442
421
  ## Integrating Polyphony with other Gems
443
422
 
@@ -504,4 +483,4 @@ reach version 1.0. Here are some of the exciting directions we're working on.
504
483
  - Support for more core and stdlib APIs
505
484
  - More adapters for gems with C-extensions, such as `mysql`, `sqlite3` etc
506
485
  - Use `io_uring` agent as alternative to the libev agent
507
- - More concurrency constructs for building highly concurrent applications
486
+ - More concurrency constructs for building highly concurrent applications
@@ -6,7 +6,11 @@ permalink: /
6
6
  next_title: Installing Polyphony
7
7
  ---
8
8
 
9
- # Polyphony - fine-grained concurrency for Ruby
9
+ # Polyphony
10
+ {:.text-center .logo-title}
11
+
12
+ ## Fine-grained concurrency for Ruby
13
+ {:.text-center .logo-title}
10
14
 
11
15
  Polyphony is a library for building concurrent applications in Ruby. Polyphony
12
16
  implements a comprehensive
Binary file
@@ -0,0 +1,9 @@
1
+ # frozen_string_literal: true
2
+
3
+ require 'bundler/setup'
4
+ require 'polyphony'
5
+ require 'concurrent'
6
+
7
+ puts "Hello, concurrent-ruby"
8
+
9
+ # this program should not hang with concurrent-ruby 1.1.6 (see issue #22)
@@ -0,0 +1,12 @@
1
+ # frozen_string_literal: true
2
+
3
+ require 'bundler/setup'
4
+ require 'polyphony/adapters/redis'
5
+ # require 'redis'
6
+
7
+ redis = Redis.new(host: ENV['REDISHOST'] || 'localhost')
8
+
9
+ redis.lpush("queue_key", "omgvalue")
10
+ puts "len: #{redis.llen("queue_key")}"
11
+ result = redis.blpop("queue_key")
12
+ puts result.inspect
@@ -0,0 +1,14 @@
1
+ # frozen_string_literal: true
2
+
3
+ require 'bundler/setup'
4
+ require 'polyphony'
5
+
6
+ Exception.__disable_sanitized_backtrace__ = true
7
+
8
+ puts "pid: #{Process.pid}"
9
+
10
+ Process.daemon(true, true)
11
+
12
+ Polyphony::ThreadPool.process do
13
+ puts "Hello world from pid #{Process.pid}"
14
+ end
@@ -4,47 +4,53 @@ def mem_usage
4
4
  `ps -o rss #{$$}`.split.last.to_i
5
5
  end
6
6
 
7
- def calculate_fiber_memory_cost(count)
7
+ def calculate_memory_cost(name, count, &block)
8
+ GC.enable
9
+ ObjectSpace.garbage_collect
10
+ sleep 0.5
8
11
  GC.disable
9
12
  rss0 = mem_usage
10
- count.times { Fiber.new { sleep 1 } }
13
+ count0 = ObjectSpace.count_objects[:TOTAL] - ObjectSpace.count_objects[:FREE]
14
+ a = []
15
+ count.times { a << block.call }
11
16
  rss1 = mem_usage
12
- GC.start
17
+ count1 = ObjectSpace.count_objects[:TOTAL] - ObjectSpace.count_objects[:FREE]
18
+ p [count0, count1]
19
+ # sleep 0.5
13
20
  cost = (rss1 - rss0).to_f / count
21
+ count_delta = (count1 - count0) / count
14
22
 
15
- puts "fiber memory cost: #{cost}KB"
23
+ puts "#{name} rss cost: #{cost}KB object count: #{count_delta}"
16
24
  end
17
25
 
18
- calculate_fiber_memory_cost(10000)
19
-
20
- def calculate_thread_memory_cost(count)
21
- GC.disable
22
- rss0 = mem_usage
23
- count.times { Thread.new { sleep 1 } }
24
- sleep 0.5
25
- rss1 = mem_usage
26
- sleep 0.5
27
- GC.start
28
- cost = (rss1 - rss0).to_f / count
26
+ f = Fiber.new { |f| f.transfer }
27
+ f.transfer Fiber.current
29
28
 
30
- puts "thread memory cost: #{cost}KB"
29
+ calculate_memory_cost('fiber', 10000) do
30
+ f = Fiber.new { |f| f.transfer :foo }
31
+ f.transfer Fiber.current
32
+ f
31
33
  end
32
34
 
33
- calculate_thread_memory_cost(500)
35
+ t = Thread.new { sleep 1}
36
+ t.kill
37
+ t.join
38
+
39
+ calculate_memory_cost('thread', 500) do
40
+ t = Thread.new { sleep 1 }
41
+ sleep 0.001
42
+ t
43
+ end
44
+ (Thread.list - [Thread.current]).each(&:kill).each(&:join)
34
45
 
35
46
  require 'bundler/setup'
36
47
  require 'polyphony'
37
48
 
38
- def calculate_extended_fiber_memory_cost(count)
39
- GC.disable
40
- rss0 = mem_usage
41
- count.times { spin { :foo } }
42
- snooze
43
- rss1 = mem_usage
44
- GC.start
45
- cost = (rss1 - rss0).to_f / count
49
+ f = spin { sleep 0.1 }
50
+ f.await
46
51
 
47
- puts "extended fiber memory cost: #{cost}KB"
52
+ calculate_memory_cost('polyphony fiber', 10000) do
53
+ f = spin { :foo }
54
+ f.await
55
+ f
48
56
  end
49
-
50
- calculate_extended_fiber_memory_cost(10000)
@@ -13,7 +13,7 @@ def handle_client(socket)
13
13
  parser.on_message_complete = proc do |env|
14
14
  reqs << Object.new # parser
15
15
  end
16
- Thread.current.agent.read_loop(socket) do |data|
16
+ socket.read_loop do |data|
17
17
  parser << data
18
18
  while (req = reqs.shift)
19
19
  handle_request(socket, req)
@@ -1,6 +1,8 @@
1
+ #include <netdb.h>
2
+ #include <sys/socket.h>
3
+
1
4
  #include "polyphony.h"
2
5
  #include "../libev/ev.h"
3
- #include <sys/socket.h>
4
6
 
5
7
  VALUE cLibevAgent = Qnil;
6
8
  VALUE cTCPSocket;
@@ -128,7 +130,7 @@ VALUE LibevAgent_poll(VALUE self, VALUE nowait, VALUE current_fiber, VALUE queue
128
130
  GetLibevAgent(self, agent);
129
131
 
130
132
  if (is_nowait) {
131
- int runnable_count = RARRAY_LEN(queue);
133
+ long runnable_count = RARRAY_LEN(queue);
132
134
  agent->run_no_wait_count++;
133
135
  if (agent->run_no_wait_count < runnable_count || agent->run_no_wait_count < 10)
134
136
  return self;
@@ -234,7 +236,7 @@ struct libev_io {
234
236
  VALUE fiber;
235
237
  };
236
238
 
237
- static void LibevAgent_io_callback(EV_P_ ev_io *w, int revents)
239
+ void LibevAgent_io_callback(EV_P_ ev_io *w, int revents)
238
240
  {
239
241
  struct libev_io *watcher = (struct libev_io *)w;
240
242
  Fiber_make_runnable(watcher->fiber, Qnil);
@@ -255,11 +257,32 @@ VALUE libev_agent_await(VALUE self) {
255
257
  return libev_await(agent);
256
258
  }
257
259
 
260
+ VALUE libev_io_wait(struct LibevAgent_t *agent, struct libev_io *watcher, rb_io_t *fptr, int flags) {
261
+ VALUE switchpoint_result;
262
+
263
+ if (watcher->fiber == Qnil) {
264
+ watcher->fiber = rb_fiber_current();
265
+ ev_io_init(&watcher->io, LibevAgent_io_callback, fptr->fd, flags);
266
+ }
267
+ ev_io_start(agent->ev_loop, &watcher->io);
268
+ switchpoint_result = libev_await(agent);
269
+ ev_io_stop(agent->ev_loop, &watcher->io);
270
+
271
+ RB_GC_GUARD(switchpoint_result);
272
+ return switchpoint_result;
273
+ }
274
+
275
+ VALUE libev_snooze() {
276
+ Fiber_make_runnable(rb_fiber_current(), Qnil);
277
+ return Thread_switch_fiber(rb_thread_current());
278
+ }
279
+
258
280
  VALUE LibevAgent_read(VALUE self, VALUE io, VALUE str, VALUE length, VALUE to_eof) {
259
281
  struct LibevAgent_t *agent;
260
282
  struct libev_io watcher;
261
283
  rb_io_t *fptr;
262
- int len = NUM2INT(length);
284
+ long dynamic_len = length == Qnil;
285
+ long len = dynamic_len ? 4096 : NUM2INT(length);
263
286
  int shrinkable = io_setstrbuf(&str, len);
264
287
  char *buf = RSTRING_PTR(str);
265
288
  long total = 0;
@@ -276,44 +299,35 @@ VALUE LibevAgent_read(VALUE self, VALUE io, VALUE str, VALUE length, VALUE to_eo
276
299
 
277
300
  OBJ_TAINT(str);
278
301
 
279
- while (len > 0) {
280
- int n = read(fptr->fd, buf, len);
281
- if (n == 0)
282
- break;
283
- if (n > 0) {
284
- total = total + n;
285
- buf += n;
286
- len -= n;
287
- if (!read_to_eof || (len == 0)) break;
288
- }
289
- else {
302
+ while (1) {
303
+ ssize_t n = read(fptr->fd, buf, len - total);
304
+ if (n < 0) {
290
305
  int e = errno;
291
- if ((e == EWOULDBLOCK || e == EAGAIN)) {
292
- if (watcher.fiber == Qnil) {
293
- watcher.fiber = rb_fiber_current();
294
- ev_io_init(&watcher.io, LibevAgent_io_callback, fptr->fd, EV_READ);
295
- }
296
- ev_io_start(agent->ev_loop, &watcher.io);
297
- switchpoint_result = libev_await(agent);
298
- ev_io_stop(agent->ev_loop, &watcher.io);
299
- if (TEST_EXCEPTION(switchpoint_result)) {
300
- goto error;
301
- }
302
- }
303
- else
304
- rb_syserr_fail(e, strerror(e));
305
- // rb_syserr_fail_path(e, fptr->pathv);
306
+ if (e != EWOULDBLOCK && e != EAGAIN) rb_syserr_fail(e, strerror(e));
307
+
308
+ switchpoint_result = libev_io_wait(agent, &watcher, fptr, EV_READ);
309
+ if (TEST_EXCEPTION(switchpoint_result)) goto error;
306
310
  }
307
- }
311
+ else {
312
+ switchpoint_result = libev_snooze();
313
+ if (TEST_EXCEPTION(switchpoint_result)) goto error;
308
314
 
309
- if (watcher.fiber == Qnil) {
310
- Fiber_make_runnable(rb_fiber_current(), Qnil);
311
- switchpoint_result = Thread_switch_fiber(rb_thread_current());
312
- if (TEST_EXCEPTION(switchpoint_result)) {
313
- goto error;
315
+ if (n == 0) break; // EOF
316
+ total = total + n;
317
+ if (!read_to_eof) break;
318
+
319
+ if (total == len) {
320
+ if (!dynamic_len) break;
321
+
322
+ rb_str_resize(str, total);
323
+ rb_str_modify_expand(str, len);
324
+ buf = RSTRING_PTR(str) + total;
325
+ shrinkable = 0;
326
+ len += len;
327
+ }
328
+ else buf += n;
314
329
  }
315
330
  }
316
-
317
331
  if (total == 0) return Qnil;
318
332
 
319
333
  io_set_read_length(str, total, shrinkable);
@@ -347,8 +361,8 @@ VALUE LibevAgent_read_loop(VALUE self, VALUE io) {
347
361
  struct libev_io watcher;
348
362
  rb_io_t *fptr;
349
363
  VALUE str;
350
- int total;
351
- int len = 8192;
364
+ long total;
365
+ long len = 8192;
352
366
  int shrinkable;
353
367
  char *buf;
354
368
  VALUE switchpoint_result = Qnil;
@@ -366,10 +380,20 @@ VALUE LibevAgent_read_loop(VALUE self, VALUE io) {
366
380
  OBJ_TAINT(str);
367
381
 
368
382
  while (1) {
369
- int n = read(fptr->fd, buf, len);
370
- if (n == 0)
371
- break;
372
- if (n > 0) {
383
+ ssize_t n = read(fptr->fd, buf, len);
384
+ if (n < 0) {
385
+ int e = errno;
386
+ if ((e != EWOULDBLOCK && e != EAGAIN)) rb_syserr_fail(e, strerror(e));
387
+
388
+ switchpoint_result = libev_io_wait(agent, &watcher, fptr, EV_READ);
389
+ if (TEST_EXCEPTION(switchpoint_result)) goto error;
390
+ }
391
+ else {
392
+ switchpoint_result = libev_snooze();
393
+ if (TEST_EXCEPTION(switchpoint_result)) goto error;
394
+
395
+ if (n == 0) break; // EOF
396
+
373
397
  total = n;
374
398
  YIELD_STR();
375
399
  Fiber_make_runnable(rb_fiber_current(), Qnil);
@@ -378,24 +402,6 @@ VALUE LibevAgent_read_loop(VALUE self, VALUE io) {
378
402
  goto error;
379
403
  }
380
404
  }
381
- else {
382
- int e = errno;
383
- if ((e == EWOULDBLOCK || e == EAGAIN)) {
384
- if (watcher.fiber == Qnil) {
385
- watcher.fiber = rb_fiber_current();
386
- ev_io_init(&watcher.io, LibevAgent_io_callback, fptr->fd, EV_READ);
387
- }
388
- ev_io_start(agent->ev_loop, &watcher.io);
389
- switchpoint_result = libev_await(agent);
390
- ev_io_stop(agent->ev_loop, &watcher.io);
391
- if (TEST_EXCEPTION(switchpoint_result)) {
392
- goto error;
393
- }
394
- }
395
- else
396
- rb_syserr_fail(e, strerror(e));
397
- // rb_syserr_fail_path(e, fptr->pathv);
398
- }
399
405
  }
400
406
 
401
407
  RB_GC_GUARD(str);
@@ -414,8 +420,8 @@ VALUE LibevAgent_write(VALUE self, VALUE io, VALUE str) {
414
420
  VALUE switchpoint_result = Qnil;
415
421
 
416
422
  char *buf = StringValuePtr(str);
417
- int len = RSTRING_LEN(str);
418
- int left = len;
423
+ long len = RSTRING_LEN(str);
424
+ long left = len;
419
425
 
420
426
  VALUE underlying_io = rb_iv_get(io, "@io");
421
427
  if (underlying_io != Qnil) io = underlying_io;
@@ -425,37 +431,20 @@ VALUE LibevAgent_write(VALUE self, VALUE io, VALUE str) {
425
431
  watcher.fiber = Qnil;
426
432
 
427
433
  while (left > 0) {
428
- int result = write(fptr->fd, buf, left);
429
- if (result < 0) {
434
+ ssize_t n = write(fptr->fd, buf, left);
435
+ if (n < 0) {
430
436
  int e = errno;
431
- if (e == EAGAIN) {
432
- if (watcher.fiber == Qnil) {
433
- watcher.fiber = rb_fiber_current();
434
- ev_io_init(&watcher.io, LibevAgent_io_callback, fptr->fd, EV_WRITE);
435
- }
436
- ev_io_start(agent->ev_loop, &watcher.io);
437
- switchpoint_result = libev_await(agent);
438
- ev_io_stop(agent->ev_loop, &watcher.io);
439
- if (TEST_EXCEPTION(switchpoint_result))
440
- goto error;
441
- }
442
- else {
443
- rb_syserr_fail(e, strerror(e));
444
- // rb_syserr_fail_path(e, fptr->pathv);
445
-
446
- }
437
+ if ((e != EWOULDBLOCK && e != EAGAIN)) rb_syserr_fail(e, strerror(e));
438
+
439
+ switchpoint_result = libev_io_wait(agent, &watcher, fptr, EV_WRITE);
440
+ if (TEST_EXCEPTION(switchpoint_result)) goto error;
447
441
  }
448
442
  else {
449
- buf += result;
450
- left -= result;
451
- }
452
- }
443
+ switchpoint_result = libev_snooze();
444
+ if (TEST_EXCEPTION(switchpoint_result)) goto error;
453
445
 
454
- if (watcher.fiber == Qnil) {
455
- Fiber_make_runnable(rb_fiber_current(), Qnil);
456
- switchpoint_result = Thread_switch_fiber(rb_thread_current());
457
- if (TEST_EXCEPTION(switchpoint_result)) {
458
- goto error;
446
+ buf += n;
447
+ left -= n;
459
448
  }
460
449
  }
461
450
 
@@ -488,50 +477,39 @@ VALUE LibevAgent_accept(VALUE self, VALUE sock) {
488
477
  fd = accept(fptr->fd, &addr, &len);
489
478
  if (fd < 0) {
490
479
  int e = errno;
491
- if (e == EWOULDBLOCK || e == EAGAIN) {
492
- if (watcher.fiber == Qnil) {
493
- watcher.fiber = rb_fiber_current();
494
- ev_io_init(&watcher.io, LibevAgent_io_callback, fptr->fd, EV_READ);
495
- }
496
- ev_io_start(agent->ev_loop, &watcher.io);
497
- switchpoint_result = libev_await(agent);
498
- ev_io_stop(agent->ev_loop, &watcher.io);
499
-
500
- TEST_RESUME_EXCEPTION(switchpoint_result);
501
- RB_GC_GUARD(watcher.fiber);
502
- RB_GC_GUARD(switchpoint_result);
503
- }
504
- else
505
- rb_syserr_fail(e, strerror(e));
506
- // rb_syserr_fail_path(e, fptr->pathv);
480
+ if ((e != EWOULDBLOCK && e != EAGAIN)) rb_syserr_fail(e, strerror(e));
481
+
482
+ switchpoint_result = libev_io_wait(agent, &watcher, fptr, EV_READ);
483
+ if (TEST_EXCEPTION(switchpoint_result)) goto error;
507
484
  }
508
485
  else {
509
- VALUE connection = rb_obj_alloc(cTCPSocket);
486
+ VALUE socket;
510
487
  rb_io_t *fp;
511
- MakeOpenFile(connection, fp);
488
+ switchpoint_result = libev_snooze();
489
+ if (TEST_EXCEPTION(switchpoint_result)) {
490
+ close(fd); // close fd since we're raising an exception
491
+ goto error;
492
+ }
493
+
494
+ socket = rb_obj_alloc(cTCPSocket);
495
+ MakeOpenFile(socket, fp);
512
496
  rb_update_max_fd(fd);
513
497
  fp->fd = fd;
514
498
  fp->mode = FMODE_READWRITE | FMODE_DUPLEX;
515
- rb_io_ascii8bit_binmode(connection);
499
+ rb_io_ascii8bit_binmode(socket);
516
500
  rb_io_set_nonblock(fp);
517
501
  rb_io_synchronized(fp);
518
502
 
519
503
  // if (rsock_do_not_reverse_lookup) {
520
504
  // fp->mode |= FMODE_NOREVLOOKUP;
521
505
  // }
522
-
523
- if (watcher.fiber == Qnil) {
524
- Fiber_make_runnable(rb_fiber_current(), Qnil);
525
- switchpoint_result = Thread_switch_fiber(rb_thread_current());
526
- if (TEST_EXCEPTION(switchpoint_result)) {
527
- return rb_funcall(rb_mKernel, ID_raise, 1, switchpoint_result);
528
- }
529
- }
530
-
531
- return connection;
506
+ return socket;
532
507
  }
533
508
  }
509
+ RB_GC_GUARD(switchpoint_result);
534
510
  return Qnil;
511
+ error:
512
+ return rb_funcall(rb_mKernel, ID_raise, 1, switchpoint_result);
535
513
  }
536
514
 
537
515
  VALUE LibevAgent_accept_loop(VALUE self, VALUE sock) {
@@ -542,7 +520,7 @@ VALUE LibevAgent_accept_loop(VALUE self, VALUE sock) {
542
520
  struct sockaddr addr;
543
521
  socklen_t len = (socklen_t)sizeof addr;
544
522
  VALUE switchpoint_result = Qnil;
545
- VALUE connection = Qnil;
523
+ VALUE socket = Qnil;
546
524
  VALUE underlying_sock = rb_iv_get(sock, "@io");
547
525
  if (underlying_sock != Qnil) sock = underlying_sock;
548
526
 
@@ -555,46 +533,82 @@ VALUE LibevAgent_accept_loop(VALUE self, VALUE sock) {
555
533
  fd = accept(fptr->fd, &addr, &len);
556
534
  if (fd < 0) {
557
535
  int e = errno;
558
- if (e == EWOULDBLOCK || e == EAGAIN) {
559
- if (watcher.fiber == Qnil) {
560
- watcher.fiber = rb_fiber_current();
561
- ev_io_init(&watcher.io, LibevAgent_io_callback, fptr->fd, EV_READ);
562
- }
563
- ev_io_start(agent->ev_loop, &watcher.io);
564
- switchpoint_result = libev_await(agent);
565
- ev_io_stop(agent->ev_loop, &watcher.io);
566
-
567
- TEST_RESUME_EXCEPTION(switchpoint_result);
568
- }
569
- else
570
- rb_syserr_fail(e, strerror(e));
571
- // rb_syserr_fail_path(e, fptr->pathv);
536
+ if ((e != EWOULDBLOCK && e != EAGAIN)) rb_syserr_fail(e, strerror(e));
537
+
538
+ switchpoint_result = libev_io_wait(agent, &watcher, fptr, EV_READ);
539
+ if (TEST_EXCEPTION(switchpoint_result)) goto error;
572
540
  }
573
541
  else {
574
542
  rb_io_t *fp;
575
- connection = rb_obj_alloc(cTCPSocket);
576
- MakeOpenFile(connection, fp);
543
+ switchpoint_result = libev_snooze();
544
+ if (TEST_EXCEPTION(switchpoint_result)) {
545
+ close(fd); // close fd since we're raising an exception
546
+ goto error;
547
+ }
548
+
549
+ socket = rb_obj_alloc(cTCPSocket);
550
+ MakeOpenFile(socket, fp);
577
551
  rb_update_max_fd(fd);
578
552
  fp->fd = fd;
579
553
  fp->mode = FMODE_READWRITE | FMODE_DUPLEX;
580
- rb_io_ascii8bit_binmode(connection);
554
+ rb_io_ascii8bit_binmode(socket);
581
555
  rb_io_set_nonblock(fp);
582
556
  rb_io_synchronized(fp);
583
557
 
584
- rb_yield(connection);
585
- connection = Qnil;
586
-
587
- Fiber_make_runnable(rb_fiber_current(), Qnil);
588
- switchpoint_result = Thread_switch_fiber(rb_thread_current());
589
- TEST_RESUME_EXCEPTION(switchpoint_result);
558
+ rb_yield(socket);
559
+ socket = Qnil;
590
560
  }
591
561
  }
592
562
 
593
- RB_GC_GUARD(connection);
563
+ RB_GC_GUARD(socket);
594
564
  RB_GC_GUARD(watcher.fiber);
595
565
  RB_GC_GUARD(switchpoint_result);
566
+ return Qnil;
567
+ error:
568
+ return rb_funcall(rb_mKernel, ID_raise, 1, switchpoint_result);
596
569
  }
597
570
 
571
+ // VALUE LibevAgent_connect(VALUE self, VALUE sock, VALUE host, VALUE port) {
572
+ // struct LibevAgent_t *agent;
573
+ // struct libev_io watcher;
574
+ // rb_io_t *fptr;
575
+ // struct sockaddr_in addr;
576
+ // char *host_buf = StringValueCStr(host);
577
+ // VALUE switchpoint_result = Qnil;
578
+ // VALUE underlying_sock = rb_iv_get(sock, "@io");
579
+ // if (underlying_sock != Qnil) sock = underlying_sock;
580
+
581
+ // GetLibevAgent(self, agent);
582
+ // GetOpenFile(sock, fptr);
583
+ // rb_io_set_nonblock(fptr);
584
+ // watcher.fiber = Qnil;
585
+
586
+ // addr.sin_family = AF_INET;
587
+ // addr.sin_addr.s_addr = inet_addr(host_buf);
588
+ // addr.sin_port = htons(NUM2INT(port));
589
+
590
+ // while (1) {
591
+ // int result = connect(fptr->fd, &addr, sizeof(addr));
592
+ // if (result < 0) {
593
+ // int e = errno;
594
+ // if ((e != EWOULDBLOCK && e != EAGAIN)) rb_syserr_fail(e, strerror(e));
595
+
596
+ // switchpoint_result = libev_io_wait(agent, &watcher, fptr, EV_WRITE);
597
+ // if (TEST_EXCEPTION(switchpoint_result)) goto error;
598
+ // }
599
+ // else {
600
+ // switchpoint_result = libev_snooze();
601
+ // if (TEST_EXCEPTION(switchpoint_result)) goto error;
602
+
603
+ // return sock;
604
+ // }
605
+ // }
606
+ // RB_GC_GUARD(switchpoint_result);
607
+ // return Qnil;
608
+ // error:
609
+ // return rb_funcall(rb_mKernel, ID_raise, 1, switchpoint_result);
610
+ // }
611
+
598
612
  VALUE LibevAgent_wait_io(VALUE self, VALUE io, VALUE write) {
599
613
  struct LibevAgent_t *agent;
600
614
  struct libev_io watcher;
@@ -624,7 +638,7 @@ struct libev_timer {
624
638
  VALUE fiber;
625
639
  };
626
640
 
627
- static void LibevAgent_timer_callback(EV_P_ ev_timer *w, int revents)
641
+ void LibevAgent_timer_callback(EV_P_ ev_timer *w, int revents)
628
642
  {
629
643
  struct libev_timer *watcher = (struct libev_timer *)w;
630
644
  Fiber_make_runnable(watcher->fiber, Qnil);
@@ -654,7 +668,7 @@ struct libev_child {
654
668
  VALUE fiber;
655
669
  };
656
670
 
657
- static void LibevAgent_child_callback(EV_P_ ev_child *w, int revents)
671
+ void LibevAgent_child_callback(EV_P_ ev_child *w, int revents)
658
672
  {
659
673
  struct libev_child *watcher = (struct libev_child *)w;
660
674
  int exit_status = w->rstatus >> 8; // weird, why should we do this?
@@ -712,6 +726,7 @@ void Init_LibevAgent() {
712
726
  rb_define_method(cLibevAgent, "write", LibevAgent_write, 2);
713
727
  rb_define_method(cLibevAgent, "accept", LibevAgent_accept, 1);
714
728
  rb_define_method(cLibevAgent, "accept_loop", LibevAgent_accept_loop, 1);
729
+ // rb_define_method(cLibevAgent, "connect", LibevAgent_accept, 3);
715
730
  rb_define_method(cLibevAgent, "wait_io", LibevAgent_wait_io, 2);
716
731
  rb_define_method(cLibevAgent, "sleep", LibevAgent_sleep, 1);
717
732
  rb_define_method(cLibevAgent, "waitpid", LibevAgent_waitpid, 1);