mirror of
https://gitlab.freedesktop.org/pipewire/pipewire.git
synced 2026-04-19 06:46:26 -04:00
When a device profile changes (e.g., Bluetooth headset switching from a2dp-sink to headset-head-unit), the active port information changes but PulseAudio compatibility layer clients don't receive the expected PA_SUBSCRIPTION_EVENT_SOURCE or PA_SUBSCRIPTION_EVENT_SINK change events. Root cause: The collect_device_info() function updates the active_port index from SPA_PARAM_Route parameters, but doesn't update the corresponding active_port_name field. When update_device_info() uses memcmp() to detect changes in the device_info structure, it compares the entire structure including active_port_name. If the pointer value doesn't change (even though the actual port changed), no change is detected, and the change_mask flag (PW_MANAGER_OBJECT_FLAG_SOURCE/SINK) is not set, preventing subscription events from being sent. Solution: After setting active_port in collect_device_info(), look up the corresponding port name from SPA_PARAM_EnumRoute parameters by matching both the port index and direction. Initialize active_port_name to NULL at the start to ensure it's always recalculated. This fix applies to all device types (Bluetooth, USB, PCI sound cards) and all profile switching scenarios, ensuring applications using the PulseAudio compatibility layer receive proper device change notifications. Tested with: - Bluetooth headset profile switching (a2dp-sink ↔ headset-head-unit) - Applications subscribing to PA_SUBSCRIPTION_MASK_SOURCE/SINK events - Verified no regression in audio initialization
645 lines
17 KiB
C
645 lines
17 KiB
C
/* PipeWire */
|
|
/* SPDX-FileCopyrightText: Copyright © 2020 Wim Taymans */
|
|
/* SPDX-License-Identifier: MIT */
|
|
|
|
#include <spa/param/props.h>
|
|
#include <spa/pod/builder.h>
|
|
#include <spa/pod/parser.h>
|
|
#include <spa/utils/string.h>
|
|
|
|
#include <spa/param/audio/format-utils.h>
|
|
|
|
#include <pipewire/pipewire.h>
|
|
|
|
#include "collect.h"
|
|
#include "defs.h"
|
|
#include "log.h"
|
|
#include "manager.h"
|
|
|
|
void select_best(struct selector *s, struct pw_manager_object *o)
|
|
{
|
|
int32_t prio = 0;
|
|
|
|
if (o->props &&
|
|
pw_properties_fetch_int32(o->props, PW_KEY_PRIORITY_SESSION, &prio) == 0) {
|
|
if (s->best == NULL || prio > s->score) {
|
|
s->best = o;
|
|
s->score = prio;
|
|
}
|
|
}
|
|
}
|
|
|
|
struct pw_manager_object *select_object(struct pw_manager *m, struct selector *s)
|
|
{
|
|
struct pw_manager_object *o;
|
|
const char *str;
|
|
|
|
spa_list_for_each(o, &m->object_list, link) {
|
|
if (o->creating || o->removing)
|
|
continue;
|
|
if (s->type != NULL && !s->type(o))
|
|
continue;
|
|
if (o->id == s->id)
|
|
return o;
|
|
if (o->index == s->index)
|
|
return o;
|
|
if (s->accumulate)
|
|
s->accumulate(s, o);
|
|
if (o->props && s->key != NULL && s->value != NULL &&
|
|
(str = pw_properties_get(o->props, s->key)) != NULL &&
|
|
spa_streq(str, s->value))
|
|
return o;
|
|
if (s->value != NULL && (uint32_t)atoi(s->value) == o->index)
|
|
return o;
|
|
}
|
|
return s->best;
|
|
}
|
|
|
|
uint32_t id_to_index(struct pw_manager *m, uint32_t id)
|
|
{
|
|
struct pw_manager_object *o;
|
|
spa_list_for_each(o, &m->object_list, link) {
|
|
if (o->id == id)
|
|
return o->index;
|
|
}
|
|
return SPA_ID_INVALID;
|
|
}
|
|
|
|
static bool collect_is_linked(struct pw_manager *m, uint32_t id, enum pw_direction direction)
|
|
{
|
|
struct pw_manager_object *o;
|
|
uint32_t in_node, out_node;
|
|
|
|
spa_list_for_each(o, &m->object_list, link) {
|
|
if (o->props == NULL || !pw_manager_object_is_link(o))
|
|
continue;
|
|
|
|
if (pw_properties_fetch_uint32(o->props, PW_KEY_LINK_OUTPUT_NODE, &out_node) != 0 ||
|
|
pw_properties_fetch_uint32(o->props, PW_KEY_LINK_INPUT_NODE, &in_node) != 0)
|
|
continue;
|
|
|
|
if ((direction == PW_DIRECTION_OUTPUT && id == out_node) ||
|
|
(direction == PW_DIRECTION_INPUT && id == in_node))
|
|
return true;
|
|
}
|
|
return false;
|
|
}
|
|
|
|
struct pw_manager_object *find_peer_for_link(struct pw_manager *m,
|
|
struct pw_manager_object *o, uint32_t id, enum pw_direction direction)
|
|
{
|
|
struct pw_manager_object *p;
|
|
uint32_t in_node, out_node;
|
|
|
|
if (o->props == NULL)
|
|
return NULL;
|
|
|
|
if (pw_properties_fetch_uint32(o->props, PW_KEY_LINK_OUTPUT_NODE, &out_node) != 0 ||
|
|
pw_properties_fetch_uint32(o->props, PW_KEY_LINK_INPUT_NODE, &in_node) != 0)
|
|
return NULL;
|
|
|
|
if (direction == PW_DIRECTION_OUTPUT && id == out_node) {
|
|
struct selector sel = { .id = in_node, .type = pw_manager_object_is_sink, };
|
|
if ((p = select_object(m, &sel)) != NULL)
|
|
return p;
|
|
}
|
|
if (direction == PW_DIRECTION_INPUT && id == in_node) {
|
|
struct selector sel = { .id = out_node, .type = pw_manager_object_is_recordable, };
|
|
if ((p = select_object(m, &sel)) != NULL)
|
|
return p;
|
|
}
|
|
return NULL;
|
|
}
|
|
|
|
struct pw_manager_object *find_linked(struct pw_manager *m, uint32_t id, enum pw_direction direction)
|
|
{
|
|
struct pw_manager_object *o, *p;
|
|
|
|
spa_list_for_each(o, &m->object_list, link) {
|
|
if (!pw_manager_object_is_link(o))
|
|
continue;
|
|
if ((p = find_peer_for_link(m, o, id, direction)) != NULL)
|
|
return p;
|
|
}
|
|
return NULL;
|
|
}
|
|
|
|
void collect_card_info(struct pw_manager_object *card, struct card_info *info)
|
|
{
|
|
struct pw_manager_param *p;
|
|
|
|
spa_list_for_each(p, &card->param_list, link) {
|
|
switch (p->id) {
|
|
case SPA_PARAM_EnumProfile:
|
|
info->n_profiles++;
|
|
break;
|
|
case SPA_PARAM_Profile:
|
|
spa_pod_parse_object(p->param,
|
|
SPA_TYPE_OBJECT_ParamProfile, NULL,
|
|
SPA_PARAM_PROFILE_index, SPA_POD_Int(&info->active_profile));
|
|
break;
|
|
case SPA_PARAM_EnumRoute:
|
|
info->n_ports++;
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
|
|
uint32_t collect_profile_info(struct pw_manager_object *card, struct card_info *card_info,
|
|
struct profile_info *profile_info)
|
|
{
|
|
struct pw_manager_param *p;
|
|
struct profile_info *pi;
|
|
uint32_t n;
|
|
|
|
n = 0;
|
|
spa_list_for_each(p, &card->param_list, link) {
|
|
struct spa_pod *classes = NULL;
|
|
|
|
if (p->id != SPA_PARAM_EnumProfile)
|
|
continue;
|
|
|
|
pi = &profile_info[n];
|
|
spa_zero(*pi);
|
|
|
|
if (spa_pod_parse_object(p->param,
|
|
SPA_TYPE_OBJECT_ParamProfile, NULL,
|
|
SPA_PARAM_PROFILE_index, SPA_POD_Int(&pi->index),
|
|
SPA_PARAM_PROFILE_name, SPA_POD_String(&pi->name),
|
|
SPA_PARAM_PROFILE_description, SPA_POD_OPT_String(&pi->description),
|
|
SPA_PARAM_PROFILE_priority, SPA_POD_OPT_Int(&pi->priority),
|
|
SPA_PARAM_PROFILE_available, SPA_POD_OPT_Id(&pi->available),
|
|
SPA_PARAM_PROFILE_classes, SPA_POD_OPT_PodStruct(&classes)) < 0) {
|
|
continue;
|
|
}
|
|
if (pi->description == NULL)
|
|
pi->description = pi->name;
|
|
if (pi->index == card_info->active_profile)
|
|
card_info->active_profile_name = pi->name;
|
|
|
|
if (classes != NULL) {
|
|
struct spa_pod *iter;
|
|
|
|
SPA_POD_STRUCT_FOREACH(classes, iter) {
|
|
struct spa_pod_parser prs;
|
|
char *class;
|
|
uint32_t count;
|
|
|
|
spa_pod_parser_pod(&prs, iter);
|
|
if (spa_pod_parser_get_struct(&prs,
|
|
SPA_POD_String(&class),
|
|
SPA_POD_Int(&count)) < 0)
|
|
continue;
|
|
|
|
if (spa_streq(class, "Audio/Sink"))
|
|
pi->n_sinks += count;
|
|
else if (spa_streq(class, "Audio/Source"))
|
|
pi->n_sources += count;
|
|
}
|
|
}
|
|
n++;
|
|
}
|
|
if (card_info->active_profile_name == NULL && n > 0)
|
|
card_info->active_profile_name = profile_info[0].name;
|
|
|
|
return n;
|
|
}
|
|
|
|
uint32_t find_profile_index(struct pw_manager_object *card, const char *name)
|
|
{
|
|
struct pw_manager_param *p;
|
|
|
|
spa_list_for_each(p, &card->param_list, link) {
|
|
uint32_t index;
|
|
const char *test_name;
|
|
|
|
if (p->id != SPA_PARAM_EnumProfile)
|
|
continue;
|
|
|
|
if (spa_pod_parse_object(p->param,
|
|
SPA_TYPE_OBJECT_ParamProfile, NULL,
|
|
SPA_PARAM_PROFILE_index, SPA_POD_Int(&index),
|
|
SPA_PARAM_PROFILE_name, SPA_POD_String(&test_name)) < 0)
|
|
continue;
|
|
|
|
if (spa_streq(test_name, name))
|
|
return index;
|
|
|
|
}
|
|
return SPA_ID_INVALID;
|
|
}
|
|
|
|
static void collect_device_info(struct pw_manager_object *device, struct pw_manager_object *card,
|
|
struct device_info *dev_info, bool monitor, struct defs *defs)
|
|
{
|
|
struct pw_manager_param *p;
|
|
dev_info->active_port_name = NULL;
|
|
|
|
if (card) {
|
|
spa_list_for_each(p, &card->param_list, link) {
|
|
uint32_t index, dev;
|
|
struct spa_pod *props;
|
|
|
|
if (p->id != SPA_PARAM_Route)
|
|
continue;
|
|
|
|
if (spa_pod_parse_object(p->param,
|
|
SPA_TYPE_OBJECT_ParamRoute, NULL,
|
|
SPA_PARAM_ROUTE_index, SPA_POD_Int(&index),
|
|
SPA_PARAM_ROUTE_device, SPA_POD_Int(&dev),
|
|
SPA_PARAM_ROUTE_props, SPA_POD_OPT_PodObject(&props)) < 0)
|
|
continue;
|
|
if (dev != dev_info->device)
|
|
continue;
|
|
dev_info->active_port = index;
|
|
if (props && !monitor) {
|
|
volume_parse_param(props, &dev_info->volume_info, monitor);
|
|
dev_info->have_volume = true;
|
|
}
|
|
}
|
|
|
|
/* Look up the port name for the active port */
|
|
if (dev_info->active_port != SPA_ID_INVALID) {
|
|
spa_list_for_each(p, &card->param_list, link) {
|
|
uint32_t index, direction;
|
|
const char *name = NULL;
|
|
|
|
if (p->id != SPA_PARAM_EnumRoute)
|
|
continue;
|
|
|
|
if (spa_pod_parse_object(p->param,
|
|
SPA_TYPE_OBJECT_ParamRoute, NULL,
|
|
SPA_PARAM_ROUTE_index, SPA_POD_Int(&index),
|
|
SPA_PARAM_ROUTE_direction, SPA_POD_Id(&direction),
|
|
SPA_PARAM_ROUTE_name, SPA_POD_String(&name)) < 0)
|
|
continue;
|
|
|
|
if (index == dev_info->active_port &&
|
|
direction == dev_info->direction) {
|
|
dev_info->active_port_name = name;
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
spa_list_for_each(p, &device->param_list, link) {
|
|
switch (p->id) {
|
|
case SPA_PARAM_EnumFormat:
|
|
{
|
|
struct spa_pod *copy = spa_pod_copy(p->param);
|
|
spa_pod_fixate(copy);
|
|
format_parse_param(copy, true, &dev_info->ss, &dev_info->map,
|
|
&defs->sample_spec, &defs->channel_map);
|
|
free(copy);
|
|
break;
|
|
}
|
|
case SPA_PARAM_Format:
|
|
format_parse_param(p->param, true, &dev_info->ss, &dev_info->map,
|
|
NULL, NULL);
|
|
break;
|
|
|
|
case SPA_PARAM_Props:
|
|
if (!dev_info->have_volume) {
|
|
volume_parse_param(p->param, &dev_info->volume_info, monitor);
|
|
dev_info->have_volume = true;
|
|
}
|
|
dev_info->have_iec958codecs = spa_pod_find_prop(p->param,
|
|
NULL, SPA_PROP_iec958Codecs) != NULL;
|
|
break;
|
|
}
|
|
}
|
|
if (dev_info->ss.channels != dev_info->map.channels)
|
|
dev_info->ss.channels = dev_info->map.channels;
|
|
if (dev_info->volume_info.volume.channels != dev_info->map.channels)
|
|
dev_info->volume_info.volume.channels = dev_info->map.channels;
|
|
}
|
|
|
|
static void update_device_info(struct pw_manager *manager, struct pw_manager_object *o,
|
|
enum pw_direction direction, bool monitor, struct defs *defs, bool stream)
|
|
{
|
|
const char *str;
|
|
const char *key = monitor ? "device.info.monitor" : "device.info";
|
|
struct pw_manager_object *card = NULL;
|
|
struct pw_node_info *info = o->info;
|
|
struct device_info *dev_info, di;
|
|
|
|
if (info == NULL)
|
|
return;
|
|
|
|
di = DEVICE_INFO_INIT(direction);
|
|
if ((str = spa_dict_lookup(info->props, PW_KEY_DEVICE_ID)) != NULL)
|
|
di.card_id = (uint32_t)atoi(str);
|
|
if ((str = spa_dict_lookup(info->props, "card.profile.device")) != NULL)
|
|
di.device = (uint32_t)atoi(str);
|
|
if (di.card_id != SPA_ID_INVALID) {
|
|
struct selector sel = { .id = di.card_id, .type = pw_manager_object_is_card, };
|
|
card = select_object(manager, &sel);
|
|
}
|
|
collect_device_info(o, card, &di, monitor, defs);
|
|
|
|
di.state = node_state(info->state);
|
|
/* running sink/source that is not linked is reported as idle */
|
|
if (!stream && di.state == STATE_RUNNING &&
|
|
!collect_is_linked(manager, o->id, pw_direction_reverse(direction)))
|
|
di.state = STATE_IDLE;
|
|
|
|
dev_info = pw_manager_object_get_data(o, key);
|
|
if (dev_info) {
|
|
if (memcmp(dev_info, &di, sizeof(di)) != 0) {
|
|
if (monitor || direction == PW_DIRECTION_INPUT)
|
|
o->change_mask |= PW_MANAGER_OBJECT_FLAG_SOURCE;
|
|
else
|
|
o->change_mask |= PW_MANAGER_OBJECT_FLAG_SINK;
|
|
}
|
|
} else {
|
|
o->change_mask = ~0;
|
|
dev_info = pw_manager_object_add_data(o, key, sizeof(*dev_info));
|
|
}
|
|
if (dev_info != NULL)
|
|
*dev_info = di;
|
|
}
|
|
|
|
void get_device_info(struct pw_manager_object *o, struct device_info *info,
|
|
enum pw_direction direction, bool monitor)
|
|
{
|
|
const char *key = monitor ? "device.info.monitor" : "device.info";
|
|
struct device_info *di;
|
|
di = pw_manager_object_get_data(o, key);
|
|
if (di != NULL)
|
|
*info = *di;
|
|
else
|
|
*info = DEVICE_INFO_INIT(direction);
|
|
}
|
|
|
|
static bool array_contains(uint32_t *vals, uint32_t n_vals, uint32_t val)
|
|
{
|
|
uint32_t n;
|
|
if (vals == NULL || n_vals == 0)
|
|
return false;
|
|
for (n = 0; n < n_vals; n++)
|
|
if (vals[n] == val)
|
|
return true;
|
|
return false;
|
|
}
|
|
|
|
uint32_t collect_port_info(struct pw_manager_object *card, struct card_info *card_info,
|
|
struct device_info *dev_info, struct port_info *port_info)
|
|
{
|
|
struct pw_manager_param *p;
|
|
uint32_t n;
|
|
|
|
if (card == NULL)
|
|
return 0;
|
|
|
|
n = 0;
|
|
spa_list_for_each(p, &card->param_list, link) {
|
|
int32_t *devices = NULL, *profiles = NULL;
|
|
uint32_t devices_size = 0, devices_type = 0, n_devices = 0;
|
|
uint32_t profiles_size = 0, profiles_type = 0, n_profiles = 0;
|
|
struct port_info *pi;
|
|
|
|
if (p->id != SPA_PARAM_EnumRoute)
|
|
continue;
|
|
|
|
pi = &port_info[n];
|
|
spa_zero(*pi);
|
|
|
|
if (spa_pod_parse_object(p->param,
|
|
SPA_TYPE_OBJECT_ParamRoute, NULL,
|
|
SPA_PARAM_ROUTE_index, SPA_POD_Int(&pi->index),
|
|
SPA_PARAM_ROUTE_direction, SPA_POD_Id(&pi->direction),
|
|
SPA_PARAM_ROUTE_name, SPA_POD_String(&pi->name),
|
|
SPA_PARAM_ROUTE_description, SPA_POD_OPT_String(&pi->description),
|
|
SPA_PARAM_ROUTE_priority, SPA_POD_OPT_Int(&pi->priority),
|
|
SPA_PARAM_ROUTE_available, SPA_POD_OPT_Id(&pi->available),
|
|
SPA_PARAM_ROUTE_info, SPA_POD_OPT_Pod(&pi->info),
|
|
SPA_PARAM_ROUTE_devices, SPA_POD_OPT_Array(&devices_size,
|
|
&devices_type, &n_devices, &devices),
|
|
SPA_PARAM_ROUTE_profiles, SPA_POD_OPT_Array(&profiles_size,
|
|
&profiles_type, &n_profiles, &profiles)) < 0)
|
|
continue;
|
|
|
|
if (pi->description == NULL)
|
|
pi->description = pi->name;
|
|
if (devices && devices_size == sizeof(pi->devices[0]) &&
|
|
devices_type == SPA_TYPE_Int) {
|
|
pi->devices = (uint32_t*)devices;
|
|
pi->n_devices = n_devices;
|
|
}
|
|
if (profiles && profiles_size == sizeof(pi->profiles[0]) &&
|
|
profiles_type == SPA_TYPE_Int) {
|
|
pi->profiles = (uint32_t*)profiles;
|
|
pi->n_profiles = n_profiles;
|
|
}
|
|
|
|
if (dev_info != NULL) {
|
|
if (pi->direction != dev_info->direction)
|
|
continue;
|
|
if (!array_contains(pi->profiles, pi->n_profiles, card_info->active_profile))
|
|
continue;
|
|
if (!array_contains(pi->devices, pi->n_devices, dev_info->device))
|
|
continue;
|
|
if (pi->index == dev_info->active_port)
|
|
dev_info->active_port_name = pi->name;
|
|
}
|
|
|
|
while (pi->info != NULL) {
|
|
struct spa_pod_parser prs;
|
|
struct spa_pod_frame f[1];
|
|
uint32_t n;
|
|
const char *key, *value;
|
|
|
|
spa_pod_parser_pod(&prs, pi->info);
|
|
if (spa_pod_parser_push_struct(&prs, &f[0]) < 0 ||
|
|
spa_pod_parser_get_int(&prs, (int32_t*)&pi->n_props) < 0)
|
|
break;
|
|
|
|
for (n = 0; n < pi->n_props; n++) {
|
|
if (spa_pod_parser_get(&prs,
|
|
SPA_POD_String(&key),
|
|
SPA_POD_String(&value),
|
|
NULL) < 0)
|
|
break;
|
|
if (spa_streq(key, "port.availability-group"))
|
|
pi->availability_group = value;
|
|
else if (spa_streq(key, "port.type"))
|
|
pi->type = port_type_value(value);
|
|
}
|
|
spa_pod_parser_pop(&prs, &f[0]);
|
|
break;
|
|
}
|
|
n++;
|
|
}
|
|
if (dev_info != NULL && dev_info->active_port_name == NULL && n > 0)
|
|
dev_info->active_port_name = port_info[0].name;
|
|
return n;
|
|
}
|
|
|
|
uint32_t find_port_index(struct pw_manager_object *card, uint32_t direction, const char *port_name)
|
|
{
|
|
struct pw_manager_param *p;
|
|
|
|
spa_list_for_each(p, &card->param_list, link) {
|
|
uint32_t index, dir;
|
|
const char *name;
|
|
|
|
if (p->id != SPA_PARAM_EnumRoute)
|
|
continue;
|
|
|
|
if (spa_pod_parse_object(p->param,
|
|
SPA_TYPE_OBJECT_ParamRoute, NULL,
|
|
SPA_PARAM_ROUTE_index, SPA_POD_Int(&index),
|
|
SPA_PARAM_ROUTE_direction, SPA_POD_Id(&dir),
|
|
SPA_PARAM_ROUTE_name, SPA_POD_String(&name)) < 0)
|
|
continue;
|
|
if (dir != direction)
|
|
continue;
|
|
if (spa_streq(name, port_name))
|
|
return index;
|
|
|
|
}
|
|
return SPA_ID_INVALID;
|
|
}
|
|
|
|
struct spa_dict *collect_props(struct spa_pod *info, struct spa_dict *dict)
|
|
{
|
|
struct spa_pod_parser prs;
|
|
struct spa_pod_frame f[1];
|
|
int32_t n, n_items;
|
|
|
|
spa_pod_parser_pod(&prs, info);
|
|
if (spa_pod_parser_push_struct(&prs, &f[0]) < 0 ||
|
|
spa_pod_parser_get_int(&prs, &n_items) < 0)
|
|
return NULL;
|
|
|
|
for (n = 0; n < n_items; n++) {
|
|
if (spa_pod_parser_get(&prs,
|
|
SPA_POD_String(&dict->items[n].key),
|
|
SPA_POD_String(&dict->items[n].value),
|
|
NULL) < 0)
|
|
break;
|
|
}
|
|
spa_pod_parser_pop(&prs, &f[0]);
|
|
dict->n_items = n;
|
|
return dict;
|
|
}
|
|
|
|
uint32_t collect_transport_codec_info(struct pw_manager_object *card,
|
|
struct transport_codec_info *codecs, uint32_t max_codecs,
|
|
uint32_t *active)
|
|
{
|
|
struct pw_manager_param *p;
|
|
uint32_t n_codecs = 0;
|
|
|
|
*active = SPA_ID_INVALID;
|
|
|
|
if (card == NULL)
|
|
return 0;
|
|
|
|
spa_list_for_each(p, &card->param_list, link) {
|
|
uint32_t iid;
|
|
const struct spa_pod_choice *type;
|
|
const struct spa_pod_struct *labels;
|
|
struct spa_pod_parser prs;
|
|
struct spa_pod_frame f;
|
|
int32_t *id;
|
|
bool first;
|
|
|
|
if (p->id != SPA_PARAM_PropInfo)
|
|
continue;
|
|
|
|
if (spa_pod_parse_object(p->param,
|
|
SPA_TYPE_OBJECT_PropInfo, NULL,
|
|
SPA_PROP_INFO_id, SPA_POD_Id(&iid),
|
|
SPA_PROP_INFO_type, SPA_POD_PodChoice(&type),
|
|
SPA_PROP_INFO_labels, SPA_POD_PodStruct(&labels)) < 0)
|
|
continue;
|
|
|
|
if (iid != SPA_PROP_bluetoothAudioCodec)
|
|
continue;
|
|
|
|
if (type->pod.size < sizeof(struct spa_pod_choice_body) +
|
|
2 * sizeof(int32_t) ||
|
|
type->body.type != SPA_CHOICE_Enum ||
|
|
type->body.child.type != SPA_TYPE_Int ||
|
|
type->body.child.size != sizeof(int32_t))
|
|
continue;
|
|
|
|
/*
|
|
* XXX: PropInfo currently uses Int, not Id, in type and labels.
|
|
*/
|
|
|
|
/* Codec name list */
|
|
first = true;
|
|
SPA_POD_CHOICE_FOREACH(type, id) {
|
|
if (first) {
|
|
/* Skip default */
|
|
first = false;
|
|
continue;
|
|
}
|
|
if (n_codecs >= max_codecs)
|
|
break;
|
|
codecs[n_codecs++].id = *id;
|
|
}
|
|
|
|
/* Codec description list */
|
|
spa_pod_parser_pod(&prs, (struct spa_pod *)labels);
|
|
if (spa_pod_parser_push_struct(&prs, &f) < 0)
|
|
continue;
|
|
|
|
while (1) {
|
|
int32_t id;
|
|
const char *desc;
|
|
uint32_t j;
|
|
|
|
if (spa_pod_parser_get_int(&prs, &id) < 0 ||
|
|
spa_pod_parser_get_string(&prs, &desc) < 0)
|
|
break;
|
|
|
|
for (j = 0; j < n_codecs; ++j) {
|
|
if (codecs[j].id == (uint32_t)id)
|
|
codecs[j].description = desc;
|
|
}
|
|
}
|
|
}
|
|
|
|
/* Active codec */
|
|
spa_list_for_each(p, &card->param_list, link) {
|
|
uint32_t j;
|
|
uint32_t id;
|
|
|
|
if (p->id != SPA_PARAM_Props)
|
|
continue;
|
|
|
|
if (spa_pod_parse_object(p->param,
|
|
SPA_TYPE_OBJECT_Props, NULL,
|
|
SPA_PROP_bluetoothAudioCodec, SPA_POD_Id(&id)) < 0)
|
|
continue;
|
|
|
|
for (j = 0; j < n_codecs; ++j) {
|
|
if (codecs[j].id == id)
|
|
*active = j;
|
|
}
|
|
}
|
|
|
|
return n_codecs;
|
|
}
|
|
|
|
void update_object_info(struct pw_manager *manager, struct pw_manager_object *o,
|
|
struct defs *defs)
|
|
{
|
|
if (pw_manager_object_is_sink(o)) {
|
|
update_device_info(manager, o, PW_DIRECTION_OUTPUT, false, defs, false);
|
|
update_device_info(manager, o, PW_DIRECTION_OUTPUT, true, defs, false);
|
|
}
|
|
if (pw_manager_object_is_source(o)) {
|
|
update_device_info(manager, o, PW_DIRECTION_INPUT, false, defs, false);
|
|
}
|
|
if (pw_manager_object_is_source_output(o)) {
|
|
update_device_info(manager, o, PW_DIRECTION_INPUT, false, defs, true);
|
|
}
|
|
if (pw_manager_object_is_sink_input(o)) {
|
|
update_device_info(manager, o, PW_DIRECTION_OUTPUT, false, defs, true);
|
|
}
|
|
}
|