⭐ 欢迎来到虫虫下载站! | 📦 资源下载 📁 资源专辑 ℹ️ 关于我们
⭐ 虫虫下载站

📄 alsa.c

📁 video linux conference
💻 C
📖 第 1 页 / 共 3 页
字号:
/***************************************************************************** * alsa.c : alsa plugin for vlc ***************************************************************************** * Copyright (C) 2000-2001 VideoLAN * $Id: alsa.c 10296 2005-03-11 09:54:47Z sigmunau $ * * Authors: Henri Fallon <henri@videolan.org> - Original Author *          Jeffrey Baker <jwbaker@acm.org> - Port to ALSA 1.0 API *          John Paul Lorenti <jpl31@columbia.edu> - Device selection *          Arnaud de Bossoreille de Ribou <bozo@via.ecp.fr> - S/PDIF and aout3 * * This program is free software; you can redistribute it and/or modify * it under the terms of the GNU General Public License as published by * the Free Software Foundation; either version 2 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 General Public License for more details. * * You should have received a copy of the GNU General Public License * along with this program; if not, write to the Free Software * Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA  02111, USA. *****************************************************************************//***************************************************************************** * Preamble *****************************************************************************/#include <errno.h>                                                 /* ENOMEM */#include <string.h>                                            /* strerror() */#include <stdlib.h>                            /* calloc(), malloc(), free() */#include <vlc/vlc.h>#include <vlc/aout.h>#include "aout_internal.h"/* ALSA part   Note: we use the new API which is available since 0.9.0beta10a. */#define ALSA_PCM_NEW_HW_PARAMS_API#define ALSA_PCM_NEW_SW_PARAMS_API#include <alsa/asoundlib.h>/***************************************************************************** * aout_sys_t: ALSA audio output method descriptor ***************************************************************************** * This structure is part of the audio output thread descriptor. * It describes the ALSA specific properties of an audio device. *****************************************************************************/struct aout_sys_t{    snd_pcm_t         * p_snd_pcm;    int                 i_period_time;#ifdef ALSA_DEBUG    snd_output_t      * p_snd_stderr;#endif    int b_playing;                                         /* playing status */    mtime_t start_date;    vlc_mutex_t lock;    vlc_cond_t  wait ;    snd_pcm_status_t *p_status;};#define A52_FRAME_NB 1536/* These values are in frames.   To convert them to a number of bytes you have to multiply them by the   number of channel(s) (eg. 2 for stereo) and the size of a sample (eg.   2 for int16_t). */#define ALSA_DEFAULT_PERIOD_SIZE        1024#define ALSA_DEFAULT_BUFFER_SIZE        ( ALSA_DEFAULT_PERIOD_SIZE << 8 )#define ALSA_SPDIF_PERIOD_SIZE          A52_FRAME_NB#define ALSA_SPDIF_BUFFER_SIZE          ( ALSA_SPDIF_PERIOD_SIZE << 4 )/* Why << 4 ? --Meuuh *//* Why not ? --Bozo *//* Right. --Meuuh */#define DEFAULT_ALSA_DEVICE N_("default")/***************************************************************************** * Local prototypes *****************************************************************************/static int  Open         ( vlc_object_t * );static void Close        ( vlc_object_t * );static void Play         ( aout_instance_t * );static int  ALSAThread   ( aout_instance_t * );static void ALSAFill     ( aout_instance_t * );static int FindDevicesCallback( vlc_object_t *p_this, char const *psz_name,                                vlc_value_t newval, vlc_value_t oldval, void *p_unused );/***************************************************************************** * Module descriptor *****************************************************************************/static char *ppsz_devices[] = { "default" };static char *ppsz_devices_text[] = { N_("Default") };vlc_module_begin();    set_shortname( "ALSA" );    set_description( _("ALSA audio output") );    set_category( CAT_AUDIO );    set_subcategory( SUBCAT_AUDIO_AOUT );    add_string( "alsadev", DEFAULT_ALSA_DEVICE, aout_FindAndRestart,                N_("ALSA Device Name"), NULL, VLC_FALSE );        change_string_list( ppsz_devices, ppsz_devices_text, FindDevicesCallback );        change_action_add( FindDevicesCallback, N_("Refresh list") );    set_capability( "audio output", 150 );    set_callbacks( Open, Close );vlc_module_end();/***************************************************************************** * Probe: probe the audio device for available formats and channels *****************************************************************************/static void Probe( aout_instance_t * p_aout,                   const char * psz_device, const char * psz_iec_device,                   int *pi_snd_pcm_format ){    struct aout_sys_t * p_sys = p_aout->output.p_sys;    vlc_value_t val, text;    int i_ret;    var_Create ( p_aout, "audio-device", VLC_VAR_INTEGER | VLC_VAR_HASCHOICE );    text.psz_string = _("Audio Device");    var_Change( p_aout, "audio-device", VLC_VAR_SETTEXT, &text, NULL );    /* We'll open the audio device in non blocking mode so we can just exit     * when it is already in use, but for the real stuff we'll still use     * the blocking mode */    /* Now test linear PCM capabilities */    if ( !(i_ret = snd_pcm_open( &p_sys->p_snd_pcm, psz_device,                                 SND_PCM_STREAM_PLAYBACK,                                 SND_PCM_NONBLOCK ) ) )    {        int i_channels;        snd_pcm_hw_params_t * p_hw;        snd_pcm_hw_params_alloca (&p_hw);        if ( snd_pcm_hw_params_any( p_sys->p_snd_pcm, p_hw ) < 0 )        {            msg_Warn( p_aout, "unable to retrieve initial hardware parameters"                              ", disabling linear PCM audio" );            snd_pcm_close( p_sys->p_snd_pcm );            var_Destroy( p_aout, "audio-device" );            return;        }        if ( snd_pcm_hw_params_set_format( p_sys->p_snd_pcm, p_hw,                                           *pi_snd_pcm_format ) < 0 )        {            if( *pi_snd_pcm_format != SND_PCM_FORMAT_S16 )            {                *pi_snd_pcm_format = SND_PCM_FORMAT_S16;                if ( snd_pcm_hw_params_set_format( p_sys->p_snd_pcm, p_hw,                                                   *pi_snd_pcm_format ) < 0 )                {                    msg_Warn( p_aout, "unable to set stream sample size and "                              "word order, disabling linear PCM audio" );                    snd_pcm_close( p_sys->p_snd_pcm );                    var_Destroy( p_aout, "audio-device" );                    return;                }            }        }        i_channels = aout_FormatNbChannels( &p_aout->output.output );        while ( i_channels > 0 )        {            if ( !snd_pcm_hw_params_test_channels( p_sys->p_snd_pcm, p_hw,                                                   i_channels ) )            {                switch ( i_channels )                {                case 1:                    val.i_int = AOUT_VAR_MONO;                    text.psz_string = N_("Mono");                    var_Change( p_aout, "audio-device",                                VLC_VAR_ADDCHOICE, &val, &text );                    break;                case 2:                    val.i_int = AOUT_VAR_STEREO;                    text.psz_string = N_("Stereo");                    var_Change( p_aout, "audio-device",                                VLC_VAR_ADDCHOICE, &val, &text );                    var_Set( p_aout, "audio-device", val );                    break;                case 4:                    val.i_int = AOUT_VAR_2F2R;                    text.psz_string = N_("2 Front 2 Rear");                    var_Change( p_aout, "audio-device",                                VLC_VAR_ADDCHOICE, &val, &text );                    break;                case 6:                    val.i_int = AOUT_VAR_5_1;                    text.psz_string = N_("5.1");                    var_Change( p_aout, "audio-device",                                VLC_VAR_ADDCHOICE, &val, &text );                    break;                }            }            --i_channels;        }        /* Special case for mono on stereo only boards */        i_channels = aout_FormatNbChannels( &p_aout->output.output );                var_Change( p_aout, "audio-device", VLC_VAR_CHOICESCOUNT, &val, NULL );        if( val.i_int <= 0 && i_channels == 1 )        {            if ( !snd_pcm_hw_params_test_channels( p_sys->p_snd_pcm, p_hw, 2 ))            {                val.i_int = AOUT_VAR_STEREO;                text.psz_string = N_("Stereo");                var_Change( p_aout, "audio-device",                            VLC_VAR_ADDCHOICE, &val, &text );                var_Set( p_aout, "audio-device", val );            }        }                /* Close the previously opened device */        snd_pcm_close( p_sys->p_snd_pcm );    }    else if ( i_ret == -EBUSY )    {        msg_Warn( p_aout, "audio device: %s is already in use", psz_device );    }    /* Test for S/PDIF device if needed */    if ( psz_iec_device )    {        /* Opening the device should be enough */        if ( !(i_ret = snd_pcm_open( &p_sys->p_snd_pcm, psz_iec_device,                                     SND_PCM_STREAM_PLAYBACK,                                     SND_PCM_NONBLOCK ) ) )        {            val.i_int = AOUT_VAR_SPDIF;            text.psz_string = N_("A/52 over S/PDIF");            var_Change( p_aout, "audio-device",                        VLC_VAR_ADDCHOICE, &val, &text );            if( config_GetInt( p_aout, "spdif" ) )                var_Set( p_aout, "audio-device", val );            snd_pcm_close( p_sys->p_snd_pcm );        }        else if ( i_ret == -EBUSY )        {            msg_Warn( p_aout, "audio device: %s is already in use",                      psz_iec_device );        }    }    var_Change( p_aout, "audio-device", VLC_VAR_CHOICESCOUNT, &val, NULL );    if( val.i_int <= 0 )    {        /* Probe() has failed. */        msg_Dbg( p_aout, "failed to find a useable alsa configuration" );        var_Destroy( p_aout, "audio-device" );        return;    }    /* Add final settings to the variable */    var_AddCallback( p_aout, "audio-device", aout_ChannelsRestart, NULL );    val.b_bool = VLC_TRUE;    var_Set( p_aout, "intf-change", val );}/***************************************************************************** * Open: create a handle and open an alsa device ***************************************************************************** * This function opens an alsa device, through the alsa API. * * Note: the only heap-allocated string is psz_device. All the other pointers * are references to psz_device or to stack-allocated data. *****************************************************************************/static int Open( vlc_object_t *p_this ){    aout_instance_t * p_aout = (aout_instance_t *)p_this;    struct aout_sys_t * p_sys;    vlc_value_t val;    char psz_default_iec_device[128]; /* Buffer used to store the default                                         S/PDIF device */    char * psz_device, * psz_iec_device; /* device names for PCM and S/PDIF                                            output */    int i_vlc_pcm_format; /* Audio format for VLC's data */    int i_snd_pcm_format; /* Audio format for ALSA's data */    snd_pcm_uframes_t i_buffer_size = 0;    snd_pcm_uframes_t i_period_size = 0;    int i_channels = 0;    snd_pcm_hw_params_t *p_hw;    snd_pcm_sw_params_t *p_sw;    int i_snd_rc = -1;    unsigned int i_old_rate;    /* Allocate structures */    p_aout->output.p_sys = p_sys = malloc( sizeof( aout_sys_t ) );    if( p_sys == NULL )    {        msg_Err( p_aout, "out of memory" );        return VLC_ENOMEM;    }    p_sys->b_playing = VLC_FALSE;    p_sys->start_date = 0;    p_sys->p_status = (snd_pcm_status_t *)malloc(snd_pcm_status_sizeof());    vlc_cond_init( p_aout, &p_sys->wait );    vlc_mutex_init( p_aout, &p_sys->lock );

⌨️ 快捷键说明

复制代码 Ctrl + C
搜索代码 Ctrl + F
全屏模式 F11
切换主题 Ctrl + Shift + D
显示快捷键 ?
增大字号 Ctrl + =
减小字号 Ctrl + -