summaryrefslogtreecommitdiffstats
path: root/flow/audioiosndio.cpp
diff options
context:
space:
mode:
Diffstat (limited to 'flow/audioiosndio.cpp')
-rw-r--r--flow/audioiosndio.cpp310
1 files changed, 310 insertions, 0 deletions
diff --git a/flow/audioiosndio.cpp b/flow/audioiosndio.cpp
new file mode 100644
index 0000000..03efd1b
--- /dev/null
+++ b/flow/audioiosndio.cpp
@@ -0,0 +1,310 @@
+/*
+ * Copyright (c) 2008 Jacob Meuser <[email protected]>
+ *
+ * Permission to use, copy, modify, and distribute this software for any
+ * purpose with or without fee is hereby granted, provided that the above
+ * copyright notice and this permission notice appear in all copies.
+ *
+ * THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES
+ * WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF
+ * MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR
+ * ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES
+ * WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN
+ * ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF
+ * OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE.
+ */
+
+#ifdef HAVE_CONFIG_H
+ #include <config.h>
+#endif
+
+#ifdef HAVE_LIBSNDIO
+
+#include <sys/types.h>
+#include <errno.h>
+#include <math.h>
+
+#include <sndio.h>
+#include <poll.h>
+
+#include "debug.h"
+#include "audioio.h"
+#include "audiosubsys.h"
+#include "iomanager.h"
+#include "dispatcher.h"
+
+#include <cstdlib>
+#include <cstring>
+
+int bps, chans;
+long long realpos, playpos, recpos;
+
+void movecb(void *addr, int delta)
+{
+ realpos += delta * (int)(bps * chans);
+}
+
+namespace Arts {
+
+class AudioIOSNDIO : public AudioIO, public TimeNotify {
+
+protected:
+ struct sio_hdl *hdl;
+ struct sio_par par;
+ int bufsz;
+ int bufused;
+ int duplex;
+
+public:
+ AudioIOSNDIO();
+
+ void notifyTime();
+
+ void setParam(AudioParam param, int& value);
+ int getParam(AudioParam param);
+
+ bool open();
+ void close();
+ int read(void *buffer, int size);
+ int write(void *buffer, int size);
+};
+
+REGISTER_AUDIO_IO(AudioIOSNDIO,"sndio","libsndio");
+}
+
+using namespace std;
+using namespace Arts;
+
+AudioIOSNDIO::AudioIOSNDIO()
+{
+ /* default parameters */
+ param(samplingRate) = 48000;
+ paramStr(deviceName) = "";
+ param(fragmentSize) = 4096;
+ param(fragmentCount) = 4;
+ param(format) = 16;
+ param(channels) = 2;
+ param(direction) = directionWrite;
+}
+
+bool AudioIOSNDIO::open()
+{
+ string& _error = paramStr(lastError);
+ string& _deviceName = paramStr(deviceName);
+ int& _channels = param(channels);
+ int& _fragmentSize = param(fragmentSize);
+ int& _fragmentCount = param(fragmentCount);
+ int& _samplingRate = param(samplingRate);
+ int& _format = param(format);
+
+ struct sio_par testpar;
+ char dev[PATH_MAX];
+ int mode, bpf;
+
+ duplex = 0;
+ if (param(direction) == (directionRead | directionWrite)) {
+ mode = SIO_PLAY | SIO_REC;
+ duplex = 1;
+ } else if (param(direction) == directionWrite) {
+ mode = SIO_PLAY;
+ } else {
+ _error = "invalid direction";
+ return false;
+ }
+
+ strlcpy(dev, _deviceName.c_str(), PATH_MAX);
+
+ if (strcmp(dev, "") == 0)
+ hdl = sio_open(NULL, mode, 0);
+ else
+ hdl = sio_open(dev, mode, 0);
+
+ if (hdl == NULL) {
+ _error = "device ";
+ if (strcmp(_deviceName.c_str(), "") == 0)
+ _error += "(default sndio device)";
+ else
+ _error += _deviceName.c_str();
+ _error += " can't be opened";
+ return false;
+ }
+
+ sio_initpar(&par);
+
+ if (_format == 8) {
+ par.bits = 8;
+ par.sig = 0;
+ } else {
+ par.bits = 16;
+ par.sig = 1;
+ par.le = 1;
+ }
+
+ if (duplex)
+ par.pchan = par.rchan = _channels;
+ else
+ par.pchan = _channels;
+
+ par.rate = _samplingRate;
+
+ /* limit the buffer size for hardware constraints */
+
+ if (_fragmentSize > 1024*16)
+ _fragmentSize = 1024*16;
+
+ while (_fragmentSize * _fragmentCount > 1024*32)
+ _fragmentCount--;
+
+ bpf = ((par.bits / NBBY) * par.pchan);
+ par.round = _fragmentSize / bpf;
+ par.appbufsz = _fragmentSize * _fragmentCount / bpf;
+
+ testpar = par;
+
+ char details[128];
+
+ snprintf(details, 128, " rate=%d pchan=%d bits=%d le=%d sig=%d sz=%d",
+ par.rate, par.pchan, par.bits, par.le, par.sig, par.appbufsz);
+
+ if (!sio_setpar(hdl, &par)) {
+ _error = "sio_setpar failed:";
+ _error += details;
+
+ close();
+ return false;
+ }
+
+ if (!sio_getpar(hdl, &par)) {
+ _error = "sio_getpar failed";
+
+ close();
+ return false;
+ }
+
+ if (testpar.bits != par.bits ||
+ testpar.sig != par.sig ||
+ testpar.le != par.le ||
+ testpar.pchan != par.pchan ||
+ fabs((testpar.rate - par.rate)/testpar.rate) > 0.05) {
+ _error = "returned params do not match requested params";
+
+ close();
+ return false;
+ }
+
+ bpf = par.pchan * par.bps;
+ bufsz = par.bufsz * bpf;
+
+ ::bps = par.bps;
+ ::chans = par.pchan;
+ ::realpos = ::playpos = ::recpos = 0;
+
+ sio_onmove(hdl, ::movecb, NULL);
+
+ if (!sio_start(hdl)) {
+ _error = "sio_start failed";
+
+ close();
+ return false;
+ }
+
+ /* use a timer instead of select() */
+ Dispatcher::the()->ioManager()->addTimer(10, this);
+
+ return true;
+}
+
+void AudioIOSNDIO::close()
+{
+ if (hdl != NULL) {
+ sio_close(hdl);
+ hdl = NULL;
+ }
+}
+
+void AudioIOSNDIO::setParam(AudioParam p, int& value)
+{
+ param(p) = value;
+}
+
+int AudioIOSNDIO::getParam(AudioParam p)
+{
+ struct pollfd gpfd;
+ int n, events;
+
+ /* update ::realpos */
+ if ((p == canRead || p == canWrite) && hdl != NULL) {
+ events = POLLOUT;
+ if (duplex)
+ events |= POLLIN;
+ n = sio_pollfd(hdl, &gpfd, events);
+ while (poll(&gpfd, n, 0) < 0 && errno == EINTR)
+ ;
+ sio_revents(hdl, &gpfd);
+ }
+
+ switch(p) {
+ case canRead:
+ bufused = (::realpos - ::recpos < 0) ? 0 : ::realpos - ::recpos;
+ return bufused;
+ break;
+ case canWrite:
+ bufused = (::realpos < 0) ? ::playpos : ::playpos - ::realpos;
+ return bufsz - bufused;
+ break;
+ case autoDetect:
+ return 15;
+ break;
+ default:
+ return param(p);
+ break;
+ }
+}
+
+int AudioIOSNDIO::read(void *buffer, int size)
+{
+ arts_assert(hdl != NULL);
+
+ size_t result;
+
+ result = sio_read(hdl, buffer, size);
+
+ ::recpos += result;
+
+ return (int)result;
+}
+
+int AudioIOSNDIO::write(void *buffer, int size)
+{
+ arts_assert(hdl != NULL);
+
+ size_t result;
+
+ result = sio_write(hdl, buffer, size);
+
+ ::playpos += result;
+
+ return (int)result;
+}
+
+void AudioIOSNDIO::notifyTime()
+{
+ int& _direction = param(direction);
+
+ for (;;) {
+ int todo = 0;
+
+ if ((_direction & directionRead) && (getParam(canRead) > 0))
+ todo |= AudioSubSystem::ioRead;
+
+ if ((_direction & directionWrite) && (getParam(canWrite) > 0))
+ todo |= AudioSubSystem::ioWrite;
+
+ if (todo == 0)
+ return;
+
+ AudioSubSystem::the()->handleIO(todo);
+ }
+}
+
+#endif