feat(pipewire): begin implementation
This is adds partial (incomplete) PipeWire support (#106). Null Sink creation is still missing!
This commit is contained in:
parent
847857bd64
commit
697a05e55b
@ -54,13 +54,16 @@ find_package(Threads REQUIRED)
|
|||||||
target_link_libraries(soundux PRIVATE Threads::Threads ${CMAKE_DL_LIBS})
|
target_link_libraries(soundux PRIVATE Threads::Threads ${CMAKE_DL_LIBS})
|
||||||
|
|
||||||
if (UNIX)
|
if (UNIX)
|
||||||
|
list(APPEND CMAKE_MODULE_PATH ${CMAKE_CURRENT_SOURCE_DIR})
|
||||||
|
find_package(PipeWire REQUIRED)
|
||||||
|
|
||||||
find_package(PulseAudio)
|
find_package(PulseAudio)
|
||||||
find_package(X11 REQUIRED)
|
find_package(X11 REQUIRED)
|
||||||
include_directories(${X11_INCLUDE_DIR} ${PULSEAUDIO_INCLUDE_DIR})
|
target_include_directories(soundux SYSTEM PRIVATE ${X11_INCLUDE_DIR} ${PULSEAUDIO_INCLUDE_DIR} ${PipeWire_INCLUDE_DIRS} ${Spa_INCLUDE_DIRS})
|
||||||
|
|
||||||
find_package(PkgConfig REQUIRED)
|
find_package(PkgConfig REQUIRED)
|
||||||
pkg_check_modules(WNCK libwnck-3.0)
|
pkg_check_modules(WNCK libwnck-3.0)
|
||||||
target_link_libraries(soundux PRIVATE ${X11_LIBRARIES} ${X11_Xinput_LIB} ${X11_XTest_LIB})
|
target_link_libraries(soundux PRIVATE ${X11_LIBRARIES} ${X11_Xinput_LIB} ${X11_XTest_LIB} ${PipeWire_LIBRARIES})
|
||||||
endif()
|
endif()
|
||||||
if (WIN32)
|
if (WIN32)
|
||||||
target_compile_definitions(soundux PRIVATE _CRT_SECURE_NO_WARNINGS=1 _SILENCE_ALL_CXX17_DEPRECATION_WARNINGS=1 _UNICODE=1)
|
target_compile_definitions(soundux PRIVATE _CRT_SECURE_NO_WARNINGS=1 _SILENCE_ALL_CXX17_DEPRECATION_WARNINGS=1 _UNICODE=1)
|
||||||
|
122
FindPipeWire.cmake
Normal file
122
FindPipeWire.cmake
Normal file
@ -0,0 +1,122 @@
|
|||||||
|
#.rst:
|
||||||
|
# FindPipeWire
|
||||||
|
# -------
|
||||||
|
#
|
||||||
|
# Try to find PipeWire on a Unix system.
|
||||||
|
#
|
||||||
|
# This will define the following variables:
|
||||||
|
#
|
||||||
|
# ``PipeWire_FOUND``
|
||||||
|
# True if (the requested version of) PipeWire is available
|
||||||
|
# ``PipeWire_VERSION``
|
||||||
|
# The version of PipeWire
|
||||||
|
# ``PipeWire_LIBRARIES``
|
||||||
|
# This can be passed to target_link_libraries() instead of the ``PipeWire::PipeWire``
|
||||||
|
# target
|
||||||
|
# ``PipeWire_INCLUDE_DIRS``
|
||||||
|
# This should be passed to target_include_directories() if the target is not
|
||||||
|
# used for linking
|
||||||
|
# ``PipeWire_DEFINITIONS``
|
||||||
|
# This should be passed to target_compile_options() if the target is not
|
||||||
|
# used for linking
|
||||||
|
#
|
||||||
|
# If ``PipeWire_FOUND`` is TRUE, it will also define the following imported target:
|
||||||
|
#
|
||||||
|
# ``PipeWire::PipeWire``
|
||||||
|
# The PipeWire library
|
||||||
|
#
|
||||||
|
# In general we recommend using the imported target, as it is easier to use.
|
||||||
|
# Bear in mind, however, that if the target is in the link interface of an
|
||||||
|
# exported library, it must be made available by the package config file.
|
||||||
|
|
||||||
|
#=============================================================================
|
||||||
|
# Copyright 2014 Alex Merry <alex.merry@kde.org>
|
||||||
|
# Copyright 2014 Martin Gräßlin <mgraesslin@kde.org>
|
||||||
|
# Copyright 2018-2020 Jan Grulich <jgrulich@redhat.com>
|
||||||
|
#
|
||||||
|
# Redistribution and use in source and binary forms, with or without
|
||||||
|
# modification, are permitted provided that the following conditions
|
||||||
|
# are met:
|
||||||
|
#
|
||||||
|
# 1. Redistributions of source code must retain the copyright
|
||||||
|
# notice, this list of conditions and the following disclaimer.
|
||||||
|
# 2. Redistributions in binary form must reproduce the copyright
|
||||||
|
# notice, this list of conditions and the following disclaimer in the
|
||||||
|
# documentation and/or other materials provided with the distribution.
|
||||||
|
# 3. The name of the author may not be used to endorse or promote products
|
||||||
|
# derived from this software without specific prior written permission.
|
||||||
|
#
|
||||||
|
# THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``AS IS'' AND ANY EXPRESS OR
|
||||||
|
# IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES
|
||||||
|
# OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED.
|
||||||
|
# IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY DIRECT, INDIRECT,
|
||||||
|
# INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT
|
||||||
|
# NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
|
||||||
|
# DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
|
||||||
|
# THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
|
||||||
|
# (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF
|
||||||
|
# THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
||||||
|
#=============================================================================
|
||||||
|
|
||||||
|
# Use pkg-config to get the directories and then use these values
|
||||||
|
# in the FIND_PATH() and FIND_LIBRARY() calls
|
||||||
|
find_package(PkgConfig QUIET)
|
||||||
|
|
||||||
|
pkg_search_module(PKG_PipeWire QUIET libpipewire-0.3 libpipewire-0.2)
|
||||||
|
pkg_search_module(PKG_Spa QUIET libspa-0.2 libspa-0.1)
|
||||||
|
|
||||||
|
set(PipeWire_DEFINITIONS "${PKG_PipeWire_CFLAGS}" "${PKG_Spa_CFLAGS}")
|
||||||
|
set(PipeWire_VERSION "${PKG_PipeWire_VERSION}")
|
||||||
|
|
||||||
|
find_path(PipeWire_INCLUDE_DIRS
|
||||||
|
NAMES
|
||||||
|
pipewire/pipewire.h
|
||||||
|
HINTS
|
||||||
|
${PKG_PipeWire_INCLUDE_DIRS}
|
||||||
|
${PKG_PipeWire_INCLUDE_DIRS}/pipewire-0.3
|
||||||
|
)
|
||||||
|
|
||||||
|
find_path(Spa_INCLUDE_DIRS
|
||||||
|
NAMES
|
||||||
|
spa/param/props.h
|
||||||
|
HINTS
|
||||||
|
${PKG_Spa_INCLUDE_DIRS}
|
||||||
|
${PKG_Spa_INCLUDE_DIRS}/spa-0.2
|
||||||
|
)
|
||||||
|
|
||||||
|
find_library(PipeWire_LIBRARIES
|
||||||
|
NAMES
|
||||||
|
pipewire-0.3
|
||||||
|
pipewire-0.2
|
||||||
|
HINTS
|
||||||
|
${PKG_PipeWire_LIBRARY_DIRS}
|
||||||
|
)
|
||||||
|
|
||||||
|
include(FindPackageHandleStandardArgs)
|
||||||
|
find_package_handle_standard_args(PipeWire
|
||||||
|
FOUND_VAR
|
||||||
|
PipeWire_FOUND
|
||||||
|
REQUIRED_VARS
|
||||||
|
PipeWire_LIBRARIES
|
||||||
|
PipeWire_INCLUDE_DIRS
|
||||||
|
Spa_INCLUDE_DIRS
|
||||||
|
VERSION_VAR
|
||||||
|
PipeWire_VERSION
|
||||||
|
)
|
||||||
|
|
||||||
|
if(PipeWire_FOUND AND NOT TARGET PipeWire::PipeWire)
|
||||||
|
add_library(PipeWire::PipeWire UNKNOWN IMPORTED)
|
||||||
|
set_target_properties(PipeWire::PipeWire PROPERTIES
|
||||||
|
IMPORTED_LOCATION "${PipeWire_LIBRARIES}"
|
||||||
|
INTERFACE_COMPILE_OPTIONS "${PipeWire_DEFINITIONS}"
|
||||||
|
INTERFACE_INCLUDE_DIRECTORIES "${PipeWire_INCLUDE_DIRS};${Spa_INCLUDE_DIRS}"
|
||||||
|
)
|
||||||
|
endif()
|
||||||
|
|
||||||
|
mark_as_advanced(PipeWire_LIBRARIES PipeWire_INCLUDE_DIRS)
|
||||||
|
|
||||||
|
include(FeatureSummary)
|
||||||
|
set_package_properties(PipeWire PROPERTIES
|
||||||
|
URL "https://www.pipewire.org"
|
||||||
|
DESCRIPTION "PipeWire - multimedia processing"
|
||||||
|
)
|
398
src/helper/audio/linux/pipewire/pipewire.cpp
Normal file
398
src/helper/audio/linux/pipewire/pipewire.cpp
Normal file
@ -0,0 +1,398 @@
|
|||||||
|
#include "pipewire.hpp"
|
||||||
|
#include <memory>
|
||||||
|
#include <optional>
|
||||||
|
#include <stdexcept>
|
||||||
|
|
||||||
|
namespace Soundux::Objects
|
||||||
|
{
|
||||||
|
void PipeWire::sync()
|
||||||
|
{
|
||||||
|
spa_hook coreListener;
|
||||||
|
int pending = 0;
|
||||||
|
|
||||||
|
pw_core_events coreEvents = {};
|
||||||
|
coreEvents.version = PW_VERSION_CORE_EVENTS;
|
||||||
|
coreEvents.done = [](void *data, uint32_t id, int seq) {
|
||||||
|
auto *info = reinterpret_cast<std::pair<PipeWire *, int *> *>(data);
|
||||||
|
if (info)
|
||||||
|
{
|
||||||
|
if (id == PW_ID_CORE && seq == *info->second)
|
||||||
|
{
|
||||||
|
*info->second = -1;
|
||||||
|
pw_main_loop_quit(info->first->loop);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
spa_zero(coreListener);
|
||||||
|
auto data = std::make_pair(this, &pending);
|
||||||
|
pw_core_add_listener(core, &coreListener, &coreEvents, &data); // NOLINT
|
||||||
|
|
||||||
|
pending = pw_core_sync(core, PW_ID_CORE, 0); // NOLINT
|
||||||
|
while (pending != -1)
|
||||||
|
{
|
||||||
|
pw_main_loop_run(loop);
|
||||||
|
}
|
||||||
|
|
||||||
|
spa_hook_remove(&coreListener);
|
||||||
|
}
|
||||||
|
|
||||||
|
void PipeWire::onGlobalAdded(void *data, std::uint32_t id, [[maybe_unused]] std::uint32_t perms, const char *type,
|
||||||
|
[[maybe_unused]] std::uint32_t version, const spa_dict *props)
|
||||||
|
{
|
||||||
|
auto *thiz = reinterpret_cast<PipeWire *>(data);
|
||||||
|
if (thiz && data && props && type)
|
||||||
|
{
|
||||||
|
if (strcmp(type, PW_TYPE_INTERFACE_Port) == 0)
|
||||||
|
{
|
||||||
|
const auto *alias = spa_dict_lookup(props, "port.alias");
|
||||||
|
const auto *portName = spa_dict_lookup(props, "port.name");
|
||||||
|
|
||||||
|
if (alias && portName)
|
||||||
|
{
|
||||||
|
//* This is the only reliable way to get the name
|
||||||
|
//* PW_KEY_APP_NAME or PW_KEY_APP_PROCESS_BINARY are almost certainly never set.
|
||||||
|
auto name = std::string(alias);
|
||||||
|
name = name.substr(0, name.find_first_of(':'));
|
||||||
|
|
||||||
|
Direction direction;
|
||||||
|
if (strstr(portName, "FR"))
|
||||||
|
{
|
||||||
|
direction = Direction::FrontRight;
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
direction = Direction::FrontLeft;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (strstr(portName, "output"))
|
||||||
|
{
|
||||||
|
auto outputApp = std::make_shared<PipeWirePlaybackApp>();
|
||||||
|
|
||||||
|
outputApp->id = id;
|
||||||
|
outputApp->name = name;
|
||||||
|
outputApp->application = name;
|
||||||
|
outputApp->direction = direction;
|
||||||
|
|
||||||
|
std::lock_guard lock(thiz->playbackMutex);
|
||||||
|
thiz->playbackApps.emplace_back(outputApp);
|
||||||
|
}
|
||||||
|
else if (strstr(portName, "input"))
|
||||||
|
{
|
||||||
|
auto recordingApp = std::make_shared<PipeWireRecordingApp>();
|
||||||
|
|
||||||
|
recordingApp->id = id;
|
||||||
|
recordingApp->name = name;
|
||||||
|
recordingApp->application = name;
|
||||||
|
recordingApp->direction = direction;
|
||||||
|
|
||||||
|
std::lock_guard lock(thiz->recordingMutex);
|
||||||
|
thiz->recordingApps.emplace_back(recordingApp);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
else if (strcmp(type, PW_TYPE_INTERFACE_Link) == 0)
|
||||||
|
{
|
||||||
|
//* Should be used later to delete left over Soundux Links.
|
||||||
|
const auto *inputPort = spa_dict_lookup(props, "link.input.port");
|
||||||
|
const auto *outputPort = spa_dict_lookup(props, "link.output.port");
|
||||||
|
|
||||||
|
if (inputPort && outputPort)
|
||||||
|
{
|
||||||
|
const auto in = std::stol(inputPort);
|
||||||
|
const auto out = std::stol(outputPort);
|
||||||
|
std::lock_guard lock(thiz->playbackMutex);
|
||||||
|
|
||||||
|
for (auto &app : thiz->playbackApps)
|
||||||
|
{
|
||||||
|
auto pipeWireApp = std::dynamic_pointer_cast<PipeWirePlaybackApp>(app);
|
||||||
|
|
||||||
|
if (pipeWireApp->id == out)
|
||||||
|
{
|
||||||
|
pipeWireApp->links.emplace_back(Link{static_cast<std::uint32_t>(in)});
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
void PipeWire::onGlobalRemoved(void *data, std::uint32_t id)
|
||||||
|
{
|
||||||
|
auto *thiz = reinterpret_cast<PipeWire *>(data);
|
||||||
|
|
||||||
|
if (thiz)
|
||||||
|
{
|
||||||
|
thiz->playbackMutex.lock();
|
||||||
|
for (auto it = thiz->playbackApps.begin(); it != thiz->playbackApps.end();)
|
||||||
|
{
|
||||||
|
auto pipeWireApp = std::dynamic_pointer_cast<PipeWirePlaybackApp>(*it);
|
||||||
|
if (pipeWireApp && pipeWireApp->id == id)
|
||||||
|
{
|
||||||
|
it = thiz->playbackApps.erase(it);
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
it++;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
thiz->playbackMutex.unlock();
|
||||||
|
|
||||||
|
thiz->recordingMutex.lock();
|
||||||
|
for (auto it = thiz->recordingApps.begin(); it != thiz->recordingApps.end();)
|
||||||
|
{
|
||||||
|
auto pipeWireApp = std::dynamic_pointer_cast<PipeWireRecordingApp>(*it);
|
||||||
|
if (pipeWireApp && pipeWireApp->id == id)
|
||||||
|
{
|
||||||
|
it = thiz->recordingApps.erase(it);
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
it++;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
thiz->recordingMutex.unlock();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
void PipeWire::setup()
|
||||||
|
{
|
||||||
|
pw_init(nullptr, nullptr);
|
||||||
|
loop = pw_main_loop_new(nullptr);
|
||||||
|
if (!loop)
|
||||||
|
{
|
||||||
|
throw std::runtime_error("Failed to create main loop");
|
||||||
|
}
|
||||||
|
context = pw_context_new(pw_main_loop_get_loop(loop), nullptr, 0);
|
||||||
|
if (!context)
|
||||||
|
{
|
||||||
|
throw std::runtime_error("Failed to create context");
|
||||||
|
}
|
||||||
|
|
||||||
|
core = pw_context_connect(context, nullptr, 0);
|
||||||
|
if (!core)
|
||||||
|
{
|
||||||
|
throw std::runtime_error("Failed to connect context");
|
||||||
|
}
|
||||||
|
registry = pw_core_get_registry(core, PW_VERSION_REGISTRY, 0);
|
||||||
|
if (!registry)
|
||||||
|
{
|
||||||
|
throw std::runtime_error("Failed to get registry");
|
||||||
|
}
|
||||||
|
|
||||||
|
spa_zero(registryListener);
|
||||||
|
registryEvents.global = onGlobalAdded;
|
||||||
|
registryEvents.global_remove = onGlobalRemoved;
|
||||||
|
registryEvents.version = PW_VERSION_REGISTRY_EVENTS;
|
||||||
|
|
||||||
|
pw_registry_add_listener(registry, ®istryListener, ®istryEvents, this); // NOLINT
|
||||||
|
sync();
|
||||||
|
}
|
||||||
|
|
||||||
|
void PipeWire::destroy()
|
||||||
|
{
|
||||||
|
pw_proxy_destroy(reinterpret_cast<pw_proxy *>(registry));
|
||||||
|
pw_core_disconnect(core);
|
||||||
|
pw_context_destroy(context);
|
||||||
|
pw_main_loop_destroy(loop);
|
||||||
|
}
|
||||||
|
|
||||||
|
bool PipeWire::deleteLink(std::uint32_t id)
|
||||||
|
{
|
||||||
|
pw_registry_destroy(registry, id); // NOLINT
|
||||||
|
sync();
|
||||||
|
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
std::optional<int> PipeWire::linkPorts(std::uint32_t in, std::uint32_t out)
|
||||||
|
{
|
||||||
|
pw_properties *props = pw_properties_new(nullptr, nullptr);
|
||||||
|
|
||||||
|
pw_properties_set(props, PW_KEY_APP_NAME, "soundux");
|
||||||
|
// pw_properties_set(props, PW_KEY_OBJECT_LINGER, "true");
|
||||||
|
pw_properties_setf(props, PW_KEY_LINK_INPUT_PORT, "%u", in);
|
||||||
|
pw_properties_setf(props, PW_KEY_LINK_OUTPUT_PORT, "%u", out);
|
||||||
|
|
||||||
|
auto *proxy = reinterpret_cast<pw_proxy *>(
|
||||||
|
pw_core_create_object(core, "link-factory", PW_TYPE_INTERFACE_Link, PW_VERSION_LINK, &props->dict, 0));
|
||||||
|
|
||||||
|
if (!proxy)
|
||||||
|
{
|
||||||
|
pw_properties_free(props);
|
||||||
|
return std::nullopt;
|
||||||
|
}
|
||||||
|
|
||||||
|
spa_hook listener;
|
||||||
|
std::optional<std::uint32_t> result;
|
||||||
|
|
||||||
|
pw_proxy_events linkEvent = {};
|
||||||
|
linkEvent.version = PW_VERSION_PROXY_EVENTS;
|
||||||
|
linkEvent.bound = [](void *data, std::uint32_t id) {
|
||||||
|
*reinterpret_cast<std::optional<std::uint32_t> *>(data) = id;
|
||||||
|
};
|
||||||
|
linkEvent.error = [](void *data, [[maybe_unused]] int a, [[maybe_unused]] int b, const char *message) {
|
||||||
|
printf("Error: %s\n", message);
|
||||||
|
*reinterpret_cast<std::optional<std::uint32_t> *>(data) = std::nullopt;
|
||||||
|
};
|
||||||
|
|
||||||
|
spa_zero(listener);
|
||||||
|
pw_proxy_add_listener(proxy, &listener, &linkEvent, &result);
|
||||||
|
|
||||||
|
sync();
|
||||||
|
|
||||||
|
spa_hook_remove(&listener);
|
||||||
|
pw_properties_free(props);
|
||||||
|
pw_proxy_destroy(proxy);
|
||||||
|
|
||||||
|
return result;
|
||||||
|
}
|
||||||
|
|
||||||
|
std::vector<std::shared_ptr<RecordingApp>> PipeWire::getRecordingApps()
|
||||||
|
{
|
||||||
|
sync();
|
||||||
|
return recordingApps;
|
||||||
|
}
|
||||||
|
|
||||||
|
std::vector<std::shared_ptr<PlaybackApp>> PipeWire::getPlaybackApps()
|
||||||
|
{
|
||||||
|
sync();
|
||||||
|
std::lock_guard lock(playbackMutex);
|
||||||
|
std::vector<std::shared_ptr<PlaybackApp>> rtn;
|
||||||
|
|
||||||
|
for (const auto &app : playbackApps)
|
||||||
|
{
|
||||||
|
auto pipeWireApp = std::dynamic_pointer_cast<PipeWirePlaybackApp>(app);
|
||||||
|
if (!pipeWireApp->links.empty())
|
||||||
|
{
|
||||||
|
rtn.emplace_back(app);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return rtn;
|
||||||
|
}
|
||||||
|
|
||||||
|
std::shared_ptr<PlaybackApp> PipeWire::getPlaybackApp(const std::string &name)
|
||||||
|
{
|
||||||
|
std::lock_guard lock(playbackMutex);
|
||||||
|
for (const auto &app : playbackApps)
|
||||||
|
{
|
||||||
|
if (app->name == name)
|
||||||
|
{
|
||||||
|
return app;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return nullptr;
|
||||||
|
}
|
||||||
|
|
||||||
|
std::shared_ptr<RecordingApp> PipeWire::getRecordingApp(const std::string &name)
|
||||||
|
{
|
||||||
|
std::lock_guard lock(recordingMutex);
|
||||||
|
for (const auto &app : recordingApps)
|
||||||
|
{
|
||||||
|
if (app->name == name)
|
||||||
|
{
|
||||||
|
return app;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return nullptr;
|
||||||
|
}
|
||||||
|
|
||||||
|
bool PipeWire::useAsDefault()
|
||||||
|
{
|
||||||
|
// TODO(pipewire): Find a way to connect the output to the microphone
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
bool PipeWire::revertDefault()
|
||||||
|
{
|
||||||
|
// TODO(pipewire): Delete link created by `useAsDefault`
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
bool PipeWire::muteInput(bool state)
|
||||||
|
{
|
||||||
|
// TODO(pipewire): Maybe we could delete any link from the microphone to the output app and recreate it?
|
||||||
|
(void)state;
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
bool PipeWire::inputSoundTo(std::shared_ptr<RecordingApp> app)
|
||||||
|
{
|
||||||
|
std::vector<PipeWireRecordingApp> toMove;
|
||||||
|
|
||||||
|
recordingMutex.lock();
|
||||||
|
for (const auto &recordingApp : recordingApps)
|
||||||
|
{
|
||||||
|
if (recordingApp->name == app->name)
|
||||||
|
{
|
||||||
|
auto pipeWireApp = std::dynamic_pointer_cast<PipeWireRecordingApp>(recordingApp);
|
||||||
|
toMove.emplace_back(*pipeWireApp);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
recordingMutex.unlock();
|
||||||
|
|
||||||
|
for (const auto &pipeWireApp : toMove)
|
||||||
|
{
|
||||||
|
(void)pipeWireApp;
|
||||||
|
// TODO(pipewire): Link null sink to each app
|
||||||
|
// TODO(pipewire): Save id of each created link
|
||||||
|
}
|
||||||
|
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
bool PipeWire::stopSoundInput()
|
||||||
|
{
|
||||||
|
for (const auto &id : soundInputLinks)
|
||||||
|
{
|
||||||
|
deleteLink(id);
|
||||||
|
}
|
||||||
|
soundInputLinks.clear();
|
||||||
|
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
bool PipeWire::passthroughFrom(std::shared_ptr<PlaybackApp> app)
|
||||||
|
{
|
||||||
|
std::vector<PipeWirePlaybackApp> toMove;
|
||||||
|
|
||||||
|
playbackMutex.lock();
|
||||||
|
for (const auto &playbackApp : playbackApps)
|
||||||
|
{
|
||||||
|
if (playbackApp->name == app->name)
|
||||||
|
{
|
||||||
|
auto pipeWireApp = std::dynamic_pointer_cast<PipeWirePlaybackApp>(playbackApp);
|
||||||
|
toMove.emplace_back(*pipeWireApp);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
playbackMutex.unlock();
|
||||||
|
|
||||||
|
for (const auto &pipeWireApp : toMove)
|
||||||
|
{
|
||||||
|
(void)pipeWireApp;
|
||||||
|
// TODO(pipewire): Link each app to null sink
|
||||||
|
// TODO(pipewire): Save id of each created link
|
||||||
|
}
|
||||||
|
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
bool PipeWire::isCurrentlyPassingThrough()
|
||||||
|
{
|
||||||
|
return !passthroughLinks.empty();
|
||||||
|
}
|
||||||
|
|
||||||
|
bool PipeWire::stopPassthrough()
|
||||||
|
{
|
||||||
|
for (const auto &id : passthroughLinks)
|
||||||
|
{
|
||||||
|
deleteLink(id);
|
||||||
|
}
|
||||||
|
passthroughLinks.clear();
|
||||||
|
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
} // namespace Soundux::Objects
|
94
src/helper/audio/linux/pipewire/pipewire.hpp
Normal file
94
src/helper/audio/linux/pipewire/pipewire.hpp
Normal file
@ -0,0 +1,94 @@
|
|||||||
|
#include "../backend.hpp"
|
||||||
|
#include <mutex>
|
||||||
|
#include <optional>
|
||||||
|
#include <pipewire/core.h>
|
||||||
|
#include <pipewire/main-loop.h>
|
||||||
|
#include <pipewire/pipewire.h>
|
||||||
|
|
||||||
|
namespace Soundux
|
||||||
|
{
|
||||||
|
namespace Objects
|
||||||
|
{
|
||||||
|
enum class Direction : std::uint8_t
|
||||||
|
{
|
||||||
|
FrontLeft,
|
||||||
|
FrontRight
|
||||||
|
};
|
||||||
|
struct Link
|
||||||
|
{
|
||||||
|
std::uint32_t destination;
|
||||||
|
};
|
||||||
|
struct PipeWirePlaybackApp : public PlaybackApp
|
||||||
|
{
|
||||||
|
std::uint32_t id;
|
||||||
|
Direction direction;
|
||||||
|
std::vector<Link> links;
|
||||||
|
~PipeWirePlaybackApp() override = default;
|
||||||
|
};
|
||||||
|
|
||||||
|
struct PipeWireRecordingApp : public RecordingApp
|
||||||
|
{
|
||||||
|
std::uint32_t id;
|
||||||
|
Direction direction;
|
||||||
|
~PipeWireRecordingApp() override = default;
|
||||||
|
};
|
||||||
|
|
||||||
|
template <class T> struct MovedApp
|
||||||
|
{
|
||||||
|
T app;
|
||||||
|
std::vector<std::uint32_t> createdLinks;
|
||||||
|
};
|
||||||
|
|
||||||
|
class PipeWire : public AudioBackend
|
||||||
|
{
|
||||||
|
pw_core *core;
|
||||||
|
pw_main_loop *loop;
|
||||||
|
pw_context *context;
|
||||||
|
|
||||||
|
pw_registry *registry;
|
||||||
|
spa_hook registryListener;
|
||||||
|
pw_registry_events registryEvents;
|
||||||
|
|
||||||
|
std::mutex playbackMutex;
|
||||||
|
std::vector<std::shared_ptr<PlaybackApp>> playbackApps;
|
||||||
|
|
||||||
|
std::mutex recordingMutex;
|
||||||
|
std::vector<std::shared_ptr<RecordingApp>> recordingApps;
|
||||||
|
|
||||||
|
private:
|
||||||
|
std::vector<std::uint32_t> soundInputLinks;
|
||||||
|
std::vector<std::uint32_t> passthroughLinks;
|
||||||
|
|
||||||
|
private:
|
||||||
|
void sync();
|
||||||
|
bool deleteLink(std::uint32_t);
|
||||||
|
std::optional<int> linkPorts(std::uint32_t, std::uint32_t);
|
||||||
|
|
||||||
|
static void onGlobalRemoved(void *, std::uint32_t);
|
||||||
|
static void onGlobalAdded(void *, std::uint32_t, std::uint32_t, const char *, std::uint32_t,
|
||||||
|
const spa_dict *);
|
||||||
|
|
||||||
|
public:
|
||||||
|
PipeWire() = default;
|
||||||
|
void setup() override;
|
||||||
|
void destroy() override;
|
||||||
|
|
||||||
|
bool useAsDefault() override;
|
||||||
|
bool revertDefault() override;
|
||||||
|
bool muteInput(bool state) override;
|
||||||
|
|
||||||
|
bool stopPassthrough() override;
|
||||||
|
bool isCurrentlyPassingThrough() override;
|
||||||
|
bool passthroughFrom(std::shared_ptr<PlaybackApp> app) override;
|
||||||
|
|
||||||
|
bool stopSoundInput() override;
|
||||||
|
bool inputSoundTo(std::shared_ptr<RecordingApp> app) override;
|
||||||
|
|
||||||
|
std::shared_ptr<PlaybackApp> getPlaybackApp(const std::string &name) override;
|
||||||
|
std::shared_ptr<RecordingApp> getRecordingApp(const std::string &name) override;
|
||||||
|
|
||||||
|
std::vector<std::shared_ptr<PlaybackApp>> getPlaybackApps() override;
|
||||||
|
std::vector<std::shared_ptr<RecordingApp>> getRecordingApps() override;
|
||||||
|
};
|
||||||
|
} // namespace Objects
|
||||||
|
} // namespace Soundux
|
Loading…
x
Reference in New Issue
Block a user