docs: convert NULL, TRUE, and FALSE to %NULL, %TRUE, and %FALSE
[platform/upstream/gstreamer.git] / gst / gstcapsfeatures.c
1 /* GStreamer
2  * Copyright (C) 2013 Collabora Ltd.
3  *   Author: Sebastian Dröge <sebastian.droege@collabora.co.uk>
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 /**
22  * SECTION:gstcapsfeatures
23  * @short_description: A set of features in caps 
24  * @see_also: #GstCaps
25  *
26  * #GstCapsFeatures can optionally be set on a #GstCaps to add requirements
27  * for additional features for a specific #GstStructure. Caps structures with
28  * the same name but with a non-equal set of caps features are not compatible.
29  * If a pad supports multiple sets of features it has to add multiple equal
30  * structures with different feature sets to the caps.
31  *
32  * Empty #GstCapsFeatures are equivalent with the #GstCapsFeatures that only
33  * contain #GST_CAPS_FEATURE_MEMORY_SYSTEM_MEMORY. ANY #GstCapsFeatures as
34  * created by gst_caps_features_new_any() are equal to any other #GstCapsFeatures
35  * and can be used to specify that any #GstCapsFeatures would be supported, e.g.
36  * for elements that don't touch buffer memory. #GstCaps with ANY #GstCapsFeatures
37  * are considered non-fixed and during negotiation some #GstCapsFeatures have
38  * to be selected.
39  *
40  * Examples for caps features would be the requirement of a specific #GstMemory
41  * types or the requirement of having a specific #GstMeta on the buffer. Features
42  * are given as a string of the format "memory:GstMemoryTypeName" or
43  * "meta:GstMetaAPIName".
44  *
45  * Since: 1.2
46  */
47
48 #ifdef HAVE_CONFIG_H
49 #include "config.h"
50 #endif
51
52 #include <string.h>
53 #include "gst_private.h"
54 #include "gstcapsfeatures.h"
55 #include <gst/gst.h>
56
57 GST_DEBUG_CATEGORY_STATIC (gst_caps_features_debug);
58 #define GST_CAT_DEFAULT gst_caps_features_debug
59
60 struct _GstCapsFeatures
61 {
62   GType type;
63   gint *parent_refcount;
64   GArray *array;
65   gboolean is_any;
66 };
67
68 GType _gst_caps_features_type = 0;
69 static gint static_caps_features_parent_refcount = G_MAXINT;
70 GstCapsFeatures *_gst_caps_features_any = NULL;
71 GstCapsFeatures *_gst_caps_features_memory_system_memory = NULL;
72 static GQuark _gst_caps_feature_memory_system_memory = 0;
73
74 G_DEFINE_BOXED_TYPE (GstCapsFeatures, gst_caps_features,
75     gst_caps_features_copy, gst_caps_features_free);
76
77 #define IS_MUTABLE(features) \
78     (!features->parent_refcount || \
79      g_atomic_int_get (features->parent_refcount) == 1)
80
81 static void
82 gst_caps_features_transform_to_string (const GValue * src_value,
83     GValue * dest_value);
84
85 void
86 _priv_gst_caps_features_initialize (void)
87 {
88   GST_DEBUG_CATEGORY_INIT (gst_caps_features_debug, "caps-features", 0,
89       "GstCapsFeatures debug");
90
91   _gst_caps_features_type = gst_caps_features_get_type ();
92   _gst_caps_feature_memory_system_memory =
93       g_quark_from_static_string (GST_CAPS_FEATURE_MEMORY_SYSTEM_MEMORY);
94
95   g_value_register_transform_func (_gst_caps_features_type, G_TYPE_STRING,
96       gst_caps_features_transform_to_string);
97
98   _gst_caps_features_any = gst_caps_features_new_any ();
99   gst_caps_features_set_parent_refcount (_gst_caps_features_any,
100       &static_caps_features_parent_refcount);
101   _gst_caps_features_memory_system_memory =
102       gst_caps_features_new_id (_gst_caps_feature_memory_system_memory, 0);
103   gst_caps_features_set_parent_refcount
104       (_gst_caps_features_memory_system_memory,
105       &static_caps_features_parent_refcount);
106 }
107
108 gboolean
109 gst_is_caps_features (gconstpointer obj)
110 {
111   const GstCapsFeatures *features = obj;
112
113   return (obj != NULL && features->type == _gst_caps_features_type);
114 }
115
116 static gboolean
117 gst_caps_feature_name_is_valid (const gchar * feature)
118 {
119 #ifndef G_DISABLE_CHECKS
120   while (TRUE) {
121     if (g_ascii_isalpha (*feature))
122       feature++;
123     else if (*feature == ':')
124       break;
125     else
126       return FALSE;
127   }
128
129   if (*feature != ':')
130     return FALSE;
131
132   feature++;
133   if (*feature == '\0' || !g_ascii_isalpha (*feature))
134     return FALSE;
135
136   while (TRUE) {
137     if (g_ascii_isalnum (*feature))
138       feature++;
139     else if (*feature == '\0')
140       break;
141     else
142       return FALSE;
143   }
144 #endif
145
146   return TRUE;
147 }
148
149 /**
150  * gst_caps_features_new_empty:
151  *
152  * Creates a new, empty #GstCapsFeatures.
153  *
154  * Free-function: gst_caps_features_free
155  *
156  * Returns: (transfer full): a new, empty #GstCapsFeatures
157  *
158  * Since: 1.2
159  */
160 GstCapsFeatures *
161 gst_caps_features_new_empty (void)
162 {
163   GstCapsFeatures *features;
164
165   features = g_slice_new (GstCapsFeatures);
166   features->type = _gst_caps_features_type;
167   features->parent_refcount = NULL;
168   features->array = g_array_new (FALSE, FALSE, sizeof (GQuark));
169   features->is_any = FALSE;
170
171   GST_TRACE ("created caps features %p", features);
172
173   return features;
174 }
175
176 /**
177  * gst_caps_features_new_any:
178  *
179  * Creates a new, ANY #GstCapsFeatures. This will be equal
180  * to any other #GstCapsFeatures but caps with these are
181  * unfixed.
182  *
183  * Free-function: gst_caps_features_free
184  *
185  * Returns: (transfer full): a new, ANY #GstCapsFeatures
186  *
187  * Since: 1.2
188  */
189 GstCapsFeatures *
190 gst_caps_features_new_any (void)
191 {
192   GstCapsFeatures *features;
193
194   features = gst_caps_features_new_empty ();
195   features->is_any = TRUE;
196
197   return features;
198 }
199
200 /**
201  * gst_caps_features_new:
202  * @feature1: name of first feature to set
203  * @...: additional features
204  *
205  * Creates a new #GstCapsFeatures with the given features.
206  * The last argument must be %NULL.
207  *
208  * Free-function: gst_caps_features_free
209  *
210  * Returns: (transfer full): a new, empty #GstCapsFeatures
211  *
212  * Since: 1.2
213  */
214 GstCapsFeatures *
215 gst_caps_features_new (const gchar * feature1, ...)
216 {
217   GstCapsFeatures *features;
218   va_list varargs;
219
220   g_return_val_if_fail (feature1 != NULL, NULL);
221
222   va_start (varargs, feature1);
223   features = gst_caps_features_new_valist (feature1, varargs);
224   va_end (varargs);
225
226   return features;
227 }
228
229 /**
230  * gst_caps_features_new_valist:
231  * @feature1: name of first feature to set
232  * @varargs: variable argument list
233  *
234  * Creates a new #GstCapsFeatures with the given features.
235  *
236  * Free-function: gst_caps_features_free
237  *
238  * Returns: (transfer full): a new, empty #GstCapsFeatures
239  *
240  * Since: 1.2
241  */
242 GstCapsFeatures *
243 gst_caps_features_new_valist (const gchar * feature1, va_list varargs)
244 {
245   GstCapsFeatures *features;
246
247   g_return_val_if_fail (feature1 != NULL, NULL);
248
249   features = gst_caps_features_new_empty ();
250
251   while (feature1) {
252     gst_caps_features_add (features, feature1);
253     feature1 = va_arg (varargs, const gchar *);
254   }
255
256   return features;
257 }
258
259 /**
260  * gst_caps_features_new_id:
261  * @feature1: name of first feature to set
262  * @...: additional features
263  *
264  * Creates a new #GstCapsFeatures with the given features.
265  * The last argument must be 0.
266  *
267  * Free-function: gst_caps_features_free
268  *
269  * Returns: (transfer full): a new, empty #GstCapsFeatures
270  *
271  * Since: 1.2
272  */
273 GstCapsFeatures *
274 gst_caps_features_new_id (GQuark feature1, ...)
275 {
276   GstCapsFeatures *features;
277   va_list varargs;
278
279   g_return_val_if_fail (feature1 != 0, NULL);
280
281   va_start (varargs, feature1);
282   features = gst_caps_features_new_id_valist (feature1, varargs);
283   va_end (varargs);
284
285   return features;
286 }
287
288 /**
289  * gst_caps_features_new_id_valist:
290  * @feature1: name of first feature to set
291  * @varargs: variable argument list
292  *
293  * Creates a new #GstCapsFeatures with the given features.
294  *
295  * Free-function: gst_caps_features_free
296  *
297  * Returns: (transfer full): a new, empty #GstCapsFeatures
298  *
299  * Since: 1.2
300  */
301 GstCapsFeatures *
302 gst_caps_features_new_id_valist (GQuark feature1, va_list varargs)
303 {
304   GstCapsFeatures *features;
305
306   g_return_val_if_fail (feature1 != 0, NULL);
307
308   features = gst_caps_features_new_empty ();
309
310   while (feature1) {
311     gst_caps_features_add_id (features, feature1);
312     feature1 = va_arg (varargs, GQuark);
313   }
314
315   return features;
316 }
317
318 /**
319  * gst_caps_features_set_parent_refcount:
320  * @features: a #GstCapsFeatures
321  * @refcount: (in): a pointer to the parent's refcount
322  *
323  * Sets the parent_refcount field of #GstCapsFeatures. This field is used to
324  * determine whether a caps features is mutable or not. This function should only be
325  * called by code implementing parent objects of #GstCapsFeatures, as described in
326  * the MT Refcounting section of the design documents.
327  *
328  * Returns: %TRUE if the parent refcount could be set.
329  *
330  * Since: 1.2
331  */
332 gboolean
333 gst_caps_features_set_parent_refcount (GstCapsFeatures * features,
334     gint * refcount)
335 {
336   g_return_val_if_fail (features != NULL, FALSE);
337
338   /* if we have a parent_refcount already, we can only clear
339    * if with a NULL refcount */
340   if (features->parent_refcount) {
341     if (refcount != NULL) {
342       g_return_val_if_fail (refcount == NULL, FALSE);
343       return FALSE;
344     }
345   } else {
346     if (refcount == NULL) {
347       g_return_val_if_fail (refcount != NULL, FALSE);
348       return FALSE;
349     }
350   }
351
352   features->parent_refcount = refcount;
353
354   return TRUE;
355 }
356
357 /**
358  * gst_caps_features_copy:
359  * @features: a #GstCapsFeatures to duplicate
360  *
361  * Duplicates a #GstCapsFeatures and all its values.
362  *
363  * Free-function: gst_caps_features_free
364  *
365  * Returns: (transfer full): a new #GstCapsFeatures.
366  *
367  * Since: 1.2
368  */
369 GstCapsFeatures *
370 gst_caps_features_copy (const GstCapsFeatures * features)
371 {
372   GstCapsFeatures *copy;
373   guint i, n;
374
375   g_return_val_if_fail (features != NULL, NULL);
376
377   copy = gst_caps_features_new_empty ();
378   n = gst_caps_features_get_size (features);
379   for (i = 0; i < n; i++)
380     gst_caps_features_add_id (copy, gst_caps_features_get_nth_id (features, i));
381   copy->is_any = features->is_any;
382
383   return copy;
384 }
385
386 /**
387  * gst_caps_features_free:
388  * @features: (in) (transfer full): the #GstCapsFeatures to free
389  *
390  * Frees a #GstCapsFeatures and all its values. The caps features must not
391  * have a parent when this function is called.
392  *
393  * Since: 1.2
394  */
395 void
396 gst_caps_features_free (GstCapsFeatures * features)
397 {
398   g_return_if_fail (features != NULL);
399   g_return_if_fail (features->parent_refcount == NULL);
400
401   g_array_free (features->array, TRUE);
402 #ifdef USE_POISONING
403   memset (features, 0xff, sizeof (GstCapsFeatures));
404 #endif
405   GST_TRACE ("free caps features %p", features);
406
407   g_slice_free (GstCapsFeatures, features);
408 }
409
410 /**
411  * gst_caps_features_to_string:
412  * @features: a #GstCapsFeatures
413  *
414  * Converts @features to a human-readable string representation.
415  *
416  * For debugging purposes its easier to do something like this:
417  * |[
418  * GST_LOG ("features is %" GST_PTR_FORMAT, features);
419  * ]|
420  * This prints the features in human readable form.
421  *
422  * Free-function: g_free
423  *
424  * Returns: (transfer full): a pointer to string allocated by g_malloc().
425  *     g_free() after usage.
426  *
427  * Since: 1.2
428  */
429 gchar *
430 gst_caps_features_to_string (const GstCapsFeatures * features)
431 {
432   GString *s;
433
434   g_return_val_if_fail (features != NULL, NULL);
435
436   s = g_string_sized_new (FEATURES_ESTIMATED_STRING_LEN (features));
437
438   priv_gst_caps_features_append_to_gstring (features, s);
439
440   return g_string_free (s, FALSE);
441 }
442
443 void
444 priv_gst_caps_features_append_to_gstring (const GstCapsFeatures * features,
445     GString * s)
446 {
447   guint i, n;
448
449   g_return_if_fail (features != NULL);
450
451   if (features->array->len == 0 && features->is_any) {
452     g_string_append (s, "ANY");
453     return;
454   }
455
456   n = features->array->len;
457   for (i = 0; i < n; i++) {
458     GQuark *quark = &g_array_index (features->array, GQuark, i);
459
460     g_string_append (s, g_quark_to_string (*quark));
461     if (i + 1 < n)
462       g_string_append (s, ", ");
463   }
464 }
465
466 /**
467  * gst_caps_features_from_string:
468  * @features: a string representation of a #GstCapsFeatures.
469  *
470  * Creates a #GstCapsFeatures from a string representation.
471  *
472  * Free-function: gst_caps_features_free
473  *
474  * Returns: (transfer full): a new #GstCapsFeatures or %NULL when the string could
475  *     not be parsed. Free with gst_caps_features_free() after use.
476  *
477  * Since: 1.2
478  */
479 GstCapsFeatures *
480 gst_caps_features_from_string (const gchar * features)
481 {
482   GstCapsFeatures *ret;
483   gboolean escape = FALSE;
484   const gchar *features_orig = features;
485   const gchar *feature;
486
487   ret = gst_caps_features_new_empty ();
488
489   if (!features || *features == '\0')
490     return ret;
491
492   if (strcmp (features, "ANY") == 0) {
493     ret->is_any = TRUE;
494     return ret;
495   }
496
497   /* Skip trailing spaces */
498   while (*features == ' ')
499     features++;
500
501   feature = features;
502   while (TRUE) {
503     gchar c = *features;
504
505     if (c == '\\') {
506       escape = TRUE;
507       features++;
508       continue;
509     } else if ((!escape && c == ',') || c == '\0') {
510       guint len = features - feature + 1;
511       gchar *tmp;
512       gchar *p;
513
514       if (len == 1) {
515         g_warning ("Failed deserialize caps features '%s'", features_orig);
516         gst_caps_features_free (ret);
517         return NULL;
518       }
519
520       tmp = g_malloc (len);
521       memcpy (tmp, feature, len - 1);
522       tmp[len - 1] = '\0';
523
524       p = tmp + len - 1;
525       while (*p == ' ') {
526         *p = '\0';
527         p--;
528       }
529
530       if (strstr (tmp, " ") != NULL || *tmp == '\0') {
531         g_free (tmp);
532         g_warning ("Failed deserialize caps features '%s'", features_orig);
533         gst_caps_features_free (ret);
534         return NULL;
535       }
536
537       gst_caps_features_add (ret, tmp);
538       g_free (tmp);
539
540       if (c == '\0')
541         break;
542
543       /* Skip to the next value */
544       features++;
545       while (*features == ' ')
546         features++;
547       feature = features;
548     } else {
549       escape = FALSE;
550       features++;
551     }
552   }
553
554   return ret;
555 }
556
557 /**
558  * gst_caps_features_get_size:
559  * @features: a #GstCapsFeatures.
560  *
561  * Returns the number of features in @features.
562  *
563  * Returns: The number of features in @features.
564  *
565  * Since: 1.2
566  */
567 guint
568 gst_caps_features_get_size (const GstCapsFeatures * features)
569 {
570   g_return_val_if_fail (features != NULL, 0);
571
572   return features->array->len;
573 }
574
575 /**
576  * gst_caps_features_get_nth:
577  * @features: a #GstCapsFeatures.
578  * @i: index of the feature
579  *
580  * Returns the @i-th feature of @features.
581  *
582  * Returns: The @i-th feature of @features.
583  *
584  * Since: 1.2
585  */
586 const gchar *
587 gst_caps_features_get_nth (const GstCapsFeatures * features, guint i)
588 {
589   const gchar *feature;
590   GQuark quark;
591
592   g_return_val_if_fail (features != NULL, NULL);
593
594   quark = gst_caps_features_get_nth_id (features, i);
595   if (!quark)
596     return NULL;
597
598   feature = g_quark_to_string (quark);
599   return feature;
600 }
601
602 /**
603  * gst_caps_features_get_nth_id:
604  * @features: a #GstCapsFeatures.
605  * @i: index of the feature
606  *
607  * Returns the @i-th feature of @features.
608  *
609  * Returns: The @i-th feature of @features.
610  *
611  * Since: 1.2
612  */
613 GQuark
614 gst_caps_features_get_nth_id (const GstCapsFeatures * features, guint i)
615 {
616   GQuark *quark;
617
618   g_return_val_if_fail (features != NULL, 0);
619   g_return_val_if_fail (i < features->array->len, 0);
620
621   quark = &g_array_index (features->array, GQuark, i);
622
623   return *quark;
624 }
625
626 /**
627  * gst_caps_features_contains:
628  * @features: a #GstCapsFeatures.
629  * @feature: a feature
630  *
631  * Check if @features contains @feature.
632  *
633  * Returns: %TRUE if @features contains @feature.
634  *
635  * Since: 1.2
636  */
637 gboolean
638 gst_caps_features_contains (const GstCapsFeatures * features,
639     const gchar * feature)
640 {
641   g_return_val_if_fail (features != NULL, FALSE);
642   g_return_val_if_fail (feature != NULL, FALSE);
643
644   return gst_caps_features_contains_id (features,
645       g_quark_from_string (feature));
646 }
647
648 /**
649  * gst_caps_features_contains_id:
650  * @features: a #GstCapsFeatures.
651  * @feature: a feature
652  *
653  * Check if @features contains @feature.
654  *
655  * Returns: %TRUE if @features contains @feature.
656  *
657  * Since: 1.2
658  */
659 gboolean
660 gst_caps_features_contains_id (const GstCapsFeatures * features, GQuark feature)
661 {
662   guint i, n;
663
664   g_return_val_if_fail (features != NULL, FALSE);
665   g_return_val_if_fail (feature != 0, FALSE);
666
667   if (features->is_any)
668     return TRUE;
669
670   n = features->array->len;
671   if (n == 0)
672     return feature == _gst_caps_feature_memory_system_memory;
673
674   for (i = 0; i < n; i++) {
675     if (gst_caps_features_get_nth_id (features, i) == feature)
676       return TRUE;
677   }
678
679   return FALSE;
680 }
681
682 /**
683  * gst_caps_features_is_equal:
684  * @features1: a #GstCapsFeatures.
685  * @features2: a #GstCapsFeatures.
686  *
687  * Check if @features1 and @features2 are equal.
688  *
689  * Returns: %TRUE if @features1 and @features2 are equal.
690  *
691  * Since: 1.2
692  */
693 gboolean
694 gst_caps_features_is_equal (const GstCapsFeatures * features1,
695     const GstCapsFeatures * features2)
696 {
697   guint i, n;
698
699   g_return_val_if_fail (features1 != NULL, FALSE);
700   g_return_val_if_fail (features2 != NULL, FALSE);
701
702   if (features1->is_any || features2->is_any)
703     return TRUE;
704
705   /* Check for the sysmem==empty case */
706   if (features1->array->len == 0 && features2->array->len == 0)
707     return TRUE;
708   if (features1->array->len == 0 && features2->array->len == 1
709       && gst_caps_features_contains_id (features2,
710           _gst_caps_feature_memory_system_memory))
711     return TRUE;
712   if (features2->array->len == 0 && features1->array->len == 1
713       && gst_caps_features_contains_id (features1,
714           _gst_caps_feature_memory_system_memory))
715     return TRUE;
716
717   if (features1->array->len != features2->array->len)
718     return FALSE;
719
720   n = features1->array->len;
721   for (i = 0; i < n; i++)
722     if (!gst_caps_features_contains_id (features2,
723             gst_caps_features_get_nth_id (features1, i)))
724       return FALSE;
725
726   return TRUE;
727 }
728
729 /**
730  * gst_caps_features_is_any:
731  * @features: a #GstCapsFeatures.
732  *
733  * Check if @features is %GST_CAPS_FEATURES_ANY.
734  *
735  * Returns: %TRUE if @features is %GST_CAPS_FEATURES_ANY.
736  *
737  * Since: 1.2
738  */
739 gboolean
740 gst_caps_features_is_any (const GstCapsFeatures * features)
741 {
742   g_return_val_if_fail (features != NULL, FALSE);
743
744   return features->is_any;
745 }
746
747 /**
748  * gst_caps_features_add:
749  * @features: a #GstCapsFeatures.
750  * @feature: a feature.
751  *
752  * Adds @feature to @features.
753  *
754  * Since: 1.2
755  */
756 void
757 gst_caps_features_add (GstCapsFeatures * features, const gchar * feature)
758 {
759   g_return_if_fail (features != NULL);
760   g_return_if_fail (IS_MUTABLE (features));
761   g_return_if_fail (feature != NULL);
762   g_return_if_fail (!features->is_any);
763
764   gst_caps_features_add_id (features, g_quark_from_string (feature));
765 }
766
767 /**
768  * gst_caps_features_add_id:
769  * @features: a #GstCapsFeatures.
770  * @feature: a feature.
771  *
772  * Adds @feature to @features.
773  *
774  * Since: 1.2
775  */
776 void
777 gst_caps_features_add_id (GstCapsFeatures * features, GQuark feature)
778 {
779   g_return_if_fail (features != NULL);
780   g_return_if_fail (IS_MUTABLE (features));
781   g_return_if_fail (feature != 0);
782   g_return_if_fail (!features->is_any);
783
784   if (!gst_caps_feature_name_is_valid (g_quark_to_string (feature))) {
785     g_warning ("Invalid caps feature name: %s", g_quark_to_string (feature));
786     return;
787   }
788
789   /* If features is empty it will contain sysmem, however
790    * we want to add it explicitely if it is tried to be
791    * added as first features
792    */
793   if (features->array->len > 0
794       && gst_caps_features_contains_id (features, feature))
795     return;
796
797   g_array_append_val (features->array, feature);
798 }
799
800 /**
801  * gst_caps_features_remove:
802  * @features: a #GstCapsFeatures.
803  * @feature: a feature.
804  *
805  * Removes @feature from @features.
806  *
807  * Since: 1.2
808  */
809 void
810 gst_caps_features_remove (GstCapsFeatures * features, const gchar * feature)
811 {
812   g_return_if_fail (features != NULL);
813   g_return_if_fail (IS_MUTABLE (features));
814   g_return_if_fail (feature != NULL);
815
816   gst_caps_features_remove_id (features, g_quark_from_string (feature));
817 }
818
819 /**
820  * gst_caps_features_remove_id:
821  * @features: a #GstCapsFeatures.
822  * @feature: a feature.
823  *
824  * Removes @feature from @features.
825  *
826  * Since: 1.2
827  */
828 void
829 gst_caps_features_remove_id (GstCapsFeatures * features, GQuark feature)
830 {
831   guint i, n;
832
833   g_return_if_fail (features != NULL);
834   g_return_if_fail (IS_MUTABLE (features));
835   g_return_if_fail (feature != 0);
836
837   n = features->array->len;
838   for (i = 0; i < n; i++) {
839     GQuark quark = gst_caps_features_get_nth_id (features, i);
840
841     if (quark == feature) {
842       g_array_remove_index_fast (features->array, i);
843       return;
844     }
845   }
846 }
847
848 static void
849 gst_caps_features_transform_to_string (const GValue * src_value,
850     GValue * dest_value)
851 {
852   g_return_if_fail (src_value != NULL);
853   g_return_if_fail (dest_value != NULL);
854
855   dest_value->data[0].v_pointer =
856       gst_caps_features_to_string (src_value->data[0].v_pointer);
857 }