blob: deb661548d291a34b0f2123ca75e86544dfa543e [file] [log] [blame]
Craig Tillerf2d39b72015-01-21 15:06:08 -08001/*
2 *
Craig Tiller06059952015-02-18 08:34:56 -08003 * Copyright 2015, Google Inc.
Craig Tillerf2d39b72015-01-21 15:06:08 -08004 * All rights reserved.
5 *
6 * Redistribution and use in source and binary forms, with or without
7 * modification, are permitted provided that the following conditions are
8 * met:
9 *
10 * * Redistributions of source code must retain the above copyright
11 * notice, this list of conditions and the following disclaimer.
12 * * Redistributions in binary form must reproduce the above
13 * copyright notice, this list of conditions and the following disclaimer
14 * in the documentation and/or other materials provided with the
15 * distribution.
16 * * Neither the name of Google Inc. nor the names of its
17 * contributors may be used to endorse or promote products derived from
18 * this software without specific prior written permission.
19 *
20 * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
21 * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
22 * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
23 * A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
24 * OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
25 * SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
26 * LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
27 * DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
28 * THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
29 * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
30 * OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
31 *
32 */
33
Craig Tillerd14a1a52015-01-21 15:26:29 -080034#include <grpc/support/port_platform.h>
Craig Tillerf2d39b72015-01-21 15:06:08 -080035
Nicolas "Pixel" Noble21f627a2015-02-04 01:31:14 +010036#ifdef GPR_WINSOCK_SOCKET
Craig Tillerf2d39b72015-01-21 15:06:08 -080037
Craig Tiller46714f92015-11-03 11:29:36 -080038#include <grpc/support/log.h>
Nicolas "Pixel" Noble21f627a2015-02-04 01:31:14 +010039#include <grpc/support/thd.h>
40
David Garcia Quintasf747bbc2015-10-04 23:09:47 -070041#include "src/core/iomgr/timer_internal.h"
Nicolas "Pixel" Noble21f627a2015-02-04 01:31:14 +010042#include "src/core/iomgr/iomgr_internal.h"
Craig Tiller3dedb702015-09-24 10:31:26 -070043#include "src/core/iomgr/iocp_windows.h"
Nicolas "Pixel" Noblefcb16e12015-06-24 23:45:30 +020044#include "src/core/iomgr/pollset.h"
Nicolas "Pixel" Noble21f627a2015-02-04 01:31:14 +010045#include "src/core/iomgr/pollset_windows.h"
46
Craig Tiller084c8082015-09-29 11:19:06 -070047gpr_mu grpc_polling_mu;
Craig Tiller3dedb702015-09-24 10:31:26 -070048static grpc_pollset_worker *g_active_poller;
49static grpc_pollset_worker g_global_root_worker;
50
51void grpc_pollset_global_init() {
Craig Tiller084c8082015-09-29 11:19:06 -070052 gpr_mu_init(&grpc_polling_mu);
Craig Tiller3dedb702015-09-24 10:31:26 -070053 g_active_poller = NULL;
54 g_global_root_worker.links[GRPC_POLLSET_WORKER_LINK_GLOBAL].next =
Craig Tiller71a0f9d2015-09-28 17:22:01 -070055 g_global_root_worker.links[GRPC_POLLSET_WORKER_LINK_GLOBAL].prev =
56 &g_global_root_worker;
Craig Tillerd1885e02015-07-30 14:32:31 -070057}
58
Craig Tillerb7a59772015-10-01 08:01:58 -070059void grpc_pollset_global_shutdown() { gpr_mu_destroy(&grpc_polling_mu); }
Craig Tillerd1885e02015-07-30 14:32:31 -070060
Craig Tiller71a0f9d2015-09-28 17:22:01 -070061static void remove_worker(grpc_pollset_worker *worker,
Craig Tiller3dedb702015-09-24 10:31:26 -070062 grpc_pollset_worker_link_type type) {
63 worker->links[type].prev->links[type].next = worker->links[type].next;
64 worker->links[type].next->links[type].prev = worker->links[type].prev;
65 worker->links[type].next = worker->links[type].prev = worker;
66}
67
Craig Tiller71a0f9d2015-09-28 17:22:01 -070068static int has_workers(grpc_pollset_worker *root,
69 grpc_pollset_worker_link_type type) {
Craig Tiller3dedb702015-09-24 10:31:26 -070070 return root->links[type].next != root;
71}
72
73static grpc_pollset_worker *pop_front_worker(
74 grpc_pollset_worker *root, grpc_pollset_worker_link_type type) {
75 if (has_workers(root, type)) {
76 grpc_pollset_worker *w = root->links[type].next;
77 remove_worker(w, type);
Craig Tillera82950e2015-09-22 12:33:20 -070078 return w;
79 } else {
80 return NULL;
81 }
Craig Tillerd1885e02015-07-30 14:32:31 -070082}
83
Craig Tiller71a0f9d2015-09-28 17:22:01 -070084static void push_back_worker(grpc_pollset_worker *root,
85 grpc_pollset_worker_link_type type,
Craig Tiller3dedb702015-09-24 10:31:26 -070086 grpc_pollset_worker *worker) {
87 worker->links[type].next = root;
88 worker->links[type].prev = worker->links[type].next->links[type].prev;
Craig Tiller71a0f9d2015-09-28 17:22:01 -070089 worker->links[type].prev->links[type].next =
90 worker->links[type].next->links[type].prev = worker;
Craig Tillerd1885e02015-07-30 14:32:31 -070091}
92
Craig Tiller71a0f9d2015-09-28 17:22:01 -070093static void push_front_worker(grpc_pollset_worker *root,
94 grpc_pollset_worker_link_type type,
Craig Tiller3dedb702015-09-24 10:31:26 -070095 grpc_pollset_worker *worker) {
96 worker->links[type].prev = root;
97 worker->links[type].next = worker->links[type].prev->links[type].next;
Craig Tiller71a0f9d2015-09-28 17:22:01 -070098 worker->links[type].prev->links[type].next =
99 worker->links[type].next->links[type].prev = worker;
Craig Tillerd1885e02015-07-30 14:32:31 -0700100}
101
Nicolas "Pixel" Noble0f981e92015-05-03 10:40:56 +0200102/* There isn't really any such thing as a pollset under Windows, due to the
103 nature of the IO completion ports. We're still going to provide a minimal
104 set of features for the sake of the rest of grpc. But grpc_pollset_work
105 won't actually do any polling, and return as quickly as possible. */
106
Craig Tillera82950e2015-09-22 12:33:20 -0700107void grpc_pollset_init(grpc_pollset *pollset) {
108 memset(pollset, 0, sizeof(*pollset));
Craig Tiller71a0f9d2015-09-28 17:22:01 -0700109 pollset->root_worker.links[GRPC_POLLSET_WORKER_LINK_POLLSET].next =
110 pollset->root_worker.links[GRPC_POLLSET_WORKER_LINK_POLLSET].prev =
111 &pollset->root_worker;
Nicolas Noblea14ebbb2015-06-22 14:17:34 -0700112}
Nicolas "Pixel" Noble21f627a2015-02-04 01:31:14 +0100113
Craig Tiller82f9bd82015-09-23 09:31:51 -0700114void grpc_pollset_shutdown(grpc_exec_ctx *exec_ctx, grpc_pollset *pollset,
115 grpc_closure *closure) {
Nicolas "Pixel" Noble857d2502015-06-25 21:58:18 +0200116 pollset->shutting_down = 1;
Craig Tillera82950e2015-09-22 12:33:20 -0700117 grpc_pollset_kick(pollset, GRPC_POLLSET_KICK_BROADCAST);
Craig Tiller3dedb702015-09-24 10:31:26 -0700118 if (!pollset->is_iocp_worker) {
119 grpc_exec_ctx_enqueue(exec_ctx, closure, 1);
120 } else {
121 pollset->on_shutdown = closure;
122 }
David Klempnerb5056612015-02-24 14:22:50 -0800123}
124
Craig Tillerb7a59772015-10-01 08:01:58 -0700125void grpc_pollset_destroy(grpc_pollset *pollset) {}
Nicolas "Pixel" Noble21f627a2015-02-04 01:31:14 +0100126
Craig Tiller46714f92015-11-03 11:29:36 -0800127void grpc_pollset_reset(grpc_pollset *pollset) {
128 GPR_ASSERT(pollset->shutting_down);
Craig Tiller3c53bb22015-11-10 14:24:36 +0000129 GPR_ASSERT(
130 !has_workers(&pollset->root_worker, GRPC_POLLSET_WORKER_LINK_POLLSET));
Craig Tiller46714f92015-11-03 11:29:36 -0800131 pollset->shutting_down = 0;
132 pollset->is_iocp_worker = 0;
133 pollset->kicked_without_pollers = 0;
134 pollset->on_shutdown = NULL;
135}
Craig Tillere8b5f622015-11-02 14:15:03 -0800136
Craig Tiller565b18b2015-09-23 10:09:42 -0700137void grpc_pollset_work(grpc_exec_ctx *exec_ctx, grpc_pollset *pollset,
138 grpc_pollset_worker *worker, gpr_timespec now,
139 gpr_timespec deadline) {
Craig Tillerd1885e02015-07-30 14:32:31 -0700140 int added_worker = 0;
Craig Tiller71a0f9d2015-09-28 17:22:01 -0700141 worker->links[GRPC_POLLSET_WORKER_LINK_POLLSET].next =
Craig Tillerb7a59772015-10-01 08:01:58 -0700142 worker->links[GRPC_POLLSET_WORKER_LINK_POLLSET].prev =
143 worker->links[GRPC_POLLSET_WORKER_LINK_GLOBAL].next =
144 worker->links[GRPC_POLLSET_WORKER_LINK_GLOBAL].prev = NULL;
Craig Tiller084c8082015-09-29 11:19:06 -0700145 worker->kicked = 0;
146 worker->pollset = pollset;
Craig Tillera82950e2015-09-22 12:33:20 -0700147 gpr_cv_init(&worker->cv);
David Garcia Quintasf747bbc2015-10-04 23:09:47 -0700148 if (grpc_timer_check(exec_ctx, now, &deadline)) {
Craig Tillera82950e2015-09-22 12:33:20 -0700149 goto done;
150 }
151 if (!pollset->kicked_without_pollers && !pollset->shutting_down) {
Craig Tiller3dedb702015-09-24 10:31:26 -0700152 if (g_active_poller == NULL) {
153 grpc_pollset_worker *next_worker;
154 /* become poller */
155 pollset->is_iocp_worker = 1;
156 g_active_poller = worker;
Craig Tiller084c8082015-09-29 11:19:06 -0700157 gpr_mu_unlock(&grpc_polling_mu);
Craig Tiller3dedb702015-09-24 10:31:26 -0700158 grpc_iocp_work(exec_ctx, deadline);
Craig Tiller0b6312e2015-09-30 08:36:27 -0700159 grpc_exec_ctx_flush(exec_ctx);
Craig Tiller084c8082015-09-29 11:19:06 -0700160 gpr_mu_lock(&grpc_polling_mu);
Craig Tiller3dedb702015-09-24 10:31:26 -0700161 pollset->is_iocp_worker = 0;
162 g_active_poller = NULL;
Craig Tiller084c8082015-09-29 11:19:06 -0700163 /* try to get a worker from this pollsets worker list */
Craig Tillerb7a59772015-10-01 08:01:58 -0700164 next_worker = pop_front_worker(&pollset->root_worker,
165 GRPC_POLLSET_WORKER_LINK_POLLSET);
166 if (next_worker == NULL) {
167 /* try to get a worker from the global list */
168 next_worker = pop_front_worker(&g_global_root_worker,
169 GRPC_POLLSET_WORKER_LINK_GLOBAL);
170 }
Craig Tiller3dedb702015-09-24 10:31:26 -0700171 if (next_worker != NULL) {
Craig Tiller084c8082015-09-29 11:19:06 -0700172 next_worker->kicked = 1;
Craig Tiller3dedb702015-09-24 10:31:26 -0700173 gpr_cv_signal(&next_worker->cv);
174 }
Craig Tiller3dedb702015-09-24 10:31:26 -0700175
176 if (pollset->shutting_down && pollset->on_shutdown != NULL) {
177 grpc_exec_ctx_enqueue(exec_ctx, pollset->on_shutdown, 1);
178 pollset->on_shutdown = NULL;
179 }
180 goto done;
181 }
Craig Tiller71a0f9d2015-09-28 17:22:01 -0700182 push_front_worker(&g_global_root_worker, GRPC_POLLSET_WORKER_LINK_GLOBAL,
183 worker);
Craig Tiller71a0f9d2015-09-28 17:22:01 -0700184 push_front_worker(&pollset->root_worker, GRPC_POLLSET_WORKER_LINK_POLLSET,
185 worker);
Craig Tillera82950e2015-09-22 12:33:20 -0700186 added_worker = 1;
Craig Tiller084c8082015-09-29 11:19:06 -0700187 while (!worker->kicked) {
188 if (gpr_cv_wait(&worker->cv, &grpc_polling_mu, deadline)) {
189 break;
190 }
191 }
Craig Tillera82950e2015-09-22 12:33:20 -0700192 } else {
193 pollset->kicked_without_pollers = 0;
194 }
Craig Tillerd1885e02015-07-30 14:32:31 -0700195done:
Craig Tiller258f8de2015-09-23 12:32:40 -0700196 if (!grpc_closure_list_empty(exec_ctx->closure_list)) {
Craig Tiller084c8082015-09-29 11:19:06 -0700197 gpr_mu_unlock(&grpc_polling_mu);
Craig Tiller258f8de2015-09-23 12:32:40 -0700198 grpc_exec_ctx_flush(exec_ctx);
Craig Tiller084c8082015-09-29 11:19:06 -0700199 gpr_mu_lock(&grpc_polling_mu);
Craig Tiller258f8de2015-09-23 12:32:40 -0700200 }
Craig Tillera82950e2015-09-22 12:33:20 -0700201 if (added_worker) {
Craig Tiller3dedb702015-09-24 10:31:26 -0700202 remove_worker(worker, GRPC_POLLSET_WORKER_LINK_GLOBAL);
203 remove_worker(worker, GRPC_POLLSET_WORKER_LINK_POLLSET);
Craig Tillera82950e2015-09-22 12:33:20 -0700204 }
Craig Tillerc3e57782015-09-29 08:37:32 -0700205 gpr_cv_destroy(&worker->cv);
Nicolas "Pixel" Noble21f627a2015-02-04 01:31:14 +0100206}
207
Craig Tillera82950e2015-09-22 12:33:20 -0700208void grpc_pollset_kick(grpc_pollset *p, grpc_pollset_worker *specific_worker) {
209 if (specific_worker != NULL) {
210 if (specific_worker == GRPC_POLLSET_KICK_BROADCAST) {
Craig Tiller71a0f9d2015-09-28 17:22:01 -0700211 for (specific_worker =
212 p->root_worker.links[GRPC_POLLSET_WORKER_LINK_POLLSET].next;
Craig Tillera82950e2015-09-22 12:33:20 -0700213 specific_worker != &p->root_worker;
Craig Tiller71a0f9d2015-09-28 17:22:01 -0700214 specific_worker =
215 specific_worker->links[GRPC_POLLSET_WORKER_LINK_POLLSET].next) {
Craig Tiller084c8082015-09-29 11:19:06 -0700216 specific_worker->kicked = 1;
Craig Tillera82950e2015-09-22 12:33:20 -0700217 gpr_cv_signal(&specific_worker->cv);
218 }
219 p->kicked_without_pollers = 1;
Craig Tiller3dedb702015-09-24 10:31:26 -0700220 if (p->is_iocp_worker) {
221 grpc_iocp_kick();
222 }
Craig Tillera82950e2015-09-22 12:33:20 -0700223 } else {
Craig Tiller3dedb702015-09-24 10:31:26 -0700224 if (p->is_iocp_worker) {
Craig Tiller3dedb702015-09-24 10:31:26 -0700225 if (g_active_poller == specific_worker) {
226 grpc_iocp_kick();
227 }
Craig Tiller3dedb702015-09-24 10:31:26 -0700228 } else {
Craig Tiller084c8082015-09-29 11:19:06 -0700229 specific_worker->kicked = 1;
Craig Tiller3dedb702015-09-24 10:31:26 -0700230 gpr_cv_signal(&specific_worker->cv);
231 }
Craig Tillerd1885e02015-07-30 14:32:31 -0700232 }
Craig Tillera82950e2015-09-22 12:33:20 -0700233 } else {
Craig Tiller71a0f9d2015-09-28 17:22:01 -0700234 specific_worker =
235 pop_front_worker(&p->root_worker, GRPC_POLLSET_WORKER_LINK_POLLSET);
Craig Tillera82950e2015-09-22 12:33:20 -0700236 if (specific_worker != NULL) {
Craig Tiller3dedb702015-09-24 10:31:26 -0700237 grpc_pollset_kick(p, specific_worker);
238 } else if (p->is_iocp_worker) {
239 grpc_iocp_kick();
Craig Tillera82950e2015-09-22 12:33:20 -0700240 } else {
241 p->kicked_without_pollers = 1;
Craig Tillerd1885e02015-07-30 14:32:31 -0700242 }
Craig Tillera82950e2015-09-22 12:33:20 -0700243 }
Craig Tillerd1885e02015-07-30 14:32:31 -0700244}
Nicolas "Pixel" Noble21f627a2015-02-04 01:31:14 +0100245
Craig Tiller71a0f9d2015-09-28 17:22:01 -0700246void grpc_kick_poller(void) { grpc_iocp_kick(); }
Craig Tiller26465e82015-09-28 17:08:30 -0700247
Craig Tiller8674cb12015-06-05 07:09:25 -0700248#endif /* GPR_WINSOCK_SOCKET */