summaryrefslogtreecommitdiff
path: root/drumgizmo
diff options
context:
space:
mode:
authorGoran Mekić <meka@tilda.center>2017-04-10 14:01:20 +0200
committerGoran Mekić <meka@tilda.center>2017-04-14 20:46:01 +0200
commit35ff1d5187da4a102406e035e8d5c02922266c16 (patch)
treeac3b91fb86cb3c2a6b4e299a4d402da2e5dcebcd /drumgizmo
parent21c0b85b57c68d591108d5f8efc1c2247c05c74f (diff)
Add OSS output support
Diffstat (limited to 'drumgizmo')
-rw-r--r--drumgizmo/Makefile.am6
-rw-r--r--drumgizmo/drumgizmoc.cc1
-rw-r--r--drumgizmo/enginefactory.cc9
-rw-r--r--drumgizmo/enginefactory.h5
-rw-r--r--drumgizmo/output/oss.cc151
-rw-r--r--drumgizmo/output/oss.h56
6 files changed, 228 insertions, 0 deletions
diff --git a/drumgizmo/Makefile.am b/drumgizmo/Makefile.am
index ee51e66..6fe9428 100644
--- a/drumgizmo/Makefile.am
+++ b/drumgizmo/Makefile.am
@@ -67,6 +67,12 @@ drumgizmo_SOURCES += output/jackaudio.cc
drumgizmo_CXXFLAGS += -DHAVE_OUTPUT_JACKAUDIO
endif # HAVE_OUTPUT_JACKAUDIO
+if HAVE_OUTPUT_OSS
+drumgizmo_SOURCES += output/oss.cc
+drumgizmo_CXXFLAGS += -DHAVE_OUTPUT_OSS
+endif # HAVE_OUTPUT_OSS
+
+
# Only compile jackclient.cc if at least one of the jack modules are included.
if HAVE_OUTPUT_JACKAUDIO
drumgizmo_SOURCES += jackclient.cc
diff --git a/drumgizmo/drumgizmoc.cc b/drumgizmo/drumgizmoc.cc
index 56b2367..01025e7 100644
--- a/drumgizmo/drumgizmoc.cc
+++ b/drumgizmo/drumgizmoc.cc
@@ -88,6 +88,7 @@ static const char usage_str[] =
" alsa: dev=<device> (default 'default'), frames=<frames> (default "
"32)\n"
" srate=<samplerate> (default 441000)\n"
+ " oss: dev=<device> (default '/dev/dsp'), srate=<samplerate>\n"
" wavfile: file=<filename> (default 'output'), srate=<samplerate> "
"(default 44100)\n"
" jackaudio:\n"
diff --git a/drumgizmo/enginefactory.cc b/drumgizmo/enginefactory.cc
index d013ef9..a1b8a0b 100644
--- a/drumgizmo/enginefactory.cc
+++ b/drumgizmo/enginefactory.cc
@@ -63,6 +63,9 @@ EngineFactory::EngineFactory()
#ifdef HAVE_OUTPUT_JACKAUDIO
output.push_back("jackaudio");
#endif
+#ifdef HAVE_OUTPUT_OSS
+ output.push_back("oss");
+#endif
}
#ifdef USE_JACK
@@ -146,6 +149,12 @@ std::unique_ptr<AudioOutputEngine> EngineFactory::createOutput(const std::string
return std::make_unique<JackAudioOutputEngine>(*jack);
}
#endif
+#ifdef HAVE_OUTPUT_OSS
+ if(name == "oss")
+ {
+ return std::make_unique<OSSOutputEngine>();
+ }
+#endif
// todo: add more engines
diff --git a/drumgizmo/enginefactory.h b/drumgizmo/enginefactory.h
index b2c0428..2a1457e 100644
--- a/drumgizmo/enginefactory.h
+++ b/drumgizmo/enginefactory.h
@@ -69,6 +69,11 @@
#include "output/jackaudio.h"
#endif
+#ifdef HAVE_OUTPUT_OSS
+#include "output/oss.h"
+#endif
+
+
//! Factory for various input- and output engines
class EngineFactory
{
diff --git a/drumgizmo/output/oss.cc b/drumgizmo/output/oss.cc
new file mode 100644
index 0000000..4e02e97
--- /dev/null
+++ b/drumgizmo/output/oss.cc
@@ -0,0 +1,151 @@
+/* -*- Mode: c++ -*- */
+/***************************************************************************
+ * oss.cc
+ *
+ * Tue Apr 11 19:42:45 CET 2017
+ * Copyright 2017 Goran Mekić
+ * meka@tilda.center
+ ****************************************************************************/
+
+/*
+ * This file is part of DrumGizmo.
+ *
+ * DrumGizmo is free software; you can redistribute it and/or modify
+ * it under the terms of the GNU Lesser General Public License as published by
+ * the Free Software Foundation; either version 3 of the License, or
+ * (at your option) any later version.
+ *
+ * DrumGizmo 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 Lesser General Public License for more details.
+ *
+ * You should have received a copy of the GNU Lesser General Public License
+ * along with DrumGizmo; if not, write to the Free Software
+ * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA.
+ */
+#include "oss.h"
+#include <sys/soundcard.h>
+#include <fcntl.h>
+#include <stdlib.h>
+#include <unistd.h>
+#include <iostream>
+
+
+OSSOutputEngine::OSSOutputEngine()
+ : dev{"/dev/dsp"}
+ , num_channels{16}
+ , srate{44100}
+ , format{AFMT_S32_NE}
+ , data{}
+{
+ data.clear();
+ data.resize(1024 * num_channels);
+}
+
+bool OSSOutputEngine::init(const Channels& channels)
+{
+ int tmp, mode = O_WRONLY;
+ num_channels = channels.size();
+
+ if((fd = open(dev.data(), mode, 0)) == -1)
+ {
+ std::cerr << dev.data() << ' ' << std::strerror(errno) << std::endl;
+ return false;
+ }
+
+ tmp = format;
+ if(ioctl(fd, SNDCTL_DSP_SETFMT, &tmp) == -1 || tmp != format)
+ {
+ std::cerr << "Setting audio format failed " << std::strerror(errno);
+ std::cerr << std::endl;
+ return false;
+ }
+
+ tmp = num_channels;
+ if(ioctl(fd, SNDCTL_DSP_CHANNELS, &tmp) == -1 || tmp != num_channels)
+ {
+ std::cerr << "Can not set number of channels to " << num_channels;
+ std::cerr << ": " << std::strerror(errno) << std::endl;
+ return false;
+ }
+
+ tmp = srate;
+ if(ioctl(fd, SNDCTL_DSP_SPEED, &tmp) == -1 || tmp != srate)
+ {
+ std::cerr << "Can not set sampling frequency to " << srate << ": ";
+ std::cerr << std::strerror(errno) << std::endl;
+ return false;
+ }
+
+ return true;
+}
+
+void OSSOutputEngine::setParm(const std::string& parm, const std::string& value)
+{
+ if(parm == "dev")
+ {
+ dev = value;
+ }
+ else if(parm == "srate")
+ {
+ try
+ {
+ srate = std::stoi(value);
+ }
+ catch(...)
+ {
+ std::cerr << "[OSSOutputEngine] Invalid samplerate ";
+ std::cerr << value << std::endl;
+ }
+ }
+ else
+ {
+ std::cerr << "[OSSOutputEngine] Unsupported parameter '";
+ std::cerr << parm << std::endl;
+ }
+}
+
+void OSSOutputEngine::run(int ch, sample_t* samples, size_t nsamples)
+{
+ // Convert to int format and
+ // write channel data in interleaved buffer
+ std::int32_t sample;
+ for(size_t i = 0; i < nsamples; ++i)
+ {
+ // Hard clip if needed
+ if(samples[i] > 1)
+ {
+ sample = INT32_MAX;
+ }
+ else if(samples[i] < -1)
+ {
+ sample = -INT32_MAX;
+ }
+ else
+ {
+ sample = samples[i] * INT32_MAX;
+ }
+ data[i * num_channels + ch] = sample;
+ }
+}
+
+void OSSOutputEngine::post(size_t nsamples)
+{
+ auto data_size = data.size() * sizeof(*data.data());
+ auto size_written = write(fd, data.data(), data_size);
+ if(size_written != data_size)
+ {
+ std::cerr << "Audio write: " << std::strerror(errno) << std::endl;
+ }
+}
+
+std::size_t OSSOutputEngine::getBufferSize() const
+{
+ return data.size() / num_channels;
+}
+
+std::size_t OSSOutputEngine::getSamplerate() const
+{
+ return srate;
+}
diff --git a/drumgizmo/output/oss.h b/drumgizmo/output/oss.h
new file mode 100644
index 0000000..d44cd1c
--- /dev/null
+++ b/drumgizmo/output/oss.h
@@ -0,0 +1,56 @@
+/* -*- Mode: c++ -*- */
+/***************************************************************************
+ * oss.h
+ *
+ * Tue Apr 11 19:42:45 CET 2017
+ * Copyright 2017 Goran Mekić
+ * meka@tilda.center
+ ****************************************************************************/
+
+/*
+ * This file is part of DrumGizmo.
+ *
+ * DrumGizmo is free software; you can redistribute it and/or modify
+ * it under the terms of the GNU Lesser General Public License as published by
+ * the Free Software Foundation; either version 3 of the License, or
+ * (at your option) any later version.
+ *
+ * DrumGizmo 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 Lesser General Public License for more details.
+ *
+ * You should have received a copy of the GNU Lesser General Public License
+ * along with DrumGizmo; if not, write to the Free Software
+ * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA.
+ */
+#pragma once
+#include "audiooutputengine.h"
+
+class OSSOutputEngine
+ : public AudioOutputEngine
+{
+public:
+ OSSOutputEngine();
+ ~OSSOutputEngine() {};
+
+ // based on AudioOutputEngine
+ bool init(const Channels& chan) override;
+ void setParm(const std::string& parm, const std::string& value) override;
+ bool start() override { return true; };
+ void stop() override {};
+ void pre(size_t nsamples) override { data.resize(nsamples * num_channels); };
+ void run(int ch, sample_t* samples, size_t nsamples) override;
+ void post(size_t nsamples) override;
+ bool isFreewheeling() const override { return false; };
+ std::size_t getBufferSize() const override;
+ std::size_t getSamplerate() const override;
+
+private:
+ std::string dev;
+ int fd;
+ std::size_t num_channels;
+ unsigned int srate;
+ unsigned int format;
+ std::vector<std::int32_t> data;
+};