Craig Tiller | af69180 | 2015-06-23 14:57:07 -0700 | [diff] [blame] | 1 | /* |
| 2 | * |
Craig Tiller | 6169d5f | 2016-03-31 07:46:18 -0700 | [diff] [blame] | 3 | * Copyright 2015, Google Inc. |
Craig Tiller | af69180 | 2015-06-23 14:57:07 -0700 | [diff] [blame] | 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, |
Craig Tiller | 892f2d3 | 2015-11-08 13:53:04 +0000 | [diff] [blame] | 25 | * SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT |
Craig Tiller | af69180 | 2015-06-23 14:57:07 -0700 | [diff] [blame] | 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 Tiller | d4c9833 | 2016-03-31 13:45:47 -0700 | [diff] [blame] | 34 | #include "src/core/ext/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> |
Craig Tiller | 694cf8b | 2016-01-15 21:13:25 -0800 | [diff] [blame] | 39 | #include <grpc/support/avl.h> |
Craig Tiller | 2595ab7 | 2015-06-25 15:26:00 -0700 | [diff] [blame] | 40 | |
Craig Tiller | d4c9833 | 2016-03-31 13:45:47 -0700 | [diff] [blame] | 41 | #include "src/core/ext/client_config/client_channel.h" |
Craig Tiller | d4c9833 | 2016-03-31 13:45:47 -0700 | [diff] [blame] | 42 | #include "src/core/ext/client_config/initial_connect_string.h" |
| 43 | #include "src/core/ext/client_config/subchannel_index.h" |
Craig Tiller | 9533d04 | 2016-03-25 17:11:06 -0700 | [diff] [blame] | 44 | #include "src/core/lib/channel/channel_args.h" |
Craig Tiller | 9533d04 | 2016-03-25 17:11:06 -0700 | [diff] [blame] | 45 | #include "src/core/lib/channel/connected_channel.h" |
Craig Tiller | 9533d04 | 2016-03-25 17:11:06 -0700 | [diff] [blame] | 46 | #include "src/core/lib/iomgr/timer.h" |
| 47 | #include "src/core/lib/profiling/timers.h" |
| 48 | #include "src/core/lib/support/backoff.h" |
| 49 | #include "src/core/lib/surface/channel.h" |
| 50 | #include "src/core/lib/surface/channel_init.h" |
| 51 | #include "src/core/lib/transport/connectivity_state.h" |
Craig Tiller | eb3b12e | 2015-06-26 14:42:49 -0700 | [diff] [blame] | 52 | |
Craig Tiller | 12ad5d6 | 2015-11-27 12:17:30 -0800 | [diff] [blame] | 53 | #define INTERNAL_REF_BITS 16 |
| 54 | #define STRONG_REF_MASK (~(gpr_atm)((1 << INTERNAL_REF_BITS) - 1)) |
| 55 | |
yang-g | b4e262c | 2015-07-21 16:11:55 -0700 | [diff] [blame] | 56 | #define GRPC_SUBCHANNEL_MIN_CONNECT_TIMEOUT_SECONDS 20 |
Craig Tiller | e0d6c57 | 2016-05-13 07:23:36 -0700 | [diff] [blame] | 57 | #define GRPC_SUBCHANNEL_INITIAL_CONNECT_BACKOFF_SECONDS 1 |
yang-g | b4e262c | 2015-07-21 16:11:55 -0700 | [diff] [blame] | 58 | #define GRPC_SUBCHANNEL_RECONNECT_BACKOFF_MULTIPLIER 1.6 |
| 59 | #define GRPC_SUBCHANNEL_RECONNECT_MAX_BACKOFF_SECONDS 120 |
| 60 | #define GRPC_SUBCHANNEL_RECONNECT_JITTER 0.2 |
| 61 | |
Craig Tiller | ab33b48 | 2015-11-21 08:11:04 -0800 | [diff] [blame] | 62 | #define GET_CONNECTED_SUBCHANNEL(subchannel, barrier) \ |
| 63 | ((grpc_connected_subchannel *)(gpr_atm_##barrier##_load( \ |
| 64 | &(subchannel)->connected_subchannel))) |
Craig Tiller | b5585d4 | 2015-11-17 07:18:31 -0800 | [diff] [blame] | 65 | |
Craig Tiller | a82950e | 2015-09-22 12:33:20 -0700 | [diff] [blame] | 66 | typedef struct { |
Craig Tiller | 3382511 | 2015-09-18 07:44:19 -0700 | [diff] [blame] | 67 | grpc_closure closure; |
Craig Tiller | 4861304 | 2015-11-29 14:45:11 -0800 | [diff] [blame] | 68 | grpc_subchannel *subchannel; |
Craig Tiller | df91ba5 | 2015-06-29 10:55:46 -0700 | [diff] [blame] | 69 | grpc_connectivity_state connectivity_state; |
| 70 | } state_watcher; |
| 71 | |
Craig Tiller | 4861304 | 2015-11-29 14:45:11 -0800 | [diff] [blame] | 72 | typedef struct external_state_watcher { |
| 73 | grpc_subchannel *subchannel; |
| 74 | grpc_pollset_set *pollset_set; |
| 75 | grpc_closure *notify; |
| 76 | grpc_closure closure; |
| 77 | struct external_state_watcher *next; |
| 78 | struct external_state_watcher *prev; |
| 79 | } external_state_watcher; |
| 80 | |
Craig Tiller | a82950e | 2015-09-22 12:33:20 -0700 | [diff] [blame] | 81 | struct grpc_subchannel { |
Craig Tiller | 9162466 | 2015-06-25 16:31:02 -0700 | [diff] [blame] | 82 | grpc_connector *connector; |
Craig Tiller | f7afa1f | 2015-06-26 09:02:20 -0700 | [diff] [blame] | 83 | |
Craig Tiller | 1d881fb | 2015-12-01 07:39:04 -0800 | [diff] [blame] | 84 | /** refcount |
| 85 | - lower INTERNAL_REF_BITS bits are for internal references: |
| 86 | these do not keep the subchannel open. |
Craig Tiller | 12ad5d6 | 2015-11-27 12:17:30 -0800 | [diff] [blame] | 87 | - upper remaining bits are for public references: these do |
| 88 | keep the subchannel open */ |
| 89 | gpr_atm ref_pair; |
Craig Tiller | b5585d4 | 2015-11-17 07:18:31 -0800 | [diff] [blame] | 90 | |
Craig Tiller | f7afa1f | 2015-06-26 09:02:20 -0700 | [diff] [blame] | 91 | /** non-transport related channel filters */ |
| 92 | const grpc_channel_filter **filters; |
Craig Tiller | 5945ee1 | 2015-06-27 10:36:09 -0700 | [diff] [blame] | 93 | size_t num_filters; |
Craig Tiller | f7afa1f | 2015-06-26 09:02:20 -0700 | [diff] [blame] | 94 | /** channel arguments */ |
| 95 | grpc_channel_args *args; |
| 96 | /** address to connect to */ |
| 97 | struct sockaddr *addr; |
| 98 | size_t addr_len; |
Craig Tiller | eb3b12e | 2015-06-26 14:42:49 -0700 | [diff] [blame] | 99 | |
Craig Tiller | 7391f13 | 2016-01-22 06:39:54 -0800 | [diff] [blame] | 100 | grpc_subchannel_key *key; |
| 101 | |
yang-g | a612412 | 2015-11-05 22:36:20 -0800 | [diff] [blame] | 102 | /** initial string to send to peer */ |
| 103 | gpr_slice initial_connect_string; |
Craig Tiller | eb3b12e | 2015-06-26 14:42:49 -0700 | [diff] [blame] | 104 | |
| 105 | /** set during connection */ |
Craig Tiller | 04c5d4b | 2015-06-26 17:21:41 -0700 | [diff] [blame] | 106 | grpc_connect_out_args connecting_result; |
Craig Tiller | eb3b12e | 2015-06-26 14:42:49 -0700 | [diff] [blame] | 107 | |
| 108 | /** callback for connection finishing */ |
Craig Tiller | 3382511 | 2015-09-18 07:44:19 -0700 | [diff] [blame] | 109 | grpc_closure connected; |
Craig Tiller | eb3b12e | 2015-06-26 14:42:49 -0700 | [diff] [blame] | 110 | |
Craig Tiller | 5f84c84 | 2015-06-26 16:08:21 -0700 | [diff] [blame] | 111 | /** pollset_set tracking who's interested in a connection |
Craig Tiller | 906e3bc | 2015-11-24 07:31:31 -0800 | [diff] [blame] | 112 | being setup */ |
Craig Tiller | 69b093b | 2016-02-25 19:04:07 -0800 | [diff] [blame] | 113 | grpc_pollset_set *pollset_set; |
Craig Tiller | 5f84c84 | 2015-06-26 16:08:21 -0700 | [diff] [blame] | 114 | |
Craig Tiller | b5585d4 | 2015-11-17 07:18:31 -0800 | [diff] [blame] | 115 | /** active connection, or null; of type grpc_connected_subchannel */ |
| 116 | gpr_atm connected_subchannel; |
| 117 | |
Craig Tiller | eb3b12e | 2015-06-26 14:42:49 -0700 | [diff] [blame] | 118 | /** mutex protecting remaining elements */ |
| 119 | gpr_mu mu; |
| 120 | |
Craig Tiller | b5585d4 | 2015-11-17 07:18:31 -0800 | [diff] [blame] | 121 | /** have we seen a disconnection? */ |
| 122 | int disconnected; |
Craig Tiller | eb3b12e | 2015-06-26 14:42:49 -0700 | [diff] [blame] | 123 | /** are we connecting */ |
| 124 | int connecting; |
Craig Tiller | c7b5f76 | 2015-06-27 11:48:42 -0700 | [diff] [blame] | 125 | /** connectivity state tracking */ |
| 126 | grpc_connectivity_state_tracker state_tracker; |
Craig Tiller | ff3ae68 | 2015-06-29 17:44:04 -0700 | [diff] [blame] | 127 | |
Craig Tiller | 4861304 | 2015-11-29 14:45:11 -0800 | [diff] [blame] | 128 | external_state_watcher root_external_state_watcher; |
| 129 | |
Craig Tiller | ff3ae68 | 2015-06-29 17:44:04 -0700 | [diff] [blame] | 130 | /** next connect attempt time */ |
| 131 | gpr_timespec next_attempt; |
Craig Tiller | 536fc53 | 2016-03-11 12:36:56 -0800 | [diff] [blame] | 132 | /** backoff state */ |
| 133 | gpr_backoff backoff_state; |
Craig Tiller | ff3ae68 | 2015-06-29 17:44:04 -0700 | [diff] [blame] | 134 | /** do we have an active alarm? */ |
| 135 | int have_alarm; |
| 136 | /** our alarm */ |
David Garcia Quintas | f747bbc | 2015-10-04 23:09:47 -0700 | [diff] [blame] | 137 | grpc_timer alarm; |
Craig Tiller | 2595ab7 | 2015-06-25 15:26:00 -0700 | [diff] [blame] | 138 | }; |
| 139 | |
Craig Tiller | a82950e | 2015-09-22 12:33:20 -0700 | [diff] [blame] | 140 | struct grpc_subchannel_call { |
Craig Tiller | b5585d4 | 2015-11-17 07:18:31 -0800 | [diff] [blame] | 141 | grpc_connected_subchannel *connection; |
Craig Tiller | 2595ab7 | 2015-06-25 15:26:00 -0700 | [diff] [blame] | 142 | }; |
| 143 | |
Craig Tiller | 04c5d4b | 2015-06-26 17:21:41 -0700 | [diff] [blame] | 144 | #define SUBCHANNEL_CALL_TO_CALL_STACK(call) ((grpc_call_stack *)((call) + 1)) |
Craig Tiller | 906e3bc | 2015-11-24 07:31:31 -0800 | [diff] [blame] | 145 | #define CHANNEL_STACK_FROM_CONNECTION(con) ((grpc_channel_stack *)(con)) |
Craig Tiller | 577c9b2 | 2015-11-02 14:11:15 -0800 | [diff] [blame] | 146 | #define CALLSTACK_TO_SUBCHANNEL_CALL(callstack) \ |
| 147 | (((grpc_subchannel_call *)(callstack)) - 1) |
Craig Tiller | 2595ab7 | 2015-06-25 15:26:00 -0700 | [diff] [blame] | 148 | |
Craig Tiller | a82950e | 2015-09-22 12:33:20 -0700 | [diff] [blame] | 149 | static void subchannel_connected(grpc_exec_ctx *exec_ctx, void *subchannel, |
Craig Tiller | 804ff71 | 2016-05-05 16:25:40 -0700 | [diff] [blame] | 150 | grpc_error *error); |
Craig Tiller | eb3b12e | 2015-06-26 14:42:49 -0700 | [diff] [blame] | 151 | |
Craig Tiller | 577c9b2 | 2015-11-02 14:11:15 -0800 | [diff] [blame] | 152 | #ifdef GRPC_STREAM_REFCOUNT_DEBUG |
Craig Tiller | 906e3bc | 2015-11-24 07:31:31 -0800 | [diff] [blame] | 153 | #define REF_REASON reason |
Craig Tiller | 079a11b | 2015-06-30 10:07:15 -0700 | [diff] [blame] | 154 | #define REF_LOG(name, p) \ |
| 155 | gpr_log(file, line, GPR_LOG_SEVERITY_DEBUG, "%s: %p ref %d -> %d %s", \ |
Craig Tiller | ab33b48 | 2015-11-21 08:11:04 -0800 | [diff] [blame] | 156 | (name), (p), (p)->refs.count, (p)->refs.count + 1, reason) |
Craig Tiller | 079a11b | 2015-06-30 10:07:15 -0700 | [diff] [blame] | 157 | #define UNREF_LOG(name, p) \ |
| 158 | gpr_log(file, line, GPR_LOG_SEVERITY_DEBUG, "%s: %p unref %d -> %d %s", \ |
Craig Tiller | ab33b48 | 2015-11-21 08:11:04 -0800 | [diff] [blame] | 159 | (name), (p), (p)->refs.count, (p)->refs.count - 1, reason) |
Craig Tiller | 1d881fb | 2015-12-01 07:39:04 -0800 | [diff] [blame] | 160 | #define REF_MUTATE_EXTRA_ARGS \ |
| 161 | GRPC_SUBCHANNEL_REF_EXTRA_ARGS, const char *purpose |
Craig Tiller | 12ad5d6 | 2015-11-27 12:17:30 -0800 | [diff] [blame] | 162 | #define REF_MUTATE_PURPOSE(x) , file, line, reason, x |
Craig Tiller | c396753 | 2015-06-29 14:59:38 -0700 | [diff] [blame] | 163 | #else |
Craig Tiller | 906e3bc | 2015-11-24 07:31:31 -0800 | [diff] [blame] | 164 | #define REF_REASON "" |
Craig Tiller | 11bf14e | 2015-06-29 16:35:41 -0700 | [diff] [blame] | 165 | #define REF_LOG(name, p) \ |
| 166 | do { \ |
| 167 | } while (0) |
| 168 | #define UNREF_LOG(name, p) \ |
| 169 | do { \ |
| 170 | } while (0) |
Craig Tiller | 12ad5d6 | 2015-11-27 12:17:30 -0800 | [diff] [blame] | 171 | #define REF_MUTATE_EXTRA_ARGS |
| 172 | #define REF_MUTATE_PURPOSE(x) |
Craig Tiller | c396753 | 2015-06-29 14:59:38 -0700 | [diff] [blame] | 173 | #endif |
| 174 | |
Craig Tiller | 2595ab7 | 2015-06-25 15:26:00 -0700 | [diff] [blame] | 175 | /* |
Craig Tiller | ca3e9d3 | 2015-06-27 18:37:27 -0700 | [diff] [blame] | 176 | * connection implementation |
| 177 | */ |
| 178 | |
Craig Tiller | ab33b48 | 2015-11-21 08:11:04 -0800 | [diff] [blame] | 179 | static void connection_destroy(grpc_exec_ctx *exec_ctx, void *arg, |
Craig Tiller | 804ff71 | 2016-05-05 16:25:40 -0700 | [diff] [blame] | 180 | grpc_error *error) { |
Craig Tiller | b5585d4 | 2015-11-17 07:18:31 -0800 | [diff] [blame] | 181 | grpc_connected_subchannel *c = arg; |
Craig Tiller | a82950e | 2015-09-22 12:33:20 -0700 | [diff] [blame] | 182 | grpc_channel_stack_destroy(exec_ctx, CHANNEL_STACK_FROM_CONNECTION(c)); |
| 183 | gpr_free(c); |
Craig Tiller | ca3e9d3 | 2015-06-27 18:37:27 -0700 | [diff] [blame] | 184 | } |
| 185 | |
Craig Tiller | f40df23 | 2016-03-25 13:38:14 -0700 | [diff] [blame] | 186 | void grpc_connected_subchannel_ref( |
| 187 | grpc_connected_subchannel *c GRPC_SUBCHANNEL_REF_EXTRA_ARGS) { |
Craig Tiller | 906e3bc | 2015-11-24 07:31:31 -0800 | [diff] [blame] | 188 | GRPC_CHANNEL_STACK_REF(CHANNEL_STACK_FROM_CONNECTION(c), REF_REASON); |
Craig Tiller | d7b68e7 | 2015-06-28 11:41:09 -0700 | [diff] [blame] | 189 | } |
| 190 | |
Craig Tiller | ab33b48 | 2015-11-21 08:11:04 -0800 | [diff] [blame] | 191 | void grpc_connected_subchannel_unref(grpc_exec_ctx *exec_ctx, |
| 192 | grpc_connected_subchannel *c |
| 193 | GRPC_SUBCHANNEL_REF_EXTRA_ARGS) { |
Craig Tiller | 7b43561 | 2015-11-24 08:15:05 -0800 | [diff] [blame] | 194 | GRPC_CHANNEL_STACK_UNREF(exec_ctx, CHANNEL_STACK_FROM_CONNECTION(c), |
| 195 | REF_REASON); |
Craig Tiller | ca3e9d3 | 2015-06-27 18:37:27 -0700 | [diff] [blame] | 196 | } |
| 197 | |
| 198 | /* |
Craig Tiller | 2595ab7 | 2015-06-25 15:26:00 -0700 | [diff] [blame] | 199 | * grpc_subchannel implementation |
| 200 | */ |
| 201 | |
Craig Tiller | ab33b48 | 2015-11-21 08:11:04 -0800 | [diff] [blame] | 202 | static void subchannel_destroy(grpc_exec_ctx *exec_ctx, void *arg, |
Craig Tiller | 804ff71 | 2016-05-05 16:25:40 -0700 | [diff] [blame] | 203 | grpc_error *error) { |
Craig Tiller | b5585d4 | 2015-11-17 07:18:31 -0800 | [diff] [blame] | 204 | grpc_subchannel *c = arg; |
Craig Tiller | 565b18b | 2015-09-23 10:09:42 -0700 | [diff] [blame] | 205 | gpr_free((void *)c->filters); |
Craig Tiller | a82950e | 2015-09-22 12:33:20 -0700 | [diff] [blame] | 206 | grpc_channel_args_destroy(c->args); |
| 207 | gpr_free(c->addr); |
yang-g | a612412 | 2015-11-05 22:36:20 -0800 | [diff] [blame] | 208 | gpr_slice_unref(c->initial_connect_string); |
Craig Tiller | a82950e | 2015-09-22 12:33:20 -0700 | [diff] [blame] | 209 | grpc_connectivity_state_destroy(exec_ctx, &c->state_tracker); |
| 210 | grpc_connector_unref(exec_ctx, c->connector); |
Craig Tiller | 69b093b | 2016-02-25 19:04:07 -0800 | [diff] [blame] | 211 | grpc_pollset_set_destroy(c->pollset_set); |
Craig Tiller | 1948244 | 2016-01-25 09:59:20 -0800 | [diff] [blame] | 212 | grpc_subchannel_key_destroy(exec_ctx, c->key); |
Craig Tiller | a82950e | 2015-09-22 12:33:20 -0700 | [diff] [blame] | 213 | gpr_free(c); |
Craig Tiller | 2595ab7 | 2015-06-25 15:26:00 -0700 | [diff] [blame] | 214 | } |
| 215 | |
Craig Tiller | 1d881fb | 2015-12-01 07:39:04 -0800 | [diff] [blame] | 216 | static gpr_atm ref_mutate(grpc_subchannel *c, gpr_atm delta, |
| 217 | int barrier REF_MUTATE_EXTRA_ARGS) { |
| 218 | gpr_atm old_val = barrier ? gpr_atm_full_fetch_add(&c->ref_pair, delta) |
| 219 | : gpr_atm_no_barrier_fetch_add(&c->ref_pair, delta); |
Craig Tiller | 12ad5d6 | 2015-11-27 12:17:30 -0800 | [diff] [blame] | 220 | #ifdef GRPC_STREAM_REFCOUNT_DEBUG |
Craig Tiller | 1d881fb | 2015-12-01 07:39:04 -0800 | [diff] [blame] | 221 | gpr_log(file, line, GPR_LOG_SEVERITY_DEBUG, |
| 222 | "SUBCHANNEL: %p % 12s 0x%08x -> 0x%08x [%s]", c, purpose, old_val, |
| 223 | old_val + delta, reason); |
Craig Tiller | 12ad5d6 | 2015-11-27 12:17:30 -0800 | [diff] [blame] | 224 | #endif |
| 225 | return old_val; |
| 226 | } |
| 227 | |
Craig Tiller | f40df23 | 2016-03-25 13:38:14 -0700 | [diff] [blame] | 228 | grpc_subchannel *grpc_subchannel_ref( |
| 229 | grpc_subchannel *c GRPC_SUBCHANNEL_REF_EXTRA_ARGS) { |
Craig Tiller | 12ad5d6 | 2015-11-27 12:17:30 -0800 | [diff] [blame] | 230 | gpr_atm old_refs; |
Craig Tiller | 1d881fb | 2015-12-01 07:39:04 -0800 | [diff] [blame] | 231 | old_refs = ref_mutate(c, (1 << INTERNAL_REF_BITS), |
| 232 | 0 REF_MUTATE_PURPOSE("STRONG_REF")); |
Craig Tiller | 12ad5d6 | 2015-11-27 12:17:30 -0800 | [diff] [blame] | 233 | GPR_ASSERT((old_refs & STRONG_REF_MASK) != 0); |
Craig Tiller | 1948244 | 2016-01-25 09:59:20 -0800 | [diff] [blame] | 234 | return c; |
Craig Tiller | 12ad5d6 | 2015-11-27 12:17:30 -0800 | [diff] [blame] | 235 | } |
| 236 | |
Craig Tiller | f40df23 | 2016-03-25 13:38:14 -0700 | [diff] [blame] | 237 | grpc_subchannel *grpc_subchannel_weak_ref( |
| 238 | grpc_subchannel *c GRPC_SUBCHANNEL_REF_EXTRA_ARGS) { |
Craig Tiller | 12ad5d6 | 2015-11-27 12:17:30 -0800 | [diff] [blame] | 239 | gpr_atm old_refs; |
| 240 | old_refs = ref_mutate(c, 1, 0 REF_MUTATE_PURPOSE("WEAK_REF")); |
| 241 | GPR_ASSERT(old_refs != 0); |
Craig Tiller | 1948244 | 2016-01-25 09:59:20 -0800 | [diff] [blame] | 242 | return c; |
Craig Tiller | 12ad5d6 | 2015-11-27 12:17:30 -0800 | [diff] [blame] | 243 | } |
| 244 | |
Craig Tiller | 5de79ee | 2016-01-25 08:16:02 -0800 | [diff] [blame] | 245 | grpc_subchannel *grpc_subchannel_ref_from_weak_ref( |
| 246 | grpc_subchannel *c GRPC_SUBCHANNEL_REF_EXTRA_ARGS) { |
Craig Tiller | 8cdba66 | 2016-01-22 20:01:55 -0800 | [diff] [blame] | 247 | if (!c) return NULL; |
| 248 | for (;;) { |
| 249 | gpr_atm old_refs = gpr_atm_acq_load(&c->ref_pair); |
| 250 | if (old_refs >= (1 << INTERNAL_REF_BITS)) { |
| 251 | gpr_atm new_refs = old_refs + (1 << INTERNAL_REF_BITS); |
| 252 | if (gpr_atm_rel_cas(&c->ref_pair, old_refs, new_refs)) { |
| 253 | return c; |
| 254 | } |
| 255 | } else { |
| 256 | return NULL; |
| 257 | } |
| 258 | } |
| 259 | } |
| 260 | |
Craig Tiller | 12ad5d6 | 2015-11-27 12:17:30 -0800 | [diff] [blame] | 261 | static void disconnect(grpc_exec_ctx *exec_ctx, grpc_subchannel *c) { |
| 262 | grpc_connected_subchannel *con; |
Craig Tiller | 7391f13 | 2016-01-22 06:39:54 -0800 | [diff] [blame] | 263 | grpc_subchannel_index_unregister(exec_ctx, c->key, c); |
Craig Tiller | 12ad5d6 | 2015-11-27 12:17:30 -0800 | [diff] [blame] | 264 | gpr_mu_lock(&c->mu); |
| 265 | GPR_ASSERT(!c->disconnected); |
| 266 | c->disconnected = 1; |
Craig Tiller | caa4e70 | 2015-12-01 06:55:25 -0800 | [diff] [blame] | 267 | grpc_connector_shutdown(exec_ctx, c->connector); |
Craig Tiller | 12ad5d6 | 2015-11-27 12:17:30 -0800 | [diff] [blame] | 268 | con = GET_CONNECTED_SUBCHANNEL(c, no_barrier); |
| 269 | if (con != NULL) { |
| 270 | GRPC_CONNECTED_SUBCHANNEL_UNREF(exec_ctx, con, "connection"); |
Yuchen Zeng | 615da64 | 2016-04-29 18:14:05 -0700 | [diff] [blame] | 271 | gpr_atm_no_barrier_store(&c->connected_subchannel, (gpr_atm)0xdeadbeef); |
Craig Tiller | 12ad5d6 | 2015-11-27 12:17:30 -0800 | [diff] [blame] | 272 | } |
| 273 | gpr_mu_unlock(&c->mu); |
Craig Tiller | b5585d4 | 2015-11-17 07:18:31 -0800 | [diff] [blame] | 274 | } |
| 275 | |
| 276 | void grpc_subchannel_unref(grpc_exec_ctx *exec_ctx, |
| 277 | grpc_subchannel *c GRPC_SUBCHANNEL_REF_EXTRA_ARGS) { |
Craig Tiller | 12ad5d6 | 2015-11-27 12:17:30 -0800 | [diff] [blame] | 278 | gpr_atm old_refs; |
Craig Tiller | 1d881fb | 2015-12-01 07:39:04 -0800 | [diff] [blame] | 279 | old_refs = ref_mutate(c, (gpr_atm)1 - (gpr_atm)(1 << INTERNAL_REF_BITS), |
| 280 | 1 REF_MUTATE_PURPOSE("STRONG_UNREF")); |
Craig Tiller | 50ec267 | 2015-11-27 21:45:11 -0800 | [diff] [blame] | 281 | if ((old_refs & STRONG_REF_MASK) == (1 << INTERNAL_REF_BITS)) { |
Craig Tiller | 12ad5d6 | 2015-11-27 12:17:30 -0800 | [diff] [blame] | 282 | disconnect(exec_ctx, c); |
| 283 | } |
| 284 | GRPC_SUBCHANNEL_WEAK_UNREF(exec_ctx, c, "strong-unref"); |
| 285 | } |
| 286 | |
| 287 | void grpc_subchannel_weak_unref(grpc_exec_ctx *exec_ctx, |
Craig Tiller | 1d881fb | 2015-12-01 07:39:04 -0800 | [diff] [blame] | 288 | grpc_subchannel *c |
| 289 | GRPC_SUBCHANNEL_REF_EXTRA_ARGS) { |
Craig Tiller | 12ad5d6 | 2015-11-27 12:17:30 -0800 | [diff] [blame] | 290 | gpr_atm old_refs; |
| 291 | old_refs = ref_mutate(c, -(gpr_atm)1, 1 REF_MUTATE_PURPOSE("WEAK_UNREF")); |
Craig Tiller | f036a64 | 2015-12-01 17:00:40 -0800 | [diff] [blame] | 292 | if (old_refs == 1) { |
Craig Tiller | 332f1b3 | 2016-05-24 13:21:21 -0700 | [diff] [blame] | 293 | grpc_exec_ctx_sched(exec_ctx, grpc_closure_create(subchannel_destroy, c), |
Craig Tiller | 77c983d | 2016-05-24 13:23:14 -0700 | [diff] [blame] | 294 | GRPC_ERROR_NONE, NULL); |
Craig Tiller | b5585d4 | 2015-11-17 07:18:31 -0800 | [diff] [blame] | 295 | } |
| 296 | } |
| 297 | |
Craig Tiller | 7391f13 | 2016-01-22 06:39:54 -0800 | [diff] [blame] | 298 | grpc_subchannel *grpc_subchannel_create(grpc_exec_ctx *exec_ctx, |
| 299 | grpc_connector *connector, |
Craig Tiller | a82950e | 2015-09-22 12:33:20 -0700 | [diff] [blame] | 300 | grpc_subchannel_args *args) { |
Craig Tiller | 7391f13 | 2016-01-22 06:39:54 -0800 | [diff] [blame] | 301 | grpc_subchannel_key *key = grpc_subchannel_key_create(connector, args); |
| 302 | grpc_subchannel *c = grpc_subchannel_index_find(exec_ctx, key); |
| 303 | if (c) { |
Craig Tiller | 1948244 | 2016-01-25 09:59:20 -0800 | [diff] [blame] | 304 | grpc_subchannel_key_destroy(exec_ctx, key); |
Craig Tiller | 7391f13 | 2016-01-22 06:39:54 -0800 | [diff] [blame] | 305 | return c; |
| 306 | } |
| 307 | |
| 308 | c = gpr_malloc(sizeof(*c)); |
Craig Tiller | a82950e | 2015-09-22 12:33:20 -0700 | [diff] [blame] | 309 | memset(c, 0, sizeof(*c)); |
Craig Tiller | 7391f13 | 2016-01-22 06:39:54 -0800 | [diff] [blame] | 310 | c->key = key; |
Craig Tiller | 12ad5d6 | 2015-11-27 12:17:30 -0800 | [diff] [blame] | 311 | gpr_atm_no_barrier_store(&c->ref_pair, 1 << INTERNAL_REF_BITS); |
Craig Tiller | f7afa1f | 2015-06-26 09:02:20 -0700 | [diff] [blame] | 312 | c->connector = connector; |
Craig Tiller | a82950e | 2015-09-22 12:33:20 -0700 | [diff] [blame] | 313 | grpc_connector_ref(c->connector); |
Craig Tiller | 5945ee1 | 2015-06-27 10:36:09 -0700 | [diff] [blame] | 314 | c->num_filters = args->filter_count; |
Craig Tiller | b9c5505 | 2016-01-27 15:22:48 -0800 | [diff] [blame] | 315 | if (c->num_filters > 0) { |
| 316 | c->filters = gpr_malloc(sizeof(grpc_channel_filter *) * c->num_filters); |
| 317 | memcpy((void *)c->filters, args->filters, |
| 318 | sizeof(grpc_channel_filter *) * c->num_filters); |
| 319 | } else { |
| 320 | c->filters = NULL; |
| 321 | } |
Craig Tiller | a82950e | 2015-09-22 12:33:20 -0700 | [diff] [blame] | 322 | c->addr = gpr_malloc(args->addr_len); |
Craig Tiller | 6889799 | 2016-05-03 23:10:07 -0700 | [diff] [blame] | 323 | if (args->addr_len) memcpy(c->addr, args->addr, args->addr_len); |
Craig Tiller | 69b093b | 2016-02-25 19:04:07 -0800 | [diff] [blame] | 324 | c->pollset_set = grpc_pollset_set_create(); |
Craig Tiller | f7afa1f | 2015-06-26 09:02:20 -0700 | [diff] [blame] | 325 | c->addr_len = args->addr_len; |
yang-g | a612412 | 2015-11-05 22:36:20 -0800 | [diff] [blame] | 326 | grpc_set_initial_connect_string(&c->addr, &c->addr_len, |
| 327 | &c->initial_connect_string); |
Craig Tiller | a82950e | 2015-09-22 12:33:20 -0700 | [diff] [blame] | 328 | c->args = grpc_channel_args_copy(args->args); |
Craig Tiller | 1d881fb | 2015-12-01 07:39:04 -0800 | [diff] [blame] | 329 | c->root_external_state_watcher.next = c->root_external_state_watcher.prev = |
| 330 | &c->root_external_state_watcher; |
Craig Tiller | a82950e | 2015-09-22 12:33:20 -0700 | [diff] [blame] | 331 | grpc_closure_init(&c->connected, subchannel_connected, c); |
| 332 | grpc_connectivity_state_init(&c->state_tracker, GRPC_CHANNEL_IDLE, |
| 333 | "subchannel"); |
Craig Tiller | 536fc53 | 2016-03-11 12:36:56 -0800 | [diff] [blame] | 334 | gpr_backoff_init(&c->backoff_state, |
| 335 | GRPC_SUBCHANNEL_RECONNECT_BACKOFF_MULTIPLIER, |
| 336 | GRPC_SUBCHANNEL_RECONNECT_JITTER, |
| 337 | GRPC_SUBCHANNEL_INITIAL_CONNECT_BACKOFF_SECONDS * 1000, |
| 338 | GRPC_SUBCHANNEL_RECONNECT_MAX_BACKOFF_SECONDS * 1000); |
| 339 | if (c->args) { |
| 340 | for (size_t i = 0; i < c->args->num_args; i++) { |
| 341 | if (0 == strcmp(c->args->args[i].key, |
| 342 | "grpc.testing.fixed_reconnect_backoff")) { |
| 343 | GPR_ASSERT(c->args->args[i].type == GRPC_ARG_INTEGER); |
| 344 | gpr_backoff_init(&c->backoff_state, 1.0, 0.0, |
| 345 | c->args->args[i].value.integer, |
| 346 | c->args->args[i].value.integer); |
| 347 | } |
Craig Tiller | 076c2b8 | 2016-03-30 17:12:29 -0700 | [diff] [blame] | 348 | if (0 == |
| 349 | strcmp(c->args->args[i].key, GRPC_ARG_MAX_RECONNECT_BACKOFF_MS)) { |
| 350 | if (c->args->args[i].type == GRPC_ARG_INTEGER) { |
| 351 | if (c->args->args[i].value.integer >= 0) { |
| 352 | gpr_backoff_init( |
| 353 | &c->backoff_state, GRPC_SUBCHANNEL_RECONNECT_BACKOFF_MULTIPLIER, |
| 354 | GRPC_SUBCHANNEL_RECONNECT_JITTER, |
| 355 | GPR_MIN(c->args->args[i].value.integer, |
| 356 | GRPC_SUBCHANNEL_INITIAL_CONNECT_BACKOFF_SECONDS * 1000), |
| 357 | c->args->args[i].value.integer); |
| 358 | } else { |
| 359 | gpr_log(GPR_ERROR, GRPC_ARG_MAX_RECONNECT_BACKOFF_MS |
| 360 | " : must be non-negative"); |
| 361 | } |
| 362 | } else { |
| 363 | gpr_log(GPR_ERROR, |
| 364 | GRPC_ARG_MAX_RECONNECT_BACKOFF_MS " : must be an integer"); |
| 365 | } |
| 366 | } |
Craig Tiller | 536fc53 | 2016-03-11 12:36:56 -0800 | [diff] [blame] | 367 | } |
| 368 | } |
Craig Tiller | a82950e | 2015-09-22 12:33:20 -0700 | [diff] [blame] | 369 | gpr_mu_init(&c->mu); |
Craig Tiller | 7391f13 | 2016-01-22 06:39:54 -0800 | [diff] [blame] | 370 | |
| 371 | return grpc_subchannel_index_register(exec_ctx, key, c); |
Craig Tiller | f7afa1f | 2015-06-26 09:02:20 -0700 | [diff] [blame] | 372 | } |
| 373 | |
Craig Tiller | a82950e | 2015-09-22 12:33:20 -0700 | [diff] [blame] | 374 | static void continue_connect(grpc_exec_ctx *exec_ctx, grpc_subchannel *c) { |
Craig Tiller | 4ab82d2 | 2015-06-29 09:40:33 -0700 | [diff] [blame] | 375 | grpc_connect_in_args args; |
Craig Tiller | 04c5d4b | 2015-06-26 17:21:41 -0700 | [diff] [blame] | 376 | |
Craig Tiller | 69b093b | 2016-02-25 19:04:07 -0800 | [diff] [blame] | 377 | args.interested_parties = c->pollset_set; |
Craig Tiller | 4ab82d2 | 2015-06-29 09:40:33 -0700 | [diff] [blame] | 378 | args.addr = c->addr; |
| 379 | args.addr_len = c->addr_len; |
Craig Tiller | 536fc53 | 2016-03-11 12:36:56 -0800 | [diff] [blame] | 380 | args.deadline = c->next_attempt; |
Craig Tiller | 4ab82d2 | 2015-06-29 09:40:33 -0700 | [diff] [blame] | 381 | args.channel_args = c->args; |
yang-g | a612412 | 2015-11-05 22:36:20 -0800 | [diff] [blame] | 382 | args.initial_connect_string = c->initial_connect_string; |
Craig Tiller | 04c5d4b | 2015-06-26 17:21:41 -0700 | [diff] [blame] | 383 | |
Craig Tiller | ab33b48 | 2015-11-21 08:11:04 -0800 | [diff] [blame] | 384 | grpc_connectivity_state_set(exec_ctx, &c->state_tracker, |
Craig Tiller | 804ff71 | 2016-05-05 16:25:40 -0700 | [diff] [blame] | 385 | GRPC_CHANNEL_CONNECTING, GRPC_ERROR_NONE, |
| 386 | "state_change"); |
Craig Tiller | a82950e | 2015-09-22 12:33:20 -0700 | [diff] [blame] | 387 | grpc_connector_connect(exec_ctx, c->connector, &args, &c->connecting_result, |
| 388 | &c->connected); |
Craig Tiller | 04c5d4b | 2015-06-26 17:21:41 -0700 | [diff] [blame] | 389 | } |
| 390 | |
Craig Tiller | a82950e | 2015-09-22 12:33:20 -0700 | [diff] [blame] | 391 | static void start_connect(grpc_exec_ctx *exec_ctx, grpc_subchannel *c) { |
Craig Tiller | a82950e | 2015-09-22 12:33:20 -0700 | [diff] [blame] | 392 | c->next_attempt = |
Craig Tiller | 536fc53 | 2016-03-11 12:36:56 -0800 | [diff] [blame] | 393 | gpr_backoff_begin(&c->backoff_state, gpr_now(GPR_CLOCK_MONOTONIC)); |
Craig Tiller | a82950e | 2015-09-22 12:33:20 -0700 | [diff] [blame] | 394 | continue_connect(exec_ctx, c); |
Craig Tiller | ff3ae68 | 2015-06-29 17:44:04 -0700 | [diff] [blame] | 395 | } |
| 396 | |
Craig Tiller | 804ff71 | 2016-05-05 16:25:40 -0700 | [diff] [blame] | 397 | grpc_connectivity_state grpc_subchannel_check_connectivity(grpc_subchannel *c, |
| 398 | grpc_error **error) { |
Craig Tiller | 5f84c84 | 2015-06-26 16:08:21 -0700 | [diff] [blame] | 399 | grpc_connectivity_state state; |
Craig Tiller | a82950e | 2015-09-22 12:33:20 -0700 | [diff] [blame] | 400 | gpr_mu_lock(&c->mu); |
Craig Tiller | 804ff71 | 2016-05-05 16:25:40 -0700 | [diff] [blame] | 401 | state = grpc_connectivity_state_check(&c->state_tracker, error); |
Craig Tiller | a82950e | 2015-09-22 12:33:20 -0700 | [diff] [blame] | 402 | gpr_mu_unlock(&c->mu); |
Craig Tiller | 5f84c84 | 2015-06-26 16:08:21 -0700 | [diff] [blame] | 403 | return state; |
| 404 | } |
| 405 | |
Craig Tiller | 1d881fb | 2015-12-01 07:39:04 -0800 | [diff] [blame] | 406 | static void on_external_state_watcher_done(grpc_exec_ctx *exec_ctx, void *arg, |
Craig Tiller | 804ff71 | 2016-05-05 16:25:40 -0700 | [diff] [blame] | 407 | grpc_error *error) { |
Craig Tiller | 86c9958 | 2015-11-25 15:22:26 -0800 | [diff] [blame] | 408 | external_state_watcher *w = arg; |
| 409 | grpc_closure *follow_up = w->notify; |
Craig Tiller | 4861304 | 2015-11-29 14:45:11 -0800 | [diff] [blame] | 410 | if (w->pollset_set != NULL) { |
Craig Tiller | 69b093b | 2016-02-25 19:04:07 -0800 | [diff] [blame] | 411 | grpc_pollset_set_del_pollset_set(exec_ctx, w->subchannel->pollset_set, |
Craig Tiller | 1d881fb | 2015-12-01 07:39:04 -0800 | [diff] [blame] | 412 | w->pollset_set); |
Craig Tiller | 4861304 | 2015-11-29 14:45:11 -0800 | [diff] [blame] | 413 | } |
| 414 | gpr_mu_lock(&w->subchannel->mu); |
| 415 | w->next->prev = w->prev; |
| 416 | w->prev->next = w->next; |
| 417 | gpr_mu_unlock(&w->subchannel->mu); |
Craig Tiller | 12ad5d6 | 2015-11-27 12:17:30 -0800 | [diff] [blame] | 418 | GRPC_SUBCHANNEL_WEAK_UNREF(exec_ctx, w->subchannel, "external_state_watcher"); |
Craig Tiller | 86c9958 | 2015-11-25 15:22:26 -0800 | [diff] [blame] | 419 | gpr_free(w); |
Craig Tiller | 804ff71 | 2016-05-05 16:25:40 -0700 | [diff] [blame] | 420 | follow_up->cb(exec_ctx, follow_up->cb_arg, error); |
Craig Tiller | 86c9958 | 2015-11-25 15:22:26 -0800 | [diff] [blame] | 421 | } |
| 422 | |
Craig Tiller | 1d881fb | 2015-12-01 07:39:04 -0800 | [diff] [blame] | 423 | void grpc_subchannel_notify_on_state_change( |
| 424 | grpc_exec_ctx *exec_ctx, grpc_subchannel *c, |
| 425 | grpc_pollset_set *interested_parties, grpc_connectivity_state *state, |
| 426 | grpc_closure *notify) { |
Craig Tiller | 4861304 | 2015-11-29 14:45:11 -0800 | [diff] [blame] | 427 | external_state_watcher *w; |
| 428 | |
| 429 | if (state == NULL) { |
| 430 | gpr_mu_lock(&c->mu); |
Craig Tiller | 1d881fb | 2015-12-01 07:39:04 -0800 | [diff] [blame] | 431 | for (w = c->root_external_state_watcher.next; |
| 432 | w != &c->root_external_state_watcher; w = w->next) { |
Craig Tiller | 4861304 | 2015-11-29 14:45:11 -0800 | [diff] [blame] | 433 | if (w->notify == notify) { |
Craig Tiller | 1d881fb | 2015-12-01 07:39:04 -0800 | [diff] [blame] | 434 | grpc_connectivity_state_notify_on_state_change( |
| 435 | exec_ctx, &c->state_tracker, NULL, &w->closure); |
Craig Tiller | 4861304 | 2015-11-29 14:45:11 -0800 | [diff] [blame] | 436 | } |
| 437 | } |
| 438 | gpr_mu_unlock(&c->mu); |
| 439 | } else { |
| 440 | w = gpr_malloc(sizeof(*w)); |
| 441 | w->subchannel = c; |
| 442 | w->pollset_set = interested_parties; |
| 443 | w->notify = notify; |
| 444 | grpc_closure_init(&w->closure, on_external_state_watcher_done, w); |
| 445 | if (interested_parties != NULL) { |
Craig Tiller | 69b093b | 2016-02-25 19:04:07 -0800 | [diff] [blame] | 446 | grpc_pollset_set_add_pollset_set(exec_ctx, c->pollset_set, |
Craig Tiller | 1d881fb | 2015-12-01 07:39:04 -0800 | [diff] [blame] | 447 | interested_parties); |
Craig Tiller | 4861304 | 2015-11-29 14:45:11 -0800 | [diff] [blame] | 448 | } |
| 449 | GRPC_SUBCHANNEL_WEAK_REF(c, "external_state_watcher"); |
| 450 | gpr_mu_lock(&c->mu); |
| 451 | w->next = &c->root_external_state_watcher; |
| 452 | w->prev = w->next->prev; |
| 453 | w->next->prev = w->prev->next = w; |
| 454 | if (grpc_connectivity_state_notify_on_state_change( |
| 455 | exec_ctx, &c->state_tracker, state, &w->closure)) { |
Craig Tiller | 4861304 | 2015-11-29 14:45:11 -0800 | [diff] [blame] | 456 | c->connecting = 1; |
| 457 | /* released by connection */ |
| 458 | GRPC_SUBCHANNEL_WEAK_REF(c, "connecting"); |
Craig Tiller | 8e19f61 | 2016-03-01 21:41:13 -0800 | [diff] [blame] | 459 | start_connect(exec_ctx, c); |
Craig Tiller | 4861304 | 2015-11-29 14:45:11 -0800 | [diff] [blame] | 460 | } |
| 461 | gpr_mu_unlock(&c->mu); |
Craig Tiller | a82950e | 2015-09-22 12:33:20 -0700 | [diff] [blame] | 462 | } |
Craig Tiller | 5f84c84 | 2015-06-26 16:08:21 -0700 | [diff] [blame] | 463 | } |
| 464 | |
Craig Tiller | ab33b48 | 2015-11-21 08:11:04 -0800 | [diff] [blame] | 465 | void grpc_connected_subchannel_process_transport_op( |
| 466 | grpc_exec_ctx *exec_ctx, grpc_connected_subchannel *con, |
| 467 | grpc_transport_op *op) { |
Craig Tiller | b5585d4 | 2015-11-17 07:18:31 -0800 | [diff] [blame] | 468 | grpc_channel_stack *channel_stack = CHANNEL_STACK_FROM_CONNECTION(con); |
Craig Tiller | ab33b48 | 2015-11-21 08:11:04 -0800 | [diff] [blame] | 469 | grpc_channel_element *top_elem = grpc_channel_stack_element(channel_stack, 0); |
Craig Tiller | b5585d4 | 2015-11-17 07:18:31 -0800 | [diff] [blame] | 470 | top_elem->filter->start_transport_op(exec_ctx, top_elem, op); |
| 471 | } |
| 472 | |
| 473 | static void subchannel_on_child_state_changed(grpc_exec_ctx *exec_ctx, void *p, |
Craig Tiller | 804ff71 | 2016-05-05 16:25:40 -0700 | [diff] [blame] | 474 | grpc_error *error) { |
Craig Tiller | df91ba5 | 2015-06-29 10:55:46 -0700 | [diff] [blame] | 475 | state_watcher *sw = p; |
Craig Tiller | 4861304 | 2015-11-29 14:45:11 -0800 | [diff] [blame] | 476 | grpc_subchannel *c = sw->subchannel; |
Craig Tiller | df91ba5 | 2015-06-29 10:55:46 -0700 | [diff] [blame] | 477 | gpr_mu *mu = &c->mu; |
Craig Tiller | df91ba5 | 2015-06-29 10:55:46 -0700 | [diff] [blame] | 478 | |
Craig Tiller | a82950e | 2015-09-22 12:33:20 -0700 | [diff] [blame] | 479 | gpr_mu_lock(mu); |
Craig Tiller | df91ba5 | 2015-06-29 10:55:46 -0700 | [diff] [blame] | 480 | |
Craig Tiller | b5585d4 | 2015-11-17 07:18:31 -0800 | [diff] [blame] | 481 | /* if we failed just leave this closure */ |
Craig Tiller | 804ff71 | 2016-05-05 16:25:40 -0700 | [diff] [blame] | 482 | if (sw->connectivity_state == GRPC_CHANNEL_TRANSIENT_FAILURE) { |
| 483 | /* any errors on a subchannel ==> we're done, create a new one */ |
Craig Tiller | d925c93 | 2016-06-06 08:38:50 -0700 | [diff] [blame] | 484 | sw->connectivity_state = GRPC_CHANNEL_SHUTDOWN; |
Craig Tiller | 804ff71 | 2016-05-05 16:25:40 -0700 | [diff] [blame] | 485 | } |
| 486 | grpc_connectivity_state_set(exec_ctx, &c->state_tracker, |
Craig Tiller | 94f8453 | 2016-05-06 21:03:29 -0700 | [diff] [blame] | 487 | sw->connectivity_state, GRPC_ERROR_REF(error), |
| 488 | "reflect_child"); |
Craig Tiller | d925c93 | 2016-06-06 08:38:50 -0700 | [diff] [blame] | 489 | if (sw->connectivity_state != GRPC_CHANNEL_SHUTDOWN) { |
Craig Tiller | 804ff71 | 2016-05-05 16:25:40 -0700 | [diff] [blame] | 490 | grpc_connected_subchannel_notify_on_state_change( |
| 491 | exec_ctx, GET_CONNECTED_SUBCHANNEL(c, no_barrier), NULL, |
| 492 | &sw->connectivity_state, &sw->closure); |
| 493 | GRPC_SUBCHANNEL_WEAK_REF(c, "state_watcher"); |
| 494 | sw = NULL; |
Craig Tiller | a82950e | 2015-09-22 12:33:20 -0700 | [diff] [blame] | 495 | } |
Craig Tiller | df91ba5 | 2015-06-29 10:55:46 -0700 | [diff] [blame] | 496 | |
Craig Tiller | a82950e | 2015-09-22 12:33:20 -0700 | [diff] [blame] | 497 | gpr_mu_unlock(mu); |
Craig Tiller | 12ad5d6 | 2015-11-27 12:17:30 -0800 | [diff] [blame] | 498 | GRPC_SUBCHANNEL_WEAK_UNREF(exec_ctx, c, "state_watcher"); |
Craig Tiller | b5585d4 | 2015-11-17 07:18:31 -0800 | [diff] [blame] | 499 | gpr_free(sw); |
| 500 | } |
| 501 | |
Craig Tiller | ab33b48 | 2015-11-21 08:11:04 -0800 | [diff] [blame] | 502 | static void connected_subchannel_state_op(grpc_exec_ctx *exec_ctx, |
| 503 | grpc_connected_subchannel *con, |
Craig Tiller | f036a64 | 2015-12-01 17:00:40 -0800 | [diff] [blame] | 504 | grpc_pollset_set *interested_parties, |
Craig Tiller | ab33b48 | 2015-11-21 08:11:04 -0800 | [diff] [blame] | 505 | grpc_connectivity_state *state, |
| 506 | grpc_closure *closure) { |
Craig Tiller | b5585d4 | 2015-11-17 07:18:31 -0800 | [diff] [blame] | 507 | grpc_transport_op op; |
| 508 | grpc_channel_element *elem; |
| 509 | memset(&op, 0, sizeof(op)); |
| 510 | op.connectivity_state = state; |
| 511 | op.on_connectivity_state_change = closure; |
Craig Tiller | f036a64 | 2015-12-01 17:00:40 -0800 | [diff] [blame] | 512 | op.bind_pollset_set = interested_parties; |
Craig Tiller | b5585d4 | 2015-11-17 07:18:31 -0800 | [diff] [blame] | 513 | elem = grpc_channel_stack_element(CHANNEL_STACK_FROM_CONNECTION(con), 0); |
| 514 | elem->filter->start_transport_op(exec_ctx, elem, &op); |
| 515 | } |
| 516 | |
Craig Tiller | ab33b48 | 2015-11-21 08:11:04 -0800 | [diff] [blame] | 517 | void grpc_connected_subchannel_notify_on_state_change( |
| 518 | grpc_exec_ctx *exec_ctx, grpc_connected_subchannel *con, |
Craig Tiller | a6bebf4 | 2015-12-01 17:02:35 -0800 | [diff] [blame] | 519 | grpc_pollset_set *interested_parties, grpc_connectivity_state *state, |
| 520 | grpc_closure *closure) { |
| 521 | connected_subchannel_state_op(exec_ctx, con, interested_parties, state, |
| 522 | closure); |
Craig Tiller | b5585d4 | 2015-11-17 07:18:31 -0800 | [diff] [blame] | 523 | } |
| 524 | |
Craig Tiller | e2c6237 | 2015-12-07 16:11:03 -0800 | [diff] [blame] | 525 | void grpc_connected_subchannel_ping(grpc_exec_ctx *exec_ctx, |
| 526 | grpc_connected_subchannel *con, |
| 527 | grpc_closure *closure) { |
Craig Tiller | 28bf891 | 2015-12-07 16:07:04 -0800 | [diff] [blame] | 528 | grpc_transport_op op; |
| 529 | grpc_channel_element *elem; |
| 530 | memset(&op, 0, sizeof(op)); |
| 531 | op.send_ping = closure; |
| 532 | elem = grpc_channel_stack_element(CHANNEL_STACK_FROM_CONNECTION(con), 0); |
| 533 | elem->filter->start_transport_op(exec_ctx, elem, &op); |
| 534 | } |
| 535 | |
Craig Tiller | 3591482 | 2016-03-11 19:29:23 -0800 | [diff] [blame] | 536 | static void publish_transport_locked(grpc_exec_ctx *exec_ctx, |
| 537 | grpc_subchannel *c) { |
Craig Tiller | b5585d4 | 2015-11-17 07:18:31 -0800 | [diff] [blame] | 538 | grpc_connected_subchannel *con; |
Craig Tiller | 4ab82d2 | 2015-06-29 09:40:33 -0700 | [diff] [blame] | 539 | grpc_channel_stack *stk; |
Craig Tiller | b5585d4 | 2015-11-17 07:18:31 -0800 | [diff] [blame] | 540 | state_watcher *sw_subchannel; |
Craig Tiller | 5945ee1 | 2015-06-27 10:36:09 -0700 | [diff] [blame] | 541 | |
Craig Tiller | df91ba5 | 2015-06-29 10:55:46 -0700 | [diff] [blame] | 542 | /* construct channel stack */ |
Craig Tiller | 839bebe | 2016-04-06 08:07:11 -0700 | [diff] [blame] | 543 | grpc_channel_stack_builder *builder = grpc_channel_stack_builder_create(); |
| 544 | grpc_channel_stack_builder_set_channel_arguments( |
| 545 | builder, c->connecting_result.channel_args); |
| 546 | grpc_channel_stack_builder_set_transport(builder, |
| 547 | c->connecting_result.transport); |
| 548 | |
| 549 | if (grpc_channel_init_create_stack(exec_ctx, builder, |
| 550 | GRPC_CLIENT_SUBCHANNEL)) { |
| 551 | con = grpc_channel_stack_builder_finish(exec_ctx, builder, 0, 1, |
| 552 | connection_destroy, NULL); |
| 553 | } else { |
| 554 | grpc_channel_stack_builder_destroy(builder); |
| 555 | abort(); /* TODO(ctiller): what to do here (previously we just crashed) */ |
| 556 | } |
Craig Tiller | 906e3bc | 2015-11-24 07:31:31 -0800 | [diff] [blame] | 557 | stk = CHANNEL_STACK_FROM_CONNECTION(con); |
Craig Tiller | a82950e | 2015-09-22 12:33:20 -0700 | [diff] [blame] | 558 | memset(&c->connecting_result, 0, sizeof(c->connecting_result)); |
Craig Tiller | ff54c92 | 2015-06-26 16:57:20 -0700 | [diff] [blame] | 559 | |
Craig Tiller | df91ba5 | 2015-06-29 10:55:46 -0700 | [diff] [blame] | 560 | /* initialize state watcher */ |
Craig Tiller | b5585d4 | 2015-11-17 07:18:31 -0800 | [diff] [blame] | 561 | sw_subchannel = gpr_malloc(sizeof(*sw_subchannel)); |
Craig Tiller | 4861304 | 2015-11-29 14:45:11 -0800 | [diff] [blame] | 562 | sw_subchannel->subchannel = c; |
Craig Tiller | b5585d4 | 2015-11-17 07:18:31 -0800 | [diff] [blame] | 563 | sw_subchannel->connectivity_state = GRPC_CHANNEL_READY; |
Craig Tiller | ab33b48 | 2015-11-21 08:11:04 -0800 | [diff] [blame] | 564 | grpc_closure_init(&sw_subchannel->closure, subchannel_on_child_state_changed, |
| 565 | sw_subchannel); |
Craig Tiller | df91ba5 | 2015-06-29 10:55:46 -0700 | [diff] [blame] | 566 | |
Craig Tiller | a82950e | 2015-09-22 12:33:20 -0700 | [diff] [blame] | 567 | if (c->disconnected) { |
Craig Tiller | b5585d4 | 2015-11-17 07:18:31 -0800 | [diff] [blame] | 568 | gpr_free(sw_subchannel); |
Craig Tiller | a82950e | 2015-09-22 12:33:20 -0700 | [diff] [blame] | 569 | grpc_channel_stack_destroy(exec_ctx, stk); |
Craig Tiller | 50ec267 | 2015-11-27 21:45:11 -0800 | [diff] [blame] | 570 | gpr_free(con); |
| 571 | GRPC_SUBCHANNEL_WEAK_UNREF(exec_ctx, c, "connecting"); |
Craig Tiller | a82950e | 2015-09-22 12:33:20 -0700 | [diff] [blame] | 572 | return; |
| 573 | } |
Craig Tiller | b6fbf1d | 2015-06-29 15:25:49 -0700 | [diff] [blame] | 574 | |
Craig Tiller | df91ba5 | 2015-06-29 10:55:46 -0700 | [diff] [blame] | 575 | /* publish */ |
Craig Tiller | 493c151 | 2016-01-27 11:55:44 -0800 | [diff] [blame] | 576 | /* TODO(ctiller): this full barrier seems to clear up a TSAN failure. |
| 577 | I'd have expected the rel_cas below to be enough, but |
| 578 | seemingly it's not. |
| 579 | Re-evaluate if we really need this. */ |
Craig Tiller | 9ebc87a | 2016-01-26 15:10:09 +0000 | [diff] [blame] | 580 | gpr_atm_full_barrier(); |
Craig Tiller | d9d474a | 2016-01-26 06:50:51 -0800 | [diff] [blame] | 581 | GPR_ASSERT(gpr_atm_rel_cas(&c->connected_subchannel, 0, (gpr_atm)con)); |
Craig Tiller | 4ab82d2 | 2015-06-29 09:40:33 -0700 | [diff] [blame] | 582 | c->connecting = 0; |
Craig Tiller | df91ba5 | 2015-06-29 10:55:46 -0700 | [diff] [blame] | 583 | |
Craig Tiller | 839bebe | 2016-04-06 08:07:11 -0700 | [diff] [blame] | 584 | /* setup subchannel watching connected subchannel for changes; subchannel |
| 585 | ref |
Craig Tiller | ab33b48 | 2015-11-21 08:11:04 -0800 | [diff] [blame] | 586 | for connecting is donated |
Craig Tiller | df91ba5 | 2015-06-29 10:55:46 -0700 | [diff] [blame] | 587 | to the state watcher */ |
Craig Tiller | 12ad5d6 | 2015-11-27 12:17:30 -0800 | [diff] [blame] | 588 | GRPC_SUBCHANNEL_WEAK_REF(c, "state_watcher"); |
Craig Tiller | 50ec267 | 2015-11-27 21:45:11 -0800 | [diff] [blame] | 589 | GRPC_SUBCHANNEL_WEAK_UNREF(exec_ctx, c, "connecting"); |
Craig Tiller | ab33b48 | 2015-11-21 08:11:04 -0800 | [diff] [blame] | 590 | grpc_connected_subchannel_notify_on_state_change( |
Craig Tiller | 69b093b | 2016-02-25 19:04:07 -0800 | [diff] [blame] | 591 | exec_ctx, con, c->pollset_set, &sw_subchannel->connectivity_state, |
Craig Tiller | ab33b48 | 2015-11-21 08:11:04 -0800 | [diff] [blame] | 592 | &sw_subchannel->closure); |
Craig Tiller | df91ba5 | 2015-06-29 10:55:46 -0700 | [diff] [blame] | 593 | |
| 594 | /* signal completion */ |
Craig Tiller | ab33b48 | 2015-11-21 08:11:04 -0800 | [diff] [blame] | 595 | grpc_connectivity_state_set(exec_ctx, &c->state_tracker, GRPC_CHANNEL_READY, |
Craig Tiller | 804ff71 | 2016-05-05 16:25:40 -0700 | [diff] [blame] | 596 | GRPC_ERROR_NONE, "connected"); |
Craig Tiller | 4ab82d2 | 2015-06-29 09:40:33 -0700 | [diff] [blame] | 597 | } |
Craig Tiller | ff54c92 | 2015-06-26 16:57:20 -0700 | [diff] [blame] | 598 | |
Craig Tiller | 804ff71 | 2016-05-05 16:25:40 -0700 | [diff] [blame] | 599 | static void on_alarm(grpc_exec_ctx *exec_ctx, void *arg, grpc_error *error) { |
Craig Tiller | ff3ae68 | 2015-06-29 17:44:04 -0700 | [diff] [blame] | 600 | grpc_subchannel *c = arg; |
Craig Tiller | a82950e | 2015-09-22 12:33:20 -0700 | [diff] [blame] | 601 | gpr_mu_lock(&c->mu); |
Craig Tiller | ff3ae68 | 2015-06-29 17:44:04 -0700 | [diff] [blame] | 602 | c->have_alarm = 0; |
Craig Tiller | a82950e | 2015-09-22 12:33:20 -0700 | [diff] [blame] | 603 | if (c->disconnected) { |
Craig Tiller | 804ff71 | 2016-05-05 16:25:40 -0700 | [diff] [blame] | 604 | error = GRPC_ERROR_CREATE_REFERENCING("Disconnected", &error, 1); |
| 605 | } else { |
Craig Tiller | f707d62 | 2016-05-06 14:26:12 -0700 | [diff] [blame] | 606 | GRPC_ERROR_REF(error); |
Craig Tiller | a82950e | 2015-09-22 12:33:20 -0700 | [diff] [blame] | 607 | } |
Craig Tiller | cfc8ae1 | 2016-05-10 20:49:54 -0700 | [diff] [blame] | 608 | if (error == GRPC_ERROR_NONE) { |
Craig Tiller | e0d6c57 | 2016-05-13 07:23:36 -0700 | [diff] [blame] | 609 | gpr_log(GPR_INFO, "Failed to connect to channel, retrying"); |
Craig Tiller | 2b59dbc | 2016-05-13 15:59:09 -0700 | [diff] [blame] | 610 | c->next_attempt = |
| 611 | gpr_backoff_step(&c->backoff_state, gpr_now(GPR_CLOCK_MONOTONIC)); |
Craig Tiller | a82950e | 2015-09-22 12:33:20 -0700 | [diff] [blame] | 612 | continue_connect(exec_ctx, c); |
Craig Tiller | 8e19f61 | 2016-03-01 21:41:13 -0800 | [diff] [blame] | 613 | gpr_mu_unlock(&c->mu); |
Craig Tiller | a82950e | 2015-09-22 12:33:20 -0700 | [diff] [blame] | 614 | } else { |
Craig Tiller | 8e19f61 | 2016-03-01 21:41:13 -0800 | [diff] [blame] | 615 | gpr_mu_unlock(&c->mu); |
Craig Tiller | 50ec267 | 2015-11-27 21:45:11 -0800 | [diff] [blame] | 616 | GRPC_SUBCHANNEL_WEAK_UNREF(exec_ctx, c, "connecting"); |
Craig Tiller | a82950e | 2015-09-22 12:33:20 -0700 | [diff] [blame] | 617 | } |
Craig Tiller | 82c63eb | 2016-05-10 15:28:01 -0700 | [diff] [blame] | 618 | GRPC_ERROR_UNREF(error); |
Craig Tiller | 45724b3 | 2015-09-22 10:42:19 -0700 | [diff] [blame] | 619 | } |
| 620 | |
Craig Tiller | a82950e | 2015-09-22 12:33:20 -0700 | [diff] [blame] | 621 | static void subchannel_connected(grpc_exec_ctx *exec_ctx, void *arg, |
Craig Tiller | 804ff71 | 2016-05-05 16:25:40 -0700 | [diff] [blame] | 622 | grpc_error *error) { |
Craig Tiller | 45724b3 | 2015-09-22 10:42:19 -0700 | [diff] [blame] | 623 | grpc_subchannel *c = arg; |
Craig Tiller | 99b642a | 2016-06-01 17:20:34 -0700 | [diff] [blame] | 624 | grpc_channel_args *delete_channel_args = c->connecting_result.channel_args; |
Craig Tiller | 50ec267 | 2015-11-27 21:45:11 -0800 | [diff] [blame] | 625 | |
Craig Tiller | 2ba4133 | 2016-03-11 17:09:12 -0800 | [diff] [blame] | 626 | GRPC_SUBCHANNEL_WEAK_REF(c, "connected"); |
| 627 | gpr_mu_lock(&c->mu); |
Craig Tiller | a82950e | 2015-09-22 12:33:20 -0700 | [diff] [blame] | 628 | if (c->connecting_result.transport != NULL) { |
Craig Tiller | 3591482 | 2016-03-11 19:29:23 -0800 | [diff] [blame] | 629 | publish_transport_locked(exec_ctx, c); |
Craig Tiller | b5585d4 | 2015-11-17 07:18:31 -0800 | [diff] [blame] | 630 | } else if (c->disconnected) { |
Craig Tiller | 50ec267 | 2015-11-27 21:45:11 -0800 | [diff] [blame] | 631 | GRPC_SUBCHANNEL_WEAK_UNREF(exec_ctx, c, "connecting"); |
Craig Tiller | a82950e | 2015-09-22 12:33:20 -0700 | [diff] [blame] | 632 | } else { |
| 633 | gpr_timespec now = gpr_now(GPR_CLOCK_MONOTONIC); |
Craig Tiller | a82950e | 2015-09-22 12:33:20 -0700 | [diff] [blame] | 634 | GPR_ASSERT(!c->have_alarm); |
| 635 | c->have_alarm = 1; |
Craig Tiller | 804ff71 | 2016-05-05 16:25:40 -0700 | [diff] [blame] | 636 | grpc_connectivity_state_set( |
| 637 | exec_ctx, &c->state_tracker, GRPC_CHANNEL_TRANSIENT_FAILURE, |
| 638 | GRPC_ERROR_CREATE_REFERENCING("Connect Failed", &error, 1), |
| 639 | "connect_failed"); |
Craig Tiller | e0d6c57 | 2016-05-13 07:23:36 -0700 | [diff] [blame] | 640 | gpr_timespec time_til_next = gpr_time_sub(c->next_attempt, now); |
Craig Tiller | cfc8ae1 | 2016-05-10 20:49:54 -0700 | [diff] [blame] | 641 | const char *errmsg = grpc_error_string(error); |
Craig Tiller | 2ec184b | 2016-06-01 13:03:17 -0700 | [diff] [blame] | 642 | gpr_log(GPR_INFO, "Connect failed: %s", errmsg); |
Craig Tiller | df30bc5 | 2016-06-01 14:08:50 -0700 | [diff] [blame] | 643 | if (gpr_time_cmp(time_til_next, gpr_time_0(time_til_next.clock_type)) <= |
| 644 | 0) { |
Craig Tiller | 2ec184b | 2016-06-01 13:03:17 -0700 | [diff] [blame] | 645 | gpr_log(GPR_INFO, "Retry immediately"); |
| 646 | } else { |
Craig Tiller | 7dce092 | 2016-06-15 10:11:11 -0700 | [diff] [blame] | 647 | gpr_log(GPR_INFO, "Retry in %" PRId64 ".%09d seconds", |
| 648 | time_til_next.tv_sec, time_til_next.tv_nsec); |
Craig Tiller | 2ec184b | 2016-06-01 13:03:17 -0700 | [diff] [blame] | 649 | } |
David Garcia Quintas | f747bbc | 2015-10-04 23:09:47 -0700 | [diff] [blame] | 650 | grpc_timer_init(exec_ctx, &c->alarm, c->next_attempt, on_alarm, c, now); |
Craig Tiller | 2ec184b | 2016-06-01 13:03:17 -0700 | [diff] [blame] | 651 | grpc_error_free_string(errmsg); |
Craig Tiller | a82950e | 2015-09-22 12:33:20 -0700 | [diff] [blame] | 652 | } |
Craig Tiller | 2ba4133 | 2016-03-11 17:09:12 -0800 | [diff] [blame] | 653 | gpr_mu_unlock(&c->mu); |
| 654 | GRPC_SUBCHANNEL_WEAK_UNREF(exec_ctx, c, "connecting"); |
Craig Tiller | 99b642a | 2016-06-01 17:20:34 -0700 | [diff] [blame] | 655 | grpc_channel_args_destroy(delete_channel_args); |
Craig Tiller | 45724b3 | 2015-09-22 10:42:19 -0700 | [diff] [blame] | 656 | } |
| 657 | |
Craig Tiller | 2595ab7 | 2015-06-25 15:26:00 -0700 | [diff] [blame] | 658 | /* |
| 659 | * grpc_subchannel_call implementation |
| 660 | */ |
| 661 | |
Craig Tiller | 577c9b2 | 2015-11-02 14:11:15 -0800 | [diff] [blame] | 662 | static void subchannel_call_destroy(grpc_exec_ctx *exec_ctx, void *call, |
Craig Tiller | 804ff71 | 2016-05-05 16:25:40 -0700 | [diff] [blame] | 663 | grpc_error *error) { |
Craig Tiller | 577c9b2 | 2015-11-02 14:11:15 -0800 | [diff] [blame] | 664 | grpc_subchannel_call *c = call; |
Craig Tiller | 577c9b2 | 2015-11-02 14:11:15 -0800 | [diff] [blame] | 665 | GPR_TIMER_BEGIN("grpc_subchannel_call_unref.destroy", 0); |
Craig Tiller | 2c8063c | 2016-03-22 22:12:15 -0700 | [diff] [blame] | 666 | grpc_connected_subchannel *connection = c->connection; |
David Garcia Quintas | 580987a | 2016-04-29 17:26:33 -0700 | [diff] [blame] | 667 | grpc_call_stack_destroy(exec_ctx, SUBCHANNEL_CALL_TO_CALL_STACK(c), NULL, c); |
Craig Tiller | 2c8063c | 2016-03-22 22:12:15 -0700 | [diff] [blame] | 668 | GRPC_CONNECTED_SUBCHANNEL_UNREF(exec_ctx, connection, "subchannel_call"); |
Craig Tiller | 577c9b2 | 2015-11-02 14:11:15 -0800 | [diff] [blame] | 669 | GPR_TIMER_END("grpc_subchannel_call_unref.destroy", 0); |
| 670 | } |
| 671 | |
Craig Tiller | f40df23 | 2016-03-25 13:38:14 -0700 | [diff] [blame] | 672 | void grpc_subchannel_call_ref( |
| 673 | grpc_subchannel_call *c GRPC_SUBCHANNEL_REF_EXTRA_ARGS) { |
Craig Tiller | 906e3bc | 2015-11-24 07:31:31 -0800 | [diff] [blame] | 674 | GRPC_CALL_STACK_REF(SUBCHANNEL_CALL_TO_CALL_STACK(c), REF_REASON); |
Craig Tiller | c396753 | 2015-06-29 14:59:38 -0700 | [diff] [blame] | 675 | } |
Craig Tiller | 2595ab7 | 2015-06-25 15:26:00 -0700 | [diff] [blame] | 676 | |
Craig Tiller | a82950e | 2015-09-22 12:33:20 -0700 | [diff] [blame] | 677 | void grpc_subchannel_call_unref(grpc_exec_ctx *exec_ctx, |
| 678 | grpc_subchannel_call *c |
| 679 | GRPC_SUBCHANNEL_REF_EXTRA_ARGS) { |
Craig Tiller | 906e3bc | 2015-11-24 07:31:31 -0800 | [diff] [blame] | 680 | GRPC_CALL_STACK_UNREF(exec_ctx, SUBCHANNEL_CALL_TO_CALL_STACK(c), REF_REASON); |
Craig Tiller | 2595ab7 | 2015-06-25 15:26:00 -0700 | [diff] [blame] | 681 | } |
| 682 | |
Craig Tiller | a82950e | 2015-09-22 12:33:20 -0700 | [diff] [blame] | 683 | char *grpc_subchannel_call_get_peer(grpc_exec_ctx *exec_ctx, |
| 684 | grpc_subchannel_call *call) { |
| 685 | grpc_call_stack *call_stack = SUBCHANNEL_CALL_TO_CALL_STACK(call); |
| 686 | grpc_call_element *top_elem = grpc_call_stack_element(call_stack, 0); |
| 687 | return top_elem->filter->get_peer(exec_ctx, top_elem); |
Craig Tiller | 1b22b9d | 2015-07-20 13:42:22 -0700 | [diff] [blame] | 688 | } |
| 689 | |
Craig Tiller | a82950e | 2015-09-22 12:33:20 -0700 | [diff] [blame] | 690 | void grpc_subchannel_call_process_op(grpc_exec_ctx *exec_ctx, |
| 691 | grpc_subchannel_call *call, |
| 692 | grpc_transport_stream_op *op) { |
Craig Tiller | bfc9adc | 2016-06-27 13:16:22 -0700 | [diff] [blame] | 693 | GPR_TIMER_BEGIN("grpc_subchannel_call_process_op", 0); |
Craig Tiller | a82950e | 2015-09-22 12:33:20 -0700 | [diff] [blame] | 694 | grpc_call_stack *call_stack = SUBCHANNEL_CALL_TO_CALL_STACK(call); |
| 695 | grpc_call_element *top_elem = grpc_call_stack_element(call_stack, 0); |
| 696 | top_elem->filter->start_transport_stream_op(exec_ctx, top_elem, op); |
Craig Tiller | bfc9adc | 2016-06-27 13:16:22 -0700 | [diff] [blame] | 697 | GPR_TIMER_END("grpc_subchannel_call_process_op", 0); |
Craig Tiller | 2595ab7 | 2015-06-25 15:26:00 -0700 | [diff] [blame] | 698 | } |
Craig Tiller | eb3b12e | 2015-06-26 14:42:49 -0700 | [diff] [blame] | 699 | |
Craig Tiller | ab33b48 | 2015-11-21 08:11:04 -0800 | [diff] [blame] | 700 | grpc_connected_subchannel *grpc_subchannel_get_connected_subchannel( |
| 701 | grpc_subchannel *c) { |
Craig Tiller | b5585d4 | 2015-11-17 07:18:31 -0800 | [diff] [blame] | 702 | return GET_CONNECTED_SUBCHANNEL(c, acq); |
| 703 | } |
| 704 | |
Craig Tiller | ab33b48 | 2015-11-21 08:11:04 -0800 | [diff] [blame] | 705 | grpc_subchannel_call *grpc_connected_subchannel_create_call( |
| 706 | grpc_exec_ctx *exec_ctx, grpc_connected_subchannel *con, |
David Garcia Quintas | 2a50dfe | 2016-05-31 15:09:12 -0700 | [diff] [blame] | 707 | grpc_polling_entity *pollent) { |
Craig Tiller | a82950e | 2015-09-22 12:33:20 -0700 | [diff] [blame] | 708 | grpc_channel_stack *chanstk = CHANNEL_STACK_FROM_CONNECTION(con); |
| 709 | grpc_subchannel_call *call = |
| 710 | gpr_malloc(sizeof(grpc_subchannel_call) + chanstk->call_stack_size); |
| 711 | grpc_call_stack *callstk = SUBCHANNEL_CALL_TO_CALL_STACK(call); |
Craig Tiller | 04c5d4b | 2015-06-26 17:21:41 -0700 | [diff] [blame] | 712 | call->connection = con; |
Craig Tiller | 53ee040 | 2015-11-25 09:07:09 -0800 | [diff] [blame] | 713 | GRPC_CONNECTED_SUBCHANNEL_REF(con, "subchannel_call"); |
Craig Tiller | 577c9b2 | 2015-11-02 14:11:15 -0800 | [diff] [blame] | 714 | grpc_call_stack_init(exec_ctx, chanstk, 1, subchannel_call_destroy, call, |
| 715 | NULL, NULL, callstk); |
David Garcia Quintas | 2a50dfe | 2016-05-31 15:09:12 -0700 | [diff] [blame] | 716 | grpc_call_stack_set_pollset_or_pollset_set(exec_ctx, callstk, pollent); |
Craig Tiller | 04c5d4b | 2015-06-26 17:21:41 -0700 | [diff] [blame] | 717 | return call; |
Craig Tiller | eb3b12e | 2015-06-26 14:42:49 -0700 | [diff] [blame] | 718 | } |
David Garcia Quintas | 7b1bd2c | 2015-10-05 18:22:10 -0700 | [diff] [blame] | 719 | |
Craig Tiller | 577c9b2 | 2015-11-02 14:11:15 -0800 | [diff] [blame] | 720 | grpc_call_stack *grpc_subchannel_call_get_call_stack( |
| 721 | grpc_subchannel_call *subchannel_call) { |
| 722 | return SUBCHANNEL_CALL_TO_CALL_STACK(subchannel_call); |
| 723 | } |