mirror of
https://gitlab.freedesktop.org/gstreamer/gstreamer.git
synced 2024-11-09 10:59:39 +00:00
802028081c
Implements a state indicating flag to preview pipeline, so that new caps are not set if the pipeline is processing a preview. The caps are set as pending and applied when the next preview post is called. In this case a wait was implemented in the post_preview function, so that new preview image buffer will wait until the other previews have been posted to the application and the new caps can be used safely.
281 lines
8.1 KiB
C
281 lines
8.1 KiB
C
/*
|
|
* GStreamer
|
|
* Copyright (C) 2008 Nokia Corporation <multimedia@maemo.org>
|
|
*
|
|
* This library is free software; you can redistribute it and/or
|
|
* modify it under the terms of the GNU Library General Public
|
|
* License as published by the Free Software Foundation; either
|
|
* version 2 of the License, or (at your option) any later version.
|
|
*
|
|
* This library is distributed in the hope that it will be useful,
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
|
* Library General Public License for more details.
|
|
*
|
|
* You should have received a copy of the GNU Library General Public
|
|
* License along with this library; if not, write to the
|
|
* Free Software Foundation, Inc., 59 Temple Place - Suite 330,
|
|
* Boston, MA 02111-1307, USA.
|
|
*/
|
|
|
|
/**
|
|
* SECTION:camerabingeneral
|
|
* @short_description: helper functions for #GstCameraBin and it's modules
|
|
*
|
|
* Common helper functions for #GstCameraBin, #GstCameraBinImage and
|
|
* #GstCameraBinVideo.
|
|
*
|
|
*/
|
|
#include <gst/app/gstappsrc.h>
|
|
#include <gst/app/gstappsink.h>
|
|
#include "gstcamerabinpreview.h"
|
|
#include "gstbasecamerasrc.h"
|
|
|
|
static void _gst_camerabin_preview_set_caps (GstCameraBinPreviewPipelineData *
|
|
preview, GstCaps * caps);
|
|
|
|
static GstFlowReturn
|
|
gst_camerabin_preview_pipeline_new_preroll (GstAppSink * appsink,
|
|
gpointer user_data)
|
|
{
|
|
GstBuffer *buffer;
|
|
|
|
buffer = gst_app_sink_pull_preroll (appsink);
|
|
gst_buffer_unref (buffer);
|
|
|
|
return GST_FLOW_OK;
|
|
}
|
|
|
|
static GstFlowReturn
|
|
gst_camerabin_preview_pipeline_new_buffer (GstAppSink * appsink,
|
|
gpointer user_data)
|
|
{
|
|
GstBuffer *buffer;
|
|
GstStructure *s;
|
|
GstMessage *msg;
|
|
GstCameraBinPreviewPipelineData *data;
|
|
|
|
data = user_data;
|
|
|
|
buffer = gst_app_sink_pull_buffer (appsink);
|
|
s = gst_structure_new (GST_BASE_CAMERA_SRC_PREVIEW_MESSAGE_NAME,
|
|
"buffer", GST_TYPE_BUFFER, buffer, NULL);
|
|
gst_buffer_unref (buffer);
|
|
msg = gst_message_new_element (GST_OBJECT (data->element), s);
|
|
|
|
GST_DEBUG_OBJECT (data->element, "sending message with preview image");
|
|
if (gst_element_post_message (data->element, msg) == FALSE) {
|
|
GST_WARNING_OBJECT (data->element,
|
|
"This element has no bus, therefore no message sent!");
|
|
}
|
|
|
|
g_mutex_lock (data->processing_lock);
|
|
|
|
data->processing--;
|
|
if (data->processing == 0)
|
|
g_cond_signal (data->processing_cond);
|
|
|
|
g_mutex_unlock (data->processing_lock);
|
|
|
|
return GST_FLOW_OK;
|
|
}
|
|
|
|
/**
|
|
* gst_camerabin_create_preview_pipeline:
|
|
* @element: Owner of this pipeline
|
|
* @filter: Custom filter to process preview data (an extra ref is taken)
|
|
*
|
|
* Creates a new previewing pipeline that can receive buffers
|
|
* to be posted as camerabin preview messages for @element
|
|
*
|
|
* Returns: The newly created #GstCameraBinPreviewPipelineData
|
|
*/
|
|
GstCameraBinPreviewPipelineData *
|
|
gst_camerabin_create_preview_pipeline (GstElement * element,
|
|
GstElement * filter)
|
|
{
|
|
GstCameraBinPreviewPipelineData *data;
|
|
GstElement *csp;
|
|
GstElement *csp2;
|
|
GstElement *vscale;
|
|
gboolean added = FALSE;
|
|
GstAppSinkCallbacks callbacks = { 0, };
|
|
|
|
data = g_new (GstCameraBinPreviewPipelineData, 1);
|
|
|
|
data->pipeline = gst_pipeline_new ("preview-pipeline");
|
|
data->appsrc = gst_element_factory_make ("appsrc", "preview-appsrc");
|
|
data->capsfilter = gst_element_factory_make ("capsfilter",
|
|
"preview-capsfilter");
|
|
data->appsink = gst_element_factory_make ("appsink", "preview-appsink");
|
|
csp = gst_element_factory_make ("ffmpegcolorspace", "preview-csp0");
|
|
csp2 = gst_element_factory_make ("ffmpegcolorspace", "preview-csp1");
|
|
vscale = gst_element_factory_make ("videoscale", "preview-vscale");
|
|
|
|
if (!data->appsrc || !data->capsfilter || !data->appsink || !csp ||
|
|
!csp2 || !vscale) {
|
|
goto error;
|
|
}
|
|
|
|
gst_bin_add_many (GST_BIN (data->pipeline), data->appsrc, data->capsfilter,
|
|
data->appsink, csp, csp2, vscale, NULL);
|
|
if (filter)
|
|
gst_bin_add (GST_BIN (data->pipeline), gst_object_ref (filter));
|
|
added = TRUE;
|
|
|
|
if (filter) {
|
|
if (!gst_element_link_many (data->appsrc, filter, csp, vscale, csp2,
|
|
data->capsfilter, data->appsink, NULL))
|
|
goto error;
|
|
} else {
|
|
if (!gst_element_link_many (data->appsrc, csp, vscale, csp2,
|
|
data->capsfilter, data->appsink, NULL))
|
|
goto error;
|
|
}
|
|
|
|
callbacks.new_preroll = gst_camerabin_preview_pipeline_new_preroll;
|
|
callbacks.new_buffer = gst_camerabin_preview_pipeline_new_buffer;
|
|
gst_app_sink_set_callbacks ((GstAppSink *) data->appsink, &callbacks, data,
|
|
NULL);
|
|
|
|
g_object_set (data->appsink, "sync", FALSE, NULL);
|
|
|
|
data->element = element;
|
|
data->filter = filter;
|
|
|
|
data->processing_lock = g_mutex_new ();
|
|
data->processing_cond = g_cond_new ();
|
|
|
|
data->pending_preview_caps = NULL;
|
|
data->processing = 0;
|
|
|
|
return data;
|
|
error:
|
|
GST_WARNING ("Failed to create camerabin's preview pipeline");
|
|
if (!added) {
|
|
if (csp)
|
|
gst_object_unref (csp);
|
|
if (csp2)
|
|
gst_object_unref (csp2);
|
|
if (vscale)
|
|
gst_object_unref (vscale);
|
|
if (data->appsrc)
|
|
gst_object_unref (data->appsrc);
|
|
if (data->capsfilter)
|
|
gst_object_unref (data->capsfilter);
|
|
if (data->appsink)
|
|
gst_object_unref (data->appsink);
|
|
}
|
|
gst_camerabin_destroy_preview_pipeline (data);
|
|
return NULL;
|
|
}
|
|
|
|
/**
|
|
* gst_camerabin_destroy_preview_pipeline:
|
|
* @preview: the #GstCameraBinPreviewPipelineData
|
|
*
|
|
* Frees a #GstCameraBinPreviewPipelineData
|
|
*/
|
|
void
|
|
gst_camerabin_destroy_preview_pipeline (GstCameraBinPreviewPipelineData *
|
|
preview)
|
|
{
|
|
if (preview->processing_lock) {
|
|
g_mutex_free (preview->processing_lock);
|
|
preview->processing_lock = NULL;
|
|
}
|
|
if (preview->processing_cond) {
|
|
g_cond_free (preview->processing_cond);
|
|
preview->processing_cond = NULL;
|
|
}
|
|
if (preview->pipeline) {
|
|
gst_element_set_state (preview->pipeline, GST_STATE_NULL);
|
|
gst_object_unref (preview->pipeline);
|
|
}
|
|
g_free (preview);
|
|
}
|
|
|
|
/**
|
|
* gst_camerabin_preview_pipeline_post:
|
|
* @preview: the #GstCameraBinPreviewPipelineData
|
|
* @buffer: the buffer to be posted as a preview
|
|
*
|
|
* Converts the @buffer to the desired format and posts the preview
|
|
* message to the bus.
|
|
*
|
|
* Returns: %TRUE on success
|
|
*/
|
|
gboolean
|
|
gst_camerabin_preview_pipeline_post (GstCameraBinPreviewPipelineData * preview,
|
|
GstBuffer * buffer)
|
|
{
|
|
g_return_val_if_fail (preview != NULL, FALSE);
|
|
g_return_val_if_fail (preview->pipeline != NULL, FALSE);
|
|
g_return_val_if_fail (buffer, FALSE);
|
|
|
|
g_mutex_lock (preview->processing_lock);
|
|
|
|
if (preview->pending_preview_caps) {
|
|
if (preview->processing > 0) {
|
|
g_cond_wait (preview->processing_cond, preview->processing_lock);
|
|
}
|
|
_gst_camerabin_preview_set_caps (preview, preview->pending_preview_caps);
|
|
gst_caps_replace (&preview->pending_preview_caps, NULL);
|
|
}
|
|
|
|
preview->processing++;
|
|
|
|
gst_app_src_push_buffer ((GstAppSrc *) preview->appsrc,
|
|
gst_buffer_ref (buffer));
|
|
|
|
g_mutex_unlock (preview->processing_lock);
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
static void
|
|
_gst_camerabin_preview_set_caps (GstCameraBinPreviewPipelineData * preview,
|
|
GstCaps * caps)
|
|
{
|
|
GstState state, pending;
|
|
GstStateChangeReturn ret;
|
|
|
|
g_return_if_fail (preview != NULL);
|
|
|
|
ret = gst_element_get_state (preview->pipeline, &state, &pending, 0);
|
|
if (ret == GST_STATE_CHANGE_FAILURE) {
|
|
/* make it try again */
|
|
state = GST_STATE_PLAYING;
|
|
pending = GST_STATE_VOID_PENDING;
|
|
}
|
|
gst_element_set_state (preview->pipeline, GST_STATE_NULL);
|
|
g_object_set (preview->capsfilter, "caps", caps, NULL);
|
|
if (pending != GST_STATE_VOID_PENDING)
|
|
state = pending;
|
|
gst_element_set_state (preview->pipeline, state);
|
|
}
|
|
|
|
/**
|
|
* gst_camerabin_preview_set_caps:
|
|
* @preview: the #GstCameraBinPreviewPipelineData
|
|
* @caps: the #GstCaps to be set (a new ref will be taken)
|
|
*
|
|
* The caps that preview buffers should have when posted
|
|
* on the bus
|
|
*/
|
|
void
|
|
gst_camerabin_preview_set_caps (GstCameraBinPreviewPipelineData * preview,
|
|
GstCaps * caps)
|
|
{
|
|
g_return_if_fail (preview != NULL);
|
|
|
|
g_mutex_lock (preview->processing_lock);
|
|
|
|
if (preview->processing == 0) {
|
|
_gst_camerabin_preview_set_caps (preview, caps);
|
|
} else {
|
|
GST_DEBUG ("Preview pipeline busy, storing new caps as pending");
|
|
gst_caps_replace (&preview->pending_preview_caps, caps);
|
|
}
|
|
g_mutex_unlock (preview->processing_lock);
|
|
}
|