openbsd-ports/audio/mpg123/files/audio_openbsd.c
espie caaef9a183 Erm... forgot the actual patch... why does cvs tell me about missing files
after the commit ? so that I can either get careless, or waste
time/bandwidth by running each commit twice...

Anyways, files/audio_openbsd.c is a `caching' audio system.
It's much smarter than the old one: uses get encoding to map the reasonable
encodings, does not even cross-check alaw/ulaw when 8 bit/16 bit work okay.

It does keep audio opened all the way instead of closing/reopening all the
time.

and it avoids gratuitous assumptions about PC broken hardware, such as not
having support for 8 bits signed and other brain-damage.

Missing part: it won't convert endianess on the fly when the only `right'
modes have the wrong endianess... this might occur with e.g., a sb16 on
an alpha.
1999-03-23 21:38:26 +00:00

249 lines
5.5 KiB
C

#include <sys/types.h>
#include <stdio.h>
#include <unistd.h>
#include <fcntl.h>
#include <stdlib.h>
#include <sys/ioctl.h>
#include "mpg123.h"
#include <sys/audioio.h>
static void audio_set_format_helper(int fmt, audio_info_t *ainfo);
static int audio_fd = -1;
static int uptodate = 1;
static int possible = 0;
static audio_info_t ainfo;
static void really_close_audio()
{
if (audio_fd != -1)
close(audio_fd);
}
static void really_open_audio(struct audio_info_struct *ai)
{
if (audio_fd == -1) {
audio_encoding_t cap;
if(!ai->device) {
if(getenv("AUDIODEV")) {
if(param.verbose > 1)
fprintf(stderr,"Using audio-device value from AUDIODEV environment variable!\n");
ai->device = getenv("AUDIODEV");
}
else
ai->device = "/dev/audio";
}
audio_fd = open(ai->device, O_WRONLY);
for (cap.index = 0; ioctl(audio_fd, AUDIO_GETENC, &cap) == 0; cap.index++) {
if (cap.flags & AUDIO_ENCODINGFLAG_EMULATED)
continue;
switch(cap.encoding) {
case AUDIO_ENCODING_ULAW:
possible |= AUDIO_FORMAT_ULAW_8;
break;
case AUDIO_ENCODING_ALAW:
possible |= AUDIO_FORMAT_ALAW_8;
break;
case AUDIO_ENCODING_SLINEAR:
case AUDIO_ENCODING_SLINEAR_BE:
case AUDIO_ENCODING_SLINEAR_LE:
if (cap.precision == 8)
possible |= AUDIO_FORMAT_SIGNED_8;
else
possible |= AUDIO_FORMAT_SIGNED_16;
break;
case AUDIO_ENCODING_ULINEAR:
case AUDIO_ENCODING_ULINEAR_BE:
case AUDIO_ENCODING_ULINEAR_LE:
if (cap.precision == 8)
possible |= AUDIO_FORMAT_UNSIGNED_8;
else
possible |= AUDIO_FORMAT_UNSIGNED_16;
break;
}
}
atexit(really_close_audio);
}
}
int audio_open(struct audio_info_struct *ai)
{
really_open_audio(ai);
ai->fn = audio_fd;
if(ai->fn < 0)
return ai->fn;
AUDIO_INITINFO(&ainfo);
uptodate = 0;
{
audio_device_t ad;
if(ioctl(ai->fn, AUDIO_GETDEV, &ad) == -1)
return -1;
if(param.verbose > 1)
fprintf(stderr,"Audio device type: %s\n",ad.name);
}
if(audio_reset_parameters(ai) < 0) {
return -1;
}
return ai->fn;
}
int audio_reset_parameters(struct audio_info_struct *ai)
{
int ret;
if(ai->rate != -1)
ainfo.play.sample_rate = ai->rate;
if(ai->channels >= 0)
ainfo.play.channels = ai->channels;
audio_set_format_helper(ai->format,&ainfo);
uptodate = 0;
return 0;
}
int audio_rate_best_match(struct audio_info_struct *ai)
{
audio_info_t ainfo;
ainfo.play.sample_rate = ai->rate;
if(ioctl(ai->fn, AUDIO_SETINFO, &ainfo) < 0) {
ai->rate = 0;
return 0;
}
if(ioctl(ai->fn, AUDIO_GETINFO, &ainfo) < 0) {
return -1;
}
ai->rate = ainfo.play.sample_rate;
uptodate = 1;
return 0;
}
int audio_set_rate(struct audio_info_struct *ai)
{
audio_info_t ainfo;
if(ai->rate != -1) {
ainfo.play.sample_rate = ai->rate;
uptodate = 0;
return 0;
}
return -1;
}
int audio_set_channels(struct audio_info_struct *ai)
{
audio_info_t ainfo;
ainfo.play.channels = ai->channels;
uptodate = 0;
return 0;
}
static void audio_set_format_helper(int fmt, audio_info_t *ainfo)
{
switch(fmt) {
case -1:
case AUDIO_FORMAT_SIGNED_16:
default:
ainfo->play.encoding = AUDIO_ENCODING_SLINEAR;
ainfo->play.precision = 16;
break;
case AUDIO_FORMAT_UNSIGNED_16:
ainfo->play.encoding = AUDIO_ENCODING_ULINEAR;
ainfo->play.precision = 16;
break;
case AUDIO_FORMAT_UNSIGNED_8:
ainfo->play.encoding = AUDIO_ENCODING_ULINEAR;
ainfo->play.precision = 8;
break;
case AUDIO_FORMAT_SIGNED_8:
ainfo->play.encoding = AUDIO_ENCODING_SLINEAR;
ainfo->play.precision = 8;
break;
case AUDIO_FORMAT_ULAW_8:
ainfo->play.encoding = AUDIO_ENCODING_ULAW;
ainfo->play.precision = 8;
break;
case AUDIO_FORMAT_ALAW_8:
ainfo->play.encoding = AUDIO_ENCODING_ALAW;
ainfo->play.precision = 8;
break;
}
}
int audio_set_format(struct audio_info_struct *ai)
{
audio_set_format_helper(ai->format,&ainfo);
uptodate = 0;
return 0;
}
static int try_format(int fmt, struct audio_info_struct *ai)
{
audio_info_t ainfo;
AUDIO_INITINFO(&ainfo);
audio_set_format_helper(fmt, &ainfo);
ainfo.play.sample_rate = ai->rate;
ainfo.play.channels = ai->channels;
if(ioctl(audio_fd, AUDIO_SETINFO, &ainfo) >= 0)
return fmt;
else
return 0;
}
int audio_get_formats(struct audio_info_struct *ai)
{
int fmts = 0;
really_open_audio(ai);
if ((possible & AUDIO_FORMAT_SIGNED_16) == AUDIO_FORMAT_SIGNED_16)
fmts |= try_format(AUDIO_FORMAT_SIGNED_16, ai);
else if ((possible & AUDIO_FORMAT_UNSIGNED_16) == AUDIO_FORMAT_UNSIGNED_16)
fmts |= try_format(AUDIO_FORMAT_UNSIGNED_16, ai);
if ((possible & AUDIO_FORMAT_SIGNED_8) == AUDIO_FORMAT_SIGNED_8)
fmts |= try_format(AUDIO_FORMAT_SIGNED_8, ai);
else if ((possible & AUDIO_FORMAT_UNSIGNED_8) == AUDIO_FORMAT_UNSIGNED_8)
fmts |= try_format(AUDIO_FORMAT_UNSIGNED_8, ai);
if (!fmts && ((possible & AUDIO_FORMAT_ULAW_8) == AUDIO_FORMAT_ULAW_8))
fmts |= try_format(AUDIO_FORMAT_ULAW_8, ai);
if (!fmts && ((possible & AUDIO_FORMAT_ALAW_8) == AUDIO_FORMAT_ALAW_8))
fmts |= try_format(AUDIO_FORMAT_ALAW_8, ai);
return fmts;
}
int audio_play_samples(struct audio_info_struct *ai,unsigned char *buf,int len)
{
if (!uptodate) {
uptodate = 1;
ioctl(ai->fn, AUDIO_SETINFO, &ainfo);
}
return write(ai->fn,buf,len);
}
int audio_close(struct audio_info_struct *ai)
{
return 0;
}
void audio_queueflush (struct audio_info_struct *ai)
{
ioctl (ai->fn, AUDIO_FLUSH, 0);
}