2018-10-10 18:50:11 +02:00
|
|
|
/* PipeWire
|
|
|
|
|
*
|
2020-02-07 15:43:13 +01:00
|
|
|
* Copyright © 2018 Wim Taymans
|
2018-10-10 18:50:11 +02:00
|
|
|
*
|
2020-02-07 15:43:13 +01:00
|
|
|
* 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:
|
2018-10-10 18:50:11 +02:00
|
|
|
*
|
2020-02-07 15:43:13 +01:00
|
|
|
* 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.
|
2018-10-10 18:50:11 +02:00
|
|
|
*/
|
|
|
|
|
|
|
|
|
|
#ifdef HAVE_CONFIG_H
|
|
|
|
|
#include "config.h"
|
|
|
|
|
#endif
|
|
|
|
|
|
|
|
|
|
#include <stdio.h>
|
|
|
|
|
#include <unistd.h>
|
|
|
|
|
#include <sys/mman.h>
|
|
|
|
|
|
|
|
|
|
#include <jack/metadata.h>
|
2019-08-21 18:37:02 +02:00
|
|
|
#include <jack/uuid.h>
|
2018-10-10 18:50:11 +02:00
|
|
|
|
|
|
|
|
#include <pipewire/pipewire.h>
|
2019-11-05 10:01:43 +01:00
|
|
|
#include <extensions/metadata.h>
|
2019-08-21 18:37:02 +02:00
|
|
|
|
|
|
|
|
static struct pw_properties * get_properties(void)
|
|
|
|
|
{
|
2019-11-05 10:01:43 +01:00
|
|
|
if (globals.properties == NULL) {
|
|
|
|
|
globals.properties = pw_properties_new(NULL, NULL);
|
2019-08-21 18:37:02 +02:00
|
|
|
}
|
2019-11-05 10:01:43 +01:00
|
|
|
return globals.properties;
|
2019-08-21 18:37:02 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
static void make_key(char *dst, jack_uuid_t subject, const char *key, int keylen)
|
|
|
|
|
{
|
|
|
|
|
int len;
|
2020-04-20 11:50:37 +02:00
|
|
|
jack_uuid_unparse(subject, dst);
|
2019-08-21 18:37:02 +02:00
|
|
|
len = strlen(dst);
|
|
|
|
|
dst[len] = '@';
|
|
|
|
|
memcpy(&dst[len+1], key, keylen+1);
|
|
|
|
|
}
|
|
|
|
|
|
2020-04-20 11:50:37 +02:00
|
|
|
static int update_property(struct client *c,
|
|
|
|
|
jack_uuid_t subject,
|
|
|
|
|
const char* key,
|
|
|
|
|
const char* value,
|
|
|
|
|
const char* type)
|
|
|
|
|
{
|
|
|
|
|
int keylen = strlen(key);
|
|
|
|
|
char *dst = alloca(JACK_UUID_STRING_SIZE + keylen);
|
|
|
|
|
struct pw_properties * props = get_properties();
|
|
|
|
|
jack_property_change_t change;
|
|
|
|
|
|
|
|
|
|
make_key(dst, subject, key, keylen);
|
|
|
|
|
|
|
|
|
|
if (value == NULL || type == NULL) {
|
|
|
|
|
pw_properties_setf(props, dst, NULL);
|
|
|
|
|
change = PropertyDeleted;
|
|
|
|
|
} else {
|
|
|
|
|
change = PropertyCreated;
|
|
|
|
|
if (pw_properties_get(props, dst) != NULL)
|
|
|
|
|
change = PropertyChanged;
|
|
|
|
|
|
|
|
|
|
pw_properties_setf(props, dst, "%s@%s", value, type);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (c->property_callback)
|
|
|
|
|
c->property_callback(subject, key, change, c->property_arg);
|
|
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
2019-04-11 16:36:52 +02:00
|
|
|
SPA_EXPORT
|
|
|
|
|
int jack_set_property(jack_client_t*client,
|
|
|
|
|
jack_uuid_t subject,
|
|
|
|
|
const char* key,
|
|
|
|
|
const char* value,
|
|
|
|
|
const char* type)
|
2018-10-10 18:50:11 +02:00
|
|
|
{
|
2019-11-05 10:01:43 +01:00
|
|
|
struct client *c = (struct client *) client;
|
|
|
|
|
uint32_t id;
|
2019-08-21 18:37:02 +02:00
|
|
|
|
2020-02-24 11:25:31 +01:00
|
|
|
spa_return_val_if_fail(c != NULL, -EINVAL);
|
|
|
|
|
spa_return_val_if_fail(key != NULL, -EINVAL);
|
|
|
|
|
spa_return_val_if_fail(value != NULL, -EINVAL);
|
2020-04-20 11:50:37 +02:00
|
|
|
|
|
|
|
|
if (c->metadata == NULL)
|
|
|
|
|
return -1;
|
2020-02-24 11:25:31 +01:00
|
|
|
|
2019-11-05 10:01:43 +01:00
|
|
|
id = jack_uuid_to_index(subject);
|
2019-08-21 18:37:02 +02:00
|
|
|
|
2020-04-20 11:50:37 +02:00
|
|
|
if (type == NULL)
|
|
|
|
|
type = "";
|
|
|
|
|
|
|
|
|
|
pw_log_info("set id:%u (%lu) '%s' to '%s@%s'", id, subject, key, value, type);
|
2019-11-05 10:01:43 +01:00
|
|
|
pw_metadata_set_property(c->metadata->proxy,
|
|
|
|
|
id, key, type, value);
|
2020-04-20 11:50:37 +02:00
|
|
|
|
2019-08-21 18:37:02 +02:00
|
|
|
return 0;
|
2018-10-10 18:50:11 +02:00
|
|
|
}
|
|
|
|
|
|
2019-04-11 16:36:52 +02:00
|
|
|
SPA_EXPORT
|
|
|
|
|
int jack_get_property(jack_uuid_t subject,
|
|
|
|
|
const char* key,
|
|
|
|
|
char** value,
|
|
|
|
|
char** type)
|
2018-10-10 18:50:11 +02:00
|
|
|
{
|
2020-02-24 11:25:31 +01:00
|
|
|
int keylen;
|
|
|
|
|
char *dst;
|
2019-08-21 18:37:02 +02:00
|
|
|
struct pw_properties * props = get_properties();
|
|
|
|
|
const char *str, *at;
|
|
|
|
|
|
2020-04-20 11:50:37 +02:00
|
|
|
spa_return_val_if_fail(props != NULL, -EIO);
|
2020-02-24 11:25:31 +01:00
|
|
|
spa_return_val_if_fail(key != NULL, -EINVAL);
|
|
|
|
|
spa_return_val_if_fail(value != NULL, -EINVAL);
|
|
|
|
|
spa_return_val_if_fail(type != NULL, -EINVAL);
|
|
|
|
|
|
|
|
|
|
keylen = strlen(key);
|
|
|
|
|
dst = alloca(JACK_UUID_STRING_SIZE + keylen);
|
2019-08-21 18:37:02 +02:00
|
|
|
make_key(dst, subject, key, keylen);
|
|
|
|
|
|
|
|
|
|
if ((str = pw_properties_get(props, dst)) == NULL) {
|
|
|
|
|
pw_log_warn("no property '%s'", dst);
|
|
|
|
|
return -1;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
at = strrchr(str, '@');
|
|
|
|
|
if (at == NULL) {
|
|
|
|
|
pw_log_warn("property '%s' invalid value '%s'", dst, str);
|
|
|
|
|
return -1;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
*value = strndup(str, at - str);
|
|
|
|
|
*type = strdup(at + 1);
|
|
|
|
|
|
2020-04-20 11:50:37 +02:00
|
|
|
pw_log_info("got '%s' with value:'%s' type:'%s'", dst, *value, *type);
|
2019-08-21 18:37:02 +02:00
|
|
|
|
|
|
|
|
return 0;
|
2018-10-10 18:50:11 +02:00
|
|
|
}
|
|
|
|
|
|
2019-04-11 16:36:52 +02:00
|
|
|
SPA_EXPORT
|
|
|
|
|
void jack_free_description (jack_description_t* desc, int free_description_itself)
|
2018-10-10 18:50:11 +02:00
|
|
|
{
|
2020-04-20 11:50:37 +02:00
|
|
|
uint32_t n;
|
|
|
|
|
|
|
|
|
|
for (n = 0; n < desc->property_cnt; ++n) {
|
|
|
|
|
free((char*)desc->properties[n].key);
|
|
|
|
|
free((char*)desc->properties[n].data);
|
|
|
|
|
if (desc->properties[n].type)
|
|
|
|
|
free((char*)desc->properties[n].type);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
free(desc->properties);
|
|
|
|
|
|
|
|
|
|
if (free_description_itself)
|
|
|
|
|
free(desc);
|
2018-10-10 18:50:11 +02:00
|
|
|
}
|
|
|
|
|
|
2019-04-11 16:36:52 +02:00
|
|
|
SPA_EXPORT
|
|
|
|
|
int jack_get_properties (jack_uuid_t subject,
|
|
|
|
|
jack_description_t* desc)
|
2018-10-10 18:50:11 +02:00
|
|
|
{
|
2020-04-20 11:50:37 +02:00
|
|
|
const struct spa_dict_item *item;
|
|
|
|
|
struct pw_properties * props = get_properties();
|
|
|
|
|
char dst[JACK_UUID_STRING_SIZE+2], *at;
|
|
|
|
|
uint32_t props_size, cnt;
|
|
|
|
|
int len;
|
|
|
|
|
|
|
|
|
|
spa_return_val_if_fail(props != NULL, -EIO);
|
|
|
|
|
spa_return_val_if_fail(desc != NULL, -EINVAL);
|
|
|
|
|
|
|
|
|
|
jack_uuid_copy(&desc->subject, subject);
|
|
|
|
|
desc->properties = NULL;
|
|
|
|
|
cnt = props_size = 0;
|
|
|
|
|
|
|
|
|
|
jack_uuid_unparse(subject, dst);
|
|
|
|
|
len = strlen(dst);
|
|
|
|
|
dst[len] = '@';
|
|
|
|
|
dst[len+1] = 0;
|
|
|
|
|
|
|
|
|
|
pw_log_debug("keys for: %s", dst);
|
|
|
|
|
|
|
|
|
|
spa_dict_for_each(item, &props->dict) {
|
|
|
|
|
jack_property_t* prop;
|
|
|
|
|
|
|
|
|
|
pw_log_debug("keys %s <-> %s", item->key, dst);
|
|
|
|
|
|
|
|
|
|
if (strstr(item->key, dst) != item->key)
|
|
|
|
|
continue;
|
|
|
|
|
|
|
|
|
|
if (cnt == props_size) {
|
|
|
|
|
props_size = props_size > 0 ? props_size * 2 : 8;
|
|
|
|
|
desc->properties = realloc(desc->properties, sizeof(jack_property_t) * props_size);
|
|
|
|
|
}
|
|
|
|
|
prop = &desc->properties[cnt];
|
|
|
|
|
|
|
|
|
|
pw_log_debug("match %s", item->value);
|
|
|
|
|
at = strrchr(item->value, '@');
|
|
|
|
|
if (at == NULL)
|
|
|
|
|
continue;
|
|
|
|
|
|
|
|
|
|
prop->key = strdup(item->key + len + 1);
|
|
|
|
|
prop->data = strndup(item->value, at - item->value);
|
|
|
|
|
prop->type = strdup(at + 1);
|
|
|
|
|
cnt++;
|
|
|
|
|
}
|
|
|
|
|
desc->property_cnt = cnt;
|
|
|
|
|
return cnt;
|
2018-10-10 18:50:11 +02:00
|
|
|
}
|
|
|
|
|
|
2019-04-11 16:36:52 +02:00
|
|
|
SPA_EXPORT
|
2020-04-20 11:50:37 +02:00
|
|
|
int jack_get_all_properties (jack_description_t** descriptions)
|
2018-10-10 18:50:11 +02:00
|
|
|
{
|
2020-04-20 11:50:37 +02:00
|
|
|
const struct spa_dict_item *item;
|
|
|
|
|
struct pw_properties * props = get_properties();
|
|
|
|
|
char *at;
|
|
|
|
|
jack_description_t *desc, *cdesc;
|
|
|
|
|
uint32_t dsize, dcnt, n, len;
|
|
|
|
|
jack_uuid_t uuid;
|
|
|
|
|
|
|
|
|
|
spa_return_val_if_fail(props != NULL, -EIO);
|
|
|
|
|
spa_return_val_if_fail(descriptions != NULL, -EINVAL);
|
|
|
|
|
|
|
|
|
|
dsize = dcnt = 0;
|
|
|
|
|
desc = NULL;
|
|
|
|
|
|
|
|
|
|
spa_dict_for_each(item, &props->dict) {
|
|
|
|
|
jack_property_t* prop;
|
|
|
|
|
|
|
|
|
|
at = strrchr(item->key, '@');
|
|
|
|
|
if (at == NULL)
|
|
|
|
|
continue;
|
|
|
|
|
|
|
|
|
|
len = at - item->key;
|
|
|
|
|
|
|
|
|
|
jack_uuid_parse(item->key, &uuid);
|
|
|
|
|
|
|
|
|
|
at = strrchr(item->value, '@');
|
|
|
|
|
if (at == NULL)
|
|
|
|
|
continue;
|
|
|
|
|
|
|
|
|
|
for (n = 0; n < dcnt; n++) {
|
|
|
|
|
if (jack_uuid_compare(uuid, desc[n].subject) == 0)
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
if (n == dcnt) {
|
|
|
|
|
if (dcnt == dsize) {
|
|
|
|
|
dsize = dsize > 0 ? dsize * 2 : 8;
|
|
|
|
|
desc = realloc (desc, sizeof(jack_description_t) * dsize);
|
|
|
|
|
}
|
|
|
|
|
desc[n].property_size = 0;
|
|
|
|
|
desc[n].property_cnt = 0;
|
|
|
|
|
desc[n].properties = NULL;
|
|
|
|
|
|
|
|
|
|
jack_uuid_copy (&desc[n].subject, uuid);
|
|
|
|
|
dcnt++;
|
|
|
|
|
}
|
|
|
|
|
cdesc = &desc[n];
|
|
|
|
|
|
|
|
|
|
if (cdesc->property_cnt == cdesc->property_size) {
|
|
|
|
|
cdesc->property_size = cdesc->property_size > 0 ? cdesc->property_size * 2 : 8;
|
|
|
|
|
cdesc->properties = realloc (cdesc->properties,
|
|
|
|
|
sizeof(jack_property_t) * cdesc->property_size);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
prop = &cdesc->properties[cdesc->property_cnt++];
|
|
|
|
|
prop->key = strdup(item->key + len + 1);
|
|
|
|
|
prop->data = strndup(item->value, at - item->value);
|
|
|
|
|
prop->type = strdup(at + 1);
|
|
|
|
|
}
|
|
|
|
|
*descriptions = desc;
|
|
|
|
|
return dcnt;
|
2018-10-10 18:50:11 +02:00
|
|
|
}
|
|
|
|
|
|
2019-04-11 16:36:52 +02:00
|
|
|
SPA_EXPORT
|
2018-10-10 18:50:11 +02:00
|
|
|
int jack_remove_property (jack_client_t* client, jack_uuid_t subject, const char* key)
|
|
|
|
|
{
|
2020-02-24 11:25:31 +01:00
|
|
|
int keylen;
|
|
|
|
|
char *dst;
|
2019-08-21 18:37:02 +02:00
|
|
|
struct pw_properties * props = get_properties();
|
|
|
|
|
|
2020-02-24 11:25:31 +01:00
|
|
|
spa_return_val_if_fail(client != NULL, -EINVAL);
|
|
|
|
|
spa_return_val_if_fail(key != NULL, -EINVAL);
|
|
|
|
|
|
|
|
|
|
keylen = strlen(key);
|
|
|
|
|
dst = alloca(JACK_UUID_STRING_SIZE + keylen);
|
2019-08-21 18:37:02 +02:00
|
|
|
make_key(dst, subject, key, keylen);
|
|
|
|
|
|
|
|
|
|
pw_properties_set(props, dst, NULL);
|
|
|
|
|
pw_log_debug("removed %s", dst);
|
|
|
|
|
|
|
|
|
|
return 0;
|
2018-10-10 18:50:11 +02:00
|
|
|
}
|
|
|
|
|
|
2019-04-11 16:36:52 +02:00
|
|
|
SPA_EXPORT
|
2018-10-10 18:50:11 +02:00
|
|
|
int jack_remove_properties (jack_client_t* client, jack_uuid_t subject)
|
|
|
|
|
{
|
|
|
|
|
pw_log_warn("not implemented");
|
|
|
|
|
return -1;
|
|
|
|
|
}
|
|
|
|
|
|
2019-04-11 16:36:52 +02:00
|
|
|
SPA_EXPORT
|
2018-10-10 18:50:11 +02:00
|
|
|
int jack_remove_all_properties (jack_client_t* client)
|
|
|
|
|
{
|
2020-04-20 11:50:37 +02:00
|
|
|
struct client *c = (struct client *) client;
|
|
|
|
|
|
|
|
|
|
spa_return_val_if_fail(c != NULL, -EINVAL);
|
|
|
|
|
|
|
|
|
|
pw_metadata_clear(c->metadata->proxy);
|
|
|
|
|
return 0;
|
2018-10-10 18:50:11 +02:00
|
|
|
}
|
|
|
|
|
|
2019-04-11 16:36:52 +02:00
|
|
|
SPA_EXPORT
|
2018-10-10 18:50:11 +02:00
|
|
|
int jack_set_property_change_callback (jack_client_t* client,
|
|
|
|
|
JackPropertyChangeCallback callback,
|
|
|
|
|
void* arg)
|
|
|
|
|
{
|
2020-04-20 11:50:37 +02:00
|
|
|
struct client *c = (struct client *) client;
|
|
|
|
|
|
|
|
|
|
spa_return_val_if_fail(c != NULL, -EINVAL);
|
|
|
|
|
|
|
|
|
|
c->property_callback = callback;
|
|
|
|
|
c->property_arg = arg;
|
|
|
|
|
return 0;
|
2018-10-10 18:50:11 +02:00
|
|
|
}
|
|
|
|
|
|
2019-04-11 16:36:52 +02:00
|
|
|
SPA_EXPORT
|
2018-10-10 18:50:11 +02:00
|
|
|
const char* JACK_METADATA_PRETTY_NAME = "http://jackaudio.org/metadata/pretty-name";
|
2019-04-11 16:36:52 +02:00
|
|
|
SPA_EXPORT
|
2018-10-10 18:50:11 +02:00
|
|
|
const char* JACK_METADATA_HARDWARE = "http://jackaudio.org/metadata/hardware";
|
2019-04-11 16:36:52 +02:00
|
|
|
SPA_EXPORT
|
2018-10-10 18:50:11 +02:00
|
|
|
const char* JACK_METADATA_CONNECTED = "http://jackaudio.org/metadata/connected";
|
2019-04-11 16:36:52 +02:00
|
|
|
SPA_EXPORT
|
2018-10-10 18:50:11 +02:00
|
|
|
const char* JACK_METADATA_PORT_GROUP = "http://jackaudio.org/metadata/port-group";
|
2019-04-11 16:36:52 +02:00
|
|
|
SPA_EXPORT
|
2018-10-10 18:50:11 +02:00
|
|
|
const char* JACK_METADATA_ICON_SMALL = "http://jackaudio.org/metadata/icon-small";
|
2019-04-11 16:36:52 +02:00
|
|
|
SPA_EXPORT
|
2018-10-10 18:50:11 +02:00
|
|
|
const char* JACK_METADATA_ICON_LARGE = "http://jackaudio.org/metadata/icon-large";
|