Add sndio mixer plugin
This commit is contained in:
parent
ae941a7665
commit
030f87c90c
@ -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
|
||||||
|
@ -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:
|
||||||
|
@ -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
|
||||||
|
63
src/mixer/plugins/SndioMixerPlugin.cxx
Normal file
63
src/mixer/plugins/SndioMixerPlugin.cxx
Normal 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,
|
||||||
|
};
|
@ -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,
|
||||||
};
|
};
|
||||||
|
@ -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;
|
||||||
|
Loading…
Reference in New Issue
Block a user