blob: 8dc7ed01e2834e958db6036333fdc1383acb8fd3 [file] [log] [blame]
Craig Tiller3bc8ebd2015-06-24 15:41:15 -07001/*
2 *
murgatroid993466c4b2016-01-12 10:26:04 -08003 * Copyright 2015-2016, Google Inc.
Craig Tiller3bc8ebd2015-06-24 15:41:15 -07004 * 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 Tillereb3b12e2015-06-26 14:42:49 -070034#include <string.h>
35
36#include <grpc/support/alloc.h>
Craig Tillerfb433852016-03-29 08:51:07 -070037#include "src/core/lib/client_config/lb_policy_registry.h"
Craig Tiller9533d042016-03-25 17:11:06 -070038#include "src/core/lib/transport/connectivity_state.h"
Craig Tillereb3b12e2015-06-26 14:42:49 -070039
Craig Tillera82950e2015-09-22 12:33:20 -070040typedef struct pending_pick {
Craig Tillereb3b12e2015-06-26 14:42:49 -070041 struct pending_pick *next;
42 grpc_pollset *pollset;
Craig Tillerb5585d42015-11-17 07:18:31 -080043 grpc_connected_subchannel **target;
Craig Tiller33825112015-09-18 07:44:19 -070044 grpc_closure *on_complete;
Craig Tillereb3b12e2015-06-26 14:42:49 -070045} pending_pick;
46
Craig Tillera82950e2015-09-22 12:33:20 -070047typedef struct {
Craig Tillereb3b12e2015-06-26 14:42:49 -070048 /** base policy: must be first */
49 grpc_lb_policy base;
Craig Tillereb3b12e2015-06-26 14:42:49 -070050 /** all our subchannels */
51 grpc_subchannel **subchannels;
52 size_t num_subchannels;
53
Craig Tiller33825112015-09-18 07:44:19 -070054 grpc_closure connectivity_changed;
Craig Tillereb3b12e2015-06-26 14:42:49 -070055
Craig Tiller86c0f8a2015-12-01 20:05:40 -080056 /** the selected channel (a grpc_connected_subchannel) */
57 gpr_atm selected;
Craig Tiller320bee02016-01-06 17:33:45 -080058
59 /** mutex protecting remaining members */
60 gpr_mu mu;
Craig Tillereb3b12e2015-06-26 14:42:49 -070061 /** have we started picking? */
62 int started_picking;
Craig Tillera14215a2015-07-17 17:21:08 -070063 /** are we shut down? */
64 int shutdown;
Craig Tillereb3b12e2015-06-26 14:42:49 -070065 /** which subchannel are we watching? */
66 size_t checking_subchannel;
67 /** what is the connectivity of that channel? */
68 grpc_connectivity_state checking_connectivity;
69 /** list of picks that are waiting on connectivity */
70 pending_pick *pending_picks;
Craig Tillerc7b5f762015-06-27 11:48:42 -070071
72 /** our connectivity state tracker */
73 grpc_connectivity_state_tracker state_tracker;
Craig Tillereb3b12e2015-06-26 14:42:49 -070074} pick_first_lb_policy;
75
Craig Tiller81afdda2016-01-11 17:09:18 -080076#define GET_SELECTED(p) \
Craig Tillerd9d474a2016-01-26 06:50:51 -080077 ((grpc_connected_subchannel *)gpr_atm_acq_load(&(p)->selected))
Craig Tiller86c0f8a2015-12-01 20:05:40 -080078
Craig Tillerfb433852016-03-29 08:51:07 -070079static void pf_destroy(grpc_exec_ctx *exec_ctx, grpc_lb_policy *pol) {
Craig Tillera82950e2015-09-22 12:33:20 -070080 pick_first_lb_policy *p = (pick_first_lb_policy *)pol;
Craig Tiller86c0f8a2015-12-01 20:05:40 -080081 grpc_connected_subchannel *selected = GET_SELECTED(p);
Craig Tillerd7b68e72015-06-28 11:41:09 -070082 size_t i;
Craig Tillera82950e2015-09-22 12:33:20 -070083 GPR_ASSERT(p->pending_picks == NULL);
84 for (i = 0; i < p->num_subchannels; i++) {
85 GRPC_SUBCHANNEL_UNREF(exec_ctx, p->subchannels[i], "pick_first");
86 }
Craig Tiller86c0f8a2015-12-01 20:05:40 -080087 if (selected != NULL) {
88 GRPC_CONNECTED_SUBCHANNEL_UNREF(exec_ctx, selected, "picked_first");
Craig Tiller89a768e2015-10-06 09:55:59 -070089 }
Craig Tillera82950e2015-09-22 12:33:20 -070090 grpc_connectivity_state_destroy(exec_ctx, &p->state_tracker);
91 gpr_free(p->subchannels);
92 gpr_mu_destroy(&p->mu);
93 gpr_free(p);
Craig Tillereb3b12e2015-06-26 14:42:49 -070094}
95
Craig Tillerfb433852016-03-29 08:51:07 -070096static void pf_shutdown(grpc_exec_ctx *exec_ctx, grpc_lb_policy *pol) {
Craig Tillera82950e2015-09-22 12:33:20 -070097 pick_first_lb_policy *p = (pick_first_lb_policy *)pol;
Craig Tillerd2cc4592015-07-01 07:50:47 -070098 pending_pick *pp;
Craig Tiller86c0f8a2015-12-01 20:05:40 -080099 grpc_connected_subchannel *selected;
Craig Tillera82950e2015-09-22 12:33:20 -0700100 gpr_mu_lock(&p->mu);
Craig Tiller86c0f8a2015-12-01 20:05:40 -0800101 selected = GET_SELECTED(p);
Craig Tillera14215a2015-07-17 17:21:08 -0700102 p->shutdown = 1;
Craig Tiller5795da72015-09-17 15:27:13 -0700103 pp = p->pending_picks;
104 p->pending_picks = NULL;
Craig Tillera82950e2015-09-22 12:33:20 -0700105 grpc_connectivity_state_set(exec_ctx, &p->state_tracker,
106 GRPC_CHANNEL_FATAL_FAILURE, "shutdown");
Craig Tillerf036a642015-12-01 17:00:40 -0800107 /* cancel subscription */
Craig Tiller86c0f8a2015-12-01 20:05:40 -0800108 if (selected != NULL) {
Craig Tiller1d881fb2015-12-01 07:39:04 -0800109 grpc_connected_subchannel_notify_on_state_change(
Craig Tiller86c0f8a2015-12-01 20:05:40 -0800110 exec_ctx, selected, NULL, NULL, &p->connectivity_changed);
Craig Tiller48613042015-11-29 14:45:11 -0800111 } else {
Craig Tiller1d881fb2015-12-01 07:39:04 -0800112 grpc_subchannel_notify_on_state_change(
113 exec_ctx, p->subchannels[p->checking_subchannel], NULL, NULL,
114 &p->connectivity_changed);
Craig Tiller48613042015-11-29 14:45:11 -0800115 }
Craig Tillera82950e2015-09-22 12:33:20 -0700116 gpr_mu_unlock(&p->mu);
117 while (pp != NULL) {
118 pending_pick *next = pp->next;
119 *pp->target = NULL;
Craig Tiller69b093b2016-02-25 19:04:07 -0800120 grpc_pollset_set_del_pollset(exec_ctx, p->base.interested_parties,
Craig Tiller1d881fb2015-12-01 07:39:04 -0800121 pp->pollset);
Craig Tiller6c396862016-01-28 13:53:40 -0800122 grpc_exec_ctx_enqueue(exec_ctx, pp->on_complete, true, NULL);
Craig Tillera82950e2015-09-22 12:33:20 -0700123 gpr_free(pp);
124 pp = next;
125 }
Craig Tillereb3b12e2015-06-26 14:42:49 -0700126}
127
Craig Tiller577c9b22015-11-02 14:11:15 -0800128static void pf_cancel_pick(grpc_exec_ctx *exec_ctx, grpc_lb_policy *pol,
Craig Tillerb5585d42015-11-17 07:18:31 -0800129 grpc_connected_subchannel **target) {
Craig Tiller577c9b22015-11-02 14:11:15 -0800130 pick_first_lb_policy *p = (pick_first_lb_policy *)pol;
131 pending_pick *pp;
132 gpr_mu_lock(&p->mu);
133 pp = p->pending_picks;
134 p->pending_picks = NULL;
135 while (pp != NULL) {
136 pending_pick *next = pp->next;
137 if (pp->target == target) {
Craig Tiller69b093b2016-02-25 19:04:07 -0800138 grpc_pollset_set_del_pollset(exec_ctx, p->base.interested_parties,
Craig Tiller1d881fb2015-12-01 07:39:04 -0800139 pp->pollset);
Craig Tiller577c9b22015-11-02 14:11:15 -0800140 *target = NULL;
Craig Tiller6c396862016-01-28 13:53:40 -0800141 grpc_exec_ctx_enqueue(exec_ctx, pp->on_complete, false, NULL);
Craig Tiller577c9b22015-11-02 14:11:15 -0800142 gpr_free(pp);
143 } else {
144 pp->next = p->pending_picks;
145 p->pending_picks = pp;
146 }
147 pp = next;
148 }
149 gpr_mu_unlock(&p->mu);
150}
151
Craig Tillera82950e2015-09-22 12:33:20 -0700152static void start_picking(grpc_exec_ctx *exec_ctx, pick_first_lb_policy *p) {
Craig Tiller48cb07c2015-07-15 16:16:15 -0700153 p->started_picking = 1;
154 p->checking_subchannel = 0;
155 p->checking_connectivity = GRPC_CHANNEL_IDLE;
Craig Tiller48613042015-11-29 14:45:11 -0800156 GRPC_LB_POLICY_WEAK_REF(&p->base, "pick_first_connectivity");
Craig Tillera82950e2015-09-22 12:33:20 -0700157 grpc_subchannel_notify_on_state_change(
158 exec_ctx, p->subchannels[p->checking_subchannel],
Craig Tiller69b093b2016-02-25 19:04:07 -0800159 p->base.interested_parties, &p->checking_connectivity,
Craig Tiller1d881fb2015-12-01 07:39:04 -0800160 &p->connectivity_changed);
Craig Tiller48cb07c2015-07-15 16:16:15 -0700161}
162
Craig Tillerfb433852016-03-29 08:51:07 -0700163static void pf_exit_idle(grpc_exec_ctx *exec_ctx, grpc_lb_policy *pol) {
Craig Tillera82950e2015-09-22 12:33:20 -0700164 pick_first_lb_policy *p = (pick_first_lb_policy *)pol;
165 gpr_mu_lock(&p->mu);
166 if (!p->started_picking) {
167 start_picking(exec_ctx, p);
168 }
169 gpr_mu_unlock(&p->mu);
Craig Tillereb3b12e2015-06-26 14:42:49 -0700170}
171
Craig Tillerfb433852016-03-29 08:51:07 -0700172static int pf_pick(grpc_exec_ctx *exec_ctx, grpc_lb_policy *pol,
173 grpc_pollset *pollset, grpc_metadata_batch *initial_metadata,
174 grpc_connected_subchannel **target,
175 grpc_closure *on_complete) {
Craig Tillera82950e2015-09-22 12:33:20 -0700176 pick_first_lb_policy *p = (pick_first_lb_policy *)pol;
Craig Tiller45724b32015-09-22 10:42:19 -0700177 pending_pick *pp;
Craig Tiller320bee02016-01-06 17:33:45 -0800178
179 /* Check atomically for a selected channel */
Craig Tiller86c0f8a2015-12-01 20:05:40 -0800180 grpc_connected_subchannel *selected = GET_SELECTED(p);
181 if (selected != NULL) {
182 *target = selected;
183 return 1;
184 }
Craig Tiller320bee02016-01-06 17:33:45 -0800185
186 /* No subchannel selected yet, so acquire lock and then attempt again */
Craig Tillera82950e2015-09-22 12:33:20 -0700187 gpr_mu_lock(&p->mu);
Craig Tiller86c0f8a2015-12-01 20:05:40 -0800188 selected = GET_SELECTED(p);
189 if (selected) {
Craig Tillera82950e2015-09-22 12:33:20 -0700190 gpr_mu_unlock(&p->mu);
Craig Tiller86c0f8a2015-12-01 20:05:40 -0800191 *target = selected;
Craig Tiller577c9b22015-11-02 14:11:15 -0800192 return 1;
Craig Tillera82950e2015-09-22 12:33:20 -0700193 } else {
194 if (!p->started_picking) {
195 start_picking(exec_ctx, p);
Craig Tiller45724b32015-09-22 10:42:19 -0700196 }
Craig Tiller69b093b2016-02-25 19:04:07 -0800197 grpc_pollset_set_add_pollset(exec_ctx, p->base.interested_parties, pollset);
Craig Tillera82950e2015-09-22 12:33:20 -0700198 pp = gpr_malloc(sizeof(*pp));
199 pp->next = p->pending_picks;
200 pp->pollset = pollset;
201 pp->target = target;
202 pp->on_complete = on_complete;
203 p->pending_picks = pp;
204 gpr_mu_unlock(&p->mu);
Craig Tiller577c9b22015-11-02 14:11:15 -0800205 return 0;
Craig Tillera82950e2015-09-22 12:33:20 -0700206 }
Craig Tiller45724b32015-09-22 10:42:19 -0700207}
208
Craig Tiller1f41b6b2015-10-09 15:07:02 -0700209static void destroy_subchannels(grpc_exec_ctx *exec_ctx, void *arg,
Craig Tiller6c396862016-01-28 13:53:40 -0800210 bool iomgr_success) {
Craig Tillerb09d84d2015-10-06 09:12:16 -0700211 pick_first_lb_policy *p = arg;
212 size_t i;
Craig Tillerb09d84d2015-10-06 09:12:16 -0700213 size_t num_subchannels = p->num_subchannels;
214 grpc_subchannel **subchannels;
Craig Tillerb09d84d2015-10-06 09:12:16 -0700215
216 gpr_mu_lock(&p->mu);
217 subchannels = p->subchannels;
Craig Tillerb09d84d2015-10-06 09:12:16 -0700218 p->num_subchannels = 0;
219 p->subchannels = NULL;
220 gpr_mu_unlock(&p->mu);
Craig Tiller48613042015-11-29 14:45:11 -0800221 GRPC_LB_POLICY_WEAK_UNREF(exec_ctx, &p->base, "destroy_subchannels");
Craig Tillerb09d84d2015-10-06 09:12:16 -0700222
223 for (i = 0; i < num_subchannels; i++) {
Craig Tillerb09d84d2015-10-06 09:12:16 -0700224 GRPC_SUBCHANNEL_UNREF(exec_ctx, subchannels[i], "pick_first");
225 }
226
227 gpr_free(subchannels);
228}
229
Craig Tillera82950e2015-09-22 12:33:20 -0700230static void pf_connectivity_changed(grpc_exec_ctx *exec_ctx, void *arg,
Craig Tiller6c396862016-01-28 13:53:40 -0800231 bool iomgr_success) {
Craig Tillereb3b12e2015-06-26 14:42:49 -0700232 pick_first_lb_policy *p = arg;
Craig Tillerb5585d42015-11-17 07:18:31 -0800233 grpc_subchannel *selected_subchannel;
Craig Tillereb3b12e2015-06-26 14:42:49 -0700234 pending_pick *pp;
Craig Tiller86c0f8a2015-12-01 20:05:40 -0800235 grpc_connected_subchannel *selected;
Craig Tillereb3b12e2015-06-26 14:42:49 -0700236
Craig Tillera82950e2015-09-22 12:33:20 -0700237 gpr_mu_lock(&p->mu);
Craig Tiller1ada6ad2015-07-16 16:19:14 -0700238
Craig Tiller86c0f8a2015-12-01 20:05:40 -0800239 selected = GET_SELECTED(p);
240
Craig Tillera82950e2015-09-22 12:33:20 -0700241 if (p->shutdown) {
242 gpr_mu_unlock(&p->mu);
Craig Tiller48613042015-11-29 14:45:11 -0800243 GRPC_LB_POLICY_WEAK_UNREF(exec_ctx, &p->base, "pick_first_connectivity");
Craig Tillera82950e2015-09-22 12:33:20 -0700244 return;
Craig Tiller86c0f8a2015-12-01 20:05:40 -0800245 } else if (selected != NULL) {
Craig Tillercb2609f2015-11-24 17:19:19 -0800246 if (p->checking_connectivity == GRPC_CHANNEL_TRANSIENT_FAILURE) {
247 /* if the selected channel goes bad, we're done */
248 p->checking_connectivity = GRPC_CHANNEL_FATAL_FAILURE;
249 }
Craig Tillera82950e2015-09-22 12:33:20 -0700250 grpc_connectivity_state_set(exec_ctx, &p->state_tracker,
251 p->checking_connectivity, "selected_changed");
252 if (p->checking_connectivity != GRPC_CHANNEL_FATAL_FAILURE) {
Craig Tillerab33b482015-11-21 08:11:04 -0800253 grpc_connected_subchannel_notify_on_state_change(
Craig Tiller69b093b2016-02-25 19:04:07 -0800254 exec_ctx, selected, p->base.interested_parties,
Craig Tillera6bebf42015-12-01 17:02:35 -0800255 &p->checking_connectivity, &p->connectivity_changed);
Craig Tillera82950e2015-09-22 12:33:20 -0700256 } else {
Craig Tiller48613042015-11-29 14:45:11 -0800257 GRPC_LB_POLICY_WEAK_UNREF(exec_ctx, &p->base, "pick_first_connectivity");
Craig Tiller1ada6ad2015-07-16 16:19:14 -0700258 }
Craig Tillera82950e2015-09-22 12:33:20 -0700259 } else {
260 loop:
261 switch (p->checking_connectivity) {
262 case GRPC_CHANNEL_READY:
263 grpc_connectivity_state_set(exec_ctx, &p->state_tracker,
264 GRPC_CHANNEL_READY, "connecting_ready");
Craig Tillerb5585d42015-11-17 07:18:31 -0800265 selected_subchannel = p->subchannels[p->checking_subchannel];
Craig Tiller81afdda2016-01-11 17:09:18 -0800266 selected =
267 grpc_subchannel_get_connected_subchannel(selected_subchannel);
Craig Tiller86c0f8a2015-12-01 20:05:40 -0800268 GPR_ASSERT(selected != NULL);
Craig Tiller86c0f8a2015-12-01 20:05:40 -0800269 GRPC_CONNECTED_SUBCHANNEL_REF(selected, "picked_first");
Craig Tillerb09d84d2015-10-06 09:12:16 -0700270 /* drop the pick list: we are connected now */
Craig Tiller48613042015-11-29 14:45:11 -0800271 GRPC_LB_POLICY_WEAK_REF(&p->base, "destroy_subchannels");
Craig Tillerd9d474a2016-01-26 06:50:51 -0800272 gpr_atm_rel_store(&p->selected, (gpr_atm)selected);
Craig Tiller6c396862016-01-28 13:53:40 -0800273 grpc_exec_ctx_enqueue(
274 exec_ctx, grpc_closure_create(destroy_subchannels, p), true, NULL);
Craig Tillerb09d84d2015-10-06 09:12:16 -0700275 /* update any calls that were waiting for a pick */
Craig Tillera82950e2015-09-22 12:33:20 -0700276 while ((pp = p->pending_picks)) {
277 p->pending_picks = pp->next;
Craig Tiller86c0f8a2015-12-01 20:05:40 -0800278 *pp->target = selected;
Craig Tiller69b093b2016-02-25 19:04:07 -0800279 grpc_pollset_set_del_pollset(exec_ctx, p->base.interested_parties,
Craig Tiller1d881fb2015-12-01 07:39:04 -0800280 pp->pollset);
Craig Tiller6c396862016-01-28 13:53:40 -0800281 grpc_exec_ctx_enqueue(exec_ctx, pp->on_complete, true, NULL);
Craig Tillera82950e2015-09-22 12:33:20 -0700282 gpr_free(pp);
283 }
Craig Tillerab33b482015-11-21 08:11:04 -0800284 grpc_connected_subchannel_notify_on_state_change(
Craig Tiller69b093b2016-02-25 19:04:07 -0800285 exec_ctx, selected, p->base.interested_parties,
Craig Tillera6bebf42015-12-01 17:02:35 -0800286 &p->checking_connectivity, &p->connectivity_changed);
Craig Tillera82950e2015-09-22 12:33:20 -0700287 break;
288 case GRPC_CHANNEL_TRANSIENT_FAILURE:
289 grpc_connectivity_state_set(exec_ctx, &p->state_tracker,
290 GRPC_CHANNEL_TRANSIENT_FAILURE,
291 "connecting_transient_failure");
Craig Tillera82950e2015-09-22 12:33:20 -0700292 p->checking_subchannel =
293 (p->checking_subchannel + 1) % p->num_subchannels;
294 p->checking_connectivity = grpc_subchannel_check_connectivity(
295 p->subchannels[p->checking_subchannel]);
Craig Tillera82950e2015-09-22 12:33:20 -0700296 if (p->checking_connectivity == GRPC_CHANNEL_TRANSIENT_FAILURE) {
297 grpc_subchannel_notify_on_state_change(
298 exec_ctx, p->subchannels[p->checking_subchannel],
Craig Tiller69b093b2016-02-25 19:04:07 -0800299 p->base.interested_parties, &p->checking_connectivity,
Craig Tiller1d881fb2015-12-01 07:39:04 -0800300 &p->connectivity_changed);
Craig Tillera82950e2015-09-22 12:33:20 -0700301 } else {
302 goto loop;
303 }
304 break;
305 case GRPC_CHANNEL_CONNECTING:
306 case GRPC_CHANNEL_IDLE:
307 grpc_connectivity_state_set(exec_ctx, &p->state_tracker,
308 GRPC_CHANNEL_CONNECTING,
309 "connecting_changed");
310 grpc_subchannel_notify_on_state_change(
311 exec_ctx, p->subchannels[p->checking_subchannel],
Craig Tiller69b093b2016-02-25 19:04:07 -0800312 p->base.interested_parties, &p->checking_connectivity,
Craig Tiller1d881fb2015-12-01 07:39:04 -0800313 &p->connectivity_changed);
Craig Tillera82950e2015-09-22 12:33:20 -0700314 break;
315 case GRPC_CHANNEL_FATAL_FAILURE:
Craig Tillera82950e2015-09-22 12:33:20 -0700316 p->num_subchannels--;
Craig Tiller86c99582015-11-25 15:22:26 -0800317 GPR_SWAP(grpc_subchannel *, p->subchannels[p->checking_subchannel],
318 p->subchannels[p->num_subchannels]);
Craig Tillera82950e2015-09-22 12:33:20 -0700319 GRPC_SUBCHANNEL_UNREF(exec_ctx, p->subchannels[p->num_subchannels],
320 "pick_first");
321 if (p->num_subchannels == 0) {
322 grpc_connectivity_state_set(exec_ctx, &p->state_tracker,
323 GRPC_CHANNEL_FATAL_FAILURE,
324 "no_more_channels");
325 while ((pp = p->pending_picks)) {
326 p->pending_picks = pp->next;
327 *pp->target = NULL;
Craig Tiller6c396862016-01-28 13:53:40 -0800328 grpc_exec_ctx_enqueue(exec_ctx, pp->on_complete, true, NULL);
Craig Tillera82950e2015-09-22 12:33:20 -0700329 gpr_free(pp);
330 }
Craig Tiller1d881fb2015-12-01 07:39:04 -0800331 GRPC_LB_POLICY_WEAK_UNREF(exec_ctx, &p->base,
332 "pick_first_connectivity");
Craig Tillera82950e2015-09-22 12:33:20 -0700333 } else {
334 grpc_connectivity_state_set(exec_ctx, &p->state_tracker,
335 GRPC_CHANNEL_TRANSIENT_FAILURE,
336 "subchannel_failed");
337 p->checking_subchannel %= p->num_subchannels;
338 p->checking_connectivity = grpc_subchannel_check_connectivity(
339 p->subchannels[p->checking_subchannel]);
Craig Tillera82950e2015-09-22 12:33:20 -0700340 goto loop;
341 }
Craig Tiller1ada6ad2015-07-16 16:19:14 -0700342 }
Craig Tillera82950e2015-09-22 12:33:20 -0700343 }
Craig Tiller1ada6ad2015-07-16 16:19:14 -0700344
Craig Tillera82950e2015-09-22 12:33:20 -0700345 gpr_mu_unlock(&p->mu);
Craig Tillereb3b12e2015-06-26 14:42:49 -0700346}
347
Craig Tillera82950e2015-09-22 12:33:20 -0700348static grpc_connectivity_state pf_check_connectivity(grpc_exec_ctx *exec_ctx,
349 grpc_lb_policy *pol) {
350 pick_first_lb_policy *p = (pick_first_lb_policy *)pol;
Craig Tillerc7b5f762015-06-27 11:48:42 -0700351 grpc_connectivity_state st;
Craig Tillera82950e2015-09-22 12:33:20 -0700352 gpr_mu_lock(&p->mu);
353 st = grpc_connectivity_state_check(&p->state_tracker);
354 gpr_mu_unlock(&p->mu);
Craig Tillerc7b5f762015-06-27 11:48:42 -0700355 return st;
356}
357
Craig Tillerfb433852016-03-29 08:51:07 -0700358static void pf_notify_on_state_change(grpc_exec_ctx *exec_ctx,
359 grpc_lb_policy *pol,
360 grpc_connectivity_state *current,
361 grpc_closure *notify) {
Craig Tillera82950e2015-09-22 12:33:20 -0700362 pick_first_lb_policy *p = (pick_first_lb_policy *)pol;
363 gpr_mu_lock(&p->mu);
364 grpc_connectivity_state_notify_on_state_change(exec_ctx, &p->state_tracker,
365 current, notify);
366 gpr_mu_unlock(&p->mu);
Craig Tillerc7b5f762015-06-27 11:48:42 -0700367}
368
Craig Tillerfb433852016-03-29 08:51:07 -0700369static void pf_ping_one(grpc_exec_ctx *exec_ctx, grpc_lb_policy *pol,
370 grpc_closure *closure) {
Craig Tiller28bf8912015-12-07 16:07:04 -0800371 pick_first_lb_policy *p = (pick_first_lb_policy *)pol;
Craig Tiller093193e2016-01-07 07:14:44 -0800372 grpc_connected_subchannel *selected = GET_SELECTED(p);
373 if (selected) {
374 grpc_connected_subchannel_ping(exec_ctx, selected, closure);
Craig Tiller28bf8912015-12-07 16:07:04 -0800375 } else {
Craig Tiller6c396862016-01-28 13:53:40 -0800376 grpc_exec_ctx_enqueue(exec_ctx, closure, false, NULL);
Craig Tiller28bf8912015-12-07 16:07:04 -0800377 }
Craig Tiller28bf8912015-12-07 16:07:04 -0800378}
379
Craig Tillereb3b12e2015-06-26 14:42:49 -0700380static const grpc_lb_policy_vtable pick_first_lb_policy_vtable = {
Craig Tillerf40df232016-03-25 13:38:14 -0700381 pf_destroy,
382 pf_shutdown,
383 pf_pick,
384 pf_cancel_pick,
385 pf_ping_one,
386 pf_exit_idle,
387 pf_check_connectivity,
388 pf_notify_on_state_change};
Craig Tillereb3b12e2015-06-26 14:42:49 -0700389
Craig Tillera82950e2015-09-22 12:33:20 -0700390static void pick_first_factory_ref(grpc_lb_policy_factory *factory) {}
David Garcia Quintas5c4543d2015-09-03 15:49:56 -0700391
Craig Tillera82950e2015-09-22 12:33:20 -0700392static void pick_first_factory_unref(grpc_lb_policy_factory *factory) {}
David Garcia Quintas5c4543d2015-09-03 15:49:56 -0700393
Craig Tillera82950e2015-09-22 12:33:20 -0700394static grpc_lb_policy *create_pick_first(grpc_lb_policy_factory *factory,
395 grpc_lb_policy_args *args) {
Craig Tiller1e55bd42016-03-11 09:47:43 -0800396 if (args->num_subchannels == 0) return NULL;
Craig Tillera82950e2015-09-22 12:33:20 -0700397 pick_first_lb_policy *p = gpr_malloc(sizeof(*p));
Craig Tillera82950e2015-09-22 12:33:20 -0700398 memset(p, 0, sizeof(*p));
399 grpc_lb_policy_init(&p->base, &pick_first_lb_policy_vtable);
400 p->subchannels =
401 gpr_malloc(sizeof(grpc_subchannel *) * args->num_subchannels);
David Garcia Quintasc7705c72015-09-09 17:21:11 -0700402 p->num_subchannels = args->num_subchannels;
Craig Tillera82950e2015-09-22 12:33:20 -0700403 grpc_connectivity_state_init(&p->state_tracker, GRPC_CHANNEL_IDLE,
404 "pick_first");
405 memcpy(p->subchannels, args->subchannels,
406 sizeof(grpc_subchannel *) * args->num_subchannels);
407 grpc_closure_init(&p->connectivity_changed, pf_connectivity_changed, p);
408 gpr_mu_init(&p->mu);
Craig Tillereb3b12e2015-06-26 14:42:49 -0700409 return &p->base;
410}
David Garcia Quintas5c4543d2015-09-03 15:49:56 -0700411
412static const grpc_lb_policy_factory_vtable pick_first_factory_vtable = {
Craig Tillera82950e2015-09-22 12:33:20 -0700413 pick_first_factory_ref, pick_first_factory_unref, create_pick_first,
414 "pick_first"};
David Garcia Quintas5c4543d2015-09-03 15:49:56 -0700415
416static grpc_lb_policy_factory pick_first_lb_policy_factory = {
Craig Tillera82950e2015-09-22 12:33:20 -0700417 &pick_first_factory_vtable};
David Garcia Quintas5c4543d2015-09-03 15:49:56 -0700418
Craig Tillerfb433852016-03-29 08:51:07 -0700419static grpc_lb_policy_factory *pick_first_lb_factory_create() {
David Garcia Quintas5c4543d2015-09-03 15:49:56 -0700420 return &pick_first_lb_policy_factory;
421}
Craig Tillerfb433852016-03-29 08:51:07 -0700422
423/* Plugin registration */
424
425void grpc_lb_policy_pick_first_init() {
Craig Tiller3113ef42016-03-29 09:03:14 -0700426 grpc_register_lb_policy(pick_first_lb_factory_create());
Craig Tillerfb433852016-03-29 08:51:07 -0700427}
428
429void grpc_lb_policy_pick_first_shutdown() {}