mirror of
https://gitlab.freedesktop.org/gstreamer/gstreamer.git
synced 2025-01-20 14:18:34 +00:00
gst-validate-scenario: Make waits optional in appsrc-push
While in many cases it's desirable to wait for a buffer to be pushed downstream when using appsrc-push, in some cases this is not possible as such pushing action is dependent on following actions that would not be executed if we wait. An example for this is prerolling: appsrc ! qtdemux ! video/x-h264 ! decodebin name=dec ! %(videosink)s description, seek=false, handles-states=true appsrc-push, target-element-name=appsrc0, file-name="raw_h264.0.mp4" set-state, state=playing appsrc-eos, target-element-name=appsrc0 In order for the preroll to occur, both the appsrc needs to push the buffer and the state needs to reach PLAYING. But `set-state` cannot finish if the buffer has not been pushed (the state transition does not finish) and conversely pushing the buffer will not finish until the state has reached. Making appsrc-push not wait for the buffer solves this problem. This patch makes appsrc-push aware of this issue by only waiting for the buffer to be pushed if the pipeline is in a state that allows buffers to flow.
This commit is contained in:
parent
af205f63b7
commit
f0abd316e2
2 changed files with 30 additions and 2 deletions
|
@ -345,6 +345,12 @@ gst_validate_action_init (GstValidateAction * action)
|
||||||
g_weak_ref_init (&action->priv->scenario, NULL);
|
g_weak_ref_init (&action->priv->scenario, NULL);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void
|
||||||
|
gst_validate_action_ref (GstValidateAction * action)
|
||||||
|
{
|
||||||
|
gst_mini_object_ref (GST_MINI_OBJECT (action));
|
||||||
|
}
|
||||||
|
|
||||||
void
|
void
|
||||||
gst_validate_action_unref (GstValidateAction * action)
|
gst_validate_action_unref (GstValidateAction * action)
|
||||||
{
|
{
|
||||||
|
@ -2659,6 +2665,7 @@ appsrc_push_chain_wrapper (GstPad * pad, GstObject * parent, GstBuffer * buffer,
|
||||||
GST_VALIDATE_SCENARIO_EOS_HANDLING_LOCK (scenario);
|
GST_VALIDATE_SCENARIO_EOS_HANDLING_LOCK (scenario);
|
||||||
ret = pad->chainfunc (pad, parent, buffer);
|
ret = pad->chainfunc (pad, parent, buffer);
|
||||||
gst_validate_action_set_done (action);
|
gst_validate_action_set_done (action);
|
||||||
|
gst_validate_action_unref (action);
|
||||||
*remove_wrapper = TRUE;
|
*remove_wrapper = TRUE;
|
||||||
GST_VALIDATE_SCENARIO_EOS_HANDLING_UNLOCK (scenario);
|
GST_VALIDATE_SCENARIO_EOS_HANDLING_UNLOCK (scenario);
|
||||||
g_object_unref (scenario);
|
g_object_unref (scenario);
|
||||||
|
@ -2698,6 +2705,12 @@ _execute_appsrc_push (GstValidateScenario * scenario,
|
||||||
guint64 offset = 0;
|
guint64 offset = 0;
|
||||||
guint64 size = -1;
|
guint64 size = -1;
|
||||||
gint push_buffer_ret;
|
gint push_buffer_ret;
|
||||||
|
gboolean wait;
|
||||||
|
|
||||||
|
/* We will only wait for the the buffer to be pushed if we are in a state
|
||||||
|
* that allows flow of buffers (>=PAUSED). Otherwise the buffer will just
|
||||||
|
* be enqueued. */
|
||||||
|
wait = scenario->priv->target_state >= GST_STATE_PAUSED;
|
||||||
|
|
||||||
target = _get_target_element (scenario, action);
|
target = _get_target_element (scenario, action);
|
||||||
if (target == NULL) {
|
if (target == NULL) {
|
||||||
|
@ -2759,6 +2772,9 @@ _execute_appsrc_push (GstValidateScenario * scenario,
|
||||||
gst_object_unref (peer_pad);
|
gst_object_unref (peer_pad);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/* Keep the action alive until set done is called. */
|
||||||
|
gst_validate_action_ref (action);
|
||||||
|
|
||||||
g_signal_emit_by_name (target, "push-buffer", buffer, &push_buffer_ret);
|
g_signal_emit_by_name (target, "push-buffer", buffer, &push_buffer_ret);
|
||||||
if (push_buffer_ret != GST_FLOW_OK) {
|
if (push_buffer_ret != GST_FLOW_OK) {
|
||||||
gchar *structure_string = gst_structure_to_string (action->structure);
|
gchar *structure_string = gst_structure_to_string (action->structure);
|
||||||
|
@ -2770,7 +2786,14 @@ _execute_appsrc_push (GstValidateScenario * scenario,
|
||||||
|
|
||||||
g_free (file_name);
|
g_free (file_name);
|
||||||
gst_object_unref (target);
|
gst_object_unref (target);
|
||||||
return GST_VALIDATE_EXECUTE_ACTION_ASYNC;
|
|
||||||
|
if (wait) {
|
||||||
|
return GST_VALIDATE_EXECUTE_ACTION_ASYNC;
|
||||||
|
} else {
|
||||||
|
gst_validate_printf (NULL,
|
||||||
|
"Pipeline is not ready to push buffers, interlacing appsrc-push action...");
|
||||||
|
return GST_VALIDATE_EXECUTE_ACTION_INTERLACED;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
static gint
|
static gint
|
||||||
|
@ -3357,6 +3380,9 @@ _load_scenario_file (GstValidateScenario * scenario,
|
||||||
gst_structure_get_boolean (structure, "handles-states",
|
gst_structure_get_boolean (structure, "handles-states",
|
||||||
&priv->handles_state);
|
&priv->handles_state);
|
||||||
|
|
||||||
|
if (!priv->handles_state)
|
||||||
|
priv->target_state = GST_STATE_PLAYING;
|
||||||
|
|
||||||
pipeline_name = gst_structure_get_string (structure, "pipeline-name");
|
pipeline_name = gst_structure_get_string (structure, "pipeline-name");
|
||||||
if (pipeline_name) {
|
if (pipeline_name) {
|
||||||
g_free (priv->pipeline_name);
|
g_free (priv->pipeline_name);
|
||||||
|
@ -5072,7 +5098,7 @@ init_scenarios (void)
|
||||||
},
|
},
|
||||||
{NULL}
|
{NULL}
|
||||||
}),
|
}),
|
||||||
"Queues a buffer from an appsrc and waits for it to be handled by downstream elements in the same streaming thread.",
|
"Queues a buffer in an appsrc. If the pipeline state allows flow of buffers, the next action is not run until the buffer has been pushed.",
|
||||||
GST_VALIDATE_ACTION_TYPE_NONE);
|
GST_VALIDATE_ACTION_TYPE_NONE);
|
||||||
|
|
||||||
REGISTER_ACTION_TYPE ("appsrc-eos", _execute_appsrc_eos,
|
REGISTER_ACTION_TYPE ("appsrc-eos", _execute_appsrc_eos,
|
||||||
|
|
|
@ -126,6 +126,8 @@ GstValidateAction * gst_validate_action_new (GstValidateScenario * sc
|
||||||
GstStructure *structure,
|
GstStructure *structure,
|
||||||
gboolean add_to_lists);
|
gboolean add_to_lists);
|
||||||
GST_VALIDATE_API
|
GST_VALIDATE_API
|
||||||
|
void gst_validate_action_ref (GstValidateAction * action);
|
||||||
|
GST_VALIDATE_API
|
||||||
void gst_validate_action_unref (GstValidateAction * action);
|
void gst_validate_action_unref (GstValidateAction * action);
|
||||||
|
|
||||||
#define GST_TYPE_VALIDATE_ACTION (gst_validate_action_get_type ())
|
#define GST_TYPE_VALIDATE_ACTION (gst_validate_action_get_type ())
|
||||||
|
|
Loading…
Reference in a new issue