mirror of
				https://gitlab.freedesktop.org/pipewire/pipewire.git
				synced 2025-11-03 09:01:54 -05:00 
			
		
		
		
	Send create_object error messages to the new resource id so that the client can better know what proxy is failing. Use resource_remove when create_object fails because the core will already send a remove_id on failure. Try to send other errors to the proxy that made the request if possible.
		
			
				
	
	
		
			1190 lines
		
	
	
	
		
			30 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			1190 lines
		
	
	
	
		
			30 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
/* PipeWire
 | 
						|
 *
 | 
						|
 * Copyright © 2018 Wim Taymans
 | 
						|
 *
 | 
						|
 * 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 <string.h>
 | 
						|
#include <stdio.h>
 | 
						|
#include <unistd.h>
 | 
						|
#include <errno.h>
 | 
						|
#include <sys/socket.h>
 | 
						|
#include <sys/un.h>
 | 
						|
#include <sys/stat.h>
 | 
						|
#include <fcntl.h>
 | 
						|
#include <sys/file.h>
 | 
						|
 | 
						|
#include <spa/pod/iter.h>
 | 
						|
#include <spa/utils/result.h>
 | 
						|
#include <spa/debug/pod.h>
 | 
						|
#include <spa/debug/types.h>
 | 
						|
 | 
						|
#include "config.h"
 | 
						|
 | 
						|
#ifdef HAVE_SYSTEMD_DAEMON
 | 
						|
#include <systemd/sd-daemon.h>
 | 
						|
#endif
 | 
						|
 | 
						|
#include <pipewire/impl.h>
 | 
						|
#include <extensions/protocol-native.h>
 | 
						|
 | 
						|
#include "pipewire/private.h"
 | 
						|
 | 
						|
#include "modules/module-protocol-native/connection.h"
 | 
						|
#include "modules/module-protocol-native/defs.h"
 | 
						|
 | 
						|
#define NAME "protocol-native"
 | 
						|
 | 
						|
#ifndef UNIX_PATH_MAX
 | 
						|
#define UNIX_PATH_MAX   108
 | 
						|
#endif
 | 
						|
 | 
						|
static const struct spa_dict_item module_props[] = {
 | 
						|
	{ PW_KEY_MODULE_AUTHOR, "Wim Taymans <wim.taymans@gmail.com>" },
 | 
						|
	{ PW_KEY_MODULE_DESCRIPTION, "Native protocol using unix sockets" },
 | 
						|
	{ PW_KEY_MODULE_VERSION, PACKAGE_VERSION },
 | 
						|
};
 | 
						|
 | 
						|
static bool debug_messages = 0;
 | 
						|
 | 
						|
#define LOCK_SUFFIX     ".lock"
 | 
						|
#define LOCK_SUFFIXLEN  5
 | 
						|
 | 
						|
void pw_protocol_native_init(struct pw_protocol *protocol);
 | 
						|
void pw_protocol_native0_init(struct pw_protocol *protocol);
 | 
						|
 | 
						|
struct protocol_data {
 | 
						|
	struct pw_impl_module *module;
 | 
						|
	struct spa_hook module_listener;
 | 
						|
	struct pw_protocol *protocol;
 | 
						|
 | 
						|
	struct server *local;
 | 
						|
};
 | 
						|
 | 
						|
struct client {
 | 
						|
	struct pw_protocol_client this;
 | 
						|
	struct pw_context *context;
 | 
						|
 | 
						|
	struct spa_source *source;
 | 
						|
 | 
						|
	struct pw_protocol_native_connection *connection;
 | 
						|
	struct spa_hook conn_listener;
 | 
						|
 | 
						|
	unsigned int disconnecting:1;
 | 
						|
	unsigned int flushing:1;
 | 
						|
	unsigned int paused:1;
 | 
						|
};
 | 
						|
 | 
						|
struct server {
 | 
						|
	struct pw_protocol_server this;
 | 
						|
 | 
						|
	int fd_lock;
 | 
						|
	struct sockaddr_un addr;
 | 
						|
	char lock_addr[UNIX_PATH_MAX + LOCK_SUFFIXLEN];
 | 
						|
 | 
						|
	struct pw_loop *loop;
 | 
						|
	struct spa_source *source;
 | 
						|
	struct spa_hook hook;
 | 
						|
	unsigned int activated:1;
 | 
						|
};
 | 
						|
 | 
						|
struct client_data {
 | 
						|
	struct pw_impl_client *client;
 | 
						|
	struct spa_hook client_listener;
 | 
						|
 | 
						|
	struct spa_list protocol_link;
 | 
						|
 | 
						|
	struct spa_source *source;
 | 
						|
	struct pw_protocol_native_connection *connection;
 | 
						|
	struct spa_hook conn_listener;
 | 
						|
 | 
						|
	unsigned int busy:1;
 | 
						|
	unsigned int need_flush:1;
 | 
						|
 | 
						|
	struct protocol_compat_v2 compat_v2;
 | 
						|
};
 | 
						|
 | 
						|
static void debug_msg(const char *prefix, const struct pw_protocol_native_message *msg, bool hex)
 | 
						|
{
 | 
						|
	struct spa_pod *pod;
 | 
						|
	fprintf(stderr, "%s: id:%d op:%d size:%d seq:%d\n", prefix,
 | 
						|
			msg->id, msg->opcode, msg->size, msg->seq);
 | 
						|
 | 
						|
	if ((pod = spa_pod_from_data(msg->data, msg->size, 0, msg->size)) != NULL)
 | 
						|
		spa_debug_pod(0, NULL, pod);
 | 
						|
	else
 | 
						|
		hex = true;
 | 
						|
	if (hex)
 | 
						|
		spa_debug_mem(0, msg->data, msg->size);
 | 
						|
}
 | 
						|
 | 
						|
static int
 | 
						|
process_messages(struct client_data *data)
 | 
						|
{
 | 
						|
	struct pw_protocol_native_connection *conn = data->connection;
 | 
						|
	struct pw_impl_client *client = data->client;
 | 
						|
	struct pw_context *context = client->context;
 | 
						|
	const struct pw_protocol_native_message *msg;
 | 
						|
	struct pw_resource *resource;
 | 
						|
	int res;
 | 
						|
 | 
						|
	context->current_client = client;
 | 
						|
 | 
						|
	/* when the client is busy processing an async action, stop processing messages
 | 
						|
	 * for the client until it finishes the action */
 | 
						|
	while (!data->busy) {
 | 
						|
		const struct pw_protocol_native_demarshal *demarshal;
 | 
						|
	        const struct pw_protocol_marshal *marshal;
 | 
						|
		uint32_t permissions, required;
 | 
						|
 | 
						|
		res = pw_protocol_native_connection_get_next(conn, &msg);
 | 
						|
		if (res < 0) {
 | 
						|
			if (res == -EAGAIN)
 | 
						|
				break;
 | 
						|
			goto error;
 | 
						|
		}
 | 
						|
		if (res == 0)
 | 
						|
			break;
 | 
						|
 | 
						|
		if (client->core_resource == NULL) {
 | 
						|
			res = -EPROTO;
 | 
						|
			goto error;
 | 
						|
		}
 | 
						|
 | 
						|
		client->recv_seq = msg->seq;
 | 
						|
 | 
						|
		pw_log_trace(NAME" %p: got message %d from %u", client->protocol,
 | 
						|
			     msg->opcode, msg->id);
 | 
						|
 | 
						|
		if (debug_messages)
 | 
						|
			debug_msg("<<<<<< in", msg, false);
 | 
						|
 | 
						|
		resource = pw_impl_client_find_resource(client, msg->id);
 | 
						|
		if (resource == NULL) {
 | 
						|
			pw_resource_errorf(client->core_resource,
 | 
						|
					-EINVAL, "unknown resource %u op:%u", msg->id, msg->opcode);
 | 
						|
			continue;
 | 
						|
		}
 | 
						|
 | 
						|
		marshal = pw_resource_get_marshal(resource);
 | 
						|
		if (marshal == NULL || msg->opcode >= marshal->n_client_methods) {
 | 
						|
			res = -EINVAL;
 | 
						|
			goto invalid_method;
 | 
						|
		}
 | 
						|
 | 
						|
		demarshal = marshal->server_demarshal;
 | 
						|
		if (!demarshal[msg->opcode].func) {
 | 
						|
			res = -ENOENT;
 | 
						|
			goto invalid_message;
 | 
						|
		}
 | 
						|
 | 
						|
		permissions = pw_resource_get_permissions(resource);
 | 
						|
		required = demarshal[msg->opcode].permissions | PW_PERM_X;
 | 
						|
 | 
						|
		if ((required & permissions) != required) {
 | 
						|
			pw_resource_errorf_id(resource, msg->id,
 | 
						|
				-EACCES, "no permission to call method %u on %u (requires %08x, have %08x)",
 | 
						|
				msg->opcode, msg->id, required, permissions);
 | 
						|
			continue;
 | 
						|
		}
 | 
						|
 | 
						|
		if ((res = demarshal[msg->opcode].func(resource, msg)) < 0)
 | 
						|
			goto invalid_message;
 | 
						|
	}
 | 
						|
	res = 0;
 | 
						|
done:
 | 
						|
	context->current_client = NULL;
 | 
						|
	return res;
 | 
						|
 | 
						|
invalid_method:
 | 
						|
	pw_resource_errorf_id(resource, msg->id, res, "invalid method id:%u op:%u",
 | 
						|
			msg->id, msg->opcode);
 | 
						|
	goto done;
 | 
						|
invalid_message:
 | 
						|
	pw_resource_errorf_id(resource, msg->id, res, "invalid message id:%u op:%u (%s)",
 | 
						|
			msg->id, msg->opcode, spa_strerror(res));
 | 
						|
	debug_msg("*invalid message*", msg, true);
 | 
						|
	goto done;
 | 
						|
error:
 | 
						|
	if (client->core_resource)
 | 
						|
		pw_resource_errorf(client->core_resource, res, "client error %d (%s)",
 | 
						|
				res, spa_strerror(res));
 | 
						|
	goto done;
 | 
						|
}
 | 
						|
 | 
						|
static void
 | 
						|
client_busy_changed(void *data, bool busy)
 | 
						|
{
 | 
						|
	struct client_data *c = data;
 | 
						|
	struct pw_impl_client *client = c->client;
 | 
						|
	uint32_t mask = c->source->mask;
 | 
						|
 | 
						|
	c->busy = busy;
 | 
						|
 | 
						|
	SPA_FLAG_UPDATE(mask, SPA_IO_IN, !busy);
 | 
						|
 | 
						|
	pw_log_debug(NAME" %p: busy changed %d", client->protocol, busy);
 | 
						|
	pw_loop_update_io(client->context->main_loop, c->source, mask);
 | 
						|
 | 
						|
	if (!busy)
 | 
						|
		process_messages(c);
 | 
						|
}
 | 
						|
 | 
						|
static void
 | 
						|
connection_data(void *data, int fd, uint32_t mask)
 | 
						|
{
 | 
						|
	struct client_data *this = data;
 | 
						|
	struct pw_impl_client *client = this->client;
 | 
						|
	int res;
 | 
						|
 | 
						|
	if (mask & SPA_IO_HUP) {
 | 
						|
		res = -EPIPE;
 | 
						|
		goto error;
 | 
						|
	}
 | 
						|
	if (mask & SPA_IO_ERR) {
 | 
						|
		res = -EIO;
 | 
						|
		goto error;
 | 
						|
	}
 | 
						|
	if (mask & SPA_IO_OUT) {
 | 
						|
		res = pw_protocol_native_connection_flush(this->connection);
 | 
						|
		if (res >= 0) {
 | 
						|
			int mask = this->source->mask;
 | 
						|
			SPA_FLAG_CLEAR(mask, SPA_IO_OUT);
 | 
						|
			pw_loop_update_io(client->context->main_loop,
 | 
						|
					this->source, mask);
 | 
						|
		} else if (res != EAGAIN)
 | 
						|
			goto error;
 | 
						|
	}
 | 
						|
	if (mask & SPA_IO_IN) {
 | 
						|
		if ((res = process_messages(this)) < 0)
 | 
						|
			goto error;
 | 
						|
	}
 | 
						|
	return;
 | 
						|
error:
 | 
						|
	if (res == -EPIPE)
 | 
						|
		pw_log_info(NAME" %p: client %p disconnected", client->protocol, client);
 | 
						|
	else
 | 
						|
		pw_log_error(NAME" %p: client %p error %d (%s)", client->protocol,
 | 
						|
				client, res, spa_strerror(res));
 | 
						|
	pw_impl_client_destroy(client);
 | 
						|
}
 | 
						|
 | 
						|
static void client_free(void *data)
 | 
						|
{
 | 
						|
	struct client_data *this = data;
 | 
						|
	struct pw_impl_client *client = this->client;
 | 
						|
 | 
						|
	spa_list_remove(&this->protocol_link);
 | 
						|
 | 
						|
	if (this->source)
 | 
						|
		pw_loop_destroy_source(client->context->main_loop, this->source);
 | 
						|
	if (this->connection)
 | 
						|
		pw_protocol_native_connection_destroy(this->connection);
 | 
						|
 | 
						|
	pw_map_clear(&this->compat_v2.types);
 | 
						|
}
 | 
						|
 | 
						|
static const struct pw_impl_client_events client_events = {
 | 
						|
	PW_VERSION_IMPL_CLIENT_EVENTS,
 | 
						|
	.free = client_free,
 | 
						|
	.busy_changed = client_busy_changed,
 | 
						|
};
 | 
						|
 | 
						|
static void on_start(void *data, uint32_t version)
 | 
						|
{
 | 
						|
	struct client_data *this = data;
 | 
						|
	struct pw_impl_client *client = this->client;
 | 
						|
 | 
						|
	pw_log_debug("version %d", version);
 | 
						|
 | 
						|
	if (client->core_resource != NULL)
 | 
						|
		pw_resource_remove(client->core_resource);
 | 
						|
 | 
						|
	if (pw_global_bind(pw_impl_core_get_global(client->core), client,
 | 
						|
			PW_PERM_RWX, version, 0) < 0)
 | 
						|
		return;
 | 
						|
 | 
						|
	if (version == 0)
 | 
						|
		client->compat_v2 = &this->compat_v2;
 | 
						|
 | 
						|
	return;
 | 
						|
}
 | 
						|
 | 
						|
static const struct pw_protocol_native_connection_events server_conn_events = {
 | 
						|
	PW_VERSION_PROTOCOL_NATIVE_CONNECTION_EVENTS,
 | 
						|
	.start = on_start,
 | 
						|
};
 | 
						|
 | 
						|
static struct client_data *client_new(struct server *s, int fd)
 | 
						|
{
 | 
						|
	struct client_data *this;
 | 
						|
	struct pw_impl_client *client;
 | 
						|
	struct pw_protocol *protocol = s->this.protocol;
 | 
						|
	socklen_t len;
 | 
						|
	struct ucred ucred;
 | 
						|
	struct pw_context *context = protocol->context;
 | 
						|
	struct pw_properties *props;
 | 
						|
	char buffer[1024];
 | 
						|
	struct protocol_data *d = pw_protocol_get_user_data(protocol);
 | 
						|
	int res;
 | 
						|
 | 
						|
	props = pw_properties_new(PW_KEY_PROTOCOL, "protocol-native", NULL);
 | 
						|
	if (props == NULL)
 | 
						|
		goto exit;
 | 
						|
 | 
						|
#ifndef __FreeBSD__
 | 
						|
	len = sizeof(ucred);
 | 
						|
	if (getsockopt(fd, SOL_SOCKET, SO_PEERCRED, &ucred, &len) < 0) {
 | 
						|
		pw_log_warn("server %p: no peercred: %m", s);
 | 
						|
	} else {
 | 
						|
		pw_properties_setf(props, PW_KEY_SEC_PID, "%d", ucred.pid);
 | 
						|
		pw_properties_setf(props, PW_KEY_SEC_UID, "%d", ucred.uid);
 | 
						|
		pw_properties_setf(props, PW_KEY_SEC_GID, "%d", ucred.gid);
 | 
						|
	}
 | 
						|
 | 
						|
	len = sizeof(buffer);
 | 
						|
	if (getsockopt(fd, SOL_SOCKET, SO_PEERSEC, buffer, &len) < 0) {
 | 
						|
		pw_log_warn("server %p: no peersec: %m", s);
 | 
						|
	} else {
 | 
						|
		/* buffer is not null terminated, must use length explicitly */
 | 
						|
		pw_properties_setf(props, PW_KEY_SEC_LABEL, "%.*s",
 | 
						|
				(int)len, buffer);
 | 
						|
	}
 | 
						|
#endif
 | 
						|
 | 
						|
	pw_properties_setf(props, PW_KEY_MODULE_ID, "%d", d->module->global->id);
 | 
						|
 | 
						|
	client = pw_context_create_client(s->this.core,
 | 
						|
			protocol, props, sizeof(struct client_data));
 | 
						|
	if (client == NULL)
 | 
						|
		goto exit;
 | 
						|
 | 
						|
 | 
						|
	this = pw_impl_client_get_user_data(client);
 | 
						|
	spa_list_append(&s->this.client_list, &this->protocol_link);
 | 
						|
 | 
						|
	this->client = client;
 | 
						|
	this->source = pw_loop_add_io(pw_context_get_main_loop(context),
 | 
						|
				      fd, SPA_IO_ERR | SPA_IO_HUP, true,
 | 
						|
				      connection_data, this);
 | 
						|
	if (this->source == NULL) {
 | 
						|
		res = -errno;
 | 
						|
		goto cleanup_client;
 | 
						|
	}
 | 
						|
 | 
						|
	this->connection = pw_protocol_native_connection_new(protocol->context, fd);
 | 
						|
	if (this->connection == NULL) {
 | 
						|
		res = -errno;
 | 
						|
		goto cleanup_client;
 | 
						|
	}
 | 
						|
 | 
						|
	pw_map_init(&this->compat_v2.types, 0, 32);
 | 
						|
 | 
						|
	pw_protocol_native_connection_add_listener(this->connection,
 | 
						|
						   &this->conn_listener,
 | 
						|
						   &server_conn_events,
 | 
						|
						   this);
 | 
						|
 | 
						|
	pw_impl_client_add_listener(client, &this->client_listener, &client_events, this);
 | 
						|
 | 
						|
	if ((res = pw_impl_client_register(client, NULL)) < 0)
 | 
						|
		goto cleanup_client;
 | 
						|
 | 
						|
	if (!client->busy)
 | 
						|
		pw_loop_update_io(pw_context_get_main_loop(context),
 | 
						|
				this->source, this->source->mask | SPA_IO_IN);
 | 
						|
 | 
						|
	return this;
 | 
						|
 | 
						|
cleanup_client:
 | 
						|
	pw_impl_client_destroy(client);
 | 
						|
	errno = -res;
 | 
						|
exit:
 | 
						|
	return NULL;
 | 
						|
}
 | 
						|
 | 
						|
static int init_socket_name(struct server *s, const char *name)
 | 
						|
{
 | 
						|
	int name_size;
 | 
						|
	const char *runtime_dir;
 | 
						|
	bool path_is_absolute;
 | 
						|
 | 
						|
	path_is_absolute = name[0] == '/';
 | 
						|
 | 
						|
	runtime_dir = getenv("XDG_RUNTIME_DIR");
 | 
						|
	if (runtime_dir == NULL && !path_is_absolute) {
 | 
						|
		pw_log_error("server %p: XDG_RUNTIME_DIR not set in the environment", s);
 | 
						|
		return -ENOENT;
 | 
						|
	}
 | 
						|
 | 
						|
	s->addr.sun_family = AF_LOCAL;
 | 
						|
	if (path_is_absolute)
 | 
						|
		name_size = snprintf(s->addr.sun_path, sizeof(s->addr.sun_path),
 | 
						|
			     "%s", name) + 1;
 | 
						|
	else
 | 
						|
		name_size = snprintf(s->addr.sun_path, sizeof(s->addr.sun_path),
 | 
						|
			     "%s/%s", runtime_dir, name) + 1;
 | 
						|
 | 
						|
	if (name_size > (int) sizeof(s->addr.sun_path)) {
 | 
						|
		if (path_is_absolute)
 | 
						|
			pw_log_error("server %p: socket path \"%s\" plus null terminator exceeds %i bytes",
 | 
						|
				s, name, (int) sizeof(s->addr.sun_path));
 | 
						|
		else
 | 
						|
			pw_log_error("server %p: socket path \"%s/%s\" plus null terminator exceeds %i bytes",
 | 
						|
				s, runtime_dir, name, (int) sizeof(s->addr.sun_path));
 | 
						|
		*s->addr.sun_path = 0;
 | 
						|
		return -ENAMETOOLONG;
 | 
						|
	}
 | 
						|
	return 0;
 | 
						|
}
 | 
						|
 | 
						|
static int lock_socket(struct server *s)
 | 
						|
{
 | 
						|
	int res;
 | 
						|
 | 
						|
	snprintf(s->lock_addr, sizeof(s->lock_addr), "%s%s", s->addr.sun_path, LOCK_SUFFIX);
 | 
						|
 | 
						|
	s->fd_lock = open(s->lock_addr, O_CREAT | O_CLOEXEC,
 | 
						|
			  (S_IRUSR | S_IWUSR | S_IRGRP | S_IWGRP));
 | 
						|
 | 
						|
	if (s->fd_lock < 0) {
 | 
						|
		res = -errno;
 | 
						|
		pw_log_error("server %p: unable to open lockfile '%s': %m", s, s->lock_addr);
 | 
						|
		goto err;
 | 
						|
	}
 | 
						|
 | 
						|
	if (flock(s->fd_lock, LOCK_EX | LOCK_NB) < 0) {
 | 
						|
		res = -errno;
 | 
						|
		pw_log_error("server %p: unable to lock lockfile '%s': %m"
 | 
						|
				" (maybe another daemon is running)",
 | 
						|
				s, s->lock_addr);
 | 
						|
		goto err_fd;
 | 
						|
	}
 | 
						|
	return 0;
 | 
						|
 | 
						|
err_fd:
 | 
						|
	close(s->fd_lock);
 | 
						|
	s->fd_lock = -1;
 | 
						|
err:
 | 
						|
	*s->lock_addr = 0;
 | 
						|
	*s->addr.sun_path = 0;
 | 
						|
	return res;
 | 
						|
}
 | 
						|
 | 
						|
static void
 | 
						|
socket_data(void *data, int fd, uint32_t mask)
 | 
						|
{
 | 
						|
	struct server *s = data;
 | 
						|
	struct client_data *client;
 | 
						|
	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("server %p: failed to accept: %m", s);
 | 
						|
		return;
 | 
						|
	}
 | 
						|
 | 
						|
	client = client_new(s, client_fd);
 | 
						|
	if (client == NULL) {
 | 
						|
		pw_log_error("server %p: failed to create client", s);
 | 
						|
		close(client_fd);
 | 
						|
		return;
 | 
						|
	}
 | 
						|
}
 | 
						|
 | 
						|
static int add_socket(struct pw_protocol *protocol, struct server *s)
 | 
						|
{
 | 
						|
	socklen_t size;
 | 
						|
	int fd = -1, res;
 | 
						|
	bool activated = false;
 | 
						|
 | 
						|
#ifdef HAVE_SYSTEMD_DAEMON
 | 
						|
	{
 | 
						|
		int i, n = sd_listen_fds(0);
 | 
						|
		for (i = 0; i < n; ++i) {
 | 
						|
			if (sd_is_socket_unix(SD_LISTEN_FDS_START + i, SOCK_STREAM,
 | 
						|
						1, s->addr.sun_path, 0) > 0) {
 | 
						|
				fd = SD_LISTEN_FDS_START + i;
 | 
						|
				activated = true;
 | 
						|
				pw_log_info("server %p: Found socket activation socket for '%s'",
 | 
						|
						s, s->addr.sun_path);
 | 
						|
				break;
 | 
						|
			}
 | 
						|
		}
 | 
						|
	}
 | 
						|
#endif
 | 
						|
 | 
						|
	if (fd < 0) {
 | 
						|
		struct stat socket_stat;
 | 
						|
 | 
						|
		if ((fd = socket(PF_LOCAL, SOCK_STREAM | SOCK_CLOEXEC | SOCK_NONBLOCK, 0)) < 0) {
 | 
						|
			res = -errno;
 | 
						|
			goto error;
 | 
						|
		}
 | 
						|
		if (stat(s->addr.sun_path, &socket_stat) < 0) {
 | 
						|
			if (errno != ENOENT) {
 | 
						|
				res = -errno;
 | 
						|
				pw_log_error("server %p: stat %s failed with error: %m",
 | 
						|
						s, s->addr.sun_path);
 | 
						|
				goto error_close;
 | 
						|
			}
 | 
						|
		} else if (socket_stat.st_mode & S_IWUSR || socket_stat.st_mode & S_IWGRP) {
 | 
						|
			unlink(s->addr.sun_path);
 | 
						|
		}
 | 
						|
 | 
						|
		size = offsetof(struct sockaddr_un, sun_path) + strlen(s->addr.sun_path);
 | 
						|
		if (bind(fd, (struct sockaddr *) &s->addr, size) < 0) {
 | 
						|
			res = -errno;
 | 
						|
			pw_log_error("server %p: bind() failed with error: %m", s);
 | 
						|
			goto error_close;
 | 
						|
		}
 | 
						|
 | 
						|
		if (listen(fd, 128) < 0) {
 | 
						|
			res = -errno;
 | 
						|
			pw_log_error("server %p: listen() failed with error: %m", s);
 | 
						|
			goto error_close;
 | 
						|
		}
 | 
						|
	}
 | 
						|
 | 
						|
	s->activated = activated;
 | 
						|
	s->loop = pw_context_get_main_loop(protocol->context);
 | 
						|
	if (s->loop == NULL) {
 | 
						|
		res = -errno;
 | 
						|
		goto error_close;
 | 
						|
	}
 | 
						|
	s->source = pw_loop_add_io(s->loop, fd, SPA_IO_IN, true, socket_data, s);
 | 
						|
	if (s->source == NULL) {
 | 
						|
		res = -errno;
 | 
						|
		goto error_close;
 | 
						|
	}
 | 
						|
	return 0;
 | 
						|
 | 
						|
error_close:
 | 
						|
	close(fd);
 | 
						|
error:
 | 
						|
	return res;
 | 
						|
 | 
						|
}
 | 
						|
 | 
						|
static int impl_steal_fd(struct pw_protocol_client *client)
 | 
						|
{
 | 
						|
	struct client *impl = SPA_CONTAINER_OF(client, struct client, this);
 | 
						|
	int fd;
 | 
						|
 | 
						|
	if (impl->source == NULL)
 | 
						|
		return -EIO;
 | 
						|
 | 
						|
	fd = dup(impl->source->fd);
 | 
						|
	if (fd < 0)
 | 
						|
		return -errno;
 | 
						|
 | 
						|
	pw_protocol_client_disconnect(client);
 | 
						|
	return fd;
 | 
						|
}
 | 
						|
 | 
						|
static int
 | 
						|
process_remote(struct client *impl)
 | 
						|
{
 | 
						|
	const struct pw_protocol_native_message *msg;
 | 
						|
	struct pw_protocol_native_connection *conn = impl->connection;
 | 
						|
	struct pw_core *this = impl->this.core;
 | 
						|
	int res = 0;
 | 
						|
 | 
						|
	while (!impl->disconnecting && !impl->paused) {
 | 
						|
		struct pw_proxy *proxy;
 | 
						|
		const struct pw_protocol_native_demarshal *demarshal;
 | 
						|
		const struct pw_protocol_marshal *marshal;
 | 
						|
 | 
						|
		res = pw_protocol_native_connection_get_next(conn, &msg);
 | 
						|
		if (res < 0) {
 | 
						|
			if (res == -EAGAIN)
 | 
						|
				break;
 | 
						|
			return res;
 | 
						|
		}
 | 
						|
		if (res == 0)
 | 
						|
			break;
 | 
						|
 | 
						|
		pw_log_trace(NAME" %p: got message %d from %u seq:%d",
 | 
						|
			this, msg->opcode, msg->id, msg->seq);
 | 
						|
 | 
						|
		this->recv_seq = msg->seq;
 | 
						|
 | 
						|
		if (debug_messages)
 | 
						|
			debug_msg("<<<<<< in", msg, false);
 | 
						|
 | 
						|
		proxy = pw_core_find_proxy(this, msg->id);
 | 
						|
		if (proxy == NULL || proxy->zombie) {
 | 
						|
			if (proxy == NULL)
 | 
						|
				pw_log_error(NAME" %p: could not find proxy %u", this, msg->id);
 | 
						|
			else
 | 
						|
				pw_log_debug(NAME" %p: zombie proxy %u", this, msg->id);
 | 
						|
 | 
						|
			/* FIXME close fds */
 | 
						|
			continue;
 | 
						|
		}
 | 
						|
 | 
						|
		marshal = pw_proxy_get_marshal(proxy);
 | 
						|
		if (marshal == NULL || msg->opcode >= marshal->n_server_methods) {
 | 
						|
			pw_log_error(NAME" %p: invalid method %u for %u (%d)",
 | 
						|
					this, msg->opcode, msg->id,
 | 
						|
					marshal ? marshal->n_server_methods : (uint32_t)-1);
 | 
						|
			continue;
 | 
						|
		}
 | 
						|
 | 
						|
                       demarshal = marshal->client_demarshal;
 | 
						|
		if (!demarshal[msg->opcode].func) {
 | 
						|
                               pw_log_error(NAME" %p: function %d not implemented on %u",
 | 
						|
					this, msg->opcode, msg->id);
 | 
						|
			continue;
 | 
						|
		}
 | 
						|
		proxy->refcount++;
 | 
						|
		res = demarshal[msg->opcode].func(proxy, msg);
 | 
						|
		pw_proxy_unref(proxy);
 | 
						|
 | 
						|
		if (res < 0) {
 | 
						|
			pw_log_error(NAME" %p: invalid message received %u for %u",
 | 
						|
					this, msg->opcode, msg->id);
 | 
						|
			debug_msg("*invalid*", msg, true);
 | 
						|
			continue;
 | 
						|
		}
 | 
						|
	}
 | 
						|
	return 0;
 | 
						|
}
 | 
						|
 | 
						|
static void
 | 
						|
on_remote_data(void *data, int fd, uint32_t mask)
 | 
						|
{
 | 
						|
	struct client *impl = data;
 | 
						|
	struct pw_core *this = impl->this.core;
 | 
						|
	struct pw_protocol_native_connection *conn = impl->connection;
 | 
						|
	struct pw_context *context = pw_core_get_context(this);
 | 
						|
	struct pw_loop *loop = pw_context_get_main_loop(context);
 | 
						|
	int res;
 | 
						|
 | 
						|
        if (mask & (SPA_IO_ERR | SPA_IO_HUP)) {
 | 
						|
		res = -EPIPE;
 | 
						|
		goto error;
 | 
						|
	}
 | 
						|
	if (mask & SPA_IO_OUT) {
 | 
						|
		res = pw_protocol_native_connection_flush(conn);
 | 
						|
		if (res >= 0) {
 | 
						|
			int mask = impl->source->mask;
 | 
						|
			SPA_FLAG_CLEAR(mask, SPA_IO_OUT);
 | 
						|
			pw_loop_update_io(loop,
 | 
						|
					impl->source, mask);
 | 
						|
			impl->flushing = false;
 | 
						|
		} else if (res != EAGAIN)
 | 
						|
			goto error;
 | 
						|
	}
 | 
						|
 | 
						|
        if (mask & SPA_IO_IN) {
 | 
						|
		if ((res = process_remote(impl)) < 0)
 | 
						|
			goto error;
 | 
						|
	}
 | 
						|
	return;
 | 
						|
error:
 | 
						|
	pw_log_debug(NAME" %p: got connection error %d (%s)", impl, res, spa_strerror(res));
 | 
						|
	pw_proxy_notify((struct pw_proxy*)this,
 | 
						|
			struct pw_core_events, error, 0, 0,
 | 
						|
			this->recv_seq, res, "connection error");
 | 
						|
	pw_loop_destroy_source(loop, impl->source);
 | 
						|
	impl->source = NULL;
 | 
						|
}
 | 
						|
 | 
						|
static void on_need_flush(void *data)
 | 
						|
{
 | 
						|
        struct client *impl = data;
 | 
						|
 | 
						|
	if (!impl->flushing && impl->source) {
 | 
						|
		int mask = impl->source->mask;
 | 
						|
		impl->flushing = true;
 | 
						|
		SPA_FLAG_SET(mask, SPA_IO_OUT);
 | 
						|
		pw_loop_update_io(impl->context->main_loop,
 | 
						|
					impl->source, mask);
 | 
						|
	}
 | 
						|
}
 | 
						|
 | 
						|
static const struct pw_protocol_native_connection_events client_conn_events = {
 | 
						|
	PW_VERSION_PROTOCOL_NATIVE_CONNECTION_EVENTS,
 | 
						|
	.need_flush = on_need_flush,
 | 
						|
};
 | 
						|
 | 
						|
static int impl_connect_fd(struct pw_protocol_client *client, int fd, bool do_close)
 | 
						|
{
 | 
						|
	struct client *impl = SPA_CONTAINER_OF(client, struct client, this);
 | 
						|
	int res;
 | 
						|
 | 
						|
	impl->disconnecting = false;
 | 
						|
 | 
						|
	pw_protocol_native_connection_set_fd(impl->connection, fd);
 | 
						|
	impl->flushing = true;
 | 
						|
	impl->source = pw_loop_add_io(impl->context->main_loop,
 | 
						|
					fd,
 | 
						|
					SPA_IO_IN | SPA_IO_OUT | SPA_IO_HUP | SPA_IO_ERR,
 | 
						|
					do_close, on_remote_data, impl);
 | 
						|
	if (impl->source == NULL) {
 | 
						|
		res = -errno;
 | 
						|
		goto error_cleanup;
 | 
						|
	}
 | 
						|
 | 
						|
	pw_protocol_native_connection_add_listener(impl->connection,
 | 
						|
						   &impl->conn_listener,
 | 
						|
						   &client_conn_events,
 | 
						|
						   impl);
 | 
						|
	return 0;
 | 
						|
 | 
						|
error_cleanup:
 | 
						|
	if (impl->connection) {
 | 
						|
		pw_protocol_native_connection_destroy(impl->connection);
 | 
						|
		impl->connection = NULL;
 | 
						|
	}
 | 
						|
	return res;
 | 
						|
}
 | 
						|
 | 
						|
static void impl_disconnect(struct pw_protocol_client *client)
 | 
						|
{
 | 
						|
	struct client *impl = SPA_CONTAINER_OF(client, struct client, this);
 | 
						|
 | 
						|
	impl->disconnecting = true;
 | 
						|
 | 
						|
	if (impl->source)
 | 
						|
                pw_loop_destroy_source(impl->context->main_loop, impl->source);
 | 
						|
	impl->source = NULL;
 | 
						|
 | 
						|
	if (impl->connection)
 | 
						|
                pw_protocol_native_connection_destroy(impl->connection);
 | 
						|
	impl->connection = NULL;
 | 
						|
}
 | 
						|
 | 
						|
static void impl_destroy(struct pw_protocol_client *client)
 | 
						|
{
 | 
						|
	struct client *impl = SPA_CONTAINER_OF(client, struct client, this);
 | 
						|
 | 
						|
	impl_disconnect(client);
 | 
						|
 | 
						|
	spa_list_remove(&client->link);
 | 
						|
	free(impl);
 | 
						|
}
 | 
						|
 | 
						|
static int impl_set_paused(struct pw_protocol_client *client, bool paused)
 | 
						|
{
 | 
						|
	struct client *impl = SPA_CONTAINER_OF(client, struct client, this);
 | 
						|
	uint32_t mask;
 | 
						|
 | 
						|
	if (impl->source == NULL)
 | 
						|
		return -EIO;
 | 
						|
 | 
						|
	mask = impl->source->mask;
 | 
						|
 | 
						|
	impl->paused = paused;
 | 
						|
 | 
						|
	SPA_FLAG_UPDATE(mask, SPA_IO_IN, !paused);
 | 
						|
 | 
						|
	pw_log_debug(NAME" %p: paused %d", client->protocol, paused);
 | 
						|
	pw_loop_update_io(impl->context->main_loop, impl->source, mask);
 | 
						|
 | 
						|
	return paused ? 0 : process_remote(impl);
 | 
						|
}
 | 
						|
 | 
						|
static int pw_protocol_native_connect_internal(struct pw_protocol_client *client,
 | 
						|
					    const struct spa_dict *props,
 | 
						|
					    void (*done_callback) (void *data, int res),
 | 
						|
					    void *data)
 | 
						|
{
 | 
						|
	int res, sv[2];
 | 
						|
	struct pw_protocol *protocol = client->protocol;
 | 
						|
	struct protocol_data *d = pw_protocol_get_user_data(protocol);
 | 
						|
	struct server *s = d->local;
 | 
						|
	struct pw_permission permissions[1];
 | 
						|
	struct client_data *c;
 | 
						|
 | 
						|
	pw_log_debug("server %p: internal connect", s);
 | 
						|
 | 
						|
	if (socketpair(PF_LOCAL, SOCK_STREAM | SOCK_CLOEXEC | SOCK_NONBLOCK, 0, sv) < 0) {
 | 
						|
		res = -errno;
 | 
						|
		pw_log_error("server %p: socketpair() failed with error: %m", s);
 | 
						|
		goto error;
 | 
						|
	}
 | 
						|
 | 
						|
	c = client_new(s, sv[0]);
 | 
						|
	if (c == NULL) {
 | 
						|
		res = -errno;
 | 
						|
		pw_log_error("server %p: failed to create client: %m", s);
 | 
						|
		goto error_close;
 | 
						|
	}
 | 
						|
	permissions[0] = PW_PERMISSION_INIT(PW_ID_ANY, PW_PERM_RWX);
 | 
						|
	pw_impl_client_update_permissions(c->client, 1, permissions);
 | 
						|
 | 
						|
	res = pw_protocol_client_connect_fd(client, sv[1], true);
 | 
						|
done:
 | 
						|
	if (done_callback)
 | 
						|
		done_callback(data, res);
 | 
						|
	return res;
 | 
						|
 | 
						|
error_close:
 | 
						|
	close(sv[0]);
 | 
						|
	close(sv[1]);
 | 
						|
error:
 | 
						|
	goto done;
 | 
						|
}
 | 
						|
 | 
						|
static struct pw_protocol_client *
 | 
						|
impl_new_client(struct pw_protocol *protocol,
 | 
						|
		struct pw_core *core,
 | 
						|
		const struct spa_dict *props)
 | 
						|
{
 | 
						|
	struct client *impl;
 | 
						|
	struct pw_protocol_client *this;
 | 
						|
	const char *str = NULL;
 | 
						|
	int res;
 | 
						|
 | 
						|
	if ((impl = calloc(1, sizeof(struct client))) == NULL)
 | 
						|
		return NULL;
 | 
						|
 | 
						|
	pw_log_debug(NAME" %p: new client %p", protocol, impl);
 | 
						|
 | 
						|
	this = &impl->this;
 | 
						|
	this->protocol = protocol;
 | 
						|
	this->core = core;
 | 
						|
 | 
						|
	impl->context = protocol->context;
 | 
						|
	impl->connection = pw_protocol_native_connection_new(protocol->context, -1);
 | 
						|
	if (impl->connection == NULL) {
 | 
						|
		res = -errno;
 | 
						|
		goto error_free;
 | 
						|
	}
 | 
						|
 | 
						|
	if (props) {
 | 
						|
		str = spa_dict_lookup(props, PW_KEY_REMOTE_INTENTION);
 | 
						|
		if (str == NULL &&
 | 
						|
		   (str = spa_dict_lookup(props, PW_KEY_REMOTE_NAME)) != NULL &&
 | 
						|
		    strcmp(str, "internal") == 0)
 | 
						|
			str = "internal";
 | 
						|
	}
 | 
						|
	if (str == NULL)
 | 
						|
		str = "generic";
 | 
						|
 | 
						|
	pw_log_debug(NAME" %p: connect %s", protocol, str);
 | 
						|
 | 
						|
	if (!strcmp(str, "screencast"))
 | 
						|
		this->connect = pw_protocol_native_connect_portal_screencast;
 | 
						|
	else if (!strcmp(str, "internal"))
 | 
						|
		this->connect = pw_protocol_native_connect_internal;
 | 
						|
	else
 | 
						|
		this->connect = pw_protocol_native_connect_local_socket;
 | 
						|
 | 
						|
	this->steal_fd = impl_steal_fd;
 | 
						|
	this->connect_fd = impl_connect_fd;
 | 
						|
	this->disconnect = impl_disconnect;
 | 
						|
	this->destroy = impl_destroy;
 | 
						|
	this->set_paused = impl_set_paused;
 | 
						|
 | 
						|
	spa_list_append(&protocol->client_list, &this->link);
 | 
						|
 | 
						|
	return this;
 | 
						|
 | 
						|
error_free:
 | 
						|
	free(impl);
 | 
						|
	errno = -res;
 | 
						|
	return NULL;
 | 
						|
}
 | 
						|
 | 
						|
static void destroy_server(struct pw_protocol_server *server)
 | 
						|
{
 | 
						|
	struct server *s = SPA_CONTAINER_OF(server, struct server, this);
 | 
						|
	struct client_data *data, *tmp;
 | 
						|
 | 
						|
	spa_list_remove(&server->link);
 | 
						|
	spa_hook_remove(&s->hook);
 | 
						|
 | 
						|
	spa_list_for_each_safe(data, tmp, &server->client_list, protocol_link)
 | 
						|
		pw_impl_client_destroy(data->client);
 | 
						|
 | 
						|
	if (s->source) {
 | 
						|
		spa_hook_remove(&s->hook);
 | 
						|
		pw_loop_destroy_source(s->loop, s->source);
 | 
						|
	}
 | 
						|
	if (s->addr.sun_path[0] && !s->activated)
 | 
						|
		unlink(s->addr.sun_path);
 | 
						|
	if (s->lock_addr[0])
 | 
						|
		unlink(s->lock_addr);
 | 
						|
	if (s->fd_lock != -1)
 | 
						|
		close(s->fd_lock);
 | 
						|
	free(s);
 | 
						|
}
 | 
						|
 | 
						|
static void on_before_hook(void *_data)
 | 
						|
{
 | 
						|
	struct server *server = _data;
 | 
						|
	struct pw_protocol_server *this = &server->this;
 | 
						|
	struct client_data *data, *tmp;
 | 
						|
	int res;
 | 
						|
 | 
						|
	spa_list_for_each_safe(data, tmp, &this->client_list, protocol_link) {
 | 
						|
		res = pw_protocol_native_connection_flush(data->connection);
 | 
						|
		if (res == -EAGAIN) {
 | 
						|
			int mask = data->source->mask;
 | 
						|
			SPA_FLAG_SET(mask, SPA_IO_OUT);
 | 
						|
			pw_loop_update_io(data->client->context->main_loop,
 | 
						|
					data->source, mask);
 | 
						|
		} else if (res < 0) {
 | 
						|
			pw_log_warn("client %p: could not flush: %s",
 | 
						|
					data->client, spa_strerror(res));
 | 
						|
			pw_impl_client_destroy(data->client);
 | 
						|
		}
 | 
						|
 | 
						|
	}
 | 
						|
}
 | 
						|
 | 
						|
static const struct spa_loop_control_hooks impl_hooks = {
 | 
						|
	SPA_VERSION_LOOP_CONTROL_HOOKS,
 | 
						|
	.before = on_before_hook,
 | 
						|
};
 | 
						|
 | 
						|
static const char *
 | 
						|
get_name(const struct spa_dict *props)
 | 
						|
{
 | 
						|
	const char *name = NULL;
 | 
						|
 | 
						|
	if (props)
 | 
						|
		name = spa_dict_lookup(props, PW_KEY_CORE_NAME);
 | 
						|
	if (name == NULL)
 | 
						|
		name = getenv("PIPEWIRE_CORE");
 | 
						|
	if (name == NULL)
 | 
						|
		name = "pipewire-0";
 | 
						|
	return name;
 | 
						|
}
 | 
						|
 | 
						|
static struct server *
 | 
						|
create_server(struct pw_protocol *protocol,
 | 
						|
		struct pw_impl_core *core,
 | 
						|
                const struct spa_dict *props)
 | 
						|
{
 | 
						|
	struct pw_protocol_server *this;
 | 
						|
	struct pw_context *context = protocol->context;
 | 
						|
	struct server *s;
 | 
						|
 | 
						|
	if ((s = calloc(1, sizeof(struct server))) == NULL)
 | 
						|
		return NULL;
 | 
						|
 | 
						|
	s->fd_lock = -1;
 | 
						|
 | 
						|
	this = &s->this;
 | 
						|
	this->protocol = protocol;
 | 
						|
	this->core = core;
 | 
						|
	spa_list_init(&this->client_list);
 | 
						|
	this->destroy = destroy_server;
 | 
						|
 | 
						|
	spa_list_append(&protocol->server_list, &this->link);
 | 
						|
 | 
						|
	pw_loop_add_hook(pw_context_get_main_loop(context), &s->hook, &impl_hooks, s);
 | 
						|
 | 
						|
	pw_log_info(NAME" %p: created server %p", protocol, this);
 | 
						|
 | 
						|
	return s;
 | 
						|
}
 | 
						|
 | 
						|
static struct pw_protocol_server *
 | 
						|
impl_add_server(struct pw_protocol *protocol,
 | 
						|
		struct pw_impl_core *core,
 | 
						|
                const struct spa_dict *props)
 | 
						|
{
 | 
						|
	struct pw_protocol_server *this;
 | 
						|
	struct server *s;
 | 
						|
	const char *name;
 | 
						|
	int res;
 | 
						|
 | 
						|
	if ((s = create_server(protocol, core, props)) == NULL)
 | 
						|
		return NULL;
 | 
						|
 | 
						|
	this = &s->this;
 | 
						|
 | 
						|
	name = get_name(props);
 | 
						|
 | 
						|
	if ((res = init_socket_name(s, name)) < 0)
 | 
						|
		goto error;
 | 
						|
 | 
						|
	if ((res = lock_socket(s)) < 0)
 | 
						|
		goto error;
 | 
						|
 | 
						|
	if ((res = add_socket(protocol, s)) < 0)
 | 
						|
		goto error;
 | 
						|
 | 
						|
	pw_log_info(NAME" %p: Listening on '%s'", protocol, name);
 | 
						|
 | 
						|
	return this;
 | 
						|
 | 
						|
error:
 | 
						|
	destroy_server(this);
 | 
						|
	errno = -res;
 | 
						|
	return NULL;
 | 
						|
}
 | 
						|
 | 
						|
const static struct pw_protocol_implementaton protocol_impl = {
 | 
						|
	PW_VERSION_PROTOCOL_IMPLEMENTATION,
 | 
						|
	.new_client = impl_new_client,
 | 
						|
	.add_server = impl_add_server,
 | 
						|
};
 | 
						|
 | 
						|
static struct spa_pod_builder *
 | 
						|
impl_ext_begin_proxy(struct pw_proxy *proxy, uint8_t opcode, struct pw_protocol_native_message **msg)
 | 
						|
{
 | 
						|
	struct client *impl = SPA_CONTAINER_OF(proxy->core->conn, struct client, this);
 | 
						|
	return pw_protocol_native_connection_begin(impl->connection, proxy->id, opcode, msg);
 | 
						|
}
 | 
						|
 | 
						|
static uint32_t impl_ext_add_proxy_fd(struct pw_proxy *proxy, int fd)
 | 
						|
{
 | 
						|
	struct client *impl = SPA_CONTAINER_OF(proxy->core->conn, struct client, 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 client *impl = SPA_CONTAINER_OF(proxy->core->conn, struct client, this);
 | 
						|
	return pw_protocol_native_connection_get_fd(impl->connection, index);
 | 
						|
}
 | 
						|
 | 
						|
static int impl_ext_end_proxy(struct pw_proxy *proxy,
 | 
						|
			       struct spa_pod_builder *builder)
 | 
						|
{
 | 
						|
	struct pw_core *core = proxy->core;
 | 
						|
	struct client *impl = SPA_CONTAINER_OF(core->conn, struct client, this);
 | 
						|
	return core->send_seq = 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 pw_protocol_native_message **msg)
 | 
						|
{
 | 
						|
	struct client_data *data = resource->client->user_data;
 | 
						|
	return pw_protocol_native_connection_begin(data->connection, resource->id, opcode, msg);
 | 
						|
}
 | 
						|
 | 
						|
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 int impl_ext_end_resource(struct pw_resource *resource,
 | 
						|
				  struct spa_pod_builder *builder)
 | 
						|
{
 | 
						|
	struct client_data *data = resource->client->user_data;
 | 
						|
	struct pw_impl_client *client = resource->client;
 | 
						|
	return client->send_seq = pw_protocol_native_connection_end(data->connection, builder);
 | 
						|
}
 | 
						|
const static struct pw_protocol_native_ext protocol_ext_impl = {
 | 
						|
	PW_VERSION_PROTOCOL_NATIVE_EXT,
 | 
						|
	.begin_proxy = impl_ext_begin_proxy,
 | 
						|
	.add_proxy_fd = impl_ext_add_proxy_fd,
 | 
						|
	.get_proxy_fd = impl_ext_get_proxy_fd,
 | 
						|
	.end_proxy = impl_ext_end_proxy,
 | 
						|
	.begin_resource = impl_ext_begin_resource,
 | 
						|
	.add_resource_fd = impl_ext_add_resource_fd,
 | 
						|
	.get_resource_fd = impl_ext_get_resource_fd,
 | 
						|
	.end_resource = impl_ext_end_resource,
 | 
						|
};
 | 
						|
 | 
						|
static void module_destroy(void *data)
 | 
						|
{
 | 
						|
	struct protocol_data *d = data;
 | 
						|
 | 
						|
	spa_hook_remove(&d->module_listener);
 | 
						|
 | 
						|
	pw_protocol_destroy(d->protocol);
 | 
						|
}
 | 
						|
 | 
						|
static const struct pw_impl_module_events module_events = {
 | 
						|
	PW_VERSION_IMPL_MODULE_EVENTS,
 | 
						|
	.destroy = module_destroy,
 | 
						|
};
 | 
						|
 | 
						|
static int need_server(struct pw_context *context, const struct spa_dict *props)
 | 
						|
{
 | 
						|
	const char *val;
 | 
						|
 | 
						|
	val = getenv("PIPEWIRE_DAEMON");
 | 
						|
	if (val == NULL)
 | 
						|
		val = spa_dict_lookup(props, PW_KEY_CORE_DAEMON);
 | 
						|
	if (val && pw_properties_parse_bool(val))
 | 
						|
		return 1;
 | 
						|
	return 0;
 | 
						|
}
 | 
						|
 | 
						|
SPA_EXPORT
 | 
						|
int pipewire__module_init(struct pw_impl_module *module, const char *args)
 | 
						|
{
 | 
						|
	struct pw_context *context = pw_impl_module_get_context(module);
 | 
						|
	struct pw_protocol *this;
 | 
						|
	struct protocol_data *d;
 | 
						|
	const struct pw_properties *props;
 | 
						|
	int res;
 | 
						|
 | 
						|
	if (pw_context_find_protocol(context, PW_TYPE_INFO_PROTOCOL_Native) != NULL)
 | 
						|
		return 0;
 | 
						|
 | 
						|
	this = pw_protocol_new(context, PW_TYPE_INFO_PROTOCOL_Native, sizeof(struct protocol_data));
 | 
						|
	if (this == NULL)
 | 
						|
		return -errno;
 | 
						|
 | 
						|
	debug_messages = pw_debug_is_category_enabled("connection");
 | 
						|
 | 
						|
	this->implementation = &protocol_impl;
 | 
						|
	this->extension = &protocol_ext_impl;
 | 
						|
 | 
						|
	pw_protocol_native_init(this);
 | 
						|
	pw_protocol_native0_init(this);
 | 
						|
 | 
						|
	pw_log_debug(NAME" %p: new debug:%d", this, debug_messages);
 | 
						|
 | 
						|
	d = pw_protocol_get_user_data(this);
 | 
						|
	d->protocol = this;
 | 
						|
	d->module = module;
 | 
						|
 | 
						|
	props = pw_context_get_properties(context);
 | 
						|
	d->local = create_server(this, context->core, &props->dict);
 | 
						|
 | 
						|
	if (need_server(context, &props->dict)) {
 | 
						|
		if (impl_add_server(this, context->core, &props->dict) == NULL) {
 | 
						|
			res = -errno;
 | 
						|
			goto error_cleanup;
 | 
						|
		}
 | 
						|
	}
 | 
						|
 | 
						|
	pw_impl_module_add_listener(module, &d->module_listener, &module_events, d);
 | 
						|
 | 
						|
	pw_impl_module_update_properties(module, &SPA_DICT_INIT_ARRAY(module_props));
 | 
						|
 | 
						|
	return 0;
 | 
						|
 | 
						|
error_cleanup:
 | 
						|
	pw_protocol_destroy(this);
 | 
						|
	return res;
 | 
						|
}
 |