context: Fix unit test for GstContext changes
[platform/upstream/gstreamer.git] / gst / gstcontext.c
1 /* GStreamer
2  * Copyright (C) 2013 Collabora Ltd.
3  *   Author: Sebastian Dröge <sebastian.droege@collabora.co.uk>
4  * Copyright (C) 2013 Sebastian Dröge <slomo@circular-chaos.org>
5  *
6  * gstcontext.h: Header for GstContext subsystem
7  *
8  * This library is free software; you can redistribute it and/or
9  * modify it under the terms of the GNU Library General Public
10  * License as published by the Free Software Foundation; either
11  * version 2 of the License, or (at your option) any later version.
12  *
13  * This library is distributed in the hope that it will be useful,
14  * but WITHOUT ANY WARRANTY; without even the implied warranty of
15  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
16  * Library General Public License for more details.
17  *
18  * You should have received a copy of the GNU Library General Public
19  * License along with this library; if not, write to the
20  * Free Software Foundation, Inc., 51 Franklin St, Fifth Floor,
21  * Boston, MA 02110-1301, USA.
22  */
23
24 /**
25  * SECTION:gstcontext
26  * @short_description: Lightweight objects to represent element contexts
27  * @see_also: #GstMiniObject, #GstElement
28  *
29  * #GstContext is a container object used to store contexts like a device
30  * context, a display server connection and similar concepts that should
31  * be shared between multiple elements.
32  *
33  * Applications can set a context on a complete pipeline by using
34  * gst_element_set_context(), which will then be propagated to all
35  * child elements. Elements can handle these in GstElement::set_context()
36  * and merge them with the context information they already have.
37  *
38  * When an element needs a context it will do the following actions in this
39  * order until one step succeeds:
40  * 1) Check if the element already has a context
41  * 2) Query downstream with GST_QUERY_CONTEXT for the context
42  * 2) Query upstream with GST_QUERY_CONTEXT for the context
43  * 3) Post a GST_MESSAGE_NEED_CONTEXT message on the bus with the required
44  *    context types and afterwards check if a usable context was set now
45  * 4) Create a context by itself and post a GST_MESSAGE_HAVE_CONTEXT message
46  *    on the bus.
47  *
48  * Bins will catch GST_MESSAGE_NEED_CONTEXT messages and will set any previously
49  * known context on the element that asks for it if possible. Otherwise the
50  * application should provide one if it can.
51  *
52  * Since: 1.2
53  */
54
55 #include "gst_private.h"
56 #include <string.h>
57 #include "gstcontext.h"
58 #include "gstquark.h"
59
60 struct _GstContext
61 {
62   GstMiniObject mini_object;
63
64   gchar *context_type;
65   GstStructure *structure;
66   gboolean persistent;
67 };
68
69 #define GST_CONTEXT_STRUCTURE(c)  (((GstContext *)(c))->structure)
70
71 static GType _gst_context_type = 0;
72 GST_DEFINE_MINI_OBJECT_TYPE (GstContext, gst_context);
73
74 void
75 _priv_gst_context_initialize (void)
76 {
77   GST_CAT_INFO (GST_CAT_GST_INIT, "init contexts");
78
79   /* the GstMiniObject types need to be class_ref'd once before it can be
80    * done from multiple threads;
81    * see http://bugzilla.gnome.org/show_bug.cgi?id=304551 */
82   gst_context_get_type ();
83
84   _gst_context_type = gst_context_get_type ();
85 }
86
87 static void
88 _gst_context_free (GstContext * context)
89 {
90   GstStructure *structure;
91
92   g_return_if_fail (context != NULL);
93
94   GST_CAT_LOG (GST_CAT_CONTEXT, "finalize context %p: %" GST_PTR_FORMAT,
95       context, GST_CONTEXT_STRUCTURE (context));
96
97   structure = GST_CONTEXT_STRUCTURE (context);
98   if (structure) {
99     gst_structure_set_parent_refcount (structure, NULL);
100     gst_structure_free (structure);
101   }
102   g_free (context->context_type);
103
104   g_slice_free1 (sizeof (GstContext), context);
105 }
106
107 static void gst_context_init (GstContext * context);
108
109 static GstContext *
110 _gst_context_copy (GstContext * context)
111 {
112   GstContext *copy;
113   GstStructure *structure;
114
115   GST_CAT_LOG (GST_CAT_CONTEXT, "copy context %p: %" GST_PTR_FORMAT, context,
116       GST_CONTEXT_STRUCTURE (context));
117
118   copy = g_slice_new0 (GstContext);
119
120   gst_context_init (copy);
121
122   copy->context_type = g_strdup (context->context_type);
123
124   structure = GST_CONTEXT_STRUCTURE (context);
125   GST_CONTEXT_STRUCTURE (copy) = gst_structure_copy (structure);
126   gst_structure_set_parent_refcount (GST_CONTEXT_STRUCTURE (copy),
127       &copy->mini_object.refcount);
128
129   copy->persistent = context->persistent;
130
131   return GST_CONTEXT_CAST (copy);
132 }
133
134 static void
135 gst_context_init (GstContext * context)
136 {
137   gst_mini_object_init (GST_MINI_OBJECT_CAST (context), 0, _gst_context_type,
138       (GstMiniObjectCopyFunction) _gst_context_copy, NULL,
139       (GstMiniObjectFreeFunction) _gst_context_free);
140 }
141
142 /**
143  * gst_context_new:
144  * @persistent: Persistent context
145  *
146  * Create a new context.
147  *
148  * Returns: (transfer full): The new context.
149  *
150  * Since: 1.2
151  */
152 GstContext *
153 gst_context_new (const gchar * context_type, gboolean persistent)
154 {
155   GstContext *context;
156   GstStructure *structure;
157
158   g_return_val_if_fail (context_type != NULL, NULL);
159
160   context = g_slice_new0 (GstContext);
161
162   GST_CAT_LOG (GST_CAT_CONTEXT, "creating new context %p", context);
163
164   structure = gst_structure_new_id_empty (GST_QUARK (CONTEXT));
165   gst_structure_set_parent_refcount (structure, &context->mini_object.refcount);
166   gst_context_init (context);
167
168   context->context_type = g_strdup (context_type);
169   GST_CONTEXT_STRUCTURE (context) = structure;
170   context->persistent = persistent;
171
172   return context;
173 }
174
175 /**
176  * gst_context_get_context_type:
177  * @context: The #GstContext.
178  *
179  * Get the type of @context.
180  *
181  * Returns: The type of the context.
182  *
183  * Since: 1.2
184  */
185 const gchar *
186 gst_context_get_context_type (const GstContext * context)
187 {
188   g_return_val_if_fail (GST_IS_CONTEXT (context), NULL);
189
190   return context->context_type;
191 }
192
193 /**
194  * gst_context_get_structure:
195  * @context: The #GstContext.
196  *
197  * Access the structure of the context.
198  *
199  * Returns: (transfer none): The structure of the context. The structure is
200  * still owned by the context, which means that you should not modify it,
201  * free it and that the pointer becomes invalid when you free the context.
202  *
203  * Since: 1.2
204  */
205 const GstStructure *
206 gst_context_get_structure (const GstContext * context)
207 {
208   g_return_val_if_fail (GST_IS_CONTEXT (context), NULL);
209
210   return GST_CONTEXT_STRUCTURE (context);
211 }
212
213 /**
214  * gst_context_writable_structure:
215  * @context: The #GstContext.
216  *
217  * Get a writable version of the structure.
218  *
219  * Returns: The structure of the context. The structure is still
220  * owned by the event, which means that you should not free it and
221  * that the pointer becomes invalid when you free the event.
222  * This function checks if @context is writable.
223  *
224  * Since: 1.2
225  */
226 GstStructure *
227 gst_context_writable_structure (GstContext * context)
228 {
229   g_return_val_if_fail (GST_IS_CONTEXT (context), NULL);
230   g_return_val_if_fail (gst_context_is_writable (context), NULL);
231
232   return GST_CONTEXT_STRUCTURE (context);
233 }
234
235 /**
236  * gst_context_is_persistent:
237  * @context: The #GstContext.
238  *
239  * Check if @context is persistent.
240  *
241  * Returns: %TRUE if the context is persistent.
242  *
243  * Since: 1.2
244  */
245 gboolean
246 gst_context_is_persistent (const GstContext * context)
247 {
248   g_return_val_if_fail (GST_IS_CONTEXT (context), NULL);
249
250   return context->persistent;
251 }