mirror of
https://github.com/kbeckmann/game-and-watch-retro-go.git
synced 2025-12-16 13:15:55 +01:00
493 lines
15 KiB
C
493 lines
15 KiB
C
/* MikMod sound library
|
|
(c) 1998, 1999, 2000 Miodrag Vallat and others - see file AUTHORS for
|
|
complete list.
|
|
|
|
This library is free software; you can redistribute it and/or modify
|
|
it under the terms of the GNU Library 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 Library General Public License for more details.
|
|
|
|
You should have received a copy of the GNU Library General Public
|
|
License along with this library; if not, write to the Free Software
|
|
Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA
|
|
02111-1307, USA.
|
|
*/
|
|
|
|
/*==============================================================================
|
|
|
|
$Id$
|
|
|
|
Driver for Advanced Linux Sound Architecture (ALSA)
|
|
|
|
==============================================================================*/
|
|
|
|
#ifdef HAVE_CONFIG_H
|
|
#include "config.h"
|
|
#endif
|
|
|
|
#include "mikmod_internals.h"
|
|
|
|
#ifdef DRV_ALSA
|
|
|
|
#ifdef HAVE_UNISTD_H
|
|
#include <unistd.h>
|
|
#endif
|
|
#ifdef HAVE_MEMORY_H
|
|
#include <memory.h>
|
|
#endif
|
|
|
|
#ifdef MIKMOD_DYNAMIC
|
|
#include <dlfcn.h>
|
|
#endif
|
|
#include <stdlib.h>
|
|
#include <string.h>
|
|
|
|
#include <alsa/asoundlib.h>
|
|
#if defined(SND_LIB_VERSION) && (SND_LIB_VERSION >= 0x20000)
|
|
#undef DRV_ALSA
|
|
#endif
|
|
|
|
#if defined(SND_LIB_VERSION) && (SND_LIB_VERSION < 0x600)
|
|
#error ALSA Version too old. Please upgrade your Linux distribution.
|
|
#endif
|
|
#endif /* DRV_ALSA */
|
|
|
|
#ifdef DRV_ALSA
|
|
|
|
#ifdef MIKMOD_DYNAMIC
|
|
/* runtime link with libasound */
|
|
#ifndef HAVE_RTLD_GLOBAL
|
|
#define RTLD_GLOBAL (0)
|
|
#endif
|
|
static int (*alsa_pcm_subformat_mask_malloc)(snd_pcm_subformat_mask_t **);
|
|
static const char * (*alsa_strerror)(int);
|
|
static int (*alsa_pcm_resume)(snd_pcm_t *);
|
|
static int (*alsa_pcm_prepare)(snd_pcm_t *);
|
|
static int (*alsa_pcm_hw_params_any)(snd_pcm_t *, snd_pcm_hw_params_t *);
|
|
static int (*alsa_pcm_hw_params)(snd_pcm_t *, snd_pcm_hw_params_t *);
|
|
static int (*alsa_pcm_hw_params_current)(snd_pcm_t *, snd_pcm_hw_params_t *);
|
|
static int (*alsa_pcm_hw_params_set_access)(snd_pcm_t *, snd_pcm_hw_params_t *, snd_pcm_access_t);
|
|
static int (*alsa_pcm_hw_params_set_format)(snd_pcm_t *, snd_pcm_hw_params_t *, snd_pcm_format_t);
|
|
static int (*alsa_pcm_hw_params_set_rate_near)(snd_pcm_t *, snd_pcm_hw_params_t *, unsigned int *, int *);
|
|
static int (*alsa_pcm_hw_params_set_channels_near)(snd_pcm_t *, snd_pcm_hw_params_t *, unsigned int *);
|
|
static int (*alsa_pcm_hw_params_set_buffer_time_near)(snd_pcm_t *, snd_pcm_hw_params_t *, unsigned int *, int *);
|
|
static int (*alsa_pcm_hw_params_set_period_time_near)(snd_pcm_t *, snd_pcm_hw_params_t *, unsigned int *, int *);
|
|
static int (*alsa_pcm_hw_params_get_buffer_size)(const snd_pcm_hw_params_t *, snd_pcm_uframes_t *);
|
|
static int (*alsa_pcm_hw_params_get_period_size)(const snd_pcm_hw_params_t *, snd_pcm_uframes_t *, int *);
|
|
static int (*alsa_pcm_sw_params_sizeof)(void);
|
|
static int (*alsa_pcm_hw_params_sizeof)(void);
|
|
static int (*alsa_pcm_open)(snd_pcm_t**, const char *, int, int);
|
|
static int (*alsa_pcm_close)(snd_pcm_t*);
|
|
static int (*alsa_pcm_nonblock)(snd_pcm_t*, int);
|
|
static int (*alsa_pcm_drop)(snd_pcm_t*);
|
|
static int (*alsa_pcm_start)(snd_pcm_t *);
|
|
static snd_pcm_sframes_t (*alsa_pcm_writei)(snd_pcm_t*,const void*,snd_pcm_uframes_t);
|
|
|
|
static void* libasound = NULL;
|
|
|
|
#else
|
|
/* compile-time link with libasound */
|
|
#define alsa_pcm_subformat_mask_malloc snd_pcm_subformat_mask_malloc
|
|
#define alsa_strerror snd_strerror
|
|
#define alsa_pcm_hw_params_any snd_pcm_hw_params_any
|
|
#define alsa_pcm_hw_params snd_pcm_hw_params
|
|
#define alsa_pcm_hw_params_current snd_pcm_hw_params_current
|
|
#define alsa_pcm_hw_params_set_access snd_pcm_hw_params_set_access
|
|
#define alsa_pcm_hw_params_set_format snd_pcm_hw_params_set_format
|
|
#define alsa_pcm_hw_params_set_rate_near snd_pcm_hw_params_set_rate_near
|
|
#define alsa_pcm_hw_params_set_channels_near snd_pcm_hw_params_set_channels_near
|
|
#define alsa_pcm_hw_params_set_buffer_time_near snd_pcm_hw_params_set_buffer_time_near
|
|
#define alsa_pcm_hw_params_set_period_time_near snd_pcm_hw_params_set_period_time_near
|
|
#define alsa_pcm_hw_params_get_buffer_size snd_pcm_hw_params_get_buffer_size
|
|
#define alsa_pcm_hw_params_get_period_size snd_pcm_hw_params_get_period_size
|
|
#define alsa_pcm_resume snd_pcm_resume
|
|
#define alsa_pcm_prepare snd_pcm_prepare
|
|
#define alsa_pcm_close snd_pcm_close
|
|
#define alsa_pcm_nonblock snd_pcm_nonblock
|
|
#define alsa_pcm_drop snd_pcm_drop
|
|
#define alsa_pcm_start snd_pcm_start
|
|
#define alsa_pcm_open snd_pcm_open
|
|
#define alsa_pcm_writei snd_pcm_writei
|
|
#endif /* MIKMOD_DYNAMIC */
|
|
|
|
#if defined(MIKMOD_DEBUG)
|
|
# define dbgprint fprintf
|
|
#elif defined (__GNUC__) && !(defined(__STDC_VERSION__) && __STDC_VERSION__ >= 199901L)
|
|
# define dbgprint(f, fmt, args...) do {} while (0)
|
|
#else
|
|
# define dbgprint(f, ...) do {} while (0)
|
|
#endif
|
|
static BOOL enabled = 0;
|
|
static snd_pcm_t *pcm_h = NULL;
|
|
static SBYTE *audiobuffer = NULL;
|
|
static snd_pcm_sframes_t period_size;
|
|
static int bytes_written = 0, bytes_played = 0;
|
|
static int global_frame_size;
|
|
|
|
#ifdef MIKMOD_DYNAMIC
|
|
static int ALSA_Link(void)
|
|
{
|
|
if (libasound) return 0;
|
|
|
|
/* load libasound.so */
|
|
libasound = dlopen("libasound.so.2",RTLD_LAZY|RTLD_GLOBAL);
|
|
if (!libasound) libasound = dlopen("libasound.so",RTLD_LAZY|RTLD_GLOBAL);
|
|
if (!libasound) return 1;
|
|
|
|
if (!(alsa_pcm_subformat_mask_malloc = (int (*)(snd_pcm_subformat_mask_t **))
|
|
dlsym(libasound,"snd_pcm_subformat_mask_malloc"))) return 1;
|
|
if (!(alsa_strerror = (const char* (*)(int))
|
|
dlsym(libasound,"snd_strerror"))) return 1;
|
|
if (!(alsa_pcm_prepare = (int (*)(snd_pcm_t *))
|
|
dlsym(libasound,"snd_pcm_prepare"))) return 1;
|
|
if (!(alsa_pcm_sw_params_sizeof = (int (*)(void))
|
|
dlsym(libasound,"snd_pcm_sw_params_sizeof"))) return 1;
|
|
if (!(alsa_pcm_hw_params_sizeof = (int (*)(void))
|
|
dlsym(libasound,"snd_pcm_hw_params_sizeof"))) return 1;
|
|
if (!(alsa_pcm_resume = (int (*)(snd_pcm_t *))
|
|
dlsym(libasound,"snd_pcm_resume"))) return 1;
|
|
if (!(alsa_pcm_hw_params_any = (int (*)(snd_pcm_t *, snd_pcm_hw_params_t *))
|
|
dlsym(libasound,"snd_pcm_hw_params_any"))) return 1;
|
|
if (!(alsa_pcm_hw_params = (int (*)(snd_pcm_t *, snd_pcm_hw_params_t *))
|
|
dlsym(libasound,"snd_pcm_hw_params"))) return 1;
|
|
if (!(alsa_pcm_hw_params_current = (int (*)(snd_pcm_t *, snd_pcm_hw_params_t *))
|
|
dlsym(libasound,"snd_pcm_hw_params_current"))) return 1;
|
|
if (!(alsa_pcm_hw_params_set_access = (int (*)(snd_pcm_t *, snd_pcm_hw_params_t *, snd_pcm_access_t))
|
|
dlsym(libasound,"snd_pcm_hw_params_set_access"))) return 1;
|
|
if (!(alsa_pcm_hw_params_set_format = (int (*)(snd_pcm_t *, snd_pcm_hw_params_t *, snd_pcm_format_t))
|
|
dlsym(libasound,"snd_pcm_hw_params_set_format"))) return 1;
|
|
if (!(alsa_pcm_hw_params_set_rate_near = (int (*)(snd_pcm_t *, snd_pcm_hw_params_t *, unsigned int *, int *))
|
|
dlsym(libasound,"snd_pcm_hw_params_set_rate_near"))) return 1;
|
|
if (!(alsa_pcm_hw_params_set_channels_near = (int (*)(snd_pcm_t *, snd_pcm_hw_params_t *, unsigned int *))
|
|
dlsym(libasound,"snd_pcm_hw_params_set_channels_near"))) return 1;
|
|
if (!(alsa_pcm_hw_params_set_buffer_time_near = (int (*)(snd_pcm_t *, snd_pcm_hw_params_t *, unsigned int *, int *))
|
|
dlsym(libasound,"snd_pcm_hw_params_set_buffer_time_near"))) return 1;
|
|
if (!(alsa_pcm_hw_params_set_period_time_near = (int (*)(snd_pcm_t *, snd_pcm_hw_params_t *, unsigned int *, int *))
|
|
dlsym(libasound,"snd_pcm_hw_params_set_period_time_near"))) return 1;
|
|
if (!(alsa_pcm_hw_params_get_buffer_size = (int (*)(const snd_pcm_hw_params_t *, snd_pcm_uframes_t *))
|
|
dlsym(libasound,"snd_pcm_hw_params_get_buffer_size"))) return 1;
|
|
if (!(alsa_pcm_hw_params_get_period_size = (int (*)(const snd_pcm_hw_params_t *, snd_pcm_uframes_t *, int *))
|
|
dlsym(libasound,"snd_pcm_hw_params_get_period_size"))) return 1;
|
|
if (!(alsa_pcm_open = (int (*)(snd_pcm_t**, const char *, int, int))
|
|
dlsym(libasound,"snd_pcm_open"))) return 1;
|
|
if (!(alsa_pcm_close = (int (*)(snd_pcm_t*))
|
|
dlsym(libasound,"snd_pcm_close"))) return 1;
|
|
if (!(alsa_pcm_nonblock = (int (*)(snd_pcm_t*, int))
|
|
dlsym(libasound,"snd_pcm_nonblock"))) return 1;
|
|
if (!(alsa_pcm_drop = (int (*)(snd_pcm_t*))
|
|
dlsym(libasound,"snd_pcm_drop"))) return 1;
|
|
if (!(alsa_pcm_start = (int (*)(snd_pcm_t *))
|
|
dlsym(libasound,"snd_pcm_start"))) return 1;
|
|
if (!(alsa_pcm_writei = (snd_pcm_sframes_t (*)(snd_pcm_t*,const void*,snd_pcm_uframes_t))
|
|
dlsym(libasound,"snd_pcm_writei"))) return 1;
|
|
|
|
return 0;
|
|
}
|
|
|
|
static void ALSA_Unlink(void)
|
|
{
|
|
alsa_pcm_subformat_mask_malloc = NULL;
|
|
alsa_strerror = NULL;
|
|
alsa_pcm_resume = NULL;
|
|
alsa_pcm_prepare = NULL;
|
|
alsa_pcm_hw_params_any = NULL;
|
|
alsa_pcm_hw_params = NULL;
|
|
alsa_pcm_hw_params_current = NULL;
|
|
alsa_pcm_hw_params_set_access = NULL;
|
|
alsa_pcm_hw_params_set_format = NULL;
|
|
alsa_pcm_hw_params_set_rate_near = NULL;
|
|
alsa_pcm_hw_params_set_channels_near = NULL;
|
|
alsa_pcm_hw_params_set_buffer_time_near = NULL;
|
|
alsa_pcm_hw_params_set_period_time_near = NULL;
|
|
alsa_pcm_hw_params_get_buffer_size = NULL;
|
|
alsa_pcm_hw_params_get_period_size = NULL;
|
|
alsa_pcm_close = NULL;
|
|
alsa_pcm_nonblock = NULL;
|
|
alsa_pcm_drop = NULL;
|
|
alsa_pcm_start = NULL;
|
|
alsa_pcm_open = NULL;
|
|
alsa_pcm_writei = NULL;
|
|
|
|
if (libasound) {
|
|
dlclose(libasound);
|
|
libasound = NULL;
|
|
}
|
|
}
|
|
|
|
/* This is done to override the identifiers expanded
|
|
* in the macros provided by the ALSA includes which are
|
|
* not available.
|
|
* */
|
|
#define snd_strerror alsa_strerror
|
|
#define snd_pcm_sw_params_sizeof alsa_pcm_sw_params_sizeof
|
|
#define snd_pcm_hw_params_sizeof alsa_pcm_hw_params_sizeof
|
|
#endif /* MIKMOD_DYNAMIC */
|
|
|
|
static void ALSA_CommandLine(const CHAR *cmdline)
|
|
{
|
|
/* no options */
|
|
}
|
|
|
|
static BOOL ALSA_IsThere(void)
|
|
{
|
|
snd_pcm_subformat_mask_t *ptr = NULL;
|
|
BOOL retval;
|
|
|
|
#ifdef MIKMOD_DYNAMIC
|
|
if (ALSA_Link()) return 0;
|
|
#endif
|
|
retval = (alsa_pcm_subformat_mask_malloc(&ptr) == 0) && (ptr != NULL);
|
|
free(ptr);
|
|
#ifdef MIKMOD_DYNAMIC
|
|
ALSA_Unlink();
|
|
#endif
|
|
return retval;
|
|
}
|
|
|
|
static int ALSA_Init_internal(void)
|
|
{
|
|
snd_pcm_format_t pformat;
|
|
unsigned int btime = 250000; /* 250ms */
|
|
unsigned int ptime = 50000; /* 50ms */
|
|
snd_pcm_uframes_t psize;
|
|
snd_pcm_uframes_t bsize;
|
|
unsigned int rate, channels;
|
|
snd_pcm_hw_params_t * hwparams;
|
|
int err;
|
|
|
|
/* setup playback format structure */
|
|
pformat = (md_mode&DMODE_FLOAT)? SND_PCM_FORMAT_FLOAT :
|
|
(md_mode&DMODE_16BITS)? SND_PCM_FORMAT_S16 : SND_PCM_FORMAT_U8;
|
|
channels = (md_mode&DMODE_STEREO)?2:1;
|
|
rate = md_mixfreq;
|
|
|
|
#define MIKMOD_ALSA_DEVICE "default"
|
|
if ((err = alsa_pcm_open(&pcm_h, MIKMOD_ALSA_DEVICE, SND_PCM_STREAM_PLAYBACK, SND_PCM_NONBLOCK)) < 0) {
|
|
_mm_errno = MMERR_OPENING_AUDIO;
|
|
goto END;
|
|
}
|
|
|
|
/* Switch to blocking mode for playback */
|
|
/* Note: this must happen before hw/sw params are set. */
|
|
alsa_pcm_nonblock(pcm_h, 0);
|
|
|
|
snd_pcm_hw_params_alloca(&hwparams);
|
|
err = alsa_pcm_hw_params_any(pcm_h, hwparams);
|
|
if (err < 0) {
|
|
_mm_errno = MMERR_ALSA_NOCONFIG;
|
|
goto END;
|
|
}
|
|
|
|
err = alsa_pcm_hw_params_set_access(pcm_h, hwparams, SND_PCM_ACCESS_RW_INTERLEAVED);
|
|
if (!err) err = alsa_pcm_hw_params_set_format(pcm_h, hwparams, pformat);
|
|
if (!err) err = alsa_pcm_hw_params_set_rate_near(pcm_h, hwparams, &rate, NULL);
|
|
if (!err) err = alsa_pcm_hw_params_set_channels_near(pcm_h, hwparams, &channels);
|
|
if (!err) err = alsa_pcm_hw_params_set_buffer_time_near(pcm_h, hwparams, &btime, NULL);
|
|
if (!err) err = alsa_pcm_hw_params_set_period_time_near(pcm_h, hwparams, &ptime, NULL);
|
|
if (!err) err = alsa_pcm_hw_params(pcm_h, hwparams);
|
|
if (err < 0) {
|
|
_mm_errno = MMERR_ALSA_SETPARAMS;
|
|
goto END;
|
|
}
|
|
|
|
if (rate != md_mixfreq) {
|
|
_mm_errno = MMERR_ALSA_SETRATE;
|
|
goto END;
|
|
}
|
|
if (!(md_mode&DMODE_STEREO) && channels != 1) {
|
|
_mm_errno = MMERR_ALSA_SETCHANNELS;
|
|
goto END;
|
|
}
|
|
if ((md_mode&DMODE_STEREO) && channels != 2) {
|
|
_mm_errno = MMERR_ALSA_SETCHANNELS;
|
|
goto END;
|
|
}
|
|
|
|
err = alsa_pcm_hw_params_current(pcm_h, hwparams);
|
|
if (!err) err = alsa_pcm_hw_params_get_buffer_size(hwparams, &bsize);
|
|
if (!err) err = alsa_pcm_hw_params_get_period_size(hwparams, &psize, NULL);
|
|
if (err < 0) {
|
|
_mm_errno = MMERR_ALSA_BUFFERSIZE;
|
|
goto END;
|
|
}
|
|
|
|
period_size = psize;
|
|
global_frame_size = channels *
|
|
((md_mode&DMODE_FLOAT)? 4 : (md_mode&DMODE_16BITS)? 2 : 1);
|
|
|
|
if (!(audiobuffer=(SBYTE*)MikMod_malloc(period_size * global_frame_size))) {
|
|
_mm_errno = MMERR_OUT_OF_MEMORY;
|
|
goto END;
|
|
}
|
|
|
|
/* sound device is ready to work */
|
|
if (!VC_Init()) {
|
|
enabled = 1;
|
|
return 0;
|
|
}
|
|
END:
|
|
alsa_pcm_close(pcm_h);
|
|
pcm_h = NULL;
|
|
return 1;
|
|
}
|
|
|
|
static int ALSA_Init(void)
|
|
{
|
|
#ifdef HAVE_SSE2
|
|
/* TODO : Detect SSE2, then set md_mode |= DMODE_SIMDMIXER;*/
|
|
#endif
|
|
#ifdef MIKMOD_DYNAMIC
|
|
if (ALSA_Link()) {
|
|
_mm_errno=MMERR_DYNAMIC_LINKING;
|
|
return 1;
|
|
}
|
|
#endif
|
|
return ALSA_Init_internal();
|
|
}
|
|
|
|
static void ALSA_Exit_internal(void)
|
|
{
|
|
enabled = 0;
|
|
VC_Exit();
|
|
if (pcm_h) {
|
|
alsa_pcm_drop(pcm_h);
|
|
alsa_pcm_close(pcm_h);
|
|
pcm_h = NULL;
|
|
}
|
|
MikMod_free(audiobuffer);
|
|
audiobuffer = NULL;
|
|
}
|
|
|
|
static void ALSA_Exit(void)
|
|
{
|
|
ALSA_Exit_internal();
|
|
#ifdef MIKMOD_DYNAMIC
|
|
ALSA_Unlink();
|
|
#endif
|
|
}
|
|
|
|
/* snd_pcm_recover() is available in alsa-lib >= 1.0.11 */
|
|
static int xrun_recovery(snd_pcm_t *handle, int err)
|
|
{
|
|
if (err == -EINTR) return 0;
|
|
if (err == -EPIPE) { /* under-run */
|
|
err = alsa_pcm_prepare(handle);
|
|
if (err < 0) return err;
|
|
return 0;
|
|
}
|
|
else if (err == -ESTRPIPE) {
|
|
while ((err = alsa_pcm_resume(handle)) == -EAGAIN)
|
|
sleep(1); /* wait until the suspend flag is released */
|
|
if (err < 0) {
|
|
err = alsa_pcm_prepare(handle);
|
|
if (err < 0) return err;
|
|
}
|
|
return 0;
|
|
}
|
|
return err;
|
|
}
|
|
|
|
static void ALSA_Update(void)
|
|
{
|
|
int err;
|
|
|
|
if (!enabled) return;
|
|
|
|
if (bytes_written == 0 || bytes_played == bytes_written) {
|
|
bytes_written = VC_WriteBytes(audiobuffer,period_size * global_frame_size);
|
|
bytes_played = 0;
|
|
}
|
|
|
|
while (bytes_played < bytes_written)
|
|
{
|
|
err = alsa_pcm_writei(pcm_h, &audiobuffer[bytes_played], (bytes_written - bytes_played) / global_frame_size);
|
|
if (err == -EAGAIN)
|
|
continue;
|
|
if (err < 0) {
|
|
if ((err = xrun_recovery(pcm_h, err)) < 0) {
|
|
_mm_errno = MMERR_ALSA_PCM_RECOVER;
|
|
enabled = 0;
|
|
dbgprint(stderr, "Write error: %s\n", alsa_strerror(err));
|
|
}
|
|
break;
|
|
}
|
|
bytes_played += err * global_frame_size;
|
|
}
|
|
}
|
|
|
|
static int ALSA_PlayStart(void)
|
|
{
|
|
int err;
|
|
|
|
if (pcm_h == NULL) return 1;
|
|
err = alsa_pcm_prepare(pcm_h);
|
|
if (err == 0)
|
|
err = alsa_pcm_start(pcm_h);
|
|
if (err < 0) {
|
|
enabled = 0;
|
|
_mm_errno = MMERR_ALSA_PCM_START;
|
|
return 1;
|
|
}
|
|
|
|
return VC_PlayStart();
|
|
}
|
|
|
|
static void ALSA_PlayStop(void)
|
|
{
|
|
VC_PlayStop();
|
|
if (pcm_h) alsa_pcm_drop(pcm_h);
|
|
}
|
|
|
|
static int ALSA_Reset(void)
|
|
{
|
|
ALSA_Exit_internal();
|
|
return ALSA_Init_internal();
|
|
}
|
|
|
|
MIKMODAPI MDRIVER drv_alsa = {
|
|
NULL,
|
|
"ALSA",
|
|
"Advanced Linux Sound Architecture (ALSA) driver v1.11",
|
|
0,255,
|
|
"alsa",
|
|
NULL,
|
|
ALSA_CommandLine,
|
|
ALSA_IsThere,
|
|
VC_SampleLoad,
|
|
VC_SampleUnload,
|
|
VC_SampleSpace,
|
|
VC_SampleLength,
|
|
ALSA_Init,
|
|
ALSA_Exit,
|
|
ALSA_Reset,
|
|
VC_SetNumVoices,
|
|
ALSA_PlayStart,
|
|
ALSA_PlayStop,
|
|
ALSA_Update,
|
|
NULL,
|
|
VC_VoiceSetVolume,
|
|
VC_VoiceGetVolume,
|
|
VC_VoiceSetFrequency,
|
|
VC_VoiceGetFrequency,
|
|
VC_VoiceSetPanning,
|
|
VC_VoiceGetPanning,
|
|
VC_VoicePlay,
|
|
VC_VoiceStop,
|
|
VC_VoiceStopped,
|
|
VC_VoiceGetPosition,
|
|
VC_VoiceRealVolume
|
|
};
|
|
|
|
#else
|
|
|
|
MISSING(drv_alsa);
|
|
|
|
#endif
|
|
|
|
/* ex:set ts=8: */
|