/* GStreamer GES plugin * * Copyright (C) 2019 Thibault Saunier * * gesbasebin.h * * 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., 51 Franklin St, Fifth Floor, * Boston, MA 02110-1301, USA. * */ #include "gesbasebin.h" static GstStaticPadTemplate video_src_template = GST_STATIC_PAD_TEMPLATE ("video_src", GST_PAD_SRC, GST_PAD_SOMETIMES, GST_STATIC_CAPS ("video/x-raw(ANY)")); static GstStaticPadTemplate audio_src_template = GST_STATIC_PAD_TEMPLATE ("audio_src", GST_PAD_SRC, GST_PAD_SOMETIMES, GST_STATIC_CAPS ("audio/x-raw(ANY);")); typedef struct { GESTimeline *timeline; GstFlowCombiner *flow_combiner; } GESBaseBinPrivate; enum { PROP_0, PROP_TIMELINE, PROP_LAST }; static GParamSpec *properties[PROP_LAST]; G_DEFINE_TYPE_WITH_PRIVATE (GESBaseBin, ges_base_bin, GST_TYPE_BIN); GST_DEBUG_CATEGORY_STATIC (gesbasebin); #define GST_CAT_DEFAULT gesbasebin static void ges_base_bin_dispose (GObject * object) { GESBaseBin *self = GES_BASE_BIN (object); GESBaseBinPrivate *priv = ges_base_bin_get_instance_private (self); if (priv->timeline) gst_clear_object (&priv->timeline); } static void ges_base_bin_finalize (GObject * object) { GESBaseBin *self = GES_BASE_BIN (object); GESBaseBinPrivate *priv = ges_base_bin_get_instance_private (self); gst_flow_combiner_free (priv->flow_combiner); } static void ges_base_bin_get_property (GObject * object, guint property_id, GValue * value, GParamSpec * pspec) { GESBaseBin *self = GES_BASE_BIN (object); GESBaseBinPrivate *priv = ges_base_bin_get_instance_private (self); switch (property_id) { case PROP_TIMELINE: g_value_set_object (value, priv->timeline); break; default: G_OBJECT_WARN_INVALID_PROPERTY_ID (object, property_id, pspec); } } static void ges_base_bin_set_property (GObject * object, guint property_id, const GValue * value, GParamSpec * pspec) { GESBaseBin *self = GES_BASE_BIN (object); switch (property_id) { case PROP_TIMELINE: ges_base_bin_set_timeline (self, g_value_get_object (value)); break; default: G_OBJECT_WARN_INVALID_PROPERTY_ID (object, property_id, pspec); } } static void ges_base_bin_class_init (GESBaseBinClass * self_class) { GObjectClass *gclass = G_OBJECT_CLASS (self_class); GstElementClass *gstelement_klass = GST_ELEMENT_CLASS (self_class); GST_DEBUG_CATEGORY_INIT (gesbasebin, "gesbasebin", 0, "ges bin element"); gclass->get_property = ges_base_bin_get_property; gclass->set_property = ges_base_bin_set_property; gclass->dispose = ges_base_bin_dispose; gclass->finalize = ges_base_bin_finalize; /** * GESBaseBin:timeline: * * Timeline to use in this bin. */ properties[PROP_TIMELINE] = g_param_spec_object ("timeline", "Timeline", "Timeline to use in this src.", GES_TYPE_TIMELINE, G_PARAM_READWRITE | G_PARAM_STATIC_STRINGS); g_object_class_install_properties (gclass, PROP_LAST, properties); gst_element_class_add_pad_template (gstelement_klass, gst_static_pad_template_get (&video_src_template)); gst_element_class_add_pad_template (gstelement_klass, gst_static_pad_template_get (&audio_src_template)); } static void ges_base_bin_init (GESBaseBin * self) { GESBaseBinPrivate *priv = ges_base_bin_get_instance_private (self); priv->flow_combiner = gst_flow_combiner_new (); } static GstFlowReturn gst_bin_src_chain (GstPad * pad, GstObject * parent, GstBuffer * buffer) { GstFlowReturn result, chain_result; GESBaseBin *self = GES_BASE_BIN (GST_OBJECT_PARENT (parent)); GESBaseBinPrivate *priv = ges_base_bin_get_instance_private (self); chain_result = gst_proxy_pad_chain_default (pad, GST_OBJECT (self), buffer); result = gst_flow_combiner_update_pad_flow (priv->flow_combiner, pad, chain_result); if (result == GST_FLOW_FLUSHING) return chain_result; return result; } gboolean ges_base_bin_set_timeline (GESBaseBin * self, GESTimeline * timeline) { GList *tmp; guint naudiopad = 0, nvideopad = 0; GstBin *sbin = GST_BIN (self); GESBaseBinPrivate *priv = ges_base_bin_get_instance_private (self); g_return_val_if_fail (GES_IS_TIMELINE (timeline), FALSE); if (priv->timeline) { GST_ERROR_OBJECT (sbin, "Implement changing timeline support"); return FALSE; } priv->timeline = gst_object_ref (timeline); GST_INFO_OBJECT (sbin, "Setting timeline: %" GST_PTR_FORMAT, timeline); if (!gst_bin_add (sbin, GST_ELEMENT (timeline))) { GST_ERROR_OBJECT (sbin, "Could not add timeline to myself!"); return FALSE; } for (tmp = timeline->tracks; tmp; tmp = tmp->next) { GstPad *gpad; gchar *name = NULL; GstElement *queue; GESTrack *track = GES_TRACK (tmp->data); GstPad *proxy_pad, *tmppad, *pad = ges_timeline_get_pad_for_track (timeline, track); GstStaticPadTemplate *template; if (!pad) { GST_WARNING_OBJECT (sbin, "No pad for track: %" GST_PTR_FORMAT, track); continue; } if (track->type == GES_TRACK_TYPE_AUDIO) { name = g_strdup_printf ("audio_%u", naudiopad++); template = &audio_src_template; } else if (track->type == GES_TRACK_TYPE_VIDEO) { name = g_strdup_printf ("video_%u", nvideopad++); template = &video_src_template; } else { GST_INFO_OBJECT (sbin, "Track type not handled: %" GST_PTR_FORMAT, track); continue; } queue = gst_element_factory_make ("queue", NULL); /* Add queues the same way as in GESPipeline */ g_object_set (G_OBJECT (queue), "max-size-buffers", 0, "max-size-bytes", 0, "max-size-time", (gint64) 2 * GST_SECOND, NULL); gst_bin_add (sbin, queue); gst_element_sync_state_with_parent (GST_ELEMENT (queue)); tmppad = gst_element_get_static_pad (queue, "sink"); if (gst_pad_link (pad, tmppad) != GST_PAD_LINK_OK) { GST_ERROR_OBJECT (sbin, "Could not link %s:%s and %s:%s", GST_DEBUG_PAD_NAME (pad), GST_DEBUG_PAD_NAME (tmppad)); gst_object_unref (tmppad); gst_object_unref (queue); continue; } tmppad = gst_element_get_static_pad (queue, "src"); gpad = gst_ghost_pad_new_from_template (name, tmppad, gst_static_pad_template_get (template)); gst_pad_set_active (gpad, TRUE); gst_element_add_pad (GST_ELEMENT (sbin), gpad); proxy_pad = GST_PAD (gst_proxy_pad_get_internal (GST_PROXY_PAD (gpad))); gst_flow_combiner_add_pad (priv->flow_combiner, proxy_pad); gst_pad_set_chain_function (proxy_pad, gst_bin_src_chain); gst_object_unref (proxy_pad); GST_DEBUG_OBJECT (sbin, "Adding pad: %" GST_PTR_FORMAT, gpad); } gst_element_no_more_pads (GST_ELEMENT (sbin)); gst_element_sync_state_with_parent (GST_ELEMENT (timeline)); return TRUE; } GESTimeline * ges_base_bin_get_timeline (GESBaseBin * self) { GESBaseBinPrivate *priv = ges_base_bin_get_instance_private (self); return priv->timeline; }