blob: b9901bf8efdae6be0ba9285897294b2f0c89a9bc [file] [log] [blame]
murgatroid999030c812016-09-16 13:25:08 -07001/*
2 *
Jan Tattermusch7897ae92017-06-07 22:57:36 +02003 * Copyright 2016 gRPC authors.
murgatroid999030c812016-09-16 13:25:08 -07004 *
Jan Tattermusch7897ae92017-06-07 22:57:36 +02005 * Licensed under the Apache License, Version 2.0 (the "License");
6 * you may not use this file except in compliance with the License.
7 * You may obtain a copy of the License at
murgatroid999030c812016-09-16 13:25:08 -07008 *
Jan Tattermusch7897ae92017-06-07 22:57:36 +02009 * http://www.apache.org/licenses/LICENSE-2.0
murgatroid999030c812016-09-16 13:25:08 -070010 *
Jan Tattermusch7897ae92017-06-07 22:57:36 +020011 * Unless required by applicable law or agreed to in writing, software
12 * distributed under the License is distributed on an "AS IS" BASIS,
13 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
14 * See the License for the specific language governing permissions and
15 * limitations under the License.
murgatroid999030c812016-09-16 13:25:08 -070016 *
17 */
18
19#include "src/core/lib/iomgr/port.h"
20
21#ifdef GRPC_UV
22
murgatroid992c287ca2016-10-07 09:55:35 -070023#include <uv.h>
24
25#include <string.h>
26
murgatroid99d0cda5c2017-03-07 18:04:52 -080027#include <grpc/support/alloc.h>
murgatroid992c287ca2016-10-07 09:55:35 -070028#include <grpc/support/log.h>
murgatroid999030c812016-09-16 13:25:08 -070029#include <grpc/support/sync.h>
30
murgatroid99b8e07ad2017-07-18 13:20:55 -070031#include "src/core/lib/iomgr/iomgr_uv.h"
murgatroid999030c812016-09-16 13:25:08 -070032#include "src/core/lib/iomgr/pollset.h"
33#include "src/core/lib/iomgr/pollset_uv.h"
34
ncteisen973863d2017-06-12 10:28:50 -070035#include "src/core/lib/debug/trace.h"
36
37#ifndef NDEBUG
ncteisen7712c7c2017-07-12 23:11:27 -070038grpc_tracer_flag grpc_trace_fd_refcount =
39 GRPC_TRACER_INITIALIZER(false, "fd_refcount");
ncteisen973863d2017-06-12 10:28:50 -070040#endif
41
murgatroid992c287ca2016-10-07 09:55:35 -070042struct grpc_pollset {
43 uv_timer_t timer;
44 int shutting_down;
45};
46
47/* Indicates that grpc_pollset_work should run an iteration of the UV loop
48 before running callbacks. This defaults to 1, and should be disabled if
49 grpc_pollset_work will be called within the callstack of uv_run */
50int grpc_pollset_work_run_loop;
51
murgatroid999030c812016-09-16 13:25:08 -070052gpr_mu grpc_polling_mu;
53
murgatroid998c2946a2017-02-27 10:19:21 -080054/* This is used solely to kick the uv loop, by setting a callback to be run
55 immediately in the next loop iteration.
56 Note: In the future, if there is a bug that involves missing wakeups in the
57 future, try adding a uv_async_t to kick the loop differently */
murgatroid99d0cda5c2017-03-07 18:04:52 -080058uv_timer_t *dummy_uv_handle;
murgatroid998c2946a2017-02-27 10:19:21 -080059
murgatroid992c287ca2016-10-07 09:55:35 -070060size_t grpc_pollset_size() { return sizeof(grpc_pollset); }
murgatroid999030c812016-09-16 13:25:08 -070061
murgatroid9906ab69d2017-02-27 16:39:27 -080062void dummy_timer_cb(uv_timer_t *handle) {}
murgatroid998c2946a2017-02-27 10:19:21 -080063
murgatroid99d0cda5c2017-03-07 18:04:52 -080064void dummy_handle_close_cb(uv_handle_t *handle) { gpr_free(handle); }
65
murgatroid992c287ca2016-10-07 09:55:35 -070066void grpc_pollset_global_init(void) {
67 gpr_mu_init(&grpc_polling_mu);
Yash Tibrewal19f0a902017-09-22 09:56:18 -070068 dummy_uv_handle = (uv_timer_t *)gpr_malloc(sizeof(uv_timer_t));
murgatroid99d0cda5c2017-03-07 18:04:52 -080069 uv_timer_init(uv_default_loop(), dummy_uv_handle);
murgatroid992c287ca2016-10-07 09:55:35 -070070 grpc_pollset_work_run_loop = 1;
71}
murgatroid999030c812016-09-16 13:25:08 -070072
murgatroid998c2946a2017-02-27 10:19:21 -080073void grpc_pollset_global_shutdown(void) {
murgatroid999d7def72017-07-20 16:41:27 -070074 GRPC_UV_ASSERT_SAME_THREAD();
murgatroid998c2946a2017-02-27 10:19:21 -080075 gpr_mu_destroy(&grpc_polling_mu);
murgatroid99d0cda5c2017-03-07 18:04:52 -080076 uv_close((uv_handle_t *)dummy_uv_handle, dummy_handle_close_cb);
murgatroid998c2946a2017-02-27 10:19:21 -080077}
murgatroid999030c812016-09-16 13:25:08 -070078
murgatroid99d0cda5c2017-03-07 18:04:52 -080079static void timer_run_cb(uv_timer_t *timer) {}
80
81static void timer_close_cb(uv_handle_t *handle) { handle->data = (void *)1; }
82
murgatroid999030c812016-09-16 13:25:08 -070083void grpc_pollset_init(grpc_pollset *pollset, gpr_mu **mu) {
murgatroid999d7def72017-07-20 16:41:27 -070084 GRPC_UV_ASSERT_SAME_THREAD();
murgatroid999030c812016-09-16 13:25:08 -070085 *mu = &grpc_polling_mu;
murgatroid992c287ca2016-10-07 09:55:35 -070086 uv_timer_init(uv_default_loop(), &pollset->timer);
87 pollset->shutting_down = 0;
88}
89
murgatroid999030c812016-09-16 13:25:08 -070090void grpc_pollset_shutdown(grpc_exec_ctx *exec_ctx, grpc_pollset *pollset,
murgatroid99dedb9232016-09-26 13:54:04 -070091 grpc_closure *closure) {
murgatroid992c287ca2016-10-07 09:55:35 -070092 GPR_ASSERT(!pollset->shutting_down);
murgatroid999d7def72017-07-20 16:41:27 -070093 GRPC_UV_ASSERT_SAME_THREAD();
murgatroid992c287ca2016-10-07 09:55:35 -070094 pollset->shutting_down = 1;
95 if (grpc_pollset_work_run_loop) {
96 // Drain any pending UV callbacks without blocking
97 uv_run(uv_default_loop(), UV_RUN_NOWAIT);
murgatroid998c2946a2017-02-27 10:19:21 -080098 } else {
99 // kick the loop once
murgatroid99d0cda5c2017-03-07 18:04:52 -0800100 uv_timer_start(dummy_uv_handle, dummy_timer_cb, 0, 0);
murgatroid992c287ca2016-10-07 09:55:35 -0700101 }
ncteisen969b46e2017-06-08 14:57:11 -0700102 GRPC_CLOSURE_SCHED(exec_ctx, closure, GRPC_ERROR_NONE);
murgatroid999030c812016-09-16 13:25:08 -0700103}
104
Craig Tillerffe51bd2017-04-21 08:30:24 -0700105void grpc_pollset_destroy(grpc_exec_ctx *exec_ctx, grpc_pollset *pollset) {
murgatroid999d7def72017-07-20 16:41:27 -0700106 GRPC_UV_ASSERT_SAME_THREAD();
murgatroid99aa9c5782016-10-10 12:16:13 -0700107 uv_close((uv_handle_t *)&pollset->timer, timer_close_cb);
murgatroid992c287ca2016-10-07 09:55:35 -0700108 // timer.data is a boolean indicating that the timer has finished closing
109 pollset->timer.data = (void *)0;
110 if (grpc_pollset_work_run_loop) {
111 while (!pollset->timer.data) {
112 uv_run(uv_default_loop(), UV_RUN_NOWAIT);
113 }
114 }
115}
murgatroid999030c812016-09-16 13:25:08 -0700116
murgatroid999030c812016-09-16 13:25:08 -0700117grpc_error *grpc_pollset_work(grpc_exec_ctx *exec_ctx, grpc_pollset *pollset,
118 grpc_pollset_worker **worker_hdl,
Craig Tillerde682c52017-09-20 22:18:16 -0700119 grpc_millis deadline) {
murgatroid992c287ca2016-10-07 09:55:35 -0700120 uint64_t timeout;
murgatroid999d7def72017-07-20 16:41:27 -0700121 GRPC_UV_ASSERT_SAME_THREAD();
murgatroid992c287ca2016-10-07 09:55:35 -0700122 gpr_mu_unlock(&grpc_polling_mu);
123 if (grpc_pollset_work_run_loop) {
Craig Tillerde682c52017-09-20 22:18:16 -0700124 grpc_millis now = grpc_exec_ctx_now(exec_ctx);
125 if (deadline >= now) {
126 timeout = deadline - now;
murgatroid992c287ca2016-10-07 09:55:35 -0700127 } else {
128 timeout = 0;
129 }
130 /* We special-case timeout=0 so that we don't bother with the timer when
131 the loop won't block anyway */
132 if (timeout > 0) {
133 uv_timer_start(&pollset->timer, timer_run_cb, timeout, 0);
134 /* Run until there is some I/O activity or the timer triggers. It doesn't
135 matter which happens */
136 uv_run(uv_default_loop(), UV_RUN_ONCE);
137 uv_timer_stop(&pollset->timer);
138 } else {
139 uv_run(uv_default_loop(), UV_RUN_NOWAIT);
140 }
murgatroid999030c812016-09-16 13:25:08 -0700141 }
murgatroid992c287ca2016-10-07 09:55:35 -0700142 if (!grpc_closure_list_empty(exec_ctx->closure_list)) {
143 grpc_exec_ctx_flush(exec_ctx);
144 }
145 gpr_mu_lock(&grpc_polling_mu);
murgatroid999030c812016-09-16 13:25:08 -0700146 return GRPC_ERROR_NONE;
147}
148
Yash Tibrewal85629d02017-09-19 16:47:12 -0700149grpc_error *grpc_pollset_kick(grpc_exec_ctx *exec_ctx, grpc_pollset *pollset,
murgatroid999030c812016-09-16 13:25:08 -0700150 grpc_pollset_worker *specific_worker) {
murgatroid999d7def72017-07-20 16:41:27 -0700151 GRPC_UV_ASSERT_SAME_THREAD();
murgatroid99d0cda5c2017-03-07 18:04:52 -0800152 uv_timer_start(dummy_uv_handle, dummy_timer_cb, 0, 0);
murgatroid999030c812016-09-16 13:25:08 -0700153 return GRPC_ERROR_NONE;
154}
155
156#endif /* GRPC_UV */