/***************************************************************************** * meta.c : Metadata handling ***************************************************************************** * Copyright (C) 1998-2004 VLC authors and VideoLAN * $Id$ * * Authors: Antoine Cellerier * Clément Stenac #include #include #include #include #include #include #include "input_internal.h" #include "../playlist/art.h" struct vlc_meta_t { char * ppsz_meta[VLC_META_TYPE_COUNT]; vlc_dictionary_t extra_tags; int i_status; }; /* FIXME bad name convention */ const char * vlc_meta_TypeToLocalizedString( vlc_meta_type_t meta_type ) { static const char posix_names[][18] = { [vlc_meta_Title] = N_("Title"), [vlc_meta_Artist] = N_("Artist"), [vlc_meta_Genre] = N_("Genre"), [vlc_meta_Copyright] = N_("Copyright"), [vlc_meta_Album] = N_("Album"), [vlc_meta_TrackNumber] = N_("Track number"), [vlc_meta_Description] = N_("Description"), [vlc_meta_Rating] = N_("Rating"), [vlc_meta_Date] = N_("Date"), [vlc_meta_Setting] = N_("Setting"), [vlc_meta_URL] = N_("URL"), [vlc_meta_Language] = N_("Language"), [vlc_meta_ESNowPlaying]= N_("Now Playing"), [vlc_meta_NowPlaying] = N_("Now Playing"), [vlc_meta_Publisher] = N_("Publisher"), [vlc_meta_EncodedBy] = N_("Encoded by"), [vlc_meta_ArtworkURL] = N_("Artwork URL"), [vlc_meta_TrackID] = N_("Track ID"), [vlc_meta_TrackTotal] = N_("Number of Tracks"), [vlc_meta_Director] = N_("Director"), [vlc_meta_Season] = N_("Season"), [vlc_meta_Episode] = N_("Episode"), [vlc_meta_ShowName] = N_("Show Name"), [vlc_meta_Actors] = N_("Actors"), [vlc_meta_AlbumArtist] = N_("Album Artist"), [vlc_meta_DiscNumber] = N_("Disc number") }; assert (meta_type < (sizeof(posix_names) / sizeof(posix_names[0]))); return vlc_gettext (posix_names[meta_type]); }; /** * vlc_meta contructor. * vlc_meta_Delete() will free the returned pointer. */ vlc_meta_t *vlc_meta_New( void ) { vlc_meta_t *m = (vlc_meta_t*)malloc( sizeof(*m) ); if( !m ) return NULL; memset( m->ppsz_meta, 0, sizeof(m->ppsz_meta) ); m->i_status = 0; vlc_dictionary_init( &m->extra_tags, 0 ); return m; } /* Free a dictonary key allocated by strdup() in vlc_meta_AddExtra() */ static void vlc_meta_FreeExtraKey( void *p_data, void *p_obj ) { VLC_UNUSED( p_obj ); free( p_data ); } void vlc_meta_Delete( vlc_meta_t *m ) { for( int i = 0; i < VLC_META_TYPE_COUNT ; i++ ) free( m->ppsz_meta[i] ); vlc_dictionary_clear( &m->extra_tags, vlc_meta_FreeExtraKey, NULL ); free( m ); } /** * vlc_meta has two kinds of meta, the one in a table, and the one in a * dictionary. * FIXME - Why don't we merge those two? */ void vlc_meta_Set( vlc_meta_t *p_meta, vlc_meta_type_t meta_type, const char *psz_val ) { free( p_meta->ppsz_meta[meta_type] ); assert( psz_val == NULL || IsUTF8( psz_val ) ); p_meta->ppsz_meta[meta_type] = psz_val ? strdup( psz_val ) : NULL; } const char *vlc_meta_Get( const vlc_meta_t *p_meta, vlc_meta_type_t meta_type ) { return p_meta->ppsz_meta[meta_type]; } void vlc_meta_AddExtra( vlc_meta_t *m, const char *psz_name, const char *psz_value ) { char *psz_oldvalue = (char *)vlc_dictionary_value_for_key( &m->extra_tags, psz_name ); if( psz_oldvalue != kVLCDictionaryNotFound ) vlc_dictionary_remove_value_for_key( &m->extra_tags, psz_name, vlc_meta_FreeExtraKey, NULL ); vlc_dictionary_insert( &m->extra_tags, psz_name, strdup(psz_value) ); } const char * vlc_meta_GetExtra( const vlc_meta_t *m, const char *psz_name ) { return (char *)vlc_dictionary_value_for_key(&m->extra_tags, psz_name); } unsigned vlc_meta_GetExtraCount( const vlc_meta_t *m ) { return vlc_dictionary_keys_count(&m->extra_tags); } char** vlc_meta_CopyExtraNames( const vlc_meta_t *m ) { return vlc_dictionary_all_keys(&m->extra_tags); } /** * vlc_meta status (see vlc_meta_status_e) */ int vlc_meta_GetStatus( vlc_meta_t *m ) { return m->i_status; } void vlc_meta_SetStatus( vlc_meta_t *m, int status ) { m->i_status = status; } /** * Merging meta */ void vlc_meta_Merge( vlc_meta_t *dst, const vlc_meta_t *src ) { if( !dst || !src ) return; for( int i = 0; i < VLC_META_TYPE_COUNT; i++ ) { if( src->ppsz_meta[i] ) { free( dst->ppsz_meta[i] ); dst->ppsz_meta[i] = strdup( src->ppsz_meta[i] ); } } /* XXX: If speed up are needed, it is possible */ char **ppsz_all_keys = vlc_dictionary_all_keys( &src->extra_tags ); for( int i = 0; ppsz_all_keys && ppsz_all_keys[i]; i++ ) { /* Always try to remove the previous value */ vlc_dictionary_remove_value_for_key( &dst->extra_tags, ppsz_all_keys[i], vlc_meta_FreeExtraKey, NULL ); void *p_value = vlc_dictionary_value_for_key( &src->extra_tags, ppsz_all_keys[i] ); vlc_dictionary_insert( &dst->extra_tags, ppsz_all_keys[i], strdup( (const char*)p_value ) ); free( ppsz_all_keys[i] ); } free( ppsz_all_keys ); } void input_ExtractAttachmentAndCacheArt( input_thread_t *p_input, const char *name ) { input_item_t *p_item = input_priv(p_input)->p_item; if( input_item_IsArtFetched( p_item ) ) { /* XXX Weird, we should not end up with attachment:// art URL * unless there is a race condition */ msg_Warn( p_input, "art already fetched" ); if( likely(playlist_FindArtInCache( p_item ) == VLC_SUCCESS) ) return; } /* */ input_attachment_t *p_attachment = NULL; vlc_mutex_lock( &p_item->lock ); for( int i_idx = 0; i_idx < input_priv(p_input)->i_attachment; i_idx++ ) { input_attachment_t *a = input_priv(p_input)->attachment[i_idx]; if( !strcmp( a->psz_name, name ) ) { p_attachment = vlc_input_attachment_Duplicate( a ); break; } } vlc_mutex_unlock( &p_item->lock ); if( p_attachment == NULL ) { msg_Warn( p_input, "art attachment %s not found", name ); return; } /* */ const char *psz_type = NULL; if( !strcmp( p_attachment->psz_mime, "image/jpeg" ) ) psz_type = ".jpg"; else if( !strcmp( p_attachment->psz_mime, "image/png" ) ) psz_type = ".png"; else if( !strcmp( p_attachment->psz_mime, "image/x-pict" ) ) psz_type = ".pct"; playlist_SaveArt( VLC_OBJECT(p_input), p_item, p_attachment->p_data, p_attachment->i_data, psz_type ); vlc_input_attachment_Delete( p_attachment ); } int input_item_WriteMeta( vlc_object_t *obj, input_item_t *p_item ) { meta_export_t *p_export = vlc_custom_create( obj, sizeof( *p_export ), "meta writer" ); if( p_export == NULL ) return VLC_ENOMEM; p_export->p_item = p_item; int type; vlc_mutex_lock( &p_item->lock ); type = p_item->i_type; vlc_mutex_unlock( &p_item->lock ); if( type != ITEM_TYPE_FILE ) goto error; char *psz_uri = input_item_GetURI( p_item ); p_export->psz_file = vlc_uri2path( psz_uri ); if( p_export->psz_file == NULL ) msg_Err( p_export, "cannot write meta to remote media %s", psz_uri ); free( psz_uri ); if( p_export->psz_file == NULL ) goto error; module_t *p_mod = module_need( p_export, "meta writer", NULL, false ); if( p_mod ) module_unneed( p_export, p_mod ); vlc_object_release( p_export ); return VLC_SUCCESS; error: vlc_object_release( p_export ); return VLC_EGENERIC; } void vlc_audio_replay_gain_MergeFromMeta( audio_replay_gain_t *p_dst, const vlc_meta_t *p_meta ) { const char * psz_value; if( !p_meta ) return; if( (psz_value = vlc_meta_GetExtra(p_meta, "REPLAYGAIN_TRACK_GAIN")) || (psz_value = vlc_meta_GetExtra(p_meta, "RG_RADIO")) ) { p_dst->pb_gain[AUDIO_REPLAY_GAIN_TRACK] = true; p_dst->pf_gain[AUDIO_REPLAY_GAIN_TRACK] = us_atof( psz_value ); } if( (psz_value = vlc_meta_GetExtra(p_meta, "REPLAYGAIN_TRACK_PEAK" )) || (psz_value = vlc_meta_GetExtra(p_meta, "RG_PEAK" )) ) { p_dst->pb_peak[AUDIO_REPLAY_GAIN_TRACK] = true; p_dst->pf_peak[AUDIO_REPLAY_GAIN_TRACK] = us_atof( psz_value ); } if( (psz_value = vlc_meta_GetExtra(p_meta, "REPLAYGAIN_ALBUM_GAIN" )) || (psz_value = vlc_meta_GetExtra(p_meta, "RG_AUDIOPHILE" )) ) { p_dst->pb_gain[AUDIO_REPLAY_GAIN_ALBUM] = true; p_dst->pf_gain[AUDIO_REPLAY_GAIN_ALBUM] = us_atof( psz_value ); } if( (psz_value = vlc_meta_GetExtra(p_meta, "REPLAYGAIN_ALBUM_PEAK" )) ) { p_dst->pb_peak[AUDIO_REPLAY_GAIN_ALBUM] = true; p_dst->pf_peak[AUDIO_REPLAY_GAIN_ALBUM] = us_atof( psz_value ); } }