Asterisk - The Open Source Telephony Project GIT-master-27fb039
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 2549 of file res_pjsip_sdp_rtp.c.

◆ __unreg_module()

static void __unreg_module ( void  )
static

Definition at line 2549 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 1648 of file res_pjsip_sdp_rtp.c.

1651{
1652 pj_str_t stmp;
1653 pjmedia_sdp_attr *attr;
1654 enum ast_rtp_dtls_hash hash;
1655 const char *crypto_attribute;
1656 struct ast_rtp_engine_dtls *dtls;
1657 struct ast_sdp_srtp *tmp;
1658 static const pj_str_t STR_NEW = { "new", 3 };
1659 static const pj_str_t STR_EXISTING = { "existing", 8 };
1660 static const pj_str_t STR_ACTIVE = { "active", 6 };
1661 static const pj_str_t STR_PASSIVE = { "passive", 7 };
1662 static const pj_str_t STR_ACTPASS = { "actpass", 7 };
1663 static const pj_str_t STR_HOLDCONN = { "holdconn", 8 };
1664 static const pj_str_t STR_MEDSECREQ = { "requested", 9 };
1665 enum ast_rtp_dtls_setup setup;
1666
1667 switch (session_media->encryption) {
1670 break;
1672 if (!session_media->srtp) {
1673 session_media->srtp = ast_sdp_srtp_alloc();
1674 if (!session_media->srtp) {
1675 return -1;
1676 }
1677 }
1678
1679 tmp = session_media->srtp;
1680
1681 do {
1682 crypto_attribute = ast_sdp_srtp_get_attrib(tmp,
1683 0 /* DTLS running? No */,
1684 session->endpoint->media.rtp.srtp_tag_32 /* 32 byte tag length? */);
1685 if (!crypto_attribute) {
1686 /* No crypto attribute to add, bad news */
1687 return -1;
1688 }
1689
1690 attr = pjmedia_sdp_attr_create(pool, "crypto",
1691 pj_cstr(&stmp, crypto_attribute));
1692 media->attr[media->attr_count++] = attr;
1693 } while ((tmp = AST_LIST_NEXT(tmp, sdp_srtp_list)));
1694
1695 if (session->endpoint->security_negotiation == AST_SIP_SECURITY_NEG_MEDIASEC) {
1696 attr = pjmedia_sdp_attr_create(pool, "3ge2ae", &STR_MEDSECREQ);
1697 media->attr[media->attr_count++] = attr;
1698 }
1699
1700 break;
1702 if (setup_dtls_srtp(session, session_media)) {
1703 return -1;
1704 }
1705
1706 dtls = ast_rtp_instance_get_dtls(session_media->rtp);
1707 if (!dtls) {
1708 return -1;
1709 }
1710
1711 switch (dtls->get_connection(session_media->rtp)) {
1713 attr = pjmedia_sdp_attr_create(pool, "connection", &STR_NEW);
1714 media->attr[media->attr_count++] = attr;
1715 break;
1717 attr = pjmedia_sdp_attr_create(pool, "connection", &STR_EXISTING);
1718 media->attr[media->attr_count++] = attr;
1719 break;
1720 default:
1721 break;
1722 }
1723
1724 /* If this is an answer we need to use our current state, if it's an offer we need to use
1725 * the configured value.
1726 */
1727 if (session->inv_session->neg
1728 && pjmedia_sdp_neg_get_state(session->inv_session->neg) != PJMEDIA_SDP_NEG_STATE_DONE) {
1729 setup = dtls->get_setup(session_media->rtp);
1730 } else {
1731 setup = session->endpoint->media.rtp.dtls_cfg.default_setup;
1732 }
1733
1734 switch (setup) {
1736 attr = pjmedia_sdp_attr_create(pool, "setup", &STR_ACTIVE);
1737 media->attr[media->attr_count++] = attr;
1738 break;
1740 attr = pjmedia_sdp_attr_create(pool, "setup", &STR_PASSIVE);
1741 media->attr[media->attr_count++] = attr;
1742 break;
1744 attr = pjmedia_sdp_attr_create(pool, "setup", &STR_ACTPASS);
1745 media->attr[media->attr_count++] = attr;
1746 break;
1748 attr = pjmedia_sdp_attr_create(pool, "setup", &STR_HOLDCONN);
1749 break;
1750 default:
1751 break;
1752 }
1753
1754 hash = dtls->get_fingerprint_hash(session_media->rtp);
1755 crypto_attribute = dtls->get_fingerprint(session_media->rtp);
1756 if (crypto_attribute && (hash == AST_RTP_DTLS_HASH_SHA1 || hash == AST_RTP_DTLS_HASH_SHA256)) {
1757 RAII_VAR(struct ast_str *, fingerprint, ast_str_create(64), ast_free);
1758 if (!fingerprint) {
1759 return -1;
1760 }
1761
1762 if (hash == AST_RTP_DTLS_HASH_SHA1) {
1763 ast_str_set(&fingerprint, 0, "SHA-1 %s", crypto_attribute);
1764 } else {
1765 ast_str_set(&fingerprint, 0, "SHA-256 %s", crypto_attribute);
1766 }
1767
1768 attr = pjmedia_sdp_attr_create(pool, "fingerprint", pj_cstr(&stmp, ast_str_buffer(fingerprint)));
1769 media->attr[media->attr_count++] = attr;
1770 }
1771 break;
1772 }
1773
1774 return 0;
1775}
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::@299 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 1412 of file res_pjsip_sdp_rtp.c.

1414{
1415 int idx;
1416 char extmap_value[256];
1417
1418 if (!session->endpoint->media.webrtc || session_media->type != AST_MEDIA_TYPE_VIDEO) {
1419 return;
1420 }
1421
1422 /* RTP extension local unique identifiers start at '1' */
1423 for (idx = 1; idx <= ast_rtp_instance_extmap_count(session_media->rtp); ++idx) {
1425 const char *direction_str = "";
1426 pj_str_t stmp;
1427 pjmedia_sdp_attr *attr;
1428
1429 /* If this is an unsupported RTP extension we can't place it into the SDP */
1431 continue;
1432 }
1433
1434 switch (ast_rtp_instance_extmap_get_direction(session_media->rtp, idx)) {
1436 /* Lack of a direction indicates sendrecv, so we leave it out */
1437 direction_str = "";
1438 break;
1440 direction_str = "/sendonly";
1441 break;
1443 direction_str = "/recvonly";
1444 break;
1446 /* It is impossible for a "none" direction extension to be negotiated but just in case
1447 * we treat it as inactive.
1448 */
1450 direction_str = "/inactive";
1451 break;
1452 }
1453
1454 snprintf(extmap_value, sizeof(extmap_value), "%d%s %s", idx, direction_str,
1455 ast_rtp_instance_extmap_get_uri(session_media->rtp, idx));
1456 attr = pjmedia_sdp_attr_create(pool, "extmap", pj_cstr(&stmp, extmap_value));
1457 pjmedia_sdp_attr_add(&media->attr_count, media->attr, attr);
1458 }
1459}
@ 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:927
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:938
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:974
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:958
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 733 of file res_pjsip_sdp_rtp.c.

735{
736 struct ast_rtp_engine_ice *ice;
737 struct ao2_container *candidates;
738 const char *username, *password;
739 pj_str_t stmp;
740 pjmedia_sdp_attr *attr;
741 struct ao2_iterator it_candidates;
742 struct ast_rtp_engine_ice_candidate *candidate;
743
744 if (!session->endpoint->media.rtp.ice_support || !(ice = ast_rtp_instance_get_ice(session_media->rtp))) {
745 return;
746 }
747
748 if (!session_media->remote_ice) {
749 ice->stop(session_media->rtp);
750 return;
751 }
752
753 if ((username = ice->get_ufrag(session_media->rtp))) {
754 attr = pjmedia_sdp_attr_create(pool, "ice-ufrag", pj_cstr(&stmp, username));
755 media->attr[media->attr_count++] = attr;
756 }
757
758 if ((password = ice->get_password(session_media->rtp))) {
759 attr = pjmedia_sdp_attr_create(pool, "ice-pwd", pj_cstr(&stmp, password));
760 media->attr[media->attr_count++] = attr;
761 }
762
763 if (!include_candidates) {
764 return;
765 }
766
767 candidates = ice->get_local_candidates(session_media->rtp);
768 if (!candidates) {
769 return;
770 }
771
772 it_candidates = ao2_iterator_init(candidates, 0);
773 for (; (candidate = ao2_iterator_next(&it_candidates)); ao2_ref(candidate, -1)) {
774 struct ast_str *attr_candidate = ast_str_create(128);
775
776 ast_str_set(&attr_candidate, -1, "%s %u %s %d %s ", candidate->foundation, candidate->id, candidate->transport,
777 candidate->priority, ast_sockaddr_stringify_addr_remote(&candidate->address));
778 ast_str_append(&attr_candidate, -1, "%s typ ", ast_sockaddr_stringify_port(&candidate->address));
779
780 switch (candidate->type) {
782 ast_str_append(&attr_candidate, -1, "host");
783 break;
785 ast_str_append(&attr_candidate, -1, "srflx");
786 break;
788 ast_str_append(&attr_candidate, -1, "relay");
789 break;
790 }
791
792 if (!ast_sockaddr_isnull(&candidate->relay_address)) {
793 ast_str_append(&attr_candidate, -1, " raddr %s rport", ast_sockaddr_stringify_addr_remote(&candidate->relay_address));
794 ast_str_append(&attr_candidate, -1, " %s", ast_sockaddr_stringify_port(&candidate->relay_address));
795 }
796
797 attr = pjmedia_sdp_attr_create(pool, "candidate", pj_cstr(&stmp, ast_str_buffer(attr_candidate)));
798 media->attr[media->attr_count++] = attr;
799
800 ast_free(attr_candidate);
801 }
802
803 ao2_iterator_destroy(&it_candidates);
804 ao2_ref(candidates, -1);
805}
#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 1330 of file res_pjsip_sdp_rtp.c.

1333{
1334 pj_str_t stmp;
1335 pjmedia_sdp_attr *attr;
1336 char msid[(AST_UUID_STR_LEN * 2) + 2];
1337 const char *stream_label = ast_stream_get_metadata(stream, "SDP:LABEL");
1338
1339 if (!session->endpoint->media.webrtc) {
1340 return;
1341 }
1342
1343 if (ast_strlen_zero(session_media->mslabel)) {
1344 /* If this stream is grouped with another then use its media stream label if possible */
1345 if (ast_stream_get_group(stream) != -1) {
1346 struct ast_sip_session_media *group_session_media = AST_VECTOR_GET(&session->pending_media_state->sessions, ast_stream_get_group(stream));
1347
1348 ast_copy_string(session_media->mslabel, group_session_media->mslabel, sizeof(session_media->mslabel));
1349 }
1350
1351 if (ast_strlen_zero(session_media->mslabel)) {
1352 ast_uuid_generate_str(session_media->mslabel, sizeof(session_media->mslabel));
1353 }
1354 }
1355
1356 if (ast_strlen_zero(session_media->label)) {
1357 ast_uuid_generate_str(session_media->label, sizeof(session_media->label));
1358 /* add for stream identification to replace stream_name */
1359 ast_stream_set_metadata(stream, "MSID:LABEL", session_media->label);
1360 }
1361
1362 snprintf(msid, sizeof(msid), "%s %s", session_media->mslabel, session_media->label);
1363 ast_debug(3, "Stream msid: %p %s %s\n", stream,
1365 attr = pjmedia_sdp_attr_create(pool, "msid", pj_cstr(&stmp, msid));
1366 pjmedia_sdp_attr_add(&media->attr_count, media->attr, attr);
1367
1368 /* 'label' must come after 'msid' */
1369 if (!ast_strlen_zero(stream_label)) {
1370 ast_debug(3, "Stream Label: %p %s %s\n", stream,
1371 ast_codec_media_type2str(ast_stream_get_type(stream)), stream_label);
1372 attr = pjmedia_sdp_attr_create(pool, "label", pj_cstr(&stmp, stream_label));
1373 pjmedia_sdp_attr_add(&media->attr_count, media->attr, attr);
1374 }
1375}
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 1377 of file res_pjsip_sdp_rtp.c.

1379{
1380 pj_str_t stmp;
1381 pjmedia_sdp_attr *attr;
1382
1383 if (!session->endpoint->media.webrtc) {
1384 return;
1385 }
1386
1387 /* transport-cc is supposed to be for the entire transport, and any media sources so
1388 * while the header does not appear in audio streams and isn't negotiated there, we still
1389 * place this attribute in as Chrome does.
1390 */
1391 attr = pjmedia_sdp_attr_create(pool, "rtcp-fb", pj_cstr(&stmp, "* transport-cc"));
1392 pjmedia_sdp_attr_add(&media->attr_count, media->attr, attr);
1393
1394 if (session_media->type != AST_MEDIA_TYPE_VIDEO) {
1395 return;
1396 }
1397
1398 /*
1399 * For now just automatically add it the stream even though it hasn't
1400 * necessarily been negotiated.
1401 */
1402 attr = pjmedia_sdp_attr_create(pool, "rtcp-fb", pj_cstr(&stmp, "* ccm fir"));
1403 pjmedia_sdp_attr_add(&media->attr_count, media->attr, attr);
1404
1405 attr = pjmedia_sdp_attr_create(pool, "rtcp-fb", pj_cstr(&stmp, "* goog-remb"));
1406 pjmedia_sdp_attr_add(&media->attr_count, media->attr, attr);
1407
1408 attr = pjmedia_sdp_attr_create(pool, "rtcp-fb", pj_cstr(&stmp, "* nack"));
1409 pjmedia_sdp_attr_add(&media->attr_count, media->attr, attr);
1410}

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 1260 of file res_pjsip_sdp_rtp.c.

1261{
1262 pj_str_t stmp;
1263 pjmedia_sdp_attr *attr;
1264 char tmp[128];
1265
1266 if (!session->endpoint->media.bundle || session_media->bundle_group == -1) {
1267 return;
1268 }
1269
1270 snprintf(tmp, sizeof(tmp), "%u cname:%s", ast_rtp_instance_get_ssrc(session_media->rtp), ast_rtp_instance_get_cname(session_media->rtp));
1271 attr = pjmedia_sdp_attr_create(pool, "ssrc", pj_cstr(&stmp, tmp));
1272 media->attr[media->attr_count++] = attr;
1273}
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 428 of file res_pjsip_sdp_rtp.c.

431{
432 if (!joint) {
433 return -1;
434 }
435
436 ast_stream_set_formats(asterisk_stream, joint);
437
438 /* If this is a bundled stream then apply the payloads to RTP instance acting as transport to prevent conflicts */
439 if (session_media_transport != session_media && session_media->bundled) {
440 int index;
441
442 for (index = 0; index < ast_format_cap_count(joint); ++index) {
443 struct ast_format *format = ast_format_cap_get_format(joint, index);
444 int rtp_code;
445
446 /* Ensure this payload is in the bundle group transport codecs, this purposely doesn't check the return value for
447 * things as the format is guaranteed to have a payload already.
448 */
449 rtp_code = ast_rtp_codecs_payload_code(ast_rtp_instance_get_codecs(session_media->rtp), 1, format, 0);
450 ast_rtp_codecs_payload_set_rx(ast_rtp_instance_get_codecs(session_media_transport->rtp), rtp_code, format);
451
452 ao2_ref(format, -1);
453 }
454 }
455
456 return 0;
457}
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:755
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 1108 of file res_pjsip_sdp_rtp.c.

1110{
1111 struct ast_rtp_engine_dtls *dtls = ast_rtp_instance_get_dtls(session_media->rtp);
1112 pj_str_t *value;
1113
1114 if (!attr->value.ptr || !dtls) {
1115 return;
1116 }
1117
1118 value = pj_strtrim(&attr->value);
1119
1120 if (!pj_strcmp2(&attr->name, "setup")) {
1121 if (!pj_stricmp2(value, "active")) {
1122 dtls->set_setup(session_media->rtp, AST_RTP_DTLS_SETUP_ACTIVE);
1123 } else if (!pj_stricmp2(value, "passive")) {
1124 dtls->set_setup(session_media->rtp, AST_RTP_DTLS_SETUP_PASSIVE);
1125 } else if (!pj_stricmp2(value, "actpass")) {
1126 dtls->set_setup(session_media->rtp, AST_RTP_DTLS_SETUP_ACTPASS);
1127 } else if (!pj_stricmp2(value, "holdconn")) {
1128 dtls->set_setup(session_media->rtp, AST_RTP_DTLS_SETUP_HOLDCONN);
1129 } else {
1130 ast_log(LOG_WARNING, "Unsupported setup attribute value '%*s'\n", (int)value->slen, value->ptr);
1131 }
1132 } else if (!pj_strcmp2(&attr->name, "connection")) {
1133 if (!pj_stricmp2(value, "new")) {
1134 dtls->reset(session_media->rtp);
1135 } else if (!pj_stricmp2(value, "existing")) {
1136 /* Do nothing */
1137 } else {
1138 ast_log(LOG_WARNING, "Unsupported connection attribute value '%*s'\n", (int)value->slen, value->ptr);
1139 }
1140 } else if (!pj_strcmp2(&attr->name, "fingerprint")) {
1141 char hash_value[256], hash[32];
1142 char fingerprint_text[value->slen + 1];
1143 ast_copy_pj_str(fingerprint_text, value, sizeof(fingerprint_text));
1144 if (sscanf(fingerprint_text, "%31s %255s", hash, hash_value) == 2) {
1145 if (!strcasecmp(hash, "sha-1")) {
1146 dtls->set_fingerprint(session_media->rtp, AST_RTP_DTLS_HASH_SHA1, hash_value);
1147 } else if (!strcasecmp(hash, "sha-256")) {
1148 dtls->set_fingerprint(session_media->rtp, AST_RTP_DTLS_HASH_SHA256, hash_value);
1149 } else {
1150 ast_log(LOG_WARNING, "Unsupported fingerprint hash type '%s'\n",
1151 hash);
1152 }
1153 }
1154 }
1155}
#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:2201
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 2216 of file res_pjsip_sdp_rtp.c.

2219{
2220 RAII_VAR(struct ast_sockaddr *, addrs, NULL, ast_free);
2221 struct pjmedia_sdp_media *remote_stream = remote->media[index];
2222 enum ast_media_type media_type = session_media->type;
2223 char host[NI_MAXHOST];
2224 int res;
2225 int rtp_timeout;
2226 struct ast_sip_session_media *session_media_transport;
2227 SCOPE_ENTER(1, "%s Stream: %s\n", ast_sip_session_get_name(session),
2228 ast_str_tmp(128, ast_stream_to_str(asterisk_stream, &STR_TMP)));
2229
2230 if (!session->channel) {
2231 SCOPE_EXIT_RTN_VALUE(1, "No channel\n");
2232 }
2233
2234 /* Ensure incoming transport is compatible with the endpoint's configuration */
2235 if (!session->endpoint->media.rtp.use_received_transport &&
2237 SCOPE_EXIT_RTN_VALUE(-1, "Incompatible transport\n");
2238 }
2239
2240 /* Create an RTP instance if need be */
2241 if (!session_media->rtp && create_rtp(session, session_media, local)) {
2242 SCOPE_EXIT_RTN_VALUE(-1, "Couldn't create rtp\n");
2243 }
2244
2245 process_ssrc_attributes(session, session_media, remote_stream);
2246 process_extmap_attributes(session, session_media, remote_stream);
2247
2248 session_media_transport = ast_sip_session_media_get_transport(session, session_media);
2249
2250 if (session_media_transport == session_media || !session_media->bundled) {
2251 session_media->remote_rtcp_mux = (pjmedia_sdp_media_find_attr2(remote_stream, "rtcp-mux", NULL) != NULL);
2252 set_ice_components(session, session_media);
2253
2254 enable_rtcp(session, session_media, remote_stream);
2255
2256 res = setup_media_encryption(session, session_media, remote, remote_stream);
2257 if (!session->endpoint->media.rtp.encryption_optimistic && res) {
2258 /* If optimistic encryption is disabled and crypto should have been enabled but was not
2259 * this session must fail.
2260 */
2261 SCOPE_EXIT_RTN_VALUE(-1, "Incompatible crypto\n");
2262 }
2263
2264 if (!remote_stream->conn && !remote->conn) {
2265 SCOPE_EXIT_RTN_VALUE(1, "No connection info\n");
2266 }
2267
2268 ast_copy_pj_str(host, remote_stream->conn ? &remote_stream->conn->addr : &remote->conn->addr, sizeof(host));
2269
2270 /* Ensure that the address provided is valid */
2271 if (ast_sockaddr_resolve(&addrs, host, PARSE_PORT_FORBID, AST_AF_UNSPEC) <= 0) {
2272 /* The provided host was actually invalid so we error out this negotiation */
2273 SCOPE_EXIT_RTN_VALUE(-1, "Host invalid\n");
2274 }
2275
2276 /* Apply connection information to the RTP instance */
2277 ast_sockaddr_set_port(addrs, remote_stream->desc.port);
2278 ast_rtp_instance_set_remote_address(session_media->rtp, addrs);
2279
2283 if (!session->endpoint->media.rtcp_mux || !session_media->remote_rtcp_mux) {
2286 }
2287
2288 /* If ICE support is enabled find all the needed attributes */
2289 process_ice_attributes(session, session_media, remote, remote_stream);
2290 } else {
2291 /* This is bundled with another session, so mark it as such */
2292 ast_rtp_instance_bundle(session_media->rtp, session_media_transport->rtp);
2294 enable_rtcp(session, session_media, remote_stream);
2295 }
2296
2297 if (set_caps(session, session_media, session_media_transport, remote_stream, 0, asterisk_stream)) {
2298 SCOPE_EXIT_RTN_VALUE(-1, "set_caps failed\n");
2299 }
2300
2301 /* Set the channel uniqueid on the RTP instance now that it is becoming active */
2302 ast_channel_lock(session->channel);
2304 ast_channel_unlock(session->channel);
2305
2306 /* Ensure the RTP instance is active */
2307 ast_rtp_instance_set_stream_num(session_media->rtp, ast_stream_get_position(asterisk_stream));
2308 ast_rtp_instance_activate(session_media->rtp);
2309
2310 /* audio stream handles music on hold */
2311 if (media_type != AST_MEDIA_TYPE_AUDIO && media_type != AST_MEDIA_TYPE_VIDEO) {
2312 if ((pjmedia_sdp_neg_was_answer_remote(session->inv_session->neg) == PJ_FALSE)
2313 && (session->inv_session->state == PJSIP_INV_STATE_CONFIRMED)) {
2315 }
2316 SCOPE_EXIT_RTN_VALUE(1, "moh\n");
2317 }
2318
2319 set_session_media_remotely_held(session_media, session, remote_stream, asterisk_stream, addrs);
2320
2321 if (session_media->remotely_held_changed) {
2322 if (session_media->remotely_held) {
2323 /* The remote side has put us on hold */
2324 if (!session->endpoint->suppress_moh_on_sendonly) {
2325 ast_queue_hold(session->channel, session->endpoint->mohsuggest);
2326 ast_rtp_instance_stop(session_media->rtp);
2328 }
2329 session_media->remotely_held_changed = 0;
2330 } else {
2331 /* The remote side has taken us off hold */
2332 if (!session->endpoint->suppress_moh_on_sendonly) {
2333 ast_queue_unhold(session->channel);
2335 }
2336 session_media->remotely_held_changed = 0;
2337 }
2338 } else if ((pjmedia_sdp_neg_was_answer_remote(session->inv_session->neg) == PJ_FALSE)
2339 && (session->inv_session->state == PJSIP_INV_STATE_CONFIRMED)) {
2341 }
2342
2343 /* This purposely resets the encryption to the configured in case it gets added later */
2344 session_media->encryption = session->endpoint->media.rtp.encryption;
2345
2346 if (session->endpoint->media.rtp.keepalive > 0 &&
2347 (session_media->type == AST_MEDIA_TYPE_AUDIO ||
2348 session_media->type == AST_MEDIA_TYPE_VIDEO)) {
2349 ast_rtp_instance_set_keepalive(session_media->rtp, session->endpoint->media.rtp.keepalive);
2350 /* Schedule the initial keepalive early in case this is being used to punch holes through
2351 * a NAT. This way there won't be an awkward delay before media starts flowing in some
2352 * scenarios.
2353 */
2354 AST_SCHED_DEL(sched, session_media->keepalive_sched_id);
2356 session_media, 1);
2357 }
2358
2359 /* As the channel lock is not held during this process the scheduled item won't block if
2360 * it is hanging up the channel at the same point we are applying this negotiated SDP.
2361 */
2362 AST_SCHED_DEL(sched, session_media->timeout_sched_id);
2363
2364 /* Due to the fact that we only ever have one scheduled timeout item for when we are both
2365 * off hold and on hold we don't need to store the two timeouts differently on the RTP
2366 * instance itself.
2367 */
2368 ast_rtp_instance_set_timeout(session_media->rtp, 0);
2369 if (session->endpoint->media.rtp.timeout && !session_media->remotely_held && !session_media->locally_held) {
2370 ast_rtp_instance_set_timeout(session_media->rtp, session->endpoint->media.rtp.timeout);
2371 } else if (session->endpoint->media.rtp.timeout_hold && (session_media->remotely_held || session_media->locally_held)) {
2372 ast_rtp_instance_set_timeout(session_media->rtp, session->endpoint->media.rtp.timeout_hold);
2373 }
2374
2375 rtp_timeout = ast_rtp_instance_get_timeout(session_media->rtp);
2376
2377 if (rtp_timeout) {
2378 session_media->timeout_sched_id = ast_sched_add_variable(sched, rtp_timeout*1000, rtp_check_timeout,
2379 session_media, 1);
2380 }
2381
2382 SCOPE_EXIT_RTN_VALUE(1, "Handled\n");
2383}
static PGconn * conn
Definition cdr_pgsql.c:87
#define ast_channel_lock(chan)
Definition channel.h:2982
int ast_queue_control(struct ast_channel *chan, enum ast_control_frame_type control)
Queue a control frame without payload.
Definition channel.c:1288
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:1170
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:1273
int ast_queue_hold(struct ast_channel *chan, const char *musicclass)
Queue a hold frame.
Definition channel.c:1248
#define ast_channel_unlock(chan)
Definition channel.h:2983
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:581
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 2549 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 2386 of file res_pjsip_sdp_rtp.c.

2387{
2389 char host[NI_MAXHOST];
2390 struct ast_sockaddr our_sdp_addr = { { 0, } };
2391
2392 /* If the stream has been rejected there will be no connection line */
2393 if (!stream->conn || !transport_state) {
2394 return;
2395 }
2396
2397 ast_copy_pj_str(host, &stream->conn->addr, sizeof(host));
2398 ast_sockaddr_parse(&our_sdp_addr, host, PARSE_PORT_FORBID);
2399
2400 /* Reversed check here. We don't check the remote endpoint being
2401 * in our local net, but whether our outgoing session IP is
2402 * local. If it is not, we won't do rewriting. No localnet
2403 * configured? Always rewrite. */
2404 if (ast_sip_transport_is_nonlocal(transport_state, &our_sdp_addr) && transport_state->localnet) {
2405 return;
2406 }
2407 ast_debug(5, "Setting media address to %s\n", ast_sockaddr_stringify_addr_remote(&transport_state->external_media_address));
2408 pj_strdup2(tdata->pool, &stream->conn->addr, ast_sockaddr_stringify_addr_remote(&transport_state->external_media_address));
2409}
#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 1028 of file res_pjsip_sdp_rtp.c.

1031{
1032 enum ast_sip_session_media_encryption incoming_encryption;
1033 char transport_end = stream->desc.transport.ptr[stream->desc.transport.slen - 1];
1034 unsigned int optimistic;
1035
1036 if ((transport_end == 'F' && !endpoint->media.rtp.use_avpf)
1037 || (transport_end != 'F' && endpoint->media.rtp.use_avpf)) {
1039 }
1040
1041 incoming_encryption = get_media_encryption_type(stream->desc.transport, stream, &optimistic);
1042
1043 if (incoming_encryption == endpoint->media.rtp.encryption) {
1044 return incoming_encryption;
1045 }
1046
1047 if (endpoint->media.rtp.force_avp ||
1048 endpoint->media.rtp.encryption_optimistic) {
1049 return incoming_encryption;
1050 }
1051
1052 /* If an optimistic offer has been made but encryption is not enabled consider it as having
1053 * no offer of crypto at all instead of invalid so the session proceeds.
1054 */
1055 if (optimistic) {
1057 }
1058
1060}
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:1003
struct ast_sip_endpoint_media_configuration media
Definition res_pjsip.h:1084
enum ast_sip_session_media_encryption encryption
Definition res_pjsip.h:946

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 808 of file res_pjsip_sdp_rtp.c.

810{
811 struct ast_rtp_engine_ice *ice;
812 const pjmedia_sdp_attr *attr;
813 unsigned int attr_i;
814
815 if (!session->endpoint->media.rtp.ice_support || !(ice = ast_rtp_instance_get_ice(session_media->rtp))) {
816 session_media->remote_ice = 0;
817 return;
818 }
819
820 /* Find all of the candidates */
821 for (attr_i = 0; attr_i < remote_stream->attr_count; ++attr_i) {
822 attr = remote_stream->attr[attr_i];
823 if (!pj_strcmp2(&attr->name, "candidate")) {
824 session_media->remote_ice = 1;
825 break;
826 }
827 }
828
829 if (attr_i == remote_stream->attr_count) {
830 session_media->remote_ice = 0;
831 }
832}
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 1778 of file res_pjsip_sdp_rtp.c.

1780{
1781 pj_pool_t *pool = session->inv_session->pool_prov;
1782 static const pj_str_t STR_RTP_AVP = { "RTP/AVP", 7 };
1783 static const pj_str_t STR_IN = { "IN", 2 };
1784 static const pj_str_t STR_IP4 = { "IP4", 3};
1785 static const pj_str_t STR_IP6 = { "IP6", 3};
1786 static const pj_str_t STR_SENDRECV = { "sendrecv", 8 };
1787 static const pj_str_t STR_SENDONLY = { "sendonly", 8 };
1788 static const pj_str_t STR_INACTIVE = { "inactive", 8 };
1789 static const pj_str_t STR_RECVONLY = { "recvonly", 8 };
1790 pjmedia_sdp_media *media;
1791 const char *hostip = NULL;
1792 struct ast_sockaddr addr;
1793 char tmp[512];
1794 pj_str_t stmp;
1795 pjmedia_sdp_attr *attr;
1796 int index = 0;
1797 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;
1798 int min_packet_size = 0, max_packet_size = 0;
1799 int rtp_code;
1800 RAII_VAR(struct ast_format_cap *, caps, NULL, ao2_cleanup);
1801 enum ast_media_type media_type = session_media->type;
1802 struct ast_sip_session_media *session_media_transport;
1803 pj_sockaddr ip;
1804 int direct_media_enabled = !ast_sockaddr_isnull(&session_media->direct_media_addr) &&
1805 ast_format_cap_count(session->direct_media_cap);
1806
1807 /* Keep track of the sample rates for offered codecs so we can build matching
1808 RFC 2833/4733 payload offers. */
1809 AST_VECTOR(, int) sample_rates;
1810 /* In case we can't init the sample rates, still try to do the rest. */
1811 int build_dtmf_sample_rates = 1;
1812
1813 SCOPE_ENTER(1, "%s Type: %s %s\n", ast_sip_session_get_name(session),
1814 ast_codec_media_type2str(media_type), ast_str_tmp(128, ast_stream_to_str(stream, &STR_TMP)));
1815
1816 media = pj_pool_zalloc(pool, sizeof(struct pjmedia_sdp_media));
1817 if (!media) {
1818 SCOPE_EXIT_RTN_VALUE(-1, "Pool alloc failure\n");
1819 }
1820 pj_strdup2(pool, &media->desc.media, ast_codec_media_type2str(session_media->type));
1821
1822 /* If this is a removed (or declined) stream OR if no formats exist then construct a minimal stream in SDP */
1825 media->desc.port = 0;
1826 media->desc.port_count = 1;
1827
1828 if (remote && remote->media[ast_stream_get_position(stream)]) {
1829 pjmedia_sdp_media *remote_media = remote->media[ast_stream_get_position(stream)];
1830 int index;
1831
1832 media->desc.transport = remote_media->desc.transport;
1833
1834 /* Preserve existing behavior by copying the formats provided from the offer */
1835 for (index = 0; index < remote_media->desc.fmt_count; ++index) {
1836 media->desc.fmt[index] = remote_media->desc.fmt[index];
1837 }
1838 media->desc.fmt_count = remote_media->desc.fmt_count;
1839 } else {
1840 /* This is actually an offer so put a dummy payload in that is ignored and sane transport */
1841 media->desc.transport = STR_RTP_AVP;
1842 pj_strdup2(pool, &media->desc.fmt[media->desc.fmt_count++], "32");
1843 }
1844
1845 sdp->media[sdp->media_count++] = media;
1847
1848 SCOPE_EXIT_RTN_VALUE(1, "Stream removed or no formats\n");
1849 }
1850
1851 if (!session_media->rtp && create_rtp(session, session_media, sdp)) {
1852 SCOPE_EXIT_RTN_VALUE(-1, "Couldn't create rtp\n");
1853 }
1854
1855 /* If this stream has not been bundled already it is new and we need to ensure there is no SSRC conflict */
1856 if (session_media->bundle_group != -1 && !session_media->bundled) {
1857 for (index = 0; index < sdp->media_count; ++index) {
1858 struct ast_sip_session_media *other_session_media;
1859
1860 other_session_media = AST_VECTOR_GET(&session->pending_media_state->sessions, index);
1861 if (!other_session_media->rtp || other_session_media->bundle_group != session_media->bundle_group) {
1862 continue;
1863 }
1864
1865 if (ast_rtp_instance_get_ssrc(session_media->rtp) == ast_rtp_instance_get_ssrc(other_session_media->rtp)) {
1866 ast_rtp_instance_change_source(session_media->rtp);
1867 /* Start the conflict check over again */
1868 index = -1;
1869 continue;
1870 }
1871 }
1872 }
1873
1874 session_media_transport = ast_sip_session_media_get_transport(session, session_media);
1875
1876 if (session_media_transport == session_media || !session_media->bundled) {
1877 set_ice_components(session, session_media);
1878 enable_rtcp(session, session_media, NULL);
1879
1880 /* Crypto has to be added before setting the media transport so that SRTP is properly
1881 * set up according to the configuration. This ends up changing the media transport.
1882 */
1883 if (add_crypto_to_stream(session, session_media, pool, media)) {
1884 SCOPE_EXIT_RTN_VALUE(-1, "Couldn't add crypto\n");
1885 }
1886
1887 if (pj_strlen(&session_media->transport)) {
1888 /* If a transport has already been specified use it */
1889 media->desc.transport = session_media->transport;
1890 } else {
1891 media->desc.transport = pj_str(ast_sdp_get_rtp_profile(
1892 /* Optimistic encryption places crypto in the normal RTP/AVP profile */
1893 !session->endpoint->media.rtp.encryption_optimistic &&
1894 (session_media->encryption == AST_SIP_MEDIA_ENCRYPT_SDES),
1895 session_media->rtp, session->endpoint->media.rtp.use_avpf,
1896 session->endpoint->media.rtp.force_avp));
1897 }
1898
1899 media->conn = pj_pool_zalloc(pool, sizeof(struct pjmedia_sdp_conn));
1900 if (!media->conn) {
1901 SCOPE_EXIT_RTN_VALUE(-1, "Pool alloc failure\n");
1902 }
1903
1904 /* Add connection level details */
1905 if (direct_media_enabled) {
1907 } else if (ast_strlen_zero(session->endpoint->media.address)) {
1908 hostip = ast_sip_get_host_ip_string(session->endpoint->media.rtp.ipv6 ? pj_AF_INET6() : pj_AF_INET());
1909 } else {
1910 hostip = session->endpoint->media.address;
1911 }
1912
1913 if (ast_strlen_zero(hostip)) {
1914 ast_log(LOG_ERROR, "No local host IP available for stream %s\n",
1915 ast_codec_media_type2str(session_media->type));
1916 SCOPE_EXIT_RTN_VALUE(-1, "No local host ip\n");
1917 }
1918
1919 media->conn->net_type = STR_IN;
1920 /* Assume that the connection will use IPv4 until proven otherwise */
1921 media->conn->addr_type = STR_IP4;
1922 pj_strdup2(pool, &media->conn->addr, hostip);
1923
1924 if ((pj_sockaddr_parse(pj_AF_UNSPEC(), 0, &media->conn->addr, &ip) == PJ_SUCCESS) &&
1925 (ip.addr.sa_family == pj_AF_INET6())) {
1926 media->conn->addr_type = STR_IP6;
1927 }
1928
1929 /* Add ICE attributes and candidates */
1930 add_ice_to_stream(session, session_media, pool, media, 1);
1931
1932 ast_rtp_instance_get_local_address(session_media->rtp, &addr);
1933 media->desc.port = direct_media_enabled ? ast_sockaddr_port(&session_media->direct_media_addr) : (pj_uint16_t) ast_sockaddr_port(&addr);
1934 media->desc.port_count = 1;
1935 } else {
1936 pjmedia_sdp_media *bundle_group_stream = sdp->media[session_media_transport->stream_num];
1937
1938 /* As this is in a bundle group it shares the same details as the group instance */
1939 media->desc.transport = bundle_group_stream->desc.transport;
1940 media->conn = bundle_group_stream->conn;
1941 media->desc.port = bundle_group_stream->desc.port;
1942
1943 if (add_crypto_to_stream(session, session_media_transport, pool, media)) {
1944 SCOPE_EXIT_RTN_VALUE(-1, "Couldn't add crypto\n");
1945 }
1946
1947 add_ice_to_stream(session, session_media_transport, pool, media, 0);
1948
1949 enable_rtcp(session, session_media, NULL);
1950 }
1951
1953 ast_log(LOG_ERROR, "Failed to allocate %s capabilities\n",
1954 ast_codec_media_type2str(session_media->type));
1955 SCOPE_EXIT_RTN_VALUE(-1, "Couldn't create caps\n");
1956 }
1957
1958 if (direct_media_enabled) {
1959 ast_format_cap_get_compatible(session->endpoint->media.codecs, session->direct_media_cap, caps);
1960 } else {
1961 ast_format_cap_append_from_cap(caps, ast_stream_get_formats(stream), media_type);
1962 }
1963
1964 /* Init the sample rates before we start adding them. Assume we will have at least one. */
1965 if (AST_VECTOR_INIT(&sample_rates, 1)) {
1966 ast_log(LOG_ERROR, "Unable to add dtmf formats to SDP!\n");
1967 build_dtmf_sample_rates = 0;
1968 }
1969
1970 for (index = 0; index < ast_format_cap_count(caps); ++index) {
1971 struct ast_format *format = ast_format_cap_get_format(caps, index);
1972
1973 if (ast_format_get_type(format) != media_type) {
1974 ao2_ref(format, -1);
1975 continue;
1976 }
1977
1978 /* It is possible for some formats not to have SDP information available for them
1979 * and if this is the case, skip over them so the SDP can still be created.
1980 */
1981 if (!ast_rtp_lookup_sample_rate2(1, format, 0)) {
1982 ast_log(LOG_WARNING, "Format '%s' can not be added to SDP, consider disallowing it on endpoint '%s'\n",
1984 ao2_ref(format, -1);
1985 continue;
1986 }
1987
1988 /* If this stream is not a transport we need to use the transport codecs structure for payload management to prevent
1989 * conflicts.
1990 */
1991 if (session_media_transport != session_media) {
1992 if ((rtp_code = ast_rtp_codecs_payload_code(ast_rtp_instance_get_codecs(session_media_transport->rtp), 1, format, 0)) == -1) {
1993 ast_log(LOG_WARNING,"Unable to get rtp codec payload code for %s\n", ast_format_get_name(format));
1994 ao2_ref(format, -1);
1995 continue;
1996 }
1997 /* Our instance has to match the payload number though */
1998 ast_rtp_codecs_payload_set_rx(ast_rtp_instance_get_codecs(session_media->rtp), rtp_code, format);
1999 } else {
2000 if ((rtp_code = ast_rtp_codecs_payload_code(ast_rtp_instance_get_codecs(session_media->rtp), 1, format, 0)) == -1) {
2001 ast_log(LOG_WARNING,"Unable to get rtp codec payload code for %s\n", ast_format_get_name(format));
2002 ao2_ref(format, -1);
2003 continue;
2004 }
2005 }
2006
2007 if ((attr = generate_rtpmap_attr(session, media, pool, rtp_code, 1, format, 0))) {
2008 int i, added = 0;
2009 int newrate = ast_rtp_lookup_sample_rate2(1, format, 0);
2010 if (build_dtmf_sample_rates) {
2011 for (i = 0; i < AST_VECTOR_SIZE(&sample_rates); i++) {
2012 /* Only add if we haven't already processed this sample rate. For instance
2013 A-law and u-law 'share' one 8K DTMF payload type. */
2014 if (newrate == AST_VECTOR_GET(&sample_rates, i)) {
2015 added = 1;
2016 break;
2017 }
2018 }
2019
2020 if (!added) {
2021 AST_VECTOR_APPEND(&sample_rates, newrate);
2022 }
2023 }
2024 media->attr[media->attr_count++] = attr;
2025 }
2026
2027 if ((attr = generate_fmtp_attr(pool, format, rtp_code))) {
2028 media->attr[media->attr_count++] = attr;
2029 }
2030
2031 if (ast_format_get_maximum_ms(format) &&
2032 ((ast_format_get_maximum_ms(format) < max_packet_size) || !max_packet_size)) {
2033 max_packet_size = ast_format_get_maximum_ms(format);
2034 }
2035 ao2_ref(format, -1);
2036
2037 if (media->desc.fmt_count == PJMEDIA_MAX_SDP_FMT) {
2038 break;
2039 }
2040 }
2041
2042 /* Add non-codec formats */
2044 && media->desc.fmt_count < PJMEDIA_MAX_SDP_FMT) {
2045 for (index = 1LL; index <= AST_RTP_MAX; index <<= 1) {
2046 if (!(noncodec & index)) {
2047 continue;
2048 }
2049
2050 if (index != AST_RTP_DTMF) {
2051 rtp_code = ast_rtp_codecs_payload_code(
2052 ast_rtp_instance_get_codecs(session_media->rtp), 0, NULL, index);
2053 if (rtp_code == -1) {
2054 continue;
2055 } else if ((attr = generate_rtpmap_attr(session, media, pool, rtp_code, 0, NULL, index))) {
2056 media->attr[media->attr_count++] = attr;
2057 }
2058 } else if (build_dtmf_sample_rates) {
2059 /*
2060 * Walk through the possible bitrates for the RFC 2833/4733 digits and generate the rtpmap
2061 * attributes.
2062 */
2063 int i, found_default_offer = 0;
2064 for (i = 0; i < AST_VECTOR_SIZE(&sample_rates); i++) {
2066 ast_rtp_instance_get_codecs(session_media->rtp), 0, NULL, index, AST_VECTOR_GET(&sample_rates, i));
2067
2068 if (rtp_code == -1) {
2069 continue;
2070 }
2071
2072 if (AST_VECTOR_GET(&sample_rates, i) == DEFAULT_DTMF_SAMPLE_RATE_MS) {
2073 /* we found and added a default offer, so no need to include a default one.*/
2074 found_default_offer = 1;
2075 }
2076
2077 if ((attr = generate_rtpmap_attr2(session, media, pool, rtp_code, 0, NULL, index, AST_VECTOR_GET(&sample_rates, i)))) {
2078 media->attr[media->attr_count++] = attr;
2079 snprintf(tmp, sizeof(tmp), "%d 0-16", (rtp_code));
2080 attr = pjmedia_sdp_attr_create(pool, "fmtp", pj_cstr(&stmp, tmp));
2081 media->attr[media->attr_count++] = attr;
2082 }
2083 }
2084
2085 /* If we weren't able to add any matching RFC 2833/4733, assume this endpoint is using a
2086 * mismatched 8K offer and try to add one as a fall-back/default.
2087 */
2088 if (!found_default_offer) {
2091
2092 if (rtp_code != -1 && (attr = generate_rtpmap_attr2(session, media, pool, rtp_code, 0, NULL, index, DEFAULT_DTMF_SAMPLE_RATE_MS))) {
2093 media->attr[media->attr_count++] = attr;
2094 snprintf(tmp, sizeof(tmp), "%d 0-16", (rtp_code));
2095 attr = pjmedia_sdp_attr_create(pool, "fmtp", pj_cstr(&stmp, tmp));
2096 media->attr[media->attr_count++] = attr;
2097 }
2098 }
2099 }
2100
2101 if (media->desc.fmt_count == PJMEDIA_MAX_SDP_FMT) {
2102 break;
2103 }
2104 }
2105 }
2106
2107 /* we are done with the sample rates */
2108 AST_VECTOR_FREE(&sample_rates);
2109
2110 /* If no formats were actually added to the media stream don't add it to the SDP */
2111 if (!media->desc.fmt_count) {
2112 SCOPE_EXIT_RTN_VALUE(1, "No formats added to stream\n");
2113 }
2114
2115 /* If ptime is set add it as an attribute */
2116 min_packet_size = ast_rtp_codecs_get_framing(ast_rtp_instance_get_codecs(session_media->rtp));
2117 if (!min_packet_size) {
2118 min_packet_size = ast_format_cap_get_framing(caps);
2119 }
2120 if (min_packet_size) {
2121 snprintf(tmp, sizeof(tmp), "%d", min_packet_size);
2122 attr = pjmedia_sdp_attr_create(pool, "ptime", pj_cstr(&stmp, tmp));
2123 media->attr[media->attr_count++] = attr;
2124 }
2125
2126 if (max_packet_size) {
2127 snprintf(tmp, sizeof(tmp), "%d", max_packet_size);
2128 attr = pjmedia_sdp_attr_create(pool, "maxptime", pj_cstr(&stmp, tmp));
2129 media->attr[media->attr_count++] = attr;
2130 }
2131
2132 attr = PJ_POOL_ZALLOC_T(pool, pjmedia_sdp_attr);
2133 if (session_media->locally_held) {
2134 if (session_media->remotely_held) {
2135 attr->name = STR_INACTIVE; /* To place on hold a recvonly stream, send inactive */
2136 } else {
2137 attr->name = STR_SENDONLY; /* Send sendonly to initate a local hold */
2138 }
2139 } else {
2140 if (session_media->remotely_held) {
2141 attr->name = STR_RECVONLY; /* Remote has sent sendonly, reply recvonly */
2142 } else if (ast_stream_get_state(stream) == AST_STREAM_STATE_SENDONLY) {
2143 attr->name = STR_SENDONLY; /* Stream has requested sendonly */
2144 } else if (ast_stream_get_state(stream) == AST_STREAM_STATE_RECVONLY) {
2145 attr->name = STR_RECVONLY; /* Stream has requested recvonly */
2146 } else if (ast_stream_get_state(stream) == AST_STREAM_STATE_INACTIVE) {
2147 attr->name = STR_INACTIVE; /* Stream has requested inactive */
2148 } else {
2149 attr->name = STR_SENDRECV; /* No hold in either direction */
2150 }
2151 }
2152 media->attr[media->attr_count++] = attr;
2153
2154 /* If we've got rtcp-mux enabled, add it unless we received an offer without it */
2155 if (session->endpoint->media.rtcp_mux && session_media->remote_rtcp_mux) {
2156 attr = pjmedia_sdp_attr_create(pool, "rtcp-mux", NULL);
2157 pjmedia_sdp_attr_add(&media->attr_count, media->attr, attr);
2158 }
2159
2160 add_ssrc_to_stream(session, session_media, pool, media);
2161 add_msid_to_stream(session, session_media, pool, media, stream);
2162 add_rtcp_fb_to_stream(session, session_media, pool, media);
2163 add_extmap_to_stream(session, session_media, pool, media);
2164
2165 /* Add the media stream to the SDP */
2166 sdp->media[sdp->media_count++] = media;
2167
2168 SCOPE_EXIT_RTN_VALUE(1, "RC: 1\n");
2169}
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:2493
@ 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:671
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_media->rtp = ast_rtp_instance_new(session->endpoint->media.rtp.engine, sched, media_address, NULL))) {
271 ast_log(LOG_ERROR, "Unable to create RTP instance using RTP engine '%s'\n", session->endpoint->media.rtp.engine);
272 return -1;
273 }
274
275 ast_rtp_instance_set_prop(session_media->rtp, AST_RTP_PROPERTY_NAT, session->endpoint->media.rtp.symmetric);
276 ast_rtp_instance_set_prop(session_media->rtp, AST_RTP_PROPERTY_ASYMMETRIC_CODEC, session->endpoint->asymmetric_rtp_codec);
277
278 if (!session->endpoint->media.rtp.ice_support && (ice = ast_rtp_instance_get_ice(session_media->rtp))) {
279 ice->stop(session_media->rtp);
280 }
281
285 } else if (session->dtmf == AST_SIP_DTMF_INBAND) {
287 }
288
289 if (session_media->type == AST_MEDIA_TYPE_AUDIO &&
290 (session->endpoint->media.tos_audio || session->endpoint->media.cos_audio)) {
291 ast_rtp_instance_set_qos(session_media->rtp, session->endpoint->media.tos_audio,
292 session->endpoint->media.cos_audio, "SIP RTP Audio");
293 } else if (session_media->type == AST_MEDIA_TYPE_VIDEO) {
294 ast_rtp_instance_set_prop(session_media->rtp, AST_RTP_PROPERTY_RETRANS_RECV, session->endpoint->media.webrtc);
295 ast_rtp_instance_set_prop(session_media->rtp, AST_RTP_PROPERTY_RETRANS_SEND, session->endpoint->media.webrtc);
296 ast_rtp_instance_set_prop(session_media->rtp, AST_RTP_PROPERTY_REMB, session->endpoint->media.webrtc);
297 if (session->endpoint->media.webrtc) {
300 }
301 if (session->endpoint->media.tos_video || session->endpoint->media.cos_video) {
302 ast_rtp_instance_set_qos(session_media->rtp, session->endpoint->media.tos_video,
303 session->endpoint->media.cos_video, "SIP RTP Video");
304 }
305 }
306
307 ast_rtp_instance_set_last_rx(session_media->rtp, time(NULL));
308
309 return 0;
310}
@ 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:493
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:733
@ 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
struct pjsip_transport * transport
Transport itself.
Definition res_pjsip.h:119
Transport to bind to.
Definition res_pjsip.h:219

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_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, 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:760
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:914

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 707 of file res_pjsip_sdp_rtp.c.

708{
709 struct ast_str *fmtp0 = ast_str_alloca(256);
710 pj_str_t fmtp1;
711 pjmedia_sdp_attr *attr = NULL;
712 char *tmp;
713
714 ast_format_generate_sdp_fmtp(format, rtp_code, &fmtp0);
715 if (ast_str_strlen(fmtp0)) {
716 tmp = ast_str_buffer(fmtp0) + ast_str_strlen(fmtp0) - 1;
717 /* remove any carriage return line feeds */
718 while (*tmp == '\r' || *tmp == '\n') --tmp;
719 *++tmp = '\0';
720 /* ast...generate gives us everything, just need value */
721 tmp = strchr(ast_str_buffer(fmtp0), ':');
722 if (tmp && tmp[1] != '\0') {
723 fmtp1 = pj_str(tmp + 1);
724 } else {
725 fmtp1 = pj_str(ast_str_buffer(fmtp0));
726 }
727 attr = pjmedia_sdp_attr_create(pool, "fmtp", &fmtp1);
728 }
729 return attr;
730}
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 636 of file res_pjsip_sdp_rtp.c.

638{
639#ifndef HAVE_PJSIP_ENDPOINT_COMPACT_FORM
640 extern pj_bool_t pjsip_use_compact_form;
641#else
642 pj_bool_t pjsip_use_compact_form = pjsip_cfg()->endpt.use_compact_form;
643#endif
644 pjmedia_sdp_rtpmap rtpmap;
645 pjmedia_sdp_attr *attr = NULL;
646 char tmp[64];
647 enum ast_rtp_options options = session->endpoint->media.g726_non_standard ?
649
650 snprintf(tmp, sizeof(tmp), "%d", rtp_code);
651 pj_strdup2(pool, &media->desc.fmt[media->desc.fmt_count++], tmp);
652
653 if (rtp_code <= AST_RTP_PT_LAST_STATIC && pjsip_use_compact_form) {
654 return NULL;
655 }
656
657 rtpmap.pt = media->desc.fmt[media->desc.fmt_count - 1];
658 rtpmap.clock_rate = ast_rtp_lookup_sample_rate2(asterisk_format, format, code);
659 pj_strdup2(pool, &rtpmap.enc_name, ast_rtp_lookup_mime_subtype2(asterisk_format, format, code, options));
660 if (!pj_stricmp2(&rtpmap.enc_name, "opus")) {
661 pj_cstr(&rtpmap.param, "2");
662 } else {
663 pj_cstr(&rtpmap.param, NULL);
664 }
665
666 pjmedia_sdp_rtpmap_to_attr(pool, &rtpmap, &attr);
667
668 return attr;
669}
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.
static struct test_options options

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 672 of file res_pjsip_sdp_rtp.c.

674{
675#ifndef HAVE_PJSIP_ENDPOINT_COMPACT_FORM
676 extern pj_bool_t pjsip_use_compact_form;
677#else
678 pj_bool_t pjsip_use_compact_form = pjsip_cfg()->endpt.use_compact_form;
679#endif
680 pjmedia_sdp_rtpmap rtpmap;
681 pjmedia_sdp_attr *attr = NULL;
682 char tmp[64];
683 enum ast_rtp_options options = session->endpoint->media.g726_non_standard ?
685
686 snprintf(tmp, sizeof(tmp), "%d", rtp_code);
687 pj_strdup2(pool, &media->desc.fmt[media->desc.fmt_count++], tmp);
688
689 if (rtp_code <= AST_RTP_PT_LAST_STATIC && pjsip_use_compact_form) {
690 return NULL;
691 }
692
693 rtpmap.pt = media->desc.fmt[media->desc.fmt_count - 1];
694 rtpmap.clock_rate = sample_rate;
695 pj_strdup2(pool, &rtpmap.enc_name, ast_rtp_lookup_mime_subtype2(asterisk_format, format, code, options));
696 if (!pj_stricmp2(&rtpmap.enc_name, "opus")) {
697 pj_cstr(&rtpmap.param, "2");
698 } else {
699 pj_cstr(&rtpmap.param, NULL);
700 }
701
702 pjmedia_sdp_rtpmap_to_attr(pool, &rtpmap, &attr);
703
704 return attr;
705}

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 312 of file res_pjsip_sdp_rtp.c.

314{
315 pjmedia_sdp_attr *attr;
316 pjmedia_sdp_rtpmap *rtpmap;
317 pjmedia_sdp_fmtp fmtp;
318 struct ast_format *format;
319 int i, num = 0, tel_event = 0;
320 char name[256];
321 char media[20];
322 char fmt_param[256];
323 enum ast_rtp_options options = session->endpoint->media.g726_non_standard ?
326
328
330
331 /* Iterate through provided formats */
332 for (i = 0; i < stream->desc.fmt_count; ++i) {
333 /* The payload is kept as a string for things like t38 but for video it is always numerical */
334 ast_rtp_codecs_payloads_set_m_type(codecs, NULL, pj_strtoul(&stream->desc.fmt[i]));
335 /* Look for the optional rtpmap attribute */
336 if (!(attr = pjmedia_sdp_media_find_attr2(stream, "rtpmap", &stream->desc.fmt[i]))) {
337 continue;
338 }
339
340 /* Interpret the attribute as an rtpmap */
341 if ((pjmedia_sdp_attr_to_rtpmap(session->inv_session->pool_prov, attr, &rtpmap)) != PJ_SUCCESS) {
342 continue;
343 }
344
345 ast_copy_pj_str(name, &rtpmap->enc_name, sizeof(name));
346 if (strcmp(name, "telephone-event") == 0) {
347 if (tel_event == 0) {
348 int dtmf_rate = 0, dtmf_code = 0;
349 char dtmf_pt[8];
350 ast_copy_pj_str(dtmf_pt, &rtpmap->pt, sizeof(dtmf_pt));
351 dtmf_code = atoi(dtmf_pt);
352 dtmf_rate = rtpmap->clock_rate;
354 }
355 tel_event++;
356 }
357
358 ast_copy_pj_str(media, (pj_str_t*)&stream->desc.media, sizeof(media));
360 pj_strtoul(&stream->desc.fmt[i]), media, name, options, rtpmap->clock_rate);
361 /* Look for an optional associated fmtp attribute */
362 if (!(attr = pjmedia_sdp_media_find_attr2(stream, "fmtp", &rtpmap->pt))) {
363 continue;
364 }
365
366 if ((pjmedia_sdp_attr_get_fmtp(attr, &fmtp)) == PJ_SUCCESS) {
367 ast_copy_pj_str(fmt_param, &fmtp.fmt, sizeof(fmt_param));
368 if (sscanf(fmt_param, "%30d", &num) != 1) {
369 continue;
370 }
371
372 if ((format = ast_rtp_codecs_get_payload_format(codecs, num))) {
373 struct ast_format *format_parsed;
374
375 ast_copy_pj_str(fmt_param, &fmtp.fmt_param, sizeof(fmt_param));
376
377 format_parsed = ast_format_parse_sdp_fmtp(format, fmt_param);
378 if (format_parsed) {
380 ao2_ref(format_parsed, -1);
381 }
382 ao2_ref(format, -1);
383 }
384 }
385 }
386
387 /* Parsing done, now fill the ast_format_cap struct in the correct order */
388 for (i = 0; i < stream->desc.fmt_count; ++i) {
389 if ((format = ast_rtp_codecs_get_payload_format(codecs, pj_strtoul(&stream->desc.fmt[i])))) {
390 ast_format_cap_append(astformats, format, 0);
391 ao2_ref(format, -1);
392 }
393 }
394
395 if (session->dtmf == AST_SIP_DTMF_AUTO) {
396 if (tel_event) {
399 } else {
402 }
403 }
404
405 if (session->dtmf == AST_SIP_DTMF_AUTO_INFO) {
406 if (tel_event) {
409 } else {
412 }
413 }
414
415
416 /* Get the packetization, if it exists */
417 if ((attr = pjmedia_sdp_media_find_attr2(stream, "ptime", NULL))) {
418 unsigned long framing = pj_strtoul(pj_strltrim(&attr->value));
419 if (framing && session->endpoint->media.rtp.use_ptime) {
421 ast_format_cap_set_framing(astformats, framing);
422 }
423 }
424
426}
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.
Definition rtp_engine.c:986
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 996 of file res_pjsip_sdp_rtp.c.

998{
999 RAII_VAR(char *, transport_str, ast_strndup(transport.ptr, transport.slen), ast_free);
1000
1001 *optimistic = 0;
1002
1003 if (!transport_str) {
1005 }
1006 if (strstr(transport_str, "UDP/TLS")) {
1008 } else if (strstr(transport_str, "SAVP")) {
1010 } else if (media_stream_has_crypto(stream)) {
1011 *optimistic = 1;
1013 } else {
1015 }
1016}
#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 2506 of file res_pjsip_sdp_rtp.c.

2507{
2508 if (ast_check_ipv6()) {
2510 } else {
2511 ast_sockaddr_parse(&address_rtp, "0.0.0.0", 0);
2512 }
2513
2514 if (!(sched = ast_sched_context_create())) {
2515 ast_log(LOG_ERROR, "Unable to create scheduler context.\n");
2516 goto end;
2517 }
2518
2520 ast_log(LOG_ERROR, "Unable to create scheduler context thread.\n");
2521 goto end;
2522 }
2523
2525 ast_log(LOG_ERROR, "Unable to register SDP handler for %s stream type\n", STR_AUDIO);
2526 goto end;
2527 }
2528
2530 ast_log(LOG_ERROR, "Unable to register SDP handler for %s stream type\n", STR_VIDEO);
2531 goto end;
2532 }
2533
2535
2537end:
2538 unload_module();
2539
2541}
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:2828

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 2189 of file res_pjsip_sdp_rtp.c.

2190{
2191 struct ast_frame *f;
2192
2193 if (!session_media->rtp) {
2194 return &ast_null_frame;
2195 }
2196
2197 f = ast_rtp_instance_read(session_media->rtp, 1);
2198 if (!f) {
2199 return NULL;
2200 }
2201
2202 ast_rtp_instance_set_last_rx(session_media->rtp, time(NULL));
2203
2204 return f;
2205}
struct ast_frame * ast_rtp_instance_read(struct ast_rtp_instance *instance, int rtcp)
Receive a frame over RTP.
Definition rtp_engine.c:606
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 2171 of file res_pjsip_sdp_rtp.c.

2172{
2173 struct ast_frame *f;
2174
2175 if (!session_media->rtp) {
2176 return &ast_null_frame;
2177 }
2178
2179 f = ast_rtp_instance_read(session_media->rtp, 0);
2180 if (!f) {
2181 return NULL;
2182 }
2183
2184 ast_rtp_instance_set_last_rx(session_media->rtp, time(NULL));
2185
2186 return f;
2187}

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 2207 of file res_pjsip_sdp_rtp.c.

2208{
2209 if (!session_media->rtp) {
2210 return 0;
2211 }
2212
2213 return ast_rtp_instance_write(session_media->rtp, frame);
2214}
int ast_rtp_instance_write(struct ast_rtp_instance *instance, struct ast_frame *frame)
Send a frame out over RTP.
Definition rtp_engine.c:596

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 976 of file res_pjsip_sdp_rtp.c.

977{
978 int i;
979
980 for (i = 0; i < stream->attr_count; i++) {
981 pjmedia_sdp_attr *attr;
982
983 /* check the stream for the required crypto attribute */
984 attr = stream->attr[i];
985 if (pj_strcmp2(&attr->name, "crypto")) {
986 continue;
987 }
988
989 return 1;
990 }
991
992 return 0;
993}

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 1543 of file res_pjsip_sdp_rtp.c.

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

1160{
1161 int i;
1162
1163 for (i = 0; i < sdp->attr_count; i++) {
1164 apply_dtls_attrib(session_media, sdp->attr[i]);
1165 }
1166
1167 for (i = 0; i < stream->attr_count; i++) {
1168 apply_dtls_attrib(session_media, stream->attr[i]);
1169 }
1170
1172
1173 return 0;
1174}
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 1462 of file res_pjsip_sdp_rtp.c.

1464{
1465 int index;
1466
1467 if (!session->endpoint->media.webrtc || session_media->type != AST_MEDIA_TYPE_VIDEO) {
1468 return;
1469 }
1470
1471 ast_rtp_instance_extmap_clear(session_media->rtp);
1472
1473 for (index = 0; index < remote_stream->attr_count; ++index) {
1474 pjmedia_sdp_attr *attr = remote_stream->attr[index];
1475 char attr_value[pj_strlen(&attr->value) + 1];
1476 char *uri;
1477 int id;
1478 char direction_str[10] = "";
1479 char *attributes;
1481
1482 /* We only care about extmap attributes */
1483 if (pj_strcmp2(&attr->name, "extmap")) {
1484 continue;
1485 }
1486
1487 ast_copy_pj_str(attr_value, &attr->value, sizeof(attr_value));
1488
1489 /* Split the combined unique identifier and direction away from the URI and attributes for easier parsing */
1490 uri = strchr(attr_value, ' ');
1491 if (ast_strlen_zero(uri)) {
1492 continue;
1493 }
1494 *uri++ = '\0';
1495
1496 if ((sscanf(attr_value, "%30d%9s", &id, direction_str) < 1) || (id < 1)) {
1497 /* We require at a minimum the unique identifier */
1498 continue;
1499 }
1500
1501 /* Convert from the string to the internal representation */
1502 if (!strcasecmp(direction_str, "/sendonly")) {
1504 } else if (!strcasecmp(direction_str, "/recvonly")) {
1506 } else if (!strcasecmp(direction_str, "/inactive")) {
1508 }
1509
1510 attributes = strchr(uri, ' ');
1511 if (!ast_strlen_zero(attributes)) {
1512 *attributes++ = '\0';
1513 }
1514
1515 ast_rtp_instance_extmap_negotiate(session_media->rtp, id, direction, uri, attributes);
1516 }
1517}
enum queue_result id
Definition app_queue.c:1771
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:846
void ast_rtp_instance_extmap_clear(struct ast_rtp_instance *instance)
Clear negotiated RTP extension information.
Definition rtp_engine.c:890

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 872 of file res_pjsip_sdp_rtp.c.

874{
875 struct ast_rtp_engine_ice *ice;
876 const pjmedia_sdp_attr *attr;
877 char attr_value[256];
878 unsigned int attr_i;
879
880 /* If ICE support is not enabled or available exit early */
881 if (!session->endpoint->media.rtp.ice_support || !(ice = ast_rtp_instance_get_ice(session_media->rtp))) {
882 return;
883 }
884
885 ast_debug_ice(2, "(%p) ICE process attributes\n", session_media->rtp);
886
887 attr = pjmedia_sdp_media_find_attr2(remote_stream, "ice-ufrag", NULL);
888 if (!attr) {
889 attr = pjmedia_sdp_attr_find2(remote->attr_count, remote->attr, "ice-ufrag", NULL);
890 }
891 if (attr) {
892 ast_copy_pj_str(attr_value, (pj_str_t*)&attr->value, sizeof(attr_value));
894 } else {
895 ast_debug_ice(2, "(%p) ICE no, or invalid ice-ufrag\n", session_media->rtp);
896 return;
897 }
898
900 if (!attr) {
901 attr = pjmedia_sdp_attr_find2(remote->attr_count, remote->attr, "ice-pwd", NULL);
902 }
903 if (attr) {
904 ast_copy_pj_str(attr_value, (pj_str_t*)&attr->value, sizeof(attr_value));
906 } else {
907 ast_debug_ice(2, "(%p) ICE no, or invalid ice-pwd\n", session_media->rtp);
908 return;
909 }
910
912 ice->ice_lite(session_media->rtp);
913 }
914
915 /* Find all of the candidates */
916 for (attr_i = 0; attr_i < remote_stream->attr_count; ++attr_i) {
917 char foundation[33], transport[32], address[PJ_INET6_ADDRSTRLEN + 1], cand_type[6], relay_address[PJ_INET6_ADDRSTRLEN + 1] = "";
918 unsigned int port, relay_port = 0;
919 struct ast_rtp_engine_ice_candidate candidate = { 0, };
920
921 attr = remote_stream->attr[attr_i];
922
923 /* If this is not a candidate line skip it */
924 if (pj_strcmp2(&attr->name, "candidate")) {
925 continue;
926 }
927
928 ast_copy_pj_str(attr_value, (pj_str_t*)&attr->value, sizeof(attr_value));
929
930 if (sscanf(attr_value, "%32s %30u %31s %30u %46s %30u typ %5s %*s %23s %*s %30u", foundation, &candidate.id, transport,
931 (unsigned *)&candidate.priority, address, &port, cand_type, relay_address, &relay_port) < 7) {
932 /* Candidate did not parse properly */
933 continue;
934 }
935
936 if (session->endpoint->media.rtcp_mux && session_media->remote_rtcp_mux && candidate.id > 1) {
937 /* Remote side may have offered RTP and RTCP candidates. However, if we're using RTCP MUX,
938 * then we should ignore RTCP candidates.
939 */
940 continue;
941 }
942
943 candidate.foundation = foundation;
944 candidate.transport = transport;
945
947 ast_sockaddr_set_port(&candidate.address, port);
948
949 if (!strcasecmp(cand_type, "host")) {
951 } else if (!strcasecmp(cand_type, "srflx")) {
953 } else if (!strcasecmp(cand_type, "relay")) {
955 } else {
956 continue;
957 }
958
961 }
962
963 if (relay_port) {
964 ast_sockaddr_set_port(&candidate.relay_address, relay_port);
965 }
966
967 ice->add_remote_candidate(session_media->rtp, &candidate);
968 }
969
970 ice->set_role(session_media->rtp, pjmedia_sdp_neg_was_answer_remote(session->inv_session->neg) == PJ_TRUE ?
972 ice->start(session_media->rtp);
973}
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 834 of file res_pjsip_sdp_rtp.c.

836{
837 struct ast_rtp_engine_ice *ice;
839 char ufrag_attr_value[256];
840 char passwd_attr_value[256];
841
842 /* If ICE support is not enabled or available exit early */
843 if (!session->endpoint->media.rtp.ice_support || !(ice = ast_rtp_instance_get_ice(session_media->rtp))) {
844 return;
845 }
846
848 if (!ufrag_attr) {
849 ufrag_attr = pjmedia_sdp_attr_find2(remote->attr_count, remote->attr, "ice-ufrag", NULL);
850 }
851 if (ufrag_attr) {
853 } else {
854 return;
855 }
857 if (!passwd_attr) {
858 passwd_attr = pjmedia_sdp_attr_find2(remote->attr_count, remote->attr, "ice-pwd", NULL);
859 }
860 if (passwd_attr) {
862 } else {
863 return;
864 }
865
866 if (ufrag_attr && passwd_attr) {
868 }
869}

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 1276 of file res_pjsip_sdp_rtp.c.

1278{
1279 int index;
1280
1281 if (!session->endpoint->media.bundle) {
1282 return;
1283 }
1284
1285 for (index = 0; index < remote_stream->attr_count; ++index) {
1286 pjmedia_sdp_attr *attr = remote_stream->attr[index];
1287 char attr_value[pj_strlen(&attr->value) + 1];
1288 char *ssrc_attribute_name, *ssrc_attribute_value = NULL;
1289 unsigned int ssrc;
1290
1291 /* We only care about ssrc attributes */
1292 if (pj_strcmp2(&attr->name, "ssrc")) {
1293 continue;
1294 }
1295
1296 ast_copy_pj_str(attr_value, &attr->value, sizeof(attr_value));
1297
1298 if ((ssrc_attribute_name = strchr(attr_value, ' '))) {
1299 /* This has an actual attribute */
1300 *ssrc_attribute_name++ = '\0';
1301 ssrc_attribute_value = strchr(ssrc_attribute_name, ':');
1302 if (ssrc_attribute_value) {
1303 /* Values are actually optional according to the spec */
1304 *ssrc_attribute_value++ = '\0';
1305 }
1306 }
1307
1308 if (sscanf(attr_value, "%30u", &ssrc) < 1) {
1309 continue;
1310 }
1311
1312 /* If we are currently negotiating as a result of the remote side renegotiating then
1313 * determine if the source for this stream has changed.
1314 */
1315 if (pjmedia_sdp_neg_get_state(session->inv_session->neg) == PJMEDIA_SDP_NEG_STATE_REMOTE_OFFER &&
1316 session->active_media_state) {
1317 struct ast_rtp_instance_stats stats = { 0, };
1318
1320 stats.remote_ssrc != ssrc) {
1321 session_media->changed = 1;
1322 }
1323 }
1324
1325 ast_rtp_instance_set_remote_ssrc(session_media->rtp, ssrc);
1326 break;
1327 }
1328}
@ 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:1416
int ast_softhangup(struct ast_channel *chan, int cause)
Softly hangup up a channel.
Definition channel.c:2461
#define ast_channel_unref(c)
Decrease channel reference count.
Definition channel.h:3018
void ast_channel_hangupcause_set(struct ast_channel *chan, int value)
@ AST_SOFTHANGUP_DEV
Definition channel.h:1141
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:576
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 504 of file res_pjsip_sdp_rtp.c.

509{
510 RAII_VAR(struct ast_format_cap *, caps, NULL, ao2_cleanup);
511 RAII_VAR(struct ast_format_cap *, peer, NULL, ao2_cleanup);
512 RAII_VAR(struct ast_format_cap *, joint, NULL, ao2_cleanup);
513 enum ast_media_type media_type = session_media->type;
515 int direct_media_enabled = !ast_sockaddr_isnull(&session_media->direct_media_addr) &&
516 ast_format_cap_count(session->direct_media_cap);
517 int dsp_features = 0;
518 SCOPE_ENTER(1, "%s %s\n", ast_sip_session_get_name(session), is_offer ? "OFFER" : "ANSWER");
519
523 ast_log(LOG_ERROR, "Failed to allocate %s capabilities\n",
524 ast_codec_media_type2str(session_media->type));
525 SCOPE_EXIT_RTN_VALUE(-1, "Couldn't create %s capabilities\n",
526 ast_codec_media_type2str(session_media->type));
527 }
528
529 /* get the endpoint capabilities */
530 if (direct_media_enabled) {
531 ast_format_cap_get_compatible(session->endpoint->media.codecs, session->direct_media_cap, caps);
532 } else {
533 ast_format_cap_append_from_cap(caps, session->endpoint->media.codecs, media_type);
534 }
535
536 /* get the capabilities on the peer */
537 get_codecs(session, stream, &codecs, session_media, peer);
538
539 /* get the joint capabilities between peer and endpoint */
540 ast_format_cap_get_compatible(caps, peer, joint);
541 if (!ast_format_cap_count(joint)) {
544
546 ast_log(LOG_NOTICE, "No joint capabilities for '%s' media stream between our configuration(%s) and incoming SDP(%s)\n",
547 ast_codec_media_type2str(session_media->type),
548 ast_format_cap_get_names(caps, &usbuf),
549 ast_format_cap_get_names(peer, &thembuf));
550 SCOPE_EXIT_RTN_VALUE(-1, "No joint capabilities for '%s' media stream between our configuration(%s) and incoming SDP(%s)\n",
551 ast_codec_media_type2str(session_media->type),
552 ast_format_cap_get_names(caps, &usbuf),
553 ast_format_cap_get_names(peer, &thembuf));
554 } else {
555 struct ast_format *preferred_fmt = ast_format_cap_get_format(joint, 0);
556
558 ao2_ref(preferred_fmt, -1);
559 }
560
561 if (is_offer) {
562 /*
563 * Setup rx payload type mapping to prefer the mapping
564 * from the peer that the RFC says we SHOULD use.
565 */
567 }
569 session_media->rtp);
570
571 apply_cap_to_bundled(session_media, session_media_transport, asterisk_stream, joint);
572
573 if (session->channel && ast_sip_session_is_pending_stream_default(session, asterisk_stream)) {
574 ast_channel_lock(session->channel);
578 ast_format_cap_remove_by_type(caps, media_type);
579
580 if (session->endpoint->preferred_codec_only) {
581 struct ast_format *preferred_fmt = ast_format_cap_get_format(joint, 0);
582 ast_format_cap_append(caps, preferred_fmt, 0);
583 ao2_ref(preferred_fmt, -1);
584 } else if (!session->endpoint->asymmetric_rtp_codec) {
585 struct ast_format *best;
586 /*
587 * If we don't allow the sending codec to be changed on our side
588 * then get the best codec from the joint capabilities of the media
589 * type and use only that. This ensures the core won't start sending
590 * out a format that we aren't currently sending.
591 */
592
593 best = ast_format_cap_get_best_by_type(joint, media_type);
594 if (best) {
596 ao2_ref(best, -1);
597 }
598 } else {
599 ast_format_cap_append_from_cap(caps, joint, media_type);
600 }
601
602 /*
603 * Apply the new formats to the channel, potentially changing
604 * raw read/write formats and translation path while doing so.
605 */
607 if (media_type == AST_MEDIA_TYPE_AUDIO) {
610 }
611
612 if ( ((session->dtmf == AST_SIP_DTMF_AUTO) || (session->dtmf == AST_SIP_DTMF_AUTO_INFO) )
614 && (session->dsp)) {
615 dsp_features = ast_dsp_get_features(session->dsp);
616 dsp_features &= ~DSP_FEATURE_DIGIT_DETECT;
617 if (dsp_features) {
618 ast_dsp_set_features(session->dsp, dsp_features);
619 } else {
620 ast_dsp_free(session->dsp);
621 session->dsp = NULL;
622 }
623 }
624
625 if (ast_channel_is_bridged(session->channel)) {
627 }
628
629 ast_channel_unlock(session->channel);
630 }
631
634}
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:5757
int ast_channel_is_bridged(const struct ast_channel *chan)
Determine if a channel is in a bridge.
Definition channel.c:10589
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:5798
struct ast_format * ast_channel_readformat(struct ast_channel *chan)
void ast_dsp_free(struct ast_dsp *dsp)
Definition dsp.c:1787
int ast_dsp_get_features(struct ast_dsp *dsp)
Get features.
Definition dsp.c:1781
void ast_dsp_set_features(struct ast_dsp *dsp, int features)
Select feature set.
Definition dsp.c:1772
#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 1239 of file res_pjsip_sdp_rtp.c.

1240{
1241 struct ast_rtp_engine_ice *ice;
1242
1243 ast_assert(session_media->rtp != NULL);
1244
1246 if (!session->endpoint->media.rtp.ice_support || !ice) {
1247 return;
1248 }
1249
1250 if (session->endpoint->media.rtcp_mux && session_media->remote_rtcp_mux) {
1251 /* We both support RTCP mux. Only one ICE component necessary */
1252 ice->change_components(session_media->rtp, 1);
1253 } else {
1254 /* They either don't support RTCP mux or we don't know if they do yet. */
1255 ice->change_components(session_media->rtp, 2);
1256 }
1257}
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 459 of file res_pjsip_sdp_rtp.c.

462{
463 struct ast_format_cap *incoming_call_offer_cap;
464 struct ast_format_cap *remote;
467
468
470 if (!remote) {
471 ast_log(LOG_ERROR, "Failed to allocate %s incoming remote capabilities\n",
472 ast_codec_media_type2str(session_media->type));
473 SCOPE_EXIT_RTN_VALUE(NULL, "Couldn't allocate caps\n");
474 }
475
476 /* Get the peer's capabilities*/
477 get_codecs(session, stream, &codecs, session_media, remote);
478
479 incoming_call_offer_cap = ast_sip_session_create_joint_call_cap(
480 session, session_media->type, remote);
481
482 ao2_ref(remote, -1);
483
484 if (!incoming_call_offer_cap || ast_format_cap_empty(incoming_call_offer_cap)) {
485 ao2_cleanup(incoming_call_offer_cap);
487 SCOPE_EXIT_RTN_VALUE(NULL, "No incoming call offer caps\n");
488 }
489
490 /*
491 * Setup rx payload type mapping to prefer the mapping
492 * from the peer that the RFC says we SHOULD use.
493 */
495
497 ast_rtp_instance_get_codecs(session_media->rtp), session_media->rtp);
498
500
501 SCOPE_EXIT_RTN_VALUE(incoming_call_offer_cap);
502}
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 1519 of file res_pjsip_sdp_rtp.c.

1524{
1526 (session_media->type == AST_MEDIA_TYPE_AUDIO)) {
1527 if (((addrs != NULL) && ast_sockaddr_isnull(addrs)) ||
1528 ((addrs != NULL) && ast_sockaddr_is_any(addrs)) ||
1529 pjmedia_sdp_media_find_attr2(media, "sendonly", NULL) ||
1530 pjmedia_sdp_media_find_attr2(media, "inactive", NULL)) {
1531 if (!session_media->remotely_held) {
1532 session_media->remotely_held = 1;
1533 session_media->remotely_held_changed = 1;
1534 }
1535 } else if (session_media->remotely_held) {
1536 session_media->remotely_held = 0;
1537 session_media->remotely_held_changed = 1;
1538 }
1539 }
1540}
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 1081 of file res_pjsip_sdp_rtp.c.

1083{
1084 struct ast_rtp_engine_dtls *dtls;
1085
1086 if (!session->endpoint->media.rtp.dtls_cfg.enabled || !session_media->rtp) {
1087 return -1;
1088 }
1089
1090 dtls = ast_rtp_instance_get_dtls(session_media->rtp);
1091 if (!dtls) {
1092 return -1;
1093 }
1094
1095 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);
1096 if (dtls->set_configuration(session_media->rtp, &session->endpoint->media.rtp.dtls_cfg)) {
1097 ast_log(LOG_ERROR, "Attempted to set an invalid DTLS-SRTP configuration on RTP instance '%p'\n",
1098 session_media->rtp);
1099 return -1;
1100 }
1101
1102 if (setup_srtp(session_media)) {
1103 return -1;
1104 }
1105 return 0;
1106}
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 1212 of file res_pjsip_sdp_rtp.c.

1216{
1217 switch (session_media->encryption) {
1219 if (setup_sdes_srtp(session_media, stream)) {
1220 return -1;
1221 }
1222 break;
1224 if (setup_dtls_srtp(session, session_media)) {
1225 return -1;
1226 }
1227 if (parse_dtls_attrib(session_media, sdp, stream)) {
1228 return -1;
1229 }
1230 break;
1233 break;
1234 }
1235
1236 return 0;
1237}
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 1176 of file res_pjsip_sdp_rtp.c.

1178{
1179 int i;
1180
1181 for (i = 0; i < stream->attr_count; i++) {
1182 pjmedia_sdp_attr *attr;
1183 RAII_VAR(char *, crypto_str, NULL, ast_free);
1184
1185 /* check the stream for the required crypto attribute */
1186 attr = stream->attr[i];
1187 if (pj_strcmp2(&attr->name, "crypto")) {
1188 continue;
1189 }
1190
1191 crypto_str = ast_strndup(attr->value.ptr, attr->value.slen);
1192 if (!crypto_str) {
1193 return -1;
1194 }
1195
1196 if (setup_srtp(session_media)) {
1197 return -1;
1198 }
1199
1200 if (!ast_sdp_crypto_process(session_media->rtp, session_media->srtp, crypto_str)) {
1201 /* found a valid crypto attribute */
1202 return 0;
1203 }
1204
1205 ast_debug(1, "Ignoring crypto offer with unsupported parameters: %s\n", crypto_str);
1206 }
1207
1208 /* no usable crypto attributes found */
1209 return -1;
1210}
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 1062 of file res_pjsip_sdp_rtp.c.

1063{
1064 if (!session_media->srtp) {
1065 session_media->srtp = ast_sdp_srtp_alloc();
1066 if (!session_media->srtp) {
1067 return -1;
1068 }
1069 }
1070
1071 if (!session_media->srtp->crypto) {
1072 session_media->srtp->crypto = ast_sdp_crypto_alloc();
1073 if (!session_media->srtp->crypto) {
1074 return -1;
1075 }
1076 }
1077
1078 return 0;
1079}
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 2424 of file res_pjsip_sdp_rtp.c.

2425{
2426 if (session_media->rtp) {
2427 stream_stop(session_media);
2428 ast_rtp_instance_destroy(session_media->rtp);
2429 }
2430 session_media->rtp = NULL;
2431}
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:464

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 2412 of file res_pjsip_sdp_rtp.c.

2413{
2414 if (!session_media->rtp) {
2415 return;
2416 }
2417
2418 AST_SCHED_DEL(sched, session_media->keepalive_sched_id);
2419 AST_SCHED_DEL(sched, session_media->timeout_sched_id);
2420 ast_rtp_instance_stop(session_media->rtp);
2421}

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 2483 of file res_pjsip_sdp_rtp.c.

2484{
2488
2489 if (sched) {
2491 }
2492
2493 return 0;
2494}
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 2455 of file res_pjsip_sdp_rtp.c.

2456{
2457 struct pjsip_transaction *tsx;
2458 pjsip_tx_data *tdata;
2459
2460 if (!session->channel
2461 || !ast_sip_are_media_types_equal(&rdata->msg_info.msg->body->content_type,
2463 return 0;
2464 }
2465
2466 tsx = pjsip_rdata_get_tsx(rdata);
2467
2469
2470 if (pjsip_dlg_create_response(session->inv_session->dlg, rdata, 200, NULL, &tdata) == PJ_SUCCESS) {
2471 pjsip_dlg_send_response(session->inv_session->dlg, tsx, tdata);
2472 }
2473
2474 return 0;
2475}
@ AST_CONTROL_VIDUPDATE
pjsip_media_type pjsip_media_type_application_media_control_xml
Definition res_pjsip.c:3910
int ast_sip_are_media_types_equal(pjsip_media_type *a, pjsip_media_type *b)
Compare pjsip media types.
Definition res_pjsip.c:2219

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 2549 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 2549 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 2434 of file res_pjsip_sdp_rtp.c.

2434 {
2435 .id = STR_AUDIO,
2436 .negotiate_incoming_sdp_stream = negotiate_incoming_sdp_stream,
2437 .create_outgoing_sdp_stream = create_outgoing_sdp_stream,
2438 .apply_negotiated_sdp_stream = apply_negotiated_sdp_stream,
2439 .change_outgoing_sdp_stream_media_address = change_outgoing_sdp_stream_media_address,
2440 .stream_stop = stream_stop,
2441 .stream_destroy = stream_destroy,
2442};
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 2477 of file res_pjsip_sdp_rtp.c.

2477 {
2478 .method = "INFO",
2479 .incoming_request = video_info_incoming_request,
2480};

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 2445 of file res_pjsip_sdp_rtp.c.

2445 {
2446 .id = STR_VIDEO,
2447 .negotiate_incoming_sdp_stream = negotiate_incoming_sdp_stream,
2448 .create_outgoing_sdp_stream = create_outgoing_sdp_stream,
2449 .apply_negotiated_sdp_stream = apply_negotiated_sdp_stream,
2450 .change_outgoing_sdp_stream_media_address = change_outgoing_sdp_stream_media_address,
2451 .stream_stop = stream_stop,
2452 .stream_destroy = stream_destroy,
2453};

Referenced by load_module(), and unload_module().