blob: 53c14f8f0a0aafa363ae88b97247f99d5e34b49c [file] [log] [blame]
Craig Tiller3bc8ebd2015-06-24 15:41:15 -07001/*
2 *
Craig Tiller6169d5f2016-03-31 07:46:18 -07003 * Copyright 2015, 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 Tiller8c0d96f2016-03-11 14:27:52 -080043 uint32_t initial_metadata_flags;
Craig Tillerb5585d42015-11-17 07:18:31 -080044 grpc_connected_subchannel **target;
Craig Tiller33825112015-09-18 07:44:19 -070045 grpc_closure *on_complete;
Craig Tillereb3b12e2015-06-26 14:42:49 -070046} pending_pick;
47
Craig Tillera82950e2015-09-22 12:33:20 -070048typedef struct {
Craig Tillereb3b12e2015-06-26 14:42:49 -070049 /** base policy: must be first */
50 grpc_lb_policy base;
Craig Tillereb3b12e2015-06-26 14:42:49 -070051 /** all our subchannels */
52 grpc_subchannel **subchannels;
53 size_t num_subchannels;
54
Craig Tiller33825112015-09-18 07:44:19 -070055 grpc_closure connectivity_changed;
Craig Tillereb3b12e2015-06-26 14:42:49 -070056
Craig Tiller86c0f8a2015-12-01 20:05:40 -080057 /** the selected channel (a grpc_connected_subchannel) */
58 gpr_atm selected;
Craig Tiller320bee02016-01-06 17:33:45 -080059
60 /** mutex protecting remaining members */
61 gpr_mu mu;
Craig Tillereb3b12e2015-06-26 14:42:49 -070062 /** have we started picking? */
63 int started_picking;
Craig Tillera14215a2015-07-17 17:21:08 -070064 /** are we shut down? */
65 int shutdown;
Craig Tillereb3b12e2015-06-26 14:42:49 -070066 /** which subchannel are we watching? */
67 size_t checking_subchannel;
68 /** what is the connectivity of that channel? */
69 grpc_connectivity_state checking_connectivity;
70 /** list of picks that are waiting on connectivity */
71 pending_pick *pending_picks;
Craig Tillerc7b5f762015-06-27 11:48:42 -070072
73 /** our connectivity state tracker */
74 grpc_connectivity_state_tracker state_tracker;
Craig Tillereb3b12e2015-06-26 14:42:49 -070075} pick_first_lb_policy;
76
Craig Tiller81afdda2016-01-11 17:09:18 -080077#define GET_SELECTED(p) \
Craig Tillerd9d474a2016-01-26 06:50:51 -080078 ((grpc_connected_subchannel *)gpr_atm_acq_load(&(p)->selected))
Craig Tiller86c0f8a2015-12-01 20:05:40 -080079
Craig Tillerfb433852016-03-29 08:51:07 -070080static void pf_destroy(grpc_exec_ctx *exec_ctx, grpc_lb_policy *pol) {
Craig Tillera82950e2015-09-22 12:33:20 -070081 pick_first_lb_policy *p = (pick_first_lb_policy *)pol;
Craig Tiller86c0f8a2015-12-01 20:05:40 -080082 grpc_connected_subchannel *selected = GET_SELECTED(p);
Craig Tillerd7b68e72015-06-28 11:41:09 -070083 size_t i;
Craig Tillera82950e2015-09-22 12:33:20 -070084 GPR_ASSERT(p->pending_picks == NULL);
85 for (i = 0; i < p->num_subchannels; i++) {
86 GRPC_SUBCHANNEL_UNREF(exec_ctx, p->subchannels[i], "pick_first");
87 }
Craig Tiller86c0f8a2015-12-01 20:05:40 -080088 if (selected != NULL) {
89 GRPC_CONNECTED_SUBCHANNEL_UNREF(exec_ctx, selected, "picked_first");
Craig Tiller89a768e2015-10-06 09:55:59 -070090 }
Craig Tillera82950e2015-09-22 12:33:20 -070091 grpc_connectivity_state_destroy(exec_ctx, &p->state_tracker);
92 gpr_free(p->subchannels);
93 gpr_mu_destroy(&p->mu);
94 gpr_free(p);
Craig Tillereb3b12e2015-06-26 14:42:49 -070095}
96
Craig Tillerfb433852016-03-29 08:51:07 -070097static void pf_shutdown(grpc_exec_ctx *exec_ctx, grpc_lb_policy *pol) {
Craig Tillera82950e2015-09-22 12:33:20 -070098 pick_first_lb_policy *p = (pick_first_lb_policy *)pol;
Craig Tillerd2cc4592015-07-01 07:50:47 -070099 pending_pick *pp;
Craig Tiller86c0f8a2015-12-01 20:05:40 -0800100 grpc_connected_subchannel *selected;
Craig Tillera82950e2015-09-22 12:33:20 -0700101 gpr_mu_lock(&p->mu);
Craig Tiller86c0f8a2015-12-01 20:05:40 -0800102 selected = GET_SELECTED(p);
Craig Tillera14215a2015-07-17 17:21:08 -0700103 p->shutdown = 1;
Craig Tiller5795da72015-09-17 15:27:13 -0700104 pp = p->pending_picks;
105 p->pending_picks = NULL;
Craig Tillera82950e2015-09-22 12:33:20 -0700106 grpc_connectivity_state_set(exec_ctx, &p->state_tracker,
107 GRPC_CHANNEL_FATAL_FAILURE, "shutdown");
Craig Tillerf036a642015-12-01 17:00:40 -0800108 /* cancel subscription */
Craig Tiller86c0f8a2015-12-01 20:05:40 -0800109 if (selected != NULL) {
Craig Tiller1d881fb2015-12-01 07:39:04 -0800110 grpc_connected_subchannel_notify_on_state_change(
Craig Tiller86c0f8a2015-12-01 20:05:40 -0800111 exec_ctx, selected, NULL, NULL, &p->connectivity_changed);
Craig Tiller48613042015-11-29 14:45:11 -0800112 } else {
Craig Tiller1d881fb2015-12-01 07:39:04 -0800113 grpc_subchannel_notify_on_state_change(
114 exec_ctx, p->subchannels[p->checking_subchannel], NULL, NULL,
115 &p->connectivity_changed);
Craig Tiller48613042015-11-29 14:45:11 -0800116 }
Craig Tillera82950e2015-09-22 12:33:20 -0700117 gpr_mu_unlock(&p->mu);
118 while (pp != NULL) {
119 pending_pick *next = pp->next;
120 *pp->target = NULL;
Craig Tiller69b093b2016-02-25 19:04:07 -0800121 grpc_pollset_set_del_pollset(exec_ctx, p->base.interested_parties,
Craig Tiller1d881fb2015-12-01 07:39:04 -0800122 pp->pollset);
Craig Tiller6c396862016-01-28 13:53:40 -0800123 grpc_exec_ctx_enqueue(exec_ctx, pp->on_complete, true, NULL);
Craig Tillera82950e2015-09-22 12:33:20 -0700124 gpr_free(pp);
125 pp = next;
126 }
Craig Tillereb3b12e2015-06-26 14:42:49 -0700127}
128
Craig Tiller577c9b22015-11-02 14:11:15 -0800129static void pf_cancel_pick(grpc_exec_ctx *exec_ctx, grpc_lb_policy *pol,
Craig Tillerb5585d42015-11-17 07:18:31 -0800130 grpc_connected_subchannel **target) {
Craig Tiller577c9b22015-11-02 14:11:15 -0800131 pick_first_lb_policy *p = (pick_first_lb_policy *)pol;
132 pending_pick *pp;
133 gpr_mu_lock(&p->mu);
134 pp = p->pending_picks;
135 p->pending_picks = NULL;
136 while (pp != NULL) {
137 pending_pick *next = pp->next;
138 if (pp->target == target) {
Craig Tiller69b093b2016-02-25 19:04:07 -0800139 grpc_pollset_set_del_pollset(exec_ctx, p->base.interested_parties,
Craig Tiller1d881fb2015-12-01 07:39:04 -0800140 pp->pollset);
Craig Tiller577c9b22015-11-02 14:11:15 -0800141 *target = NULL;
Craig Tiller6c396862016-01-28 13:53:40 -0800142 grpc_exec_ctx_enqueue(exec_ctx, pp->on_complete, false, NULL);
Craig Tiller577c9b22015-11-02 14:11:15 -0800143 gpr_free(pp);
144 } else {
145 pp->next = p->pending_picks;
146 p->pending_picks = pp;
147 }
148 pp = next;
149 }
150 gpr_mu_unlock(&p->mu);
151}
152
Craig Tiller8c0d96f2016-03-11 14:27:52 -0800153static void pf_cancel_picks(grpc_exec_ctx *exec_ctx, grpc_lb_policy *pol,
154 uint32_t initial_metadata_flags_mask,
155 uint32_t initial_metadata_flags_eq) {
156 pick_first_lb_policy *p = (pick_first_lb_policy *)pol;
157 pending_pick *pp;
158 gpr_mu_lock(&p->mu);
159 pp = p->pending_picks;
160 p->pending_picks = NULL;
161 while (pp != NULL) {
162 pending_pick *next = pp->next;
163 if ((pp->initial_metadata_flags & initial_metadata_flags_mask) ==
164 initial_metadata_flags_eq) {
165 grpc_pollset_set_del_pollset(exec_ctx, p->base.interested_parties,
166 pp->pollset);
167 *pp->target = NULL;
168 grpc_exec_ctx_enqueue(exec_ctx, pp->on_complete, false, NULL);
169 gpr_free(pp);
170 } else {
171 pp->next = p->pending_picks;
172 p->pending_picks = pp;
173 }
174 pp = next;
175 }
176 gpr_mu_unlock(&p->mu);
177}
178
Craig Tillera82950e2015-09-22 12:33:20 -0700179static void start_picking(grpc_exec_ctx *exec_ctx, pick_first_lb_policy *p) {
Craig Tiller48cb07c2015-07-15 16:16:15 -0700180 p->started_picking = 1;
181 p->checking_subchannel = 0;
182 p->checking_connectivity = GRPC_CHANNEL_IDLE;
Craig Tiller48613042015-11-29 14:45:11 -0800183 GRPC_LB_POLICY_WEAK_REF(&p->base, "pick_first_connectivity");
Craig Tillera82950e2015-09-22 12:33:20 -0700184 grpc_subchannel_notify_on_state_change(
185 exec_ctx, p->subchannels[p->checking_subchannel],
Craig Tiller69b093b2016-02-25 19:04:07 -0800186 p->base.interested_parties, &p->checking_connectivity,
Craig Tiller1d881fb2015-12-01 07:39:04 -0800187 &p->connectivity_changed);
Craig Tiller48cb07c2015-07-15 16:16:15 -0700188}
189
Craig Tillerfb433852016-03-29 08:51:07 -0700190static void pf_exit_idle(grpc_exec_ctx *exec_ctx, grpc_lb_policy *pol) {
Craig Tillera82950e2015-09-22 12:33:20 -0700191 pick_first_lb_policy *p = (pick_first_lb_policy *)pol;
192 gpr_mu_lock(&p->mu);
193 if (!p->started_picking) {
194 start_picking(exec_ctx, p);
195 }
196 gpr_mu_unlock(&p->mu);
Craig Tillereb3b12e2015-06-26 14:42:49 -0700197}
198
Craig Tillerfb433852016-03-29 08:51:07 -0700199static int pf_pick(grpc_exec_ctx *exec_ctx, grpc_lb_policy *pol,
200 grpc_pollset *pollset, grpc_metadata_batch *initial_metadata,
Craig Tiller41dac842016-03-31 14:01:53 -0700201 uint32_t initial_metadata_flags,
Craig Tillerfb433852016-03-29 08:51:07 -0700202 grpc_connected_subchannel **target,
203 grpc_closure *on_complete) {
Craig Tillera82950e2015-09-22 12:33:20 -0700204 pick_first_lb_policy *p = (pick_first_lb_policy *)pol;
Craig Tiller45724b32015-09-22 10:42:19 -0700205 pending_pick *pp;
Craig Tiller320bee02016-01-06 17:33:45 -0800206
207 /* Check atomically for a selected channel */
Craig Tiller86c0f8a2015-12-01 20:05:40 -0800208 grpc_connected_subchannel *selected = GET_SELECTED(p);
209 if (selected != NULL) {
210 *target = selected;
211 return 1;
212 }
Craig Tiller320bee02016-01-06 17:33:45 -0800213
214 /* No subchannel selected yet, so acquire lock and then attempt again */
Craig Tillera82950e2015-09-22 12:33:20 -0700215 gpr_mu_lock(&p->mu);
Craig Tiller86c0f8a2015-12-01 20:05:40 -0800216 selected = GET_SELECTED(p);
217 if (selected) {
Craig Tillera82950e2015-09-22 12:33:20 -0700218 gpr_mu_unlock(&p->mu);
Craig Tiller86c0f8a2015-12-01 20:05:40 -0800219 *target = selected;
Craig Tiller577c9b22015-11-02 14:11:15 -0800220 return 1;
Craig Tillera82950e2015-09-22 12:33:20 -0700221 } else {
222 if (!p->started_picking) {
223 start_picking(exec_ctx, p);
Craig Tiller45724b32015-09-22 10:42:19 -0700224 }
Craig Tiller69b093b2016-02-25 19:04:07 -0800225 grpc_pollset_set_add_pollset(exec_ctx, p->base.interested_parties, pollset);
Craig Tillera82950e2015-09-22 12:33:20 -0700226 pp = gpr_malloc(sizeof(*pp));
227 pp->next = p->pending_picks;
228 pp->pollset = pollset;
229 pp->target = target;
Craig Tiller8c0d96f2016-03-11 14:27:52 -0800230 pp->initial_metadata_flags = initial_metadata_flags;
Craig Tillera82950e2015-09-22 12:33:20 -0700231 pp->on_complete = on_complete;
232 p->pending_picks = pp;
233 gpr_mu_unlock(&p->mu);
Craig Tiller577c9b22015-11-02 14:11:15 -0800234 return 0;
Craig Tillera82950e2015-09-22 12:33:20 -0700235 }
Craig Tiller45724b32015-09-22 10:42:19 -0700236}
237
Craig Tiller1f41b6b2015-10-09 15:07:02 -0700238static void destroy_subchannels(grpc_exec_ctx *exec_ctx, void *arg,
Craig Tiller6c396862016-01-28 13:53:40 -0800239 bool iomgr_success) {
Craig Tillerb09d84d2015-10-06 09:12:16 -0700240 pick_first_lb_policy *p = arg;
241 size_t i;
Craig Tillerb09d84d2015-10-06 09:12:16 -0700242 size_t num_subchannels = p->num_subchannels;
243 grpc_subchannel **subchannels;
Craig Tillerb09d84d2015-10-06 09:12:16 -0700244
245 gpr_mu_lock(&p->mu);
246 subchannels = p->subchannels;
Craig Tillerb09d84d2015-10-06 09:12:16 -0700247 p->num_subchannels = 0;
248 p->subchannels = NULL;
249 gpr_mu_unlock(&p->mu);
Craig Tiller48613042015-11-29 14:45:11 -0800250 GRPC_LB_POLICY_WEAK_UNREF(exec_ctx, &p->base, "destroy_subchannels");
Craig Tillerb09d84d2015-10-06 09:12:16 -0700251
252 for (i = 0; i < num_subchannels; i++) {
Craig Tillerb09d84d2015-10-06 09:12:16 -0700253 GRPC_SUBCHANNEL_UNREF(exec_ctx, subchannels[i], "pick_first");
254 }
255
256 gpr_free(subchannels);
257}
258
Craig Tillera82950e2015-09-22 12:33:20 -0700259static void pf_connectivity_changed(grpc_exec_ctx *exec_ctx, void *arg,
Craig Tiller6c396862016-01-28 13:53:40 -0800260 bool iomgr_success) {
Craig Tillereb3b12e2015-06-26 14:42:49 -0700261 pick_first_lb_policy *p = arg;
Craig Tillerb5585d42015-11-17 07:18:31 -0800262 grpc_subchannel *selected_subchannel;
Craig Tillereb3b12e2015-06-26 14:42:49 -0700263 pending_pick *pp;
Craig Tiller86c0f8a2015-12-01 20:05:40 -0800264 grpc_connected_subchannel *selected;
Craig Tillereb3b12e2015-06-26 14:42:49 -0700265
Craig Tillera82950e2015-09-22 12:33:20 -0700266 gpr_mu_lock(&p->mu);
Craig Tiller1ada6ad2015-07-16 16:19:14 -0700267
Craig Tiller86c0f8a2015-12-01 20:05:40 -0800268 selected = GET_SELECTED(p);
269
Craig Tillera82950e2015-09-22 12:33:20 -0700270 if (p->shutdown) {
271 gpr_mu_unlock(&p->mu);
Craig Tiller48613042015-11-29 14:45:11 -0800272 GRPC_LB_POLICY_WEAK_UNREF(exec_ctx, &p->base, "pick_first_connectivity");
Craig Tillera82950e2015-09-22 12:33:20 -0700273 return;
Craig Tiller86c0f8a2015-12-01 20:05:40 -0800274 } else if (selected != NULL) {
Craig Tillercb2609f2015-11-24 17:19:19 -0800275 if (p->checking_connectivity == GRPC_CHANNEL_TRANSIENT_FAILURE) {
276 /* if the selected channel goes bad, we're done */
277 p->checking_connectivity = GRPC_CHANNEL_FATAL_FAILURE;
278 }
Craig Tillera82950e2015-09-22 12:33:20 -0700279 grpc_connectivity_state_set(exec_ctx, &p->state_tracker,
280 p->checking_connectivity, "selected_changed");
281 if (p->checking_connectivity != GRPC_CHANNEL_FATAL_FAILURE) {
Craig Tillerab33b482015-11-21 08:11:04 -0800282 grpc_connected_subchannel_notify_on_state_change(
Craig Tiller69b093b2016-02-25 19:04:07 -0800283 exec_ctx, selected, p->base.interested_parties,
Craig Tillera6bebf42015-12-01 17:02:35 -0800284 &p->checking_connectivity, &p->connectivity_changed);
Craig Tillera82950e2015-09-22 12:33:20 -0700285 } else {
Craig Tiller48613042015-11-29 14:45:11 -0800286 GRPC_LB_POLICY_WEAK_UNREF(exec_ctx, &p->base, "pick_first_connectivity");
Craig Tiller1ada6ad2015-07-16 16:19:14 -0700287 }
Craig Tillera82950e2015-09-22 12:33:20 -0700288 } else {
289 loop:
290 switch (p->checking_connectivity) {
291 case GRPC_CHANNEL_READY:
292 grpc_connectivity_state_set(exec_ctx, &p->state_tracker,
293 GRPC_CHANNEL_READY, "connecting_ready");
Craig Tillerb5585d42015-11-17 07:18:31 -0800294 selected_subchannel = p->subchannels[p->checking_subchannel];
Craig Tiller81afdda2016-01-11 17:09:18 -0800295 selected =
296 grpc_subchannel_get_connected_subchannel(selected_subchannel);
Craig Tiller86c0f8a2015-12-01 20:05:40 -0800297 GPR_ASSERT(selected != NULL);
Craig Tiller86c0f8a2015-12-01 20:05:40 -0800298 GRPC_CONNECTED_SUBCHANNEL_REF(selected, "picked_first");
Craig Tillerb09d84d2015-10-06 09:12:16 -0700299 /* drop the pick list: we are connected now */
Craig Tiller48613042015-11-29 14:45:11 -0800300 GRPC_LB_POLICY_WEAK_REF(&p->base, "destroy_subchannels");
Craig Tillerd9d474a2016-01-26 06:50:51 -0800301 gpr_atm_rel_store(&p->selected, (gpr_atm)selected);
Craig Tiller6c396862016-01-28 13:53:40 -0800302 grpc_exec_ctx_enqueue(
303 exec_ctx, grpc_closure_create(destroy_subchannels, p), true, NULL);
Craig Tillerb09d84d2015-10-06 09:12:16 -0700304 /* update any calls that were waiting for a pick */
Craig Tillera82950e2015-09-22 12:33:20 -0700305 while ((pp = p->pending_picks)) {
306 p->pending_picks = pp->next;
Craig Tiller86c0f8a2015-12-01 20:05:40 -0800307 *pp->target = selected;
Craig Tiller69b093b2016-02-25 19:04:07 -0800308 grpc_pollset_set_del_pollset(exec_ctx, p->base.interested_parties,
Craig Tiller1d881fb2015-12-01 07:39:04 -0800309 pp->pollset);
Craig Tiller6c396862016-01-28 13:53:40 -0800310 grpc_exec_ctx_enqueue(exec_ctx, pp->on_complete, true, NULL);
Craig Tillera82950e2015-09-22 12:33:20 -0700311 gpr_free(pp);
312 }
Craig Tillerab33b482015-11-21 08:11:04 -0800313 grpc_connected_subchannel_notify_on_state_change(
Craig Tiller69b093b2016-02-25 19:04:07 -0800314 exec_ctx, selected, p->base.interested_parties,
Craig Tillera6bebf42015-12-01 17:02:35 -0800315 &p->checking_connectivity, &p->connectivity_changed);
Craig Tillera82950e2015-09-22 12:33:20 -0700316 break;
317 case GRPC_CHANNEL_TRANSIENT_FAILURE:
Craig Tillera82950e2015-09-22 12:33:20 -0700318 p->checking_subchannel =
319 (p->checking_subchannel + 1) % p->num_subchannels;
Craig Tiller131b6de2016-03-31 17:05:28 -0700320 if (p->checking_subchannel == 0) {
321 /* only trigger transient failure when we've tried all alternatives */
322 grpc_connectivity_state_set(exec_ctx, &p->state_tracker,
323 GRPC_CHANNEL_TRANSIENT_FAILURE,
324 "connecting_transient_failure");
325 }
Craig Tillera82950e2015-09-22 12:33:20 -0700326 p->checking_connectivity = grpc_subchannel_check_connectivity(
327 p->subchannels[p->checking_subchannel]);
Craig Tillera82950e2015-09-22 12:33:20 -0700328 if (p->checking_connectivity == GRPC_CHANNEL_TRANSIENT_FAILURE) {
329 grpc_subchannel_notify_on_state_change(
330 exec_ctx, p->subchannels[p->checking_subchannel],
Craig Tiller69b093b2016-02-25 19:04:07 -0800331 p->base.interested_parties, &p->checking_connectivity,
Craig Tiller1d881fb2015-12-01 07:39:04 -0800332 &p->connectivity_changed);
Craig Tillera82950e2015-09-22 12:33:20 -0700333 } else {
334 goto loop;
335 }
336 break;
337 case GRPC_CHANNEL_CONNECTING:
338 case GRPC_CHANNEL_IDLE:
339 grpc_connectivity_state_set(exec_ctx, &p->state_tracker,
340 GRPC_CHANNEL_CONNECTING,
341 "connecting_changed");
342 grpc_subchannel_notify_on_state_change(
343 exec_ctx, p->subchannels[p->checking_subchannel],
Craig Tiller69b093b2016-02-25 19:04:07 -0800344 p->base.interested_parties, &p->checking_connectivity,
Craig Tiller1d881fb2015-12-01 07:39:04 -0800345 &p->connectivity_changed);
Craig Tillera82950e2015-09-22 12:33:20 -0700346 break;
347 case GRPC_CHANNEL_FATAL_FAILURE:
Craig Tillera82950e2015-09-22 12:33:20 -0700348 p->num_subchannels--;
Craig Tiller86c99582015-11-25 15:22:26 -0800349 GPR_SWAP(grpc_subchannel *, p->subchannels[p->checking_subchannel],
350 p->subchannels[p->num_subchannels]);
Craig Tillera82950e2015-09-22 12:33:20 -0700351 GRPC_SUBCHANNEL_UNREF(exec_ctx, p->subchannels[p->num_subchannels],
352 "pick_first");
353 if (p->num_subchannels == 0) {
354 grpc_connectivity_state_set(exec_ctx, &p->state_tracker,
355 GRPC_CHANNEL_FATAL_FAILURE,
356 "no_more_channels");
357 while ((pp = p->pending_picks)) {
358 p->pending_picks = pp->next;
359 *pp->target = NULL;
Craig Tiller6c396862016-01-28 13:53:40 -0800360 grpc_exec_ctx_enqueue(exec_ctx, pp->on_complete, true, NULL);
Craig Tillera82950e2015-09-22 12:33:20 -0700361 gpr_free(pp);
362 }
Craig Tiller1d881fb2015-12-01 07:39:04 -0800363 GRPC_LB_POLICY_WEAK_UNREF(exec_ctx, &p->base,
364 "pick_first_connectivity");
Craig Tillera82950e2015-09-22 12:33:20 -0700365 } else {
366 grpc_connectivity_state_set(exec_ctx, &p->state_tracker,
367 GRPC_CHANNEL_TRANSIENT_FAILURE,
368 "subchannel_failed");
369 p->checking_subchannel %= p->num_subchannels;
370 p->checking_connectivity = grpc_subchannel_check_connectivity(
371 p->subchannels[p->checking_subchannel]);
Craig Tillera82950e2015-09-22 12:33:20 -0700372 goto loop;
373 }
Craig Tiller1ada6ad2015-07-16 16:19:14 -0700374 }
Craig Tillera82950e2015-09-22 12:33:20 -0700375 }
Craig Tiller1ada6ad2015-07-16 16:19:14 -0700376
Craig Tillera82950e2015-09-22 12:33:20 -0700377 gpr_mu_unlock(&p->mu);
Craig Tillereb3b12e2015-06-26 14:42:49 -0700378}
379
Craig Tillera82950e2015-09-22 12:33:20 -0700380static grpc_connectivity_state pf_check_connectivity(grpc_exec_ctx *exec_ctx,
381 grpc_lb_policy *pol) {
382 pick_first_lb_policy *p = (pick_first_lb_policy *)pol;
Craig Tillerc7b5f762015-06-27 11:48:42 -0700383 grpc_connectivity_state st;
Craig Tillera82950e2015-09-22 12:33:20 -0700384 gpr_mu_lock(&p->mu);
385 st = grpc_connectivity_state_check(&p->state_tracker);
386 gpr_mu_unlock(&p->mu);
Craig Tillerc7b5f762015-06-27 11:48:42 -0700387 return st;
388}
389
Craig Tillerfb433852016-03-29 08:51:07 -0700390static void pf_notify_on_state_change(grpc_exec_ctx *exec_ctx,
391 grpc_lb_policy *pol,
392 grpc_connectivity_state *current,
393 grpc_closure *notify) {
Craig Tillera82950e2015-09-22 12:33:20 -0700394 pick_first_lb_policy *p = (pick_first_lb_policy *)pol;
395 gpr_mu_lock(&p->mu);
396 grpc_connectivity_state_notify_on_state_change(exec_ctx, &p->state_tracker,
397 current, notify);
398 gpr_mu_unlock(&p->mu);
Craig Tillerc7b5f762015-06-27 11:48:42 -0700399}
400
Craig Tillerfb433852016-03-29 08:51:07 -0700401static void pf_ping_one(grpc_exec_ctx *exec_ctx, grpc_lb_policy *pol,
402 grpc_closure *closure) {
Craig Tiller28bf8912015-12-07 16:07:04 -0800403 pick_first_lb_policy *p = (pick_first_lb_policy *)pol;
Craig Tiller093193e2016-01-07 07:14:44 -0800404 grpc_connected_subchannel *selected = GET_SELECTED(p);
405 if (selected) {
406 grpc_connected_subchannel_ping(exec_ctx, selected, closure);
Craig Tiller28bf8912015-12-07 16:07:04 -0800407 } else {
Craig Tiller6c396862016-01-28 13:53:40 -0800408 grpc_exec_ctx_enqueue(exec_ctx, closure, false, NULL);
Craig Tiller28bf8912015-12-07 16:07:04 -0800409 }
Craig Tiller28bf8912015-12-07 16:07:04 -0800410}
411
Craig Tillereb3b12e2015-06-26 14:42:49 -0700412static const grpc_lb_policy_vtable pick_first_lb_policy_vtable = {
Craig Tillerc5ff7812016-03-28 12:45:55 -0700413 pf_destroy, pf_shutdown, pf_pick,
414 pf_cancel_pick, pf_cancel_picks, pf_ping_one,
415 pf_exit_idle, pf_check_connectivity, pf_notify_on_state_change};
Craig Tillereb3b12e2015-06-26 14:42:49 -0700416
Craig Tillera82950e2015-09-22 12:33:20 -0700417static void pick_first_factory_ref(grpc_lb_policy_factory *factory) {}
David Garcia Quintas5c4543d2015-09-03 15:49:56 -0700418
Craig Tillera82950e2015-09-22 12:33:20 -0700419static void pick_first_factory_unref(grpc_lb_policy_factory *factory) {}
David Garcia Quintas5c4543d2015-09-03 15:49:56 -0700420
David Garcia Quintas67c0d042016-03-25 01:37:53 -0700421static grpc_lb_policy *create_pick_first(grpc_exec_ctx *exec_ctx,
422 grpc_lb_policy_factory *factory,
Craig Tillera82950e2015-09-22 12:33:20 -0700423 grpc_lb_policy_args *args) {
David Garcia Quintas67c0d042016-03-25 01:37:53 -0700424 GPR_ASSERT(args->addresses != NULL);
425 GPR_ASSERT(args->subchannel_factory != NULL);
426
427 if (args->addresses->naddrs == 0) return NULL;
428
Craig Tillera82950e2015-09-22 12:33:20 -0700429 pick_first_lb_policy *p = gpr_malloc(sizeof(*p));
Craig Tillera82950e2015-09-22 12:33:20 -0700430 memset(p, 0, sizeof(*p));
David Garcia Quintas67c0d042016-03-25 01:37:53 -0700431
Craig Tillera82950e2015-09-22 12:33:20 -0700432 p->subchannels =
David Garcia Quintas67c0d042016-03-25 01:37:53 -0700433 gpr_malloc(sizeof(grpc_subchannel *) * args->addresses->naddrs);
434 memset(p->subchannels, 0, sizeof(*p->subchannels) * args->addresses->naddrs);
435 grpc_subchannel_args sc_args;
436 size_t subchannel_idx = 0;
437 for (size_t i = 0; i < args->addresses->naddrs; i++) {
438 memset(&sc_args, 0, sizeof(grpc_subchannel_args));
439 sc_args.addr = (struct sockaddr *)(args->addresses->addrs[i].addr);
440 sc_args.addr_len = (size_t)args->addresses->addrs[i].len;
441
442 grpc_subchannel *subchannel = grpc_subchannel_factory_create_subchannel(
443 exec_ctx, args->subchannel_factory, &sc_args);
444
445 if (subchannel != NULL) {
446 p->subchannels[subchannel_idx++] = subchannel;
447 }
448 }
449 if (subchannel_idx == 0) {
450 gpr_free(p->subchannels);
451 gpr_free(p);
452 return NULL;
453 }
454 p->num_subchannels = subchannel_idx;
455
456 grpc_lb_policy_init(&p->base, &pick_first_lb_policy_vtable);
Craig Tillera82950e2015-09-22 12:33:20 -0700457 grpc_closure_init(&p->connectivity_changed, pf_connectivity_changed, p);
458 gpr_mu_init(&p->mu);
Craig Tillereb3b12e2015-06-26 14:42:49 -0700459 return &p->base;
460}
David Garcia Quintas5c4543d2015-09-03 15:49:56 -0700461
462static const grpc_lb_policy_factory_vtable pick_first_factory_vtable = {
Craig Tillera82950e2015-09-22 12:33:20 -0700463 pick_first_factory_ref, pick_first_factory_unref, create_pick_first,
464 "pick_first"};
David Garcia Quintas5c4543d2015-09-03 15:49:56 -0700465
466static grpc_lb_policy_factory pick_first_lb_policy_factory = {
Craig Tillera82950e2015-09-22 12:33:20 -0700467 &pick_first_factory_vtable};
David Garcia Quintas5c4543d2015-09-03 15:49:56 -0700468
Craig Tillerfb433852016-03-29 08:51:07 -0700469static grpc_lb_policy_factory *pick_first_lb_factory_create() {
David Garcia Quintas5c4543d2015-09-03 15:49:56 -0700470 return &pick_first_lb_policy_factory;
471}
Craig Tillerfb433852016-03-29 08:51:07 -0700472
473/* Plugin registration */
474
475void grpc_lb_policy_pick_first_init() {
Craig Tiller3113ef42016-03-29 09:03:14 -0700476 grpc_register_lb_policy(pick_first_lb_factory_create());
Craig Tillerfb433852016-03-29 08:51:07 -0700477}
478
479void grpc_lb_policy_pick_first_shutdown() {}