Craig Tiller | 577c9b2 | 2015-11-02 14:11:15 -0800 | [diff] [blame] | 1 | /* |
| 2 | * |
Craig Tiller | 6169d5f | 2016-03-31 07:46:18 -0700 | [diff] [blame] | 3 | * Copyright 2015, Google Inc. |
Craig Tiller | 577c9b2 | 2015-11-02 14:11:15 -0800 | [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, |
| 25 | * SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT |
| 26 | * LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, |
| 27 | * DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY |
| 28 | * THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT |
| 29 | * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE |
| 30 | * OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. |
| 31 | * |
| 32 | */ |
| 33 | |
Craig Tiller | b112f39 | 2016-04-05 12:44:04 -0700 | [diff] [blame] | 34 | #include "src/core/ext/client_config/subchannel_call_holder.h" |
Craig Tiller | 577c9b2 | 2015-11-02 14:11:15 -0800 | [diff] [blame] | 35 | |
| 36 | #include <grpc/support/alloc.h> |
| 37 | |
Craig Tiller | 9533d04 | 2016-03-25 17:11:06 -0700 | [diff] [blame] | 38 | #include "src/core/lib/profiling/timers.h" |
Craig Tiller | 577c9b2 | 2015-11-02 14:11:15 -0800 | [diff] [blame] | 39 | |
| 40 | #define GET_CALL(holder) \ |
| 41 | ((grpc_subchannel_call *)(gpr_atm_acq_load(&(holder)->subchannel_call))) |
| 42 | |
| 43 | #define CANCELLED_CALL ((grpc_subchannel_call *)1) |
| 44 | |
| 45 | static void subchannel_ready(grpc_exec_ctx *exec_ctx, void *holder, |
Craig Tiller | 804ff71 | 2016-05-05 16:25:40 -0700 | [diff] [blame] | 46 | grpc_error *error); |
Craig Tiller | 577c9b2 | 2015-11-02 14:11:15 -0800 | [diff] [blame] | 47 | static void retry_ops(grpc_exec_ctx *exec_ctx, void *retry_ops_args, |
Craig Tiller | 804ff71 | 2016-05-05 16:25:40 -0700 | [diff] [blame] | 48 | grpc_error *error); |
Craig Tiller | 577c9b2 | 2015-11-02 14:11:15 -0800 | [diff] [blame] | 49 | |
| 50 | static void add_waiting_locked(grpc_subchannel_call_holder *holder, |
| 51 | grpc_transport_stream_op *op); |
| 52 | static void fail_locked(grpc_exec_ctx *exec_ctx, |
Craig Tiller | 804ff71 | 2016-05-05 16:25:40 -0700 | [diff] [blame] | 53 | grpc_subchannel_call_holder *holder, grpc_error *error); |
Craig Tiller | 577c9b2 | 2015-11-02 14:11:15 -0800 | [diff] [blame] | 54 | static void retry_waiting_locked(grpc_exec_ctx *exec_ctx, |
| 55 | grpc_subchannel_call_holder *holder); |
| 56 | |
| 57 | void grpc_subchannel_call_holder_init( |
| 58 | grpc_subchannel_call_holder *holder, |
| 59 | grpc_subchannel_call_holder_pick_subchannel pick_subchannel, |
Craig Tiller | 11beb9a | 2015-11-24 10:29:32 -0800 | [diff] [blame] | 60 | void *pick_subchannel_arg, grpc_call_stack *owning_call) { |
Craig Tiller | 577c9b2 | 2015-11-02 14:11:15 -0800 | [diff] [blame] | 61 | gpr_atm_rel_store(&holder->subchannel_call, 0); |
| 62 | holder->pick_subchannel = pick_subchannel; |
| 63 | holder->pick_subchannel_arg = pick_subchannel_arg; |
| 64 | gpr_mu_init(&holder->mu); |
Craig Tiller | b5585d4 | 2015-11-17 07:18:31 -0800 | [diff] [blame] | 65 | holder->connected_subchannel = NULL; |
Craig Tiller | 577c9b2 | 2015-11-02 14:11:15 -0800 | [diff] [blame] | 66 | holder->waiting_ops = NULL; |
| 67 | holder->waiting_ops_count = 0; |
| 68 | holder->waiting_ops_capacity = 0; |
| 69 | holder->creation_phase = GRPC_SUBCHANNEL_CALL_HOLDER_NOT_CREATING; |
Craig Tiller | 11beb9a | 2015-11-24 10:29:32 -0800 | [diff] [blame] | 70 | holder->owning_call = owning_call; |
David Garcia Quintas | 2a50dfe | 2016-05-31 15:09:12 -0700 | [diff] [blame] | 71 | holder->pollent = NULL; |
Craig Tiller | 577c9b2 | 2015-11-02 14:11:15 -0800 | [diff] [blame] | 72 | } |
| 73 | |
| 74 | void grpc_subchannel_call_holder_destroy(grpc_exec_ctx *exec_ctx, |
| 75 | grpc_subchannel_call_holder *holder) { |
| 76 | grpc_subchannel_call *call = GET_CALL(holder); |
| 77 | if (call != NULL && call != CANCELLED_CALL) { |
| 78 | GRPC_SUBCHANNEL_CALL_UNREF(exec_ctx, call, "holder"); |
| 79 | } |
| 80 | GPR_ASSERT(holder->creation_phase == |
| 81 | GRPC_SUBCHANNEL_CALL_HOLDER_NOT_CREATING); |
| 82 | gpr_mu_destroy(&holder->mu); |
| 83 | GPR_ASSERT(holder->waiting_ops_count == 0); |
| 84 | gpr_free(holder->waiting_ops); |
| 85 | } |
| 86 | |
Mark D. Roth | 09e6698 | 2016-06-23 11:14:18 -0700 | [diff] [blame] | 87 | // The logic here is fairly complicated, due to (a) the fact that we |
| 88 | // need to handle the case where we receive the send op before the |
| 89 | // initial metadata op, and (b) the need for efficiency, especially in |
| 90 | // the streaming case. |
| 91 | // TODO(ctiller): Explain this more thoroughly. |
Craig Tiller | 577c9b2 | 2015-11-02 14:11:15 -0800 | [diff] [blame] | 92 | void grpc_subchannel_call_holder_perform_op(grpc_exec_ctx *exec_ctx, |
| 93 | grpc_subchannel_call_holder *holder, |
| 94 | grpc_transport_stream_op *op) { |
| 95 | /* try to (atomically) get the call */ |
| 96 | grpc_subchannel_call *call = GET_CALL(holder); |
| 97 | GPR_TIMER_BEGIN("grpc_subchannel_call_holder_perform_op", 0); |
| 98 | if (call == CANCELLED_CALL) { |
Craig Tiller | 804ff71 | 2016-05-05 16:25:40 -0700 | [diff] [blame] | 99 | grpc_transport_stream_op_finish_with_failure(exec_ctx, op, |
| 100 | GRPC_ERROR_CANCELLED); |
Craig Tiller | 577c9b2 | 2015-11-02 14:11:15 -0800 | [diff] [blame] | 101 | GPR_TIMER_END("grpc_subchannel_call_holder_perform_op", 0); |
| 102 | return; |
| 103 | } |
| 104 | if (call != NULL) { |
| 105 | grpc_subchannel_call_process_op(exec_ctx, call, op); |
| 106 | GPR_TIMER_END("grpc_subchannel_call_holder_perform_op", 0); |
| 107 | return; |
| 108 | } |
| 109 | /* we failed; lock and figure out what to do */ |
| 110 | gpr_mu_lock(&holder->mu); |
| 111 | retry: |
| 112 | /* need to recheck that another thread hasn't set the call */ |
| 113 | call = GET_CALL(holder); |
| 114 | if (call == CANCELLED_CALL) { |
| 115 | gpr_mu_unlock(&holder->mu); |
Craig Tiller | 804ff71 | 2016-05-05 16:25:40 -0700 | [diff] [blame] | 116 | grpc_transport_stream_op_finish_with_failure(exec_ctx, op, |
| 117 | GRPC_ERROR_CANCELLED); |
Craig Tiller | 577c9b2 | 2015-11-02 14:11:15 -0800 | [diff] [blame] | 118 | GPR_TIMER_END("grpc_subchannel_call_holder_perform_op", 0); |
| 119 | return; |
| 120 | } |
| 121 | if (call != NULL) { |
| 122 | gpr_mu_unlock(&holder->mu); |
| 123 | grpc_subchannel_call_process_op(exec_ctx, call, op); |
| 124 | GPR_TIMER_END("grpc_subchannel_call_holder_perform_op", 0); |
| 125 | return; |
| 126 | } |
| 127 | /* if this is a cancellation, then we can raise our cancelled flag */ |
Craig Tiller | f0f70a8 | 2016-06-23 13:55:06 -0700 | [diff] [blame] | 128 | if (op->cancel_error != GRPC_ERROR_NONE) { |
Mark D. Roth | 09e6698 | 2016-06-23 11:14:18 -0700 | [diff] [blame] | 129 | if (!gpr_atm_rel_cas(&holder->subchannel_call, 0, |
| 130 | (gpr_atm)(uintptr_t)CANCELLED_CALL)) { |
Craig Tiller | 577c9b2 | 2015-11-02 14:11:15 -0800 | [diff] [blame] | 131 | goto retry; |
| 132 | } else { |
| 133 | switch (holder->creation_phase) { |
| 134 | case GRPC_SUBCHANNEL_CALL_HOLDER_NOT_CREATING: |
Craig Tiller | f0f70a8 | 2016-06-23 13:55:06 -0700 | [diff] [blame] | 135 | fail_locked(exec_ctx, holder, GRPC_ERROR_REF(op->cancel_error)); |
Craig Tiller | 577c9b2 | 2015-11-02 14:11:15 -0800 | [diff] [blame] | 136 | break; |
Craig Tiller | 577c9b2 | 2015-11-02 14:11:15 -0800 | [diff] [blame] | 137 | case GRPC_SUBCHANNEL_CALL_HOLDER_PICKING_SUBCHANNEL: |
| 138 | holder->pick_subchannel(exec_ctx, holder->pick_subchannel_arg, NULL, |
Craig Tiller | 8c0d96f | 2016-03-11 14:27:52 -0800 | [diff] [blame] | 139 | 0, &holder->connected_subchannel, NULL); |
Craig Tiller | 577c9b2 | 2015-11-02 14:11:15 -0800 | [diff] [blame] | 140 | break; |
| 141 | } |
| 142 | gpr_mu_unlock(&holder->mu); |
Craig Tiller | 804ff71 | 2016-05-05 16:25:40 -0700 | [diff] [blame] | 143 | grpc_transport_stream_op_finish_with_failure(exec_ctx, op, |
| 144 | GRPC_ERROR_CANCELLED); |
Craig Tiller | 577c9b2 | 2015-11-02 14:11:15 -0800 | [diff] [blame] | 145 | GPR_TIMER_END("grpc_subchannel_call_holder_perform_op", 0); |
| 146 | return; |
| 147 | } |
| 148 | } |
| 149 | /* if we don't have a subchannel, try to get one */ |
| 150 | if (holder->creation_phase == GRPC_SUBCHANNEL_CALL_HOLDER_NOT_CREATING && |
Craig Tiller | ab33b48 | 2015-11-21 08:11:04 -0800 | [diff] [blame] | 151 | holder->connected_subchannel == NULL && |
| 152 | op->send_initial_metadata != NULL) { |
Craig Tiller | 577c9b2 | 2015-11-02 14:11:15 -0800 | [diff] [blame] | 153 | holder->creation_phase = GRPC_SUBCHANNEL_CALL_HOLDER_PICKING_SUBCHANNEL; |
| 154 | grpc_closure_init(&holder->next_step, subchannel_ready, holder); |
Craig Tiller | 11beb9a | 2015-11-24 10:29:32 -0800 | [diff] [blame] | 155 | GRPC_CALL_STACK_REF(holder->owning_call, "pick_subchannel"); |
Craig Tiller | ab33b48 | 2015-11-21 08:11:04 -0800 | [diff] [blame] | 156 | if (holder->pick_subchannel( |
| 157 | exec_ctx, holder->pick_subchannel_arg, op->send_initial_metadata, |
Craig Tiller | 8c0d96f | 2016-03-11 14:27:52 -0800 | [diff] [blame] | 158 | op->send_initial_metadata_flags, &holder->connected_subchannel, |
| 159 | &holder->next_step)) { |
Craig Tiller | 577c9b2 | 2015-11-02 14:11:15 -0800 | [diff] [blame] | 160 | holder->creation_phase = GRPC_SUBCHANNEL_CALL_HOLDER_NOT_CREATING; |
Craig Tiller | 11beb9a | 2015-11-24 10:29:32 -0800 | [diff] [blame] | 161 | GRPC_CALL_STACK_UNREF(exec_ctx, holder->owning_call, "pick_subchannel"); |
Craig Tiller | 577c9b2 | 2015-11-02 14:11:15 -0800 | [diff] [blame] | 162 | } |
| 163 | } |
| 164 | /* if we've got a subchannel, then let's ask it to create a call */ |
| 165 | if (holder->creation_phase == GRPC_SUBCHANNEL_CALL_HOLDER_NOT_CREATING && |
Craig Tiller | b5585d4 | 2015-11-17 07:18:31 -0800 | [diff] [blame] | 166 | holder->connected_subchannel != NULL) { |
Mark D. Roth | 76d2442 | 2016-06-23 13:22:10 -0700 | [diff] [blame] | 167 | grpc_subchannel_call *subchannel_call = NULL; |
| 168 | grpc_error *error = grpc_connected_subchannel_create_call( |
Mark D. Roth | 09e6698 | 2016-06-23 11:14:18 -0700 | [diff] [blame] | 169 | exec_ctx, holder->connected_subchannel, holder->pollent, |
| 170 | &subchannel_call); |
| 171 | if (error != GRPC_ERROR_NONE) { |
| 172 | subchannel_call = CANCELLED_CALL; |
Mark D. Roth | 05d73af | 2016-07-27 15:52:46 +0000 | [diff] [blame] | 173 | fail_locked(exec_ctx, holder, GRPC_ERROR_REF(error)); |
Mark D. Roth | 09e6698 | 2016-06-23 11:14:18 -0700 | [diff] [blame] | 174 | grpc_transport_stream_op_finish_with_failure(exec_ctx, op, error); |
| 175 | } |
| 176 | gpr_atm_rel_store(&holder->subchannel_call, |
| 177 | (gpr_atm)(uintptr_t)subchannel_call); |
Craig Tiller | b5585d4 | 2015-11-17 07:18:31 -0800 | [diff] [blame] | 178 | retry_waiting_locked(exec_ctx, holder); |
| 179 | goto retry; |
Craig Tiller | 577c9b2 | 2015-11-02 14:11:15 -0800 | [diff] [blame] | 180 | } |
| 181 | /* nothing to be done but wait */ |
| 182 | add_waiting_locked(holder, op); |
| 183 | gpr_mu_unlock(&holder->mu); |
| 184 | GPR_TIMER_END("grpc_subchannel_call_holder_perform_op", 0); |
| 185 | } |
| 186 | |
Craig Tiller | 804ff71 | 2016-05-05 16:25:40 -0700 | [diff] [blame] | 187 | static void subchannel_ready(grpc_exec_ctx *exec_ctx, void *arg, |
| 188 | grpc_error *error) { |
Craig Tiller | 577c9b2 | 2015-11-02 14:11:15 -0800 | [diff] [blame] | 189 | grpc_subchannel_call_holder *holder = arg; |
Craig Tiller | 577c9b2 | 2015-11-02 14:11:15 -0800 | [diff] [blame] | 190 | gpr_mu_lock(&holder->mu); |
| 191 | GPR_ASSERT(holder->creation_phase == |
| 192 | GRPC_SUBCHANNEL_CALL_HOLDER_PICKING_SUBCHANNEL); |
Craig Tiller | 11beb9a | 2015-11-24 10:29:32 -0800 | [diff] [blame] | 193 | holder->creation_phase = GRPC_SUBCHANNEL_CALL_HOLDER_NOT_CREATING; |
Craig Tiller | b5585d4 | 2015-11-17 07:18:31 -0800 | [diff] [blame] | 194 | if (holder->connected_subchannel == NULL) { |
Sree Kuchibhotla | ad0f792 | 2016-05-04 19:49:31 -0700 | [diff] [blame] | 195 | gpr_atm_no_barrier_store(&holder->subchannel_call, 1); |
Craig Tiller | 804ff71 | 2016-05-05 16:25:40 -0700 | [diff] [blame] | 196 | fail_locked(exec_ctx, holder, |
| 197 | GRPC_ERROR_CREATE_REFERENCING("Failed to create subchannel", |
| 198 | &error, 1)); |
Craig Tiller | d426d9d | 2016-03-09 09:30:18 -0800 | [diff] [blame] | 199 | } else if (1 == gpr_atm_acq_load(&holder->subchannel_call)) { |
| 200 | /* already cancelled before subchannel became ready */ |
Craig Tiller | 804ff71 | 2016-05-05 16:25:40 -0700 | [diff] [blame] | 201 | fail_locked(exec_ctx, holder, |
| 202 | GRPC_ERROR_CREATE_REFERENCING( |
| 203 | "Cancelled before creating subchannel", &error, 1)); |
Craig Tiller | 577c9b2 | 2015-11-02 14:11:15 -0800 | [diff] [blame] | 204 | } else { |
Mark D. Roth | 76d2442 | 2016-06-23 13:22:10 -0700 | [diff] [blame] | 205 | grpc_subchannel_call *subchannel_call = NULL; |
| 206 | grpc_error *new_error = grpc_connected_subchannel_create_call( |
Mark D. Roth | 09e6698 | 2016-06-23 11:14:18 -0700 | [diff] [blame] | 207 | exec_ctx, holder->connected_subchannel, holder->pollent, |
| 208 | &subchannel_call); |
| 209 | if (new_error != GRPC_ERROR_NONE) { |
Mark D. Roth | 3c945ee | 2016-07-01 14:17:38 -0700 | [diff] [blame] | 210 | new_error = grpc_error_add_child(new_error, error); |
Mark D. Roth | 09e6698 | 2016-06-23 11:14:18 -0700 | [diff] [blame] | 211 | subchannel_call = CANCELLED_CALL; |
| 212 | fail_locked(exec_ctx, holder, new_error); |
| 213 | } |
| 214 | gpr_atm_rel_store(&holder->subchannel_call, |
| 215 | (gpr_atm)(uintptr_t)subchannel_call); |
Craig Tiller | d426d9d | 2016-03-09 09:30:18 -0800 | [diff] [blame] | 216 | retry_waiting_locked(exec_ctx, holder); |
Craig Tiller | 577c9b2 | 2015-11-02 14:11:15 -0800 | [diff] [blame] | 217 | } |
| 218 | gpr_mu_unlock(&holder->mu); |
Craig Tiller | 11beb9a | 2015-11-24 10:29:32 -0800 | [diff] [blame] | 219 | GRPC_CALL_STACK_UNREF(exec_ctx, holder->owning_call, "pick_subchannel"); |
Craig Tiller | 577c9b2 | 2015-11-02 14:11:15 -0800 | [diff] [blame] | 220 | } |
| 221 | |
| 222 | typedef struct { |
| 223 | grpc_transport_stream_op *ops; |
| 224 | size_t nops; |
| 225 | grpc_subchannel_call *call; |
| 226 | } retry_ops_args; |
| 227 | |
| 228 | static void retry_waiting_locked(grpc_exec_ctx *exec_ctx, |
| 229 | grpc_subchannel_call_holder *holder) { |
| 230 | retry_ops_args *a = gpr_malloc(sizeof(*a)); |
| 231 | a->ops = holder->waiting_ops; |
| 232 | a->nops = holder->waiting_ops_count; |
| 233 | a->call = GET_CALL(holder); |
| 234 | if (a->call == CANCELLED_CALL) { |
| 235 | gpr_free(a); |
Craig Tiller | 804ff71 | 2016-05-05 16:25:40 -0700 | [diff] [blame] | 236 | fail_locked(exec_ctx, holder, GRPC_ERROR_CANCELLED); |
Craig Tiller | 577c9b2 | 2015-11-02 14:11:15 -0800 | [diff] [blame] | 237 | return; |
| 238 | } |
| 239 | holder->waiting_ops = NULL; |
| 240 | holder->waiting_ops_count = 0; |
| 241 | holder->waiting_ops_capacity = 0; |
| 242 | GRPC_SUBCHANNEL_CALL_REF(a->call, "retry_ops"); |
Craig Tiller | 332f1b3 | 2016-05-24 13:21:21 -0700 | [diff] [blame] | 243 | grpc_exec_ctx_sched(exec_ctx, grpc_closure_create(retry_ops, a), |
| 244 | GRPC_ERROR_NONE, NULL); |
Craig Tiller | 577c9b2 | 2015-11-02 14:11:15 -0800 | [diff] [blame] | 245 | } |
| 246 | |
Craig Tiller | 804ff71 | 2016-05-05 16:25:40 -0700 | [diff] [blame] | 247 | static void retry_ops(grpc_exec_ctx *exec_ctx, void *args, grpc_error *error) { |
Craig Tiller | 577c9b2 | 2015-11-02 14:11:15 -0800 | [diff] [blame] | 248 | retry_ops_args *a = args; |
| 249 | size_t i; |
| 250 | for (i = 0; i < a->nops; i++) { |
| 251 | grpc_subchannel_call_process_op(exec_ctx, a->call, &a->ops[i]); |
| 252 | } |
| 253 | GRPC_SUBCHANNEL_CALL_UNREF(exec_ctx, a->call, "retry_ops"); |
| 254 | gpr_free(a->ops); |
| 255 | gpr_free(a); |
| 256 | } |
| 257 | |
| 258 | static void add_waiting_locked(grpc_subchannel_call_holder *holder, |
| 259 | grpc_transport_stream_op *op) { |
| 260 | GPR_TIMER_BEGIN("add_waiting_locked", 0); |
| 261 | if (holder->waiting_ops_count == holder->waiting_ops_capacity) { |
| 262 | holder->waiting_ops_capacity = GPR_MAX(3, 2 * holder->waiting_ops_capacity); |
| 263 | holder->waiting_ops = |
| 264 | gpr_realloc(holder->waiting_ops, holder->waiting_ops_capacity * |
| 265 | sizeof(*holder->waiting_ops)); |
| 266 | } |
| 267 | holder->waiting_ops[holder->waiting_ops_count++] = *op; |
| 268 | GPR_TIMER_END("add_waiting_locked", 0); |
| 269 | } |
| 270 | |
| 271 | static void fail_locked(grpc_exec_ctx *exec_ctx, |
Craig Tiller | 804ff71 | 2016-05-05 16:25:40 -0700 | [diff] [blame] | 272 | grpc_subchannel_call_holder *holder, |
| 273 | grpc_error *error) { |
Craig Tiller | 577c9b2 | 2015-11-02 14:11:15 -0800 | [diff] [blame] | 274 | size_t i; |
| 275 | for (i = 0; i < holder->waiting_ops_count; i++) { |
Craig Tiller | 804ff71 | 2016-05-05 16:25:40 -0700 | [diff] [blame] | 276 | grpc_transport_stream_op_finish_with_failure( |
Craig Tiller | f707d62 | 2016-05-06 14:26:12 -0700 | [diff] [blame] | 277 | exec_ctx, &holder->waiting_ops[i], GRPC_ERROR_REF(error)); |
Craig Tiller | 577c9b2 | 2015-11-02 14:11:15 -0800 | [diff] [blame] | 278 | } |
| 279 | holder->waiting_ops_count = 0; |
Craig Tiller | f707d62 | 2016-05-06 14:26:12 -0700 | [diff] [blame] | 280 | GRPC_ERROR_UNREF(error); |
Craig Tiller | 577c9b2 | 2015-11-02 14:11:15 -0800 | [diff] [blame] | 281 | } |
| 282 | |
Craig Tiller | 7b43561 | 2015-11-24 08:15:05 -0800 | [diff] [blame] | 283 | char *grpc_subchannel_call_holder_get_peer( |
| 284 | grpc_exec_ctx *exec_ctx, grpc_subchannel_call_holder *holder) { |
Craig Tiller | 577c9b2 | 2015-11-02 14:11:15 -0800 | [diff] [blame] | 285 | grpc_subchannel_call *subchannel_call = GET_CALL(holder); |
| 286 | |
Craig Tiller | 347e9f9 | 2016-04-19 17:09:13 -0700 | [diff] [blame] | 287 | if (subchannel_call == NULL || subchannel_call == CANCELLED_CALL) { |
Craig Tiller | 906e3bc | 2015-11-24 07:31:31 -0800 | [diff] [blame] | 288 | return NULL; |
Craig Tiller | 347e9f9 | 2016-04-19 17:09:13 -0700 | [diff] [blame] | 289 | } else { |
| 290 | return grpc_subchannel_call_get_peer(exec_ctx, subchannel_call); |
Craig Tiller | 577c9b2 | 2015-11-02 14:11:15 -0800 | [diff] [blame] | 291 | } |
| 292 | } |