/** Generate MRCP descriptor by RTSP request */ MRCP_DECLARE(mrcp_session_descriptor_t*) mrcp_descriptor_generate_by_rtsp_request( const rtsp_message_t *request, const char *force_destination_ip, const apr_table_t *resource_map, apr_pool_t *pool, su_home_t *home) { mrcp_session_descriptor_t *descriptor = NULL; const char *resource_name = mrcp_name_get_by_rtsp_name( resource_map, request->start_line.common.request_line.resource_name); if(!resource_name) { return NULL; } if(request->start_line.common.request_line.method_id == RTSP_METHOD_SETUP) { if(rtsp_header_property_check(&request->header.property_set,RTSP_HEADER_FIELD_CONTENT_TYPE) == TRUE && rtsp_header_property_check(&request->header.property_set,RTSP_HEADER_FIELD_CONTENT_LENGTH) == TRUE && request->body.buf) { sdp_parser_t *parser; sdp_session_t *sdp; parser = sdp_parse(home,request->body.buf,request->body.length,0); sdp = sdp_session(parser); if(sdp) { descriptor = mrcp_session_descriptor_create(pool); mrcp_descriptor_generate_by_sdp_session(descriptor,sdp,force_destination_ip,pool); } else { apt_log(APT_LOG_MARK,APT_PRIO_WARNING,"Failed to Parse SDP Message"); } sdp_parser_free(parser); } else { /* create default descriptor in case RTSP SETUP contains no SDP */ mpf_rtp_media_descriptor_t *media; descriptor = mrcp_session_descriptor_create(pool); media = apr_palloc(pool,sizeof(mpf_rtp_media_descriptor_t)); mpf_rtp_media_descriptor_init(media); media->state = MPF_MEDIA_ENABLED; media->id = mrcp_session_audio_media_add(descriptor,media); if(rtsp_header_property_check(&request->header.property_set,RTSP_HEADER_FIELD_TRANSPORT) == TRUE) { media->port = request->header.transport.client_port_range.min; media->ip = request->header.transport.destination; } } if(descriptor) { apt_string_assign(&descriptor->resource_name,resource_name,pool); descriptor->resource_state = TRUE; } } else if(request->start_line.common.request_line.method_id == RTSP_METHOD_TEARDOWN) { descriptor = mrcp_session_descriptor_create(pool); apt_string_assign(&descriptor->resource_name,resource_name,pool); descriptor->resource_state = FALSE; } return descriptor; }
static void mrcp_sofia_on_resource_discover( int status, mrcp_sofia_agent_t *sofia_agent, nua_handle_t *nh, mrcp_sofia_session_t *sofia_session, sip_t const *sip, tagi_t tags[]) { mrcp_session_t *session = sofia_session->session; if(session) { const char *remote_sdp_str = NULL; mrcp_session_descriptor_t *descriptor = NULL; tl_gets(tags, SOATAG_REMOTE_SDP_STR_REF(remote_sdp_str), TAG_END()); if(remote_sdp_str) { sdp_parser_t *parser = NULL; sdp_session_t *sdp = NULL; apt_log(APT_LOG_MARK,APT_PRIO_INFO,"Resource Discovery SDP %s\n%s", session->name, remote_sdp_str); parser = sdp_parse(sofia_session->home,remote_sdp_str,(int)strlen(remote_sdp_str),0); sdp = sdp_session(parser); descriptor = mrcp_descriptor_generate_by_sdp_session(sdp,NULL,session->pool); sdp_parser_free(parser); } mrcp_session_discover_response(session,descriptor); } }
static void mrcp_sofia_on_resource_discover( int status, mrcp_sofia_agent_t *sofia_agent, nua_handle_t *nh, mrcp_sofia_session_t *sofia_session, sip_t const *sip, tagi_t tags[]) { mrcp_session_t *session = sofia_session->session; if(session) { const char *remote_sdp_str = NULL; mrcp_session_descriptor_t *descriptor = NULL; if(sip->sip_payload) { remote_sdp_str = sip->sip_payload->pl_data; } if(remote_sdp_str) { sdp_parser_t *parser = NULL; sdp_session_t *sdp = NULL; apt_obj_log(APT_LOG_MARK,APT_PRIO_INFO,session->obj,"Resource Discovery SDP %s\n%s", session->name, remote_sdp_str); parser = sdp_parse(sofia_session->home,remote_sdp_str,(int)strlen(remote_sdp_str),0); sdp = sdp_session(parser); descriptor = mrcp_descriptor_generate_by_sdp_session(sdp,NULL,session->pool); descriptor->response_code = status; sdp_parser_free(parser); } mrcp_session_discover_response(session,descriptor); } }
static int priv_set_remote_sdp(SscMedia *self, const gchar *str) { su_home_t *home = self->sm_home; const char *pa_error; int res = 0, dlen = strlen(str); g_debug(__func__); if (self->sm_sdp_remote) sdp_parser_free(self->sm_sdp_remote); /* XXX: only update if SDP has really changed */ /* g_message("parsing SDP:\n%s\n---", str); */ self->sm_sdp_remote = sdp_parse(home, str, dlen, sdp_f_insane); pa_error = sdp_parsing_error(self->sm_sdp_remote); if (pa_error) { g_warning("%s: error parsing SDP: %s\n", __func__, pa_error); res = -1; } else { if (self->sm_sdp_remote_str) g_free(self->sm_sdp_remote_str); } return res; }
static void mrcp_sofia_on_session_ready( int status, mrcp_sofia_agent_t *sofia_agent, nua_handle_t *nh, mrcp_sofia_session_t *sofia_session, sip_t const *sip, tagi_t tags[]) { const char *local_sdp_str = NULL, *remote_sdp_str = NULL; mrcp_session_descriptor_t *descriptor = NULL; tl_gets(tags, SOATAG_LOCAL_SDP_STR_REF(local_sdp_str), SOATAG_REMOTE_SDP_STR_REF(remote_sdp_str), TAG_END()); if(remote_sdp_str) { sdp_parser_t *parser = NULL; sdp_session_t *sdp = NULL; apt_log(APT_LOG_MARK,APT_PRIO_INFO,"Remote SDP\n%s", remote_sdp_str); parser = sdp_parse(sofia_session->home,remote_sdp_str,(int)strlen(remote_sdp_str),0); sdp = sdp_session(parser); descriptor = mrcp_descriptor_generate_by_sdp_session(sdp,sofia_session->session->pool); sdp_parser_free(parser); } mrcp_session_answer(sofia_session->session,descriptor); }
/* Pre-parse SDP: is this SDP valid? how many audio/video lines? */ janus_sdp *janus_sdp_preparse(const char *jsep_sdp, int *audio, int *video) { sdp_parser_t *parser = sdp_parse(home, jsep_sdp, strlen(jsep_sdp), 0); sdp_session_t *parsed_sdp = sdp_session(parser); if(!parsed_sdp) { JANUS_DEBUG(" Error parsing SDP? %s\n", sdp_parsing_error(parser)); sdp_parser_free(parser); /* Invalid SDP */ return NULL; } sdp_media_t *m = parsed_sdp->sdp_media; while(m) { if(m->m_type == sdp_media_audio) { *audio = *audio + 1; } else if(m->m_type == sdp_media_video) { *video = *video + 1; } m = m->m_next; } janus_sdp *sdp = (janus_sdp *)calloc(1, sizeof(janus_sdp)); if(sdp == NULL) { JANUS_DEBUG("Memory error!\n"); return NULL; } sdp->parser = parser; sdp->sdp = parsed_sdp; return sdp; }
/* SDP parser */ void janus_sdp_free(janus_sdp *sdp) { if(!sdp) return; sdp_parser_t *parser = (sdp_parser_t *)sdp->parser; if(parser) sdp_parser_free(parser); sdp->parser = NULL; sdp->sdp = NULL; free(sdp); sdp = NULL; }
/** Generate MRCP descriptor by RTSP response */ MRCP_DECLARE(mrcp_session_descriptor_t*) mrcp_descriptor_generate_by_rtsp_response( const rtsp_message_t *request, const rtsp_message_t *response, const char *force_destination_ip, const apr_table_t *resource_map, apr_pool_t *pool, su_home_t *home) { mrcp_session_descriptor_t *descriptor = NULL; const char *resource_name = mrcp_name_get_by_rtsp_name( resource_map, request->start_line.common.request_line.resource_name); if(!resource_name) { return NULL; } if(request->start_line.common.request_line.method_id == RTSP_METHOD_SETUP) { if(rtsp_header_property_check(&response->header,RTSP_HEADER_FIELD_CONTENT_TYPE) == TRUE && rtsp_header_property_check(&response->header,RTSP_HEADER_FIELD_CONTENT_LENGTH) == TRUE && response->body.buf) { sdp_parser_t *parser; sdp_session_t *sdp; parser = sdp_parse(home,response->body.buf,response->body.length,0); sdp = sdp_session(parser); if(sdp) { descriptor = mrcp_session_descriptor_create(pool); mrcp_descriptor_generate_by_sdp_session(descriptor,sdp,force_destination_ip,pool); apt_string_assign(&descriptor->resource_name,resource_name,pool); descriptor->resource_state = TRUE; descriptor->response_code = response->start_line.common.status_line.status_code; } else { apt_log(APT_LOG_MARK,APT_PRIO_WARNING,"Failed to Parse SDP Message"); } sdp_parser_free(parser); } else { descriptor = mrcp_session_descriptor_create(pool); apt_string_assign(&descriptor->resource_name,resource_name,pool); descriptor->resource_state = FALSE; } } else if(request->start_line.common.request_line.method_id == RTSP_METHOD_TEARDOWN) { descriptor = mrcp_session_descriptor_create(pool); apt_string_assign(&descriptor->resource_name,resource_name,pool); descriptor->resource_state = FALSE; } return descriptor; }
static void mrcp_sofia_on_call_receive(mrcp_sofia_agent_t *sofia_agent, nua_handle_t *nh, mrcp_sofia_session_t *sofia_session, sip_t const *sip, tagi_t tags[]) { int offer_recv = 0, answer_recv = 0, offer_sent = 0, answer_sent = 0; const char *local_sdp_str = NULL, *remote_sdp_str = NULL; mrcp_session_descriptor_t *descriptor = NULL; tl_gets(tags, NUTAG_OFFER_RECV_REF(offer_recv), NUTAG_ANSWER_RECV_REF(answer_recv), NUTAG_OFFER_SENT_REF(offer_sent), NUTAG_ANSWER_SENT_REF(answer_sent), SOATAG_LOCAL_SDP_STR_REF(local_sdp_str), SOATAG_REMOTE_SDP_STR_REF(remote_sdp_str), TAG_END()); if(!sofia_session) { sofia_session = mrcp_sofia_session_create(sofia_agent,nh); if(!sofia_session) { nua_respond(nh, SIP_488_NOT_ACCEPTABLE, TAG_END()); return; } } if(remote_sdp_str) { sdp_parser_t *parser = NULL; sdp_session_t *sdp = NULL; apt_log(APT_LOG_MARK,APT_PRIO_INFO,"Remote SDP\n%s", remote_sdp_str); parser = sdp_parse(sofia_session->home,remote_sdp_str,(int)strlen(remote_sdp_str),0); sdp = sdp_session(parser); descriptor = mrcp_descriptor_generate_by_sdp_session(sdp,NULL,sofia_session->session->pool); sdp_parser_free(parser); } if(!descriptor) { nua_respond(nh, SIP_400_BAD_REQUEST, TAG_END()); return; } mrcp_session_offer(sofia_session->session,descriptor); }
static void mrcp_sofia_on_call_receive(mrcp_sofia_agent_t *sofia_agent, nua_handle_t *nh, mrcp_sofia_session_t *sofia_session, sip_t const *sip, tagi_t tags[]) { apt_bool_t status = FALSE; const char *remote_sdp_str = NULL; mrcp_session_descriptor_t *descriptor; if(!sofia_session) { sofia_session = mrcp_sofia_session_create(sofia_agent,nh); if(!sofia_session) { nua_respond(nh, SIP_488_NOT_ACCEPTABLE, TAG_END()); return; } } descriptor = mrcp_session_descriptor_create(sofia_session->session->pool); tl_gets(tags, SOATAG_REMOTE_SDP_STR_REF(remote_sdp_str), TAG_END()); if(remote_sdp_str) { sdp_parser_t *parser = NULL; sdp_session_t *sdp = NULL; apt_log(APT_LOG_MARK,APT_PRIO_INFO,"Remote SDP "APT_NAMESID_FMT"\n%s", sofia_session->session->name, MRCP_SESSION_SID(sofia_session->session), remote_sdp_str); parser = sdp_parse(sofia_session->home,remote_sdp_str,(int)strlen(remote_sdp_str),0); sdp = sdp_session(parser); status = mrcp_descriptor_generate_by_sdp_session(descriptor,sdp,NULL,sofia_session->session->pool); sdp_parser_free(parser); } if(status == FALSE) { nua_respond(nh, SIP_400_BAD_REQUEST, TAG_END()); return; } mrcp_session_offer(sofia_session->session,descriptor); }
/** Generate MRCP descriptor by RTSP response */ MRCP_DECLARE(mrcp_session_descriptor_t*) mrcp_descriptor_generate_by_rtsp_response(const rtsp_message_t *request, const rtsp_message_t *response, apr_pool_t *pool, su_home_t *home) { mrcp_session_descriptor_t *descriptor = NULL; const char *resource_name = request->start_line.common.request_line.resource_name; if(!resource_name) { return NULL; } if(request->start_line.common.request_line.method_id == RTSP_METHOD_SETUP) { if(rtsp_header_property_check(&response->header.property_set,RTSP_HEADER_FIELD_CONTENT_TYPE) == TRUE && rtsp_header_property_check(&response->header.property_set,RTSP_HEADER_FIELD_CONTENT_LENGTH) == TRUE && response->body.buf) { sdp_parser_t *parser; sdp_session_t *sdp; parser = sdp_parse(home,response->body.buf,response->body.length,0); sdp = sdp_session(parser); descriptor = mrcp_descriptor_generate_by_sdp_session(sdp,pool); if(descriptor) { apt_string_assign(&descriptor->resource_name,resource_name,pool); descriptor->resource_state = TRUE; } sdp_parser_free(parser); } else { descriptor = mrcp_session_descriptor_create(pool); if(descriptor) { apt_string_assign(&descriptor->resource_name,resource_name,pool); descriptor->resource_state = FALSE; } } } else if(request->start_line.common.request_line.method_id == RTSP_METHOD_TEARDOWN) { descriptor = mrcp_session_descriptor_create(pool); if(descriptor) { apt_string_assign(&descriptor->resource_name,resource_name,pool); descriptor->resource_state = FALSE; } } return descriptor; }
/** Generate resource discovery descriptor by RTSP response */ MRCP_DECLARE(mrcp_session_descriptor_t*) mrcp_resource_discovery_response_generate( const rtsp_message_t *request, const rtsp_message_t *response, const apr_table_t *resource_map, apr_pool_t *pool, su_home_t *home) { mrcp_session_descriptor_t *descriptor = NULL; const char *resource_name = mrcp_name_get_by_rtsp_name( resource_map, request->start_line.common.request_line.resource_name); if(!resource_name) { return NULL; } descriptor = mrcp_session_descriptor_create(pool); apt_string_assign(&descriptor->resource_name,resource_name,pool); if(rtsp_header_property_check(&response->header,RTSP_HEADER_FIELD_CONTENT_TYPE) == TRUE && rtsp_header_property_check(&response->header,RTSP_HEADER_FIELD_CONTENT_LENGTH) == TRUE && response->body.buf) { sdp_parser_t *parser; sdp_session_t *sdp; parser = sdp_parse(home,response->body.buf,response->body.length,0); sdp = sdp_session(parser); if(sdp) { mrcp_descriptor_generate_by_sdp_session(descriptor,sdp,0,pool); descriptor->resource_state = TRUE; descriptor->response_code = response->start_line.common.status_line.status_code; } else { apt_string_assign(&descriptor->resource_name,resource_name,pool); descriptor->resource_state = TRUE; } sdp_parser_free(parser); } else { descriptor->resource_state = FALSE; } return descriptor; }
/* Pre-parse SDP: is this SDP valid? how many audio/video lines? any features to take into account? */ janus_sdp *janus_sdp_preparse(const char *jsep_sdp, int *audio, int *video, int *data, int *bundle, int *rtcpmux, int *trickle) { if(!jsep_sdp || !audio || !video || !data || !bundle || !rtcpmux || !trickle) { JANUS_LOG(LOG_ERR, " Can't preparse, invalid arduments\n"); return NULL; } sdp_parser_t *parser = sdp_parse(home, jsep_sdp, strlen(jsep_sdp), 0); sdp_session_t *parsed_sdp = sdp_session(parser); if(!parsed_sdp) { JANUS_LOG(LOG_ERR, " Error parsing SDP? %s\n", sdp_parsing_error(parser)); sdp_parser_free(parser); /* Invalid SDP */ return NULL; } sdp_media_t *m = parsed_sdp->sdp_media; while(m) { if(m->m_type == sdp_media_audio && m->m_port > 0) { *audio = *audio + 1; } else if(m->m_type == sdp_media_video && m->m_port > 0) { *video = *video + 1; } m = m->m_next; } #ifdef HAVE_SCTP *data = (strstr(jsep_sdp, "DTLS/SCTP") && !strstr(jsep_sdp, " 0 DTLS/SCTP")) ? 1 : 0; /* FIXME This is a really hacky way of checking... */ #else *data = 0; #endif *bundle = strstr(jsep_sdp, "a=group:BUNDLE") ? 1 : 0; /* FIXME This is a really hacky way of checking... */ *rtcpmux = strstr(jsep_sdp, "a=rtcp-mux") ? 1 : 0; /* FIXME Should we make this check per-medium? */ //~ *trickle = (strstr(jsep_sdp, "trickle") || strstr(jsep_sdp, "google-ice") || strstr(jsep_sdp, "Mozilla")) ? 1 : 0; /* FIXME This is a really hacky way of checking... */ /* FIXME We're assuming trickle is always supported, see https://github.com/meetecho/janus-gateway/issues/83 */ *trickle = 1; janus_sdp *sdp = (janus_sdp *)calloc(1, sizeof(janus_sdp)); if(sdp == NULL) { JANUS_LOG(LOG_FATAL, "Memory error!\n"); return NULL; } sdp->parser = parser; sdp->sdp = parsed_sdp; return sdp; }
static void mrcp_sofia_on_session_ready( int status, mrcp_sofia_agent_t *sofia_agent, nua_handle_t *nh, mrcp_sofia_session_t *sofia_session, sip_t const *sip, tagi_t tags[]) { mrcp_session_t *session = sofia_session->session; if(session) { const char *local_sdp_str = NULL, *remote_sdp_str = NULL; mrcp_session_descriptor_t *descriptor = NULL; tl_gets(tags, SOATAG_LOCAL_SDP_STR_REF(local_sdp_str), SOATAG_REMOTE_SDP_STR_REF(remote_sdp_str), TAG_END()); if(remote_sdp_str) { sdp_parser_t *parser = NULL; sdp_session_t *sdp = NULL; const char *force_destination_ip = NULL; apt_log(APT_LOG_MARK,APT_PRIO_INFO,"Remote SDP "APT_NAMESID_FMT"\n%s", session->name, MRCP_SESSION_SID(session), remote_sdp_str); parser = sdp_parse(sofia_session->home,remote_sdp_str,(int)strlen(remote_sdp_str),0); sdp = sdp_session(parser); if(sofia_session->sip_settings->force_destination == TRUE) { force_destination_ip = sofia_session->sip_settings->server_ip; } descriptor = mrcp_descriptor_generate_by_sdp_session(sdp,force_destination_ip,session->pool); sdp_parser_free(parser); } mrcp_session_answer(session,descriptor); } }
char *janus_sdp_merge(janus_ice_handle *handle, const char *origsdp) { if(handle == NULL || origsdp == NULL) return NULL; sdp_session_t *anon = NULL; sdp_parser_t *parser = sdp_parse(home, origsdp, strlen(origsdp), 0); if(!(anon = sdp_session(parser))) { JANUS_LOG(LOG_ERR, "[%"SCNu64"] Error parsing/merging SDP: %s\n", handle->handle_id, sdp_parsing_error(parser)); sdp_parser_free(parser); return NULL; } /* Prepare SDP to merge */ gchar buffer[512]; memset(buffer, 0, 512); char *sdp = (char*)calloc(BUFSIZE, sizeof(char)); if(sdp == NULL) { JANUS_LOG(LOG_FATAL, "Memory error!\n"); sdp_parser_free(parser); return NULL; } sdp[0] = '\0'; /* FIXME Any Plan B to take into account? */ int planb = strstr(origsdp, "a=planb:") ? 1 : 0; if(planb) { janus_flags_set(&handle->webrtc_flags, JANUS_ICE_HANDLE_WEBRTC_PLAN_B); } else { janus_flags_clear(&handle->webrtc_flags, JANUS_ICE_HANDLE_WEBRTC_PLAN_B); } /* Version v= */ g_strlcat(sdp, "v=0\r\n", BUFSIZE); /* Origin o= */ if(anon->sdp_origin) { g_snprintf(buffer, 512, "o=%s %"SCNu64" %"SCNu64" IN IP4 127.0.0.1\r\n", /* FIXME Should we fix the address? */ anon->sdp_origin->o_username ? anon->sdp_origin->o_username : "******", anon->sdp_origin->o_id, anon->sdp_origin->o_version); g_strlcat(sdp, buffer, BUFSIZE); } else { gint64 sessid = janus_get_monotonic_time(); gint64 version = sessid; /* FIXME This needs to be increased when it changes, so time should be ok */ g_snprintf(buffer, 512, "o=%s %"SCNi64" %"SCNi64" IN IP4 127.0.0.1\r\n", /* FIXME Should we fix the address? */ "-", sessid, version); g_strlcat(sdp, buffer, BUFSIZE); } /* Session name s= */ if(anon->sdp_subject && strlen(anon->sdp_subject) > 0) { g_snprintf(buffer, 512, "s=%s\r\n", anon->sdp_subject); } else { g_snprintf(buffer, 512, "s=%s\r\n", "Meetecho Janus"); } g_strlcat(sdp, buffer, BUFSIZE); /* Timing t= */ g_snprintf(buffer, 512, "t=%lu %lu\r\n", anon->sdp_time ? anon->sdp_time->t_start : 0, anon->sdp_time ? anon->sdp_time->t_stop : 0); g_strlcat(sdp, buffer, BUFSIZE); /* ICE Full or Lite? */ if(janus_ice_is_ice_lite_enabled()) { /* Janus is acting in ICE Lite mode, advertize this */ g_strlcat(sdp, "a=ice-lite\r\n", BUFSIZE); } /* bundle: add new global attribute */ int audio = (strstr(origsdp, "m=audio") != NULL); int video = (strstr(origsdp, "m=video") != NULL); #ifdef HAVE_SCTP int data = (strstr(origsdp, "DTLS/SCTP") && !strstr(origsdp, " 0 DTLS/SCTP")); #else int data = 0; #endif g_strlcat(sdp, "a=group:BUNDLE", BUFSIZE); if(audio) { g_snprintf(buffer, 512, " %s", handle->audio_mid ? handle->audio_mid : "audio"); g_strlcat(sdp, buffer, BUFSIZE); } if(video) { g_snprintf(buffer, 512, " %s", handle->video_mid ? handle->video_mid : "video"); g_strlcat(sdp, buffer, BUFSIZE); } if(data) { g_snprintf(buffer, 512, " %s", handle->data_mid ? handle->data_mid : "data"); g_strlcat(sdp, buffer, BUFSIZE); } g_strlcat(sdp, "\r\n", BUFSIZE); /* msid-semantic: add new global attribute */ g_strlcat(sdp, "a=msid-semantic: WMS janus\r\n", BUFSIZE); char wms[BUFSIZE]; memset(wms, 0, BUFSIZE); g_strlcat(wms, "WMS", BUFSIZE); /* Copy other global attributes, if any */ if(anon->sdp_attributes) { sdp_attribute_t *a = anon->sdp_attributes; while(a) { if(a->a_value == NULL) { g_snprintf(buffer, 512, "a=%s\r\n", a->a_name); g_strlcat(sdp, buffer, BUFSIZE); } else { g_snprintf(buffer, 512, "a=%s:%s\r\n", a->a_name, a->a_value); g_strlcat(sdp, buffer, BUFSIZE); } a = a->a_next; } } gboolean ipv6 = strstr(janus_get_public_ip(), ":") != NULL; /* Media lines now */ if(anon->sdp_media) { int audio = 0, video = 0; #ifdef HAVE_SCTP int data = 0; #endif sdp_media_t *m = anon->sdp_media; janus_ice_stream *stream = NULL; while(m) { if(m->m_type == sdp_media_audio && m->m_port > 0) { audio++; if(audio > 1 || !handle->audio_id) { JANUS_LOG(LOG_WARN, "[%"SCNu64"] Skipping audio line (we have %d audio lines, and the id is %d)\n", handle->handle_id, audio, handle->audio_id); g_strlcat(sdp, "m=audio 0 RTP/SAVPF 0\r\n", BUFSIZE); /* FIXME Adding a c-line anyway because otherwise Firefox complains? ("c= connection line not specified for every media level, validation failed") */ g_snprintf(buffer, 512, "c=IN %s %s\r\n", ipv6 ? "IP6" : "IP4", janus_get_public_ip()); g_strlcat(sdp, buffer, BUFSIZE); m = m->m_next; continue; } /* Audio */ stream = g_hash_table_lookup(handle->streams, GUINT_TO_POINTER(handle->audio_id)); if(stream == NULL) { JANUS_LOG(LOG_WARN, "[%"SCNu64"] Skipping audio line (invalid stream %d)\n", handle->handle_id, handle->audio_id); g_strlcat(sdp, "m=audio 0 RTP/SAVPF 0\r\n", BUFSIZE); /* FIXME Adding a c-line anyway because otherwise Firefox complains? ("c= connection line not specified for every media level, validation failed") */ g_snprintf(buffer, 512, "c=IN %s %s\r\n", ipv6 ? "IP6" : "IP4", janus_get_public_ip()); g_strlcat(sdp, buffer, BUFSIZE); m = m->m_next; continue; } g_strlcat(sdp, "m=audio 1 RTP/SAVPF", BUFSIZE); } else if(m->m_type == sdp_media_video && m->m_port > 0) { video++; gint id = handle->video_id; if(id == 0 && janus_flags_is_set(&handle->webrtc_flags, JANUS_ICE_HANDLE_WEBRTC_BUNDLE)) id = handle->audio_id > 0 ? handle->audio_id : handle->video_id; if(video > 1 || !id) { JANUS_LOG(LOG_WARN, "[%"SCNu64"] Skipping video line (we have %d video lines, and the id is %d)\n", handle->handle_id, video, janus_flags_is_set(&handle->webrtc_flags, JANUS_ICE_HANDLE_WEBRTC_BUNDLE) ? handle->audio_id : handle->video_id); g_strlcat(sdp, "m=video 0 RTP/SAVPF 0\r\n", BUFSIZE); /* FIXME Adding a c-line anyway because otherwise Firefox complains? ("c= connection line not specified for every media level, validation failed") */ g_snprintf(buffer, 512, "c=IN %s %s\r\n", ipv6 ? "IP6" : "IP4", janus_get_public_ip()); g_strlcat(sdp, buffer, BUFSIZE); m = m->m_next; continue; } /* Video */ stream = g_hash_table_lookup(handle->streams, GUINT_TO_POINTER(id)); if(stream == NULL) { JANUS_LOG(LOG_WARN, "[%"SCNu64"] Skipping video line (invalid stream %d)\n", handle->handle_id, id); g_strlcat(sdp, "m=video 0 RTP/SAVPF 0\r\n", BUFSIZE); /* FIXME Adding a c-line anyway because otherwise Firefox complains? ("c= connection line not specified for every media level, validation failed") */ g_snprintf(buffer, 512, "c=IN %s %s\r\n", ipv6 ? "IP6" : "IP4", janus_get_public_ip()); g_strlcat(sdp, buffer, BUFSIZE); m = m->m_next; continue; } g_strlcat(sdp, "m=video 1 RTP/SAVPF", BUFSIZE); #ifdef HAVE_SCTP } else if(m->m_type == sdp_media_application) { /* Is this SCTP for DataChannels? */ if(m->m_port > 0 && m->m_proto_name != NULL && !strcasecmp(m->m_proto_name, "DTLS/SCTP") && m->m_port > 0) { /* Yep */ data++; gint id = handle->data_id; if(id == 0 && janus_flags_is_set(&handle->webrtc_flags, JANUS_ICE_HANDLE_WEBRTC_BUNDLE)) id = handle->audio_id > 0 ? handle->audio_id : handle->video_id; if(data > 1 || !id) { JANUS_LOG(LOG_WARN, "[%"SCNu64"] Skipping SCTP line (we have %d SCTP lines, and the id is %d)\n", handle->handle_id, data, id); g_snprintf(buffer, 512, "m=%s 0 %s 0\r\n", m->m_type_name, m->m_proto_name); g_strlcat(sdp, buffer, BUFSIZE); /* FIXME Adding a c-line anyway because otherwise Firefox complains? ("c= connection line not specified for every media level, validation failed") */ g_snprintf(buffer, 512, "c=IN %s %s\r\n", ipv6 ? "IP6" : "IP4", janus_get_public_ip()); g_strlcat(sdp, buffer, BUFSIZE); m = m->m_next; continue; } /* SCTP */ stream = g_hash_table_lookup(handle->streams, GUINT_TO_POINTER(id)); if(stream == NULL) { JANUS_LOG(LOG_WARN, "[%"SCNu64"] Skipping SCTP line (invalid stream %d)\n", handle->handle_id, id); g_snprintf(buffer, 512, "m=%s 0 %s 0\r\n", m->m_type_name, m->m_proto_name); g_strlcat(sdp, buffer, BUFSIZE); /* FIXME Adding a c-line anyway because otherwise Firefox complains? ("c= connection line not specified for every media level, validation failed") */ g_snprintf(buffer, 512, "c=IN %s %s\r\n", ipv6 ? "IP6" : "IP4", janus_get_public_ip()); g_strlcat(sdp, buffer, BUFSIZE); m = m->m_next; continue; } g_strlcat(sdp, "m=application 1 DTLS/SCTP", BUFSIZE); } else { JANUS_LOG(LOG_WARN, "[%"SCNu64"] Skipping unsupported application media line...\n", handle->handle_id); g_snprintf(buffer, 512, "m=%s 0 %s 0\r\n", m->m_type_name, m->m_proto_name); g_strlcat(sdp, buffer, BUFSIZE); m = m->m_next; continue; } #endif } else { JANUS_LOG(LOG_WARN, "[%"SCNu64"] Skipping disabled/unsupported media line...\n", handle->handle_id); g_snprintf(buffer, 512, "m=%s 0 %s 0\r\n", m->m_type_name, m->m_proto_name); g_strlcat(sdp, buffer, BUFSIZE); /* FIXME Adding a c-line anyway because otherwise Firefox complains? ("c= connection line not specified for every media level, validation failed") */ g_snprintf(buffer, 512, "c=IN %s %s\r\n", ipv6 ? "IP6" : "IP4", janus_get_public_ip()); g_strlcat(sdp, buffer, BUFSIZE); m = m->m_next; continue; } /* Add formats now */ if(!m->m_rtpmaps) { JANUS_LOG(LOG_VERB, "[%"SCNu64"] No RTP maps?? trying formats...\n", handle->handle_id); if(!m->m_format) { JANUS_LOG(LOG_ERR, "[%"SCNu64"] No formats either?? this sucks!\n", handle->handle_id); g_strlcat(sdp, " 0", BUFSIZE); /* FIXME Won't work apparently */ } else { sdp_list_t *fmt = m->m_format; while(fmt) { g_snprintf(buffer, 512, " %s", fmt->l_text); g_strlcat(sdp, buffer, BUFSIZE); fmt = fmt->l_next; } } } else { sdp_rtpmap_t *r = m->m_rtpmaps; while(r) { g_snprintf(buffer, 512, " %d", r->rm_pt); g_strlcat(sdp, buffer, BUFSIZE); r = r->rm_next; } } g_strlcat(sdp, "\r\n", BUFSIZE); /* Media connection c= */ g_snprintf(buffer, 512, "c=IN %s %s\r\n", ipv6 ? "IP6" : "IP4", janus_get_public_ip()); g_strlcat(sdp, buffer, BUFSIZE); /* Any bandwidth? */ if(m->m_bandwidths) { g_snprintf(buffer, 512, "b=%s:%lu\r\n", /* FIXME Are we doing this correctly? */ m->m_bandwidths->b_modifier_name ? m->m_bandwidths->b_modifier_name : "AS", m->m_bandwidths->b_value); g_strlcat(sdp, buffer, BUFSIZE); } /* a=mid:(audio|video|data) */ switch(m->m_type) { case sdp_media_audio: g_snprintf(buffer, 512, "a=mid:%s\r\n", handle->audio_mid ? handle->audio_mid : "audio"); break; case sdp_media_video: g_snprintf(buffer, 512, "a=mid:%s\r\n", handle->video_mid ? handle->video_mid : "video"); break; #ifdef HAVE_SCTP case sdp_media_application: /* FIXME sctpmap and webrtc-datachannel should be dynamic */ g_snprintf(buffer, 512, "a=mid:%s\r\na=sctpmap:5000 webrtc-datachannel 16\r\n", handle->data_mid ? handle->data_mid : "data"); break; #endif default: break; } g_strlcat(sdp, buffer, BUFSIZE); if(m->m_type != sdp_media_application) { /* What is the direction? */ switch(m->m_mode) { case sdp_sendonly: g_strlcat(sdp, "a=sendonly\r\n", BUFSIZE); break; case sdp_recvonly: g_strlcat(sdp, "a=recvonly\r\n", BUFSIZE); break; case sdp_inactive: g_strlcat(sdp, "a=inactive\r\n", BUFSIZE); break; case sdp_sendrecv: default: g_strlcat(sdp, "a=sendrecv\r\n", BUFSIZE); break; } /* rtcp-mux */ g_snprintf(buffer, 512, "a=rtcp-mux\n"); g_strlcat(sdp, buffer, BUFSIZE); /* RTP maps */ if(m->m_rtpmaps) { sdp_rtpmap_t *rm = NULL; for(rm = m->m_rtpmaps; rm; rm = rm->rm_next) { g_snprintf(buffer, 512, "a=rtpmap:%u %s/%lu%s%s\r\n", rm->rm_pt, rm->rm_encoding, rm->rm_rate, rm->rm_params ? "/" : "", rm->rm_params ? rm->rm_params : ""); g_strlcat(sdp, buffer, BUFSIZE); } for(rm = m->m_rtpmaps; rm; rm = rm->rm_next) { if(rm->rm_fmtp) { g_snprintf(buffer, 512, "a=fmtp:%u %s\r\n", rm->rm_pt, rm->rm_fmtp); g_strlcat(sdp, buffer, BUFSIZE); } } } } /* ICE ufrag and pwd, DTLS fingerprint setup and connection a= */ gchar *ufrag = NULL; gchar *password = NULL; nice_agent_get_local_credentials(handle->agent, stream->stream_id, &ufrag, &password); memset(buffer, 0, 100); g_snprintf(buffer, 512, "a=ice-ufrag:%s\r\n" "a=ice-pwd:%s\r\n" "a=ice-options:trickle\r\n" "a=fingerprint:sha-256 %s\r\n" "a=setup:%s\r\n" "a=connection:new\r\n", ufrag, password, janus_dtls_get_local_fingerprint(), janus_get_dtls_srtp_role(stream->dtls_role)); if(ufrag != NULL) g_free(ufrag); ufrag = NULL; if(password != NULL) g_free(password); password = NULL; g_strlcat(sdp, buffer, BUFSIZE); /* Copy existing media attributes, if any */ if(m->m_attributes) { sdp_attribute_t *a = m->m_attributes; while(a) { if(!strcmp(a->a_name, "planb")) { /* Skip the fake planb attribute, it's for internal use only */ a = a->a_next; continue; } if(a->a_value == NULL) { g_snprintf(buffer, 512, "a=%s\r\n", a->a_name); g_strlcat(sdp, buffer, BUFSIZE); } else { g_snprintf(buffer, 512, "a=%s:%s\r\n", a->a_name, a->a_value); g_strlcat(sdp, buffer, BUFSIZE); } a = a->a_next; } } /* Add last attributes, rtcp and ssrc (msid) */ if(!planb) { /* Single SSRC */ if(m->m_type == sdp_media_audio && m->m_mode != sdp_inactive && m->m_mode != sdp_recvonly) { g_snprintf(buffer, 512, "a=ssrc:%"SCNu32" cname:janusaudio\r\n" "a=ssrc:%"SCNu32" msid:janus janusa0\r\n" "a=ssrc:%"SCNu32" mslabel:janus\r\n" "a=ssrc:%"SCNu32" label:janusa0\r\n", stream->audio_ssrc, stream->audio_ssrc, stream->audio_ssrc, stream->audio_ssrc); g_strlcat(sdp, buffer, BUFSIZE); } else if(m->m_type == sdp_media_video && m->m_mode != sdp_inactive && m->m_mode != sdp_recvonly) { g_snprintf(buffer, 512, "a=ssrc:%"SCNu32" cname:janusvideo\r\n" "a=ssrc:%"SCNu32" msid:janus janusv0\r\n" "a=ssrc:%"SCNu32" mslabel:janus\r\n" "a=ssrc:%"SCNu32" label:janusv0\r\n", stream->video_ssrc, stream->video_ssrc, stream->video_ssrc, stream->video_ssrc); g_strlcat(sdp, buffer, BUFSIZE); } } else { /* Multiple SSRCs */ char mslabel[255]; memset(mslabel, 0, 255); if(m->m_attributes) { char id[256]; uint32_t ssrc = 0; sdp_attribute_t *a = m->m_attributes; while(a) { if(a->a_name == NULL || a->a_value == NULL || strcmp(a->a_name, "planb")) { a = a->a_next; continue; } if(sscanf(a->a_value, "%255s %"SCNu32"", id, &ssrc) != 2) { JANUS_LOG(LOG_ERR, "Error parsing 'planb' attribute, skipping it...\n"); a = a->a_next; continue; } JANUS_LOG(LOG_VERB, "Parsing 'planb' attribute: %s\n", a->a_value); /* Add proper SSRC attributes */ if(m->m_type == sdp_media_audio) { g_snprintf(buffer, 512, "a=ssrc:%"SCNu32" cname:%saudio\r\n" "a=ssrc:%"SCNu32" msid:%s %sa0\r\n" "a=ssrc:%"SCNu32" mslabel:%s\r\n" "a=ssrc:%"SCNu32" label:%sa0\r\n", ssrc, id, ssrc, id, id, ssrc, id, ssrc, id); } else if(m->m_type == sdp_media_video) { g_snprintf(buffer, 512, "a=ssrc:%"SCNu32" cname:%svideo\r\n" "a=ssrc:%"SCNu32" msid:%s %sv0\r\n" "a=ssrc:%"SCNu32" mslabel:%s\r\n" "a=ssrc:%"SCNu32" label:%sv0\r\n", ssrc, id, ssrc, id, id, ssrc, id, ssrc, id); } g_strlcat(sdp, buffer, BUFSIZE); /* Add to msid-semantic, if needed */ if(!strstr(wms, id)) { g_snprintf(mslabel, 255, " %s", id); g_strlcat(wms, mslabel, BUFSIZE); } /* Go on */ a = a->a_next; } } } /* And now the candidates */ janus_ice_candidates_to_sdp(handle, sdp, stream->stream_id, 1); if(!janus_flags_is_set(&handle->webrtc_flags, JANUS_ICE_HANDLE_WEBRTC_RTCPMUX) && m->m_type != sdp_media_application) janus_ice_candidates_to_sdp(handle, sdp, stream->stream_id, 2); /* Next */ m = m->m_next; } } /* Do we need to update the msid-semantic attribute? */ if(planb) { sdp = janus_string_replace(sdp, "WMS janus", wms); } sdp_parser_free(parser); JANUS_LOG(LOG_VERB, " -------------------------------------------\n"); JANUS_LOG(LOG_VERB, " >> Merged (%zu --> %zu bytes)\n", strlen(origsdp), strlen(sdp)); JANUS_LOG(LOG_VERB, " -------------------------------------------\n"); JANUS_LOG(LOG_VERB, "%s\n", sdp); return sdp; }
char *janus_sdp_anonymize(const char *sdp) { if(sdp == NULL) return NULL; sdp_session_t *anon = NULL; sdp_parser_t *parser = sdp_parse(home, sdp, strlen(sdp), 0); if(!(anon = sdp_session(parser))) { JANUS_LOG(LOG_ERR, "Error parsing/merging SDP: %s\n", sdp_parsing_error(parser)); sdp_parser_free(parser); return NULL; } /* c= */ if(anon->sdp_connection && anon->sdp_connection->c_address) { anon->sdp_connection->c_address = "1.1.1.1"; } /* a= */ if(anon->sdp_attributes) { /* These are attributes we handle ourselves, the plugins don't need them */ while(sdp_attribute_find(anon->sdp_attributes, "ice-ufrag")) sdp_attribute_remove(&anon->sdp_attributes, "ice-ufrag"); while(sdp_attribute_find(anon->sdp_attributes, "ice-pwd")) sdp_attribute_remove(&anon->sdp_attributes, "ice-pwd"); while(sdp_attribute_find(anon->sdp_attributes, "ice-options")) sdp_attribute_remove(&anon->sdp_attributes, "ice-options"); while(sdp_attribute_find(anon->sdp_attributes, "fingerprint")) sdp_attribute_remove(&anon->sdp_attributes, "fingerprint"); while(sdp_attribute_find(anon->sdp_attributes, "group")) sdp_attribute_remove(&anon->sdp_attributes, "group"); while(sdp_attribute_find(anon->sdp_attributes, "msid-semantic")) sdp_attribute_remove(&anon->sdp_attributes, "msid-semantic"); } /* m= */ if(anon->sdp_media) { int audio = 0, video = 0; #ifdef HAVE_SCTP int data = 0; #endif sdp_media_t *m = anon->sdp_media; while(m) { if(m->m_type == sdp_media_audio && m->m_port > 0) { audio++; m->m_port = audio == 1 ? 1 : 0; } else if(m->m_type == sdp_media_video && m->m_port > 0) { video++; m->m_port = video == 1 ? 1 : 0; #ifdef HAVE_SCTP } else if(m->m_type == sdp_media_application) { if(m->m_proto_name != NULL && !strcasecmp(m->m_proto_name, "DTLS/SCTP") && m->m_port != 0) { data++; m->m_port = data == 1 ? 1 : 0; } else { m->m_port = 0; } #endif } else { m->m_port = 0; } /* c= */ if(m->m_connections) { sdp_connection_t *c = m->m_connections; while(c) { if(c->c_address) { c->c_address = "1.1.1.1"; } c = c->c_next; } } /* a= */ if(m->m_attributes) { /* These are attributes we handle ourselves, the plugins don't need them */ while(sdp_attribute_find(m->m_attributes, "ice-ufrag")) sdp_attribute_remove(&m->m_attributes, "ice-ufrag"); while(sdp_attribute_find(m->m_attributes, "ice-pwd")) sdp_attribute_remove(&m->m_attributes, "ice-pwd"); while(sdp_attribute_find(m->m_attributes, "ice-options")) sdp_attribute_remove(&m->m_attributes, "ice-options"); while(sdp_attribute_find(m->m_attributes, "crypto")) sdp_attribute_remove(&m->m_attributes, "crypto"); while(sdp_attribute_find(m->m_attributes, "fingerprint")) sdp_attribute_remove(&m->m_attributes, "fingerprint"); while(sdp_attribute_find(m->m_attributes, "setup")) sdp_attribute_remove(&m->m_attributes, "setup"); while(sdp_attribute_find(m->m_attributes, "connection")) sdp_attribute_remove(&m->m_attributes, "connection"); while(sdp_attribute_find(m->m_attributes, "group")) sdp_attribute_remove(&m->m_attributes, "group"); while(sdp_attribute_find(m->m_attributes, "mid")) sdp_attribute_remove(&m->m_attributes, "mid"); while(sdp_attribute_find(m->m_attributes, "msid")) sdp_attribute_remove(&m->m_attributes, "msid"); while(sdp_attribute_find(m->m_attributes, "msid-semantic")) sdp_attribute_remove(&m->m_attributes, "msid-semantic"); while(sdp_attribute_find(m->m_attributes, "rtcp")) sdp_attribute_remove(&m->m_attributes, "rtcp"); while(sdp_attribute_find(m->m_attributes, "rtcp-mux")) sdp_attribute_remove(&m->m_attributes, "rtcp-mux"); while(sdp_attribute_find(m->m_attributes, "candidate")) sdp_attribute_remove(&m->m_attributes, "candidate"); while(sdp_attribute_find(m->m_attributes, "ssrc")) sdp_attribute_remove(&m->m_attributes, "ssrc"); while(sdp_attribute_find(m->m_attributes, "ssrc-group")) sdp_attribute_remove(&m->m_attributes, "ssrc-group"); while(sdp_attribute_find(m->m_attributes, "extmap")) /* TODO Actually implement RTP extensions */ sdp_attribute_remove(&m->m_attributes, "extmap"); while(sdp_attribute_find(m->m_attributes, "sctpmap")) sdp_attribute_remove(&m->m_attributes, "sctpmap"); } if(m->m_type != sdp_media_application && m->m_mode == sdp_sendrecv) { /* FIXME sendrecv hack: sofia-sdp doesn't print sendrecv, but we want it to */ sdp_attribute_t *fakedir = calloc(1, sizeof(sdp_attribute_t)); fakedir->a_size = sizeof(sdp_attribute_t); fakedir->a_name = g_strdup("jfmod"); fakedir->a_value = g_strdup("sr"); sdp_attribute_append(&m->m_attributes, fakedir); } m = m->m_next; } } char buf[BUFSIZE]; sdp_printer_t *printer = sdp_print(home, anon, buf, BUFSIZE, 0); if(sdp_message(printer)) { int retval = sdp_message_size(printer); sdp_printer_free(printer); sdp_parser_free(parser); /* FIXME Take care of the sendrecv hack, if needed */ char *replace = strstr(buf, "a=jfmod:sr"); while(replace != NULL) { memcpy(replace, "a=sendrecv", strlen("a=sendrecv")); replace++; replace = strstr(replace, "a=jfmod:sr"); } JANUS_LOG(LOG_VERB, " -------------------------------------------\n"); JANUS_LOG(LOG_VERB, " >> Anonymized (%zu --> %d bytes)\n", strlen(sdp), retval); JANUS_LOG(LOG_VERB, " -------------------------------------------\n"); JANUS_LOG(LOG_VERB, "%s\n", buf); return g_strdup(buf); } else { JANUS_LOG(LOG_ERR, "Error anonymizing SDP: %s\n", sdp_printing_error(printer)); sdp_printer_free(printer); sdp_parser_free(parser); return NULL; } }