Class: Event::Backend::KQueue
- Inherits:
-
Object
- Object
- Event::Backend::KQueue
- Defined in:
- ext/event/backend/kqueue.c
Instance Method Summary collapse
- #close ⇒ Object
- #initialize(loop) ⇒ Object constructor
- #io_wait(fiber, io, events) ⇒ Object
- #process_wait(fiber, pid, flags) ⇒ Object
- #select(duration) ⇒ Object
Constructor Details
#initialize(loop) ⇒ Object
88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 |
# File 'ext/event/backend/kqueue.c', line 88 VALUE Event_Backend_KQueue_initialize(VALUE self, VALUE loop) { struct Event_Backend_KQueue *data = NULL; TypedData_Get_Struct(self, struct Event_Backend_KQueue, &Event_Backend_KQueue_Type, data); data->loop = loop; int result = kqueue(); if (result == -1) { rb_sys_fail("kqueue"); } else { ioctl(result, FIOCLEX); data->descriptor = result; rb_update_max_fd(data->descriptor); } return self; } |
Instance Method Details
#close ⇒ Object
107 108 109 110 111 112 113 114 |
# File 'ext/event/backend/kqueue.c', line 107 VALUE Event_Backend_KQueue_close(VALUE self) { struct Event_Backend_KQueue *data = NULL; TypedData_Get_Struct(self, struct Event_Backend_KQueue, &Event_Backend_KQueue_Type, data); close_internal(data); return Qnil; } |
#io_wait(fiber, io, events) ⇒ Object
289 290 291 292 293 294 295 296 297 298 299 300 301 302 |
# File 'ext/event/backend/kqueue.c', line 289 VALUE Event_Backend_KQueue_io_wait(VALUE self, VALUE fiber, VALUE io, VALUE events) { struct Event_Backend_KQueue *data = NULL; TypedData_Get_Struct(self, struct Event_Backend_KQueue, &Event_Backend_KQueue_Type, data); int descriptor = NUM2INT(rb_funcall(io, id_fileno, 0)); struct io_wait_arguments io_wait_arguments = { .events = io_add_filters(data->descriptor, descriptor, NUM2INT(events), fiber), .data = data, .descriptor = descriptor, }; return rb_rescue(io_wait_transfer, (VALUE)&io_wait_arguments, io_wait_rescue, (VALUE)&io_wait_arguments); } |
#process_wait(fiber, pid, flags) ⇒ Object
177 178 179 180 181 182 183 184 185 186 187 188 189 190 191 192 193 194 |
# File 'ext/event/backend/kqueue.c', line 177 VALUE Event_Backend_KQueue_process_wait(VALUE self, VALUE fiber, VALUE pid, VALUE flags) { struct Event_Backend_KQueue *data = NULL; TypedData_Get_Struct(self, struct Event_Backend_KQueue, &Event_Backend_KQueue_Type, data); struct process_wait_arguments process_wait_arguments = { .data = data, .pid = NUM2PIDT(pid), .flags = NUM2INT(flags), }; int waiting = process_add_filters(data->descriptor, process_wait_arguments.pid, fiber); if (waiting) { return rb_rescue(process_wait_transfer, (VALUE)&process_wait_arguments, process_wait_rescue, (VALUE)&process_wait_arguments); } else { return Event_Backend_process_status_wait(process_wait_arguments.pid); } } |
#select(duration) ⇒ Object
372 373 374 375 376 377 378 379 380 381 382 383 384 385 386 387 388 389 390 391 392 393 394 395 396 397 398 399 400 401 402 403 404 405 406 407 408 409 410 411 412 413 414 |
# File 'ext/event/backend/kqueue.c', line 372 VALUE Event_Backend_KQueue_select(VALUE self, VALUE duration) { struct Event_Backend_KQueue *data = NULL; TypedData_Get_Struct(self, struct Event_Backend_KQueue, &Event_Backend_KQueue_Type, data); struct select_arguments arguments = { .data = data, .count = KQUEUE_MAX_EVENTS, .storage = { .tv_sec = 0, .tv_nsec = 0 } }; // We break this implementation into two parts. // (1) count = kevent(..., timeout = 0) // (2) without gvl: kevent(..., timeout = 0) if count == 0 and timeout != 0 // This allows us to avoid releasing and reacquiring the GVL. // Non-comprehensive testing shows this gives a 1.5x speedup. arguments.timeout = &arguments.storage; // First do the syscall with no timeout to get any immediately available events: select_internal_with_gvl(&arguments); // If there were no pending events, if we have a timeout, wait for more events: if (arguments.count == 0) { arguments.timeout = make_timeout(duration, &arguments.storage); if (!timeout_nonblocking(arguments.timeout)) { arguments.count = KQUEUE_MAX_EVENTS; select_internal_without_gvl(&arguments); } } for (int i = 0; i < arguments.count; i += 1) { VALUE fiber = (VALUE)arguments.events[i].udata; VALUE result = INT2NUM(arguments.events[i].filter); Event_Backend_transfer_result(fiber, result); } return INT2NUM(arguments.count); } |