ref: 2515cb2de68b10eff93b876159dcbf180474f146
dir: /src/sunaudio.c/
/* Direct to Sun Audio Driver * * Added by Chris Bagwell (cbagwell@sprynet.com) on 2/26/96 * Based on oss driver. * * Cleaned up changes of format somewhat in sunstartwrite on 03/31/98 * */ /* * Copyright 1997 Chris Bagwell And Sundry Contributors * This source code is freely redistributable and may be used for * any purpose. This copyright notice must be maintained. * Rick Richardson, Lance Norskog And Sundry Contributors are not * responsible for the consequences of using this software. */ #include "st_i.h" #ifdef HAVE_SUN_AUDIO #include <sys/ioctl.h> #include <sys/types.h> #ifdef HAVE_SUN_AUDIOIO_H #include <sun/audioio.h> #else #ifdef HAVE_SYS_AUDIOIO_H #include <sys/audioio.h> #endif #endif #include <errno.h> #if !defined(__NetBSD__) && !defined(__OpenBSD__) #include <stropts.h> #endif #include <unistd.h> #include <stdlib.h> #include <fcntl.h> #include <string.h> /* * Do anything required before you start reading samples. * Read file header. * Find out sampling rate, * size and encoding of samples, * mono/stereo/quad. */ static int st_sunstartread(ft_t ft) { st_fileinfo_t *file = (st_fileinfo_t *)ft->priv; st_size_t samplesize, encoding; audio_info_t audio_if; #ifdef __SVR4 audio_device_t audio_dev; #endif char simple_hw=0; /* Hard-code for now. */ file->count = 0; file->pos = 0; file->eof = 0; file->size = 1024; if ((file->buf = malloc (file->size)) == NULL) { st_fail_errno(ft,ST_ENOMEM,"unable to allocate input buffer of size %d", file->size); return ST_EOF; } if (ft->info.rate == 0.0) ft->info.rate = 8000; if (ft->info.size == -1) ft->info.size = ST_SIZE_BYTE; if (ft->info.encoding == ST_ENCODING_UNKNOWN) ft->info.encoding = ST_ENCODING_ULAW; #ifdef __SVR4 /* Read in old values, change to what we need and then send back */ if (ioctl(fileno(ft->fp), AUDIO_GETDEV, &audio_dev) < 0) { st_fail_errno(ft,errno,"Unable to get device information."); return(ST_EOF); } st_report("Hardware detected: %s",audio_dev.name); if (strcmp("SUNW,am79c30",audio_dev.name) == 0) { simple_hw = 1; } #endif /* If simple hardware detected in force data to ulaw. */ if (simple_hw) { if (ft->info.size == ST_SIZE_BYTE) { if (ft->info.encoding != ST_ENCODING_ULAW && ft->info.encoding != ST_ENCODING_ALAW) { st_report("Warning: Detected simple hardware. Forcing output to ULAW"); ft->info.encoding = ST_ENCODING_ULAW; } } else if (ft->info.size == ST_SIZE_WORD) { st_report("Warning: Detected simple hardware. Forcing output to ULAW"); ft->info.size = ST_SIZE_BYTE; ft->info.encoding = ST_ENCODING_ULAW; } } if (ft->info.size == ST_SIZE_BYTE) { samplesize = 8; if (ft->info.encoding != ST_ENCODING_ULAW && ft->info.encoding != ST_ENCODING_ALAW && ft->info.encoding != ST_ENCODING_SIGN2) { st_fail_errno(ft,ST_EFMT,"Sun audio driver only supports ULAW, ALAW, and signed linear for bytes."); return (ST_EOF); } if ((ft->info.encoding == ST_ENCODING_ULAW || ft->info.encoding == ST_ENCODING_ALAW) && ft->info.channels == 2) { st_report("Warning: only support mono for ULAW and ALAW data. Forcing to mono."); ft->info.channels = 1; } } else if (ft->info.size == ST_SIZE_WORD) { samplesize = 16; if (ft->info.encoding != ST_ENCODING_SIGN2) { st_fail_errno(ft,ST_EFMT,"Sun audio driver only supports signed linear for words."); return(ST_EOF); } } else { st_fail_errno(ft,ST_EFMT,"Sun audio driver only supports bytes and words"); return(ST_EOF); } if (ft->info.channels == 0) ft->info.channels = 1; else if (ft->info.channels > 1) { st_report("Warning: some Sun audio devices can not play stereo"); st_report("at all or sometimes only with signed words. If the"); st_report("sound seems sluggish then this is probably the case."); st_report("Try forcing output to signed words or use the avg"); st_report("filter to reduce the number of channels."); ft->info.channels = 2; } /* Read in old values, change to what we need and then send back */ if (ioctl(fileno(ft->fp), AUDIO_GETINFO, &audio_if) < 0) { st_fail_errno(ft,errno,"Unable to initialize /dev/audio"); return(ST_EOF); } audio_if.record.precision = samplesize; audio_if.record.channels = ft->info.channels; audio_if.record.sample_rate = ft->info.rate; if (ft->info.encoding == ST_ENCODING_ULAW) encoding = AUDIO_ENCODING_ULAW; else if (ft->info.encoding == ST_ENCODING_ALAW) encoding = AUDIO_ENCODING_ALAW; else encoding = AUDIO_ENCODING_LINEAR; audio_if.record.encoding = encoding; ioctl(fileno(ft->fp), AUDIO_SETINFO, &audio_if); if (audio_if.record.precision != samplesize) { st_fail_errno(ft,errno,"Unable to initialize sample size for /dev/audio"); return(ST_EOF); } if (audio_if.record.channels != ft->info.channels) { st_fail_errno(ft,errno,"Unable to initialize number of channels for /dev/audio"); return(ST_EOF); } if (audio_if.record.sample_rate != ft->info.rate) { st_fail_errno(ft,errno,"Unable to initialize rate for /dev/audio"); return(ST_EOF); } if (audio_if.record.encoding != encoding) { st_fail_errno(ft,errno,"Unable to initialize encoding for /dev/audio"); return(ST_EOF); } /* Flush any data in the buffers - its probably in the wrong format */ #if defined(__NetBSD__) || defined(__OpenBSD__) ioctl(fileno(ft->fp), AUDIO_FLUSH); #else ioctl(fileno(ft->fp), I_FLUSH, FLUSHR); #endif /* Change to non-buffered I/O*/ setvbuf(ft->fp, NULL, _IONBF, sizeof(char) * file->size); return (ST_SUCCESS); } static int st_sunstartwrite(ft_t ft) { st_fileinfo_t *file = (st_fileinfo_t *)ft->priv; st_size_t samplesize, encoding; audio_info_t audio_if; #ifdef __SVR4 audio_device_t audio_dev; #endif char simple_hw=0; /* Hard-code for now. */ file->count = 0; file->pos = 0; file->eof = 0; file->size = 1024; if ((file->buf = malloc (file->size)) == NULL) { st_fail_errno(ft,ST_ENOMEM,"unable to allocate output buffer of size %d", file->size); return(ST_EOF); } #ifdef __SVR4 /* Read in old values, change to what we need and then send back */ if (ioctl(fileno(ft->fp), AUDIO_GETDEV, &audio_dev) < 0) { st_fail_errno(ft,errno,"Unable to get device information."); return(ST_EOF); } st_report("Hardware detected: %s",audio_dev.name); if (strcmp("SUNW,am79c30",audio_dev.name) == 0) { simple_hw = 1; } #endif if (simple_hw) { if (ft->info.size == ST_SIZE_BYTE) { if (ft->info.encoding != ST_ENCODING_ULAW && ft->info.encoding != ST_ENCODING_ALAW) { st_report("Warning: Detected simple hardware. Forcing output to ULAW"); ft->info.encoding = ST_ENCODING_ULAW; } } else if (ft->info.size == ST_SIZE_WORD) { st_report("Warning: Detected simple hardware. Forcing output to ULAW"); ft->info.size = ST_SIZE_BYTE; ft->info.encoding = ST_ENCODING_ULAW; } } if (ft->info.rate == 0.0) ft->info.rate = 8000; if (ft->info.size == -1) ft->info.size = ST_SIZE_BYTE; if (ft->info.encoding == ST_ENCODING_UNKNOWN) ft->info.encoding = ST_ENCODING_ULAW; if (ft->info.size == ST_SIZE_BYTE) { samplesize = 8; if (ft->info.encoding != ST_ENCODING_ULAW && ft->info.encoding != ST_ENCODING_ALAW && ft->info.encoding != ST_ENCODING_SIGN2) { st_report("Sun Audio driver only supports ULAW, ALAW, and Signed Linear for bytes."); st_report("Forcing to ULAW"); ft->info.encoding = ST_ENCODING_ULAW; } if ((ft->info.encoding == ST_ENCODING_ULAW || ft->info.encoding == ST_ENCODING_ALAW) && ft->info.channels == 2) { st_report("Warning: only support mono for ULAW and ALAW data. Forcing to mono."); ft->info.channels = 1; } } else if (ft->info.size == ST_SIZE_WORD) { samplesize = 16; if (ft->info.encoding != ST_ENCODING_SIGN2) { st_report("Sun Audio driver only supports Signed Linear for words."); st_report("Forcing to Signed Linear"); ft->info.encoding = ST_ENCODING_SIGN2; } } else { st_report("Sun Audio driver only supports bytes and words"); ft->info.size = ST_SIZE_WORD; samplesize = 16; } if (ft->info.channels == 0) ft->info.channels = 1; else if (ft->info.channels > 1) ft->info.channels = 2; /* Read in old values, change to what we need and then send back */ if (ioctl(fileno(ft->fp), AUDIO_GETINFO, &audio_if) < 0) { st_fail_errno(ft,errno,"Unable to initialize /dev/audio"); return(ST_EOF); } audio_if.play.precision = samplesize; audio_if.play.channels = ft->info.channels; audio_if.play.sample_rate = ft->info.rate; if (ft->info.encoding == ST_ENCODING_ULAW) encoding = AUDIO_ENCODING_ULAW; else if (ft->info.encoding == ST_ENCODING_ALAW) encoding = AUDIO_ENCODING_ALAW; else encoding = AUDIO_ENCODING_LINEAR; audio_if.play.encoding = encoding; ioctl(fileno(ft->fp), AUDIO_SETINFO, &audio_if); if (audio_if.play.precision != samplesize) { st_fail_errno(ft,errno,"Unable to initialize sample size for /dev/audio"); return(ST_EOF); } if (audio_if.play.channels != ft->info.channels) { st_fail_errno(ft,errno,"Unable to initialize number of channels for /dev/audio"); return(ST_EOF); } if (audio_if.play.sample_rate != ft->info.rate) { st_fail_errno(ft,errno,"Unable to initialize rate for /dev/audio"); return(ST_EOF); } if (audio_if.play.encoding != encoding) { st_fail_errno(ft,errno,"Unable to initialize encoding for /dev/audio"); return(ST_EOF); } /* Change to non-buffered I/O */ setvbuf(ft->fp, NULL, _IONBF, sizeof(char) * file->size); return (ST_SUCCESS); } /* Sun /dev/audio player */ static const char *sunnames[] = { "sunau", NULL }; static st_format_t st_sun_format = { sunnames, NULL, ST_FILE_STEREO, st_sunstartread, st_rawread, st_rawstopread, st_sunstartwrite, st_rawwrite, st_rawstopwrite, st_format_nothing_seek }; const st_format_t *st_sun_format_fn(void) { return &st_sun_format; } #endif