blob: 20c2dd8bf343233c0dae9dc072f57ef0e9271369 [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
Vijay Paie9ef5362016-02-25 18:10:24 -080034#include "src/core/client_config/lb_policies/pick_first.h"
Craig Tiller69b093b2016-02-25 19:04:07 -080035#include "src/core/client_config/lb_policy_factory.h"
Craig Tillereb3b12e2015-06-26 14:42:49 -070036
37#include <string.h>
38
39#include <grpc/support/alloc.h>
Craig Tiller08a1cf82015-06-29 09:37:52 -070040#include "src/core/transport/connectivity_state.h"
Craig Tillereb3b12e2015-06-26 14:42:49 -070041
Craig Tillera82950e2015-09-22 12:33:20 -070042typedef struct pending_pick {
Craig Tillereb3b12e2015-06-26 14:42:49 -070043 struct pending_pick *next;
44 grpc_pollset *pollset;
Craig Tiller8c0d96f2016-03-11 14:27:52 -080045 uint32_t initial_metadata_flags;
Craig Tillerb5585d42015-11-17 07:18:31 -080046 grpc_connected_subchannel **target;
Craig Tiller33825112015-09-18 07:44:19 -070047 grpc_closure *on_complete;
Craig Tillereb3b12e2015-06-26 14:42:49 -070048} pending_pick;
49
Craig Tillera82950e2015-09-22 12:33:20 -070050typedef struct {
Craig Tillereb3b12e2015-06-26 14:42:49 -070051 /** base policy: must be first */
52 grpc_lb_policy base;
Craig Tillereb3b12e2015-06-26 14:42:49 -070053 /** all our subchannels */
54 grpc_subchannel **subchannels;
55 size_t num_subchannels;
56
Craig Tiller33825112015-09-18 07:44:19 -070057 grpc_closure connectivity_changed;
Craig Tillereb3b12e2015-06-26 14:42:49 -070058
Craig Tiller86c0f8a2015-12-01 20:05:40 -080059 /** the selected channel (a grpc_connected_subchannel) */
60 gpr_atm selected;
Craig Tiller320bee02016-01-06 17:33:45 -080061
62 /** mutex protecting remaining members */
63 gpr_mu mu;
Craig Tillereb3b12e2015-06-26 14:42:49 -070064 /** have we started picking? */
65 int started_picking;
Craig Tillera14215a2015-07-17 17:21:08 -070066 /** are we shut down? */
67 int shutdown;
Craig Tillereb3b12e2015-06-26 14:42:49 -070068 /** which subchannel are we watching? */
69 size_t checking_subchannel;
70 /** what is the connectivity of that channel? */
71 grpc_connectivity_state checking_connectivity;
72 /** list of picks that are waiting on connectivity */
73 pending_pick *pending_picks;
Craig Tillerc7b5f762015-06-27 11:48:42 -070074
75 /** our connectivity state tracker */
76 grpc_connectivity_state_tracker state_tracker;
Craig Tillereb3b12e2015-06-26 14:42:49 -070077} pick_first_lb_policy;
78
Craig Tiller81afdda2016-01-11 17:09:18 -080079#define GET_SELECTED(p) \
Craig Tillerd9d474a2016-01-26 06:50:51 -080080 ((grpc_connected_subchannel *)gpr_atm_acq_load(&(p)->selected))
Craig Tiller86c0f8a2015-12-01 20:05:40 -080081
Craig Tillera82950e2015-09-22 12:33:20 -070082void pf_destroy(grpc_exec_ctx *exec_ctx, grpc_lb_policy *pol) {
83 pick_first_lb_policy *p = (pick_first_lb_policy *)pol;
Craig Tiller86c0f8a2015-12-01 20:05:40 -080084 grpc_connected_subchannel *selected = GET_SELECTED(p);
Craig Tillerd7b68e72015-06-28 11:41:09 -070085 size_t i;
Craig Tillera82950e2015-09-22 12:33:20 -070086 GPR_ASSERT(p->pending_picks == NULL);
87 for (i = 0; i < p->num_subchannels; i++) {
88 GRPC_SUBCHANNEL_UNREF(exec_ctx, p->subchannels[i], "pick_first");
89 }
Craig Tiller86c0f8a2015-12-01 20:05:40 -080090 if (selected != NULL) {
91 GRPC_CONNECTED_SUBCHANNEL_UNREF(exec_ctx, selected, "picked_first");
Craig Tiller89a768e2015-10-06 09:55:59 -070092 }
Craig Tillera82950e2015-09-22 12:33:20 -070093 grpc_connectivity_state_destroy(exec_ctx, &p->state_tracker);
94 gpr_free(p->subchannels);
95 gpr_mu_destroy(&p->mu);
96 gpr_free(p);
Craig Tillereb3b12e2015-06-26 14:42:49 -070097}
98
Craig Tillera82950e2015-09-22 12:33:20 -070099void pf_shutdown(grpc_exec_ctx *exec_ctx, grpc_lb_policy *pol) {
100 pick_first_lb_policy *p = (pick_first_lb_policy *)pol;
Craig Tillerd2cc4592015-07-01 07:50:47 -0700101 pending_pick *pp;
Craig Tiller86c0f8a2015-12-01 20:05:40 -0800102 grpc_connected_subchannel *selected;
Craig Tillera82950e2015-09-22 12:33:20 -0700103 gpr_mu_lock(&p->mu);
Craig Tiller86c0f8a2015-12-01 20:05:40 -0800104 selected = GET_SELECTED(p);
Craig Tillera14215a2015-07-17 17:21:08 -0700105 p->shutdown = 1;
Craig Tiller5795da72015-09-17 15:27:13 -0700106 pp = p->pending_picks;
107 p->pending_picks = NULL;
Craig Tillera82950e2015-09-22 12:33:20 -0700108 grpc_connectivity_state_set(exec_ctx, &p->state_tracker,
109 GRPC_CHANNEL_FATAL_FAILURE, "shutdown");
Craig Tillerf036a642015-12-01 17:00:40 -0800110 /* cancel subscription */
Craig Tiller86c0f8a2015-12-01 20:05:40 -0800111 if (selected != NULL) {
Craig Tiller1d881fb2015-12-01 07:39:04 -0800112 grpc_connected_subchannel_notify_on_state_change(
Craig Tiller86c0f8a2015-12-01 20:05:40 -0800113 exec_ctx, selected, NULL, NULL, &p->connectivity_changed);
Craig Tiller48613042015-11-29 14:45:11 -0800114 } else {
Craig Tiller1d881fb2015-12-01 07:39:04 -0800115 grpc_subchannel_notify_on_state_change(
116 exec_ctx, p->subchannels[p->checking_subchannel], NULL, NULL,
117 &p->connectivity_changed);
Craig Tiller48613042015-11-29 14:45:11 -0800118 }
Craig Tillera82950e2015-09-22 12:33:20 -0700119 gpr_mu_unlock(&p->mu);
120 while (pp != NULL) {
121 pending_pick *next = pp->next;
122 *pp->target = NULL;
Craig Tiller69b093b2016-02-25 19:04:07 -0800123 grpc_pollset_set_del_pollset(exec_ctx, p->base.interested_parties,
Craig Tiller1d881fb2015-12-01 07:39:04 -0800124 pp->pollset);
Craig Tiller6c396862016-01-28 13:53:40 -0800125 grpc_exec_ctx_enqueue(exec_ctx, pp->on_complete, true, NULL);
Craig Tillera82950e2015-09-22 12:33:20 -0700126 gpr_free(pp);
127 pp = next;
128 }
Craig Tillereb3b12e2015-06-26 14:42:49 -0700129}
130
Craig Tiller577c9b22015-11-02 14:11:15 -0800131static void pf_cancel_pick(grpc_exec_ctx *exec_ctx, grpc_lb_policy *pol,
Craig Tillerb5585d42015-11-17 07:18:31 -0800132 grpc_connected_subchannel **target) {
Craig Tiller577c9b22015-11-02 14:11:15 -0800133 pick_first_lb_policy *p = (pick_first_lb_policy *)pol;
134 pending_pick *pp;
135 gpr_mu_lock(&p->mu);
136 pp = p->pending_picks;
137 p->pending_picks = NULL;
138 while (pp != NULL) {
139 pending_pick *next = pp->next;
140 if (pp->target == target) {
Craig Tiller69b093b2016-02-25 19:04:07 -0800141 grpc_pollset_set_del_pollset(exec_ctx, p->base.interested_parties,
Craig Tiller1d881fb2015-12-01 07:39:04 -0800142 pp->pollset);
Craig Tiller577c9b22015-11-02 14:11:15 -0800143 *target = NULL;
Craig Tiller6c396862016-01-28 13:53:40 -0800144 grpc_exec_ctx_enqueue(exec_ctx, pp->on_complete, false, NULL);
Craig Tiller577c9b22015-11-02 14:11:15 -0800145 gpr_free(pp);
146 } else {
147 pp->next = p->pending_picks;
148 p->pending_picks = pp;
149 }
150 pp = next;
151 }
152 gpr_mu_unlock(&p->mu);
153}
154
Craig Tiller8c0d96f2016-03-11 14:27:52 -0800155static void pf_cancel_picks(grpc_exec_ctx *exec_ctx, grpc_lb_policy *pol,
156 uint32_t initial_metadata_flags_mask,
157 uint32_t initial_metadata_flags_eq) {
158 pick_first_lb_policy *p = (pick_first_lb_policy *)pol;
159 pending_pick *pp;
160 gpr_mu_lock(&p->mu);
161 pp = p->pending_picks;
162 p->pending_picks = NULL;
163 while (pp != NULL) {
164 pending_pick *next = pp->next;
165 if ((pp->initial_metadata_flags & initial_metadata_flags_mask) ==
166 initial_metadata_flags_eq) {
167 grpc_pollset_set_del_pollset(exec_ctx, p->base.interested_parties,
168 pp->pollset);
169 *pp->target = NULL;
170 grpc_exec_ctx_enqueue(exec_ctx, pp->on_complete, false, NULL);
171 gpr_free(pp);
172 } else {
173 pp->next = p->pending_picks;
174 p->pending_picks = pp;
175 }
176 pp = next;
177 }
178 gpr_mu_unlock(&p->mu);
179}
180
Craig Tillera82950e2015-09-22 12:33:20 -0700181static void start_picking(grpc_exec_ctx *exec_ctx, pick_first_lb_policy *p) {
Craig Tiller48cb07c2015-07-15 16:16:15 -0700182 p->started_picking = 1;
183 p->checking_subchannel = 0;
184 p->checking_connectivity = GRPC_CHANNEL_IDLE;
Craig Tiller48613042015-11-29 14:45:11 -0800185 GRPC_LB_POLICY_WEAK_REF(&p->base, "pick_first_connectivity");
Craig Tillera82950e2015-09-22 12:33:20 -0700186 grpc_subchannel_notify_on_state_change(
187 exec_ctx, p->subchannels[p->checking_subchannel],
Craig Tiller69b093b2016-02-25 19:04:07 -0800188 p->base.interested_parties, &p->checking_connectivity,
Craig Tiller1d881fb2015-12-01 07:39:04 -0800189 &p->connectivity_changed);
Craig Tiller48cb07c2015-07-15 16:16:15 -0700190}
191
Craig Tillera82950e2015-09-22 12:33:20 -0700192void pf_exit_idle(grpc_exec_ctx *exec_ctx, grpc_lb_policy *pol) {
193 pick_first_lb_policy *p = (pick_first_lb_policy *)pol;
194 gpr_mu_lock(&p->mu);
195 if (!p->started_picking) {
196 start_picking(exec_ctx, p);
197 }
198 gpr_mu_unlock(&p->mu);
Craig Tillereb3b12e2015-06-26 14:42:49 -0700199}
200
Craig Tiller577c9b22015-11-02 14:11:15 -0800201int pf_pick(grpc_exec_ctx *exec_ctx, grpc_lb_policy *pol, grpc_pollset *pollset,
Craig Tillerab33b482015-11-21 08:11:04 -0800202 grpc_metadata_batch *initial_metadata,
Craig Tiller8c0d96f2016-03-11 14:27:52 -0800203 uint32_t initial_metadata_flags, grpc_connected_subchannel **target,
204 grpc_closure *on_complete) {
Craig Tillera82950e2015-09-22 12:33:20 -0700205 pick_first_lb_policy *p = (pick_first_lb_policy *)pol;
Craig Tiller45724b32015-09-22 10:42:19 -0700206 pending_pick *pp;
Craig Tiller320bee02016-01-06 17:33:45 -0800207
208 /* Check atomically for a selected channel */
Craig Tiller86c0f8a2015-12-01 20:05:40 -0800209 grpc_connected_subchannel *selected = GET_SELECTED(p);
210 if (selected != NULL) {
211 *target = selected;
212 return 1;
213 }
Craig Tiller320bee02016-01-06 17:33:45 -0800214
215 /* No subchannel selected yet, so acquire lock and then attempt again */
Craig Tillera82950e2015-09-22 12:33:20 -0700216 gpr_mu_lock(&p->mu);
Craig Tiller86c0f8a2015-12-01 20:05:40 -0800217 selected = GET_SELECTED(p);
218 if (selected) {
Craig Tillera82950e2015-09-22 12:33:20 -0700219 gpr_mu_unlock(&p->mu);
Craig Tiller86c0f8a2015-12-01 20:05:40 -0800220 *target = selected;
Craig Tiller577c9b22015-11-02 14:11:15 -0800221 return 1;
Craig Tillera82950e2015-09-22 12:33:20 -0700222 } else {
223 if (!p->started_picking) {
224 start_picking(exec_ctx, p);
Craig Tiller45724b32015-09-22 10:42:19 -0700225 }
Craig Tiller69b093b2016-02-25 19:04:07 -0800226 grpc_pollset_set_add_pollset(exec_ctx, p->base.interested_parties, pollset);
Craig Tillera82950e2015-09-22 12:33:20 -0700227 pp = gpr_malloc(sizeof(*pp));
228 pp->next = p->pending_picks;
229 pp->pollset = pollset;
230 pp->target = target;
Craig Tiller8c0d96f2016-03-11 14:27:52 -0800231 pp->initial_metadata_flags = initial_metadata_flags;
Craig Tillera82950e2015-09-22 12:33:20 -0700232 pp->on_complete = on_complete;
233 p->pending_picks = pp;
234 gpr_mu_unlock(&p->mu);
Craig Tiller577c9b22015-11-02 14:11:15 -0800235 return 0;
Craig Tillera82950e2015-09-22 12:33:20 -0700236 }
Craig Tiller45724b32015-09-22 10:42:19 -0700237}
238
Craig Tiller1f41b6b2015-10-09 15:07:02 -0700239static void destroy_subchannels(grpc_exec_ctx *exec_ctx, void *arg,
Craig Tiller6c396862016-01-28 13:53:40 -0800240 bool iomgr_success) {
Craig Tillerb09d84d2015-10-06 09:12:16 -0700241 pick_first_lb_policy *p = arg;
242 size_t i;
Craig Tillerb09d84d2015-10-06 09:12:16 -0700243 size_t num_subchannels = p->num_subchannels;
244 grpc_subchannel **subchannels;
Craig Tillerb09d84d2015-10-06 09:12:16 -0700245
246 gpr_mu_lock(&p->mu);
247 subchannels = p->subchannels;
Craig Tillerb09d84d2015-10-06 09:12:16 -0700248 p->num_subchannels = 0;
249 p->subchannels = NULL;
250 gpr_mu_unlock(&p->mu);
Craig Tiller48613042015-11-29 14:45:11 -0800251 GRPC_LB_POLICY_WEAK_UNREF(exec_ctx, &p->base, "destroy_subchannels");
Craig Tillerb09d84d2015-10-06 09:12:16 -0700252
253 for (i = 0; i < num_subchannels; i++) {
Craig Tillerb09d84d2015-10-06 09:12:16 -0700254 GRPC_SUBCHANNEL_UNREF(exec_ctx, subchannels[i], "pick_first");
255 }
256
257 gpr_free(subchannels);
258}
259
Craig Tillera82950e2015-09-22 12:33:20 -0700260static void pf_connectivity_changed(grpc_exec_ctx *exec_ctx, void *arg,
Craig Tiller6c396862016-01-28 13:53:40 -0800261 bool iomgr_success) {
Craig Tillereb3b12e2015-06-26 14:42:49 -0700262 pick_first_lb_policy *p = arg;
Craig Tillerb5585d42015-11-17 07:18:31 -0800263 grpc_subchannel *selected_subchannel;
Craig Tillereb3b12e2015-06-26 14:42:49 -0700264 pending_pick *pp;
Craig Tiller86c0f8a2015-12-01 20:05:40 -0800265 grpc_connected_subchannel *selected;
Craig Tillereb3b12e2015-06-26 14:42:49 -0700266
Craig Tillera82950e2015-09-22 12:33:20 -0700267 gpr_mu_lock(&p->mu);
Craig Tiller1ada6ad2015-07-16 16:19:14 -0700268
Craig Tiller86c0f8a2015-12-01 20:05:40 -0800269 selected = GET_SELECTED(p);
270
Craig Tillera82950e2015-09-22 12:33:20 -0700271 if (p->shutdown) {
272 gpr_mu_unlock(&p->mu);
Craig Tiller48613042015-11-29 14:45:11 -0800273 GRPC_LB_POLICY_WEAK_UNREF(exec_ctx, &p->base, "pick_first_connectivity");
Craig Tillera82950e2015-09-22 12:33:20 -0700274 return;
Craig Tiller86c0f8a2015-12-01 20:05:40 -0800275 } else if (selected != NULL) {
Craig Tillercb2609f2015-11-24 17:19:19 -0800276 if (p->checking_connectivity == GRPC_CHANNEL_TRANSIENT_FAILURE) {
277 /* if the selected channel goes bad, we're done */
278 p->checking_connectivity = GRPC_CHANNEL_FATAL_FAILURE;
279 }
Craig Tillera82950e2015-09-22 12:33:20 -0700280 grpc_connectivity_state_set(exec_ctx, &p->state_tracker,
281 p->checking_connectivity, "selected_changed");
282 if (p->checking_connectivity != GRPC_CHANNEL_FATAL_FAILURE) {
Craig Tillerab33b482015-11-21 08:11:04 -0800283 grpc_connected_subchannel_notify_on_state_change(
Craig Tiller69b093b2016-02-25 19:04:07 -0800284 exec_ctx, selected, p->base.interested_parties,
Craig Tillera6bebf42015-12-01 17:02:35 -0800285 &p->checking_connectivity, &p->connectivity_changed);
Craig Tillera82950e2015-09-22 12:33:20 -0700286 } else {
Craig Tiller48613042015-11-29 14:45:11 -0800287 GRPC_LB_POLICY_WEAK_UNREF(exec_ctx, &p->base, "pick_first_connectivity");
Craig Tiller1ada6ad2015-07-16 16:19:14 -0700288 }
Craig Tillera82950e2015-09-22 12:33:20 -0700289 } else {
290 loop:
291 switch (p->checking_connectivity) {
292 case GRPC_CHANNEL_READY:
293 grpc_connectivity_state_set(exec_ctx, &p->state_tracker,
294 GRPC_CHANNEL_READY, "connecting_ready");
Craig Tillerb5585d42015-11-17 07:18:31 -0800295 selected_subchannel = p->subchannels[p->checking_subchannel];
Craig Tiller81afdda2016-01-11 17:09:18 -0800296 selected =
297 grpc_subchannel_get_connected_subchannel(selected_subchannel);
Craig Tiller86c0f8a2015-12-01 20:05:40 -0800298 GPR_ASSERT(selected != NULL);
Craig Tiller86c0f8a2015-12-01 20:05:40 -0800299 GRPC_CONNECTED_SUBCHANNEL_REF(selected, "picked_first");
Craig Tillerb09d84d2015-10-06 09:12:16 -0700300 /* drop the pick list: we are connected now */
Craig Tiller48613042015-11-29 14:45:11 -0800301 GRPC_LB_POLICY_WEAK_REF(&p->base, "destroy_subchannels");
Craig Tillerd9d474a2016-01-26 06:50:51 -0800302 gpr_atm_rel_store(&p->selected, (gpr_atm)selected);
Craig Tiller6c396862016-01-28 13:53:40 -0800303 grpc_exec_ctx_enqueue(
304 exec_ctx, grpc_closure_create(destroy_subchannels, p), true, NULL);
Craig Tillerb09d84d2015-10-06 09:12:16 -0700305 /* update any calls that were waiting for a pick */
Craig Tillera82950e2015-09-22 12:33:20 -0700306 while ((pp = p->pending_picks)) {
307 p->pending_picks = pp->next;
Craig Tiller86c0f8a2015-12-01 20:05:40 -0800308 *pp->target = selected;
Craig Tiller69b093b2016-02-25 19:04:07 -0800309 grpc_pollset_set_del_pollset(exec_ctx, p->base.interested_parties,
Craig Tiller1d881fb2015-12-01 07:39:04 -0800310 pp->pollset);
Craig Tiller6c396862016-01-28 13:53:40 -0800311 grpc_exec_ctx_enqueue(exec_ctx, pp->on_complete, true, NULL);
Craig Tillera82950e2015-09-22 12:33:20 -0700312 gpr_free(pp);
313 }
Craig Tillerab33b482015-11-21 08:11:04 -0800314 grpc_connected_subchannel_notify_on_state_change(
Craig Tiller69b093b2016-02-25 19:04:07 -0800315 exec_ctx, selected, p->base.interested_parties,
Craig Tillera6bebf42015-12-01 17:02:35 -0800316 &p->checking_connectivity, &p->connectivity_changed);
Craig Tillera82950e2015-09-22 12:33:20 -0700317 break;
318 case GRPC_CHANNEL_TRANSIENT_FAILURE:
319 grpc_connectivity_state_set(exec_ctx, &p->state_tracker,
320 GRPC_CHANNEL_TRANSIENT_FAILURE,
321 "connecting_transient_failure");
Craig Tillera82950e2015-09-22 12:33:20 -0700322 p->checking_subchannel =
323 (p->checking_subchannel + 1) % p->num_subchannels;
324 p->checking_connectivity = grpc_subchannel_check_connectivity(
325 p->subchannels[p->checking_subchannel]);
Craig Tillera82950e2015-09-22 12:33:20 -0700326 if (p->checking_connectivity == GRPC_CHANNEL_TRANSIENT_FAILURE) {
327 grpc_subchannel_notify_on_state_change(
328 exec_ctx, p->subchannels[p->checking_subchannel],
Craig Tiller69b093b2016-02-25 19:04:07 -0800329 p->base.interested_parties, &p->checking_connectivity,
Craig Tiller1d881fb2015-12-01 07:39:04 -0800330 &p->connectivity_changed);
Craig Tillera82950e2015-09-22 12:33:20 -0700331 } else {
332 goto loop;
333 }
334 break;
335 case GRPC_CHANNEL_CONNECTING:
336 case GRPC_CHANNEL_IDLE:
337 grpc_connectivity_state_set(exec_ctx, &p->state_tracker,
338 GRPC_CHANNEL_CONNECTING,
339 "connecting_changed");
340 grpc_subchannel_notify_on_state_change(
341 exec_ctx, p->subchannels[p->checking_subchannel],
Craig Tiller69b093b2016-02-25 19:04:07 -0800342 p->base.interested_parties, &p->checking_connectivity,
Craig Tiller1d881fb2015-12-01 07:39:04 -0800343 &p->connectivity_changed);
Craig Tillera82950e2015-09-22 12:33:20 -0700344 break;
345 case GRPC_CHANNEL_FATAL_FAILURE:
Craig Tillera82950e2015-09-22 12:33:20 -0700346 p->num_subchannels--;
Craig Tiller86c99582015-11-25 15:22:26 -0800347 GPR_SWAP(grpc_subchannel *, p->subchannels[p->checking_subchannel],
348 p->subchannels[p->num_subchannels]);
Craig Tillera82950e2015-09-22 12:33:20 -0700349 GRPC_SUBCHANNEL_UNREF(exec_ctx, p->subchannels[p->num_subchannels],
350 "pick_first");
351 if (p->num_subchannels == 0) {
352 grpc_connectivity_state_set(exec_ctx, &p->state_tracker,
353 GRPC_CHANNEL_FATAL_FAILURE,
354 "no_more_channels");
355 while ((pp = p->pending_picks)) {
356 p->pending_picks = pp->next;
357 *pp->target = NULL;
Craig Tiller6c396862016-01-28 13:53:40 -0800358 grpc_exec_ctx_enqueue(exec_ctx, pp->on_complete, true, NULL);
Craig Tillera82950e2015-09-22 12:33:20 -0700359 gpr_free(pp);
360 }
Craig Tiller1d881fb2015-12-01 07:39:04 -0800361 GRPC_LB_POLICY_WEAK_UNREF(exec_ctx, &p->base,
362 "pick_first_connectivity");
Craig Tillera82950e2015-09-22 12:33:20 -0700363 } else {
364 grpc_connectivity_state_set(exec_ctx, &p->state_tracker,
365 GRPC_CHANNEL_TRANSIENT_FAILURE,
366 "subchannel_failed");
367 p->checking_subchannel %= p->num_subchannels;
368 p->checking_connectivity = grpc_subchannel_check_connectivity(
369 p->subchannels[p->checking_subchannel]);
Craig Tillera82950e2015-09-22 12:33:20 -0700370 goto loop;
371 }
Craig Tiller1ada6ad2015-07-16 16:19:14 -0700372 }
Craig Tillera82950e2015-09-22 12:33:20 -0700373 }
Craig Tiller1ada6ad2015-07-16 16:19:14 -0700374
Craig Tillera82950e2015-09-22 12:33:20 -0700375 gpr_mu_unlock(&p->mu);
Craig Tillereb3b12e2015-06-26 14:42:49 -0700376}
377
Craig Tillera82950e2015-09-22 12:33:20 -0700378static grpc_connectivity_state pf_check_connectivity(grpc_exec_ctx *exec_ctx,
379 grpc_lb_policy *pol) {
380 pick_first_lb_policy *p = (pick_first_lb_policy *)pol;
Craig Tillerc7b5f762015-06-27 11:48:42 -0700381 grpc_connectivity_state st;
Craig Tillera82950e2015-09-22 12:33:20 -0700382 gpr_mu_lock(&p->mu);
383 st = grpc_connectivity_state_check(&p->state_tracker);
384 gpr_mu_unlock(&p->mu);
Craig Tillerc7b5f762015-06-27 11:48:42 -0700385 return st;
386}
387
Craig Tillera82950e2015-09-22 12:33:20 -0700388void pf_notify_on_state_change(grpc_exec_ctx *exec_ctx, grpc_lb_policy *pol,
389 grpc_connectivity_state *current,
390 grpc_closure *notify) {
391 pick_first_lb_policy *p = (pick_first_lb_policy *)pol;
392 gpr_mu_lock(&p->mu);
393 grpc_connectivity_state_notify_on_state_change(exec_ctx, &p->state_tracker,
394 current, notify);
395 gpr_mu_unlock(&p->mu);
Craig Tillerc7b5f762015-06-27 11:48:42 -0700396}
397
Craig Tillere2c62372015-12-07 16:11:03 -0800398void pf_ping_one(grpc_exec_ctx *exec_ctx, grpc_lb_policy *pol,
399 grpc_closure *closure) {
Craig Tiller28bf8912015-12-07 16:07:04 -0800400 pick_first_lb_policy *p = (pick_first_lb_policy *)pol;
Craig Tiller093193e2016-01-07 07:14:44 -0800401 grpc_connected_subchannel *selected = GET_SELECTED(p);
402 if (selected) {
403 grpc_connected_subchannel_ping(exec_ctx, selected, closure);
Craig Tiller28bf8912015-12-07 16:07:04 -0800404 } else {
Craig Tiller6c396862016-01-28 13:53:40 -0800405 grpc_exec_ctx_enqueue(exec_ctx, closure, false, NULL);
Craig Tiller28bf8912015-12-07 16:07:04 -0800406 }
Craig Tiller28bf8912015-12-07 16:07:04 -0800407}
408
Craig Tillereb3b12e2015-06-26 14:42:49 -0700409static const grpc_lb_policy_vtable pick_first_lb_policy_vtable = {
Craig Tiller8c0d96f2016-03-11 14:27:52 -0800410 pf_destroy, pf_shutdown, pf_pick, pf_cancel_pick, pf_cancel_picks,
411 pf_ping_one, pf_exit_idle, pf_check_connectivity,
412 pf_notify_on_state_change};
Craig Tillereb3b12e2015-06-26 14:42:49 -0700413
Craig Tillera82950e2015-09-22 12:33:20 -0700414static void pick_first_factory_ref(grpc_lb_policy_factory *factory) {}
David Garcia Quintas5c4543d2015-09-03 15:49:56 -0700415
Craig Tillera82950e2015-09-22 12:33:20 -0700416static void pick_first_factory_unref(grpc_lb_policy_factory *factory) {}
David Garcia Quintas5c4543d2015-09-03 15:49:56 -0700417
Craig Tillera82950e2015-09-22 12:33:20 -0700418static grpc_lb_policy *create_pick_first(grpc_lb_policy_factory *factory,
419 grpc_lb_policy_args *args) {
420 pick_first_lb_policy *p = gpr_malloc(sizeof(*p));
421 GPR_ASSERT(args->num_subchannels > 0);
422 memset(p, 0, sizeof(*p));
423 grpc_lb_policy_init(&p->base, &pick_first_lb_policy_vtable);
424 p->subchannels =
425 gpr_malloc(sizeof(grpc_subchannel *) * args->num_subchannels);
David Garcia Quintasc7705c72015-09-09 17:21:11 -0700426 p->num_subchannels = args->num_subchannels;
Craig Tillera82950e2015-09-22 12:33:20 -0700427 grpc_connectivity_state_init(&p->state_tracker, GRPC_CHANNEL_IDLE,
428 "pick_first");
429 memcpy(p->subchannels, args->subchannels,
430 sizeof(grpc_subchannel *) * args->num_subchannels);
431 grpc_closure_init(&p->connectivity_changed, pf_connectivity_changed, p);
432 gpr_mu_init(&p->mu);
Craig Tillereb3b12e2015-06-26 14:42:49 -0700433 return &p->base;
434}
David Garcia Quintas5c4543d2015-09-03 15:49:56 -0700435
436static const grpc_lb_policy_factory_vtable pick_first_factory_vtable = {
Craig Tillera82950e2015-09-22 12:33:20 -0700437 pick_first_factory_ref, pick_first_factory_unref, create_pick_first,
438 "pick_first"};
David Garcia Quintas5c4543d2015-09-03 15:49:56 -0700439
440static grpc_lb_policy_factory pick_first_lb_policy_factory = {
Craig Tillera82950e2015-09-22 12:33:20 -0700441 &pick_first_factory_vtable};
David Garcia Quintas5c4543d2015-09-03 15:49:56 -0700442
Craig Tillera82950e2015-09-22 12:33:20 -0700443grpc_lb_policy_factory *grpc_pick_first_lb_factory_create() {
David Garcia Quintas5c4543d2015-09-03 15:49:56 -0700444 return &pick_first_lb_policy_factory;
445}