X-Git-Url: http://www.chiark.greenend.org.uk/ucgi/~mdw/git/disorder/blobdiff_plain/515cd2bb4f4ec548f464edeb5be7564fa5b0139f..63ad732fd2031c8775ddf98cd538c8759c4537df:/clients/playrtp-oss.c diff --git a/clients/playrtp-oss.c b/clients/playrtp-oss.c index 8a8f4c6..b40947d 100644 --- a/clients/playrtp-oss.c +++ b/clients/playrtp-oss.c @@ -1,6 +1,7 @@ /* * 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 * it under the terms of the GNU General Public License as published by @@ -18,26 +19,26 @@ * USA */ /** @file clients/playrtp-oss.c - * @brief RTP player - OSS support + * @brief RTP player - OSS and empeg support */ #include -#if HAVE_SYS_SOUNDCARD_H +#if HAVE_SYS_SOUNDCARD_H || EMPEG_HOST #include "types.h" #include #include +#if !EMPEG_HOST #include +#endif #include #include #include #include #include #include -#if HAVE_LINUX_EMPEG_H -# include -#endif +#include #include "mem.h" #include "log.h" @@ -61,6 +62,16 @@ 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) @@ -82,9 +93,10 @@ static void playrtp_oss_enable(void) { 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; - info("OSS buffer size %d", playrtp_oss_bufsize); nonblock(playrtp_oss_fd); } } @@ -100,6 +112,16 @@ static int playrtp_oss_flush(void) { /* 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); @@ -117,6 +139,15 @@ static int playrtp_oss_flush(void) { 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; } @@ -143,8 +174,11 @@ static void playrtp_oss_wait(void) { */ static void playrtp_oss_disable(int 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);