[Asterisk-code-review] res/res pjsip session: allow SDP answer to be regenerated (asterisk[14])
Torrey Searle
asteriskteam at digium.com
Fri Aug 25 02:53:15 CDT 2017
Torrey Searle has uploaded this change for review. ( https://gerrit.asterisk.org/6301
Change subject: res/res_pjsip_session: allow SDP answer to be regenerated
......................................................................
res/res_pjsip_session: allow SDP answer to be regenerated
If an SDP answer hasn't been sent yet, it's legal to change it.
This is required for PJSIP_DTMF_MODE to work correctly, and can
also have use in the future for updating codecs too.
ASTERISK-27209 #close
Change-Id: Idbbfb7cb3f72fbd96c94d10d93540f69bd51e7a1
---
M channels/pjsip/dialplan_functions.c
M include/asterisk/res_pjsip_session.h
M res/res_pjsip_session.c
M res/res_pjsip_session.exports.in
4 files changed, 62 insertions(+), 1 deletion(-)
git pull ssh://gerrit.asterisk.org:29418/asterisk refs/changes/01/6301/1
diff --git a/channels/pjsip/dialplan_functions.c b/channels/pjsip/dialplan_functions.c
index 13d9fab..857684a 100644
--- a/channels/pjsip/dialplan_functions.c
+++ b/channels/pjsip/dialplan_functions.c
@@ -1113,10 +1113,13 @@
struct refresh_data *data = obj;
if (data->session->inv_session->state == PJSIP_INV_STATE_CONFIRMED) {
- ast_debug(3, "Changing DTMF mode on channel %s after OFFER/ANSER completion. Sending session refresh\n", ast_channel_name(data->session->channel));
+ ast_debug(3, "Changing DTMF mode on channel %s after OFFER/ANSWER completion. Sending session refresh\n", ast_channel_name(data->session->channel));
ast_sip_session_refresh(data->session, NULL, NULL,
sip_session_response_cb, data->method, 1);
+ } else if (data->session->inv_session->state == PJSIP_INV_STATE_INCOMING) {
+ ast_debug(3, "Changing DTMF mode on channel %s during OFFER/ANSWER exchange. Updating SDP answer\n", ast_channel_name(data->session->channel));
+ ast_sip_session_regenerate_answer(data->session, NULL);
}
return 0;
diff --git a/include/asterisk/res_pjsip_session.h b/include/asterisk/res_pjsip_session.h
index ca331cb..d275c25 100644
--- a/include/asterisk/res_pjsip_session.h
+++ b/include/asterisk/res_pjsip_session.h
@@ -626,6 +626,23 @@
int generate_new_sdp);
/*!
+ * \brief Regenerate SDP Answer
+ *
+ * This method is used when an SDP offer has been received but an SDP answer
+ * has not been sent yet. It requests that a new local SDP be created and
+ * set as the SDP answer. As with any outgoing request in res_pjsip_session,
+ * this will call into registered supplements in case they wish to add anything.
+ *
+ * \param session The session on which the answer will be updated
+ * \param on_sdp_creation Callback called when SDP is created
+ * \param generate_new_sdp Boolean to indicate if a new SDP should be created
+ * \retval 0 Successfully updated the SDP answer
+ * \retval -1 Failure to updated the SDP answer
+ */
+int ast_sip_session_regenerate_answer(struct ast_sip_session *session,
+ ast_sip_session_sdp_creation_cb on_sdp_creation);
+
+/*!
* \brief Send a SIP response
*
* This will send the SIP response specified in tdata and
diff --git a/res/res_pjsip_session.c b/res/res_pjsip_session.c
index 2eb111c..42d37fe 100644
--- a/res/res_pjsip_session.c
+++ b/res/res_pjsip_session.c
@@ -968,6 +968,46 @@
return 0;
}
+int ast_sip_session_regenerate_answer(struct ast_sip_session *session,
+ ast_sip_session_sdp_creation_cb on_sdp_creation)
+{
+ pjsip_inv_session *inv_session = session->inv_session;
+ pjmedia_sdp_session *new_answer = NULL;
+ const pjmedia_sdp_session *previous_offer = NULL;
+
+ /* The SDP answer can only be regenerated if it is still pending to be sent */
+ if (!inv_session->neg || (pjmedia_sdp_neg_get_state(inv_session->neg) != PJMEDIA_SDP_NEG_STATE_REMOTE_OFFER &&
+ pjmedia_sdp_neg_get_state(inv_session->neg) != PJMEDIA_SDP_NEG_STATE_WAIT_NEGO)) {
+ ast_log(LOG_WARNING, "Requested to regenerate local SDP answer for channel '%s' but negotiation in state '%s'\n",
+ ast_channel_name(session->channel), pjmedia_sdp_neg_state_str(pjmedia_sdp_neg_get_state(inv_session->neg)));
+ return -1;
+ }
+
+ pjmedia_sdp_neg_get_neg_remote(inv_session->neg, &previous_offer);
+ if (pjmedia_sdp_neg_get_state(inv_session->neg) == PJMEDIA_SDP_NEG_STATE_WAIT_NEGO) {
+ /* Transition the SDP negotiator back to when it received the remote offer */
+ pjmedia_sdp_neg_negotiate(inv_session->pool, inv_session->neg, 0);
+ pjmedia_sdp_neg_set_remote_offer(inv_session->pool, inv_session->neg, previous_offer);
+ }
+
+ new_answer = create_local_sdp(inv_session, session, previous_offer);
+ if (!new_answer) {
+ ast_log(LOG_WARNING, "Could not create a new local SDP answer for channel '%s'\n",
+ ast_channel_name(session->channel));
+ return -1;
+ }
+
+ if (on_sdp_creation) {
+ if (on_sdp_creation(session, new_answer)) {
+ return -1;
+ }
+ }
+
+ pjsip_inv_set_sdp_answer(inv_session, new_answer);
+
+ return 0;
+}
+
void ast_sip_session_send_response(struct ast_sip_session *session, pjsip_tx_data *tdata)
{
handle_outgoing_response(session, tdata);
diff --git a/res/res_pjsip_session.exports.in b/res/res_pjsip_session.exports.in
index fdfc5fb..5bc0bf4 100644
--- a/res/res_pjsip_session.exports.in
+++ b/res/res_pjsip_session.exports.in
@@ -14,6 +14,7 @@
LINKER_SYMBOL_PREFIXast_sip_session_remove_datastore;
LINKER_SYMBOL_PREFIXast_sip_session_get_identity;
LINKER_SYMBOL_PREFIXast_sip_session_refresh;
+ LINKER_SYMBOL_PREFIXast_sip_session_regenerate_answer;
LINKER_SYMBOL_PREFIXast_sip_session_send_response;
LINKER_SYMBOL_PREFIXast_sip_session_send_request;
LINKER_SYMBOL_PREFIXast_sip_session_create_invite;
--
To view, visit https://gerrit.asterisk.org/6301
To unsubscribe, visit https://gerrit.asterisk.org/settings
Gerrit-Project: asterisk
Gerrit-Branch: 14
Gerrit-MessageType: newchange
Gerrit-Change-Id: Idbbfb7cb3f72fbd96c94d10d93540f69bd51e7a1
Gerrit-Change-Number: 6301
Gerrit-PatchSet: 1
Gerrit-Owner: Torrey Searle <tsearle at gmail.com>
-------------- next part --------------
An HTML attachment was scrubbed...
URL: <http://lists.digium.com/pipermail/asterisk-code-review/attachments/20170825/2ae84e62/attachment.html>
More information about the asterisk-code-review
mailing list