X-Git-Url: https://git.sesse.net/?a=blobdiff_plain;f=lib%2Faudio.c;h=0165d16214de009e7fbc154ce7e2470203cb3553;hb=b70d2acf4c2e9be5af44d88c4968f4173f75376d;hp=18ff2e7ccb04783e3c2627afb0e2ae9333730be8;hpb=b27ba687c24fd2071ad25ef8eac2613fce11af61;p=vlc diff --git a/lib/audio.c b/lib/audio.c index 18ff2e7ccb..0165d16214 100644 --- a/lib/audio.c +++ b/lib/audio.c @@ -35,7 +35,6 @@ #include #include -#include #include #include @@ -50,12 +49,7 @@ static audio_output_t *GetAOut( libvlc_media_player_t *mp ) { assert( mp != NULL ); - input_thread_t *p_input = libvlc_get_input_thread( mp ); - if( p_input == NULL ) - return NULL; - - audio_output_t * p_aout = input_GetAout( p_input ); - vlc_object_release( p_input ); + audio_output_t *p_aout = input_resource_HoldAout( mp->input.p_resource ); if( p_aout == NULL ) libvlc_printerr( "No active audio output" ); return p_aout; @@ -67,11 +61,11 @@ static audio_output_t *GetAOut( libvlc_media_player_t *mp ) libvlc_audio_output_t * libvlc_audio_output_list_get( libvlc_instance_t *p_instance ) { - VLC_UNUSED( p_instance ); + size_t count; + module_t **module_list = module_list_get( &count ); libvlc_audio_output_t *list = NULL; - module_t **module_list = module_list_get( NULL ); - for (size_t i = 0; module_list[i]; i++) + for (size_t i = 0; i < count; i++) { module_t *module = module_list[i]; @@ -100,6 +94,7 @@ libvlc_audio_output_t * } module_list_free( module_list ); + VLC_UNUSED( p_instance ); return list; } @@ -132,122 +127,121 @@ int libvlc_audio_output_set( libvlc_media_player_t *mp, const char *psz_name ) return -1; var_SetString( mp, "aout", value ); free( value ); - return 0; -} - -/**************************** - * Get count of devices. - *****************************/ -int libvlc_audio_output_device_count( libvlc_instance_t *p_instance, - const char *psz_audio_output ) -{ - char *psz_config_name; - if( !psz_audio_output ) - return 0; - if( asprintf( &psz_config_name, "%s-audio-device", psz_audio_output ) == -1 ) - return 0; - - module_config_t *p_module_config = config_FindConfig( - VLC_OBJECT( p_instance->p_libvlc_int ), psz_config_name ); - - if( p_module_config && p_module_config->pf_update_list ) - { - vlc_value_t val; - val.psz_string = strdup( p_module_config->value.psz ); - p_module_config->pf_update_list( - VLC_OBJECT( p_instance->p_libvlc_int ), psz_config_name, val, val, NULL ); - free( val.psz_string ); - free( psz_config_name ); + /* Forget the existing audio output */ + input_resource_ResetAout(mp->input.p_resource); - return p_module_config->i_list; - } + /* Create a new audio output */ + audio_output_t *aout = input_resource_GetAout(mp->input.p_resource); + if( aout != NULL ) + input_resource_PutAout(mp->input.p_resource, aout); - free( psz_config_name ); return 0; } -/******************************** - * Get long name of device - *********************************/ -char * libvlc_audio_output_device_longname( libvlc_instance_t *p_instance, - const char *psz_audio_output, - int i_device ) +libvlc_audio_output_device_t * +libvlc_audio_output_device_enum( libvlc_media_player_t *mp ) { - char *psz_config_name; - if( !psz_audio_output ) - return NULL; - if( asprintf( &psz_config_name, "%s-audio-device", psz_audio_output ) == -1 ) + audio_output_t *aout = GetAOut( mp ); + if( aout == NULL ) return NULL; - module_config_t *p_module_config = config_FindConfig( - VLC_OBJECT( p_instance->p_libvlc_int ), psz_config_name ); + libvlc_audio_output_device_t *list, **pp = &list; + char **values, **texts; - if( p_module_config ) + int n = aout_DevicesList( aout, &values, &texts ); + vlc_object_release( aout ); + if( n < 0 ) + goto err; + + for (int i = 0; i < n; i++) { - // refresh if there arent devices - if( p_module_config->i_list < 2 && p_module_config->pf_update_list ) + libvlc_audio_output_device_t *item = malloc( sizeof(*item) ); + if( unlikely(item == NULL) ) { - vlc_value_t val; - val.psz_string = strdup( p_module_config->value.psz ); - - p_module_config->pf_update_list( - VLC_OBJECT( p_instance->p_libvlc_int ), psz_config_name, val, val, NULL ); - free( val.psz_string ); + free( texts[i] ); + free( values[i] ); + continue; } - if( i_device >= 0 && i_device < p_module_config->i_list ) - { - free( psz_config_name ); - - if( p_module_config->ppsz_list_text[i_device] ) - return strdup( p_module_config->ppsz_list_text[i_device] ); - else - return strdup( p_module_config->ppsz_list[i_device] ); - } + *pp = item; + pp = &item->p_next; + item->psz_device = values[i]; + item->psz_description = texts[i]; } - free( psz_config_name ); - return NULL; + free( texts ); + free( values ); +err: + *pp = NULL; + return list; } -/******************************** - * Get id name of device - *********************************/ -char * libvlc_audio_output_device_id( libvlc_instance_t *p_instance, - const char *psz_audio_output, - int i_device ) +libvlc_audio_output_device_t * +libvlc_audio_output_device_list_get( libvlc_instance_t *p_instance, + const char *aout ) { - char *psz_config_name; - if( !psz_audio_output ) + char varname[32]; + if( (size_t)snprintf( varname, sizeof(varname), "%s-audio-device", aout ) + >= sizeof(varname) ) return NULL; - if( asprintf( &psz_config_name, "%s-audio-device", psz_audio_output ) == -1) - return NULL; - - module_config_t *p_module_config = config_FindConfig( - VLC_OBJECT( p_instance->p_libvlc_int ), psz_config_name ); - if( p_module_config ) + libvlc_audio_output_device_t *list = NULL, **pp = &list; + char **values, **texts; + ssize_t count = config_GetPszChoices( VLC_OBJECT(p_instance->p_libvlc_int), + varname, &values, &texts ); + for( ssize_t i = 0; i < count; i++ ) { - // refresh if there arent devices - if( p_module_config->i_list < 2 && p_module_config->pf_update_list ) - { - vlc_value_t val; - val.psz_string = strdup( p_module_config->value.psz ); + libvlc_audio_output_device_t *item = malloc( sizeof(*item) ); + if( unlikely(item == NULL) ) + break; - p_module_config->pf_update_list( - VLC_OBJECT( p_instance->p_libvlc_int ), psz_config_name, val, val, NULL ); - free( val.psz_string ); - } + *pp = item; + pp = &item->p_next; + item->psz_device = values[i]; + item->psz_description = texts[i]; + } - if( i_device >= 0 && i_device < p_module_config->i_list ) - { - free( psz_config_name ); - return strdup( p_module_config->ppsz_list[i_device] ); - } + *pp = NULL; + free( texts ); + free( values ); + (void) p_instance; + return list; +} + +void libvlc_audio_output_device_list_release( libvlc_audio_output_device_t *l ) +{ + while( l != NULL ) + { + libvlc_audio_output_device_t *next = l->p_next; + + free( l->psz_description ); + free( l->psz_device ); + free( l ); + l = next; } +} + +int libvlc_audio_output_device_count( libvlc_instance_t *p_instance, + const char *psz_audio_output ) +{ + (void) p_instance; (void) psz_audio_output; + return 0; +} + +char *libvlc_audio_output_device_longname( libvlc_instance_t *p_instance, + const char *psz_audio_output, + int i_device ) +{ + (void) p_instance; (void) psz_audio_output; (void) i_device; + return NULL; +} - free( psz_config_name ); +char *libvlc_audio_output_device_id( libvlc_instance_t *p_instance, + const char *psz_audio_output, + int i_device ) +{ + (void) p_instance; (void) psz_audio_output; (void) i_device; return NULL; } @@ -255,81 +249,120 @@ char * libvlc_audio_output_device_id( libvlc_instance_t *p_instance, * Set device for using *****************************/ void libvlc_audio_output_device_set( libvlc_media_player_t *mp, - const char *psz_audio_output, - const char *psz_device_id ) + const char *module, const char *devid ) { - char *psz_config_name; - if( !psz_audio_output || !psz_device_id ) + if( devid == NULL ) + return; + + if( module != NULL ) + { + char *cfg_name; + + if( asprintf( &cfg_name, "%s-audio-device", module ) == -1 ) + return; + + if( !var_Type( mp, cfg_name ) ) + /* Don't recreate the same variable over and over and over... */ + var_Create( mp, cfg_name, VLC_VAR_STRING ); + var_SetString( mp, cfg_name, devid ); + free( cfg_name ); return; - if( asprintf( &psz_config_name, "%s-audio-device", psz_audio_output ) == -1 ) + } + + audio_output_t *aout = GetAOut( mp ); + if( aout == NULL ) return; - if( !var_Type( mp, psz_config_name ) ) - /* Don't recreate the same variable over and over and over... */ - var_Create( mp, psz_config_name, VLC_VAR_STRING ); - var_SetString( mp, psz_config_name, psz_device_id ); - free( psz_config_name ); + + aout_DeviceSet( aout, devid ); + vlc_object_release( aout ); +} + +char *libvlc_audio_output_device_get( libvlc_media_player_t *mp ) +{ + audio_output_t *aout = GetAOut( mp ); + if( aout == NULL ) + return NULL; + + char *devid = aout_DeviceGet( aout ); + + vlc_object_release( aout ); + + return devid; } -/***************************************************************************** - * libvlc_audio_output_get_device_type : Get the current audio device type - *****************************************************************************/ int libvlc_audio_output_get_device_type( libvlc_media_player_t *mp ) { - audio_output_t *p_aout = GetAOut( mp ); - if( p_aout ) - { - int i_device_type = var_GetInteger( p_aout, "audio-device" ); - vlc_object_release( p_aout ); - return i_device_type; - } + (void) mp; return libvlc_AudioOutputDevice_Error; } -/***************************************************************************** - * libvlc_audio_output_set_device_type : Set the audio device type - *****************************************************************************/ void libvlc_audio_output_set_device_type( libvlc_media_player_t *mp, int device_type ) { - audio_output_t *p_aout = GetAOut( mp ); - if( !p_aout ) - return; - if( var_SetInteger( p_aout, "audio-device", device_type ) < 0 ) - libvlc_printerr( "Error setting audio device" ); - vlc_object_release( p_aout ); + (void) mp; (void) device_type; } void libvlc_audio_toggle_mute( libvlc_media_player_t *mp ) { - aout_MuteToggle( mp ); + int mute = libvlc_audio_get_mute( mp ); + if( mute != -1 ) + libvlc_audio_set_mute( mp, !mute ); } int libvlc_audio_get_mute( libvlc_media_player_t *mp ) { - return aout_MuteGet( mp ); + int mute = -1; + + audio_output_t *aout = GetAOut( mp ); + if( aout != NULL ) + { + mute = aout_MuteGet( aout ); + vlc_object_release( aout ); + } + return mute; } void libvlc_audio_set_mute( libvlc_media_player_t *mp, int mute ) { - aout_MuteSet( VLC_OBJECT(mp), mute != 0 ); + audio_output_t *aout = GetAOut( mp ); + if( aout != NULL ) + { + mute = aout_MuteSet( aout, mute ); + vlc_object_release( aout ); + } } int libvlc_audio_get_volume( libvlc_media_player_t *mp ) { - float vol = aout_VolumeGet( mp ); - return ( vol >= 0.f ) ? lroundf( vol * 100.f ) : -1; + int volume = -1; + + audio_output_t *aout = GetAOut( mp ); + if( aout != NULL ) + { + float vol = aout_VolumeGet( aout ); + vlc_object_release( aout ); + volume = lroundf( vol * 100.f ); + } + return volume; } int libvlc_audio_set_volume( libvlc_media_player_t *mp, int volume ) { float vol = volume / 100.f; - if (vol < 0.f) + if (!isgreaterequal(vol, 0.f)) { libvlc_printerr( "Volume out of range" ); return -1; } - aout_VolumeSet (mp, vol); - return 0; + + int ret = -1; + audio_output_t *aout = GetAOut( mp ); + if( aout != NULL ) + { + ret = aout_VolumeSet( aout, vol ); + vlc_object_release( aout ); + } + return ret; } /***************************************************************************** @@ -364,33 +397,12 @@ libvlc_track_description_t * int libvlc_audio_get_track( libvlc_media_player_t *p_mi ) { input_thread_t *p_input_thread = libvlc_get_input_thread( p_mi ); - vlc_value_t val_list; - vlc_value_t val; - int i_track = -1; - int i; - if( !p_input_thread ) return -1; - if( var_Get( p_input_thread, "audio-es", &val ) < 0 ) - { - vlc_object_release( p_input_thread ); - libvlc_printerr( "Audio track information not found" ); - return -1; - } - - var_Change( p_input_thread, "audio-es", VLC_VAR_GETCHOICES, &val_list, NULL ); - for( i = 0; i < val_list.p_list->i_count; i++ ) - { - if( val_list.p_list->p_values[i].i_int == val.i_int ) - { - i_track = i; - break; - } - } - var_FreeList( &val_list, NULL ); + int id = var_GetInteger( p_input_thread, "audio-es" ); vlc_object_release( p_input_thread ); - return i_track; + return id; } /***************************************************************************** @@ -400,30 +412,23 @@ int libvlc_audio_set_track( libvlc_media_player_t *p_mi, int i_track ) { input_thread_t *p_input_thread = libvlc_get_input_thread( p_mi ); vlc_value_t val_list; - vlc_value_t newval; - int i_ret; + int i_ret = -1; if( !p_input_thread ) return -1; var_Change( p_input_thread, "audio-es", VLC_VAR_GETCHOICES, &val_list, NULL ); - if( (i_track < 0) || (i_track > val_list.p_list->i_count) ) + for( int i = 0; i < val_list.p_list->i_count; i++ ) { - libvlc_printerr( "Audio track out of range" ); - i_ret = -1; - goto end; - } - - newval = val_list.p_list->p_values[i_track]; - i_ret = var_Set( p_input_thread, "audio-es", newval ); - if( i_ret < 0 ) - { - libvlc_printerr( "Audio track out of range" ); /* Race... */ - i_ret = -1; - goto end; + if( i_track == val_list.p_list->p_values[i].i_int ) + { + if( var_SetInteger( p_input_thread, "audio-es", i_track ) < 0 ) + break; + i_ret = 0; + goto end; + } } - i_ret = 0; - + libvlc_printerr( "Track identifier not found" ); end: var_FreeList( &val_list, NULL ); vlc_object_release( p_input_thread ); @@ -439,7 +444,7 @@ int libvlc_audio_get_channel( libvlc_media_player_t *mp ) if( !p_aout ) return 0; - int val = var_GetInteger( p_aout, "audio-channels" ); + int val = var_GetInteger( p_aout, "stereo-mode" ); vlc_object_release( p_aout ); return val; } @@ -455,7 +460,7 @@ int libvlc_audio_set_channel( libvlc_media_player_t *mp, int channel ) if( !p_aout ) return -1; - if( var_SetInteger( p_aout, "audio-channels", channel ) < 0 ) + if( var_SetInteger( p_aout, "stereo-mode", channel ) < 0 ) { libvlc_printerr( "Audio channel out of range" ); ret = -1; @@ -497,3 +502,141 @@ int libvlc_audio_set_delay( libvlc_media_player_t *p_mi, int64_t i_delay ) } return ret; } + +/***************************************************************************** + * libvlc_audio_equalizer_get_preset_count : Get the number of equalizer presets + *****************************************************************************/ +unsigned libvlc_audio_equalizer_get_preset_count( void ) +{ + return NB_PRESETS; +} + +/***************************************************************************** + * libvlc_audio_equalizer_get_preset_name : Get the name for a preset + *****************************************************************************/ +const char *libvlc_audio_equalizer_get_preset_name( unsigned u_index ) +{ + if ( u_index >= NB_PRESETS ) + return NULL; + + return preset_list_text[ u_index ]; +} + +/***************************************************************************** + * libvlc_audio_equalizer_get_band_count : Get the number of equalizer frequency bands + *****************************************************************************/ +unsigned libvlc_audio_equalizer_get_band_count( void ) +{ + return EQZ_BANDS_MAX; +} + +/***************************************************************************** + * libvlc_audio_equalizer_get_band_frequency : Get the frequency for a band + *****************************************************************************/ +float libvlc_audio_equalizer_get_band_frequency( unsigned u_index ) +{ + if ( u_index >= EQZ_BANDS_MAX ) + return -1.f; + + return f_iso_frequency_table_10b[ u_index ]; +} + +/***************************************************************************** + * libvlc_audio_equalizer_new : Create a new audio equalizer with zeroed values + *****************************************************************************/ +libvlc_equalizer_t *libvlc_audio_equalizer_new( void ) +{ + libvlc_equalizer_t *p_equalizer; + p_equalizer = malloc( sizeof( *p_equalizer ) ); + if ( unlikely( p_equalizer == NULL ) ) + return NULL; + + p_equalizer->f_preamp = 0.f; + for ( unsigned i = 0; i < EQZ_BANDS_MAX; i++ ) + p_equalizer->f_amp[ i ] = 0.f; + + return p_equalizer; +} + +/***************************************************************************** + * libvlc_audio_equalizer_new_from_preset : Create a new audio equalizer based on a preset + *****************************************************************************/ +libvlc_equalizer_t *libvlc_audio_equalizer_new_from_preset( unsigned u_index ) +{ + libvlc_equalizer_t *p_equalizer; + + if ( u_index >= NB_PRESETS ) + return NULL; + + p_equalizer = malloc( sizeof( *p_equalizer ) ); + if ( unlikely( p_equalizer == NULL ) ) + return NULL; + + p_equalizer->f_preamp = eqz_preset_10b[ u_index ].f_preamp; + + for ( unsigned i = 0; i < EQZ_BANDS_MAX; i++ ) + p_equalizer->f_amp[ i ] = eqz_preset_10b[ u_index ].f_amp[ i ]; + + return p_equalizer; +} + +/***************************************************************************** + * libvlc_audio_equalizer_release : Release a previously created equalizer + *****************************************************************************/ +void libvlc_audio_equalizer_release( libvlc_equalizer_t *p_equalizer ) +{ + free( p_equalizer ); +} + +/***************************************************************************** + * libvlc_audio_equalizer_set_preamp : Set the preamp value for an equalizer + *****************************************************************************/ +int libvlc_audio_equalizer_set_preamp( libvlc_equalizer_t *p_equalizer, float f_preamp ) +{ + if( isnan(f_preamp) ) + return -1; + if( f_preamp < -20.f ) + f_preamp = -20.f; + else if( f_preamp > 20.f ) + f_preamp = 20.f; + + p_equalizer->f_preamp = f_preamp; + return 0; +} + +/***************************************************************************** + * libvlc_audio_equalizer_get_preamp : Get the preamp value for an equalizer + *****************************************************************************/ +float libvlc_audio_equalizer_get_preamp( libvlc_equalizer_t *p_equalizer ) +{ + return p_equalizer->f_preamp; +} + +/***************************************************************************** + * libvlc_audio_equalizer_set_amp_at_index : Set the amplification value for an equalizer band + *****************************************************************************/ +int libvlc_audio_equalizer_set_amp_at_index( libvlc_equalizer_t *p_equalizer, float f_amp, unsigned u_band ) +{ + if( u_band >= EQZ_BANDS_MAX || isnan(f_amp) ) + return -1; + + + if( f_amp < -20.f ) + f_amp = -20.f; + else if( f_amp > 20.f ) + f_amp = 20.f; + + p_equalizer->f_amp[ u_band ] = f_amp; + return 0; +} + +/***************************************************************************** + * libvlc_audio_equalizer_get_amp_at_index : Get the amplification value for an equalizer band + *****************************************************************************/ +float libvlc_audio_equalizer_get_amp_at_index( libvlc_equalizer_t *p_equalizer, unsigned u_band ) +{ + if ( u_band >= EQZ_BANDS_MAX ) + return nanf(""); + + return p_equalizer->f_amp[ u_band ]; +}