<p>Richard Mudgett has uploaded this change for <strong>review</strong>.</p><p><a href="https://gerrit.asterisk.org/6206">View Change</a></p><pre style="font-family: monospace,monospace; white-space: pre-wrap;">res_pjsip_outbound_registration.c: Re-REGISTER on transport shutdown.<br><br>The fix for the issue is broken up into three parts.<br><br>This is part three which handles the client side of REGISTER requests.<br>The registered contact may no longer be valid on the server when the<br>transport used is reliable and the connection is broken.<br><br>* Re-REGISTER our contact if the reliable transport is broken after<br>registration completes. We attempt to re-REGISTER immediately to minimize<br>the time we are unreachable. Time may have already passed between the<br>connection being broken and the loss being detected.<br><br>* Reorder sip_outbound_registration_state_alloc() so the STATSD_GUAGE's<br>are still correct if an allocation failure happens.<br><br>ASTERISK-27147<br><br>Change-Id: I3668405b1ee75dfefb07c0d637826176f741ce83<br>---<br>M res/res_pjsip_outbound_registration.c<br>1 file changed, 105 insertions(+), 9 deletions(-)<br><br></pre><pre style="font-family: monospace,monospace; white-space: pre-wrap;">git pull ssh://gerrit.asterisk.org:29418/asterisk refs/changes/06/6206/1</pre><pre style="font-family: monospace,monospace; white-space: pre-wrap;">diff --git a/res/res_pjsip_outbound_registration.c b/res/res_pjsip_outbound_registration.c<br>index 7b605b9..bbb286a 100644<br>--- a/res/res_pjsip_outbound_registration.c<br>+++ b/res/res_pjsip_outbound_registration.c<br>@@ -358,6 +358,8 @@<br> unsigned int auth_attempted:1;<br> /*! \brief The name of the transport to be used for the registration */<br> char *transport_name;<br>+ /*! \brief The name of the registration sorcery object */<br>+ char *registration_name;<br> };<br> <br> /*! \brief Outbound registration state information (persists for lifetime that registration should exist) */<br>@@ -795,6 +797,82 @@<br> }<br> }<br> <br>+static int reregister_immediately_cb(void *obj)<br>+{<br>+ struct sip_outbound_registration_state *state = obj;<br>+<br>+ if (state->client_state->status != SIP_REGISTRATION_REGISTERED) {<br>+ ao2_ref(state, -1);<br>+ return 0;<br>+ }<br>+<br>+ if (DEBUG_ATLEAST(1)) {<br>+ pjsip_regc_info info;<br>+<br>+ pjsip_regc_get_info(state->client_state->client, &info);<br>+ ast_log(LOG_DEBUG,<br>+ "Outbound registration transport to server '%.*s' from client '%.*s' shutdown\n",<br>+ (int) info.server_uri.slen, info.server_uri.ptr,<br>+ (int) info.client_uri.slen, info.client_uri.ptr);<br>+ }<br>+<br>+ cancel_registration(state->client_state);<br>+<br>+ ao2_ref(state->client_state, +1);<br>+ handle_client_registration(state->client_state);<br>+<br>+ ao2_ref(state, -1);<br>+ return 0;<br>+}<br>+<br>+/*!<br>+ * \internal<br>+ * \brief The reliable transport we registered using has shutdown.<br>+ * \since 13.18.0<br>+ *<br>+ * \param obj What is needed to initiate a reregister attempt.<br>+ *<br>+ * \return Nothing<br>+ */<br>+static void registration_transport_shutdown_cb(void *obj)<br>+{<br>+ const char *registration_name = obj;<br>+ struct sip_outbound_registration_state *state;<br>+<br>+ state = get_state(registration_name);<br>+ if (!state) {<br>+ /* Registration no longer exists or shutting down. */<br>+ return;<br>+ }<br>+ if (ast_sip_push_task(state->client_state->serializer, reregister_immediately_cb, state)) {<br>+ ao2_ref(state, -1);<br>+ }<br>+}<br>+<br>+static void registration_transport_monitor_setup(pjsip_transport *transport, const char *registration_name)<br>+{<br>+ char *monitor;<br>+<br>+ if (!PJSIP_TRANSPORT_IS_RELIABLE(transport)) {<br>+ return;<br>+ }<br>+ monitor = ao2_alloc_options(strlen(registration_name) + 1, NULL,<br>+ AO2_ALLOC_OPT_LOCK_NOLOCK);<br>+ if (!monitor) {<br>+ return;<br>+ }<br>+ strcpy(monitor, registration_name);/* Safe */<br>+<br>+ /*<br>+ * We'll ignore if the transport has already been shutdown before we<br>+ * register the monitor. We might get into a message spamming infinite<br>+ * loop of registration, shutdown, reregistration...<br>+ */<br>+ ast_sip_transport_monitor_register(transport, registration_transport_shutdown_cb,<br>+ monitor);<br>+ ao2_ref(monitor, -1);<br>+}<br>+<br> /*! \brief Callback function for handling a response to a registration attempt */<br> static int handle_registration_response(void *data)<br> {<br>@@ -863,9 +941,15 @@<br> next_registration_round = 0;<br> }<br> schedule_registration(response->client_state, next_registration_round);<br>+<br>+ /* See if we should monitor for transport shutdown */<br>+ registration_transport_monitor_setup(response->rdata->tp_info.transport,<br>+ response->client_state->registration_name);<br> } else {<br> ast_debug(1, "Outbound unregistration to '%s' with client '%s' successful\n", server_uri, client_uri);<br> update_client_state_status(response->client_state, SIP_REGISTRATION_UNREGISTERED);<br>+ ast_sip_transport_monitor_unregister(response->rdata->tp_info.transport,<br>+ registration_transport_shutdown_cb);<br> }<br> } else if (response->client_state->destroy) {<br> /* We need to deal with the pending destruction instead. */<br>@@ -1008,12 +1092,13 @@<br> {<br> struct sip_outbound_registration_client_state *client_state = obj;<br> <br>- ast_free(client_state->transport_name);<br> ast_statsd_log_string("PJSIP.registrations.count", AST_STATSD_GAUGE, "-1", 1.0);<br> ast_statsd_log_string_va("PJSIP.registrations.state.%s", AST_STATSD_GAUGE, "-1", 1.0,<br> sip_outbound_registration_status_str(client_state->status));<br> <br> ast_taskprocessor_unreference(client_state->serializer);<br>+ ast_free(client_state->transport_name);<br>+ ast_free(client_state->registration_name);<br> }<br> <br> /*! \brief Allocator function for registration state */<br>@@ -1033,6 +1118,23 @@<br> return NULL;<br> }<br> <br>+ state->client_state->status = SIP_REGISTRATION_UNREGISTERED;<br>+ state->client_state->timer.user_data = state->client_state;<br>+ state->client_state->timer.cb = sip_outbound_registration_timer_cb;<br>+ state->client_state->transport_name = ast_strdup(registration->transport);<br>+ state->client_state->registration_name =<br>+ ast_strdup(ast_sorcery_object_get_id(registration));<br>+<br>+ ast_statsd_log_string("PJSIP.registrations.count", AST_STATSD_GAUGE, "+1", 1.0);<br>+ ast_statsd_log_string_va("PJSIP.registrations.state.%s", AST_STATSD_GAUGE, "+1", 1.0,<br>+ sip_outbound_registration_status_str(state->client_state->status));<br>+<br>+ if (!state->client_state->transport_name<br>+ || !state->client_state->registration_name) {<br>+ ao2_cleanup(state);<br>+ return NULL;<br>+ }<br>+<br> /* Create name with seq number appended. */<br> ast_taskprocessor_build_name(tps_name, sizeof(tps_name), "pjsip/outreg/%s",<br> ast_sorcery_object_get_id(registration));<br>@@ -1043,14 +1145,6 @@<br> ao2_cleanup(state);<br> return NULL;<br> }<br>- state->client_state->status = SIP_REGISTRATION_UNREGISTERED;<br>- state->client_state->timer.user_data = state->client_state;<br>- state->client_state->timer.cb = sip_outbound_registration_timer_cb;<br>- state->client_state->transport_name = ast_strdup(registration->transport);<br>-<br>- ast_statsd_log_string("PJSIP.registrations.count", AST_STATSD_GAUGE, "+1", 1.0);<br>- ast_statsd_log_string_va("PJSIP.registrations.state.%s", AST_STATSD_GAUGE, "+1", 1.0,<br>- sip_outbound_registration_status_str(state->client_state->status));<br> <br> state->registration = ao2_bump(registration);<br> return state;<br>@@ -2054,6 +2148,8 @@<br> <br> ao2_global_obj_release(current_states);<br> <br>+ ast_sip_transport_monitor_unregister_all(registration_transport_shutdown_cb);<br>+<br> /* Wait for registration serializers to get destroyed. */<br> ast_debug(2, "Waiting for registration transactions to complete for unload.\n");<br> remaining = ast_serializer_shutdown_group_join(shutdown_group, MAX_UNLOAD_TIMEOUT_TIME);<br></pre><p>To view, visit <a href="https://gerrit.asterisk.org/6206">change 6206</a>. To unsubscribe, visit <a href="https://gerrit.asterisk.org/settings">settings</a>.</p><div itemscope itemtype="http://schema.org/EmailMessage"><div itemscope itemprop="action" itemtype="http://schema.org/ViewAction"><link itemprop="url" href="https://gerrit.asterisk.org/6206"/><meta itemprop="name" content="View Change"/></div></div>
<div style="display:none"> Gerrit-Project: asterisk </div>
<div style="display:none"> Gerrit-Branch: master </div>
<div style="display:none"> Gerrit-MessageType: newchange </div>
<div style="display:none"> Gerrit-Change-Id: I3668405b1ee75dfefb07c0d637826176f741ce83 </div>
<div style="display:none"> Gerrit-Change-Number: 6206 </div>
<div style="display:none"> Gerrit-PatchSet: 1 </div>
<div style="display:none"> Gerrit-Owner: Richard Mudgett <rmudgett@digium.com> </div>