mirror of
https://gitlab.freedesktop.org/pipewire/pipewire.git
synced 2025-11-03 09:01:54 -05:00
Move send and dispatch functions to the implementation. This makes it possible to place an access check before sending and dispatching. Add module-access that allows to bind and notify on globals owned by the client.
205 lines
5.9 KiB
C
205 lines
5.9 KiB
C
/* Pinos
|
|
* 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 "pinos/server/resource.h"
|
|
|
|
typedef struct {
|
|
PinosResource this;
|
|
|
|
PinosDispatchFunc dispatch_func;
|
|
void *dispatch_data;
|
|
} PinosResourceImpl;
|
|
|
|
PinosResource *
|
|
pinos_resource_new (PinosClient *client,
|
|
uint32_t id,
|
|
uint32_t type,
|
|
void *object,
|
|
PinosDestroy destroy)
|
|
{
|
|
PinosResourceImpl *impl;
|
|
PinosResource *this;
|
|
|
|
impl = calloc (1, sizeof (PinosResourceImpl));
|
|
if (impl == NULL)
|
|
return NULL;
|
|
|
|
this = &impl->this;
|
|
|
|
this->core = client->core;
|
|
this->client = client;
|
|
this->id = id;
|
|
this->type = type;
|
|
this->object = object;
|
|
this->destroy = destroy;
|
|
|
|
pinos_signal_init (&this->destroy_signal);
|
|
|
|
this->id = pinos_map_insert_new (&client->objects, this);
|
|
pinos_log_debug ("resource %p: new for client %p id %u", this, client, this->id);
|
|
|
|
return this;
|
|
}
|
|
|
|
static void
|
|
sync_destroy (void *object,
|
|
void *data,
|
|
SpaResult res,
|
|
uint32_t id)
|
|
{
|
|
PinosResource *resource = object;
|
|
pinos_log_debug ("resource %p: sync destroy", resource);
|
|
free (resource);
|
|
}
|
|
|
|
SpaResult
|
|
pinos_resource_destroy (PinosResource *resource)
|
|
{
|
|
pinos_log_debug ("resource %p: destroy", resource);
|
|
pinos_signal_emit (&resource->destroy_signal, resource);
|
|
|
|
pinos_map_remove (&resource->client->objects, resource->id);
|
|
|
|
if (resource->destroy)
|
|
resource->destroy (resource);
|
|
|
|
if (resource->client->core_resource) {
|
|
PinosMessageRemoveId m;
|
|
m.id = resource->id;
|
|
pinos_resource_send_message (resource->client->core_resource,
|
|
PINOS_MESSAGE_REMOVE_ID,
|
|
&m,
|
|
true);
|
|
}
|
|
|
|
pinos_main_loop_defer (resource->core->main_loop,
|
|
resource,
|
|
SPA_RESULT_WAIT_SYNC,
|
|
sync_destroy,
|
|
resource);
|
|
|
|
return SPA_RESULT_OK;
|
|
}
|
|
|
|
void
|
|
pinos_resource_set_dispatch (PinosResource *resource,
|
|
PinosDispatchFunc func,
|
|
void *data)
|
|
{
|
|
PinosResourceImpl *impl = SPA_CONTAINER_OF (resource, PinosResourceImpl, this);
|
|
|
|
impl->dispatch_func = func;
|
|
impl->dispatch_data = data;
|
|
}
|
|
|
|
static SpaResult
|
|
do_dispatch_message (PinosAccessData *data)
|
|
{
|
|
PinosResourceImpl *impl = SPA_CONTAINER_OF (data->resource, PinosResourceImpl, this);
|
|
|
|
if (data->res == SPA_RESULT_NO_PERMISSION) {
|
|
pinos_resource_send_error (data->resource,
|
|
data->res,
|
|
"no permission");
|
|
} else if (SPA_RESULT_IS_ERROR (data->res)) {
|
|
pinos_resource_send_error (data->resource,
|
|
data->res,
|
|
"error %d", data->res);
|
|
} else {
|
|
data->res = impl->dispatch_func (data->resource,
|
|
data->opcode,
|
|
data->message,
|
|
impl->dispatch_data);
|
|
}
|
|
return data->res;
|
|
}
|
|
|
|
SpaResult
|
|
pinos_resource_dispatch (PinosResource *resource,
|
|
uint32_t opcode,
|
|
void *message)
|
|
{
|
|
PinosResourceImpl *impl = SPA_CONTAINER_OF (resource, PinosResourceImpl, this);
|
|
|
|
if (impl->dispatch_func) {
|
|
PinosAccessData data;
|
|
|
|
data.client = resource->client;
|
|
data.resource = resource;
|
|
data.opcode = opcode;
|
|
data.message = message;
|
|
data.flush = false;
|
|
|
|
data.res = SPA_RESULT_OK;
|
|
pinos_signal_emit (&resource->core->access.check_dispatch,
|
|
do_dispatch_message,
|
|
&data);
|
|
|
|
if (SPA_RESULT_IS_ASYNC (data.res))
|
|
return data.res;
|
|
|
|
return do_dispatch_message (&data);
|
|
}
|
|
|
|
pinos_log_error ("resource %p: dispatch func not implemented", resource);
|
|
|
|
return SPA_RESULT_NOT_IMPLEMENTED;
|
|
}
|
|
|
|
SpaResult
|
|
pinos_resource_send_message (PinosResource *resource,
|
|
uint32_t opcode,
|
|
void *message,
|
|
bool flush)
|
|
{
|
|
return pinos_client_send_message (resource->client,
|
|
resource,
|
|
opcode,
|
|
message,
|
|
flush);
|
|
}
|
|
|
|
SpaResult
|
|
pinos_resource_send_error (PinosResource *resource,
|
|
SpaResult res,
|
|
const char *message,
|
|
...)
|
|
{
|
|
PinosClient *client = resource->client;
|
|
PinosMessageError m;
|
|
char buffer[128];
|
|
va_list ap;
|
|
|
|
va_start (ap, message);
|
|
vsnprintf (buffer, sizeof (buffer), message, ap);
|
|
va_end (ap);
|
|
|
|
m.id = resource->id;
|
|
m.res = res;
|
|
m.error = buffer;
|
|
|
|
pinos_log_error ("resource %p: %u send error %d %s", resource, resource->id, res, buffer);
|
|
|
|
return pinos_resource_send_message (client->core_resource,
|
|
PINOS_MESSAGE_ERROR,
|
|
&m,
|
|
true);
|
|
}
|