Asterisk - The Open Source Telephony Project GIT-master-7988d11
Loading...
Searching...
No Matches
Functions | Variables
res_pjsip_sdp_rtp.c File Reference

SIP SDP media stream handling. More...

#include "asterisk.h"
#include <pjsip.h>
#include <pjsip_ua.h>
#include <pjmedia.h>
#include <pjlib.h>
#include "asterisk/utils.h"
#include "asterisk/module.h"
#include "asterisk/format.h"
#include "asterisk/format_cap.h"
#include "asterisk/rtp_engine.h"
#include "asterisk/netsock2.h"
#include "asterisk/channel.h"
#include "asterisk/causes.h"
#include "asterisk/sched.h"
#include "asterisk/acl.h"
#include "asterisk/sdp_srtp.h"
#include "asterisk/dsp.h"
#include "asterisk/linkedlists.h"
#include "asterisk/stream.h"
#include "asterisk/logger_category.h"
#include "asterisk/format_cache.h"
#include "asterisk/res_pjsip.h"
#include "asterisk/res_pjsip_session.h"
#include "asterisk/res_pjsip_session_caps.h"
Include dependency graph for res_pjsip_sdp_rtp.c:

Go to the source code of this file.

Functions

static void __reg_module (void)
 
static void __unreg_module (void)
 
static int add_crypto_to_stream (struct ast_sip_session *session, struct ast_sip_session_media *session_media, pj_pool_t *pool, pjmedia_sdp_media *media)
 
static void add_extmap_to_stream (struct ast_sip_session *session, struct ast_sip_session_media *session_media, pj_pool_t *pool, pjmedia_sdp_media *media)
 
static void add_ice_to_stream (struct ast_sip_session *session, struct ast_sip_session_media *session_media, pj_pool_t *pool, pjmedia_sdp_media *media, unsigned int include_candidates)
 Function which adds ICE attributes to a media stream.
 
static void add_msid_to_stream (struct ast_sip_session *session, struct ast_sip_session_media *session_media, pj_pool_t *pool, pjmedia_sdp_media *media, struct ast_stream *stream)
 
static void add_rtcp_fb_to_stream (struct ast_sip_session *session, struct ast_sip_session_media *session_media, pj_pool_t *pool, pjmedia_sdp_media *media)
 
static void add_ssrc_to_stream (struct ast_sip_session *session, struct ast_sip_session_media *session_media, pj_pool_t *pool, pjmedia_sdp_media *media)
 Function which adds ssrc attributes to a media stream.
 
static int apply_cap_to_bundled (struct ast_sip_session_media *session_media, struct ast_sip_session_media *session_media_transport, struct ast_stream *asterisk_stream, struct ast_format_cap *joint)
 
static void apply_dtls_attrib (struct ast_sip_session_media *session_media, pjmedia_sdp_attr *attr)
 
static int apply_negotiated_sdp_stream (struct ast_sip_session *session, struct ast_sip_session_media *session_media, const struct pjmedia_sdp_session *local, const struct pjmedia_sdp_session *remote, int index, struct ast_stream *asterisk_stream)
 
struct ast_moduleAST_MODULE_SELF_SYM (void)
 
static void change_outgoing_sdp_stream_media_address (pjsip_tx_data *tdata, struct pjmedia_sdp_media *stream, struct ast_sip_transport *transport)
 Function which updates the media stream with external media address, if applicable.
 
static enum ast_sip_session_media_encryption check_endpoint_media_transport (struct ast_sip_endpoint *endpoint, const struct pjmedia_sdp_media *stream)
 Checks whether the encryption offered in SDP is compatible with the endpoint's configuration.
 
static void check_ice_support (struct ast_sip_session *session, struct ast_sip_session_media *session_media, const struct pjmedia_sdp_media *remote_stream)
 Function which checks for ice attributes in an audio stream.
 
static int create_outgoing_sdp_stream (struct ast_sip_session *session, struct ast_sip_session_media *session_media, struct pjmedia_sdp_session *sdp, const struct pjmedia_sdp_session *remote, struct ast_stream *stream)
 Function which creates an outgoing stream.
 
static int create_rtp (struct ast_sip_session *session, struct ast_sip_session_media *session_media, const pjmedia_sdp_session *sdp)
 Internal function which creates an RTP instance.
 
static void enable_rtcp (struct ast_sip_session *session, struct ast_sip_session_media *session_media, const struct pjmedia_sdp_media *remote_media)
 Enable RTCP on an RTP session.
 
static void enable_rtp_extension (struct ast_sip_session *session, struct ast_sip_session_media *session_media, enum ast_rtp_extension extension, enum ast_rtp_extension_direction direction, const pjmedia_sdp_session *sdp)
 Enable an RTP extension on an RTP session.
 
static pjmedia_sdp_attr * generate_fmtp_attr (pj_pool_t *pool, struct ast_format *format, int rtp_code)
 
static pjmedia_sdp_attr * generate_rtpmap_attr (struct ast_sip_session *session, pjmedia_sdp_media *media, pj_pool_t *pool, int rtp_code, int asterisk_format, struct ast_format *format, int code)
 
static pjmedia_sdp_attr * generate_rtpmap_attr2 (struct ast_sip_session *session, pjmedia_sdp_media *media, pj_pool_t *pool, int rtp_code, int asterisk_format, struct ast_format *format, int code, int sample_rate)
 
static void get_codecs (struct ast_sip_session *session, const struct pjmedia_sdp_media *stream, struct ast_rtp_codecs *codecs, struct ast_sip_session_media *session_media, struct ast_format_cap *astformats)
 
static enum ast_sip_session_media_encryption get_media_encryption_type (pj_str_t transport, const struct pjmedia_sdp_media *stream, unsigned int *optimistic)
 figure out media transport encryption type from the media transport string
 
static int load_module (void)
 Load the module.
 
static struct ast_framemedia_session_rtcp_read_callback (struct ast_sip_session *session, struct ast_sip_session_media *session_media)
 
static struct ast_framemedia_session_rtp_read_callback (struct ast_sip_session *session, struct ast_sip_session_media *session_media)
 
static int media_session_rtp_write_callback (struct ast_sip_session *session, struct ast_sip_session_media *session_media, struct ast_frame *frame)
 
static int media_stream_has_crypto (const struct pjmedia_sdp_media *stream)
 figure out if media stream has crypto lines for sdes
 
static int negotiate_incoming_sdp_stream (struct ast_sip_session *session, struct ast_sip_session_media *session_media, const pjmedia_sdp_session *sdp, int index, struct ast_stream *asterisk_stream)
 Function which negotiates an incoming media stream.
 
static int parse_dtls_attrib (struct ast_sip_session_media *session_media, const struct pjmedia_sdp_session *sdp, const struct pjmedia_sdp_media *stream)
 
static void process_extmap_attributes (struct ast_sip_session *session, struct ast_sip_session_media *session_media, const struct pjmedia_sdp_media *remote_stream)
 Function which processes extmap attributes in a stream.
 
static void process_ice_attributes (struct ast_sip_session *session, struct ast_sip_session_media *session_media, const struct pjmedia_sdp_session *remote, const struct pjmedia_sdp_media *remote_stream)
 Function which processes ICE attributes in an audio stream.
 
static void process_ice_auth_attrb (struct ast_sip_session *session, struct ast_sip_session_media *session_media, const struct pjmedia_sdp_session *remote, const struct pjmedia_sdp_media *remote_stream)
 
static void process_ssrc_attributes (struct ast_sip_session *session, struct ast_sip_session_media *session_media, const struct pjmedia_sdp_media *remote_stream)
 Function which processes ssrc attributes in a stream.
 
static int rtp_check_timeout (const void *data)
 Check whether RTP is being received or not.
 
static int send_keepalive (const void *data)
 
static int set_caps (struct ast_sip_session *session, struct ast_sip_session_media *session_media, struct ast_sip_session_media *session_media_transport, const struct pjmedia_sdp_media *stream, int is_offer, struct ast_stream *asterisk_stream)
 
static void set_ice_components (struct ast_sip_session *session, struct ast_sip_session_media *session_media)
 
static struct ast_format_capset_incoming_call_offer_cap (struct ast_sip_session *session, struct ast_sip_session_media *session_media, const struct pjmedia_sdp_media *stream)
 
static void set_session_media_remotely_held (struct ast_sip_session_media *session_media, const struct ast_sip_session *session, const pjmedia_sdp_media *media, const struct ast_stream *stream, const struct ast_sockaddr *addrs)
 
static int setup_dtls_srtp (struct ast_sip_session *session, struct ast_sip_session_media *session_media)
 
static int setup_media_encryption (struct ast_sip_session *session, struct ast_sip_session_media *session_media, const struct pjmedia_sdp_session *sdp, const struct pjmedia_sdp_media *stream)
 
static int setup_sdes_srtp (struct ast_sip_session_media *session_media, const struct pjmedia_sdp_media *stream)
 
static int setup_srtp (struct ast_sip_session_media *session_media)
 
static void stream_destroy (struct ast_sip_session_media *session_media)
 Function which destroys the RTP instance when session ends.
 
static void stream_stop (struct ast_sip_session_media *session_media)
 Function which stops the RTP instance.
 
static int unload_module (void)
 Unloads the sdp RTP/AVP module from Asterisk.
 
static int video_info_incoming_request (struct ast_sip_session *session, struct pjsip_rx_data *rdata)
 

Variables

static struct ast_module_info __mod_info = { .name = AST_MODULE, .flags = AST_MODFLAG_LOAD_ORDER , .description = "PJSIP SDP RTP/AVP stream handler" , .key = ASTERISK_GPL_KEY , .buildopt_sum = AST_BUILDOPT_SUM, .support_level = AST_MODULE_SUPPORT_CORE, .load = load_module, .unload = unload_module, .load_pri = AST_MODPRI_CHANNEL_DRIVER, .requires = "res_pjsip,res_pjsip_session", }
 
static struct ast_sockaddr address_rtp
 Address for RTP.
 
static const struct ast_module_infoast_module_info = &__mod_info
 
static struct ast_sip_session_sdp_handler audio_sdp_handler
 SDP handler for 'audio' media stream.
 
static struct ast_sched_contextsched
 Scheduler for RTCP purposes.
 
static const char STR_AUDIO [] = "audio"
 
static const char STR_VIDEO [] = "video"
 
static struct ast_sip_session_supplement video_info_supplement
 
static struct ast_sip_session_sdp_handler video_sdp_handler
 SDP handler for 'video' media stream.
 

Detailed Description

SIP SDP media stream handling.

Author
Joshua Colp jcolp.nosp@m.@dig.nosp@m.ium.c.nosp@m.om

Definition in file res_pjsip_sdp_rtp.c.

Function Documentation

◆ __reg_module()

static void __reg_module ( void  )
static

Definition at line 2565 of file res_pjsip_sdp_rtp.c.

◆ __unreg_module()

static void __unreg_module ( void  )
static

Definition at line 2565 of file res_pjsip_sdp_rtp.c.

◆ add_crypto_to_stream()

static int add_crypto_to_stream ( struct ast_sip_session session,
struct ast_sip_session_media session_media,
pj_pool_t *  pool,
pjmedia_sdp_media *  media 
)
static

Definition at line 1664 of file res_pjsip_sdp_rtp.c.

1667{
1668 pj_str_t stmp;
1669 pjmedia_sdp_attr *attr;
1670 enum ast_rtp_dtls_hash hash;
1671 const char *crypto_attribute;
1672 struct ast_rtp_engine_dtls *dtls;
1673 struct ast_sdp_srtp *tmp;
1674 static const pj_str_t STR_NEW = { "new", 3 };
1675 static const pj_str_t STR_EXISTING = { "existing", 8 };
1676 static const pj_str_t STR_ACTIVE = { "active", 6 };
1677 static const pj_str_t STR_PASSIVE = { "passive", 7 };
1678 static const pj_str_t STR_ACTPASS = { "actpass", 7 };
1679 static const pj_str_t STR_HOLDCONN = { "holdconn", 8 };
1680 static const pj_str_t STR_MEDSECREQ = { "requested", 9 };
1681 enum ast_rtp_dtls_setup setup;
1682
1683 switch (session_media->encryption) {
1686 break;
1688 if (!session_media->srtp) {
1689 session_media->srtp = ast_sdp_srtp_alloc();
1690 if (!session_media->srtp) {
1691 return -1;
1692 }
1693 }
1694
1695 tmp = session_media->srtp;
1696
1697 do {
1698 crypto_attribute = ast_sdp_srtp_get_attrib(tmp,
1699 0 /* DTLS running? No */,
1700 session->endpoint->media.rtp.srtp_tag_32 /* 32 byte tag length? */);
1701 if (!crypto_attribute) {
1702 /* No crypto attribute to add, bad news */
1703 return -1;
1704 }
1705
1706 attr = pjmedia_sdp_attr_create(pool, "crypto",
1707 pj_cstr(&stmp, crypto_attribute));
1708 media->attr[media->attr_count++] = attr;
1709 } while ((tmp = AST_LIST_NEXT(tmp, sdp_srtp_list)));
1710
1711 if (session->endpoint->security_negotiation == AST_SIP_SECURITY_NEG_MEDIASEC) {
1712 attr = pjmedia_sdp_attr_create(pool, "3ge2ae", &STR_MEDSECREQ);
1713 media->attr[media->attr_count++] = attr;
1714 }
1715
1716 break;
1718 if (setup_dtls_srtp(session, session_media)) {
1719 return -1;
1720 }
1721
1722 dtls = ast_rtp_instance_get_dtls(session_media->rtp);
1723 if (!dtls) {
1724 return -1;
1725 }
1726
1727 switch (dtls->get_connection(session_media->rtp)) {
1729 attr = pjmedia_sdp_attr_create(pool, "connection", &STR_NEW);
1730 media->attr[media->attr_count++] = attr;
1731 break;
1733 attr = pjmedia_sdp_attr_create(pool, "connection", &STR_EXISTING);
1734 media->attr[media->attr_count++] = attr;
1735 break;
1736 default:
1737 break;
1738 }
1739
1740 /* If this is an answer we need to use our current state, if it's an offer we need to use
1741 * the configured value.
1742 */
1743 if (session->inv_session->neg
1744 && pjmedia_sdp_neg_get_state(session->inv_session->neg) != PJMEDIA_SDP_NEG_STATE_DONE) {
1745 setup = dtls->get_setup(session_media->rtp);
1746 } else {
1747 setup = session->endpoint->media.rtp.dtls_cfg.default_setup;
1748 }
1749
1750 switch (setup) {
1752 attr = pjmedia_sdp_attr_create(pool, "setup", &STR_ACTIVE);
1753 media->attr[media->attr_count++] = attr;
1754 break;
1756 attr = pjmedia_sdp_attr_create(pool, "setup", &STR_PASSIVE);
1757 media->attr[media->attr_count++] = attr;
1758 break;
1760 attr = pjmedia_sdp_attr_create(pool, "setup", &STR_ACTPASS);
1761 media->attr[media->attr_count++] = attr;
1762 break;
1764 attr = pjmedia_sdp_attr_create(pool, "setup", &STR_HOLDCONN);
1765 break;
1766 default:
1767 break;
1768 }
1769
1770 hash = dtls->get_fingerprint_hash(session_media->rtp);
1771 crypto_attribute = dtls->get_fingerprint(session_media->rtp);
1772 if (crypto_attribute && (hash == AST_RTP_DTLS_HASH_SHA1 || hash == AST_RTP_DTLS_HASH_SHA256)) {
1773 RAII_VAR(struct ast_str *, fingerprint, ast_str_create(64), ast_free);
1774 if (!fingerprint) {
1775 return -1;
1776 }
1777
1778 if (hash == AST_RTP_DTLS_HASH_SHA1) {
1779 ast_str_set(&fingerprint, 0, "SHA-1 %s", crypto_attribute);
1780 } else {
1781 ast_str_set(&fingerprint, 0, "SHA-256 %s", crypto_attribute);
1782 }
1783
1784 attr = pjmedia_sdp_attr_create(pool, "fingerprint", pj_cstr(&stmp, ast_str_buffer(fingerprint)));
1785 media->attr[media->attr_count++] = attr;
1786 }
1787 break;
1788 }
1789
1790 return 0;
1791}
static struct ast_mansession session
#define ast_free(a)
Definition astmm.h:180
#define AST_LIST_NEXT(elm, field)
Returns the next entry in the list after the given entry.
@ AST_SIP_MEDIA_ENCRYPT_SDES
Definition res_pjsip.h:739
@ AST_SIP_MEDIA_TRANSPORT_INVALID
Definition res_pjsip.h:735
@ AST_SIP_MEDIA_ENCRYPT_NONE
Definition res_pjsip.h:737
@ AST_SIP_MEDIA_ENCRYPT_DTLS
Definition res_pjsip.h:741
@ AST_SIP_SECURITY_NEG_MEDIASEC
Definition res_pjsip.h:355
static int setup_dtls_srtp(struct ast_sip_session *session, struct ast_sip_session_media *session_media)
ast_rtp_dtls_setup
DTLS setup types.
Definition rtp_engine.h:564
@ AST_RTP_DTLS_SETUP_PASSIVE
Definition rtp_engine.h:566
@ AST_RTP_DTLS_SETUP_HOLDCONN
Definition rtp_engine.h:568
@ AST_RTP_DTLS_SETUP_ACTPASS
Definition rtp_engine.h:567
@ AST_RTP_DTLS_SETUP_ACTIVE
Definition rtp_engine.h:565
ast_rtp_dtls_hash
DTLS fingerprint hashes.
Definition rtp_engine.h:578
@ AST_RTP_DTLS_HASH_SHA1
Definition rtp_engine.h:580
@ AST_RTP_DTLS_HASH_SHA256
Definition rtp_engine.h:579
struct ast_rtp_engine_dtls * ast_rtp_instance_get_dtls(struct ast_rtp_instance *instance)
Obtain a pointer to the DTLS support present on an RTP instance.
@ AST_RTP_DTLS_CONNECTION_NEW
Definition rtp_engine.h:573
@ AST_RTP_DTLS_CONNECTION_EXISTING
Definition rtp_engine.h:574
const char * ast_sdp_srtp_get_attrib(struct ast_sdp_srtp *srtp, int dtls_enabled, int default_taglen_32)
Get the crypto attribute line for the srtp structure.
Definition sdp_srtp.c:95
struct ast_sdp_srtp * ast_sdp_srtp_alloc(void)
allocate a ast_sdp_srtp structure
Definition sdp_srtp.c:41
#define ast_str_create(init_len)
Create a malloc'ed dynamic length string.
Definition strings.h:659
int ast_str_set(struct ast_str **buf, ssize_t max_len, const char *fmt,...)
Set a dynamic string using variable arguments.
Definition strings.h:1113
char *attribute_pure ast_str_buffer(const struct ast_str *buf)
Returns the string buffer within the ast_str buf.
Definition strings.h:761
Structure that represents the optional DTLS SRTP support within an RTP engine.
Definition rtp_engine.h:621
enum ast_rtp_dtls_setup(* get_setup)(struct ast_rtp_instance *instance)
Definition rtp_engine.h:633
enum ast_rtp_dtls_hash(* get_fingerprint_hash)(struct ast_rtp_instance *instance)
Definition rtp_engine.h:639
const char *(* get_fingerprint)(struct ast_rtp_instance *instance)
Definition rtp_engine.h:641
enum ast_rtp_dtls_connection(* get_connection)(struct ast_rtp_instance *instance)
Definition rtp_engine.h:631
structure for secure RTP audio
Definition sdp_srtp.h:38
struct ast_sdp_srtp::@295 sdp_srtp_list
struct ast_sdp_srtp * srtp
Holds SRTP information.
struct ast_rtp_instance * rtp
RTP instance itself.
enum ast_sip_session_media_encryption encryption
What type of encryption is in use on this stream.
Support for dynamic strings.
Definition strings.h:623
#define RAII_VAR(vartype, varname, initval, dtor)
Declare a variable that will call a destructor function when it goes out of scope.
Definition utils.h:981

References ast_free, AST_LIST_NEXT, AST_RTP_DTLS_CONNECTION_EXISTING, AST_RTP_DTLS_CONNECTION_NEW, AST_RTP_DTLS_HASH_SHA1, AST_RTP_DTLS_HASH_SHA256, AST_RTP_DTLS_SETUP_ACTIVE, AST_RTP_DTLS_SETUP_ACTPASS, AST_RTP_DTLS_SETUP_HOLDCONN, AST_RTP_DTLS_SETUP_PASSIVE, ast_rtp_instance_get_dtls(), ast_sdp_srtp_alloc(), ast_sdp_srtp_get_attrib(), AST_SIP_MEDIA_ENCRYPT_DTLS, AST_SIP_MEDIA_ENCRYPT_NONE, AST_SIP_MEDIA_ENCRYPT_SDES, AST_SIP_MEDIA_TRANSPORT_INVALID, AST_SIP_SECURITY_NEG_MEDIASEC, ast_str_buffer(), ast_str_create, ast_str_set(), ast_sip_session_media::encryption, ast_rtp_engine_dtls::get_connection, ast_rtp_engine_dtls::get_fingerprint, ast_rtp_engine_dtls::get_fingerprint_hash, ast_rtp_engine_dtls::get_setup, RAII_VAR, ast_sip_session_media::rtp, ast_sdp_srtp::sdp_srtp_list, session, setup_dtls_srtp(), and ast_sip_session_media::srtp.

Referenced by create_outgoing_sdp_stream().

◆ add_extmap_to_stream()

static void add_extmap_to_stream ( struct ast_sip_session session,
struct ast_sip_session_media session_media,
pj_pool_t *  pool,
pjmedia_sdp_media *  media 
)
static

Definition at line 1428 of file res_pjsip_sdp_rtp.c.

1430{
1431 int idx;
1432 char extmap_value[256];
1433
1434 if (!session->endpoint->media.webrtc || session_media->type != AST_MEDIA_TYPE_VIDEO) {
1435 return;
1436 }
1437
1438 /* RTP extension local unique identifiers start at '1' */
1439 for (idx = 1; idx <= ast_rtp_instance_extmap_count(session_media->rtp); ++idx) {
1441 const char *direction_str = "";
1442 pj_str_t stmp;
1443 pjmedia_sdp_attr *attr;
1444
1445 /* If this is an unsupported RTP extension we can't place it into the SDP */
1447 continue;
1448 }
1449
1450 switch (ast_rtp_instance_extmap_get_direction(session_media->rtp, idx)) {
1452 /* Lack of a direction indicates sendrecv, so we leave it out */
1453 direction_str = "";
1454 break;
1456 direction_str = "/sendonly";
1457 break;
1459 direction_str = "/recvonly";
1460 break;
1462 /* It is impossible for a "none" direction extension to be negotiated but just in case
1463 * we treat it as inactive.
1464 */
1466 direction_str = "/inactive";
1467 break;
1468 }
1469
1470 snprintf(extmap_value, sizeof(extmap_value), "%d%s %s", idx, direction_str,
1471 ast_rtp_instance_extmap_get_uri(session_media->rtp, idx));
1472 attr = pjmedia_sdp_attr_create(pool, "extmap", pj_cstr(&stmp, extmap_value));
1473 pjmedia_sdp_attr_add(&media->attr_count, media->attr, attr);
1474 }
1475}
@ AST_MEDIA_TYPE_VIDEO
Definition codec.h:33
size_t ast_rtp_instance_extmap_count(struct ast_rtp_instance *instance)
Get the number of known unique identifiers.
Definition rtp_engine.c:950
enum ast_rtp_extension ast_rtp_instance_extmap_get_extension(struct ast_rtp_instance *instance, int id)
Retrieve the extension for an RTP extension id.
Definition rtp_engine.c:961
ast_rtp_extension
Known RTP extensions.
Definition rtp_engine.h:593
@ AST_RTP_EXTENSION_UNSUPPORTED
Definition rtp_engine.h:595
@ AST_RTP_EXTENSION_DIRECTION_SENDRECV
Definition rtp_engine.h:831
@ AST_RTP_EXTENSION_DIRECTION_NONE
Definition rtp_engine.h:829
@ AST_RTP_EXTENSION_DIRECTION_INACTIVE
Definition rtp_engine.h:837
@ AST_RTP_EXTENSION_DIRECTION_RECVONLY
Definition rtp_engine.h:835
@ AST_RTP_EXTENSION_DIRECTION_SENDONLY
Definition rtp_engine.h:833
const char * ast_rtp_instance_extmap_get_uri(struct ast_rtp_instance *instance, int id)
Retrieve the URI for an RTP extension id.
Definition rtp_engine.c:997
enum ast_rtp_extension_direction ast_rtp_instance_extmap_get_direction(struct ast_rtp_instance *instance, int id)
Retrieve the negotiated direction for an RTP extension id.
Definition rtp_engine.c:981
enum ast_media_type type
Media type of this session media.
structure to hold extensions

References AST_MEDIA_TYPE_VIDEO, AST_RTP_EXTENSION_DIRECTION_INACTIVE, AST_RTP_EXTENSION_DIRECTION_NONE, AST_RTP_EXTENSION_DIRECTION_RECVONLY, AST_RTP_EXTENSION_DIRECTION_SENDONLY, AST_RTP_EXTENSION_DIRECTION_SENDRECV, AST_RTP_EXTENSION_UNSUPPORTED, ast_rtp_instance_extmap_count(), ast_rtp_instance_extmap_get_direction(), ast_rtp_instance_extmap_get_extension(), ast_rtp_instance_extmap_get_uri(), ast_sip_session_media::rtp, session, and ast_sip_session_media::type.

Referenced by create_outgoing_sdp_stream().

◆ add_ice_to_stream()

static void add_ice_to_stream ( struct ast_sip_session session,
struct ast_sip_session_media session_media,
pj_pool_t *  pool,
pjmedia_sdp_media *  media,
unsigned int  include_candidates 
)
static

Function which adds ICE attributes to a media stream.

Definition at line 749 of file res_pjsip_sdp_rtp.c.

751{
752 struct ast_rtp_engine_ice *ice;
753 struct ao2_container *candidates;
754 const char *username, *password;
755 pj_str_t stmp;
756 pjmedia_sdp_attr *attr;
757 struct ao2_iterator it_candidates;
758 struct ast_rtp_engine_ice_candidate *candidate;
759
760 if (!session->endpoint->media.rtp.ice_support || !(ice = ast_rtp_instance_get_ice(session_media->rtp))) {
761 return;
762 }
763
764 if (!session_media->remote_ice) {
765 ice->stop(session_media->rtp);
766 return;
767 }
768
769 if ((username = ice->get_ufrag(session_media->rtp))) {
770 attr = pjmedia_sdp_attr_create(pool, "ice-ufrag", pj_cstr(&stmp, username));
771 media->attr[media->attr_count++] = attr;
772 }
773
774 if ((password = ice->get_password(session_media->rtp))) {
775 attr = pjmedia_sdp_attr_create(pool, "ice-pwd", pj_cstr(&stmp, password));
776 media->attr[media->attr_count++] = attr;
777 }
778
779 if (!include_candidates) {
780 return;
781 }
782
783 candidates = ice->get_local_candidates(session_media->rtp);
784 if (!candidates) {
785 return;
786 }
787
788 it_candidates = ao2_iterator_init(candidates, 0);
789 for (; (candidate = ao2_iterator_next(&it_candidates)); ao2_ref(candidate, -1)) {
790 struct ast_str *attr_candidate = ast_str_create(128);
791
792 ast_str_set(&attr_candidate, -1, "%s %u %s %d %s ", candidate->foundation, candidate->id, candidate->transport,
793 candidate->priority, ast_sockaddr_stringify_addr_remote(&candidate->address));
794 ast_str_append(&attr_candidate, -1, "%s typ ", ast_sockaddr_stringify_port(&candidate->address));
795
796 switch (candidate->type) {
798 ast_str_append(&attr_candidate, -1, "host");
799 break;
801 ast_str_append(&attr_candidate, -1, "srflx");
802 break;
804 ast_str_append(&attr_candidate, -1, "relay");
805 break;
806 }
807
808 if (!ast_sockaddr_isnull(&candidate->relay_address)) {
809 ast_str_append(&attr_candidate, -1, " raddr %s rport", ast_sockaddr_stringify_addr_remote(&candidate->relay_address));
810 ast_str_append(&attr_candidate, -1, " %s", ast_sockaddr_stringify_port(&candidate->relay_address));
811 }
812
813 attr = pjmedia_sdp_attr_create(pool, "candidate", pj_cstr(&stmp, ast_str_buffer(attr_candidate)));
814 media->attr[media->attr_count++] = attr;
815
816 ast_free(attr_candidate);
817 }
818
819 ao2_iterator_destroy(&it_candidates);
820 ao2_ref(candidates, -1);
821}
#define ao2_iterator_next(iter)
Definition astobj2.h:1911
struct ao2_iterator ao2_iterator_init(struct ao2_container *c, int flags) attribute_warn_unused_result
Create an iterator for a container.
#define ao2_ref(o, delta)
Reference/unreference an object and return the old refcount.
Definition astobj2.h:459
void ao2_iterator_destroy(struct ao2_iterator *iter)
Destroy a container iterator.
static int ast_sockaddr_isnull(const struct ast_sockaddr *addr)
Checks if the ast_sockaddr is null. "null" in this sense essentially means uninitialized,...
Definition netsock2.h:127
static char * ast_sockaddr_stringify_port(const struct ast_sockaddr *addr)
Wrapper around ast_sockaddr_stringify_fmt() to return a port only.
Definition netsock2.h:358
static char * ast_sockaddr_stringify_addr_remote(const struct ast_sockaddr *addr)
Wrapper around ast_sockaddr_stringify_fmt() to return an address only.
Definition netsock2.h:313
@ AST_RTP_ICE_CANDIDATE_TYPE_RELAYED
Definition rtp_engine.h:509
@ AST_RTP_ICE_CANDIDATE_TYPE_SRFLX
Definition rtp_engine.h:508
@ AST_RTP_ICE_CANDIDATE_TYPE_HOST
Definition rtp_engine.h:507
struct ast_rtp_engine_ice * ast_rtp_instance_get_ice(struct ast_rtp_instance *instance)
Obtain a pointer to the ICE support present on an RTP instance.
int ast_str_append(struct ast_str **buf, ssize_t max_len, const char *fmt,...)
Append to a thread local dynamic string.
Definition strings.h:1139
Generic container type.
When we need to walk through a container, we use an ao2_iterator to keep track of the current positio...
Definition astobj2.h:1821
Structure for an ICE candidate.
Definition rtp_engine.h:525
struct ast_sockaddr address
Definition rtp_engine.h:530
enum ast_rtp_ice_component_type id
Definition rtp_engine.h:527
struct ast_sockaddr relay_address
Definition rtp_engine.h:531
enum ast_rtp_ice_candidate_type type
Definition rtp_engine.h:532
Structure that represents the optional ICE support within an RTP engine.
Definition rtp_engine.h:536
void(* stop)(struct ast_rtp_instance *instance)
Definition rtp_engine.h:544
const char *(* get_ufrag)(struct ast_rtp_instance *instance)
Definition rtp_engine.h:546
const char *(* get_password)(struct ast_rtp_instance *instance)
Definition rtp_engine.h:548
struct ao2_container *(* get_local_candidates)(struct ast_rtp_instance *instance)
Definition rtp_engine.h:550
unsigned int remote_ice
Does remote support ice.

References ast_rtp_engine_ice_candidate::address, ao2_iterator_destroy(), ao2_iterator_init(), ao2_iterator_next, ao2_ref, ast_free, AST_RTP_ICE_CANDIDATE_TYPE_HOST, AST_RTP_ICE_CANDIDATE_TYPE_RELAYED, AST_RTP_ICE_CANDIDATE_TYPE_SRFLX, ast_rtp_instance_get_ice(), ast_sockaddr_isnull(), ast_sockaddr_stringify_addr_remote(), ast_sockaddr_stringify_port(), ast_str_append(), ast_str_buffer(), ast_str_create, ast_str_set(), ast_rtp_engine_ice_candidate::foundation, ast_rtp_engine_ice::get_local_candidates, ast_rtp_engine_ice::get_password, ast_rtp_engine_ice::get_ufrag, ast_rtp_engine_ice_candidate::id, ast_rtp_engine_ice_candidate::priority, ast_rtp_engine_ice_candidate::relay_address, ast_sip_session_media::remote_ice, ast_sip_session_media::rtp, session, ast_rtp_engine_ice::stop, ast_rtp_engine_ice_candidate::transport, and ast_rtp_engine_ice_candidate::type.

Referenced by create_outgoing_sdp_stream().

◆ add_msid_to_stream()

static void add_msid_to_stream ( struct ast_sip_session session,
struct ast_sip_session_media session_media,
pj_pool_t *  pool,
pjmedia_sdp_media *  media,
struct ast_stream stream 
)
static

Definition at line 1346 of file res_pjsip_sdp_rtp.c.

1349{
1350 pj_str_t stmp;
1351 pjmedia_sdp_attr *attr;
1352 char msid[(AST_UUID_STR_LEN * 2) + 2];
1353 const char *stream_label = ast_stream_get_metadata(stream, "SDP:LABEL");
1354
1355 if (!session->endpoint->media.webrtc) {
1356 return;
1357 }
1358
1359 if (ast_strlen_zero(session_media->mslabel)) {
1360 /* If this stream is grouped with another then use its media stream label if possible */
1361 if (ast_stream_get_group(stream) != -1) {
1362 struct ast_sip_session_media *group_session_media = AST_VECTOR_GET(&session->pending_media_state->sessions, ast_stream_get_group(stream));
1363
1364 ast_copy_string(session_media->mslabel, group_session_media->mslabel, sizeof(session_media->mslabel));
1365 }
1366
1367 if (ast_strlen_zero(session_media->mslabel)) {
1368 ast_uuid_generate_str(session_media->mslabel, sizeof(session_media->mslabel));
1369 }
1370 }
1371
1372 if (ast_strlen_zero(session_media->label)) {
1373 ast_uuid_generate_str(session_media->label, sizeof(session_media->label));
1374 /* add for stream identification to replace stream_name */
1375 ast_stream_set_metadata(stream, "MSID:LABEL", session_media->label);
1376 }
1377
1378 snprintf(msid, sizeof(msid), "%s %s", session_media->mslabel, session_media->label);
1379 ast_debug(3, "Stream msid: %p %s %s\n", stream,
1381 attr = pjmedia_sdp_attr_create(pool, "msid", pj_cstr(&stmp, msid));
1382 pjmedia_sdp_attr_add(&media->attr_count, media->attr, attr);
1383
1384 /* 'label' must come after 'msid' */
1385 if (!ast_strlen_zero(stream_label)) {
1386 ast_debug(3, "Stream Label: %p %s %s\n", stream,
1387 ast_codec_media_type2str(ast_stream_get_type(stream)), stream_label);
1388 attr = pjmedia_sdp_attr_create(pool, "label", pj_cstr(&stmp, stream_label));
1389 pjmedia_sdp_attr_add(&media->attr_count, media->attr, attr);
1390 }
1391}
const char * ast_codec_media_type2str(enum ast_media_type type)
Conversion function to take a media type and turn it into a string.
Definition codec.c:348
#define ast_debug(level,...)
Log a DEBUG message.
const char * ast_stream_get_metadata(const struct ast_stream *stream, const char *m_key)
Get a stream metadata value.
Definition stream.c:423
int ast_stream_set_metadata(struct ast_stream *stream, const char *m_key, const char *value)
Set a stream metadata value.
Definition stream.c:460
int ast_stream_get_group(const struct ast_stream *stream)
Get the stream group that a stream is part of.
Definition stream.c:1080
enum ast_media_type ast_stream_get_type(const struct ast_stream *stream)
Get the media type of a stream.
Definition stream.c:316
static force_inline int attribute_pure ast_strlen_zero(const char *s)
Definition strings.h:65
void ast_copy_string(char *dst, const char *src, size_t size)
Size-limited null-terminating string copy.
Definition strings.h:425
A structure containing SIP session media information.
char label[AST_UUID_STR_LEN]
Track label.
char mslabel[AST_UUID_STR_LEN]
Media stream label.
#define AST_UUID_STR_LEN
Definition uuid.h:27
char * ast_uuid_generate_str(char *buf, size_t size)
Generate a UUID string.
Definition uuid.c:141
#define AST_VECTOR_GET(vec, idx)
Get an element from a vector.
Definition vector.h:691

References ast_codec_media_type2str(), ast_copy_string(), ast_debug, ast_stream_get_group(), ast_stream_get_metadata(), ast_stream_get_type(), ast_stream_set_metadata(), ast_strlen_zero(), ast_uuid_generate_str(), AST_UUID_STR_LEN, AST_VECTOR_GET, ast_sip_session_media::label, ast_sip_session_media::mslabel, and session.

Referenced by create_outgoing_sdp_stream().

◆ add_rtcp_fb_to_stream()

static void add_rtcp_fb_to_stream ( struct ast_sip_session session,
struct ast_sip_session_media session_media,
pj_pool_t *  pool,
pjmedia_sdp_media *  media 
)
static

Definition at line 1393 of file res_pjsip_sdp_rtp.c.

1395{
1396 pj_str_t stmp;
1397 pjmedia_sdp_attr *attr;
1398
1399 if (!session->endpoint->media.webrtc) {
1400 return;
1401 }
1402
1403 /* transport-cc is supposed to be for the entire transport, and any media sources so
1404 * while the header does not appear in audio streams and isn't negotiated there, we still
1405 * place this attribute in as Chrome does.
1406 */
1407 attr = pjmedia_sdp_attr_create(pool, "rtcp-fb", pj_cstr(&stmp, "* transport-cc"));
1408 pjmedia_sdp_attr_add(&media->attr_count, media->attr, attr);
1409
1410 if (session_media->type != AST_MEDIA_TYPE_VIDEO) {
1411 return;
1412 }
1413
1414 /*
1415 * For now just automatically add it the stream even though it hasn't
1416 * necessarily been negotiated.
1417 */
1418 attr = pjmedia_sdp_attr_create(pool, "rtcp-fb", pj_cstr(&stmp, "* ccm fir"));
1419 pjmedia_sdp_attr_add(&media->attr_count, media->attr, attr);
1420
1421 attr = pjmedia_sdp_attr_create(pool, "rtcp-fb", pj_cstr(&stmp, "* goog-remb"));
1422 pjmedia_sdp_attr_add(&media->attr_count, media->attr, attr);
1423
1424 attr = pjmedia_sdp_attr_create(pool, "rtcp-fb", pj_cstr(&stmp, "* nack"));
1425 pjmedia_sdp_attr_add(&media->attr_count, media->attr, attr);
1426}

References AST_MEDIA_TYPE_VIDEO, session, and ast_sip_session_media::type.

Referenced by create_outgoing_sdp_stream().

◆ add_ssrc_to_stream()

static void add_ssrc_to_stream ( struct ast_sip_session session,
struct ast_sip_session_media session_media,
pj_pool_t *  pool,
pjmedia_sdp_media *  media 
)
static

Function which adds ssrc attributes to a media stream.

Definition at line 1276 of file res_pjsip_sdp_rtp.c.

1277{
1278 pj_str_t stmp;
1279 pjmedia_sdp_attr *attr;
1280 char tmp[128];
1281
1282 if (!session->endpoint->media.bundle || session_media->bundle_group == -1) {
1283 return;
1284 }
1285
1286 snprintf(tmp, sizeof(tmp), "%u cname:%s", ast_rtp_instance_get_ssrc(session_media->rtp), ast_rtp_instance_get_cname(session_media->rtp));
1287 attr = pjmedia_sdp_attr_create(pool, "ssrc", pj_cstr(&stmp, tmp));
1288 media->attr[media->attr_count++] = attr;
1289}
const char * ast_rtp_instance_get_cname(struct ast_rtp_instance *rtp)
Retrieve the CNAME used in RTCP SDES items.
unsigned int ast_rtp_instance_get_ssrc(struct ast_rtp_instance *rtp)
Retrieve the local SSRC value that we will be using.
int bundle_group
The bundle group the stream belongs to.

References ast_rtp_instance_get_cname(), ast_rtp_instance_get_ssrc(), session, and ast_rtp_engine_ice::start.

Referenced by create_outgoing_sdp_stream().

◆ apply_cap_to_bundled()

static int apply_cap_to_bundled ( struct ast_sip_session_media session_media,
struct ast_sip_session_media session_media_transport,
struct ast_stream asterisk_stream,
struct ast_format_cap joint 
)
static

Definition at line 444 of file res_pjsip_sdp_rtp.c.

447{
448 if (!joint) {
449 return -1;
450 }
451
452 ast_stream_set_formats(asterisk_stream, joint);
453
454 /* If this is a bundled stream then apply the payloads to RTP instance acting as transport to prevent conflicts */
455 if (session_media_transport != session_media && session_media->bundled) {
456 int index;
457
458 for (index = 0; index < ast_format_cap_count(joint); ++index) {
459 struct ast_format *format = ast_format_cap_get_format(joint, index);
460 int rtp_code;
461
462 /* Ensure this payload is in the bundle group transport codecs, this purposely doesn't check the return value for
463 * things as the format is guaranteed to have a payload already.
464 */
465 rtp_code = ast_rtp_codecs_payload_code(ast_rtp_instance_get_codecs(session_media->rtp), 1, format, 0);
466 ast_rtp_codecs_payload_set_rx(ast_rtp_instance_get_codecs(session_media_transport->rtp), rtp_code, format);
467
468 ao2_ref(format, -1);
469 }
470 }
471
472 return 0;
473}
struct ast_format * ast_format_cap_get_format(const struct ast_format_cap *cap, int position)
Get the format at a specific index.
Definition format_cap.c:400
size_t ast_format_cap_count(const struct ast_format_cap *cap)
Get the number of formats present within the capabilities structure.
Definition format_cap.c:395
int ast_rtp_codecs_payload_set_rx(struct ast_rtp_codecs *codecs, int code, struct ast_format *format)
Set a payload code for use with a specific Asterisk format.
struct ast_rtp_codecs * ast_rtp_instance_get_codecs(struct ast_rtp_instance *instance)
Get the codecs structure of an RTP instance.
Definition rtp_engine.c:778
int ast_rtp_codecs_payload_code(struct ast_rtp_codecs *codecs, int asterisk_format, struct ast_format *format, int code)
Retrieve a rx mapped payload type based on whether it is an Asterisk format and the code.
void ast_stream_set_formats(struct ast_stream *stream, struct ast_format_cap *caps)
Set the current negotiated formats of a stream.
Definition stream.c:365
Definition of a media format.
Definition format.c:43
unsigned int bundled
Whether this stream is currently bundled or not.

References ao2_ref, ast_format_cap_count(), ast_format_cap_get_format(), ast_rtp_codecs_payload_code(), ast_rtp_codecs_payload_set_rx(), ast_rtp_instance_get_codecs(), ast_stream_set_formats(), ast_sip_session_media::bundled, and ast_sip_session_media::rtp.

Referenced by negotiate_incoming_sdp_stream(), and set_caps().

◆ apply_dtls_attrib()

static void apply_dtls_attrib ( struct ast_sip_session_media session_media,
pjmedia_sdp_attr *  attr 
)
static

Definition at line 1124 of file res_pjsip_sdp_rtp.c.

1126{
1127 struct ast_rtp_engine_dtls *dtls = ast_rtp_instance_get_dtls(session_media->rtp);
1128 pj_str_t *value;
1129
1130 if (!attr->value.ptr || !dtls) {
1131 return;
1132 }
1133
1134 value = pj_strtrim(&attr->value);
1135
1136 if (!pj_strcmp2(&attr->name, "setup")) {
1137 if (!pj_stricmp2(value, "active")) {
1138 dtls->set_setup(session_media->rtp, AST_RTP_DTLS_SETUP_ACTIVE);
1139 } else if (!pj_stricmp2(value, "passive")) {
1140 dtls->set_setup(session_media->rtp, AST_RTP_DTLS_SETUP_PASSIVE);
1141 } else if (!pj_stricmp2(value, "actpass")) {
1142 dtls->set_setup(session_media->rtp, AST_RTP_DTLS_SETUP_ACTPASS);
1143 } else if (!pj_stricmp2(value, "holdconn")) {
1144 dtls->set_setup(session_media->rtp, AST_RTP_DTLS_SETUP_HOLDCONN);
1145 } else {
1146 ast_log(LOG_WARNING, "Unsupported setup attribute value '%*s'\n", (int)value->slen, value->ptr);
1147 }
1148 } else if (!pj_strcmp2(&attr->name, "connection")) {
1149 if (!pj_stricmp2(value, "new")) {
1150 dtls->reset(session_media->rtp);
1151 } else if (!pj_stricmp2(value, "existing")) {
1152 /* Do nothing */
1153 } else {
1154 ast_log(LOG_WARNING, "Unsupported connection attribute value '%*s'\n", (int)value->slen, value->ptr);
1155 }
1156 } else if (!pj_strcmp2(&attr->name, "fingerprint")) {
1157 char hash_value[256], hash[32];
1158 char fingerprint_text[value->slen + 1];
1159 ast_copy_pj_str(fingerprint_text, value, sizeof(fingerprint_text));
1160 if (sscanf(fingerprint_text, "%31s %255s", hash, hash_value) == 2) {
1161 if (!strcasecmp(hash, "sha-1")) {
1162 dtls->set_fingerprint(session_media->rtp, AST_RTP_DTLS_HASH_SHA1, hash_value);
1163 } else if (!strcasecmp(hash, "sha-256")) {
1164 dtls->set_fingerprint(session_media->rtp, AST_RTP_DTLS_HASH_SHA256, hash_value);
1165 } else {
1166 ast_log(LOG_WARNING, "Unsupported fingerprint hash type '%s'\n",
1167 hash);
1168 }
1169 }
1170 }
1171}
#define ast_log
Definition astobj2.c:42
#define LOG_WARNING
void ast_copy_pj_str(char *dest, const pj_str_t *src, size_t size)
Copy a pj_str_t into a standard character buffer.
Definition res_pjsip.c:2172
void(* reset)(struct ast_rtp_instance *instance)
Definition rtp_engine.h:629
void(* set_fingerprint)(struct ast_rtp_instance *instance, enum ast_rtp_dtls_hash hash, const char *fingerprint)
Definition rtp_engine.h:637
void(* set_setup)(struct ast_rtp_instance *instance, enum ast_rtp_dtls_setup setup)
Definition rtp_engine.h:635
int value
Definition syslog.c:37

References ast_copy_pj_str(), ast_log, AST_RTP_DTLS_HASH_SHA1, AST_RTP_DTLS_HASH_SHA256, AST_RTP_DTLS_SETUP_ACTIVE, AST_RTP_DTLS_SETUP_ACTPASS, AST_RTP_DTLS_SETUP_HOLDCONN, AST_RTP_DTLS_SETUP_PASSIVE, ast_rtp_instance_get_dtls(), LOG_WARNING, ast_rtp_engine_dtls::reset, ast_sip_session_media::rtp, ast_rtp_engine_dtls::set_fingerprint, ast_rtp_engine_dtls::set_setup, and value.

Referenced by parse_dtls_attrib().

◆ apply_negotiated_sdp_stream()

static int apply_negotiated_sdp_stream ( struct ast_sip_session session,
struct ast_sip_session_media session_media,
const struct pjmedia_sdp_session *  local,
const struct pjmedia_sdp_session *  remote,
int  index,
struct ast_stream asterisk_stream 
)
static

Definition at line 2232 of file res_pjsip_sdp_rtp.c.

2235{
2236 RAII_VAR(struct ast_sockaddr *, addrs, NULL, ast_free);
2237 struct pjmedia_sdp_media *remote_stream = remote->media[index];
2238 enum ast_media_type media_type = session_media->type;
2239 char host[NI_MAXHOST];
2240 int res;
2241 int rtp_timeout;
2242 struct ast_sip_session_media *session_media_transport;
2243 SCOPE_ENTER(1, "%s Stream: %s\n", ast_sip_session_get_name(session),
2244 ast_str_tmp(128, ast_stream_to_str(asterisk_stream, &STR_TMP)));
2245
2246 if (!session->channel) {
2247 SCOPE_EXIT_RTN_VALUE(1, "No channel\n");
2248 }
2249
2250 /* Ensure incoming transport is compatible with the endpoint's configuration */
2251 if (!session->endpoint->media.rtp.use_received_transport &&
2253 SCOPE_EXIT_RTN_VALUE(-1, "Incompatible transport\n");
2254 }
2255
2256 /* Create an RTP instance if need be */
2257 if (!session_media->rtp && create_rtp(session, session_media, local)) {
2258 SCOPE_EXIT_RTN_VALUE(-1, "Couldn't create rtp\n");
2259 }
2260
2261 process_ssrc_attributes(session, session_media, remote_stream);
2262 process_extmap_attributes(session, session_media, remote_stream);
2263
2264 session_media_transport = ast_sip_session_media_get_transport(session, session_media);
2265
2266 if (session_media_transport == session_media || !session_media->bundled) {
2267 session_media->remote_rtcp_mux = (pjmedia_sdp_media_find_attr2(remote_stream, "rtcp-mux", NULL) != NULL);
2268 set_ice_components(session, session_media);
2269
2270 enable_rtcp(session, session_media, remote_stream);
2271
2272 res = setup_media_encryption(session, session_media, remote, remote_stream);
2273 if (!session->endpoint->media.rtp.encryption_optimistic && res) {
2274 /* If optimistic encryption is disabled and crypto should have been enabled but was not
2275 * this session must fail.
2276 */
2277 SCOPE_EXIT_RTN_VALUE(-1, "Incompatible crypto\n");
2278 }
2279
2280 if (!remote_stream->conn && !remote->conn) {
2281 SCOPE_EXIT_RTN_VALUE(1, "No connection info\n");
2282 }
2283
2284 ast_copy_pj_str(host, remote_stream->conn ? &remote_stream->conn->addr : &remote->conn->addr, sizeof(host));
2285
2286 /* Ensure that the address provided is valid */
2287 if (ast_sockaddr_resolve(&addrs, host, PARSE_PORT_FORBID, AST_AF_UNSPEC) <= 0) {
2288 /* The provided host was actually invalid so we error out this negotiation */
2289 SCOPE_EXIT_RTN_VALUE(-1, "Host invalid\n");
2290 }
2291
2292 /* Apply connection information to the RTP instance */
2293 ast_sockaddr_set_port(addrs, remote_stream->desc.port);
2294 ast_rtp_instance_set_remote_address(session_media->rtp, addrs);
2295
2299 if (!session->endpoint->media.rtcp_mux || !session_media->remote_rtcp_mux) {
2302 }
2303
2304 /* If ICE support is enabled find all the needed attributes */
2305 process_ice_attributes(session, session_media, remote, remote_stream);
2306 } else {
2307 /* This is bundled with another session, so mark it as such */
2308 ast_rtp_instance_bundle(session_media->rtp, session_media_transport->rtp);
2310 enable_rtcp(session, session_media, remote_stream);
2311 }
2312
2313 if (set_caps(session, session_media, session_media_transport, remote_stream, 0, asterisk_stream)) {
2314 SCOPE_EXIT_RTN_VALUE(-1, "set_caps failed\n");
2315 }
2316
2317 /* Set the channel uniqueid on the RTP instance now that it is becoming active */
2318 ast_channel_lock(session->channel);
2320 ast_channel_unlock(session->channel);
2321
2322 /* Ensure the RTP instance is active */
2323 ast_rtp_instance_set_stream_num(session_media->rtp, ast_stream_get_position(asterisk_stream));
2324 ast_rtp_instance_activate(session_media->rtp);
2325
2326 /* audio stream handles music on hold */
2327 if (media_type != AST_MEDIA_TYPE_AUDIO && media_type != AST_MEDIA_TYPE_VIDEO) {
2328 if ((pjmedia_sdp_neg_was_answer_remote(session->inv_session->neg) == PJ_FALSE)
2329 && (session->inv_session->state == PJSIP_INV_STATE_CONFIRMED)) {
2331 }
2332 SCOPE_EXIT_RTN_VALUE(1, "moh\n");
2333 }
2334
2335 set_session_media_remotely_held(session_media, session, remote_stream, asterisk_stream, addrs);
2336
2337 if (session_media->remotely_held_changed) {
2338 if (session_media->remotely_held) {
2339 /* The remote side has put us on hold */
2340 if (!session->endpoint->suppress_moh_on_sendonly) {
2341 ast_queue_hold(session->channel, session->endpoint->mohsuggest);
2342 ast_rtp_instance_stop(session_media->rtp);
2344 }
2345 session_media->remotely_held_changed = 0;
2346 } else {
2347 /* The remote side has taken us off hold */
2348 if (!session->endpoint->suppress_moh_on_sendonly) {
2349 ast_queue_unhold(session->channel);
2351 }
2352 session_media->remotely_held_changed = 0;
2353 }
2354 } else if ((pjmedia_sdp_neg_was_answer_remote(session->inv_session->neg) == PJ_FALSE)
2355 && (session->inv_session->state == PJSIP_INV_STATE_CONFIRMED)) {
2357 }
2358
2359 /* This purposely resets the encryption to the configured in case it gets added later */
2360 session_media->encryption = session->endpoint->media.rtp.encryption;
2361
2362 if (session->endpoint->media.rtp.keepalive > 0 &&
2363 (session_media->type == AST_MEDIA_TYPE_AUDIO ||
2364 session_media->type == AST_MEDIA_TYPE_VIDEO)) {
2365 ast_rtp_instance_set_keepalive(session_media->rtp, session->endpoint->media.rtp.keepalive);
2366 /* Schedule the initial keepalive early in case this is being used to punch holes through
2367 * a NAT. This way there won't be an awkward delay before media starts flowing in some
2368 * scenarios.
2369 */
2370 AST_SCHED_DEL(sched, session_media->keepalive_sched_id);
2372 session_media, 1);
2373 }
2374
2375 /* As the channel lock is not held during this process the scheduled item won't block if
2376 * it is hanging up the channel at the same point we are applying this negotiated SDP.
2377 */
2378 AST_SCHED_DEL(sched, session_media->timeout_sched_id);
2379
2380 /* Due to the fact that we only ever have one scheduled timeout item for when we are both
2381 * off hold and on hold we don't need to store the two timeouts differently on the RTP
2382 * instance itself.
2383 */
2384 ast_rtp_instance_set_timeout(session_media->rtp, 0);
2385 if (session->endpoint->media.rtp.timeout && !session_media->remotely_held && !session_media->locally_held) {
2386 ast_rtp_instance_set_timeout(session_media->rtp, session->endpoint->media.rtp.timeout);
2387 } else if (session->endpoint->media.rtp.timeout_hold && (session_media->remotely_held || session_media->locally_held)) {
2388 ast_rtp_instance_set_timeout(session_media->rtp, session->endpoint->media.rtp.timeout_hold);
2389 }
2390
2391 rtp_timeout = ast_rtp_instance_get_timeout(session_media->rtp);
2392
2393 if (rtp_timeout) {
2394 session_media->timeout_sched_id = ast_sched_add_variable(sched, rtp_timeout*1000, rtp_check_timeout,
2395 session_media, 1);
2396 }
2397
2398 SCOPE_EXIT_RTN_VALUE(1, "Handled\n");
2399}
static PGconn * conn
Definition cdr_pgsql.c:87
#define ast_channel_lock(chan)
Definition channel.h:2989
int ast_queue_control(struct ast_channel *chan, enum ast_control_frame_type control)
Queue a control frame without payload.
Definition channel.c:1289
int ast_queue_frame(struct ast_channel *chan, struct ast_frame *f)
Queue one or more frames to a channel's frame queue.
Definition channel.c:1171
const char * ast_channel_uniqueid(const struct ast_channel *chan)
int ast_queue_unhold(struct ast_channel *chan)
Queue an unhold frame.
Definition channel.c:1274
int ast_queue_hold(struct ast_channel *chan, const char *musicclass)
Queue a hold frame.
Definition channel.c:1249
#define ast_channel_unlock(chan)
Definition channel.h:2990
ast_media_type
Types of media.
Definition codec.h:30
@ AST_MEDIA_TYPE_AUDIO
Definition codec.h:32
#define SCOPE_EXIT_RTN_VALUE(__return_value,...)
#define SCOPE_ENTER(level,...)
@ AST_CONTROL_UPDATE_RTP_PEER
struct ast_frame ast_null_frame
Definition main/frame.c:79
@ AST_AF_UNSPEC
Definition netsock2.h:54
int ast_sockaddr_resolve(struct ast_sockaddr **addrs, const char *str, int flags, int family)
Parses a string with an IPv4 or IPv6 address and place results into an array.
Definition netsock2.c:280
#define ast_sockaddr_set_port(addr, port)
Sets the port number of a socket address.
Definition netsock2.h:532
static int send_keepalive(const void *data)
static void set_session_media_remotely_held(struct ast_sip_session_media *session_media, const struct ast_sip_session *session, const pjmedia_sdp_media *media, const struct ast_stream *stream, const struct ast_sockaddr *addrs)
static enum ast_sip_session_media_encryption check_endpoint_media_transport(struct ast_sip_endpoint *endpoint, const struct pjmedia_sdp_media *stream)
Checks whether the encryption offered in SDP is compatible with the endpoint's configuration.
static struct ast_frame * media_session_rtcp_read_callback(struct ast_sip_session *session, struct ast_sip_session_media *session_media)
static void process_ice_attributes(struct ast_sip_session *session, struct ast_sip_session_media *session_media, const struct pjmedia_sdp_session *remote, const struct pjmedia_sdp_media *remote_stream)
Function which processes ICE attributes in an audio stream.
static int set_caps(struct ast_sip_session *session, struct ast_sip_session_media *session_media, struct ast_sip_session_media *session_media_transport, const struct pjmedia_sdp_media *stream, int is_offer, struct ast_stream *asterisk_stream)
static int media_session_rtp_write_callback(struct ast_sip_session *session, struct ast_sip_session_media *session_media, struct ast_frame *frame)
static int rtp_check_timeout(const void *data)
Check whether RTP is being received or not.
static int setup_media_encryption(struct ast_sip_session *session, struct ast_sip_session_media *session_media, const struct pjmedia_sdp_session *sdp, const struct pjmedia_sdp_media *stream)
static void process_extmap_attributes(struct ast_sip_session *session, struct ast_sip_session_media *session_media, const struct pjmedia_sdp_media *remote_stream)
Function which processes extmap attributes in a stream.
static struct ast_frame * media_session_rtp_read_callback(struct ast_sip_session *session, struct ast_sip_session_media *session_media)
static void enable_rtcp(struct ast_sip_session *session, struct ast_sip_session_media *session_media, const struct pjmedia_sdp_media *remote_media)
Enable RTCP on an RTP session.
static int create_rtp(struct ast_sip_session *session, struct ast_sip_session_media *session_media, const pjmedia_sdp_session *sdp)
Internal function which creates an RTP instance.
static void set_ice_components(struct ast_sip_session *session, struct ast_sip_session_media *session_media)
static void process_ssrc_attributes(struct ast_sip_session *session, struct ast_sip_session_media *session_media, const struct pjmedia_sdp_media *remote_stream)
Function which processes ssrc attributes in a stream.
int ast_sip_session_media_add_read_callback(struct ast_sip_session *session, struct ast_sip_session_media *session_media, int fd, ast_sip_session_media_read_cb callback)
Set a read callback for a media session with a specific file descriptor.
struct ast_sip_session_media * ast_sip_session_media_get_transport(struct ast_sip_session *session, struct ast_sip_session_media *session_media)
Retrieve the underlying media session that is acting as transport for a media session.
int ast_sip_session_media_set_write_callback(struct ast_sip_session *session, struct ast_sip_session_media *session_media, ast_sip_session_media_write_cb callback)
Set a write callback for a media session.
const char * ast_sip_session_get_name(const struct ast_sip_session *session)
Get the channel or endpoint name associated with the session.
#define NULL
Definition resample.c:96
void ast_rtp_instance_stop(struct ast_rtp_instance *instance)
Stop an RTP instance.
void ast_rtp_instance_set_stream_num(struct ast_rtp_instance *instance, int stream_num)
Set the stream number for an RTP instance.
int ast_rtp_instance_get_timeout(struct ast_rtp_instance *instance)
Get the RTP timeout value.
int ast_rtp_instance_bundle(struct ast_rtp_instance *child, struct ast_rtp_instance *parent)
Request that an RTP instance be bundled with another.
#define ast_rtp_instance_set_remote_address(instance, address)
Set the address of the remote endpoint that we are sending RTP to.
int ast_rtp_instance_activate(struct ast_rtp_instance *instance)
Indicate to the RTP engine that packets are now expected to be sent/received on the RTP instance.
void ast_rtp_instance_set_channel_id(struct ast_rtp_instance *instance, const char *uniqueid)
Set the channel that owns this RTP instance.
Definition rtp_engine.c:604
int ast_rtp_instance_fd(struct ast_rtp_instance *instance, int rtcp)
Get the file descriptor for an RTP session (or RTCP)
void ast_rtp_instance_set_keepalive(struct ast_rtp_instance *instance, int timeout)
Set the RTP keepalive interval.
void ast_rtp_instance_set_timeout(struct ast_rtp_instance *instance, int timeout)
Set the RTP timeout value.
#define AST_SCHED_DEL(sched, id)
Remove a scheduler entry.
Definition sched.h:46
int ast_sched_add_variable(struct ast_sched_context *con, int when, ast_sched_cb callback, const void *data, int variable) attribute_warn_unused_result
Adds a scheduled event with rescheduling support.
Definition sched.c:526
const char * ast_stream_to_str(const struct ast_stream *stream, struct ast_str **buf)
Get a string representing the stream for debugging/display purposes.
Definition stream.c:337
int ast_stream_get_position(const struct ast_stream *stream)
Get the position of the stream in the topology.
Definition stream.c:500
#define ast_str_tmp(init_len, __expr)
Provides a temporary ast_str and returns a copy of its buffer.
Definition strings.h:1189
unsigned int remotely_held_changed
Stream is held by remote side changed during this negotiation.
unsigned int locally_held
Stream is on hold by local side.
unsigned int remote_rtcp_mux
Does remote support rtcp_mux.
int timeout_sched_id
Scheduler ID for RTP timeout.
unsigned int remotely_held
Stream is on hold by remote side.
int keepalive_sched_id
Scheduler ID for RTP keepalive.
Socket address structure.
Definition netsock2.h:97
Definition sched.c:76

References AST_AF_UNSPEC, ast_channel_lock, ast_channel_uniqueid(), ast_channel_unlock, AST_CONTROL_UPDATE_RTP_PEER, ast_copy_pj_str(), ast_free, AST_MEDIA_TYPE_AUDIO, AST_MEDIA_TYPE_VIDEO, ast_null_frame, ast_queue_control(), ast_queue_frame(), ast_queue_hold(), ast_queue_unhold(), ast_rtp_instance_activate(), ast_rtp_instance_bundle(), ast_rtp_instance_fd(), ast_rtp_instance_get_timeout(), ast_rtp_instance_set_channel_id(), ast_rtp_instance_set_keepalive(), ast_rtp_instance_set_remote_address, ast_rtp_instance_set_stream_num(), ast_rtp_instance_set_timeout(), ast_rtp_instance_stop(), ast_sched_add_variable(), AST_SCHED_DEL, AST_SIP_MEDIA_TRANSPORT_INVALID, ast_sip_session_get_name(), ast_sip_session_media_add_read_callback(), ast_sip_session_media_get_transport(), ast_sip_session_media_set_write_callback(), ast_sockaddr_resolve(), ast_sockaddr_set_port, ast_str_tmp, ast_stream_get_position(), ast_stream_to_str(), ast_sip_session_media::bundled, check_endpoint_media_transport(), create_rtp(), enable_rtcp(), ast_sip_session_media::encryption, ast_sip_session_media::keepalive_sched_id, ast_sip_session_media::locally_held, media_session_rtcp_read_callback(), media_session_rtp_read_callback(), media_session_rtp_write_callback(), NULL, PARSE_PORT_FORBID, process_extmap_attributes(), process_ice_attributes(), process_ssrc_attributes(), RAII_VAR, ast_sip_session_media::remote_rtcp_mux, ast_sip_session_media::remotely_held, ast_sip_session_media::remotely_held_changed, ast_sip_session_media::rtp, rtp_check_timeout(), SCOPE_ENTER, SCOPE_EXIT_RTN_VALUE, send_keepalive(), session, set_caps(), set_ice_components(), set_session_media_remotely_held(), setup_media_encryption(), ast_sip_session_media::timeout_sched_id, and ast_sip_session_media::type.

◆ AST_MODULE_SELF_SYM()

struct ast_module * AST_MODULE_SELF_SYM ( void  )

Definition at line 2565 of file res_pjsip_sdp_rtp.c.

◆ change_outgoing_sdp_stream_media_address()

static void change_outgoing_sdp_stream_media_address ( pjsip_tx_data *  tdata,
struct pjmedia_sdp_media *  stream,
struct ast_sip_transport transport 
)
static

Function which updates the media stream with external media address, if applicable.

Definition at line 2402 of file res_pjsip_sdp_rtp.c.

2403{
2405 char host[NI_MAXHOST];
2406 struct ast_sockaddr our_sdp_addr = { { 0, } };
2407
2408 /* If the stream has been rejected there will be no connection line */
2409 if (!stream->conn || !transport_state) {
2410 return;
2411 }
2412
2413 ast_copy_pj_str(host, &stream->conn->addr, sizeof(host));
2414 ast_sockaddr_parse(&our_sdp_addr, host, PARSE_PORT_FORBID);
2415
2416 /* Reversed check here. We don't check the remote endpoint being
2417 * in our local net, but whether our outgoing session IP is
2418 * local. If it is not, we won't do rewriting. No localnet
2419 * configured? Always rewrite. */
2420 if (ast_sip_transport_is_nonlocal(transport_state, &our_sdp_addr) && transport_state->localnet) {
2421 return;
2422 }
2423 ast_debug(5, "Setting media address to %s\n", ast_sockaddr_stringify_addr_remote(&transport_state->external_media_address));
2424 pj_strdup2(tdata->pool, &stream->conn->addr, ast_sockaddr_stringify_addr_remote(&transport_state->external_media_address));
2425}
#define ao2_cleanup(obj)
Definition astobj2.h:1934
int ast_sockaddr_parse(struct ast_sockaddr *addr, const char *str, int flags)
Parse an IPv4 or IPv6 address string.
Definition netsock2.c:230
#define ast_sip_transport_is_nonlocal(transport_state, addr)
Definition res_pjsip.h:210
struct ast_sip_transport_state * ast_sip_get_transport_state(const char *transport_id)
Retrieve transport state.
const char * ast_sorcery_object_get_id(const void *object)
Get the unique identifier of a sorcery object.
Definition sorcery.c:2381
Structure for SIP transport information.
Definition res_pjsip.h:117

References ao2_cleanup, ast_copy_pj_str(), ast_debug, ast_sip_get_transport_state(), ast_sip_transport_is_nonlocal, ast_sockaddr_parse(), ast_sockaddr_stringify_addr_remote(), ast_sorcery_object_get_id(), PARSE_PORT_FORBID, RAII_VAR, and ast_sip_session_media::transport.

◆ check_endpoint_media_transport()

static enum ast_sip_session_media_encryption check_endpoint_media_transport ( struct ast_sip_endpoint endpoint,
const struct pjmedia_sdp_media *  stream 
)
static

Checks whether the encryption offered in SDP is compatible with the endpoint's configuration.

Definition at line 1044 of file res_pjsip_sdp_rtp.c.

1047{
1048 enum ast_sip_session_media_encryption incoming_encryption;
1049 char transport_end = stream->desc.transport.ptr[stream->desc.transport.slen - 1];
1050 unsigned int optimistic;
1051
1052 if ((transport_end == 'F' && !endpoint->media.rtp.use_avpf)
1053 || (transport_end != 'F' && endpoint->media.rtp.use_avpf)) {
1055 }
1056
1057 incoming_encryption = get_media_encryption_type(stream->desc.transport, stream, &optimistic);
1058
1059 if (incoming_encryption == endpoint->media.rtp.encryption) {
1060 return incoming_encryption;
1061 }
1062
1063 if (endpoint->media.rtp.force_avp ||
1064 endpoint->media.rtp.encryption_optimistic) {
1065 return incoming_encryption;
1066 }
1067
1068 /* If an optimistic offer has been made but encryption is not enabled consider it as having
1069 * no offer of crypto at all instead of invalid so the session proceeds.
1070 */
1071 if (optimistic) {
1073 }
1074
1076}
ast_sip_session_media_encryption
Definition res_pjsip.h:733
static enum ast_sip_session_media_encryption get_media_encryption_type(pj_str_t transport, const struct pjmedia_sdp_media *stream, unsigned int *optimistic)
figure out media transport encryption type from the media transport string
struct ast_sip_media_rtp_configuration rtp
Definition res_pjsip.h:1017
struct ast_sip_endpoint_media_configuration media
Definition res_pjsip.h:1098
enum ast_sip_session_media_encryption encryption
Definition res_pjsip.h:956

References AST_SIP_MEDIA_ENCRYPT_NONE, AST_SIP_MEDIA_TRANSPORT_INVALID, ast_sip_media_rtp_configuration::encryption, ast_sip_media_rtp_configuration::encryption_optimistic, ast_sip_media_rtp_configuration::force_avp, get_media_encryption_type(), ast_sip_endpoint::media, ast_sip_endpoint_media_configuration::rtp, and ast_sip_media_rtp_configuration::use_avpf.

Referenced by apply_negotiated_sdp_stream(), and negotiate_incoming_sdp_stream().

◆ check_ice_support()

static void check_ice_support ( struct ast_sip_session session,
struct ast_sip_session_media session_media,
const struct pjmedia_sdp_media *  remote_stream 
)
static

Function which checks for ice attributes in an audio stream.

Definition at line 824 of file res_pjsip_sdp_rtp.c.

826{
827 struct ast_rtp_engine_ice *ice;
828 const pjmedia_sdp_attr *attr;
829 unsigned int attr_i;
830
831 if (!session->endpoint->media.rtp.ice_support || !(ice = ast_rtp_instance_get_ice(session_media->rtp))) {
832 session_media->remote_ice = 0;
833 return;
834 }
835
836 /* Find all of the candidates */
837 for (attr_i = 0; attr_i < remote_stream->attr_count; ++attr_i) {
838 attr = remote_stream->attr[attr_i];
839 if (!pj_strcmp2(&attr->name, "candidate")) {
840 session_media->remote_ice = 1;
841 break;
842 }
843 }
844
845 if (attr_i == remote_stream->attr_count) {
846 session_media->remote_ice = 0;
847 }
848}
void(* start)(struct ast_rtp_instance *instance)
Definition rtp_engine.h:542

References ast_rtp_instance_get_ice(), session, and ast_rtp_engine_ice::start.

Referenced by negotiate_incoming_sdp_stream().

◆ create_outgoing_sdp_stream()

static int create_outgoing_sdp_stream ( struct ast_sip_session session,
struct ast_sip_session_media session_media,
struct pjmedia_sdp_session *  sdp,
const struct pjmedia_sdp_session *  remote,
struct ast_stream stream 
)
static

Function which creates an outgoing stream.

Definition at line 1794 of file res_pjsip_sdp_rtp.c.

1796{
1797 pj_pool_t *pool = session->inv_session->pool_prov;
1798 static const pj_str_t STR_RTP_AVP = { "RTP/AVP", 7 };
1799 static const pj_str_t STR_IN = { "IN", 2 };
1800 static const pj_str_t STR_IP4 = { "IP4", 3};
1801 static const pj_str_t STR_IP6 = { "IP6", 3};
1802 static const pj_str_t STR_SENDRECV = { "sendrecv", 8 };
1803 static const pj_str_t STR_SENDONLY = { "sendonly", 8 };
1804 static const pj_str_t STR_INACTIVE = { "inactive", 8 };
1805 static const pj_str_t STR_RECVONLY = { "recvonly", 8 };
1806 pjmedia_sdp_media *media;
1807 const char *hostip = NULL;
1808 struct ast_sockaddr addr;
1809 char tmp[512];
1810 pj_str_t stmp;
1811 pjmedia_sdp_attr *attr;
1812 int index = 0;
1813 int noncodec = (session->dtmf == AST_SIP_DTMF_RFC_4733 || session->dtmf == AST_SIP_DTMF_AUTO || session->dtmf == AST_SIP_DTMF_AUTO_INFO) ? AST_RTP_DTMF : 0;
1814 int min_packet_size = 0, max_packet_size = 0;
1815 int rtp_code;
1816 RAII_VAR(struct ast_format_cap *, caps, NULL, ao2_cleanup);
1817 enum ast_media_type media_type = session_media->type;
1818 struct ast_sip_session_media *session_media_transport;
1819 pj_sockaddr ip;
1820 int direct_media_enabled = !ast_sockaddr_isnull(&session_media->direct_media_addr) &&
1821 ast_format_cap_count(session->direct_media_cap);
1822
1823 /* Keep track of the sample rates for offered codecs so we can build matching
1824 RFC 2833/4733 payload offers. */
1825 AST_VECTOR(, int) sample_rates;
1826 /* In case we can't init the sample rates, still try to do the rest. */
1827 int build_dtmf_sample_rates = 1;
1828
1829 SCOPE_ENTER(1, "%s Type: %s %s\n", ast_sip_session_get_name(session),
1830 ast_codec_media_type2str(media_type), ast_str_tmp(128, ast_stream_to_str(stream, &STR_TMP)));
1831
1832 media = pj_pool_zalloc(pool, sizeof(struct pjmedia_sdp_media));
1833 if (!media) {
1834 SCOPE_EXIT_RTN_VALUE(-1, "Pool alloc failure\n");
1835 }
1836 pj_strdup2(pool, &media->desc.media, ast_codec_media_type2str(session_media->type));
1837
1838 /* If this is a removed (or declined) stream OR if no formats exist then construct a minimal stream in SDP */
1841 media->desc.port = 0;
1842 media->desc.port_count = 1;
1843
1844 if (remote && remote->media[ast_stream_get_position(stream)]) {
1845 pjmedia_sdp_media *remote_media = remote->media[ast_stream_get_position(stream)];
1846 int index;
1847
1848 media->desc.transport = remote_media->desc.transport;
1849
1850 /* Preserve existing behavior by copying the formats provided from the offer */
1851 for (index = 0; index < remote_media->desc.fmt_count; ++index) {
1852 media->desc.fmt[index] = remote_media->desc.fmt[index];
1853 }
1854 media->desc.fmt_count = remote_media->desc.fmt_count;
1855 } else {
1856 /* This is actually an offer so put a dummy payload in that is ignored and sane transport */
1857 media->desc.transport = STR_RTP_AVP;
1858 pj_strdup2(pool, &media->desc.fmt[media->desc.fmt_count++], "32");
1859 }
1860
1861 sdp->media[sdp->media_count++] = media;
1863
1864 SCOPE_EXIT_RTN_VALUE(1, "Stream removed or no formats\n");
1865 }
1866
1867 if (!session_media->rtp && create_rtp(session, session_media, sdp)) {
1868 SCOPE_EXIT_RTN_VALUE(-1, "Couldn't create rtp\n");
1869 }
1870
1871 /* If this stream has not been bundled already it is new and we need to ensure there is no SSRC conflict */
1872 if (session_media->bundle_group != -1 && !session_media->bundled) {
1873 for (index = 0; index < sdp->media_count; ++index) {
1874 struct ast_sip_session_media *other_session_media;
1875
1876 other_session_media = AST_VECTOR_GET(&session->pending_media_state->sessions, index);
1877 if (!other_session_media->rtp || other_session_media->bundle_group != session_media->bundle_group) {
1878 continue;
1879 }
1880
1881 if (ast_rtp_instance_get_ssrc(session_media->rtp) == ast_rtp_instance_get_ssrc(other_session_media->rtp)) {
1882 ast_rtp_instance_change_source(session_media->rtp);
1883 /* Start the conflict check over again */
1884 index = -1;
1885 continue;
1886 }
1887 }
1888 }
1889
1890 session_media_transport = ast_sip_session_media_get_transport(session, session_media);
1891
1892 if (session_media_transport == session_media || !session_media->bundled) {
1893 set_ice_components(session, session_media);
1894 enable_rtcp(session, session_media, NULL);
1895
1896 /* Crypto has to be added before setting the media transport so that SRTP is properly
1897 * set up according to the configuration. This ends up changing the media transport.
1898 */
1899 if (add_crypto_to_stream(session, session_media, pool, media)) {
1900 SCOPE_EXIT_RTN_VALUE(-1, "Couldn't add crypto\n");
1901 }
1902
1903 if (pj_strlen(&session_media->transport)) {
1904 /* If a transport has already been specified use it */
1905 media->desc.transport = session_media->transport;
1906 } else {
1907 media->desc.transport = pj_str(ast_sdp_get_rtp_profile(
1908 /* Optimistic encryption places crypto in the normal RTP/AVP profile */
1909 !session->endpoint->media.rtp.encryption_optimistic &&
1910 (session_media->encryption == AST_SIP_MEDIA_ENCRYPT_SDES),
1911 session_media->rtp, session->endpoint->media.rtp.use_avpf,
1912 session->endpoint->media.rtp.force_avp));
1913 }
1914
1915 media->conn = pj_pool_zalloc(pool, sizeof(struct pjmedia_sdp_conn));
1916 if (!media->conn) {
1917 SCOPE_EXIT_RTN_VALUE(-1, "Pool alloc failure\n");
1918 }
1919
1920 /* Add connection level details */
1921 if (direct_media_enabled) {
1923 } else if (ast_strlen_zero(session->endpoint->media.address)) {
1924 hostip = ast_sip_get_host_ip_string(session->endpoint->media.rtp.ipv6 ? pj_AF_INET6() : pj_AF_INET());
1925 } else {
1926 hostip = session->endpoint->media.address;
1927 }
1928
1929 if (ast_strlen_zero(hostip)) {
1930 ast_log(LOG_ERROR, "No local host IP available for stream %s\n",
1931 ast_codec_media_type2str(session_media->type));
1932 SCOPE_EXIT_RTN_VALUE(-1, "No local host ip\n");
1933 }
1934
1935 media->conn->net_type = STR_IN;
1936 /* Assume that the connection will use IPv4 until proven otherwise */
1937 media->conn->addr_type = STR_IP4;
1938 pj_strdup2(pool, &media->conn->addr, hostip);
1939
1940 if ((pj_sockaddr_parse(pj_AF_UNSPEC(), 0, &media->conn->addr, &ip) == PJ_SUCCESS) &&
1941 (ip.addr.sa_family == pj_AF_INET6())) {
1942 media->conn->addr_type = STR_IP6;
1943 }
1944
1945 /* Add ICE attributes and candidates */
1946 add_ice_to_stream(session, session_media, pool, media, 1);
1947
1948 ast_rtp_instance_get_local_address(session_media->rtp, &addr);
1949 media->desc.port = direct_media_enabled ? ast_sockaddr_port(&session_media->direct_media_addr) : (pj_uint16_t) ast_sockaddr_port(&addr);
1950 media->desc.port_count = 1;
1951 } else {
1952 pjmedia_sdp_media *bundle_group_stream = sdp->media[session_media_transport->stream_num];
1953
1954 /* As this is in a bundle group it shares the same details as the group instance */
1955 media->desc.transport = bundle_group_stream->desc.transport;
1956 media->conn = bundle_group_stream->conn;
1957 media->desc.port = bundle_group_stream->desc.port;
1958
1959 if (add_crypto_to_stream(session, session_media_transport, pool, media)) {
1960 SCOPE_EXIT_RTN_VALUE(-1, "Couldn't add crypto\n");
1961 }
1962
1963 add_ice_to_stream(session, session_media_transport, pool, media, 0);
1964
1965 enable_rtcp(session, session_media, NULL);
1966 }
1967
1969 ast_log(LOG_ERROR, "Failed to allocate %s capabilities\n",
1970 ast_codec_media_type2str(session_media->type));
1971 SCOPE_EXIT_RTN_VALUE(-1, "Couldn't create caps\n");
1972 }
1973
1974 if (direct_media_enabled) {
1975 ast_format_cap_get_compatible(session->endpoint->media.codecs, session->direct_media_cap, caps);
1976 } else {
1977 ast_format_cap_append_from_cap(caps, ast_stream_get_formats(stream), media_type);
1978 }
1979
1980 /* Init the sample rates before we start adding them. Assume we will have at least one. */
1981 if (AST_VECTOR_INIT(&sample_rates, 1)) {
1982 ast_log(LOG_ERROR, "Unable to add dtmf formats to SDP!\n");
1983 build_dtmf_sample_rates = 0;
1984 }
1985
1986 for (index = 0; index < ast_format_cap_count(caps); ++index) {
1987 struct ast_format *format = ast_format_cap_get_format(caps, index);
1988
1989 if (ast_format_get_type(format) != media_type) {
1990 ao2_ref(format, -1);
1991 continue;
1992 }
1993
1994 /* It is possible for some formats not to have SDP information available for them
1995 * and if this is the case, skip over them so the SDP can still be created.
1996 */
1997 if (!ast_rtp_lookup_sample_rate2(1, format, 0)) {
1998 ast_log(LOG_WARNING, "Format '%s' can not be added to SDP, consider disallowing it on endpoint '%s'\n",
2000 ao2_ref(format, -1);
2001 continue;
2002 }
2003
2004 /* If this stream is not a transport we need to use the transport codecs structure for payload management to prevent
2005 * conflicts.
2006 */
2007 if (session_media_transport != session_media) {
2008 if ((rtp_code = ast_rtp_codecs_payload_code(ast_rtp_instance_get_codecs(session_media_transport->rtp), 1, format, 0)) == -1) {
2009 ast_log(LOG_WARNING,"Unable to get rtp codec payload code for %s\n", ast_format_get_name(format));
2010 ao2_ref(format, -1);
2011 continue;
2012 }
2013 /* Our instance has to match the payload number though */
2014 ast_rtp_codecs_payload_set_rx(ast_rtp_instance_get_codecs(session_media->rtp), rtp_code, format);
2015 } else {
2016 if ((rtp_code = ast_rtp_codecs_payload_code(ast_rtp_instance_get_codecs(session_media->rtp), 1, format, 0)) == -1) {
2017 ast_log(LOG_WARNING,"Unable to get rtp codec payload code for %s\n", ast_format_get_name(format));
2018 ao2_ref(format, -1);
2019 continue;
2020 }
2021 }
2022
2023 if ((attr = generate_rtpmap_attr(session, media, pool, rtp_code, 1, format, 0))) {
2024 int i, added = 0;
2025 int newrate = ast_rtp_lookup_sample_rate2(1, format, 0);
2026 if (build_dtmf_sample_rates) {
2027 for (i = 0; i < AST_VECTOR_SIZE(&sample_rates); i++) {
2028 /* Only add if we haven't already processed this sample rate. For instance
2029 A-law and u-law 'share' one 8K DTMF payload type. */
2030 if (newrate == AST_VECTOR_GET(&sample_rates, i)) {
2031 added = 1;
2032 break;
2033 }
2034 }
2035
2036 if (!added) {
2037 AST_VECTOR_APPEND(&sample_rates, newrate);
2038 }
2039 }
2040 media->attr[media->attr_count++] = attr;
2041 }
2042
2043 if ((attr = generate_fmtp_attr(pool, format, rtp_code))) {
2044 media->attr[media->attr_count++] = attr;
2045 }
2046
2047 if (ast_format_get_maximum_ms(format) &&
2048 ((ast_format_get_maximum_ms(format) < max_packet_size) || !max_packet_size)) {
2049 max_packet_size = ast_format_get_maximum_ms(format);
2050 }
2051 ao2_ref(format, -1);
2052
2053 if (media->desc.fmt_count == PJMEDIA_MAX_SDP_FMT) {
2054 break;
2055 }
2056 }
2057
2058 /* Add non-codec formats */
2060 && media->desc.fmt_count < PJMEDIA_MAX_SDP_FMT) {
2061 for (index = 1LL; index <= AST_RTP_MAX; index <<= 1) {
2062 if (!(noncodec & index)) {
2063 continue;
2064 }
2065
2066 if (index != AST_RTP_DTMF) {
2067 rtp_code = ast_rtp_codecs_payload_code(
2068 ast_rtp_instance_get_codecs(session_media->rtp), 0, NULL, index);
2069 if (rtp_code == -1) {
2070 continue;
2071 } else if ((attr = generate_rtpmap_attr(session, media, pool, rtp_code, 0, NULL, index))) {
2072 media->attr[media->attr_count++] = attr;
2073 }
2074 } else if (build_dtmf_sample_rates) {
2075 /*
2076 * Walk through the possible bitrates for the RFC 2833/4733 digits and generate the rtpmap
2077 * attributes.
2078 */
2079 int i, found_default_offer = 0;
2080 for (i = 0; i < AST_VECTOR_SIZE(&sample_rates); i++) {
2082 ast_rtp_instance_get_codecs(session_media->rtp), 0, NULL, index, AST_VECTOR_GET(&sample_rates, i));
2083
2084 if (rtp_code == -1) {
2085 continue;
2086 }
2087
2088 if (AST_VECTOR_GET(&sample_rates, i) == DEFAULT_DTMF_SAMPLE_RATE_MS) {
2089 /* we found and added a default offer, so no need to include a default one.*/
2090 found_default_offer = 1;
2091 }
2092
2093 if ((attr = generate_rtpmap_attr2(session, media, pool, rtp_code, 0, NULL, index, AST_VECTOR_GET(&sample_rates, i)))) {
2094 media->attr[media->attr_count++] = attr;
2095 snprintf(tmp, sizeof(tmp), "%d 0-16", (rtp_code));
2096 attr = pjmedia_sdp_attr_create(pool, "fmtp", pj_cstr(&stmp, tmp));
2097 media->attr[media->attr_count++] = attr;
2098 }
2099 }
2100
2101 /* If we weren't able to add any matching RFC 2833/4733, assume this endpoint is using a
2102 * mismatched 8K offer and try to add one as a fall-back/default.
2103 */
2104 if (!found_default_offer) {
2107
2108 if (rtp_code != -1 && (attr = generate_rtpmap_attr2(session, media, pool, rtp_code, 0, NULL, index, DEFAULT_DTMF_SAMPLE_RATE_MS))) {
2109 media->attr[media->attr_count++] = attr;
2110 snprintf(tmp, sizeof(tmp), "%d 0-16", (rtp_code));
2111 attr = pjmedia_sdp_attr_create(pool, "fmtp", pj_cstr(&stmp, tmp));
2112 media->attr[media->attr_count++] = attr;
2113 }
2114 }
2115 }
2116
2117 if (media->desc.fmt_count == PJMEDIA_MAX_SDP_FMT) {
2118 break;
2119 }
2120 }
2121 }
2122
2123 /* we are done with the sample rates */
2124 AST_VECTOR_FREE(&sample_rates);
2125
2126 /* If no formats were actually added to the media stream don't add it to the SDP */
2127 if (!media->desc.fmt_count) {
2128 SCOPE_EXIT_RTN_VALUE(1, "No formats added to stream\n");
2129 }
2130
2131 /* If ptime is set add it as an attribute */
2132 min_packet_size = ast_rtp_codecs_get_framing(ast_rtp_instance_get_codecs(session_media->rtp));
2133 if (!min_packet_size) {
2134 min_packet_size = ast_format_cap_get_framing(caps);
2135 }
2136 if (min_packet_size) {
2137 snprintf(tmp, sizeof(tmp), "%d", min_packet_size);
2138 attr = pjmedia_sdp_attr_create(pool, "ptime", pj_cstr(&stmp, tmp));
2139 media->attr[media->attr_count++] = attr;
2140 }
2141
2142 if (max_packet_size) {
2143 snprintf(tmp, sizeof(tmp), "%d", max_packet_size);
2144 attr = pjmedia_sdp_attr_create(pool, "maxptime", pj_cstr(&stmp, tmp));
2145 media->attr[media->attr_count++] = attr;
2146 }
2147
2148 attr = PJ_POOL_ZALLOC_T(pool, pjmedia_sdp_attr);
2149 if (session_media->locally_held) {
2150 if (session_media->remotely_held) {
2151 attr->name = STR_INACTIVE; /* To place on hold a recvonly stream, send inactive */
2152 } else {
2153 attr->name = STR_SENDONLY; /* Send sendonly to initate a local hold */
2154 }
2155 } else {
2156 if (session_media->remotely_held) {
2157 attr->name = STR_RECVONLY; /* Remote has sent sendonly, reply recvonly */
2158 } else if (ast_stream_get_state(stream) == AST_STREAM_STATE_SENDONLY) {
2159 attr->name = STR_SENDONLY; /* Stream has requested sendonly */
2160 } else if (ast_stream_get_state(stream) == AST_STREAM_STATE_RECVONLY) {
2161 attr->name = STR_RECVONLY; /* Stream has requested recvonly */
2162 } else if (ast_stream_get_state(stream) == AST_STREAM_STATE_INACTIVE) {
2163 attr->name = STR_INACTIVE; /* Stream has requested inactive */
2164 } else {
2165 attr->name = STR_SENDRECV; /* No hold in either direction */
2166 }
2167 }
2168 media->attr[media->attr_count++] = attr;
2169
2170 /* If we've got rtcp-mux enabled, add it unless we received an offer without it */
2171 if (session->endpoint->media.rtcp_mux && session_media->remote_rtcp_mux) {
2172 attr = pjmedia_sdp_attr_create(pool, "rtcp-mux", NULL);
2173 pjmedia_sdp_attr_add(&media->attr_count, media->attr, attr);
2174 }
2175
2176 add_ssrc_to_stream(session, session_media, pool, media);
2177 add_msid_to_stream(session, session_media, pool, media, stream);
2178 add_rtcp_fb_to_stream(session, session_media, pool, media);
2179 add_extmap_to_stream(session, session_media, pool, media);
2180
2181 /* Add the media stream to the SDP */
2182 sdp->media[sdp->media_count++] = media;
2183
2184 SCOPE_EXIT_RTN_VALUE(1, "RC: 1\n");
2185}
enum ast_media_type ast_format_get_type(const struct ast_format *format)
Get the media type of a format.
Definition format.c:354
const char * ast_format_get_name(const struct ast_format *format)
Get the name associated with a format.
Definition format.c:334
unsigned int ast_format_get_maximum_ms(const struct ast_format *format)
Get the maximum amount of media carried in this format.
Definition format.c:369
unsigned int ast_format_cap_get_framing(const struct ast_format_cap *cap)
Get the global framing.
Definition format_cap.c:438
int ast_format_cap_get_compatible(const struct ast_format_cap *cap1, const struct ast_format_cap *cap2, struct ast_format_cap *result)
Find the compatible formats between two capabilities structures.
Definition format_cap.c:628
@ AST_FORMAT_CAP_FLAG_DEFAULT
Definition format_cap.h:38
int ast_format_cap_append_from_cap(struct ast_format_cap *dst, const struct ast_format_cap *src, enum ast_media_type type)
Append the formats of provided type in src to dst.
Definition format_cap.c:269
#define ast_format_cap_alloc(flags)
Allocate a new ast_format_cap structure.
Definition format_cap.h:49
#define LOG_ERROR
char * ast_sockaddr_stringify_fmt(const struct ast_sockaddr *addr, int format)
Convert a socket address to a string.
Definition netsock2.c:65
#define AST_SOCKADDR_STR_ADDR
Definition netsock2.h:198
#define ast_sockaddr_port(addr)
Get the port number of a socket address.
Definition netsock2.h:517
const char * ast_sip_get_host_ip_string(int af)
Retrieve the local host address in string form.
Definition res_pjsip.c:2464
@ AST_SIP_DTMF_AUTO_INFO
Definition res_pjsip.h:558
@ AST_SIP_DTMF_AUTO
Definition res_pjsip.h:556
@ AST_SIP_DTMF_RFC_4733
Definition res_pjsip.h:550
static pjmedia_sdp_attr * generate_rtpmap_attr2(struct ast_sip_session *session, pjmedia_sdp_media *media, pj_pool_t *pool, int rtp_code, int asterisk_format, struct ast_format *format, int code, int sample_rate)
static void add_ice_to_stream(struct ast_sip_session *session, struct ast_sip_session_media *session_media, pj_pool_t *pool, pjmedia_sdp_media *media, unsigned int include_candidates)
Function which adds ICE attributes to a media stream.
static void add_extmap_to_stream(struct ast_sip_session *session, struct ast_sip_session_media *session_media, pj_pool_t *pool, pjmedia_sdp_media *media)
static void add_rtcp_fb_to_stream(struct ast_sip_session *session, struct ast_sip_session_media *session_media, pj_pool_t *pool, pjmedia_sdp_media *media)
static void add_msid_to_stream(struct ast_sip_session *session, struct ast_sip_session_media *session_media, pj_pool_t *pool, pjmedia_sdp_media *media, struct ast_stream *stream)
static pjmedia_sdp_attr * generate_rtpmap_attr(struct ast_sip_session *session, pjmedia_sdp_media *media, pj_pool_t *pool, int rtp_code, int asterisk_format, struct ast_format *format, int code)
static int add_crypto_to_stream(struct ast_sip_session *session, struct ast_sip_session_media *session_media, pj_pool_t *pool, pjmedia_sdp_media *media)
static void add_ssrc_to_stream(struct ast_sip_session *session, struct ast_sip_session_media *session_media, pj_pool_t *pool, pjmedia_sdp_media *media)
Function which adds ssrc attributes to a media stream.
static pjmedia_sdp_attr * generate_fmtp_attr(pj_pool_t *pool, struct ast_format *format, int rtp_code)
int ast_sip_session_is_pending_stream_default(const struct ast_sip_session *session, const struct ast_stream *stream)
Determines if a provided pending stream will be the default stream or not.
unsigned int ast_rtp_lookup_sample_rate2(int asterisk_format, const struct ast_format *format, int code)
Get the sample rate associated with known RTP payload types.
#define AST_RTP_DTMF
Definition rtp_engine.h:294
void ast_rtp_instance_get_local_address(struct ast_rtp_instance *instance, struct ast_sockaddr *address)
Get the local address that we are expecting RTP on.
Definition rtp_engine.c:694
int ast_rtp_codecs_payload_code_sample_rate(struct ast_rtp_codecs *codecs, int asterisk_format, struct ast_format *format, int code, unsigned int sample_rate)
Retrieve a rx mapped payload type based on whether it is an Asterisk format, the code and the sample ...
void ast_rtp_instance_change_source(struct ast_rtp_instance *instance)
Indicate a new source of audio has dropped in and the ssrc should change.
#define AST_RTP_MAX
Definition rtp_engine.h:300
unsigned int ast_rtp_codecs_get_framing(struct ast_rtp_codecs *codecs)
Get the framing used for a set of codecs.
#define DEFAULT_DTMF_SAMPLE_RATE_MS
Definition rtp_engine.h:110
char * ast_sdp_get_rtp_profile(unsigned int sdes_active, struct ast_rtp_instance *instance, unsigned int using_avpf, unsigned int force_avp)
Get the RTP profile in use by a media session.
Definition sdp_srtp.c:103
@ AST_STREAM_STATE_RECVONLY
Set when the stream is receiving media only.
Definition stream.h:90
@ AST_STREAM_STATE_INACTIVE
Set when the stream is not sending OR receiving media.
Definition stream.h:94
@ AST_STREAM_STATE_REMOVED
Set when the stream has been removed/declined.
Definition stream.h:78
@ AST_STREAM_STATE_SENDONLY
Set when the stream is sending media only.
Definition stream.h:86
void ast_stream_set_state(struct ast_stream *stream, enum ast_stream_state state)
Set the state of a stream.
Definition stream.c:380
enum ast_stream_state ast_stream_get_state(const struct ast_stream *stream)
Get the current state of a stream.
Definition stream.c:373
const struct ast_format_cap * ast_stream_get_formats(const struct ast_stream *stream)
Get the current negotiated formats of a stream.
Definition stream.c:330
Format capabilities structure, holds formats + preference order + etc.
Definition format_cap.c:54
int stream_num
The stream number to place into any resulting frames.
struct ast_sockaddr direct_media_addr
Direct media address.
pj_str_t transport
The media transport in use for this stream.
#define AST_VECTOR_SIZE(vec)
Get the number of elements in a vector.
Definition vector.h:620
#define AST_VECTOR_FREE(vec)
Deallocates this vector.
Definition vector.h:185
#define AST_VECTOR_INIT(vec, size)
Initialize a vector.
Definition vector.h:124
#define AST_VECTOR_APPEND(vec, elem)
Append an element to a vector, growing the vector if needed.
Definition vector.h:267
#define AST_VECTOR(name, type)
Define a vector structure.
Definition vector.h:44

References add_crypto_to_stream(), add_extmap_to_stream(), add_ice_to_stream(), add_msid_to_stream(), add_rtcp_fb_to_stream(), add_ssrc_to_stream(), ao2_cleanup, ao2_ref, ast_codec_media_type2str(), ast_format_cap_alloc, ast_format_cap_append_from_cap(), ast_format_cap_count(), AST_FORMAT_CAP_FLAG_DEFAULT, ast_format_cap_get_compatible(), ast_format_cap_get_format(), ast_format_cap_get_framing(), ast_format_get_maximum_ms(), ast_format_get_name(), ast_format_get_type(), ast_log, AST_MEDIA_TYPE_VIDEO, ast_rtp_codecs_get_framing(), ast_rtp_codecs_payload_code(), ast_rtp_codecs_payload_code_sample_rate(), ast_rtp_codecs_payload_set_rx(), AST_RTP_DTMF, ast_rtp_instance_change_source(), ast_rtp_instance_get_codecs(), ast_rtp_instance_get_local_address(), ast_rtp_instance_get_ssrc(), ast_rtp_lookup_sample_rate2(), AST_RTP_MAX, ast_sdp_get_rtp_profile(), AST_SIP_DTMF_AUTO, AST_SIP_DTMF_AUTO_INFO, AST_SIP_DTMF_RFC_4733, ast_sip_get_host_ip_string(), AST_SIP_MEDIA_ENCRYPT_SDES, ast_sip_session_get_name(), ast_sip_session_is_pending_stream_default(), ast_sip_session_media_get_transport(), ast_sockaddr_isnull(), ast_sockaddr_port, AST_SOCKADDR_STR_ADDR, ast_sockaddr_stringify_fmt(), ast_sorcery_object_get_id(), ast_str_tmp, ast_stream_get_formats(), ast_stream_get_position(), ast_stream_get_state(), ast_stream_set_state(), AST_STREAM_STATE_INACTIVE, AST_STREAM_STATE_RECVONLY, AST_STREAM_STATE_REMOVED, AST_STREAM_STATE_SENDONLY, ast_stream_to_str(), ast_strlen_zero(), AST_VECTOR, AST_VECTOR_APPEND, AST_VECTOR_FREE, AST_VECTOR_GET, AST_VECTOR_INIT, AST_VECTOR_SIZE, ast_sip_session_media::bundle_group, ast_sip_session_media::bundled, create_rtp(), DEFAULT_DTMF_SAMPLE_RATE_MS, ast_sip_session_media::direct_media_addr, enable_rtcp(), ast_sip_session_media::encryption, generate_fmtp_attr(), generate_rtpmap_attr(), generate_rtpmap_attr2(), ast_sip_session_media::locally_held, LOG_ERROR, LOG_WARNING, NULL, RAII_VAR, ast_sip_session_media::remote_rtcp_mux, ast_sip_session_media::remotely_held, ast_sip_session_media::rtp, SCOPE_ENTER, SCOPE_EXIT_RTN_VALUE, session, set_ice_components(), ast_sip_session_media::stream_num, ast_sip_session_media::transport, and ast_sip_session_media::type.

◆ create_rtp()

static int create_rtp ( struct ast_sip_session session,
struct ast_sip_session_media session_media,
const pjmedia_sdp_session *  sdp 
)
static

Internal function which creates an RTP instance.

Definition at line 225 of file res_pjsip_sdp_rtp.c.

227{
228 struct ast_rtp_engine_ice *ice;
229 struct ast_sockaddr temp_media_address;
230 struct ast_sockaddr *media_address = &address_rtp;
231
232 if (session->endpoint->media.bind_rtp_to_media_address && !ast_strlen_zero(session->endpoint->media.address)) {
233 if (ast_sockaddr_parse(&temp_media_address, session->endpoint->media.address, 0)) {
234 ast_debug_rtp(1, "Endpoint %s: Binding RTP media to %s\n",
236 session->endpoint->media.address);
237 media_address = &temp_media_address;
238 } else {
239 ast_debug_rtp(1, "Endpoint %s: RTP media address invalid: %s\n",
241 session->endpoint->media.address);
242 }
243 } else {
244 struct ast_sip_transport *transport;
245
246 transport = ast_sorcery_retrieve_by_id(ast_sip_get_sorcery(), "transport",
247 session->endpoint->transport);
248 if (transport) {
249 struct ast_sip_transport_state *trans_state;
250
252 if (trans_state) {
253 char hoststr[PJ_INET6_ADDRSTRLEN];
254
255 pj_sockaddr_print(&trans_state->host, hoststr, sizeof(hoststr), 0);
256 if (ast_sockaddr_parse(&temp_media_address, hoststr, 0)) {
257 ast_debug_rtp(1, "Transport %s bound to %s: Using it for RTP media.\n",
258 session->endpoint->transport, hoststr);
259 media_address = &temp_media_address;
260 } else {
261 ast_debug_rtp(1, "Transport %s bound to %s: Invalid for RTP media.\n",
262 session->endpoint->transport, hoststr);
263 }
264 ao2_ref(trans_state, -1);
265 }
266 ao2_ref(transport, -1);
267 }
268 }
269
270 if (session->endpoint->media.rtp.port_start && session->endpoint->media.rtp.port_end) {
272 .port_start = session->endpoint->media.rtp.port_start,
273 .port_end = session->endpoint->media.rtp.port_end,
274 };
275 if (!(session_media->rtp = ast_rtp_instance_new_with_options(
276 session->endpoint->media.rtp.engine, sched, media_address, NULL,
277 &options))) {
278 ast_log(LOG_ERROR, "Unable to create RTP instance using RTP engine '%s' with port range %u-%u\n",
279 session->endpoint->media.rtp.engine,
280 session->endpoint->media.rtp.port_start,
281 session->endpoint->media.rtp.port_end);
282 return -1;
283 }
284 } else {
285 if (!(session_media->rtp = ast_rtp_instance_new(session->endpoint->media.rtp.engine, sched, media_address, NULL))) {
286 ast_log(LOG_ERROR, "Unable to create RTP instance using RTP engine '%s'\n", session->endpoint->media.rtp.engine);
287 return -1;
288 }
289 }
290
291 ast_rtp_instance_set_prop(session_media->rtp, AST_RTP_PROPERTY_NAT, session->endpoint->media.rtp.symmetric);
292 ast_rtp_instance_set_prop(session_media->rtp, AST_RTP_PROPERTY_ASYMMETRIC_CODEC, session->endpoint->asymmetric_rtp_codec);
293
294 if (!session->endpoint->media.rtp.ice_support && (ice = ast_rtp_instance_get_ice(session_media->rtp))) {
295 ice->stop(session_media->rtp);
296 }
297
301 } else if (session->dtmf == AST_SIP_DTMF_INBAND) {
303 }
304
305 if (session_media->type == AST_MEDIA_TYPE_AUDIO &&
306 (session->endpoint->media.tos_audio || session->endpoint->media.cos_audio)) {
307 ast_rtp_instance_set_qos(session_media->rtp, session->endpoint->media.tos_audio,
308 session->endpoint->media.cos_audio, "SIP RTP Audio");
309 } else if (session_media->type == AST_MEDIA_TYPE_VIDEO) {
310 ast_rtp_instance_set_prop(session_media->rtp, AST_RTP_PROPERTY_RETRANS_RECV, session->endpoint->media.webrtc);
311 ast_rtp_instance_set_prop(session_media->rtp, AST_RTP_PROPERTY_RETRANS_SEND, session->endpoint->media.webrtc);
312 ast_rtp_instance_set_prop(session_media->rtp, AST_RTP_PROPERTY_REMB, session->endpoint->media.webrtc);
313 if (session->endpoint->media.webrtc) {
316 }
317 if (session->endpoint->media.tos_video || session->endpoint->media.cos_video) {
318 ast_rtp_instance_set_qos(session_media->rtp, session->endpoint->media.tos_video,
319 session->endpoint->media.cos_video, "SIP RTP Video");
320 }
321 }
322
323 ast_rtp_instance_set_last_rx(session_media->rtp, time(NULL));
324
325 return 0;
326}
@ AST_SIP_DTMF_INBAND
Definition res_pjsip.h:552
struct ast_sorcery * ast_sip_get_sorcery(void)
Get a pointer to the SIP sorcery structure.
static struct ast_sockaddr address_rtp
Address for RTP.
static void enable_rtp_extension(struct ast_sip_session *session, struct ast_sip_session_media *session_media, enum ast_rtp_extension extension, enum ast_rtp_extension_direction direction, const pjmedia_sdp_session *sdp)
Enable an RTP extension on an RTP session.
struct ast_rtp_instance * ast_rtp_instance_new(const char *engine_name, struct ast_sched_context *sched, const struct ast_sockaddr *sa, void *data)
Create a new RTP instance.
Definition rtp_engine.c:497
void ast_rtp_instance_set_last_rx(struct ast_rtp_instance *rtp, time_t time)
Set the last RTP reception time.
@ AST_RTP_DTMF_MODE_RFC2833
Definition rtp_engine.h:155
@ AST_RTP_DTMF_MODE_INBAND
Definition rtp_engine.h:157
int ast_rtp_instance_dtmf_mode_set(struct ast_rtp_instance *instance, enum ast_rtp_dtmf_mode dtmf_mode)
Set the DTMF mode that should be used.
#define ast_debug_rtp(sublevel,...)
Log debug level RTP information.
void ast_rtp_instance_set_prop(struct ast_rtp_instance *instance, enum ast_rtp_property property, int value)
Set the value of an RTP instance property.
Definition rtp_engine.c:756
struct ast_rtp_instance * ast_rtp_instance_new_with_options(const char *engine_name, struct ast_sched_context *sched, const struct ast_sockaddr *sa, void *data, const struct ast_rtp_instance_options *options)
Create a new RTP instance with additional options.
Definition rtp_engine.c:505
@ AST_RTP_EXTENSION_TRANSPORT_WIDE_CC
Definition rtp_engine.h:599
@ AST_RTP_EXTENSION_ABS_SEND_TIME
Definition rtp_engine.h:597
@ AST_RTP_PROPERTY_NAT
Definition rtp_engine.h:118
@ AST_RTP_PROPERTY_RETRANS_RECV
Definition rtp_engine.h:130
@ AST_RTP_PROPERTY_RETRANS_SEND
Definition rtp_engine.h:132
@ AST_RTP_PROPERTY_ASYMMETRIC_CODEC
Definition rtp_engine.h:128
@ AST_RTP_PROPERTY_DTMF
Definition rtp_engine.h:120
@ AST_RTP_PROPERTY_REMB
Definition rtp_engine.h:134
int ast_rtp_instance_set_qos(struct ast_rtp_instance *instance, int tos, int cos, const char *desc)
Set QoS parameters on an RTP session.
void * ast_sorcery_retrieve_by_id(const struct ast_sorcery *sorcery, const char *type, const char *id)
Retrieve an object using its unique identifier.
Definition sorcery.c:1917
Options for creating a new RTP instance.
Definition rtp_engine.h:994
struct pjsip_transport * transport
Transport itself.
Definition res_pjsip.h:119
Transport to bind to.
Definition res_pjsip.h:219
static struct test_options options

References address_rtp, ao2_ref, ast_debug_rtp, ast_log, AST_MEDIA_TYPE_AUDIO, AST_MEDIA_TYPE_VIDEO, AST_RTP_DTMF_MODE_INBAND, AST_RTP_DTMF_MODE_RFC2833, AST_RTP_EXTENSION_ABS_SEND_TIME, AST_RTP_EXTENSION_DIRECTION_SENDRECV, AST_RTP_EXTENSION_TRANSPORT_WIDE_CC, ast_rtp_instance_dtmf_mode_set(), ast_rtp_instance_get_ice(), ast_rtp_instance_new(), ast_rtp_instance_new_with_options(), ast_rtp_instance_set_last_rx(), ast_rtp_instance_set_prop(), ast_rtp_instance_set_qos(), AST_RTP_PROPERTY_ASYMMETRIC_CODEC, AST_RTP_PROPERTY_DTMF, AST_RTP_PROPERTY_NAT, AST_RTP_PROPERTY_REMB, AST_RTP_PROPERTY_RETRANS_RECV, AST_RTP_PROPERTY_RETRANS_SEND, AST_SIP_DTMF_AUTO, AST_SIP_DTMF_AUTO_INFO, AST_SIP_DTMF_INBAND, AST_SIP_DTMF_RFC_4733, ast_sip_get_sorcery(), ast_sip_get_transport_state(), ast_sockaddr_parse(), ast_sorcery_object_get_id(), ast_sorcery_retrieve_by_id(), ast_strlen_zero(), enable_rtp_extension(), ast_sip_transport_state::host, LOG_ERROR, NULL, options, ast_sip_session_media::rtp, session, ast_rtp_engine_ice::stop, ast_sip_transport_state::transport, and ast_sip_session_media::type.

Referenced by apply_negotiated_sdp_stream(), create_outgoing_sdp_stream(), and negotiate_incoming_sdp_stream().

◆ enable_rtcp()

static void enable_rtcp ( struct ast_sip_session session,
struct ast_sip_session_media session_media,
const struct pjmedia_sdp_media *  remote_media 
)
static

Enable RTCP on an RTP session.

Definition at line 162 of file res_pjsip_sdp_rtp.c.

164{
165 enum ast_rtp_instance_rtcp rtcp_type;
166
167 if (session->endpoint->media.rtcp_mux && session_media->remote_rtcp_mux) {
168 rtcp_type = AST_RTP_INSTANCE_RTCP_MUX;
169 } else {
171 }
172
173 ast_rtp_instance_set_prop(session_media->rtp, AST_RTP_PROPERTY_RTCP, rtcp_type);
174}
ast_rtp_instance_rtcp
Definition rtp_engine.h:283
@ AST_RTP_INSTANCE_RTCP_MUX
Definition rtp_engine.h:289
@ AST_RTP_INSTANCE_RTCP_STANDARD
Definition rtp_engine.h:287
@ AST_RTP_PROPERTY_RTCP
Definition rtp_engine.h:126

References AST_RTP_INSTANCE_RTCP_MUX, AST_RTP_INSTANCE_RTCP_STANDARD, ast_rtp_instance_set_prop(), AST_RTP_PROPERTY_RTCP, ast_sip_session_media::remote_rtcp_mux, ast_sip_session_media::rtp, and session.

Referenced by apply_negotiated_sdp_stream(), create_outgoing_sdp_stream(), and negotiate_incoming_sdp_stream().

◆ enable_rtp_extension()

static void enable_rtp_extension ( struct ast_sip_session session,
struct ast_sip_session_media session_media,
enum ast_rtp_extension  extension,
enum ast_rtp_extension_direction  direction,
const pjmedia_sdp_session *  sdp 
)
static

Enable an RTP extension on an RTP session.

Definition at line 179 of file res_pjsip_sdp_rtp.c.

182{
183 int id = -1;
184
185 /* For a bundle group the local unique identifier space is shared across all streams within
186 * it.
187 */
188 if (session_media->bundle_group != -1) {
189 int index;
190
191 for (index = 0; index < sdp->media_count; ++index) {
192 struct ast_sip_session_media *other_session_media;
193 int other_id;
194
195 if (index >= AST_VECTOR_SIZE(&session->pending_media_state->sessions)) {
196 break;
197 }
198
199 other_session_media = AST_VECTOR_GET(&session->pending_media_state->sessions, index);
200 if (!other_session_media->rtp || other_session_media->bundle_group != session_media->bundle_group) {
201 continue;
202 }
203
204 other_id = ast_rtp_instance_extmap_get_id(other_session_media->rtp, extension);
205 if (other_id == -1) {
206 /* Worst case we have to fall back to the highest available free local unique identifier
207 * for the bundle group.
208 */
209 other_id = ast_rtp_instance_extmap_count(other_session_media->rtp) + 1;
210 if (id < other_id) {
211 id = other_id;
212 }
213 continue;
214 }
215
216 id = other_id;
217 break;
218 }
219 }
220
222}
direction
int ast_rtp_instance_extmap_enable(struct ast_rtp_instance *instance, int id, enum ast_rtp_extension extension, enum ast_rtp_extension_direction direction)
Enable support for an RTP extension on an instance.
Definition rtp_engine.c:783
int ast_rtp_instance_extmap_get_id(struct ast_rtp_instance *instance, enum ast_rtp_extension extension)
Retrieve the id for an RTP extension.
Definition rtp_engine.c:937

References ast_rtp_instance_extmap_count(), ast_rtp_instance_extmap_enable(), ast_rtp_instance_extmap_get_id(), AST_VECTOR_GET, AST_VECTOR_SIZE, ast_sip_session_media::bundle_group, ast_sip_session_media::rtp, and session.

Referenced by create_rtp().

◆ generate_fmtp_attr()

static pjmedia_sdp_attr * generate_fmtp_attr ( pj_pool_t *  pool,
struct ast_format format,
int  rtp_code 
)
static

Definition at line 723 of file res_pjsip_sdp_rtp.c.

724{
725 struct ast_str *fmtp0 = ast_str_alloca(256);
726 pj_str_t fmtp1;
727 pjmedia_sdp_attr *attr = NULL;
728 char *tmp;
729
730 ast_format_generate_sdp_fmtp(format, rtp_code, &fmtp0);
731 if (ast_str_strlen(fmtp0)) {
732 tmp = ast_str_buffer(fmtp0) + ast_str_strlen(fmtp0) - 1;
733 /* remove any carriage return line feeds */
734 while (*tmp == '\r' || *tmp == '\n') --tmp;
735 *++tmp = '\0';
736 /* ast...generate gives us everything, just need value */
737 tmp = strchr(ast_str_buffer(fmtp0), ':');
738 if (tmp && tmp[1] != '\0') {
739 fmtp1 = pj_str(tmp + 1);
740 } else {
741 fmtp1 = pj_str(ast_str_buffer(fmtp0));
742 }
743 attr = pjmedia_sdp_attr_create(pool, "fmtp", &fmtp1);
744 }
745 return attr;
746}
void ast_format_generate_sdp_fmtp(const struct ast_format *format, unsigned int payload, struct ast_str **str)
This function is used to produce an fmtp SDP line for an Asterisk format. The attributes present on t...
Definition format.c:305
size_t attribute_pure ast_str_strlen(const struct ast_str *buf)
Returns the current length of the string stored within buf.
Definition strings.h:730
#define ast_str_alloca(init_len)
Definition strings.h:848

References ast_format_generate_sdp_fmtp(), ast_str_alloca, ast_str_buffer(), ast_str_strlen(), and NULL.

Referenced by create_outgoing_sdp_stream().

◆ generate_rtpmap_attr()

static pjmedia_sdp_attr * generate_rtpmap_attr ( struct ast_sip_session session,
pjmedia_sdp_media *  media,
pj_pool_t *  pool,
int  rtp_code,
int  asterisk_format,
struct ast_format format,
int  code 
)
static

Definition at line 652 of file res_pjsip_sdp_rtp.c.

654{
655#ifndef HAVE_PJSIP_ENDPOINT_COMPACT_FORM
656 extern pj_bool_t pjsip_use_compact_form;
657#else
658 pj_bool_t pjsip_use_compact_form = pjsip_cfg()->endpt.use_compact_form;
659#endif
660 pjmedia_sdp_rtpmap rtpmap;
661 pjmedia_sdp_attr *attr = NULL;
662 char tmp[64];
663 enum ast_rtp_options options = session->endpoint->media.g726_non_standard ?
665
666 snprintf(tmp, sizeof(tmp), "%d", rtp_code);
667 pj_strdup2(pool, &media->desc.fmt[media->desc.fmt_count++], tmp);
668
669 if (rtp_code <= AST_RTP_PT_LAST_STATIC && pjsip_use_compact_form) {
670 return NULL;
671 }
672
673 rtpmap.pt = media->desc.fmt[media->desc.fmt_count - 1];
674 rtpmap.clock_rate = ast_rtp_lookup_sample_rate2(asterisk_format, format, code);
675 pj_strdup2(pool, &rtpmap.enc_name, ast_rtp_lookup_mime_subtype2(asterisk_format, format, code, options));
676 if (!pj_stricmp2(&rtpmap.enc_name, "opus")) {
677 pj_cstr(&rtpmap.param, "2");
678 } else {
679 pj_cstr(&rtpmap.param, NULL);
680 }
681
682 pjmedia_sdp_rtpmap_to_attr(pool, &rtpmap, &attr);
683
684 return attr;
685}
ast_rtp_options
Definition rtp_engine.h:145
@ AST_RTP_OPT_G726_NONSTANDARD
Definition rtp_engine.h:147
#define AST_RTP_PT_LAST_STATIC
Definition rtp_engine.h:89
const char * ast_rtp_lookup_mime_subtype2(const int asterisk_format, const struct ast_format *format, int code, enum ast_rtp_options options)
Retrieve mime subtype information on a payload.

References ast_rtp_lookup_mime_subtype2(), ast_rtp_lookup_sample_rate2(), AST_RTP_OPT_G726_NONSTANDARD, AST_RTP_PT_LAST_STATIC, NULL, options, and session.

Referenced by create_outgoing_sdp_stream().

◆ generate_rtpmap_attr2()

static pjmedia_sdp_attr * generate_rtpmap_attr2 ( struct ast_sip_session session,
pjmedia_sdp_media *  media,
pj_pool_t *  pool,
int  rtp_code,
int  asterisk_format,
struct ast_format format,
int  code,
int  sample_rate 
)
static

Definition at line 688 of file res_pjsip_sdp_rtp.c.

690{
691#ifndef HAVE_PJSIP_ENDPOINT_COMPACT_FORM
692 extern pj_bool_t pjsip_use_compact_form;
693#else
694 pj_bool_t pjsip_use_compact_form = pjsip_cfg()->endpt.use_compact_form;
695#endif
696 pjmedia_sdp_rtpmap rtpmap;
697 pjmedia_sdp_attr *attr = NULL;
698 char tmp[64];
699 enum ast_rtp_options options = session->endpoint->media.g726_non_standard ?
701
702 snprintf(tmp, sizeof(tmp), "%d", rtp_code);
703 pj_strdup2(pool, &media->desc.fmt[media->desc.fmt_count++], tmp);
704
705 if (rtp_code <= AST_RTP_PT_LAST_STATIC && pjsip_use_compact_form) {
706 return NULL;
707 }
708
709 rtpmap.pt = media->desc.fmt[media->desc.fmt_count - 1];
710 rtpmap.clock_rate = sample_rate;
711 pj_strdup2(pool, &rtpmap.enc_name, ast_rtp_lookup_mime_subtype2(asterisk_format, format, code, options));
712 if (!pj_stricmp2(&rtpmap.enc_name, "opus")) {
713 pj_cstr(&rtpmap.param, "2");
714 } else {
715 pj_cstr(&rtpmap.param, NULL);
716 }
717
718 pjmedia_sdp_rtpmap_to_attr(pool, &rtpmap, &attr);
719
720 return attr;
721}

References ast_rtp_lookup_mime_subtype2(), AST_RTP_OPT_G726_NONSTANDARD, AST_RTP_PT_LAST_STATIC, NULL, options, and session.

Referenced by create_outgoing_sdp_stream().

◆ get_codecs()

static void get_codecs ( struct ast_sip_session session,
const struct pjmedia_sdp_media *  stream,
struct ast_rtp_codecs codecs,
struct ast_sip_session_media session_media,
struct ast_format_cap astformats 
)
static

Definition at line 328 of file res_pjsip_sdp_rtp.c.

330{
331 pjmedia_sdp_attr *attr;
332 pjmedia_sdp_rtpmap *rtpmap;
333 pjmedia_sdp_fmtp fmtp;
334 struct ast_format *format;
335 int i, num = 0, tel_event = 0;
336 char name[256];
337 char media[20];
338 char fmt_param[256];
339 enum ast_rtp_options options = session->endpoint->media.g726_non_standard ?
342
344
346
347 /* Iterate through provided formats */
348 for (i = 0; i < stream->desc.fmt_count; ++i) {
349 /* The payload is kept as a string for things like t38 but for video it is always numerical */
350 ast_rtp_codecs_payloads_set_m_type(codecs, NULL, pj_strtoul(&stream->desc.fmt[i]));
351 /* Look for the optional rtpmap attribute */
352 if (!(attr = pjmedia_sdp_media_find_attr2(stream, "rtpmap", &stream->desc.fmt[i]))) {
353 continue;
354 }
355
356 /* Interpret the attribute as an rtpmap */
357 if ((pjmedia_sdp_attr_to_rtpmap(session->inv_session->pool_prov, attr, &rtpmap)) != PJ_SUCCESS) {
358 continue;
359 }
360
361 ast_copy_pj_str(name, &rtpmap->enc_name, sizeof(name));
362 if (strcmp(name, "telephone-event") == 0) {
363 if (tel_event == 0) {
364 int dtmf_rate = 0, dtmf_code = 0;
365 char dtmf_pt[8];
366 ast_copy_pj_str(dtmf_pt, &rtpmap->pt, sizeof(dtmf_pt));
367 dtmf_code = atoi(dtmf_pt);
368 dtmf_rate = rtpmap->clock_rate;
370 }
371 tel_event++;
372 }
373
374 ast_copy_pj_str(media, (pj_str_t*)&stream->desc.media, sizeof(media));
376 pj_strtoul(&stream->desc.fmt[i]), media, name, options, rtpmap->clock_rate);
377 /* Look for an optional associated fmtp attribute */
378 if (!(attr = pjmedia_sdp_media_find_attr2(stream, "fmtp", &rtpmap->pt))) {
379 continue;
380 }
381
382 if ((pjmedia_sdp_attr_get_fmtp(attr, &fmtp)) == PJ_SUCCESS) {
383 ast_copy_pj_str(fmt_param, &fmtp.fmt, sizeof(fmt_param));
384 if (sscanf(fmt_param, "%30d", &num) != 1) {
385 continue;
386 }
387
388 if ((format = ast_rtp_codecs_get_payload_format(codecs, num))) {
389 struct ast_format *format_parsed;
390
391 ast_copy_pj_str(fmt_param, &fmtp.fmt_param, sizeof(fmt_param));
392
393 format_parsed = ast_format_parse_sdp_fmtp(format, fmt_param);
394 if (format_parsed) {
396 ao2_ref(format_parsed, -1);
397 }
398 ao2_ref(format, -1);
399 }
400 }
401 }
402
403 /* Parsing done, now fill the ast_format_cap struct in the correct order */
404 for (i = 0; i < stream->desc.fmt_count; ++i) {
405 if ((format = ast_rtp_codecs_get_payload_format(codecs, pj_strtoul(&stream->desc.fmt[i])))) {
406 ast_format_cap_append(astformats, format, 0);
407 ao2_ref(format, -1);
408 }
409 }
410
411 if (session->dtmf == AST_SIP_DTMF_AUTO) {
412 if (tel_event) {
415 } else {
418 }
419 }
420
421 if (session->dtmf == AST_SIP_DTMF_AUTO_INFO) {
422 if (tel_event) {
425 } else {
428 }
429 }
430
431
432 /* Get the packetization, if it exists */
433 if ((attr = pjmedia_sdp_media_find_attr2(stream, "ptime", NULL))) {
434 unsigned long framing = pj_strtoul(pj_strltrim(&attr->value));
435 if (framing && session->endpoint->media.rtp.use_ptime) {
437 ast_format_cap_set_framing(astformats, framing);
438 }
439 }
440
442}
static struct ao2_container * codecs
Registered codecs.
Definition codec.c:48
@ AST_MEDIA_TYPE_UNKNOWN
Definition codec.h:31
struct ast_format * ast_format_parse_sdp_fmtp(const struct ast_format *format, const char *attributes)
This function is used to have a media format aware module parse and interpret SDP attribute informati...
Definition format.c:286
void ast_format_cap_remove_by_type(struct ast_format_cap *cap, enum ast_media_type type)
Remove all formats matching a specific format type.
Definition format_cap.c:523
#define ast_format_cap_append(cap, format, framing)
Add format capability to capabilities structure.
Definition format_cap.h:99
void ast_format_cap_set_framing(struct ast_format_cap *cap, unsigned int framing)
Set the global framing.
Definition format_cap.c:136
static const char name[]
Definition format_mp3.c:68
#define SCOPE_EXIT_RTN(...)
struct ast_format * ast_rtp_codecs_get_payload_format(struct ast_rtp_codecs *codecs, int payload)
Retrieve the actual ast_format stored on the codecs structure for a specific tx payload type.
@ AST_RTP_DTMF_MODE_NONE
Definition rtp_engine.h:153
void ast_rtp_codecs_set_framing(struct ast_rtp_codecs *codecs, unsigned int framing)
Set the framing used for a set of codecs.
void ast_rtp_codecs_payloads_set_m_type(struct ast_rtp_codecs *codecs, struct ast_rtp_instance *instance, int payload)
Record tx payload type information that was seen in an m= SDP line.
int ast_rtp_codecs_payloads_initialize(struct ast_rtp_codecs *codecs)
Initialize an RTP codecs structure.
int ast_rtp_codecs_payload_replace_format(struct ast_rtp_codecs *codecs, int payload, struct ast_format *format)
Update the format associated with a tx payload type in a codecs structure.
int ast_rtp_codecs_payloads_set_rtpmap_type_rate(struct ast_rtp_codecs *codecs, struct ast_rtp_instance *instance, int pt, char *mimetype, char *mimesubtype, enum ast_rtp_options options, unsigned int sample_rate)
Set tx payload type to a known MIME media type for a codec with a specific sample rate.
int ast_rtp_codecs_set_preferred_dtmf_format(struct ast_rtp_codecs *codecs, int pt, int rate)
Set the preferred dtmf format pt and sample rate.

References ao2_ref, ast_copy_pj_str(), ast_format_cap_append, ast_format_cap_remove_by_type(), ast_format_cap_set_framing(), ast_format_parse_sdp_fmtp(), AST_MEDIA_TYPE_UNKNOWN, ast_rtp_codecs_get_payload_format(), ast_rtp_codecs_payload_replace_format(), ast_rtp_codecs_payloads_initialize(), ast_rtp_codecs_payloads_set_m_type(), ast_rtp_codecs_payloads_set_rtpmap_type_rate(), ast_rtp_codecs_set_framing(), ast_rtp_codecs_set_preferred_dtmf_format(), AST_RTP_DTMF_MODE_INBAND, AST_RTP_DTMF_MODE_NONE, AST_RTP_DTMF_MODE_RFC2833, ast_rtp_instance_dtmf_mode_set(), ast_rtp_instance_set_prop(), AST_RTP_OPT_G726_NONSTANDARD, AST_RTP_PROPERTY_DTMF, AST_SIP_DTMF_AUTO, AST_SIP_DTMF_AUTO_INFO, ast_sip_session_get_name(), codecs, name, NULL, options, ast_sip_session_media::rtp, SCOPE_ENTER, SCOPE_EXIT_RTN, and session.

Referenced by set_caps(), and set_incoming_call_offer_cap().

◆ get_media_encryption_type()

static enum ast_sip_session_media_encryption get_media_encryption_type ( pj_str_t  transport,
const struct pjmedia_sdp_media *  stream,
unsigned int *  optimistic 
)
static

figure out media transport encryption type from the media transport string

Definition at line 1012 of file res_pjsip_sdp_rtp.c.

1014{
1015 RAII_VAR(char *, transport_str, ast_strndup(transport.ptr, transport.slen), ast_free);
1016
1017 *optimistic = 0;
1018
1019 if (!transport_str) {
1021 }
1022 if (strstr(transport_str, "UDP/TLS")) {
1024 } else if (strstr(transport_str, "SAVP")) {
1026 } else if (media_stream_has_crypto(stream)) {
1027 *optimistic = 1;
1029 } else {
1031 }
1032}
#define ast_strndup(str, len)
A wrapper for strndup()
Definition astmm.h:256
static int media_stream_has_crypto(const struct pjmedia_sdp_media *stream)
figure out if media stream has crypto lines for sdes

References ast_free, AST_SIP_MEDIA_ENCRYPT_DTLS, AST_SIP_MEDIA_ENCRYPT_NONE, AST_SIP_MEDIA_ENCRYPT_SDES, AST_SIP_MEDIA_TRANSPORT_INVALID, ast_strndup, media_stream_has_crypto(), RAII_VAR, and ast_rtp_engine_ice_candidate::transport.

Referenced by check_endpoint_media_transport().

◆ load_module()

static int load_module ( void  )
static

Load the module.

Module loading including tests for configuration or dependencies. This function can return AST_MODULE_LOAD_FAILURE, AST_MODULE_LOAD_DECLINE, or AST_MODULE_LOAD_SUCCESS. If a dependency or environment variable fails tests return AST_MODULE_LOAD_FAILURE. If the module can not load the configuration file or other non-critical problem return AST_MODULE_LOAD_DECLINE. On success return AST_MODULE_LOAD_SUCCESS.

Definition at line 2522 of file res_pjsip_sdp_rtp.c.

2523{
2524 if (ast_check_ipv6()) {
2526 } else {
2527 ast_sockaddr_parse(&address_rtp, "0.0.0.0", 0);
2528 }
2529
2530 if (!(sched = ast_sched_context_create())) {
2531 ast_log(LOG_ERROR, "Unable to create scheduler context.\n");
2532 goto end;
2533 }
2534
2536 ast_log(LOG_ERROR, "Unable to create scheduler context thread.\n");
2537 goto end;
2538 }
2539
2541 ast_log(LOG_ERROR, "Unable to register SDP handler for %s stream type\n", STR_AUDIO);
2542 goto end;
2543 }
2544
2546 ast_log(LOG_ERROR, "Unable to register SDP handler for %s stream type\n", STR_VIDEO);
2547 goto end;
2548 }
2549
2551
2553end:
2554 unload_module();
2555
2557}
char * end
Definition eagi_proxy.c:73
@ AST_MODULE_LOAD_SUCCESS
Definition module.h:70
@ AST_MODULE_LOAD_DECLINE
Module has failed to load, may be in an inconsistent state.
Definition module.h:78
static const char STR_AUDIO[]
static struct ast_sip_session_sdp_handler audio_sdp_handler
SDP handler for 'audio' media stream.
static const char STR_VIDEO[]
static struct ast_sip_session_sdp_handler video_sdp_handler
SDP handler for 'video' media stream.
static int unload_module(void)
Unloads the sdp RTP/AVP module from Asterisk.
static struct ast_sip_session_supplement video_info_supplement
int ast_sip_session_register_sdp_handler(struct ast_sip_session_sdp_handler *handler, const char *stream_type)
Register an SDP handler.
#define ast_sip_session_register_supplement(supplement)
int ast_sched_start_thread(struct ast_sched_context *con)
Start a thread for processing scheduler entries.
Definition sched.c:197
struct ast_sched_context * ast_sched_context_create(void)
Create a scheduler context.
Definition sched.c:238
int ast_check_ipv6(void)
Test that an OS supports IPv6 Networking.
Definition utils.c:2826

References address_rtp, ast_check_ipv6(), ast_log, AST_MODULE_LOAD_DECLINE, AST_MODULE_LOAD_SUCCESS, ast_sched_context_create(), ast_sched_start_thread(), ast_sip_session_register_sdp_handler(), ast_sip_session_register_supplement, ast_sockaddr_parse(), audio_sdp_handler, end, LOG_ERROR, STR_AUDIO, STR_VIDEO, unload_module(), video_info_supplement, and video_sdp_handler.

◆ media_session_rtcp_read_callback()

static struct ast_frame * media_session_rtcp_read_callback ( struct ast_sip_session session,
struct ast_sip_session_media session_media 
)
static

Definition at line 2205 of file res_pjsip_sdp_rtp.c.

2206{
2207 struct ast_frame *f;
2208
2209 if (!session_media->rtp) {
2210 return &ast_null_frame;
2211 }
2212
2213 f = ast_rtp_instance_read(session_media->rtp, 1);
2214 if (!f) {
2215 return NULL;
2216 }
2217
2218 ast_rtp_instance_set_last_rx(session_media->rtp, time(NULL));
2219
2220 return f;
2221}
struct ast_frame * ast_rtp_instance_read(struct ast_rtp_instance *instance, int rtcp)
Receive a frame over RTP.
Definition rtp_engine.c:629
Data structure associated with a single frame of data.

References ast_null_frame, ast_rtp_instance_read(), ast_rtp_instance_set_last_rx(), NULL, and ast_sip_session_media::rtp.

Referenced by apply_negotiated_sdp_stream().

◆ media_session_rtp_read_callback()

static struct ast_frame * media_session_rtp_read_callback ( struct ast_sip_session session,
struct ast_sip_session_media session_media 
)
static

Definition at line 2187 of file res_pjsip_sdp_rtp.c.

2188{
2189 struct ast_frame *f;
2190
2191 if (!session_media->rtp) {
2192 return &ast_null_frame;
2193 }
2194
2195 f = ast_rtp_instance_read(session_media->rtp, 0);
2196 if (!f) {
2197 return NULL;
2198 }
2199
2200 ast_rtp_instance_set_last_rx(session_media->rtp, time(NULL));
2201
2202 return f;
2203}

References ast_null_frame, ast_rtp_instance_read(), ast_rtp_instance_set_last_rx(), NULL, and ast_sip_session_media::rtp.

Referenced by apply_negotiated_sdp_stream().

◆ media_session_rtp_write_callback()

static int media_session_rtp_write_callback ( struct ast_sip_session session,
struct ast_sip_session_media session_media,
struct ast_frame frame 
)
static

Definition at line 2223 of file res_pjsip_sdp_rtp.c.

2224{
2225 if (!session_media->rtp) {
2226 return 0;
2227 }
2228
2229 return ast_rtp_instance_write(session_media->rtp, frame);
2230}
int ast_rtp_instance_write(struct ast_rtp_instance *instance, struct ast_frame *frame)
Send a frame out over RTP.
Definition rtp_engine.c:619

References ast_rtp_instance_write(), and ast_sip_session_media::rtp.

Referenced by apply_negotiated_sdp_stream().

◆ media_stream_has_crypto()

static int media_stream_has_crypto ( const struct pjmedia_sdp_media *  stream)
static

figure out if media stream has crypto lines for sdes

Definition at line 992 of file res_pjsip_sdp_rtp.c.

993{
994 int i;
995
996 for (i = 0; i < stream->attr_count; i++) {
997 pjmedia_sdp_attr *attr;
998
999 /* check the stream for the required crypto attribute */
1000 attr = stream->attr[i];
1001 if (pj_strcmp2(&attr->name, "crypto")) {
1002 continue;
1003 }
1004
1005 return 1;
1006 }
1007
1008 return 0;
1009}

Referenced by get_media_encryption_type().

◆ negotiate_incoming_sdp_stream()

static int negotiate_incoming_sdp_stream ( struct ast_sip_session session,
struct ast_sip_session_media session_media,
const pjmedia_sdp_session *  sdp,
int  index,
struct ast_stream asterisk_stream 
)
static

Function which negotiates an incoming media stream.

Definition at line 1559 of file res_pjsip_sdp_rtp.c.

1562{
1563 char host[NI_MAXHOST];
1564 RAII_VAR(struct ast_sockaddr *, addrs, NULL, ast_free);
1565 pjmedia_sdp_media *stream = sdp->media[index];
1566 struct ast_sip_session_media *session_media_transport;
1567 enum ast_media_type media_type = session_media->type;
1569 struct ast_format_cap *joint;
1570 int res;
1572
1573 /* If no type formats have been configured reject this stream */
1574 if (!ast_format_cap_has_type(session->endpoint->media.codecs, media_type)) {
1575 ast_debug(3, "Endpoint has no codecs for media type '%s', declining stream\n",
1576 ast_codec_media_type2str(session_media->type));
1577 SCOPE_EXIT_RTN_VALUE(0, "Endpoint has no codecs\n");
1578 }
1579
1580 /* Ensure incoming transport is compatible with the endpoint's configuration */
1581 if (!session->endpoint->media.rtp.use_received_transport) {
1582 encryption = check_endpoint_media_transport(session->endpoint, stream);
1583
1584 if (encryption == AST_SIP_MEDIA_TRANSPORT_INVALID) {
1585 SCOPE_EXIT_RTN_VALUE(-1, "Incompatible transport\n");
1586 }
1587 }
1588
1589 ast_copy_pj_str(host, stream->conn ? &stream->conn->addr : &sdp->conn->addr, sizeof(host));
1590
1591 /* Ensure that the address provided is valid */
1592 if (ast_sockaddr_resolve(&addrs, host, PARSE_PORT_FORBID, AST_AF_UNSPEC) <= 0) {
1593 /* The provided host was actually invalid so we error out this negotiation */
1594 SCOPE_EXIT_RTN_VALUE(-1, "Invalid host\n");
1595 }
1596
1597 /* Using the connection information create an appropriate RTP instance */
1598 if (!session_media->rtp && create_rtp(session, session_media, sdp)) {
1599 SCOPE_EXIT_RTN_VALUE(-1, "Couldn't create rtp\n");
1600 }
1601
1602 process_ssrc_attributes(session, session_media, stream);
1603 process_extmap_attributes(session, session_media, stream);
1604 session_media_transport = ast_sip_session_media_get_transport(session, session_media);
1605
1606 if (session_media_transport == session_media || !session_media->bundled) {
1607 /* If this media session is carrying actual traffic then set up those aspects */
1608 session_media->remote_rtcp_mux = (pjmedia_sdp_media_find_attr2(stream, "rtcp-mux", NULL) != NULL);
1609 set_ice_components(session, session_media);
1610
1611 enable_rtcp(session, session_media, stream);
1612
1613 res = setup_media_encryption(session, session_media, sdp, stream);
1614 if (res) {
1615 if (!session->endpoint->media.rtp.encryption_optimistic ||
1616 !pj_strncmp2(&stream->desc.transport, "RTP/SAVP", 8)) {
1617 /* If optimistic encryption is disabled and crypto should have been enabled
1618 * but was not this session must fail. This must also fail if crypto was
1619 * required in the offer but could not be set up.
1620 */
1621 SCOPE_EXIT_RTN_VALUE(-1, "Incompatible crypto\n");
1622 }
1623 /* There is no encryption, sad. */
1624 session_media->encryption = AST_SIP_MEDIA_ENCRYPT_NONE;
1625 }
1626
1627 /* If we've been explicitly configured to use the received transport OR if
1628 * encryption is on and crypto is present use the received transport.
1629 * This is done in case of optimistic because it may come in as RTP/AVP or RTP/SAVP depending
1630 * on the configuration of the remote endpoint (optimistic themselves or mandatory).
1631 */
1632 if ((session->endpoint->media.rtp.use_received_transport) ||
1633 ((encryption == AST_SIP_MEDIA_ENCRYPT_SDES) && !res)) {
1634 pj_strdup(session->inv_session->pool, &session_media->transport, &stream->desc.transport);
1635 }
1636 } else {
1637 /* This is bundled with another session, so mark it as such */
1638 ast_rtp_instance_bundle(session_media->rtp, session_media_transport->rtp);
1639
1640 enable_rtcp(session, session_media, stream);
1641 }
1642
1643 /* If ICE support is enabled find all the needed attributes */
1644 check_ice_support(session, session_media, stream);
1645
1646 /* If ICE support is enabled then check remote ICE started? */
1647 if (session_media->remote_ice) {
1648 process_ice_auth_attrb(session, session_media, sdp, stream);
1649 }
1650
1651 /* Check if incoming SDP is changing the remotely held state */
1652 set_session_media_remotely_held(session_media, session, stream, asterisk_stream, addrs);
1653
1654 joint = set_incoming_call_offer_cap(session, session_media, stream);
1655 res = apply_cap_to_bundled(session_media, session_media_transport, asterisk_stream, joint);
1656 ao2_cleanup(joint);
1657 if (res != 0) {
1658 SCOPE_EXIT_RTN_VALUE(0, "Something failed\n");
1659 }
1660
1662}
int ast_format_cap_has_type(const struct ast_format_cap *cap, enum ast_media_type type)
Find out if the capabilities structure has any formats of a specific type.
Definition format_cap.c:613
static void check_ice_support(struct ast_sip_session *session, struct ast_sip_session_media *session_media, const struct pjmedia_sdp_media *remote_stream)
Function which checks for ice attributes in an audio stream.
static int apply_cap_to_bundled(struct ast_sip_session_media *session_media, struct ast_sip_session_media *session_media_transport, struct ast_stream *asterisk_stream, struct ast_format_cap *joint)
static struct ast_format_cap * set_incoming_call_offer_cap(struct ast_sip_session *session, struct ast_sip_session_media *session_media, const struct pjmedia_sdp_media *stream)
static void process_ice_auth_attrb(struct ast_sip_session *session, struct ast_sip_session_media *session_media, const struct pjmedia_sdp_session *remote, const struct pjmedia_sdp_media *remote_stream)

References ao2_cleanup, apply_cap_to_bundled(), AST_AF_UNSPEC, ast_codec_media_type2str(), ast_copy_pj_str(), ast_debug, ast_format_cap_has_type(), ast_free, ast_rtp_instance_bundle(), AST_SIP_MEDIA_ENCRYPT_NONE, AST_SIP_MEDIA_ENCRYPT_SDES, AST_SIP_MEDIA_TRANSPORT_INVALID, ast_sip_session_get_name(), ast_sip_session_media_get_transport(), ast_sockaddr_resolve(), ast_sip_session_media::bundled, check_endpoint_media_transport(), check_ice_support(), create_rtp(), enable_rtcp(), ast_sip_session_media::encryption, NULL, PARSE_PORT_FORBID, process_extmap_attributes(), process_ice_auth_attrb(), process_ssrc_attributes(), RAII_VAR, ast_sip_session_media::remote_ice, ast_sip_session_media::remote_rtcp_mux, ast_sip_session_media::rtp, SCOPE_ENTER, SCOPE_EXIT_RTN_VALUE, session, set_ice_components(), set_incoming_call_offer_cap(), set_session_media_remotely_held(), setup_media_encryption(), ast_sip_session_media::transport, and ast_sip_session_media::type.

◆ parse_dtls_attrib()

static int parse_dtls_attrib ( struct ast_sip_session_media session_media,
const struct pjmedia_sdp_session *  sdp,
const struct pjmedia_sdp_media *  stream 
)
static

Definition at line 1173 of file res_pjsip_sdp_rtp.c.

1176{
1177 int i;
1178
1179 for (i = 0; i < sdp->attr_count; i++) {
1180 apply_dtls_attrib(session_media, sdp->attr[i]);
1181 }
1182
1183 for (i = 0; i < stream->attr_count; i++) {
1184 apply_dtls_attrib(session_media, stream->attr[i]);
1185 }
1186
1188
1189 return 0;
1190}
static void apply_dtls_attrib(struct ast_sip_session_media *session_media, pjmedia_sdp_attr *attr)
#define AST_SRTP_CRYPTO_OFFER_OK
Definition sdp_srtp.h:45
#define ast_set_flag(p, flag)
Definition utils.h:71

References apply_dtls_attrib(), ast_set_flag, AST_SRTP_CRYPTO_OFFER_OK, and ast_sip_session_media::srtp.

Referenced by setup_media_encryption().

◆ process_extmap_attributes()

static void process_extmap_attributes ( struct ast_sip_session session,
struct ast_sip_session_media session_media,
const struct pjmedia_sdp_media *  remote_stream 
)
static

Function which processes extmap attributes in a stream.

Definition at line 1478 of file res_pjsip_sdp_rtp.c.

1480{
1481 int index;
1482
1483 if (!session->endpoint->media.webrtc || session_media->type != AST_MEDIA_TYPE_VIDEO) {
1484 return;
1485 }
1486
1487 ast_rtp_instance_extmap_clear(session_media->rtp);
1488
1489 for (index = 0; index < remote_stream->attr_count; ++index) {
1490 pjmedia_sdp_attr *attr = remote_stream->attr[index];
1491 char attr_value[pj_strlen(&attr->value) + 1];
1492 char *uri;
1493 int id;
1494 char direction_str[10] = "";
1495 char *attributes;
1497
1498 /* We only care about extmap attributes */
1499 if (pj_strcmp2(&attr->name, "extmap")) {
1500 continue;
1501 }
1502
1503 ast_copy_pj_str(attr_value, &attr->value, sizeof(attr_value));
1504
1505 /* Split the combined unique identifier and direction away from the URI and attributes for easier parsing */
1506 uri = strchr(attr_value, ' ');
1507 if (ast_strlen_zero(uri)) {
1508 continue;
1509 }
1510 *uri++ = '\0';
1511
1512 if ((sscanf(attr_value, "%30d%9s", &id, direction_str) < 1) || (id < 1)) {
1513 /* We require at a minimum the unique identifier */
1514 continue;
1515 }
1516
1517 /* Convert from the string to the internal representation */
1518 if (!strcasecmp(direction_str, "/sendonly")) {
1520 } else if (!strcasecmp(direction_str, "/recvonly")) {
1522 } else if (!strcasecmp(direction_str, "/inactive")) {
1524 }
1525
1526 attributes = strchr(uri, ' ');
1527 if (!ast_strlen_zero(attributes)) {
1528 *attributes++ = '\0';
1529 }
1530
1531 ast_rtp_instance_extmap_negotiate(session_media->rtp, id, direction, uri, attributes);
1532 }
1533}
enum queue_result id
Definition app_queue.c:1790
ast_rtp_extension_direction
Directions for RTP extensions.
Definition rtp_engine.h:827
int ast_rtp_instance_extmap_negotiate(struct ast_rtp_instance *instance, int id, enum ast_rtp_extension_direction direction, const char *uri, const char *attributes)
Negotiate received RTP extension information.
Definition rtp_engine.c:869
void ast_rtp_instance_extmap_clear(struct ast_rtp_instance *instance)
Clear negotiated RTP extension information.
Definition rtp_engine.c:913

References ast_copy_pj_str(), AST_MEDIA_TYPE_VIDEO, AST_RTP_EXTENSION_DIRECTION_INACTIVE, AST_RTP_EXTENSION_DIRECTION_RECVONLY, AST_RTP_EXTENSION_DIRECTION_SENDONLY, AST_RTP_EXTENSION_DIRECTION_SENDRECV, ast_rtp_instance_extmap_clear(), ast_rtp_instance_extmap_negotiate(), ast_strlen_zero(), id, ast_sip_session_media::rtp, session, and ast_sip_session_media::type.

Referenced by apply_negotiated_sdp_stream(), and negotiate_incoming_sdp_stream().

◆ process_ice_attributes()

static void process_ice_attributes ( struct ast_sip_session session,
struct ast_sip_session_media session_media,
const struct pjmedia_sdp_session *  remote,
const struct pjmedia_sdp_media *  remote_stream 
)
static

Function which processes ICE attributes in an audio stream.

Definition at line 888 of file res_pjsip_sdp_rtp.c.

890{
891 struct ast_rtp_engine_ice *ice;
892 const pjmedia_sdp_attr *attr;
893 char attr_value[256];
894 unsigned int attr_i;
895
896 /* If ICE support is not enabled or available exit early */
897 if (!session->endpoint->media.rtp.ice_support || !(ice = ast_rtp_instance_get_ice(session_media->rtp))) {
898 return;
899 }
900
901 ast_debug_ice(2, "(%p) ICE process attributes\n", session_media->rtp);
902
903 attr = pjmedia_sdp_media_find_attr2(remote_stream, "ice-ufrag", NULL);
904 if (!attr) {
905 attr = pjmedia_sdp_attr_find2(remote->attr_count, remote->attr, "ice-ufrag", NULL);
906 }
907 if (attr) {
908 ast_copy_pj_str(attr_value, (pj_str_t*)&attr->value, sizeof(attr_value));
910 } else {
911 ast_debug_ice(2, "(%p) ICE no, or invalid ice-ufrag\n", session_media->rtp);
912 return;
913 }
914
916 if (!attr) {
917 attr = pjmedia_sdp_attr_find2(remote->attr_count, remote->attr, "ice-pwd", NULL);
918 }
919 if (attr) {
920 ast_copy_pj_str(attr_value, (pj_str_t*)&attr->value, sizeof(attr_value));
922 } else {
923 ast_debug_ice(2, "(%p) ICE no, or invalid ice-pwd\n", session_media->rtp);
924 return;
925 }
926
928 ice->ice_lite(session_media->rtp);
929 }
930
931 /* Find all of the candidates */
932 for (attr_i = 0; attr_i < remote_stream->attr_count; ++attr_i) {
933 char foundation[33], transport[32], address[PJ_INET6_ADDRSTRLEN + 1], cand_type[6], relay_address[PJ_INET6_ADDRSTRLEN + 1] = "";
934 unsigned int port, relay_port = 0;
935 struct ast_rtp_engine_ice_candidate candidate = { 0, };
936
937 attr = remote_stream->attr[attr_i];
938
939 /* If this is not a candidate line skip it */
940 if (pj_strcmp2(&attr->name, "candidate")) {
941 continue;
942 }
943
944 ast_copy_pj_str(attr_value, (pj_str_t*)&attr->value, sizeof(attr_value));
945
946 if (sscanf(attr_value, "%32s %30u %31s %30u %46s %30u typ %5s %*s %23s %*s %30u", foundation, &candidate.id, transport,
947 (unsigned *)&candidate.priority, address, &port, cand_type, relay_address, &relay_port) < 7) {
948 /* Candidate did not parse properly */
949 continue;
950 }
951
952 if (session->endpoint->media.rtcp_mux && session_media->remote_rtcp_mux && candidate.id > 1) {
953 /* Remote side may have offered RTP and RTCP candidates. However, if we're using RTCP MUX,
954 * then we should ignore RTCP candidates.
955 */
956 continue;
957 }
958
959 candidate.foundation = foundation;
960 candidate.transport = transport;
961
963 ast_sockaddr_set_port(&candidate.address, port);
964
965 if (!strcasecmp(cand_type, "host")) {
967 } else if (!strcasecmp(cand_type, "srflx")) {
969 } else if (!strcasecmp(cand_type, "relay")) {
971 } else {
972 continue;
973 }
974
977 }
978
979 if (relay_port) {
980 ast_sockaddr_set_port(&candidate.relay_address, relay_port);
981 }
982
983 ice->add_remote_candidate(session_media->rtp, &candidate);
984 }
985
986 ice->set_role(session_media->rtp, pjmedia_sdp_neg_was_answer_remote(session->inv_session->neg) == PJ_TRUE ?
988 ice->start(session_media->rtp);
989}
char * address
Definition f2c.h:59
@ AST_RTP_ICE_ROLE_CONTROLLING
Definition rtp_engine.h:521
@ AST_RTP_ICE_ROLE_CONTROLLED
Definition rtp_engine.h:520
#define ast_debug_ice(sublevel,...)
Log debug level ICE information.
void(* ice_lite)(struct ast_rtp_instance *instance)
Definition rtp_engine.h:552
void(* set_authentication)(struct ast_rtp_instance *instance, const char *ufrag, const char *password)
Definition rtp_engine.h:538
void(* add_remote_candidate)(struct ast_rtp_instance *instance, const struct ast_rtp_engine_ice_candidate *candidate)
Definition rtp_engine.h:540
void(* set_role)(struct ast_rtp_instance *instance, enum ast_rtp_ice_role role)
Definition rtp_engine.h:554

References ast_rtp_engine_ice::add_remote_candidate, ast_rtp_engine_ice_candidate::address, ast_copy_pj_str(), ast_debug_ice, AST_RTP_ICE_CANDIDATE_TYPE_HOST, AST_RTP_ICE_CANDIDATE_TYPE_RELAYED, AST_RTP_ICE_CANDIDATE_TYPE_SRFLX, AST_RTP_ICE_ROLE_CONTROLLED, AST_RTP_ICE_ROLE_CONTROLLING, ast_rtp_instance_get_ice(), ast_sockaddr_parse(), ast_sockaddr_set_port, ast_strlen_zero(), ast_rtp_engine_ice_candidate::foundation, ast_rtp_engine_ice::ice_lite, ast_rtp_engine_ice_candidate::id, NULL, PARSE_PORT_FORBID, ast_rtp_engine_ice_candidate::priority, ast_rtp_engine_ice_candidate::relay_address, ast_sip_session_media::remote_rtcp_mux, ast_sip_session_media::rtp, session, ast_rtp_engine_ice::set_authentication, ast_rtp_engine_ice::set_role, ast_rtp_engine_ice::start, ast_rtp_engine_ice_candidate::transport, and ast_rtp_engine_ice_candidate::type.

Referenced by apply_negotiated_sdp_stream().

◆ process_ice_auth_attrb()

static void process_ice_auth_attrb ( struct ast_sip_session session,
struct ast_sip_session_media session_media,
const struct pjmedia_sdp_session *  remote,
const struct pjmedia_sdp_media *  remote_stream 
)
static

Definition at line 850 of file res_pjsip_sdp_rtp.c.

852{
853 struct ast_rtp_engine_ice *ice;
855 char ufrag_attr_value[256];
856 char passwd_attr_value[256];
857
858 /* If ICE support is not enabled or available exit early */
859 if (!session->endpoint->media.rtp.ice_support || !(ice = ast_rtp_instance_get_ice(session_media->rtp))) {
860 return;
861 }
862
864 if (!ufrag_attr) {
865 ufrag_attr = pjmedia_sdp_attr_find2(remote->attr_count, remote->attr, "ice-ufrag", NULL);
866 }
867 if (ufrag_attr) {
869 } else {
870 return;
871 }
873 if (!passwd_attr) {
874 passwd_attr = pjmedia_sdp_attr_find2(remote->attr_count, remote->attr, "ice-pwd", NULL);
875 }
876 if (passwd_attr) {
878 } else {
879 return;
880 }
881
882 if (ufrag_attr && passwd_attr) {
884 }
885}

References ast_copy_pj_str(), ast_rtp_instance_get_ice(), NULL, session, ast_rtp_engine_ice::set_authentication, and ast_rtp_engine_ice::start.

Referenced by negotiate_incoming_sdp_stream().

◆ process_ssrc_attributes()

static void process_ssrc_attributes ( struct ast_sip_session session,
struct ast_sip_session_media session_media,
const struct pjmedia_sdp_media *  remote_stream 
)
static

Function which processes ssrc attributes in a stream.

Definition at line 1292 of file res_pjsip_sdp_rtp.c.

1294{
1295 int index;
1296
1297 if (!session->endpoint->media.bundle) {
1298 return;
1299 }
1300
1301 for (index = 0; index < remote_stream->attr_count; ++index) {
1302 pjmedia_sdp_attr *attr = remote_stream->attr[index];
1303 char attr_value[pj_strlen(&attr->value) + 1];
1304 char *ssrc_attribute_name, *ssrc_attribute_value = NULL;
1305 unsigned int ssrc;
1306
1307 /* We only care about ssrc attributes */
1308 if (pj_strcmp2(&attr->name, "ssrc")) {
1309 continue;
1310 }
1311
1312 ast_copy_pj_str(attr_value, &attr->value, sizeof(attr_value));
1313
1314 if ((ssrc_attribute_name = strchr(attr_value, ' '))) {
1315 /* This has an actual attribute */
1316 *ssrc_attribute_name++ = '\0';
1317 ssrc_attribute_value = strchr(ssrc_attribute_name, ':');
1318 if (ssrc_attribute_value) {
1319 /* Values are actually optional according to the spec */
1320 *ssrc_attribute_value++ = '\0';
1321 }
1322 }
1323
1324 if (sscanf(attr_value, "%30u", &ssrc) < 1) {
1325 continue;
1326 }
1327
1328 /* If we are currently negotiating as a result of the remote side renegotiating then
1329 * determine if the source for this stream has changed.
1330 */
1331 if (pjmedia_sdp_neg_get_state(session->inv_session->neg) == PJMEDIA_SDP_NEG_STATE_REMOTE_OFFER &&
1332 session->active_media_state) {
1333 struct ast_rtp_instance_stats stats = { 0, };
1334
1336 stats.remote_ssrc != ssrc) {
1337 session_media->changed = 1;
1338 }
1339 }
1340
1341 ast_rtp_instance_set_remote_ssrc(session_media->rtp, ssrc);
1342 break;
1343 }
1344}
@ AST_RTP_INSTANCE_STAT_REMOTE_SSRC
Definition rtp_engine.h:251
int ast_rtp_instance_get_stats(struct ast_rtp_instance *instance, struct ast_rtp_instance_stats *stats, enum ast_rtp_instance_stat stat)
Retrieve statistics about an RTP instance.
void ast_rtp_instance_set_remote_ssrc(struct ast_rtp_instance *rtp, unsigned int ssrc)
Set the remote SSRC for an RTP instance.
unsigned int remote_ssrc
Definition rtp_engine.h:454
unsigned int changed
The underlying session has been changed in some fashion.

References ast_copy_pj_str(), ast_rtp_instance_get_stats(), ast_rtp_instance_set_remote_ssrc(), AST_RTP_INSTANCE_STAT_REMOTE_SSRC, ast_sip_session_media::changed, NULL, ast_rtp_instance_stats::remote_ssrc, ast_sip_session_media::rtp, session, and ast_rtp_engine_ice::start.

Referenced by apply_negotiated_sdp_stream(), and negotiate_incoming_sdp_stream().

◆ rtp_check_timeout()

static int rtp_check_timeout ( const void *  data)
static

Check whether RTP is being received or not.

Definition at line 105 of file res_pjsip_sdp_rtp.c.

106{
107 struct ast_sip_session_media *session_media = (struct ast_sip_session_media *)data;
108 struct ast_rtp_instance *rtp = session_media->rtp;
109 struct ast_channel *chan;
110 int elapsed;
111 int now;
112 int timeout;
113
114 if (!rtp) {
115 return 0;
116 }
117
119 if (!chan) {
120 return 0;
121 }
122
123 /* Store these values locally to avoid multiple function calls */
124 now = time(NULL);
125 timeout = ast_rtp_instance_get_timeout(rtp);
126
127 /* If the channel is not in UP state or call is redirected
128 * outside Asterisk return for later check.
129 */
130 if (ast_channel_state(chan) != AST_STATE_UP || !ast_sockaddr_isnull(&session_media->direct_media_addr)) {
131 /* Avoiding immediately disconnect after channel up or direct media has been stopped */
133 ast_channel_unref(chan);
134 /* Recheck after half timeout for avoiding possible races
135 * and faster reacting to cases while there is no an RTP at all.
136 */
137 return timeout * 500;
138 }
139
140 elapsed = now - ast_rtp_instance_get_last_rx(rtp);
141 if (elapsed < timeout) {
142 ast_channel_unref(chan);
143 return (timeout - elapsed) * 1000;
144 }
145
146 ast_log(LOG_NOTICE, "Disconnecting channel '%s' for lack of %s RTP activity in %d seconds\n",
147 ast_channel_name(chan), ast_codec_media_type2str(session_media->type), elapsed);
148
149 ast_channel_lock(chan);
151 ast_channel_unlock(chan);
152
154 ast_channel_unref(chan);
155
156 return 0;
157}
#define AST_CAUSE_REQUESTED_CHAN_UNAVAIL
Definition causes.h:125
const char * ast_channel_name(const struct ast_channel *chan)
struct ast_channel * ast_channel_get_by_name(const char *search)
Find a channel by name or uniqueid.
Definition channel.c:1417
@ AST_SOFTHANGUP_DEV
Definition channel.h:1141
int ast_softhangup(struct ast_channel *chan, int cause)
Softly hangup up a channel.
Definition channel.c:2462
#define ast_channel_unref(c)
Decrease channel reference count.
Definition channel.h:3025
void ast_channel_hangupcause_set(struct ast_channel *chan, int value)
ast_channel_state
ast_channel states
@ AST_STATE_UP
#define LOG_NOTICE
time_t ast_rtp_instance_get_last_rx(const struct ast_rtp_instance *rtp)
Get the last RTP reception time.
const char * ast_rtp_instance_get_channel_id(struct ast_rtp_instance *instance)
Get the unique ID of the channel that owns this RTP instance.
Definition rtp_engine.c:599
Main Channel structure associated with a channel.

References AST_CAUSE_REQUESTED_CHAN_UNAVAIL, ast_channel_get_by_name(), ast_channel_hangupcause_set(), ast_channel_lock, ast_channel_name(), ast_channel_unlock, ast_channel_unref, ast_codec_media_type2str(), ast_log, ast_rtp_instance_get_channel_id(), ast_rtp_instance_get_last_rx(), ast_rtp_instance_get_timeout(), ast_rtp_instance_set_last_rx(), ast_sockaddr_isnull(), ast_softhangup(), AST_SOFTHANGUP_DEV, AST_STATE_UP, ast_sip_session_media::direct_media_addr, LOG_NOTICE, NULL, ast_sip_session_media::rtp, and ast_sip_session_media::type.

Referenced by apply_negotiated_sdp_stream().

◆ send_keepalive()

static int send_keepalive ( const void *  data)
static

Definition at line 71 of file res_pjsip_sdp_rtp.c.

72{
73 struct ast_sip_session_media *session_media = (struct ast_sip_session_media *) data;
74 struct ast_rtp_instance *rtp = session_media->rtp;
75 int keepalive;
76 time_t interval;
78
79 if (!rtp) {
80 return 0;
81 }
82
84
85 if (!ast_sockaddr_isnull(&session_media->direct_media_addr)) {
86 ast_debug_rtp(3, "(%p) RTP not sending keepalive since direct media is in use\n", rtp);
87 return keepalive * 1000;
88 }
89
90 interval = time(NULL) - ast_rtp_instance_get_last_tx(rtp);
91 send_keepalive = interval >= keepalive;
92
93 ast_debug_rtp(3, "(%p) RTP it has been %d seconds since RTP was last sent. %sending keepalive\n",
94 rtp, (int) interval, send_keepalive ? "S" : "Not s");
95
96 if (send_keepalive) {
98 return keepalive * 1000;
99 }
100
101 return (keepalive - interval) * 1000;
102}
int ast_rtp_instance_get_keepalive(struct ast_rtp_instance *instance)
Get the RTP keepalive interval.
time_t ast_rtp_instance_get_last_tx(const struct ast_rtp_instance *rtp)
Get the last RTP transmission time.
int ast_rtp_instance_sendcng(struct ast_rtp_instance *instance, int level)
Send a comfort noise packet to the RTP instance.

References ast_debug_rtp, ast_rtp_instance_get_keepalive(), ast_rtp_instance_get_last_tx(), ast_rtp_instance_sendcng(), ast_sockaddr_isnull(), ast_sip_session_media::direct_media_addr, ast_rtp_instance::keepalive, NULL, ast_sip_session_media::rtp, and send_keepalive().

Referenced by apply_negotiated_sdp_stream(), and send_keepalive().

◆ set_caps()

static int set_caps ( struct ast_sip_session session,
struct ast_sip_session_media session_media,
struct ast_sip_session_media session_media_transport,
const struct pjmedia_sdp_media *  stream,
int  is_offer,
struct ast_stream asterisk_stream 
)
static

Definition at line 520 of file res_pjsip_sdp_rtp.c.

525{
526 RAII_VAR(struct ast_format_cap *, caps, NULL, ao2_cleanup);
527 RAII_VAR(struct ast_format_cap *, peer, NULL, ao2_cleanup);
528 RAII_VAR(struct ast_format_cap *, joint, NULL, ao2_cleanup);
529 enum ast_media_type media_type = session_media->type;
531 int direct_media_enabled = !ast_sockaddr_isnull(&session_media->direct_media_addr) &&
532 ast_format_cap_count(session->direct_media_cap);
533 int dsp_features = 0;
534 SCOPE_ENTER(1, "%s %s\n", ast_sip_session_get_name(session), is_offer ? "OFFER" : "ANSWER");
535
539 ast_log(LOG_ERROR, "Failed to allocate %s capabilities\n",
540 ast_codec_media_type2str(session_media->type));
541 SCOPE_EXIT_RTN_VALUE(-1, "Couldn't create %s capabilities\n",
542 ast_codec_media_type2str(session_media->type));
543 }
544
545 /* get the endpoint capabilities */
546 if (direct_media_enabled) {
547 ast_format_cap_get_compatible(session->endpoint->media.codecs, session->direct_media_cap, caps);
548 } else {
549 ast_format_cap_append_from_cap(caps, session->endpoint->media.codecs, media_type);
550 }
551
552 /* get the capabilities on the peer */
553 get_codecs(session, stream, &codecs, session_media, peer);
554
555 /* get the joint capabilities between peer and endpoint */
556 ast_format_cap_get_compatible(caps, peer, joint);
557 if (!ast_format_cap_count(joint)) {
560
562 ast_log(LOG_NOTICE, "No joint capabilities for '%s' media stream between our configuration(%s) and incoming SDP(%s)\n",
563 ast_codec_media_type2str(session_media->type),
564 ast_format_cap_get_names(caps, &usbuf),
565 ast_format_cap_get_names(peer, &thembuf));
566 SCOPE_EXIT_RTN_VALUE(-1, "No joint capabilities for '%s' media stream between our configuration(%s) and incoming SDP(%s)\n",
567 ast_codec_media_type2str(session_media->type),
568 ast_format_cap_get_names(caps, &usbuf),
569 ast_format_cap_get_names(peer, &thembuf));
570 } else {
571 struct ast_format *preferred_fmt = ast_format_cap_get_format(joint, 0);
572
574 ao2_ref(preferred_fmt, -1);
575 }
576
577 if (is_offer) {
578 /*
579 * Setup rx payload type mapping to prefer the mapping
580 * from the peer that the RFC says we SHOULD use.
581 */
583 }
585 session_media->rtp);
586
587 apply_cap_to_bundled(session_media, session_media_transport, asterisk_stream, joint);
588
589 if (session->channel && ast_sip_session_is_pending_stream_default(session, asterisk_stream)) {
590 ast_channel_lock(session->channel);
594 ast_format_cap_remove_by_type(caps, media_type);
595
596 if (session->endpoint->preferred_codec_only) {
597 struct ast_format *preferred_fmt = ast_format_cap_get_format(joint, 0);
598 ast_format_cap_append(caps, preferred_fmt, 0);
599 ao2_ref(preferred_fmt, -1);
600 } else if (!session->endpoint->asymmetric_rtp_codec) {
601 struct ast_format *best;
602 /*
603 * If we don't allow the sending codec to be changed on our side
604 * then get the best codec from the joint capabilities of the media
605 * type and use only that. This ensures the core won't start sending
606 * out a format that we aren't currently sending.
607 */
608
609 best = ast_format_cap_get_best_by_type(joint, media_type);
610 if (best) {
612 ao2_ref(best, -1);
613 }
614 } else {
615 ast_format_cap_append_from_cap(caps, joint, media_type);
616 }
617
618 /*
619 * Apply the new formats to the channel, potentially changing
620 * raw read/write formats and translation path while doing so.
621 */
623 if (media_type == AST_MEDIA_TYPE_AUDIO) {
626 }
627
628 if ( ((session->dtmf == AST_SIP_DTMF_AUTO) || (session->dtmf == AST_SIP_DTMF_AUTO_INFO) )
630 && (session->dsp)) {
631 dsp_features = ast_dsp_get_features(session->dsp);
632 dsp_features &= ~DSP_FEATURE_DIGIT_DETECT;
633 if (dsp_features) {
634 ast_dsp_set_features(session->dsp, dsp_features);
635 } else {
636 ast_dsp_free(session->dsp);
637 session->dsp = NULL;
638 }
639 }
640
641 if (ast_channel_is_bridged(session->channel)) {
643 }
644
645 ast_channel_unlock(session->channel);
646 }
647
650}
void ast_channel_set_unbridged_nolock(struct ast_channel *chan, int value)
Variant of ast_channel_set_unbridged. Use this if the channel is already locked prior to calling.
void ast_channel_nativeformats_set(struct ast_channel *chan, struct ast_format_cap *value)
struct ast_format_cap * ast_channel_nativeformats(const struct ast_channel *chan)
int ast_set_read_format(struct ast_channel *chan, struct ast_format *format)
Sets read format on channel chan.
Definition channel.c:5819
int ast_channel_is_bridged(const struct ast_channel *chan)
Determine if a channel is in a bridge.
Definition channel.c:10655
struct ast_format * ast_channel_writeformat(struct ast_channel *chan)
int ast_set_write_format(struct ast_channel *chan, struct ast_format *format)
Sets write format on channel chan.
Definition channel.c:5860
struct ast_format * ast_channel_readformat(struct ast_channel *chan)
void ast_dsp_free(struct ast_dsp *dsp)
Definition dsp.c:1968
int ast_dsp_get_features(struct ast_dsp *dsp)
Get features.
Definition dsp.c:1962
void ast_dsp_set_features(struct ast_dsp *dsp, int features)
Select feature set.
Definition dsp.c:1953
#define AST_FORMAT_CAP_NAMES_LEN
Definition format_cap.h:324
struct ast_format * ast_format_cap_get_best_by_type(const struct ast_format_cap *cap, enum ast_media_type type)
Get the most preferred format for a particular media type.
Definition format_cap.c:417
const char * ast_format_cap_get_names(const struct ast_format_cap *cap, struct ast_str **buf)
Get the names of codecs of a set of formats.
Definition format_cap.c:734
static void get_codecs(struct ast_sip_session *session, const struct pjmedia_sdp_media *stream, struct ast_rtp_codecs *codecs, struct ast_sip_session_media *session_media, struct ast_format_cap *astformats)
void ast_rtp_codecs_payloads_destroy(struct ast_rtp_codecs *codecs)
Destroy the contents of an RTP codecs structure (but not the structure itself)
enum ast_rtp_dtmf_mode ast_rtp_instance_dtmf_mode_get(struct ast_rtp_instance *instance)
Get the DTMF mode of an RTP instance.
void ast_rtp_codecs_payloads_copy(struct ast_rtp_codecs *src, struct ast_rtp_codecs *dest, struct ast_rtp_instance *instance)
Copy payload information from one RTP instance to another.
#define AST_RTP_CODECS_NULL_INIT
Definition rtp_engine.h:773
void ast_rtp_codecs_payloads_xover(struct ast_rtp_codecs *src, struct ast_rtp_codecs *dest, struct ast_rtp_instance *instance)
Crossover copy the tx payload mapping of src to the rx payload mapping of dest.
int ast_rtp_codecs_set_preferred_format(struct ast_rtp_codecs *codecs, struct ast_format *format)
Set the preferred format.

References ao2_cleanup, ao2_ref, apply_cap_to_bundled(), ast_channel_is_bridged(), ast_channel_lock, ast_channel_nativeformats(), ast_channel_nativeformats_set(), ast_channel_readformat(), ast_channel_set_unbridged_nolock(), ast_channel_unlock, ast_channel_writeformat(), ast_codec_media_type2str(), ast_dsp_free(), ast_dsp_get_features(), ast_dsp_set_features(), ast_format_cap_alloc, ast_format_cap_append, ast_format_cap_append_from_cap(), ast_format_cap_count(), AST_FORMAT_CAP_FLAG_DEFAULT, ast_format_cap_get_best_by_type(), ast_format_cap_get_compatible(), ast_format_cap_get_format(), ast_format_cap_get_framing(), ast_format_cap_get_names(), AST_FORMAT_CAP_NAMES_LEN, ast_format_cap_remove_by_type(), ast_log, AST_MEDIA_TYPE_AUDIO, AST_MEDIA_TYPE_UNKNOWN, AST_RTP_CODECS_NULL_INIT, ast_rtp_codecs_payloads_copy(), ast_rtp_codecs_payloads_destroy(), ast_rtp_codecs_payloads_xover(), ast_rtp_codecs_set_preferred_format(), AST_RTP_DTMF_MODE_RFC2833, ast_rtp_instance_dtmf_mode_get(), ast_rtp_instance_get_codecs(), ast_set_read_format(), ast_set_write_format(), AST_SIP_DTMF_AUTO, AST_SIP_DTMF_AUTO_INFO, ast_sip_session_get_name(), ast_sip_session_is_pending_stream_default(), ast_sockaddr_isnull(), ast_str_alloca, codecs, ast_sip_session_media::direct_media_addr, get_codecs(), LOG_ERROR, LOG_NOTICE, NULL, RAII_VAR, ast_sip_session_media::rtp, SCOPE_ENTER, SCOPE_EXIT_RTN_VALUE, session, and ast_sip_session_media::type.

Referenced by apply_negotiated_sdp_stream().

◆ set_ice_components()

static void set_ice_components ( struct ast_sip_session session,
struct ast_sip_session_media session_media 
)
static

Definition at line 1255 of file res_pjsip_sdp_rtp.c.

1256{
1257 struct ast_rtp_engine_ice *ice;
1258
1259 ast_assert(session_media->rtp != NULL);
1260
1262 if (!session->endpoint->media.rtp.ice_support || !ice) {
1263 return;
1264 }
1265
1266 if (session->endpoint->media.rtcp_mux && session_media->remote_rtcp_mux) {
1267 /* We both support RTCP mux. Only one ICE component necessary */
1268 ice->change_components(session_media->rtp, 1);
1269 } else {
1270 /* They either don't support RTCP mux or we don't know if they do yet. */
1271 ice->change_components(session_media->rtp, 2);
1272 }
1273}
void(* change_components)(struct ast_rtp_instance *instance, int num_components)
Definition rtp_engine.h:560
#define ast_assert(a)
Definition utils.h:779

References ast_assert, ast_rtp_instance_get_ice(), ast_rtp_engine_ice::change_components, NULL, session, and ast_rtp_engine_ice::start.

Referenced by apply_negotiated_sdp_stream(), create_outgoing_sdp_stream(), and negotiate_incoming_sdp_stream().

◆ set_incoming_call_offer_cap()

static struct ast_format_cap * set_incoming_call_offer_cap ( struct ast_sip_session session,
struct ast_sip_session_media session_media,
const struct pjmedia_sdp_media *  stream 
)
static

Definition at line 475 of file res_pjsip_sdp_rtp.c.

478{
479 struct ast_format_cap *incoming_call_offer_cap;
480 struct ast_format_cap *remote;
483
484
486 if (!remote) {
487 ast_log(LOG_ERROR, "Failed to allocate %s incoming remote capabilities\n",
488 ast_codec_media_type2str(session_media->type));
489 SCOPE_EXIT_RTN_VALUE(NULL, "Couldn't allocate caps\n");
490 }
491
492 /* Get the peer's capabilities*/
493 get_codecs(session, stream, &codecs, session_media, remote);
494
495 incoming_call_offer_cap = ast_sip_session_create_joint_call_cap(
496 session, session_media->type, remote);
497
498 ao2_ref(remote, -1);
499
500 if (!incoming_call_offer_cap || ast_format_cap_empty(incoming_call_offer_cap)) {
501 ao2_cleanup(incoming_call_offer_cap);
503 SCOPE_EXIT_RTN_VALUE(NULL, "No incoming call offer caps\n");
504 }
505
506 /*
507 * Setup rx payload type mapping to prefer the mapping
508 * from the peer that the RFC says we SHOULD use.
509 */
511
513 ast_rtp_instance_get_codecs(session_media->rtp), session_media->rtp);
514
516
517 SCOPE_EXIT_RTN_VALUE(incoming_call_offer_cap);
518}
int ast_format_cap_empty(const struct ast_format_cap *cap)
Determine if a format cap has no formats in it.
Definition format_cap.c:744
struct ast_format_cap * ast_sip_session_create_joint_call_cap(const struct ast_sip_session *session, enum ast_media_type media_type, const struct ast_format_cap *remote)
Create joint capabilities.

References ao2_cleanup, ao2_ref, ast_codec_media_type2str(), ast_format_cap_alloc, ast_format_cap_empty(), AST_FORMAT_CAP_FLAG_DEFAULT, ast_log, AST_RTP_CODECS_NULL_INIT, ast_rtp_codecs_payloads_copy(), ast_rtp_codecs_payloads_destroy(), ast_rtp_codecs_payloads_xover(), ast_rtp_instance_get_codecs(), ast_sip_session_create_joint_call_cap(), ast_sip_session_get_name(), codecs, get_codecs(), LOG_ERROR, NULL, ast_sip_session_media::rtp, SCOPE_ENTER, SCOPE_EXIT_RTN_VALUE, session, and ast_sip_session_media::type.

Referenced by negotiate_incoming_sdp_stream().

◆ set_session_media_remotely_held()

static void set_session_media_remotely_held ( struct ast_sip_session_media session_media,
const struct ast_sip_session session,
const pjmedia_sdp_media *  media,
const struct ast_stream stream,
const struct ast_sockaddr addrs 
)
static

Definition at line 1535 of file res_pjsip_sdp_rtp.c.

1540{
1542 (session_media->type == AST_MEDIA_TYPE_AUDIO)) {
1543 if (((addrs != NULL) && ast_sockaddr_isnull(addrs)) ||
1544 ((addrs != NULL) && ast_sockaddr_is_any(addrs)) ||
1545 pjmedia_sdp_media_find_attr2(media, "sendonly", NULL) ||
1546 pjmedia_sdp_media_find_attr2(media, "inactive", NULL)) {
1547 if (!session_media->remotely_held) {
1548 session_media->remotely_held = 1;
1549 session_media->remotely_held_changed = 1;
1550 }
1551 } else if (session_media->remotely_held) {
1552 session_media->remotely_held = 0;
1553 session_media->remotely_held_changed = 1;
1554 }
1555 }
1556}
int ast_sockaddr_is_any(const struct ast_sockaddr *addr)
Determine if the address type is unspecified, or "any" address.
Definition netsock2.c:534

References AST_MEDIA_TYPE_AUDIO, ast_sip_session_is_pending_stream_default(), ast_sockaddr_is_any(), ast_sockaddr_isnull(), NULL, ast_sip_session_media::remotely_held, ast_sip_session_media::remotely_held_changed, session, and ast_sip_session_media::type.

Referenced by apply_negotiated_sdp_stream(), and negotiate_incoming_sdp_stream().

◆ setup_dtls_srtp()

static int setup_dtls_srtp ( struct ast_sip_session session,
struct ast_sip_session_media session_media 
)
static

Definition at line 1097 of file res_pjsip_sdp_rtp.c.

1099{
1100 struct ast_rtp_engine_dtls *dtls;
1101
1102 if (!session->endpoint->media.rtp.dtls_cfg.enabled || !session_media->rtp) {
1103 return -1;
1104 }
1105
1106 dtls = ast_rtp_instance_get_dtls(session_media->rtp);
1107 if (!dtls) {
1108 return -1;
1109 }
1110
1111 session->endpoint->media.rtp.dtls_cfg.suite = ((session->endpoint->media.rtp.srtp_tag_32) ? AST_AES_CM_128_HMAC_SHA1_32 : AST_AES_CM_128_HMAC_SHA1_80);
1112 if (dtls->set_configuration(session_media->rtp, &session->endpoint->media.rtp.dtls_cfg)) {
1113 ast_log(LOG_ERROR, "Attempted to set an invalid DTLS-SRTP configuration on RTP instance '%p'\n",
1114 session_media->rtp);
1115 return -1;
1116 }
1117
1118 if (setup_srtp(session_media)) {
1119 return -1;
1120 }
1121 return 0;
1122}
static int setup_srtp(struct ast_sip_session_media *session_media)
@ AST_AES_CM_128_HMAC_SHA1_80
Definition res_srtp.h:58
@ AST_AES_CM_128_HMAC_SHA1_32
Definition res_srtp.h:59
int(* set_configuration)(struct ast_rtp_instance *instance, const struct ast_rtp_dtls_cfg *dtls_cfg)
Definition rtp_engine.h:623

References AST_AES_CM_128_HMAC_SHA1_32, AST_AES_CM_128_HMAC_SHA1_80, ast_log, ast_rtp_instance_get_dtls(), LOG_ERROR, ast_sip_session_media::rtp, session, ast_rtp_engine_dtls::set_configuration, and setup_srtp().

Referenced by add_crypto_to_stream(), and setup_media_encryption().

◆ setup_media_encryption()

static int setup_media_encryption ( struct ast_sip_session session,
struct ast_sip_session_media session_media,
const struct pjmedia_sdp_session *  sdp,
const struct pjmedia_sdp_media *  stream 
)
static

Definition at line 1228 of file res_pjsip_sdp_rtp.c.

1232{
1233 switch (session_media->encryption) {
1235 if (setup_sdes_srtp(session_media, stream)) {
1236 return -1;
1237 }
1238 break;
1240 if (setup_dtls_srtp(session, session_media)) {
1241 return -1;
1242 }
1243 if (parse_dtls_attrib(session_media, sdp, stream)) {
1244 return -1;
1245 }
1246 break;
1249 break;
1250 }
1251
1252 return 0;
1253}
static int parse_dtls_attrib(struct ast_sip_session_media *session_media, const struct pjmedia_sdp_session *sdp, const struct pjmedia_sdp_media *stream)
static int setup_sdes_srtp(struct ast_sip_session_media *session_media, const struct pjmedia_sdp_media *stream)

References AST_SIP_MEDIA_ENCRYPT_DTLS, AST_SIP_MEDIA_ENCRYPT_NONE, AST_SIP_MEDIA_ENCRYPT_SDES, AST_SIP_MEDIA_TRANSPORT_INVALID, ast_sip_session_media::encryption, parse_dtls_attrib(), session, setup_dtls_srtp(), and setup_sdes_srtp().

Referenced by apply_negotiated_sdp_stream(), and negotiate_incoming_sdp_stream().

◆ setup_sdes_srtp()

static int setup_sdes_srtp ( struct ast_sip_session_media session_media,
const struct pjmedia_sdp_media *  stream 
)
static

Definition at line 1192 of file res_pjsip_sdp_rtp.c.

1194{
1195 int i;
1196
1197 for (i = 0; i < stream->attr_count; i++) {
1198 pjmedia_sdp_attr *attr;
1199 RAII_VAR(char *, crypto_str, NULL, ast_free);
1200
1201 /* check the stream for the required crypto attribute */
1202 attr = stream->attr[i];
1203 if (pj_strcmp2(&attr->name, "crypto")) {
1204 continue;
1205 }
1206
1207 crypto_str = ast_strndup(attr->value.ptr, attr->value.slen);
1208 if (!crypto_str) {
1209 return -1;
1210 }
1211
1212 if (setup_srtp(session_media)) {
1213 return -1;
1214 }
1215
1216 if (!ast_sdp_crypto_process(session_media->rtp, session_media->srtp, crypto_str)) {
1217 /* found a valid crypto attribute */
1218 return 0;
1219 }
1220
1221 ast_debug(1, "Ignoring crypto offer with unsupported parameters: %s\n", crypto_str);
1222 }
1223
1224 /* no usable crypto attributes found */
1225 return -1;
1226}
int ast_sdp_crypto_process(struct ast_rtp_instance *rtp, struct ast_sdp_srtp *srtp, const char *attr)
Parse the a=crypto line from SDP and set appropriate values on the ast_sdp_crypto struct.
Definition sdp_srtp.c:79

References ast_debug, ast_free, ast_sdp_crypto_process(), ast_strndup, NULL, RAII_VAR, ast_sip_session_media::rtp, setup_srtp(), and ast_sip_session_media::srtp.

Referenced by setup_media_encryption().

◆ setup_srtp()

static int setup_srtp ( struct ast_sip_session_media session_media)
static

Definition at line 1078 of file res_pjsip_sdp_rtp.c.

1079{
1080 if (!session_media->srtp) {
1081 session_media->srtp = ast_sdp_srtp_alloc();
1082 if (!session_media->srtp) {
1083 return -1;
1084 }
1085 }
1086
1087 if (!session_media->srtp->crypto) {
1088 session_media->srtp->crypto = ast_sdp_crypto_alloc();
1089 if (!session_media->srtp->crypto) {
1090 return -1;
1091 }
1092 }
1093
1094 return 0;
1095}
struct ast_sdp_crypto * ast_sdp_crypto_alloc(void)
Initialize an return an ast_sdp_crypto struct.
Definition sdp_srtp.c:71
struct ast_sdp_crypto * crypto
Definition sdp_srtp.h:40

References ast_sdp_crypto_alloc(), ast_sdp_srtp_alloc(), ast_sdp_srtp::crypto, and ast_sip_session_media::srtp.

Referenced by setup_dtls_srtp(), and setup_sdes_srtp().

◆ stream_destroy()

static void stream_destroy ( struct ast_sip_session_media session_media)
static

Function which destroys the RTP instance when session ends.

Definition at line 2440 of file res_pjsip_sdp_rtp.c.

2441{
2442 if (session_media->rtp) {
2443 stream_stop(session_media);
2444 ast_rtp_instance_destroy(session_media->rtp);
2445 }
2446 session_media->rtp = NULL;
2447}
static void stream_stop(struct ast_sip_session_media *session_media)
Function which stops the RTP instance.
int ast_rtp_instance_destroy(struct ast_rtp_instance *instance)
Destroy an RTP instance.
Definition rtp_engine.c:468

References ast_rtp_instance_destroy(), NULL, ast_sip_session_media::rtp, and stream_stop().

◆ stream_stop()

static void stream_stop ( struct ast_sip_session_media session_media)
static

Function which stops the RTP instance.

Definition at line 2428 of file res_pjsip_sdp_rtp.c.

2429{
2430 if (!session_media->rtp) {
2431 return;
2432 }
2433
2434 AST_SCHED_DEL(sched, session_media->keepalive_sched_id);
2435 AST_SCHED_DEL(sched, session_media->timeout_sched_id);
2436 ast_rtp_instance_stop(session_media->rtp);
2437}

References ast_rtp_instance_stop(), AST_SCHED_DEL, ast_sip_session_media::keepalive_sched_id, ast_sip_session_media::rtp, and ast_sip_session_media::timeout_sched_id.

Referenced by stream_destroy().

◆ unload_module()

static int unload_module ( void  )
static

Unloads the sdp RTP/AVP module from Asterisk.

Definition at line 2499 of file res_pjsip_sdp_rtp.c.

2500{
2504
2505 if (sched) {
2507 }
2508
2509 return 0;
2510}
void ast_sip_session_unregister_sdp_handler(struct ast_sip_session_sdp_handler *handler, const char *stream_type)
Unregister an SDP handler.
void ast_sip_session_unregister_supplement(struct ast_sip_session_supplement *supplement)
Unregister a an supplement to SIP session processing.
void ast_sched_context_destroy(struct ast_sched_context *c)
destroys a schedule context
Definition sched.c:271

References ast_sched_context_destroy(), ast_sip_session_unregister_sdp_handler(), ast_sip_session_unregister_supplement(), audio_sdp_handler, STR_AUDIO, STR_VIDEO, video_info_supplement, and video_sdp_handler.

Referenced by load_module().

◆ video_info_incoming_request()

static int video_info_incoming_request ( struct ast_sip_session session,
struct pjsip_rx_data *  rdata 
)
static

Definition at line 2471 of file res_pjsip_sdp_rtp.c.

2472{
2473 struct pjsip_transaction *tsx;
2474 pjsip_tx_data *tdata;
2475
2476 if (!session->channel
2477 || !ast_sip_are_media_types_equal(&rdata->msg_info.msg->body->content_type,
2479 return 0;
2480 }
2481
2482 tsx = pjsip_rdata_get_tsx(rdata);
2483
2485
2486 if (pjsip_dlg_create_response(session->inv_session->dlg, rdata, 200, NULL, &tdata) == PJ_SUCCESS) {
2487 pjsip_dlg_send_response(session->inv_session->dlg, tsx, tdata);
2488 }
2489
2490 return 0;
2491}
@ AST_CONTROL_VIDUPDATE
pjsip_media_type pjsip_media_type_application_media_control_xml
Definition res_pjsip.c:3901
int ast_sip_are_media_types_equal(pjsip_media_type *a, pjsip_media_type *b)
Compare pjsip media types.
Definition res_pjsip.c:2190

References AST_CONTROL_VIDUPDATE, ast_queue_control(), ast_sip_are_media_types_equal(), NULL, pjsip_media_type_application_media_control_xml, and session.

Variable Documentation

◆ __mod_info

struct ast_module_info __mod_info = { .name = AST_MODULE, .flags = AST_MODFLAG_LOAD_ORDER , .description = "PJSIP SDP RTP/AVP stream handler" , .key = ASTERISK_GPL_KEY , .buildopt_sum = AST_BUILDOPT_SUM, .support_level = AST_MODULE_SUPPORT_CORE, .load = load_module, .unload = unload_module, .load_pri = AST_MODPRI_CHANNEL_DRIVER, .requires = "res_pjsip,res_pjsip_session", }
static

Definition at line 2565 of file res_pjsip_sdp_rtp.c.

◆ address_rtp

struct ast_sockaddr address_rtp
static

Address for RTP.

Definition at line 66 of file res_pjsip_sdp_rtp.c.

Referenced by create_rtp(), and load_module().

◆ ast_module_info

const struct ast_module_info* ast_module_info = &__mod_info
static

Definition at line 2565 of file res_pjsip_sdp_rtp.c.

◆ audio_sdp_handler

struct ast_sip_session_sdp_handler audio_sdp_handler
static

SDP handler for 'audio' media stream.

Definition at line 2450 of file res_pjsip_sdp_rtp.c.

2450 {
2451 .id = STR_AUDIO,
2452 .negotiate_incoming_sdp_stream = negotiate_incoming_sdp_stream,
2453 .create_outgoing_sdp_stream = create_outgoing_sdp_stream,
2454 .apply_negotiated_sdp_stream = apply_negotiated_sdp_stream,
2455 .change_outgoing_sdp_stream_media_address = change_outgoing_sdp_stream_media_address,
2456 .stream_stop = stream_stop,
2457 .stream_destroy = stream_destroy,
2458};
static int negotiate_incoming_sdp_stream(struct ast_sip_session *session, struct ast_sip_session_media *session_media, const pjmedia_sdp_session *sdp, int index, struct ast_stream *asterisk_stream)
Function which negotiates an incoming media stream.
static int apply_negotiated_sdp_stream(struct ast_sip_session *session, struct ast_sip_session_media *session_media, const struct pjmedia_sdp_session *local, const struct pjmedia_sdp_session *remote, int index, struct ast_stream *asterisk_stream)
static void stream_destroy(struct ast_sip_session_media *session_media)
Function which destroys the RTP instance when session ends.
static void change_outgoing_sdp_stream_media_address(pjsip_tx_data *tdata, struct pjmedia_sdp_media *stream, struct ast_sip_transport *transport)
Function which updates the media stream with external media address, if applicable.
static int create_outgoing_sdp_stream(struct ast_sip_session *session, struct ast_sip_session_media *session_media, struct pjmedia_sdp_session *sdp, const struct pjmedia_sdp_session *remote, struct ast_stream *stream)
Function which creates an outgoing stream.

Referenced by load_module(), and unload_module().

◆ sched

struct ast_sched_context* sched
static

Scheduler for RTCP purposes.

Definition at line 63 of file res_pjsip_sdp_rtp.c.

◆ STR_AUDIO

const char STR_AUDIO[] = "audio"
static

Definition at line 68 of file res_pjsip_sdp_rtp.c.

Referenced by load_module(), and unload_module().

◆ STR_VIDEO

const char STR_VIDEO[] = "video"
static

Definition at line 69 of file res_pjsip_sdp_rtp.c.

Referenced by load_module(), and unload_module().

◆ video_info_supplement

struct ast_sip_session_supplement video_info_supplement
static
Initial value:
= {
.method = "INFO",
.incoming_request = video_info_incoming_request,
}
static int video_info_incoming_request(struct ast_sip_session *session, struct pjsip_rx_data *rdata)

Definition at line 2493 of file res_pjsip_sdp_rtp.c.

2493 {
2494 .method = "INFO",
2495 .incoming_request = video_info_incoming_request,
2496};

Referenced by load_module(), and unload_module().

◆ video_sdp_handler

struct ast_sip_session_sdp_handler video_sdp_handler
static

SDP handler for 'video' media stream.

Definition at line 2461 of file res_pjsip_sdp_rtp.c.

2461 {
2462 .id = STR_VIDEO,
2463 .negotiate_incoming_sdp_stream = negotiate_incoming_sdp_stream,
2464 .create_outgoing_sdp_stream = create_outgoing_sdp_stream,
2465 .apply_negotiated_sdp_stream = apply_negotiated_sdp_stream,
2466 .change_outgoing_sdp_stream_media_address = change_outgoing_sdp_stream_media_address,
2467 .stream_stop = stream_stop,
2468 .stream_destroy = stream_destroy,
2469};

Referenced by load_module(), and unload_module().