mirror of
https://gitlab.freedesktop.org/gstreamer/gstreamer.git
synced 2025-01-01 13:08:49 +00:00
c86d662b23
Implements a message handling function to preview pipeline bus. If GST_MESSAGE_ERROR is seen, considers preview pipeline unable to do its job and posts an error message to application. Sets pipeline element to NULL so that subsequent calls to post_preview and set_caps functions just returns without pushing anything to the disposed preview pipeline. Leaves further actions to the application.
329 lines
9.6 KiB
C
329 lines
9.6 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 gboolean
|
|
bus_callback (GstBus * bus, GstMessage * message, gpointer user_data)
|
|
{
|
|
switch (GST_MESSAGE_TYPE (message)) {
|
|
case GST_MESSAGE_ERROR:{
|
|
GError *err;
|
|
GstCameraBinPreviewPipelineData *data;
|
|
|
|
data = user_data;
|
|
|
|
gst_message_parse_error (message, &err, NULL);
|
|
GST_WARNING ("Error from preview pipeline: %s", err->message);
|
|
g_error_free (err);
|
|
|
|
/* TODO Not sure if we should post an Error or Warning here */
|
|
GST_ELEMENT_ERROR (data, CORE, FAILED,
|
|
("fatal error in preview pipeline, disposing the pipeline"), (NULL));
|
|
|
|
/* Possible error situations:
|
|
* 1) cond_wait pending. prevent deadlock by signalling the cond
|
|
* 2) preview_pipeline_post called with new buffer to handle. returns
|
|
* because data->pipeline is set to null
|
|
* 3) new preview caps incoming. returns because data->pipeline is null
|
|
*/
|
|
|
|
if (data->pipeline) {
|
|
gst_element_set_state (data->pipeline, GST_STATE_NULL);
|
|
gst_object_unref (data->pipeline);
|
|
data->pipeline = NULL;
|
|
}
|
|
|
|
g_cond_signal (data->processing_cond);
|
|
|
|
break;
|
|
}
|
|
default:
|
|
break;
|
|
}
|
|
return TRUE;
|
|
}
|
|
|
|
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;
|
|
GstBus *bus;
|
|
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);
|
|
|
|
bus = gst_pipeline_get_bus (GST_PIPELINE (data->pipeline));
|
|
gst_bus_add_watch (bus, bus_callback, data);
|
|
gst_object_unref (bus);
|
|
|
|
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);
|
|
g_return_val_if_fail (preview->pipeline != NULL, FALSE);
|
|
|
|
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);
|
|
g_return_if_fail (preview->pipeline != 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);
|
|
}
|