/** Generate MRCP descriptor by SDP session */ static mrcp_session_descriptor_t* mrcp_descriptor_generate_by_sdp_session(const sdp_session_t *sdp, apr_pool_t *pool) { sdp_media_t *sdp_media; mrcp_session_descriptor_t *descriptor = mrcp_session_descriptor_create(pool); if(sdp->sdp_connection) { apt_string_assign(&descriptor->ip,sdp->sdp_connection->c_address,pool); } for(sdp_media=sdp->sdp_media; sdp_media; sdp_media=sdp_media->m_next) { switch(sdp_media->m_type) { case sdp_media_audio: { mpf_rtp_media_descriptor_t *media = apr_palloc(pool,sizeof(mpf_rtp_media_descriptor_t)); mpf_rtp_media_descriptor_init(media); media->base.id = mrcp_session_audio_media_add(descriptor,media); mpf_rtp_media_generate(media,sdp_media,&descriptor->ip,pool); break; } case sdp_media_video: { mpf_rtp_media_descriptor_t *media = apr_palloc(pool,sizeof(mpf_rtp_media_descriptor_t)); mpf_rtp_media_descriptor_init(media); media->base.id = mrcp_session_video_media_add(descriptor,media); mpf_rtp_media_generate(media,sdp_media,&descriptor->ip,pool); break; } default: apt_log(APT_PRIO_INFO,"Not Supported SDP Media [%s]", sdp_media->m_type_name); break; } } return descriptor; }
/** Generate MRCP descriptor by SDP session */ MRCP_DECLARE(mrcp_session_descriptor_t*) mrcp_descriptor_generate_by_sdp_session(const sdp_session_t *sdp, const char *force_destination_ip, apr_pool_t *pool) { sdp_media_t *sdp_media; mrcp_session_descriptor_t *descriptor; if(!sdp) { apt_log(APT_LOG_MARK,APT_PRIO_WARNING,"Invalid SDP Message"); return NULL; } descriptor = mrcp_session_descriptor_create(pool); if(force_destination_ip) { apt_string_assign(&descriptor->ip,force_destination_ip,pool); } else if(sdp->sdp_connection) { apt_string_assign(&descriptor->ip,sdp->sdp_connection->c_address,pool); } for(sdp_media=sdp->sdp_media; sdp_media; sdp_media=sdp_media->m_next) { switch(sdp_media->m_type) { case sdp_media_audio: { mpf_rtp_media_descriptor_t *media = apr_palloc(pool,sizeof(mpf_rtp_media_descriptor_t)); mpf_rtp_media_descriptor_init(media); media->id = mrcp_session_audio_media_add(descriptor,media); mpf_rtp_media_generate(media,sdp_media,&descriptor->ip,pool); break; } case sdp_media_video: { mpf_rtp_media_descriptor_t *media = apr_palloc(pool,sizeof(mpf_rtp_media_descriptor_t)); mpf_rtp_media_descriptor_init(media); media->id = mrcp_session_video_media_add(descriptor,media); mpf_rtp_media_generate(media,sdp_media,&descriptor->ip,pool); break; } case sdp_media_application: { mrcp_control_descriptor_t *control_media = mrcp_control_descriptor_create(pool); control_media->id = mrcp_session_control_media_add(descriptor,control_media); mrcp_control_media_generate(control_media,sdp_media,&descriptor->ip,pool); break; } default: apt_log(APT_LOG_MARK,APT_PRIO_INFO,"Not Supported SDP Media [%s]", sdp_media->m_type_name); break; } } return descriptor; }
/** 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; }
/** Create demo RTP termination descriptor */ mpf_rtp_termination_descriptor_t* demo_rtp_descriptor_create(apr_pool_t *pool) { mpf_codec_descriptor_t *codec_descriptor; mpf_rtp_media_descriptor_t *media; /* create rtp descriptor */ mpf_rtp_termination_descriptor_t *rtp_descriptor = apr_palloc(pool,sizeof(mpf_rtp_termination_descriptor_t)); mpf_rtp_termination_descriptor_init(rtp_descriptor); /* create rtp local media */ media = apr_palloc(pool,sizeof(mpf_rtp_media_descriptor_t)); mpf_rtp_media_descriptor_init(media); apt_string_assign(&media->ip,"127.0.0.1",pool); media->port = 6000; media->state = MPF_MEDIA_ENABLED; media->direction = STREAM_DIRECTION_RECEIVE; /* initialize codec list */ mpf_codec_list_init(&media->codec_list,2,pool); /* set codec descriptor */ codec_descriptor = mpf_codec_list_add(&media->codec_list); if(codec_descriptor) { codec_descriptor->payload_type = 0; } /* set another codec descriptor */ codec_descriptor = mpf_codec_list_add(&media->codec_list); if(codec_descriptor) { codec_descriptor->payload_type = 96; apt_string_set(&codec_descriptor->name,"PCMU"); codec_descriptor->sampling_rate = 16000; codec_descriptor->channel_count = 1; } rtp_descriptor->audio.local = media; return rtp_descriptor; }
/** Create sample RTP remote descriptor */ static mpf_rtp_stream_descriptor_t* mpf_rtp_remote_descriptor_create(mpf_suite_session_t *session) { mpf_codec_list_t *codec_list; mpf_codec_descriptor_t *codec_descriptor; mpf_rtp_stream_descriptor_t *descriptor = apr_palloc(session->pool,sizeof(mpf_rtp_stream_descriptor_t)); mpf_rtp_stream_descriptor_init(descriptor); descriptor->remote = apr_palloc(session->pool,sizeof(mpf_rtp_media_descriptor_t)); mpf_rtp_media_descriptor_init(descriptor->remote); descriptor->remote->direction = STREAM_DIRECTION_DUPLEX; apt_string_set(&descriptor->remote->ip,"127.0.0.1"); descriptor->remote->port = 5002; codec_list = &descriptor->remote->codec_list; mpf_codec_list_init(codec_list,2,session->pool); codec_descriptor = mpf_codec_list_add(codec_list); if(codec_descriptor) { codec_descriptor->payload_type = 0; } codec_descriptor = mpf_codec_list_add(codec_list); if(codec_descriptor) { codec_descriptor->payload_type = 96; apt_string_set(&codec_descriptor->name,"PCMU"); codec_descriptor->sampling_rate = 16000; codec_descriptor->channel_count = 1; } return descriptor; }
/** Create RTP tx remote descriptor */ static mpf_rtp_stream_descriptor_t* mpf_rtp_tx_remote_descriptor_create(const mpf_suite_agent_t *agent, const mpf_suite_session_t *session) { mpf_codec_list_t *codec_list; mpf_codec_descriptor_t *codec_descriptor; mpf_rtp_media_descriptor_t *media_descriptor; mpf_rtp_stream_descriptor_t *stream_descriptor; media_descriptor = apr_palloc(session->pool,sizeof(mpf_rtp_media_descriptor_t)); mpf_rtp_media_descriptor_init(media_descriptor); media_descriptor->state = MPF_MEDIA_ENABLED; media_descriptor->direction = STREAM_DIRECTION_RECEIVE; apt_string_set(&media_descriptor->ip,"127.0.0.1"); media_descriptor->port = 5000; codec_list = &media_descriptor->codec_list; mpf_codec_list_init(codec_list,1,session->pool); codec_descriptor = mpf_codec_list_add(codec_list); if(codec_descriptor) { codec_descriptor->payload_type = 0; apt_string_set(&codec_descriptor->name,"PCMU"); codec_descriptor->sampling_rate = 8000; codec_descriptor->channel_count = 1; } stream_descriptor = apr_palloc(session->pool,sizeof(mpf_rtp_stream_descriptor_t)); mpf_rtp_stream_descriptor_init(stream_descriptor); stream_descriptor->remote = media_descriptor; stream_descriptor->settings = agent->rtp_settings; return stream_descriptor; }
/** Create sample RTP local descriptor */ static mpf_rtp_stream_descriptor_t* mpf_rtp_local_descriptor_create(mpf_suite_session_t *session) { mpf_rtp_stream_descriptor_t *descriptor = apr_palloc(session->pool,sizeof(mpf_rtp_stream_descriptor_t)); mpf_rtp_stream_descriptor_init(descriptor); descriptor->local = apr_palloc(session->pool,sizeof(mpf_rtp_media_descriptor_t)); mpf_rtp_media_descriptor_init(descriptor->local); descriptor->local->direction = STREAM_DIRECTION_NONE; apt_string_set(&descriptor->local->ip,"127.0.0.1"); descriptor->local->port = 5000; return descriptor; }
/** Generate MRCP descriptor by SDP session */ static apt_bool_t mrcp_descriptor_generate_by_sdp_session(mrcp_session_descriptor_t *descriptor, const sdp_session_t *sdp, const char *force_destination_ip, apr_pool_t *pool) { sdp_media_t *sdp_media; if(force_destination_ip) { apt_string_assign(&descriptor->ip,force_destination_ip,pool); } else if(sdp->sdp_connection) { apt_string_assign(&descriptor->ip,sdp->sdp_connection->c_address,pool); } for(sdp_media=sdp->sdp_media; sdp_media; sdp_media=sdp_media->m_next) { switch(sdp_media->m_type) { case sdp_media_audio: { mpf_rtp_media_descriptor_t *media = apr_palloc(pool,sizeof(mpf_rtp_media_descriptor_t)); mpf_rtp_media_descriptor_init(media); media->id = mrcp_session_audio_media_add(descriptor,media); mpf_rtp_media_generate(media,sdp_media,&descriptor->ip,pool); break; } case sdp_media_video: { mpf_rtp_media_descriptor_t *media = apr_palloc(pool,sizeof(mpf_rtp_media_descriptor_t)); mpf_rtp_media_descriptor_init(media); media->id = mrcp_session_video_media_add(descriptor,media); mpf_rtp_media_generate(media,sdp_media,&descriptor->ip,pool); break; } default: apt_log(APT_LOG_MARK,APT_PRIO_INFO,"Not Supported SDP Media [%s]", sdp_media->m_type_name); break; } } return TRUE; }
/** Create RTP tx local descriptor */ static mpf_rtp_stream_descriptor_t* mpf_rtp_tx_local_descriptor_create(const mpf_suite_agent_t *agent, const mpf_suite_session_t *session) { mpf_rtp_media_descriptor_t *media_descriptor; mpf_rtp_stream_descriptor_t *stream_descriptor; media_descriptor = apr_palloc(session->pool,sizeof(mpf_rtp_media_descriptor_t)); mpf_rtp_media_descriptor_init(media_descriptor); media_descriptor->state = MPF_MEDIA_ENABLED; media_descriptor->direction = STREAM_DIRECTION_SEND; apt_string_set(&media_descriptor->ip,"127.0.0.1"); media_descriptor->port = 5002; stream_descriptor = apr_palloc(session->pool,sizeof(mpf_rtp_stream_descriptor_t)); mpf_rtp_stream_descriptor_init(stream_descriptor); stream_descriptor->local = media_descriptor; stream_descriptor->settings = agent->rtp_settings; return stream_descriptor; }
static apt_bool_t mrcp_client_channel_add(mrcp_client_session_t *session, mrcp_channel_t *channel) { mpf_rtp_termination_descriptor_t *rtp_descriptor = NULL; rtp_termination_slot_t *slot; apr_pool_t *pool = session->base.pool; mrcp_profile_t *profile = session->profile; if(mrcp_client_channel_find(session,channel,NULL) == TRUE) { /* update */ return mrcp_client_channel_modify(session,channel,TRUE); } if(!session->offer) { session->offer = mrcp_session_descriptor_create(pool); } mrcp_client_session_state_set(session,SESSION_STATE_GENERATING_OFFER); if(mrcp_session_version_get(session) == MRCP_VERSION_1) { session->offer->resource_name = channel->resource->name; session->offer->resource_state = TRUE; } else { mrcp_control_descriptor_t *control_media; if(!channel->control_channel) { channel->control_channel = mrcp_client_control_channel_create(profile->connection_agent,channel,pool); mrcp_client_control_channel_log_obj_set(channel->control_channel,session->base.log_obj); } control_media = mrcp_control_offer_create(pool); control_media->id = mrcp_session_control_media_add(session->offer,control_media); mrcp_cmid_add(control_media->cmid_arr,session->offer->control_media_arr->nelts); control_media->resource_name = channel->resource->name; if(mrcp_client_control_channel_add(channel->control_channel,control_media) == TRUE) { channel->waiting_for_channel = TRUE; mrcp_client_session_subrequest_add(session); } } apt_obj_log(APT_LOG_MARK,APT_PRIO_NOTICE,session->base.log_obj,"Add Control Channel "APT_NAMESIDRES_FMT, MRCP_SESSION_NAMESID(session), channel->resource->name.buf); /* add control channel */ APR_ARRAY_PUSH(session->channels,mrcp_channel_t*) = channel; /* add rtp termination slot */ slot = apr_array_push(session->terminations); slot->waiting = FALSE; slot->termination = NULL; slot->descriptor = NULL; slot->channel = channel; slot->id = 0; if(channel->termination) { /* media termination mode */ mpf_termination_t *termination; mpf_audio_stream_t *audio_stream; if(!session->context) { /* create media context first */ session->context = mpf_engine_context_create( profile->media_engine, session->base.name, session,5,pool); } apt_obj_log(APT_LOG_MARK,APT_PRIO_DEBUG,session->base.log_obj,"Add Media Termination "APT_NAMESIDRES_FMT, MRCP_SESSION_NAMESID(session), mpf_termination_name_get(channel->termination)); if(mpf_engine_termination_message_add( profile->media_engine, MPF_ADD_TERMINATION,session->context,channel->termination,NULL, &session->mpf_task_msg) == TRUE) { channel->waiting_for_termination = TRUE; mrcp_client_session_subrequest_add(session); } /* initialize rtp descriptor */ rtp_descriptor = apr_palloc(pool,sizeof(mpf_rtp_termination_descriptor_t)); mpf_rtp_termination_descriptor_init(rtp_descriptor); rtp_descriptor->audio.settings = profile->rtp_settings; audio_stream = mpf_termination_audio_stream_get(channel->termination); if(audio_stream) { mpf_rtp_media_descriptor_t *media; media = apr_palloc(pool,sizeof(mpf_rtp_media_descriptor_t)); mpf_rtp_media_descriptor_init(media); media->state = MPF_MEDIA_ENABLED; media->direction = mpf_stream_reverse_direction_get(audio_stream->direction); rtp_descriptor->audio.local = media; if(audio_stream->capabilities) { rtp_descriptor->audio.capabilities = mpf_stream_capabilities_clone(audio_stream->capabilities,pool); rtp_descriptor->audio.capabilities->direction = media->direction; } } /* create rtp termination */ termination = mpf_termination_create(profile->rtp_termination_factory,session,pool); slot->termination = termination; apt_obj_log(APT_LOG_MARK,APT_PRIO_DEBUG,session->base.log_obj,"Add Media Termination "APT_NAMESIDRES_FMT, MRCP_SESSION_NAMESID(session), mpf_termination_name_get(termination)); /* send add termination request (add to media context) */ if(mpf_engine_termination_message_add( profile->media_engine, MPF_ADD_TERMINATION,session->context,termination,rtp_descriptor, &session->mpf_task_msg) == TRUE) { slot->waiting = TRUE; mrcp_client_session_subrequest_add(session); } mpf_engine_message_send(profile->media_engine,&session->mpf_task_msg); } else { /* bypass media mode */ if(channel->rtp_termination_slot) { rtp_descriptor = channel->rtp_termination_slot->descriptor; if(rtp_descriptor) { if(rtp_descriptor->audio.local) { session->offer->ip = rtp_descriptor->audio.local->ip; session->offer->ext_ip = rtp_descriptor->audio.local->ext_ip; rtp_descriptor->audio.local->id = mrcp_session_audio_media_add(session->offer,rtp_descriptor->audio.local); rtp_descriptor->audio.local->mid = session->offer->audio_media_arr->nelts; slot->id = session->offer->audio_media_arr->nelts - 1; } } } } slot->descriptor = rtp_descriptor; channel->rtp_termination_slot = slot; if(!session->subrequest_count) { /* send offer to server */ mrcp_client_session_offer_send(session); } return TRUE; }
static apt_bool_t mrcp_client_channel_add(mrcp_client_session_t *session, mrcp_channel_t *channel) { mrcp_channel_t **channel_slot; mrcp_control_descriptor_t *control_media; mpf_rtp_termination_descriptor_t *rtp_descriptor = NULL; rtp_termination_slot_t *termination_slot; apr_pool_t *pool = session->base.pool; mrcp_profile_t *profile = session->profile; if(mrcp_client_channel_find(session,channel,NULL) == TRUE) { /* update */ return mrcp_client_channel_modify(session,channel,TRUE); } if(!session->offer) { session->base.signaling_agent = profile->signaling_agent; session->base.signaling_agent->create_client_session(&session->base); session->offer = mrcp_session_descriptor_create(pool); session->context = mpf_context_create(session,5,pool); } if(!channel->resource) { channel->resource = mrcp_resource_get(profile->resource_factory,channel->resource_id); if(!channel->resource) { return FALSE; } channel->resource_name = mrcp_resource_name_get(profile->resource_factory,channel->resource_id); if(!channel->resource_name) { return FALSE; } } if(session->base.signaling_agent->mrcp_version == MRCP_VERSION_1) { session->offer->resource_name = *channel->resource_name; session->offer->resource_state = TRUE; } else { if(!channel->control_channel) { channel->control_channel = mrcp_client_control_channel_create(profile->connection_agent,channel,pool); } control_media = mrcp_control_offer_create(pool); control_media->id = mrcp_session_control_media_add(session->offer,control_media); control_media->cmid = session->offer->control_media_arr->nelts; control_media->resource_name = *channel->resource_name; if(mrcp_client_control_channel_add(channel->control_channel,control_media) == TRUE) { channel->waiting_for_channel = TRUE; session->offer_flag_count++; } } /* add to channel array */ apt_log(APT_LOG_MARK,APT_PRIO_NOTICE,"Add Control Channel <%s@%s>", mrcp_session_str(session), channel->resource_name->buf); channel_slot = apr_array_push(session->channels); *channel_slot = channel; if(channel->termination) { if(mrcp_client_mpf_request_send(profile->media_engine,MPF_COMMAND_ADD,session->context,channel->termination,NULL) == TRUE) { channel->waiting_for_termination = TRUE; session->offer_flag_count++; } } if(channel->rtp_termination_slot) { rtp_descriptor = channel->rtp_termination_slot->descriptor; } /* add to rtp termination array */ apt_log(APT_LOG_MARK,APT_PRIO_DEBUG,"Add RTP Termination"); termination_slot = apr_array_push(session->terminations); termination_slot->waiting = FALSE; termination_slot->termination = NULL; termination_slot->descriptor = NULL; if(rtp_descriptor) { if(rtp_descriptor->audio.local) { session->offer->ip = rtp_descriptor->audio.local->base.ip; rtp_descriptor->audio.local->base.id = mrcp_session_audio_media_add(session->offer,rtp_descriptor->audio.local); rtp_descriptor->audio.local->mid = session->offer->audio_media_arr->nelts; } } else { /* create rtp termination */ mpf_termination_t *termination = mpf_termination_create(profile->rtp_termination_factory,session,session->base.pool); termination_slot->termination = termination; /* initialize rtp descriptor */ rtp_descriptor = apr_palloc(pool,sizeof(mpf_rtp_termination_descriptor_t)); mpf_rtp_termination_descriptor_init(rtp_descriptor); if(channel->termination && channel->termination->audio_stream) { mpf_rtp_media_descriptor_t *media; media = apr_palloc(pool,sizeof(mpf_rtp_media_descriptor_t)); mpf_rtp_media_descriptor_init(media); media->base.state = MPF_MEDIA_ENABLED; media->mode = mpf_stream_mode_negotiate(channel->termination->audio_stream->mode); rtp_descriptor->audio.local = media; } /* send add termination request (add to media context) */ if(mrcp_client_mpf_request_send(profile->media_engine,MPF_COMMAND_ADD,session->context,termination,rtp_descriptor) == TRUE) { termination_slot->waiting = TRUE; session->offer_flag_count++; } } termination_slot->descriptor = rtp_descriptor; channel->rtp_termination_slot = termination_slot; if(!session->offer_flag_count) { /* send offer to server */ mrcp_client_session_offer_send(session); } return TRUE; }
static apt_bool_t mpf_rtp_stream_local_media_create(mpf_rtp_stream_t *rtp_stream, mpf_rtp_media_descriptor_t *local_media, mpf_rtp_media_descriptor_t *remote_media, mpf_stream_capabilities_t *capabilities) { apt_bool_t status = TRUE; if(!local_media) { /* local media is not specified, create the default one */ local_media = apr_palloc(rtp_stream->pool,sizeof(mpf_rtp_media_descriptor_t)); mpf_rtp_media_descriptor_init(local_media); local_media->state = MPF_MEDIA_ENABLED; local_media->direction = STREAM_DIRECTION_DUPLEX; } if(remote_media) { local_media->id = remote_media->id; } if(local_media->ip.length == 0) { local_media->ip = rtp_stream->config->ip; local_media->ext_ip = rtp_stream->config->ext_ip; } if(local_media->port == 0) { /* RTP port management */ apr_port_t first_port_in_search = rtp_stream->config->rtp_port_cur; apt_bool_t is_port_ok = FALSE; do { local_media->port = rtp_stream->config->rtp_port_cur; rtp_stream->config->rtp_port_cur += 2; if(rtp_stream->config->rtp_port_cur == rtp_stream->config->rtp_port_max) { rtp_stream->config->rtp_port_cur = rtp_stream->config->rtp_port_min; } if(mpf_rtp_socket_pair_create(rtp_stream,local_media) == TRUE) { is_port_ok = TRUE; } } while((is_port_ok == FALSE) && (first_port_in_search != rtp_stream->config->rtp_port_cur)); if(is_port_ok == FALSE) { apt_log(APT_LOG_MARK,APT_PRIO_WARNING,"Failed to Find Free RTP Port"); local_media->state = MPF_MEDIA_DISABLED; status = FALSE; } } else if(mpf_rtp_socket_pair_create(rtp_stream,local_media) == FALSE) { local_media->state = MPF_MEDIA_DISABLED; status = FALSE; } if(rtp_stream->config->ptime) { local_media->ptime = rtp_stream->config->ptime; } if(mpf_codec_list_is_empty(&local_media->codec_list) == TRUE) { if(mpf_codec_list_is_empty(&rtp_stream->config->codec_list) == TRUE) { mpf_codec_manager_codec_list_get( rtp_stream->base->termination->codec_manager, &local_media->codec_list, rtp_stream->pool); } else { mpf_codec_list_copy(&local_media->codec_list, &rtp_stream->config->codec_list, rtp_stream->pool); } if(capabilities) { mpf_codec_list_modify(&local_media->codec_list,&capabilities->codecs); } } rtp_stream->local_media = local_media; return status; }