summaryrefslogtreecommitdiff
path: root/tests/check/elements/wasapi.c
diff options
context:
space:
mode:
Diffstat (limited to 'tests/check/elements/wasapi.c')
-rw-r--r--tests/check/elements/wasapi.c205
1 files changed, 184 insertions, 21 deletions
diff --git a/tests/check/elements/wasapi.c b/tests/check/elements/wasapi.c
index 9132b19ad..881bb682c 100644
--- a/tests/check/elements/wasapi.c
+++ b/tests/check/elements/wasapi.c
@@ -28,14 +28,143 @@
typedef struct
{
GMainLoop *loop;
+ GstElement *pipeline;
+ guint n_buffers;
+ guint restart_count;
+ GstState reuse_state;
+} SrcReuseTestData;
+
+static gboolean
+src_reuse_bus_handler (GstBus * bus, GstMessage * message,
+ SrcReuseTestData * data)
+{
+ if (GST_MESSAGE_TYPE (message) == GST_MESSAGE_ERROR) {
+ GST_ERROR ("Got error message from pipeline");
+ g_main_loop_quit (data->loop);
+ }
+
+ return TRUE;
+}
+
+static void
+start_pipeline (SrcReuseTestData * data)
+{
+ GstStateChangeReturn ret;
+
+ GST_INFO ("Start pipeline");
+ ret = gst_element_set_state (data->pipeline, GST_STATE_PLAYING);
+ fail_unless (ret != GST_STATE_CHANGE_FAILURE);
+}
+
+static gboolean
+restart_pipeline (SrcReuseTestData * data)
+{
+ data->restart_count++;
+ start_pipeline (data);
+
+ return G_SOURCE_REMOVE;
+}
+
+static gboolean
+handle_handoff (SrcReuseTestData * data)
+{
+ data->n_buffers++;
+
+ /* Restart every 10 packets */
+ if (data->n_buffers > 10) {
+ GstStateChangeReturn ret;
+ data->n_buffers = 0;
+
+ ret = gst_element_set_state (data->pipeline, data->reuse_state);
+ fail_unless (ret != GST_STATE_CHANGE_FAILURE);
+
+ if (data->restart_count < 2) {
+ GST_INFO ("Restart pipeline, current restart count %d",
+ data->restart_count);
+ g_timeout_add_seconds (1, (GSourceFunc) restart_pipeline, data);
+ } else {
+ GST_INFO ("Finish test");
+ g_main_loop_quit (data->loop);
+ }
+ }
+
+ return G_SOURCE_REMOVE;
+}
+
+static void
+on_sink_handoff (GstElement * element, GstBuffer * buffer, GstPad * pad,
+ SrcReuseTestData * data)
+{
+ g_idle_add ((GSourceFunc) handle_handoff, data);
+}
+
+static void
+wasapisrc_reuse (GstState reuse_state)
+{
+ GstBus *bus;
+ GstElement *sink;
+ SrcReuseTestData data;
+
+ memset (&data, 0, sizeof (SrcReuseTestData));
+
+ data.loop = g_main_loop_new (NULL, FALSE);
+
+ data.pipeline = gst_parse_launch ("wasapisrc provide-clock=false ! queue ! "
+ "fakesink name=sink async=false", NULL);
+ fail_unless (data.pipeline != NULL);
+ data.reuse_state = reuse_state;
+
+ sink = gst_bin_get_by_name (GST_BIN (data.pipeline), "sink");
+ fail_unless (sink);
+
+ g_object_set (G_OBJECT (sink), "signal-handoffs", TRUE, NULL);
+ g_signal_connect (sink, "handoff", G_CALLBACK (on_sink_handoff), &data);
+
+ bus = gst_element_get_bus (GST_ELEMENT (data.pipeline));
+ fail_unless (bus != NULL);
+
+ gst_bus_add_watch (bus, (GstBusFunc) src_reuse_bus_handler, &data);
+ start_pipeline (&data);
+ g_main_loop_run (data.loop);
+
+ fail_unless (data.restart_count == 2);
+
+ gst_element_set_start_time (data.pipeline, GST_STATE_NULL);
+ gst_bus_remove_watch (bus);
+ gst_object_unref (bus);
+
+ gst_object_unref (data.pipeline);
+ g_main_loop_unref (data.loop);
+}
+
+/* https://gitlab.freedesktop.org/gstreamer/gst-plugins-bad/-/issues/1110 */
+GST_START_TEST (test_wasapisrc_reuse_null)
+{
+ wasapisrc_reuse (GST_STATE_NULL);
+}
+
+GST_END_TEST;
+
+GST_START_TEST (test_wasapisrc_reuse_ready)
+{
+ wasapisrc_reuse (GST_STATE_READY);
+}
+
+GST_END_TEST;
+
+typedef struct
+{
+ GMainLoop *loop;
GstElement *pipe;
guint rem_st_changes;
+ GstState reuse_state;
} SinkPlayReadyTData;
static gboolean
-bus_watch_cb (GstBus * bus, GstMessage * message, gpointer user_data)
+sink_reuse_bus_watch_cb (GstBus * bus, GstMessage * message, gpointer user_data)
{
fail_unless (message->type != GST_MESSAGE_ERROR);
+
return G_SOURCE_CONTINUE;
}
@@ -44,7 +173,7 @@ state_timer_cb (gpointer user_data)
{
SinkPlayReadyTData *tdata = user_data;
GstState nxt_st = tdata->rem_st_changes % 2 == 1 ?
- GST_STATE_READY : GST_STATE_PLAYING;
+ tdata->reuse_state : GST_STATE_PLAYING;
ASSERT_SET_STATE (tdata->pipe, nxt_st, GST_STATE_CHANGE_SUCCESS);
tdata->rem_st_changes--;
@@ -58,7 +187,8 @@ state_timer_cb (gpointer user_data)
/* Test that the wasapisink can survive the state change
* from PLAYING to READY and then back to PLAYING */
-GST_START_TEST (test_sink_play_ready)
+static void
+wasapisink_reuse (GstState reuse_state)
{
SinkPlayReadyTData tdata;
GstBus *bus;
@@ -67,7 +197,9 @@ GST_START_TEST (test_sink_play_ready)
fail_unless (tdata.pipe != NULL);
bus = gst_element_get_bus (tdata.pipe);
fail_unless (bus != NULL);
- gst_bus_add_watch (bus, bus_watch_cb, NULL);
+ gst_bus_add_watch (bus, sink_reuse_bus_watch_cb, NULL);
+
+ tdata.reuse_state = reuse_state;
ASSERT_SET_STATE (tdata.pipe, GST_STATE_PLAYING, GST_STATE_CHANGE_SUCCESS);
tdata.rem_st_changes = 3; /* -> READY -> PLAYING -> QUIT */
@@ -82,47 +214,78 @@ GST_START_TEST (test_sink_play_ready)
gst_object_unref (tdata.pipe);
}
+GST_START_TEST (test_wasapisink_reuse_null)
+{
+ wasapisink_reuse (GST_STATE_NULL);
+}
+
+GST_END_TEST;
+
+GST_START_TEST (test_wasapisink_reuse_ready)
+{
+ wasapisink_reuse (GST_STATE_READY);
+}
+
GST_END_TEST;
static gboolean
-device_is_available (const gchar * factory_name)
+check_wasapi_element (gboolean is_src)
{
+ gboolean ret = TRUE;
GstElement *elem;
- gboolean avail;
+ const gchar *elem_name;
+
+ if (is_src)
+ elem_name = "wasapisrc";
+ else
+ elem_name = "wasapisink";
- elem = gst_element_factory_make (factory_name, NULL);
- if (elem == NULL) {
- GST_INFO ("%s: not available", factory_name);
+ elem = gst_element_factory_make (elem_name, NULL);
+ if (!elem) {
+ GST_INFO ("%s is not available", elem_name);
return FALSE;
}
- avail = gst_element_set_state (elem, GST_STATE_READY)
- == GST_STATE_CHANGE_SUCCESS;
- if (!avail) {
- GST_INFO ("%s: cannot change state to ready", factory_name);
+ /* GST_STATE_READY is meaning that camera is available */
+ if (gst_element_set_state (elem, GST_STATE_READY) != GST_STATE_CHANGE_SUCCESS) {
+ GST_INFO ("cannot open device");
+ ret = FALSE;
}
gst_element_set_state (elem, GST_STATE_NULL);
gst_object_unref (elem);
- return avail;
+ return ret;
}
static Suite *
wasapi_suite (void)
{
- Suite *suite = suite_create ("wasapi");
- TCase *tc_sink = tcase_create ("sink");
+ Suite *s = suite_create ("wasapi");
+ TCase *tc_basic = tcase_create ("general");
+ gboolean have_src = FALSE;
+ gboolean have_sink = FALSE;
- suite_add_tcase (suite, tc_sink);
+ suite_add_tcase (s, tc_basic);
- if (device_is_available ("wasapisink")) {
- tcase_add_test (tc_sink, test_sink_play_ready);
+ have_src = check_wasapi_element (TRUE);
+ have_sink = check_wasapi_element (FALSE);
+
+ if (!have_src && !have_sink) {
+ GST_INFO ("Skipping tests, wasapisrc/wasapisink are unavailable");
} else {
- GST_INFO ("Sink not available, skipping sink tests");
+ if (have_src) {
+ tcase_add_test (tc_basic, test_wasapisrc_reuse_null);
+ tcase_add_test (tc_basic, test_wasapisrc_reuse_ready);
+ }
+
+ if (have_sink) {
+ tcase_add_test (tc_basic, test_wasapisink_reuse_null);
+ tcase_add_test (tc_basic, test_wasapisink_reuse_ready);
+ }
}
- return suite;
+ return s;
}
GST_CHECK_MAIN (wasapi)