/** * Initializes the raw dump pseudo-demuxer. */ static int Open( vlc_object_t * p_this ) { demux_t *p_demux = (demux_t*)p_this; /* Accept only if forced */ if( !p_demux->b_force ) return VLC_EGENERIC; char *access = var_InheritString( p_demux, "demuxdump-access" ); if( access == NULL ) return VLC_EGENERIC; /* --sout-file-append (defaults to false) */ var_Create( p_demux, "sout-file-append", VLC_VAR_BOOL ); if( var_InheritBool( p_demux, "demuxdump-append" ) ) var_SetBool( p_demux, "sout-file-append", true ); /* --sout-file-format (always false) */ var_Create( p_demux, "sout-file-format", VLC_VAR_BOOL ); char *path = var_InheritString( p_demux, "demuxdump-file" ); if( path == NULL ) { free( access ); msg_Err( p_demux, "no dump file name given" ); return VLC_EGENERIC; } sout_access_out_t *out = sout_AccessOutNew( p_demux, access, path ); free( path ); free( access ); if( out == NULL ) { msg_Err( p_demux, "cannot create output" ); return VLC_EGENERIC; } p_demux->p_sys = (void *)out; p_demux->pf_demux = Demux; p_demux->pf_control = Control; return VLC_SUCCESS; }
static sout_stream_id_t *Add( sout_stream_t *p_stream, es_format_t *p_fmt ) { sout_stream_sys_t *p_sys = p_stream->p_sys; sout_instance_t *p_sout = p_stream->p_sout; sout_stream_id_t *id; char *psz_access; char *psz_mux; char *psz_dst; sout_access_out_t *p_access; sout_mux_t *p_mux; /* *** get access name *** */ if( p_fmt->i_cat == AUDIO_ES && p_sys->psz_access_audio && *p_sys->psz_access_audio ) { psz_access = p_sys->psz_access_audio; } else if( p_fmt->i_cat == VIDEO_ES && p_sys->psz_access_video && *p_sys->psz_access_video ) { psz_access = p_sys->psz_access_video; } else { psz_access = p_sys->psz_access; } /* *** get mux name *** */ if( p_fmt->i_cat == AUDIO_ES && p_sys->psz_mux_audio && *p_sys->psz_mux_audio ) { psz_mux = p_sys->psz_mux_audio; } else if( p_fmt->i_cat == VIDEO_ES && p_sys->psz_mux_video && *p_sys->psz_mux_video ) { psz_mux = p_sys->psz_mux_video; } else { psz_mux = p_sys->psz_mux; } /* Get url (%d expanded as a codec count, %c expanded as codec fcc ) */ if( p_fmt->i_cat == AUDIO_ES && p_sys->psz_dst_audio && *p_sys->psz_dst_audio ) { psz_dst = es_print_url( p_sys->psz_dst_audio, p_fmt->i_codec, p_sys->i_count_audio, psz_access, psz_mux ); } else if( p_fmt->i_cat == VIDEO_ES && p_sys->psz_dst_video && *p_sys->psz_dst_video ) { psz_dst = es_print_url( p_sys->psz_dst_video, p_fmt->i_codec, p_sys->i_count_video, psz_access, psz_mux ); } else { int i_count; if( p_fmt->i_cat == VIDEO_ES ) { i_count = p_sys->i_count_video; } else if( p_fmt->i_cat == AUDIO_ES ) { i_count = p_sys->i_count_audio; } else { i_count = p_sys->i_count; } psz_dst = es_print_url( p_sys->psz_dst, p_fmt->i_codec, i_count, psz_access, psz_mux ); } p_sys->i_count++; if( p_fmt->i_cat == VIDEO_ES ) { p_sys->i_count_video++; } else if( p_fmt->i_cat == AUDIO_ES ) { p_sys->i_count_audio++; } msg_Dbg( p_stream, "creating `%s/%s://%s'", psz_access, psz_mux, psz_dst ); /* *** find and open appropriate access module *** */ p_access = sout_AccessOutNew( p_sout, psz_access, psz_dst ); if( p_access == NULL ) { msg_Err( p_stream, "no suitable sout access module for `%s/%s://%s'", psz_access, psz_mux, psz_dst ); dialog_Fatal( p_stream, _("Streaming / Transcoding failed"), _("There is no suitable stream-output access module for \"%s/%s://%s\"."), psz_access, psz_mux, psz_dst ); free( psz_dst ); return( NULL ); } /* *** find and open appropriate mux module *** */ p_mux = sout_MuxNew( p_sout, psz_mux, p_access ); if( p_mux == NULL ) { msg_Err( p_stream, "no suitable sout mux module for `%s/%s://%s'", psz_access, psz_mux, psz_dst ); dialog_Fatal( p_stream, _("Streaming / Transcoding failed"), _("There is no suitable stream-output access module "\ "for \"%s/%s://%s\"."), psz_access, psz_mux, psz_dst ); sout_AccessOutDelete( p_access ); free( psz_dst ); return( NULL ); } free( psz_dst ); id = malloc( sizeof( sout_stream_id_t ) ); if( !id ) { sout_MuxDelete( p_mux ); sout_AccessOutDelete( p_access ); return NULL; } id->p_mux = p_mux; id->p_input = sout_MuxAddStream( p_mux, p_fmt ); if( id->p_input == NULL ) { sout_MuxDelete( p_mux ); sout_AccessOutDelete( p_access ); free( id ); return NULL; } if( !sout_AccessOutCanControlPace( p_access ) ) p_sout->i_out_pace_nocontrol++; return id; }
/***************************************************************************** * Open: *****************************************************************************/ static int Open( vlc_object_t *p_this ) { sout_stream_t *p_stream = (sout_stream_t*)p_this; sout_instance_t *p_sout = p_stream->p_sout; sout_stream_sys_t *p_sys; char *psz_mux, *psz_access, *psz_url; sout_access_out_t *p_access; int ret = VLC_EGENERIC; config_ChainParse( p_stream, SOUT_CFG_PREFIX, ppsz_sout_options, p_stream->p_cfg ); psz_mux = var_GetNonEmptyString( p_stream, SOUT_CFG_PREFIX "mux" ); psz_access = var_GetNonEmptyString( p_stream, SOUT_CFG_PREFIX "access" ); if( !psz_access ) { if( !strcmp( p_stream->psz_name, "http" ) ) psz_access = strdup("http"); else if (!strcmp (p_stream->psz_name, "udp")) psz_access = strdup("udp"); else if (!strcmp (p_stream->psz_name, "file")) psz_access = strdup("file"); } psz_url = var_GetNonEmptyString( p_stream, SOUT_CFG_PREFIX "dst" ); if (!psz_url) { char *psz_bind = var_GetNonEmptyString( p_stream, SOUT_CFG_PREFIX "bind" ); if( psz_bind ) { char *psz_path = var_GetNonEmptyString( p_stream, SOUT_CFG_PREFIX "path" ); if( psz_path ) { if( asprintf( &psz_url, "%s/%s", psz_bind, psz_path ) == -1 ) psz_url = NULL; free(psz_bind); free( psz_path ); } else psz_url = psz_bind; } } p_sys = p_stream->p_sys = malloc( sizeof( sout_stream_sys_t) ); if( !p_sys ) { ret = VLC_ENOMEM; goto end; } p_sys->p_session = NULL; if( fixAccessMux( p_stream, &psz_mux, &psz_access, psz_url ) ) goto end; checkAccessMux( p_stream, psz_access, psz_mux ); p_access = sout_AccessOutNew( p_sout, psz_access, psz_url ); if( p_access == NULL ) { msg_Err( p_stream, "no suitable sout access module for `%s/%s://%s'", psz_access, psz_mux, psz_url ); goto end; } p_sys->p_mux = sout_MuxNew( p_sout, psz_mux, p_access ); if( !p_sys->p_mux ) { const char *psz_mux_guess = getMuxFromAlias( psz_mux ); if( psz_mux_guess && strcmp( psz_mux_guess, psz_mux ) ) { msg_Dbg( p_stream, "Couldn't open mux `%s', trying `%s' instead", psz_mux, psz_mux_guess ); p_sys->p_mux = sout_MuxNew( p_sout, psz_mux_guess, p_access ); } if( !p_sys->p_mux ) { msg_Err( p_stream, "no suitable sout mux module for `%s/%s://%s'", psz_access, psz_mux, psz_url ); sout_AccessOutDelete( p_access ); goto end; } } if( var_GetBool( p_stream, SOUT_CFG_PREFIX"sap" ) ) create_SDP( p_stream, p_access ); if( !sout_AccessOutCanControlPace( p_access ) ) p_sout->i_out_pace_nocontrol++; p_stream->pf_add = Add; p_stream->pf_del = Del; p_stream->pf_send = Send; ret = VLC_SUCCESS; msg_Dbg( p_this, "using `%s/%s://%s'", psz_access, psz_mux, psz_url ); end: if( ret != VLC_SUCCESS ) free( p_sys ); free( psz_access ); free( psz_mux ); free( psz_url ); return ret; }
/***************************************************************************** * Open: *****************************************************************************/ static int Open( vlc_object_t *p_this ) { sout_stream_t *p_stream = (sout_stream_t*)p_this; sout_instance_t *p_sout = p_stream->p_sout; slp_session_t *p_slp = NULL; char *psz_mux; char *psz_access; char *psz_url; vlc_value_t val; sout_access_out_t *p_access; sout_mux_t *p_mux; char *psz_mux_byext = NULL; sout_CfgParse( p_stream, SOUT_CFG_PREFIX, ppsz_sout_options, p_stream->p_cfg ); var_Get( p_stream, SOUT_CFG_PREFIX "access", &val ); psz_access = *val.psz_string ? val.psz_string : NULL; if( val.psz_string && !*val.psz_string ) free( val.psz_string ); var_Get( p_stream, SOUT_CFG_PREFIX "mux", &val ); psz_mux = *val.psz_string ? val.psz_string : NULL; if( val.psz_string && !*val.psz_string ) free( val.psz_string ); var_Get( p_stream, SOUT_CFG_PREFIX "url", &val ); psz_url = *val.psz_string ? val.psz_string : NULL; if( val.psz_string && !*val.psz_string ) free( val.psz_string ); p_stream->p_sys = malloc( sizeof( sout_stream_sys_t) ); p_stream->p_sys->p_session = NULL; msg_Dbg( p_this, "creating `%s/%s://%s'", psz_access, psz_mux, psz_url ); /* ext -> muxer name */ if( psz_url && strrchr( psz_url, '.' ) ) { /* by extention */ static struct { char *ext; char *mux; } exttomux[] = { { "avi", "avi" }, { "ogg", "ogg" }, { "ogm", "ogg" }, { "mp4", "mp4" }, { "mov", "mov" }, { "moov","mov" }, { "asf", "asf" }, { "wma", "asf" }, { "wmv", "asf" }, { "trp", "ts" }, { "ts", "ts" }, { "mpg", "ps" }, { "mpeg","ps" }, { "ps", "ps" }, { "mpeg1","mpeg1" }, { NULL, NULL } }; char *psz_ext = strrchr( psz_url, '.' ) + 1; int i; msg_Dbg( p_this, "extention is %s", psz_ext ); for( i = 0; exttomux[i].ext != NULL; i++ ) { if( !strcasecmp( psz_ext, exttomux[i].ext ) ) { psz_mux_byext = exttomux[i].mux; break; } } msg_Dbg( p_this, "extention -> mux=%s", psz_mux_byext ); } /* We fix access/mux to valid couple */ if( !psz_access && !psz_mux ) { if( psz_mux_byext ) { msg_Warn( p_stream, "no access _and_ no muxer, extention gives file/%s", psz_mux_byext ); psz_access = strdup("file"); psz_mux = strdup(psz_mux_byext); } else { msg_Err( p_stream, "no access _and_ no muxer (fatal error)" ); return VLC_EGENERIC; } } if( psz_access && !psz_mux ) { /* access given, no mux */ if( !strncmp( psz_access, "mmsh", 4 ) ) { psz_mux = strdup("asfh"); } else if( !strncmp( psz_access, "udp", 3 ) ) { psz_mux = strdup("ts"); } else if( psz_mux_byext ) { psz_mux = strdup(psz_mux_byext); } else { msg_Err( p_stream, "no mux specified or found by extention" ); return VLC_EGENERIC; } } else if( psz_mux && !psz_access ) { /* mux given, no access */ if( !strncmp( psz_mux, "asfh", 4 ) ) { psz_access = strdup("mmsh"); } else { /* default file */ psz_access = strdup("file"); } } /* fix or warm of incompatible couple */ if( psz_mux && psz_access ) { if( !strncmp( psz_access, "mmsh", 4 ) && strncmp( psz_mux, "asfh", 4 ) ) { char *p = strchr( psz_mux,'{' ); msg_Warn( p_stream, "fixing to mmsh/asfh" ); if( p ) { /* -> a little memleak but ... */ psz_mux = malloc( strlen( "asfh" ) + strlen( p ) + 1); sprintf( psz_mux, "asfh%s", p ); } else { psz_mux = strdup("asfh"); } } else if( ( !strncmp( psz_access, "rtp", 3 ) || !strncmp( psz_access, "udp", 3 ) ) && strncmp( psz_mux, "ts", 2 ) ) { msg_Err( p_stream, "for now udp and rtp are only valid with TS" ); } else if( strncmp( psz_access, "file", 4 ) && ( !strncmp( psz_mux, "mov", 3 ) || !strncmp( psz_mux, "mp4", 3 ) ) ) { msg_Err( p_stream, "mov and mp4 work only with file output" ); } } msg_Dbg( p_this, "using `%s/%s://%s'", psz_access, psz_mux, psz_url ); /* *** find and open appropriate access module *** */ p_access = sout_AccessOutNew( p_sout, psz_access, psz_url ); if( p_access == NULL ) { msg_Err( p_stream, "no suitable sout access module for `%s/%s://%s'", psz_access, psz_mux, psz_url ); if( psz_access ) free( psz_access ); if( psz_mux ) free( psz_mux ); return VLC_EGENERIC; } msg_Dbg( p_stream, "access opened" ); /* *** find and open appropriate mux module *** */ p_mux = sout_MuxNew( p_sout, psz_mux, p_access ); if( p_mux == NULL ) { msg_Err( p_stream, "no suitable sout mux module for `%s/%s://%s'", psz_access, psz_mux, psz_url ); sout_AccessOutDelete( p_access ); if( psz_access ) free( psz_access ); if( psz_mux ) free( psz_mux ); return VLC_EGENERIC; } msg_Dbg( p_stream, "mux opened" ); /* *** Create the SAP Session structure *** */ var_Get( p_stream, SOUT_CFG_PREFIX "sap", &val ); if( val.b_bool && ( strstr( psz_access, "udp" ) || strstr( psz_access , "rtp" ) ) ) { session_descriptor_t *p_session = sout_AnnounceSessionCreate(); announce_method_t *p_method = sout_AnnounceMethodCreate( METHOD_TYPE_SAP ); vlc_url_t url; var_Get( p_stream, SOUT_CFG_PREFIX "name", &val ); if( *val.psz_string ) { p_session->psz_name = strdup( val.psz_string ); } else { p_session->psz_name = strdup( psz_url ); } free( val.psz_string ); var_Get( p_stream, SOUT_CFG_PREFIX "group", &val ); if( *val.psz_string ) { p_session->psz_group = strdup( val.psz_string ); } free( val.psz_string ); var_Get( p_stream, SOUT_CFG_PREFIX "sap-ipv6", &val ); p_method->i_ip_version = val.b_bool ? 6 : 4; /* Now, parse the URL to extract host and port */ vlc_UrlParse( &url, psz_url , 0); if( url.psz_host ) { if( url.i_port == 0 ) url.i_port = DEFAULT_PORT; p_session->psz_uri = url.psz_host; p_session->i_port = url.i_port; p_session->psz_sdp = NULL; p_session->i_ttl = config_GetInt( p_sout, "ttl" ); p_session->i_payload = 33; msg_Info( p_this, "SAP Enabled"); sout_AnnounceRegister( p_sout, p_session, p_method ); /* FIXME: Free p_method */ p_stream->p_sys->p_session = p_session; } vlc_UrlClean( &url ); if( p_method->psz_address) free( p_method->psz_address ); free( p_method ); } /* *** Register with slp *** */ #ifdef HAVE_SLP_H var_Get( p_stream, SOUT_CFG_PREFIX "slp", &val ); if( val.b_bool && ( strstr( psz_access, "udp" ) || strstr( psz_access , "rtp" ) ) ) { int i_ret; msg_Info( p_this, "SLP Enabled"); var_Get( p_stream, SOUT_CFG_PREFIX "name", &val ); if( *val.psz_string ) { i_ret = sout_SLPReg( p_sout, psz_url, val.psz_string ); } else { i_ret = sout_SLPReg( p_sout, psz_url, psz_url ); } if( i_ret ) { msg_Warn( p_sout, "SLP Registering failed"); } else { p_slp = malloc(sizeof(slp_session_t)); p_slp->psz_url = strdup( psz_url ); p_slp->psz_name = strdup( *val.psz_string ? val.psz_string : psz_url ); } free( val.psz_string ); } #endif p_stream->pf_add = Add; p_stream->pf_del = Del; p_stream->pf_send = Send; p_stream->p_sys->p_mux = p_mux; p_stream->p_sys->p_slp = p_slp; if( psz_access ) free( psz_access ); if( psz_mux ) free( psz_mux ); if( psz_url ) free( psz_url ); return VLC_SUCCESS; }