mirror of
https://gitlab.freedesktop.org/pipewire/pipewire.git
synced 2025-11-02 09:01:50 -05:00
module-protocol-pulse: Add Avahi zeroconf publish module
This commit is contained in:
parent
e66125ede0
commit
cf93fd7f9a
4 changed files with 721 additions and 18 deletions
|
|
@ -131,25 +131,37 @@ if dbus_dep.found()
|
||||||
pipewire_module_protocol_pulse_deps += dbus_dep
|
pipewire_module_protocol_pulse_deps += dbus_dep
|
||||||
endif
|
endif
|
||||||
|
|
||||||
|
pipewire_module_protocol_pulse_sources = [
|
||||||
|
'module-protocol-pulse.c',
|
||||||
|
'module-protocol-pulse/manager.c',
|
||||||
|
'module-protocol-pulse/pulse-server.c',
|
||||||
|
'module-protocol-pulse/modules/module-combine-sink.c',
|
||||||
|
'module-protocol-pulse/modules/module-echo-cancel.c',
|
||||||
|
'module-protocol-pulse/modules/module-ladspa-sink.c',
|
||||||
|
'module-protocol-pulse/modules/module-ladspa-source.c',
|
||||||
|
'module-protocol-pulse/modules/module-loopback.c',
|
||||||
|
'module-protocol-pulse/modules/module-native-protocol-tcp.c',
|
||||||
|
'module-protocol-pulse/modules/module-null-sink.c',
|
||||||
|
'module-protocol-pulse/modules/module-pipe-sink.c',
|
||||||
|
'module-protocol-pulse/modules/module-remap-sink.c',
|
||||||
|
'module-protocol-pulse/modules/module-remap-source.c',
|
||||||
|
'module-protocol-pulse/modules/module-simple-protocol-tcp.c',
|
||||||
|
'module-protocol-pulse/modules/module-tunnel-sink.c',
|
||||||
|
'module-protocol-pulse/modules/module-tunnel-source.c',
|
||||||
|
'module-protocol-pulse/modules/module-zeroconf-discover.c',
|
||||||
|
]
|
||||||
|
|
||||||
|
if avahi_dep.found()
|
||||||
|
pipewire_module_protocol_pulse_sources += [
|
||||||
|
'module-protocol-pulse/modules/module-zeroconf-publish.c',
|
||||||
|
'module-zeroconf-discover/avahi-poll.c',
|
||||||
|
]
|
||||||
|
pipewire_module_protocol_pulse_deps += avahi_dep
|
||||||
|
cdata.set('HAVE_AVAHI', 1)
|
||||||
|
endif
|
||||||
|
|
||||||
pipewire_module_protocol_pulse = shared_library('pipewire-module-protocol-pulse',
|
pipewire_module_protocol_pulse = shared_library('pipewire-module-protocol-pulse',
|
||||||
[ 'module-protocol-pulse.c',
|
pipewire_module_protocol_pulse_sources,
|
||||||
'module-protocol-pulse/manager.c',
|
|
||||||
'module-protocol-pulse/pulse-server.c',
|
|
||||||
'module-protocol-pulse/modules/module-combine-sink.c',
|
|
||||||
'module-protocol-pulse/modules/module-echo-cancel.c',
|
|
||||||
'module-protocol-pulse/modules/module-ladspa-sink.c',
|
|
||||||
'module-protocol-pulse/modules/module-ladspa-source.c',
|
|
||||||
'module-protocol-pulse/modules/module-loopback.c',
|
|
||||||
'module-protocol-pulse/modules/module-native-protocol-tcp.c',
|
|
||||||
'module-protocol-pulse/modules/module-null-sink.c',
|
|
||||||
'module-protocol-pulse/modules/module-pipe-sink.c',
|
|
||||||
'module-protocol-pulse/modules/module-remap-sink.c',
|
|
||||||
'module-protocol-pulse/modules/module-remap-source.c',
|
|
||||||
'module-protocol-pulse/modules/module-simple-protocol-tcp.c',
|
|
||||||
'module-protocol-pulse/modules/module-tunnel-sink.c',
|
|
||||||
'module-protocol-pulse/modules/module-tunnel-source.c',
|
|
||||||
'module-protocol-pulse/modules/module-zeroconf-discover.c',
|
|
||||||
],
|
|
||||||
include_directories : [configinc, spa_inc],
|
include_directories : [configinc, spa_inc],
|
||||||
install : true,
|
install : true,
|
||||||
install_dir : modules_install_dir,
|
install_dir : modules_install_dir,
|
||||||
|
|
|
||||||
|
|
@ -232,6 +232,9 @@ static const struct module_info module_list[] = {
|
||||||
{ "module-tunnel-sink", create_module_tunnel_sink, },
|
{ "module-tunnel-sink", create_module_tunnel_sink, },
|
||||||
{ "module-tunnel-source", create_module_tunnel_source, },
|
{ "module-tunnel-source", create_module_tunnel_source, },
|
||||||
{ "module-zeroconf-discover", create_module_zeroconf_discover, },
|
{ "module-zeroconf-discover", create_module_zeroconf_discover, },
|
||||||
|
#ifdef HAVE_AVAHI
|
||||||
|
{ "module-zeroconf-publish", create_module_zeroconf_publish, },
|
||||||
|
#endif
|
||||||
{ NULL, }
|
{ NULL, }
|
||||||
};
|
};
|
||||||
|
|
||||||
|
|
|
||||||
|
|
@ -0,0 +1,687 @@
|
||||||
|
/* PipeWire
|
||||||
|
*
|
||||||
|
* Copyright © 2021 Wim Taymans <wim.taymans@gmail.com>
|
||||||
|
* Copyright © 2021 Sanchayan Maity <sanchayan@asymptotic.io>
|
||||||
|
*
|
||||||
|
* Permission is hereby granted, free of charge, to any person obtaining a
|
||||||
|
* copy of this software and associated documentation files (the "Software"),
|
||||||
|
* to deal in the Software without restriction, including without limitation
|
||||||
|
* the rights to use, copy, modify, merge, publish, distribute, sublicense,
|
||||||
|
* and/or sell copies of the Software, and to permit persons to whom the
|
||||||
|
* Software is furnished to do so, subject to the following conditions:
|
||||||
|
*
|
||||||
|
* The above copyright notice and this permission notice (including the next
|
||||||
|
* paragraph) shall be included in all copies or substantial portions of the
|
||||||
|
* Software.
|
||||||
|
*
|
||||||
|
* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
||||||
|
* IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
||||||
|
* FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL
|
||||||
|
* THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
||||||
|
* LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING
|
||||||
|
* FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER
|
||||||
|
* DEALINGS IN THE SOFTWARE.
|
||||||
|
*/
|
||||||
|
|
||||||
|
#include <pipewire/pipewire.h>
|
||||||
|
|
||||||
|
#include "../collect.h"
|
||||||
|
#include "../defs.h"
|
||||||
|
#include "../manager.h"
|
||||||
|
#include "../module.h"
|
||||||
|
#include "../pulse-server.h"
|
||||||
|
#include "registry.h"
|
||||||
|
#include "sys/utsname.h"
|
||||||
|
|
||||||
|
#include "../../module-zeroconf-discover/avahi-poll.h"
|
||||||
|
#include <avahi-client/client.h>
|
||||||
|
#include <avahi-client/publish.h>
|
||||||
|
#include <avahi-common/alternative.h>
|
||||||
|
#include <avahi-common/error.h>
|
||||||
|
#include <avahi-common/domain.h>
|
||||||
|
|
||||||
|
#define SERVICE_TYPE_SINK "_pulse-sink._tcp"
|
||||||
|
#define SERVICE_TYPE_SOURCE "_pulse-source._tcp"
|
||||||
|
#define SERVICE_TYPE_SERVER "_pulse-server._tcp"
|
||||||
|
#define SERVICE_SUBTYPE_SINK_HARDWARE "_hardware._sub."SERVICE_TYPE_SINK
|
||||||
|
#define SERVICE_SUBTYPE_SINK_VIRTUAL "_virtual._sub."SERVICE_TYPE_SINK
|
||||||
|
#define SERVICE_SUBTYPE_SOURCE_HARDWARE "_hardware._sub."SERVICE_TYPE_SOURCE
|
||||||
|
#define SERVICE_SUBTYPE_SOURCE_VIRTUAL "_virtual._sub."SERVICE_TYPE_SOURCE
|
||||||
|
#define SERVICE_SUBTYPE_SOURCE_MONITOR "_monitor._sub."SERVICE_TYPE_SOURCE
|
||||||
|
#define SERVICE_SUBTYPE_SOURCE_NON_MONITOR "_non-monitor._sub."SERVICE_TYPE_SOURCE
|
||||||
|
|
||||||
|
enum service_subtype {
|
||||||
|
SUBTYPE_HARDWARE,
|
||||||
|
SUBTYPE_VIRTUAL,
|
||||||
|
SUBTYPE_MONITOR
|
||||||
|
};
|
||||||
|
|
||||||
|
struct service {
|
||||||
|
void *key;
|
||||||
|
|
||||||
|
struct module_zeroconf_publish_data *userdata;
|
||||||
|
AvahiEntryGroup *entry_group;
|
||||||
|
char *service_name;
|
||||||
|
const char *service_type;
|
||||||
|
enum service_subtype subtype;
|
||||||
|
|
||||||
|
char *name;
|
||||||
|
bool is_sink;
|
||||||
|
|
||||||
|
struct sample_spec ss;
|
||||||
|
struct channel_map cm;
|
||||||
|
struct pw_properties *props;
|
||||||
|
};
|
||||||
|
|
||||||
|
struct module_zeroconf_publish_data {
|
||||||
|
struct module *module;
|
||||||
|
|
||||||
|
struct pw_core *core;
|
||||||
|
struct pw_manager *manager;
|
||||||
|
|
||||||
|
struct spa_hook core_listener;
|
||||||
|
struct spa_hook manager_listener;
|
||||||
|
|
||||||
|
unsigned int port;
|
||||||
|
|
||||||
|
AvahiPoll *avahi_poll;
|
||||||
|
AvahiClient *client;
|
||||||
|
bool entry_group_free;
|
||||||
|
};
|
||||||
|
|
||||||
|
static void on_core_error(void *data, uint32_t id, int seq, int res, const char *message)
|
||||||
|
{
|
||||||
|
struct module_zeroconf_publish_data *d = data;
|
||||||
|
struct module *module = d->module;
|
||||||
|
|
||||||
|
pw_log_error("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 = on_core_error,
|
||||||
|
};
|
||||||
|
|
||||||
|
static char *get_service_name(struct pw_manager_object *o)
|
||||||
|
{
|
||||||
|
const char *hn, *un, *n;
|
||||||
|
char *service_name;
|
||||||
|
|
||||||
|
hn = pw_get_host_name();
|
||||||
|
un = pw_get_user_name();
|
||||||
|
n = pw_properties_get(o->props, PW_KEY_NODE_DESCRIPTION);
|
||||||
|
|
||||||
|
service_name = calloc(1, AVAHI_LABEL_MAX - 1);
|
||||||
|
snprintf(service_name, AVAHI_LABEL_MAX - 1, "%s@%s: %s", un, hn, n);
|
||||||
|
|
||||||
|
return service_name;
|
||||||
|
}
|
||||||
|
|
||||||
|
static int service_free(void *d, struct pw_manager_object *o) {
|
||||||
|
struct service *s;
|
||||||
|
char *service_name;
|
||||||
|
|
||||||
|
if (!pw_manager_object_is_sink(o) && !pw_manager_object_is_source(o))
|
||||||
|
return 0;
|
||||||
|
|
||||||
|
service_name = get_service_name(o);
|
||||||
|
|
||||||
|
s = pw_manager_object_add_data(o, service_name, sizeof(struct service));
|
||||||
|
if (s == NULL) {
|
||||||
|
pw_log_error("Could not find service %s to remove", service_name);
|
||||||
|
free(service_name);
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
free(service_name);
|
||||||
|
spa_assert(s);
|
||||||
|
|
||||||
|
if (s->entry_group) {
|
||||||
|
pw_log_debug("Removing entry group for %s.", s->service_name);
|
||||||
|
avahi_entry_group_free(s->entry_group);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (s->service_name) {
|
||||||
|
pw_log_debug("Removing service: %s", s->service_name);
|
||||||
|
free(s->service_name);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (s->name) {
|
||||||
|
pw_log_debug("Removing service for node: %s", s->name);
|
||||||
|
free(s->name);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (s->props)
|
||||||
|
pw_properties_free(s->props);
|
||||||
|
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
static int unpublish_service(void *data, struct pw_manager_object *o) {
|
||||||
|
struct module_zeroconf_publish_data *d = data;
|
||||||
|
struct service *s;
|
||||||
|
char *service_name;
|
||||||
|
|
||||||
|
if (!pw_manager_object_is_sink(o) && !pw_manager_object_is_source(o))
|
||||||
|
return 0;
|
||||||
|
|
||||||
|
service_name = get_service_name(o);
|
||||||
|
|
||||||
|
s = pw_manager_object_add_data(o, service_name, sizeof(struct service));
|
||||||
|
if (s == NULL) {
|
||||||
|
pw_log_error("Could not find service %s to remove", service_name);
|
||||||
|
free(service_name);
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
free(service_name);
|
||||||
|
spa_assert(s);
|
||||||
|
|
||||||
|
if (s->entry_group) {
|
||||||
|
if (d->entry_group_free) {
|
||||||
|
pw_log_debug("Removing entry group for %s.", s->service_name);
|
||||||
|
avahi_entry_group_free(s->entry_group);
|
||||||
|
s->entry_group = NULL;
|
||||||
|
} else {
|
||||||
|
avahi_entry_group_reset(s->entry_group);
|
||||||
|
pw_log_debug("Resetting entry group for %s.", s->service_name);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
static void unpublish_all_services(struct module_zeroconf_publish_data *d, bool entry_group_free)
|
||||||
|
{
|
||||||
|
d->entry_group_free = entry_group_free;
|
||||||
|
pw_manager_for_each_object(d->manager, unpublish_service, d);
|
||||||
|
}
|
||||||
|
|
||||||
|
static char* channel_map_snprint(char *s, size_t l, const struct channel_map *map)
|
||||||
|
{
|
||||||
|
unsigned channel;
|
||||||
|
bool first = true;
|
||||||
|
char *e;
|
||||||
|
|
||||||
|
spa_assert(s);
|
||||||
|
spa_assert(l > 0);
|
||||||
|
spa_assert(map);
|
||||||
|
|
||||||
|
if (!channel_map_valid(map)) {
|
||||||
|
snprintf(s, l, "(invalid)");
|
||||||
|
return s;
|
||||||
|
}
|
||||||
|
|
||||||
|
*(e = s) = 0;
|
||||||
|
|
||||||
|
for (channel = 0; channel < map->channels && l > 1; channel++) {
|
||||||
|
l -= snprintf(e, l, "%s%s",
|
||||||
|
first ? "" : ",",
|
||||||
|
channel_id2name(channel_pa2id(map->map[channel])));
|
||||||
|
|
||||||
|
e = strchr(e, 0);
|
||||||
|
first = false;
|
||||||
|
}
|
||||||
|
|
||||||
|
return s;
|
||||||
|
}
|
||||||
|
|
||||||
|
static void get_service_data(struct module_zeroconf_publish_data *d,
|
||||||
|
struct service *s, struct pw_manager_object *o)
|
||||||
|
{
|
||||||
|
bool is_sink = pw_manager_object_is_sink(o);
|
||||||
|
bool is_source = pw_manager_object_is_source(o);
|
||||||
|
struct pw_node_info *info = o->info;
|
||||||
|
const char *name, *desc, *str;
|
||||||
|
uint32_t card_id = SPA_ID_INVALID;
|
||||||
|
struct pw_manager *manager = d->manager;
|
||||||
|
struct pw_manager_object *card = NULL;
|
||||||
|
struct card_info card_info = CARD_INFO_INIT;
|
||||||
|
struct device_info dev_info = is_sink ?
|
||||||
|
DEVICE_INFO_INIT(PW_DIRECTION_OUTPUT) : DEVICE_INFO_INIT(PW_DIRECTION_INPUT);
|
||||||
|
uint32_t flags = 0;
|
||||||
|
|
||||||
|
if (info == NULL || info->props == NULL)
|
||||||
|
return;
|
||||||
|
|
||||||
|
name = spa_dict_lookup(info->props, PW_KEY_NODE_NAME);
|
||||||
|
if ((desc = spa_dict_lookup(info->props, PW_KEY_NODE_DESCRIPTION)) == NULL)
|
||||||
|
desc = name ? name : "Unknown";
|
||||||
|
if (name == NULL)
|
||||||
|
name = "unknown";
|
||||||
|
|
||||||
|
if ((str = spa_dict_lookup(info->props, PW_KEY_DEVICE_ID)) != NULL)
|
||||||
|
card_id = (uint32_t)atoi(str);
|
||||||
|
if ((str = spa_dict_lookup(info->props, "card.profile.device")) != NULL)
|
||||||
|
dev_info.device = (uint32_t)atoi(str);
|
||||||
|
if (card_id != SPA_ID_INVALID) {
|
||||||
|
struct selector sel = { .id = card_id, .type = pw_manager_object_is_card, };
|
||||||
|
card = select_object(manager, &sel);
|
||||||
|
}
|
||||||
|
if (card)
|
||||||
|
collect_card_info(card, &card_info);
|
||||||
|
|
||||||
|
collect_device_info(o, card, &dev_info, false);
|
||||||
|
|
||||||
|
if ((str = spa_dict_lookup(info->props, PW_KEY_DEVICE_API)) != NULL) {
|
||||||
|
if (pw_manager_object_is_sink(o))
|
||||||
|
flags |= SINK_HARDWARE;
|
||||||
|
else if (pw_manager_object_is_source(o))
|
||||||
|
flags |= SOURCE_HARDWARE;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (is_sink) {
|
||||||
|
s->is_sink = true;
|
||||||
|
s->service_type = SERVICE_TYPE_SINK;
|
||||||
|
s->ss = dev_info.ss;
|
||||||
|
s->cm = dev_info.map;
|
||||||
|
s->name = strdup(name);
|
||||||
|
s->props = pw_properties_copy(o->props);
|
||||||
|
s->subtype = flags & SINK_HARDWARE ? SUBTYPE_HARDWARE : SUBTYPE_VIRTUAL;
|
||||||
|
} else if (is_source) {
|
||||||
|
s->is_sink = false;
|
||||||
|
s->service_type = SERVICE_TYPE_SOURCE;
|
||||||
|
s->name = strdup(name);
|
||||||
|
s->props = pw_properties_copy(o->props);
|
||||||
|
s->subtype = flags & SOURCE_HARDWARE ? SUBTYPE_HARDWARE : SUBTYPE_VIRTUAL;
|
||||||
|
} else
|
||||||
|
spa_assert_not_reached();
|
||||||
|
}
|
||||||
|
|
||||||
|
static struct service *get_service(struct module_zeroconf_publish_data *d, struct pw_manager_object *o)
|
||||||
|
{
|
||||||
|
struct service *s;
|
||||||
|
char *service_name;
|
||||||
|
|
||||||
|
service_name = get_service_name(o);
|
||||||
|
|
||||||
|
s = pw_manager_object_add_data(o, service_name, sizeof(struct service));
|
||||||
|
spa_assert(s);
|
||||||
|
|
||||||
|
s->key = o;
|
||||||
|
s->userdata = d;
|
||||||
|
s->entry_group = NULL;
|
||||||
|
s->service_name = service_name;
|
||||||
|
|
||||||
|
get_service_data(d, s, o);
|
||||||
|
|
||||||
|
return s;
|
||||||
|
}
|
||||||
|
|
||||||
|
static AvahiStringList* txt_record_server_data(struct pw_core_info *info, AvahiStringList *l)
|
||||||
|
{
|
||||||
|
const char *t;
|
||||||
|
struct utsname u;
|
||||||
|
static char sysname[256];
|
||||||
|
|
||||||
|
spa_assert(info);
|
||||||
|
spa_assert(uname(&u) >= 0);
|
||||||
|
|
||||||
|
l = avahi_string_list_add_pair(l, "server-version", PACKAGE_NAME" "PACKAGE_VERSION);
|
||||||
|
|
||||||
|
t = pw_get_user_name();
|
||||||
|
l = avahi_string_list_add_pair(l, "user-name", t);
|
||||||
|
|
||||||
|
snprintf(sysname, sizeof(sysname), "%s %s %s", u.sysname, u.machine, u.release);
|
||||||
|
l = avahi_string_list_add_pair(l, "uname", sysname);
|
||||||
|
|
||||||
|
t = pw_get_host_name();
|
||||||
|
l = avahi_string_list_add_pair(l, "fqdn", t);
|
||||||
|
l = avahi_string_list_add_printf(l, "cookie=0x%08x", info->cookie);
|
||||||
|
|
||||||
|
return l;
|
||||||
|
}
|
||||||
|
|
||||||
|
static void publish_service(struct service *s);
|
||||||
|
|
||||||
|
static void service_entry_group_callback(AvahiEntryGroup *g, AvahiEntryGroupState state, void *userdata)
|
||||||
|
{
|
||||||
|
struct service *s = userdata;
|
||||||
|
|
||||||
|
spa_assert(s);
|
||||||
|
|
||||||
|
switch (state) {
|
||||||
|
case AVAHI_ENTRY_GROUP_ESTABLISHED:
|
||||||
|
pw_log_info("Successfully established service %s.", s->service_name);
|
||||||
|
break;
|
||||||
|
|
||||||
|
case AVAHI_ENTRY_GROUP_COLLISION:
|
||||||
|
char *t;
|
||||||
|
|
||||||
|
t = avahi_alternative_service_name(s->service_name);
|
||||||
|
pw_log_info("Name collision, renaming %s to %s.", s->service_name, t);
|
||||||
|
free(s->service_name);
|
||||||
|
s->service_name = t;
|
||||||
|
|
||||||
|
publish_service(s);
|
||||||
|
break;
|
||||||
|
|
||||||
|
case AVAHI_ENTRY_GROUP_FAILURE:
|
||||||
|
pw_log_error("Failed to register service: %s",
|
||||||
|
avahi_strerror(avahi_client_errno(avahi_entry_group_get_client(g))));
|
||||||
|
avahi_entry_group_free(g);
|
||||||
|
s->entry_group = NULL;
|
||||||
|
break;
|
||||||
|
|
||||||
|
case AVAHI_ENTRY_GROUP_UNCOMMITED:
|
||||||
|
case AVAHI_ENTRY_GROUP_REGISTERING:
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
#define PA_CHANNEL_MAP_SNPRINT_MAX 336
|
||||||
|
|
||||||
|
static void publish_service(struct service *s)
|
||||||
|
{
|
||||||
|
AvahiStringList *txt = NULL;
|
||||||
|
const char *t;
|
||||||
|
char cm[PA_CHANNEL_MAP_SNPRINT_MAX];
|
||||||
|
|
||||||
|
const char * const subtype_text[] = {
|
||||||
|
[SUBTYPE_HARDWARE] = "hardware",
|
||||||
|
[SUBTYPE_VIRTUAL] = "virtual",
|
||||||
|
[SUBTYPE_MONITOR] = "monitor"
|
||||||
|
};
|
||||||
|
|
||||||
|
spa_assert(s);
|
||||||
|
|
||||||
|
if (!s->userdata->client || avahi_client_get_state(s->userdata->client) != AVAHI_CLIENT_S_RUNNING)
|
||||||
|
return;
|
||||||
|
|
||||||
|
if (!s->entry_group) {
|
||||||
|
if (!(s->entry_group =
|
||||||
|
avahi_entry_group_new(s->userdata->client, service_entry_group_callback, s))) {
|
||||||
|
pw_log_error("avahi_entry_group_new(): %s",
|
||||||
|
avahi_strerror(avahi_client_errno(s->userdata->client)));
|
||||||
|
goto finish;
|
||||||
|
}
|
||||||
|
} else
|
||||||
|
avahi_entry_group_reset(s->entry_group);
|
||||||
|
|
||||||
|
txt = txt_record_server_data(s->userdata->manager->info, txt);
|
||||||
|
|
||||||
|
txt = avahi_string_list_add_pair(txt, "device", s->name);
|
||||||
|
txt = avahi_string_list_add_printf(txt, "rate=%u", s->ss.rate);
|
||||||
|
txt = avahi_string_list_add_printf(txt, "channels=%u", s->ss.channels);
|
||||||
|
txt = avahi_string_list_add_pair(txt, "format", format_id2name(s->ss.format));
|
||||||
|
txt = avahi_string_list_add_pair(txt, "channel_map", channel_map_snprint(cm, sizeof(cm), &s->cm));
|
||||||
|
txt = avahi_string_list_add_pair(txt, "subtype", subtype_text[s->subtype]);
|
||||||
|
|
||||||
|
if ((t = pw_properties_get(s->props, PW_KEY_NODE_DESCRIPTION)))
|
||||||
|
txt = avahi_string_list_add_pair(txt, "description", t);
|
||||||
|
if ((t = pw_properties_get(s->props, PW_KEY_DEVICE_VENDOR_NAME)))
|
||||||
|
txt = avahi_string_list_add_pair(txt, "vendor-name", t);
|
||||||
|
if ((t = pw_properties_get(s->props, PW_KEY_DEVICE_PRODUCT_NAME)))
|
||||||
|
txt = avahi_string_list_add_pair(txt, "product-name", t);
|
||||||
|
if ((t = pw_properties_get(s->props, PW_KEY_DEVICE_CLASS)))
|
||||||
|
txt = avahi_string_list_add_pair(txt, "class", t);
|
||||||
|
if ((t = pw_properties_get(s->props, PW_KEY_DEVICE_FORM_FACTOR)))
|
||||||
|
txt = avahi_string_list_add_pair(txt, "form-factor", t);
|
||||||
|
if ((t = pw_properties_get(s->props, PW_KEY_DEVICE_ICON_NAME)))
|
||||||
|
txt = avahi_string_list_add_pair(txt, "icon-name", t);
|
||||||
|
|
||||||
|
if (avahi_entry_group_add_service_strlst(
|
||||||
|
s->entry_group,
|
||||||
|
AVAHI_IF_UNSPEC, AVAHI_PROTO_UNSPEC,
|
||||||
|
0,
|
||||||
|
s->service_name,
|
||||||
|
s->service_type,
|
||||||
|
NULL,
|
||||||
|
NULL,
|
||||||
|
s->userdata->port,
|
||||||
|
txt) < 0) {
|
||||||
|
pw_log_error("avahi_entry_group_add_service_strlst(): %s",
|
||||||
|
avahi_strerror(avahi_client_errno(s->userdata->client)));
|
||||||
|
goto finish;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (avahi_entry_group_add_service_subtype(
|
||||||
|
s->entry_group,
|
||||||
|
AVAHI_IF_UNSPEC, AVAHI_PROTO_UNSPEC,
|
||||||
|
0,
|
||||||
|
s->service_name,
|
||||||
|
s->service_type,
|
||||||
|
NULL,
|
||||||
|
s->is_sink ? (s->subtype == SUBTYPE_HARDWARE ? SERVICE_SUBTYPE_SINK_HARDWARE : SERVICE_SUBTYPE_SINK_VIRTUAL) :
|
||||||
|
(s->subtype == SUBTYPE_HARDWARE ? SERVICE_SUBTYPE_SOURCE_HARDWARE : (s->subtype == SUBTYPE_VIRTUAL ? SERVICE_SUBTYPE_SOURCE_VIRTUAL : SERVICE_SUBTYPE_SOURCE_MONITOR))) < 0) {
|
||||||
|
|
||||||
|
pw_log_error("avahi_entry_group_add_service_subtype(): %s",
|
||||||
|
avahi_strerror(avahi_client_errno(s->userdata->client)));
|
||||||
|
goto finish;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!s->is_sink && s->subtype != SUBTYPE_MONITOR) {
|
||||||
|
if (avahi_entry_group_add_service_subtype(
|
||||||
|
s->entry_group,
|
||||||
|
AVAHI_IF_UNSPEC, AVAHI_PROTO_UNSPEC,
|
||||||
|
0,
|
||||||
|
s->service_name,
|
||||||
|
SERVICE_TYPE_SOURCE,
|
||||||
|
NULL,
|
||||||
|
SERVICE_SUBTYPE_SOURCE_NON_MONITOR) < 0) {
|
||||||
|
pw_log_error("avahi_entry_group_add_service_subtype(): %s",
|
||||||
|
avahi_strerror(avahi_client_errno(s->userdata->client)));
|
||||||
|
goto finish;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if (avahi_entry_group_commit(s->entry_group) < 0) {
|
||||||
|
pw_log_error("avahi_entry_group_commit(): %s",
|
||||||
|
avahi_strerror(avahi_client_errno(s->userdata->client)));
|
||||||
|
goto finish;
|
||||||
|
}
|
||||||
|
|
||||||
|
pw_log_info("Successfully created entry group for %s.", s->service_name);
|
||||||
|
|
||||||
|
finish:
|
||||||
|
avahi_string_list_free(txt);
|
||||||
|
}
|
||||||
|
|
||||||
|
static void client_callback(AvahiClient *c, AvahiClientState state, void *d)
|
||||||
|
{
|
||||||
|
struct module_zeroconf_publish_data *data = d;
|
||||||
|
|
||||||
|
spa_assert(c);
|
||||||
|
spa_assert(data);
|
||||||
|
|
||||||
|
data->client = c;
|
||||||
|
|
||||||
|
switch (state) {
|
||||||
|
case AVAHI_CLIENT_S_RUNNING:
|
||||||
|
break;
|
||||||
|
case AVAHI_CLIENT_S_COLLISION:
|
||||||
|
pw_log_error("Host name collision");
|
||||||
|
unpublish_all_services(d, false);
|
||||||
|
break;
|
||||||
|
case AVAHI_CLIENT_FAILURE:
|
||||||
|
if (avahi_client_errno(c) == AVAHI_ERR_DISCONNECTED) {
|
||||||
|
int error;
|
||||||
|
|
||||||
|
pw_log_debug("Avahi daemon disconnected.");
|
||||||
|
|
||||||
|
unpublish_all_services(d, true);
|
||||||
|
avahi_client_free(data->client);
|
||||||
|
|
||||||
|
if (!(data->client = avahi_client_new(data->avahi_poll,
|
||||||
|
AVAHI_CLIENT_NO_FAIL, client_callback, data, &error))) {
|
||||||
|
pw_log_error("avahi_client_new() failed: %s",
|
||||||
|
avahi_strerror(error));
|
||||||
|
module_schedule_unload(data->module);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
default:
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
static void manager_updated(void *d, struct pw_manager_object *o)
|
||||||
|
{
|
||||||
|
struct service *s;
|
||||||
|
char *service_name;
|
||||||
|
|
||||||
|
if (!pw_manager_object_is_sink(o) && !pw_manager_object_is_source(o))
|
||||||
|
return;
|
||||||
|
|
||||||
|
service_name = get_service_name(o);
|
||||||
|
|
||||||
|
s = pw_manager_object_add_data(o, service_name, sizeof(struct service));
|
||||||
|
|
||||||
|
free(service_name);
|
||||||
|
spa_assert(s);
|
||||||
|
|
||||||
|
publish_service(s);
|
||||||
|
}
|
||||||
|
|
||||||
|
static void manager_removed(void *d, struct pw_manager_object *o)
|
||||||
|
{
|
||||||
|
if (!pw_manager_object_is_sink(o) && !pw_manager_object_is_source(o))
|
||||||
|
return;
|
||||||
|
|
||||||
|
service_free(d, o);
|
||||||
|
}
|
||||||
|
|
||||||
|
static void manager_added(void *d, struct pw_manager_object *o)
|
||||||
|
{
|
||||||
|
struct service *s;
|
||||||
|
|
||||||
|
if (!pw_manager_object_is_sink(o) && !pw_manager_object_is_source(o))
|
||||||
|
return;
|
||||||
|
|
||||||
|
s = get_service(d, o);
|
||||||
|
|
||||||
|
publish_service(s);
|
||||||
|
}
|
||||||
|
|
||||||
|
static const struct pw_manager_events manager_events = {
|
||||||
|
PW_VERSION_MANAGER_EVENTS,
|
||||||
|
.added = manager_added,
|
||||||
|
.updated = manager_updated,
|
||||||
|
.removed = manager_removed,
|
||||||
|
};
|
||||||
|
|
||||||
|
static int module_zeroconf_publish_load(struct client *client, struct module *module)
|
||||||
|
{
|
||||||
|
struct module_zeroconf_publish_data *data = module->user_data;
|
||||||
|
struct pw_loop *loop;
|
||||||
|
int error;
|
||||||
|
|
||||||
|
data->core = pw_context_connect(module->impl->context,
|
||||||
|
pw_properties_copy(client->props), 0);
|
||||||
|
if (data->core == NULL) {
|
||||||
|
pw_log_error("Failed to connect to pipewire context");
|
||||||
|
return -errno;
|
||||||
|
}
|
||||||
|
|
||||||
|
pw_core_add_listener(data->core,
|
||||||
|
&data->core_listener,
|
||||||
|
&core_events, data);
|
||||||
|
|
||||||
|
loop = pw_context_get_main_loop(module->impl->context);
|
||||||
|
data->avahi_poll = pw_avahi_poll_new(loop);
|
||||||
|
|
||||||
|
data->client = avahi_client_new(data->avahi_poll, AVAHI_CLIENT_NO_FAIL,
|
||||||
|
client_callback, data, &error);
|
||||||
|
if (!data->client) {
|
||||||
|
pw_log_error("avahi_client_new() failed: %s", avahi_strerror(error));
|
||||||
|
pw_avahi_poll_free(data->avahi_poll);
|
||||||
|
return -errno;
|
||||||
|
}
|
||||||
|
|
||||||
|
data->manager = pw_manager_new(data->core);
|
||||||
|
if (client->manager == NULL) {
|
||||||
|
pw_log_error("Failed to create pipewire manager");
|
||||||
|
avahi_client_free(data->client);
|
||||||
|
pw_avahi_poll_free(data->avahi_poll);
|
||||||
|
return -errno;
|
||||||
|
}
|
||||||
|
|
||||||
|
pw_manager_add_listener(data->manager, &data->manager_listener,
|
||||||
|
&manager_events, data);
|
||||||
|
|
||||||
|
pw_log_info("loaded module %p id:%u name:%s", module, module->idx, module->name);
|
||||||
|
module_emit_loaded(module, 0);
|
||||||
|
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
static int module_zeroconf_publish_unload(struct client *client, struct module *module)
|
||||||
|
{
|
||||||
|
struct module_zeroconf_publish_data *d = module->user_data;
|
||||||
|
|
||||||
|
pw_log_info("unload module %p id:%u name:%s", module, module->idx, module->name);
|
||||||
|
|
||||||
|
pw_manager_for_each_object(d->manager, service_free, d);
|
||||||
|
|
||||||
|
if (d->client)
|
||||||
|
avahi_client_free(d->client);
|
||||||
|
|
||||||
|
if (d->avahi_poll)
|
||||||
|
pw_avahi_poll_free(d->avahi_poll);
|
||||||
|
|
||||||
|
if (d->manager != NULL) {
|
||||||
|
spa_hook_remove(&d->manager_listener);
|
||||||
|
pw_manager_destroy(d->manager);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (d->core != NULL) {
|
||||||
|
spa_hook_remove(&d->core_listener);
|
||||||
|
pw_core_disconnect(d->core);
|
||||||
|
}
|
||||||
|
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
static const struct module_methods module_zeroconf_publish_methods = {
|
||||||
|
VERSION_MODULE_METHODS,
|
||||||
|
.load = module_zeroconf_publish_load,
|
||||||
|
.unload = module_zeroconf_publish_unload,
|
||||||
|
};
|
||||||
|
|
||||||
|
static const struct spa_dict_item module_zeroconf_publish_info[] = {
|
||||||
|
{ PW_KEY_MODULE_AUTHOR, "Sanchayan Maity <sanchayan@asymptotic.io" },
|
||||||
|
{ PW_KEY_MODULE_DESCRIPTION, "mDNS/DNS-SD Service Publish" },
|
||||||
|
{ PW_KEY_MODULE_USAGE, "port=<TCP port number>" },
|
||||||
|
{ PW_KEY_MODULE_VERSION, PACKAGE_VERSION },
|
||||||
|
};
|
||||||
|
|
||||||
|
struct module *create_module_zeroconf_publish(struct impl *impl, const char *argument)
|
||||||
|
{
|
||||||
|
struct module *module;
|
||||||
|
struct module_zeroconf_publish_data *d;
|
||||||
|
struct pw_properties *props = NULL;
|
||||||
|
const char *port;
|
||||||
|
int res;
|
||||||
|
|
||||||
|
props = pw_properties_new_dict(&SPA_DICT_INIT_ARRAY(module_zeroconf_publish_info));
|
||||||
|
if (!props) {
|
||||||
|
res = -errno;
|
||||||
|
goto out;
|
||||||
|
}
|
||||||
|
if (argument)
|
||||||
|
module_args_add_props(props, argument);
|
||||||
|
|
||||||
|
module = module_new(impl, &module_zeroconf_publish_methods, sizeof(*d));
|
||||||
|
if (module == NULL) {
|
||||||
|
res = -errno;
|
||||||
|
goto out;
|
||||||
|
}
|
||||||
|
|
||||||
|
module->props = props;
|
||||||
|
d = module->user_data;
|
||||||
|
d->module = module;
|
||||||
|
|
||||||
|
if ((port = pw_properties_get(props, "port")) == NULL)
|
||||||
|
d->port = PW_PROTOCOL_PULSE_DEFAULT_PORT;
|
||||||
|
else
|
||||||
|
d->port = atoi(port);
|
||||||
|
|
||||||
|
|
||||||
|
return module;
|
||||||
|
out:
|
||||||
|
pw_properties_free(props);
|
||||||
|
errno = -res;
|
||||||
|
return NULL;
|
||||||
|
}
|
||||||
|
|
@ -42,5 +42,6 @@ struct module *create_module_tunnel_source(struct impl *impl, const char *argume
|
||||||
struct module *create_module_simple_protocol_tcp(struct impl *impl, const char *argument);
|
struct module *create_module_simple_protocol_tcp(struct impl *impl, const char *argument);
|
||||||
struct module *create_module_pipe_sink(struct impl *impl, const char *argument);
|
struct module *create_module_pipe_sink(struct impl *impl, const char *argument);
|
||||||
struct module *create_module_zeroconf_discover(struct impl *impl, const char *argument);
|
struct module *create_module_zeroconf_discover(struct impl *impl, const char *argument);
|
||||||
|
struct module *create_module_zeroconf_publish(struct impl *impl, const char *argument);
|
||||||
|
|
||||||
#endif
|
#endif
|
||||||
|
|
|
||||||
Loading…
Add table
Add a link
Reference in a new issue