/*****************************************************************************
* waveout.c : Windows waveOut plugin for vlc
*****************************************************************************
- * Copyright (C) 2001 VideoLAN
- * $Id: waveout.c,v 1.22 2003/04/07 16:02:08 gbazin Exp $
+ * Copyright (C) 2001-2009 the VideoLAN team
+ * $Id$
+ *
+ * Authors: Gildas Bazin <gbazin@videolan.org>
+ * André Weber
*
- * Authors: Gildas Bazin <gbazin@netcourrier.com>
- *
* 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
*
* 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.
+ * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston MA 02110-1301, USA.
*****************************************************************************/
/*****************************************************************************
* Preamble
*****************************************************************************/
-#include <string.h> /* strerror() */
-#include <stdlib.h> /* calloc(), malloc(), free() */
-
-#include <vlc/vlc.h>
-#include <vlc/aout.h>
-#include "aout_internal.h"
-
-#include <windows.h>
-#include <mmsystem.h>
-
-#define FRAME_SIZE 1024 /* The size is in samples, not in bytes */
-#define FRAMES_NUM 4
-
-/*****************************************************************************
- * Useful macros
- *****************************************************************************/
-#ifdef UNDER_CE
-# define DWORD_PTR DWORD
-#endif
-
-#ifndef WAVE_FORMAT_IEEE_FLOAT
-# define WAVE_FORMAT_IEEE_FLOAT 0x0003
-#endif
-#ifndef WAVE_FORMAT_DOLBY_AC3_SPDIF
-# define WAVE_FORMAT_DOLBY_AC3_SPDIF 0x0092
+#ifdef HAVE_CONFIG_H
+# include "config.h"
#endif
-#ifndef WAVE_FORMAT_EXTENSIBLE
-#define WAVE_FORMAT_EXTENSIBLE 0xFFFE
-#endif
-
-#ifndef SPEAKER_FRONT_LEFT
-# define SPEAKER_FRONT_LEFT 0x1
-# define SPEAKER_FRONT_RIGHT 0x2
-# define SPEAKER_FRONT_CENTER 0x4
-# define SPEAKER_LOW_FREQUENCY 0x8
-# define SPEAKER_BACK_LEFT 0x10
-# define SPEAKER_BACK_RIGHT 0x20
-# define SPEAKER_FRONT_LEFT_OF_CENTER 0x40
-# define SPEAKER_FRONT_RIGHT_OF_CENTER 0x80
-# define SPEAKER_BACK_CENTER 0x100
-# define SPEAKER_SIDE_LEFT 0x200
-# define SPEAKER_SIDE_RIGHT 0x400
-# define SPEAKER_TOP_CENTER 0x800
-# define SPEAKER_TOP_FRONT_LEFT 0x1000
-# define SPEAKER_TOP_FRONT_CENTER 0x2000
-# define SPEAKER_TOP_FRONT_RIGHT 0x4000
-# define SPEAKER_TOP_BACK_LEFT 0x8000
-# define SPEAKER_TOP_BACK_CENTER 0x10000
-# define SPEAKER_TOP_BACK_RIGHT 0x20000
-# define SPEAKER_RESERVED 0x80000000
-#endif
+#include <vlc_common.h>
+#include <vlc_plugin.h>
+#include <vlc_aout.h>
+#include <vlc_charset.h>
-#ifndef _WAVEFORMATEXTENSIBLE_
-typedef struct {
- WAVEFORMATEX Format;
- union {
- WORD wValidBitsPerSample; /* bits of precision */
- WORD wSamplesPerBlock; /* valid if wBitsPerSample==0 */
- WORD wReserved; /* If neither applies, set to zero. */
- } Samples;
- DWORD dwChannelMask; /* which channels are */
- /* present in stream */
- GUID SubFormat;
-} WAVEFORMATEXTENSIBLE, *PWAVEFORMATEXTENSIBLE;
-#endif
+#include "windows_audio_common.h"
-#include <initguid.h>
-DEFINE_GUID( _KSDATAFORMAT_SUBTYPE_IEEE_FLOAT, WAVE_FORMAT_IEEE_FLOAT, 0x0000, 0x0010, 0x80, 0x00, 0x00, 0xaa, 0x00, 0x38, 0x9b, 0x71 );
-DEFINE_GUID( _KSDATAFORMAT_SUBTYPE_PCM, WAVE_FORMAT_PCM, 0x0000, 0x0010, 0x80, 0x00, 0x00, 0xaa, 0x00, 0x38, 0x9b, 0x71 );
-DEFINE_GUID( _KSDATAFORMAT_SUBTYPE_DOLBY_AC3_SPDIF, WAVE_FORMAT_DOLBY_AC3_SPDIF, 0x0000, 0x0010, 0x80, 0x00, 0x00, 0xaa, 0x00, 0x38, 0x9b, 0x71 );
+#define FRAME_SIZE 4096 /* The size is in samples, not in bytes */
/*****************************************************************************
* Local prototypes
static void Close ( vlc_object_t * );
static void Play ( aout_instance_t * );
+/*****************************************************************************
+ * notification_thread_t: waveOut event thread
+ *****************************************************************************/
+typedef struct notification_thread_t
+{
+ VLC_COMMON_MEMBERS
+ aout_instance_t *p_aout;
+
+} notification_thread_t;
+
/* local functions */
static void Probe ( aout_instance_t * );
-static int OpenWaveOut ( aout_instance_t *, int, int, int, int, vlc_bool_t );
-static int OpenWaveOutPCM( aout_instance_t *, int*, int, int, int, vlc_bool_t );
-static void CheckReordering( aout_instance_t *, int );
+static int OpenWaveOut ( aout_instance_t *, uint32_t,
+ int, int, int, int, bool );
+static int OpenWaveOutPCM( aout_instance_t *, uint32_t,
+ vlc_fourcc_t*, int, int, int, bool );
static int PlayWaveOut ( aout_instance_t *, HWAVEOUT, WAVEHDR *,
- aout_buffer_t * );
+ aout_buffer_t *, bool );
static void CALLBACK WaveOutCallback ( HWAVEOUT, UINT, DWORD, DWORD, DWORD );
+static void* WaveOutThread( vlc_object_t * );
+
+static int VolumeGet( aout_instance_t *, audio_volume_t * );
+static int VolumeSet( aout_instance_t *, audio_volume_t );
+
+static int WaveOutClearDoneBuffers(aout_sys_t *p_sys);
+
+static int ReloadWaveoutDevices( vlc_object_t *, char const *,
+ vlc_value_t, vlc_value_t, void * );
+static uint32_t findDeviceID(char *);
+
+static const char psz_device_name_fmt[] = "%s ($%x,$%x)";
+
+static const char *const ppsz_adev[] = { "wavemapper", };
+static const char *const ppsz_adev_text[] = { N_("Microsoft Soundmapper") };
+
-static void InterleaveFloat32( float *, int *, int );
-static void InterleaveS16( int16_t *, int *, int );
/*****************************************************************************
* Module descriptor
*****************************************************************************/
-vlc_module_begin();
- set_description( _("Win32 waveOut extension output") );
- set_capability( "audio output", 50 );
- set_callbacks( Open, Close );
-vlc_module_end();
+#define DEVICE_TEXT N_("Select Audio Device")
+#define DEVICE_LONG N_("Select special Audio device, or let windows "\
+ "decide (default), change needs VLC restart "\
+ "to apply.")
+#define DEFAULT_AUDIO_DEVICE N_("Default Audio Device")
+
+vlc_module_begin ()
+ set_shortname( "WaveOut" )
+ set_description( N_("Win32 waveOut extension output") )
+ set_capability( "audio output", 50 )
+ set_category( CAT_AUDIO )
+ set_subcategory( SUBCAT_AUDIO_AOUT )
+ add_bool( "waveout-float32", true, NULL, FLOAT_TEXT, FLOAT_LONGTEXT, true )
+
+ add_string( "waveout-audio-device", "wavemapper", NULL,
+ DEVICE_TEXT, DEVICE_LONG, false )
+ add_deprecated_alias( "waveout-dev" ) /* deprecated since 0.9.3 */
+ change_string_list( ppsz_adev, ppsz_adev_text, ReloadWaveoutDevices )
+ change_need_restart ()
+ change_action_add( ReloadWaveoutDevices, N_("Refresh list") )
+
+
+ set_callbacks( Open, Close )
+vlc_module_end ()
/*****************************************************************************
* aout_sys_t: waveOut audio output method descriptor
*****************************************************************************/
struct aout_sys_t
{
+ uint32_t i_wave_device_id; /* ID of selected output device */
+
HWAVEOUT h_waveout; /* handle to waveout instance */
WAVEFORMATEXTENSIBLE waveformat; /* audio format */
WAVEHDR waveheader[FRAMES_NUM];
+ notification_thread_t *p_notif; /* WaveOutThread id */
+ HANDLE event;
+ HANDLE new_buffer_event;
+
+ // rental from alsa.c to synchronize startup of audiothread
+ int b_playing; /* playing status */
+ mtime_t start_date;
+
+ int i_repeat_counter;
+
int i_buffer_size;
- byte_t *p_silence_buffer; /* buffer we use to play silence */
+ uint8_t *p_silence_buffer; /* buffer we use to play silence */
- vlc_bool_t b_chan_reorder; /* do we need channel reordering */
- int *pi_chan_table;
+ bool b_chan_reorder; /* do we need channel reordering */
+ int pi_chan_table[AOUT_CHAN_MAX];
};
-static const uint32_t pi_channels_in[] =
- { AOUT_CHAN_LEFT, AOUT_CHAN_RIGHT,
- AOUT_CHAN_REARLEFT, AOUT_CHAN_REARRIGHT,
- AOUT_CHAN_CENTER, AOUT_CHAN_LFE };
-static const uint32_t pi_channels_out[] =
- { SPEAKER_FRONT_LEFT, SPEAKER_FRONT_RIGHT,
- SPEAKER_BACK_LEFT, SPEAKER_BACK_RIGHT,
- SPEAKER_FRONT_CENTER, SPEAKER_LOW_FREQUENCY };
-static const uint32_t pi_channels_ordered[] =
- { SPEAKER_FRONT_LEFT, SPEAKER_FRONT_RIGHT, SPEAKER_FRONT_CENTER,
- SPEAKER_LOW_FREQUENCY,
- SPEAKER_BACK_LEFT, SPEAKER_BACK_RIGHT };
-
/*****************************************************************************
* Open: open the audio device
*****************************************************************************
{
aout_instance_t *p_aout = (aout_instance_t *)p_this;
vlc_value_t val;
- int i;
/* Allocate structure */
p_aout->output.p_sys = malloc( sizeof( aout_sys_t ) );
if( p_aout->output.p_sys == NULL )
+ return VLC_ENOMEM;
+
+ p_aout->output.pf_play = Play;
+ p_aout->b_die = false;
+
+
+ /*
+ initialize/update Device selection List
+ */
+ ReloadWaveoutDevices( p_this, "waveout-audio-device", val, val, NULL);
+
+
+ /*
+ check for configured audio device!
+ */
+ char *psz_waveout_dev = var_CreateGetString( p_aout, "waveout-audio-device");
+
+ p_aout->output.p_sys->i_wave_device_id =
+ findDeviceID( psz_waveout_dev );
+
+ if(p_aout->output.p_sys->i_wave_device_id == WAVE_MAPPER)
{
- msg_Err( p_aout, "out of memory" );
- return VLC_EGENERIC;
+ if(psz_waveout_dev &&
+ stricmp(psz_waveout_dev,"wavemapper"))
+ {
+ msg_Warn( p_aout, "configured audio device '%s' not available, "\
+ "use default instead", psz_waveout_dev );
+ }
+ }
+ free( psz_waveout_dev );
+
+
+ WAVEOUTCAPS waveoutcaps;
+ if(waveOutGetDevCaps( p_aout->output.p_sys->i_wave_device_id,
+ &waveoutcaps,
+ sizeof(WAVEOUTCAPS)) == MMSYSERR_NOERROR)
+ {
+ /* log debug some infos about driver, to know who to blame
+ if it doesn't work */
+ msg_Dbg( p_aout, "Drivername: %s", waveoutcaps.szPname);
+ msg_Dbg( p_aout, "Driver Version: %d.%d",
+ (waveoutcaps.vDriverVersion>>8)&255,
+ waveoutcaps.vDriverVersion & 255);
+ msg_Dbg( p_aout, "Manufacturer identifier: 0x%x", waveoutcaps.wMid );
+ msg_Dbg( p_aout, "Product identifier: 0x%x", waveoutcaps.wPid );
}
- p_aout->output.pf_play = Play;
- p_aout->output.p_sys->pi_chan_table = NULL;
- p_aout->b_die = VLC_FALSE;
+
if( var_Type( p_aout, "audio-device" ) == 0 )
{
if( var_Get( p_aout, "audio-device", &val ) < 0 )
{
/* Probe() has failed. */
+ var_Destroy( p_aout, "waveout-audio-device");
free( p_aout->output.p_sys );
return VLC_EGENERIC;
}
+
/* Open the device */
- if( !strcmp( val.psz_string, N_("A/52 over S/PDIF") ) )
+ if( val.i_int == AOUT_VAR_SPDIF )
{
- p_aout->output.output.i_format = VLC_FOURCC('s','p','d','i');
- free( val.psz_string );
+ p_aout->output.output.i_format = VLC_CODEC_SPDIFL;
- if( OpenWaveOut( p_aout, VLC_FOURCC('s','p','d','i'),
+ if( OpenWaveOut( p_aout,
+ p_aout->output.p_sys->i_wave_device_id,
+ VLC_CODEC_SPDIFL,
p_aout->output.output.i_physical_channels,
aout_FormatNbChannels( &p_aout->output.output ),
- p_aout->output.output.i_rate, VLC_FALSE )
+ p_aout->output.output.i_rate, false )
!= VLC_SUCCESS )
{
msg_Err( p_aout, "cannot open waveout audio device" );
}
else
{
- if( !strcmp( val.psz_string, N_("5.1") ) )
+ WAVEOUTCAPS wocaps;
+
+ switch( val.i_int )
{
+ case AOUT_VAR_5_1:
p_aout->output.output.i_physical_channels
- = AOUT_CHAN_LEFT | AOUT_CHAN_RIGHT | AOUT_CHAN_CENTER
- | AOUT_CHAN_REARLEFT | AOUT_CHAN_REARRIGHT
- | AOUT_CHAN_LFE;
- }
- else if( !strcmp( val.psz_string, N_("2 Front 2 Rear") ) )
- {
+ = AOUT_CHAN_LEFT | AOUT_CHAN_RIGHT | AOUT_CHAN_CENTER
+ | AOUT_CHAN_REARLEFT | AOUT_CHAN_REARRIGHT
+ | AOUT_CHAN_LFE;
+ break;
+ case AOUT_VAR_2F2R:
p_aout->output.output.i_physical_channels
- = AOUT_CHAN_LEFT | AOUT_CHAN_RIGHT
- | AOUT_CHAN_REARLEFT | AOUT_CHAN_REARRIGHT;
- }
- else if( !strcmp( val.psz_string, "Mono" ) )
- {
+ = AOUT_CHAN_LEFT | AOUT_CHAN_RIGHT
+ | AOUT_CHAN_REARLEFT | AOUT_CHAN_REARRIGHT;
+ break;
+ case AOUT_VAR_MONO:
p_aout->output.output.i_physical_channels = AOUT_CHAN_CENTER;
- }
- else
- {
+ break;
+ default:
p_aout->output.output.i_physical_channels
- = AOUT_CHAN_LEFT | AOUT_CHAN_RIGHT;
+ = AOUT_CHAN_LEFT | AOUT_CHAN_RIGHT;
}
- free( val.psz_string );
- if( OpenWaveOutPCM( p_aout, &p_aout->output.output.i_format,
+ if( OpenWaveOutPCM( p_aout,
+ p_aout->output.p_sys->i_wave_device_id,
+ &p_aout->output.output.i_format,
p_aout->output.output.i_physical_channels,
aout_FormatNbChannels( &p_aout->output.output ),
- p_aout->output.output.i_rate, VLC_FALSE )
+ p_aout->output.output.i_rate, false )
!= VLC_SUCCESS )
{
msg_Err( p_aout, "cannot open waveout audio device" );
p_aout->output.output.i_bytes_per_frame;
aout_VolumeSoftInit( p_aout );
+
+ /* Check for hardware volume support */
+ if( waveOutGetDevCaps( (UINT_PTR)p_aout->output.p_sys->h_waveout,
+ &wocaps, sizeof(wocaps) ) == MMSYSERR_NOERROR &&
+ wocaps.dwSupport & WAVECAPS_VOLUME )
+ {
+ DWORD i_dummy;
+ if( waveOutGetVolume( p_aout->output.p_sys->h_waveout, &i_dummy )
+ == MMSYSERR_NOERROR )
+ {
+ p_aout->output.pf_volume_get = VolumeGet;
+ p_aout->output.pf_volume_set = VolumeSet;
+ }
+ }
}
if( p_aout->output.p_sys->p_silence_buffer == NULL )
{
free( p_aout->output.p_sys );
- msg_Err( p_aout, "out of memory" );
- return 1;
+ return VLC_ENOMEM;
}
+ p_aout->output.p_sys->i_repeat_counter = 0;
+
/* Zero the buffer. WinCE doesn't have calloc(). */
memset( p_aout->output.p_sys->p_silence_buffer, 0,
p_aout->output.p_sys->i_buffer_size );
+ /* Now we need to setup our waveOut play notification structure */
+ p_aout->output.p_sys->p_notif =
+ vlc_object_create( p_aout, sizeof(notification_thread_t) );
+ p_aout->output.p_sys->p_notif->p_aout = p_aout;
+ p_aout->output.p_sys->event = CreateEvent( NULL, FALSE, FALSE, NULL );
+ p_aout->output.p_sys->new_buffer_event = CreateEvent( NULL, FALSE, FALSE, NULL );
+
+ /* define startpoint of playback on first call to play()
+ like alsa does (instead of playing a blank sample) */
+ p_aout->output.p_sys->b_playing = 0;
+ p_aout->output.p_sys->start_date = 0;
+
+
+ /* Then launch the notification thread */
+ if( vlc_thread_create( p_aout->output.p_sys->p_notif,
+ "waveOut Notification Thread", WaveOutThread,
+ VLC_THREAD_PRIORITY_OUTPUT ) )
+ {
+ msg_Err( p_aout, "cannot create WaveOutThread" );
+ }
+
/* We need to kick off the playback in order to have the callback properly
* working */
- for( i = 0; i < FRAMES_NUM; i++ )
+ for( int i = 0; i < FRAMES_NUM; i++ )
{
p_aout->output.p_sys->waveheader[i].dwFlags = WHDR_DONE;
+ p_aout->output.p_sys->waveheader[i].dwUser = 0;
}
- PlayWaveOut( p_aout, p_aout->output.p_sys->h_waveout,
- &p_aout->output.p_sys->waveheader[0], NULL );
- return 0;
+ return VLC_SUCCESS;
}
/*****************************************************************************
*****************************************************************************/
static void Probe( aout_instance_t * p_aout )
{
- vlc_value_t val;
- int i_format;
+ vlc_value_t val, text;
+ vlc_fourcc_t i_format;
unsigned int i_physical_channels;
- var_Create( p_aout, "audio-device", VLC_VAR_STRING | VLC_VAR_HASCHOICE );
+ 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 );
/* Test for 5.1 support */
i_physical_channels = AOUT_CHAN_LEFT | AOUT_CHAN_RIGHT |
AOUT_CHAN_REARRIGHT | AOUT_CHAN_LFE;
if( p_aout->output.output.i_physical_channels == i_physical_channels )
{
- if( OpenWaveOutPCM( p_aout, &i_format,
+ if( OpenWaveOutPCM( p_aout,
+ p_aout->output.p_sys->i_wave_device_id,
+ &i_format,
i_physical_channels, 6,
- p_aout->output.output.i_rate, VLC_TRUE )
+ p_aout->output.output.i_rate, true )
== VLC_SUCCESS )
{
- val.psz_string = N_("5.1");
- var_Change( p_aout, "audio-device", VLC_VAR_ADDCHOICE, &val );
+ val.i_int = AOUT_VAR_5_1;
+ text.psz_string = (char *)_("5.1");
+ var_Change( p_aout, "audio-device",
+ VLC_VAR_ADDCHOICE, &val, &text );
msg_Dbg( p_aout, "device supports 5.1 channels" );
}
}
if( ( p_aout->output.output.i_physical_channels & i_physical_channels )
== i_physical_channels )
{
- if( OpenWaveOutPCM( p_aout, &i_format,
+ if( OpenWaveOutPCM( p_aout,
+ p_aout->output.p_sys->i_wave_device_id,
+ &i_format,
i_physical_channels, 4,
- p_aout->output.output.i_rate, VLC_TRUE )
+ p_aout->output.output.i_rate, true )
== VLC_SUCCESS )
{
- val.psz_string = N_("2 Front 2 Rear");
- var_Change( p_aout, "audio-device", VLC_VAR_ADDCHOICE, &val );
+ val.i_int = AOUT_VAR_2F2R;
+ text.psz_string = (char *)_("2 Front 2 Rear");
+ var_Change( p_aout, "audio-device",
+ VLC_VAR_ADDCHOICE, &val, &text );
msg_Dbg( p_aout, "device supports 4 channels" );
}
}
/* Test for stereo support */
i_physical_channels = AOUT_CHAN_LEFT | AOUT_CHAN_RIGHT;
- if( OpenWaveOutPCM( p_aout, &i_format,
+ if( OpenWaveOutPCM( p_aout,
+ p_aout->output.p_sys->i_wave_device_id,
+ &i_format,
i_physical_channels, 2,
- p_aout->output.output.i_rate, VLC_TRUE )
+ p_aout->output.output.i_rate, true )
== VLC_SUCCESS )
{
- val.psz_string = N_("Stereo");
- var_Change( p_aout, "audio-device", VLC_VAR_ADDCHOICE, &val );
- var_Set( p_aout, "audio-device", val );
+ val.i_int = AOUT_VAR_STEREO;
+ text.psz_string = (char *)_("Stereo");
+ var_Change( p_aout, "audio-device", VLC_VAR_ADDCHOICE, &val, &text );
msg_Dbg( p_aout, "device supports 2 channels" );
}
/* Test for mono support */
i_physical_channels = AOUT_CHAN_CENTER;
- if( OpenWaveOutPCM( p_aout, &i_format,
+ if( OpenWaveOutPCM( p_aout,
+ p_aout->output.p_sys->i_wave_device_id,
+ &i_format,
i_physical_channels, 1,
- p_aout->output.output.i_rate, VLC_TRUE )
+ p_aout->output.output.i_rate, true )
== VLC_SUCCESS )
{
- val.psz_string = N_("Mono");
- var_Change( p_aout, "audio-device", VLC_VAR_ADDCHOICE, &val );
+ val.i_int = AOUT_VAR_MONO;
+ text.psz_string = (char *)_("Mono");
+ var_Change( p_aout, "audio-device", VLC_VAR_ADDCHOICE, &val, &text );
msg_Dbg( p_aout, "device supports 1 channel" );
}
/* Test for SPDIF support */
if ( AOUT_FMT_NON_LINEAR( &p_aout->output.output ) )
{
- if( OpenWaveOut( p_aout, VLC_FOURCC('s','p','d','i'),
+ if( OpenWaveOut( p_aout,
+ p_aout->output.p_sys->i_wave_device_id,
+ VLC_CODEC_SPDIFL,
p_aout->output.output.i_physical_channels,
aout_FormatNbChannels( &p_aout->output.output ),
- p_aout->output.output.i_rate, VLC_TRUE )
+ p_aout->output.output.i_rate, true )
== VLC_SUCCESS )
{
msg_Dbg( p_aout, "device supports A/52 over S/PDIF" );
- val.psz_string = N_("A/52 over S/PDIF");
- var_Change( p_aout, "audio-device", VLC_VAR_ADDCHOICE, &val );
- if( config_GetInt( p_aout, "spdif" ) )
+ val.i_int = AOUT_VAR_SPDIF;
+ text.psz_string = (char *)_("A/52 over S/PDIF");
+ var_Change( p_aout, "audio-device",
+ VLC_VAR_ADDCHOICE, &val, &text );
+ if( var_InheritBool( p_aout, "spdif" ) )
var_Set( p_aout, "audio-device", val );
}
}
- var_AddCallback( p_aout, "audio-device", aout_ChannelsRestart, NULL );
+ var_Change( p_aout, "audio-device", VLC_VAR_CHOICESCOUNT, &val, NULL );
+ if( val.i_int <= 0 )
+ {
+ /* Probe() has failed. */
+ var_Destroy( p_aout, "audio-device" );
+ return;
+ }
- val.b_bool = VLC_TRUE;
- var_Set( p_aout, "intf-change", val );
+ var_AddCallback( p_aout, "audio-device", aout_ChannelsRestart, NULL );
+ var_SetBool( p_aout, "intf-change", true );
}
/*****************************************************************************
* This doesn't actually play the buffer. This just stores the buffer so it
* can be played by the callback thread.
*****************************************************************************/
-static void Play( aout_instance_t *p_aout )
+static void Play( aout_instance_t *_p_aout )
{
+ if( !_p_aout->output.p_sys->b_playing )
+ {
+ _p_aout->output.p_sys->b_playing = 1;
+
+ /* get the playing date of the first aout buffer */
+ _p_aout->output.p_sys->start_date =
+ aout_FifoFirstDate( _p_aout, &_p_aout->output.fifo );
+
+ msg_Dbg( _p_aout, "Wakeup sleeping output thread.");
+
+ /* wake up the audio output thread */
+ SetEvent( _p_aout->output.p_sys->event );
+ } else {
+ SetEvent( _p_aout->output.p_sys->new_buffer_event );
+ }
}
/*****************************************************************************
static void Close( vlc_object_t *p_this )
{
aout_instance_t *p_aout = (aout_instance_t *)p_this;
+ aout_sys_t *p_sys = p_aout->output.p_sys;
/* Before calling waveOutClose we must reset the device */
- p_aout->b_die = VLC_TRUE;
-
- /* Wait for the waveout buffers to be freed */
- while( VLC_TRUE )
+ vlc_object_kill( p_aout );
+
+ /* wake up the audio thread, to recognize that p_aout died */
+ SetEvent( p_sys->event );
+ SetEvent( p_sys->new_buffer_event );
+
+ vlc_thread_join( p_sys->p_notif );
+ vlc_object_release( p_sys->p_notif );
+
+ /*
+ kill the real output then - when the feed thread
+ is surely terminated!
+ old code could be too early in case that "feeding"
+ was running on termination
+
+ at this point now its sure, that there will be no new
+ data send to the driver, and we can cancel the last
+ running playbuffers
+ */
+ MMRESULT result = waveOutReset( p_sys->h_waveout );
+ if(result != MMSYSERR_NOERROR)
{
- int i;
- vlc_bool_t b_not_done = VLC_FALSE;
-
- for( i = 0; i < FRAMES_NUM; i++ )
- {
- if( !(p_aout->output.p_sys->waveheader[i].dwFlags & WHDR_DONE) )
- b_not_done = VLC_TRUE;
- }
-
- if( !b_not_done )
- break;
-
- msleep( 1000 );
+ msg_Err( p_aout, "waveOutReset failed 0x%x", result );
+ /*
+ now we must wait, that all buffers are played
+ because cancel doesn't work in this case...
+ */
+ if(result == MMSYSERR_NOTSUPPORTED)
+ {
+ /*
+ clear currently played (done) buffers,
+ if returnvalue > 0 (means some buffer still playing)
+ wait for the driver event callback that one buffer
+ is finished with playing, and check again
+ the timeout of 5000ms is just, an emergency exit
+ of this loop, to avoid deadlock in case of other
+ (currently not known bugs, problems, errors cases?)
+ */
+ while(
+ (WaveOutClearDoneBuffers( p_sys ) > 0)
+ &&
+ (WaitForSingleObject( p_sys->event, 5000) == WAIT_OBJECT_0)
+ )
+ {
+ msg_Dbg( p_aout, "Wait for waveout device...");
+ }
+ }
+ } else {
+ WaveOutClearDoneBuffers( p_sys );
}
- waveOutReset( p_aout->output.p_sys->h_waveout );
-
- /* Close the device */
- if( waveOutClose( p_aout->output.p_sys->h_waveout ) != MMSYSERR_NOERROR )
+ /* now we can Close the device */
+ if( waveOutClose( p_sys->h_waveout ) != MMSYSERR_NOERROR )
{
msg_Err( p_aout, "waveOutClose failed" );
}
- free( p_aout->output.p_sys->p_silence_buffer );
- if( p_aout->output.p_sys->pi_chan_table )
- free( p_aout->output.p_sys->pi_chan_table );
- free( p_aout->output.p_sys );
+ /*
+ because so long, the waveout device is playing, the callback
+ could occur and need the events
+ */
+ CloseHandle( p_sys->event );
+ CloseHandle( p_sys->new_buffer_event);
+
+ free( p_sys->p_silence_buffer );
+ free( p_sys );
}
/*****************************************************************************
* OpenWaveOut: open the waveout sound device
****************************************************************************/
-static int OpenWaveOut( aout_instance_t *p_aout, int i_format,
+static int OpenWaveOut( aout_instance_t *p_aout, uint32_t i_device_id, int i_format,
int i_channels, int i_nb_channels, int i_rate,
- vlc_bool_t b_probe )
+ bool b_probe )
{
MMRESULT result;
unsigned int i;
#define waveformat p_aout->output.p_sys->waveformat
waveformat.dwChannelMask = 0;
- for( i = 0; i < sizeof(pi_channels_in)/sizeof(uint32_t); i++ )
+ for( i = 0; i < sizeof(pi_channels_src)/sizeof(uint32_t); i++ )
{
- if( i_channels & pi_channels_in[i] )
- waveformat.dwChannelMask |= pi_channels_out[i];
+ if( i_channels & pi_channels_src[i] )
+ waveformat.dwChannelMask |= pi_channels_in[i];
}
switch( i_format )
{
- case VLC_FOURCC('s','p','d','i'):
+ case VLC_CODEC_SPDIFL:
i_nb_channels = 2;
/* To prevent channel re-ordering */
waveformat.dwChannelMask = SPEAKER_FRONT_LEFT | SPEAKER_FRONT_RIGHT;
waveformat.Samples.wValidBitsPerSample =
waveformat.Format.wBitsPerSample;
waveformat.Format.wFormatTag = WAVE_FORMAT_DOLBY_AC3_SPDIF;
- waveformat.SubFormat = _KSDATAFORMAT_SUBTYPE_DOLBY_AC3_SPDIF;
+ waveformat.SubFormat = __KSDATAFORMAT_SUBTYPE_DOLBY_AC3_SPDIF;
break;
- case VLC_FOURCC('f','l','3','2'):
+ case VLC_CODEC_FL32:
waveformat.Format.wBitsPerSample = sizeof(float) * 8;
waveformat.Samples.wValidBitsPerSample =
waveformat.Format.wBitsPerSample;
waveformat.Format.wFormatTag = WAVE_FORMAT_IEEE_FLOAT;
- waveformat.SubFormat = _KSDATAFORMAT_SUBTYPE_IEEE_FLOAT;
+ waveformat.SubFormat = __KSDATAFORMAT_SUBTYPE_IEEE_FLOAT;
break;
- case VLC_FOURCC('s','1','6','l'):
+ case VLC_CODEC_S16L:
waveformat.Format.wBitsPerSample = 16;
waveformat.Samples.wValidBitsPerSample =
waveformat.Format.wBitsPerSample;
waveformat.Format.wFormatTag = WAVE_FORMAT_PCM;
- waveformat.SubFormat = _KSDATAFORMAT_SUBTYPE_PCM;
+ waveformat.SubFormat = __KSDATAFORMAT_SUBTYPE_PCM;
break;
}
sizeof(WAVEFORMATEXTENSIBLE) - sizeof(WAVEFORMATEX);
}
+ if(!b_probe) {
+ msg_Dbg( p_aout, "OpenWaveDevice-ID: %u", i_device_id);
+ msg_Dbg( p_aout,"waveformat.Format.cbSize = %d",
+ waveformat.Format.cbSize);
+ msg_Dbg( p_aout,"waveformat.Format.wFormatTag = %u",
+ waveformat.Format.wFormatTag);
+ msg_Dbg( p_aout,"waveformat.Format.nChannels = %u",
+ waveformat.Format.nChannels);
+ msg_Dbg( p_aout,"waveformat.Format.nSamplesPerSec = %d",
+ (int)waveformat.Format.nSamplesPerSec);
+ msg_Dbg( p_aout,"waveformat.Format.nAvgBytesPerSec = %u",
+ (int)waveformat.Format.nAvgBytesPerSec);
+ msg_Dbg( p_aout,"waveformat.Format.nBlockAlign = %d",
+ waveformat.Format.nBlockAlign);
+ msg_Dbg( p_aout,"waveformat.Format.wBitsPerSample = %d",
+ waveformat.Format.wBitsPerSample);
+ msg_Dbg( p_aout,"waveformat.Samples.wValidBitsPerSample = %d",
+ waveformat.Samples.wValidBitsPerSample);
+ msg_Dbg( p_aout,"waveformat.Samples.wSamplesPerBlock = %d",
+ waveformat.Samples.wSamplesPerBlock);
+ msg_Dbg( p_aout,"waveformat.dwChannelMask = %lu",
+ waveformat.dwChannelMask);
+ }
+
/* Open the device */
- result = waveOutOpen( &p_aout->output.p_sys->h_waveout, WAVE_MAPPER,
+ result = waveOutOpen( &p_aout->output.p_sys->h_waveout, i_device_id,
(WAVEFORMATEX *)&waveformat,
(DWORD_PTR)WaveOutCallback, (DWORD_PTR)p_aout,
CALLBACK_FUNCTION | (b_probe?WAVE_FORMAT_QUERY:0) );
return VLC_EGENERIC;
}
- CheckReordering( p_aout, i_nb_channels );
+ p_aout->output.p_sys->b_chan_reorder =
+ aout_CheckChannelReorder( pi_channels_in, pi_channels_out,
+ waveformat.dwChannelMask, i_nb_channels,
+ p_aout->output.p_sys->pi_chan_table );
+
+ if( p_aout->output.p_sys->b_chan_reorder )
+ {
+ msg_Dbg( p_aout, "channel reordering needed" );
+ }
return VLC_SUCCESS;
/*****************************************************************************
* OpenWaveOutPCM: open a PCM waveout sound device
****************************************************************************/
-static int OpenWaveOutPCM( aout_instance_t *p_aout, int *i_format,
+static int OpenWaveOutPCM( aout_instance_t *p_aout, uint32_t i_device_id,
+ vlc_fourcc_t *i_format,
int i_channels, int i_nb_channels, int i_rate,
- vlc_bool_t b_probe )
+ bool b_probe )
{
- if( OpenWaveOut( p_aout, VLC_FOURCC('f','l','3','2'),
- i_channels, i_nb_channels, i_rate, b_probe )
+ bool b_use_float32 = var_CreateGetBool( p_aout, "waveout-float32");
+
+ if( !b_use_float32 || OpenWaveOut( p_aout, i_device_id, VLC_CODEC_FL32,
+ i_channels, i_nb_channels, i_rate, b_probe )
!= VLC_SUCCESS )
{
- if ( OpenWaveOut( p_aout, VLC_FOURCC('s','1','6','l'),
+ if ( OpenWaveOut( p_aout, i_device_id, VLC_CODEC_S16L,
i_channels, i_nb_channels, i_rate, b_probe )
!= VLC_SUCCESS )
{
}
else
{
- *i_format = VLC_FOURCC('s','1','6','l');
+ *i_format = VLC_CODEC_S16L;
return VLC_SUCCESS;
}
}
else
{
- *i_format = VLC_FOURCC('f','l','3','2');
+ *i_format = VLC_CODEC_FL32;
return VLC_SUCCESS;
}
}
-/*****************************************************************************
- * CheckReordering: Check if we need to do some channel re-ordering (the ac3
- * channel order is different from the one chosen by
- * Microsoft).
- *****************************************************************************/
-static void CheckReordering( aout_instance_t *p_aout, int i_nb_channels )
-{
- int i, j, k, l;
-
-#define waveformat p_aout->output.p_sys->waveformat
-#define pi_chan_table p_aout->output.p_sys->pi_chan_table
-
- p_aout->output.p_sys->b_chan_reorder = VLC_FALSE;
-
- pi_chan_table = malloc( i_nb_channels * sizeof(int) );
- if( !pi_chan_table )
- {
- return;
- }
-
- for( i = 0, j = 0;
- i < (int)(sizeof(pi_channels_out)/sizeof(uint32_t)); i++ )
- {
- if( waveformat.dwChannelMask & pi_channels_out[i] )
- {
- for( k = 0, l = 0;
- pi_channels_out[i] != pi_channels_ordered[k]; k++ )
- {
- if( waveformat.dwChannelMask & pi_channels_ordered[k] )
- {
- l++;
- }
- }
-
- pi_chan_table[j] = l;
-
- j++;
- }
- }
-
- for( i = 0; i < i_nb_channels; i++ )
- {
- if( pi_chan_table[i] != i )
- {
- p_aout->output.p_sys->b_chan_reorder = VLC_TRUE;
- }
- }
-
- if( p_aout->output.p_sys->b_chan_reorder )
- {
- msg_Dbg( p_aout, "channel reordering needed" );
- }
-
-#undef pi_chan_table
-#undef waveformat
-}
-
/*****************************************************************************
* PlayWaveOut: play a buffer through the WaveOut device
*****************************************************************************/
static int PlayWaveOut( aout_instance_t *p_aout, HWAVEOUT h_waveout,
- WAVEHDR *p_waveheader, aout_buffer_t *p_buffer )
+ WAVEHDR *p_waveheader, aout_buffer_t *p_buffer,
+ bool b_spdif)
{
MMRESULT result;
/* Prepare the buffer */
if( p_buffer != NULL )
+ {
p_waveheader->lpData = p_buffer->p_buffer;
- else
+ /*
+ copy the buffer to the silence buffer :) so in case we don't
+ get the next buffer fast enough (I will repeat this one a time
+ for AC3 / DTS and SPDIF this will sound better instead of
+ a hickup)
+ */
+ if(b_spdif)
+ {
+ vlc_memcpy( p_aout->output.p_sys->p_silence_buffer,
+ p_buffer->p_buffer,
+ p_aout->output.p_sys->i_buffer_size );
+ p_aout->output.p_sys->i_repeat_counter = 2;
+ }
+ } else {
/* Use silence buffer instead */
+ if(p_aout->output.p_sys->i_repeat_counter)
+ {
+ p_aout->output.p_sys->i_repeat_counter--;
+ if(!p_aout->output.p_sys->i_repeat_counter)
+ {
+ vlc_memset( p_aout->output.p_sys->p_silence_buffer,
+ 0x00, p_aout->output.p_sys->i_buffer_size );
+ }
+ }
p_waveheader->lpData = p_aout->output.p_sys->p_silence_buffer;
+ }
- p_waveheader->dwUser = (DWORD_PTR)p_buffer;
+ p_waveheader->dwUser = p_buffer ? (DWORD_PTR)p_buffer : (DWORD_PTR)1;
p_waveheader->dwBufferLength = p_aout->output.p_sys->i_buffer_size;
p_waveheader->dwFlags = 0;
DWORD _p_aout,
DWORD dwParam1, DWORD dwParam2 )
{
+ (void)h_waveout; (void)dwParam1; (void)dwParam2;
aout_instance_t *p_aout = (aout_instance_t *)_p_aout;
- WAVEHDR *p_waveheader = (WAVEHDR *)dwParam1;
- aout_buffer_t *p_buffer = NULL;
- vlc_bool_t b_sleek;
int i, i_queued_frames = 0;
if( uMsg != WOM_DONE ) return;
- /* Unprepare and free the buffer which has just been played */
- waveOutUnprepareHeader( h_waveout, p_waveheader, sizeof(WAVEHDR) );
- if( p_waveheader->dwUser )
- aout_BufferFree( (aout_buffer_t *)p_waveheader->dwUser );
-
- if( p_aout->b_die ) return;
-
- /* We don't want any resampling when using S/PDIF */
- b_sleek = p_aout->output.output.i_format == VLC_FOURCC('s','p','d','i');
+ if( !vlc_object_alive (p_aout) ) return;
/* Find out the current latency */
for( i = 0; i < FRAMES_NUM; i++ )
}
}
- /* Try to fill in as many frame buffers as possible */
- for( i = 0; i < FRAMES_NUM; i++ )
+ /* Don't wake up the thread too much */
+ if( i_queued_frames <= FRAMES_NUM/2 )
+ SetEvent( p_aout->output.p_sys->event );
+}
+
+
+/****************************************************************************
+ * WaveOutClearDoneBuffers: Clear all done marked buffers, and free aout_bufer
+ ****************************************************************************
+ * return value is the number of still playing buffers in the queue
+ ****************************************************************************/
+static int WaveOutClearDoneBuffers(aout_sys_t *p_sys)
+{
+ WAVEHDR *p_waveheader = p_sys->waveheader;
+ int i_queued_frames = 0;
+
+ for( int i = 0; i < FRAMES_NUM; i++ )
{
- /* Check if frame buf is available */
- if( p_aout->output.p_sys->waveheader[i].dwFlags & WHDR_DONE )
+ if( (p_waveheader[i].dwFlags & WHDR_DONE) &&
+ p_waveheader[i].dwUser )
{
- /* Take into account the latency */
- p_buffer = aout_OutputNextBuffer( p_aout,
- mdate() + 1000000 * i_queued_frames /
- p_aout->output.output.i_rate * p_aout->output.i_nb_samples,
- b_sleek );
+ aout_buffer_t *p_buffer =
+ (aout_buffer_t *)(p_waveheader[i].dwUser);
+ /* Unprepare and free the buffers which has just been played */
+ waveOutUnprepareHeader( p_sys->h_waveout, &p_waveheader[i],
+ sizeof(WAVEHDR) );
- if( !p_buffer && i_queued_frames )
- {
- /* We aren't late so no need to play a blank sample */
- return;
- }
-
- /* Do the channel reordering here */
- if( p_buffer && p_aout->output.p_sys->b_chan_reorder )
- {
- if( p_aout->output.output.i_format ==
- VLC_FOURCC('s','1','6','l') )
- InterleaveS16( (int16_t *)p_buffer->p_buffer,
- p_aout->output.p_sys->pi_chan_table,
- aout_FormatNbChannels( &p_aout->output.output ) );
- else
- InterleaveFloat32( (float *)p_buffer->p_buffer,
- p_aout->output.p_sys->pi_chan_table,
- aout_FormatNbChannels( &p_aout->output.output ) );
- }
+ if( p_waveheader[i].dwUser != 1 )
+ aout_BufferFree( p_buffer );
- PlayWaveOut( p_aout, h_waveout,
- &p_aout->output.p_sys->waveheader[i] , p_buffer );
+ p_waveheader[i].dwUser = 0;
+ }
+ /* Check if frame buf is available */
+ if( !(p_waveheader[i].dwFlags & WHDR_DONE) )
+ {
i_queued_frames++;
}
}
+ return i_queued_frames;
}
/*****************************************************************************
- * InterleaveFloat32/S16: change the channel order to the Microsoft one.
+ * WaveOutThread: this thread will capture play notification events.
+ *****************************************************************************
+ * We use this thread to feed new audio samples to the sound card because
+ * we are not authorized to use waveOutWrite() directly in the waveout
+ * callback.
*****************************************************************************/
-static void InterleaveFloat32( float *p_buf, int *pi_chan_table,
- int i_nb_channels )
+static void* WaveOutThread( vlc_object_t *p_this )
{
- int i, j;
- float p_tmp[10];
+ notification_thread_t *p_notif = (notification_thread_t*)p_this;
+ aout_instance_t *p_aout = p_notif->p_aout;
+ aout_sys_t *p_sys = p_aout->output.p_sys;
+ aout_buffer_t *p_buffer = NULL;
+ WAVEHDR *p_waveheader = p_sys->waveheader;
+ int i, i_queued_frames;
+ bool b_sleek;
+ mtime_t next_date;
+ uint32_t i_buffer_length = 64;
+ int canc = vlc_savecancel ();
+
+ /* We don't want any resampling when using S/PDIF */
+ b_sleek = p_aout->output.output.i_format == VLC_CODEC_SPDIFL;
+
+ // wait for first call to "play()"
+ while( !p_sys->start_date && vlc_object_alive (p_aout) )
+ WaitForSingleObject( p_sys->event, INFINITE );
+ if( !vlc_object_alive (p_aout) )
+ return NULL;
+
+ msg_Dbg( p_aout, "will start to play in %"PRId64" us",
+ (p_sys->start_date - AOUT_PTS_TOLERANCE/4)-mdate());
+
+ // than wait a short time... before grabbing first frames
+ mwait( p_sys->start_date - AOUT_PTS_TOLERANCE/4 );
+
+#define waveout_warn(msg) msg_Warn( p_aout, "aout_OutputNextBuffer no buffer "\
+ "got next_date=%d ms, "\
+ "%d frames to play, "\
+ "starving? %d, %s",(int)(next_date/(mtime_t)1000), \
+ i_queued_frames, \
+ p_aout->output.b_starving, msg);
+ next_date = mdate();
- for( i = 0; i < FRAME_SIZE; i++ )
+ while( vlc_object_alive (p_aout) )
{
- for( j = 0; j < i_nb_channels; j++ )
+ /* Cleanup and find out the current latency */
+ i_queued_frames = WaveOutClearDoneBuffers( p_sys );
+
+ if( !vlc_object_alive (p_aout) ) return NULL;
+
+ /* Try to fill in as many frame buffers as possible */
+ for( i = 0; i < FRAMES_NUM; i++ )
{
- p_tmp[pi_chan_table[j]] = p_buf[i*i_nb_channels + j];
+ /* Check if frame buf is available */
+ if( p_waveheader[i].dwFlags & WHDR_DONE )
+ {
+ // next_date = mdate() + 1000000 * i_queued_frames /
+ // p_aout->output.output.i_rate * p_aout->output.i_nb_samples;
+
+ // the realtime has got our back-site:) to come in sync
+ if(next_date < mdate())
+ next_date = mdate();
+
+
+ /* Take into account the latency */
+ p_buffer = aout_OutputNextBuffer( p_aout,
+ next_date,
+ b_sleek );
+
+ if(!p_buffer)
+ {
+#if 0
+ msg_Dbg( p_aout, "aout_OutputNextBuffer no buffer "\
+ "got next_date=%d ms, "\
+ "%d frames to play, "\
+ "starving? %d",(int)(next_date/(mtime_t)1000),
+ i_queued_frames,
+ p_aout->output.b_starving);
+#endif
+ if(p_aout->output.b_starving)
+ {
+ // means we are too early to request a new buffer?
+ waveout_warn("waiting...")
+ next_date = aout_FifoFirstDate( p_aout, &p_aout->output.fifo );
+ mwait( next_date - AOUT_PTS_TOLERANCE/4 );
+ next_date = mdate();
+ p_buffer = aout_OutputNextBuffer( p_aout,
+ next_date,
+ b_sleek
+ );
+ }
+ }
+
+ if( !p_buffer && i_queued_frames )
+ {
+ /* We aren't late so no need to play a blank sample */
+ break;
+ }
+
+ if( p_buffer )
+ {
+ mtime_t buffer_length = p_buffer->i_length;
+ next_date = next_date + buffer_length;
+ i_buffer_length = buffer_length/1000;
+ }
+
+ /* Do the channel reordering */
+ if( p_buffer && p_sys->b_chan_reorder )
+ {
+ aout_ChannelReorder( p_buffer->p_buffer,
+ p_buffer->i_buffer,
+ p_sys->waveformat.Format.nChannels,
+ p_sys->pi_chan_table,
+ p_sys->waveformat.Format.wBitsPerSample );
+ }
+
+ PlayWaveOut( p_aout, p_sys->h_waveout,
+ &p_waveheader[i], p_buffer, b_sleek );
+
+ i_queued_frames++;
+ }
}
- memcpy( &p_buf[i*i_nb_channels], p_tmp,
- i_nb_channels * sizeof(float) );
+ if( !vlc_object_alive (p_aout) ) return NULL;
+
+ /*
+ deal with the case that the loop didn't fillup the buffer to the
+ max - instead of waiting that half the buffer is played before
+ fillup the waveout buffers, wait only for the next sample buffer
+ to arrive at the play method...
+
+ this will also avoid, that the last buffer is play until the
+ end, and then trying to get more data, so it will also
+ work - if the next buffer will arrive some ms before the
+ last buffer is finished.
+ */
+ if(i_queued_frames < FRAMES_NUM)
+ WaitForSingleObject( p_sys->new_buffer_event, INFINITE );
+ else
+ WaitForSingleObject( p_sys->event, INFINITE );
+
}
+
+#undef waveout_warn
+ vlc_restorecancel (canc);
+ return NULL;
}
-static void InterleaveS16( int16_t *p_buf, int *pi_chan_table,
- int i_nb_channels )
+static int VolumeGet( aout_instance_t * p_aout, audio_volume_t * pi_volume )
{
- int i, j;
- int16_t p_tmp[10];
+ DWORD i_waveout_vol;
+
+#ifdef UNDER_CE
+ waveOutGetVolume( 0, &i_waveout_vol );
+#else
+ waveOutGetVolume( p_aout->output.p_sys->h_waveout, &i_waveout_vol );
+#endif
- for( i = 0; i < FRAME_SIZE; i++ )
+ i_waveout_vol &= 0xFFFF;
+ *pi_volume = p_aout->output.i_volume =
+ (i_waveout_vol * AOUT_VOLUME_MAX + 0xFFFF /*rounding*/) / 2 / 0xFFFF;
+ return 0;
+}
+
+static int VolumeSet( aout_instance_t * p_aout, audio_volume_t i_volume )
+{
+ unsigned long i_waveout_vol = i_volume * 0xFFFF * 2 / AOUT_VOLUME_MAX;
+ i_waveout_vol |= (i_waveout_vol << 16);
+
+#ifdef UNDER_CE
+ waveOutSetVolume( 0, i_waveout_vol );
+#else
+ waveOutSetVolume( p_aout->output.p_sys->h_waveout, i_waveout_vol );
+#endif
+
+ p_aout->output.i_volume = i_volume;
+ return 0;
+}
+
+
+/*
+ reload the configuration drop down list, of the Audio Devices
+*/
+static int ReloadWaveoutDevices( vlc_object_t *p_this, char const *psz_name,
+ vlc_value_t newval, vlc_value_t oldval, void *data )
+{
+ VLC_UNUSED( newval ); VLC_UNUSED( oldval ); VLC_UNUSED( data );
+
+ module_config_t *p_item = config_FindConfig( p_this, psz_name );
+ if( !p_item ) return VLC_SUCCESS;
+
+ /* Clear-up the current list */
+ if( p_item->i_list )
+ {
+ int i;
+
+ /* Keep the first entry */
+ for( i = 1; i < p_item->i_list; i++ )
+ {
+ free((char *)(p_item->ppsz_list[i]) );
+ free((char *)(p_item->ppsz_list_text[i]) );
+ }
+ /* TODO: Remove when no more needed */
+ p_item->ppsz_list[i] = NULL;
+ p_item->ppsz_list_text[i] = NULL;
+ }
+ p_item->i_list = 1;
+
+ int wave_devices = waveOutGetNumDevs();
+
+ p_item->ppsz_list = xrealloc( p_item->ppsz_list,
+ (wave_devices+2) * sizeof(char *) );
+ p_item->ppsz_list_text = xrealloc( p_item->ppsz_list_text,
+ (wave_devices+2) * sizeof(char *) );
+
+ WAVEOUTCAPS caps;
+ char sz_dev_name[MAXPNAMELEN+32];
+ int j=1;
+ for(int i=0; i<wave_devices; i++)
{
- for( j = 0; j < i_nb_channels; j++ )
+ if(waveOutGetDevCaps(i, &caps, sizeof(WAVEOUTCAPS))
+ == MMSYSERR_NOERROR)
{
- p_tmp[pi_chan_table[j]] = p_buf[i*i_nb_channels + j];
+ sprintf( sz_dev_name, psz_device_name_fmt, caps.szPname,
+ caps.wMid,
+ caps.wPid
+ );
+ p_item->ppsz_list[j] = FromLocaleDup( sz_dev_name );
+ p_item->ppsz_list_text[j] = FromLocaleDup( sz_dev_name );
+ p_item->i_list++;
+ j++;
}
- memcpy( &p_buf[i*i_nb_channels], p_tmp,
- i_nb_channels * sizeof(int16_t) );
}
+ p_item->ppsz_list[j] = NULL;
+ p_item->ppsz_list_text[j] = NULL;
+
+ /* Signal change to the interface */
+ p_item->b_dirty = true;
+
+ return VLC_SUCCESS;
+}
+
+/*
+ convert devicename to device ID for output
+ if device not found return WAVE_MAPPER, so let
+ windows decide which preferred audio device
+ should be used.
+*/
+static uint32_t findDeviceID(char *psz_device_name)
+{
+ if(!psz_device_name)
+ return WAVE_MAPPER;
+
+ uint32_t wave_devices = waveOutGetNumDevs();
+ WAVEOUTCAPS caps;
+ char sz_dev_name[MAXPNAMELEN+32];
+ for(uint32_t i=0; i<wave_devices; i++)
+ {
+ if(waveOutGetDevCaps(i, &caps, sizeof(WAVEOUTCAPS))
+ == MMSYSERR_NOERROR)
+ {
+ sprintf(sz_dev_name, psz_device_name_fmt, caps.szPname,
+ caps.wMid,
+ caps.wPid
+ );
+ char *psz_temp = FromLocaleDup(sz_dev_name);
+
+ if( !stricmp(psz_temp, psz_device_name) )
+ {
+ LocaleFree( psz_temp );
+ return i;
+ }
+ LocaleFree( psz_temp );
+ }
+ }
+
+ return WAVE_MAPPER;
}