• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
1 /*
2  * GStreamer
3  * Copyright (C) 2015 Matthew Waters <matthew@centricular.com>
4  *
5  * This library is free software; you can redistribute it and/or
6  * modify it under the terms of the GNU Library General Public
7  * License as published by the Free Software Foundation; either
8  * version 2 of the License, or (at your option) any later version.
9  *
10  * This library is distributed in the hope that it will be useful,
11  * but WITHOUT ANY WARRANTY; without even the implied warranty of
12  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
13  * Library General Public License for more details.
14  *
15  * You should have received a copy of the GNU Library General Public
16  * License along with this library; if not, write to the
17  * Free Software Foundation, Inc., 51 Franklin St, Fifth Floor,
18  * Boston, MA 02110-1301, USA.
19  */
20 
21 #ifdef HAVE_CONFIG_H
22 #include "config.h"
23 #endif
24 
25 #include "gstglelements.h"
26 #include "gstglsrcbin.h"
27 
28 GST_DEBUG_CATEGORY (gst_debug_gl_src_bin);
29 #define GST_CAT_DEFAULT gst_debug_gl_src_bin
30 
31 static void gst_gl_src_bin_finalize (GObject * object);
32 static void gst_gl_src_bin_set_property (GObject * object, guint prop_id,
33     const GValue * value, GParamSpec * param_spec);
34 static void gst_gl_src_bin_get_property (GObject * object, guint prop_id,
35     GValue * value, GParamSpec * param_spec);
36 
37 static GstStateChangeReturn gst_gl_src_bin_change_state (GstElement * element,
38     GstStateChange transition);
39 
40 static GstStaticPadTemplate gst_gl_src_bin_template =
41 GST_STATIC_PAD_TEMPLATE ("src",
42     GST_PAD_SRC,
43     GST_PAD_ALWAYS,
44     GST_STATIC_CAPS ("video/x-raw(ANY)"));
45 
46 enum
47 {
48   PROP_0,
49   PROP_SRC,
50 };
51 
52 enum
53 {
54   SIGNAL_0,
55   SIGNAL_CREATE_ELEMENT,
56   SIGNAL_LAST,
57 };
58 
59 static guint gst_gl_src_bin_signals[SIGNAL_LAST] = { 0, };
60 
61 #define gst_gl_src_bin_parent_class parent_class
62 G_DEFINE_TYPE_WITH_CODE (GstGLSrcBin, gst_gl_src_bin,
63     GST_TYPE_BIN,
64     GST_DEBUG_CATEGORY_INIT (gst_debug_gl_src_bin, "glsrcbin", 0,
65         "OpenGL Video Src Bin"));
66 GST_ELEMENT_REGISTER_DEFINE_WITH_CODE (glsrcbin, "glsrcbin",
67     GST_RANK_NONE, GST_TYPE_GL_SRC_BIN, gl_element_init (plugin));
68 
69 static void
gst_gl_src_bin_class_init(GstGLSrcBinClass * klass)70 gst_gl_src_bin_class_init (GstGLSrcBinClass * klass)
71 {
72   GObjectClass *gobject_class;
73   GstElementClass *element_class;
74 
75   gobject_class = (GObjectClass *) klass;
76   element_class = GST_ELEMENT_CLASS (klass);
77 
78   element_class->change_state = gst_gl_src_bin_change_state;
79 
80   gobject_class->set_property = gst_gl_src_bin_set_property;
81   gobject_class->get_property = gst_gl_src_bin_get_property;
82   gobject_class->finalize = gst_gl_src_bin_finalize;
83 
84   g_object_class_install_property (gobject_class, PROP_SRC,
85       g_param_spec_object ("src",
86           "GL src element",
87           "The GL src chain to use",
88           GST_TYPE_ELEMENT,
89           GST_PARAM_MUTABLE_READY | G_PARAM_READWRITE |
90           G_PARAM_STATIC_STRINGS));
91 
92   /**
93    * GstGLSrcBin::create-element:
94    * @object: the #GstGLSrcBin
95    *
96    * Will be emitted when we need the processing element/s that this bin will use
97    *
98    * Returns: a new #GstElement
99    */
100   gst_gl_src_bin_signals[SIGNAL_CREATE_ELEMENT] =
101       g_signal_new ("create-element", G_TYPE_FROM_CLASS (klass),
102       G_SIGNAL_RUN_LAST, 0, NULL, NULL, NULL, GST_TYPE_ELEMENT, 0);
103 
104   gst_element_class_set_metadata (element_class,
105       "GL Src Bin", "Src/Video",
106       "Infrastructure to process GL textures",
107       "Matthew Waters <matthew@centricular.com>");
108 
109   gst_element_class_add_static_pad_template (element_class,
110       &gst_gl_src_bin_template);
111 }
112 
113 static void
gst_gl_src_bin_init(GstGLSrcBin * self)114 gst_gl_src_bin_init (GstGLSrcBin * self)
115 {
116   gboolean res = TRUE;
117   GstPad *pad;
118 
119   self->download = gst_element_factory_make ("gldownload", NULL);
120   self->convert = gst_element_factory_make ("glcolorconvert", NULL);
121 
122   res &= gst_bin_add (GST_BIN (self), self->download);
123   res &= gst_bin_add (GST_BIN (self), self->convert);
124 
125   res &= gst_element_link_pads (self->convert, "src", self->download, "sink");
126 
127   pad = gst_element_get_static_pad (self->download, "src");
128   if (!pad) {
129     res = FALSE;
130   } else {
131     GST_DEBUG_OBJECT (self, "setting target src pad %" GST_PTR_FORMAT, pad);
132     self->srcpad = gst_ghost_pad_new ("src", pad);
133     gst_element_add_pad (GST_ELEMENT_CAST (self), self->srcpad);
134     gst_object_unref (pad);
135   }
136 
137   if (!res) {
138     GST_WARNING_OBJECT (self, "Failed to add/connect the necessary machinery");
139   }
140 }
141 
142 static void
gst_gl_src_bin_finalize(GObject * object)143 gst_gl_src_bin_finalize (GObject * object)
144 {
145   GstGLSrcBin *self = GST_GL_SRC_BIN (object);
146 
147   if (self->src)
148     gst_object_unref (self->src);
149 
150   G_OBJECT_CLASS (parent_class)->finalize (object);
151 }
152 
153 static gboolean
_connect_src_element(GstGLSrcBin * self)154 _connect_src_element (GstGLSrcBin * self)
155 {
156   gboolean res = TRUE;
157 
158   gst_object_set_name (GST_OBJECT (self->src), "src");
159   res &= gst_bin_add (GST_BIN (self), self->src);
160 
161   res &= gst_element_link_pads (self->src, "src", self->convert, "sink");
162 
163   if (!res)
164     GST_ERROR_OBJECT (self, "Failed to link src element into the pipeline");
165 
166   return res;
167 }
168 
169 /*
170  * @src: (transfer floating):
171  */
172 static gboolean
gst_gl_src_bin_set_src(GstGLSrcBin * self,GstElement * src)173 gst_gl_src_bin_set_src (GstGLSrcBin * self, GstElement * src)
174 {
175   g_return_val_if_fail (GST_IS_ELEMENT (src), FALSE);
176 
177   if (self->src) {
178     gst_element_set_locked_state (self->src, TRUE);
179     gst_bin_remove (GST_BIN (self), self->src);
180     gst_element_set_state (self->src, GST_STATE_NULL);
181     gst_object_unref (self->src);
182     self->src = NULL;
183   }
184   self->src = src;
185 
186   gst_object_ref_sink (src);
187 
188   if (src && !_connect_src_element (self)) {
189     gst_object_unref (self->src);
190     self->src = NULL;
191     return FALSE;
192   }
193 
194   return TRUE;
195 }
196 
197 void
gst_gl_src_bin_finish_init_with_element(GstGLSrcBin * self,GstElement * element)198 gst_gl_src_bin_finish_init_with_element (GstGLSrcBin * self,
199     GstElement * element)
200 {
201   gst_gl_src_bin_set_src (self, element);
202 }
203 
204 void
gst_gl_src_bin_finish_init(GstGLSrcBin * self)205 gst_gl_src_bin_finish_init (GstGLSrcBin * self)
206 {
207   GstGLSrcBinClass *klass = GST_GL_SRC_BIN_GET_CLASS (self);
208   GstElement *element = NULL;
209 
210   if (klass->create_element)
211     element = klass->create_element ();
212 
213   if (element)
214     gst_gl_src_bin_finish_init_with_element (self, element);
215 }
216 
217 static void
gst_gl_src_bin_set_property(GObject * object,guint prop_id,const GValue * value,GParamSpec * pspec)218 gst_gl_src_bin_set_property (GObject * object, guint prop_id,
219     const GValue * value, GParamSpec * pspec)
220 {
221   GstGLSrcBin *self = GST_GL_SRC_BIN (object);
222 
223   switch (prop_id) {
224     case PROP_SRC:
225       gst_gl_src_bin_set_src (self, g_value_get_object (value));
226       break;
227     default:
228       if (self->src)
229         g_object_set_property (G_OBJECT (self->src), pspec->name, value);
230       break;
231   }
232 }
233 
234 static void
gst_gl_src_bin_get_property(GObject * object,guint prop_id,GValue * value,GParamSpec * pspec)235 gst_gl_src_bin_get_property (GObject * object, guint prop_id,
236     GValue * value, GParamSpec * pspec)
237 {
238   GstGLSrcBin *self = GST_GL_SRC_BIN (object);
239 
240   switch (prop_id) {
241     case PROP_SRC:
242       g_value_set_object (value, self->src);
243       break;
244     default:
245       if (self->src)
246         g_object_get_property (G_OBJECT (self->src), pspec->name, value);
247       break;
248   }
249 }
250 
251 static GstStateChangeReturn
gst_gl_src_bin_change_state(GstElement * element,GstStateChange transition)252 gst_gl_src_bin_change_state (GstElement * element, GstStateChange transition)
253 {
254   GstGLSrcBin *self = GST_GL_SRC_BIN (element);
255   GstGLSrcBinClass *klass = GST_GL_SRC_BIN_GET_CLASS (self);
256   GstStateChangeReturn ret = GST_STATE_CHANGE_SUCCESS;
257 
258   GST_DEBUG ("changing state: %s => %s",
259       gst_element_state_get_name (GST_STATE_TRANSITION_CURRENT (transition)),
260       gst_element_state_get_name (GST_STATE_TRANSITION_NEXT (transition)));
261 
262   switch (transition) {
263     case GST_STATE_CHANGE_NULL_TO_READY:
264       if (!self->src) {
265         if (klass->create_element)
266           self->src = klass->create_element ();
267 
268         if (!self->src) {
269           g_signal_emit (element,
270               gst_gl_src_bin_signals[SIGNAL_CREATE_ELEMENT], 0, &self->src);
271           if (self->src && g_object_is_floating (self->src))
272             gst_object_ref_sink (self->src);
273         }
274 
275         if (!self->src) {
276           GST_ERROR_OBJECT (element, "Failed to retrieve element");
277           return GST_STATE_CHANGE_FAILURE;
278         }
279         if (!_connect_src_element (self))
280           return GST_STATE_CHANGE_FAILURE;
281       }
282       break;
283     default:
284       break;
285   }
286 
287   ret = GST_ELEMENT_CLASS (parent_class)->change_state (element, transition);
288   if (ret == GST_STATE_CHANGE_FAILURE)
289     return ret;
290 
291   switch (transition) {
292     default:
293       break;
294   }
295 
296   return ret;
297 }
298