Craig Tiller | 48cb07c | 2015-07-15 16:16:15 -0700 | [diff] [blame] | 1 | /* |
| 2 | * |
Craig Tiller | 6169d5f | 2016-03-31 07:46:18 -0700 | [diff] [blame] | 3 | * Copyright 2015, Google Inc. |
Craig Tiller | 48cb07c | 2015-07-15 16:16:15 -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, |
| 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 | 9533d04 | 2016-03-25 17:11:06 -0700 | [diff] [blame] | 34 | #include "src/core/lib/surface/channel.h" |
Craig Tiller | 48cb07c | 2015-07-15 16:16:15 -0700 | [diff] [blame] | 35 | |
| 36 | #include <grpc/support/alloc.h> |
| 37 | #include <grpc/support/log.h> |
| 38 | |
Craig Tiller | d4c9833 | 2016-03-31 13:45:47 -0700 | [diff] [blame] | 39 | #include "src/core/ext/client_config/client_channel.h" |
Craig Tiller | 9533d04 | 2016-03-25 17:11:06 -0700 | [diff] [blame] | 40 | #include "src/core/lib/iomgr/timer.h" |
| 41 | #include "src/core/lib/surface/api_trace.h" |
| 42 | #include "src/core/lib/surface/completion_queue.h" |
Craig Tiller | 48cb07c | 2015-07-15 16:16:15 -0700 | [diff] [blame] | 43 | |
Craig Tiller | a82950e | 2015-09-22 12:33:20 -0700 | [diff] [blame] | 44 | grpc_connectivity_state grpc_channel_check_connectivity_state( |
| 45 | grpc_channel *channel, int try_to_connect) { |
Craig Tiller | 48cb07c | 2015-07-15 16:16:15 -0700 | [diff] [blame] | 46 | /* forward through to the underlying client channel */ |
Craig Tiller | a82950e | 2015-09-22 12:33:20 -0700 | [diff] [blame] | 47 | grpc_channel_element *client_channel_elem = |
| 48 | grpc_channel_stack_last_element(grpc_channel_get_channel_stack(channel)); |
Craig Tiller | f5768a6 | 2015-09-22 10:54:34 -0700 | [diff] [blame] | 49 | grpc_exec_ctx exec_ctx = GRPC_EXEC_CTX_INIT; |
Craig Tiller | 000cd8f | 2015-09-18 07:20:29 -0700 | [diff] [blame] | 50 | grpc_connectivity_state state; |
Masood Malekghassemi | 76c3d74 | 2015-08-19 18:22:53 -0700 | [diff] [blame] | 51 | GRPC_API_TRACE( |
Craig Tiller | 4de3e4f | 2015-10-05 08:55:50 -0700 | [diff] [blame] | 52 | "grpc_channel_check_connectivity_state(channel=%p, try_to_connect=%d)", 2, |
| 53 | (channel, try_to_connect)); |
David Garcia Quintas | 7b1bd2c | 2015-10-05 18:22:10 -0700 | [diff] [blame] | 54 | if (client_channel_elem->filter == &grpc_client_channel_filter) { |
| 55 | state = grpc_client_channel_check_connectivity_state( |
| 56 | &exec_ctx, client_channel_elem, try_to_connect); |
Craig Tiller | 6301038 | 2015-09-24 15:00:58 -0700 | [diff] [blame] | 57 | grpc_exec_ctx_finish(&exec_ctx); |
David Garcia Quintas | 7b1bd2c | 2015-10-05 18:22:10 -0700 | [diff] [blame] | 58 | return state; |
Craig Tiller | a82950e | 2015-09-22 12:33:20 -0700 | [diff] [blame] | 59 | } |
David Garcia Quintas | 7b1bd2c | 2015-10-05 18:22:10 -0700 | [diff] [blame] | 60 | gpr_log(GPR_ERROR, |
| 61 | "grpc_channel_check_connectivity_state called on something that is " |
| 62 | "not a (u)client channel, but '%s'", |
| 63 | client_channel_elem->filter->name); |
Craig Tiller | a82950e | 2015-09-22 12:33:20 -0700 | [diff] [blame] | 64 | grpc_exec_ctx_finish(&exec_ctx); |
David Garcia Quintas | 7b1bd2c | 2015-10-05 18:22:10 -0700 | [diff] [blame] | 65 | return GRPC_CHANNEL_FATAL_FAILURE; |
Craig Tiller | 48cb07c | 2015-07-15 16:16:15 -0700 | [diff] [blame] | 66 | } |
| 67 | |
Craig Tiller | a82950e | 2015-09-22 12:33:20 -0700 | [diff] [blame] | 68 | typedef enum { |
Craig Tiller | 48cb07c | 2015-07-15 16:16:15 -0700 | [diff] [blame] | 69 | WAITING, |
| 70 | CALLING_BACK, |
| 71 | CALLING_BACK_AND_FINISHED, |
| 72 | CALLED_BACK |
| 73 | } callback_phase; |
| 74 | |
Craig Tiller | a82950e | 2015-09-22 12:33:20 -0700 | [diff] [blame] | 75 | typedef struct { |
Craig Tiller | 48cb07c | 2015-07-15 16:16:15 -0700 | [diff] [blame] | 76 | gpr_mu mu; |
| 77 | callback_phase phase; |
Craig Tiller | 804ff71 | 2016-05-05 16:25:40 -0700 | [diff] [blame] | 78 | grpc_error *error; |
Craig Tiller | 3382511 | 2015-09-18 07:44:19 -0700 | [diff] [blame] | 79 | grpc_closure on_complete; |
David Garcia Quintas | f747bbc | 2015-10-04 23:09:47 -0700 | [diff] [blame] | 80 | grpc_timer alarm; |
Craig Tiller | 48cb07c | 2015-07-15 16:16:15 -0700 | [diff] [blame] | 81 | grpc_connectivity_state state; |
Craig Tiller | 48cb07c | 2015-07-15 16:16:15 -0700 | [diff] [blame] | 82 | grpc_completion_queue *cq; |
| 83 | grpc_cq_completion completion_storage; |
Craig Tiller | 1ada6ad | 2015-07-16 16:19:14 -0700 | [diff] [blame] | 84 | grpc_channel *channel; |
Craig Tiller | 48cb07c | 2015-07-15 16:16:15 -0700 | [diff] [blame] | 85 | void *tag; |
| 86 | } state_watcher; |
| 87 | |
Craig Tiller | a82950e | 2015-09-22 12:33:20 -0700 | [diff] [blame] | 88 | static void delete_state_watcher(grpc_exec_ctx *exec_ctx, state_watcher *w) { |
David Garcia Quintas | 7b1bd2c | 2015-10-05 18:22:10 -0700 | [diff] [blame] | 89 | grpc_channel_element *client_channel_elem = grpc_channel_stack_last_element( |
| 90 | grpc_channel_get_channel_stack(w->channel)); |
| 91 | if (client_channel_elem->filter == &grpc_client_channel_filter) { |
| 92 | GRPC_CHANNEL_INTERNAL_UNREF(exec_ctx, w->channel, |
| 93 | "watch_channel_connectivity"); |
David Garcia Quintas | 7b1bd2c | 2015-10-05 18:22:10 -0700 | [diff] [blame] | 94 | } else { |
| 95 | abort(); |
| 96 | } |
Craig Tiller | a82950e | 2015-09-22 12:33:20 -0700 | [diff] [blame] | 97 | gpr_mu_destroy(&w->mu); |
Craig Tiller | d7faf55 | 2016-05-08 09:55:28 -0700 | [diff] [blame] | 98 | GRPC_ERROR_UNREF(w->error); |
Craig Tiller | a82950e | 2015-09-22 12:33:20 -0700 | [diff] [blame] | 99 | gpr_free(w); |
Craig Tiller | 48cb07c | 2015-07-15 16:16:15 -0700 | [diff] [blame] | 100 | } |
| 101 | |
Craig Tiller | a82950e | 2015-09-22 12:33:20 -0700 | [diff] [blame] | 102 | static void finished_completion(grpc_exec_ctx *exec_ctx, void *pw, |
| 103 | grpc_cq_completion *ignored) { |
Craig Tiller | 48cb07c | 2015-07-15 16:16:15 -0700 | [diff] [blame] | 104 | int delete = 0; |
| 105 | state_watcher *w = pw; |
Craig Tiller | a82950e | 2015-09-22 12:33:20 -0700 | [diff] [blame] | 106 | gpr_mu_lock(&w->mu); |
| 107 | switch (w->phase) { |
Craig Tiller | 48cb07c | 2015-07-15 16:16:15 -0700 | [diff] [blame] | 108 | case WAITING: |
| 109 | case CALLED_BACK: |
yang-g | b063c87 | 2015-10-07 11:40:13 -0700 | [diff] [blame] | 110 | GPR_UNREACHABLE_CODE(return ); |
Craig Tiller | 48cb07c | 2015-07-15 16:16:15 -0700 | [diff] [blame] | 111 | case CALLING_BACK: |
| 112 | w->phase = CALLED_BACK; |
| 113 | break; |
| 114 | case CALLING_BACK_AND_FINISHED: |
| 115 | delete = 1; |
| 116 | break; |
Craig Tiller | a82950e | 2015-09-22 12:33:20 -0700 | [diff] [blame] | 117 | } |
| 118 | gpr_mu_unlock(&w->mu); |
Craig Tiller | 48cb07c | 2015-07-15 16:16:15 -0700 | [diff] [blame] | 119 | |
Craig Tiller | a82950e | 2015-09-22 12:33:20 -0700 | [diff] [blame] | 120 | if (delete) { |
| 121 | delete_state_watcher(exec_ctx, w); |
| 122 | } |
Craig Tiller | 48cb07c | 2015-07-15 16:16:15 -0700 | [diff] [blame] | 123 | } |
| 124 | |
Craig Tiller | a82950e | 2015-09-22 12:33:20 -0700 | [diff] [blame] | 125 | static void partly_done(grpc_exec_ctx *exec_ctx, state_watcher *w, |
Craig Tiller | 804ff71 | 2016-05-05 16:25:40 -0700 | [diff] [blame] | 126 | bool due_to_completion, grpc_error *error) { |
Craig Tiller | 48cb07c | 2015-07-15 16:16:15 -0700 | [diff] [blame] | 127 | int delete = 0; |
| 128 | |
Craig Tiller | a82950e | 2015-09-22 12:33:20 -0700 | [diff] [blame] | 129 | if (due_to_completion) { |
David Garcia Quintas | f747bbc | 2015-10-04 23:09:47 -0700 | [diff] [blame] | 130 | grpc_timer_cancel(exec_ctx, &w->alarm); |
Craig Tiller | a82950e | 2015-09-22 12:33:20 -0700 | [diff] [blame] | 131 | } |
Craig Tiller | 48cb07c | 2015-07-15 16:16:15 -0700 | [diff] [blame] | 132 | |
Craig Tiller | a82950e | 2015-09-22 12:33:20 -0700 | [diff] [blame] | 133 | gpr_mu_lock(&w->mu); |
Craig Tiller | cfc8ae1 | 2016-05-10 20:49:54 -0700 | [diff] [blame^] | 134 | const char *msg = grpc_error_string(error); |
| 135 | gpr_log(GPR_DEBUG, "partly_done: d2c=%d phs=%d err=%s", due_to_completion, w->phase, msg); |
| 136 | grpc_error_free_string(msg); |
| 137 | |
Craig Tiller | 7b43561 | 2015-11-24 08:15:05 -0800 | [diff] [blame] | 138 | if (due_to_completion) { |
Craig Tiller | f707d62 | 2016-05-06 14:26:12 -0700 | [diff] [blame] | 139 | GRPC_ERROR_UNREF(w->error); |
Craig Tiller | 804ff71 | 2016-05-05 16:25:40 -0700 | [diff] [blame] | 140 | w->error = GRPC_ERROR_NONE; |
Craig Tiller | 7b43561 | 2015-11-24 08:15:05 -0800 | [diff] [blame] | 141 | } |
Craig Tiller | a82950e | 2015-09-22 12:33:20 -0700 | [diff] [blame] | 142 | switch (w->phase) { |
Craig Tiller | 48cb07c | 2015-07-15 16:16:15 -0700 | [diff] [blame] | 143 | case WAITING: |
| 144 | w->phase = CALLING_BACK; |
Craig Tiller | f707d62 | 2016-05-06 14:26:12 -0700 | [diff] [blame] | 145 | grpc_cq_end_op(exec_ctx, w->cq, w->tag, GRPC_ERROR_REF(w->error), |
Craig Tiller | 804ff71 | 2016-05-05 16:25:40 -0700 | [diff] [blame] | 146 | finished_completion, w, &w->completion_storage); |
Craig Tiller | 48cb07c | 2015-07-15 16:16:15 -0700 | [diff] [blame] | 147 | break; |
| 148 | case CALLING_BACK: |
| 149 | w->phase = CALLING_BACK_AND_FINISHED; |
| 150 | break; |
| 151 | case CALLING_BACK_AND_FINISHED: |
yang-g | b063c87 | 2015-10-07 11:40:13 -0700 | [diff] [blame] | 152 | GPR_UNREACHABLE_CODE(return ); |
Craig Tiller | 48cb07c | 2015-07-15 16:16:15 -0700 | [diff] [blame] | 153 | case CALLED_BACK: |
| 154 | delete = 1; |
| 155 | break; |
Craig Tiller | a82950e | 2015-09-22 12:33:20 -0700 | [diff] [blame] | 156 | } |
| 157 | gpr_mu_unlock(&w->mu); |
Craig Tiller | 48cb07c | 2015-07-15 16:16:15 -0700 | [diff] [blame] | 158 | |
Craig Tiller | a82950e | 2015-09-22 12:33:20 -0700 | [diff] [blame] | 159 | if (delete) { |
| 160 | delete_state_watcher(exec_ctx, w); |
| 161 | } |
Craig Tiller | cfc8ae1 | 2016-05-10 20:49:54 -0700 | [diff] [blame^] | 162 | |
| 163 | GRPC_ERROR_UNREF(error); |
Craig Tiller | 48cb07c | 2015-07-15 16:16:15 -0700 | [diff] [blame] | 164 | } |
| 165 | |
Craig Tiller | 804ff71 | 2016-05-05 16:25:40 -0700 | [diff] [blame] | 166 | static void watch_complete(grpc_exec_ctx *exec_ctx, void *pw, |
| 167 | grpc_error *error) { |
Craig Tiller | cfc8ae1 | 2016-05-10 20:49:54 -0700 | [diff] [blame^] | 168 | partly_done(exec_ctx, pw, true, GRPC_ERROR_REF(error)); |
Craig Tiller | dfff1b8 | 2015-09-21 14:39:57 -0700 | [diff] [blame] | 169 | } |
Craig Tiller | 48cb07c | 2015-07-15 16:16:15 -0700 | [diff] [blame] | 170 | |
Craig Tiller | 804ff71 | 2016-05-05 16:25:40 -0700 | [diff] [blame] | 171 | static void timeout_complete(grpc_exec_ctx *exec_ctx, void *pw, |
| 172 | grpc_error *error) { |
Craig Tiller | cfc8ae1 | 2016-05-10 20:49:54 -0700 | [diff] [blame^] | 173 | partly_done(exec_ctx, pw, false, GRPC_ERROR_REF(error)); |
Craig Tiller | dfff1b8 | 2015-09-21 14:39:57 -0700 | [diff] [blame] | 174 | } |
Craig Tiller | 48cb07c | 2015-07-15 16:16:15 -0700 | [diff] [blame] | 175 | |
Craig Tiller | a82950e | 2015-09-22 12:33:20 -0700 | [diff] [blame] | 176 | void grpc_channel_watch_connectivity_state( |
| 177 | grpc_channel *channel, grpc_connectivity_state last_observed_state, |
| 178 | gpr_timespec deadline, grpc_completion_queue *cq, void *tag) { |
| 179 | grpc_channel_element *client_channel_elem = |
| 180 | grpc_channel_stack_last_element(grpc_channel_get_channel_stack(channel)); |
Craig Tiller | f5768a6 | 2015-09-22 10:54:34 -0700 | [diff] [blame] | 181 | grpc_exec_ctx exec_ctx = GRPC_EXEC_CTX_INIT; |
Craig Tiller | a82950e | 2015-09-22 12:33:20 -0700 | [diff] [blame] | 182 | state_watcher *w = gpr_malloc(sizeof(*w)); |
Craig Tiller | 48cb07c | 2015-07-15 16:16:15 -0700 | [diff] [blame] | 183 | |
Masood Malekghassemi | 76c3d74 | 2015-08-19 18:22:53 -0700 | [diff] [blame] | 184 | GRPC_API_TRACE( |
| 185 | "grpc_channel_watch_connectivity_state(" |
Craig Tiller | 4de3e4f | 2015-10-05 08:55:50 -0700 | [diff] [blame] | 186 | "channel=%p, last_observed_state=%d, " |
Jan Tattermusch | 8808637 | 2015-12-10 10:54:12 -0800 | [diff] [blame] | 187 | "deadline=gpr_timespec { tv_sec: %lld, tv_nsec: %d, clock_type: %d }, " |
Craig Tiller | 4de3e4f | 2015-10-05 08:55:50 -0700 | [diff] [blame] | 188 | "cq=%p, tag=%p)", |
Jan Tattermusch | 8808637 | 2015-12-10 10:54:12 -0800 | [diff] [blame] | 189 | 7, (channel, (int)last_observed_state, (long long)deadline.tv_sec, |
| 190 | (int)deadline.tv_nsec, (int)deadline.clock_type, cq, tag)); |
Masood Malekghassemi | 76c3d74 | 2015-08-19 18:22:53 -0700 | [diff] [blame] | 191 | |
Craig Tiller | 4bf2928 | 2015-12-14 11:25:48 -0800 | [diff] [blame] | 192 | grpc_cq_begin_op(cq, tag); |
Craig Tiller | 48cb07c | 2015-07-15 16:16:15 -0700 | [diff] [blame] | 193 | |
Craig Tiller | a82950e | 2015-09-22 12:33:20 -0700 | [diff] [blame] | 194 | gpr_mu_init(&w->mu); |
| 195 | grpc_closure_init(&w->on_complete, watch_complete, w); |
Craig Tiller | 48cb07c | 2015-07-15 16:16:15 -0700 | [diff] [blame] | 196 | w->phase = WAITING; |
| 197 | w->state = last_observed_state; |
Craig Tiller | cfc8ae1 | 2016-05-10 20:49:54 -0700 | [diff] [blame^] | 198 | w->error = GRPC_ERROR_CREATE("Timeout waiting for channel state"); |
Craig Tiller | 48cb07c | 2015-07-15 16:16:15 -0700 | [diff] [blame] | 199 | w->cq = cq; |
| 200 | w->tag = tag; |
Craig Tiller | 1ada6ad | 2015-07-16 16:19:14 -0700 | [diff] [blame] | 201 | w->channel = channel; |
Craig Tiller | 48cb07c | 2015-07-15 16:16:15 -0700 | [diff] [blame] | 202 | |
David Garcia Quintas | f747bbc | 2015-10-04 23:09:47 -0700 | [diff] [blame] | 203 | grpc_timer_init(&exec_ctx, &w->alarm, |
Craig Tiller | a82950e | 2015-09-22 12:33:20 -0700 | [diff] [blame] | 204 | gpr_convert_clock_type(deadline, GPR_CLOCK_MONOTONIC), |
| 205 | timeout_complete, w, gpr_now(GPR_CLOCK_MONOTONIC)); |
Craig Tiller | 48cb07c | 2015-07-15 16:16:15 -0700 | [diff] [blame] | 206 | |
David Garcia Quintas | 7b1bd2c | 2015-10-05 18:22:10 -0700 | [diff] [blame] | 207 | if (client_channel_elem->filter == &grpc_client_channel_filter) { |
| 208 | GRPC_CHANNEL_INTERNAL_REF(channel, "watch_channel_connectivity"); |
Craig Tiller | a82950e | 2015-09-22 12:33:20 -0700 | [diff] [blame] | 209 | grpc_client_channel_watch_connectivity_state(&exec_ctx, client_channel_elem, |
Craig Tiller | 7b43561 | 2015-11-24 08:15:05 -0800 | [diff] [blame] | 210 | grpc_cq_pollset(cq), &w->state, |
| 211 | &w->on_complete); |
David Garcia Quintas | fb349b9 | 2016-03-21 15:37:20 -0700 | [diff] [blame] | 212 | } else { |
| 213 | abort(); |
Craig Tiller | a82950e | 2015-09-22 12:33:20 -0700 | [diff] [blame] | 214 | } |
Craig Tiller | 000cd8f | 2015-09-18 07:20:29 -0700 | [diff] [blame] | 215 | |
Craig Tiller | a82950e | 2015-09-22 12:33:20 -0700 | [diff] [blame] | 216 | grpc_exec_ctx_finish(&exec_ctx); |
Craig Tiller | 48cb07c | 2015-07-15 16:16:15 -0700 | [diff] [blame] | 217 | } |