pulse-server: add alsa source and sink module

See #3456
This commit is contained in:
Wim Taymans 2023-08-23 11:14:30 +02:00
parent 293e1f85e6
commit dbf1ea0e78
3 changed files with 480 additions and 0 deletions

View file

@ -356,6 +356,8 @@ pipewire_module_protocol_pulse_sources = [
'module-protocol-pulse/stream.c',
'module-protocol-pulse/utils.c',
'module-protocol-pulse/volume.c',
'module-protocol-pulse/modules/module-alsa-sink.c',
'module-protocol-pulse/modules/module-alsa-source.c',
'module-protocol-pulse/modules/module-always-sink.c',
'module-protocol-pulse/modules/module-combine-sink.c',
'module-protocol-pulse/modules/module-echo-cancel.c',

View file

@ -0,0 +1,239 @@
/* PipeWire */
/* SPDX-FileCopyrightText: Copyright © 2023 Wim Taymans <wim.taymans@gmail.com> */
/* SPDX-License-Identifier: MIT */
#include <pipewire/pipewire.h>
#include "../manager.h"
#include "../module.h"
#define NAME "alsa-sink"
#define DEFAULT_DEVICE "default"
PW_LOG_TOPIC_STATIC(mod_topic, "mod." NAME);
#define PW_LOG_TOPIC_DEFAULT mod_topic
struct module_alsa_sink_data {
struct pw_core *core;
struct spa_hook core_listener;
struct pw_proxy *proxy;
struct spa_hook proxy_listener;
};
static void module_alsa_sink_proxy_removed(void *data)
{
struct module *module = data;
struct module_alsa_sink_data *d = module->user_data;
pw_proxy_destroy(d->proxy);
}
static void module_alsa_sink_proxy_destroy(void *data)
{
struct module *module = data;
struct module_alsa_sink_data *d = module->user_data;
pw_log_info("proxy %p destroy", d->proxy);
spa_hook_remove(&d->proxy_listener);
d->proxy = NULL;
module_schedule_unload(module);
}
static void module_alsa_sink_proxy_bound_props(void *data, uint32_t global_id, const struct spa_dict *props)
{
struct module *module = data;
struct module_alsa_sink_data *d = module->user_data;
pw_log_info("proxy %p bound", d->proxy);
module_emit_loaded(module, 0);
}
static void module_alsa_sink_proxy_error(void *data, int seq, int res, const char *message)
{
struct module *module = data;
struct module_alsa_sink_data *d = module->user_data;
pw_log_info("proxy %p error %d", d->proxy, res);
pw_proxy_destroy(d->proxy);
}
static const struct pw_proxy_events proxy_events = {
PW_VERSION_PROXY_EVENTS,
.removed = module_alsa_sink_proxy_removed,
.bound_props = module_alsa_sink_proxy_bound_props,
.error = module_alsa_sink_proxy_error,
.destroy = module_alsa_sink_proxy_destroy,
};
static void module_alsa_sink_core_error(void *data, uint32_t id, int seq, int res, const char *message)
{
struct module *module = data;
pw_log_warn("error id:%u seq:%d res:%d (%s): %s",
id, seq, res, spa_strerror(res), message);
if (id == PW_ID_CORE && res == -EPIPE)
module_schedule_unload(module);
}
static const struct pw_core_events core_events = {
PW_VERSION_CORE_EVENTS,
.error = module_alsa_sink_core_error,
};
static int module_alsa_sink_load(struct module *module)
{
struct module_alsa_sink_data *d = module->user_data;
d->core = pw_context_connect(module->impl->context, NULL, 0);
if (d->core == NULL)
return -errno;
pw_core_add_listener(d->core, &d->core_listener, &core_events, module);
pw_properties_setf(module->props, "pulse.module.id", "%u", module->index);
d->proxy = pw_core_create_object(d->core,
"adapter", PW_TYPE_INTERFACE_Node, PW_VERSION_NODE,
module->props ? &module->props->dict : NULL, 0);
if (d->proxy == NULL)
return -errno;
pw_proxy_add_listener(d->proxy, &d->proxy_listener, &proxy_events, module);
return SPA_RESULT_RETURN_ASYNC(0);
}
static int module_alsa_sink_unload(struct module *module)
{
struct module_alsa_sink_data *d = module->user_data;
if (d->proxy != NULL) {
spa_hook_remove(&d->proxy_listener);
pw_proxy_destroy(d->proxy);
d->proxy = NULL;
}
if (d->core != NULL) {
spa_hook_remove(&d->core_listener);
pw_core_disconnect(d->core);
d->core = NULL;
}
return 0;
}
static const struct spa_dict_item module_alsa_sink_info[] = {
{ PW_KEY_MODULE_AUTHOR, "Wim Taymans <wim.taymans@gmail.com>" },
{ PW_KEY_MODULE_DESCRIPTION, "An ALSA sink" },
{ PW_KEY_MODULE_USAGE,
"name=<name of the sink, to be prefixed> "
"sink_name=<name for the sink> "
"sink_properties=<properties for the sink> "
"namereg_fail=<when false attempt to synthesise new sink_name if it is already taken> "
"device=<ALSA device> "
"device_id=<ALSA card index> "
"format=<sample format> "
"rate=<sample rate> "
"alternate_rate=<alternate sample rate> "
"channels=<number of channels> "
"channel_map=<channel map> "
"fragments=<number of fragments> "
"fragment_size=<fragment size> "
"mmap=<enable memory mapping?> "
"tsched=<enable system timer based scheduling mode?> "
"tsched_buffer_size=<buffer size when using timer based scheduling> "
"tsched_buffer_watermark=<lower fill watermark> "
"ignore_dB=<ignore dB information from the device?> "
"control=<name of mixer control, or name and index separated by a comma> "
"rewind_safeguard=<number of bytes that cannot be rewound> "
"deferred_volume=<Synchronize software and hardware volume changes to avoid momentary jumps?> "
"deferred_volume_safety_margin=<usec adjustment depending on volume direction> "
"deferred_volume_extra_delay=<usec adjustment to HW volume changes> "
"fixed_latency_range=<disable latency range changes on underrun?>" },
{ PW_KEY_MODULE_VERSION, PACKAGE_VERSION },
};
static int module_alsa_sink_prepare(struct module * const module)
{
struct pw_properties * const props = module->props;
const char *str, *dev_id;
struct spa_audio_info_raw info = { 0 };
PW_LOG_TOPIC_INIT(mod_topic);
dev_id = pw_properties_get(props, "device_id");
if (dev_id == NULL)
dev_id = pw_properties_get(props, "device");
if (dev_id == NULL)
dev_id = DEFAULT_DEVICE;
pw_properties_set(props, "api.alsa.path", dev_id);
if ((str = pw_properties_get(props, "sink_name")) != NULL) {
pw_properties_set(props, PW_KEY_NODE_NAME, str);
pw_properties_set(props, "sink_name", NULL);
}
else if ((str = pw_properties_get(props, "name")) != NULL) {
pw_properties_setf(props, PW_KEY_NODE_NAME, "alsa_output.%s", str);
pw_properties_set(props, "name", NULL);
}
else {
pw_properties_setf(props, PW_KEY_NODE_NAME, "alsa_output.%s", dev_id);
}
if ((str = pw_properties_get(props, "sink_properties")) != NULL) {
module_args_add_props(props, str);
pw_properties_set(props, "sink_properties", NULL);
}
if ((str = pw_properties_get(props, "fragments")) != NULL) {
pw_properties_set(props, "api.alsa.period-num", str);
pw_properties_set(props, "fragments", NULL);
}
if ((str = pw_properties_get(props, "fragment_size")) != NULL) {
pw_properties_set(props, "api.alsa.period-size", str);
pw_properties_set(props, "fragment_size", NULL);
}
if ((str = pw_properties_get(props, "mmap")) != NULL) {
pw_properties_setf(props, "api.alsa.disable-mmap",
spa_atob(str) ? "false" : "true");
pw_properties_set(props, "mmap", NULL);
}
if (module_args_to_audioinfo_keys(module->impl, props,
"format", "rate", "channels", "channel_map", &info) < 0)
return -EINVAL;
audioinfo_to_properties(&info, props);
if (pw_properties_get(props, PW_KEY_MEDIA_CLASS) == NULL)
pw_properties_set(props, PW_KEY_MEDIA_CLASS, "Audio/Sink");
if ((str = pw_properties_get(props, PW_KEY_NODE_DESCRIPTION)) == NULL)
pw_properties_setf(props, PW_KEY_NODE_DESCRIPTION,
"ALSA Sink on %s", dev_id);
pw_properties_set(props, PW_KEY_FACTORY_NAME, "api.alsa.pcm.sink");
if (pw_properties_get(props, "monitor.channel-volumes") == NULL)
pw_properties_set(props, "monitor.channel-volumes", "true");
if (pw_properties_get(props, "node.suspend-on-idle") == NULL)
pw_properties_set(props, "node.suspend-on-idle", "true");
return 0;
}
DEFINE_MODULE_INFO(module_alsa_sink) = {
.name = "module-alsa-sink",
.prepare = module_alsa_sink_prepare,
.load = module_alsa_sink_load,
.unload = module_alsa_sink_unload,
.properties = &SPA_DICT_INIT_ARRAY(module_alsa_sink_info),
.data_size = sizeof(struct module_alsa_sink_data),
};

View file

@ -0,0 +1,239 @@
/* PipeWire */
/* SPDX-FileCopyrightText: Copyright © 2023 Wim Taymans <wim.taymans@gmail.com> */
/* SPDX-License-Identifier: MIT */
#include <pipewire/pipewire.h>
#include "../manager.h"
#include "../module.h"
#define NAME "alsa-source"
#define DEFAULT_DEVICE "default"
PW_LOG_TOPIC_STATIC(mod_topic, "mod." NAME);
#define PW_LOG_TOPIC_DEFAULT mod_topic
struct module_alsa_source_data {
struct pw_core *core;
struct spa_hook core_listener;
struct pw_proxy *proxy;
struct spa_hook proxy_listener;
};
static void module_alsa_source_proxy_removed(void *data)
{
struct module *module = data;
struct module_alsa_source_data *d = module->user_data;
pw_proxy_destroy(d->proxy);
}
static void module_alsa_source_proxy_destroy(void *data)
{
struct module *module = data;
struct module_alsa_source_data *d = module->user_data;
pw_log_info("proxy %p destroy", d->proxy);
spa_hook_remove(&d->proxy_listener);
d->proxy = NULL;
module_schedule_unload(module);
}
static void module_alsa_source_proxy_bound_props(void *data, uint32_t global_id, const struct spa_dict *props)
{
struct module *module = data;
struct module_alsa_source_data *d = module->user_data;
pw_log_info("proxy %p bound", d->proxy);
module_emit_loaded(module, 0);
}
static void module_alsa_source_proxy_error(void *data, int seq, int res, const char *message)
{
struct module *module = data;
struct module_alsa_source_data *d = module->user_data;
pw_log_info("proxy %p error %d", d->proxy, res);
pw_proxy_destroy(d->proxy);
}
static const struct pw_proxy_events proxy_events = {
PW_VERSION_PROXY_EVENTS,
.removed = module_alsa_source_proxy_removed,
.bound_props = module_alsa_source_proxy_bound_props,
.error = module_alsa_source_proxy_error,
.destroy = module_alsa_source_proxy_destroy,
};
static void module_alsa_source_core_error(void *data, uint32_t id, int seq, int res, const char *message)
{
struct module *module = data;
pw_log_warn("error id:%u seq:%d res:%d (%s): %s",
id, seq, res, spa_strerror(res), message);
if (id == PW_ID_CORE && res == -EPIPE)
module_schedule_unload(module);
}
static const struct pw_core_events core_events = {
PW_VERSION_CORE_EVENTS,
.error = module_alsa_source_core_error,
};
static int module_alsa_source_load(struct module *module)
{
struct module_alsa_source_data *d = module->user_data;
d->core = pw_context_connect(module->impl->context, NULL, 0);
if (d->core == NULL)
return -errno;
pw_core_add_listener(d->core, &d->core_listener, &core_events, module);
pw_properties_setf(module->props, "pulse.module.id", "%u", module->index);
d->proxy = pw_core_create_object(d->core,
"adapter", PW_TYPE_INTERFACE_Node, PW_VERSION_NODE,
module->props ? &module->props->dict : NULL, 0);
if (d->proxy == NULL)
return -errno;
pw_proxy_add_listener(d->proxy, &d->proxy_listener, &proxy_events, module);
return SPA_RESULT_RETURN_ASYNC(0);
}
static int module_alsa_source_unload(struct module *module)
{
struct module_alsa_source_data *d = module->user_data;
if (d->proxy != NULL) {
spa_hook_remove(&d->proxy_listener);
pw_proxy_destroy(d->proxy);
d->proxy = NULL;
}
if (d->core != NULL) {
spa_hook_remove(&d->core_listener);
pw_core_disconnect(d->core);
d->core = NULL;
}
return 0;
}
static const struct spa_dict_item module_alsa_source_info[] = {
{ PW_KEY_MODULE_AUTHOR, "Wim Taymans <wim.taymans@gmail.com>" },
{ PW_KEY_MODULE_DESCRIPTION, "An ALSA source" },
{ PW_KEY_MODULE_USAGE,
"name=<name of the source, to be prefixed> "
"source_name=<name for the source> "
"source_properties=<properties for the source> "
"namereg_fail=<when false attempt to synthesise new source_name if it is already taken> "
"device=<ALSA device> "
"device_id=<ALSA card index> "
"format=<sample format> "
"rate=<sample rate> "
"alternate_rate=<alternate sample rate> "
"channels=<number of channels> "
"channel_map=<channel map> "
"fragments=<number of fragments> "
"fragment_size=<fragment size> "
"mmap=<enable memory mapping?> "
"tsched=<enable system timer based scheduling mode?> "
"tsched_buffer_size=<buffer size when using timer based scheduling> "
"tsched_buffer_watermark=<lower fill watermark> "
"ignore_dB=<ignore dB information from the device?> "
"control=<name of mixer control, or name and index separated by a comma> "
"rewind_safeguard=<number of bytes that cannot be rewound> "
"deferred_volume=<Synchronize software and hardware volume changes to avoid momentary jumps?> "
"deferred_volume_safety_margin=<usec adjustment depending on volume direction> "
"deferred_volume_extra_delay=<usec adjustment to HW volume changes> "
"fixed_latency_range=<disable latency range changes on underrun?>" },
{ PW_KEY_MODULE_VERSION, PACKAGE_VERSION },
};
static int module_alsa_source_prepare(struct module * const module)
{
struct pw_properties * const props = module->props;
const char *str, *dev_id;
struct spa_audio_info_raw info = { 0 };
PW_LOG_TOPIC_INIT(mod_topic);
dev_id = pw_properties_get(props, "device_id");
if (dev_id == NULL)
dev_id = pw_properties_get(props, "device");
if (dev_id == NULL)
dev_id = DEFAULT_DEVICE;
pw_properties_set(props, "api.alsa.path", dev_id);
if ((str = pw_properties_get(props, "source_name")) != NULL) {
pw_properties_set(props, PW_KEY_NODE_NAME, str);
pw_properties_set(props, "source_name", NULL);
}
else if ((str = pw_properties_get(props, "name")) != NULL) {
pw_properties_setf(props, PW_KEY_NODE_NAME, "alsa_output.%s", str);
pw_properties_set(props, "name", NULL);
}
else {
pw_properties_setf(props, PW_KEY_NODE_NAME, "alsa_output.%s", dev_id);
}
if ((str = pw_properties_get(props, "source_properties")) != NULL) {
module_args_add_props(props, str);
pw_properties_set(props, "source_properties", NULL);
}
if ((str = pw_properties_get(props, "fragments")) != NULL) {
pw_properties_set(props, "api.alsa.period-num", str);
pw_properties_set(props, "fragments", NULL);
}
if ((str = pw_properties_get(props, "fragment_size")) != NULL) {
pw_properties_set(props, "api.alsa.period-size", str);
pw_properties_set(props, "fragment_size", NULL);
}
if ((str = pw_properties_get(props, "mmap")) != NULL) {
pw_properties_setf(props, "api.alsa.disable-mmap",
spa_atob(str) ? "false" : "true");
pw_properties_set(props, "mmap", NULL);
}
if (module_args_to_audioinfo_keys(module->impl, props,
"format", "rate", "channels", "channel_map", &info) < 0)
return -EINVAL;
audioinfo_to_properties(&info, props);
if (pw_properties_get(props, PW_KEY_MEDIA_CLASS) == NULL)
pw_properties_set(props, PW_KEY_MEDIA_CLASS, "Audio/Source");
if ((str = pw_properties_get(props, PW_KEY_NODE_DESCRIPTION)) == NULL)
pw_properties_setf(props, PW_KEY_NODE_DESCRIPTION,
"ALSA Source on %s", dev_id);
pw_properties_set(props, PW_KEY_FACTORY_NAME, "api.alsa.pcm.source");
if (pw_properties_get(props, "monitor.channel-volumes") == NULL)
pw_properties_set(props, "monitor.channel-volumes", "true");
if (pw_properties_get(props, "node.suspend-on-idle") == NULL)
pw_properties_set(props, "node.suspend-on-idle", "true");
return 0;
}
DEFINE_MODULE_INFO(module_alsa_source) = {
.name = "module-alsa-source",
.prepare = module_alsa_source_prepare,
.load = module_alsa_source_load,
.unload = module_alsa_source_unload,
.properties = &SPA_DICT_INIT_ARRAY(module_alsa_source_info),
.data_size = sizeof(struct module_alsa_source_data),
};