Add sndio mixer plugin

This commit is contained in:
Christopher Zimmermann 2017-09-16 08:21:07 +02:00
parent ae941a7665
commit 030f87c90c
6 changed files with 125 additions and 5 deletions

View File

@ -1469,6 +1469,7 @@ if ENABLE_SNDIO
liboutput_plugins_a_SOURCES += \ liboutput_plugins_a_SOURCES += \
src/output/plugins/SndioOutputPlugin.cxx \ src/output/plugins/SndioOutputPlugin.cxx \
src/output/plugins/SndioOutputPlugin.hxx src/output/plugins/SndioOutputPlugin.hxx
libmixer_plugins_a_SOURCES += src/mixer/plugins/SndioMixerPlugin.cxx
endif endif
if ENABLE_HAIKU if ENABLE_HAIKU

View File

@ -310,7 +310,7 @@ input {
#audio_output { #audio_output {
# type "sndio" # type "sndio"
# name "sndio output" # name "sndio output"
# mixer_type "software" # mixer_type "hardware"
#} #}
# #
# An example of an OS X output: # An example of an OS X output:

View File

@ -36,5 +36,6 @@ extern const MixerPlugin osx_mixer_plugin;
extern const MixerPlugin roar_mixer_plugin; extern const MixerPlugin roar_mixer_plugin;
extern const MixerPlugin pulse_mixer_plugin; extern const MixerPlugin pulse_mixer_plugin;
extern const MixerPlugin winmm_mixer_plugin; extern const MixerPlugin winmm_mixer_plugin;
extern const MixerPlugin sndio_mixer_plugin;
#endif #endif

View File

@ -0,0 +1,63 @@
/*
* Copyright 2017 Christopher Zimmermann <christopher@gmerlin.de>
*
* 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
* (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.
*
* 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.,
* 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA.
*/
#include "config.h"
#include "mixer/MixerInternal.hxx"
#include "mixer/Listener.hxx"
#include "output/plugins/SndioOutputPlugin.hxx"
#include <sndio.h>
class SndioMixer final : public Mixer {
SndioOutput &output;
public:
SndioMixer(SndioOutput &_output, MixerListener &_listener)
:Mixer(sndio_mixer_plugin, _listener), output(_output)
{
output.RegisterMixerListener(this, &_listener);
}
/* virtual methods from class Mixer */
void Open() override {}
void Close() noexcept override {}
int GetVolume() override {
return output.GetVolume();
}
void SetVolume(unsigned volume) override {
output.SetVolume(volume);
}
};
static Mixer *
sndio_mixer_init(gcc_unused EventLoop &event_loop,
AudioOutput &ao,
MixerListener &listener,
gcc_unused const ConfigBlock &block)
{
return new SndioMixer((SndioOutput &)ao, listener);
}
const MixerPlugin sndio_mixer_plugin = {
sndio_mixer_init,
false,
};

View File

@ -19,6 +19,7 @@
#include "config.h" #include "config.h"
#include "SndioOutputPlugin.hxx" #include "SndioOutputPlugin.hxx"
#include "mixer/MixerList.hxx"
#include "util/Domain.hxx" #include "util/Domain.hxx"
#include "Log.hxx" #include "Log.hxx"
@ -47,10 +48,16 @@ SndioOutput::SndioOutput(const ConfigBlock &block)
:AudioOutput(0), :AudioOutput(0),
device(block.GetBlockValue("device", SIO_DEVANY)), device(block.GetBlockValue("device", SIO_DEVANY)),
buffer_time(block.GetBlockValue("buffer_time", buffer_time(block.GetBlockValue("buffer_time",
MPD_SNDIO_BUFFER_TIME_MS)) MPD_SNDIO_BUFFER_TIME_MS)),
raw_volume(SIO_MAXVOL)
{ {
} }
static void
VolumeCallback(void *arg, unsigned int volume) {
((SndioOutput *)arg)->VolumeChanged(volume);
}
AudioOutput * AudioOutput *
SndioOutput::Create(EventLoop &, const ConfigBlock &block) { SndioOutput::Create(EventLoop &, const ConfigBlock &block) {
return new SndioOutput(block); return new SndioOutput(block);
@ -125,6 +132,15 @@ SndioOutput::Open(AudioFormat &audio_format)
throw std::runtime_error("Requested audio params cannot be satisfied"); throw std::runtime_error("Requested audio params cannot be satisfied");
} }
// Set volume after opening fresh audio stream which does
// know nothing about previous audio streams.
sio_setvol(sio_hdl, raw_volume);
// sio_onvol returns 0 if no volume knob is available.
// This is the case on raw audio devices rather than
// the sndiod audio server.
if (sio_onvol(sio_hdl, VolumeCallback, this) == 0)
raw_volume = -1;
if (!sio_start(sio_hdl)) { if (!sio_start(sio_hdl)) {
sio_close(sio_hdl); sio_close(sio_hdl);
throw std::runtime_error("Failed to start audio device"); throw std::runtime_error("Failed to start audio device");
@ -148,9 +164,39 @@ SndioOutput::Play(const void *chunk, size_t size)
return n; return n;
} }
void
SndioOutput::SetVolume(unsigned int volume) {
sio_setvol(sio_hdl, volume * SIO_MAXVOL / 100);
}
static inline unsigned int
RawToPercent(int raw_volume) {
return raw_volume < 0 ? 100 : raw_volume * 100 / SIO_MAXVOL;
}
void
SndioOutput::VolumeChanged(int _raw_volume) {
if (raw_volume >= 0 && listener != nullptr && mixer != nullptr) {
raw_volume = _raw_volume;
listener->OnMixerVolumeChanged(*mixer,
RawToPercent(raw_volume));
}
}
unsigned int
SndioOutput::GetVolume() {
return RawToPercent(raw_volume);
}
void
SndioOutput::RegisterMixerListener(Mixer *_mixer, MixerListener *_listener) {
mixer = _mixer;
listener = _listener;
}
const struct AudioOutputPlugin sndio_output_plugin = { const struct AudioOutputPlugin sndio_output_plugin = {
"sndio", "sndio",
sndio_test_default_device, sndio_test_default_device,
&SndioOutput::Create, SndioOutput::Create,
nullptr, &sndio_mixer_plugin,
}; };

View File

@ -18,6 +18,7 @@
*/ */
#include "../OutputAPI.hxx" #include "../OutputAPI.hxx"
#include "mixer/Listener.hxx"
#ifndef MPD_SNDIO_OUTPUT_PLUGIN_HXX #ifndef MPD_SNDIO_OUTPUT_PLUGIN_HXX
#define MPD_SNDIO_OUTPUT_PLUGIN_HXX #define MPD_SNDIO_OUTPUT_PLUGIN_HXX
@ -25,9 +26,12 @@
extern const struct AudioOutputPlugin sndio_output_plugin; extern const struct AudioOutputPlugin sndio_output_plugin;
class SndioOutput final : AudioOutput { class SndioOutput final : AudioOutput {
Mixer *mixer = nullptr;
MixerListener *listener = nullptr;
const char *const device; const char *const device;
const unsigned buffer_time; /* in ms */ const unsigned buffer_time; /* in ms */
struct sio_hdl *sio_hdl; struct sio_hdl *sio_hdl;
int raw_volume;
public: public:
SndioOutput(const ConfigBlock &block); SndioOutput(const ConfigBlock &block);
@ -35,6 +39,11 @@ public:
static AudioOutput *Create(EventLoop &, static AudioOutput *Create(EventLoop &,
const ConfigBlock &block); const ConfigBlock &block);
void SetVolume(unsigned int _volume);
unsigned int GetVolume();
void VolumeChanged(int _volume);
void RegisterMixerListener(Mixer *_mixer, MixerListener *_listener);
private: private:
void Open(AudioFormat &audio_format) override; void Open(AudioFormat &audio_format) override;
void Close() noexcept override; void Close() noexcept override;