41 #include <sofia-sip/sdp.h> 42 #include <sofia-sip/su.h> 51 #define MAX_CODEC_CHECK_FRAMES 50//x:mod_sofia.h 52 #define MAX_MISMATCH_FRAMES 5//x:mod_sofia.h 53 #define type2str(type) type == SWITCH_MEDIA_TYPE_VIDEO ? "video" : (type == SWITCH_MEDIA_TYPE_AUDIO ? "audio" : "text") 54 #define VIDEO_REFRESH_FREQ 1000000 56 #define TEXT_TIMER_MS 100 57 #define TEXT_TIMER_SAMPLES 10 58 #define TEXT_PERIOD_TIMEOUT 3000 59 #define MAX_RED_FRAMES 25 60 #define RED_PACKET_SIZE 100 210 #define MAX_REJ_STREAMS 10 296 ice_resolve_candidate = resolve_ice;
310 if (!strncasecmp(str, SUITES[i].
name, strlen(SUITES[i].name)) || (SUITES[i].alias && strlen(SUITES[i].alias) && !strncasecmp(str, SUITES[i].alias, strlen(SUITES[i].alias)))) {
311 return SUITES[i].
type;
322 return SUITES[type].
name;
365 return dft ? dft : 1;
372 uint32_t
fps, elapsed = 0;
414 sdp_attribute_t *attr;
432 if (sdp->sdp_origin) {
438 if (m->m_connections && m->m_connections->c_address) {
440 }
else if (sdp->sdp_connection && sdp->sdp_connection->c_address) {
444 for (attr = m->m_attributes; attr; attr = attr->a_next) {
445 if (!strcasecmp(attr->a_name,
"T38FaxVersion") && attr->a_value) {
447 }
else if (!strcasecmp(attr->a_name,
"T38MaxBitRate") && attr->a_value) {
449 }
else if (!strcasecmp(attr->a_name,
"T38FaxFillBitRemoval")) {
451 }
else if (!strcasecmp(attr->a_name,
"T38FaxTranscodingMMR")) {
453 }
else if (!strcasecmp(attr->a_name,
"T38FaxTranscodingJBIG")) {
455 }
else if (!strcasecmp(attr->a_name,
"T38FaxRateManagement") && attr->a_value) {
457 }
else if (!strcasecmp(attr->a_name,
"T38FaxMaxBuffer") && attr->a_value) {
459 }
else if (!strcasecmp(attr->a_name,
"T38FaxMaxDatagram") && attr->a_value) {
461 }
else if (!strcasecmp(attr->a_name,
"T38FaxUdpEC") && attr->a_value) {
463 }
else if (!strcasecmp(attr->a_name,
"T38VendorInfo") && attr->a_value) {
541 sdp_parser_t *parser = NULL;
545 if (!(parser = sdp_parse(NULL, r_sdp, (
int) strlen(r_sdp), 0))) {
549 if (!(sdp = sdp_session(parser))) {
550 sdp_parser_free(parser);
554 for (m = sdp->sdp_media; m; m = m->m_next) {
555 if (m->m_proto == sdp_proto_udptl && m->m_type == sdp_media_image && m->m_port) {
561 sdp_parser_free(parser);
596 "Audio params are unchanged for %s.\n",
599 const char *err = NULL;
602 "Audio params changed for %s from %s:%d to %s:%d\n",
625 const char *iananame,
653 if (!fmtp_a) fmtp_a =
"";
654 if (!fmtp_in) fmtp_in =
"";
657 if (!strcasecmp(pmap->
iananame, iananame) && !strcasecmp(fmtp_a, fmtp_in) && (!rate || (rate == pmap->
rate))) {
675 if (!
zstr(fmtp) && fmtpP) {
720 exists = (type == pmap->
type && !strcasecmp(name, pmap->
iananame));
723 exists = (type == pmap->
type && !strcasecmp(name, pmap->
iananame) && pmap->
pt == pt && (!pmap->
rate || rate == pmap->
rate) && (!pmap->
ptime || pmap->
ptime == ptime));
739 if (strcmp(pmap->
rm_fmtp, fmtp)) {
750 exists = (type == pmap->
type && !strcasecmp(name, pmap->
iananame) && pmap->
pt == pt);
752 exists = (type == pmap->
type && !strcasecmp(name, pmap->
iananame) && pmap->
pt == pt && (!pmap->
rate || rate == pmap->
rate) && (!pmap->
ptime || pmap->
ptime == ptime));
757 if (strcmp(pmap->
rm_fmtp, fmtp)) {
844 const char *preferred = NULL, *fallback = NULL;
870 return !
zstr(preferred) ? preferred : fallback;
878 const char *vars[] = {
"rtp_last_audio_local_crypto_key",
879 "srtp_remote_audio_crypto_key",
880 "srtp_remote_audio_crypto_tag",
881 "srtp_remote_audio_crypto_type",
882 "srtp_remote_video_crypto_key",
883 "srtp_remote_video_crypto_tag",
884 "srtp_remote_video_crypto_type",
885 "srtp_remote_text_crypto_key",
886 "srtp_remote_text_crypto_tag",
887 "srtp_remote_text_crypto_type",
889 "rtp_secure_media_inbound",
890 "rtp_secure_media_outbound",
893 for(i = 0; vars[i] ;i++) {
1038 unsigned char b64_key[512] =
"";
1079 p = strrchr((
char *) b64_key,
'=');
1081 while (p && *p && *p ==
'=') {
1111 #define CRYPTO_KEY_MATERIAL_LIFETIME_MKI_ERR 0x0u 1112 #define CRYPTO_KEY_MATERIAL_MKI 0x1u 1113 #define CRYPTO_KEY_MATERIAL_LIFETIME 0x2u 1115 #define RAW_BITS_PER_64_ENCODED_CHAR 6 1127 const char *field_begin;
1128 const char *field_end;
1129 const char *sep, *space;
1138 field_begin = strchr(*p,
'|');
1140 if (field_begin && (field_begin + 1 < end)) {
1141 space = strchr(field_begin,
' ');
1142 field_end = strchr(field_begin + 2,
'|');
1149 if ((field_end == NULL) || (space < field_end)) {
1156 sep = strchr(field_begin,
':');
1157 if (sep && (sep + 1 < field_end) && isdigit(*(sep + 1))) {
1160 for (i = 1, *p = sep - 1; *p != field_begin; --(*p), i *= 10) {
1161 val += ((**p) -
'0') * i;
1164 res |= ((val << 8) & 0x00ffff00);
1167 for (i = 1, *p = field_end - 1; *p != sep; --(*p), i *= 10) {
1168 val += ((**p) -
'0') * i;
1170 res |= (val & 0x000000ff);
1171 }
else if (isdigit(*(field_begin + 1)) && (*(field_begin + 2) ==
'^') && isdigit(*(field_begin + 3))) {
1173 val = ((uint32_t) (*(field_begin + 1) -
'0')) << 8;
1177 for (i = 1, *p = field_end - 1; *p != (field_begin + 2); --(*p), i *= 10) {
1178 val += ((**p) -
'0') * i;
1181 res |= (val & 0x000000ff);
1197 const char *key_material,
1198 int key_material_len,
1200 unsigned int mki_id,
1201 unsigned int mki_size)
1206 if (new_key_material == NULL) {
1217 new_key_material->
next = tail;
1219 return new_key_material;
1228 const char *end = NULL;
1233 end = strchr(p,
' ');
1235 end = p + strlen(p);
1246 const char *p, *delimit;
1247 const char *key_material_begin;
1248 const char *key_material_end = NULL;
1256 uint16_t lifetime_base = 0;
1257 uint16_t lifetime_exp = 0;
1261 unsigned long *key_material_n = NULL;
1263 bool multiple_keys =
false;
1265 const char *key_param;
1278 if (
zstr(key_param)) {
1279 goto no_crypto_found;
1282 *key_material_n = 0;
1284 p = strchr(key_param,
' ');
1286 if (!(p && *p && *(p + 1))) {
1287 goto no_crypto_found;
1314 key_material_begin = p;
1319 if ((delimit = strchr(p,
':')) == NULL) {
1320 goto bad_error_parsing_near;
1323 method_len = delimit - p;
1326 goto bad_key_param_method;
1334 if (!(p && *p && *(p + 1))) {
1339 if ((opts = strchr(p,
'|')) && (opts < key_material_end)) {
1340 keysalt_len = opts - p;
1342 keysalt_len = key_material_end - p;
1345 if (keysalt_len >
sizeof(key)) {
1346 goto bad_keysalt_len;
1351 if (!multiple_keys) {
1357 multiple_keys =
true;
1362 if (!(p < key_material_end)) {
1372 for (
int i = 0; i < 2 && (*opts ==
'|'); ++i) {
1376 switch ((opt_field >> 24) & 0x3) {
1380 lifetime_base = ((opt_field & 0x00ffff00) >> 8) & 0xffff;
1381 lifetime_exp = (opt_field & 0x000000ff) & 0xffff;
1382 lifetime = pow(lifetime_base, lifetime_exp);
1388 mki_id = ((opt_field & 0x00ffff00) >> 8) & 0xffff;
1389 mki_size = (opt_field & 0x000000ff) & 0xffff;
1394 goto bad_key_lifetime_or_mki;
1398 if (mki_id == 0 && lifetime == 0) {
1401 }
else if (mki_id == 0 || lifetime == 0) {
1404 goto bad_key_no_mki_index;
1409 goto bad_key_no_mki_size;
1419 goto bad_key_no_mki_index;
1422 if (mki_size != key_material->
mki_size) {
1423 goto bad_key_mki_size;
1428 SUITES[type].keysalt_len, (
char*) key_material_begin, key_material_end - key_material_begin, lifetime, mki_id, mki_size);
1429 *key_material_n = *key_material_n + 1;
1445 bad_error_parsing_near:
1463 bad_key_lifetime_or_mki:
1468 bad_key_no_mki_index:
1472 bad_key_no_mki_size:
1480 bad_key_param_method:
1492 engine->
type = type;
1506 char *keyvar, *tagvar, *ctypevar;
1512 keyvar =
"srtp_remote_audio_crypto_key";
1513 tagvar =
"srtp_remote_audio_crypto_tag";
1514 ctypevar =
"srtp_remote_audio_crypto_type";
1516 keyvar =
"srtp_remote_video_crypto_key";
1517 tagvar =
"srtp_remote_video_crypto_tag";
1518 ctypevar =
"srtp_remote_video_crypto_type";
1520 keyvar =
"srtp_remote_text_crypto_key";
1521 tagvar =
"srtp_remote_text_crypto_tag";
1522 ctypevar =
"srtp_remote_text_crypto_type";
1548 const char *varname;
1551 varname =
"rtp_secure_audio_confirmed";
1553 varname =
"rtp_secure_video_confirmed";
1555 varname =
"rtp_secure_text_confirmed";
1591 const char *var = NULL;
1592 const char *val = NULL;
1593 char *suites = NULL;
1596 int argc = 0, i = 0, j = 0, k = 0;
1607 var =
"rtp_secure_media_inbound";
1609 var =
"rtp_secure_media_outbound";
1613 var =
"rtp_secure_media";
1617 if (!
zstr(val) && (suites = strchr(val,
':'))) {
1633 if (!strcasecmp(val,
"optional")) {
1635 }
else if (
switch_true(val) || !strcasecmp(val,
"mandatory")) {
1639 if (!
switch_false(val) && strcasecmp(val,
"forbidden")) {
1647 for (i = 0; i < argc; i++) {
1651 if (!strcasecmp(fields[i], SUITES[j].
name)) {
1671 const char *varname,
1677 const char *vval = NULL;
1709 ctype = SUITES[j].
type;
1710 vval = use_alias ? SUITES[j].
alias : SUITES[j].
name;
1739 if (a && b && !strncasecmp(a, b, 23)) {
1847 #define add_stat(_i, _s) \ 1848 switch_snprintf(var_name, sizeof(var_name), "rtp_%s_%s", switch_str_nil(prefix), _s) ; \ 1849 switch_snprintf(var_val, sizeof(var_val), "%" SWITCH_SIZE_T_FMT, _i); \ 1850 switch_channel_set_variable(channel, var_name, var_val) 1852 #define add_stat_double(_i, _s) \ 1853 switch_snprintf(var_name, sizeof(var_name), "rtp_%s_%s", switch_str_nil(prefix), _s) ; \ 1854 switch_snprintf(var_val, sizeof(var_val), "%0.2f", _i); \ 1855 switch_channel_set_variable(channel, var_name, var_val) 1862 char var_name[256] =
"", var_val[35] =
"";
2154 flow = engine->
smode;
2179 flow = engine->
rmode;
2201 if (!engine)
return 0;
2210 if (!engine)
return NULL;
2241 const char *abs, *codec_string = NULL;
2242 const char *ocodec = NULL, *val;
2244 char *tmp_codec_string;
2278 codec_string = ocodec;
2286 if (codec_string && *codec_string ==
'=') {
2297 if (!codec_string) {
2298 codec_string =
"PCMU@20i,PCMA@20i,speex@20i";
2328 if (!strcasecmp(input,
"pause")) {
2331 }
else if (!strcasecmp(input,
"resume")) {
2334 }
else if (!strcasecmp(input,
"stop")) {
2337 }
else if (!strncasecmp(input,
"debug:", 6)) {
2339 if (s && !strcmp(s,
"off")) {
2349 if (v_engine->rtp_session) {
2350 if (!strncasecmp(input,
"vbsize:", 7)) {
2351 int frames = 0, max_frames = 0;
2356 if ((s = strchr(s,
':')) && *(s+1) !=
'\0') {
2357 max_frames = atoi(s+1);
2366 }
else if (!strncasecmp(input,
"vdebug:", 7)) {
2369 if (s && !strcmp(s,
"off")) {
2377 if (t_engine->rtp_session) {
2378 if (!strncasecmp(input,
"tbsize:", 7)) {
2379 int frames = 0, max_frames = 0;
2384 if ((s = strchr(s,
':')) && *(s+1) !=
'\0') {
2385 max_frames = atoi(s+1);
2392 }
else if (!strncasecmp(input,
"tdebug:", 7)) {
2395 if (s && !strcmp(s,
"off")) {
2409 jb_msec = atoi(val);
2411 if (strchr(val,
'p') && jb_msec > 0) {
2413 if (!maxlen) maxlen = jb_msec * 50;
2416 if ((p = strchr(val,
':'))) {
2420 if (strchr(p,
'p') && maxlen > 0) {
2426 if (!maxlen) maxlen = jb_msec * 50;
2428 if (jb_msec < 0 && jb_msec > -1000) {
2432 if (maxlen < 0 && maxlen > -1000) {
2437 if (jb_msec < 10 || jb_msec > 10000) {
2441 maxlen = jb_msec * 100;
2444 if (jb_msec && maxlen) {
2445 int qlen, maxqlen = 50;
2453 if (maxqlen < qlen) {
2461 SWITCH_LOG_DEBUG,
"Setting Jitterbuffer to %dms (%d frames) (%d max frames)\n",
2462 jb_msec, qlen, maxqlen);
2468 }
else if (!silent) {
2470 SWITCH_LOG_WARNING,
"Error Setting Jitterbuffer to %dms (%d frames)\n", jb_msec, qlen);
2480 int32_t jb_sync_msec = 0;
2481 uint32_t
fps = 0, frames = 0;
2482 uint32_t min_frames = 0;
2483 uint32_t max_frames = 0;
2484 uint32_t cur_frames = 0;
2507 if (!strcasecmp(var,
"disabled")) {
2513 if (tmp && tmp > -50 && tmp < 10000) {
2517 if ((p = strchr(var,
':'))) {
2525 fps = video_globals.
fps;
2527 if (fps < 15)
return;
2532 if (cur_frames && min_frames && cur_frames >= min_frames) {
2533 frames = cur_frames;
2534 }
else if (min_frames) {
2535 frames = min_frames;
2542 if (!jb_sync_msec && frames) {
2543 jb_sync_msec = ((double)1000 / fps) * frames;
2547 SWITCH_LOG_DEBUG1,
"%s %s \"%s\" Sync A/V JB to %dms %u VFrames, FPS %u a:%s sync_ms:%d\n",
2551 jb_sync_msec, frames, video_globals.
fps, sync_audio ?
"yes" :
"no", jb_sync_msec);
2615 (*tfP)->pool =
pool;
2617 (*tfP)->text_write_frame.packet = (*tfP)->text_write_frame_data;
2618 (*tfP)->text_write_frame.data = (
switch_byte_t *)(*tfP)->text_write_frame.packet + 12;
2621 (*tfP)->red_max = 5;
2626 for(x = 0; x < (*tfP)->red_max; x++) {
2645 unsigned char *
buf = data;
2647 unsigned char *e = (buf + datalen);
2653 *new_datalen = datalen;
2655 *(buf + datalen) =
'\0';
2657 while (*buf & 0x80) {
2663 pt[
count] = *buf & 0x7F;
2664 len[
count] = (ntohs(*(uint16_t *)(buf + 2)) & 0x03ff);
2671 idx = count - (seq - want_seq);
2679 *new_datalen = len[idx];
2683 for(x = 0; x < idx; x++) {
2687 *new_datalen = len[idx];
2688 *new_payload = pt[idx];
2690 memcpy(new_data, buf, len[idx]);
2692 *(((
char *)new_data) + len[idx]) =
'\0';
2700 unsigned char *
buf = data;
2701 int bytes = 0,
count = 0, pt = 0, len = 0;
2702 unsigned char *e = (buf + datalen);
2704 *new_datalen = datalen;
2707 while (*buf & 0x80) {
2715 len = (ntohs(*(uint16_t *)(buf + 2)) & 0x03ff);
2720 *new_datalen = datalen - bytes - 1 - (
count *4);
2845 memset((*frame)->data, 0, (*frame)->datalen);
2885 int rtp_timeout_sec = 0;
2886 int rtp_hold_timeout_sec = 0;
2909 "rtp_timeout_sec deprecated use media_timeout variable.\n");
2910 rtp_timeout_sec = v;
2918 "rtp_hold_timeout_sec deprecated use media_timeout variable.\n");
2919 rtp_hold_timeout_sec = v;
2923 if (rtp_timeout_sec) {
2928 if (!rtp_hold_timeout_sec) {
2929 rtp_hold_timeout_sec = rtp_timeout_sec * 10;
2933 if (rtp_hold_timeout_sec) {
2955 memset((*frame)->data, 0, (*frame)->datalen);
2974 snprintf(value,
sizeof(value),
"%.8x", rtcp_frame.
ssrc);
2977 snprintf(value,
sizeof(value),
"%u", rtcp_frame.
ntp_msw);
2980 snprintf(value,
sizeof(value),
"%u", rtcp_frame.
ntp_lsw);
2983 snprintf(value,
sizeof(value),
"%u", rtcp_frame.
timestamp);
2986 snprintf(value,
sizeof(value),
"%u", rtcp_frame.
packet_count);
2989 snprintf(value,
sizeof(value),
"%u", rtcp_frame.
octect_count);
3003 snprintf(header,
sizeof(header),
"Source%u-SSRC", i);
3004 snprintf(value,
sizeof(value),
"%.8x", rtcp_frame.
reports[i].
ssrc);
3006 snprintf(header,
sizeof(header),
"Source%u-Fraction", i);
3009 snprintf(header,
sizeof(header),
"Source%u-Lost", i);
3010 snprintf(value,
sizeof(value),
"%u", rtcp_frame.
reports[i].
lost);
3012 snprintf(header,
sizeof(header),
"Source%u-Loss-Avg", i);
3015 snprintf(header,
sizeof(header),
"Source%u-Highest-Sequence-Number-Received", i);
3018 snprintf(header,
sizeof(header),
"Source%u-Jitter", i);
3019 snprintf(value,
sizeof(value),
"%u", rtcp_frame.
reports[i].
jitter);
3021 snprintf(header,
sizeof(header),
"Source%u-LSR", i);
3022 snprintf(value,
sizeof(value),
"%u", rtcp_frame.
reports[i].
lsr);
3024 snprintf(header,
sizeof(header),
"Source%u-DLSR", i);
3025 snprintf(value,
sizeof(value),
"%u", rtcp_frame.
reports[i].
dlsr);
3027 snprintf(header,
sizeof(header),
"Rtt%u-Avg", i);
3028 snprintf(value,
sizeof(value),
"%f", rtcp_frame.
reports[i].
rtt_avg);
3097 if (codec_ms > 120) {
3099 "[CBR]: Your phone is trying to send timestamps that suggest an increment of %dms per packet\n" 3100 "That seems hard to believe so I am going to go on ahead and um ignore that, mmkay?\n",
3110 "[CBR]: Asynchronous PTIME not supported, changing our end from %d to %d\n",
3156 if (codec_ms > 120) {
3159 "[VBR]: Remote party is trying to send timestamps that suggest an increment of [%d] ms per packet, which is too high. Ignoring.\n",
3168 "[VBR]: Packet size change detected. Remote PTIME changed from [%d] to [%d]\n",
3207 "alternate payload received (received %d, expecting %d)\n",
3218 "Changing current codec to %s (payload type %d).\n",
3230 "Could not change to payload type %d, ignoring...\n",
3268 for (i = 1; i < 3; i++) {
3306 if ((*frame)->datalen == 0) {
3308 (*frame)->data =
"";
3332 int bytes = 0, samples = 0, frames = 0;
3379 frames = ((int) frame->
datalen / bytes);
3406 if (!local_t38_options) {
3466 if (!(engine = &smh->
engines[type]))
return;
3472 uint32_t system_bw = 0;
3473 const char *var = NULL, *bwv;
3587 msg.
from = __FILE__;
3651 "Changing Codec from %s@%dms@%dhz to %s@%dms@%luhz\n",
3950 #pragma warning(push) 3951 #pragma warning(disable:4702) 4007 if (strchr(ip,
':')) {
4035 if (strchr(ip,
':')) {
4048 sdp_attribute_t *attr = NULL, *attrs[2] = { 0 };
4049 int i = 0, got_rtcp_mux = 0;
4051 int ice_seen = 0, cid = 0, ai = 0, attr_idx = 0, cand_seen = 0, relay_ok = 0;
4053 int ice_resolve = 0;
4073 attrs[0] = m->m_attributes;
4074 attrs[1] = sdp->sdp_attributes;
4076 attrs[0] = sdp->sdp_attributes;
4081 for (attr_idx = 0; attr_idx < 2 && !(ice_seen && cand_seen); attr_idx++) {
4082 for (attr = attrs[attr_idx]; attr; attr = attr->a_next) {
4084 char *fields[32] = {0};
4085 int argc = 0, j = 0;
4087 if (
zstr(attr->a_name)) {
4091 if (!strcasecmp(attr->a_name,
"ice-ufrag")) {
4099 }
else if (!strcasecmp(attr->a_name,
"ice-pwd")) {
4103 }
else if (!strcasecmp(attr->a_name,
"ice-options")) {
4105 }
else if (!strcasecmp(attr->a_name,
"setup")) {
4106 if (!strcasecmp(attr->a_value,
"passive") ||
4113 }
else if (!strcasecmp(attr->a_value,
"active")) {
4149 }
else if (!engine->
remote_ssrc && !strcasecmp(attr->a_name,
"ssrc") && attr->a_value) {
4150 engine->
remote_ssrc = (uint32_t) atol(attr->a_value);
4158 }
else if (!strcasecmp(attr->a_name,
"rtcp-mux")) {
4167 }
else if (!strcasecmp(attr->a_name,
"candidate")) {
4189 cid = fields[1] ? atoi(fields[1]) - 1 : 0;
4196 for (i = 0; i < argc; i++) {
4203 }
else if (fields[4] && ice_resolve) {
4208 "Drop %s Candidate cid: %d proto: %s type: %s addr: %s:%s (cannot resolve)\n",
4210 cid+1, fields[2], fields[7], fields[4], fields[5]);
4215 "Drop %s Candidate cid: %d proto: %s type: %s addr: %s:%s (not an IP address)\n",
4217 cid+1, fields[2], fields[7], fields[4] ? fields[4] :
"(null)", fields[5]);
4223 "Drop %s Candidate cid: %d proto: %s type: %s addr: %s:%s (no network path)\n",
4225 cid+1, fields[2], fields[7] ? fields[7] :
"N/A", con_addr, fields[5]);
4229 "Save %s Candidate cid: %d proto: %s type: %s addr: %s:%s\n",
4231 cid+1, fields[2], fields[7] ? fields[7] :
"N/A", con_addr, fields[5]);
4244 while(j < argc && j <=
sizeof(fields)/
sizeof(
char*) && fields[j+1] && engine->
ice_in.
cand_idx[cid] <
MAX_CAND - 1) {
4245 if (!strcasecmp(fields[j],
"typ")) {
4247 }
else if (!strcasecmp(fields[j],
"raddr")) {
4249 }
else if (!strcasecmp(fields[j],
"rport")) {
4251 }
else if (!strcasecmp(fields[j],
"generation")) {
4279 if (relay_ok != is_relay)
continue;
4283 "Choose %s candidate, index %d, %s:%d\n", cid ?
"rtcp" :
"rtp", i,
4294 "Choose same candidate, index %d, for rtcp based on rtcp-mux attribute %s:%d\n", engine->
ice_in.
cand_idx[1],
4335 for (i = 0; i < 2; i++) {
4346 const char *media_varname = NULL, *port_varname = NULL;
4350 "setting remote %s ice addr to index %d %s:%d based on candidate\n",
type2str(type), engine->
ice_in.
chosen[0],
4365 media_varname =
"remote_video_ip";
4366 port_varname =
"remote_video_port";
4368 media_varname =
"remote_audio_ip";
4369 port_varname =
"remote_audio_port";
4371 media_varname =
"remote_text_ip";
4372 port_varname =
"remote_text_port";
4381 const char *media_varname = NULL, *port_varname = NULL;
4385 "Setting remote rtcp %s addr to %s:%d based on candidate\n",
type2str(type),
4394 media_varname =
"remote_video_rtcp_ip";
4395 port_varname =
"remote_video_rtcp_port";
4397 media_varname =
"remote_audio_rtcp_ip";
4398 port_varname =
"remote_audio_rtcp_port";
4400 media_varname =
"remote_text_rtcp_ip";
4401 port_varname =
"remote_text_rtcp_port";
4411 if (m && !got_rtcp_mux) {
4441 "rtcp_video_interval_msec" :
"rtcp_audio_interval_msec"))
4447 if (remote_rtcp_port) {
4448 if (!strcasecmp(val,
"passthru")) {
4453 int interval = atoi(val);
4454 if (interval < 100 || interval > 500000) {
4456 "Invalid rtcp interval spec [%d] must be between 100 and 500000\n", interval);
4496 #pragma warning(pop) 4517 #define MAX_MATCHES 30 4526 #define MIN(a,b) (((a) < (b)) ? (a) : (b)) 4531 int j = 0, f = 0, g;
4532 struct matches mtmp[MAX_MATCHES] = { { 0 } };
4534 m_idx =
MIN(m_idx, MAX_MATCHES);
4536 for(j = 0; j < m_idx; j++) {
4537 *&mtmp[j] = *&matches[j];
4543 for(j = 0; j < m_idx; j++) {
4544 if (mtmp[j].imp && mtmp[j].imp == imp) {
4545 *&matches[f++] = *&mtmp[j];
4569 if (!top++) pmap->
current = 1;
4575 const char *varname =
"invalid";
4579 varname =
"audio_media_flow";
4582 varname =
"video_media_flow";
4585 varname =
"text_media_flow";
4594 const char *varname =
"invalid";
4598 varname =
"remote_audio_media_flow";
4601 varname =
"remote_video_media_flow";
4604 varname =
"remote_text_media_flow";
4613 const char *smode_str =
"";
4619 smode_str =
"sendonly";
4623 smode_str =
"recvonly";
4626 smode_str =
"inactive";
4629 smode_str =
"disabled";
4632 smode_str =
"sendrecv";
4636 *mode_str = smode_str;
4637 *opp_mode = opp_smode;
4653 if (other_session) {
4672 msg->
from = __FILE__;
4677 if (other_session) {
4682 uint8_t proceed = 1;
4683 const char *sdp_in, *other_ep;
4696 msg->
from = __FILE__;
4711 const char *varname = NULL, *smode_str = NULL;
4714 int pass_codecs = 0;
4726 old_smode = engine->
smode;
4728 engine->
smode = smode;
4755 const char *varname = NULL, *rmode_str = NULL;
4768 if (engine->
rmode != rmode) {
4772 engine->
rmode = rmode;
4789 uint8_t match = 0, vmatch = 0, almost_vmatch = 0, tmatch = 0, fmatch = 0;
4791 unsigned long best_te_rate = 8000, cng_rate = 8000;
4793 sdp_attribute_t *attr;
4794 int ptime = 0, dptime = 0, maxptime = 0, dmaxptime = 0;
4795 int sendonly = 0, recvonly = 0, inactive = 0;
4796 int greedy = 0, x = 0, skip = 0;
4799 const char *crypto = NULL;
4800 int got_crypto = 0, got_video_crypto = 0, got_audio = 0, saw_audio = 0, saw_video = 0, got_avp = 0, got_savp = 0, got_udptl = 0, got_webrtc = 0, got_text = 0, got_text_crypto = 0, got_msrp = 0;
4802 sdp_parser_t *parser = NULL;
4808 uint32_t near_rate = 0;
4810 sdp_rtpmap_t *mmap = NULL, *near_map = NULL;
4811 struct matches matches[MAX_MATCHES] = { { 0 } };
4812 struct matches near_matches[MAX_MATCHES] = { { 0 } };
4814 uint32_t remote_codec_rate = 0, fmtp_remote_codec_rate = 0;
4816 int m_idx = 0, skip_rtcp = 0, skip_video_rtcp = 0, got_rtcp_mux = 0, got_video_rtcp_mux = 0;
4818 int vmatch_pt = 1, consider_video_fmtp = 1;
4819 int rtcp_auto_audio = 0, rtcp_auto_video = 0;
4820 int got_audio_rtcp = 0, got_video_rtcp = 0;
4841 codec_array = smh->
codecs;
4844 if (!(parser = sdp_parse(NULL, r_sdp, (
int) strlen(r_sdp), 0))) {
4848 if (!(sdp = sdp_session(parser))) {
4849 sdp_parser_free(parser);
4862 rtcp_auto_video = 1;
4863 rtcp_auto_audio = 1;
4879 if (proceed) *proceed = 1;
4885 if (!strcasecmp(val,
"generous")) {
4888 }
else if (!strcasecmp(val,
"greedy")) {
4891 }
else if (!strcasecmp(val,
"scrooge")) {
4903 if (strstr(smh->
origin,
"CiscoSystemsSIP-GW-UserAgent")) {
4910 if (strstr(smh->
origin,
"Sonus_UAC")) {
4913 "Hello,\nI see you have a Sonus!\n" 4914 "FYI, Sonus cannot follow the RFC on the proper way to send DTMF.\n" 4915 "Sadly, my creator had to spend several hours figuring this out so I thought you'd like to know that!\n" 4916 "Don't worry, DTMF will work but you may want to ask them to fix it......\n");
4937 if ((sdp->sdp_connection && sdp->sdp_connection->c_address && !strcmp(sdp->sdp_connection->c_address,
"0.0.0.0"))) {
4948 for (m = sdp->sdp_media; m; m = m->m_next) {
4949 sdp_connection_t *connection;
4955 case sdp_media_audio:
4958 case sdp_media_video:
4961 case sdp_media_image:
4969 if (m->m_type == sdp_media_audio) {
4973 if (m->m_type == sdp_media_video) {
4978 maxptime = dmaxptime;
4980 if (m->m_proto == sdp_proto_extended_srtp || m->m_proto == sdp_proto_extended_rtp) {
4985 if (m->m_proto_name && !strcasecmp(m->m_proto_name,
"UDP/TLS/RTP/SAVPF")) {
4989 if (m->m_proto_name && !strcasecmp(m->m_proto_name,
"UDP/RTP/AVPF")) {
4993 if (m->m_proto == sdp_proto_srtp || m->m_proto == sdp_proto_extended_srtp) {
4994 if (m->m_type == sdp_media_audio) {
4997 }
else if (m->m_proto == sdp_proto_rtp) {
4998 if (m->m_type == sdp_media_audio) {
5001 }
else if (m->m_proto == sdp_proto_udptl) {
5003 }
else if (m->m_proto == sdp_proto_msrp || m->m_proto == sdp_proto_msrps){
5007 if (got_msrp && m->m_type == sdp_media_message) {
5015 for (attr = m->m_attributes; attr; attr = attr->a_next) {
5017 if (!strcasecmp(attr->a_name,
"path") && attr->a_value) {
5020 }
else if (!strcasecmp(attr->a_name,
"accept-types") && attr->a_value) {
5023 }
else if (!strcasecmp(attr->a_name,
"accept-wrapped-types") && attr->a_value) {
5026 }
else if (!strcasecmp(attr->a_name,
"setup") && attr->a_value) {
5029 if (!strcmp(attr->a_value,
"passive")) {
5032 }
else if (!strcasecmp(attr->a_name,
"file-selector") && attr->a_value) {
5034 char *argv[4] = { 0 };
5043 for(i = 0; i<argc; i++) {
5045 if (
zstr(argv[i])) {
5049 if (!strncasecmp(argv[i],
"name:", 5)) {
5050 char *p = argv[i] + 5;
5051 int len = strlen(p);
5054 *(p + len - 1) =
'\0';
5058 }
else if (!strncasecmp(argv[i],
"type:", 5)) {
5061 if (!strncasecmp(argv[i],
"size:", 5)) {
5064 if (!strncasecmp(argv[i],
"hash:", 5)) {
5069 }
else if (!strcasecmp(attr->a_name,
"file-transfer-id") && attr->a_value) {
5071 }
else if (!strcasecmp(attr->a_name,
"file-disposition") && attr->a_value) {
5073 }
else if (!strcasecmp(attr->a_name,
"file-date") && attr->a_value) {
5075 }
else if (!strcasecmp(attr->a_name,
"file-icon") && attr->a_value) {
5077 }
else if (!strcasecmp(attr->a_name,
"file-range") && attr->a_value) {
5091 if (m->m_proto == sdp_proto_msrps) {
5099 "msrp%s://%s:%d/%s;tcp",
5110 if (got_udptl && m->m_type == sdp_media_image) {
5123 sdp_type ==
SDP_ANSWER ?
"response" :
"request");
5136 sdp_type ==
SDP_ANSWER ?
"response" :
"request");
5139 if (proceed) *proceed = 0;
5144 if (!strcasecmp(var,
"once")) {
5193 const char *err = NULL;
5222 msg->
from = __FILE__;
5240 fmatch ?
"IS" :
"IS NOT",
5241 sdp_type ==
SDP_ANSWER ?
"response" :
"request");
5246 }
else if (m->m_type == sdp_media_audio && m->m_port && got_audio && got_savp) {
5248 }
else if (m->m_type == sdp_media_audio && m->m_port && !got_audio) {
5254 memset(matches, 0,
sizeof(matches[0]) * MAX_MATCHES);
5255 memset(near_matches, 0,
sizeof(near_matches[0]) * MAX_MATCHES);
5257 audio_port = m->m_port;
5259 if (!sendonly && (m->m_mode == sdp_sendonly || m->m_mode == sdp_inactive)) {
5261 if (m->m_mode == sdp_inactive) {
5266 if (!sendonly && m->m_connections && m->m_connections->c_address && !strcmp(m->m_connections->c_address,
"0.0.0.0")) {
5274 switch(a_engine->
rmode) {
5290 for (attr = sdp->sdp_attributes; attr; attr = attr->a_next) {
5291 if (
zstr(attr->a_name)) {
5296 if (!strncasecmp(attr->a_name,
"ice", 3)) {
5298 }
else if (sendonly < 2 && !strcasecmp(attr->a_name,
"sendonly")) {
5301 }
else if (sendonly < 2 && !strcasecmp(attr->a_name,
"inactive")) {
5303 }
else if (!strcasecmp(attr->a_name,
"recvonly")) {
5321 }
else if (sendonly < 2 && !strcasecmp(attr->a_name,
"sendrecv")) {
5323 }
else if (!strcasecmp(attr->a_name,
"ptime")) {
5324 ptime = dptime = atoi(attr->a_value);
5325 }
else if (!strcasecmp(attr->a_name,
"maxptime")) {
5326 maxptime = dmaxptime = atoi(attr->a_value);
5330 if (sendonly == 2 && ice) {
5335 if (sendonly != 1 && recvonly != 1 && inactive != 1) {
5347 }
else if (sendonly) {
5349 }
else if (recvonly) {
5365 for (attr = m->m_attributes; attr; attr = attr->a_next) {
5367 if (!strcasecmp(attr->a_name,
"fingerprint") && !
zstr(attr->a_value)) {
5375 for (attr = m->m_attributes; attr; attr = attr->a_next) {
5376 if (!strcasecmp(attr->a_name,
"rtcp-mux")) {
5382 }
else if (!strcasecmp(attr->a_name,
"ice-ufrag")) {
5387 if (!got_rtcp_mux) {
5391 for (attr = m->m_attributes; attr; attr = attr->a_next) {
5392 if (!strcasecmp(attr->a_name,
"rtcp") && attr->a_value && !skip_rtcp) {
5400 }
else if (!strcasecmp(attr->a_name,
"ptime") && attr->a_value) {
5401 ptime = atoi(attr->a_value);
5402 }
else if (!strcasecmp(attr->a_name,
"maxptime") && attr->a_value) {
5403 maxptime = atoi(attr->a_value);
5404 }
else if (got_crypto < 1 && !strcasecmp(attr->a_name,
"crypto") && !
zstr(attr->a_value)) {
5409 if (m->m_proto != sdp_proto_srtp && !got_webrtc) {
5416 crypto = attr->a_value;
5417 crypto_tag = atoi(crypto);
5424 if (got_crypto == -1 && got_savp && !got_avp && !got_webrtc) {
5430 connection = sdp->sdp_connection;
5431 if (m->m_connections) {
5432 connection = m->m_connections;
5443 for (map = m->m_rtpmaps; map; map = map->rm_next) {
5445 const char *rm_encoding;
5446 uint32_t map_bit_rate = 0;
5448 int map_channels = map->rm_params ? atoi(map->rm_params) : 1;
5450 if (!(rm_encoding = map->rm_encoding)) {
5455 if (!strcasecmp(rm_encoding,
"telephone-event")) {
5458 best_te_rate = map->rm_rate;
5489 if (maxptime && (!codec_ms || codec_ms > maxptime)) {
5490 codec_ms = maxptime;
5499 if (!ptime && !strcasecmp(map->rm_encoding,
"g723")) {
5503 remote_codec_rate = map->rm_rate;
5504 fmtp_remote_codec_rate = 0;
5505 memset(&codec_fmtp, 0,
sizeof(codec_fmtp));
5507 if (
zstr(map->rm_fmtp)) {
5508 if (!strcasecmp(map->rm_encoding,
"ilbc")) {
5510 map_bit_rate = 13330;
5511 }
else if (!strcasecmp(map->rm_encoding,
"isac")) {
5513 map_bit_rate = 32000;
5528 }
else if (!strcasecmp(map->rm_encoding,
"opus")) {
5548 rm_encoding, map->rm_pt, (
int) remote_codec_rate, codec_ms, map_bit_rate, map_channels,
5551 match = (map->rm_pt == imp->
ianacode) ? 1 : 0;
5553 match = (!strcasecmp(rm_encoding, imp->
iananame) &&
5554 ((map->rm_pt < 96 && imp->
ianacode < 96) || (map->rm_pt > 95 && imp->
ianacode > 95)) &&
5556 if (fmtp_remote_codec_rate) {
5557 remote_codec_rate = fmtp_remote_codec_rate;
5561 if (match && bit_rate && map_bit_rate && map_bit_rate != bit_rate && strcasecmp(map->rm_encoding,
"ilbc") &&
5562 strcasecmp(map->rm_encoding,
"isac")) {
5567 if (match && remote_codec_rate && codec_rate && remote_codec_rate != codec_rate && (!strcasecmp(map->rm_encoding,
"pcma") ||
5568 !strcasecmp(map->rm_encoding,
"pcmu"))) {
5577 "Bah HUMBUG! Sticking with %s@%uh@%ui\n",
5579 }
else if ((ptime && codec_ms && codec_ms * 1000 != imp->
microseconds_per_packet) || remote_codec_rate != codec_rate) {
5583 if (nm_idx >= MAX_MATCHES) {
5585 "Audio Codec Compare [%s:%d:%u:%u:%d:%u:%d] was not saved as a near-match. Too many. Ignoring.\n",
5591 "Audio Codec Compare [%s:%d:%u:%d:%u:%d] is saved as a near-match\n",
5595 near_matches[nm_idx].
rate = remote_codec_rate;
5596 near_matches[nm_idx].
imp =
imp;
5597 near_matches[nm_idx].
map =
map;
5604 matches[m_idx].
rate = codec_rate;
5605 matches[m_idx].
imp =
imp;
5606 matches[m_idx].
map =
map;
5610 "Audio Codec Compare [%s:%d:%u:%d:%u:%d] ++++ is saved as a match\n",
5613 if (m_idx >= MAX_MATCHES) {
5621 if (m_idx >= MAX_MATCHES) {
5633 if (!m_idx && nm_idx) {
5636 for(j = 0; j < nm_idx; j++) {
5644 near_rate = near_matches[j].
rate;
5645 near_match = near_matches[j].
imp;
5646 near_map = near_matches[j].
map;
5648 switch_snprintf(tmp,
sizeof(tmp),
"%s@%uh@%ui%dc", near_match->iananame, near_rate ? near_rate : near_match->samples_per_second,
5649 codec_ms, near_match->number_of_channels);
5666 matches[m_idx].
rate = near_rate;
5667 matches[m_idx].
imp = timp;
5668 matches[m_idx].
map = near_map;
5680 greedy_sort(smh, matches, m_idx, codec_array, total_codecs);
5689 matches[j].map->rm_encoding,
5691 matches[j].
map->rm_fmtp,
5693 matches[j].
map->rm_pt,
5699 mimp = matches[j].
imp;
5700 mmap = matches[j].
map;
5720 pmap->
channels = mmap->rm_params ? atoi(mmap->rm_params) : 1;
5722 if (!strcasecmp((
char *) mmap->rm_encoding,
"opus")) {
5729 if (!
zstr((
char *) mmap->rm_fmtp) &&
switch_stristr(
"stereo=1", (
char *) mmap->rm_fmtp)) {
5731 if (!allow_channels || allow_channels >= 2) {
5770 if (pmap->
pt > pl) {
5808 for (map = m->m_rtpmaps; map; map = map->rm_next) {
5809 const char *rm_encoding;
5811 if (!(rm_encoding = map->rm_encoding)) {
5815 if (!strcasecmp(rm_encoding,
"telephone-event")) {
5818 best_te_rate = map->rm_rate;
5828 cng_rate = map->rm_rate;
5844 best_te_rate = 8000;
5849 "No 2833 in SDP. Liberal DTMF mode adding %d as telephone-event.\n", smh->
mparams->
te);
5895 }
else if (!got_text && m->m_type == sdp_media_text && m->m_port) {
5901 connection = sdp->sdp_connection;
5902 if (m->m_connections) {
5903 connection = m->m_connections;
5918 for (map = m->m_rtpmaps; map; map = map->rm_next) {
5943 if (!strcasecmp(map->rm_encoding,
"red")) {
5952 for (attr = m->m_attributes; attr; attr = attr->a_next) {
5953 if (!strcasecmp(attr->a_name,
"rtcp") && attr->a_value) {
5956 }
else if (!got_text_crypto && !strcasecmp(attr->a_name,
"crypto") && !
zstr(attr->a_value)) {
5961 if (m->m_proto != sdp_proto_srtp && !got_webrtc) {
5968 crypto = attr->a_value;
5969 crypto_tag = atoi(crypto);
5972 "rtp_has_text_crypto",
5995 }
else if (m->m_type == sdp_media_video) {
5997 const char *rm_encoding;
6000 const char *inherit_video_fmtp = NULL;
6005 sdp_bandwidth_t *bw;
6008 for (bw = m->m_bandwidths; bw; bw = bw->b_next) {
6009 if (bw->b_modifier == sdp_bw_as && !tias) {
6010 v_engine->
sdp_bw = bw->b_value;
6011 }
else if (bw->b_modifier == sdp_bw_tias) {
6013 v_engine->
sdp_bw = bw->b_value / 1024;
6017 switch(v_engine->
rmode) {
6041 memset(matches, 0,
sizeof(matches[0]) * MAX_MATCHES);
6042 memset(near_matches, 0,
sizeof(near_matches[0]) * MAX_MATCHES);
6048 connection = sdp->sdp_connection;
6049 if (m->m_connections) {
6050 connection = m->m_connections;
6061 skip_video_rtcp = 0;
6062 got_video_rtcp_mux = 0;
6063 for (attr = m->m_attributes; attr; attr = attr->a_next) {
6064 if (!strcasecmp(attr->a_name,
"rtcp-mux")) {
6065 got_video_rtcp_mux = 1;
6066 skip_video_rtcp = 1;
6067 }
else if (!strcasecmp(attr->a_name,
"ice-ufrag")) {
6068 skip_video_rtcp = 1;
6072 if (!got_video_rtcp_mux) {
6076 for (attr = m->m_attributes; attr; attr = attr->a_next) {
6077 if (!strcasecmp(attr->a_name,
"framerate") && attr->a_value) {
6079 }
else if (!strcasecmp(attr->a_name,
"rtcp-fb")) {
6080 if (!
zstr(attr->a_value)) {
6097 rtcp_auto_video = 1;
6100 }
else if (!strcasecmp(attr->a_name,
"rtcp") && attr->a_value && !skip_video_rtcp) {
6107 }
else if (!got_video_crypto && !strcasecmp(attr->a_name,
"crypto") && !
zstr(attr->a_value)) {
6112 if (m->m_proto != sdp_proto_srtp && !got_webrtc) {
6119 crypto = attr->a_value;
6120 crypto_tag = atoi(crypto);
6123 "rtp_has_video_crypto",
6136 for (map = m->m_rtpmaps; map; map = map->rm_next) {
6139 for (attr = m->m_attributes; attr; attr = attr->a_next) {
6140 if (!strcasecmp(attr->a_name,
"fingerprint") && !
zstr(attr->a_value)) {
6141 got_video_crypto = 1;
6146 if (!(rm_encoding = map->rm_encoding)) {
6150 for (i = 0; i < total_codecs; i++) {
6165 vmatch = (map->rm_pt == imp->
ianacode) ? 1 : 0;
6167 vmatch = strcasecmp(rm_encoding, imp->
iananame) ? 0 : 1;
6172 vmatch = !strcasecmp(smh->
fmtps[i], map->rm_fmtp);
6175 if (vmatch && vmatch_pt) {
6179 int opt = atoi(other_pt);
6180 if (map->rm_pt != opt) {
6191 matches[m_idx].
imp =
imp;
6192 matches[m_idx].
map =
map;
6199 if (m_idx >= MAX_MATCHES) {
6207 if (m_idx >= MAX_MATCHES) {
6212 if (consider_video_fmtp && (!m_idx || almost_vmatch)) {
6216 consider_video_fmtp = 0;
6220 if (vmatch_pt && !m_idx) {
6238 greedy_sort(smh, matches, m_idx, codec_array, total_codecs);
6248 matches[j].map->rm_encoding,
6250 consider_video_fmtp ? matches[j].
map->rm_fmtp : NULL,
6252 matches[j].
map->rm_pt,
6266 mimp = matches[j].
imp;
6267 map = matches[j].
map;
6308 video_port = m->m_port;
6314 if (rtcp_auto_audio || rtcp_auto_video) {
6315 if (rtcp_auto_audio && !skip_rtcp && !got_audio_rtcp && audio_port) {
6323 if (rtcp_auto_video && !skip_video_rtcp && !got_video_rtcp && video_port) {
6365 if (v_engine->
fir) {
6371 if (v_engine->
pli) {
6377 if (v_engine->
nack) {
6383 if (v_engine->
tmmbr) {
6424 sdp_parser_free(parser);
6432 return match || vmatch || tmatch || fmatch;
6486 const char *msg =
"hold";
6501 msg =
"hold-private";
6532 stream =
"local_stream://moh";
6537 if (!strcasecmp(stream,
"indicate_hold")) {
6569 int bypass_after_hold_a = 0;
6570 int bypass_after_hold_b = 0;
6572 if (media_on_hold_a) {
6698 uint32_t frames = 0, frame_ms = 0;
6699 uint32_t
fps, width, height;
6710 if (!width) width = 352;
6711 if (!height) height = 288;
6724 if (fps < 15) fps = 15;
6725 frame_ms = (uint32_t) 1000 / fps;
6726 if (frame_ms <= 0) frame_ms = 66;
6727 frames = (uint32_t) ms / frame_ms;
6730 for (i = 0; i < frames; i++) {
6745 unsigned char *
buf = NULL;
6755 int last_w = 0, last_h = 0, kps = 0;
6796 }
else if (video_globals.
fps) {
6797 fps = video_globals.
fps;
6804 int min = 0, max = 10000000;
6811 if (min < 0) min = 0;
6816 if (max < 0) max = 10000000;
6819 if (min && kps < min) kps = min;
6820 if (max != 10000000 && kps > max) kps = max;
6847 if (!kps && fr.
img && (last_w != fr.
img->
d_w || last_h != fr.
img->
d_h)) {
6864 last_frame = fr.
img;
6881 fr.
img = last_frame;
6883 for (x = 0; x < (int)(fps_data.
fps / 2); x++) {
7132 for (bp = session->
bugs; bp; bp = bp->
next) {
7456 uint32_t loops = 0, xloops = 0;
7459 unsigned char *
buf = NULL;
7464 int blank_enabled = 1;
7599 }
else if (blank_img) {
7776 #define RA_PTR_LEN 512 7784 char *p, *ip_ptr = NULL, *port_ptr = NULL, *vid_port_ptr = NULL, *text_port_ptr = NULL, *pe;
7801 if (
zstr(sdp_str)) {
7805 if (
zstr(sdp_str)) {
7820 if (tmp && atoi(tmp)) {
7826 vid_port_ptr = p + 8;
7830 text_port_ptr = p + 7;
7833 if (!(ip_ptr && port_ptr)) {
7840 while (x <
sizeof(rip) - 1 && p && *p && ((*p >=
'0' && *p <=
'9') || *p ==
'.' || *p ==
':' || (*p >=
'a' && *p <=
'f') || (*p >=
'A' && *p <=
'F'))) {
7851 while (x <
sizeof(rp) - 1 && p && *p && (*p >=
'0' && *p <=
'9')) {
7863 while (x <
sizeof(rvp) - 1 && p && *p && (*p >=
'0' && *p <=
'9')) {
7872 if (text_port_ptr) {
7875 while (x <
sizeof(rtp) - 1 && p && *p && (*p >=
'0' && *p <=
'9')) {
7884 if (!(*rip && *rp)) {
7914 const char *rport = NULL;
7917 if (!remote_rtcp_port) {
7953 const char *rport = NULL;
7956 if (!remote_rtcp_port) {
7987 const char *rport = NULL;
7994 }
else if (remote_host && ( (strcmp(remote_host,
"0.0.0.0") == 0) ||
7998 "Remote address changed from [%s] to [%s]. Ignoring...\n",
8052 char *stun_ip = NULL;
8066 if (!strncasecmp(sourceip,
"host:", 5)) {
8068 }
else if (!strncasecmp(sourceip,
"stun:", 5)) {
8071 stun_ip = strdup(sourceip + 5);
8073 if ((p = strchr(stun_ip,
':'))) {
8077 if (iport > 0 && iport < 0xFFFF) {
8082 if (
zstr(stun_ip)) {
8087 for (x = 0; x < 5; x++) {
8105 if (myport == *port && !strcmp(*ip, smh->
mparams->
rtpip)) {
8113 *ip = (
char *) sourceip;
8150 const char *use_ip = NULL;
8154 char vname[128] =
"";
8166 if (!lookup_rtpip) {
8218 use_ip = lookup_rtpip;
8317 a_engine->
mh.
up = 0;
8328 v_engine->
mh.
up = 0;
8350 t_engine->
mh.
up = 0;
8508 #ifdef HAVE_OPENSSL_DTLSv1_2_method 8509 uint8_t want_DTLSv1_2 = 1;
8511 uint8_t want_DTLSv1_2 = 0;
8512 #endif // HAVE_OPENSSL_DTLSv1_2_method 8542 const char *err = NULL;
8543 const char *val = NULL;
8547 char *timer_name = NULL;
8551 int is_reinvite = 0;
8553 #ifdef HAVE_OPENSSL_DTLSv1_2_method 8554 uint8_t want_DTLSv1_2 = 1;
8556 uint8_t want_DTLSv1_2 = 0;
8590 if (want_DTLSv1_2) {
8657 #if __BYTE_ORDER == __LITTLE_ENDIAN 8709 const char *rport = NULL;
8712 if (!remote_rtcp_port) {
8738 if (session && a_engine) {
8748 memset(flags, 0,
sizeof(flags));
8760 "PROXY AUDIO RTP [%s] %s:%d->%s:%d codec: %u ms: %d\n",
8775 timer_name = (
char *) var;
8811 uint32_t ssrc_ul = (uint32_t) strtoul(ssrc, NULL, 10);
8813 a_engine->
ssrc = ssrc_ul;
8862 if ((vad_in && inb) || (vad_out && !inb)) {
8899 if (!remote_rtcp_port && rport) {
8903 if (!strcasecmp(val,
"passthru")) {
8907 int interval = atoi(val);
8908 if (interval < 100 || interval > 500000) {
8910 "Invalid rtcp interval spec [%d] must be between 100 and 500000\n", interval);
8978 "rtp_timeout_sec deprecated use media_timeout variable.\n");
8987 "rtp_hold_timeout_sec deprecated use media_hold_timeout variable.\n");
9028 int delayi = atoi(val);
9029 if (delayi < 0) delayi = 0;
9102 const char *rport = NULL;
9107 if (!remote_rtcp_port) {
9133 memset(flags, 0,
sizeof(flags));
9143 "PROXY TEXT RTP [%s] %s:%d->%s:%d codec: %u ms: %d\n",
9166 memset(flags, 0,
sizeof(flags));
9209 if (!t_engine->
tf) {
9220 uint32_t ssrc_ul = (uint32_t) strtoul(ssrc, NULL, 10);
9222 t_engine->
ssrc = ssrc_ul;
9262 if (!strcasecmp(val,
"passthru")) {
9266 int interval = atoi(val);
9267 if (interval < 100 || interval > 500000) {
9269 "Invalid rtcp interval spec [%d] must be between 100 and 500000\n", interval);
9273 "Activating TEXT RTCP PORT %d interval %d mux %d\n", remote_port, interval, t_engine->
rtcp_mux);
9404 const char *rport = NULL;
9409 if (!remote_rtcp_port) {
9435 memset(flags, 0,
sizeof(flags));
9445 "PROXY VIDEO RTP [%s] %s:%d->%s:%d codec: %u ms: %d\n",
9468 memset(flags, 0,
sizeof(flags));
9487 if (v_engine->
fir) {
9491 if (v_engine->
pli) {
9499 if (v_engine->
tmmbr) {
9525 if (v_engine->
fir) {
9529 if (v_engine->
pli) {
9539 uint32_t ssrc_ul = (uint32_t) strtoul(ssrc, NULL, 10);
9541 v_engine->
ssrc = ssrc_ul;
9583 if (!strcasecmp(val,
"passthru")) {
9587 int interval = atoi(val);
9588 if (interval < 100 || interval > 500000) {
9590 "Invalid rtcp interval spec [%d] must be between 100 and 500000\n", interval);
9594 "Activating VIDEO RTCP PORT %d interval %d mux %d\n", remote_port, interval, v_engine->
rtcp_mux);
9689 if (session && v_engine) {
9712 return "UDP/TLS/RTP/SAVPF";
9758 int cur_ptime,
const char *append_audio,
const char *sr,
int use_cng,
int cng_type,
switch_event_t *map,
int secure,
9763 int already_did[128] = { 0 };
9764 int ptime = 0, noptime = 0;
9767 int include_external;
9795 switch_snprintf(buf + strlen(buf), buflen - strlen(buf),
"m=audio %d %s", port,
9805 if (!strcasecmp(imp->
iananame,
"ilbc") || !strcasecmp(imp->
iananame,
"isac") ) {
9819 if (this_ptime != cur_ptime) {
9850 if (pmap->
pt >= 128 || already_did[pmap->
pt]) {
9854 if (!strncasecmp(pmap->
iananame,
"telephone-event", 15)) {
9856 already_did[pmap->
pt] = 1;
9883 memset(already_did, 0,
sizeof(already_did));
9888 char *fmtp = imp->
fmtp;
9895 if (!strcasecmp(imp->
iananame,
"ilbc") || !strcasecmp(imp->
iananame,
"isac")) {
9905 if (this_ptime != cur_ptime) {
9928 if (smh->
fmtps[i]) {
9929 fmtp = smh->
fmtps[i];
9956 if (!strncasecmp(pmap->
iananame,
"telephone-event", 15)) {
9957 switch_snprintf(buf + strlen(buf), buflen - strlen(buf),
"a=rtpmap:%d telephone-event/%d\r\n",
9967 switch_snprintf(buf + strlen(buf), buflen - strlen(buf),
"a=rtpmap:%d telephone-event/%d\r\n",
9971 switch_snprintf(buf + strlen(buf), buflen - strlen(buf),
"a=rtpmap:%d telephone-event/%d\r\na=fmtp:%d 0-%d\r\n",
9985 switch_snprintf(buf + strlen(buf), buflen - strlen(buf),
"a=rtcp-mux\r\n");
9986 switch_snprintf(buf + strlen(buf), buflen - strlen(buf),
"a=rtcp:%d IN %s %s\r\n", port, family, ip);
9988 switch_snprintf(buf + strlen(buf), buflen - strlen(buf),
"a=rtcp:%d IN %s %s\r\n", port + 1, family, ip);
9998 uint32_t c1 = (1<<24)*126 + (1<<8)*65535 + (1<<0)*(256 - 1);
9999 uint32_t c2 = c1 - 1;
10011 ice_out = &a_engine->
ice_out;
10014 switch_snprintf(buf + strlen(buf), buflen - strlen(buf),
"a=ssrc:%u msid:%s a0\r\n", a_engine->
ssrc, smh->
msid);
10015 switch_snprintf(buf + strlen(buf), buflen - strlen(buf),
"a=ssrc:%u mslabel:%s\r\n", a_engine->
ssrc, smh->
msid);
10016 switch_snprintf(buf + strlen(buf), buflen - strlen(buf),
"a=ssrc:%u label:%sa0\r\n", a_engine->
ssrc, smh->
msid);
10019 switch_snprintf(buf + strlen(buf), buflen - strlen(buf),
"a=ice-ufrag:%s\r\n", ice_out->
ufrag);
10020 switch_snprintf(buf + strlen(buf), buflen - strlen(buf),
"a=ice-pwd:%s\r\n", ice_out->
pwd);
10023 switch_snprintf(buf + strlen(buf), buflen - strlen(buf),
"a=candidate:%s 1 %s %u %s %d typ host generation 0\r\n",
10029 switch_snprintf(buf + strlen(buf), buflen - strlen(buf),
"a=candidate:%s 1 %s %u %s %d typ host generation 0\r\n",
10039 switch_snprintf(buf + strlen(buf), buflen - strlen(buf),
"a=candidate:%s 1 %s %u %s %d typ srflx raddr %s rport %d generation 0\r\n",
10049 switch_snprintf(buf + strlen(buf), buflen - strlen(buf),
"a=candidate:%s 2 %s %u %s %d typ host generation 0\r\n",
10055 switch_snprintf(buf + strlen(buf), buflen - strlen(buf),
"a=candidate:%s 2 %s %u %s %d typ host generation 0\r\n",
10065 switch_snprintf(buf + strlen(buf), buflen - strlen(buf),
"a=candidate:%s 2 %s %u %s %d typ srflx raddr %s rport %d generation 0\r\n",
10076 switch_snprintf(buf + strlen(buf), buflen - strlen(buf),
"a=ice-options:google-ice\r\n");
10104 switch_snprintf(buf + strlen(buf), buflen - strlen(buf),
"a=silenceSupp:off - - - -\r\n");
10108 if (append_audio) {
10109 switch_snprintf(buf + strlen(buf), buflen - strlen(buf),
"%s%s", append_audio,
end_of(append_audio) ==
'\n' ?
"" :
"\r\n");
10116 if (!noptime && cur_ptime) {
10117 switch_snprintf(buf + strlen(buf), buflen - strlen(buf),
"a=ptime:%d\r\n", cur_ptime);
10121 switch_snprintf(buf + strlen(buf), buflen - strlen(buf),
"a=%s\r\n", sr);
10138 if (!strcasecmp(val,
"rfc2833")) {
10140 }
else if (!strcasecmp(val,
"info")) {
10142 }
else if (!strcasecmp(val,
"none")) {
10152 sdp_parser_t *parser = NULL;
10153 sdp_session_t *sdp;
10155 if (!(parser = sdp_parse(NULL, r_sdp, (
int) strlen(r_sdp), 0))) {
10159 if (!(sdp = sdp_session(parser))) {
10160 sdp_parser_free(parser);
10167 for (m = sdp->sdp_media; m; m = m->m_next) {
10168 if (m->m_proto == sdp_proto_rtp) {
10171 for (map = m->m_rtpmaps; map; map = map->rm_next) {
10172 if (map->rm_encoding) {
10174 char key[128] =
"";
10177 if (map->rm_fmtp) {
10178 if ((br = strstr(map->rm_fmtp,
"bitrate="))) {
10193 if (map->rm_fmtp) {
10201 sdp_parser_free(parser);
10227 switch_snprintf(buf + strlen(buf), buflen - strlen(buf),
"a=rtcp-fb:%d ccm fir\r\n", pt);
10231 switch_snprintf(buf + strlen(buf), buflen - strlen(buf),
"a=rtcp-fb:%d ccm tmmbr\r\n", pt);
10235 switch_snprintf(buf + strlen(buf), buflen - strlen(buf),
"a=rtcp-fb:%d nack\r\n", pt);
10239 switch_snprintf(buf + strlen(buf), buflen - strlen(buf),
"a=rtcp-fb:%d nack pli\r\n", pt);
10245 #define SDPBUFLEN 65536 10251 uint32_t v_port, t_port;
10254 const char *family;
10259 char srbuf[128] =
"";
10260 const char *var_val;
10261 const char *username;
10262 const char *fmtp_out;
10276 int bw = 256, i = 0;
10277 uint8_t fir = 0, nack = 0, pli = 0, tmmbr = 0, has_vid = 0;
10278 const char *use_rtcp_mux = NULL;
10279 int include_external;
10385 if (smh->
rates[j] == 0) {
10416 char *orig_fmtp = NULL;
10427 if (orig_session &&
10438 if (!
zstr(orig_fmtp)) {
10479 if (orig_session) {
10485 if (fmtp_out_var) {
10486 fmtp_out = fmtp_out_var;
10495 if (!force && !ip &&
zstr(sr)
10550 if (!strcasecmp(sr,
"sendonly") || !strcasecmp(sr,
"recvonly") || !strcasecmp(sr,
"sendrecv") || !strcasecmp(sr,
"inactive")) {
10583 family = strchr(ip,
':') ?
"IP6" :
"IP4";
10586 "o=%s %010u %010u IN %s %s\r\n" 10679 switch_snprintf(buf + strlen(buf),
SDPBUFLEN - strlen(buf),
"a=rtpmap:%d telephone-event/%d\r\na=fmtp:%d 0-%d\r\n",
10694 if (append_audio) {
10726 char tmp1[11] =
"";
10727 char tmp2[11] =
"";
10728 char tmp3[11] =
"";
10729 uint32_t c1 = (1<<24)*126 + (1<<8)*65535 + (1<<0)*(256 - 1);
10730 uint32_t c2 = c1 - 1;
10731 uint32_t c3 = c1 - 2;
10732 uint32_t c4 = c1 - 3;
10741 ice_out = &a_engine->
ice_out;
10748 switch_snprintf(buf + strlen(buf),
SDPBUFLEN - strlen(buf),
"a=candidate:%s 1 %s %u %s %d typ host generation 0\r\n",
10754 switch_snprintf(buf + strlen(buf),
SDPBUFLEN - strlen(buf),
"a=candidate:%s 1 %s %u %s %d typ host generation 0\r\n",
10764 switch_snprintf(buf + strlen(buf),
SDPBUFLEN - strlen(buf),
"a=candidate:%s 1 %s %u %s %d typ srflx raddr %s rport %d generation 0\r\n",
10774 switch_snprintf(buf + strlen(buf),
SDPBUFLEN - strlen(buf),
"a=candidate:%s 2 %s %u %s %d typ host generation 0\r\n",
10780 switch_snprintf(buf + strlen(buf),
SDPBUFLEN - strlen(buf),
"a=candidate:%s 2 %s %u %s %d typ host generation 0\r\n",
10792 switch_snprintf(buf + strlen(buf),
SDPBUFLEN - strlen(buf),
"a=candidate:%s 2 %s %u %s %d typ srflx raddr %s rport %d generation 0\r\n",
10825 int cur_ptime = 0, this_ptime = 0, cng_type = 0;
10845 generate_m(session, buf,
SDPBUFLEN, port, family, ip, 0, append_audio, sr, use_cng, cng_type, map, 1, sdp_type);
10846 bp = (buf + strlen(buf));
10855 generate_m(session, bp,
SDPBUFLEN - strlen(buf), port, family, ip, 0, append_audio, sr, use_cng, cng_type, map, 0, sdp_type);
10869 if (!strcasecmp(imp->
iananame,
"ilbc") || !strcasecmp(imp->
iananame,
"isac")) {
10873 if (cur_ptime != this_ptime) {
10877 cur_ptime = this_ptime;
10881 generate_m(session, bp,
SDPBUFLEN - strlen(buf), port, family, ip, cur_ptime, append_audio, sr, use_cng, cng_type, map, 1, sdp_type);
10882 bp = (buf + strlen(buf));
10894 generate_m(session, bp,
SDPBUFLEN - strlen(buf), port, family, ip, cur_ptime, append_audio, sr, use_cng, cng_type, map, 0, sdp_type);
10957 for (loops = 0; loops < 2; loops++) {
10991 int already_did[128] = { 0 };
11076 pass_fmtp = ov_fmtp;
11105 if (append_video) {
11110 int already_did[128] = { 0 };
11127 if (ianacode < 128) {
11128 if (already_did[ianacode]) {
11131 already_did[ianacode] = 1;
11148 if (channels > 1) {
11158 if (!
zstr(ov_fmtp)) {
11159 fmtp = (
char *) ov_fmtp;
11167 fmtp = smh->
fmtp[i];
11168 }
else if (smh->
fmtps[i]) {
11169 fmtp = smh->
fmtps[i];
11174 if (
zstr(fmtp)) fmtp = (
char *) pass_fmtp;
11177 if (!
zstr(fmtp) && strcasecmp(fmtp,
"_blank_")) {
11230 v_engine->
nack || nack, v_engine->
pli || pli, v_engine->
tmmbr || tmmbr);
11243 int already_did[128] = { 0 };
11272 char tmp1[11] =
"";
11273 char tmp2[11] =
"";
11274 char tmp3[11] =
"";
11275 uint32_t c1 = (1<<24)*126 + (1<<8)*65535 + (1<<0)*(256 - 1);
11276 uint32_t c2 = c1 - 1;
11277 uint32_t c3 = c1 - 2;
11278 uint32_t c4 = c1 - 3;
11287 ice_out = &v_engine->
ice_out;
11301 switch_snprintf(buf + strlen(buf),
SDPBUFLEN - strlen(buf),
"a=candidate:%s 1 %s %u %s %d typ host generation 0\r\n",
11307 switch_snprintf(buf + strlen(buf),
SDPBUFLEN - strlen(buf),
"a=candidate:%s 1 %s %u %s %d typ host generation 0\r\n",
11317 switch_snprintf(buf + strlen(buf),
SDPBUFLEN - strlen(buf),
"a=candidate:%s 1 %s %u %s %d typ srflx raddr %s rport %d generation 0\r\n",
11327 switch_snprintf(buf + strlen(buf),
SDPBUFLEN - strlen(buf),
"a=candidate:%s 2 %s %u %s %d typ host generation 0\r\n",
11333 switch_snprintf(buf + strlen(buf),
SDPBUFLEN - strlen(buf),
"a=candidate:%s 2 %s %u %s %d typ host generation 0\r\n",
11344 switch_snprintf(buf + strlen(buf),
SDPBUFLEN - strlen(buf),
"a=candidate:%s 2 %s %u %s %d typ srflx generation 0\r\n",
11398 if (want_msrp || want_msrps) {
11423 "msrp%s://%s:%d/%s;tcp",
11424 msrp_session->
secure ?
"s" :
"",
11429 "m=message %d TCP/%sMSRP *\r\n" 11431 "a=accept-types:%s\r\n" 11432 "a=accept-wrapped-types:%s\r\n" 11435 msrp_session->
secure ?
"TLS/" :
"",
11439 msrp_session->
active ?
"active" :
"passive");
11446 msrp_session->
active = 1;
11452 "msrp%s://%s:%d/%s;tcp",
11453 msrp_session->
secure ?
"s" :
"",
11458 "m=message %d TCP/%sMSRP *\r\n" 11460 "a=accept-types:message/cpim text/* application/im-iscomposing+xml\r\n" 11461 "a=accept-wrapped-types:*\r\n" 11464 msrp_session->
secure ?
"TLS/" :
"",
11466 msrp_session->
active ?
"active" :
"passive");
11468 if (!
zstr(file_selector)) {
11470 "a=sendonly\r\na=file-selector:%s\r\n", file_selector);
11537 for (loops = 0; loops < 2; loops++) {
11580 if (!strcasecmp(pmap->
iananame,
"t140")) {
11584 if (!strcasecmp(pmap->
iananame,
"red")) {
11632 char tmp1[11] =
"";
11633 char tmp2[11] =
"";
11634 uint32_t c1 = (1<<24)*126 + (1<<8)*65535 + (1<<0)*(256 - 1);
11635 uint32_t c2 = c1 - 1;
11636 uint32_t c3 = c1 - 2;
11637 uint32_t c4 = c1 - 3;
11644 ice_out = &t_engine->
ice_out;
11658 switch_snprintf(buf + strlen(buf),
SDPBUFLEN - strlen(buf),
"a=candidate:%s 1 %s %u %s %d typ host generation 0\r\n",
11667 switch_snprintf(buf + strlen(buf),
SDPBUFLEN - strlen(buf),
"a=candidate:%s 1 %s %u %s %d typ srflx raddr %s rport %d generation 0\r\n",
11677 switch_snprintf(buf + strlen(buf),
SDPBUFLEN - strlen(buf),
"a=candidate:%s 2 %s %u %s %d typ host generation 0\r\n",
11687 switch_snprintf(buf + strlen(buf),
SDPBUFLEN - strlen(buf),
"a=candidate:%s 2 %s %u %s %d typ srflx generation 0\r\n",
11747 const char *sdp_str;
11760 sdp_parser_t *parser;
11761 sdp_session_t *sdp;
11763 sdp_connection_t *connection;
11765 if ((parser = sdp_parse(NULL, sdp_str, (
int) strlen(sdp_str), 0))) {
11766 if ((sdp = sdp_session(parser))) {
11767 for (m = sdp->sdp_media; m; m = m->m_next) {
11768 if (m->m_type != sdp_media_audio || !m->m_port) {
11772 connection = sdp->sdp_connection;
11773 if (m->m_connections) {
11774 connection = m->m_connections;
11786 sdp_parser_free(parser);
11796 for (x = 0; x < smh->
rej_idx; x++) {
11808 char buf[2048] =
"";
11809 char max_buf[128] =
"";
11810 char max_data[128] =
"";
11813 const char *family =
"IP4";
11814 const char *username;
11815 const char *bit_removal_on =
"a=T38FaxFillBitRemoval\r\n";
11816 const char *bit_removal_off =
"";
11818 const char *mmr_on =
"a=T38FaxTranscodingMMR\r\n";
11819 const char *mmr_off =
"";
11821 const char *jbig_on =
"a=T38FaxTranscodingJBIG\r\n";
11822 const char *jbig_off =
"";
11824 int broken_boolean;
11881 family = strchr(ip,
':') ?
"IP6" :
"IP4";
11886 "o=%s %010u %010u IN %s %s\r\n" 11887 "s=%s\r\n" "c=IN %s %s\r\n" "t=0 0\r\n", username, smh->
owner_id, smh->
session_id, family, ip, username, family, ip);
11900 if (broken_boolean) {
11901 bit_removal_on =
"a=T38FaxFillBitRemoval:1\r\n";
11902 bit_removal_off =
"a=T38FaxFillBitRemoval:0\r\n";
11904 mmr_on =
"a=T38FaxTranscodingMMR:1\r\n";
11905 mmr_off =
"a=T38FaxTranscodingMMR:0\r\n";
11907 jbig_on =
"a=T38FaxTranscodingJBIG:1\r\n";
11908 jbig_off =
"a=T38FaxTranscodingJBIG:0\r\n";
11914 "m=audio 0 RTP/AVP 0\r\n");
11918 "m=image %d udptl t38\r\n" 11919 "a=T38FaxVersion:%d\r\n" 11920 "a=T38MaxBitRate:%d\r\n" 11924 "a=T38FaxRateManagement:%s\r\n" 11927 "a=T38FaxUdpEC:%s\r\n",
11945 switch_snprintf(buf + strlen(buf),
sizeof(buf) - strlen(buf),
"m=audio 0 RTP/AVP 19\r\n");
11963 char *p, *q, *pe, *qe;
11964 int has_video = 0, has_audio = 0, has_text = 0, has_ip = 0;
11965 char port_buf[25] =
"";
11966 char vport_buf[25] =
"";
11967 char tport_buf[25] =
"";
12031 pe = p + strlen(p);
12046 if (a_engine->
local_sdp_ip && !strncmp(
"c=IN IP", p, 7)) {
12050 memcpy(q, strchr(a_engine->
adv_sdp_ip,
':') ?
"6 " :
"4 ", 2);
12053 snprintf(q, qe - q,
"%s", a_engine->
adv_sdp_ip);
12056 while (p && *p && ((*p >=
'0' && *p <=
'9') || *p ==
'.' || *p ==
':' || (*p >=
'A' && *p <=
'F') || (*p >=
'a' && *p <=
'f'))) {
12066 }
else if (!strncmp(
"o=", p, 2)) {
12067 char *oe = strchr(p,
'\n');
12071 const char *family =
"IP4";
12072 char o_line[1024] =
"";
12083 family = strchr(smh->
mparams->
sipip,
':') ?
"IP6" :
"IP4";
12096 snprintf(o_line,
sizeof(o_line),
"o=%s %010u %010u IN %s %s\r\n",
12099 snprintf(q, qe-q,
"%s", o_line);
12100 q += strlen(o_line) - 1;
12104 }
else if (!strncmp(
"s=", p, 2)) {
12105 char *se = strchr(p,
'\n');
12109 char s_line[1024] =
"";
12120 snprintf(q, qe-q,
"%s", s_line);
12122 q += strlen(s_line) - 1;
12126 }
else if ((!strncmp(
"m=audio ", p, 8) && *(p + 8) !=
'0') || (!strncmp(
"m=image ", p, 8) && *(p + 8) !=
'0')) {
12144 snprintf(q, qe - q,
"%s", port_buf);
12145 q += strlen(port_buf);
12152 while (p && *p && (*p >=
'0' && *p <=
'9')) {
12162 }
else if (!strncmp(
"m=video ", p, 8) && *(p + 8) !=
'0') {
12206 snprintf(q, qe-q,
"%s", vport_buf);
12207 q += strlen(vport_buf);
12214 while (p && *p && (*p >=
'0' && *p <=
'9')) {
12225 }
else if (!strncmp(
"m=text ", p, 7) && *(p + 7) !=
'0') {
12269 snprintf(q, qe-q,
"%s", tport_buf);
12270 q += strlen(tport_buf);
12277 while (p && *p && (*p >=
'0' && *p <=
'9')) {
12291 while (p && *p && *p !=
'\n') {
12333 if (!has_ip && !has_audio) {
12369 const char *err, *val;
12374 if (!t38_options || !t38_options->
remote_ip) {
12379 if (remote_host && remote_port && remote_port == t38_options->
remote_port && !strcmp(remote_host, t38_options->
remote_ip)) {
12405 msg.
from = __FILE__;
12414 cJSON *ret = NULL, *n = NULL;
12416 if ((!ideal && !max)) {
12417 ret = cJSON_CreateNumber(min);
12419 ret = cJSON_CreateObject();
12420 n = cJSON_CreateNumber(min);
12421 cJSON_AddItemToObject(ret,
"min", n);
12424 n = cJSON_CreateNumber(ideal);
12425 cJSON_AddItemToObject(ret,
"ideal", n);
12429 n = cJSON_CreateNumber(max);
12430 cJSON_AddItemToObject(ret,
"max", n);
12440 float min = 0, ideal = 0, max = 0;
12445 min = atof(argv[0]);
12449 ideal = atof(argv[1]);
12453 max = atof(argv[2]);
12463 char *parse = NULL;
12468 char *aspect = NULL, *
fps = NULL, *width = NULL, *height = NULL, *jtmp = NULL;
12480 parse = strdup(json);
12483 for(i = 0; i < argc; i++) {
12485 if ((val = strchr(name,
'='))) {
12490 if (!strcmp(name,
"aspect")) {
12493 }
else if (!strcmp(name,
"fps")) {
12496 }
else if (!strcmp(name,
"width")) {
12499 }
else if (!strcmp(name,
"height")) {
12506 obj = cJSON_CreateObject();
12509 video = cJSON_CreateObject();
12513 cJSON_AddItemToObject(video,
"frameRate", p);
12518 cJSON_AddItemToObject(video,
"width", p);
12523 cJSON_AddItemToObject(video,
"height", p);
12527 p = cJSON_CreateNumber(atof(aspect));
12528 cJSON_AddItemToObject(video,
"aspectRatio", p);
12531 cJSON_AddItemToObject(obj,
"video", video);
12534 jtmp = cJSON_PrintUnformatted(obj);
12540 msg.
from = __FILE__;
12592 engine = &smh->
engines[type];
12622 engine = &smh->
engines[type];
12636 uint32_t new_bitrate;
12646 engine = &smh->
engines[type];
12648 new_bitrate = bitrate - bitrate * engine->
bw_mult;
12711 engine = &smh->
engines[type];
12874 void *reply = NULL;
12884 engine = &smh->
engines[type];
12910 if (direction && *direction ==
'v') {
12913 }
else if (direction && *direction ==
't' && t_engine) {
12920 int both = !strcasecmp(direction,
"both");
12923 if (both || !strcasecmp(direction,
"read")) {
12928 if (both || !strcasecmp(direction,
"write")) {
13113 const char *
ip = NULL, *port = NULL;
13480 sdp_parser_t *parser;
13481 sdp_session_t *sdp;
13494 if (
zstr(codec_string)) {
13498 if ((parser = sdp_parse(NULL, r_sdp, (
int) strlen(r_sdp), 0))) {
13500 if ((sdp = sdp_session(parser))) {
13504 sdp_parser_free(parser);
13511 int codec_ms = ptime;
13512 uint32_t map_bit_rate = 0, map_channels = 1;
13513 char ptstr[20] =
"";
13514 char ratestr[20] =
"";
13515 char bitstr[20] =
"";
13522 map_channels = map->rm_params ? atoi(map->rm_params) : 1;
13525 if (!ptime && !strcasecmp(map->rm_encoding,
"g723")) {
13529 if (
zstr(map->rm_fmtp)) {
13530 if (!strcasecmp(map->rm_encoding,
"ilbc")) {
13532 map_bit_rate = 13330;
13533 }
else if (!strcasecmp(map->rm_encoding,
"isac")) {
13535 map_bit_rate = 32000;
13548 if (map->rm_rate) {
13549 switch_snprintf(ratestr,
sizeof(ratestr),
"@%uh", (
unsigned int) map->rm_rate);
13556 if (map_bit_rate) {
13560 if (map_channels > 1) {
13564 switch_snprintf(buf + strlen(buf), buflen - strlen(buf),
",%s.%s%s%s%s", imp->
modname, map->rm_encoding, ratestr, ptstr, bitstr);
13571 char buf[1024] = { 0 };
13573 sdp_attribute_t *attr;
13574 int ptime = 0, dptime = 0;
13575 sdp_connection_t *connection;
13577 short int match = 0;
13579 int already_did[128] = { 0 };
13580 int num_codecs = 0;
13586 int prefer_sdp = 0;
13602 if (!
zstr(codec_string)) {
13603 char *tmp_codec_string;
13605 if (*codec_string ==
'=') codec_string++;
13607 if ((tmp_codec_string = strdup(codec_string))) {
13616 if (!channel || !num_codecs) {
13620 for (attr = sdp->sdp_attributes; attr; attr = attr->a_next) {
13621 if (
zstr(attr->a_name)) {
13625 if (!strcasecmp(attr->a_name,
"ptime")) {
13626 dptime = atoi(attr->a_value);
13631 for (m = sdp->sdp_media; m; m = m->m_next) {
13634 if ((m->m_type == sdp_media_audio || m->m_type == sdp_media_video) && m->m_port) {
13635 for (map = m->m_rtpmaps; map; map = map->rm_next) {
13638 for (attr = m->m_attributes; attr && found < 2; attr = attr->a_next) {
13639 if (
zstr(attr->a_name)) {
13643 if (!strcasecmp(attr->a_name,
"ptime") && attr->a_value) {
13644 ptime = atoi(attr->a_value);
13648 if (!strcasecmp(attr->a_name,
"rtcp-mux")) {
13667 map->rm_params ? atoi(map->rm_params) : 1,
13673 for (m = sdp->sdp_media; m; m = m->m_next) {
13676 if (m->m_type == sdp_media_image && m->m_port) {
13677 switch_snprintf(buf + strlen(buf),
sizeof(buf) - strlen(buf),
",t38");
13678 }
else if (m->m_type == sdp_media_audio && m->m_port) {
13679 for (attr = m->m_attributes; attr; attr = attr->a_next) {
13680 if (
zstr(attr->a_name)) {
13684 if (!strcasecmp(attr->a_name,
"ptime") && attr->a_value) {
13685 ptime = atoi(attr->a_value);
13690 connection = sdp->sdp_connection;
13691 if (m->m_connections) {
13692 connection = m->m_connections;
13701 for (map = m->m_rtpmaps; map; map = map->rm_next) {
13703 if (already_did[map->rm_pt]) {
13707 for (i = 0; i < num_codecs; i++) {
13711 match = (map->rm_pt == imp->
ianacode) ? 1 : 0;
13713 if (map->rm_encoding) {
13714 match = !strcasecmp(map->rm_encoding, imp->
iananame) &&
13715 ((map->rm_pt < 96 && imp->
ianacode < 96) || (map->rm_pt > 95 && imp->
ianacode > 95));
13728 for (i = 0; i < num_codecs; i++) {
13735 for (map = m->m_rtpmaps; map; map = map->rm_next) {
13737 if (already_did[map->rm_pt]) {
13742 match = (map->rm_pt == imp->
ianacode) ? 1 : 0;
13744 if (map->rm_encoding) {
13745 match = !strcasecmp(map->rm_encoding, imp->
iananame) &&
13746 ((map->rm_pt < 96 && imp->
ianacode < 96) || (map->rm_pt > 95 && imp->
ianacode > 95));
13759 }
else if (m->m_type == sdp_media_video && m->m_port) {
13760 connection = sdp->sdp_connection;
13761 if (m->m_connections) {
13762 connection = m->m_connections;
13771 for (map = m->m_rtpmaps; map; map = map->rm_next) {
13773 if (already_did[map->rm_pt]) {
13777 for (i = 0; i < num_codecs; i++) {
13781 match = (map->rm_pt == imp->
ianacode) ? 1 : 0;
13783 if (map->rm_encoding) {
13784 match = !strcasecmp(map->rm_encoding, imp->
iananame) &&
13785 ((map->rm_pt < 96 && imp->
ianacode < 96) || (map->rm_pt > 95 && imp->
ianacode > 95));
13792 if (map->rm_fmtp) {
13803 for (i = 0; i < num_codecs; i++) {
13815 for (map = m->m_rtpmaps; map; map = map->rm_next) {
13817 if (already_did[map->rm_pt]) {
13822 match = (map->rm_pt == imp->
ianacode) ? 1 : 0;
13824 if (map->rm_encoding) {
13825 match = !strcasecmp(map->rm_encoding, imp->
iananame) &&
13826 ((map->rm_pt < 96 && imp->
ianacode < 96) || (map->rm_pt > 95 && imp->
ianacode > 95));
13833 if (map->rm_fmtp) {
13844 }
else if (m->m_proto == sdp_proto_msrp) {
13846 }
else if (m->m_proto == sdp_proto_msrps) {
13848 }
else if (m->m_type == sdp_media_text) {
13853 if (buf[0] ==
',') {
13870 engine = &smh->
engines[type];
13971 const char *r_sdp = NULL;
14058 const char *r_port;
14149 if (r_ip && r_port) {
14185 if (r_ip && r_port) {
14203 int idx = atoi(tmp);
14240 if (video_globals.
pool) {
14247 if (!strcasecmp(name,
"pcmu")) {
14251 if (!strcasecmp(name,
"pcma")) {
14255 if (!strcasecmp(name,
"gsm")) {
14259 if (!strcasecmp(name,
"g722")) {
14263 if (!strcasecmp(name,
"g729")) {
14267 if (!strcasecmp(name,
"dvi4")) {
14271 if (!strcasecmp(name,
"h261")) {
14275 if (!strcasecmp(name,
"h263")) {
14287 if (p <
end_of_p(sdp) && *(p+strlen(name)) ==
'/' && *(p-1) ==
' ') {
14290 while(*p > 47 && *p < 58) {
14307 char *new_sdp = NULL;
14308 int pt = -1, te = -1;
14312 int in_m = 0, slash = 0;
14313 int number = 0, skip = 0;
14314 int remove = !strcasecmp(cmd,
"remove");
14315 int only = !strcasecmp(cmd,
"only");
14316 char *end =
end_of_p((
char *)sdp_str);
14321 if (
remove || only) {
14332 te =
find_pt(sdp_str,
"telephone-event");
14336 len = strlen(sdp_str) + 2;
14337 new_sdp = malloc(len);
14342 while(i && *i && i < end) {
14344 if (*i ==
'm' && *(i+1) ==
'=') {
14349 if (*i ==
'\r' || *i ==
'\n') {
14355 while(*i !=
' ' && i < end) {
14367 while(i < end && ((*i > 47 && *i < 58) || *i ==
' ')) {
14370 tst = (number != pt);
14372 tst = (number == pt || number == te);
14387 tst = (number == pt);
14389 tst = (number != pt && number != te);
14399 while (i < end && !strncasecmp(i,
"a=rtpmap:", 9)) {
14400 const char *t = i + 9;
14405 tst = (number == pt);
14407 tst = (number != pt && number != te);
14410 while(i < end && (*i !=
'\r' && *i !=
'\n')) {
14411 if (!tst) *o++ = *i;
14415 while(i < end && (*i ==
'\r' || *i ==
'\n')) {
14416 if (!tst) *o++ = *i;
14421 while (i < end && !strncasecmp(i,
"a=fmtp:", 7)) {
14422 const char *t = i + 7;
14427 tst = (number == pt);
14429 tst = (number != pt && number != te);
14432 while(i < end && (*i !=
'\r' && *i !=
'\n')) {
14433 if (!tst) *o++ = *i;
14437 while(i < end && (*i ==
'\r' || *i ==
'\n')) {
14438 if (!tst) *o++ = *i;
14464 char *patched_sdp = NULL;
14468 for (x = 0; x < argc; x++) {
14469 char *command = argv[x];
14470 char *arg = strchr(command,
'(');
14481 char *tmp_sdp = NULL;
14491 "%s Filter command %s(%s)\nFROM:\n==========\n%s\nTO:\n==========\n%s\n\n",
14493 command, arg, patched_sdp ? patched_sdp : sdp, tmp_sdp);
14498 patched_sdp = tmp_sdp;
14503 return patched_sdp;
14519 if (!(engine = &smh->
engines[mtype])) {
14557 msg.
from = __FILE__;
14599 if (!(engine = &smh->
engines[mtype])) {
14639 if (!(engine = &smh->
engines[mtype])) {
14669 for (ptr = session->
event_hooks.video_write_frame; ptr; ptr = ptr->
next) {
14789 if (vid_params.
width && vid_params.
height && ((vid_params.
width != img->d_w) || (vid_params.
height != img->d_h))) {
14791 if (!(img = dup_img)) {
14802 if (session->
bugs) {
14808 for (bp = session->
bugs; bp; bp = bp->
next) {
14837 if (bp->
ready && img &&
14841 bug_frame.
img = img;
14852 if (bug_frame.
img && bug_frame.
img != img) {
14854 img = bug_frame.
img;
14879 write_frame = *frame;
14880 frame = &write_frame;
14913 if (frame->
datalen == 0)
break;
14964 if (video_globals.
synced &&
14995 if (read_impl_a.
impl_id && read_impl_b.impl_id) {
15008 return transcoding;
15050 int is_keyframe = 0;
15066 for (ptr = session->
event_hooks.video_read_frame; ptr; ptr = ptr->
next) {
15099 "VIDEO: seq: %d ts: %u len: %4d %02x %02x %02x %02x %02x %02x %02x %02x %02x %02x %02x mark: %d\n",
15100 (*frame)->seq, (*frame)->timestamp, (*frame)->datalen,
15101 *((uint8_t *)(*frame)->data), *((uint8_t *)(*frame)->data + 1),
15102 *((uint8_t *)(*frame)->data + 2), *((uint8_t *)(*frame)->data + 3),
15103 *((uint8_t *)(*frame)->data + 4), *((uint8_t *)(*frame)->data + 5),
15104 *((uint8_t *)(*frame)->data + 6), *((uint8_t *)(*frame)->data + 7),
15105 *((uint8_t *)(*frame)->data + 8), *((uint8_t *)(*frame)->data + 9),
15106 *((uint8_t *)(*frame)->data + 10), (*frame)->m);
15119 (*frame)->img = NULL;
15127 (*frame)->img->w, (*frame)->img->h, (*frame)->img->d_w, (*frame)->img->d_h);
15130 if ((*frame)->img && (*frame)->img->d_w && (*frame)->img->d_h) {
15131 int new_w = 0, new_h = 0;
15134 new_w = (*frame)->img->d_w;
15135 new_h = (*frame)->img->d_h;
15137 if (new_w && new_h) {
15152 if (!(*frame)->img) {
15166 if ((*frame)->img) {
15169 }
else if ((*frame)->m || ++smh->
ready_loops > 5) {
15176 if (*frame && is_keyframe) {
15180 if (session->
bugs) {
15186 for (bp = session->
bugs; bp; bp = bp->
next) {
15203 if ((*frame) && (*frame)->img) {
15214 if (bp->
ready && (*frame) && (*frame)->img &&
15246 if ((*frame) && (*frame)->codec) {
15247 (*frame)->pmap = NULL;
15383 if (read_text_frame) {
15385 for (ptr = session->
event_hooks.text_read_frame; ptr; ptr = ptr->
next) {
15416 unsigned char *p = (*frame)->data;
15423 if (*p ==
'\r' || *p ==
'\n') {
15428 if (*p == 0xE2 && *(p+1) == 0x80 && *(p+2) == 0xA8) {
15437 if ((*frame)->data && (*frame)->datalen && !((*frame)->flags &
SFF_CNG)) {
15445 if (session->
bugs) {
15450 for (bp = session->
bugs; bp; bp = bp->
next) {
15472 if ((*frame)->data && (*frame)->datalen && !((*frame)->flags &
SFF_CNG)) {
15484 void *tmp = malloc(inuse + 1024);
15524 void *data = (*frame)->data;
15525 char eof[1] = {
'\0'};
15587 uint32_t plen = 0, loops = 0;
15601 *buf = t_engine->
t140_pt & 0x7f;
15607 u16 = (uint16_t *) buf;
15608 *u16 = htons(ts << 2);
15610 *buf += (len & 0x300) >> 8;
15617 if (pos == t_engine->
tf->
red_pos)
break;
15622 if (pos == t_engine->
tf->
red_max) pos = 0;
15626 plen = ((loops - 1) * 4) + 1;
15629 if (pos == t_engine->
tf->
red_max) pos = 0;
15638 if (pos == t_engine->
tf->
red_pos)
break;
15642 if (pos == t_engine->
tf->
red_max) pos = 0;
15647 *(buf+plen) =
'\0';
15693 if (!t_engine || !t_engine->
tf) {
15699 char *str = (
char *) frame->
data;
15718 for(pos = 0; pos < t_engine->
tf->
red_max; pos++) {
15749 if (write_text_frame) {
15751 for (ptr = session->
event_hooks.text_write_frame; ptr; ptr = ptr->
next) {
15761 if (!t_engine || (t_engine->
red_pt && !t_engine->
tf)) {
15800 frame.
datalen = strlen(data);
15823 frame.
data = (
char *) data;
15824 frame.
datalen = strlen(data);
15845 unsigned int flag = 0, need_codec = 0, perfect = 0, do_bugs = 0, do_write = 0, do_resample = 0, ptime_mismatch = 0, pass_cng = 0, resample = 0;
15846 int did_write_resample = 0;
15918 ptime_mismatch =
TRUE;
15944 do_resample =
TRUE;
15976 write_frame = frame;
15988 if (frame->
codec) {
16072 write_frame = frame;
16096 short *data = write_frame->data;
16122 did_write_resample = 1;
16129 if (session->
bugs) {
16134 for (bp = session->
bugs; bp; bp = bp->
next) {
16193 write_frame = frame;
16198 if (!ptime_mismatch && write_frame->codec && write_frame->codec->implementation &&
16212 enc_frame = write_frame;
16254 enc_frame->
m = frame->
m;
16255 enc_frame->
seq = frame->
seq;
16258 write_frame = enc_frame;
16262 write_frame = NULL;
16267 write_frame = NULL;
16274 status =
perform_write(session, write_frame, flags, stream_id);
16280 "Engaging Write Buffer at %u bytes to accommodate %u->%u\n",
16408 enc_frame->
m = frame->
m;
16411 write_frame = enc_frame;
16415 write_frame = NULL;
16420 write_frame = NULL;
16425 short *data = write_frame->data;
16443 if (ptime_mismatch || resample) {
16444 write_frame->timestamp = 0;
16463 if (ptime_mismatch || resample) {
16464 write_frame->timestamp = 0;
16468 status =
perform_write(session, write_frame, flags, stream_id);
16482 if (!session)
return NULL;
16491 if (!engine)
return NULL;
struct switch_video_codec_settings video
switch_status_t switch_channel_set_variable_printf(switch_channel_t *channel, const char *varname, const char *fmt,...)
switch_mutex_t * codec_init_mutex
switch_status_t switch_core_codec_decode_video(switch_codec_t *codec, switch_frame_t *frame)
Decode video data using a codec handle.
#define SWITCH_MAX_CODECS
switch_core_session_t * session
switch_time_t switch_micro_time_now(void)
Get the current epoch time in microseconds.
#define switch_event_fire(event)
Fire an event filling in most of the arguements with obvious values.
#define switch_channel_hangup(channel, hangup_cause)
Hangup a channel flagging it's state machine to end.
switch_status_t switch_channel_execute_on(switch_channel_t *channel, const char *variable_prefix)
void switch_core_codec_unlock_full(switch_core_session_t *session)
Unlock codec read mutex and codec write mutex.
#define switch_core_new_memory_pool(p)
Create a new sub memory pool from the core's master pool.
void switch_channel_set_flag_recursive(switch_channel_t *channel, switch_channel_flag_t flag)
uint32_t T38FaxMaxDatagram
switch_status_t switch_thread_rwlock_unlock(switch_thread_rwlock_t *rwlock)
switch_status_t switch_rtp_activate_rtcp(switch_rtp_t *rtp_session, int send_rate, switch_port_t remote_port, switch_bool_t mux)
Activate sending RTCP Sender Reports (SR's)
char * switch_core_session_sprintf(_In_ switch_core_session_t *session, _In_z_ _Printf_format_string_ const char *fmt,...)
printf-style style printing routine. The data is output to a string allocated from the session ...
unsigned char remote_raw_key[SWITCH_RTP_MAX_CRYPTO_LEN]
switch_status_t switch_msrp_session_destroy(switch_msrp_session_t **ms)
switch_status_t switch_thread_cond_create(switch_thread_cond_t **cond, switch_memory_pool_t *pool)
switch_audio_resampler_t * write_resampler
switch_frame_t dummy_cng_frame
#define SWITCH_BUFFER_START_FRAMES
static switch_bool_t switch_true(const char *expr)
Evaluate the truthfullness of a string expression.
void switch_ivr_bg_media(const char *uuid, switch_media_flag_t flags, switch_bool_t on, switch_bool_t is3p, uint32_t delay)
#define SWITCH_CHANNEL_SESSION_LOG(x)
switch_status_t switch_core_codec_parse_fmtp(const char *codec_name, const char *fmtp, uint32_t rate, switch_codec_fmtp_t *codec_fmtp)
void switch_rtp_set_media_timeout(switch_rtp_t *rtp_session, uint32_t ms)
switch_thread_rwlock_t * bug_rwlock
#define switch_set_flag(obj, flag)
Set a flag on an arbitrary object.
switch_rtp_numbers_t inbound
uint8_t switch_rtp_ready(switch_rtp_t *rtp_session)
Test if an RTP session is ready.
switch_status_t switch_core_timer_init(switch_timer_t *timer, const char *timer_name, int interval, int samples, switch_memory_pool_t *pool)
Request a timer handle using given time module.
switch_core_session_message_types_t message_id
switch_status_t switch_frame_buffer_free(switch_frame_buffer_t *fb, switch_frame_t **frameP)
#define SWITCH_CHANNEL_LOG
void switch_core_session_reset(_In_ switch_core_session_t *session, switch_bool_t flush_dtmf, switch_bool_t reset_read_codec)
Reset the buffers and resampler on a session.
switch_size_t switch_buffer_read(_In_ switch_buffer_t *buffer, _In_ void *data, _In_ switch_size_t datalen)
Read data from a switch_buffer_t up to the ammount of datalen if it is available. Remove read data fr...
switch_rtp_crypto_key_type_t
void switch_core_session_passthru(switch_core_session_t *session, switch_media_type_t type, switch_bool_t on)
switch_status_t switch_mutex_trylock(switch_mutex_t *lock)
switch_status_t switch_channel_set_variable_name_printf(switch_channel_t *channel, const char *val, const char *fmt,...)
const char *const const double number
switch_status_t switch_channel_queue_dtmf(_In_ switch_channel_t *channel, _In_ const switch_dtmf_t *dtmf)
Queue DTMF on a given channel.
uint32_t switch_channel_test_flag_partner(switch_channel_t *channel, switch_channel_flag_t flag)
char * local_accept_types
struct switch_io_event_hook_text_write_frame * next
char * switch_find_end_paren(const char *s, char open, char close)
void switch_img_free(switch_image_t **img)
Close an image descriptor.
const char * switch_msrp_listen_ip(void)
switch_mutex_t * codec_write_mutex
const char * switch_channel_get_partner_uuid(switch_channel_t *channel)
switch_status_t switch_channel_set_private(switch_channel_t *channel, const char *key, const void *private_info)
Set private data on channel.
switch_status_t switch_core_session_set_video_read_callback(switch_core_session_t *session, switch_core_video_thread_callback_func_t func, void *user_data)
void switch_rtp_reset(switch_rtp_t *rtp_session)
switch_status_t switch_buffer_create_dynamic(_Out_ switch_buffer_t **buffer, _In_ switch_size_t blocksize, _In_ switch_size_t start_len, _In_ switch_size_t max_len)
Allocate a new dynamic switch_buffer.
uint32_t switch_io_flag_t
char * remote_rtcp_ice_addr
switch_status_t switch_rtp_ack_bitrate(switch_rtp_t *rtp_session, uint32_t bps)
switch_video_write_frame_hook_t text_write_frame
int switch_loadable_module_get_codecs_sorted(const switch_codec_implementation_t **array, char fmtp_array[SWITCH_MAX_CODECS][MAX_FMTP_LEN], int arraylen, char **prefs, int preflen)
Retrieve the list of loaded codecs into an array based on another array showing the sorted order...
switch_size_t switch_b64_decode(const char *in, char *out, switch_size_t olen)
switch_buffer_t * raw_write_buffer
switch_read_frame_hook_t video_read_frame
switch_status_t switch_frame_buffer_trypop(switch_frame_buffer_t *fb, void **ptr)
#define switch_channel_stop_broadcast(_channel)
switch_status_t switch_core_session_set_video_read_codec(_In_ switch_core_session_t *session, switch_codec_t *codec)
Assign the video_read codec to a given session.
switch_rtp_t * switch_rtp_new(const char *rx_host, switch_port_t rx_port, const char *tx_host, switch_port_t tx_port, switch_payload_t payload, uint32_t samples_per_interval, uint32_t ms_per_packet, switch_rtp_flag_t flags[SWITCH_RTP_FLAG_INVALID], char *timer_name, const char **err, switch_memory_pool_t *pool, switch_port_t bundle_internal_ports, switch_port_t bundle_external_port)
prepare a new RTP session handle and fully initilize it
#define switch_channel_up(_channel)
#define SWITCH_RECOMMENDED_BUFFER_SIZE
switch_status_t switch_jb_peek_frame(switch_jb_t *jb, uint32_t ts, uint16_t seq, int peek, switch_frame_t *frame)
void switch_channel_event_set_data(_In_ switch_channel_t *channel, _In_ switch_event_t *event)
Add information about a given channel to an event object.
switch_bool_t switch_is_leading_number(const char *str)
unsigned long adv_rm_rate
#define switch_channel_presence(_a, _b, _c, _d)
#define switch_core_strdup(_pool, _todup)
Copy a string using memory allocation from a given pool.
switch_status_t switch_core_codec_encode(switch_codec_t *codec, switch_codec_t *other_codec, void *decoded_data, uint32_t decoded_data_len, uint32_t decoded_rate, void *encoded_data, uint32_t *encoded_data_len, uint32_t *encoded_rate, unsigned int *flag)
Encode data using a codec handle.
switch_size_t largest_jb_size
uint32_t media_hold_timeout
#define switch_split(_data, _delim, _array)
switch_status_t switch_core_session_set_read_impl(switch_core_session_t *session, const switch_codec_implementation_t *impp)
switch_status_t switch_ivr_broadcast(const char *uuid, const char *path, switch_media_flag_t flags)
Signal the session to broadcast audio.
switch_status_t switch_core_session_set_video_write_codec(_In_ switch_core_session_t *session, switch_codec_t *codec)
Assign the video_write codec to a given session.
#define TEXT_UNICODE_LINEFEED
switch_thread_id_t thread_write_lock
switch_status_t switch_threadattr_stacksize_set(switch_threadattr_t *attr, switch_size_t stacksize)
Node in which to store custom video_write_frame channel callback hooks.
switch_status_t switch_core_media_bug_flush_all(_In_ switch_core_session_t *session)
Flush the read/write buffers for all media bugs on the session.
switch_bool_t switch_is_uint_in_range(const char *str, unsigned int from, unsigned int to)
Check if a 32 bit unsigned number is in a range.
#define switch_core_destroy_memory_pool(p)
Returns a subpool back to the main pool.
#define SWITCH_LOCAL_VIDEO_IP_VARIABLE
uint32_t decoded_bytes_per_packet
switch_codec_t * read_codec
Representation of an event.
Node in which to store custom read frame channel callback hooks.
switch_read_frame_hook_t text_read_frame
#define switch_channel_ready(_channel)
switch_status_t switch_event_add_body(switch_event_t *event, const char *fmt,...) PRINTF_FUNCTION(2
Add a body to an event.
switch_frame_t * cur_frame
switch_io_routines_t * io_routines
struct switch_io_event_hook_video_read_frame * next
struct switch_rtcp_report_block_frame reports[MAX_REPORT_BLOCKS]
const char *const const char *const const cJSON *const value
switch_status_t switch_core_session_read_lock(_In_ switch_core_session_t *session)
Acquire a read lock on the session.
#define SWITCH_RTP_MAX_BUF_LEN
switch_rtp_crypto_direction_t
static int switch_channel_var_false(switch_channel_t *channel, const char *variable)
switch_media_flow_t smode
switch_buffer_t * write_buffer
switch_status_t switch_rtp_pause_jitter_buffer(switch_rtp_t *rtp_session, switch_bool_t pause)
switch_bool_t T38FaxTranscodingMMR
#define switch_channel_media_ready(_channel)
switch_rtp_crypto_key_param_method_type_t
const char * T38FaxRateManagement
Node in which to store custom read frame channel callback hooks.
dtls_state_t switch_rtp_dtls_state(switch_rtp_t *rtp_session, dtls_type_t type)
switch_status_t switch_core_session_video_read_callback(switch_core_session_t *session, switch_frame_t *frame)
switch_status_t switch_nat_add_mapping(switch_port_t port, switch_nat_ip_proto_t proto, switch_port_t *external_port, switch_bool_t sticky)
Maps a port through the NAT Traversal System.
unsigned int switch_ci_hashfunc_default(const char *char_key, switch_ssize_t *klen)
switch_status_t switch_rtp_set_payload_map(switch_rtp_t *rtp_session, payload_map_t **pmap)
switch_engine_function_t engine_function
switch_status_t switch_core_codec_decode(switch_codec_t *codec, switch_codec_t *other_codec, void *encoded_data, uint32_t encoded_data_len, uint32_t encoded_rate, void *decoded_data, uint32_t *decoded_data_len, uint32_t *decoded_rate, unsigned int *flag)
Decode data using a codec handle.
#define SWITCH_RESAMPLE_QUALITY
switch_video_write_frame_hook_t video_write_frame
switch_codec_settings_t codec_settings
switch_codec_implementation_t read_impl
void switch_rtp_intentional_bugs(switch_rtp_t *rtp_session, switch_rtp_bug_flag_t bugs)
switch_status_t switch_core_codec_destroy(switch_codec_t *codec)
Destroy an initalized codec handle.
switch_status_t switch_thread_cond_wait(switch_thread_cond_t *cond, switch_mutex_t *mutex)
const char * switch_channel_get_variable_dup(switch_channel_t *channel, const char *varname, switch_bool_t dup, int idx)
Retrieve a variable from a given channel.
const char * string_array_arg[MESSAGE_STRING_ARG_MAX]
#define switch_calc_video_fps(fpsP, fps)
#define switch_core_session_get_name(_s)
struct switch_io_event_hook_video_write_frame * next
uint32_t switch_core_cpu_count(void)
switch_status_t switch_core_session_queue_message(_In_ switch_core_session_t *session, _In_ switch_core_session_message_t *message)
Queue a message on a session.
char * switch_stun_host_lookup(const char *host, switch_memory_pool_t *pool)
#define switch_resample_calc_buffer_size(_to, _from, _srclen)
switch_bool_t T38FaxTranscodingJBIG
switch_frame_buffer_t * write_fb
switch_size_t media_bytes
#define SWITCH_LOCAL_VIDEO_PORT_VARIABLE
void switch_rtp_set_telephony_recv_event(switch_rtp_t *rtp_session, switch_payload_t te)
switch_status_t switch_core_media_bug_patch_spy_frame(switch_media_bug_t *bug, switch_image_t *img, switch_rw_t rw)
struct switch_crypto_key_material_s * local_key_material_next
switch_status_t switch_event_dup(switch_event_t **event, switch_event_t *todup)
Duplicate an event.
static int32_t switch_calc_bitrate(int w, int h, float quality, double fps)
icand_t cands[MAX_CAND][MAX_CAND_IDX_COUNT]
uint32_t max_missed_hold_packets
Abstract handler to a timer module.
#define SWITCH_DEFAULT_VIDEO_SIZE
switch_size_t dtmf_packet_count
void switch_resample_destroy(switch_audio_resampler_t **resampler)
Destroy an existing resampler handle.
static switch_thread_t * thread
int switch_snprintf(_Out_z_cap_(len) char *buf, _In_ switch_size_t len, _In_z_ _Printf_format_string_ const char *format,...)
switch_size_t switch_buffer_write(_In_ switch_buffer_t *buffer, _In_bytecount_(datalen) const void *data, _In_ switch_size_t datalen)
Write data into a switch_buffer_t up to the length of datalen.
struct switch_runtime runtime
switch_status_t switch_frame_buffer_dup(switch_frame_buffer_t *fb, switch_frame_t *orig, switch_frame_t **clone)
uint32_t switch_channel_test_flag(switch_channel_t *channel, switch_channel_flag_t flag)
Test for presence of given flag on a given channel.
static int32_t switch_parse_bandwidth_string(const char *bwv)
struct switch_crypto_key_material_s * next
switch_thread_id_t thread_id
switch_secure_settings_t ssec[CRYPTO_INVALID+1]
#define switch_check_network_list_ip(_ip_str, _list_name)
payload_map_t * payload_map
switch_status_t switch_frame_buffer_destroy(switch_frame_buffer_t **fbP)
switch_status_t(* switch_io_write_text_frame_t)(switch_core_session_t *, switch_frame_t *, switch_io_flag_t, int)
switch_io_read_text_frame_t read_text_frame
switch_status_t switch_core_session_read_frame(_In_ switch_core_session_t *session, switch_frame_t **frame, switch_io_flag_t flags, int stream_id)
Read a frame from a session.
static uint32_t switch_known_bitrate(switch_payload_t payload)
switch_buffer_t * text_buffer
#define SWITCH_ADVERTISED_MEDIA_IP_VARIABLE
A message object designed to allow unlike technologies to exchange data.
switch_image_t * switch_img_alloc(switch_image_t *img, switch_img_fmt_t fmt, unsigned int d_w, unsigned int d_h, unsigned int align)
Open a descriptor, allocating storage for the underlying image.
Node in which to store custom write_frame channel callback hooks.
#define SWITCH_LOCAL_TEXT_PORT_VARIABLE
void switch_rtp_set_max_missed_packets(switch_rtp_t *rtp_session, uint32_t max)
switch_port_t switch_rtp_get_remote_port(switch_rtp_t *rtp_session)
struct payload_map_s * next
switch_media_handle_t * media_handle
void rtp_flush_read_buffer(switch_rtp_t *rtp_session, switch_rtp_flush_t flush)
switch_status_t switch_core_thread_set_cpu_affinity(int cpu)
void switch_core_session_unset_write_codec(_In_ switch_core_session_t *session)
#define SWITCH_LOCAL_MEDIA_PORT_VARIABLE
static int switch_safe_atoi(const char *nptr, int dft)
Turn a string into an integer (default if NULL)
uint32_t switch_codec_flag_t
switch_status_t switch_rtp_udptl_mode(switch_rtp_t *rtp_session)
switch_status_t switch_resolve_host(const char *host, char *buf, size_t buflen)
void switch_rtp_reset_media_timer(switch_rtp_t *rtp_session)
void switch_rtp_set_telephony_event(switch_rtp_t *rtp_session, switch_payload_t te)
Set the payload type to consider RFC2833 DTMF.
switch_status_t switch_core_codec_reset(switch_codec_t *codec)
switch_io_event_hooks_t event_hooks
switch_status_t switch_mutex_unlock(switch_mutex_t *lock)
struct switch_io_event_hook_text_read_frame * next
switch_status_t switch_thread_rwlock_rdlock(switch_thread_rwlock_t *rwlock)
switch_status_t switch_core_session_get_read_impl(switch_core_session_t *session, switch_codec_implementation_t *impp)
switch_frame_t read_frame
_Ret_ switch_channel_t * switch_core_session_get_channel(_In_ switch_core_session_t *session)
Retrieve a pointer to the channel object associated with a given session.
switch_codec_control_command_t
void switch_rtp_destroy(switch_rtp_t **rtp_session)
Destroy an RTP session.
void switch_core_recovery_track(switch_core_session_t *session)
#define SWITCH_ORIGINATOR_CODEC_VARIABLE
void switch_channel_clear_app_flag_key(const char *app, switch_channel_t *channel, uint32_t flags)
#define switch_clear_flag(obj, flag)
Clear a flag on an arbitrary object while locked.
unsigned char local_raw_key[SWITCH_RTP_MAX_CRYPTO_LEN]
int switch_core_cert_gen_fingerprint(const char *prefix, dtls_fingerprint_t *fp)
void switch_channel_mark_hold(switch_channel_t *channel, switch_bool_t on)
switch_codec_implementation_t write_impl
#define SWITCH_MUTEX_NESTED
#define SWITCH_RTCP_AUDIO_INTERVAL_MSEC
switch_status_t switch_rtp_deactivate_jitter_buffer(switch_rtp_t *rtp_session)
switch_jb_t * switch_rtp_get_jitter_buffer(switch_rtp_t *rtp_session)
void switch_core_session_enable_heartbeat(switch_core_session_t *session, uint32_t seconds)
char * remote_accept_types
switch_codec_t write_codec
switch_status_t switch_rtp_zerocopy_read_frame(switch_rtp_t *rtp_session, switch_frame_t *frame, switch_io_flag_t io_flags)
Read data from a given RTP session without copying.
switch_memory_pool_t * pool
switch_codec_control_type_t
void(* switch_engine_function_t)(switch_core_session_t *session, void *user_data)
const switch_codec_implementation_t * implementation
#define SWITCH_REMOTE_MEDIA_IP_VARIABLE
switch_port_t remote_rtp_ice_port
switch_rtcp_numbers_t rtcp
switch_byte_t switch_byte_t * buf
#define SWITCH_STUN_DEFAULT_PORT
int microseconds_per_packet
if((uint32_t)(unpack->cur - unpack->buf) > unpack->buflen)
#define switch_yield(ms)
Wait a desired number of microseconds and yield the CPU.
uint32_t switch_core_media_bug_prune(switch_core_session_t *session)
#define switch_channel_audio_sync(_c)
unsigned int switch_separate_string(_In_ char *buf, char delim, _Post_count_(return) char **array, unsigned int arraylen)
Separate a string into an array based on a character delimiter.
switch_channel_t * channel
switch_rtp_t * rtp_session
switch_write_frame_hook_t write_frame
switch_audio_resampler_t * read_resampler
uint32_t highest_sequence_number_received
void switch_stun_random_string(char *buf, uint16_t len, char *set)
Writes random characters into a buffer.
const char * switch_channel_get_hold_music(switch_channel_t *channel)
switch_codec_interface_t * codec_interface
switch_timer_interface_t * timer_interface
switch_status_t switch_rtp_activate_jitter_buffer(switch_rtp_t *rtp_session, uint32_t queue_frames, uint32_t max_queue_frames, uint32_t samples_per_packet, uint32_t samples_per_second)
Acvite a jitter buffer on an RTP session.
int switch_rtp_has_dtls(void)
switch_codec_t read_codec
switch_frame_flag_t flags
switch_status_t switch_mutex_lock(switch_mutex_t *lock)
switch_media_flow_t rmode
switch_status_t switch_b64_encode(unsigned char *in, switch_size_t ilen, unsigned char *out, switch_size_t olen)
switch_status_t switch_rtp_get_video_buffer_size(switch_rtp_t *rtp_session, uint32_t *min_frame_len, uint32_t *max_frame_len, uint32_t *cur_frame_len, uint32_t *highest_frame_len)
uint32_t red_ts[MAX_RED_FRAMES]
#define switch_core_session_request_video_refresh(_s)
switch_codec_type_t codec_type
unsigned long local_key_material_n
uint32_t switch_channel_test_cap_partner(switch_channel_t *channel, switch_channel_cap_t cap)
#define switch_core_alloc(_pool, _mem)
Allocate memory directly from a memory pool.
uint32_t actual_samples_per_second
switch_status_t switch_rtp_activate_ice(switch_rtp_t *rtp_session, char *login, char *rlogin, const char *password, const char *rpassword, ice_proto_t proto, switch_core_media_ice_type_t type, ice_t *ice_params)
Acvite ICE on an RTP session.
#define switch_channel_get_variable(_c, _v)
#define SWITCH_THREAD_STACKSIZE
#define switch_channel_video_sync(_c)
void * video_read_user_data
void switch_rtp_set_flags(switch_rtp_t *rtp_session, switch_rtp_flag_t flags[SWITCH_RTP_FLAG_INVALID])
switch_status_t switch_event_add_header_string(switch_event_t *event, switch_stack_t stack, const char *header_name, const char *data)
Add a string header to an event.
#define switch_zmalloc(ptr, len)
char * remote_accept_wrapped_types
void switch_img_copy(switch_image_t *img, switch_image_t **new_img)
Copy image to a new image.
payload_map_t * cur_payload_map
#define SWITCH_RTP_MAX_CRYPTO_LEN
switch_codec_t * write_codec
#define switch_safe_free(it)
Free a pointer and set it to NULL unless it already is NULL.
switch_rtp_numbers_t outbound
switch_status_t switch_img_fit(switch_image_t **srcP, int width, int height, switch_img_fit_t fit)
switch_core_video_thread_callback_func_t text_read_callback
void switch_core_codec_lock_full(switch_core_session_t *session)
Lock codec read mutex and codec write mutex using trylock in an infinite loop.
void switch_rtp_video_loss(switch_rtp_t *rtp_session)
dtls_fingerprint_t local_dtls_fingerprint
switch_status_t switch_mutex_init(switch_mutex_t **lock, unsigned int flags, switch_memory_pool_t *pool)
switch_status_t switch_rtp_set_remote_ssrc(switch_rtp_t *rtp_session, uint32_t ssrc)
switch_rtp_text_factory_t * tf
switch_status_t switch_thread_join(switch_status_t *retval, switch_thread_t *thd)
switch_image_t * switch_img_read_png(const char *file_name, switch_img_fmt_t img_fmt)
switch_media_bug_t * bugs
int switch_loadable_module_get_codecs(const switch_codec_implementation_t **array, int arraylen)
Retrieve the list of loaded codecs into an array.
switch_status_t switch_rtp_set_ssrc(switch_rtp_t *rtp_session, uint32_t ssrc)
void switch_buffer_zero(_In_ switch_buffer_t *buffer)
Remove all data from the buffer.
switch_mutex_t * resample_mutex
#define switch_channel_down_nosig(_channel)
switch_sdp_type_t sdp_type
switch_mutex_t * text_mutex
uint32_t switch_core_max_audio_channels(uint32_t limit)
switch_io_routines_t * io_override
An abstraction of a data frame.
switch_status_t switch_rtp_add_dtls(switch_rtp_t *rtp_session, dtls_fingerprint_t *local_fp, dtls_fingerprint_t *remote_fp, dtls_type_t type, uint8_t want_DTLSv1_2)
int switch_vasprintf(_Out_opt_ char **buf, _In_z_ _Printf_format_string_ const char *format, _In_ va_list ap)
#define switch_set_flag_locked(obj, flag)
Set a flag on an arbitrary object while locked.
uint32_t switch_default_ptime(const char *name, uint32_t number)
switch_status_t switch_channel_wait_for_app_flag(switch_channel_t *channel, uint32_t app_flag, const char *key, switch_bool_t pres, uint32_t to)
const char * T38VendorInfo
switch_codec_t * switch_core_session_get_video_write_codec(_In_ switch_core_session_t *session)
Retrieve the video_write codec from a given session.
switch_size_t switch_rtp_has_dtmf(switch_rtp_t *rtp_session)
Test for presence of DTMF on a given RTP session.
switch_byte_t switch_byte_t uint32_t buflen
#define switch_core_codec_init(_codec, _codec_name, _modname, _fmtp, _rate, _ms, _channels, _flags, _codec_settings, _pool)
Initialize a codec handle.
switch_port_t local_sdp_port
pthread_t switch_thread_id_t
int switch_inet_pton(int af, const char *src, void *dst)
void switch_cond_next(void)
#define switch_core_session_get_partner(_session, _partner)
switch_status_t switch_img_letterbox(switch_image_t *img, switch_image_t **imgP, int width, int height, const char *color)
switch_payload_t agreed_pt
int microseconds_per_packet
char * switch_core_get_variable(_In_z_ const char *varname)
Retrieve a global variable from the core.
char * switch_copy_string(_Out_z_cap_(dst_size) char *dst, _In_z_ const char *src, _In_ switch_size_t dst_size)
switch_status_t switch_nat_del_mapping(switch_port_t port, switch_nat_ip_proto_t proto)
Deletes a NAT mapping.
switch_size_t packet_count
switch_status_t switch_core_session_queue_event(_In_ switch_core_session_t *session, _Inout_ switch_event_t **event)
Queue an event on a given session.
switch_status_t switch_rtp_set_remote_address(switch_rtp_t *rtp_session, const char *host, switch_port_t port, switch_port_t remote_rtcp_port, switch_bool_t change_adv_addr, const char **err)
Assign a remote address to the RTP session.
payload_map_t * pmap_tail
switch_memory_pool_t * pool
void switch_color_set_rgb(switch_rgb_color_t *color, const char *color_str)
Set RGB color with a string.
void * text_read_user_data
unsigned long remote_key_material_n
switch_status_t switch_core_timer_destroy(switch_timer_t *timer)
Destroy an allocated timer.
uint32_t actual_samples_per_second
int cand_idx[MAX_CAND_IDX_COUNT]
#define switch_core_session_receive_message(_session, _message)
void switch_core_session_rwunlock(_In_ switch_core_session_t *session)
Unlock a read or write lock on as given session.
void switch_core_session_video_reset(switch_core_session_t *session)
char * switch_core_session_get_uuid(_In_ switch_core_session_t *session)
Retrieve the unique identifier from a session.
switch_size_t jb_packet_count
#define SWITCH_RTCP_VIDEO_INTERVAL_MSEC
uint32_t switch_channel_test_cap(switch_channel_t *channel, switch_channel_cap_t cap)
switch_status_t switch_core_codec_encode_video(switch_codec_t *codec, switch_frame_t *frame)
Encode video data using a codec handle.
switch_size_t skip_packet_count
void switch_rtp_set_cng_pt(switch_rtp_t *rtp_session, switch_payload_t pt)
Set the payload type for comfort noise.
void switch_channel_set_app_flag_key(const char *app, switch_channel_t *channel, uint32_t flags)
#define SWITCH_LOCAL_MEDIA_IP_VARIABLE
#define switch_str_nil(s)
Make a null string a blank string instead.
struct fspr_thread_mutex_t switch_mutex_t
void switch_mux_channels(int16_t *data, switch_size_t samples, uint32_t orig_channels, uint32_t channels)
switch_status_t switch_core_timer_next(switch_timer_t *timer)
Wait for one cycle on an existing timer.
switch_bool_t switch_core_session_transcoding(switch_core_session_t *session_a, switch_core_session_t *session_b, switch_media_type_t type)
static int switch_channel_var_true(switch_channel_t *channel, const char *variable)
switch_status_t switch_core_codec_init_with_bitrate(switch_codec_t *codec, const char *codec_name, const char *fmtp, const char *modname, uint32_t rate, int ms, int channels, uint32_t bitrate, uint32_t flags, const switch_codec_settings_t *codec_settings, switch_memory_pool_t *pool)
int switch_thread_equal(switch_thread_id_t tid1, switch_thread_id_t tid2)
Compare two thread ids.
void switch_rtp_clear_flag(switch_rtp_t *rtp_session, switch_rtp_flag_t flag)
Clear an RTP Flag.
const char * switch_channel_get_variable_partner(switch_channel_t *channel, const char *varname)
switch_core_video_thread_callback_func_t video_read_callback
dtls_fingerprint_t remote_dtls_fingerprint
switch_status_t switch_frame_buffer_push(switch_frame_buffer_t *fb, void *ptr)
switch_port_t switch_rtp_request_port(const char *ip)
Request a new port to be used for media.
#define SWITCH_R_SDP_VARIABLE
switch_io_write_frame_t write_frame
uint32_t samples_per_packet
void switch_rtp_set_flag(switch_rtp_t *rtp_session, switch_rtp_flag_t flag)
Set an RTP Flag.
char * remote_file_selector
#define switch_channel_down(_channel)
uint8_t number_of_channels
uint32_t samples_per_second
#define SWITCH_CRYPTO_MKI_MAX
char * switch_rtp_get_remote_host(switch_rtp_t *rtp_session)
switch_memory_pool_t * pool
char * local_accept_wrapped_types
switch_status_t
Common return values.
switch_timer_t * switch_rtp_get_media_timer(switch_rtp_t *rtp_session)
void * switch_channel_get_private(switch_channel_t *channel, const char *key)
Retrieve private from a given channel.
#define switch_goto_status(_status, _label)
uint32_t switch_rtp_get_ssrc(switch_rtp_t *rtp_session)
Retrieve the SSRC from a given RTP session.
switch_size_t switch_rtp_dequeue_dtmf(switch_rtp_t *rtp_session, switch_dtmf_t *dtmf)
Retrieve DTMF digits from a given RTP session.
void switch_rtp_flush(switch_rtp_t *rtp_session)
switch_status_t switch_ivr_unhold(switch_core_session_t *session)
Signal the session with a protocol specific unhold message.
switch_status_t switch_core_session_write_encoded_video_frame(switch_core_session_t *session, switch_frame_t *frame, switch_io_flag_t flags, int stream_id)
const switch_codec_implementation_t * imp
switch_status_t switch_frame_buffer_create(switch_frame_buffer_t **fbP, switch_size_t qlen)
int8_t engine_function_running
char * remote_rtp_ice_addr
switch_frame_t text_frame
void switch_rtp_break(switch_rtp_t *rtp_session)
void switch_rtp_reset_jb(switch_rtp_t *rtp_session)
void switch_img_fill(switch_image_t *img, int x, int y, int w, int h, switch_rgb_color_t *color)
Fill image with color.
An abstraction of a rtcp frame.
switch_payload_t ianacode
#define switch_core_session_locate(uuid_str)
Locate a session based on it's uuid.
switch_rtp_flag_t
RTP Related Flags.
#define SWITCH_B_SDP_VARIABLE
void switch_rtp_get_random(void *buf, uint32_t len)
switch_status_t switch_rtp_queue_rfc2833(switch_rtp_t *rtp_session, const switch_dtmf_t *dtmf)
Queue RFC2833 DTMF data into an RTP Session.
#define MAX_CAND_IDX_COUNT
void switch_rtp_set_default_payload(switch_rtp_t *rtp_session, switch_payload_t payload)
Set the default payload number for a given RTP session.
#define switch_event_create(event, id)
Create a new event assuming it will not be custom event and therefore hiding the unused parameters...
uint32_t encoded_bytes_per_packet
char * cand_acl[SWITCH_MAX_CAND_ACL]
switch_size_t flush_packet_count
switch_byte_t * text_write_frame_data
Node in which to store custom video_write_frame channel callback hooks.
switch_port_t adv_sdp_port
switch_frame_t enc_write_frame
struct fspr_thread_cond_t switch_thread_cond_t
#define switch_event_get_header(_e, _h)
#define SWITCH_REMOTE_VIDEO_PORT_VARIABLE
#define switch_channel_set_flag(_c, _f)
#define SWITCH_IMG_FMT_I420
switch_thread_t * media_thread
#define SWITCH_MEDIA_TYPE_TOTAL
void switch_rtp_clear_flags(switch_rtp_t *rtp_session, switch_rtp_flag_t flags[SWITCH_RTP_FLAG_INVALID])
uint32_t max_missed_packets
switch_status_t(* switch_core_video_thread_callback_func_t)(switch_core_session_t *session, switch_frame_t *frame, void *user_data)
switch_io_read_video_frame_t read_video_frame
int switch_rtp_write_frame(switch_rtp_t *rtp_session, switch_frame_t *frame)
Write data to a given RTP session.
switch_size_t media_packet_count
switch_rtp_stats_t * switch_rtp_get_stats(switch_rtp_t *rtp_session, switch_memory_pool_t *pool)
switch_call_direction_t switch_channel_direction(switch_channel_t *channel)
switch_codec_implementation_t read_impl
#define switch_set_string(_dst, _src)
const char * interface_name
#define SWITCH_REMOTE_MEDIA_PORT_VARIABLE
switch_status_t switch_rtp_set_video_buffer_size(switch_rtp_t *rtp_session, uint32_t frames, uint32_t max_frames)
switch_port_t remote_rtcp_port
switch_status_t(* switch_core_text_thread_callback_func_t)(switch_core_session_t *session, switch_frame_t *frame, void *user_data)
switch_status_t switch_rtp_change_interval(switch_rtp_t *rtp_session, uint32_t ms_per_packet, uint32_t samples_per_interval)
struct switch_io_event_hook_write_frame * next
int is_chosen[MAX_CAND_IDX_COUNT]
switch_status_t switch_rtp_debug_jitter_buffer(switch_rtp_t *rtp_session, const char *name)
time_t switch_epoch_time_now(time_t *t)
Get the current epoch time.
switch_rtp_bug_flag_t rtp_bugs
switch_status_t switch_core_session_set_write_codec(_In_ switch_core_session_t *session, switch_codec_t *codec)
Assign the write codec to a given session.
#define switch_core_session_alloc(_session, _memory)
Allocate memory from a session's pool.
switch_status_t switch_queue_push(switch_queue_t *queue, void *data)
switch_status_t switch_rtp_enable_vad(switch_rtp_t *rtp_session, switch_core_session_t *session, switch_codec_t *codec, switch_vad_flag_t flags)
Enable VAD on an RTP Session.
void switch_rtp_release_port(const char *ip, switch_port_t port)
#define switch_test_flag(obj, flag)
Test for the existance of a flag on an arbitary object.
void switch_channel_clear_flag_recursive(switch_channel_t *channel, switch_channel_flag_t flag)
static switch_bool_t switch_core_codec_ready(switch_codec_t *codec)
void switch_log_printf(_In_ switch_text_channel_t channel, _In_z_ const char *file, _In_z_ const char *func, _In_ int line, _In_opt_z_ const char *userdata, _In_ switch_log_level_t level, _In_z_ _Printf_format_string_ const char *fmt,...) PRINTF_FUNCTION(7
Write log data to the logging engine.
const char * switch_stristr(const char *instr, const char *str)
void switch_core_session_unset_read_codec(_In_ switch_core_session_t *session)
uint32_t switch_rtp_test_flag(switch_rtp_t *rtp_session, switch_rtp_flag_t flags)
Test an RTP Flag.
switch_status_t switch_threadattr_create(switch_threadattr_t **new_attr, switch_memory_pool_t *pool)
switch_status_t switch_thread_create(switch_thread_t **new_thread, switch_threadattr_t *attr, switch_thread_start_t func, void *data, switch_memory_pool_t *cont)
A table of settings and callbacks that define a paticular implementation of a codec.
switch_io_write_video_frame_t write_video_frame
switch_msrp_session_t * switch_msrp_session_new(switch_memory_pool_t *pool, const char *call_id, switch_bool_t secure)
switch_bool_t T38FaxFillBitRemoval
switch_port_t remote_sdp_port
switch_endpoint_interface_t * endpoint_interface
unsigned char raw_key[SWITCH_RTP_MAX_CRYPTO_LEN]
static int switch_false(const char *expr)
Evaluate the falsefullness of a string expression.
struct fspr_pool_t switch_memory_pool_t
uint32_t switch_core_media_bug_test_flag(_In_ switch_media_bug_t *bug, _In_ uint32_t flag)
Test for the existance of a flag on an media bug.
#define switch_channel_up_nosig(_channel)
#define switch_core_session_strdup(_session, _todup)
Copy a string using memory allocation from a session's pool.
int chosen[MAX_CAND_IDX_COUNT]
switch_thread_id_t switch_thread_self(void)
void switch_event_destroy(switch_event_t **event)
Destroy an event.
switch_port_t remote_rtcp_ice_port
switch_status_t switch_thread_cond_broadcast(switch_thread_cond_t *cond)
int red_buflen[MAX_RED_FRAMES]
switch_port_t proxy_sdp_port
switch_bool_t switch_core_check_dtls_pem(const char *file)
switch_status_t switch_rtp_queue_rfc2833_in(switch_rtp_t *rtp_session, const switch_dtmf_t *dtmf)
Queue RFC2833 DTMF data into an RTP Session.
switch_rtp_crypto_key_param_method_type_t method
void switch_rtp_kill_socket(switch_rtp_t *rtp_session)
Kill the socket on an existing RTP session.
int switch_core_gen_certs(const char *prefix)
static uint32_t switch_round_to_step(uint32_t num, uint32_t step)
switch_status_t switch_channel_wait_for_flag(switch_channel_t *channel, switch_channel_flag_t want_flag, switch_bool_t pres, uint32_t to, switch_channel_t *super_channel)
void switch_channel_clear_flag(switch_channel_t *channel, switch_channel_flag_t flag)
Clear given flag(s) from a channel.
switch_core_media_ice_type_t
int switch_channel_test_app_flag_key(const char *app, switch_channel_t *channel, uint32_t flags)
struct fspr_thread_t switch_thread_t
#define SWITCH_LOCAL_TEXT_IP_VARIABLE
#define switch_channel_set_variable(_channel, _var, _val)
void switch_rtp_video_refresh(switch_rtp_t *rtp_session)
switch_time_t switch_time_now(void)
switch_frame_t raw_write_frame
static switch_status_t ice_out(switch_rtp_t *rtp_session, switch_rtp_ice_t *ice, switch_bool_t force)
char * switch_strip_spaces(char *str, switch_bool_t dup)
switch_codec_implementation_t write_impl
switch_rtp_crypto_key_type_t crypto_type
#define SWITCH_BUFFER_BLOCK_FRAMES
switch_size_t switch_buffer_inuse(_In_ switch_buffer_t *buffer)
Get the in use amount of a switch_buffer_t.
switch_status_t switch_channel_api_on(switch_channel_t *channel, const char *variable_prefix)
void switch_buffer_destroy(switch_buffer_t **buffer)
Destroy the buffer.
char * switch_channel_get_name(switch_channel_t *channel)
Retrieve the name of a given channel.
switch_status_t switch_ivr_hold_uuid(const char *uuid, const char *message, switch_bool_t moh)
Signal the session with a protocol specific hold message.
switch_status_t switch_rtp_req_bitrate(switch_rtp_t *rtp_session, uint32_t bps)
SWITCH_BEGIN_EXTERN_C char * switch_mprintf(const char *zFormat,...)
#define SWITCH_READ_ACCEPTABLE(status)
switch_status_t switch_stun_lookup(char **ip, switch_port_t *port, char *stunip, switch_port_t stunport, char **err, switch_memory_pool_t *pool)
Perform a stun lookup.
switch_status_t switch_rtcp_zerocopy_read_frame(switch_rtp_t *rtp_session, switch_rtcp_frame_t *frame)
Read RTCP data from a given RTP session without copying.
void switch_rtp_set_interdigit_delay(switch_rtp_t *rtp_session, uint32_t delay)
switch_status_t(* switch_io_read_text_frame_t)(switch_core_session_t *, switch_frame_t **, switch_io_flag_t, int)
switch_status_t switch_core_session_set_write_impl(switch_core_session_t *session, const switch_codec_implementation_t *impp)
#define switch_channel_media_up(_channel)
void * switch_buffer_get_head_pointer(switch_buffer_t *buffer)
switch_status_t switch_rtp_sync_stats(switch_rtp_t *rtp_session)
switch_buffer_t * text_line_buffer
#define SWITCH_CHANNEL_CHANNEL_LOG(x)
switch_memory_pool_t * switch_core_session_get_pool(_In_ switch_core_session_t *session)
Retrieve the memory pool from a session.
switch_status_t switch_rtp_add_crypto_key(switch_rtp_t *rtp_session, switch_rtp_crypto_direction_t direction, uint32_t index, switch_secure_settings_t *ssec)
#define switch_resample_create(_n, _fr, _tr, _ts, _q, _c)
uint32_t switch_resample_process(switch_audio_resampler_t *resampler, int16_t *src, uint32_t srclen)
Resample one float buffer into another using specifications of a given handle.
switch_status_t switch_msrp_start_client(switch_msrp_session_t *msrp_session)
uint8_t try_hardware_encoder
switch_io_write_text_frame_t write_text_frame
#define SWITCH_REMOTE_VIDEO_IP_VARIABLE
switch_byte_t * red_buf[MAX_RED_FRAMES]
switch_status_t switch_core_codec_control(switch_codec_t *codec, switch_codec_control_command_t cmd, switch_codec_control_type_t ctype, void *cmd_data, switch_codec_control_type_t atype, void *cmd_arg, switch_codec_control_type_t *rtype, void **ret_data)
send control data using a codec handle
switch_status_t switch_core_session_set_real_read_codec(_In_ switch_core_session_t *session, switch_codec_t *codec)
Assign the original read codec to a given session. This is the read codec used by an endpoint...
switch_status_t switch_core_timer_sync(switch_timer_t *timer)
switch_size_t cng_packet_count
switch_rtp_crypto_key_type_t type
switch_frame_t text_write_frame
switch_jb_t * switch_core_session_get_jb(switch_core_session_t *session, switch_media_type_t type)
switch_rtp_crypto_key_type_t crypto_type
struct switch_crypto_key_material_s * remote_key_material_next