aboutsummaryrefslogtreecommitdiff
path: root/pd/src/s_audio_alsa.c
diff options
context:
space:
mode:
authorMiller Puckette <millerpuckette@users.sourceforge.net>2004-09-06 20:20:36 +0000
committerMiller Puckette <millerpuckette@users.sourceforge.net>2004-09-06 20:20:36 +0000
commited932acb5860bf8b9296169676499562a55d139e (patch)
treedc6a40dba908deb07c175cd40ee19c197318f72d /pd/src/s_audio_alsa.c
parentdad636821f6e7d3ead02c157f308c0ceeba9af3d (diff)
checking in version 0.38test5.
Oops, I realize I forgot some more nice files, will add them and re-commit. svn path=/trunk/; revision=2010
Diffstat (limited to 'pd/src/s_audio_alsa.c')
-rw-r--r--pd/src/s_audio_alsa.c1298
1 files changed, 578 insertions, 720 deletions
diff --git a/pd/src/s_audio_alsa.c b/pd/src/s_audio_alsa.c
index f42b6436..10577ad5 100644
--- a/pd/src/s_audio_alsa.c
+++ b/pd/src/s_audio_alsa.c
@@ -5,6 +5,9 @@
/* this file inputs and outputs audio using the ALSA API available on linux. */
+/* support for ALSA pcmv2 api by Karl MacMillan<karlmac@peabody.jhu.edu> */
+/* support for ALSA MMAP noninterleaved by Winfried Ritsch, IEM */
+
#include <alsa/asoundlib.h>
#include "m_pd.h"
@@ -21,729 +24,536 @@
#include <fcntl.h>
#include <sched.h>
#include <sys/mman.h>
-
-typedef int16_t t_alsa_sample16;
-typedef int32_t t_alsa_sample32;
-#define ALSA_SAMPLEWIDTH_16 sizeof(t_alsa_sample16)
-#define ALSA_SAMPLEWIDTH_32 sizeof(t_alsa_sample32)
-#define ALSA_XFERSIZE16 (signed int)(sizeof(t_alsa_sample16) * DEFDACBLKSIZE)
-#define ALSA_XFERSIZE32 (signed int)(sizeof(t_alsa_sample32) * DEFDACBLKSIZE)
-#define ALSA_MAXDEV 1
-#define ALSA_JITTER 1024
-#define ALSA_EXTRABUFFER 2048
-#define ALSA_DEFFRAGSIZE 64
-#define ALSA_DEFNFRAG 12
-
-#ifndef INT32_MAX
-#define INT32_MAX 0x7fffffff
-#endif
-
-#if (SND_LIB_MAJOR < 1)
-#define ALSAAPI9
-#endif
-
-typedef struct _alsa_dev
-{
- snd_pcm_t *inhandle;
- snd_pcm_t *outhandle;
- int innoninterleave; /* true if we're set for noninterleaved read */
- int outnoninterleave; /* same for write */
-} t_alsa_dev;
-
-t_alsa_dev alsa_device;
-static void *alsa_snd_buf = 0;
-static void **alsa_buf_ptrs;
-static int alsa_samplewidth;
-static snd_pcm_status_t* in_status;
-static snd_pcm_status_t* out_status;
-
-static int alsa_mode;
-static int alsa_buf_samps; /* believed actual ALSA bufsize in sample frames */
-static int alsa_inchannels;
-static int alsa_outchannels;
+#include "s_audio_alsa.h"
/* Defines */
#define DEBUG(x) x
#define DEBUG2(x) {x;}
static void alsa_checkiosync( void);
-static void alsa_numbertoname(int devno, char *devname, int nchar);
+static void alsa_numbertoname(int iodev, char *devname, int nchar);
/* don't assume we can turn all 31 bits when doing float-to-fix;
otherwise some audio drivers (e.g. Midiman/ALSA) wrap around. */
#define FMAX 0x7ffff000
#define CLIP32(x) (((x)>FMAX)?FMAX:((x) < -FMAX)?-FMAX:(x))
-/* support for ALSA pcmv2 api by Karl MacMillan<karlmac@peabody.jhu.edu> */
+static char *alsa_snd_buf;
+static int alsa_snd_bufsize;
+static int alsa_buf_samps;
+static snd_pcm_status_t *alsa_status;
+static int alsa_usemmap;
+
+t_alsa_dev alsa_indev[ALSA_MAXDEV];
+t_alsa_dev alsa_outdev[ALSA_MAXDEV];
+int alsa_nindev;
+int alsa_noutdev;
static void check_error(int err, const char *why)
{
if (err < 0)
- fprintf(stderr, "%s: %s\n", why, snd_strerror(err));
+ fprintf(stderr, "%s: %s\n", why, snd_strerror(err));
+}
+
+static int alsaio_canmmap(t_alsa_dev *dev)
+{
+ snd_pcm_hw_params_t *hw_params;
+ int err;
+
+ snd_pcm_hw_params_alloca(&hw_params);
+
+ err = snd_pcm_hw_params_any(dev->a_handle, hw_params);
+ if (err < 0) {
+ check_error(err,"Broken configuration: no configurations available");
+ return (0);
+ }
+
+ err = snd_pcm_hw_params_set_access(dev->a_handle,
+ hw_params, SND_PCM_ACCESS_MMAP_NONINTERLEAVED);
+ return (err >= 0);
+}
+
+static int alsaio_setup(t_alsa_dev *dev, int out, int *channels, int *rate,
+ int nfrags, int frag_size)
+{
+ int bufsizeforthis, err;
+ snd_pcm_hw_params_t* hw_params;
+ unsigned int tmp_uint;
+ snd_pcm_uframes_t tmp_snd_pcm_uframes;
+
+ if (sys_verbose)
+ {
+ if (out)
+ post("configuring sound output...");
+ else post("configuring sound input...");
+ }
+
+ /* set hardware parameters... */
+ snd_pcm_hw_params_alloca(&hw_params);
+
+ /* get the default params */
+ err = snd_pcm_hw_params_any(dev->a_handle, hw_params);
+ check_error(err, "snd_pcm_hw_params_any");
+
+ /* try to set interleaved access */
+ err = snd_pcm_hw_params_set_access(dev->a_handle,
+ hw_params, SND_PCM_ACCESS_RW_INTERLEAVED);
+ if (err < 0)
+ return (-1);
+ check_error(err, "snd_pcm_hw_params_set_access");
+ /* Try to set 32 bit format first */
+ err = snd_pcm_hw_params_set_format(dev->a_handle,
+ hw_params, SND_PCM_FORMAT_S32);
+ if (err < 0)
+ {
+ /* fprintf(stderr,
+ "PD-ALSA: 32 bit format not available - using 16\n"); */
+ err = snd_pcm_hw_params_set_format(dev->a_handle, hw_params,
+ SND_PCM_FORMAT_S16);
+ check_error(err, "snd_pcm_hw_params_set_format");
+ dev->a_sampwidth = 2;
+ }
+ else dev->a_sampwidth = 4;
+
+ if (sys_verbose)
+ post("Sample width set to %d bytes", dev->a_sampwidth);
+
+ /* set the subformat */
+ err = snd_pcm_hw_params_set_subformat(dev->a_handle,
+ hw_params, SND_PCM_SUBFORMAT_STD);
+ check_error(err, "snd_pcm_hw_params_set_subformat");
+
+ /* set the number of channels */
+ tmp_uint = *channels;
+ err = snd_pcm_hw_params_set_channels_min(dev->a_handle,
+ hw_params, &tmp_uint);
+ check_error(err, "snd_pcm_hw_params_set_channels");
+ if (tmp_uint != (unsigned)*channels)
+ post("ALSA: set input channels to %d", tmp_uint);
+ *channels = tmp_uint;
+ dev->a_channels = *channels;
+
+ /* set the sampling rate */
+ err = snd_pcm_hw_params_set_rate_min(dev->a_handle, hw_params, rate, 0);
+ check_error(err, "snd_pcm_hw_params_set_rate_min (input)");
+#if 0
+ err = snd_pcm_hw_params_get_rate(hw_params, &subunitdir);
+ post("input sample rate %d", err);
+#endif
+
+ /* set the period - ie frag size */
+ /* LATER try this to get a recommended period size...
+ right now, it trips an assertion failure in ALSA lib */
+ tmp_snd_pcm_uframes = frag_size;
+ err = snd_pcm_hw_params_set_period_size_near(dev->a_handle,
+ hw_params, &tmp_snd_pcm_uframes, 0);
+ check_error(err, "snd_pcm_hw_params_set_period_size_near (input)");
+
+ /* set the number of periods - ie numfrags */
+ tmp_uint = nfrags;
+ err = snd_pcm_hw_params_set_periods_near(dev->a_handle,
+ hw_params, &tmp_uint, 0);
+ check_error(err, "snd_pcm_hw_params_set_periods_near (input)");
+
+ /* set the buffer size */
+ tmp_snd_pcm_uframes = nfrags * frag_size;
+ err = snd_pcm_hw_params_set_buffer_size_near(dev->a_handle,
+ hw_params, &tmp_snd_pcm_uframes);
+ check_error(err, "snd_pcm_hw_params_set_buffer_size_near (input)");
+
+ err = snd_pcm_hw_params(dev->a_handle, hw_params);
+ check_error(err, "snd_pcm_hw_params (input)");
+
+ /* set up the buffer */
+ bufsizeforthis = DEFDACBLKSIZE * dev->a_sampwidth * *channels;
+ if (alsa_snd_buf)
+ {
+ if (alsa_snd_bufsize < bufsizeforthis)
+ {
+ if (!(alsa_snd_buf = realloc(alsa_snd_buf, bufsizeforthis)))
+ {
+ post("out of memory");
+ return (0);
+ }
+ memset(alsa_snd_buf, 0, bufsizeforthis);
+ alsa_snd_bufsize = bufsizeforthis;
+ }
+ }
+ else
+ {
+ if (!(alsa_snd_buf = (void *)malloc(bufsizeforthis)))
+ {
+ post("out of memory");
+ return (0);
+ }
+ memset(alsa_snd_buf, 0, bufsizeforthis);
+ alsa_snd_bufsize = bufsizeforthis;
+ }
+ return (1);
}
-/* was: alsa_open_audio(int wantinchans, int wantoutchans, int srate) */
+ /* return 0 on success */
int alsa_open_audio(int naudioindev, int *audioindev, int nchindev,
int *chindev, int naudiooutdev, int *audiooutdev, int nchoutdev,
int *choutdev, int rate)
{
int err, inchans = 0, outchans = 0, subunitdir;
char devname[512];
- snd_pcm_hw_params_t* hw_params;
- snd_pcm_sw_params_t* sw_params;
snd_output_t* out;
int frag_size = (sys_blocksize ? sys_blocksize : ALSA_DEFFRAGSIZE);
- int nfrags, i;
- short* tmp_buf;
- unsigned int tmp_uint;
- snd_pcm_uframes_t tmp_snd_pcm_uframes;
- int wantinchans, wantoutchans, devno;
-
- if (naudioindev >= 2 || naudiooutdev >= 2)
- post("alsa: only one input and output device allowed (extras ignored");
- if (naudioindev >= 1 && naudiooutdev >= 1 &&
- audioindev[0] != audiooutdev[0])
- post("alsa: changing output device to agree with input device");
- if (nchindev)
- wantinchans = chindev[0];
- else wantinchans = (naudioindev ? 2 : 0);
- if (nchoutdev)
- wantoutchans = choutdev[0];
- else wantoutchans = (naudiooutdev ? 2 : 0);
- devno = (naudioindev > 0 ? audioindev[0] :
- (naudiooutdev > 0 ? audiooutdev[0] : 0));
-
- alsa_numbertoname(devno, devname, 512);
+ int nfrags, i, iodev, dev2;
+ int wantinchans, wantoutchans, device;
- if (sys_verbose)
- post("device name %s; channels in %d, out %d", devname, wantinchans,
- wantoutchans);
-
nfrags = sys_schedadvance * (float)rate / (1e6 * frag_size);
- /* save our belief as to ALSA's buffer size for later */
+ /* save our belief as to ALSA's buffer size for later */
alsa_buf_samps = nfrags * frag_size;
+ alsa_nindev = alsa_noutdev = 0;
if (sys_verbose)
- post("audio buffer set to %d", (int)(0.001 * sys_schedadvance));
-
- alsa_device.innoninterleave = alsa_device.outnoninterleave = 0;
- if (wantinchans)
- {
- err = snd_pcm_open(&alsa_device.inhandle, devname,
- SND_PCM_STREAM_CAPTURE, SND_PCM_NONBLOCK);
-
- check_error(err, "snd_pcm_open (input)");
- if (err < 0)
- inchans = 0;
- else
- {
- inchans = wantinchans;
- snd_pcm_nonblock(alsa_device.inhandle, 1);
- }
- }
- if (wantoutchans)
- {
- err = snd_pcm_open(&alsa_device.outhandle, devname,
- SND_PCM_STREAM_PLAYBACK, SND_PCM_NONBLOCK);
-
- check_error(err, "snd_pcm_open (output)");
- if (err < 0)
- outchans = 0;
- else
- {
- outchans = wantoutchans;
- snd_pcm_nonblock(alsa_device.outhandle, 1);
- }
- }
- if (inchans)
- {
- if (sys_verbose)
- post("opening sound input...");
- err = snd_pcm_hw_params_malloc(&hw_params);
- check_error(err, "snd_pcm_hw_params_malloc (input)");
-
- // get the default params
- err = snd_pcm_hw_params_any(alsa_device.inhandle, hw_params);
- check_error(err, "snd_pcm_hw_params_any (input)");
-
- /* try to set interleaved access */
- err = snd_pcm_hw_params_set_access(alsa_device.inhandle,
- hw_params, SND_PCM_ACCESS_RW_INTERLEAVED);
- if (err < 0)
- {
- /* OK, so try non-interleaved */
- err = snd_pcm_hw_params_set_access(alsa_device.inhandle,
- hw_params, SND_PCM_ACCESS_RW_NONINTERLEAVED);
- if (err >= 0)
- {
- post("using non-interleaved audio input");
- alsa_device.innoninterleave = 1;
- }
- }
- check_error(err, "snd_pcm_hw_params_set_access (input)");
- // Try to set 32 bit format first
- err = snd_pcm_hw_params_set_format(alsa_device.inhandle, hw_params,
- SND_PCM_FORMAT_S32);
- if (err < 0)
- {
- /* fprintf(stderr,
- "PD-ALSA: 32 bit format not available - using 16\n"); */
- err = snd_pcm_hw_params_set_format(alsa_device.inhandle, hw_params,
- SND_PCM_FORMAT_S16);
- check_error(err, "snd_pcm_hw_params_set_format (input)");
- alsa_samplewidth = 2;
- }
- else
- {
- alsa_samplewidth = 4;
- }
- post("Sample width set to %d bytes", alsa_samplewidth);
- // set the subformat
- err = snd_pcm_hw_params_set_subformat(alsa_device.inhandle, hw_params,
- SND_PCM_SUBFORMAT_STD);
- check_error(err, "snd_pcm_hw_params_set_subformat (input)");
- // set the number of channels
- tmp_uint = inchans;
- err = snd_pcm_hw_params_set_channels_min(alsa_device.inhandle,
- hw_params, &tmp_uint);
- check_error(err, "snd_pcm_hw_params_set_channels (input)");
- if (tmp_uint != (unsigned)inchans)
- post("ALSA: set input channels to %d", tmp_uint);
- inchans = tmp_uint;
- // set the sampling rate
- err = snd_pcm_hw_params_set_rate_min(alsa_device.inhandle, hw_params,
- &rate, 0);
- check_error(err, "snd_pcm_hw_params_set_rate_min (input)");
-#if 0
- err = snd_pcm_hw_params_get_rate(hw_params, &subunitdir);
- post("input sample rate %d", err);
-#endif
- // set the period - ie frag size
- // post("fragsize a %d", frag_size);
+ post("audio buffer set to %d", (int)(0.001 * sys_schedadvance));
- /* LATER try this to get a recommended period size...
- right now, it trips an assertion failure in ALSA lib */
-#if 0
- post("input period was %d, min %d, max %d\n",
- snd_pcm_hw_params_get_period_size(hw_params, 0),
- snd_pcm_hw_params_get_period_size_min(hw_params, 0),
- snd_pcm_hw_params_get_period_size_max(hw_params, 0));
-#endif
-#ifdef ALSAAPI9
- err = snd_pcm_hw_params_set_period_size_near(alsa_device.inhandle,
- hw_params,
- (snd_pcm_uframes_t)
- frag_size, 0);
-#else
- tmp_snd_pcm_uframes = frag_size;
- err = snd_pcm_hw_params_set_period_size_near(alsa_device.inhandle,
- hw_params, &tmp_snd_pcm_uframes, 0);
-#endif
- check_error(err, "snd_pcm_hw_params_set_period_size_near (input)");
- // post("fragsize b %d", frag_size);
- // set the number of periods - ie numfrags
- // post("nfrags a %d", nfrags);
-#ifdef ALSAAPI9
- err = snd_pcm_hw_params_set_periods_near(alsa_device.inhandle,
- hw_params, nfrags, 0);
-#else
- tmp_uint = nfrags;
- err = snd_pcm_hw_params_set_periods_near(alsa_device.inhandle,
- hw_params, &tmp_uint, 0);
-#endif
- check_error(err, "snd_pcm_hw_params_set_periods_near (input)");
- // set the buffer size
-#ifdef ALSAAPI9
- err = snd_pcm_hw_params_set_buffer_size_near(alsa_device.inhandle,
- hw_params, nfrags * frag_size);
-#else
- tmp_snd_pcm_uframes = nfrags * frag_size;
- err = snd_pcm_hw_params_set_buffer_size_near(alsa_device.inhandle,
- hw_params, &tmp_snd_pcm_uframes);
-#endif
- check_error(err, "snd_pcm_hw_params_set_buffer_size_near (input)");
-
- err = snd_pcm_hw_params(alsa_device.inhandle, hw_params);
- check_error(err, "snd_pcm_hw_params (input)");
-
- snd_pcm_hw_params_free(hw_params);
-
- err = snd_pcm_sw_params_malloc(&sw_params);
- check_error(err, "snd_pcm_sw_params_malloc (input)");
- err = snd_pcm_sw_params_current(alsa_device.inhandle, sw_params);
- check_error(err, "snd_pcm_sw_params_current (input)");
- err = snd_pcm_sw_params_set_start_threshold(alsa_device.inhandle,
- sw_params, nfrags * frag_size);
- check_error(err, "snd_pcm_sw_params_set_start_threshold (input)");
- err = snd_pcm_sw_params_set_stop_threshold(alsa_device.inhandle,
- sw_params, 0x7fffffff);
- check_error(err, "snd_pcm_sw_params_set_stop_threshold (input)");
- err = snd_pcm_sw_params_set_avail_min(alsa_device.inhandle, sw_params,
- frag_size);
- check_error(err, "snd_pcm_sw_params_set_avail_min (input)");
- err = snd_pcm_sw_params(alsa_device.inhandle, sw_params);
- check_error(err, "snd_pcm_sw_params (input)");
-
- snd_pcm_sw_params_free(sw_params);
-
- snd_output_stdio_attach(&out, stderr, 0);
-#if 0
- if (sys_verbose)
- {
- snd_pcm_dump_hw_setup(alsa_device.inhandle, out);
- snd_pcm_dump_sw_setup(alsa_device.inhandle, out);
- }
-#endif
+ for (iodev = 0; iodev < naudioindev; iodev++)
+ {
+ alsa_numbertoname(audioindev[iodev], devname, 512);
+ err = snd_pcm_open(&alsa_indev[alsa_nindev].a_handle, devname,
+ SND_PCM_STREAM_CAPTURE, SND_PCM_NONBLOCK);
+ check_error(err, "snd_pcm_open (input)");
+ if (err < 0)
+ continue;
+ alsa_indev[alsa_nindev].a_devno = audioindev[iodev];
+ snd_pcm_nonblock(alsa_indev[alsa_nindev].a_handle, 1);
+ if (sys_verbose)
+ post("opened input device name %s", devname);
+ alsa_nindev++;
}
-
- if (outchans)
+ for (iodev = 0; iodev < naudiooutdev; iodev++)
{
- int foo;
- if (sys_verbose)
- post("opening sound output...");
- err = snd_pcm_hw_params_malloc(&hw_params);
- check_error(err, "snd_pcm_sw_params (output)");
-
- // get the default params
- err = snd_pcm_hw_params_any(alsa_device.outhandle, hw_params);
- check_error(err, "snd_pcm_hw_params_any (output)");
- // set interleaved access - FIXME deal with other access types
- err = snd_pcm_hw_params_set_access(alsa_device.outhandle, hw_params,
- SND_PCM_ACCESS_RW_INTERLEAVED);
- check_error(err, "snd_pcm_hw_params_set_access (output)");
-
- /* try to set interleaved access */
- err = snd_pcm_hw_params_set_access(alsa_device.outhandle,
- hw_params, SND_PCM_ACCESS_RW_INTERLEAVED);
- if (err < 0)
- {
- /* OK, so try non-interleaved */
- err = snd_pcm_hw_params_set_access(alsa_device.outhandle,
- hw_params, SND_PCM_ACCESS_RW_NONINTERLEAVED);
- if (err >= 0)
- {
- post("using non-interleaved audio");
- alsa_device.outnoninterleave = 1;
- }
- }
- check_error(err, "snd_pcm_hw_params_set_access (output)");
-
-
- // Try to set 32 bit format first
- err = snd_pcm_hw_params_set_format(alsa_device.outhandle, hw_params,
- SND_PCM_FORMAT_S32);
- if (err < 0)
- {
- err = snd_pcm_hw_params_set_format(alsa_device.outhandle,
- hw_params,SND_PCM_FORMAT_S16);
- check_error(err, "snd_pcm_hw_params_set_format (output)");
- /* fprintf(stderr,
- "PD-ALSA: 32 bit format not available - using 16\n"); */
- alsa_samplewidth = 2;
- }
- else
- {
- alsa_samplewidth = 4;
- }
- // set the subformat
- err = snd_pcm_hw_params_set_subformat(alsa_device.outhandle, hw_params,
- SND_PCM_SUBFORMAT_STD);
- check_error(err, "snd_pcm_hw_params_set_subformat (output)");
- // set the number of channels
- tmp_uint = outchans;
- err = snd_pcm_hw_params_set_channels_min(alsa_device.outhandle,
- hw_params, &tmp_uint);
- check_error(err, "snd_pcm_hw_params_set_channels (output)");
- if (tmp_uint != (unsigned)outchans)
- post("alsa: set output channels to %d", tmp_uint);
- outchans = tmp_uint;
- // set the sampling rate
- err = snd_pcm_hw_params_set_rate_min(alsa_device.outhandle, hw_params,
- &rate, 0);
- check_error(err, "snd_pcm_hw_params_set_rate_min (output)");
-#if 0
- err = snd_pcm_hw_params_get_rate(hw_params, &subunitdir);
- post("output sample rate %d", err);
-#endif
- // set the period - ie frag size
-#if 0
- post("output period was %d, min %d, max %d\n",
- snd_pcm_hw_params_get_period_size(hw_params, 0),
- snd_pcm_hw_params_get_period_size_min(hw_params, 0),
- snd_pcm_hw_params_get_period_size_max(hw_params, 0));
-#endif
- // post("fragsize c %d", frag_size);
-#ifdef ALSAAPI9
- err = snd_pcm_hw_params_set_period_size_near(alsa_device.outhandle,
- hw_params,
- (snd_pcm_uframes_t)
- frag_size, 0);
-#else
- tmp_snd_pcm_uframes = frag_size;
- err = snd_pcm_hw_params_set_period_size_near(alsa_device.outhandle,
- hw_params, &tmp_snd_pcm_uframes, 0);
-#endif
- // post("fragsize d %d", frag_size);
- check_error(err, "snd_pcm_hw_params_set_period_size_near (output)");
- // set the number of periods - ie numfrags
-#ifdef ALSAAPI9
- err = snd_pcm_hw_params_set_periods_near(alsa_device.outhandle,
- hw_params, nfrags, 0);
-#else
- tmp_uint = nfrags;
- err = snd_pcm_hw_params_set_periods_near(alsa_device.outhandle,
- hw_params, &tmp_uint, 0);
-#endif
- check_error(err, "snd_pcm_hw_params_set_periods_near (output)");
- // set the buffer size
-#ifdef ALSAAPI9
- err = snd_pcm_hw_params_set_buffer_size_near(alsa_device.outhandle,
- hw_params, nfrags * frag_size);
-#else
- tmp_snd_pcm_uframes = nfrags * frag_size;
- err = snd_pcm_hw_params_set_buffer_size_near(alsa_device.outhandle,
- hw_params, &tmp_snd_pcm_uframes);
-#endif
- check_error(err, "snd_pcm_hw_params_set_buffer_size_near (output)");
-
- err = snd_pcm_hw_params(alsa_device.outhandle, hw_params);
- check_error(err, "snd_pcm_hw_params (output)");
-
- snd_pcm_hw_params_free(hw_params);
-
- err = snd_pcm_sw_params_malloc(&sw_params);
- check_error(err, "snd_pcm_sw_params_malloc (output)");
- err = snd_pcm_sw_params_current(alsa_device.outhandle, sw_params);
- check_error(err, "snd_pcm_sw_params_current (output)");
- err = snd_pcm_sw_params_set_start_threshold(alsa_device.outhandle,
- sw_params, nfrags * frag_size);
- check_error(err, "snd_pcm_sw_params_set_start_threshold (output)");
- err = snd_pcm_sw_params_set_stop_threshold(alsa_device.outhandle,
- sw_params, 0x7fffffff);
- check_error(err, "snd_pcm_sw_params_set_stop_threshold (output)");
- err = snd_pcm_sw_params_set_avail_min(alsa_device.outhandle, sw_params,
- frag_size);
- check_error(err, "snd_pcm_sw_params_set_avail_min (output)");
- err = snd_pcm_sw_params(alsa_device.outhandle, sw_params);
- check_error(err, "snd_pcm_sw_params (output)");
- snd_pcm_sw_params_free(sw_params);
-
- snd_output_stdio_attach(&out, stderr, 0);
-#if 0
- if (sys_verbose)
- {
- snd_pcm_dump_hw_setup(alsa_device.outhandle, out);
- snd_pcm_dump_sw_setup(alsa_device.outhandle, out);
- }
-#endif
+ alsa_numbertoname(audiooutdev[iodev], devname, 512);
+ err = snd_pcm_open(&alsa_outdev[alsa_noutdev].a_handle, devname,
+ SND_PCM_STREAM_PLAYBACK, SND_PCM_NONBLOCK);
+ check_error(err, "snd_pcm_open (output)");
+ if (err < 0)
+ continue;
+ alsa_outdev[alsa_noutdev].a_devno = audiooutdev[iodev];
+ snd_pcm_nonblock(alsa_outdev[alsa_noutdev].a_handle, 1);
+ alsa_noutdev++;
+ }
+ if (!alsa_nindev && !alsa_noutdev)
+ goto blewit;
+
+ /* If all the open devices support mmap_noninterleaved, let's call
+ Wini's code in s_audio_alsamm.c */
+ alsa_usemmap = 1;
+ for (iodev = 0; iodev < alsa_nindev; iodev++)
+ if (!alsaio_canmmap(&alsa_indev[iodev]))
+ alsa_usemmap = 0;
+ for (iodev = 0; iodev < alsa_noutdev; iodev++)
+ if (!alsaio_canmmap(&alsa_outdev[iodev]))
+ alsa_usemmap = 0;
+ if (alsa_usemmap)
+ {
+ post("using mmap audio interface");
+ if (alsamm_open_audio(rate))
+ goto blewit;
+ else return (0);
+ }
+ for (iodev = 0; iodev < alsa_nindev; iodev++)
+ {
+ int channels = chindev[iodev];
+ if (alsaio_setup(&alsa_indev[iodev], 0, &channels, &rate,
+ nfrags, frag_size) < 0)
+ goto blewit;
+ inchans += channels;
}
+ for (iodev = 0; iodev < alsa_noutdev; iodev++)
+ {
+ int channels = choutdev[iodev];
+ if (alsaio_setup(&alsa_outdev[iodev], 1, &channels, &rate,
+ nfrags, frag_size) < 0)
+ goto blewit;
+ outchans += channels;
+ }
+ if (!inchans && !outchans)
+ goto blewit;
- if (inchans)
- snd_pcm_prepare(alsa_device.inhandle);
- if (outchans)
- snd_pcm_prepare(alsa_device.outhandle);
+ for (iodev = 0; iodev < alsa_nindev; iodev++)
+ snd_pcm_prepare(alsa_indev[iodev].a_handle);
+ for (iodev = 0; iodev < alsa_noutdev; iodev++)
+ snd_pcm_prepare(alsa_outdev[iodev].a_handle);
- // if duplex we can link the channels so they start together
- if (inchans && outchans)
- snd_pcm_link(alsa_device.inhandle, alsa_device.outhandle);
+ /* if duplex we can link the channels so they start together */
+ for (iodev = 0; iodev < alsa_nindev; iodev++)
+ for (dev2 = 0; dev2 < alsa_noutdev; dev2++)
+ {
+ if (alsa_indev[iodev].a_devno == alsa_outdev[iodev].a_devno)
+ {
+ snd_pcm_link(alsa_indev[iodev].a_handle,
+ alsa_outdev[iodev].a_handle);
+ }
+ }
+
+ /* allocate the status variables */
+ if (!alsa_status)
+ {
+ err = snd_pcm_status_malloc(&alsa_status);
+ check_error(err, "snd_pcm_status_malloc");
+ }
- // set up the status variables
- err = snd_pcm_status_malloc(&in_status);
- check_error(err, "snd_pcm_status_malloc");
- err = snd_pcm_status_malloc(&out_status);
- check_error(err, "snd_pcm_status_malloc");
+ /* fill the buffer with silence */
+ memset(alsa_snd_buf, 0, alsa_snd_bufsize);
- // set up the buffer
- if (alsa_snd_buf)
- free(alsa_snd_buf);
- alsa_snd_buf = (void *)malloc(
- sizeof(char) * alsa_samplewidth * DEFDACBLKSIZE *
- (outchans > inchans ? outchans : inchans));
- memset(alsa_snd_buf, 0, sizeof(char) * alsa_samplewidth * DEFDACBLKSIZE *
- (outchans > inchans ? outchans : inchans));
- /* make an array of pointers too in case we need them */
- if (alsa_buf_ptrs)
- free(alsa_buf_ptrs);
- alsa_buf_ptrs = (void **)malloc(
- sizeof(void *) * (outchans > inchans ? outchans : inchans));
- for (i = 0; i < (outchans > inchans ? outchans : inchans); i++)
- alsa_buf_ptrs[i] = (t_alsa_sample32 *)alsa_snd_buf + i * DEFDACBLKSIZE;
-
- // fill the buffer with silence
if (outchans)
{
- i = (frag_size * nfrags)/DEFDACBLKSIZE + 1;
- while (i--)
- {
- if (alsa_device.outnoninterleave)
- snd_pcm_writen(alsa_device.outhandle, alsa_buf_ptrs,
- DEFDACBLKSIZE);
- else snd_pcm_writei(alsa_device.outhandle, alsa_snd_buf,
- DEFDACBLKSIZE);
- }
- /* confused about this: */
- /* if ((err = snd_pcm_start(alsa_device.outhandle) < 0))
- check_error(err, "output start failed\n"); */
+ i = (frag_size * nfrags)/DEFDACBLKSIZE + 1;
+ while (i--)
+ {
+ for (iodev = 0; iodev < alsa_noutdev; iodev++)
+ snd_pcm_writei(alsa_outdev[iodev].a_handle, alsa_snd_buf,
+ DEFDACBLKSIZE);
+ }
}
else if (inchans)
{
- if (snd_pcm_start(alsa_device.inhandle) < 0)
- check_error(err, "input start failed\n");
+ for (iodev = 0; iodev < alsa_nindev; iodev++)
+ if ((err = snd_pcm_start(alsa_indev[iodev].a_handle)) < 0)
+ check_error(err, "input start failed\n");
}
- alsa_outchannels = outchans;
- alsa_inchannels = inchans;
-
- return (!(inchans || outchans));
+ return (0);
+blewit:
+ sys_inchannels = 0;
+ sys_outchannels = 0;
+ alsa_close_audio();
+ return (1);
}
void alsa_close_audio(void)
{
- int err;
- if (alsa_inchannels)
+ int err, iodev;
+ if (alsa_usemmap)
+ {
+ alsamm_close_audio();
+ return;
+ }
+ for (iodev = 0; iodev < alsa_nindev; iodev++)
{
- err = snd_pcm_close(alsa_device.inhandle);
- check_error(err, "snd_pcm_close (input)");
+ err = snd_pcm_close(alsa_indev[iodev].a_handle);
+ check_error(err, "snd_pcm_close (input)");
}
- if (alsa_outchannels)
+ for (iodev = 0; iodev < alsa_noutdev; iodev++)
{
- err = snd_pcm_close(alsa_device.outhandle);
- check_error(err, "snd_pcm_close (output)");
+ err = snd_pcm_close(alsa_outdev[iodev].a_handle);
+ check_error(err, "snd_pcm_close (output)");
}
+ alsa_nindev = alsa_noutdev = 0;
}
// #define DEBUG_ALSA_XFER
int alsa_send_dacs(void)
{
- static int16_t *sp;
+#ifdef DEBUG_ALSA_XFER
static int xferno = 0;
static int callno = 0;
+#endif
static double timenow;
double timelast;
t_sample *fp, *fp1, *fp2;
- int i, j, k, err, devno = 0;
- int inputcount = 0, outputcount = 0, inputlate = 0, outputlate = 0;
- int result;
- int inchannels = (sys_inchannels > alsa_inchannels ?
- alsa_inchannels : sys_inchannels);
- int outchannels = (sys_outchannels > alsa_outchannels ?
- alsa_outchannels : sys_outchannels);
- unsigned int intransfersize = DEFDACBLKSIZE;
- unsigned int outtransfersize = DEFDACBLKSIZE;
+ int i, j, k, err, iodev, result, ch;
+ int chansintogo, chansouttogo;
+ unsigned int transfersize;
+
+ if (alsa_usemmap)
+ return (alsamm_send_dacs());
+
+ if (!alsa_nindev && !alsa_noutdev)
+ return (SENDDACS_NO);
+
+ chansintogo = sys_inchannels;
+ chansouttogo = sys_outchannels;
+ transfersize = DEFDACBLKSIZE;
- // get the status
- if (!inchannels && !outchannels)
- {
- return SENDDACS_NO;
- }
-
timelast = timenow;
timenow = sys_getrealtime();
#ifdef DEBUG_ALSA_XFER
if (timenow - timelast > 0.050)
- fprintf(stderr, "(%d)",
- (int)(1000 * (timenow - timelast))), fflush(stderr);
-#endif
-
+ fprintf(stderr, "(%d)",
+ (int)(1000 * (timenow - timelast))), fflush(stderr);
callno++;
+#endif
alsa_checkiosync(); /* check I/O are in sync and data not late */
- if (alsa_inchannels)
+ for (iodev = 0; iodev < alsa_nindev; iodev++)
{
- snd_pcm_status(alsa_device.inhandle, in_status);
- if (snd_pcm_status_get_avail(in_status) < intransfersize)
- return SENDDACS_NO;
+ snd_pcm_status(alsa_indev[iodev].a_handle, alsa_status);
+ if (snd_pcm_status_get_avail(alsa_status) < transfersize)
+ return SENDDACS_NO;
}
- if (alsa_outchannels)
+ for (iodev = 0; iodev < alsa_noutdev; iodev++)
{
- snd_pcm_status(alsa_device.outhandle, out_status);
- if (snd_pcm_status_get_avail(out_status) < outtransfersize)
- return SENDDACS_NO;
+ snd_pcm_status(alsa_outdev[iodev].a_handle, alsa_status);
+ if (snd_pcm_status_get_avail(alsa_status) < transfersize)
+ return SENDDACS_NO;
}
/* do output */
- if (alsa_outchannels)
- {
- fp = sys_soundout;
- if (alsa_samplewidth == 4)
- {
- if (alsa_device.outnoninterleave)
- {
- int n = outchannels * DEFDACBLKSIZE;
- for (i = 0, fp1 = fp; i < n; i++)
- {
- float s1 = *fp1 * INT32_MAX;
- ((t_alsa_sample32 *)alsa_snd_buf)[i] = CLIP32(s1);
- }
- n = alsa_outchannels * DEFDACBLKSIZE;
- for (; i < n; i++)
- ((t_alsa_sample32 *)alsa_snd_buf)[i] = 0;
- }
- else
- {
- for (i = 0, fp1 = fp; i < outchannels; i++,
- fp1 += DEFDACBLKSIZE)
- {
- for (j = i, k = DEFDACBLKSIZE, fp2 = fp1; k--;
- j += alsa_outchannels, fp2++)
- {
- float s1 = *fp2 * INT32_MAX;
- ((t_alsa_sample32 *)alsa_snd_buf)[j] = CLIP32(s1);
- }
- }
- }
- }
- else
- {
- for (i = 0, fp1 = fp; i < outchannels; i++, fp1 += DEFDACBLKSIZE)
- {
- for (j = i, k = DEFDACBLKSIZE, fp2 = fp1; k--;
- j += alsa_outchannels, fp2++)
- {
- int s = *fp2 * 32767.;
- if (s > 32767)
- s = 32767;
- else if (s < -32767)
- s = -32767;
- ((t_alsa_sample16 *)alsa_snd_buf)[j] = s;
- }
- }
- }
-
- if (alsa_device.outnoninterleave)
- result = snd_pcm_writen(alsa_device.outhandle, alsa_buf_ptrs,
- outtransfersize);
- else result = snd_pcm_writei(alsa_device.outhandle, alsa_snd_buf,
- outtransfersize);
-
- if (result != (int)outtransfersize)
- {
+ for (iodev = 0, fp1 = sys_soundout, ch = 0; iodev < alsa_noutdev; iodev++)
+ {
+ int thisdevchans = alsa_outdev[iodev].a_channels;
+ int chans = (chansouttogo < thisdevchans ? chansouttogo : thisdevchans);
+ chansouttogo -= chans;
+
+ if (alsa_outdev[iodev].a_sampwidth == 4)
+ {
+ for (i = 0; i < chans; i++, ch++, fp1 += DEFDACBLKSIZE)
+ for (j = ch, k = DEFDACBLKSIZE, fp2 = fp1; k--;
+ j += thisdevchans, fp2++)
+ {
+ float s1 = *fp2 * INT32_MAX;
+ ((t_alsa_sample32 *)alsa_snd_buf)[j] = CLIP32(s1);
+ }
+ for (; i < thisdevchans; i++, ch++)
+ for (j = ch, k = DEFDACBLKSIZE; k--; j += thisdevchans)
+ ((t_alsa_sample32 *)alsa_snd_buf)[j] = 0;
+ }
+ else
+ {
+ for (i = 0; i < chans; i++, ch++, fp1 += DEFDACBLKSIZE)
+ for (j = ch, k = DEFDACBLKSIZE, fp2 = fp1; k--;
+ j += thisdevchans, fp2++)
+ {
+ int s = *fp2 * 32767.;
+ if (s > 32767)
+ s = 32767;
+ else if (s < -32767)
+ s = -32767;
+ ((t_alsa_sample16 *)alsa_snd_buf)[j] = s;
+ }
+ for (; i < thisdevchans; i++, ch++)
+ for (j = ch, k = DEFDACBLKSIZE; k--; j += thisdevchans)
+ ((t_alsa_sample16 *)alsa_snd_buf)[j] = 0;
+ }
+ result = snd_pcm_writei(alsa_outdev[iodev].a_handle, alsa_snd_buf,
+ transfersize);
+
+ if (result != (int)transfersize)
+ {
#ifdef DEBUG_ALSA_XFER
- if (result >= 0 || errno == EAGAIN)
- fprintf(stderr, "ALSA: write returned %d of %d\n",
- result, outtransfersize);
- else fprintf(stderr, "ALSA: write: %s\n",
- snd_strerror(errno));
- fprintf(stderr,
- "inputcount %d, outputcount %d, outbufsize %d\n",
- inputcount, outputcount,
- (ALSA_EXTRABUFFER + sys_advance_samples)
- * alsa_samplewidth * outchannels);
+ if (result >= 0 || errno == EAGAIN)
+ fprintf(stderr, "ALSA: write returned %d of %d\n",
+ result, transfersize);
+ else fprintf(stderr, "ALSA: write: %s\n",
+ snd_strerror(errno));
+ fprintf(stderr,
+ "inputcount %d, outputcount %d, outbufsize %d\n",
+ inputcount, outputcount,
+ (ALSA_EXTRABUFFER + sys_advance_samples)
+ * alsa_outdev[iodev].a_sampwidth * outchannels);
#endif
- sys_log_error(ERR_DACSLEPT);
- return (SENDDACS_NO);
- }
-
- /* zero out the output buffer */
- memset(sys_soundout, 0, DEFDACBLKSIZE * sizeof(*sys_soundout) *
- sys_outchannels);
- if (sys_getrealtime() - timenow > 0.002)
- {
+ sys_log_error(ERR_DACSLEPT);
+ return (SENDDACS_NO);
+ }
+
+ /* zero out the output buffer */
+ memset(sys_soundout, 0, DEFDACBLKSIZE * sizeof(*sys_soundout) *
+ sys_outchannels);
+ if (sys_getrealtime() - timenow > 0.002)
+ {
#ifdef DEBUG_ALSA_XFER
- fprintf(stderr, "output %d took %d msec\n",
- callno, (int)(1000 * (timenow - timelast))), fflush(stderr);
+ fprintf(stderr, "output %d took %d msec\n",
+ callno, (int)(1000 * (timenow - timelast))), fflush(stderr);
#endif
- timenow = sys_getrealtime();
- sys_log_error(ERR_DACSLEPT);
- }
- }
- /* do input */
- if (alsa_inchannels)
- {
- if (alsa_device.innoninterleave)
- result = snd_pcm_readn(alsa_device.inhandle, alsa_buf_ptrs,
- intransfersize);
- else result = snd_pcm_readi(alsa_device.inhandle, alsa_snd_buf,
- intransfersize);
- if (result < (int)intransfersize)
- {
+ timenow = sys_getrealtime();
+ sys_log_error(ERR_DACSLEPT);
+ }
+ }
+
+ /* do input */
+ for (iodev = 0, fp1 = sys_soundin, ch = 0; iodev < alsa_nindev; iodev++)
+ {
+ int thisdevchans = alsa_indev[iodev].a_channels;
+ int chans = (chansintogo < thisdevchans ? chansintogo : thisdevchans);
+ chansouttogo -= chans;
+ result = snd_pcm_readi(alsa_indev[iodev].a_handle, alsa_snd_buf,
+ transfersize);
+ if (result < (int)transfersize)
+ {
#ifdef DEBUG_ALSA_XFER
- if (result < 0)
- fprintf(stderr,
- "snd_pcm_read %d %d: %s\n",
- callno, xferno, snd_strerror(errno));
- else fprintf(stderr,
- "snd_pcm_read %d %d returned only %d\n",
- callno, xferno, result);
- fprintf(stderr,
- "inputcount %d, outputcount %d, inbufsize %d\n",
- inputcount, outputcount,
- (ALSA_EXTRABUFFER + sys_advance_samples)
- * alsa_samplewidth * inchannels);
+ if (result < 0)
+ fprintf(stderr,
+ "snd_pcm_read %d %d: %s\n",
+ callno, xferno, snd_strerror(errno));
+ else fprintf(stderr,
+ "snd_pcm_read %d %d returned only %d\n",
+ callno, xferno, result);
+ fprintf(stderr,
+ "inputcount %d, outputcount %d, inbufsize %d\n",
+ inputcount, outputcount,
+ (ALSA_EXTRABUFFER + sys_advance_samples)
+ * alsa_indev[iodev].a_sampwidth * inchannels);
#endif
- sys_log_error(ERR_ADCSLEPT);
- return (SENDDACS_NO);
- }
- fp = sys_soundin;
- if (alsa_samplewidth == 4)
- {
- if (alsa_device.innoninterleave)
- {
- int n = inchannels * DEFDACBLKSIZE;
- for (i = 0, fp1 = fp; i < n; i++)
- *fp1 = (float) ((t_alsa_sample32 *)alsa_snd_buf)[i]
- * (1./ INT32_MAX);
- }
- else
- {
- for (i = 0, fp1 = fp; i < inchannels;
- i++, fp1 += DEFDACBLKSIZE)
- {
- for (j = i, k = DEFDACBLKSIZE, fp2 = fp1; k--;
- j += alsa_inchannels, fp2++)
- *fp2 = (float) ((t_alsa_sample32 *)alsa_snd_buf)[j]
- * (1./ INT32_MAX);
- }
- }
- }
- else
- {
- for (i = 0, fp1 = fp; i < inchannels; i++, fp1 += DEFDACBLKSIZE)
- {
- for (j = i, k = DEFDACBLKSIZE, fp2 = fp1; k--;
- j += alsa_inchannels, fp2++)
- *fp2 = (float) ((t_alsa_sample16 *)alsa_snd_buf)[j]
- * 3.051850e-05;
- }
- }
+ sys_log_error(ERR_ADCSLEPT);
+ return (SENDDACS_NO);
+ }
+ if (alsa_indev[iodev].a_sampwidth == 4)
+ {
+ for (i = 0; i < chans; i++, ch++, fp1 += DEFDACBLKSIZE)
+ {
+ for (j = ch, k = DEFDACBLKSIZE, fp2 = fp1; k--;
+ j += thisdevchans, fp2++)
+ *fp2 = (float) ((t_alsa_sample32 *)alsa_snd_buf)[j]
+ * (1./ INT32_MAX);
+ }
+ }
+ else
+ {
+ for (i = 0; i < chans; i++, ch++, fp1 += DEFDACBLKSIZE)
+ {
+ for (j = ch, k = DEFDACBLKSIZE, fp2 = fp1; k--;
+ j += thisdevchans, fp2++)
+ *fp2 = (float) ((t_alsa_sample16 *)alsa_snd_buf)[j]
+ * 3.051850e-05;
+ }
+ }
}
+#ifdef DEBUG_ALSA_XFER
xferno++;
+#endif
if (sys_getrealtime() - timenow > 0.002)
{
#ifdef DEBUG_ALSA_XFER
- fprintf(stderr, "routine took %d msec\n",
- (int)(1000 * (sys_getrealtime() - timenow)));
+ fprintf(stderr, "routine took %d msec\n",
+ (int)(1000 * (sys_getrealtime() - timenow)));
#endif
- sys_log_error(ERR_ADCSLEPT);
+ sys_log_error(ERR_ADCSLEPT);
}
return SENDDACS_YES;
}
void alsa_printstate( void)
{
- int i, result;
+ int i, result, iodev = 0;
snd_pcm_sframes_t indelay, outdelay;
if (sys_audioapi != API_ALSA)
{
- error("restart-audio: implemented for ALSA only.");
- return;
+ error("restart-audio: implemented for ALSA only.");
+ return;
}
if (sys_inchannels)
{
- result = snd_pcm_delay(alsa_device.inhandle, &indelay);
- if (result < 0)
- post("snd_pcm_delay 1 failed");
- else post("in delay %d", indelay);
+ result = snd_pcm_delay(alsa_indev[iodev].a_handle, &indelay);
+ if (result < 0)
+ post("snd_pcm_delay 1 failed");
+ else post("in delay %d", indelay);
}
if (sys_outchannels)
{
- result = snd_pcm_delay(alsa_device.outhandle, &outdelay);
- if (result < 0)
- post("snd_pcm_delay 2 failed");
- else post("out delay %d", outdelay);
+ result = snd_pcm_delay(alsa_outdev[iodev].a_handle, &outdelay);
+ if (result < 0)
+ post("snd_pcm_delay 2 failed");
+ else post("out delay %d", outdelay);
}
post("sum %d (%d mod 64)\n", indelay + outdelay, (indelay+outdelay)%64);
@@ -753,57 +563,53 @@ void alsa_printstate( void)
void alsa_resync( void)
{
- int i, result;
+ int i, result, iodev = 0;
if (sys_audioapi != API_ALSA)
{
- error("restart-audio: implemented for ALSA only.");
- return;
+ error("restart-audio: implemented for ALSA only.");
+ return;
}
memset(alsa_snd_buf, 0,
- sizeof(char) * alsa_samplewidth * DEFDACBLKSIZE * sys_outchannels);
+ alsa_indev[iodev].a_sampwidth *
+ DEFDACBLKSIZE * sys_outchannels);
for (i = 0; i < 1000000; i++)
{
- if (alsa_device.outnoninterleave)
- result = snd_pcm_writen(alsa_device.outhandle, alsa_buf_ptrs,
- DEFDACBLKSIZE);
- else result = snd_pcm_writei(alsa_device.outhandle, alsa_snd_buf,
- DEFDACBLKSIZE);
- if (result != (int)DEFDACBLKSIZE)
- break;
+ result = snd_pcm_writei(alsa_outdev[iodev].a_handle, alsa_snd_buf,
+ DEFDACBLKSIZE);
+ if (result != (int)DEFDACBLKSIZE)
+ break;
}
post("%d written", i);
}
-void alsa_putzeros(int n)
+void alsa_putzeros(int iodev, int n)
{
int i, result;
memset(alsa_snd_buf, 0,
- sizeof(char) * alsa_samplewidth * DEFDACBLKSIZE * alsa_outchannels);
+ alsa_outdev[iodev].a_sampwidth * DEFDACBLKSIZE *
+ alsa_outdev[iodev].a_channels);
for (i = 0; i < n; i++)
{
- if (alsa_device.outnoninterleave)
- result = snd_pcm_writen(alsa_device.outhandle, alsa_buf_ptrs,
- DEFDACBLKSIZE);
- else result = snd_pcm_writei(alsa_device.outhandle, alsa_snd_buf,
- DEFDACBLKSIZE);
+ result = snd_pcm_writei(alsa_outdev[iodev].a_handle, alsa_snd_buf,
+ DEFDACBLKSIZE);
#if 0
- if (result != DEFDACBLKSIZE)
- post("result %d", result);
+ if (result != DEFDACBLKSIZE)
+ post("result %d", result);
#endif
}
/* post ("putzeros %d", n); */
}
-void alsa_getzeros(int n)
+void alsa_getzeros(int iodev, int n)
{
int i, result;
for (i = 0; i < n; i++)
{
- result = snd_pcm_readi(alsa_device.inhandle, alsa_snd_buf,
- DEFDACBLKSIZE);
+ result = snd_pcm_readi(alsa_indev[iodev].a_handle, alsa_snd_buf,
+ DEFDACBLKSIZE);
#if 0
- if (result != DEFDACBLKSIZE)
- post("result %d", result);
+ if (result != DEFDACBLKSIZE)
+ post("result %d", result);
#endif
}
/* post ("getzeros %d", n); */
@@ -812,56 +618,108 @@ void alsa_getzeros(int n)
/* call this only if both input and output are open */
static void alsa_checkiosync( void)
{
- int i, result, checkit = 1, giveup = 1000, alreadylogged = 0;
- snd_pcm_sframes_t indelay, outdelay, defect;
-
- if (!(alsa_outchannels && alsa_inchannels))
- return;
- while (checkit)
- {
- checkit = 0;
- if (giveup-- <= 0)
- return;
- result = snd_pcm_delay(alsa_device.outhandle, &outdelay);
- if (result < 0)
- {
- post("output snd_pcm_delay failed: %s", snd_strerror(result));
- if (snd_pcm_status(alsa_device.outhandle, out_status) < 0)
- post("output snd_pcm_status failed");
- else post("astate %d",
- snd_pcm_status_get_state(out_status));
- return;
- }
- if (outdelay < 0)
- sys_log_error(ERR_DATALATE), alreadylogged = 1;
-
- if (sys_inchannels)
- {
- result = snd_pcm_delay(alsa_device.inhandle, &indelay);
- if (result < 0)
- {
- post("input snd_pcm_delay failed");
- return;
- }
- defect = indelay + outdelay - alsa_buf_samps;
- if (defect < -(3 * DEFDACBLKSIZE / 2) )
- {
- checkit = 1;
- alsa_putzeros(1);
- if (!alreadylogged)
- sys_log_error(ERR_RESYNC), alreadylogged = 1;
- }
- else if (defect > 0)
- {
- checkit = 1;
- alsa_getzeros(1);
- if (!alreadylogged)
- sys_log_error(ERR_RESYNC), alreadylogged = 1;
- }
- /* if (alreadylogged)
- post("in %d out %d defect %d", indelay, outdelay, defect); */
- }
+ int i, result, giveup = 1000, alreadylogged = 0, iodev = 0;
+ snd_pcm_sframes_t minphase, maxphase, thisphase, outdelay;
+
+ while (1)
+ {
+ if (giveup-- <= 0)
+ {
+ post("tried but couldn't sync A/D/A");
+ return;
+ }
+ minphase = 0x7fffffff;
+ maxphase = -0x7fffffff;
+ for (iodev = 0; iodev < alsa_noutdev; iodev++)
+ {
+ result = snd_pcm_delay(alsa_outdev[iodev].a_handle, &outdelay);
+ if (result < 0)
+ {
+ snd_pcm_prepare(alsa_outdev[iodev].a_handle);
+ result = snd_pcm_delay(alsa_outdev[iodev].a_handle, &outdelay);
+ }
+ if (result < 0)
+ {
+ post("output snd_pcm_delay failed: %s", snd_strerror(result));
+ if (snd_pcm_status(alsa_outdev[iodev].a_handle,
+ alsa_status) < 0)
+ post("output snd_pcm_status failed");
+ else post("astate %d",
+ snd_pcm_status_get_state(alsa_status));
+ return;
+ }
+ thisphase = alsa_buf_samps - outdelay;
+ if (thisphase < minphase)
+ minphase = thisphase;
+ if (thisphase > maxphase)
+ maxphase = thisphase;
+ if (outdelay < 0)
+ sys_log_error(ERR_DATALATE), alreadylogged = 1;
+ }
+ for (iodev = 0; iodev < alsa_nindev; iodev++)
+ {
+ result = snd_pcm_delay(alsa_indev[iodev].a_handle, &thisphase);
+ if (result < 0)
+ {
+ snd_pcm_prepare(alsa_indev[iodev].a_handle);
+ result = snd_pcm_delay(alsa_indev[iodev].a_handle, &thisphase);
+ }
+ if (result < 0)
+ {
+ post("output snd_pcm_delay failed: %s", snd_strerror(result));
+ if (snd_pcm_status(alsa_outdev[iodev].a_handle,
+ alsa_status) < 0)
+ post("output snd_pcm_status failed");
+ else post("astate %d",
+ snd_pcm_status_get_state(alsa_status));
+ return;
+ }
+ if (thisphase < minphase)
+ minphase = thisphase;
+ if (thisphase > maxphase)
+ maxphase = thisphase;
+ }
+ /* the "correct" position is for all the phases to be exactly
+ equal; but since we only make corrections DEFDACBLKSIZE samples
+ at a time, we just ask that the spread be not more than 3/4
+ of a block. */
+ if (maxphase <= minphase + (3 * DEFDACBLKSIZE / 4))
+ break;
+
+ if (!alreadylogged)
+ sys_log_error(ERR_RESYNC), alreadylogged = 1;
+ for (iodev = 0; iodev < alsa_noutdev; iodev++)
+ {
+ result = snd_pcm_delay(alsa_outdev[iodev].a_handle, &outdelay);
+ if (result < 0)
+ break;
+ thisphase = alsa_buf_samps - outdelay;
+ if (thisphase > minphase + DEFDACBLKSIZE)
+ {
+ alsa_putzeros(iodev, 1);
+#if DEBUGSYNC
+ fprintf(stderr, "putz %d %d\n", (int)thisphase, (int)minphase);
+#endif
+ }
+ }
+ for (iodev = 0; iodev < alsa_nindev; iodev++)
+ {
+ result = snd_pcm_delay(alsa_indev[iodev].a_handle, &thisphase);
+ if (result < 0)
+ break;
+ if (thisphase > minphase + DEFDACBLKSIZE)
+ {
+ alsa_getzeros(iodev, 1);
+#if DEBUGSYNC
+ fprintf(stderr, "getz %d %d\n", (int)thisphase, (int)minphase);
+#endif
+ }
+ }
}
+#if DEBUGSYNC
+ if (alreadylogged)
+ fprintf(stderr, "done\n");
+#endif
}
static int alsa_nnames = 0;
@@ -870,9 +728,9 @@ static char **alsa_names = 0;
void alsa_adddev(char *name)
{
if (alsa_nnames)
- alsa_names = (char **)t_resizebytes(alsa_names,
- alsa_nnames * sizeof(char *),
- (alsa_nnames+1) * sizeof(char *));
+ alsa_names = (char **)t_resizebytes(alsa_names,
+ alsa_nnames * sizeof(char *),
+ (alsa_nnames+1) * sizeof(char *));
else alsa_names = (char **)t_getbytes(sizeof(char *));
alsa_names[alsa_nnames] = gensym(name)->s_name;
alsa_nnames++;
@@ -882,15 +740,15 @@ static void alsa_numbertoname(int devno, char *devname, int nchar)
{
int ndev = 0, cardno = -1;
while (!snd_card_next(&cardno) && cardno >= 0)
- ndev++;
+ ndev++;
if (devno < 2*ndev)
{
- if (devno & 1)
- snprintf(devname, nchar, "plughw:%d", devno/2);
- else snprintf(devname, nchar, "hw:%d", devno/2);
+ if (devno & 1)
+ snprintf(devname, nchar, "plughw:%d", devno/2);
+ else snprintf(devname, nchar, "hw:%d", devno/2);
}
else if (devno <2*ndev + alsa_nnames)
- snprintf(devname, nchar, "%s", alsa_names[devno - 2*ndev]);
+ snprintf(devname, nchar, "%s", alsa_names[devno - 2*ndev]);
else snprintf(devname, nchar, "???");
}
@@ -898,48 +756,48 @@ static void alsa_numbertoname(int devno, char *devname, int nchar)
"plug" one. The card scan is derived from portaudio code. */
void alsa_getdevs(char *indevlist, int *nindevs,
char *outdevlist, int *noutdevs, int *canmulti,
- int maxndev, int devdescsize)
+ int maxndev, int devdescsize)
{
int ndev = 0, cardno = -1, i, j;
- *canmulti = 0; /* only one device; must be the same for input&output */
+ *canmulti = 2; /* supports multiple devices */
while (!snd_card_next(&cardno) && cardno >= 0)
{
- snd_ctl_t *ctl;
- snd_ctl_card_info_t *info;
+ snd_ctl_t *ctl;
+ snd_ctl_card_info_t *info;
char devname[80];
- const char *desc;
- if (2 * ndev + 2 > maxndev)
- break;
- /* apparently, "cardno" is just a counter; but check that here */
- if (ndev != cardno)
- fprintf(stderr, "oops: ALSA cards not reported in order?\n");
+ const char *desc;
+ if (2 * ndev + 2 > maxndev)
+ break;
+ /* apparently, "cardno" is just a counter; but check that here */
+ if (ndev != cardno)
+ fprintf(stderr, "oops: ALSA cards not reported in order?\n");
sprintf(devname, "hw:%d", cardno );
/* fprintf(stderr, "\ntry %s...\n", devname); */
if (snd_ctl_open(&ctl, devname, 0) >= 0)
- {
+ {
snd_ctl_card_info_malloc(&info);
snd_ctl_card_info(ctl, info);
- desc = snd_ctl_card_info_get_name(info);
- snd_ctl_card_info_free(info);
- }
- else
- {
- fprintf(stderr, "ALSA card scan error\n");
- desc = "???";
- }
+ desc = snd_ctl_card_info_get_name(info);
+ snd_ctl_card_info_free(info);
+ }
+ else
+ {
+ fprintf(stderr, "ALSA card scan error\n");
+ desc = "???";
+ }
/* fprintf(stderr, "name: %s\n", snd_ctl_card_info_get_name(info)); */
sprintf(indevlist + 2*ndev * devdescsize, "%s (hardware)", desc);
sprintf(indevlist + (2*ndev + 1) * devdescsize, "%s (plug-in)", desc);
sprintf(outdevlist + 2*ndev * devdescsize, "%s (hardware)", desc);
sprintf(outdevlist + (2*ndev + 1) * devdescsize, "%s (plug-in)", desc);
- ndev++;
+ ndev++;
}
for (i = 0, j = 2*ndev; i < alsa_nnames; i++, j++)
{
- if (j >= maxndev)
- break;
+ if (j >= maxndev)
+ break;
snprintf(indevlist + j * devdescsize, devdescsize, "%s",
- alsa_names[i]);
+ alsa_names[i]);
}
*nindevs = *noutdevs = j;
}