<p>Boris P. Korzun has uploaded this change for <strong>review</strong>.</p><p><a href="https://gerrit.asterisk.org/c/asterisk/+/18231">View Change</a></p><pre style="font-family: monospace,monospace; white-space: pre-wrap;">res_pjsip_sdp_rtp: Improve detecting of lack of RTP activity<br><br>Change RTP timer behavior for detecting RTP only after two-way<br>SDP channel establishment. Ignore detecting after receiving 183<br>with SDP or while direct media is used.<br>Make rtp_timeout and rtp_timeout_hold options consistent to rtptimeout<br>and rtpholdtimeout options in chan_sip.<br><br>ASTERISK-26689 #close<br>ASTERISK-29929 #close<br><br>Change-Id: I07326d5b9c40f25db717fd6075f6f3a8d77279eb<br>---<br>M channels/chan_pjsip.c<br>M res/res_pjsip_sdp_rtp.c<br>2 files changed, 23 insertions(+), 35 deletions(-)<br><br></pre><pre style="font-family: monospace,monospace; white-space: pre-wrap;">git pull ssh://gerrit.asterisk.org:29418/asterisk refs/changes/31/18231/1</pre><pre style="font-family: monospace,monospace; white-space: pre-wrap;"><span>diff --git a/channels/chan_pjsip.c b/channels/chan_pjsip.c</span><br><span>index e8fbb3d..61c4cfb 100644</span><br><span>--- a/channels/chan_pjsip.c</span><br><span>+++ b/channels/chan_pjsip.c</span><br><span>@@ -337,14 +337,6 @@</span><br><span> ast_sockaddr_setnull(&media->direct_media_addr);</span><br><span> changed = 1;</span><br><span> if (media->rtp) {</span><br><span style="color: hsl(0, 100%, 40%);">- /* Direct media has ended - reset time of last received RTP packet</span><br><span style="color: hsl(0, 100%, 40%);">- * to avoid premature RTP timeout. Synchronisation between the</span><br><span style="color: hsl(0, 100%, 40%);">- * modification of direct_mdedia_addr+last_rx here and reading the</span><br><span style="color: hsl(0, 100%, 40%);">- * values in res_pjsip_sdp_rtp.c:rtp_check_timeout() is provided</span><br><span style="color: hsl(0, 100%, 40%);">- * by the channel's lock (which is held while this function is</span><br><span style="color: hsl(0, 100%, 40%);">- * executed).</span><br><span style="color: hsl(0, 100%, 40%);">- */</span><br><span style="color: hsl(0, 100%, 40%);">- ast_rtp_instance_set_last_rx(media->rtp, time(NULL));</span><br><span> ast_rtp_instance_set_prop(media->rtp, AST_RTP_PROPERTY_RTCP, 1);</span><br><span> if (position != -1) {</span><br><span> ast_channel_set_fd(chan, position + AST_EXTENDED_FDS, ast_rtp_instance_fd(media->rtp, 1));</span><br><span>diff --git a/res/res_pjsip_sdp_rtp.c b/res/res_pjsip_sdp_rtp.c</span><br><span>index e1d1701..3e10cfd 100644</span><br><span>--- a/res/res_pjsip_sdp_rtp.c</span><br><span>+++ b/res/res_pjsip_sdp_rtp.c</span><br><span>@@ -106,9 +106,10 @@</span><br><span> {</span><br><span> struct ast_sip_session_media *session_media = (struct ast_sip_session_media *)data;</span><br><span> struct ast_rtp_instance *rtp = session_media->rtp;</span><br><span style="color: hsl(0, 100%, 40%);">- int elapsed;</span><br><span style="color: hsl(0, 100%, 40%);">- int timeout;</span><br><span> struct ast_channel *chan;</span><br><span style="color: hsl(120, 100%, 40%);">+ int elapsed;</span><br><span style="color: hsl(120, 100%, 40%);">+ int now;</span><br><span style="color: hsl(120, 100%, 40%);">+ int timeout;</span><br><span> </span><br><span> if (!rtp) {</span><br><span> return 0;</span><br><span>@@ -119,41 +120,37 @@</span><br><span> return 0;</span><br><span> }</span><br><span> </span><br><span style="color: hsl(0, 100%, 40%);">- /* Get channel lock to make sure that we access a consistent set of values</span><br><span style="color: hsl(0, 100%, 40%);">- * (last_rx and direct_media_addr) - the lock is held when values are modified</span><br><span style="color: hsl(0, 100%, 40%);">- * (see send_direct_media_request()/check_for_rtp_changes() in chan_pjsip.c). We</span><br><span style="color: hsl(0, 100%, 40%);">- * are trying to avoid a situation where direct_media_addr has been reset but the</span><br><span style="color: hsl(0, 100%, 40%);">- * last-rx time was not set yet.</span><br><span style="color: hsl(0, 100%, 40%);">- */</span><br><span style="color: hsl(0, 100%, 40%);">- ast_channel_lock(chan);</span><br><span style="color: hsl(0, 100%, 40%);">-</span><br><span style="color: hsl(0, 100%, 40%);">- elapsed = time(NULL) - ast_rtp_instance_get_last_rx(rtp);</span><br><span style="color: hsl(120, 100%, 40%);">+ /* Store these values locally to avoid multiple function calls */</span><br><span style="color: hsl(120, 100%, 40%);">+ now = time(NULL);</span><br><span> timeout = ast_rtp_instance_get_timeout(rtp);</span><br><span style="color: hsl(0, 100%, 40%);">- if (elapsed < timeout) {</span><br><span style="color: hsl(0, 100%, 40%);">- ast_channel_unlock(chan);</span><br><span style="color: hsl(120, 100%, 40%);">+</span><br><span style="color: hsl(120, 100%, 40%);">+ /* If the channel is not in UP state or call is redirected</span><br><span style="color: hsl(120, 100%, 40%);">+ * outside Asterisk return for later check.</span><br><span style="color: hsl(120, 100%, 40%);">+ */</span><br><span style="color: hsl(120, 100%, 40%);">+ if (ast_channel_state(chan) != AST_STATE_UP || !ast_sockaddr_isnull(&session_media->direct_media_addr)) {</span><br><span style="color: hsl(120, 100%, 40%);">+ /* Avoiding immediately disconnect after channel up or direct media has been stopped */</span><br><span style="color: hsl(120, 100%, 40%);">+ ast_rtp_instance_set_last_rx(rtp, now);</span><br><span> ast_channel_unref(chan);</span><br><span style="color: hsl(0, 100%, 40%);">- return (timeout - elapsed) * 1000;</span><br><span style="color: hsl(120, 100%, 40%);">+ /* Recheck after half timeout for avoiding possible races</span><br><span style="color: hsl(120, 100%, 40%);">+ * and faster reacting to cases while there is no an RTP at all.</span><br><span style="color: hsl(120, 100%, 40%);">+ */</span><br><span style="color: hsl(120, 100%, 40%);">+ return timeout * 500;</span><br><span> }</span><br><span> </span><br><span style="color: hsl(0, 100%, 40%);">- /* Last RTP packet was received too long ago</span><br><span style="color: hsl(0, 100%, 40%);">- * - disconnect channel unless direct media is in use.</span><br><span style="color: hsl(0, 100%, 40%);">- */</span><br><span style="color: hsl(0, 100%, 40%);">- if (!ast_sockaddr_isnull(&session_media->direct_media_addr)) {</span><br><span style="color: hsl(0, 100%, 40%);">- ast_debug_rtp(3, "(%p) RTP not disconnecting channel '%s' for lack of %s RTP activity in %d seconds "</span><br><span style="color: hsl(0, 100%, 40%);">- "since direct media is in use\n", rtp, ast_channel_name(chan),</span><br><span style="color: hsl(0, 100%, 40%);">- ast_codec_media_type2str(session_media->type), elapsed);</span><br><span style="color: hsl(0, 100%, 40%);">- ast_channel_unlock(chan);</span><br><span style="color: hsl(120, 100%, 40%);">+ elapsed = now - ast_rtp_instance_get_last_rx(rtp);</span><br><span style="color: hsl(120, 100%, 40%);">+ if (elapsed < timeout) {</span><br><span> ast_channel_unref(chan);</span><br><span style="color: hsl(0, 100%, 40%);">- return timeout * 1000; /* recheck later, direct media may have ended then */</span><br><span style="color: hsl(120, 100%, 40%);">+ return (timeout - elapsed) * 1000;</span><br><span> }</span><br><span> </span><br><span> ast_log(LOG_NOTICE, "Disconnecting channel '%s' for lack of %s RTP activity in %d seconds\n",</span><br><span> ast_channel_name(chan), ast_codec_media_type2str(session_media->type), elapsed);</span><br><span> </span><br><span style="color: hsl(120, 100%, 40%);">+ ast_channel_lock(chan);</span><br><span> ast_channel_hangupcause_set(chan, AST_CAUSE_REQUESTED_CHAN_UNAVAIL);</span><br><span style="color: hsl(0, 100%, 40%);">- ast_softhangup(chan, AST_SOFTHANGUP_DEV);</span><br><span style="color: hsl(0, 100%, 40%);">-</span><br><span> ast_channel_unlock(chan);</span><br><span style="color: hsl(120, 100%, 40%);">+</span><br><span style="color: hsl(120, 100%, 40%);">+ ast_softhangup(chan, AST_SOFTHANGUP_DEV);</span><br><span> ast_channel_unref(chan);</span><br><span> </span><br><span> return 0;</span><br><span>@@ -2234,8 +2231,7 @@</span><br><span> }</span><br><span> </span><br><span> if (ast_rtp_instance_get_timeout(session_media->rtp)) {</span><br><span style="color: hsl(0, 100%, 40%);">- session_media->timeout_sched_id = ast_sched_add_variable(sched,</span><br><span style="color: hsl(0, 100%, 40%);">- ast_rtp_instance_get_timeout(session_media->rtp) * 1000, rtp_check_timeout,</span><br><span style="color: hsl(120, 100%, 40%);">+ session_media->timeout_sched_id = ast_sched_add_variable(sched, 500, rtp_check_timeout,</span><br><span> session_media, 1);</span><br><span> }</span><br><span> </span><br><span></span><br></pre><p>To view, visit <a href="https://gerrit.asterisk.org/c/asterisk/+/18231">change 18231</a>. To unsubscribe, or for help writing mail filters, 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/c/asterisk/+/18231"/><meta itemprop="name" content="View Change"/></div></div>
<div style="display:none"> Gerrit-Project: asterisk </div>
<div style="display:none"> Gerrit-Branch: 19 </div>
<div style="display:none"> Gerrit-Change-Id: I07326d5b9c40f25db717fd6075f6f3a8d77279eb </div>
<div style="display:none"> Gerrit-Change-Number: 18231 </div>
<div style="display:none"> Gerrit-PatchSet: 1 </div>
<div style="display:none"> Gerrit-Owner: Boris P. Korzun <drtr0jan@yandex.ru> </div>
<div style="display:none"> Gerrit-MessageType: newchange </div>