Commit 01aca2cc authored by Clément Stenac's avatar Clément Stenac

*experimental* input pre-parsing support.

Won't work for many things (only those for which metadata can be gotten during initialization).

Processing is done asynchronously, use playlist_PreparseEnqueue to add an item to the queue.
parent 3e46b414
...@@ -104,8 +104,8 @@ struct access_t ...@@ -104,8 +104,8 @@ struct access_t
access_sys_t *p_sys; access_sys_t *p_sys;
}; };
#define access2_New( a, b, c, d ) __access2_New(VLC_OBJECT(a), b, c, d ) #define access2_New( a, b, c, d, e ) __access2_New(VLC_OBJECT(a), b, c, d, e )
VLC_EXPORT( access_t *, __access2_New, ( vlc_object_t *p_obj, char *psz_access, char *psz_demux, char *psz_path ) ); VLC_EXPORT( access_t *, __access2_New, ( vlc_object_t *p_obj, char *psz_access, char *psz_demux, char *psz_path, vlc_bool_t b_quick ) );
VLC_EXPORT( void, access2_Delete, ( access_t * ) ); VLC_EXPORT( void, access2_Delete, ( access_t * ) );
static inline int access2_vaControl( access_t *p_access, int i_query, va_list args ) static inline int access2_vaControl( access_t *p_access, int i_query, va_list args )
......
...@@ -225,6 +225,7 @@ typedef struct playlist_export_t playlist_export_t; ...@@ -225,6 +225,7 @@ typedef struct playlist_export_t playlist_export_t;
typedef struct services_discovery_t services_discovery_t; typedef struct services_discovery_t services_discovery_t;
typedef struct services_discovery_sys_t services_discovery_sys_t; typedef struct services_discovery_sys_t services_discovery_sys_t;
typedef struct playlist_add_t playlist_add_t; typedef struct playlist_add_t playlist_add_t;
typedef struct playlist_preparse_t playlist_preparse_t;
/* Modules */ /* Modules */
typedef struct module_bank_t module_bank_t; typedef struct module_bank_t module_bank_t;
......
...@@ -109,8 +109,8 @@ enum demux_query_e ...@@ -109,8 +109,8 @@ enum demux_query_e
}; };
/* stream_t *s could be null and then it mean a access+demux in one */ /* stream_t *s could be null and then it mean a access+demux in one */
#define demux2_New( a, b, c, d, e, f ) __demux2_New(VLC_OBJECT(a),b,c,d,e,f) #define demux2_New( a, b, c, d, e, f,g ) __demux2_New(VLC_OBJECT(a),b,c,d,e,f,g)
VLC_EXPORT( demux_t *, __demux2_New, ( vlc_object_t *p_obj, char *psz_access, char *psz_demux, char *psz_path, stream_t *s, es_out_t *out ) ); VLC_EXPORT( demux_t *, __demux2_New, ( vlc_object_t *p_obj, char *psz_access, char *psz_demux, char *psz_path, stream_t *s, es_out_t *out, vlc_bool_t ) );
VLC_EXPORT( void, demux2_Delete, ( demux_t * ) ); VLC_EXPORT( void, demux2_Delete, ( demux_t * ) );
VLC_EXPORT( int, demux2_vaControlHelper, ( stream_t *, int64_t i_start, int64_t i_end, int i_bitrate, int i_align, int i_query, va_list args ) ); VLC_EXPORT( int, demux2_vaControlHelper, ( stream_t *, int64_t i_start, int64_t i_end, int i_bitrate, int i_align, int i_query, va_list args ) );
......
...@@ -398,6 +398,8 @@ struct input_thread_t ...@@ -398,6 +398,8 @@ struct input_thread_t
*****************************************************************************/ *****************************************************************************/
#define input_CreateThread(a,b) __input_CreateThread(VLC_OBJECT(a),b) #define input_CreateThread(a,b) __input_CreateThread(VLC_OBJECT(a),b)
VLC_EXPORT( input_thread_t *, __input_CreateThread, ( vlc_object_t *, input_item_t * ) ); VLC_EXPORT( input_thread_t *, __input_CreateThread, ( vlc_object_t *, input_item_t * ) );
#define input_Preparse(a,b) __input_Preparse(VLC_OBJECT(a),b)
VLC_EXPORT( int, __input_Preparse, ( vlc_object_t *, input_item_t * ) );
VLC_EXPORT( void, input_StopThread, ( input_thread_t * ) ); VLC_EXPORT( void, input_StopThread, ( input_thread_t * ) );
VLC_EXPORT( void, input_DestroyThread, ( input_thread_t * ) ); VLC_EXPORT( void, input_DestroyThread, ( input_thread_t * ) );
......
...@@ -127,6 +127,14 @@ struct services_discovery_t ...@@ -127,6 +127,14 @@ struct services_discovery_t
void (*pf_run) ( services_discovery_t *); void (*pf_run) ( services_discovery_t *);
}; };
struct playlist_preparse_t
{
VLC_COMMON_MEMBERS
vlc_mutex_t lock;
int i_waiting;
input_item_t **pp_waiting;
};
/** /**
* Structure containing information about the playlist * Structure containing information about the playlist
...@@ -194,6 +202,8 @@ struct playlist_t ...@@ -194,6 +202,8 @@ struct playlist_t
vlc_mutex_t lock; /**< Lock to protect request */ vlc_mutex_t lock; /**< Lock to protect request */
} request; } request;
playlist_preparse_t *p_preparse;
/*@}*/ /*@}*/
}; };
...@@ -238,6 +248,7 @@ VLC_EXPORT( int, playlist_Control, ( playlist_t *, int, ... ) ); ...@@ -238,6 +248,7 @@ VLC_EXPORT( int, playlist_Control, ( playlist_t *, int, ... ) );
VLC_EXPORT( int, playlist_Clear, ( playlist_t * ) ); VLC_EXPORT( int, playlist_Clear, ( playlist_t * ) );
VLC_EXPORT( int, playlist_PreparseEnqueue, (playlist_t *, input_item_t *) );
/* Services discovery */ /* Services discovery */
......
...@@ -90,6 +90,7 @@ enum ...@@ -90,6 +90,7 @@ enum
PopupPlay_Event, PopupPlay_Event,
PopupPlayThis_Event, PopupPlayThis_Event,
PopupPreparse_Event,
PopupSort_Event, PopupSort_Event,
PopupDel_Event, PopupDel_Event,
PopupEna_Event, PopupEna_Event,
...@@ -150,6 +151,7 @@ BEGIN_EVENT_TABLE(Playlist, wxFrame) ...@@ -150,6 +151,7 @@ BEGIN_EVENT_TABLE(Playlist, wxFrame)
/* Popup events */ /* Popup events */
EVT_MENU( PopupPlay_Event, Playlist::OnPopupPlay) EVT_MENU( PopupPlay_Event, Playlist::OnPopupPlay)
EVT_MENU( PopupPlayThis_Event, Playlist::OnPopupPlay) EVT_MENU( PopupPlayThis_Event, Playlist::OnPopupPlay)
EVT_MENU( PopupPreparse_Event, Playlist::OnPopupPreparse)
EVT_MENU( PopupSort_Event, Playlist::OnPopupSort) EVT_MENU( PopupSort_Event, Playlist::OnPopupSort)
EVT_MENU( PopupDel_Event, Playlist::OnPopupDel) EVT_MENU( PopupDel_Event, Playlist::OnPopupDel)
EVT_MENU( PopupEna_Event, Playlist::OnPopupEna) EVT_MENU( PopupEna_Event, Playlist::OnPopupEna)
...@@ -266,6 +268,7 @@ Playlist::Playlist( intf_thread_t *_p_intf, wxWindow *p_parent ): ...@@ -266,6 +268,7 @@ Playlist::Playlist( intf_thread_t *_p_intf, wxWindow *p_parent ):
popup_menu = new wxMenu; popup_menu = new wxMenu;
popup_menu->Append( PopupPlay_Event, wxU(_("Play")) ); popup_menu->Append( PopupPlay_Event, wxU(_("Play")) );
popup_menu->Append( PopupPlayThis_Event, wxU(_("Play this branch")) ); popup_menu->Append( PopupPlayThis_Event, wxU(_("Play this branch")) );
popup_menu->Append( PopupPreparse_Event, wxU(_("Preparse")) );
popup_menu->Append( PopupSort_Event, wxU(_("Sort this branch")) ); popup_menu->Append( PopupSort_Event, wxU(_("Sort this branch")) );
popup_menu->Append( PopupDel_Event, wxU(_("Delete")) ); popup_menu->Append( PopupDel_Event, wxU(_("Delete")) );
popup_menu->Append( PopupEna_Event, wxU(_("Enable/Disable")) ); popup_menu->Append( PopupEna_Event, wxU(_("Enable/Disable")) );
...@@ -1139,6 +1142,7 @@ void Playlist::OnActivateItem( wxTreeEvent& event ) ...@@ -1139,6 +1142,7 @@ void Playlist::OnActivateItem( wxTreeEvent& event )
playlist_item_t *p_item,*p_node; playlist_item_t *p_item,*p_node;
playlist_t *p_playlist = (playlist_t *)vlc_object_find( p_intf, playlist_t *p_playlist = (playlist_t *)vlc_object_find( p_intf,
VLC_OBJECT_PLAYLIST,FIND_ANYWHERE ); VLC_OBJECT_PLAYLIST,FIND_ANYWHERE );
PlaylistItem *p_wxitem = (PlaylistItem *)treectrl->GetItemData( PlaylistItem *p_wxitem = (PlaylistItem *)treectrl->GetItemData(
event.GetItem() ); event.GetItem() );
wxTreeItemId parent = treectrl->GetItemParent( event.GetItem() ); wxTreeItemId parent = treectrl->GetItemParent( event.GetItem() );
...@@ -1380,9 +1384,6 @@ void Playlist::OnPopup( wxContextMenuEvent& event ) ...@@ -1380,9 +1384,6 @@ void Playlist::OnPopup( wxContextMenuEvent& event )
Playlist::PopupMenu( popup_menu, Playlist::PopupMenu( popup_menu,
ScreenToClient( wxGetMousePosition() ) ); ScreenToClient( wxGetMousePosition() ) );
} }
else
{
}
} }
void Playlist::OnPopupPlay( wxMenuEvent& event ) void Playlist::OnPopupPlay( wxMenuEvent& event )
...@@ -1424,6 +1425,38 @@ void Playlist::OnPopupPlay( wxMenuEvent& event ) ...@@ -1424,6 +1425,38 @@ void Playlist::OnPopupPlay( wxMenuEvent& event )
vlc_object_release( p_playlist ); vlc_object_release( p_playlist );
} }
void Playlist::OnPopupPreparse( wxMenuEvent& event )
{
playlist_t *p_playlist =
(playlist_t *)vlc_object_find( p_intf, VLC_OBJECT_PLAYLIST,
FIND_ANYWHERE );
if( p_playlist == NULL )
{
return;
}
if( p_popup_item != NULL )
{
if( p_popup_item->i_children == -1 )
{
playlist_PreparseEnqueue( p_playlist, &p_popup_item->input );
}
else
{
int i = 0;
playlist_item_t *p_parent = p_popup_item;
for( i = 0; i< p_parent->i_children ; i++ )
{
wxMenuEvent dummy;
i_popup_item = FindItem( treectrl->GetRootItem(),
p_parent->pp_children[i] );
p_popup_item = p_parent->pp_children[i];
OnPopupPreparse( dummy );
}
}
}
vlc_object_release( p_playlist );
}
void Playlist::OnPopupDel( wxMenuEvent& event ) void Playlist::OnPopupDel( wxMenuEvent& event )
{ {
PlaylistItem *p_wxitem; PlaylistItem *p_wxitem;
......
...@@ -108,9 +108,14 @@ void Timer::Notify() ...@@ -108,9 +108,14 @@ void Timer::Notify()
/* Update the input */ /* Update the input */
if( p_intf->p_sys->p_input == NULL ) if( p_intf->p_sys->p_input == NULL )
{ {
p_intf->p_sys->p_input = playlist_t *p_playlist =
(input_thread_t *)vlc_object_find( p_intf, VLC_OBJECT_INPUT, (playlist_t *)vlc_object_find( p_intf, VLC_OBJECT_PLAYLIST,
FIND_ANYWHERE ); FIND_ANYWHERE );
if( p_playlist != NULL )
{
p_intf->p_sys->p_input = p_playlist->p_input;
vlc_object_release( p_playlist );
}
/* Refresh interface */ /* Refresh interface */
if( p_intf->p_sys->p_input ) if( p_intf->p_sys->p_input )
...@@ -140,7 +145,6 @@ void Timer::Notify() ...@@ -140,7 +145,6 @@ void Timer::Notify()
p_main_interface->statusbar->SetStatusText( wxT(""), 0 ); p_main_interface->statusbar->SetStatusText( wxT(""), 0 );
p_main_interface->statusbar->SetStatusText( wxT(""), 2 ); p_main_interface->statusbar->SetStatusText( wxT(""), 2 );
vlc_object_release( p_intf->p_sys->p_input );
p_intf->p_sys->p_input = NULL; p_intf->p_sys->p_input = NULL;
} }
......
...@@ -175,11 +175,6 @@ static void Close( vlc_object_t *p_this ) ...@@ -175,11 +175,6 @@ static void Close( vlc_object_t *p_this )
{ {
intf_thread_t *p_intf = (intf_thread_t *)p_this; intf_thread_t *p_intf = (intf_thread_t *)p_this;
if( p_intf->p_sys->p_input )
{
vlc_object_release( p_intf->p_sys->p_input );
}
vlc_mutex_lock( &p_intf->object_lock ); vlc_mutex_lock( &p_intf->object_lock );
p_intf->b_dead = VLC_TRUE; p_intf->b_dead = VLC_TRUE;
vlc_mutex_unlock( &p_intf->object_lock ); vlc_mutex_unlock( &p_intf->object_lock );
......
...@@ -846,6 +846,7 @@ private: ...@@ -846,6 +846,7 @@ private:
playlist_item_t *p_popup_parent; playlist_item_t *p_popup_parent;
void OnPopup( wxContextMenuEvent& event ); void OnPopup( wxContextMenuEvent& event );
void OnPopupPlay( wxMenuEvent& event ); void OnPopupPlay( wxMenuEvent& event );
void OnPopupPreparse( wxMenuEvent& event );
void OnPopupSort( wxMenuEvent& event ); void OnPopupSort( wxMenuEvent& event );
void OnPopupDel( wxMenuEvent& event ); void OnPopupDel( wxMenuEvent& event );
void OnPopupEna( wxMenuEvent& event ); void OnPopupEna( wxMenuEvent& event );
......
...@@ -31,7 +31,8 @@ ...@@ -31,7 +31,8 @@
* access2_New: * access2_New:
*****************************************************************************/ *****************************************************************************/
access_t *__access2_New( vlc_object_t *p_obj, access_t *__access2_New( vlc_object_t *p_obj,
char *psz_access, char *psz_demux, char *psz_path ) char *psz_access, char *psz_demux, char *psz_path,
vlc_bool_t b_quick )
{ {
access_t *p_access = vlc_object_create( p_obj, VLC_OBJECT_ACCESS ); access_t *p_access = vlc_object_create( p_obj, VLC_OBJECT_ACCESS );
...@@ -42,10 +43,11 @@ access_t *__access2_New( vlc_object_t *p_obj, ...@@ -42,10 +43,11 @@ access_t *__access2_New( vlc_object_t *p_obj,
} }
/* Parse URL */ /* Parse URL */
p_access->psz_access = strdup( psz_access ); p_access->psz_access = b_quick ? strdup( "file" ) : strdup( psz_access );
p_access->psz_path = strdup( psz_path ); p_access->psz_path = strdup( psz_path );
p_access->psz_demux = strdup( "" ); p_access->psz_demux = strdup( "" );
if( !b_quick )
msg_Dbg( p_obj, "access2_New: access='%s' path='%s'", msg_Dbg( p_obj, "access2_New: access='%s' path='%s'",
p_access->psz_access, p_access->psz_path ); p_access->psz_access, p_access->psz_path );
...@@ -66,7 +68,8 @@ access_t *__access2_New( vlc_object_t *p_obj, ...@@ -66,7 +68,8 @@ access_t *__access2_New( vlc_object_t *p_obj,
vlc_object_attach( p_access, p_obj ); vlc_object_attach( p_access, p_obj );
p_access->p_module = p_access->p_module =
module_Need( p_access, "access2", p_access->psz_access, VLC_FALSE ); module_Need( p_access, "access2", p_access->psz_access,
b_quick ? VLC_TRUE : VLC_FALSE );
if( p_access->p_module == NULL ) if( p_access->p_module == NULL )
{ {
......
...@@ -33,7 +33,7 @@ ...@@ -33,7 +33,7 @@
*****************************************************************************/ *****************************************************************************/
demux_t *__demux2_New( vlc_object_t *p_obj, demux_t *__demux2_New( vlc_object_t *p_obj,
char *psz_access, char *psz_demux, char *psz_path, char *psz_access, char *psz_demux, char *psz_path,
stream_t *s, es_out_t *out ) stream_t *s, es_out_t *out, vlc_bool_t b_quick )
{ {
demux_t *p_demux = vlc_object_create( p_obj, VLC_OBJECT_DEMUX ); demux_t *p_demux = vlc_object_create( p_obj, VLC_OBJECT_DEMUX );
char *psz_module; char *psz_module;
...@@ -55,8 +55,11 @@ demux_t *__demux2_New( vlc_object_t *p_obj, ...@@ -55,8 +55,11 @@ demux_t *__demux2_New( vlc_object_t *p_obj,
p_demux->psz_demux = var_GetString( p_obj, "demux" ); p_demux->psz_demux = var_GetString( p_obj, "demux" );
} }
if( !b_quick )
{
msg_Dbg( p_obj, "demux2_New: access='%s' demux='%s' path='%s'", msg_Dbg( p_obj, "demux2_New: access='%s' demux='%s' path='%s'",
p_demux->psz_access, p_demux->psz_demux, p_demux->psz_path ); p_demux->psz_access, p_demux->psz_demux, p_demux->psz_path );
}
p_demux->s = s; p_demux->s = s;
p_demux->out = out; p_demux->out = out;
...@@ -98,10 +101,20 @@ demux_t *__demux2_New( vlc_object_t *p_obj, ...@@ -98,10 +101,20 @@ demux_t *__demux2_New( vlc_object_t *p_obj,
{ "rm", "rm" }, { "rm", "rm" },
{ NULL, NULL }, { NULL, NULL },
}; };
/* Here, we don't mind if it does not work, it must be quick */
static struct { char *ext; char *demux; } exttodemux_quick[] =
{
{ "mp3", "mpga" },
{ "ogg", "ogg" },
{ "wma", "asf" },
{ NULL, NULL }
};
char *psz_ext = strrchr( p_demux->psz_path, '.' ) + 1; char *psz_ext = strrchr( p_demux->psz_path, '.' ) + 1;
int i; int i;
if( !b_quick )
{
for( i = 0; exttodemux[i].ext != NULL; i++ ) for( i = 0; exttodemux[i].ext != NULL; i++ )
{ {
if( !strcasecmp( psz_ext, exttodemux[i].ext ) ) if( !strcasecmp( psz_ext, exttodemux[i].ext ) )
...@@ -111,6 +124,19 @@ demux_t *__demux2_New( vlc_object_t *p_obj, ...@@ -111,6 +124,19 @@ demux_t *__demux2_New( vlc_object_t *p_obj,
} }
} }
} }
else
{
for( i = 0; exttodemux_quick[i].ext != NULL; i++ )
{
if( !strcasecmp( psz_ext, exttodemux_quick[i].ext ) )
{
psz_module = exttodemux_quick[i].demux;
break;
}
}
}
}
/* Before module_Need (for var_Create...) */ /* Before module_Need (for var_Create...) */
vlc_object_attach( p_demux, p_obj ); vlc_object_attach( p_demux, p_obj );
...@@ -486,7 +512,8 @@ static int DStreamThread( stream_t *s ) ...@@ -486,7 +512,8 @@ static int DStreamThread( stream_t *s )
demux_t *p_demux; demux_t *p_demux;
/* Create the demuxer */ /* Create the demuxer */
if( !(p_demux = demux2_New( s, "", p_sys->psz_name, "", s, p_sys->out )) ) if( !(p_demux = demux2_New( s, "", p_sys->psz_name, "", s, p_sys->out,
VLC_FALSE )) )
{ {
return VLC_EGENERIC; return VLC_EGENERIC;
} }
......
...@@ -44,7 +44,7 @@ ...@@ -44,7 +44,7 @@
*****************************************************************************/ *****************************************************************************/
static int Run ( input_thread_t *p_input ); static int Run ( input_thread_t *p_input );
static int Init ( input_thread_t *p_input ); static int Init ( input_thread_t *p_input, vlc_bool_t b_quick );
static void Error( input_thread_t *p_input ); static void Error( input_thread_t *p_input );
static void End ( input_thread_t *p_input ); static void End ( input_thread_t *p_input );
...@@ -55,9 +55,9 @@ static vlc_bool_t Control( input_thread_t *, int, vlc_value_t ); ...@@ -55,9 +55,9 @@ static vlc_bool_t Control( input_thread_t *, int, vlc_value_t );
static int UpdateFromAccess( input_thread_t * ); static int UpdateFromAccess( input_thread_t * );
static int UpdateFromDemux( input_thread_t * ); static int UpdateFromDemux( input_thread_t * );
static int UpdateMeta( input_thread_t * ); static int UpdateMeta( input_thread_t *, vlc_bool_t );
static void UpdateItemLength( input_thread_t *, int64_t i_length ); static void UpdateItemLength( input_thread_t *, int64_t i_length, vlc_bool_t );
static void ParseOption( input_thread_t *p_input, const char *psz_option ); static void ParseOption( input_thread_t *p_input, const char *psz_option );
...@@ -67,7 +67,8 @@ static void MRLSections( input_thread_t *, char *, int *, int *, int *, int *); ...@@ -67,7 +67,8 @@ static void MRLSections( input_thread_t *, char *, int *, int *, int *, int *);
static input_source_t *InputSourceNew( input_thread_t *); static input_source_t *InputSourceNew( input_thread_t *);
static int InputSourceInit( input_thread_t *, input_source_t *, static int InputSourceInit( input_thread_t *, input_source_t *,
char *, char *psz_forced_demux ); char *, char *psz_forced_demux,
vlc_bool_t b_quick );
static void InputSourceClean( input_thread_t *, input_source_t * ); static void InputSourceClean( input_thread_t *, input_source_t * );
static void SlaveDemux( input_thread_t *p_input ); static void SlaveDemux( input_thread_t *p_input );
...@@ -237,6 +238,99 @@ input_thread_t *__input_CreateThread( vlc_object_t *p_parent, ...@@ -237,6 +238,99 @@ input_thread_t *__input_CreateThread( vlc_object_t *p_parent,
return p_input; return p_input;
} }
/*****************************************************************************
* input_PreParse: Lightweight input for playlist item preparsing
*****************************************************************************/
int __input_Preparse( vlc_object_t *p_parent, input_item_t *p_item )
{
input_thread_t *p_input; /* thread descriptor */
int i;
/* Allocate descriptor */
p_input = vlc_object_create( p_parent, VLC_OBJECT_INPUT );
if( p_input == NULL )
{
msg_Err( p_parent, "out of memory" );
return VLC_EGENERIC;
}
/* Init Common fields */
p_input->b_eof = VLC_FALSE;
p_input->b_can_pace_control = VLC_TRUE;
p_input->i_start = 0;
p_input->i_time = 0;
p_input->i_stop = 0;
p_input->i_title = 0;
p_input->title = NULL;
p_input->i_title_offset = p_input->i_seekpoint_offset = 0;
p_input->i_state = INIT_S;
p_input->i_rate = INPUT_RATE_DEFAULT;
p_input->i_bookmark = 0;
p_input->bookmark = NULL;
p_input->p_meta = NULL;
p_input->p_es_out = NULL;
p_input->p_sout = NULL;
p_input->b_out_pace_control = VLC_FALSE;
p_input->i_pts_delay = 0;
/* Init Input fields */
p_input->input.p_item = p_item;
p_input->input.p_access = NULL;
p_input->input.p_stream = NULL;
p_input->input.p_demux = NULL;
p_input->input.b_title_demux = VLC_FALSE;
p_input->input.i_title = 0;
p_input->input.title = NULL;
p_input->input.i_title_offset = p_input->input.i_seekpoint_offset = 0;
p_input->input.b_can_pace_control = VLC_TRUE;
p_input->input.b_eof = VLC_FALSE;
p_input->input.i_cr_average = 0;
/* No slave */
p_input->i_slave = 0;
p_input->slave = NULL;
/* Init control buffer */
vlc_mutex_init( p_input, &p_input->lock_control );
p_input->i_control = 0;
/* Parse input options */
vlc_mutex_lock( &p_item->lock );
for( i = 0; i < p_item->i_options; i++ )
{
ParseOption( p_input, p_item->ppsz_options[i] );
}
vlc_mutex_unlock( &p_item->lock );
/* Create Object Variables for private use only */
input_ConfigVarInit( p_input );
p_input->input.i_cr_average = var_GetInteger( p_input, "cr-average" );
/* Now we can attach our new input */
vlc_object_attach( p_input, p_parent );
Init( p_input, VLC_TRUE );
/* Clean up master */
InputSourceClean( p_input, &p_input->input );
/* Kill access and demux */
if( p_input->input.p_access ) p_input->input.p_access->b_die = VLC_TRUE;
if( p_input->input.p_demux ) p_input->input.p_access->b_die = VLC_TRUE;
/* Unload all modules */
if( p_input->p_es_out ) input_EsOutDelete( p_input->p_es_out );
/* Delete meta */
if( p_input->p_meta ) vlc_meta_Delete( p_input->p_meta );
vlc_object_detach( p_input );
vlc_object_destroy( p_input );
return VLC_SUCCESS;
}
/***************************************************************************** /*****************************************************************************
* input_StopThread: mark an input thread as zombie * input_StopThread: mark an input thread as zombie
***************************************************************************** *****************************************************************************
...@@ -313,7 +407,7 @@ static int Run( input_thread_t *p_input ) ...@@ -313,7 +407,7 @@ static int Run( input_thread_t *p_input )
/* Signal that the thread is launched */ /* Signal that the thread is launched */
vlc_thread_ready( p_input ); vlc_thread_ready( p_input );
if( Init( p_input ) ) if( Init( p_input, VLC_FALSE ) )
{ {
/* If we failed, wait before we are killed, and exit */ /* If we failed, wait before we are killed, and exit */
p_input->b_error = VLC_TRUE; p_input->b_error = VLC_TRUE;
...@@ -467,7 +561,7 @@ static int Run( input_thread_t *p_input ) ...@@ -467,7 +561,7 @@ static int Run( input_thread_t *p_input )
if( old_val.i_time != val.i_time ) if( old_val.i_time != val.i_time )
{ {
UpdateItemLength( p_input, i_length ); UpdateItemLength( p_input, i_length, VLC_TRUE );
} }
} }
...@@ -505,10 +599,8 @@ static int Run( input_thread_t *p_input ) ...@@ -505,10 +599,8 @@ static int Run( input_thread_t *p_input )
return 0; return 0;
} }
/*****************************************************************************
* Init: init the input Thread static int Init( input_thread_t * p_input, vlc_bool_t b_quick )
*****************************************************************************/
static int Init( input_thread_t * p_input )
{ {
char *psz; char *psz;
char *psz_subtitle; char *psz_subtitle;
...@@ -519,18 +611,22 @@ static int Init( input_thread_t * p_input ) ...@@ -519,18 +611,22 @@ static int Init( input_thread_t * p_input )
int i, i_delay; int i, i_delay;
/* Initialize optional stream output. (before access/demuxer) */ /* Initialize optional stream output. (before access/demuxer) */
if( !b_quick )
{
psz = var_GetString( p_input, "sout" ); psz = var_GetString( p_input, "sout" );
if( *psz ) if( *psz )
{ {
p_input->p_sout = sout_NewInstance( p_input, psz ); p_input->p_sout = sout_NewInstance( p_input, psz );
if( p_input->p_sout == NULL ) if( p_input->p_sout == NULL )
{ {
msg_Err( p_input, "cannot start stream output instance, aborting" ); msg_Err( p_input, "cannot start stream output instance," \
"aborting" );
free( psz ); free( psz );
return VLC_EGENERIC; return VLC_EGENERIC;
} }
} }
free( psz ); free( psz );
}
/* Create es out */ /* Create es out */
p_input->p_es_out = input_EsOutNew( p_input ); p_input->p_es_out = input_EsOutNew( p_input );
...@@ -538,12 +634,14 @@ static int Init( input_thread_t * p_input ) ...@@ -538,12 +634,14 @@ static int Init( input_thread_t * p_input )
es_out_Control( p_input->p_es_out, ES_OUT_SET_MODE, ES_OUT_MODE_NONE ); es_out_Control( p_input->p_es_out, ES_OUT_SET_MODE, ES_OUT_MODE_NONE );
if( InputSourceInit( p_input, &p_input->input, if( InputSourceInit( p_input, &p_input->input,
p_input->input.p_item->psz_uri, NULL ) ) p_input->input.p_item->psz_uri, NULL, b_quick ) )
{ {
goto error; goto error;
} }
/* Create global title (from master) */ /* Create global title (from master) */
if( !b_quick )
{
p_input->i_title = p_input->input.i_title; p_input->i_title = p_input->input.i_title;
p_input->title = p_input->input.title; p_input->title = p_input->input.title;
p_input->i_title_offset = p_input->input.i_title_offset; p_input->i_title_offset = p_input->input.i_title_offset;
...@@ -572,6 +670,7 @@ static int Init( input_thread_t * p_input ) ...@@ -572,6 +670,7 @@ static int Init( input_thread_t * p_input )
p_input->input.i_cr_average *= (10 * p_input->i_pts_delay / 200000); p_input->input.i_cr_average *= (10 * p_input->i_pts_delay / 200000);
p_input->input.i_cr_average /= 10; p_input->input.i_cr_average /= 10;
if( p_input->input.i_cr_average <= 0 ) p_input->input.i_cr_average = 1; if( p_input->input.i_cr_average <= 0 ) p_input->input.i_cr_average = 1;
}
/* Load master infos */ /* Load master infos */
/* Init length */ /* Init length */
...@@ -579,12 +678,13 @@ static int Init( input_thread_t * p_input ) ...@@ -579,12 +678,13 @@ static int Init( input_thread_t * p_input )
&val.i_time ) && val.i_time > 0 ) &val.i_time ) && val.i_time > 0 )
{ {
var_Change( p_input, "length", VLC_VAR_SETVALUE, &val, NULL ); var_Change( p_input, "length", VLC_VAR_SETVALUE, &val, NULL );
UpdateItemLength( p_input, val.i_time, b_quick );
UpdateItemLength( p_input, val.i_time );
p_input->input.p_item->i_duration = val.i_time; p_input->input.p_item->i_duration = val.i_time;
} }
/* Start title/chapter */ /* Start title/chapter */
if( !b_quick )
{
val.i_int = p_input->input.i_title_start - val.i_int = p_input->input.i_title_start -
p_input->input.i_title_offset; p_input->input.i_title_offset;
if( val.i_int > 0 && val.i_int < p_input->input.i_title ) if( val.i_int > 0 && val.i_int < p_input->input.i_title )
...@@ -630,7 +730,6 @@ static int Init( input_thread_t * p_input ) ...@@ -630,7 +730,6 @@ static int Init( input_thread_t * p_input )
if( !demux2_Control( p_input->input.p_demux, DEMUX_GET_FPS, &f_fps ) && if( !demux2_Control( p_input->input.p_demux, DEMUX_GET_FPS, &f_fps ) &&
f_fps > 1.0 ) f_fps > 1.0 )
{ {
vlc_value_t fps;
float f_requested_fps; float f_requested_fps;
var_Create( p_input, "sub-original-fps", VLC_VAR_FLOAT ); var_Create( p_input, "sub-original-fps", VLC_VAR_FLOAT );
...@@ -639,13 +738,13 @@ static int Init( input_thread_t * p_input ) ...@@ -639,13 +738,13 @@ static int Init( input_thread_t * p_input )
f_requested_fps = var_CreateGetFloat( p_input, "sub-fps" ); f_requested_fps = var_CreateGetFloat( p_input, "sub-fps" );
if( f_requested_fps != f_fps ) if( f_requested_fps != f_fps )
{ {
var_Create( p_input, "sub-fps", VLC_VAR_FLOAT| VLC_VAR_DOINHERIT ); var_Create( p_input, "sub-fps", VLC_VAR_FLOAT|
VLC_VAR_DOINHERIT );
var_SetFloat( p_input, "sub-fps", f_requested_fps ); var_SetFloat( p_input, "sub-fps", f_requested_fps );
} }
} }
i_delay = var_CreateGetInteger( p_input, "sub-delay" ); i_delay = var_CreateGetInteger( p_input, "sub-delay" );
if( i_delay != 0 ) if( i_delay != 0 )
{ {
var_SetTime( p_input, "spu-delay", (mtime_t)i_delay * 100000 ); var_SetTime( p_input, "spu-delay", (mtime_t)i_delay * 100000 );
...@@ -666,22 +765,26 @@ static int Init( input_thread_t * p_input ) ...@@ -666,22 +765,26 @@ static int Init( input_thread_t * p_input )
/* */ /* */
sub = InputSourceNew( p_input ); sub = InputSourceNew( p_input );
if( !InputSourceInit( p_input, sub, psz_subtitle, "subtitle" ) ) if( !InputSourceInit( p_input, sub, psz_subtitle, "subtitle",
VLC_FALSE ) )
{ {
TAB_APPEND( p_input->i_slave, p_input->slave, sub ); TAB_APPEND( p_input->i_slave, p_input->slave, sub );
/* Select the ES */ /* Select the ES */
if( !var_Change( p_input, "spu-es", VLC_VAR_GETLIST, &list, NULL ) ) if( !var_Change( p_input, "spu-es", VLC_VAR_GETLIST, &list,
NULL ) )
{ {
if( count.i_int == 0 ) if( count.i_int == 0 )
count.i_int++; /* if it was first one, there is disable too */ count.i_int++;
/* if it was first one, there is disable too */
if( count.i_int < list.p_list->i_count ) if( count.i_int < list.p_list->i_count )
{ {
input_ControlPush( p_input, INPUT_CONTROL_SET_ES, input_ControlPush( p_input, INPUT_CONTROL_SET_ES,
&list.p_list->p_values[count.i_int] ); &list.p_list->p_values[count.i_int] );
} }
var_Change( p_input, "spu-es", VLC_VAR_FREELIST, &list, NULL ); var_Change( p_input, "spu-es", VLC_VAR_FREELIST, &list,
NULL );
} }
} }
} }
...@@ -699,7 +802,8 @@ static int Init( input_thread_t * p_input ) ...@@ -699,7 +802,8 @@ static int Init( input_thread_t * p_input )
if( strcmp( psz_subtitle, subs[i] ) ) if( strcmp( psz_subtitle, subs[i] ) )
{ {
sub = InputSourceNew( p_input ); sub = InputSourceNew( p_input );
if( !InputSourceInit( p_input, sub, subs[i], "subtitle" ) ) if( !InputSourceInit( p_input, sub, subs[i], "subtitle",
VLC_FALSE ) )
{ {
TAB_APPEND( p_input->i_slave, p_input->slave, sub ); TAB_APPEND( p_input->i_slave, p_input->slave, sub );
} }
...@@ -734,7 +838,7 @@ static int Init( input_thread_t * p_input ) ...@@ -734,7 +838,7 @@ static int Init( input_thread_t * p_input )
msg_Dbg( p_input, "adding slave '%s'", psz ); msg_Dbg( p_input, "adding slave '%s'", psz );
slave = InputSourceNew( p_input ); slave = InputSourceNew( p_input );
if( !InputSourceInit( p_input, slave, psz, NULL ) ) if( !InputSourceInit( p_input, slave, psz, NULL, VLC_FALSE ) )
{ {
TAB_APPEND( p_input->i_slave, p_input->slave, slave ); TAB_APPEND( p_input->i_slave, p_input->slave, slave );
} }
...@@ -742,8 +846,16 @@ static int Init( input_thread_t * p_input ) ...@@ -742,8 +846,16 @@ static int Init( input_thread_t * p_input )
} }
} }
if( psz ) free( psz ); if( psz ) free( psz );
}
else
{
p_input->i_start = 0;
p_input->i_start = 0;
}
/* Set up es_out */ /* Set up es_out */
if( !b_quick )
{
es_out_Control( p_input->p_es_out, ES_OUT_SET_ACTIVE, VLC_TRUE ); es_out_Control( p_input->p_es_out, ES_OUT_SET_ACTIVE, VLC_TRUE );
i_es_out_mode = ES_OUT_MODE_AUTO; i_es_out_mode = ES_OUT_MODE_AUTO;
val.p_list = NULL; val.p_list = NULL;
...@@ -764,7 +876,8 @@ static int Init( input_thread_t * p_input ) ...@@ -764,7 +876,8 @@ static int Init( input_thread_t * p_input )
/* Note : we should remove the "program" callback. */ /* Note : we should remove the "program" callback. */
} }
else else
var_Change( p_input, "programs", VLC_VAR_FREELIST, &val, NULL ); var_Change( p_input, "programs", VLC_VAR_FREELIST, &val,
NULL );
} }
} }
es_out_Control( p_input->p_es_out, ES_OUT_SET_MODE, i_es_out_mode ); es_out_Control( p_input->p_es_out, ES_OUT_SET_MODE, i_es_out_mode );
...@@ -798,6 +911,7 @@ static int Init( input_thread_t * p_input ) ...@@ -798,6 +911,7 @@ static int Init( input_thread_t * p_input )
msg_Dbg( p_input, "starting in %s mode", msg_Dbg( p_input, "starting in %s mode",
p_input->b_out_pace_control ? "asynch" : "synch" ); p_input->b_out_pace_control ? "asynch" : "synch" );
} }
}
/* Get meta data from users */ /* Get meta data from users */
p_meta_tmp = InputMetaUser( p_input ); p_meta_tmp = InputMetaUser( p_input );
...@@ -817,8 +931,12 @@ static int Init( input_thread_t * p_input ) ...@@ -817,8 +931,12 @@ static int Init( input_thread_t * p_input )
vlc_meta_Delete( p_meta_tmp ); vlc_meta_Delete( p_meta_tmp );
} }
/* Access_file does not give any meta, and there are no slave */
if( !b_quick )
{
if( !p_input->input.p_access || if( !p_input->input.p_access ||
access2_Control( p_input->input.p_access, ACCESS_GET_META, &p_meta_tmp)) access2_Control( p_input->input.p_access, ACCESS_GET_META,
&p_meta_tmp))
p_meta_tmp = NULL; p_meta_tmp = NULL;
if( p_meta == NULL ) if( p_meta == NULL )
...@@ -864,15 +982,19 @@ static int Init( input_thread_t * p_input ) ...@@ -864,15 +982,19 @@ static int Init( input_thread_t * p_input )
vlc_meta_Delete( p_meta_slave ); vlc_meta_Delete( p_meta_slave );
} }
} }
}
} }
p_input->p_meta = p_meta; p_input->p_meta = p_meta;
UpdateMeta( p_input ); UpdateMeta( p_input, b_quick );
if( !b_quick )
{
msg_Dbg( p_input, "`%s' sucessfully opened", msg_Dbg( p_input, "`%s' sucessfully opened",
p_input->input.p_item->psz_uri ); p_input->input.p_item->psz_uri );
}
/* Trigger intf update for this item */ /* Trigger intf update for this item */
/* Playlist has a callback on this variable and will forward /* Playlist has a callback on this variable and will forward
* it to intf */ * it to intf */
...@@ -1407,7 +1529,8 @@ static vlc_bool_t Control( input_thread_t *p_input, int i_type, ...@@ -1407,7 +1529,8 @@ static vlc_bool_t Control( input_thread_t *p_input, int i_type,
{ {
input_source_t *slave = InputSourceNew( p_input ); input_source_t *slave = InputSourceNew( p_input );
if( !InputSourceInit( p_input, slave, val.psz_string, NULL ) ) if( !InputSourceInit( p_input, slave, val.psz_string, NULL,
VLC_FALSE ) )
{ {
vlc_meta_t *p_meta_new = NULL; vlc_meta_t *p_meta_new = NULL;
vlc_meta_t *p_meta; vlc_meta_t *p_meta;
...@@ -1465,7 +1588,7 @@ static vlc_bool_t Control( input_thread_t *p_input, int i_type, ...@@ -1465,7 +1588,7 @@ static vlc_bool_t Control( input_thread_t *p_input, int i_type,
{ {
p_input->p_meta = p_meta_new; p_input->p_meta = p_meta_new;
} }
UpdateMeta( p_input ); UpdateMeta( p_input, VLC_FALSE );
} }
TAB_APPEND( p_input->i_slave, p_input->slave, slave ); TAB_APPEND( p_input->i_slave, p_input->slave, slave );
...@@ -1584,7 +1707,7 @@ static int UpdateFromAccess( input_thread_t *p_input ) ...@@ -1584,7 +1707,7 @@ static int UpdateFromAccess( input_thread_t *p_input )
p_input->p_meta = p_meta; p_input->p_meta = p_meta;
} }
UpdateMeta( p_input ); UpdateMeta( p_input, VLC_FALSE );
var_SetBool( p_input, "item-change", p_input->input.p_item->i_id ); var_SetBool( p_input, "item-change", p_input->input.p_item->i_id );
} }
p_access->info.i_update &= ~INPUT_UPDATE_META; p_access->info.i_update &= ~INPUT_UPDATE_META;
...@@ -1622,7 +1745,7 @@ static int UpdateFromAccess( input_thread_t *p_input ) ...@@ -1622,7 +1745,7 @@ static int UpdateFromAccess( input_thread_t *p_input )
/***************************************************************************** /*****************************************************************************
* UpdateMeta: * UpdateMeta:
*****************************************************************************/ *****************************************************************************/
static int UpdateMeta( input_thread_t *p_input ) static int UpdateMeta( input_thread_t *p_input, vlc_bool_t b_quick )
{ {
vlc_meta_t *p_meta = p_input->p_meta; vlc_meta_t *p_meta = p_input->p_meta;
int i; int i;
...@@ -1630,9 +1753,10 @@ static int UpdateMeta( input_thread_t *p_input ) ...@@ -1630,9 +1753,10 @@ static int UpdateMeta( input_thread_t *p_input )
if( !p_meta || p_meta->i_meta == 0 ) if( !p_meta || p_meta->i_meta == 0 )
return VLC_SUCCESS; return VLC_SUCCESS;
msg_Dbg( p_input, "meta information:" ); if( !b_quick ) msg_Dbg( p_input, "meta information:" );
for( i = 0; i < p_meta->i_meta; i++ ) for( i = 0; i < p_meta->i_meta; i++ )
{ {
if( !b_quick )
msg_Dbg( p_input, " - '%s' = '%s'", msg_Dbg( p_input, " - '%s' = '%s'",
_(p_meta->name[i]), p_meta->value[i] ); _(p_meta->name[i]), p_meta->value[i] );
...@@ -1682,7 +1806,8 @@ static int UpdateMeta( input_thread_t *p_input ) ...@@ -1682,7 +1806,8 @@ static int UpdateMeta( input_thread_t *p_input )
/***************************************************************************** /*****************************************************************************
* UpdateItemLength: * UpdateItemLength:
*****************************************************************************/ *****************************************************************************/
static void UpdateItemLength( input_thread_t *p_input, int64_t i_length ) static void UpdateItemLength( input_thread_t *p_input, int64_t i_length,
vlc_bool_t b_quick )
{ {
playlist_t *p_playlist; playlist_t *p_playlist;
char psz_buffer[MSTRTIME_MAX_SIZE]; char psz_buffer[MSTRTIME_MAX_SIZE];
...@@ -1690,11 +1815,15 @@ static void UpdateItemLength( input_thread_t *p_input, int64_t i_length ) ...@@ -1690,11 +1815,15 @@ static void UpdateItemLength( input_thread_t *p_input, int64_t i_length )
vlc_mutex_lock( &p_input->input.p_item->lock ); vlc_mutex_lock( &p_input->input.p_item->lock );
p_input->input.p_item->i_duration = i_length; p_input->input.p_item->i_duration = i_length;
vlc_mutex_unlock( &p_input->input.p_item->lock ); vlc_mutex_unlock( &p_input->input.p_item->lock );
p_playlist = vlc_object_find( p_input, VLC_OBJECT_PLAYLIST, FIND_PARENT);
p_playlist = vlc_object_find( p_input, VLC_OBJECT_PLAYLIST,
FIND_PARENT);
if( p_playlist )
{
var_SetInteger( p_playlist, "item-change", var_SetInteger( p_playlist, "item-change",
p_input->input.p_item->i_id ); p_input->input.p_item->i_id );
vlc_object_release( p_playlist ); vlc_object_release( p_playlist );
}
input_Control( p_input, INPUT_ADD_INFO, _("General"), _("Duration"), input_Control( p_input, INPUT_ADD_INFO, _("General"), _("Duration"),
msecstotimestr( psz_buffer, i_length / 1000 ) ); msecstotimestr( psz_buffer, i_length / 1000 ) );
...@@ -1726,7 +1855,7 @@ static input_source_t *InputSourceNew( input_thread_t *p_input ) ...@@ -1726,7 +1855,7 @@ static input_source_t *InputSourceNew( input_thread_t *p_input )
*****************************************************************************/ *****************************************************************************/
static int InputSourceInit( input_thread_t *p_input, static int InputSourceInit( input_thread_t *p_input,
input_source_t *in, char *psz_mrl, input_source_t *in, char *psz_mrl,
char *psz_forced_demux ) char *psz_forced_demux, vlc_bool_t b_quick )
{ {
char *psz_dup = strdup( psz_mrl ); char *psz_dup = strdup( psz_mrl );
char *psz_access; char *psz_access;
...@@ -1735,6 +1864,8 @@ static int InputSourceInit( input_thread_t *p_input, ...@@ -1735,6 +1864,8 @@ static int InputSourceInit( input_thread_t *p_input,
vlc_value_t val; vlc_value_t val;
/* Split uri */ /* Split uri */
if( !b_quick )
{
MRLSplit( p_input, psz_dup, &psz_access, &psz_demux, &psz_path ); MRLSplit( p_input, psz_dup, &psz_access, &psz_demux, &psz_path );
msg_Dbg( p_input, "`%s' gives access `%s' demux `%s' path `%s'", msg_Dbg( p_input, "`%s' gives access `%s' demux `%s' path `%s'",
...@@ -1742,7 +1873,8 @@ static int InputSourceInit( input_thread_t *p_input, ...@@ -1742,7 +1873,8 @@ static int InputSourceInit( input_thread_t *p_input,
/* Hack to allow udp://@:port syntax */ /* Hack to allow udp://@:port syntax */
if( !psz_access || if( !psz_access ||
(strncmp( psz_access, "udp", 3 ) && strncmp( psz_access, "rtp", 3 )) ) (strncmp( psz_access, "udp", 3 ) &&
strncmp( psz_access, "rtp", 3 )) )
/* Find optional titles and seekpoints */ /* Find optional titles and seekpoints */
MRLSections( p_input, psz_path, &in->i_title_start, &in->i_title_end, MRLSections( p_input, psz_path, &in->i_title_start, &in->i_title_end,
...@@ -1755,7 +1887,15 @@ static int InputSourceInit( input_thread_t *p_input, ...@@ -1755,7 +1887,15 @@ static int InputSourceInit( input_thread_t *p_input,
if( *psz_demux == '\0' ) if( *psz_demux == '\0' )
{ {
in->p_demux = demux2_New( p_input, psz_access, psz_demux, psz_path, in->p_demux = demux2_New( p_input, psz_access, psz_demux, psz_path,
NULL, p_input->p_es_out ); NULL, p_input->p_es_out, VLC_FALSE );
}
}
else
{
psz_path = psz_mrl;
msg_Dbg( p_input, "trying to preparse %s", psz_path );
psz_demux = strdup( "" );
psz_access = strdup( "file" );
} }
if( in->p_demux ) if( in->p_demux )
...@@ -1790,18 +1930,20 @@ static int InputSourceInit( input_thread_t *p_input, ...@@ -1790,18 +1930,20 @@ static int InputSourceInit( input_thread_t *p_input,
int64_t i_pts_delay; int64_t i_pts_delay;
/* Now try a real access */ /* Now try a real access */
in->p_access = access2_New( p_input, psz_access, psz_demux, psz_path ); in->p_access = access2_New( p_input, psz_access, psz_demux, psz_path,
b_quick );
/* Access failed, URL encoded ? */ /* Access failed, URL encoded ? */
if( in->p_access == NULL && strchr( psz_path, '%' ) ) if( in->p_access == NULL && strchr( psz_path, '%' ) )
{ {
DecodeUrl( psz_path ); DecodeUrl( psz_path );
msg_Dbg( p_input, "retying with access `%s' demux `%s' path `%s'", msg_Dbg( p_input, "retrying with access `%s' demux `%s' path `%s'",
psz_access, psz_demux, psz_path ); psz_access, psz_demux, psz_path );
in->p_access = access2_New( p_input, in->p_access = access2_New( p_input,
psz_access, psz_demux, psz_path ); psz_access, psz_demux, psz_path,
b_quick );
} }
#ifndef WIN32 /* Remove this gross hack from the win32 build as colons #ifndef WIN32 /* Remove this gross hack from the win32 build as colons
* are forbidden in filenames on Win32. */ * are forbidden in filenames on Win32. */
...@@ -1817,7 +1959,8 @@ static int InputSourceInit( input_thread_t *p_input, ...@@ -1817,7 +1959,8 @@ static int InputSourceInit( input_thread_t *p_input,
psz_path = psz_dup; psz_path = psz_dup;
in->p_access = access2_New( p_input, in->p_access = access2_New( p_input,
psz_access, psz_demux, psz_path ); psz_access, psz_demux, psz_path,
b_quick );
} }
#endif #endif
...@@ -1828,6 +1971,8 @@ static int InputSourceInit( input_thread_t *p_input, ...@@ -1828,6 +1971,8 @@ static int InputSourceInit( input_thread_t *p_input,
} }
/* Get infos from access */ /* Get infos from access */
if( !b_quick )
{
access2_Control( in->p_access, access2_Control( in->p_access,
ACCESS_GET_PTS_DELAY, &i_pts_delay ); ACCESS_GET_PTS_DELAY, &i_pts_delay );
p_input->i_pts_delay = __MAX( p_input->i_pts_delay, i_pts_delay ); p_input->i_pts_delay = __MAX( p_input->i_pts_delay, i_pts_delay );
...@@ -1848,9 +1993,10 @@ static int InputSourceInit( input_thread_t *p_input, ...@@ -1848,9 +1993,10 @@ static int InputSourceInit( input_thread_t *p_input,
access2_Control( in->p_access, ACCESS_CAN_SEEK, access2_Control( in->p_access, ACCESS_CAN_SEEK,
&val.b_bool ); &val.b_bool );
var_Set( p_input, "seekable", val ); var_Set( p_input, "seekable", val );
}
/* Create the stream_t */ /* Create the stream_t */
in->p_stream = stream_AccessNew( in->p_access ); in->p_stream = stream_AccessNew( in->p_access, b_quick );
if( in->p_stream == NULL ) if( in->p_stream == NULL )
{ {
msg_Warn( p_input, "cannot create a stream_t from access" ); msg_Warn( p_input, "cannot create a stream_t from access" );
...@@ -1863,7 +2009,7 @@ static int InputSourceInit( input_thread_t *p_input, ...@@ -1863,7 +2009,7 @@ static int InputSourceInit( input_thread_t *p_input,
psz_demux = in->p_access->psz_demux; psz_demux = in->p_access->psz_demux;
} }
in->p_demux = demux2_New( p_input, psz_access, psz_demux, psz_path, in->p_demux = demux2_New( p_input, psz_access, psz_demux, psz_path,
in->p_stream, p_input->p_es_out ); in->p_stream, p_input->p_es_out, b_quick );
if( in->p_demux == NULL ) if( in->p_demux == NULL )
{ {
msg_Err( p_input, "no suitable demux module for `%s/%s://%s'", msg_Err( p_input, "no suitable demux module for `%s/%s://%s'",
...@@ -1872,7 +2018,7 @@ static int InputSourceInit( input_thread_t *p_input, ...@@ -1872,7 +2018,7 @@ static int InputSourceInit( input_thread_t *p_input,
} }
/* TODO get title from demux */ /* TODO get title from demux */
if( in->i_title <= 0 ) if( !b_quick && in->i_title <= 0 )
{ {
if( demux2_Control( in->p_demux, DEMUX_GET_TITLE_INFO, if( demux2_Control( in->p_demux, DEMUX_GET_TITLE_INFO,
&in->title, &in->i_title, &in->title, &in->i_title,
......
...@@ -102,7 +102,7 @@ void input_ControlVarTitle( input_thread_t *, int i_title ); ...@@ -102,7 +102,7 @@ void input_ControlVarTitle( input_thread_t *, int i_title );
void input_ConfigVarInit ( input_thread_t * ); void input_ConfigVarInit ( input_thread_t * );
/* stream.c */ /* stream.c */
stream_t *stream_AccessNew( access_t *p_access ); stream_t *stream_AccessNew( access_t *p_access, vlc_bool_t );
void stream_AccessDelete( stream_t *s ); void stream_AccessDelete( stream_t *s );
void stream_AccessReset( stream_t *s ); void stream_AccessReset( stream_t *s );
void stream_AccessUpdate( stream_t *s ); void stream_AccessUpdate( stream_t *s );
......
...@@ -136,6 +136,9 @@ struct stream_sys_t ...@@ -136,6 +136,9 @@ struct stream_sys_t
int i_seek_count; int i_seek_count;
int64_t i_seek_time; int64_t i_seek_time;
} stat; } stat;
/* Preparse mode ? */
vlc_bool_t b_quick;
}; };
/* Method 1: */ /* Method 1: */
...@@ -157,7 +160,7 @@ static int AStreamControl( stream_t *, int i_query, va_list ); ...@@ -157,7 +160,7 @@ static int AStreamControl( stream_t *, int i_query, va_list );
/**************************************************************************** /****************************************************************************
* stream_AccessNew: create a stream from a access * stream_AccessNew: create a stream from a access
****************************************************************************/ ****************************************************************************/
stream_t *stream_AccessNew( access_t *p_access ) stream_t *stream_AccessNew( access_t *p_access, vlc_bool_t b_quick )
{ {
stream_t *s = vlc_object_create( p_access, VLC_OBJECT_STREAM ); stream_t *s = vlc_object_create( p_access, VLC_OBJECT_STREAM );
stream_sys_t *p_sys; stream_sys_t *p_sys;
...@@ -188,6 +191,8 @@ stream_t *stream_AccessNew( access_t *p_access ) ...@@ -188,6 +191,8 @@ stream_t *stream_AccessNew( access_t *p_access )
p_sys->stat.i_seek_count = 0; p_sys->stat.i_seek_count = 0;
p_sys->stat.i_seek_time = 0; p_sys->stat.i_seek_time = 0;
p_sys->b_quick = b_quick;
/* Peek */ /* Peek */
p_sys->i_peek = 0; p_sys->i_peek = 0;
p_sys->p_peek = NULL; p_sys->p_peek = NULL;
...@@ -1129,8 +1134,9 @@ static void AStreamPrebufferStream( stream_t *s ) ...@@ -1129,8 +1134,9 @@ static void AStreamPrebufferStream( stream_t *s )
int64_t i_first = 0; int64_t i_first = 0;
int64_t i_start; int64_t i_start;
int64_t i_prebuffer = (s->p_sys->p_access->info.i_title > 1 || int64_t i_prebuffer = p_sys->b_quick ? STREAM_CACHE_TRACK_SIZE /100 :
s->p_sys->p_access->info.i_seekpoint > 1) ? STREAM_CACHE_PREBUFFER_SIZE : STREAM_CACHE_TRACK_SIZE / 3; ((s->p_sys->p_access->info.i_title > 1 ||
s->p_sys->p_access->info.i_seekpoint > 1) ? STREAM_CACHE_PREBUFFER_SIZE : STREAM_CACHE_TRACK_SIZE / 3);
msg_Dbg( s, "pre buffering" ); msg_Dbg( s, "pre buffering" );
i_start = mdate(); i_start = mdate();
......
...@@ -43,6 +43,7 @@ ...@@ -43,6 +43,7 @@
* Local prototypes * Local prototypes
*****************************************************************************/ *****************************************************************************/
static void RunThread ( playlist_t * ); static void RunThread ( playlist_t * );
static void RunPreparse( playlist_preparse_t * );
static playlist_item_t * NextItem ( playlist_t * ); static playlist_item_t * NextItem ( playlist_t * );
static int PlayItem ( playlist_t *, playlist_item_t * ); static int PlayItem ( playlist_t *, playlist_item_t * );
...@@ -147,6 +148,31 @@ playlist_t * __playlist_Create ( vlc_object_t *p_parent ) ...@@ -147,6 +148,31 @@ playlist_t * __playlist_Create ( vlc_object_t *p_parent )
return NULL; return NULL;
} }
/* Preparsing stuff */
p_playlist->p_preparse = vlc_object_create( p_playlist,
sizeof( playlist_preparse_t ) );
if( !p_playlist->p_preparse )
{
msg_Err( p_playlist, "unable to create preparser" );
vlc_object_destroy( p_playlist );
return NULL;
}
p_playlist->p_preparse->i_waiting = 0;
p_playlist->p_preparse->pp_waiting = NULL;
vlc_mutex_init( p_playlist->p_preparse,
&p_playlist->p_preparse->object_lock );
vlc_object_attach( p_playlist->p_preparse, p_playlist );
if( vlc_thread_create( p_playlist->p_preparse, "preparser",
RunPreparse, VLC_THREAD_PRIORITY_LOW, VLC_TRUE ) )
{
msg_Err( p_playlist, "cannot spawn preparse thread" );
vlc_object_detach( p_playlist->p_preparse );
vlc_object_destroy( p_playlist->p_preparse );
return NULL;
}
/* The object has been initialized, now attach it */ /* The object has been initialized, now attach it */
vlc_object_attach( p_playlist, p_parent ); vlc_object_attach( p_playlist, p_parent );
...@@ -171,8 +197,13 @@ int playlist_Destroy( playlist_t * p_playlist ) ...@@ -171,8 +197,13 @@ int playlist_Destroy( playlist_t * p_playlist )
p_playlist->pp_sds[i]->psz_module ); p_playlist->pp_sds[i]->psz_module );
} }
vlc_thread_join( p_playlist->p_preparse );
vlc_thread_join( p_playlist ); vlc_thread_join( p_playlist );
vlc_object_detach( p_playlist->p_preparse );
vlc_mutex_destroy( &p_playlist->p_preparse->object_lock );
var_Destroy( p_playlist, "intf-change" ); var_Destroy( p_playlist, "intf-change" );
var_Destroy( p_playlist, "item-change" ); var_Destroy( p_playlist, "item-change" );
var_Destroy( p_playlist, "playlist-current" ); var_Destroy( p_playlist, "playlist-current" );
...@@ -195,6 +226,7 @@ int playlist_Destroy( playlist_t * p_playlist ) ...@@ -195,6 +226,7 @@ int playlist_Destroy( playlist_t * p_playlist )
free( p_view ); free( p_view );
} }
vlc_object_destroy( p_playlist->p_preparse );
vlc_object_destroy( p_playlist ); vlc_object_destroy( p_playlist );
return VLC_SUCCESS; return VLC_SUCCESS;
...@@ -368,6 +400,18 @@ int playlist_vaControl( playlist_t * p_playlist, int i_query, va_list args ) ...@@ -368,6 +400,18 @@ int playlist_vaControl( playlist_t * p_playlist, int i_query, va_list args )
return VLC_SUCCESS; return VLC_SUCCESS;
} }
int playlist_PreparseEnqueue( playlist_t *p_playlist,
input_item_t *p_item )
{
vlc_mutex_lock( &p_playlist->p_preparse->object_lock );
INSERT_ELEM( p_playlist->p_preparse->pp_waiting,
p_playlist->p_preparse->i_waiting,
p_playlist->p_preparse->i_waiting,
p_item );
vlc_mutex_unlock( &p_playlist->p_preparse->object_lock );
return VLC_SUCCESS;
}
/* Destroy remaining objects */ /* Destroy remaining objects */
static mtime_t ObjectGarbageCollector( playlist_t *p_playlist, int i_type, static mtime_t ObjectGarbageCollector( playlist_t *p_playlist, int i_type,
...@@ -641,6 +685,37 @@ static void RunThread ( playlist_t *p_playlist ) ...@@ -641,6 +685,37 @@ static void RunThread ( playlist_t *p_playlist )
} }
} }
/* Queue for items to preparse */
static void RunPreparse ( playlist_preparse_t *p_obj )
{
playlist_t *p_playlist = p_obj->p_parent;
vlc_bool_t b_sleep;
/* Tell above that we're ready */
vlc_thread_ready( p_obj );
while( !p_playlist->b_die )
{
vlc_mutex_lock( &p_obj->object_lock );
if( p_obj->i_waiting > 0 )
{
input_Preparse( p_playlist, p_obj->pp_waiting[0] );
var_SetInteger( p_playlist, "item-change",
p_obj->pp_waiting[0]->i_id );
REMOVE_ELEM( p_obj->pp_waiting, p_obj->i_waiting, 0 );
}
b_sleep = ( p_obj->i_waiting == 0 );
vlc_mutex_unlock( &p_obj->object_lock );
if( p_obj->i_waiting == 0 )
{
msleep( INTF_IDLE_SLEEP );
}
}
}
/***************************************************************************** /*****************************************************************************
* NextItem * NextItem
***************************************************************************** *****************************************************************************
...@@ -840,6 +915,10 @@ static playlist_item_t * NextItem( playlist_t *p_playlist ) ...@@ -840,6 +915,10 @@ static playlist_item_t * NextItem( playlist_t *p_playlist )
playlist_view_t *p_view = playlist_view_t *p_view =
playlist_ViewFind( p_playlist, playlist_ViewFind( p_playlist,
p_playlist->status.i_view ); p_playlist->status.i_view );
fprintf(stderr,"Finding next of %s within %s, view %i\n",
p_playlist->status.p_item ? p_playlist->status.p_item->input.psz_name : "coincoin",
p_playlist->status.p_node->input.psz_name,
p_playlist->status.i_view) ;
p_new = playlist_FindNextFromParent( p_playlist, p_new = playlist_FindNextFromParent( p_playlist,
p_playlist->status.i_view, p_playlist->status.i_view,
p_view->p_root, p_view->p_root,
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment