mirror of
https://gitlab.freedesktop.org/pipewire/wireplumber.git
synced 2025-12-22 16:50:19 +01:00
463 lines
16 KiB
C
463 lines
16 KiB
C
/* WirePlumber
|
|
*
|
|
* Copyright © 2020 Collabora Ltd.
|
|
* @author George Kiagiadakis <george.kiagiadakis@collabora.com>
|
|
*
|
|
* SPDX-License-Identifier: MIT
|
|
*/
|
|
|
|
#include "../common/base-test-fixture.h"
|
|
|
|
typedef struct {
|
|
WpBaseTestFixture base;
|
|
|
|
WpSession *session;
|
|
WpSessionItem *src_item;
|
|
WpSessionItem *sink_item;
|
|
gint activation_state;
|
|
|
|
} TestFixture;
|
|
|
|
static WpSessionItem *
|
|
load_endpoint (TestFixture * f, const gchar * factory, const gchar * media_class)
|
|
{
|
|
g_autoptr (WpNode) target_node = NULL;
|
|
g_autoptr (WpSessionItem) target = NULL;
|
|
g_autoptr (WpSessionItem) endpoint = NULL;
|
|
|
|
/* create target node */
|
|
|
|
target_node = wp_node_new_from_factory (f->base.core,
|
|
"adapter",
|
|
wp_properties_new (
|
|
"factory.name", factory,
|
|
"node.name", factory,
|
|
NULL));
|
|
g_assert_nonnull (target_node);
|
|
|
|
/* create target */
|
|
|
|
target = wp_session_item_make (f->base.core, "si-audio-adapter");
|
|
g_assert_nonnull (target);
|
|
g_assert_true (WP_IS_SI_PORT_INFO (target));
|
|
|
|
/* configure target */
|
|
|
|
{
|
|
WpProperties *props = wp_properties_new_empty ();
|
|
wp_properties_setf (props, "node", "%p", target_node);
|
|
wp_properties_set (props, "name", factory);
|
|
wp_properties_set (props, "media-class", media_class);
|
|
wp_properties_set (props, "role", "role");
|
|
g_assert_true (wp_session_item_configure (target, props));
|
|
g_assert_true (wp_session_item_is_configured (target));
|
|
}
|
|
|
|
/* create endpoint */
|
|
|
|
endpoint = wp_session_item_make (f->base.core, "si-audio-endpoint");
|
|
g_assert_nonnull (endpoint);
|
|
g_assert_true (WP_IS_SI_ENDPOINT (endpoint));
|
|
|
|
/* configure endpoint */
|
|
{
|
|
WpProperties *props = wp_properties_new_empty ();
|
|
wp_properties_set (props, "name", factory);
|
|
wp_properties_setf (props, "target", "%p", target);
|
|
wp_properties_setf (props, "session", "%p", f->session);
|
|
g_assert_true (wp_session_item_configure (endpoint, props));
|
|
g_assert_true (wp_session_item_is_configured (endpoint));
|
|
}
|
|
|
|
/* activate and export endpoint */
|
|
|
|
wp_object_activate (WP_OBJECT (endpoint),
|
|
WP_SESSION_ITEM_FEATURE_ACTIVE | WP_SESSION_ITEM_FEATURE_EXPORTED,
|
|
NULL, (GAsyncReadyCallback) test_object_activate_finish_cb, f);
|
|
g_main_loop_run (f->base.loop);
|
|
g_assert_cmphex (wp_object_get_active_features (WP_OBJECT (endpoint)), ==,
|
|
WP_SESSION_ITEM_FEATURE_ACTIVE | WP_SESSION_ITEM_FEATURE_EXPORTED);
|
|
|
|
return g_steal_pointer (&endpoint);
|
|
}
|
|
|
|
static void
|
|
test_si_standard_link_setup (TestFixture * f, gconstpointer user_data)
|
|
{
|
|
wp_base_test_fixture_setup (&f->base, WP_BASE_TEST_FLAG_CLIENT_CORE);
|
|
|
|
/* load modules */
|
|
{
|
|
g_autoptr (WpTestServerLocker) lock =
|
|
wp_test_server_locker_new (&f->base.server);
|
|
|
|
g_assert_cmpint (pw_context_add_spa_lib (f->base.server.context,
|
|
"fake*", "test/libspa-test"), ==, 0);
|
|
g_assert_cmpint (pw_context_add_spa_lib (f->base.server.context,
|
|
"audiotestsrc", "audiotestsrc/libspa-audiotestsrc"), ==, 0);
|
|
g_assert_nonnull (pw_context_load_module (f->base.server.context,
|
|
"libpipewire-module-spa-node-factory", NULL, NULL));
|
|
g_assert_nonnull (pw_context_load_module (f->base.server.context,
|
|
"libpipewire-module-link-factory", NULL, NULL));
|
|
}
|
|
{
|
|
g_autoptr (GError) error = NULL;
|
|
wp_core_load_component (f->base.core,
|
|
"libwireplumber-module-si-audio-adapter", "module", NULL, &error);
|
|
g_assert_no_error (error);
|
|
|
|
wp_core_load_component (f->base.core,
|
|
"libwireplumber-module-si-audio-endpoint", "module", NULL, &error);
|
|
g_assert_no_error (error);
|
|
|
|
wp_core_load_component (f->base.core,
|
|
"libwireplumber-module-si-standard-link", "module", NULL, &error);
|
|
g_assert_no_error (error);
|
|
}
|
|
|
|
g_assert_nonnull (
|
|
f->session = WP_SESSION (wp_impl_session_new (f->base.core)));
|
|
wp_impl_session_set_property (WP_IMPL_SESSION (f->session),
|
|
"session.name", "audio");
|
|
wp_object_activate (WP_OBJECT (f->session), WP_OBJECT_FEATURES_ALL, NULL,
|
|
(GAsyncReadyCallback) test_object_activate_finish_cb, f);
|
|
g_main_loop_run (f->base.loop);
|
|
|
|
f->src_item = load_endpoint (f, "audiotestsrc", "Audio/Source");
|
|
f->sink_item = load_endpoint (f, "fakesink", "Audio/Sink");
|
|
}
|
|
|
|
static void
|
|
test_si_standard_link_teardown (TestFixture * f, gconstpointer user_data)
|
|
{
|
|
g_clear_object (&f->sink_item);
|
|
g_clear_object (&f->src_item);
|
|
g_clear_object (&f->session);
|
|
wp_base_test_fixture_teardown (&f->base);
|
|
}
|
|
|
|
static void
|
|
on_link_state_changed (WpEndpointLink * link, WpEndpointLinkState old,
|
|
WpEndpointLinkState new, const gchar * error, TestFixture * f)
|
|
{
|
|
g_assert_null (error);
|
|
switch (f->activation_state++) {
|
|
case 0:
|
|
g_assert_cmpuint (old, ==, WP_ENDPOINT_LINK_STATE_INACTIVE);
|
|
g_assert_cmpuint (new, ==, WP_ENDPOINT_LINK_STATE_ACTIVE);
|
|
g_main_loop_quit (f->base.loop);
|
|
break;
|
|
case 1:
|
|
g_assert_cmpuint (old, ==, WP_ENDPOINT_LINK_STATE_ACTIVE);
|
|
g_assert_cmpuint (new, ==, WP_ENDPOINT_LINK_STATE_INACTIVE);
|
|
g_main_loop_quit (f->base.loop);
|
|
break;
|
|
default:
|
|
g_assert_not_reached ();
|
|
}
|
|
}
|
|
|
|
static void
|
|
test_si_standard_link_main (TestFixture * f, gconstpointer user_data)
|
|
{
|
|
g_autoptr (WpSession) session_proxy = NULL;
|
|
g_autoptr (WpEndpoint) src_ep = NULL;
|
|
g_autoptr (WpEndpoint) sink_ep = NULL;
|
|
g_autoptr (WpEndpointLink) ep_link = NULL;
|
|
|
|
/* find the "audio" session from the client */
|
|
{
|
|
g_autoptr (WpObjectManager) om = wp_object_manager_new ();
|
|
wp_object_manager_add_interest (om, WP_TYPE_SESSION, NULL);
|
|
wp_object_manager_request_object_features (om, WP_TYPE_SESSION,
|
|
WP_OBJECT_FEATURES_ALL);
|
|
test_ensure_object_manager_is_installed (om, f->base.client_core,
|
|
f->base.loop);
|
|
|
|
g_assert_nonnull (session_proxy =
|
|
wp_object_manager_lookup (om, WP_TYPE_SESSION,
|
|
WP_CONSTRAINT_TYPE_PW_PROPERTY, "session.name", "=s", "audio", NULL));
|
|
g_assert_cmpint (wp_proxy_get_bound_id (WP_PROXY (session_proxy)), ==,
|
|
wp_proxy_get_bound_id (WP_PROXY (f->session)));
|
|
}
|
|
|
|
/* find the endpoints */
|
|
|
|
g_assert_nonnull (src_ep = wp_session_lookup_endpoint (session_proxy,
|
|
WP_CONSTRAINT_TYPE_PW_PROPERTY, "endpoint.name", "=s", "audiotestsrc",
|
|
NULL));
|
|
g_assert_nonnull (sink_ep = wp_session_lookup_endpoint (session_proxy,
|
|
WP_CONSTRAINT_TYPE_PW_PROPERTY, "endpoint.name", "=s", "fakesink",
|
|
NULL));
|
|
|
|
/* create the link */
|
|
{
|
|
g_autoptr (WpProperties) props = NULL;
|
|
g_autofree gchar * id =
|
|
g_strdup_printf ("%u", wp_proxy_get_bound_id (WP_PROXY (sink_ep)));
|
|
|
|
/* only the peer endpoint id is required,
|
|
everything else will be discovered */
|
|
props = wp_properties_new ("endpoint-link.input.endpoint", id, NULL);
|
|
wp_endpoint_create_link (src_ep, props);
|
|
}
|
|
|
|
g_signal_connect_swapped (session_proxy, "links-changed",
|
|
G_CALLBACK (g_main_loop_quit), f->base.loop);
|
|
g_main_loop_run (f->base.loop);
|
|
|
|
/* verify */
|
|
|
|
g_assert_cmpuint (wp_session_get_n_links (session_proxy), ==, 1);
|
|
g_assert_nonnull (ep_link = wp_session_lookup_link (session_proxy, NULL));
|
|
|
|
{
|
|
guint32 out_ep, in_ep;
|
|
|
|
wp_endpoint_link_get_linked_object_ids (ep_link, &out_ep, &in_ep);
|
|
g_assert_cmpuint (out_ep, ==, wp_proxy_get_bound_id (WP_PROXY (src_ep)));
|
|
g_assert_cmpuint (in_ep, ==, wp_proxy_get_bound_id (WP_PROXY (sink_ep)));
|
|
}
|
|
|
|
{
|
|
g_autoptr (WpProperties) p =
|
|
wp_pipewire_object_get_properties (WP_PIPEWIRE_OBJECT (ep_link));
|
|
|
|
g_assert_nonnull (p);
|
|
g_assert_nonnull (wp_properties_get (p, "endpoint-link.input.endpoint"));
|
|
g_assert_nonnull (wp_properties_get (p, "endpoint-link.output.endpoint"));
|
|
}
|
|
|
|
{
|
|
const gchar *error = NULL;
|
|
g_assert_cmpuint (wp_endpoint_link_get_state (ep_link, &error), ==,
|
|
WP_ENDPOINT_LINK_STATE_INACTIVE);
|
|
g_assert_null (error);
|
|
}
|
|
|
|
/* activate */
|
|
|
|
g_signal_connect (ep_link, "state-changed",
|
|
G_CALLBACK (on_link_state_changed), f);
|
|
wp_endpoint_link_request_state (ep_link, WP_ENDPOINT_LINK_STATE_ACTIVE);
|
|
g_main_loop_run (f->base.loop);
|
|
|
|
{
|
|
const gchar *error = NULL;
|
|
g_assert_cmpuint (wp_endpoint_link_get_state (ep_link, &error), ==,
|
|
WP_ENDPOINT_LINK_STATE_ACTIVE);
|
|
g_assert_null (error);
|
|
g_assert_cmpint (f->activation_state, ==, 1);
|
|
}
|
|
|
|
/* verify the graph state */
|
|
{
|
|
g_autoptr (WpNode) out_node = NULL;
|
|
g_autoptr (WpNode) in_node = NULL;
|
|
g_autoptr (WpIterator) it = NULL;
|
|
g_auto (GValue) val = G_VALUE_INIT;
|
|
g_autoptr (WpObjectManager) om = wp_object_manager_new ();
|
|
guint total_links = 0;
|
|
|
|
wp_object_manager_add_interest (om, WP_TYPE_NODE, NULL);
|
|
wp_object_manager_add_interest (om, WP_TYPE_PORT, NULL);
|
|
wp_object_manager_add_interest (om, WP_TYPE_LINK, NULL);
|
|
wp_object_manager_request_object_features (om, WP_TYPE_PROXY,
|
|
WP_PIPEWIRE_OBJECT_FEATURES_MINIMAL);
|
|
test_ensure_object_manager_is_installed (om, f->base.client_core,
|
|
f->base.loop);
|
|
|
|
g_assert_nonnull (out_node = wp_object_manager_lookup (om, WP_TYPE_NODE,
|
|
WP_CONSTRAINT_TYPE_PW_PROPERTY, "node.name", "=s", "control.audiotestsrc",
|
|
NULL));
|
|
g_assert_nonnull (in_node = wp_object_manager_lookup (om, WP_TYPE_NODE,
|
|
WP_CONSTRAINT_TYPE_PW_PROPERTY, "node.name", "=s", "control.fakesink",
|
|
NULL));
|
|
g_assert_cmpuint (wp_object_manager_get_n_objects (om), ==, 11);
|
|
|
|
it = wp_object_manager_new_filtered_iterator (om, WP_TYPE_LINK, NULL);
|
|
for (; wp_iterator_next (it, &val); g_value_unset (&val)) {
|
|
guint32 out_nd_id, out_pt_id, in_nd_id, in_pt_id;
|
|
g_autoptr (WpPort) out_port = NULL;
|
|
g_autoptr (WpPort) in_port = NULL;
|
|
WpLink *link = g_value_get_object (&val);
|
|
wp_link_get_linked_object_ids (link, &out_nd_id, &out_pt_id, &in_nd_id,
|
|
&in_pt_id);
|
|
g_assert_cmpuint (out_nd_id, ==, wp_proxy_get_bound_id (WP_PROXY (out_node)));
|
|
g_assert_cmpuint (in_nd_id, ==, wp_proxy_get_bound_id (WP_PROXY (in_node)));
|
|
g_assert_nonnull (out_port = wp_object_manager_lookup (om, WP_TYPE_PORT,
|
|
WP_CONSTRAINT_TYPE_G_PROPERTY, "bound-id", "=u", out_pt_id,
|
|
WP_CONSTRAINT_TYPE_PW_PROPERTY, "node.id", "=u", out_nd_id,
|
|
NULL));
|
|
g_assert_nonnull (in_port = wp_object_manager_lookup (om, WP_TYPE_PORT,
|
|
WP_CONSTRAINT_TYPE_G_PROPERTY, "bound-id", "=u", in_pt_id,
|
|
WP_CONSTRAINT_TYPE_PW_PROPERTY, "node.id", "=u", in_nd_id,
|
|
NULL));
|
|
total_links++;
|
|
}
|
|
g_assert_cmpuint (total_links, ==, 2);
|
|
}
|
|
|
|
/* deactivate */
|
|
|
|
wp_endpoint_link_request_state (ep_link, WP_ENDPOINT_LINK_STATE_INACTIVE);
|
|
g_main_loop_run (f->base.loop);
|
|
|
|
{
|
|
const gchar *error = NULL;
|
|
g_assert_cmpuint (wp_endpoint_link_get_state (ep_link, &error), ==,
|
|
WP_ENDPOINT_LINK_STATE_INACTIVE);
|
|
g_assert_null (error);
|
|
g_assert_cmpint (f->activation_state, ==, 2);
|
|
}
|
|
|
|
/* verify the graph state */
|
|
{
|
|
g_autoptr (WpNode) out_node = NULL;
|
|
g_autoptr (WpNode) in_node = NULL;
|
|
g_autoptr (WpPort) out_port = NULL;
|
|
g_autoptr (WpPort) in_port = NULL;
|
|
g_autoptr (WpLink) link = NULL;
|
|
g_autoptr (WpObjectManager) om = wp_object_manager_new ();
|
|
|
|
wp_object_manager_add_interest (om, WP_TYPE_NODE, NULL);
|
|
wp_object_manager_add_interest (om, WP_TYPE_PORT, NULL);
|
|
wp_object_manager_add_interest (om, WP_TYPE_LINK, NULL);
|
|
wp_object_manager_request_object_features (om, WP_TYPE_PROXY,
|
|
WP_PIPEWIRE_OBJECT_FEATURES_MINIMAL);
|
|
test_ensure_object_manager_is_installed (om, f->base.client_core,
|
|
f->base.loop);
|
|
|
|
g_assert_nonnull (out_node = wp_object_manager_lookup (om, WP_TYPE_NODE,
|
|
WP_CONSTRAINT_TYPE_PW_PROPERTY, "node.name", "=s", "control.audiotestsrc",
|
|
NULL));
|
|
g_assert_nonnull (in_node = wp_object_manager_lookup (om, WP_TYPE_NODE,
|
|
WP_CONSTRAINT_TYPE_PW_PROPERTY, "node.name", "=s", "control.fakesink",
|
|
NULL));
|
|
g_assert_nonnull (out_port = wp_object_manager_lookup (om, WP_TYPE_PORT,
|
|
WP_CONSTRAINT_TYPE_PW_PROPERTY, "port.direction", "=s", "out",
|
|
NULL));
|
|
g_assert_nonnull (in_port = wp_object_manager_lookup (om, WP_TYPE_PORT,
|
|
WP_CONSTRAINT_TYPE_PW_PROPERTY, "port.direction", "=s", "in",
|
|
NULL));
|
|
g_assert_null (link = wp_object_manager_lookup (om, WP_TYPE_LINK, NULL));
|
|
g_assert_cmpuint (wp_object_manager_get_n_objects (om), ==, 9);
|
|
}
|
|
}
|
|
|
|
static void
|
|
on_link_destroyed (WpEndpointLink * link, TestFixture * f)
|
|
{
|
|
f->activation_state = 10;
|
|
}
|
|
|
|
static void
|
|
test_si_standard_link_destroy (TestFixture * f, gconstpointer user_data)
|
|
{
|
|
g_autoptr (WpSession) session_proxy = NULL;
|
|
g_autoptr (WpEndpoint) src_ep = NULL;
|
|
g_autoptr (WpEndpoint) sink_ep = NULL;
|
|
g_autoptr (WpEndpointLink) ep_link = NULL;
|
|
|
|
/* find the "audio" session from the client */
|
|
{
|
|
g_autoptr (WpObjectManager) om = wp_object_manager_new ();
|
|
wp_object_manager_add_interest (om, WP_TYPE_SESSION, NULL);
|
|
wp_object_manager_request_object_features (om, WP_TYPE_SESSION,
|
|
WP_OBJECT_FEATURES_ALL);
|
|
test_ensure_object_manager_is_installed (om, f->base.client_core,
|
|
f->base.loop);
|
|
|
|
g_assert_nonnull (session_proxy =
|
|
wp_object_manager_lookup (om, WP_TYPE_SESSION,
|
|
WP_CONSTRAINT_TYPE_PW_PROPERTY, "session.name", "=s", "audio", NULL));
|
|
g_assert_cmpint (wp_proxy_get_bound_id (WP_PROXY (session_proxy)), ==,
|
|
wp_proxy_get_bound_id (WP_PROXY (f->session)));
|
|
}
|
|
|
|
/* find the endpoints */
|
|
|
|
g_assert_nonnull (src_ep = wp_session_lookup_endpoint (session_proxy,
|
|
WP_CONSTRAINT_TYPE_PW_PROPERTY, "endpoint.name", "=s", "audiotestsrc",
|
|
NULL));
|
|
g_assert_nonnull (sink_ep = wp_session_lookup_endpoint (session_proxy,
|
|
WP_CONSTRAINT_TYPE_PW_PROPERTY, "endpoint.name", "=s", "fakesink",
|
|
NULL));
|
|
|
|
/* create the link */
|
|
{
|
|
g_autoptr (WpProperties) props = NULL;
|
|
g_autofree gchar * id =
|
|
g_strdup_printf ("%u", wp_proxy_get_bound_id (WP_PROXY (sink_ep)));
|
|
|
|
/* only the peer endpoint id is required,
|
|
everything else will be discovered */
|
|
props = wp_properties_new ("endpoint-link.input.endpoint", id, NULL);
|
|
wp_endpoint_create_link (src_ep, props);
|
|
}
|
|
|
|
g_signal_connect_swapped (session_proxy, "links-changed",
|
|
G_CALLBACK (g_main_loop_quit), f->base.loop);
|
|
g_main_loop_run (f->base.loop);
|
|
|
|
/* verify */
|
|
|
|
g_assert_cmpuint (wp_session_get_n_links (session_proxy), ==, 1);
|
|
g_assert_nonnull (ep_link = wp_session_lookup_link (session_proxy, NULL));
|
|
g_assert_cmpuint (wp_endpoint_link_get_state (ep_link, NULL), ==,
|
|
WP_ENDPOINT_LINK_STATE_INACTIVE);
|
|
|
|
/* activate */
|
|
|
|
g_signal_connect (ep_link, "state-changed",
|
|
G_CALLBACK (on_link_state_changed), f);
|
|
wp_endpoint_link_request_state (ep_link, WP_ENDPOINT_LINK_STATE_ACTIVE);
|
|
g_main_loop_run (f->base.loop);
|
|
g_assert_cmpuint (wp_endpoint_link_get_state (ep_link, NULL), ==,
|
|
WP_ENDPOINT_LINK_STATE_ACTIVE);
|
|
|
|
/* destroy */
|
|
|
|
g_signal_connect (ep_link, "pw-proxy-destroyed",
|
|
G_CALLBACK (on_link_destroyed), f);
|
|
wp_global_proxy_request_destroy (WP_GLOBAL_PROXY (ep_link));
|
|
|
|
/* loop will quit because the "links-changed" signal from the session
|
|
is still connected to quit() from earlier */
|
|
g_main_loop_run (f->base.loop);
|
|
|
|
g_assert_cmpint (f->activation_state, ==, 10);
|
|
g_assert_cmpuint (wp_session_get_n_links (session_proxy), ==, 0);
|
|
g_assert_cmphex (wp_object_get_active_features (WP_OBJECT (ep_link)), ==, 0);
|
|
|
|
/* verify the link was also destroyed on the session manager core */
|
|
{
|
|
g_autoptr (WpObjectManager) om = wp_object_manager_new ();
|
|
|
|
wp_object_manager_add_interest (om, WP_TYPE_ENDPOINT_LINK, NULL);
|
|
test_ensure_object_manager_is_installed (om, f->base.core, f->base.loop);
|
|
|
|
g_assert_cmpuint (wp_object_manager_get_n_objects (om), ==, 0);
|
|
}
|
|
}
|
|
|
|
gint
|
|
main (gint argc, gchar *argv[])
|
|
{
|
|
g_test_init (&argc, &argv, NULL);
|
|
wp_init (WP_INIT_ALL);
|
|
|
|
g_test_add ("/modules/si-standard-link/main",
|
|
TestFixture, NULL,
|
|
test_si_standard_link_setup,
|
|
test_si_standard_link_main,
|
|
test_si_standard_link_teardown);
|
|
|
|
g_test_add ("/modules/si-standard-link/destroy",
|
|
TestFixture, NULL,
|
|
test_si_standard_link_setup,
|
|
test_si_standard_link_destroy,
|
|
test_si_standard_link_teardown);
|
|
|
|
return g_test_run ();
|
|
}
|