1 /*****************************************************************************
3 *****************************************************************************
4 * Copyright (C) 1999-2004 VideoLAN
7 * Author: Laurent Aimar <fenrir@via.ecp.fr>
9 * This program is free software; you can redistribute it and/or modify
10 * it under the terms of the GNU General Public License as published by
11 * the Free Software Foundation; either version 2 of the License, or
12 * (at your option) any later version.
14 * This program is distributed in the hope that it will be useful,
15 * but WITHOUT ANY WARRANTY; without even the implied warranty of
16 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
17 * GNU General Public License for more details.
19 * You should have received a copy of the GNU General Public License
20 * along with this program; if not, write to the Free Software
21 * Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111, USA.
22 *****************************************************************************/
26 #include <vlc/input.h>
30 int demux_vaControl( input_thread_t *p_input, int i_query, va_list args )
32 if( p_input->pf_demux_control )
34 return p_input->pf_demux_control( p_input, i_query, args );
39 int demux_Control( input_thread_t *p_input, int i_query, ... )
44 va_start( args, i_query );
45 i_result = demux_vaControl( p_input, i_query, args );
51 static void SeekOffset( input_thread_t *p_input, int64_t i_pos );
53 int demux_vaControlDefault( input_thread_t *p_input, int i_query,
60 vlc_mutex_lock( &p_input->stream.stream_lock );
63 case DEMUX_GET_POSITION:
64 pf = (double*)va_arg( args, double * );
65 if( p_input->stream.p_selected_area->i_size <= 0 )
71 *pf = (double)p_input->stream.p_selected_area->i_tell /
72 (double)p_input->stream.p_selected_area->i_size;
77 case DEMUX_SET_POSITION:
78 f = (double)va_arg( args, double );
79 if( p_input->stream.b_seekable && p_input->pf_seek != NULL &&
80 f >= 0.0 && f <= 1.0 )
82 SeekOffset( p_input, (int64_t)(f *
83 (double)p_input->stream.p_selected_area->i_size) );
93 pi64 = (int64_t*)va_arg( args, int64_t * );
94 if( p_input->stream.i_mux_rate > 0 )
96 *pi64 = (int64_t)1000000 *
97 ( p_input->stream.p_selected_area->i_tell / 50 ) /
98 p_input->stream.i_mux_rate;
104 i_ret = VLC_EGENERIC;
109 i64 = (int64_t)va_arg( args, int64_t );
110 if( p_input->stream.i_mux_rate > 0 &&
111 p_input->stream.b_seekable &&
112 p_input->pf_seek != NULL && i64 >= 0 )
114 SeekOffset( p_input, i64 * 50 *
115 (int64_t)p_input->stream.i_mux_rate /
121 i_ret = VLC_EGENERIC;
125 case DEMUX_GET_LENGTH:
126 pi64 = (int64_t*)va_arg( args, int64_t * );
127 if( p_input->stream.i_mux_rate > 0 )
129 *pi64 = (int64_t)1000000 *
130 ( p_input->stream.p_selected_area->i_size / 50 ) /
131 p_input->stream.i_mux_rate;
137 i_ret = VLC_EGENERIC;
141 i_ret = VLC_EGENERIC;
144 i_ret = VLC_EGENERIC;
148 msg_Err( p_input, "unknown query in demux_vaControlDefault" );
149 i_ret = VLC_EGENERIC;
152 vlc_mutex_unlock( &p_input->stream.stream_lock );
157 static void SeekOffset( input_thread_t *p_input, int64_t i_pos )
159 /* Reinitialize buffer manager. */
160 input_AccessReinit( p_input );
162 vlc_mutex_unlock( &p_input->stream.stream_lock );
163 p_input->pf_seek( p_input, i_pos );
164 vlc_mutex_lock( &p_input->stream.stream_lock );
168 /*****************************************************************************
170 *****************************************************************************/
171 demux_t *__demux2_New( vlc_object_t *p_obj,
172 char *psz_mrl, stream_t *s, es_out_t *out )
174 demux_t *p_demux = vlc_object_create( p_obj, VLC_OBJECT_DEMUX );
176 char *psz_dup = strdup( psz_mrl ? psz_mrl : "" );
177 char *psz = strchr( psz_dup, ':' );
180 if( p_demux == NULL )
187 p_demux->psz_access = NULL;
188 p_demux->psz_demux = NULL;
189 p_demux->psz_path = NULL;
195 if( psz[0] == '/' && psz[1] == '/' )
199 p_demux->psz_path = strdup( psz );
201 psz = strchr( psz_dup, '/' );
205 p_demux->psz_access = strdup( psz_dup );
206 p_demux->psz_demux = strdup( psz );
211 p_demux->psz_path = strdup( psz_mrl );
216 if( p_demux->psz_access == NULL )
218 p_demux->psz_access = strdup( "" );
220 if( p_demux->psz_demux == NULL )
222 p_demux->psz_demux = strdup( "" );
224 if( p_demux->psz_path == NULL )
226 p_demux->psz_path = strdup( "" );
228 msg_Dbg( p_obj, "demux2_New: '%s' -> access='%s' demux='%s' path='%s'",
230 p_demux->psz_access, p_demux->psz_demux, p_demux->psz_path );
235 p_demux->pf_demux = NULL;
236 p_demux->pf_control = NULL;
237 p_demux->p_sys = NULL;
239 psz_module = p_demux->psz_demux;
240 if( *psz_module == '\0' && strrchr( p_demux->psz_path, '.' ) )
242 /* XXX: add only file without any problem here and with strong detection.
243 * - no .mp3, .a52, ... (aac is added as it works only by file ext anyway
244 * - wav can't be added 'cause of a52 and dts in them as raw audio
246 static struct { char *ext; char *demux; } exttodemux[] =
250 { "asf", "asf" }, { "wmv", "asf" }, { "wma", "asf" },
256 { "mkv", "mkv" }, { "mka", "mkv" }, { "mks", "mkv" },
257 { "mp4", "mp4" }, { "m4a", "mp4" }, { "mov", "mp4" }, { "moov", "mp4" },
258 { "mod", "mod" }, { "xm", "mod" },
260 { "ogg", "ogg" }, { "ogm", "ogg" },
266 char *psz_ext = strrchr( p_demux->psz_path, '.' ) + 1;
269 for( i = 0; exttodemux[i].ext != NULL; i++ )
271 if( !strcasecmp( psz_ext, exttodemux[i].ext ) )
273 psz_module = exttodemux[i].demux;
279 /* Before module_Need (for var_Create...) */
280 vlc_object_attach( p_demux, p_obj );
283 module_Need( p_demux, "demux2", psz_module,
284 !strcmp( psz_module, p_demux->psz_demux ) ? VLC_TRUE : VLC_FALSE );
286 if( p_demux->p_module == NULL )
288 vlc_object_detach( p_demux );
289 free( p_demux->psz_path );
290 free( p_demux->psz_demux );
291 free( p_demux->psz_access );
292 vlc_object_destroy( p_demux );
299 /*****************************************************************************
301 *****************************************************************************/
302 void demux2_Delete( demux_t *p_demux )
304 module_Unneed( p_demux, p_demux->p_module );
305 vlc_object_detach( p_demux );
307 free( p_demux->psz_path );
308 free( p_demux->psz_demux );
309 free( p_demux->psz_access );
311 vlc_object_destroy( p_demux );
314 /*****************************************************************************
315 * demux2_vaControlHelper:
316 *****************************************************************************/
317 int demux2_vaControlHelper( stream_t *s,
318 int64_t i_start, int64_t i_end,
319 int i_bitrate, int i_align,
320 int i_query, va_list args )
326 if( i_end < 0 ) i_end = stream_Size( s );
327 if( i_start < 0 ) i_start = 0;
328 if( i_align <= 0 ) i_align = 1;
329 i_tell = stream_Tell( s );
333 case DEMUX_GET_LENGTH:
334 pi64 = (int64_t*)va_arg( args, int64_t * );
335 if( i_bitrate > 0 && i_end > i_start )
337 *pi64 = I64C(8000000) * (i_end - i_start) / i_bitrate;
343 pi64 = (int64_t*)va_arg( args, int64_t * );
344 if( i_bitrate > 0 && i_end > i_start )
346 *pi64 = I64C(8000000) * (i_tell - i_start) / i_bitrate;
351 case DEMUX_GET_POSITION:
352 pf = (double*)va_arg( args, double * );
353 if( i_start < i_end )
355 *pf = (double)( i_tell - i_start ) /
356 (double)( i_end - i_start );
362 case DEMUX_SET_POSITION:
363 f = (double)va_arg( args, double );
364 if( i_start < i_end && f >= 0.0 && f <= 1.0 )
366 int64_t i_block = (f * ( i_end - i_start )) / i_align;
368 if( stream_Seek( s, i_start + i_block * i_align ) )
377 i64 = (int64_t)va_arg( args, int64_t );
378 if( i_bitrate > 0 && i64 >= 0 )
380 int64_t i_block = i64 * i_bitrate / I64C(8000000) / i_align;
381 if( stream_Seek( s, i_start + i_block * i_align ) )
394 msg_Err( s, "unknown query in demux_vaControlDefault" );