openbsd-ports/audio/tracker/files/audio.c
jakemsr 56cbd2c6ca convert to sndio(7), which allows this port to be used with all
non-"isdn bullshit" audio devices OpenBSD currently (and most
likely ever will) supports.  sndio(7) doesn't support audioamd(4)
(which is an isdn device), but the audio(4) backend here didn't
support that device either.

tested on a 195 MHz r10k O2 extracting install tarballs +
'cvs co ports' + repeatedly compiling tracker in a loop +
'find /', all running simultaneously over the built-in mec0 NIC,
a zaurus with hw.setperf set to 0 + 'find /' running on the
console + tracker and an audio device monitoring program run
from an ssh session through a wi(4) pcmcia card, a 233MHz
Toshiba Tecra 8000 with wss(4), and various other machines.
2010-06-30 11:50:51 +00:00

213 lines
3.8 KiB
C

/* openbsd/audio.c
vi:ts=3 sw=3:
*/
/* sndio(7) interface */
#include "defs.h"
#include <unistd.h>
#include "extern.h"
#include "prefs.h"
#include "autoinit.h"
#include "watched_var.h"
struct options_set *port_options=0;
#define DEFAULT_BUFFERS
#define UNSIGNED8
#define DEFAULT_SET_MIX
#define NEW_OUTPUT_SAMPLES_AWARE
#define NEW_FUNCS
/* fine-tune to get the scrolling display in sync with the music */
#define ADVANCE_TAGS 20000
#include "Arch/common.c"
#include <sndio.h>
LOCAL unsigned long samples_max;
LOCAL long long realpos;
LOCAL struct sio_hdl *hdl;
LOCAL unsigned long current_freq;
unsigned long total;
LOCAL int dsp_samplesize = 0;
static void
movecb(void *arg, int delta)
{
realpos += delta * dsize * (stereo ? 2 : 1);
}
unsigned long open_audio(unsigned long f, int s)
{
struct sio_par par;
int buf_max;
hdl = sio_open(NULL, SIO_PLAY, 0);
if (hdl == NULL)
end_all("Error opening audio device");
realpos = 0;
sio_onmove(hdl, movecb, NULL);
sio_initpar(&par);
par.rate = f ? f : 22050;
if (!sio_setpar(hdl, &par) || !sio_getpar(hdl, &par) || !sio_start(hdl) ||
(par.bits != 8 && par.bits != 16))
end_all("Sorry, no audio format supported by this binary is available");
buf_max = par.appbufsz * par.bps * par.pchan;
current_freq = par.rate;
stereo = par.pchan == 2 ? 1 : 0;
dsp_samplesize = par.bits;
dsize = par.bps;
buffer = malloc(buf_max);
buffer16 = (short *)buffer;
idx = 0;
samples_max = buf_max / dsize / par.pchan;
set_watched_scalar(FREQUENCY, current_freq);
total = 0;
return current_freq;
}
/* synchronize stuff with audio output */
LOCAL struct tagged
{
struct tagged *next; /* simply linked list */
void (*f)(GENERIC p); /* function to call */
void (*f2)(GENERIC p); /* function to call for flush */
GENERIC p; /* and parameter */
unsigned long when; /* number of bytes to let go before calling */
}
*start, /* what still to output */
*end; /* where to add new tags */
/* flush_tags: use tags that have gone by recently */
LOCAL void flush_tags(void)
{
if (start)
{
while (start && start->when <= realpos + ADVANCE_TAGS)
{
struct tagged *tofree;
(*start->f)(start->p);
tofree = start;
start = start->next;
free(tofree);
}
}
}
/* remove unused tags at end */
LOCAL void remove_pending_tags(void)
{
while (start)
{
struct tagged *tofree;
(*start->f2)(start->p);
tofree = start;
start = start->next;
free(tofree);
}
}
void sync_audio(void (*function)(void *p), void (*f2)(void *p), void *parameter)
{
struct tagged *t;
if (hdl)
{
t = malloc(sizeof(struct tagged));
if (!t)
{
(*function)(parameter);
return;
}
/* build new tag */
t->next = 0;
t->f = function;
t->f2 = f2;
t->p = parameter;
t->when = total;
/* add it to list */
if (start)
end->next = t;
else
start = t;
end = t;
/* set up for next tag */
}
else
(*function)(parameter);
}
LOCAL void actually_flush_buffer(void)
{
int l,i;
if (idx)
{
total += idx * dsize;
sio_write(hdl, buffer, dsize * idx);
}
idx = 0;
}
void output_samples(long left, long right, int n)
{
if (idx >= samples_max - 1)
actually_flush_buffer();
switch(dsp_samplesize)
{
case 16: /* Cool! 16 bits samples */
add_samples16(left, right, n);
break;
case 8:
add_samples8(left, right, n);
break;
default: /* should not happen */
;
}
}
void flush_buffer(void)
{
actually_flush_buffer();
flush_tags();
}
/*
* Closing the Linux sound device waits for all pending samples to play.
*/
void close_audio(void)
{
actually_flush_buffer();
sio_close(hdl);
free(buffer);
}
unsigned long update_frequency(void)
{
return 0;
}
void discard_buffer(void)
{
remove_pending_tags();
total = 0;
}
void audio_ui(char c)
{
}