Commit 96862403 authored by Max Kellermann's avatar Max Kellermann

mixer/pipewire: new plugin

parent b838bf31
/*
* Copyright 2003-2021 The Music Player Daemon Project
* http://www.musicpd.org
*
* 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 "PipeWireMixerPlugin.hxx"
#include "mixer/MixerInternal.hxx"
#include "mixer/Listener.hxx"
#include "output/plugins/PipeWireOutputPlugin.hxx"
#include <cmath>
class PipeWireMixer final : public Mixer {
PipeWireOutput &output;
int volume = 100;
public:
PipeWireMixer(PipeWireOutput &_output,
MixerListener &_listener) noexcept
:Mixer(pipewire_mixer_plugin, _listener),
output(_output)
{
}
PipeWireMixer(const PipeWireMixer &) = delete;
PipeWireMixer &operator=(const PipeWireMixer &) = delete;
void OnVolumeChanged(float new_volume) noexcept {
volume = std::lround(new_volume * 100.f);
listener.OnMixerVolumeChanged(*this, volume);
}
/* virtual methods from class Mixer */
void Open() override {
}
void Close() noexcept override {
}
int GetVolume() override;
void SetVolume(unsigned volume) override;
};
void
pipewire_mixer_on_change(PipeWireMixer &pm, float new_volume) noexcept
{
pm.OnVolumeChanged(new_volume);
}
int
PipeWireMixer::GetVolume()
{
return volume;
}
void
PipeWireMixer::SetVolume(unsigned new_volume)
{
pipewire_output_set_volume(output, float(new_volume) * 0.01f);
volume = new_volume;
}
static Mixer *
pipewire_mixer_init([[maybe_unused]] EventLoop &event_loop, AudioOutput &ao,
MixerListener &listener,
const ConfigBlock &)
{
auto &po = (PipeWireOutput &)ao;
return new PipeWireMixer(po, listener);
}
const MixerPlugin pipewire_mixer_plugin = {
pipewire_mixer_init,
true,
};
/*
* Copyright 2003-2021 The Music Player Daemon Project
* http://www.musicpd.org
*
* 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.
*/
#ifndef MPD_PIPEWIRE_MIXER_PLUGIN_HXX
#define MPD_PIPEWIRE_MIXER_PLUGIN_HXX
struct MixerPlugin;
class PipeWireMixer;
extern const MixerPlugin pipewire_mixer_plugin;
void
pipewire_mixer_on_change(PipeWireMixer &pm, float new_volume) noexcept;
#endif
...@@ -22,6 +22,10 @@ if is_darwin ...@@ -22,6 +22,10 @@ if is_darwin
mixer_plugins_sources += 'OSXMixerPlugin.cxx' mixer_plugins_sources += 'OSXMixerPlugin.cxx'
endif endif
if pipewire_dep.found()
mixer_plugins_sources += 'PipeWireMixerPlugin.cxx'
endif
if pulse_dep.found() if pulse_dep.found()
mixer_plugins_sources += 'PulseMixerPlugin.cxx' mixer_plugins_sources += 'PulseMixerPlugin.cxx'
endif endif
......
...@@ -21,6 +21,7 @@ ...@@ -21,6 +21,7 @@
#include "lib/pipewire/ThreadLoop.hxx" #include "lib/pipewire/ThreadLoop.hxx"
#include "../OutputAPI.hxx" #include "../OutputAPI.hxx"
#include "../Error.hxx" #include "../Error.hxx"
#include "mixer/plugins/PipeWireMixerPlugin.hxx"
#ifdef __GNUC__ #ifdef __GNUC__
#pragma GCC diagnostic push #pragma GCC diagnostic push
...@@ -32,6 +33,7 @@ ...@@ -32,6 +33,7 @@
#include <pipewire/pipewire.h> #include <pipewire/pipewire.h>
#include <spa/param/audio/format-utils.h> #include <spa/param/audio/format-utils.h>
#include <spa/param/props.h>
#ifdef __GNUC__ #ifdef __GNUC__
#pragma GCC diagnostic pop #pragma GCC diagnostic pop
...@@ -56,7 +58,18 @@ class PipeWireOutput final : AudioOutput { ...@@ -56,7 +58,18 @@ class PipeWireOutput final : AudioOutput {
const uint32_t target_id; const uint32_t target_id;
float volume = 1.0;
bool disconnected; bool disconnected;
/**
* Shall the previously known volume be restored as soon as
* PW_STREAM_STATE_STREAMING is reached? This needs to be
* done each time after the pw_stream got created, thus this
* flag gets set by Open().
*/
bool restore_volume;
bool interrupted; bool interrupted;
bool paused; bool paused;
bool drained; bool drained;
...@@ -80,6 +93,8 @@ public: ...@@ -80,6 +93,8 @@ public:
return events; return events;
} }
void SetVolume(float volume);
private: private:
void CheckThrowError() { void CheckThrowError() {
if (disconnected) if (disconnected)
...@@ -148,6 +163,20 @@ PipeWireOutput::PipeWireOutput(const ConfigBlock &block) ...@@ -148,6 +163,20 @@ PipeWireOutput::PipeWireOutput(const ConfigBlock &block)
} }
void void
PipeWireOutput::SetVolume(float _volume)
{
const PipeWire::ThreadLoopLock lock(thread_loop);
if (stream != nullptr && !restore_volume &&
pw_stream_set_control(stream,
SPA_PROP_volume, 1, &_volume,
0) != 0)
throw std::runtime_error("pw_stream_set_control() failed");
volume = _volume;
}
void
PipeWireOutput::Enable() PipeWireOutput::Enable()
{ {
thread_loop = pw_thread_loop_new(name, nullptr); thread_loop = pw_thread_loop_new(name, nullptr);
...@@ -282,6 +311,7 @@ void ...@@ -282,6 +311,7 @@ void
PipeWireOutput::Open(AudioFormat &audio_format) PipeWireOutput::Open(AudioFormat &audio_format)
{ {
disconnected = false; disconnected = false;
restore_volume = true;
paused = false; paused = false;
drained = true; drained = true;
...@@ -292,6 +322,8 @@ PipeWireOutput::Open(AudioFormat &audio_format) ...@@ -292,6 +322,8 @@ PipeWireOutput::Open(AudioFormat &audio_format)
PW_KEY_NODE_NAME, "mpd", PW_KEY_NODE_NAME, "mpd",
nullptr); nullptr);
const PipeWire::ThreadLoopLock lock(thread_loop);
stream = pw_stream_new_simple(pw_thread_loop_get_loop(thread_loop), stream = pw_stream_new_simple(pw_thread_loop_get_loop(thread_loop),
"mpd", "mpd",
props, props,
...@@ -317,7 +349,6 @@ PipeWireOutput::Open(AudioFormat &audio_format) ...@@ -317,7 +349,6 @@ PipeWireOutput::Open(AudioFormat &audio_format)
params[0] = spa_format_audio_raw_build(&pod_builder, params[0] = spa_format_audio_raw_build(&pod_builder,
SPA_PARAM_EnumFormat, &raw); SPA_PARAM_EnumFormat, &raw);
const PipeWire::ThreadLoopLock lock(thread_loop);
pw_stream_connect(stream, pw_stream_connect(stream,
PW_DIRECTION_OUTPUT, PW_DIRECTION_OUTPUT,
target_id, target_id,
...@@ -333,6 +364,7 @@ PipeWireOutput::Close() noexcept ...@@ -333,6 +364,7 @@ PipeWireOutput::Close() noexcept
{ {
const PipeWire::ThreadLoopLock lock(thread_loop); const PipeWire::ThreadLoopLock lock(thread_loop);
pw_stream_destroy(stream); pw_stream_destroy(stream);
stream = nullptr;
} }
delete ring_buffer; delete ring_buffer;
...@@ -347,6 +379,15 @@ PipeWireOutput::StateChanged(enum pw_stream_state state, ...@@ -347,6 +379,15 @@ PipeWireOutput::StateChanged(enum pw_stream_state state,
state == PW_STREAM_STATE_UNCONNECTED; state == PW_STREAM_STATE_UNCONNECTED;
if (!was_disconnected && disconnected) if (!was_disconnected && disconnected)
pw_thread_loop_signal(thread_loop, false); pw_thread_loop_signal(thread_loop, false);
if (state == PW_STREAM_STATE_STREAMING && restore_volume) {
/* restore the last known volume after creating a new
pw_stream */
restore_volume = false;
pw_stream_set_control(stream,
SPA_PROP_volume, 1, &volume,
0);
}
} }
inline void inline void
...@@ -442,5 +483,11 @@ const struct AudioOutputPlugin pipewire_output_plugin = { ...@@ -442,5 +483,11 @@ const struct AudioOutputPlugin pipewire_output_plugin = {
"pipewire", "pipewire",
nullptr, nullptr,
&PipeWireOutput::Create, &PipeWireOutput::Create,
nullptr, &pipewire_mixer_plugin,
}; };
void
pipewire_output_set_volume(PipeWireOutput &output, float volume)
{
output.SetVolume(volume);
}
...@@ -20,6 +20,11 @@ ...@@ -20,6 +20,11 @@
#ifndef MPD_PIPEWIRE_OUTPUT_PLUGIN_HXX #ifndef MPD_PIPEWIRE_OUTPUT_PLUGIN_HXX
#define MPD_PIPEWIRE_OUTPUT_PLUGIN_HXX #define MPD_PIPEWIRE_OUTPUT_PLUGIN_HXX
class PipeWireOutput;
extern const struct AudioOutputPlugin pipewire_output_plugin; extern const struct AudioOutputPlugin pipewire_output_plugin;
void
pipewire_output_set_volume(PipeWireOutput &output, float volume);
#endif #endif
Markdown is supported
0% or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment