X-Git-Url: http://www.chiark.greenend.org.uk/ucgi/~mdw/git/disorder/blobdiff_plain/cfa7dda12a65898b5eab2ed504fd765fef5c7527..c28e0bca3c88d7b9288397a9dc27f8a6e5b90c1a:/clients/playrtp-oss.c diff --git a/clients/playrtp-oss.c b/clients/playrtp-oss.c index a59e749..4b0ab83 100644 --- a/clients/playrtp-oss.c +++ b/clients/playrtp-oss.c @@ -1,40 +1,39 @@ /* * This file is part of DisOrder. * Copyright (C) 2007 Richard Kettlewell + * Portions copyright (C) 2007 Ross Younger * - * This program is free software; you can redistribute it and/or modify + * This program is free software: you can redistribute it and/or modify * it under the terms of the GNU General Public License as published by - * the Free Software Foundation; either version 2 of the License, or + * the Free Software Foundation, either version 3 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 - * General Public License for more details. - * + * + * 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 General Public License for more details. + * * You should have received a copy of the GNU General Public License - * along with this program; if not, write to the Free Software - * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 - * USA + * along with this program. If not, see . */ /** @file clients/playrtp-oss.c - * @brief RTP player - OSS support + * @brief RTP player - OSS and empeg support */ -#include +#include "common.h" -#if HAVE_SYS_SOUNDCARD_H -#include "types.h" +#if HAVE_SYS_SOUNDCARD_H || EMPEG_HOST #include #include +#if !EMPEG_HOST #include -#include +#endif #include -#include #include #include #include +#include #include "mem.h" #include "log.h" @@ -46,9 +45,28 @@ /** @brief /dev/dsp (or whatever) */ static int playrtp_oss_fd = -1; +/** @brief Audio buffer */ +static char *playrtp_oss_buffer; + +/** @brief Size of @ref playrtp_oss_buffer in bytes */ +static int playrtp_oss_bufsize; + +/** @brief Number of bytes used in @ref playrtp_oss_buffer */ +static int playrtp_oss_bufused; + /** @brief Open and configure the OSS audio device */ static void playrtp_oss_enable(void) { if(playrtp_oss_fd == -1) { +#if EMPEG_HOST + /* empeg audio driver only knows /dev/audio, only supports the equivalent + * of AFMT_S16_NE, has a fixed buffer size, and does not support the + * SNDCTL_ ioctls. */ + if(!device) + device = "/dev/audio"; + if((playrtp_oss_fd = open(device, O_WRONLY)) < 0) + fatal(errno, "error opening %s", device); + playrtp_oss_bufsize = 4608; +#else int rate = 44100, stereo = 1, format = AFMT_S16_BE; if(!device) { if(access("/dev/dsp", W_OK) == 0) @@ -68,10 +86,69 @@ static void playrtp_oss_enable(void) { fatal(errno, "ioctl SNDCTL_DSP_SPEED"); if(rate != 44100) error(0, "asking for 44100Hz, got %dHz", rate); + if(ioctl(playrtp_oss_fd, SNDCTL_DSP_GETBLKSIZE, &playrtp_oss_bufsize) < 0) + fatal(errno, "ioctl SNDCTL_DSP_GETBLKSIZE"); + info("OSS buffer size %d", playrtp_oss_bufsize); +#endif + playrtp_oss_buffer = xmalloc(playrtp_oss_bufsize); + playrtp_oss_bufused = 0; nonblock(playrtp_oss_fd); } } +/** @brief Flush the OSS output buffer + * @return 0 on success, non-0 on error + */ +static int playrtp_oss_flush(void) { + int nbyteswritten; + + if(!playrtp_oss_bufused) + return 0; /* nothing to do */ + /* 0 out the unused portion of the buffer */ + memset(playrtp_oss_buffer + playrtp_oss_bufused, 0, + playrtp_oss_bufsize - playrtp_oss_bufused); +#if EMPEG_HOST + /* empeg audio driver insists on native-endian samples */ + { + uint16_t *ptr, + *const limit = (uint16_t *)(playrtp_oss_buffer + playrtp_oss_bufused); + + for(ptr = (uint16_t *)playrtp_oss_buffer; ptr < limit; ++ptr) + *ptr = ntohs(*ptr); + } +#endif + for(;;) { + nbyteswritten = write(playrtp_oss_fd, + playrtp_oss_buffer, playrtp_oss_bufsize); + if(nbyteswritten < 0) { + switch(errno) { + case EINTR: + break; /* try again */ + case EAGAIN: + return 0; /* try later */ + default: + error(errno, "error writing to %s", device); + return -1; + } + } else { + if(nbyteswritten < playrtp_oss_bufsize) + error(0, "%s: short write (%d/%d)", + device, nbyteswritten, playrtp_oss_bufsize); + if(dump_buffer) { + int count; + const int16_t *sp = (const int16_t *)playrtp_oss_buffer; + + for(count = 0; count < playrtp_oss_bufsize; count += sizeof(int16_t)) { + dump_buffer[dump_index++] = (int16_t)ntohs(*sp++); + dump_index %= dump_size; + } + } + playrtp_oss_bufused = 0; + return 0; + } + } +} + /** @brief Wait until the audio device can accept more data */ static void playrtp_oss_wait(void) { struct pollfd fds[1]; @@ -91,35 +168,42 @@ static void playrtp_oss_wait(void) { * @param hard If nonzero, drop pending data */ static void playrtp_oss_disable(int hard) { - if(hard) + if(hard) { +#if !EMPEG_HOST + /* No SNDCTL_DSP_ ioctls on empeg */ if(ioctl(playrtp_oss_fd, SNDCTL_DSP_RESET, 0) < 0) error(errno, "ioctl SNDCTL_DSP_RESET"); +#endif + } else + playrtp_oss_flush(); xclose(playrtp_oss_fd); playrtp_oss_fd = -1; + free(playrtp_oss_buffer); + playrtp_oss_buffer = 0; } /** @brief Write samples to OSS output device * @param data Pointer to sample data - * @param nsamples Number of samples + * @param samples Number of samples * @return 0 on success, non-0 on error */ -static int playrtp_oss_write(const void *data, size_t samples) { - const ssize_t nbyteswritten = write(playrtp_oss_fd, data, - samples * sizeof (int16_t)); - - if(nbyteswritten < 0) { - switch(errno) { - case EAGAIN: - case EINTR: - return 0; - default: - error(errno, "error writing to %s", device); - return -1; - } - } else { - next_timestamp += nbyteswritten / 2; - return 0; +static int playrtp_oss_write(const char *data, size_t samples) { + long bytes = samples * sizeof(int16_t); + while(bytes > 0) { + int n = playrtp_oss_bufsize - playrtp_oss_bufused; + + if(n > bytes) + n = bytes; + memcpy(playrtp_oss_buffer + playrtp_oss_bufused, data, n); + bytes -= n; + data += n; + playrtp_oss_bufused += n; + if(playrtp_oss_bufused == playrtp_oss_bufsize) + if(playrtp_oss_flush()) + return -1; } + next_timestamp += samples; + return 0; } /** @brief Play some data from packet @p p @@ -127,8 +211,9 @@ static int playrtp_oss_write(const void *data, size_t samples) { * @p p is assumed to contain @ref next_timestamp. */ static int playrtp_oss_play(const struct packet *p) { - return playrtp_oss_write(p->samples_raw + next_timestamp - p->timestamp, - (p->timestamp + p->nsamples) - next_timestamp); + return playrtp_oss_write + ((const char *)(p->samples_raw + next_timestamp - p->timestamp), + (p->timestamp + p->nsamples) - next_timestamp); } /** @brief Play some silence before packet @p p @@ -136,7 +221,7 @@ static int playrtp_oss_play(const struct packet *p) { * @p p is assumed to be entirely before @ref next_timestamp. */ static int playrtp_oss_infill(const struct packet *p) { - static const uint16_t zeros[INFILL_SAMPLES]; + static const char zeros[INFILL_SAMPLES * sizeof(int16_t)]; size_t samples_available = INFILL_SAMPLES; if(p && samples_available > p->timestamp - next_timestamp)