2017-05-23 19:15:33 +02:00
|
|
|
/* PipeWire
|
2016-11-22 13:06:22 +01:00
|
|
|
* Copyright (C) 2015 Wim Taymans <wim.taymans@gmail.com>
|
|
|
|
|
*
|
|
|
|
|
* This library is free software; you can redistribute it and/or
|
|
|
|
|
* modify it under the terms of the GNU Library General Public
|
|
|
|
|
* License as published by the Free Software Foundation; either
|
|
|
|
|
* version 2 of the License, or (at your option) any later version.
|
|
|
|
|
*
|
|
|
|
|
* This library is distributed in the hope that it will be useful,
|
|
|
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
|
|
|
|
* Library General Public License for more details.
|
|
|
|
|
*
|
|
|
|
|
* You should have received a copy of the GNU Library General Public
|
|
|
|
|
* License along with this library; if not, write to the
|
|
|
|
|
* Free Software Foundation, Inc., 51 Franklin St, Fifth Floor,
|
|
|
|
|
* Boston, MA 02110-1301, USA.
|
|
|
|
|
*/
|
|
|
|
|
|
|
|
|
|
#include <string.h>
|
|
|
|
|
#include <stdio.h>
|
2016-11-24 17:00:42 +01:00
|
|
|
#include <unistd.h>
|
2016-11-22 13:06:22 +01:00
|
|
|
#include <errno.h>
|
|
|
|
|
#include <sys/socket.h>
|
|
|
|
|
#include <sys/un.h>
|
|
|
|
|
#include <sys/stat.h>
|
|
|
|
|
#include <fcntl.h>
|
|
|
|
|
#include <sys/file.h>
|
|
|
|
|
|
|
|
|
|
#include "config.h"
|
|
|
|
|
|
2017-07-11 15:57:20 +02:00
|
|
|
#include "pipewire/pipewire.h"
|
2017-08-04 10:18:54 +02:00
|
|
|
#include "pipewire/private.h"
|
2017-07-11 15:57:20 +02:00
|
|
|
#include "pipewire/log.h"
|
|
|
|
|
#include "pipewire/interfaces.h"
|
|
|
|
|
#include "pipewire/core.h"
|
|
|
|
|
#include "pipewire/node.h"
|
|
|
|
|
#include "pipewire/module.h"
|
|
|
|
|
#include "pipewire/client.h"
|
|
|
|
|
#include "pipewire/resource.h"
|
|
|
|
|
#include "pipewire/link.h"
|
|
|
|
|
#include "pipewire/node-factory.h"
|
|
|
|
|
#include "pipewire/data-loop.h"
|
|
|
|
|
#include "pipewire/main-loop.h"
|
|
|
|
|
|
2017-07-12 18:04:00 +02:00
|
|
|
#include "extensions/protocol-native.h"
|
2017-07-11 15:57:20 +02:00
|
|
|
#include "modules/module-protocol-native/connection.h"
|
2016-11-22 13:06:22 +01:00
|
|
|
|
|
|
|
|
#ifndef UNIX_PATH_MAX
|
|
|
|
|
#define UNIX_PATH_MAX 108
|
|
|
|
|
#endif
|
|
|
|
|
|
|
|
|
|
#define LOCK_SUFFIX ".lock"
|
|
|
|
|
#define LOCK_SUFFIXLEN 5
|
|
|
|
|
|
2017-07-12 18:04:00 +02:00
|
|
|
void pw_protocol_native_init(struct pw_protocol *protocol);
|
2017-06-21 12:11:54 +02:00
|
|
|
|
2017-07-18 14:58:14 +02:00
|
|
|
struct protocol_data {
|
|
|
|
|
struct pw_module *module;
|
|
|
|
|
};
|
|
|
|
|
|
2017-07-11 12:24:03 +02:00
|
|
|
struct connection {
|
|
|
|
|
struct pw_protocol_connection this;
|
|
|
|
|
|
|
|
|
|
int fd;
|
|
|
|
|
|
|
|
|
|
struct spa_source *source;
|
2017-08-04 16:49:13 +02:00
|
|
|
|
2017-07-12 18:04:00 +02:00
|
|
|
struct pw_protocol_native_connection *connection;
|
2017-08-04 16:49:13 +02:00
|
|
|
struct pw_listener conn_listener;
|
2017-07-11 12:24:03 +02:00
|
|
|
|
|
|
|
|
bool disconnecting;
|
2017-07-19 10:44:03 +02:00
|
|
|
bool flush_signaled;
|
2017-07-11 12:24:03 +02:00
|
|
|
struct spa_source *flush_event;
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
struct listener {
|
|
|
|
|
struct pw_protocol_listener this;
|
|
|
|
|
|
2017-05-26 08:05:01 +02:00
|
|
|
int fd;
|
|
|
|
|
int fd_lock;
|
|
|
|
|
struct sockaddr_un addr;
|
|
|
|
|
char lock_addr[UNIX_PATH_MAX + LOCK_SUFFIXLEN];
|
|
|
|
|
|
|
|
|
|
struct pw_loop *loop;
|
|
|
|
|
struct spa_source *source;
|
2017-07-12 18:04:00 +02:00
|
|
|
struct spa_loop_control_hooks hooks;
|
2017-05-23 19:15:33 +02:00
|
|
|
};
|
2016-11-22 13:06:22 +01:00
|
|
|
|
2017-07-12 18:04:00 +02:00
|
|
|
struct client_data {
|
2017-08-04 10:18:54 +02:00
|
|
|
struct pw_client *client;
|
2017-08-04 16:49:13 +02:00
|
|
|
struct pw_listener client_listener;
|
2017-05-26 08:05:01 +02:00
|
|
|
int fd;
|
|
|
|
|
struct spa_source *source;
|
2017-07-12 18:04:00 +02:00
|
|
|
struct pw_protocol_native_connection *connection;
|
2017-05-23 19:15:33 +02:00
|
|
|
};
|
2016-11-22 13:06:22 +01:00
|
|
|
|
2017-06-09 18:22:57 +02:00
|
|
|
static void
|
2017-07-12 18:04:00 +02:00
|
|
|
process_messages(struct pw_client *client)
|
2017-06-09 18:22:57 +02:00
|
|
|
{
|
2017-07-12 18:04:00 +02:00
|
|
|
struct client_data *data = client->user_data;
|
|
|
|
|
struct pw_protocol_native_connection *conn = data->connection;
|
2017-06-09 18:22:57 +02:00
|
|
|
uint8_t opcode;
|
|
|
|
|
uint32_t id;
|
|
|
|
|
uint32_t size;
|
|
|
|
|
void *message;
|
|
|
|
|
|
2017-07-12 18:04:00 +02:00
|
|
|
while (pw_protocol_native_connection_get_next(conn, &opcode, &id, &message, &size)) {
|
2017-06-09 18:22:57 +02:00
|
|
|
struct pw_resource *resource;
|
2017-08-01 20:11:38 +02:00
|
|
|
const struct pw_protocol_native_demarshal *demarshal;
|
2017-06-09 18:22:57 +02:00
|
|
|
|
2017-07-12 18:04:00 +02:00
|
|
|
pw_log_trace("protocol-native %p: got message %d from %u", client->protocol,
|
2017-06-09 18:22:57 +02:00
|
|
|
opcode, id);
|
|
|
|
|
|
2017-07-12 18:04:00 +02:00
|
|
|
resource = pw_map_lookup(&client->objects, id);
|
2017-06-09 18:22:57 +02:00
|
|
|
if (resource == NULL) {
|
|
|
|
|
pw_log_error("protocol-native %p: unknown resource %u",
|
2017-07-12 18:04:00 +02:00
|
|
|
client->protocol, id);
|
2017-06-09 18:22:57 +02:00
|
|
|
continue;
|
|
|
|
|
}
|
2017-08-01 20:11:38 +02:00
|
|
|
if ((resource->permissions & PW_PERM_X) == 0) {
|
|
|
|
|
pw_log_error("protocol-native %p: execute not allowed on resource %u",
|
|
|
|
|
client->protocol, id);
|
|
|
|
|
continue;
|
2017-06-09 18:22:57 +02:00
|
|
|
}
|
2017-08-01 20:11:38 +02:00
|
|
|
|
|
|
|
|
if (opcode >= resource->marshal->n_methods)
|
|
|
|
|
goto invalid_method;
|
|
|
|
|
|
2017-07-13 15:21:52 +02:00
|
|
|
demarshal = resource->marshal->method_demarshal;
|
2017-08-01 20:11:38 +02:00
|
|
|
if (!demarshal[opcode].func)
|
|
|
|
|
goto invalid_message;
|
|
|
|
|
|
|
|
|
|
if ((demarshal[opcode].flags & PW_PROTOCOL_NATIVE_PERM_W) &&
|
|
|
|
|
((resource->permissions & PW_PERM_X) == 0)) {
|
|
|
|
|
pw_log_error("protocol-native %p: method %u requires write access on %u",
|
|
|
|
|
client->protocol, opcode, id);
|
|
|
|
|
continue;
|
2017-06-09 18:22:57 +02:00
|
|
|
}
|
2017-08-01 20:11:38 +02:00
|
|
|
|
|
|
|
|
if (demarshal[opcode].flags & PW_PROTOCOL_NATIVE_REMAP)
|
|
|
|
|
if (!pw_pod_remap_data(SPA_POD_TYPE_STRUCT, message, size, &client->types))
|
|
|
|
|
goto invalid_message;
|
|
|
|
|
|
|
|
|
|
if (!demarshal[opcode].func (resource, message, size))
|
|
|
|
|
goto invalid_message;
|
|
|
|
|
|
|
|
|
|
/* when the client is busy processing an async action, stop processing messages
|
|
|
|
|
* for the client until it finishes the action */
|
|
|
|
|
if (client->busy)
|
2017-06-09 18:22:57 +02:00
|
|
|
break;
|
|
|
|
|
}
|
2017-08-01 20:11:38 +02:00
|
|
|
return;
|
|
|
|
|
|
|
|
|
|
invalid_method:
|
|
|
|
|
pw_log_error("protocol-native %p: invalid method %u on resource %u",
|
|
|
|
|
client->protocol, opcode, id);
|
|
|
|
|
pw_client_destroy(client);
|
|
|
|
|
return;
|
|
|
|
|
invalid_message:
|
|
|
|
|
pw_log_error("protocol-native %p: invalid message received %u %u",
|
|
|
|
|
client->protocol, id, opcode);
|
|
|
|
|
pw_client_destroy(client);
|
|
|
|
|
return;
|
2017-06-09 18:22:57 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
static void
|
2017-08-04 10:18:54 +02:00
|
|
|
client_busy_changed(void *data, bool busy)
|
2017-06-09 18:22:57 +02:00
|
|
|
{
|
2017-08-04 10:18:54 +02:00
|
|
|
struct pw_client *client = data;
|
|
|
|
|
struct client_data *c = client->user_data;
|
2017-06-09 18:22:57 +02:00
|
|
|
enum spa_io mask = SPA_IO_ERR | SPA_IO_HUP;
|
|
|
|
|
|
2017-08-04 10:18:54 +02:00
|
|
|
if (!busy)
|
2017-06-09 18:22:57 +02:00
|
|
|
mask |= SPA_IO_IN;
|
|
|
|
|
|
2017-08-04 10:18:54 +02:00
|
|
|
pw_loop_update_io(client->core->main_loop, c->source, mask);
|
2017-06-09 18:22:57 +02:00
|
|
|
|
2017-08-04 10:18:54 +02:00
|
|
|
if (busy)
|
2017-07-12 18:04:00 +02:00
|
|
|
process_messages(client);
|
2017-06-09 18:22:57 +02:00
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
2017-06-14 16:10:07 +02:00
|
|
|
static void on_before_hook(const struct spa_loop_control_hooks *hooks)
|
2017-03-14 16:13:29 +01:00
|
|
|
{
|
2017-07-12 18:04:00 +02:00
|
|
|
struct listener *listener = SPA_CONTAINER_OF(hooks, struct listener, hooks);
|
|
|
|
|
struct pw_protocol_listener *this = &listener->this;
|
|
|
|
|
struct pw_client *client, *tmp;
|
|
|
|
|
struct client_data *data;
|
|
|
|
|
|
|
|
|
|
spa_list_for_each_safe(client, tmp, &this->client_list, protocol_link) {
|
|
|
|
|
data = client->user_data;
|
|
|
|
|
pw_protocol_native_connection_flush(data->connection);
|
|
|
|
|
}
|
2017-03-14 16:13:29 +01:00
|
|
|
}
|
|
|
|
|
|
2016-11-22 13:06:22 +01:00
|
|
|
static void
|
2017-05-26 08:05:01 +02:00
|
|
|
connection_data(struct spa_loop_utils *utils,
|
|
|
|
|
struct spa_source *source, int fd, enum spa_io mask, void *data)
|
2016-11-22 13:06:22 +01:00
|
|
|
{
|
2017-07-12 18:04:00 +02:00
|
|
|
struct pw_client *client = data;
|
2017-05-26 08:05:01 +02:00
|
|
|
|
|
|
|
|
if (mask & (SPA_IO_ERR | SPA_IO_HUP)) {
|
2017-07-12 18:04:00 +02:00
|
|
|
pw_log_error("protocol-native %p: got connection error", client->protocol);
|
|
|
|
|
pw_client_destroy(client);
|
2017-05-26 08:05:01 +02:00
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
|
2017-06-09 18:22:57 +02:00
|
|
|
if (mask & SPA_IO_IN)
|
|
|
|
|
process_messages(client);
|
2016-11-22 13:06:22 +01:00
|
|
|
}
|
|
|
|
|
|
2017-08-04 10:18:54 +02:00
|
|
|
static void client_free(void *data)
|
2016-11-22 13:06:22 +01:00
|
|
|
{
|
2017-07-12 18:04:00 +02:00
|
|
|
struct pw_client *client = data;
|
|
|
|
|
struct client_data *this = client->user_data;
|
|
|
|
|
|
|
|
|
|
pw_loop_destroy_source(client->protocol->core->main_loop, this->source);
|
|
|
|
|
spa_list_remove(&client->protocol_link);
|
|
|
|
|
|
|
|
|
|
pw_protocol_native_connection_destroy(this->connection);
|
|
|
|
|
close(this->fd);
|
|
|
|
|
}
|
|
|
|
|
|
2017-08-04 16:49:13 +02:00
|
|
|
static const struct pw_client_events client_events = {
|
|
|
|
|
PW_VERSION_CLIENT_EVENTS,
|
2017-08-04 10:18:54 +02:00
|
|
|
.free = client_free,
|
|
|
|
|
.busy_changed = client_busy_changed,
|
|
|
|
|
};
|
|
|
|
|
|
2017-07-12 18:04:00 +02:00
|
|
|
static struct pw_client *client_new(struct listener *l, int fd)
|
|
|
|
|
{
|
|
|
|
|
struct client_data *this;
|
2017-05-26 08:05:01 +02:00
|
|
|
struct pw_client *client;
|
2017-07-12 18:04:00 +02:00
|
|
|
struct pw_protocol *protocol = l->this.protocol;
|
2017-07-18 14:58:14 +02:00
|
|
|
struct protocol_data *pd = protocol->user_data;
|
2017-05-26 08:05:01 +02:00
|
|
|
socklen_t len;
|
|
|
|
|
struct ucred ucred, *ucredp;
|
|
|
|
|
|
2017-06-15 17:54:55 +02:00
|
|
|
len = sizeof(ucred);
|
|
|
|
|
if (getsockopt(fd, SOL_SOCKET, SO_PEERCRED, &ucred, &len) < 0) {
|
|
|
|
|
pw_log_error("no peercred: %m");
|
|
|
|
|
ucredp = NULL;
|
|
|
|
|
} else {
|
|
|
|
|
ucredp = &ucred;
|
|
|
|
|
}
|
|
|
|
|
|
2017-07-18 14:58:14 +02:00
|
|
|
client = pw_client_new(protocol->core, pd->module->global, ucredp, NULL, sizeof(struct client_data));
|
2017-06-15 17:54:55 +02:00
|
|
|
if (client == NULL)
|
|
|
|
|
goto no_client;
|
2017-05-26 08:05:01 +02:00
|
|
|
|
2017-08-04 10:18:54 +02:00
|
|
|
this = pw_client_get_user_data(client);
|
|
|
|
|
this->client = client;
|
2017-05-26 08:05:01 +02:00
|
|
|
this->fd = fd;
|
2017-07-12 18:04:00 +02:00
|
|
|
this->source = pw_loop_add_io(protocol->core->main_loop,
|
2017-05-26 08:05:01 +02:00
|
|
|
this->fd,
|
2017-07-12 18:04:00 +02:00
|
|
|
SPA_IO_ERR | SPA_IO_HUP, false, connection_data, client);
|
2017-05-26 08:05:01 +02:00
|
|
|
if (this->source == NULL)
|
|
|
|
|
goto no_source;
|
|
|
|
|
|
2017-07-12 18:04:00 +02:00
|
|
|
this->connection = pw_protocol_native_connection_new(fd);
|
2017-05-26 08:05:01 +02:00
|
|
|
if (this->connection == NULL)
|
|
|
|
|
goto no_connection;
|
|
|
|
|
|
2017-07-12 18:04:00 +02:00
|
|
|
client->protocol = protocol;
|
|
|
|
|
spa_list_insert(l->this.client_list.prev, &client->protocol_link);
|
2017-05-26 08:05:01 +02:00
|
|
|
|
2017-08-04 16:49:13 +02:00
|
|
|
pw_client_add_listener(client, &this->client_listener, &client_events, client);
|
2017-05-26 08:05:01 +02:00
|
|
|
|
2017-08-01 17:09:57 +02:00
|
|
|
pw_global_bind(protocol->core->global, client, PW_PERM_RWX, PW_VERSION_CORE, 0);
|
2017-06-15 17:54:55 +02:00
|
|
|
|
2017-07-12 18:04:00 +02:00
|
|
|
return client;
|
2017-05-26 08:05:01 +02:00
|
|
|
|
|
|
|
|
no_connection:
|
2017-07-12 18:04:00 +02:00
|
|
|
pw_loop_destroy_source(protocol->core->main_loop, this->source);
|
2017-05-26 08:05:01 +02:00
|
|
|
no_source:
|
2017-07-12 18:04:00 +02:00
|
|
|
pw_client_destroy(client);
|
2017-06-15 17:54:55 +02:00
|
|
|
no_client:
|
2017-05-26 08:05:01 +02:00
|
|
|
return NULL;
|
2016-11-22 13:06:22 +01:00
|
|
|
}
|
|
|
|
|
|
2017-07-11 12:24:03 +02:00
|
|
|
static bool init_socket_name(struct listener *l, const char *name)
|
2016-11-22 13:06:22 +01:00
|
|
|
{
|
2017-05-26 08:05:01 +02:00
|
|
|
int name_size;
|
|
|
|
|
const char *runtime_dir;
|
|
|
|
|
|
|
|
|
|
if ((runtime_dir = getenv("XDG_RUNTIME_DIR")) == NULL) {
|
|
|
|
|
pw_log_error("XDG_RUNTIME_DIR not set in the environment");
|
|
|
|
|
return false;
|
|
|
|
|
}
|
|
|
|
|
|
2017-07-11 12:24:03 +02:00
|
|
|
l->addr.sun_family = AF_LOCAL;
|
|
|
|
|
name_size = snprintf(l->addr.sun_path, sizeof(l->addr.sun_path),
|
2017-05-26 08:05:01 +02:00
|
|
|
"%s/%s", runtime_dir, name) + 1;
|
|
|
|
|
|
2017-07-11 12:24:03 +02:00
|
|
|
if (name_size > (int) sizeof(l->addr.sun_path)) {
|
2017-05-26 08:05:01 +02:00
|
|
|
pw_log_error("socket path \"%s/%s\" plus null terminator exceeds 108 bytes",
|
|
|
|
|
runtime_dir, name);
|
2017-07-11 12:24:03 +02:00
|
|
|
*l->addr.sun_path = 0;
|
2017-05-26 08:05:01 +02:00
|
|
|
return false;
|
|
|
|
|
}
|
|
|
|
|
return true;
|
2016-11-22 13:06:22 +01:00
|
|
|
}
|
|
|
|
|
|
2017-07-11 12:24:03 +02:00
|
|
|
static bool lock_socket(struct listener *l)
|
2016-11-22 13:06:22 +01:00
|
|
|
{
|
2017-05-26 08:05:01 +02:00
|
|
|
struct stat socket_stat;
|
|
|
|
|
|
2017-07-11 12:24:03 +02:00
|
|
|
snprintf(l->lock_addr, sizeof(l->lock_addr), "%s%s", l->addr.sun_path, LOCK_SUFFIX);
|
2017-05-26 08:05:01 +02:00
|
|
|
|
2017-07-11 12:24:03 +02:00
|
|
|
l->fd_lock = open(l->lock_addr, O_CREAT | O_CLOEXEC,
|
2017-05-26 08:05:01 +02:00
|
|
|
(S_IRUSR | S_IWUSR | S_IRGRP | S_IWGRP));
|
|
|
|
|
|
2017-07-11 12:24:03 +02:00
|
|
|
if (l->fd_lock < 0) {
|
|
|
|
|
pw_log_error("unable to open lockfile %s check permissions", l->lock_addr);
|
2017-05-26 08:05:01 +02:00
|
|
|
goto err;
|
|
|
|
|
}
|
|
|
|
|
|
2017-07-11 12:24:03 +02:00
|
|
|
if (flock(l->fd_lock, LOCK_EX | LOCK_NB) < 0) {
|
2017-05-26 08:05:01 +02:00
|
|
|
pw_log_error("unable to lock lockfile %s, maybe another daemon is running",
|
2017-07-11 12:24:03 +02:00
|
|
|
l->lock_addr);
|
2017-05-26 08:05:01 +02:00
|
|
|
goto err_fd;
|
|
|
|
|
}
|
|
|
|
|
|
2017-07-11 12:24:03 +02:00
|
|
|
if (stat(l->addr.sun_path, &socket_stat) < 0) {
|
2017-05-26 08:05:01 +02:00
|
|
|
if (errno != ENOENT) {
|
2017-07-11 12:24:03 +02:00
|
|
|
pw_log_error("did not manage to stat file %s\n", l->addr.sun_path);
|
2017-05-26 08:05:01 +02:00
|
|
|
goto err_fd;
|
|
|
|
|
}
|
|
|
|
|
} else if (socket_stat.st_mode & S_IWUSR || socket_stat.st_mode & S_IWGRP) {
|
2017-07-11 12:24:03 +02:00
|
|
|
unlink(l->addr.sun_path);
|
2017-05-26 08:05:01 +02:00
|
|
|
}
|
|
|
|
|
return true;
|
|
|
|
|
|
|
|
|
|
err_fd:
|
2017-07-11 12:24:03 +02:00
|
|
|
close(l->fd_lock);
|
|
|
|
|
l->fd_lock = -1;
|
2017-05-26 08:05:01 +02:00
|
|
|
err:
|
2017-07-11 12:24:03 +02:00
|
|
|
*l->lock_addr = 0;
|
|
|
|
|
*l->addr.sun_path = 0;
|
2017-05-26 08:05:01 +02:00
|
|
|
return false;
|
2016-11-22 13:06:22 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
static void
|
2017-05-26 08:05:01 +02:00
|
|
|
socket_data(struct spa_loop_utils *utils,
|
|
|
|
|
struct spa_source *source, int fd, enum spa_io mask, void *data)
|
2016-11-22 13:06:22 +01:00
|
|
|
{
|
2017-07-12 18:04:00 +02:00
|
|
|
struct listener *l = data;
|
|
|
|
|
struct pw_client *client;
|
|
|
|
|
struct client_data *c;
|
2017-05-26 08:05:01 +02:00
|
|
|
struct sockaddr_un name;
|
|
|
|
|
socklen_t length;
|
|
|
|
|
int client_fd;
|
|
|
|
|
|
|
|
|
|
length = sizeof(name);
|
|
|
|
|
client_fd = accept4(fd, (struct sockaddr *) &name, &length, SOCK_CLOEXEC);
|
|
|
|
|
if (client_fd < 0) {
|
|
|
|
|
pw_log_error("failed to accept: %m");
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
|
2017-07-12 18:04:00 +02:00
|
|
|
client = client_new(l, client_fd);
|
2017-05-26 08:05:01 +02:00
|
|
|
if (client == NULL) {
|
|
|
|
|
pw_log_error("failed to create client");
|
|
|
|
|
close(client_fd);
|
|
|
|
|
return;
|
|
|
|
|
}
|
2017-07-12 18:04:00 +02:00
|
|
|
c = client->user_data;
|
2017-05-26 08:05:01 +02:00
|
|
|
|
2017-07-12 18:04:00 +02:00
|
|
|
pw_loop_update_io(client->protocol->core->main_loop,
|
|
|
|
|
c->source, SPA_IO_IN | SPA_IO_ERR | SPA_IO_HUP);
|
2016-11-22 13:06:22 +01:00
|
|
|
}
|
|
|
|
|
|
2017-07-12 18:04:00 +02:00
|
|
|
static bool add_socket(struct pw_protocol *protocol, struct listener *l)
|
2016-11-22 13:06:22 +01:00
|
|
|
{
|
2017-05-26 08:05:01 +02:00
|
|
|
socklen_t size;
|
|
|
|
|
|
2017-07-11 12:24:03 +02:00
|
|
|
if ((l->fd = socket(PF_LOCAL, SOCK_STREAM | SOCK_CLOEXEC | SOCK_NONBLOCK, 0)) < 0)
|
2017-05-26 08:05:01 +02:00
|
|
|
return false;
|
|
|
|
|
|
2017-07-11 12:24:03 +02:00
|
|
|
size = offsetof(struct sockaddr_un, sun_path) + strlen(l->addr.sun_path);
|
|
|
|
|
if (bind(l->fd, (struct sockaddr *) &l->addr, size) < 0) {
|
2017-05-26 08:05:01 +02:00
|
|
|
pw_log_error("bind() failed with error: %m");
|
|
|
|
|
return false;
|
|
|
|
|
}
|
|
|
|
|
|
2017-07-11 12:24:03 +02:00
|
|
|
if (listen(l->fd, 128) < 0) {
|
2017-05-26 08:05:01 +02:00
|
|
|
pw_log_error("listen() failed with error: %m");
|
|
|
|
|
return false;
|
|
|
|
|
}
|
|
|
|
|
|
2017-07-12 18:04:00 +02:00
|
|
|
l->loop = protocol->core->main_loop;
|
|
|
|
|
l->source = pw_loop_add_io(l->loop, l->fd, SPA_IO_IN, false, socket_data, l);
|
2017-07-11 12:24:03 +02:00
|
|
|
if (l->source == NULL)
|
2017-05-26 08:05:01 +02:00
|
|
|
return false;
|
|
|
|
|
|
|
|
|
|
return true;
|
2016-11-22 13:06:22 +01:00
|
|
|
}
|
|
|
|
|
|
2017-07-11 12:24:03 +02:00
|
|
|
static const char *
|
|
|
|
|
get_name(struct pw_properties *properties)
|
2016-11-22 13:06:22 +01:00
|
|
|
{
|
2017-07-11 12:24:03 +02:00
|
|
|
const char *name = NULL;
|
2016-11-22 13:06:22 +01:00
|
|
|
|
2017-07-11 12:24:03 +02:00
|
|
|
if (properties)
|
|
|
|
|
name = pw_properties_get(properties, "pipewire.core.name");
|
2017-05-26 08:05:01 +02:00
|
|
|
if (name == NULL)
|
|
|
|
|
name = getenv("PIPEWIRE_CORE");
|
|
|
|
|
if (name == NULL)
|
|
|
|
|
name = "pipewire-0";
|
2017-07-11 12:24:03 +02:00
|
|
|
return name;
|
|
|
|
|
}
|
2016-11-22 13:06:22 +01:00
|
|
|
|
2017-07-11 12:24:03 +02:00
|
|
|
static int impl_connect(struct pw_protocol_connection *conn)
|
|
|
|
|
{
|
|
|
|
|
struct sockaddr_un addr;
|
|
|
|
|
socklen_t size;
|
|
|
|
|
const char *runtime_dir, *name = NULL;
|
|
|
|
|
int name_size, fd;
|
|
|
|
|
|
|
|
|
|
if ((runtime_dir = getenv("XDG_RUNTIME_DIR")) == NULL) {
|
|
|
|
|
pw_log_error("connect failed: XDG_RUNTIME_DIR not set in the environment");
|
|
|
|
|
return -1;
|
|
|
|
|
}
|
|
|
|
|
|
2017-07-11 20:38:48 +02:00
|
|
|
name = get_name(NULL);
|
2017-07-11 12:24:03 +02:00
|
|
|
|
|
|
|
|
if ((fd = socket(PF_LOCAL, SOCK_STREAM | SOCK_CLOEXEC | SOCK_NONBLOCK, 0)) < 0)
|
|
|
|
|
return -1;
|
|
|
|
|
|
|
|
|
|
memset(&addr, 0, sizeof(addr));
|
|
|
|
|
addr.sun_family = AF_LOCAL;
|
|
|
|
|
name_size = snprintf(addr.sun_path, sizeof(addr.sun_path), "%s/%s", runtime_dir, name) + 1;
|
|
|
|
|
|
|
|
|
|
if (name_size > (int) sizeof addr.sun_path) {
|
|
|
|
|
pw_log_error("socket path \"%s/%s\" plus null terminator exceeds 108 bytes",
|
|
|
|
|
runtime_dir, name);
|
|
|
|
|
goto error_close;
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
size = offsetof(struct sockaddr_un, sun_path) + name_size;
|
|
|
|
|
|
|
|
|
|
if (connect(fd, (struct sockaddr *) &addr, size) < 0) {
|
|
|
|
|
goto error_close;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return conn->connect_fd(conn, fd);
|
|
|
|
|
|
|
|
|
|
error_close:
|
|
|
|
|
close(fd);
|
|
|
|
|
return -1;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
static void
|
|
|
|
|
on_remote_data(struct spa_loop_utils *utils,
|
2017-07-12 18:04:00 +02:00
|
|
|
struct spa_source *source, int fd, enum spa_io mask, void *data)
|
2017-07-11 12:24:03 +02:00
|
|
|
{
|
|
|
|
|
struct connection *impl = data;
|
|
|
|
|
struct pw_remote *this = impl->this.remote;
|
2017-07-12 18:04:00 +02:00
|
|
|
struct pw_protocol_native_connection *conn = impl->connection;
|
2017-07-11 12:24:03 +02:00
|
|
|
|
|
|
|
|
if (mask & (SPA_IO_ERR | SPA_IO_HUP)) {
|
2017-07-11 18:41:22 +02:00
|
|
|
pw_log_error("protocol-native %p: got connection error", impl);
|
2017-07-18 15:28:14 +02:00
|
|
|
pw_loop_destroy_source(this->core->main_loop, impl->source);
|
|
|
|
|
impl->source = NULL;
|
|
|
|
|
pw_remote_update_state(this, PW_REMOTE_STATE_ERROR, "connection error");
|
|
|
|
|
return;
|
2017-07-11 12:24:03 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (mask & SPA_IO_IN) {
|
|
|
|
|
uint8_t opcode;
|
|
|
|
|
uint32_t id;
|
|
|
|
|
uint32_t size;
|
|
|
|
|
void *message;
|
|
|
|
|
|
|
|
|
|
while (!impl->disconnecting
|
2017-07-12 18:04:00 +02:00
|
|
|
&& pw_protocol_native_connection_get_next(conn, &opcode, &id, &message, &size)) {
|
2017-07-11 12:24:03 +02:00
|
|
|
struct pw_proxy *proxy;
|
2017-08-01 20:11:38 +02:00
|
|
|
const struct pw_protocol_native_demarshal *demarshal;
|
2017-07-11 12:24:03 +02:00
|
|
|
|
|
|
|
|
pw_log_trace("protocol-native %p: got message %d from %u", this, opcode, id);
|
|
|
|
|
|
|
|
|
|
proxy = pw_map_lookup(&this->objects, id);
|
2017-08-04 10:18:54 +02:00
|
|
|
if (proxy == NULL || proxy->marshal == NULL) {
|
2017-07-11 12:24:03 +02:00
|
|
|
pw_log_error("protocol-native %p: could not find proxy %u", this, id);
|
|
|
|
|
continue;
|
|
|
|
|
}
|
2017-07-13 15:21:52 +02:00
|
|
|
if (opcode >= proxy->marshal->n_events) {
|
2017-07-11 12:24:03 +02:00
|
|
|
pw_log_error("protocol-native %p: invalid method %u for %u", this, opcode,
|
|
|
|
|
id);
|
|
|
|
|
continue;
|
|
|
|
|
}
|
|
|
|
|
|
2017-07-13 15:21:52 +02:00
|
|
|
demarshal = proxy->marshal->event_demarshal;
|
2017-08-01 20:11:38 +02:00
|
|
|
if (!demarshal[opcode].func) {
|
2017-07-11 12:24:03 +02:00
|
|
|
pw_log_error("protocol-native %p: function %d not implemented on %u", this,
|
|
|
|
|
opcode, id);
|
2017-08-01 20:11:38 +02:00
|
|
|
continue;
|
|
|
|
|
}
|
2017-07-11 12:24:03 +02:00
|
|
|
|
2017-08-01 20:11:38 +02:00
|
|
|
if (demarshal[opcode].flags & PW_PROTOCOL_NATIVE_REMAP) {
|
|
|
|
|
if (!pw_pod_remap_data(SPA_POD_TYPE_STRUCT, message, size, &this->types)) {
|
|
|
|
|
pw_log_error
|
|
|
|
|
("protocol-native %p: invalid message received %u for %u", this,
|
|
|
|
|
opcode, id);
|
|
|
|
|
continue;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
if (!demarshal[opcode].func(proxy, message, size)) {
|
|
|
|
|
pw_log_error ("protocol-native %p: invalid message received %u for %u", this,
|
|
|
|
|
opcode, id);
|
|
|
|
|
continue;
|
|
|
|
|
}
|
2017-07-11 12:24:03 +02:00
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
2017-07-19 10:44:03 +02:00
|
|
|
static void do_flush_event(struct spa_loop_utils *utils, struct spa_source *source,
|
|
|
|
|
uint64_t count, void *data)
|
2017-07-11 12:24:03 +02:00
|
|
|
{
|
|
|
|
|
struct connection *impl = data;
|
2017-07-19 10:44:03 +02:00
|
|
|
impl->flush_signaled = false;
|
2017-07-11 12:24:03 +02:00
|
|
|
if (impl->connection)
|
2017-07-12 18:04:00 +02:00
|
|
|
if (!pw_protocol_native_connection_flush(impl->connection))
|
2017-07-11 12:24:03 +02:00
|
|
|
impl->this.disconnect(&impl->this);
|
|
|
|
|
}
|
|
|
|
|
|
2017-08-04 16:49:13 +02:00
|
|
|
static void on_need_flush(void *data)
|
2017-07-11 12:24:03 +02:00
|
|
|
{
|
2017-08-04 16:49:13 +02:00
|
|
|
struct connection *impl = data;
|
2017-07-11 12:24:03 +02:00
|
|
|
struct pw_remote *remote = impl->this.remote;
|
2017-07-19 10:44:03 +02:00
|
|
|
|
|
|
|
|
if (!impl->flush_signaled) {
|
|
|
|
|
impl->flush_signaled = true;
|
|
|
|
|
pw_loop_signal_event(remote->core->main_loop, impl->flush_event);
|
|
|
|
|
}
|
2017-07-11 12:24:03 +02:00
|
|
|
}
|
|
|
|
|
|
2017-08-04 16:49:13 +02:00
|
|
|
static const struct pw_protocol_native_connection_events conn_events = {
|
|
|
|
|
PW_VERSION_PROTOCOL_NATIVE_CONNECTION_EVENTS,
|
|
|
|
|
.need_flush = on_need_flush,
|
|
|
|
|
};
|
|
|
|
|
|
2017-07-11 12:24:03 +02:00
|
|
|
static int impl_connect_fd(struct pw_protocol_connection *conn, int fd)
|
|
|
|
|
{
|
|
|
|
|
struct connection *impl = SPA_CONTAINER_OF(conn, struct connection, this);
|
2017-07-12 18:04:00 +02:00
|
|
|
struct pw_remote *remote = conn->remote;
|
2017-07-11 12:24:03 +02:00
|
|
|
|
2017-07-12 18:04:00 +02:00
|
|
|
impl->connection = pw_protocol_native_connection_new(fd);
|
2017-07-11 12:24:03 +02:00
|
|
|
if (impl->connection == NULL)
|
|
|
|
|
goto error_close;
|
|
|
|
|
|
2017-08-04 16:49:13 +02:00
|
|
|
pw_protocol_native_connection_add_listener(impl->connection,
|
|
|
|
|
&impl->conn_listener,
|
|
|
|
|
&conn_events,
|
|
|
|
|
impl);
|
2017-07-11 12:24:03 +02:00
|
|
|
|
|
|
|
|
impl->fd = fd;
|
|
|
|
|
impl->source = pw_loop_add_io(remote->core->main_loop,
|
|
|
|
|
fd,
|
|
|
|
|
SPA_IO_IN | SPA_IO_HUP | SPA_IO_ERR,
|
|
|
|
|
false, on_remote_data, impl);
|
2016-11-22 13:06:22 +01:00
|
|
|
|
2017-07-11 12:24:03 +02:00
|
|
|
return 0;
|
|
|
|
|
|
|
|
|
|
error_close:
|
|
|
|
|
close(fd);
|
|
|
|
|
return -1;
|
|
|
|
|
}
|
|
|
|
|
|
2017-07-12 18:04:00 +02:00
|
|
|
static void impl_disconnect(struct pw_protocol_connection *conn)
|
2017-07-11 12:24:03 +02:00
|
|
|
{
|
|
|
|
|
struct connection *impl = SPA_CONTAINER_OF(conn, struct connection, this);
|
2017-07-12 18:04:00 +02:00
|
|
|
struct pw_remote *remote = conn->remote;
|
2017-07-11 12:24:03 +02:00
|
|
|
|
|
|
|
|
impl->disconnecting = true;
|
|
|
|
|
|
|
|
|
|
if (impl->source)
|
|
|
|
|
pw_loop_destroy_source(remote->core->main_loop, impl->source);
|
|
|
|
|
impl->source = NULL;
|
|
|
|
|
|
|
|
|
|
if (impl->connection)
|
2017-07-12 18:04:00 +02:00
|
|
|
pw_protocol_native_connection_destroy(impl->connection);
|
2017-07-11 12:24:03 +02:00
|
|
|
impl->connection = NULL;
|
|
|
|
|
|
|
|
|
|
if (impl->fd != -1)
|
|
|
|
|
close(impl->fd);
|
|
|
|
|
impl->fd = -1;
|
|
|
|
|
}
|
|
|
|
|
|
2017-07-12 18:04:00 +02:00
|
|
|
static void impl_destroy(struct pw_protocol_connection *conn)
|
2017-07-11 12:24:03 +02:00
|
|
|
{
|
|
|
|
|
struct connection *impl = SPA_CONTAINER_OF(conn, struct connection, this);
|
|
|
|
|
struct pw_remote *remote = conn->remote;
|
|
|
|
|
|
|
|
|
|
pw_loop_destroy_source(remote->core->main_loop, impl->flush_event);
|
|
|
|
|
|
|
|
|
|
spa_list_remove(&conn->link);
|
|
|
|
|
free(impl);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
static struct pw_protocol_connection *
|
|
|
|
|
impl_new_connection(struct pw_protocol *protocol,
|
|
|
|
|
struct pw_remote *remote,
|
|
|
|
|
struct pw_properties *properties)
|
|
|
|
|
{
|
2017-07-12 18:04:00 +02:00
|
|
|
struct connection *impl;
|
2017-07-11 12:24:03 +02:00
|
|
|
struct pw_protocol_connection *this;
|
|
|
|
|
|
2017-07-12 18:04:00 +02:00
|
|
|
if ((impl = calloc(1, sizeof(struct connection))) == NULL)
|
2017-07-11 12:24:03 +02:00
|
|
|
return NULL;
|
|
|
|
|
|
2017-07-12 18:04:00 +02:00
|
|
|
this = &impl->this;
|
|
|
|
|
this->protocol = protocol;
|
2017-07-11 12:24:03 +02:00
|
|
|
this->remote = remote;
|
|
|
|
|
|
|
|
|
|
this->connect = impl_connect;
|
|
|
|
|
this->connect_fd = impl_connect_fd;
|
|
|
|
|
this->disconnect = impl_disconnect;
|
|
|
|
|
this->destroy = impl_destroy;
|
|
|
|
|
|
2017-07-12 18:04:00 +02:00
|
|
|
impl->flush_event = pw_loop_add_event(remote->core->main_loop, do_flush_event, impl);
|
2017-07-11 12:24:03 +02:00
|
|
|
|
2017-07-12 18:04:00 +02:00
|
|
|
spa_list_insert(protocol->connection_list.prev, &this->link);
|
2017-07-11 12:24:03 +02:00
|
|
|
|
|
|
|
|
return this;
|
|
|
|
|
}
|
|
|
|
|
|
2017-07-12 18:04:00 +02:00
|
|
|
static void destroy_listener(struct pw_protocol_listener *listener)
|
|
|
|
|
{
|
|
|
|
|
struct listener *l = SPA_CONTAINER_OF(listener, struct listener, this);
|
|
|
|
|
|
|
|
|
|
if (l->source)
|
|
|
|
|
pw_loop_destroy_source(l->loop, l->source);
|
|
|
|
|
if (l->addr.sun_path[0])
|
|
|
|
|
unlink(l->addr.sun_path);
|
|
|
|
|
if (l->fd >= 0)
|
|
|
|
|
close(l->fd);
|
|
|
|
|
if (l->lock_addr[0])
|
|
|
|
|
unlink(l->lock_addr);
|
|
|
|
|
if (l->fd_lock >= 0)
|
|
|
|
|
close(l->fd_lock);
|
|
|
|
|
free(l);
|
|
|
|
|
}
|
|
|
|
|
|
2017-07-11 12:24:03 +02:00
|
|
|
static struct pw_protocol_listener *
|
|
|
|
|
impl_add_listener(struct pw_protocol *protocol,
|
|
|
|
|
struct pw_core *core,
|
|
|
|
|
struct pw_properties *properties)
|
|
|
|
|
{
|
2017-07-12 18:04:00 +02:00
|
|
|
struct pw_protocol_listener *this;
|
2017-07-11 12:24:03 +02:00
|
|
|
struct listener *l;
|
|
|
|
|
const char *name;
|
|
|
|
|
|
|
|
|
|
if ((l = calloc(1, sizeof(struct listener))) == NULL)
|
|
|
|
|
return NULL;
|
|
|
|
|
|
|
|
|
|
l->fd = -1;
|
|
|
|
|
l->fd_lock = -1;
|
|
|
|
|
|
2017-07-12 18:04:00 +02:00
|
|
|
this = &l->this;
|
|
|
|
|
this->protocol = protocol;
|
|
|
|
|
spa_list_init(&this->client_list);
|
|
|
|
|
this->destroy = destroy_listener;
|
|
|
|
|
|
2017-07-11 20:38:48 +02:00
|
|
|
name = get_name(core->properties);
|
2017-07-11 12:24:03 +02:00
|
|
|
|
|
|
|
|
if (!init_socket_name(l, name))
|
2017-05-26 08:05:01 +02:00
|
|
|
goto error;
|
2016-11-22 13:06:22 +01:00
|
|
|
|
2017-07-11 12:24:03 +02:00
|
|
|
if (!lock_socket(l))
|
2017-05-26 08:05:01 +02:00
|
|
|
goto error;
|
2016-11-22 13:06:22 +01:00
|
|
|
|
2017-07-12 18:04:00 +02:00
|
|
|
if (!add_socket(protocol, l))
|
2017-05-26 08:05:01 +02:00
|
|
|
goto error;
|
2016-11-22 13:06:22 +01:00
|
|
|
|
2017-07-12 18:04:00 +02:00
|
|
|
spa_list_insert(protocol->listener_list.prev, &this->link);
|
2017-07-11 12:24:03 +02:00
|
|
|
|
2017-07-12 18:04:00 +02:00
|
|
|
l->hooks.before = on_before_hook;
|
|
|
|
|
pw_loop_add_hooks(protocol->core->main_loop, &l->hooks);
|
2017-03-14 16:13:29 +01:00
|
|
|
|
2017-07-12 18:04:00 +02:00
|
|
|
pw_log_info("protocol-native %p: Added listener %p", protocol, this);
|
2017-07-11 12:24:03 +02:00
|
|
|
|
2017-07-12 18:04:00 +02:00
|
|
|
return this;
|
2016-11-22 13:06:22 +01:00
|
|
|
|
2017-05-26 08:05:01 +02:00
|
|
|
error:
|
2017-07-12 18:04:00 +02:00
|
|
|
destroy_listener(this);
|
2017-05-26 08:05:01 +02:00
|
|
|
return NULL;
|
2016-11-22 13:06:22 +01:00
|
|
|
}
|
|
|
|
|
|
2017-07-12 18:04:00 +02:00
|
|
|
const static struct pw_protocol_implementaton protocol_impl = {
|
|
|
|
|
PW_VERSION_PROTOCOL_IMPLEMENTATION,
|
|
|
|
|
impl_new_connection,
|
|
|
|
|
impl_add_listener,
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
static struct spa_pod_builder *
|
|
|
|
|
impl_ext_begin_proxy(struct pw_proxy *proxy, uint8_t opcode)
|
|
|
|
|
{
|
|
|
|
|
struct connection *impl = SPA_CONTAINER_OF(proxy->remote->conn, struct connection, this);
|
|
|
|
|
return pw_protocol_native_connection_begin_proxy(impl->connection, proxy, opcode);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
static uint32_t impl_ext_add_proxy_fd(struct pw_proxy *proxy, int fd)
|
|
|
|
|
{
|
|
|
|
|
struct connection *impl = SPA_CONTAINER_OF(proxy->remote->conn, struct connection, this);
|
|
|
|
|
return pw_protocol_native_connection_add_fd(impl->connection, fd);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
static int impl_ext_get_proxy_fd(struct pw_proxy *proxy, uint32_t index)
|
|
|
|
|
{
|
|
|
|
|
struct connection *impl = SPA_CONTAINER_OF(proxy->remote->conn, struct connection, this);
|
|
|
|
|
return pw_protocol_native_connection_get_fd(impl->connection, index);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
static void impl_ext_end_proxy(struct pw_proxy *proxy,
|
|
|
|
|
struct spa_pod_builder *builder)
|
|
|
|
|
{
|
|
|
|
|
struct connection *impl = SPA_CONTAINER_OF(proxy->remote->conn, struct connection, this);
|
|
|
|
|
pw_protocol_native_connection_end(impl->connection, builder);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
static struct spa_pod_builder *
|
|
|
|
|
impl_ext_begin_resource(struct pw_resource *resource, uint8_t opcode)
|
|
|
|
|
{
|
|
|
|
|
struct client_data *data = resource->client->user_data;
|
|
|
|
|
return pw_protocol_native_connection_begin_resource(data->connection, resource, opcode);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
static uint32_t impl_ext_add_resource_fd(struct pw_resource *resource, int fd)
|
|
|
|
|
{
|
|
|
|
|
struct client_data *data = resource->client->user_data;
|
|
|
|
|
return pw_protocol_native_connection_add_fd(data->connection, fd);
|
|
|
|
|
}
|
|
|
|
|
static int impl_ext_get_resource_fd(struct pw_resource *resource, uint32_t index)
|
|
|
|
|
{
|
|
|
|
|
struct client_data *data = resource->client->user_data;
|
|
|
|
|
return pw_protocol_native_connection_get_fd(data->connection, index);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
static void impl_ext_end_resource(struct pw_resource *resource,
|
|
|
|
|
struct spa_pod_builder *builder)
|
|
|
|
|
{
|
|
|
|
|
struct client_data *data = resource->client->user_data;
|
|
|
|
|
pw_protocol_native_connection_end(data->connection, builder);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
const static struct pw_protocol_native_ext protocol_ext_impl = {
|
|
|
|
|
PW_VERSION_PROTOCOL_NATIVE_EXT,
|
|
|
|
|
impl_ext_begin_proxy,
|
|
|
|
|
impl_ext_add_proxy_fd,
|
|
|
|
|
impl_ext_get_proxy_fd,
|
|
|
|
|
impl_ext_end_proxy,
|
|
|
|
|
impl_ext_begin_resource,
|
|
|
|
|
impl_ext_add_resource_fd,
|
|
|
|
|
impl_ext_get_resource_fd,
|
|
|
|
|
impl_ext_end_resource,
|
|
|
|
|
};
|
2017-07-11 12:24:03 +02:00
|
|
|
|
2017-07-18 14:58:14 +02:00
|
|
|
static bool module_init(struct pw_module *module, struct pw_properties *properties)
|
2017-07-11 12:24:03 +02:00
|
|
|
{
|
2017-07-18 14:58:14 +02:00
|
|
|
struct pw_core *core = module->core;
|
2017-07-12 18:04:00 +02:00
|
|
|
struct pw_protocol *this;
|
2017-07-11 20:38:48 +02:00
|
|
|
const char *val;
|
2017-07-18 14:58:14 +02:00
|
|
|
struct protocol_data *d;
|
2017-07-11 12:24:03 +02:00
|
|
|
|
2017-07-19 12:44:10 +02:00
|
|
|
if (pw_core_find_protocol(core, PW_TYPE_PROTOCOL__Native) != NULL)
|
|
|
|
|
return true;
|
|
|
|
|
|
2017-07-12 18:04:00 +02:00
|
|
|
this = pw_protocol_new(core, PW_TYPE_PROTOCOL__Native, sizeof(struct protocol_data));
|
|
|
|
|
if (this == NULL)
|
2017-07-18 14:58:14 +02:00
|
|
|
return false;
|
2017-07-12 18:04:00 +02:00
|
|
|
|
|
|
|
|
this->implementation = &protocol_impl;
|
|
|
|
|
this->extension = &protocol_ext_impl;
|
2017-07-11 12:24:03 +02:00
|
|
|
|
2017-07-12 18:04:00 +02:00
|
|
|
pw_protocol_native_init(this);
|
2017-07-11 12:24:03 +02:00
|
|
|
|
2017-07-12 18:04:00 +02:00
|
|
|
pw_log_debug("protocol-native %p: new", this);
|
2017-07-11 12:24:03 +02:00
|
|
|
|
2017-08-04 10:18:54 +02:00
|
|
|
d = pw_protocol_get_user_data(this);
|
2017-07-18 14:58:14 +02:00
|
|
|
d->module = module;
|
|
|
|
|
|
2017-07-11 20:38:48 +02:00
|
|
|
if ((val = pw_properties_get(core->properties, "pipewire.daemon"))) {
|
|
|
|
|
if (atoi(val) == 1)
|
2017-07-12 18:04:00 +02:00
|
|
|
impl_add_listener(this, core, properties);
|
2017-07-11 20:38:48 +02:00
|
|
|
}
|
2017-07-18 14:58:14 +02:00
|
|
|
return true;
|
2017-07-11 12:24:03 +02:00
|
|
|
}
|
|
|
|
|
|
2016-11-22 13:06:22 +01:00
|
|
|
#if 0
|
2017-05-26 08:05:01 +02:00
|
|
|
static void pw_protocol_native_destroy(struct impl *impl)
|
2016-11-22 13:06:22 +01:00
|
|
|
{
|
2017-05-26 08:05:01 +02:00
|
|
|
struct impl *object, *tmp;
|
2016-11-22 13:06:22 +01:00
|
|
|
|
2017-05-26 08:05:01 +02:00
|
|
|
pw_log_debug("protocol-native %p: destroy", impl);
|
2016-11-22 13:06:22 +01:00
|
|
|
|
2017-05-26 08:05:01 +02:00
|
|
|
spa_list_for_each_safe(object, tmp, &impl->object_list, link)
|
|
|
|
|
object_destroy(object);
|
2016-11-22 13:06:22 +01:00
|
|
|
|
2017-05-26 08:05:01 +02:00
|
|
|
free(impl);
|
2016-11-22 13:06:22 +01:00
|
|
|
}
|
|
|
|
|
#endif
|
|
|
|
|
|
2017-05-26 08:05:01 +02:00
|
|
|
bool pipewire__module_init(struct pw_module *module, const char *args)
|
2016-11-22 13:06:22 +01:00
|
|
|
{
|
2017-07-18 14:58:14 +02:00
|
|
|
return module_init(module, NULL);
|
2016-11-22 13:06:22 +01:00
|
|
|
}
|