Commit 001cae59 authored by Francois Cartegnie's avatar Francois Cartegnie

parser/fetcher: split in 2 pass mode

Changes from a depth first (local then network) per item to
an horizontal parsing. Allows displaying all local files first
and let the thread slowly process all other resources.
parent 44c6d792
...@@ -43,20 +43,31 @@ ...@@ -43,20 +43,31 @@
/***************************************************************************** /*****************************************************************************
* Structures/definitions * Structures/definitions
*****************************************************************************/ *****************************************************************************/
typedef enum
{
PASS1_LOCAL = 0,
PASS2_NETWORK
} fetcher_pass_t;
#define PASS_COUNT 2
typedef struct typedef struct
{ {
char *psz_artist; char *psz_artist;
char *psz_album; char *psz_album;
char *psz_arturl; char *psz_arturl;
bool b_found; bool b_found;
meta_fetcher_scope_t e_scope; /* max scope */
} playlist_album_t; } playlist_album_t;
typedef struct playlist_fetcher_entry_t typedef struct fetcher_entry_t fetcher_entry_t;
struct fetcher_entry_t
{ {
input_item_t *p_item; input_item_t *p_item;
input_item_meta_request_option_t i_options; input_item_meta_request_option_t i_options;
} playlist_fetcher_entry_t; fetcher_entry_t *p_next;
};
struct playlist_fetcher_t struct playlist_fetcher_t
{ {
...@@ -64,7 +75,10 @@ struct playlist_fetcher_t ...@@ -64,7 +75,10 @@ struct playlist_fetcher_t
vlc_mutex_t lock; vlc_mutex_t lock;
vlc_cond_t wait; vlc_cond_t wait;
bool b_live; bool b_live;
DECL_ARRAY(playlist_fetcher_entry_t) waiting;
fetcher_entry_t *p_waiting_head[PASS_COUNT];
fetcher_entry_t *p_waiting_tail[PASS_COUNT];
DECL_ARRAY(playlist_album_t) albums; DECL_ARRAY(playlist_album_t) albums;
meta_fetcher_scope_t e_scope; meta_fetcher_scope_t e_scope;
}; };
...@@ -92,8 +106,10 @@ playlist_fetcher_t *playlist_fetcher_New( vlc_object_t *parent ) ...@@ -92,8 +106,10 @@ playlist_fetcher_t *playlist_fetcher_New( vlc_object_t *parent )
p_fetcher->e_scope = ( b_access ) ? FETCHER_SCOPE_ANY : FETCHER_SCOPE_LOCAL; p_fetcher->e_scope = ( b_access ) ? FETCHER_SCOPE_ANY : FETCHER_SCOPE_LOCAL;
memset( p_fetcher->p_waiting_head, 0, PASS_COUNT * sizeof(fetcher_entry_t *) );
memset( p_fetcher->p_waiting_tail, 0, PASS_COUNT * sizeof(fetcher_entry_t *) );
ARRAY_INIT( p_fetcher->albums ); ARRAY_INIT( p_fetcher->albums );
ARRAY_INIT( p_fetcher->waiting );
return p_fetcher; return p_fetcher;
} }
...@@ -101,13 +117,24 @@ playlist_fetcher_t *playlist_fetcher_New( vlc_object_t *parent ) ...@@ -101,13 +117,24 @@ playlist_fetcher_t *playlist_fetcher_New( vlc_object_t *parent )
void playlist_fetcher_Push( playlist_fetcher_t *p_fetcher, input_item_t *p_item, void playlist_fetcher_Push( playlist_fetcher_t *p_fetcher, input_item_t *p_item,
input_item_meta_request_option_t i_options ) input_item_meta_request_option_t i_options )
{ {
vlc_gc_incref( p_item ); fetcher_entry_t *p_entry = malloc( sizeof(fetcher_entry_t) );
if ( !p_entry ) return;
playlist_fetcher_entry_t entry = { p_item, i_options }; vlc_gc_incref( p_item );
p_entry->p_item = p_item;
p_entry->p_next = NULL;
p_entry->i_options = i_options;
vlc_mutex_lock( &p_fetcher->lock ); vlc_mutex_lock( &p_fetcher->lock );
ARRAY_APPEND( p_fetcher->waiting, entry ); /* Append last */
if ( p_fetcher->p_waiting_head[PASS1_LOCAL] )
p_fetcher->p_waiting_tail[PASS1_LOCAL]->p_next = p_entry;
else
p_fetcher->p_waiting_head[PASS1_LOCAL] = p_entry;
p_fetcher->p_waiting_tail[PASS1_LOCAL] = p_entry;
if( !p_fetcher->b_live ) if( !p_fetcher->b_live )
{ {
assert( p_fetcher->p_waiting_head[PASS1_LOCAL] );
if( vlc_clone_detach( NULL, Thread, p_fetcher, if( vlc_clone_detach( NULL, Thread, p_fetcher,
VLC_THREAD_PRIORITY_LOW ) ) VLC_THREAD_PRIORITY_LOW ) )
msg_Err( p_fetcher->object, msg_Err( p_fetcher->object,
...@@ -120,11 +147,20 @@ void playlist_fetcher_Push( playlist_fetcher_t *p_fetcher, input_item_t *p_item, ...@@ -120,11 +147,20 @@ void playlist_fetcher_Push( playlist_fetcher_t *p_fetcher, input_item_t *p_item,
void playlist_fetcher_Delete( playlist_fetcher_t *p_fetcher ) void playlist_fetcher_Delete( playlist_fetcher_t *p_fetcher )
{ {
fetcher_entry_t *p_next;
vlc_mutex_lock( &p_fetcher->lock ); vlc_mutex_lock( &p_fetcher->lock );
/* Remove any left-over item, the fetcher will exit */ /* Remove any left-over item, the fetcher will exit */
for (int i=0;i<p_fetcher->waiting.i_size; i++) for ( int i_queue=0; i_queue<PASS_COUNT; i_queue++ )
vlc_gc_decref( p_fetcher->waiting.p_elems[i].p_item ); {
ARRAY_RESET( p_fetcher->waiting ); while( p_fetcher->p_waiting_head[i_queue] )
{
p_next = p_fetcher->p_waiting_head[i_queue]->p_next;
vlc_gc_decref( p_fetcher->p_waiting_head[i_queue]->p_item );
free( p_fetcher->p_waiting_head[i_queue] );
p_fetcher->p_waiting_head[i_queue] = p_next;
}
p_fetcher->p_waiting_head[i_queue] = NULL;
}
while( p_fetcher->b_live ) while( p_fetcher->b_live )
vlc_cond_wait( &p_fetcher->wait, &p_fetcher->lock ); vlc_cond_wait( &p_fetcher->wait, &p_fetcher->lock );
...@@ -151,6 +187,7 @@ static int FindArt( playlist_fetcher_t *p_fetcher, input_item_t *p_item ) ...@@ -151,6 +187,7 @@ static int FindArt( playlist_fetcher_t *p_fetcher, input_item_t *p_item )
{ {
int i_ret; int i_ret;
playlist_album_t *p_album = NULL;
char *psz_artist = input_item_GetArtist( p_item ); char *psz_artist = input_item_GetArtist( p_item );
char *psz_album = input_item_GetAlbum( p_item ); char *psz_album = input_item_GetAlbum( p_item );
char *psz_title = input_item_GetTitle( p_item ); char *psz_title = input_item_GetTitle( p_item );
...@@ -183,15 +220,22 @@ static int FindArt( playlist_fetcher_t *p_fetcher, input_item_t *p_item ) ...@@ -183,15 +220,22 @@ static int FindArt( playlist_fetcher_t *p_fetcher, input_item_t *p_item )
playlist_FindArtInCache( p_item ); playlist_FindArtInCache( p_item );
return 0; return 0;
} }
else else if ( album.e_scope >= p_fetcher->e_scope )
{ {
return VLC_EGENERIC; return VLC_EGENERIC;
} }
msg_Dbg( p_fetcher->object,
" will search at higher scope, if possible" );
p_album = &album;
break;
} }
FOREACH_END(); FOREACH_END();
} }
else
{
free( psz_artist ); free( psz_artist );
free( psz_album ); free( psz_album );
}
if ( playlist_FindArtInCacheUsingItemUID( p_item ) != VLC_SUCCESS ) if ( playlist_FindArtInCacheUsingItemUID( p_item ) != VLC_SUCCESS )
playlist_FindArtInCache( p_item ); playlist_FindArtInCache( p_item );
...@@ -260,14 +304,27 @@ static int FindArt( playlist_fetcher_t *p_fetcher, input_item_t *p_item ) ...@@ -260,14 +304,27 @@ static int FindArt( playlist_fetcher_t *p_fetcher, input_item_t *p_item )
/* Record this album */ /* Record this album */
if( psz_artist && psz_album ) if( psz_artist && psz_album )
{
if ( p_album )
{
p_album->e_scope = p_fetcher->e_scope;
free( p_album->psz_arturl );
p_album->psz_arturl = input_item_GetArtURL( p_item );
p_album->b_found = (i_ret == VLC_EGENERIC ? false : true );
free( psz_artist );
free( psz_album );
}
else
{ {
playlist_album_t a; playlist_album_t a;
a.psz_artist = psz_artist; a.psz_artist = psz_artist;
a.psz_album = psz_album; a.psz_album = psz_album;
a.psz_arturl = input_item_GetArtURL( p_item ); a.psz_arturl = input_item_GetArtURL( p_item );
a.b_found = (i_ret == VLC_EGENERIC ? false : true ); a.b_found = (i_ret == VLC_EGENERIC ? false : true );
a.e_scope = p_fetcher->e_scope;
ARRAY_APPEND( p_fetcher->albums, a ); ARRAY_APPEND( p_fetcher->albums, a );
} }
}
else else
{ {
free( psz_artist ); free( psz_artist );
...@@ -371,18 +428,28 @@ static void *Thread( void *p_data ) ...@@ -371,18 +428,28 @@ static void *Thread( void *p_data )
{ {
playlist_fetcher_t *p_fetcher = p_data; playlist_fetcher_t *p_fetcher = p_data;
vlc_object_t *obj = p_fetcher->object; vlc_object_t *obj = p_fetcher->object;
fetcher_pass_t e_pass = PASS1_LOCAL;
for( ;; ) for( ;; )
{ {
input_item_t *p_item = NULL; fetcher_entry_t *p_entry = NULL;
input_item_meta_request_option_t i_options;
vlc_mutex_lock( &p_fetcher->lock ); vlc_mutex_lock( &p_fetcher->lock );
if( p_fetcher->waiting.i_size != 0 ) for ( int i=0; i<PASS_COUNT; i++ )
{ {
p_item = p_fetcher->waiting.p_elems[0].p_item; if ( p_fetcher->p_waiting_head[i] )
i_options = p_fetcher->waiting.p_elems[0].i_options; {
ARRAY_REMOVE( p_fetcher->waiting, 0 ); e_pass = i;
break;
}
}
if( p_fetcher->p_waiting_head[e_pass] )
{
p_entry = p_fetcher->p_waiting_head[e_pass];
p_fetcher->p_waiting_head[e_pass] = p_entry->p_next;
if ( p_entry->p_next == NULL )
p_fetcher->p_waiting_tail[e_pass] = NULL;
p_entry->p_next = NULL;
} }
else else
{ {
...@@ -391,20 +458,20 @@ static void *Thread( void *p_data ) ...@@ -391,20 +458,20 @@ static void *Thread( void *p_data )
} }
vlc_mutex_unlock( &p_fetcher->lock ); vlc_mutex_unlock( &p_fetcher->lock );
if( !p_item ) if( !p_entry )
break; break;
meta_fetcher_scope_t e_prev_scope = p_fetcher->e_scope; meta_fetcher_scope_t e_prev_scope = p_fetcher->e_scope;
/* scope override */ /* scope override */
switch ( i_options ) { switch ( p_entry->i_options ) {
case META_REQUEST_OPTION_ANY: case META_REQUEST_OPTION_SCOPE_ANY:
p_fetcher->e_scope = FETCHER_SCOPE_ANY; p_fetcher->e_scope = FETCHER_SCOPE_ANY;
break; break;
case META_REQUEST_OPTION_LOCAL: case META_REQUEST_OPTION_SCOPE_LOCAL:
p_fetcher->e_scope = FETCHER_SCOPE_LOCAL; p_fetcher->e_scope = FETCHER_SCOPE_LOCAL;
break; break;
case META_REQUEST_OPTION_NETWORK: case META_REQUEST_OPTION_SCOPE_NETWORK:
p_fetcher->e_scope = FETCHER_SCOPE_NETWORK; p_fetcher->e_scope = FETCHER_SCOPE_NETWORK;
break; break;
case META_REQUEST_OPTION_NONE: case META_REQUEST_OPTION_NONE:
...@@ -415,29 +482,70 @@ static void *Thread( void *p_data ) ...@@ -415,29 +482,70 @@ static void *Thread( void *p_data )
* They are identical to "meta reader" expect that may actually * They are identical to "meta reader" expect that may actually
* takes time. That's why they are running here. * takes time. That's why they are running here.
* The result of this fetch is not cached. */ * The result of this fetch is not cached. */
FetchMeta( p_fetcher, p_item );
/* Find art, and download it if needed */ int i_ret = -1;
int i_ret = FindArt( p_fetcher, p_item );
if( i_ret == 1 ) if( e_pass == PASS1_LOCAL && ( p_fetcher->e_scope & FETCHER_SCOPE_LOCAL ) )
i_ret = DownloadArt( p_fetcher, p_item ); {
/* only fetch from local */
p_fetcher->e_scope = FETCHER_SCOPE_LOCAL;
}
else if( e_pass == PASS2_NETWORK && ( p_fetcher->e_scope & FETCHER_SCOPE_NETWORK ) )
{
/* only fetch from network */
p_fetcher->e_scope = FETCHER_SCOPE_NETWORK;
}
else
p_fetcher->e_scope = 0;
if ( p_fetcher->e_scope & FETCHER_SCOPE_ANY )
{
FetchMeta( p_fetcher, p_entry->p_item );
i_ret = FindArt( p_fetcher, p_entry->p_item );
switch( i_ret )
{
case 1: /* Found, need to dl */
i_ret = DownloadArt( p_fetcher, p_entry->p_item );
break;
case 0: /* Is in cache */
i_ret = VLC_SUCCESS;
//ft
default:// error
break;
}
}
p_fetcher->e_scope = e_prev_scope; p_fetcher->e_scope = e_prev_scope;
/* */ /* */
char *psz_name = input_item_GetName( p_item ); if ( i_ret != VLC_SUCCESS && (e_pass != PASS2_NETWORK) )
if( !i_ret ) /* Art is now in cache */ {
/* Move our entry to next pass queue */
vlc_mutex_lock( &p_fetcher->lock );
if ( p_fetcher->p_waiting_head[e_pass + 1] )
p_fetcher->p_waiting_tail[e_pass + 1]->p_next = p_entry;
else
p_fetcher->p_waiting_head[e_pass + 1] = p_entry;
p_fetcher->p_waiting_tail[e_pass + 1] = p_entry;
vlc_mutex_unlock( &p_fetcher->lock );
}
else
{
/* */
char *psz_name = input_item_GetName( p_entry->p_item );
if( i_ret == VLC_SUCCESS ) /* Art is now in cache */
{ {
msg_Dbg( obj, "found art for %s in cache", psz_name ); msg_Dbg( obj, "found art for %s in cache", psz_name );
input_item_SetArtFetched( p_item, true ); input_item_SetArtFetched( p_entry->p_item, true );
var_SetAddress( obj, "item-change", p_item ); var_SetAddress( obj, "item-change", p_entry->p_item );
} }
else else
{ {
msg_Dbg( obj, "art not found for %s", psz_name ); msg_Dbg( obj, "art not found for %s", psz_name );
input_item_SetArtNotFound( p_item, true ); input_item_SetArtNotFound( p_entry->p_item, true );
} }
free( psz_name ); free( psz_name );
vlc_gc_decref( p_item ); vlc_gc_decref( p_entry->p_item );
free( p_entry );
}
} }
return NULL; return NULL;
} }
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