Craig Tiller | af69180 | 2015-06-23 14:57:07 -0700 | [diff] [blame] | 1 | /* |
| 2 | * |
| 3 | * Copyright 2015, Google Inc. |
| 4 | * 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 | |
| 34 | #include "src/core/client_config/subchannel.h" |
Craig Tiller | 2595ab7 | 2015-06-25 15:26:00 -0700 | [diff] [blame] | 35 | |
Craig Tiller | f7afa1f | 2015-06-26 09:02:20 -0700 | [diff] [blame] | 36 | #include <string.h> |
| 37 | |
Craig Tiller | 2595ab7 | 2015-06-25 15:26:00 -0700 | [diff] [blame] | 38 | #include <grpc/support/alloc.h> |
| 39 | |
Craig Tiller | eb3b12e | 2015-06-26 14:42:49 -0700 | [diff] [blame] | 40 | #include "src/core/channel/channel_args.h" |
Craig Tiller | ff54c92 | 2015-06-26 16:57:20 -0700 | [diff] [blame] | 41 | #include "src/core/channel/connected_channel.h" |
Craig Tiller | ff3ae68 | 2015-06-29 17:44:04 -0700 | [diff] [blame] | 42 | #include "src/core/iomgr/alarm.h" |
Craig Tiller | 08a1cf8 | 2015-06-29 09:37:52 -0700 | [diff] [blame] | 43 | #include "src/core/transport/connectivity_state.h" |
Craig Tiller | eb3b12e | 2015-06-26 14:42:49 -0700 | [diff] [blame] | 44 | |
| 45 | typedef struct { |
Craig Tiller | 4ab82d2 | 2015-06-29 09:40:33 -0700 | [diff] [blame] | 46 | /* all fields protected by subchannel->mu */ |
| 47 | /** refcount */ |
| 48 | int refs; |
| 49 | /** parent subchannel */ |
Craig Tiller | eb3b12e | 2015-06-26 14:42:49 -0700 | [diff] [blame] | 50 | grpc_subchannel *subchannel; |
| 51 | } connection; |
| 52 | |
Craig Tiller | df91ba5 | 2015-06-29 10:55:46 -0700 | [diff] [blame] | 53 | typedef struct { |
| 54 | grpc_iomgr_closure closure; |
| 55 | size_t version; |
| 56 | grpc_subchannel *subchannel; |
| 57 | grpc_connectivity_state connectivity_state; |
| 58 | } state_watcher; |
| 59 | |
Craig Tiller | 5f84c84 | 2015-06-26 16:08:21 -0700 | [diff] [blame] | 60 | typedef struct waiting_for_connect { |
| 61 | struct waiting_for_connect *next; |
| 62 | grpc_iomgr_closure *notify; |
Craig Tiller | abf3638 | 2015-06-29 16:13:27 -0700 | [diff] [blame] | 63 | grpc_pollset *pollset; |
Craig Tiller | 5f84c84 | 2015-06-26 16:08:21 -0700 | [diff] [blame] | 64 | grpc_subchannel_call **target; |
Craig Tiller | df91ba5 | 2015-06-29 10:55:46 -0700 | [diff] [blame] | 65 | grpc_subchannel *subchannel; |
| 66 | grpc_iomgr_closure continuation; |
Craig Tiller | 5f84c84 | 2015-06-26 16:08:21 -0700 | [diff] [blame] | 67 | } waiting_for_connect; |
| 68 | |
Craig Tiller | 2595ab7 | 2015-06-25 15:26:00 -0700 | [diff] [blame] | 69 | struct grpc_subchannel { |
Craig Tiller | 9162466 | 2015-06-25 16:31:02 -0700 | [diff] [blame] | 70 | grpc_connector *connector; |
Craig Tiller | f7afa1f | 2015-06-26 09:02:20 -0700 | [diff] [blame] | 71 | |
| 72 | /** non-transport related channel filters */ |
| 73 | const grpc_channel_filter **filters; |
Craig Tiller | 5945ee1 | 2015-06-27 10:36:09 -0700 | [diff] [blame] | 74 | size_t num_filters; |
Craig Tiller | f7afa1f | 2015-06-26 09:02:20 -0700 | [diff] [blame] | 75 | /** channel arguments */ |
| 76 | grpc_channel_args *args; |
| 77 | /** address to connect to */ |
| 78 | struct sockaddr *addr; |
| 79 | size_t addr_len; |
Craig Tiller | 5f84c84 | 2015-06-26 16:08:21 -0700 | [diff] [blame] | 80 | /** metadata context */ |
| 81 | grpc_mdctx *mdctx; |
Craig Tiller | f037011 | 2015-07-01 14:26:11 -0700 | [diff] [blame] | 82 | /** master channel - the grpc_channel instance that ultimately owns |
| 83 | this channel_data via its channel stack. |
| 84 | We occasionally use this to bump the refcount on the master channel |
| 85 | to keep ourselves alive through an asynchronous operation. */ |
Craig Tiller | 9846503 | 2015-06-29 14:36:42 -0700 | [diff] [blame] | 86 | grpc_channel *master; |
Craig Tiller | b6fbf1d | 2015-06-29 15:25:49 -0700 | [diff] [blame] | 87 | /** have we seen a disconnection? */ |
| 88 | int disconnected; |
Craig Tiller | eb3b12e | 2015-06-26 14:42:49 -0700 | [diff] [blame] | 89 | |
| 90 | /** set during connection */ |
Craig Tiller | 04c5d4b | 2015-06-26 17:21:41 -0700 | [diff] [blame] | 91 | grpc_connect_out_args connecting_result; |
Craig Tiller | eb3b12e | 2015-06-26 14:42:49 -0700 | [diff] [blame] | 92 | |
| 93 | /** callback for connection finishing */ |
| 94 | grpc_iomgr_closure connected; |
| 95 | |
Craig Tiller | 5f84c84 | 2015-06-26 16:08:21 -0700 | [diff] [blame] | 96 | /** pollset_set tracking who's interested in a connection |
| 97 | being setup */ |
| 98 | grpc_pollset_set pollset_set; |
| 99 | |
Craig Tiller | eb3b12e | 2015-06-26 14:42:49 -0700 | [diff] [blame] | 100 | /** mutex protecting remaining elements */ |
| 101 | gpr_mu mu; |
| 102 | |
| 103 | /** active connection */ |
| 104 | connection *active; |
Craig Tiller | df91ba5 | 2015-06-29 10:55:46 -0700 | [diff] [blame] | 105 | /** version number for the active connection */ |
| 106 | size_t active_version; |
Craig Tiller | d7b68e7 | 2015-06-28 11:41:09 -0700 | [diff] [blame] | 107 | /** refcount */ |
| 108 | int refs; |
Craig Tiller | eb3b12e | 2015-06-26 14:42:49 -0700 | [diff] [blame] | 109 | /** are we connecting */ |
| 110 | int connecting; |
Craig Tiller | 5f84c84 | 2015-06-26 16:08:21 -0700 | [diff] [blame] | 111 | /** things waiting for a connection */ |
| 112 | waiting_for_connect *waiting; |
Craig Tiller | c7b5f76 | 2015-06-27 11:48:42 -0700 | [diff] [blame] | 113 | /** connectivity state tracking */ |
| 114 | grpc_connectivity_state_tracker state_tracker; |
Craig Tiller | ff3ae68 | 2015-06-29 17:44:04 -0700 | [diff] [blame] | 115 | |
| 116 | /** next connect attempt time */ |
| 117 | gpr_timespec next_attempt; |
| 118 | /** amount to backoff each failure */ |
| 119 | gpr_timespec backoff_delta; |
| 120 | /** do we have an active alarm? */ |
| 121 | int have_alarm; |
| 122 | /** our alarm */ |
| 123 | grpc_alarm alarm; |
Craig Tiller | 2595ab7 | 2015-06-25 15:26:00 -0700 | [diff] [blame] | 124 | }; |
| 125 | |
| 126 | struct grpc_subchannel_call { |
Craig Tiller | eb3b12e | 2015-06-26 14:42:49 -0700 | [diff] [blame] | 127 | connection *connection; |
Craig Tiller | 2595ab7 | 2015-06-25 15:26:00 -0700 | [diff] [blame] | 128 | gpr_refcount refs; |
| 129 | }; |
| 130 | |
Craig Tiller | 04c5d4b | 2015-06-26 17:21:41 -0700 | [diff] [blame] | 131 | #define SUBCHANNEL_CALL_TO_CALL_STACK(call) ((grpc_call_stack *)((call) + 1)) |
| 132 | #define CHANNEL_STACK_FROM_CONNECTION(con) ((grpc_channel_stack *)((con) + 1)) |
Craig Tiller | 2595ab7 | 2015-06-25 15:26:00 -0700 | [diff] [blame] | 133 | |
Craig Tiller | abf3638 | 2015-06-29 16:13:27 -0700 | [diff] [blame] | 134 | static grpc_subchannel_call *create_call(connection *con); |
Craig Tiller | 5f84c84 | 2015-06-26 16:08:21 -0700 | [diff] [blame] | 135 | static void connectivity_state_changed_locked(grpc_subchannel *c); |
| 136 | static grpc_connectivity_state compute_connectivity_locked(grpc_subchannel *c); |
| 137 | static gpr_timespec compute_connect_deadline(grpc_subchannel *c); |
Craig Tiller | ff54c92 | 2015-06-26 16:57:20 -0700 | [diff] [blame] | 138 | static void subchannel_connected(void *subchannel, int iomgr_success); |
Craig Tiller | eb3b12e | 2015-06-26 14:42:49 -0700 | [diff] [blame] | 139 | |
Craig Tiller | 079a11b | 2015-06-30 10:07:15 -0700 | [diff] [blame] | 140 | static void subchannel_ref_locked( |
| 141 | grpc_subchannel *c GRPC_SUBCHANNEL_REF_EXTRA_ARGS); |
| 142 | static int subchannel_unref_locked( |
| 143 | grpc_subchannel *c GRPC_SUBCHANNEL_REF_EXTRA_ARGS) GRPC_MUST_USE_RESULT; |
Craig Tiller | c396753 | 2015-06-29 14:59:38 -0700 | [diff] [blame] | 144 | static void connection_ref_locked(connection *c GRPC_SUBCHANNEL_REF_EXTRA_ARGS); |
Craig Tiller | 079a11b | 2015-06-30 10:07:15 -0700 | [diff] [blame] | 145 | static grpc_subchannel *connection_unref_locked( |
| 146 | connection *c GRPC_SUBCHANNEL_REF_EXTRA_ARGS) GRPC_MUST_USE_RESULT; |
Craig Tiller | d7b68e7 | 2015-06-28 11:41:09 -0700 | [diff] [blame] | 147 | static void subchannel_destroy(grpc_subchannel *c); |
| 148 | |
Craig Tiller | c396753 | 2015-06-29 14:59:38 -0700 | [diff] [blame] | 149 | #ifdef GRPC_SUBCHANNEL_REFCOUNT_DEBUG |
Craig Tiller | 079a11b | 2015-06-30 10:07:15 -0700 | [diff] [blame] | 150 | #define SUBCHANNEL_REF_LOCKED(p, r) \ |
| 151 | subchannel_ref_locked((p), __FILE__, __LINE__, (r)) |
| 152 | #define SUBCHANNEL_UNREF_LOCKED(p, r) \ |
| 153 | subchannel_unref_locked((p), __FILE__, __LINE__, (r)) |
| 154 | #define CONNECTION_REF_LOCKED(p, r) \ |
| 155 | connection_ref_locked((p), __FILE__, __LINE__, (r)) |
| 156 | #define CONNECTION_UNREF_LOCKED(p, r) \ |
| 157 | connection_unref_locked((p), __FILE__, __LINE__, (r)) |
Craig Tiller | c396753 | 2015-06-29 14:59:38 -0700 | [diff] [blame] | 158 | #define REF_PASS_ARGS , file, line, reason |
Craig Tiller | 079a11b | 2015-06-30 10:07:15 -0700 | [diff] [blame] | 159 | #define REF_LOG(name, p) \ |
| 160 | gpr_log(file, line, GPR_LOG_SEVERITY_DEBUG, "%s: %p ref %d -> %d %s", \ |
| 161 | (name), (p), (p)->refs, (p)->refs + 1, reason) |
| 162 | #define UNREF_LOG(name, p) \ |
| 163 | gpr_log(file, line, GPR_LOG_SEVERITY_DEBUG, "%s: %p unref %d -> %d %s", \ |
| 164 | (name), (p), (p)->refs, (p)->refs - 1, reason) |
Craig Tiller | c396753 | 2015-06-29 14:59:38 -0700 | [diff] [blame] | 165 | #else |
| 166 | #define SUBCHANNEL_REF_LOCKED(p, r) subchannel_ref_locked((p)) |
| 167 | #define SUBCHANNEL_UNREF_LOCKED(p, r) subchannel_unref_locked((p)) |
Craig Tiller | 11bf14e | 2015-06-29 16:35:41 -0700 | [diff] [blame] | 168 | #define CONNECTION_REF_LOCKED(p, r) connection_ref_locked((p)) |
| 169 | #define CONNECTION_UNREF_LOCKED(p, r) connection_unref_locked((p)) |
Craig Tiller | c396753 | 2015-06-29 14:59:38 -0700 | [diff] [blame] | 170 | #define REF_PASS_ARGS |
Craig Tiller | 11bf14e | 2015-06-29 16:35:41 -0700 | [diff] [blame] | 171 | #define REF_LOG(name, p) \ |
| 172 | do { \ |
| 173 | } while (0) |
| 174 | #define UNREF_LOG(name, p) \ |
| 175 | do { \ |
| 176 | } while (0) |
Craig Tiller | c396753 | 2015-06-29 14:59:38 -0700 | [diff] [blame] | 177 | #endif |
| 178 | |
Craig Tiller | 2595ab7 | 2015-06-25 15:26:00 -0700 | [diff] [blame] | 179 | /* |
Craig Tiller | ca3e9d3 | 2015-06-27 18:37:27 -0700 | [diff] [blame] | 180 | * connection implementation |
| 181 | */ |
| 182 | |
Craig Tiller | d7b68e7 | 2015-06-28 11:41:09 -0700 | [diff] [blame] | 183 | static void connection_destroy(connection *c) { |
Craig Tiller | 4ab82d2 | 2015-06-29 09:40:33 -0700 | [diff] [blame] | 184 | GPR_ASSERT(c->refs == 0); |
| 185 | grpc_channel_stack_destroy(CHANNEL_STACK_FROM_CONNECTION(c)); |
Craig Tiller | d7b68e7 | 2015-06-28 11:41:09 -0700 | [diff] [blame] | 186 | gpr_free(c); |
Craig Tiller | ca3e9d3 | 2015-06-27 18:37:27 -0700 | [diff] [blame] | 187 | } |
| 188 | |
Craig Tiller | 079a11b | 2015-06-30 10:07:15 -0700 | [diff] [blame] | 189 | static void connection_ref_locked( |
| 190 | connection *c GRPC_SUBCHANNEL_REF_EXTRA_ARGS) { |
Craig Tiller | c396753 | 2015-06-29 14:59:38 -0700 | [diff] [blame] | 191 | REF_LOG("CONNECTION", c); |
| 192 | subchannel_ref_locked(c->subchannel REF_PASS_ARGS); |
Craig Tiller | 4ab82d2 | 2015-06-29 09:40:33 -0700 | [diff] [blame] | 193 | ++c->refs; |
Craig Tiller | d7b68e7 | 2015-06-28 11:41:09 -0700 | [diff] [blame] | 194 | } |
| 195 | |
Craig Tiller | 079a11b | 2015-06-30 10:07:15 -0700 | [diff] [blame] | 196 | static grpc_subchannel *connection_unref_locked( |
| 197 | connection *c GRPC_SUBCHANNEL_REF_EXTRA_ARGS) { |
Craig Tiller | 4ab82d2 | 2015-06-29 09:40:33 -0700 | [diff] [blame] | 198 | grpc_subchannel *destroy = NULL; |
Craig Tiller | c396753 | 2015-06-29 14:59:38 -0700 | [diff] [blame] | 199 | UNREF_LOG("CONNECTION", c); |
| 200 | if (subchannel_unref_locked(c->subchannel REF_PASS_ARGS)) { |
Craig Tiller | 4ab82d2 | 2015-06-29 09:40:33 -0700 | [diff] [blame] | 201 | destroy = c->subchannel; |
| 202 | } |
Craig Tiller | d7b68e7 | 2015-06-28 11:41:09 -0700 | [diff] [blame] | 203 | if (--c->refs == 0 && c->subchannel->active != c) { |
Craig Tiller | 4ab82d2 | 2015-06-29 09:40:33 -0700 | [diff] [blame] | 204 | connection_destroy(c); |
Craig Tiller | ca3e9d3 | 2015-06-27 18:37:27 -0700 | [diff] [blame] | 205 | } |
Craig Tiller | d7b68e7 | 2015-06-28 11:41:09 -0700 | [diff] [blame] | 206 | return destroy; |
Craig Tiller | ca3e9d3 | 2015-06-27 18:37:27 -0700 | [diff] [blame] | 207 | } |
| 208 | |
| 209 | /* |
Craig Tiller | 2595ab7 | 2015-06-25 15:26:00 -0700 | [diff] [blame] | 210 | * grpc_subchannel implementation |
| 211 | */ |
| 212 | |
Craig Tiller | 079a11b | 2015-06-30 10:07:15 -0700 | [diff] [blame] | 213 | static void subchannel_ref_locked( |
| 214 | grpc_subchannel *c GRPC_SUBCHANNEL_REF_EXTRA_ARGS) { |
Craig Tiller | c396753 | 2015-06-29 14:59:38 -0700 | [diff] [blame] | 215 | REF_LOG("SUBCHANNEL", c); |
Craig Tiller | 079a11b | 2015-06-30 10:07:15 -0700 | [diff] [blame] | 216 | ++c->refs; |
Craig Tiller | c396753 | 2015-06-29 14:59:38 -0700 | [diff] [blame] | 217 | } |
Craig Tiller | 2595ab7 | 2015-06-25 15:26:00 -0700 | [diff] [blame] | 218 | |
Craig Tiller | 079a11b | 2015-06-30 10:07:15 -0700 | [diff] [blame] | 219 | static int subchannel_unref_locked( |
| 220 | grpc_subchannel *c GRPC_SUBCHANNEL_REF_EXTRA_ARGS) { |
Craig Tiller | c396753 | 2015-06-29 14:59:38 -0700 | [diff] [blame] | 221 | UNREF_LOG("SUBCHANNEL", c); |
Craig Tiller | 4ab82d2 | 2015-06-29 09:40:33 -0700 | [diff] [blame] | 222 | return --c->refs == 0; |
Craig Tiller | d7b68e7 | 2015-06-28 11:41:09 -0700 | [diff] [blame] | 223 | } |
| 224 | |
Craig Tiller | c396753 | 2015-06-29 14:59:38 -0700 | [diff] [blame] | 225 | void grpc_subchannel_ref(grpc_subchannel *c GRPC_SUBCHANNEL_REF_EXTRA_ARGS) { |
Craig Tiller | 4ab82d2 | 2015-06-29 09:40:33 -0700 | [diff] [blame] | 226 | gpr_mu_lock(&c->mu); |
Craig Tiller | c396753 | 2015-06-29 14:59:38 -0700 | [diff] [blame] | 227 | subchannel_ref_locked(c REF_PASS_ARGS); |
Craig Tiller | 4ab82d2 | 2015-06-29 09:40:33 -0700 | [diff] [blame] | 228 | gpr_mu_unlock(&c->mu); |
Craig Tiller | d7b68e7 | 2015-06-28 11:41:09 -0700 | [diff] [blame] | 229 | } |
| 230 | |
Craig Tiller | c396753 | 2015-06-29 14:59:38 -0700 | [diff] [blame] | 231 | void grpc_subchannel_unref(grpc_subchannel *c GRPC_SUBCHANNEL_REF_EXTRA_ARGS) { |
Craig Tiller | 4ab82d2 | 2015-06-29 09:40:33 -0700 | [diff] [blame] | 232 | int destroy; |
| 233 | gpr_mu_lock(&c->mu); |
Craig Tiller | c396753 | 2015-06-29 14:59:38 -0700 | [diff] [blame] | 234 | destroy = subchannel_unref_locked(c REF_PASS_ARGS); |
Craig Tiller | 4ab82d2 | 2015-06-29 09:40:33 -0700 | [diff] [blame] | 235 | gpr_mu_unlock(&c->mu); |
| 236 | if (destroy) subchannel_destroy(c); |
Craig Tiller | d7b68e7 | 2015-06-28 11:41:09 -0700 | [diff] [blame] | 237 | } |
| 238 | |
| 239 | static void subchannel_destroy(grpc_subchannel *c) { |
Craig Tiller | 4ab82d2 | 2015-06-29 09:40:33 -0700 | [diff] [blame] | 240 | if (c->active != NULL) { |
| 241 | connection_destroy(c->active); |
| 242 | } |
Craig Tiller | d7b68e7 | 2015-06-28 11:41:09 -0700 | [diff] [blame] | 243 | gpr_free(c->filters); |
| 244 | grpc_channel_args_destroy(c->args); |
| 245 | gpr_free(c->addr); |
| 246 | grpc_mdctx_unref(c->mdctx); |
| 247 | grpc_pollset_set_destroy(&c->pollset_set); |
| 248 | grpc_connectivity_state_destroy(&c->state_tracker); |
Craig Tiller | c396753 | 2015-06-29 14:59:38 -0700 | [diff] [blame] | 249 | grpc_connector_unref(c->connector); |
Craig Tiller | d7b68e7 | 2015-06-28 11:41:09 -0700 | [diff] [blame] | 250 | gpr_free(c); |
Craig Tiller | 2595ab7 | 2015-06-25 15:26:00 -0700 | [diff] [blame] | 251 | } |
| 252 | |
Craig Tiller | 5f84c84 | 2015-06-26 16:08:21 -0700 | [diff] [blame] | 253 | void grpc_subchannel_add_interested_party(grpc_subchannel *c, |
| 254 | grpc_pollset *pollset) { |
| 255 | grpc_pollset_set_add_pollset(&c->pollset_set, pollset); |
| 256 | } |
| 257 | |
| 258 | void grpc_subchannel_del_interested_party(grpc_subchannel *c, |
| 259 | grpc_pollset *pollset) { |
| 260 | grpc_pollset_set_del_pollset(&c->pollset_set, pollset); |
| 261 | } |
| 262 | |
Craig Tiller | f7afa1f | 2015-06-26 09:02:20 -0700 | [diff] [blame] | 263 | grpc_subchannel *grpc_subchannel_create(grpc_connector *connector, |
| 264 | grpc_subchannel_args *args) { |
| 265 | grpc_subchannel *c = gpr_malloc(sizeof(*c)); |
| 266 | memset(c, 0, sizeof(*c)); |
Craig Tiller | d7b68e7 | 2015-06-28 11:41:09 -0700 | [diff] [blame] | 267 | c->refs = 1; |
Craig Tiller | f7afa1f | 2015-06-26 09:02:20 -0700 | [diff] [blame] | 268 | c->connector = connector; |
| 269 | grpc_connector_ref(c->connector); |
Craig Tiller | 5945ee1 | 2015-06-27 10:36:09 -0700 | [diff] [blame] | 270 | c->num_filters = args->filter_count; |
| 271 | c->filters = gpr_malloc(sizeof(grpc_channel_filter *) * c->num_filters); |
Craig Tiller | f7afa1f | 2015-06-26 09:02:20 -0700 | [diff] [blame] | 272 | memcpy(c->filters, args->filters, |
Craig Tiller | 5945ee1 | 2015-06-27 10:36:09 -0700 | [diff] [blame] | 273 | sizeof(grpc_channel_filter *) * c->num_filters); |
Craig Tiller | f7afa1f | 2015-06-26 09:02:20 -0700 | [diff] [blame] | 274 | c->addr = gpr_malloc(args->addr_len); |
| 275 | memcpy(c->addr, args->addr, args->addr_len); |
| 276 | c->addr_len = args->addr_len; |
Craig Tiller | eb3b12e | 2015-06-26 14:42:49 -0700 | [diff] [blame] | 277 | c->args = grpc_channel_args_copy(args->args); |
Craig Tiller | 5f84c84 | 2015-06-26 16:08:21 -0700 | [diff] [blame] | 278 | c->mdctx = args->mdctx; |
Craig Tiller | 9846503 | 2015-06-29 14:36:42 -0700 | [diff] [blame] | 279 | c->master = args->master; |
Craig Tiller | 5f84c84 | 2015-06-26 16:08:21 -0700 | [diff] [blame] | 280 | grpc_mdctx_ref(c->mdctx); |
Craig Tiller | ff54c92 | 2015-06-26 16:57:20 -0700 | [diff] [blame] | 281 | grpc_pollset_set_init(&c->pollset_set); |
| 282 | grpc_iomgr_closure_init(&c->connected, subchannel_connected, c); |
Craig Tiller | c7b5f76 | 2015-06-27 11:48:42 -0700 | [diff] [blame] | 283 | grpc_connectivity_state_init(&c->state_tracker, GRPC_CHANNEL_IDLE); |
Craig Tiller | eb3b12e | 2015-06-26 14:42:49 -0700 | [diff] [blame] | 284 | gpr_mu_init(&c->mu); |
Craig Tiller | f7afa1f | 2015-06-26 09:02:20 -0700 | [diff] [blame] | 285 | return c; |
| 286 | } |
| 287 | |
Craig Tiller | ff3ae68 | 2015-06-29 17:44:04 -0700 | [diff] [blame] | 288 | static void continue_connect(grpc_subchannel *c) { |
Craig Tiller | 4ab82d2 | 2015-06-29 09:40:33 -0700 | [diff] [blame] | 289 | grpc_connect_in_args args; |
Craig Tiller | 04c5d4b | 2015-06-26 17:21:41 -0700 | [diff] [blame] | 290 | |
Craig Tiller | 4ab82d2 | 2015-06-29 09:40:33 -0700 | [diff] [blame] | 291 | args.interested_parties = &c->pollset_set; |
| 292 | args.addr = c->addr; |
| 293 | args.addr_len = c->addr_len; |
| 294 | args.deadline = compute_connect_deadline(c); |
| 295 | args.channel_args = c->args; |
| 296 | args.metadata_context = c->mdctx; |
Craig Tiller | 04c5d4b | 2015-06-26 17:21:41 -0700 | [diff] [blame] | 297 | |
Craig Tiller | 4ab82d2 | 2015-06-29 09:40:33 -0700 | [diff] [blame] | 298 | grpc_connector_connect(c->connector, &args, &c->connecting_result, |
| 299 | &c->connected); |
Craig Tiller | 04c5d4b | 2015-06-26 17:21:41 -0700 | [diff] [blame] | 300 | } |
| 301 | |
Craig Tiller | ff3ae68 | 2015-06-29 17:44:04 -0700 | [diff] [blame] | 302 | static void start_connect(grpc_subchannel *c) { |
Craig Tiller | a25ca0b | 2015-07-07 13:54:12 -0700 | [diff] [blame] | 303 | gpr_timespec now = gpr_now(GPR_CLOCK_REALTIME); |
Craig Tiller | ff3ae68 | 2015-06-29 17:44:04 -0700 | [diff] [blame] | 304 | c->next_attempt = now; |
Craig Tiller | 58bbc86 | 2015-07-13 09:51:17 -0700 | [diff] [blame] | 305 | c->backoff_delta = gpr_time_from_seconds(1, GPR_TIMESPAN); |
Craig Tiller | ff3ae68 | 2015-06-29 17:44:04 -0700 | [diff] [blame] | 306 | |
| 307 | continue_connect(c); |
| 308 | } |
| 309 | |
Craig Tiller | df91ba5 | 2015-06-29 10:55:46 -0700 | [diff] [blame] | 310 | static void continue_creating_call(void *arg, int iomgr_success) { |
| 311 | waiting_for_connect *w4c = arg; |
Craig Tiller | abf3638 | 2015-06-29 16:13:27 -0700 | [diff] [blame] | 312 | grpc_subchannel_create_call(w4c->subchannel, w4c->pollset, w4c->target, |
Craig Tiller | f62d6fc | 2015-06-29 10:55:59 -0700 | [diff] [blame] | 313 | w4c->notify); |
Craig Tiller | c396753 | 2015-06-29 14:59:38 -0700 | [diff] [blame] | 314 | GRPC_SUBCHANNEL_UNREF(w4c->subchannel, "waiting_for_connect"); |
Craig Tiller | df91ba5 | 2015-06-29 10:55:46 -0700 | [diff] [blame] | 315 | gpr_free(w4c); |
| 316 | } |
| 317 | |
Craig Tiller | abf3638 | 2015-06-29 16:13:27 -0700 | [diff] [blame] | 318 | void grpc_subchannel_create_call(grpc_subchannel *c, grpc_pollset *pollset, |
Craig Tiller | eb3b12e | 2015-06-26 14:42:49 -0700 | [diff] [blame] | 319 | grpc_subchannel_call **target, |
| 320 | grpc_iomgr_closure *notify) { |
| 321 | connection *con; |
| 322 | gpr_mu_lock(&c->mu); |
| 323 | if (c->active != NULL) { |
| 324 | con = c->active; |
Craig Tiller | c396753 | 2015-06-29 14:59:38 -0700 | [diff] [blame] | 325 | CONNECTION_REF_LOCKED(con, "call"); |
Craig Tiller | eb3b12e | 2015-06-26 14:42:49 -0700 | [diff] [blame] | 326 | gpr_mu_unlock(&c->mu); |
| 327 | |
Craig Tiller | abf3638 | 2015-06-29 16:13:27 -0700 | [diff] [blame] | 328 | *target = create_call(con); |
Craig Tiller | eb3b12e | 2015-06-26 14:42:49 -0700 | [diff] [blame] | 329 | notify->cb(notify->cb_arg, 1); |
| 330 | } else { |
Craig Tiller | 5f84c84 | 2015-06-26 16:08:21 -0700 | [diff] [blame] | 331 | waiting_for_connect *w4c = gpr_malloc(sizeof(*w4c)); |
| 332 | w4c->next = c->waiting; |
| 333 | w4c->notify = notify; |
Craig Tiller | abf3638 | 2015-06-29 16:13:27 -0700 | [diff] [blame] | 334 | w4c->pollset = pollset; |
Craig Tiller | 5f84c84 | 2015-06-26 16:08:21 -0700 | [diff] [blame] | 335 | w4c->target = target; |
Craig Tiller | df91ba5 | 2015-06-29 10:55:46 -0700 | [diff] [blame] | 336 | w4c->subchannel = c; |
Craig Tiller | 9846503 | 2015-06-29 14:36:42 -0700 | [diff] [blame] | 337 | /* released when clearing w4c */ |
Craig Tiller | c396753 | 2015-06-29 14:59:38 -0700 | [diff] [blame] | 338 | SUBCHANNEL_REF_LOCKED(c, "waiting_for_connect"); |
Craig Tiller | df91ba5 | 2015-06-29 10:55:46 -0700 | [diff] [blame] | 339 | grpc_iomgr_closure_init(&w4c->continuation, continue_creating_call, w4c); |
Craig Tiller | 5f84c84 | 2015-06-26 16:08:21 -0700 | [diff] [blame] | 340 | c->waiting = w4c; |
Craig Tiller | abf3638 | 2015-06-29 16:13:27 -0700 | [diff] [blame] | 341 | grpc_subchannel_add_interested_party(c, pollset); |
Craig Tiller | eb3b12e | 2015-06-26 14:42:49 -0700 | [diff] [blame] | 342 | if (!c->connecting) { |
| 343 | c->connecting = 1; |
Craig Tiller | 5f84c84 | 2015-06-26 16:08:21 -0700 | [diff] [blame] | 344 | connectivity_state_changed_locked(c); |
Craig Tiller | 9846503 | 2015-06-29 14:36:42 -0700 | [diff] [blame] | 345 | /* released by connection */ |
Craig Tiller | c396753 | 2015-06-29 14:59:38 -0700 | [diff] [blame] | 346 | SUBCHANNEL_REF_LOCKED(c, "connecting"); |
Craig Tiller | eb3b12e | 2015-06-26 14:42:49 -0700 | [diff] [blame] | 347 | gpr_mu_unlock(&c->mu); |
| 348 | |
Craig Tiller | 04c5d4b | 2015-06-26 17:21:41 -0700 | [diff] [blame] | 349 | start_connect(c); |
Craig Tiller | eb3b12e | 2015-06-26 14:42:49 -0700 | [diff] [blame] | 350 | } else { |
| 351 | gpr_mu_unlock(&c->mu); |
| 352 | } |
| 353 | } |
| 354 | } |
| 355 | |
Craig Tiller | 5f84c84 | 2015-06-26 16:08:21 -0700 | [diff] [blame] | 356 | grpc_connectivity_state grpc_subchannel_check_connectivity(grpc_subchannel *c) { |
| 357 | grpc_connectivity_state state; |
| 358 | gpr_mu_lock(&c->mu); |
Craig Tiller | c7b5f76 | 2015-06-27 11:48:42 -0700 | [diff] [blame] | 359 | state = grpc_connectivity_state_check(&c->state_tracker); |
Craig Tiller | 5f84c84 | 2015-06-26 16:08:21 -0700 | [diff] [blame] | 360 | gpr_mu_unlock(&c->mu); |
| 361 | return state; |
| 362 | } |
| 363 | |
| 364 | void grpc_subchannel_notify_on_state_change(grpc_subchannel *c, |
| 365 | grpc_connectivity_state *state, |
| 366 | grpc_iomgr_closure *notify) { |
Craig Tiller | 5f84c84 | 2015-06-26 16:08:21 -0700 | [diff] [blame] | 367 | int do_connect = 0; |
Craig Tiller | 5f84c84 | 2015-06-26 16:08:21 -0700 | [diff] [blame] | 368 | gpr_mu_lock(&c->mu); |
Craig Tiller | 4ab82d2 | 2015-06-29 09:40:33 -0700 | [diff] [blame] | 369 | if (grpc_connectivity_state_notify_on_state_change(&c->state_tracker, state, |
| 370 | notify)) { |
| 371 | do_connect = 1; |
Craig Tiller | 5f84c84 | 2015-06-26 16:08:21 -0700 | [diff] [blame] | 372 | c->connecting = 1; |
Craig Tiller | 9846503 | 2015-06-29 14:36:42 -0700 | [diff] [blame] | 373 | /* released by connection */ |
Craig Tiller | c396753 | 2015-06-29 14:59:38 -0700 | [diff] [blame] | 374 | SUBCHANNEL_REF_LOCKED(c, "connecting"); |
Craig Tiller | 11bf14e | 2015-06-29 16:35:41 -0700 | [diff] [blame] | 375 | connectivity_state_changed_locked(c); |
Craig Tiller | 5f84c84 | 2015-06-26 16:08:21 -0700 | [diff] [blame] | 376 | } |
Craig Tiller | c7b5f76 | 2015-06-27 11:48:42 -0700 | [diff] [blame] | 377 | gpr_mu_unlock(&c->mu); |
Craig Tiller | 5f84c84 | 2015-06-26 16:08:21 -0700 | [diff] [blame] | 378 | if (do_connect) { |
Craig Tiller | 4ab82d2 | 2015-06-29 09:40:33 -0700 | [diff] [blame] | 379 | start_connect(c); |
Craig Tiller | 5f84c84 | 2015-06-26 16:08:21 -0700 | [diff] [blame] | 380 | } |
| 381 | } |
| 382 | |
Craig Tiller | 4ab82d2 | 2015-06-29 09:40:33 -0700 | [diff] [blame] | 383 | void grpc_subchannel_process_transport_op(grpc_subchannel *c, |
| 384 | grpc_transport_op *op) { |
Craig Tiller | b6fbf1d | 2015-06-29 15:25:49 -0700 | [diff] [blame] | 385 | connection *con = NULL; |
| 386 | grpc_subchannel *destroy; |
Craig Tiller | ff3ae68 | 2015-06-29 17:44:04 -0700 | [diff] [blame] | 387 | int cancel_alarm = 0; |
Craig Tiller | b6fbf1d | 2015-06-29 15:25:49 -0700 | [diff] [blame] | 388 | gpr_mu_lock(&c->mu); |
| 389 | if (op->disconnect) { |
| 390 | c->disconnected = 1; |
Craig Tiller | 11bf14e | 2015-06-29 16:35:41 -0700 | [diff] [blame] | 391 | connectivity_state_changed_locked(c); |
Craig Tiller | ff3ae68 | 2015-06-29 17:44:04 -0700 | [diff] [blame] | 392 | if (c->have_alarm) { |
| 393 | cancel_alarm = 1; |
| 394 | } |
Craig Tiller | b6fbf1d | 2015-06-29 15:25:49 -0700 | [diff] [blame] | 395 | } |
| 396 | if (c->active != NULL) { |
| 397 | con = c->active; |
| 398 | CONNECTION_REF_LOCKED(con, "transport-op"); |
| 399 | } |
| 400 | gpr_mu_unlock(&c->mu); |
| 401 | |
| 402 | if (con != NULL) { |
| 403 | grpc_channel_stack *channel_stack = CHANNEL_STACK_FROM_CONNECTION(con); |
Craig Tiller | 079a11b | 2015-06-30 10:07:15 -0700 | [diff] [blame] | 404 | grpc_channel_element *top_elem = |
| 405 | grpc_channel_stack_element(channel_stack, 0); |
Craig Tiller | b6fbf1d | 2015-06-29 15:25:49 -0700 | [diff] [blame] | 406 | top_elem->filter->start_transport_op(top_elem, op); |
| 407 | |
| 408 | gpr_mu_lock(&c->mu); |
| 409 | destroy = CONNECTION_UNREF_LOCKED(con, "transport-op"); |
| 410 | gpr_mu_unlock(&c->mu); |
| 411 | if (destroy) { |
| 412 | subchannel_destroy(destroy); |
| 413 | } |
| 414 | } |
Craig Tiller | ff3ae68 | 2015-06-29 17:44:04 -0700 | [diff] [blame] | 415 | |
| 416 | if (cancel_alarm) { |
| 417 | grpc_alarm_cancel(&c->alarm); |
| 418 | } |
Craig Tiller | df91ba5 | 2015-06-29 10:55:46 -0700 | [diff] [blame] | 419 | } |
| 420 | |
| 421 | static void on_state_changed(void *p, int iomgr_success) { |
| 422 | state_watcher *sw = p; |
| 423 | grpc_subchannel *c = sw->subchannel; |
| 424 | gpr_mu *mu = &c->mu; |
| 425 | int destroy; |
| 426 | grpc_transport_op op; |
| 427 | grpc_channel_element *elem; |
| 428 | connection *destroy_connection = NULL; |
Craig Tiller | df91ba5 | 2015-06-29 10:55:46 -0700 | [diff] [blame] | 429 | |
| 430 | gpr_mu_lock(mu); |
| 431 | |
| 432 | /* if we failed or there is a version number mismatch, just leave |
| 433 | this closure */ |
| 434 | if (!iomgr_success || sw->subchannel->active_version != sw->version) { |
| 435 | goto done; |
| 436 | } |
| 437 | |
| 438 | switch (sw->connectivity_state) { |
| 439 | case GRPC_CHANNEL_CONNECTING: |
| 440 | case GRPC_CHANNEL_READY: |
| 441 | case GRPC_CHANNEL_IDLE: |
| 442 | /* all is still good: keep watching */ |
| 443 | memset(&op, 0, sizeof(op)); |
| 444 | op.connectivity_state = &sw->connectivity_state; |
| 445 | op.on_connectivity_state_change = &sw->closure; |
Craig Tiller | f62d6fc | 2015-06-29 10:55:59 -0700 | [diff] [blame] | 446 | elem = grpc_channel_stack_element( |
| 447 | CHANNEL_STACK_FROM_CONNECTION(c->active), 0); |
Craig Tiller | df91ba5 | 2015-06-29 10:55:46 -0700 | [diff] [blame] | 448 | elem->filter->start_transport_op(elem, &op); |
| 449 | /* early out */ |
| 450 | gpr_mu_unlock(mu); |
| 451 | return; |
| 452 | case GRPC_CHANNEL_FATAL_FAILURE: |
Craig Tiller | 49924e0 | 2015-06-29 22:42:33 -0700 | [diff] [blame] | 453 | case GRPC_CHANNEL_TRANSIENT_FAILURE: |
Craig Tiller | df91ba5 | 2015-06-29 10:55:46 -0700 | [diff] [blame] | 454 | /* things have gone wrong, deactivate and enter idle */ |
| 455 | if (sw->subchannel->active->refs == 0) { |
| 456 | destroy_connection = sw->subchannel->active; |
| 457 | } |
| 458 | sw->subchannel->active = NULL; |
Craig Tiller | 11bf14e | 2015-06-29 16:35:41 -0700 | [diff] [blame] | 459 | grpc_connectivity_state_set(&c->state_tracker, |
| 460 | GRPC_CHANNEL_TRANSIENT_FAILURE); |
Craig Tiller | df91ba5 | 2015-06-29 10:55:46 -0700 | [diff] [blame] | 461 | break; |
Craig Tiller | df91ba5 | 2015-06-29 10:55:46 -0700 | [diff] [blame] | 462 | } |
| 463 | |
| 464 | done: |
Craig Tiller | 11bf14e | 2015-06-29 16:35:41 -0700 | [diff] [blame] | 465 | connectivity_state_changed_locked(c); |
Craig Tiller | b6fbf1d | 2015-06-29 15:25:49 -0700 | [diff] [blame] | 466 | destroy = SUBCHANNEL_UNREF_LOCKED(c, "state_watcher"); |
Craig Tiller | df91ba5 | 2015-06-29 10:55:46 -0700 | [diff] [blame] | 467 | gpr_free(sw); |
| 468 | gpr_mu_unlock(mu); |
Craig Tiller | df91ba5 | 2015-06-29 10:55:46 -0700 | [diff] [blame] | 469 | if (destroy) { |
| 470 | subchannel_destroy(c); |
| 471 | } |
| 472 | if (destroy_connection != NULL) { |
| 473 | connection_destroy(destroy_connection); |
| 474 | } |
Craig Tiller | c7b5f76 | 2015-06-27 11:48:42 -0700 | [diff] [blame] | 475 | } |
| 476 | |
Craig Tiller | ff54c92 | 2015-06-26 16:57:20 -0700 | [diff] [blame] | 477 | static void publish_transport(grpc_subchannel *c) { |
Craig Tiller | 4ab82d2 | 2015-06-29 09:40:33 -0700 | [diff] [blame] | 478 | size_t channel_stack_size; |
| 479 | connection *con; |
| 480 | grpc_channel_stack *stk; |
| 481 | size_t num_filters; |
| 482 | const grpc_channel_filter **filters; |
| 483 | waiting_for_connect *w4c; |
Craig Tiller | df91ba5 | 2015-06-29 10:55:46 -0700 | [diff] [blame] | 484 | grpc_transport_op op; |
| 485 | state_watcher *sw; |
| 486 | connection *destroy_connection = NULL; |
| 487 | grpc_channel_element *elem; |
Craig Tiller | 5945ee1 | 2015-06-27 10:36:09 -0700 | [diff] [blame] | 488 | |
Craig Tiller | df91ba5 | 2015-06-29 10:55:46 -0700 | [diff] [blame] | 489 | /* build final filter list */ |
Craig Tiller | 4ab82d2 | 2015-06-29 09:40:33 -0700 | [diff] [blame] | 490 | num_filters = c->num_filters + c->connecting_result.num_filters + 1; |
| 491 | filters = gpr_malloc(sizeof(*filters) * num_filters); |
| 492 | memcpy(filters, c->filters, sizeof(*filters) * c->num_filters); |
| 493 | memcpy(filters + c->num_filters, c->connecting_result.filters, |
| 494 | sizeof(*filters) * c->connecting_result.num_filters); |
| 495 | filters[num_filters - 1] = &grpc_connected_channel_filter; |
Craig Tiller | 5945ee1 | 2015-06-27 10:36:09 -0700 | [diff] [blame] | 496 | |
Craig Tiller | df91ba5 | 2015-06-29 10:55:46 -0700 | [diff] [blame] | 497 | /* construct channel stack */ |
Craig Tiller | 4ab82d2 | 2015-06-29 09:40:33 -0700 | [diff] [blame] | 498 | channel_stack_size = grpc_channel_stack_size(filters, num_filters); |
| 499 | con = gpr_malloc(sizeof(connection) + channel_stack_size); |
| 500 | stk = (grpc_channel_stack *)(con + 1); |
Craig Tiller | 4ab82d2 | 2015-06-29 09:40:33 -0700 | [diff] [blame] | 501 | con->refs = 0; |
| 502 | con->subchannel = c; |
Craig Tiller | 079a11b | 2015-06-30 10:07:15 -0700 | [diff] [blame] | 503 | grpc_channel_stack_init(filters, num_filters, c->master, c->args, c->mdctx, |
| 504 | stk); |
Craig Tiller | 4ab82d2 | 2015-06-29 09:40:33 -0700 | [diff] [blame] | 505 | grpc_connected_channel_bind_transport(stk, c->connecting_result.transport); |
Craig Tiller | 9846503 | 2015-06-29 14:36:42 -0700 | [diff] [blame] | 506 | gpr_free(c->connecting_result.filters); |
Craig Tiller | 4ab82d2 | 2015-06-29 09:40:33 -0700 | [diff] [blame] | 507 | memset(&c->connecting_result, 0, sizeof(c->connecting_result)); |
Craig Tiller | ff54c92 | 2015-06-26 16:57:20 -0700 | [diff] [blame] | 508 | |
Craig Tiller | df91ba5 | 2015-06-29 10:55:46 -0700 | [diff] [blame] | 509 | /* initialize state watcher */ |
| 510 | sw = gpr_malloc(sizeof(*sw)); |
| 511 | grpc_iomgr_closure_init(&sw->closure, on_state_changed, sw); |
| 512 | sw->subchannel = c; |
| 513 | sw->connectivity_state = GRPC_CHANNEL_READY; |
| 514 | |
Craig Tiller | 4ab82d2 | 2015-06-29 09:40:33 -0700 | [diff] [blame] | 515 | gpr_mu_lock(&c->mu); |
Craig Tiller | df91ba5 | 2015-06-29 10:55:46 -0700 | [diff] [blame] | 516 | |
Craig Tiller | b6fbf1d | 2015-06-29 15:25:49 -0700 | [diff] [blame] | 517 | if (c->disconnected) { |
| 518 | gpr_mu_unlock(&c->mu); |
| 519 | gpr_free(sw); |
| 520 | gpr_free(filters); |
| 521 | grpc_channel_stack_destroy(stk); |
| 522 | return; |
| 523 | } |
| 524 | |
Craig Tiller | df91ba5 | 2015-06-29 10:55:46 -0700 | [diff] [blame] | 525 | /* publish */ |
| 526 | if (c->active != NULL && c->active->refs == 0) { |
| 527 | destroy_connection = c->active; |
| 528 | } |
Craig Tiller | 4ab82d2 | 2015-06-29 09:40:33 -0700 | [diff] [blame] | 529 | c->active = con; |
Craig Tiller | df91ba5 | 2015-06-29 10:55:46 -0700 | [diff] [blame] | 530 | c->active_version++; |
| 531 | sw->version = c->active_version; |
Craig Tiller | 4ab82d2 | 2015-06-29 09:40:33 -0700 | [diff] [blame] | 532 | c->connecting = 0; |
Craig Tiller | df91ba5 | 2015-06-29 10:55:46 -0700 | [diff] [blame] | 533 | |
| 534 | /* watch for changes; subchannel ref for connecting is donated |
| 535 | to the state watcher */ |
| 536 | memset(&op, 0, sizeof(op)); |
| 537 | op.connectivity_state = &sw->connectivity_state; |
| 538 | op.on_connectivity_state_change = &sw->closure; |
Craig Tiller | b6fbf1d | 2015-06-29 15:25:49 -0700 | [diff] [blame] | 539 | SUBCHANNEL_REF_LOCKED(c, "state_watcher"); |
| 540 | GPR_ASSERT(!SUBCHANNEL_UNREF_LOCKED(c, "connecting")); |
Craig Tiller | f62d6fc | 2015-06-29 10:55:59 -0700 | [diff] [blame] | 541 | elem = |
| 542 | grpc_channel_stack_element(CHANNEL_STACK_FROM_CONNECTION(c->active), 0); |
Craig Tiller | df91ba5 | 2015-06-29 10:55:46 -0700 | [diff] [blame] | 543 | elem->filter->start_transport_op(elem, &op); |
| 544 | |
| 545 | /* signal completion */ |
Craig Tiller | 4ab82d2 | 2015-06-29 09:40:33 -0700 | [diff] [blame] | 546 | connectivity_state_changed_locked(c); |
| 547 | while ((w4c = c->waiting)) { |
Craig Tiller | df91ba5 | 2015-06-29 10:55:46 -0700 | [diff] [blame] | 548 | c->waiting = w4c->next; |
| 549 | grpc_iomgr_add_callback(&w4c->continuation); |
Craig Tiller | 4ab82d2 | 2015-06-29 09:40:33 -0700 | [diff] [blame] | 550 | } |
Craig Tiller | df91ba5 | 2015-06-29 10:55:46 -0700 | [diff] [blame] | 551 | |
Craig Tiller | 4ab82d2 | 2015-06-29 09:40:33 -0700 | [diff] [blame] | 552 | gpr_mu_unlock(&c->mu); |
Craig Tiller | 5945ee1 | 2015-06-27 10:36:09 -0700 | [diff] [blame] | 553 | |
Craig Tiller | 4ab82d2 | 2015-06-29 09:40:33 -0700 | [diff] [blame] | 554 | gpr_free(filters); |
Craig Tiller | d7b68e7 | 2015-06-28 11:41:09 -0700 | [diff] [blame] | 555 | |
Craig Tiller | df91ba5 | 2015-06-29 10:55:46 -0700 | [diff] [blame] | 556 | if (destroy_connection != NULL) { |
| 557 | connection_destroy(destroy_connection); |
Craig Tiller | 4ab82d2 | 2015-06-29 09:40:33 -0700 | [diff] [blame] | 558 | } |
| 559 | } |
Craig Tiller | ff54c92 | 2015-06-26 16:57:20 -0700 | [diff] [blame] | 560 | |
Craig Tiller | ff3ae68 | 2015-06-29 17:44:04 -0700 | [diff] [blame] | 561 | static void on_alarm(void *arg, int iomgr_success) { |
| 562 | grpc_subchannel *c = arg; |
| 563 | gpr_mu_lock(&c->mu); |
| 564 | c->have_alarm = 0; |
Craig Tiller | d2cc459 | 2015-07-01 07:50:47 -0700 | [diff] [blame] | 565 | if (c->disconnected) { |
| 566 | iomgr_success = 0; |
| 567 | } |
Craig Tiller | 87cc084 | 2015-06-30 08:15:55 -0700 | [diff] [blame] | 568 | connectivity_state_changed_locked(c); |
Craig Tiller | ff3ae68 | 2015-06-29 17:44:04 -0700 | [diff] [blame] | 569 | gpr_mu_unlock(&c->mu); |
| 570 | if (iomgr_success) { |
| 571 | continue_connect(c); |
| 572 | } else { |
| 573 | GRPC_SUBCHANNEL_UNREF(c, "connecting"); |
| 574 | } |
| 575 | } |
| 576 | |
Craig Tiller | ff54c92 | 2015-06-26 16:57:20 -0700 | [diff] [blame] | 577 | static void subchannel_connected(void *arg, int iomgr_success) { |
Craig Tiller | 4ab82d2 | 2015-06-29 09:40:33 -0700 | [diff] [blame] | 578 | grpc_subchannel *c = arg; |
Craig Tiller | 11bf14e | 2015-06-29 16:35:41 -0700 | [diff] [blame] | 579 | if (c->connecting_result.transport != NULL) { |
Craig Tiller | 4ab82d2 | 2015-06-29 09:40:33 -0700 | [diff] [blame] | 580 | publish_transport(c); |
| 581 | } else { |
Craig Tiller | 4ab82d2 | 2015-06-29 09:40:33 -0700 | [diff] [blame] | 582 | gpr_mu_lock(&c->mu); |
Craig Tiller | 87cc084 | 2015-06-30 08:15:55 -0700 | [diff] [blame] | 583 | connectivity_state_changed_locked(c); |
Craig Tiller | ff3ae68 | 2015-06-29 17:44:04 -0700 | [diff] [blame] | 584 | GPR_ASSERT(!c->have_alarm); |
| 585 | c->have_alarm = 1; |
| 586 | c->next_attempt = gpr_time_add(c->next_attempt, c->backoff_delta); |
| 587 | c->backoff_delta = gpr_time_add(c->backoff_delta, c->backoff_delta); |
Craig Tiller | a25ca0b | 2015-07-07 13:54:12 -0700 | [diff] [blame] | 588 | grpc_alarm_init(&c->alarm, c->next_attempt, on_alarm, c, gpr_now(GPR_CLOCK_REALTIME)); |
Craig Tiller | 4ab82d2 | 2015-06-29 09:40:33 -0700 | [diff] [blame] | 589 | gpr_mu_unlock(&c->mu); |
Craig Tiller | 4ab82d2 | 2015-06-29 09:40:33 -0700 | [diff] [blame] | 590 | } |
Craig Tiller | ff54c92 | 2015-06-26 16:57:20 -0700 | [diff] [blame] | 591 | } |
| 592 | |
Craig Tiller | 5f84c84 | 2015-06-26 16:08:21 -0700 | [diff] [blame] | 593 | static gpr_timespec compute_connect_deadline(grpc_subchannel *c) { |
Craig Tiller | 87cc084 | 2015-06-30 08:15:55 -0700 | [diff] [blame] | 594 | return gpr_time_add(c->next_attempt, c->backoff_delta); |
Craig Tiller | 5f84c84 | 2015-06-26 16:08:21 -0700 | [diff] [blame] | 595 | } |
| 596 | |
| 597 | static grpc_connectivity_state compute_connectivity_locked(grpc_subchannel *c) { |
Craig Tiller | b6fbf1d | 2015-06-29 15:25:49 -0700 | [diff] [blame] | 598 | if (c->disconnected) { |
| 599 | return GRPC_CHANNEL_FATAL_FAILURE; |
| 600 | } |
Craig Tiller | 5f84c84 | 2015-06-26 16:08:21 -0700 | [diff] [blame] | 601 | if (c->connecting) { |
Craig Tiller | 87cc084 | 2015-06-30 08:15:55 -0700 | [diff] [blame] | 602 | if (c->have_alarm) { |
| 603 | return GRPC_CHANNEL_TRANSIENT_FAILURE; |
| 604 | } |
Craig Tiller | 5f84c84 | 2015-06-26 16:08:21 -0700 | [diff] [blame] | 605 | return GRPC_CHANNEL_CONNECTING; |
| 606 | } |
| 607 | if (c->active) { |
| 608 | return GRPC_CHANNEL_READY; |
| 609 | } |
| 610 | return GRPC_CHANNEL_IDLE; |
| 611 | } |
| 612 | |
| 613 | static void connectivity_state_changed_locked(grpc_subchannel *c) { |
| 614 | grpc_connectivity_state current = compute_connectivity_locked(c); |
Craig Tiller | c7b5f76 | 2015-06-27 11:48:42 -0700 | [diff] [blame] | 615 | grpc_connectivity_state_set(&c->state_tracker, current); |
Craig Tiller | 5f84c84 | 2015-06-26 16:08:21 -0700 | [diff] [blame] | 616 | } |
| 617 | |
Craig Tiller | 2595ab7 | 2015-06-25 15:26:00 -0700 | [diff] [blame] | 618 | /* |
| 619 | * grpc_subchannel_call implementation |
| 620 | */ |
| 621 | |
Craig Tiller | 079a11b | 2015-06-30 10:07:15 -0700 | [diff] [blame] | 622 | void grpc_subchannel_call_ref( |
| 623 | grpc_subchannel_call *c GRPC_SUBCHANNEL_REF_EXTRA_ARGS) { |
| 624 | gpr_ref(&c->refs); |
Craig Tiller | c396753 | 2015-06-29 14:59:38 -0700 | [diff] [blame] | 625 | } |
Craig Tiller | 2595ab7 | 2015-06-25 15:26:00 -0700 | [diff] [blame] | 626 | |
Craig Tiller | 079a11b | 2015-06-30 10:07:15 -0700 | [diff] [blame] | 627 | void grpc_subchannel_call_unref( |
| 628 | grpc_subchannel_call *c GRPC_SUBCHANNEL_REF_EXTRA_ARGS) { |
Craig Tiller | ca3e9d3 | 2015-06-27 18:37:27 -0700 | [diff] [blame] | 629 | if (gpr_unref(&c->refs)) { |
Craig Tiller | 4ab82d2 | 2015-06-29 09:40:33 -0700 | [diff] [blame] | 630 | gpr_mu *mu = &c->connection->subchannel->mu; |
| 631 | grpc_subchannel *destroy; |
Craig Tiller | ca3e9d3 | 2015-06-27 18:37:27 -0700 | [diff] [blame] | 632 | grpc_call_stack_destroy(SUBCHANNEL_CALL_TO_CALL_STACK(c)); |
Craig Tiller | d7b68e7 | 2015-06-28 11:41:09 -0700 | [diff] [blame] | 633 | gpr_mu_lock(mu); |
Craig Tiller | c396753 | 2015-06-29 14:59:38 -0700 | [diff] [blame] | 634 | destroy = CONNECTION_UNREF_LOCKED(c->connection, "call"); |
Craig Tiller | d7b68e7 | 2015-06-28 11:41:09 -0700 | [diff] [blame] | 635 | gpr_mu_unlock(mu); |
Craig Tiller | ca3e9d3 | 2015-06-27 18:37:27 -0700 | [diff] [blame] | 636 | gpr_free(c); |
Craig Tiller | c396753 | 2015-06-29 14:59:38 -0700 | [diff] [blame] | 637 | if (destroy != NULL) { |
Craig Tiller | 4ab82d2 | 2015-06-29 09:40:33 -0700 | [diff] [blame] | 638 | subchannel_destroy(destroy); |
Craig Tiller | d7b68e7 | 2015-06-28 11:41:09 -0700 | [diff] [blame] | 639 | } |
Craig Tiller | 2595ab7 | 2015-06-25 15:26:00 -0700 | [diff] [blame] | 640 | } |
| 641 | } |
| 642 | |
Craig Tiller | 1b22b9d | 2015-07-20 13:42:22 -0700 | [diff] [blame^] | 643 | char *grpc_subchannel_call_get_peer(grpc_subchannel_call *call) { |
| 644 | grpc_call_stack *call_stack = SUBCHANNEL_CALL_TO_CALL_STACK(call); |
| 645 | grpc_call_element *top_elem = grpc_call_stack_element(call_stack, 0); |
| 646 | return top_elem->filter->get_peer(top_elem); |
| 647 | } |
| 648 | |
Craig Tiller | 2595ab7 | 2015-06-25 15:26:00 -0700 | [diff] [blame] | 649 | void grpc_subchannel_call_process_op(grpc_subchannel_call *call, |
| 650 | grpc_transport_stream_op *op) { |
| 651 | grpc_call_stack *call_stack = SUBCHANNEL_CALL_TO_CALL_STACK(call); |
| 652 | grpc_call_element *top_elem = grpc_call_stack_element(call_stack, 0); |
| 653 | top_elem->filter->start_transport_stream_op(top_elem, op); |
| 654 | } |
Craig Tiller | eb3b12e | 2015-06-26 14:42:49 -0700 | [diff] [blame] | 655 | |
Craig Tiller | abf3638 | 2015-06-29 16:13:27 -0700 | [diff] [blame] | 656 | grpc_subchannel_call *create_call(connection *con) { |
Craig Tiller | 4ab82d2 | 2015-06-29 09:40:33 -0700 | [diff] [blame] | 657 | grpc_channel_stack *chanstk = CHANNEL_STACK_FROM_CONNECTION(con); |
| 658 | grpc_subchannel_call *call = |
| 659 | gpr_malloc(sizeof(grpc_subchannel_call) + chanstk->call_stack_size); |
Craig Tiller | 04c5d4b | 2015-06-26 17:21:41 -0700 | [diff] [blame] | 660 | grpc_call_stack *callstk = SUBCHANNEL_CALL_TO_CALL_STACK(call); |
| 661 | call->connection = con; |
| 662 | gpr_ref_init(&call->refs, 1); |
Craig Tiller | abf3638 | 2015-06-29 16:13:27 -0700 | [diff] [blame] | 663 | grpc_call_stack_init(chanstk, NULL, NULL, callstk); |
Craig Tiller | 04c5d4b | 2015-06-26 17:21:41 -0700 | [diff] [blame] | 664 | return call; |
Craig Tiller | eb3b12e | 2015-06-26 14:42:49 -0700 | [diff] [blame] | 665 | } |