wireplumber/tests/modules/si-standard-link.c
2021-03-26 16:16:02 -04:00

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 ();
}