mirror of
https://gitlab.freedesktop.org/pipewire/pipewire.git
synced 2025-10-29 05:40:27 -04:00
Remove _remove from properties, we can do the same with set of a NULL value. Add signals to the stream API to manage the buffers. Wrap those buffers in a GstBuffer in the pinossrc and pinossink elements and pool them in a bufferpool. Remove SPA_EVENT_TYPE_PULL_INPUT, we can do the same with NEED_INPUT and by using a ringbuffer. Do more complete allocation of buffers in the link. Use the buffer allocator if none of the nodes can allocate. Follow the node state to trigger negotiation and allocation. Remove offset and size when refering to buffers, we want to always deal with the complete buffer and use a ringbuffer for ranges or change the offset/size in the buffer data when needed. Serialize port_info structures as part of the port_update Print both the enum number and the name when debuging properties or formats.
993 lines
27 KiB
C
993 lines
27 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 <stdlib.h>
|
|
|
|
#include <gio/gio.h>
|
|
#include <gio/gunixfdlist.h>
|
|
|
|
#include "pinos/client/pinos.h"
|
|
#include "pinos/client/enumtypes.h"
|
|
|
|
#include "pinos/server/node.h"
|
|
#include "pinos/server/daemon.h"
|
|
|
|
#include "pinos/dbus/org-pinos.h"
|
|
|
|
#define PINOS_NODE_GET_PRIVATE(node) \
|
|
(G_TYPE_INSTANCE_GET_PRIVATE ((node), PINOS_TYPE_NODE, PinosNodePrivate))
|
|
|
|
struct _PinosNodePrivate
|
|
{
|
|
PinosDaemon *daemon;
|
|
PinosNode1 *iface;
|
|
|
|
gchar *sender;
|
|
gchar *object_path;
|
|
gchar *name;
|
|
|
|
PinosNodeState state;
|
|
GError *error;
|
|
guint idle_timeout;
|
|
|
|
PinosProperties *properties;
|
|
|
|
GHashTable *ports;
|
|
};
|
|
|
|
G_DEFINE_ABSTRACT_TYPE (PinosNode, pinos_node, G_TYPE_OBJECT);
|
|
|
|
enum
|
|
{
|
|
PROP_0,
|
|
PROP_DAEMON,
|
|
PROP_SENDER,
|
|
PROP_OBJECT_PATH,
|
|
PROP_NAME,
|
|
PROP_STATE,
|
|
PROP_PROPERTIES,
|
|
PROP_NODE,
|
|
PROP_NODE_STATE,
|
|
};
|
|
|
|
enum
|
|
{
|
|
SIGNAL_REMOVE,
|
|
SIGNAL_PORT_ADDED,
|
|
SIGNAL_PORT_REMOVED,
|
|
LAST_SIGNAL
|
|
};
|
|
|
|
static guint signals[LAST_SIGNAL] = { 0 };
|
|
|
|
static gboolean
|
|
node_set_state (PinosNode *node,
|
|
PinosNodeState state)
|
|
{
|
|
return FALSE;
|
|
}
|
|
|
|
static void
|
|
do_remove_port (PinosPort *port, PinosNode *node)
|
|
{
|
|
pinos_node_remove_port (node, port->id);
|
|
}
|
|
|
|
static PinosPort *
|
|
node_add_port (PinosNode *node,
|
|
PinosDirection direction,
|
|
guint id,
|
|
GError **error)
|
|
{
|
|
PinosNodePrivate *priv = node->priv;
|
|
PinosPort *port;
|
|
|
|
port = g_object_new (PINOS_TYPE_PORT,
|
|
"daemon", priv->daemon,
|
|
"node", node,
|
|
"direction", direction,
|
|
"id", id,
|
|
NULL);
|
|
if (port) {
|
|
g_hash_table_insert (priv->ports, GUINT_TO_POINTER (port->id), port);
|
|
g_signal_connect (port, "remove", (GCallback) do_remove_port, node);
|
|
g_signal_emit (node, signals[SIGNAL_PORT_ADDED], 0, port);
|
|
}
|
|
return port;
|
|
}
|
|
|
|
static gboolean
|
|
node_remove_port (PinosNode *node,
|
|
guint id)
|
|
{
|
|
PinosNodePrivate *priv = node->priv;
|
|
PinosPort *port;
|
|
|
|
g_debug ("node %p: removed port %u", node, id);
|
|
port = g_hash_table_lookup (priv->ports, GUINT_TO_POINTER (id));
|
|
if (port) {
|
|
g_signal_emit (node, signals[SIGNAL_PORT_REMOVED], 0, port);
|
|
g_hash_table_remove (priv->ports, GUINT_TO_POINTER (id));
|
|
}
|
|
return TRUE;
|
|
}
|
|
|
|
static gboolean
|
|
handle_add_port (PinosNode1 *interface,
|
|
GDBusMethodInvocation *invocation,
|
|
PinosDirection arg_direction,
|
|
guint arg_id,
|
|
gpointer user_data)
|
|
{
|
|
PinosNode *node = user_data;
|
|
PinosNodePrivate *priv = node->priv;
|
|
const gchar *sender;
|
|
PinosPort *port;
|
|
GError *error = NULL;
|
|
|
|
sender = g_dbus_method_invocation_get_sender (invocation);
|
|
if (g_strcmp0 (priv->sender, sender) != 0)
|
|
goto not_allowed;
|
|
|
|
port = pinos_node_add_port (node, arg_direction, arg_id, &error);
|
|
if (port == NULL)
|
|
goto no_port;
|
|
|
|
g_debug ("node %p: add port %p", node, port);
|
|
g_dbus_method_invocation_return_value (invocation,
|
|
g_variant_new ("()"));
|
|
|
|
return TRUE;
|
|
|
|
/* ERRORS */
|
|
not_allowed:
|
|
{
|
|
g_debug ("sender %s is not owner of node with sender %s", sender, priv->sender);
|
|
g_dbus_method_invocation_return_dbus_error (invocation,
|
|
"org.pinos.Error", "not node owner");
|
|
return TRUE;
|
|
}
|
|
no_port:
|
|
{
|
|
g_debug ("node %p: could create port", node);
|
|
g_dbus_method_invocation_return_dbus_error (invocation,
|
|
"org.pinos.Error", "can't create port");
|
|
return TRUE;
|
|
}
|
|
}
|
|
|
|
static gboolean
|
|
handle_remove_port (PinosNode1 *interface,
|
|
GDBusMethodInvocation *invocation,
|
|
guint arg_id,
|
|
gpointer user_data)
|
|
{
|
|
PinosNode *node = user_data;
|
|
PinosNodePrivate *priv = node->priv;
|
|
const gchar *sender;
|
|
|
|
sender = g_dbus_method_invocation_get_sender (invocation);
|
|
if (g_strcmp0 (priv->sender, sender) != 0)
|
|
goto not_allowed;
|
|
|
|
if (!pinos_node_remove_port (node, arg_id))
|
|
goto no_port;
|
|
|
|
g_debug ("node %p: remove port %u", node, arg_id);
|
|
g_dbus_method_invocation_return_value (invocation,
|
|
g_variant_new ("()"));
|
|
|
|
|
|
return TRUE;
|
|
|
|
not_allowed:
|
|
{
|
|
g_debug ("sender %s is not owner of node with sender %s", sender, priv->sender);
|
|
g_dbus_method_invocation_return_dbus_error (invocation,
|
|
"org.pinos.Error", "not node owner");
|
|
return TRUE;
|
|
}
|
|
no_port:
|
|
{
|
|
g_debug ("node %p: could remove port", node);
|
|
g_dbus_method_invocation_return_dbus_error (invocation,
|
|
"org.pinos.Error", "can't remove port");
|
|
return TRUE;
|
|
}
|
|
}
|
|
|
|
|
|
static gboolean
|
|
handle_remove (PinosNode1 *interface,
|
|
GDBusMethodInvocation *invocation,
|
|
gpointer user_data)
|
|
{
|
|
PinosNode *node = user_data;
|
|
|
|
g_debug ("node %p: remove", node);
|
|
pinos_node_remove (node);
|
|
|
|
g_dbus_method_invocation_return_value (invocation,
|
|
g_variant_new ("()"));
|
|
return TRUE;
|
|
}
|
|
|
|
static void
|
|
pinos_node_get_property (GObject *_object,
|
|
guint prop_id,
|
|
GValue *value,
|
|
GParamSpec *pspec)
|
|
{
|
|
PinosNode *node = PINOS_NODE (_object);
|
|
PinosNodePrivate *priv = node->priv;
|
|
|
|
switch (prop_id) {
|
|
case PROP_DAEMON:
|
|
g_value_set_object (value, priv->daemon);
|
|
break;
|
|
|
|
case PROP_SENDER:
|
|
g_value_set_string (value, priv->sender);
|
|
break;
|
|
|
|
case PROP_OBJECT_PATH:
|
|
g_value_set_string (value, priv->object_path);
|
|
break;
|
|
|
|
case PROP_NAME:
|
|
g_value_set_string (value, priv->name);
|
|
break;
|
|
|
|
case PROP_STATE:
|
|
g_value_set_enum (value, priv->state);
|
|
break;
|
|
|
|
case PROP_PROPERTIES:
|
|
g_value_set_boxed (value, priv->properties);
|
|
break;
|
|
|
|
case PROP_NODE:
|
|
g_value_set_pointer (value, node->node);
|
|
break;
|
|
|
|
case PROP_NODE_STATE:
|
|
g_value_set_uint (value, node->node_state);
|
|
break;
|
|
|
|
default:
|
|
G_OBJECT_WARN_INVALID_PROPERTY_ID (node, prop_id, pspec);
|
|
break;
|
|
}
|
|
}
|
|
|
|
static void
|
|
pinos_node_set_property (GObject *_object,
|
|
guint prop_id,
|
|
const GValue *value,
|
|
GParamSpec *pspec)
|
|
{
|
|
PinosNode *node = PINOS_NODE (_object);
|
|
PinosNodePrivate *priv = node->priv;
|
|
|
|
switch (prop_id) {
|
|
case PROP_DAEMON:
|
|
priv->daemon = g_value_dup_object (value);
|
|
break;
|
|
|
|
case PROP_SENDER:
|
|
priv->sender = g_value_dup_string (value);
|
|
break;
|
|
|
|
case PROP_NAME:
|
|
priv->name = g_value_dup_string (value);
|
|
break;
|
|
|
|
case PROP_PROPERTIES:
|
|
if (priv->properties)
|
|
pinos_properties_free (priv->properties);
|
|
priv->properties = g_value_dup_boxed (value);
|
|
break;
|
|
|
|
case PROP_NODE:
|
|
node->node = g_value_get_pointer (value);
|
|
break;
|
|
|
|
default:
|
|
G_OBJECT_WARN_INVALID_PROPERTY_ID (node, prop_id, pspec);
|
|
break;
|
|
}
|
|
}
|
|
|
|
static void
|
|
node_register_object (PinosNode *node)
|
|
{
|
|
PinosNodePrivate *priv = node->priv;
|
|
PinosDaemon *daemon = priv->daemon;
|
|
PinosObjectSkeleton *skel;
|
|
|
|
skel = pinos_object_skeleton_new (PINOS_DBUS_OBJECT_NODE);
|
|
|
|
pinos_object_skeleton_set_node1 (skel, priv->iface);
|
|
|
|
g_free (priv->object_path);
|
|
priv->object_path = pinos_daemon_export_uniquely (daemon, G_DBUS_OBJECT_SKELETON (skel));
|
|
g_object_unref (skel);
|
|
|
|
g_debug ("node %p: register object %s", node, priv->object_path);
|
|
pinos_daemon_add_node (daemon, node);
|
|
|
|
return;
|
|
}
|
|
|
|
static void
|
|
node_unregister_object (PinosNode *node)
|
|
{
|
|
PinosNodePrivate *priv = node->priv;
|
|
|
|
g_debug ("node %p: unregister object %s", node, priv->object_path);
|
|
pinos_daemon_unexport (priv->daemon, priv->object_path);
|
|
pinos_daemon_remove_node (priv->daemon, node);
|
|
}
|
|
|
|
static void
|
|
on_property_notify (GObject *obj,
|
|
GParamSpec *pspec,
|
|
gpointer user_data)
|
|
{
|
|
PinosNode *node = user_data;
|
|
PinosNodePrivate *priv = node->priv;
|
|
|
|
if (pspec == NULL || strcmp (g_param_spec_get_name (pspec), "sender") == 0) {
|
|
pinos_node1_set_owner (priv->iface, priv->sender);
|
|
}
|
|
if (pspec == NULL || strcmp (g_param_spec_get_name (pspec), "name") == 0) {
|
|
pinos_node1_set_name (priv->iface, pinos_node_get_name (node));
|
|
}
|
|
if (pspec == NULL || strcmp (g_param_spec_get_name (pspec), "properties") == 0) {
|
|
PinosProperties *props = pinos_node_get_properties (node);
|
|
pinos_node1_set_properties (priv->iface, props ? pinos_properties_to_variant (props) : NULL);
|
|
}
|
|
}
|
|
|
|
static void
|
|
pinos_node_constructed (GObject * obj)
|
|
{
|
|
PinosNode *node = PINOS_NODE (obj);
|
|
PinosNodePrivate *priv = node->priv;
|
|
|
|
g_debug ("node %p: constructed", node);
|
|
|
|
g_signal_connect (node, "notify", (GCallback) on_property_notify, node);
|
|
G_OBJECT_CLASS (pinos_node_parent_class)->constructed (obj);
|
|
|
|
if (priv->sender == NULL) {
|
|
priv->sender = g_strdup (pinos_daemon_get_sender (priv->daemon));
|
|
}
|
|
on_property_notify (G_OBJECT (node), NULL, node);
|
|
|
|
node_register_object (node);
|
|
}
|
|
|
|
static void
|
|
pinos_node_dispose (GObject * obj)
|
|
{
|
|
PinosNode *node = PINOS_NODE (obj);
|
|
PinosNodePrivate *priv = node->priv;
|
|
|
|
pinos_node_set_state (node, PINOS_NODE_STATE_SUSPENDED);
|
|
|
|
g_debug ("node %p: dispose", node);
|
|
node_unregister_object (node);
|
|
|
|
g_hash_table_unref (priv->ports);
|
|
|
|
G_OBJECT_CLASS (pinos_node_parent_class)->dispose (obj);
|
|
}
|
|
|
|
static void
|
|
pinos_node_finalize (GObject * obj)
|
|
{
|
|
PinosNode *node = PINOS_NODE (obj);
|
|
PinosNodePrivate *priv = node->priv;
|
|
|
|
g_debug ("node %p: finalize", node);
|
|
g_clear_object (&priv->daemon);
|
|
g_clear_object (&priv->iface);
|
|
g_free (priv->sender);
|
|
g_free (priv->name);
|
|
g_clear_error (&priv->error);
|
|
if (priv->properties)
|
|
pinos_properties_free (priv->properties);
|
|
|
|
G_OBJECT_CLASS (pinos_node_parent_class)->finalize (obj);
|
|
}
|
|
|
|
static void
|
|
pinos_node_class_init (PinosNodeClass * klass)
|
|
{
|
|
GObjectClass *gobject_class = G_OBJECT_CLASS (klass);
|
|
PinosNodeClass *node_class = PINOS_NODE_CLASS (klass);
|
|
|
|
g_type_class_add_private (klass, sizeof (PinosNodePrivate));
|
|
|
|
gobject_class->constructed = pinos_node_constructed;
|
|
gobject_class->dispose = pinos_node_dispose;
|
|
gobject_class->finalize = pinos_node_finalize;
|
|
gobject_class->set_property = pinos_node_set_property;
|
|
gobject_class->get_property = pinos_node_get_property;
|
|
|
|
g_object_class_install_property (gobject_class,
|
|
PROP_DAEMON,
|
|
g_param_spec_object ("daemon",
|
|
"Daemon",
|
|
"The Daemon",
|
|
PINOS_TYPE_DAEMON,
|
|
G_PARAM_READWRITE |
|
|
G_PARAM_CONSTRUCT_ONLY |
|
|
G_PARAM_STATIC_STRINGS));
|
|
|
|
g_object_class_install_property (gobject_class,
|
|
PROP_SENDER,
|
|
g_param_spec_string ("sender",
|
|
"Sender",
|
|
"The Sender",
|
|
NULL,
|
|
G_PARAM_READWRITE |
|
|
G_PARAM_CONSTRUCT_ONLY |
|
|
G_PARAM_STATIC_STRINGS));
|
|
|
|
g_object_class_install_property (gobject_class,
|
|
PROP_OBJECT_PATH,
|
|
g_param_spec_string ("object-path",
|
|
"Object Path",
|
|
"The object path",
|
|
NULL,
|
|
G_PARAM_READABLE |
|
|
G_PARAM_STATIC_STRINGS));
|
|
|
|
g_object_class_install_property (gobject_class,
|
|
PROP_NAME,
|
|
g_param_spec_string ("name",
|
|
"Name",
|
|
"The node name",
|
|
NULL,
|
|
G_PARAM_READWRITE |
|
|
G_PARAM_CONSTRUCT_ONLY |
|
|
G_PARAM_STATIC_STRINGS));
|
|
|
|
g_object_class_install_property (gobject_class,
|
|
PROP_STATE,
|
|
g_param_spec_enum ("state",
|
|
"State",
|
|
"The state of the node",
|
|
PINOS_TYPE_NODE_STATE,
|
|
PINOS_NODE_STATE_SUSPENDED,
|
|
G_PARAM_READABLE |
|
|
G_PARAM_STATIC_STRINGS));
|
|
|
|
g_object_class_install_property (gobject_class,
|
|
PROP_PROPERTIES,
|
|
g_param_spec_boxed ("properties",
|
|
"Properties",
|
|
"The properties of the node",
|
|
PINOS_TYPE_PROPERTIES,
|
|
G_PARAM_READWRITE |
|
|
G_PARAM_CONSTRUCT |
|
|
G_PARAM_STATIC_STRINGS));
|
|
|
|
g_object_class_install_property (gobject_class,
|
|
PROP_NODE,
|
|
g_param_spec_pointer ("node",
|
|
"Node",
|
|
"The SPA node",
|
|
G_PARAM_READWRITE |
|
|
G_PARAM_CONSTRUCT_ONLY |
|
|
G_PARAM_STATIC_STRINGS));
|
|
|
|
g_object_class_install_property (gobject_class,
|
|
PROP_NODE_STATE,
|
|
g_param_spec_uint ("node-state",
|
|
"Node State",
|
|
"The state of the SPA node",
|
|
0,
|
|
G_MAXUINT,
|
|
SPA_NODE_STATE_INIT,
|
|
G_PARAM_READABLE |
|
|
G_PARAM_STATIC_STRINGS));
|
|
|
|
signals[SIGNAL_REMOVE] = g_signal_new ("remove",
|
|
G_TYPE_FROM_CLASS (klass),
|
|
G_SIGNAL_RUN_LAST,
|
|
0,
|
|
NULL,
|
|
NULL,
|
|
g_cclosure_marshal_generic,
|
|
G_TYPE_NONE,
|
|
0,
|
|
G_TYPE_NONE);
|
|
signals[SIGNAL_PORT_ADDED] = g_signal_new ("port-added",
|
|
G_TYPE_FROM_CLASS (klass),
|
|
G_SIGNAL_RUN_LAST,
|
|
0,
|
|
NULL,
|
|
NULL,
|
|
g_cclosure_marshal_generic,
|
|
G_TYPE_NONE,
|
|
1,
|
|
PINOS_TYPE_PORT);
|
|
signals[SIGNAL_PORT_REMOVED] = g_signal_new ("port-removed",
|
|
G_TYPE_FROM_CLASS (klass),
|
|
G_SIGNAL_RUN_LAST,
|
|
0,
|
|
NULL,
|
|
NULL,
|
|
g_cclosure_marshal_generic,
|
|
G_TYPE_NONE,
|
|
1,
|
|
PINOS_TYPE_PORT);
|
|
|
|
node_class->set_state = node_set_state;
|
|
node_class->add_port = node_add_port;
|
|
node_class->remove_port = node_remove_port;
|
|
}
|
|
|
|
static void
|
|
pinos_node_init (PinosNode * node)
|
|
{
|
|
PinosNodePrivate *priv = node->priv = PINOS_NODE_GET_PRIVATE (node);
|
|
|
|
g_debug ("node %p: new", node);
|
|
priv->iface = pinos_node1_skeleton_new ();
|
|
g_signal_connect (priv->iface, "handle-add-port",
|
|
(GCallback) handle_add_port,
|
|
node);
|
|
g_signal_connect (priv->iface, "handle-remove-port",
|
|
(GCallback) handle_remove_port,
|
|
node);
|
|
g_signal_connect (priv->iface, "handle-remove",
|
|
(GCallback) handle_remove,
|
|
node);
|
|
priv->state = PINOS_NODE_STATE_SUSPENDED;
|
|
pinos_node1_set_state (priv->iface, PINOS_NODE_STATE_SUSPENDED);
|
|
|
|
priv->ports = g_hash_table_new_full (g_direct_hash,
|
|
g_direct_equal,
|
|
NULL,
|
|
(GDestroyNotify) g_object_unref);
|
|
}
|
|
|
|
/**
|
|
* pinos_node_new:
|
|
* @daemon: a #PinosDaemon
|
|
* @sender: the path of the owner
|
|
* @name: a name
|
|
* @properties: extra properties
|
|
*
|
|
* Create a new #PinosNode.
|
|
*
|
|
* Returns: a new #PinosNode
|
|
*/
|
|
PinosNode *
|
|
pinos_node_new (PinosDaemon *daemon,
|
|
const gchar *sender,
|
|
const gchar *name,
|
|
PinosProperties *properties,
|
|
SpaNode *node)
|
|
{
|
|
g_return_val_if_fail (PINOS_IS_DAEMON (daemon), NULL);
|
|
|
|
return g_object_new (PINOS_TYPE_NODE,
|
|
"daemon", daemon,
|
|
"sender", sender,
|
|
"name", name,
|
|
"properties", properties,
|
|
"node", node,
|
|
NULL);
|
|
}
|
|
|
|
/**
|
|
* pinos_node_get_name:
|
|
* @node: a #PinosNode
|
|
*
|
|
* Get the name of @node
|
|
*
|
|
* Returns: the name of @node
|
|
*/
|
|
const gchar *
|
|
pinos_node_get_name (PinosNode *node)
|
|
{
|
|
PinosNodePrivate *priv;
|
|
|
|
g_return_val_if_fail (PINOS_IS_NODE (node), NULL);
|
|
priv = node->priv;
|
|
|
|
return priv->name;
|
|
}
|
|
|
|
/**
|
|
* pinos_node_get_state:
|
|
* @node: a #PinosNode
|
|
*
|
|
* Get the state of @node
|
|
*
|
|
* Returns: the state of @node
|
|
*/
|
|
PinosNodeState
|
|
pinos_node_get_state (PinosNode *node)
|
|
{
|
|
PinosNodePrivate *priv;
|
|
|
|
g_return_val_if_fail (PINOS_IS_NODE (node), PINOS_NODE_STATE_ERROR);
|
|
priv = node->priv;
|
|
|
|
return priv->state;
|
|
}
|
|
|
|
/**
|
|
* pinos_node_get_properties:
|
|
* @node: a #PinosNode
|
|
*
|
|
* Get the properties of @node
|
|
*
|
|
* Returns: the properties of @node
|
|
*/
|
|
PinosProperties *
|
|
pinos_node_get_properties (PinosNode *node)
|
|
{
|
|
PinosNodePrivate *priv;
|
|
|
|
g_return_val_if_fail (PINOS_IS_NODE (node), NULL);
|
|
priv = node->priv;
|
|
|
|
return priv->properties;
|
|
}
|
|
|
|
/**
|
|
* pinos_node_get_daemon:
|
|
* @node: a #PinosNode
|
|
*
|
|
* Get the daemon of @node.
|
|
*
|
|
* Returns: the daemon of @node.
|
|
*/
|
|
PinosDaemon *
|
|
pinos_node_get_daemon (PinosNode *node)
|
|
{
|
|
PinosNodePrivate *priv;
|
|
|
|
g_return_val_if_fail (PINOS_IS_NODE (node), NULL);
|
|
priv = node->priv;
|
|
|
|
return priv->daemon;
|
|
}
|
|
|
|
/**
|
|
* pinos_node_get_sender:
|
|
* @node: a #PinosNode
|
|
*
|
|
* Get the owner path of @node.
|
|
*
|
|
* Returns: the owner path of @node.
|
|
*/
|
|
const gchar *
|
|
pinos_node_get_sender (PinosNode *node)
|
|
{
|
|
PinosNodePrivate *priv;
|
|
|
|
g_return_val_if_fail (PINOS_IS_NODE (node), NULL);
|
|
priv = node->priv;
|
|
|
|
return priv->sender;
|
|
}
|
|
/**
|
|
* pinos_node_get_object_path:
|
|
* @node: a #PinosNode
|
|
*
|
|
* Get the object path of @node.
|
|
*
|
|
* Returns: the object path of @node.
|
|
*/
|
|
const gchar *
|
|
pinos_node_get_object_path (PinosNode *node)
|
|
{
|
|
PinosNodePrivate *priv;
|
|
|
|
g_return_val_if_fail (PINOS_IS_NODE (node), NULL);
|
|
priv = node->priv;
|
|
|
|
return priv->object_path;
|
|
}
|
|
|
|
/**
|
|
* pinos_node_remove:
|
|
* @node: a #PinosNode
|
|
*
|
|
* Remove @node. This will stop the transfer on the node and
|
|
* free the resources allocated by @node.
|
|
*/
|
|
void
|
|
pinos_node_remove (PinosNode *node)
|
|
{
|
|
g_return_if_fail (PINOS_IS_NODE (node));
|
|
|
|
g_debug ("node %p: remove", node);
|
|
g_signal_emit (node, signals[SIGNAL_REMOVE], 0, NULL);
|
|
}
|
|
|
|
/**
|
|
* pinos_node_add_port:
|
|
* @node: a #PinosNode
|
|
* @direction: the direction of the port
|
|
* @error: location of #GError
|
|
*
|
|
* Add the #PinosPort to @node
|
|
*
|
|
* Returns: a new #PinosPort or %NULL
|
|
*/
|
|
PinosPort *
|
|
pinos_node_add_port (PinosNode *node,
|
|
PinosDirection direction,
|
|
guint id,
|
|
GError **error)
|
|
{
|
|
PinosNodeClass *klass;
|
|
PinosPort *port;
|
|
|
|
g_return_val_if_fail (PINOS_IS_NODE (node), NULL);
|
|
|
|
klass = PINOS_NODE_GET_CLASS (node);
|
|
if (!klass->add_port) {
|
|
g_set_error (error, G_IO_ERROR, G_IO_ERROR_NOT_SUPPORTED, "add-port not implemented");
|
|
return NULL;
|
|
}
|
|
|
|
g_debug ("node %p: add port", node);
|
|
port = klass->add_port (node, direction, id, error);
|
|
|
|
return port;
|
|
}
|
|
|
|
/**
|
|
* pinos_node_remove_port:
|
|
* @node: a #PinosNode
|
|
* @id: a #PinosPort id
|
|
*
|
|
* Remove the #PinosPort with @id from @node
|
|
*
|
|
* Returns: %TRUE when the port was removed
|
|
*/
|
|
gboolean
|
|
pinos_node_remove_port (PinosNode *node, guint id)
|
|
{
|
|
PinosNodeClass *klass;
|
|
gboolean res = FALSE;
|
|
|
|
g_return_val_if_fail (PINOS_IS_NODE (node), FALSE);
|
|
|
|
klass = PINOS_NODE_GET_CLASS (node);
|
|
|
|
if (!klass->remove_port)
|
|
return FALSE;
|
|
|
|
res = klass->remove_port (node, id);
|
|
return res;
|
|
}
|
|
|
|
/**
|
|
* pinos_node_find_port:
|
|
* @node: a #PinosNode
|
|
* @id: a #PinosPort id
|
|
*
|
|
* Get the port with @id @node.
|
|
*
|
|
* Returns: a #PinosPort with @id or %NULL when not found
|
|
*/
|
|
PinosPort *
|
|
pinos_node_find_port (PinosNode *node, guint id)
|
|
{
|
|
PinosNodePrivate *priv;
|
|
|
|
g_return_val_if_fail (PINOS_IS_NODE (node), NULL);
|
|
priv = node->priv;
|
|
|
|
return g_hash_table_lookup (priv->ports, GUINT_TO_POINTER (id));
|
|
}
|
|
|
|
/**
|
|
* pinos_node_get_ports:
|
|
* @node: a #PinosNode
|
|
*
|
|
* Get the ports in @node.
|
|
*
|
|
* Returns: a #GList of ports g_list_free after usage.
|
|
*/
|
|
GList *
|
|
pinos_node_get_ports (PinosNode *node)
|
|
{
|
|
PinosNodePrivate *priv;
|
|
|
|
g_return_val_if_fail (PINOS_IS_NODE (node), NULL);
|
|
priv = node->priv;
|
|
|
|
return g_hash_table_get_values (priv->ports);
|
|
}
|
|
|
|
static void
|
|
remove_idle_timeout (PinosNode *node)
|
|
{
|
|
PinosNodePrivate *priv = node->priv;
|
|
|
|
if (priv->idle_timeout) {
|
|
g_source_remove (priv->idle_timeout);
|
|
priv->idle_timeout = 0;
|
|
}
|
|
}
|
|
|
|
/**
|
|
* pinos_node_set_state:
|
|
* @node: a #PinosNode
|
|
* @state: a #PinosNodeState
|
|
*
|
|
* Set the state of @node to @state.
|
|
*
|
|
* Returns: %TRUE on success.
|
|
*/
|
|
gboolean
|
|
pinos_node_set_state (PinosNode *node,
|
|
PinosNodeState state)
|
|
{
|
|
PinosNodeClass *klass;
|
|
gboolean res;
|
|
|
|
g_return_val_if_fail (PINOS_IS_NODE (node), FALSE);
|
|
|
|
klass = PINOS_NODE_GET_CLASS (node);
|
|
|
|
remove_idle_timeout (node);
|
|
|
|
g_debug ("node %p: set state to %s", node, pinos_node_state_as_string (state));
|
|
if (klass->set_state)
|
|
res = klass->set_state (node, state);
|
|
else
|
|
res = FALSE;
|
|
|
|
return res;
|
|
}
|
|
|
|
/**
|
|
* pinos_node_update_state:
|
|
* @node: a #PinosNode
|
|
* @state: a #PinosNodeState
|
|
*
|
|
* Update the state of a node. This method is used from
|
|
* inside @node itself.
|
|
*/
|
|
void
|
|
pinos_node_update_state (PinosNode *node,
|
|
PinosNodeState state)
|
|
{
|
|
PinosNodePrivate *priv;
|
|
|
|
g_return_if_fail (PINOS_IS_NODE (node));
|
|
priv = node->priv;
|
|
|
|
if (priv->state != state) {
|
|
g_debug ("node %p: update state to %s", node, pinos_node_state_as_string (state));
|
|
priv->state = state;
|
|
pinos_node1_set_state (priv->iface, state);
|
|
g_object_notify (G_OBJECT (node), "state");
|
|
}
|
|
}
|
|
|
|
/**
|
|
* pinos_node_report_error:
|
|
* @node: a #PinosNode
|
|
* @error: a #GError
|
|
*
|
|
* Report an error from within @node.
|
|
*/
|
|
void
|
|
pinos_node_report_error (PinosNode *node,
|
|
GError *error)
|
|
{
|
|
PinosNodePrivate *priv;
|
|
|
|
g_return_if_fail (PINOS_IS_NODE (node));
|
|
priv = node->priv;
|
|
|
|
g_clear_error (&priv->error);
|
|
remove_idle_timeout (node);
|
|
priv->error = error;
|
|
priv->state = PINOS_NODE_STATE_ERROR;
|
|
g_debug ("node %p: got error state %s", node, error->message);
|
|
g_object_notify (G_OBJECT (node), "state");
|
|
}
|
|
|
|
static gboolean
|
|
idle_timeout (PinosNode *node)
|
|
{
|
|
PinosNodePrivate *priv = node->priv;
|
|
|
|
priv->idle_timeout = 0;
|
|
g_debug ("node %p: idle timeout", node);
|
|
pinos_node_set_state (node, PINOS_NODE_STATE_SUSPENDED);
|
|
|
|
return G_SOURCE_REMOVE;
|
|
}
|
|
|
|
/**
|
|
* pinos_node_report_idle:
|
|
* @node: a #PinosNode
|
|
*
|
|
* Mark @node as being idle. This will start a timeout that will
|
|
* set the node to SUSPENDED.
|
|
*/
|
|
void
|
|
pinos_node_report_idle (PinosNode *node)
|
|
{
|
|
PinosNodePrivate *priv;
|
|
|
|
g_return_if_fail (PINOS_IS_NODE (node));
|
|
priv = node->priv;
|
|
|
|
g_debug ("node %p: report idle", node);
|
|
pinos_node_set_state (node, PINOS_NODE_STATE_IDLE);
|
|
|
|
priv->idle_timeout = g_timeout_add_seconds (3,
|
|
(GSourceFunc) idle_timeout,
|
|
node);
|
|
}
|
|
|
|
/**
|
|
* pinos_node_report_busy:
|
|
* @node: a #PinosNode
|
|
*
|
|
* Mark @node as being busy. This will set the state of the node
|
|
* to the RUNNING state.
|
|
*/
|
|
void
|
|
pinos_node_report_busy (PinosNode *node)
|
|
{
|
|
g_return_if_fail (PINOS_IS_NODE (node));
|
|
|
|
g_debug ("node %p: report busy", node);
|
|
pinos_node_set_state (node, PINOS_NODE_STATE_RUNNING);
|
|
}
|
|
|
|
/**
|
|
* pinos_node_update_node_state:
|
|
* @node: a #PinosNode
|
|
* @state: a #SpaNodeState
|
|
*
|
|
* Update the state of a SPA node. This method is used from
|
|
* inside @node itself.
|
|
*/
|
|
void
|
|
pinos_node_update_node_state (PinosNode *node,
|
|
SpaNodeState state)
|
|
{
|
|
g_return_if_fail (PINOS_IS_NODE (node));
|
|
|
|
if (node->node_state != state) {
|
|
g_debug ("node %p: update SPA state to %d", node, state);
|
|
node->node_state = state;
|
|
g_object_notify (G_OBJECT (node), "node-state");
|
|
}
|
|
}
|