You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
430 lines
14 KiB
430 lines
14 KiB
/*****************************************************************************
|
|
* fingerprinter.c: Audio fingerprinter module
|
|
*****************************************************************************
|
|
* Copyright (C) 2012 VLC authors and VideoLAN
|
|
*
|
|
* This program is free software; you can redistribute it and/or modify it
|
|
* under the terms of the GNU Lesser General Public License as published by
|
|
* the Free Software Foundation; either version 2.1 of the License, or
|
|
* (at your option) any later version.
|
|
*
|
|
* This program is distributed in the hope that it will be useful,
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
* GNU Lesser General Public License for more details.
|
|
*
|
|
* You should have received a copy of the GNU Lesser General Public License
|
|
* along with this program; if not, write to the Free Software Foundation,
|
|
* Inc., 51 Franklin Street, Fifth Floor, Boston MA 02110-1301, USA.
|
|
*****************************************************************************/
|
|
#ifdef HAVE_CONFIG_H
|
|
# include "config.h"
|
|
#endif
|
|
|
|
#include <assert.h>
|
|
#include <stdatomic.h>
|
|
|
|
#include <vlc_common.h>
|
|
#include <vlc_arrays.h>
|
|
#include <vlc_plugin.h>
|
|
#include <vlc_stream.h>
|
|
#include <vlc_modules.h>
|
|
#include <vlc_meta.h>
|
|
#include <vlc_url.h>
|
|
|
|
#include <vlc_player.h>
|
|
#include <vlc_fingerprinter.h>
|
|
#include "webservices/acoustid.h"
|
|
#include "../stream_out/chromaprint_data.h"
|
|
|
|
/*****************************************************************************
|
|
* Local prototypes
|
|
*****************************************************************************/
|
|
|
|
struct fingerprinter_sys_t
|
|
{
|
|
vlc_thread_t thread;
|
|
vlc_player_t *player;
|
|
vlc_player_listener_id *listener_id;
|
|
|
|
atomic_bool abort;
|
|
|
|
struct
|
|
{
|
|
vlc_array_t queue;
|
|
} incoming;
|
|
|
|
struct
|
|
{
|
|
vlc_array_t queue;
|
|
vlc_mutex_t lock;
|
|
} results;
|
|
|
|
vlc_cond_t incoming_cond;
|
|
|
|
struct
|
|
{
|
|
vlc_array_t queue;
|
|
vlc_cond_t cond;
|
|
bool b_working;
|
|
} processing;
|
|
};
|
|
|
|
static int Open (vlc_object_t *);
|
|
static void Close (vlc_object_t *);
|
|
static void CleanSys (fingerprinter_sys_t *);
|
|
static void *Run(void *);
|
|
|
|
/*****************************************************************************
|
|
* Module descriptor
|
|
****************************************************************************/
|
|
vlc_module_begin ()
|
|
set_subcategory(SUBCAT_ADVANCED_MISC)
|
|
set_shortname(N_("acoustid"))
|
|
set_description(N_("Track fingerprinter (based on Acoustid)"))
|
|
set_capability("fingerprinter", 10)
|
|
set_callbacks(Open, Close)
|
|
vlc_module_end ()
|
|
|
|
/*****************************************************************************
|
|
* Requests lifecycle
|
|
*****************************************************************************/
|
|
|
|
static int EnqueueRequest( fingerprinter_thread_t *f, fingerprint_request_t *r )
|
|
{
|
|
fingerprinter_sys_t *p_sys = f->p_sys;
|
|
vlc_player_Lock( p_sys->player );
|
|
int i_ret = vlc_array_append( &p_sys->incoming.queue, r );
|
|
vlc_cond_signal( &p_sys->incoming_cond );
|
|
vlc_player_Unlock( p_sys->player );
|
|
return i_ret;
|
|
}
|
|
|
|
static void QueueIncomingRequests( fingerprinter_sys_t *p_sys )
|
|
{
|
|
for( size_t i = vlc_array_count( &p_sys->incoming.queue ); i > 0 ; i-- )
|
|
{
|
|
fingerprint_request_t *r = vlc_array_item_at_index( &p_sys->incoming.queue, i - 1 );
|
|
if( vlc_array_append( &p_sys->processing.queue, r ) )
|
|
fingerprint_request_Delete( r );
|
|
}
|
|
vlc_array_clear( &p_sys->incoming.queue );
|
|
}
|
|
|
|
static fingerprint_request_t * GetResult( fingerprinter_thread_t *f )
|
|
{
|
|
fingerprint_request_t *r = NULL;
|
|
fingerprinter_sys_t *p_sys = f->p_sys;
|
|
vlc_mutex_lock( &p_sys->results.lock );
|
|
if ( vlc_array_count( &p_sys->results.queue ) )
|
|
{
|
|
r = vlc_array_item_at_index( &p_sys->results.queue, 0 );
|
|
vlc_array_remove( &p_sys->results.queue, 0 );
|
|
}
|
|
vlc_mutex_unlock( &p_sys->results.lock );
|
|
return r;
|
|
}
|
|
|
|
static void ApplyResult( fingerprint_request_t *p_r, size_t i_resultid )
|
|
{
|
|
if ( i_resultid >= vlc_array_count( & p_r->results.metas_array ) ) return;
|
|
|
|
vlc_meta_t *p_meta = (vlc_meta_t *)
|
|
vlc_array_item_at_index( & p_r->results.metas_array, i_resultid );
|
|
input_item_t *p_item = p_r->p_item;
|
|
vlc_mutex_lock( &p_item->lock );
|
|
vlc_meta_Merge( p_item->p_meta, p_meta );
|
|
vlc_mutex_unlock( &p_item->lock );
|
|
}
|
|
|
|
static void player_on_state_changed(vlc_player_t *player,
|
|
enum vlc_player_state new_state,
|
|
void *p_user_data)
|
|
{
|
|
VLC_UNUSED(player);
|
|
fingerprinter_sys_t *p_sys = p_user_data;
|
|
if (new_state == VLC_PLAYER_STATE_STOPPED)
|
|
{
|
|
p_sys->processing.b_working = false;
|
|
vlc_cond_signal( &p_sys->processing.cond );
|
|
}
|
|
}
|
|
|
|
static void DoFingerprint( fingerprinter_thread_t *p_fingerprinter,
|
|
acoustid_fingerprint_t *fp,
|
|
const char *psz_uri )
|
|
{
|
|
input_item_t *p_item = input_item_New( NULL, NULL );
|
|
if ( unlikely(p_item == NULL) )
|
|
return;
|
|
|
|
char *psz_sout_option;
|
|
/* Note: need at -max- 2 channels, but we can't guess it before playing */
|
|
/* the stereo upmix could make the mono tracks fingerprint to differ :/ */
|
|
if ( asprintf( &psz_sout_option,
|
|
"sout=#transcode{acodec=%s,channels=2}:chromaprint",
|
|
( VLC_CODEC_S16L == VLC_CODEC_S16N ) ? "s16l" : "s16b" )
|
|
== -1 )
|
|
{
|
|
input_item_Release( p_item );
|
|
return;
|
|
}
|
|
|
|
input_item_AddOption( p_item, psz_sout_option, VLC_INPUT_OPTION_TRUSTED );
|
|
free( psz_sout_option );
|
|
if ( fp->i_duration )
|
|
{
|
|
if ( asprintf( &psz_sout_option, "stop-time=%u", fp->i_duration ) == -1 )
|
|
{
|
|
input_item_Release( p_item );
|
|
return;
|
|
}
|
|
input_item_AddOption( p_item, psz_sout_option, VLC_INPUT_OPTION_TRUSTED );
|
|
free( psz_sout_option );
|
|
}
|
|
input_item_SetURI( p_item, psz_uri ) ;
|
|
|
|
chromaprint_fingerprint_t chroma_fingerprint;
|
|
|
|
chroma_fingerprint.psz_fingerprint = NULL;
|
|
chroma_fingerprint.i_duration = fp->i_duration;
|
|
|
|
var_Create( p_fingerprinter, "fingerprint-data", VLC_VAR_ADDRESS );
|
|
var_SetAddress( p_fingerprinter, "fingerprint-data", &chroma_fingerprint );
|
|
|
|
vlc_player_t *player = p_fingerprinter->p_sys->player;
|
|
vlc_player_Lock(player);
|
|
|
|
p_fingerprinter->p_sys->processing.b_working = true;
|
|
|
|
int ret = vlc_player_SetCurrentMedia(player, p_item);
|
|
if (ret == VLC_SUCCESS)
|
|
ret = vlc_player_Start(player);
|
|
input_item_Release(p_item);
|
|
|
|
if (ret == VLC_SUCCESS)
|
|
{
|
|
while( !p_fingerprinter->p_sys->abort && p_fingerprinter->p_sys->processing.b_working )
|
|
vlc_player_CondWait(player,
|
|
&p_fingerprinter->p_sys->processing.cond);
|
|
|
|
fp->psz_fingerprint = chroma_fingerprint.psz_fingerprint;
|
|
if( !fp->i_duration ) /* had not given hint */
|
|
fp->i_duration = chroma_fingerprint.i_duration;
|
|
}
|
|
|
|
vlc_player_Unlock(player);
|
|
}
|
|
|
|
/*****************************************************************************
|
|
* Open:
|
|
*****************************************************************************/
|
|
static int Open(vlc_object_t *p_this)
|
|
{
|
|
fingerprinter_thread_t *p_fingerprinter = (fingerprinter_thread_t*) p_this;
|
|
fingerprinter_sys_t *p_sys = calloc(1, sizeof(fingerprinter_sys_t));
|
|
|
|
if ( !p_sys )
|
|
return VLC_ENOMEM;
|
|
|
|
p_fingerprinter->p_sys = p_sys;
|
|
|
|
var_Create(p_fingerprinter, "vout", VLC_VAR_STRING);
|
|
var_SetString(p_fingerprinter, "vout", "dummy");
|
|
var_Create(p_fingerprinter, "aout", VLC_VAR_STRING);
|
|
var_SetString(p_fingerprinter, "aout", "dummy");
|
|
p_sys->player = vlc_player_New(VLC_OBJECT(p_fingerprinter),
|
|
VLC_PLAYER_LOCK_NORMAL);
|
|
if (!p_sys->player)
|
|
{
|
|
free(p_sys);
|
|
return VLC_ENOMEM;
|
|
}
|
|
|
|
static const struct vlc_player_cbs cbs = {
|
|
.on_state_changed = player_on_state_changed,
|
|
};
|
|
|
|
vlc_player_Lock(p_sys->player);
|
|
p_sys->listener_id =
|
|
vlc_player_AddListener(p_sys->player, &cbs, p_fingerprinter->p_sys);
|
|
vlc_player_Unlock(p_sys->player);
|
|
if (!p_sys->listener_id)
|
|
{
|
|
vlc_player_Delete(p_sys->player);
|
|
free(p_sys);
|
|
return VLC_ENOMEM;
|
|
}
|
|
|
|
p_sys->abort = false;
|
|
vlc_array_init( &p_sys->incoming.queue );
|
|
vlc_cond_init( &p_sys->incoming_cond );
|
|
|
|
vlc_array_init( &p_sys->processing.queue );
|
|
vlc_cond_init( &p_sys->processing.cond );
|
|
|
|
vlc_array_init( &p_sys->results.queue );
|
|
vlc_mutex_init( &p_sys->results.lock );
|
|
|
|
p_fingerprinter->pf_enqueue = EnqueueRequest;
|
|
p_fingerprinter->pf_getresults = GetResult;
|
|
p_fingerprinter->pf_apply = ApplyResult;
|
|
|
|
var_Create( p_fingerprinter, "results-available", VLC_VAR_BOOL );
|
|
if( vlc_clone( &p_sys->thread, Run, p_fingerprinter ) )
|
|
{
|
|
msg_Err( p_fingerprinter, "cannot spawn fingerprinter thread" );
|
|
goto error;
|
|
}
|
|
|
|
return VLC_SUCCESS;
|
|
|
|
error:
|
|
CleanSys( p_sys );
|
|
free( p_sys );
|
|
return VLC_EGENERIC;
|
|
}
|
|
|
|
/*****************************************************************************
|
|
* Close:
|
|
*****************************************************************************/
|
|
static void Close(vlc_object_t *p_this)
|
|
{
|
|
fingerprinter_thread_t *p_fingerprinter = (fingerprinter_thread_t*) p_this;
|
|
fingerprinter_sys_t *p_sys = p_fingerprinter->p_sys;
|
|
|
|
vlc_player_Lock( p_sys->player );
|
|
p_sys->abort = true;
|
|
vlc_cond_signal( &p_sys->processing.cond );
|
|
vlc_cond_signal( &p_sys->incoming_cond );
|
|
vlc_player_Unlock( p_sys->player );
|
|
vlc_join( p_sys->thread, NULL );
|
|
|
|
CleanSys( p_sys );
|
|
free( p_sys );
|
|
}
|
|
|
|
static void CleanSys( fingerprinter_sys_t *p_sys )
|
|
{
|
|
for ( size_t i = 0; i < vlc_array_count( &p_sys->incoming.queue ); i++ )
|
|
fingerprint_request_Delete( vlc_array_item_at_index( &p_sys->incoming.queue, i ) );
|
|
vlc_array_clear( &p_sys->incoming.queue );
|
|
|
|
for ( size_t i = 0; i < vlc_array_count( &p_sys->processing.queue ); i++ )
|
|
fingerprint_request_Delete( vlc_array_item_at_index( &p_sys->processing.queue, i ) );
|
|
vlc_array_clear( &p_sys->processing.queue );
|
|
|
|
for ( size_t i = 0; i < vlc_array_count( &p_sys->results.queue ); i++ )
|
|
fingerprint_request_Delete( vlc_array_item_at_index( &p_sys->results.queue, i ) );
|
|
vlc_array_clear( &p_sys->results.queue );
|
|
|
|
vlc_player_Lock(p_sys->player);
|
|
vlc_player_RemoveListener(p_sys->player, p_sys->listener_id);
|
|
vlc_player_Unlock(p_sys->player);
|
|
vlc_player_Delete(p_sys->player);
|
|
}
|
|
|
|
static void fill_metas_with_results( fingerprint_request_t *p_r, acoustid_fingerprint_t *p_f )
|
|
{
|
|
for( unsigned int i=0 ; i < p_f->results.count; i++ )
|
|
{
|
|
acoustid_result_t *p_result = & p_f->results.p_results[ i ];
|
|
for ( unsigned int j=0 ; j < p_result->recordings.count; j++ )
|
|
{
|
|
acoustid_mb_result_t *p_record = & p_result->recordings.p_recordings[ j ];
|
|
vlc_meta_t *p_meta = vlc_meta_New();
|
|
if ( p_meta )
|
|
{
|
|
vlc_meta_Set( p_meta, vlc_meta_Title, p_record->psz_title );
|
|
vlc_meta_Set( p_meta, vlc_meta_Artist, p_record->psz_artist );
|
|
vlc_meta_SetExtra( p_meta, "musicbrainz-id", p_record->s_musicbrainz_id );
|
|
if( vlc_array_append( & p_r->results.metas_array, p_meta ) )
|
|
vlc_meta_Delete( p_meta );
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
/*****************************************************************************
|
|
* Run :
|
|
*****************************************************************************/
|
|
static void *Run( void *opaque )
|
|
{
|
|
vlc_thread_set_name("vlc-fingerprint");
|
|
|
|
fingerprinter_thread_t *p_fingerprinter = opaque;
|
|
fingerprinter_sys_t *p_sys = p_fingerprinter->p_sys;
|
|
|
|
/* main loop */
|
|
for (;;)
|
|
{
|
|
vlc_player_Lock( p_sys->player );
|
|
|
|
while( vlc_array_count( &p_sys->incoming.queue ) == 0 )
|
|
{
|
|
if( p_sys->abort )
|
|
{
|
|
vlc_player_Unlock( p_sys->player );
|
|
return NULL;
|
|
}
|
|
vlc_player_CondWait( p_sys->player, &p_sys->incoming_cond );
|
|
}
|
|
|
|
QueueIncomingRequests( p_sys );
|
|
|
|
vlc_player_Unlock( p_sys->player );
|
|
|
|
bool results_available = false;
|
|
while( vlc_array_count( &p_sys->processing.queue ) )
|
|
{
|
|
fingerprint_request_t *p_data = vlc_array_item_at_index( &p_sys->processing.queue, 0 );
|
|
|
|
char *psz_uri = input_item_GetURI( p_data->p_item );
|
|
if ( psz_uri != NULL )
|
|
{
|
|
acoustid_fingerprint_t acoustid_print = {0};
|
|
|
|
/* overwrite with hint, as in this case, fingerprint's session will be truncated */
|
|
if ( p_data->i_duration )
|
|
acoustid_print.i_duration = p_data->i_duration;
|
|
|
|
DoFingerprint( p_fingerprinter, &acoustid_print, psz_uri );
|
|
free( psz_uri );
|
|
|
|
acoustid_config_t cfg = { .p_obj = VLC_OBJECT(p_fingerprinter),
|
|
.psz_server = NULL, .psz_apikey = NULL };
|
|
acoustid_lookup_fingerprint( &cfg, &acoustid_print );
|
|
fill_metas_with_results( p_data, &acoustid_print );
|
|
|
|
for( unsigned j = 0; j < acoustid_print.results.count; j++ )
|
|
acoustid_result_release( &acoustid_print.results.p_results[j] );
|
|
if( acoustid_print.results.count )
|
|
free( acoustid_print.results.p_results );
|
|
free( acoustid_print.psz_fingerprint );
|
|
}
|
|
|
|
/* copy results */
|
|
vlc_mutex_lock( &p_sys->results.lock );
|
|
if( vlc_array_append( &p_sys->results.queue, p_data ) )
|
|
fingerprint_request_Delete( p_data );
|
|
else
|
|
results_available = true;
|
|
vlc_mutex_unlock( &p_sys->results.lock );
|
|
|
|
// the fingerprint request must not exist both in the
|
|
// processing and results queue, even in case of thread
|
|
// cancellation, so remove it immediately
|
|
vlc_array_remove( &p_sys->processing.queue, 0 );
|
|
|
|
if( p_sys->abort )
|
|
return NULL;
|
|
}
|
|
|
|
if ( results_available )
|
|
{
|
|
var_TriggerCallback( p_fingerprinter, "results-available" );
|
|
}
|
|
}
|
|
|
|
vlc_assert_unreachable();
|
|
}
|
|
|