Files
wireplumber/tests/modules/reserve-device.c
George Kiagiadakis c7cb193588 core: introduce the notion of provided features on components
Each component can optionally "provide" a feature, which is basically
a string that describes the feature (ex. "support.dbus"). If the
component loads successfully, the feature is marked as provided and
can be tested for its presence with wp_core_test_feature()
2023-06-20 12:39:29 +03:00

340 lines
10 KiB
C

/* WirePlumber
*
* Copyright © 2021 Collabora Ltd.
* @author George Kiagiadakis <george.kiagiadakis@collabora.com>
*
* SPDX-License-Identifier: MIT
*/
#include <wp/wp.h>
#include "../../modules/dbus-connection-state.h"
#include "../common/base-test-fixture.h"
typedef struct {
WpBaseTestFixture base;
GTestDBus *test_dbus;
WpPlugin *dbus_1;
WpPlugin *dbus_2;
WpPlugin *rd_plugin_1;
WpPlugin *rd_plugin_2;
gint expected_rd1_state;
gint expected_rd2_state;
} RdTestFixture;
static void
on_plugin_loaded (WpCore * core, GAsyncResult * res, RdTestFixture *f)
{
gboolean loaded;
GError *error = NULL;
loaded = wp_core_load_component_finish (core, res, &error);
g_assert_no_error (error);
g_assert_true (loaded);
g_main_loop_quit (f->base.loop);
}
static void
test_rd_setup (RdTestFixture *f, gconstpointer data)
{
WpDBusConnectionState state = -1;
wp_base_test_fixture_setup (&f->base,
WP_BASE_TEST_FLAG_CLIENT_CORE | WP_BASE_TEST_FLAG_DONT_CONNECT);
f->test_dbus = g_test_dbus_new (G_TEST_DBUS_NONE);
g_test_dbus_up (f->test_dbus);
{
wp_core_load_component (f->base.core,
"libwireplumber-module-dbus-connection", "module", NULL, NULL, NULL,
(GAsyncReadyCallback) on_plugin_loaded, f);
g_main_loop_run (f->base.loop);
}
{
wp_core_load_component (f->base.client_core,
"libwireplumber-module-dbus-connection", "module", NULL, NULL, NULL,
(GAsyncReadyCallback) on_plugin_loaded, f);
g_main_loop_run (f->base.loop);
}
{
wp_core_load_component (f->base.core,
"libwireplumber-module-reserve-device", "module", NULL, NULL, NULL,
(GAsyncReadyCallback) on_plugin_loaded, f);
g_main_loop_run (f->base.loop);
}
{
wp_core_load_component (f->base.client_core,
"libwireplumber-module-reserve-device", "module", NULL, NULL, NULL,
(GAsyncReadyCallback) on_plugin_loaded, f);
g_main_loop_run (f->base.loop);
}
f->dbus_1 = wp_plugin_find (f->base.core, "dbus-connection");
g_assert_nonnull (f->dbus_1);
f->dbus_2 = wp_plugin_find (f->base.client_core, "dbus-connection");
g_assert_nonnull (f->dbus_2);
f->rd_plugin_1 = wp_plugin_find (f->base.core, "reserve-device");
g_assert_nonnull (f->rd_plugin_1);
f->rd_plugin_2 = wp_plugin_find (f->base.client_core, "reserve-device");
g_assert_nonnull (f->rd_plugin_2);
g_object_get (f->dbus_1, "state", &state, NULL);
g_assert_cmpint (state, ==, WP_DBUS_CONNECTION_STATE_CONNECTED);
g_object_get (f->dbus_2, "state", &state, NULL);
g_assert_cmpint (state, ==, WP_DBUS_CONNECTION_STATE_CONNECTED);
}
static void
test_rd_teardown (RdTestFixture *f, gconstpointer data)
{
g_clear_object (&f->dbus_1);
g_clear_object (&f->dbus_2);
g_clear_object (&f->rd_plugin_1);
g_clear_object (&f->rd_plugin_2);
g_test_dbus_down (f->test_dbus);
g_clear_object (&f->test_dbus);
wp_base_test_fixture_teardown (&f->base);
}
static void
ensure_plugins_stable_state (GObject * obj, GParamSpec * spec, RdTestFixture *f)
{
WpDBusConnectionState state1 = -1, state2 = -1;
g_object_get (f->dbus_1, "state", &state1, NULL);
g_object_get (f->dbus_2, "state", &state2, NULL);
if (state1 != WP_DBUS_CONNECTION_STATE_CONNECTING &&
state2 != WP_DBUS_CONNECTION_STATE_CONNECTING &&
state1 == state2)
g_main_loop_quit (f->base.loop);
}
static void
test_rd_plugin (RdTestFixture *f, gconstpointer data)
{
GObject *rd1 = NULL, *rd2 = NULL, *rd_video = NULL, *tmp = NULL;
gint priority = 0;
gchar *str;
g_signal_emit_by_name (f->rd_plugin_1, "create-reservation",
"Audio0", "WirePlumber", "hw:0,0", 10, &rd1);
g_assert_nonnull (rd1);
g_signal_emit_by_name (f->rd_plugin_2, "create-reservation",
"Audio0", "Other Server", "hw:0,0", 15, &rd2);
g_assert_nonnull (rd2);
g_signal_emit_by_name (f->rd_plugin_1, "create-reservation",
"Video0", "WirePlumber", "/dev/video0", 10, &rd_video);
g_assert_nonnull (rd_video);
g_signal_emit_by_name (f->rd_plugin_1, "get-reservation", "Video1", &tmp);
g_assert_null (tmp);
g_signal_emit_by_name (f->rd_plugin_2, "get-reservation", "Video0", &tmp);
g_assert_null (tmp);
g_signal_emit_by_name (f->rd_plugin_1, "get-reservation", "Audio0", &tmp);
g_assert_nonnull (tmp);
g_assert_true (tmp == rd1);
g_clear_object (&tmp);
g_object_get (rd1, "name", &str, NULL);
g_assert_cmpstr (str, ==, "Audio0");
g_free (str);
g_object_get (rd2, "name", &str, NULL);
g_assert_cmpstr (str, ==, "Audio0");
g_free (str);
g_object_get (rd_video, "name", &str, NULL);
g_assert_cmpstr (str, ==, "Video0");
g_free (str);
g_object_get (rd1, "application-name", &str, NULL);
g_assert_cmpstr (str, ==, "WirePlumber");
g_free (str);
g_object_get (rd1, "application-device-name", &str, NULL);
g_assert_cmpstr (str, ==, "hw:0,0");
g_free (str);
g_object_get (rd1, "priority", &priority, NULL);
g_assert_cmpint (priority, ==, 10);
g_object_get (rd2, "priority", &priority, NULL);
g_assert_cmpint (priority, ==, 15);
g_signal_emit_by_name (f->rd_plugin_1, "destroy-reservation", "Audio0");
g_signal_emit_by_name (f->rd_plugin_1, "get-reservation", "Audio0", &tmp);
g_assert_null (tmp);
g_signal_emit_by_name (f->rd_plugin_2, "get-reservation", "Audio0", &tmp);
g_assert_nonnull (tmp);
g_assert_true (tmp == rd2);
g_clear_object (&tmp);
g_clear_object (&rd2);
g_clear_object (&rd1);
g_clear_object (&rd_video);
}
static void
test_rd_conn_closed (RdTestFixture *f, gconstpointer data)
{
GObject *rd1 = NULL;
WpDBusConnectionState state = -1;
g_signal_connect (f->dbus_1, "notify::state",
G_CALLBACK (ensure_plugins_stable_state), f);
g_signal_connect (f->dbus_2, "notify::state",
G_CALLBACK (ensure_plugins_stable_state), f);
g_signal_emit_by_name (f->rd_plugin_1, "create-reservation",
"Audio0", "WirePlumber", "hw:0,0", 10, &rd1);
g_assert_nonnull (rd1);
g_clear_object (&rd1);
/* stop the bus, expect the connections to close
and state to go back to CLOSED */
g_test_dbus_stop (f->test_dbus);
g_main_loop_run (f->base.loop);
g_object_get (f->dbus_1, "state", &state, NULL);
g_assert_cmpint (state, ==, WP_DBUS_CONNECTION_STATE_CLOSED);
g_object_get (f->dbus_2, "state", &state, NULL);
g_assert_cmpint (state, ==, WP_DBUS_CONNECTION_STATE_CLOSED);
g_signal_emit_by_name (f->rd_plugin_1, "get-reservation", "Audio0", &rd1);
g_assert_null (rd1);
}
static void
expect_rd1_state (GObject * rd, GParamSpec * spec, RdTestFixture *f)
{
gint state;
g_object_get (rd, "state", &state, NULL);
if (state == f->expected_rd1_state)
g_main_loop_quit (f->base.loop);
}
static void
expect_rd2_state (GObject * rd, GParamSpec * spec, RdTestFixture *f)
{
gint state;
g_object_get (rd, "state", &state, NULL);
if (state == f->expected_rd2_state)
g_main_loop_quit (f->base.loop);
}
static void
handle_release_requested (GObject * rd, gboolean forced, RdTestFixture *f)
{
gint state = 0xffff;
g_signal_emit_by_name (rd, "release");
g_object_get (rd, "state", &state, NULL);
g_assert_cmpint (state, ==, 2);
g_main_loop_quit (f->base.loop);
}
static void
test_rd_acquire_release (RdTestFixture *f, gconstpointer data)
{
GObject *rd1 = NULL, *rd2 = NULL;
gint state = 0;
gchar *str = NULL;
g_signal_emit_by_name (f->rd_plugin_1, "create-reservation",
"Audio0", "WirePlumber", "hw:0,0", 10, &rd1);
g_assert_nonnull (rd1);
g_signal_emit_by_name (f->rd_plugin_2, "create-reservation",
"Audio0", "Other Server", "hw:0,0", 15, &rd2);
g_assert_nonnull (rd2);
g_signal_connect (rd1, "notify::state", G_CALLBACK (expect_rd1_state), f);
g_signal_connect (rd2, "notify::state", G_CALLBACK (expect_rd2_state), f);
/* acquire */
wp_info ("rd1 acquire");
f->expected_rd1_state = 3;
g_signal_emit_by_name (rd1, "acquire");
g_main_loop_run (f->base.loop);
g_object_get (rd1, "state", &state, NULL);
g_assert_cmpint (state, ==, 3);
g_object_get (rd1, "owner-application-name", &str, NULL);
g_assert_cmpstr (str, ==, "WirePlumber");
g_free (str);
g_signal_connect (rd1, "release-requested",
G_CALLBACK (handle_release_requested), f);
/* acquire with higher priority */
wp_info ("rd2 acquire, higher prio");
g_signal_emit_by_name (rd2, "acquire");
/* rd1 is now released */
g_main_loop_run (f->base.loop);
g_object_get (rd1, "state", &state, NULL);
g_assert_cmpint (state, ==, 2);
/* rd2 acquired */
f->expected_rd2_state = 3;
g_main_loop_run (f->base.loop);
g_object_get (rd2, "state", &state, NULL);
g_assert_cmpint (state, ==, 3);
/* rd1 busy */
g_signal_connect_swapped (rd1, "notify::owner-application-name",
G_CALLBACK (g_main_loop_quit), f->base.loop);
g_main_loop_run (f->base.loop);
g_object_get (rd1, "state", &state, NULL);
g_assert_cmpint (state, ==, 1);
g_object_get (rd1, "owner-application-name", &str, NULL);
g_assert_cmpstr (str, ==, "Other Server");
g_free (str);
g_signal_handlers_disconnect_by_func (rd1, G_CALLBACK (g_main_loop_quit),
f->base.loop);
/* try to acquire back with lower priority */
wp_info ("rd1 acquire, lower prio");
g_signal_emit_by_name (rd1, "acquire");
/* ... expect this to fail */
f->expected_rd1_state = 1;
g_main_loop_run (f->base.loop);
g_object_get (rd1, "state", &state, NULL);
g_assert_cmpint (state, ==, 1);
g_object_get (rd1, "owner-application-name", &str, NULL);
g_assert_cmpstr (str, ==, "Other Server");
g_free (str);
/* release */
wp_info ("rd2 release");
g_signal_emit_by_name (rd2, "release");
g_object_get (rd2, "state", &state, NULL);
g_assert_cmpint (state, ==, 2);
f->expected_rd1_state = 2;
g_main_loop_run (f->base.loop);
g_object_get (rd1, "state", &state, NULL);
g_assert_cmpint (state, ==, 2);
g_object_get (rd1, "owner-application-name", &str, NULL);
g_assert_null (str);
g_clear_object (&rd1);
g_clear_object (&rd2);
}
gint
main (gint argc, gchar *argv[])
{
g_test_init (&argc, &argv, NULL);
wp_init (WP_INIT_ALL);
g_test_add ("/modules/rd/plugin", RdTestFixture, NULL,
test_rd_setup, test_rd_plugin, test_rd_teardown);
g_test_add ("/modules/rd/conn_closed", RdTestFixture, NULL,
test_rd_setup, test_rd_conn_closed, test_rd_teardown);
g_test_add ("/modules/rd/acquire_release", RdTestFixture, NULL,
test_rd_setup, test_rd_acquire_release, test_rd_teardown);
return g_test_run ();
}