Blame gst/gstcapsfeatures.c

Packit Service 963350
/* GStreamer
Packit Service 963350
 * Copyright (C) 2013 Collabora Ltd.
Packit Service 963350
 *   Author: Sebastian Dröge <sebastian.droege@collabora.co.uk>
Packit Service 963350
 *
Packit Service 963350
 * This library is free software; you can redistribute it and/or
Packit Service 963350
 * modify it under the terms of the GNU Library General Public
Packit Service 963350
 * License as published by the Free Software Foundation; either
Packit Service 963350
 * version 2 of the License, or (at your option) any later version.
Packit Service 963350
 *
Packit Service 963350
 * This library is distributed in the hope that it will be useful,
Packit Service 963350
 * but WITHOUT ANY WARRANTY; without even the implied warranty of
Packit Service 963350
 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
Packit Service 963350
 * Library General Public License for more details.
Packit Service 963350
 *
Packit Service 963350
 * You should have received a copy of the GNU Library General Public
Packit Service 963350
 * License along with this library; if not, write to the
Packit Service 963350
 * Free Software Foundation, Inc., 51 Franklin St, Fifth Floor,
Packit Service 963350
 * Boston, MA 02110-1301, USA.
Packit Service 963350
 */
Packit Service 963350
Packit Service 963350
/**
Packit Service 963350
 * SECTION:gstcapsfeatures
Packit Service 963350
 * @title: GstCapsFeatures
Packit Service 963350
 * @short_description: A set of features in caps
Packit Service 963350
 * @see_also: #GstCaps
Packit Service 963350
 *
Packit Service 963350
 * #GstCapsFeatures can optionally be set on a #GstCaps to add requirements
Packit Service 963350
 * for additional features for a specific #GstStructure. Caps structures with
Packit Service 963350
 * the same name but with a non-equal set of caps features are not compatible.
Packit Service 963350
 * If a pad supports multiple sets of features it has to add multiple equal
Packit Service 963350
 * structures with different feature sets to the caps.
Packit Service 963350
 *
Packit Service 963350
 * Empty #GstCapsFeatures are equivalent with the #GstCapsFeatures that only
Packit Service 963350
 * contain #GST_CAPS_FEATURE_MEMORY_SYSTEM_MEMORY. ANY #GstCapsFeatures as
Packit Service 963350
 * created by gst_caps_features_new_any() are equal to any other #GstCapsFeatures
Packit Service 963350
 * and can be used to specify that any #GstCapsFeatures would be supported, e.g.
Packit Service 963350
 * for elements that don't touch buffer memory. #GstCaps with ANY #GstCapsFeatures
Packit Service 963350
 * are considered non-fixed and during negotiation some #GstCapsFeatures have
Packit Service 963350
 * to be selected.
Packit Service 963350
 *
Packit Service 963350
 * Examples for caps features would be the requirement of a specific #GstMemory
Packit Service 963350
 * types or the requirement of having a specific #GstMeta on the buffer. Features
Packit Service 963350
 * are given as a string of the format "memory:GstMemoryTypeName" or
Packit Service 963350
 * "meta:GstMetaAPIName".
Packit Service 963350
 *
Packit Service 963350
 * Since: 1.2
Packit Service 963350
 */
Packit Service 963350
Packit Service 963350
#ifdef HAVE_CONFIG_H
Packit Service 963350
#include "config.h"
Packit Service 963350
#endif
Packit Service 963350
Packit Service 963350
#include <string.h>
Packit Service 963350
#include "gst_private.h"
Packit Service 963350
#include "gstcapsfeatures.h"
Packit Service 963350
#include <gst/gst.h>
Packit Service 963350
Packit Service 963350
GST_DEBUG_CATEGORY_STATIC (gst_caps_features_debug);
Packit Service 963350
#define GST_CAT_DEFAULT gst_caps_features_debug
Packit Service 963350
Packit Service 963350
struct _GstCapsFeatures
Packit Service 963350
{
Packit Service 963350
  GType type;
Packit Service 963350
  gint *parent_refcount;
Packit Service 963350
  GArray *array;
Packit Service 963350
  gboolean is_any;
Packit Service 963350
};
Packit Service 963350
Packit Service 963350
GType _gst_caps_features_type = 0;
Packit Service 963350
static gint static_caps_features_parent_refcount = G_MAXINT;
Packit Service 963350
GstCapsFeatures *_gst_caps_features_any = NULL;
Packit Service 963350
GstCapsFeatures *_gst_caps_features_memory_system_memory = NULL;
Packit Service 963350
static GQuark _gst_caps_feature_memory_system_memory = 0;
Packit Service 963350
Packit Service 963350
G_DEFINE_BOXED_TYPE (GstCapsFeatures, gst_caps_features,
Packit Service 963350
    gst_caps_features_copy, gst_caps_features_free);
Packit Service 963350
Packit Service 963350
#define IS_MUTABLE(features) \
Packit Service 963350
    (!features->parent_refcount || \
Packit Service 963350
     g_atomic_int_get (features->parent_refcount) == 1)
Packit Service 963350
Packit Service 963350
static void
Packit Service 963350
gst_caps_features_transform_to_string (const GValue * src_value,
Packit Service 963350
    GValue * dest_value);
Packit Service 963350
Packit Service 963350
void
Packit Service 963350
_priv_gst_caps_features_initialize (void)
Packit Service 963350
{
Packit Service 963350
  GST_DEBUG_CATEGORY_INIT (gst_caps_features_debug, "caps-features", 0,
Packit Service 963350
      "GstCapsFeatures debug");
Packit Service 963350
Packit Service 963350
  _gst_caps_features_type = gst_caps_features_get_type ();
Packit Service 963350
  _gst_caps_feature_memory_system_memory =
Packit Service 963350
      g_quark_from_static_string (GST_CAPS_FEATURE_MEMORY_SYSTEM_MEMORY);
Packit Service 963350
Packit Service 963350
  g_value_register_transform_func (_gst_caps_features_type, G_TYPE_STRING,
Packit Service 963350
      gst_caps_features_transform_to_string);
Packit Service 963350
Packit Service 963350
  _gst_caps_features_any = gst_caps_features_new_any ();
Packit Service 963350
  gst_caps_features_set_parent_refcount (_gst_caps_features_any,
Packit Service 963350
      &static_caps_features_parent_refcount);
Packit Service 963350
  _gst_caps_features_memory_system_memory =
Packit Service 963350
      gst_caps_features_new_id (_gst_caps_feature_memory_system_memory, 0);
Packit Service 963350
  gst_caps_features_set_parent_refcount
Packit Service 963350
      (_gst_caps_features_memory_system_memory,
Packit Service 963350
      &static_caps_features_parent_refcount);
Packit Service 963350
}
Packit Service 963350
Packit Service 963350
void
Packit Service 963350
_priv_gst_caps_features_cleanup (void)
Packit Service 963350
{
Packit Service 963350
  gst_caps_features_set_parent_refcount (_gst_caps_features_any, NULL);
Packit Service 963350
  gst_caps_features_free (_gst_caps_features_any);
Packit Service 963350
  _gst_caps_features_any = NULL;
Packit Service 963350
  gst_caps_features_set_parent_refcount
Packit Service 963350
      (_gst_caps_features_memory_system_memory, NULL);
Packit Service 963350
  gst_caps_features_free (_gst_caps_features_memory_system_memory);
Packit Service 963350
  _gst_caps_features_memory_system_memory = NULL;
Packit Service 963350
}
Packit Service 963350
Packit Service 963350
gboolean
Packit Service 963350
gst_is_caps_features (gconstpointer obj)
Packit Service 963350
{
Packit Service 963350
  const GstCapsFeatures *features = obj;
Packit Service 963350
Packit Service 963350
  return (obj != NULL && features->type == _gst_caps_features_type);
Packit Service 963350
}
Packit Service 963350
Packit Service 963350
static gboolean
Packit Service 963350
gst_caps_feature_name_is_valid (const gchar * feature)
Packit Service 963350
{
Packit Service 963350
#ifndef G_DISABLE_CHECKS
Packit Service 963350
  while (TRUE) {
Packit Service 963350
    if (g_ascii_isalpha (*feature))
Packit Service 963350
      feature++;
Packit Service 963350
    else if (*feature == ':')
Packit Service 963350
      break;
Packit Service 963350
    else
Packit Service 963350
      return FALSE;
Packit Service 963350
  }
Packit Service 963350
Packit Service 963350
  if (*feature != ':')
Packit Service 963350
    return FALSE;
Packit Service 963350
Packit Service 963350
  feature++;
Packit Service 963350
  if (*feature == '\0' || !g_ascii_isalpha (*feature))
Packit Service 963350
    return FALSE;
Packit Service 963350
Packit Service 963350
  while (TRUE) {
Packit Service 963350
    if (g_ascii_isalnum (*feature))
Packit Service 963350
      feature++;
Packit Service 963350
    else if (*feature == '\0')
Packit Service 963350
      break;
Packit Service 963350
    else
Packit Service 963350
      return FALSE;
Packit Service 963350
  }
Packit Service 963350
#endif
Packit Service 963350
Packit Service 963350
  return TRUE;
Packit Service 963350
}
Packit Service 963350
Packit Service 963350
/**
Packit Service 963350
 * gst_caps_features_new_empty:
Packit Service 963350
 *
Packit Service 963350
 * Creates a new, empty #GstCapsFeatures.
Packit Service 963350
 *
Packit Service 963350
 * Free-function: gst_caps_features_free
Packit Service 963350
 *
Packit Service 963350
 * Returns: (transfer full): a new, empty #GstCapsFeatures
Packit Service 963350
 *
Packit Service 963350
 * Since: 1.2
Packit Service 963350
 */
Packit Service 963350
GstCapsFeatures *
Packit Service 963350
gst_caps_features_new_empty (void)
Packit Service 963350
{
Packit Service 963350
  GstCapsFeatures *features;
Packit Service 963350
Packit Service 963350
  features = g_slice_new (GstCapsFeatures);
Packit Service 963350
  features->type = _gst_caps_features_type;
Packit Service 963350
  features->parent_refcount = NULL;
Packit Service 963350
  features->array = g_array_new (FALSE, FALSE, sizeof (GQuark));
Packit Service 963350
  features->is_any = FALSE;
Packit Service 963350
Packit Service 963350
  GST_TRACE ("created caps features %p", features);
Packit Service 963350
Packit Service 963350
  return features;
Packit Service 963350
}
Packit Service 963350
Packit Service 963350
/**
Packit Service 963350
 * gst_caps_features_new_any:
Packit Service 963350
 *
Packit Service 963350
 * Creates a new, ANY #GstCapsFeatures. This will be equal
Packit Service 963350
 * to any other #GstCapsFeatures but caps with these are
Packit Service 963350
 * unfixed.
Packit Service 963350
 *
Packit Service 963350
 * Free-function: gst_caps_features_free
Packit Service 963350
 *
Packit Service 963350
 * Returns: (transfer full): a new, ANY #GstCapsFeatures
Packit Service 963350
 *
Packit Service 963350
 * Since: 1.2
Packit Service 963350
 */
Packit Service 963350
GstCapsFeatures *
Packit Service 963350
gst_caps_features_new_any (void)
Packit Service 963350
{
Packit Service 963350
  GstCapsFeatures *features;
Packit Service 963350
Packit Service 963350
  features = gst_caps_features_new_empty ();
Packit Service 963350
  features->is_any = TRUE;
Packit Service 963350
Packit Service 963350
  return features;
Packit Service 963350
}
Packit Service 963350
Packit Service 963350
/**
Packit Service 963350
 * gst_caps_features_new:
Packit Service 963350
 * @feature1: name of first feature to set
Packit Service 963350
 * @...: additional features
Packit Service 963350
 *
Packit Service 963350
 * Creates a new #GstCapsFeatures with the given features.
Packit Service 963350
 * The last argument must be %NULL.
Packit Service 963350
 *
Packit Service 963350
 * Free-function: gst_caps_features_free
Packit Service 963350
 *
Packit Service 963350
 * Returns: (transfer full): a new, empty #GstCapsFeatures
Packit Service 963350
 *
Packit Service 963350
 * Since: 1.2
Packit Service 963350
 */
Packit Service 963350
GstCapsFeatures *
Packit Service 963350
gst_caps_features_new (const gchar * feature1, ...)
Packit Service 963350
{
Packit Service 963350
  GstCapsFeatures *features;
Packit Service 963350
  va_list varargs;
Packit Service 963350
Packit Service 963350
  g_return_val_if_fail (feature1 != NULL, NULL);
Packit Service 963350
Packit Service 963350
  va_start (varargs, feature1);
Packit Service 963350
  features = gst_caps_features_new_valist (feature1, varargs);
Packit Service 963350
  va_end (varargs);
Packit Service 963350
Packit Service 963350
  return features;
Packit Service 963350
}
Packit Service 963350
Packit Service 963350
/**
Packit Service 963350
 * gst_caps_features_new_valist:
Packit Service 963350
 * @feature1: name of first feature to set
Packit Service 963350
 * @varargs: variable argument list
Packit Service 963350
 *
Packit Service 963350
 * Creates a new #GstCapsFeatures with the given features.
Packit Service 963350
 *
Packit Service 963350
 * Free-function: gst_caps_features_free
Packit Service 963350
 *
Packit Service 963350
 * Returns: (transfer full): a new, empty #GstCapsFeatures
Packit Service 963350
 *
Packit Service 963350
 * Since: 1.2
Packit Service 963350
 */
Packit Service 963350
GstCapsFeatures *
Packit Service 963350
gst_caps_features_new_valist (const gchar * feature1, va_list varargs)
Packit Service 963350
{
Packit Service 963350
  GstCapsFeatures *features;
Packit Service 963350
Packit Service 963350
  g_return_val_if_fail (feature1 != NULL, NULL);
Packit Service 963350
Packit Service 963350
  features = gst_caps_features_new_empty ();
Packit Service 963350
Packit Service 963350
  while (feature1) {
Packit Service 963350
    gst_caps_features_add (features, feature1);
Packit Service 963350
    feature1 = va_arg (varargs, const gchar *);
Packit Service 963350
  }
Packit Service 963350
Packit Service 963350
  return features;
Packit Service 963350
}
Packit Service 963350
Packit Service 963350
/**
Packit Service 963350
 * gst_caps_features_new_id:
Packit Service 963350
 * @feature1: name of first feature to set
Packit Service 963350
 * @...: additional features
Packit Service 963350
 *
Packit Service 963350
 * Creates a new #GstCapsFeatures with the given features.
Packit Service 963350
 * The last argument must be 0.
Packit Service 963350
 *
Packit Service 963350
 * Free-function: gst_caps_features_free
Packit Service 963350
 *
Packit Service 963350
 * Returns: (transfer full): a new, empty #GstCapsFeatures
Packit Service 963350
 *
Packit Service 963350
 * Since: 1.2
Packit Service 963350
 */
Packit Service 963350
GstCapsFeatures *
Packit Service 963350
gst_caps_features_new_id (GQuark feature1, ...)
Packit Service 963350
{
Packit Service 963350
  GstCapsFeatures *features;
Packit Service 963350
  va_list varargs;
Packit Service 963350
Packit Service 963350
  g_return_val_if_fail (feature1 != 0, NULL);
Packit Service 963350
Packit Service 963350
  va_start (varargs, feature1);
Packit Service 963350
  features = gst_caps_features_new_id_valist (feature1, varargs);
Packit Service 963350
  va_end (varargs);
Packit Service 963350
Packit Service 963350
  return features;
Packit Service 963350
}
Packit Service 963350
Packit Service 963350
/**
Packit Service 963350
 * gst_caps_features_new_id_valist:
Packit Service 963350
 * @feature1: name of first feature to set
Packit Service 963350
 * @varargs: variable argument list
Packit Service 963350
 *
Packit Service 963350
 * Creates a new #GstCapsFeatures with the given features.
Packit Service 963350
 *
Packit Service 963350
 * Free-function: gst_caps_features_free
Packit Service 963350
 *
Packit Service 963350
 * Returns: (transfer full): a new, empty #GstCapsFeatures
Packit Service 963350
 *
Packit Service 963350
 * Since: 1.2
Packit Service 963350
 */
Packit Service 963350
GstCapsFeatures *
Packit Service 963350
gst_caps_features_new_id_valist (GQuark feature1, va_list varargs)
Packit Service 963350
{
Packit Service 963350
  GstCapsFeatures *features;
Packit Service 963350
Packit Service 963350
  g_return_val_if_fail (feature1 != 0, NULL);
Packit Service 963350
Packit Service 963350
  features = gst_caps_features_new_empty ();
Packit Service 963350
Packit Service 963350
  while (feature1) {
Packit Service 963350
    gst_caps_features_add_id (features, feature1);
Packit Service 963350
    feature1 = va_arg (varargs, GQuark);
Packit Service 963350
  }
Packit Service 963350
Packit Service 963350
  return features;
Packit Service 963350
}
Packit Service 963350
Packit Service 963350
/**
Packit Service 963350
 * gst_caps_features_set_parent_refcount:
Packit Service 963350
 * @features: a #GstCapsFeatures
Packit Service 963350
 * @refcount: (in): a pointer to the parent's refcount
Packit Service 963350
 *
Packit Service 963350
 * Sets the parent_refcount field of #GstCapsFeatures. This field is used to
Packit Service 963350
 * determine whether a caps features is mutable or not. This function should only be
Packit Service 963350
 * called by code implementing parent objects of #GstCapsFeatures, as described in
Packit Service 963350
 * the MT Refcounting section of the design documents.
Packit Service 963350
 *
Packit Service 963350
 * Returns: %TRUE if the parent refcount could be set.
Packit Service 963350
 *
Packit Service 963350
 * Since: 1.2
Packit Service 963350
 */
Packit Service 963350
gboolean
Packit Service 963350
gst_caps_features_set_parent_refcount (GstCapsFeatures * features,
Packit Service 963350
    gint * refcount)
Packit Service 963350
{
Packit Service 963350
  g_return_val_if_fail (features != NULL, FALSE);
Packit Service 963350
Packit Service 963350
  /* if we have a parent_refcount already, we can only clear
Packit Service 963350
   * if with a NULL refcount */
Packit Service 963350
  if (features->parent_refcount) {
Packit Service 963350
    if (refcount != NULL) {
Packit Service 963350
      g_return_val_if_fail (refcount == NULL, FALSE);
Packit Service 963350
      return FALSE;
Packit Service 963350
    }
Packit Service 963350
  } else {
Packit Service 963350
    if (refcount == NULL) {
Packit Service 963350
      g_return_val_if_fail (refcount != NULL, FALSE);
Packit Service 963350
      return FALSE;
Packit Service 963350
    }
Packit Service 963350
  }
Packit Service 963350
Packit Service 963350
  features->parent_refcount = refcount;
Packit Service 963350
Packit Service 963350
  return TRUE;
Packit Service 963350
}
Packit Service 963350
Packit Service 963350
/**
Packit Service 963350
 * gst_caps_features_copy:
Packit Service 963350
 * @features: a #GstCapsFeatures to duplicate
Packit Service 963350
 *
Packit Service 963350
 * Duplicates a #GstCapsFeatures and all its values.
Packit Service 963350
 *
Packit Service 963350
 * Free-function: gst_caps_features_free
Packit Service 963350
 *
Packit Service 963350
 * Returns: (transfer full): a new #GstCapsFeatures.
Packit Service 963350
 *
Packit Service 963350
 * Since: 1.2
Packit Service 963350
 */
Packit Service 963350
GstCapsFeatures *
Packit Service 963350
gst_caps_features_copy (const GstCapsFeatures * features)
Packit Service 963350
{
Packit Service 963350
  GstCapsFeatures *copy;
Packit Service 963350
  guint i, n;
Packit Service 963350
Packit Service 963350
  g_return_val_if_fail (features != NULL, NULL);
Packit Service 963350
Packit Service 963350
  copy = gst_caps_features_new_empty ();
Packit Service 963350
  n = gst_caps_features_get_size (features);
Packit Service 963350
  for (i = 0; i < n; i++)
Packit Service 963350
    gst_caps_features_add_id (copy, gst_caps_features_get_nth_id (features, i));
Packit Service 963350
  copy->is_any = features->is_any;
Packit Service 963350
Packit Service 963350
  return copy;
Packit Service 963350
}
Packit Service 963350
Packit Service 963350
/**
Packit Service 963350
 * gst_caps_features_free:
Packit Service 963350
 * @features: (in) (transfer full): the #GstCapsFeatures to free
Packit Service 963350
 *
Packit Service 963350
 * Frees a #GstCapsFeatures and all its values. The caps features must not
Packit Service 963350
 * have a parent when this function is called.
Packit Service 963350
 *
Packit Service 963350
 * Since: 1.2
Packit Service 963350
 */
Packit Service 963350
void
Packit Service 963350
gst_caps_features_free (GstCapsFeatures * features)
Packit Service 963350
{
Packit Service 963350
  g_return_if_fail (features != NULL);
Packit Service 963350
  g_return_if_fail (features->parent_refcount == NULL);
Packit Service 963350
Packit Service 963350
  g_array_free (features->array, TRUE);
Packit Service 963350
#ifdef USE_POISONING
Packit Service 963350
  memset (features, 0xff, sizeof (GstCapsFeatures));
Packit Service 963350
#endif
Packit Service 963350
  GST_TRACE ("free caps features %p", features);
Packit Service 963350
Packit Service 963350
  g_slice_free (GstCapsFeatures, features);
Packit Service 963350
}
Packit Service 963350
Packit Service 963350
/**
Packit Service 963350
 * gst_caps_features_to_string:
Packit Service 963350
 * @features: a #GstCapsFeatures
Packit Service 963350
 *
Packit Service 963350
 * Converts @features to a human-readable string representation.
Packit Service 963350
 *
Packit Service 963350
 * For debugging purposes its easier to do something like this:
Packit Service 963350
 * |[
Packit Service 963350
 * GST_LOG ("features is %" GST_PTR_FORMAT, features);
Packit Service 963350
 * ]|
Packit Service 963350
 * This prints the features in human readable form.
Packit Service 963350
 *
Packit Service 963350
 * Free-function: g_free
Packit Service 963350
 *
Packit Service 963350
 * Returns: (transfer full): a pointer to string allocated by g_malloc().
Packit Service 963350
 *     g_free() after usage.
Packit Service 963350
 *
Packit Service 963350
 * Since: 1.2
Packit Service 963350
 */
Packit Service 963350
gchar *
Packit Service 963350
gst_caps_features_to_string (const GstCapsFeatures * features)
Packit Service 963350
{
Packit Service 963350
  GString *s;
Packit Service 963350
Packit Service 963350
  g_return_val_if_fail (features != NULL, NULL);
Packit Service 963350
Packit Service 963350
  s = g_string_sized_new (FEATURES_ESTIMATED_STRING_LEN (features));
Packit Service 963350
Packit Service 963350
  priv_gst_caps_features_append_to_gstring (features, s);
Packit Service 963350
Packit Service 963350
  return g_string_free (s, FALSE);
Packit Service 963350
}
Packit Service 963350
Packit Service 963350
void
Packit Service 963350
priv_gst_caps_features_append_to_gstring (const GstCapsFeatures * features,
Packit Service 963350
    GString * s)
Packit Service 963350
{
Packit Service 963350
  guint i, n;
Packit Service 963350
Packit Service 963350
  g_return_if_fail (features != NULL);
Packit Service 963350
Packit Service 963350
  if (features->array->len == 0 && features->is_any) {
Packit Service 963350
    g_string_append (s, "ANY");
Packit Service 963350
    return;
Packit Service 963350
  }
Packit Service 963350
Packit Service 963350
  n = features->array->len;
Packit Service 963350
  for (i = 0; i < n; i++) {
Packit Service 963350
    GQuark *quark = &g_array_index (features->array, GQuark, i);
Packit Service 963350
Packit Service 963350
    g_string_append (s, g_quark_to_string (*quark));
Packit Service 963350
    if (i + 1 < n)
Packit Service 963350
      g_string_append (s, ", ");
Packit Service 963350
  }
Packit Service 963350
}
Packit Service 963350
Packit Service 963350
/**
Packit Service 963350
 * gst_caps_features_from_string:
Packit Service 963350
 * @features: a string representation of a #GstCapsFeatures.
Packit Service 963350
 *
Packit Service 963350
 * Creates a #GstCapsFeatures from a string representation.
Packit Service 963350
 *
Packit Service 963350
 * Free-function: gst_caps_features_free
Packit Service 963350
 *
Packit Service 963350
 * Returns: (transfer full) (nullable): a new #GstCapsFeatures or
Packit Service 963350
 *     %NULL when the string could not be parsed. Free with
Packit Service 963350
 *     gst_caps_features_free() after use.
Packit Service 963350
 *
Packit Service 963350
 * Since: 1.2
Packit Service 963350
 */
Packit Service 963350
GstCapsFeatures *
Packit Service 963350
gst_caps_features_from_string (const gchar * features)
Packit Service 963350
{
Packit Service 963350
  GstCapsFeatures *ret;
Packit Service 963350
  gboolean escape = FALSE;
Packit Service 963350
  const gchar *features_orig = features;
Packit Service 963350
  const gchar *feature;
Packit Service 963350
Packit Service 963350
  ret = gst_caps_features_new_empty ();
Packit Service 963350
Packit Service 963350
  if (!features || *features == '\0')
Packit Service 963350
    return ret;
Packit Service 963350
Packit Service 963350
  if (strcmp (features, "ANY") == 0) {
Packit Service 963350
    ret->is_any = TRUE;
Packit Service 963350
    return ret;
Packit Service 963350
  }
Packit Service 963350
Packit Service 963350
  /* Skip trailing spaces */
Packit Service 963350
  while (*features == ' ')
Packit Service 963350
    features++;
Packit Service 963350
Packit Service 963350
  feature = features;
Packit Service 963350
  while (TRUE) {
Packit Service 963350
    gchar c = *features;
Packit Service 963350
Packit Service 963350
    if (c == '\\') {
Packit Service 963350
      escape = TRUE;
Packit Service 963350
      features++;
Packit Service 963350
      continue;
Packit Service 963350
    } else if ((!escape && c == ',') || c == '\0') {
Packit Service 963350
      guint len = features - feature + 1;
Packit Service 963350
      gchar *tmp;
Packit Service 963350
      gchar *p;
Packit Service 963350
Packit Service 963350
      if (len == 1) {
Packit Service 963350
        g_warning ("Failed deserialize caps features '%s'", features_orig);
Packit Service 963350
        gst_caps_features_free (ret);
Packit Service 963350
        return NULL;
Packit Service 963350
      }
Packit Service 963350
Packit Service 963350
      tmp = g_malloc (len);
Packit Service 963350
      memcpy (tmp, feature, len - 1);
Packit Service 963350
      tmp[len - 1] = '\0';
Packit Service 963350
Packit Service 963350
      p = tmp + len - 1;
Packit Service 963350
      while (*p == ' ') {
Packit Service 963350
        *p = '\0';
Packit Service 963350
        p--;
Packit Service 963350
      }
Packit Service 963350
Packit Service 963350
      if (strstr (tmp, " ") != NULL || *tmp == '\0') {
Packit Service 963350
        g_free (tmp);
Packit Service 963350
        g_warning ("Failed deserialize caps features '%s'", features_orig);
Packit Service 963350
        gst_caps_features_free (ret);
Packit Service 963350
        return NULL;
Packit Service 963350
      }
Packit Service 963350
Packit Service 963350
      gst_caps_features_add (ret, tmp);
Packit Service 963350
      g_free (tmp);
Packit Service 963350
Packit Service 963350
      if (c == '\0')
Packit Service 963350
        break;
Packit Service 963350
Packit Service 963350
      /* Skip to the next value */
Packit Service 963350
      features++;
Packit Service 963350
      while (*features == ' ')
Packit Service 963350
        features++;
Packit Service 963350
      feature = features;
Packit Service 963350
    } else {
Packit Service 963350
      escape = FALSE;
Packit Service 963350
      features++;
Packit Service 963350
    }
Packit Service 963350
  }
Packit Service 963350
Packit Service 963350
  return ret;
Packit Service 963350
}
Packit Service 963350
Packit Service 963350
/**
Packit Service 963350
 * gst_caps_features_get_size:
Packit Service 963350
 * @features: a #GstCapsFeatures.
Packit Service 963350
 *
Packit Service 963350
 * Returns the number of features in @features.
Packit Service 963350
 *
Packit Service 963350
 * Returns: The number of features in @features.
Packit Service 963350
 *
Packit Service 963350
 * Since: 1.2
Packit Service 963350
 */
Packit Service 963350
guint
Packit Service 963350
gst_caps_features_get_size (const GstCapsFeatures * features)
Packit Service 963350
{
Packit Service 963350
  g_return_val_if_fail (features != NULL, 0);
Packit Service 963350
Packit Service 963350
  return features->array->len;
Packit Service 963350
}
Packit Service 963350
Packit Service 963350
/**
Packit Service 963350
 * gst_caps_features_get_nth:
Packit Service 963350
 * @features: a #GstCapsFeatures.
Packit Service 963350
 * @i: index of the feature
Packit Service 963350
 *
Packit Service 963350
 * Returns the @i-th feature of @features.
Packit Service 963350
 *
Packit Service 963350
 * Returns: (nullable): The @i-th feature of @features.
Packit Service 963350
 *
Packit Service 963350
 * Since: 1.2
Packit Service 963350
 */
Packit Service 963350
const gchar *
Packit Service 963350
gst_caps_features_get_nth (const GstCapsFeatures * features, guint i)
Packit Service 963350
{
Packit Service 963350
  const gchar *feature;
Packit Service 963350
  GQuark quark;
Packit Service 963350
Packit Service 963350
  g_return_val_if_fail (features != NULL, NULL);
Packit Service 963350
Packit Service 963350
  quark = gst_caps_features_get_nth_id (features, i);
Packit Service 963350
  if (!quark)
Packit Service 963350
    return NULL;
Packit Service 963350
Packit Service 963350
  feature = g_quark_to_string (quark);
Packit Service 963350
  return feature;
Packit Service 963350
}
Packit Service 963350
Packit Service 963350
/**
Packit Service 963350
 * gst_caps_features_get_nth_id:
Packit Service 963350
 * @features: a #GstCapsFeatures.
Packit Service 963350
 * @i: index of the feature
Packit Service 963350
 *
Packit Service 963350
 * Returns the @i-th feature of @features.
Packit Service 963350
 *
Packit Service 963350
 * Returns: The @i-th feature of @features.
Packit Service 963350
 *
Packit Service 963350
 * Since: 1.2
Packit Service 963350
 */
Packit Service 963350
GQuark
Packit Service 963350
gst_caps_features_get_nth_id (const GstCapsFeatures * features, guint i)
Packit Service 963350
{
Packit Service 963350
  GQuark *quark;
Packit Service 963350
Packit Service 963350
  g_return_val_if_fail (features != NULL, 0);
Packit Service 963350
  g_return_val_if_fail (i < features->array->len, 0);
Packit Service 963350
Packit Service 963350
  quark = &g_array_index (features->array, GQuark, i);
Packit Service 963350
Packit Service 963350
  return *quark;
Packit Service 963350
}
Packit Service 963350
Packit Service 963350
/**
Packit Service 963350
 * gst_caps_features_contains:
Packit Service 963350
 * @features: a #GstCapsFeatures.
Packit Service 963350
 * @feature: a feature
Packit Service 963350
 *
Packit Service 963350
 * Check if @features contains @feature.
Packit Service 963350
 *
Packit Service 963350
 * Returns: %TRUE if @features contains @feature.
Packit Service 963350
 *
Packit Service 963350
 * Since: 1.2
Packit Service 963350
 */
Packit Service 963350
gboolean
Packit Service 963350
gst_caps_features_contains (const GstCapsFeatures * features,
Packit Service 963350
    const gchar * feature)
Packit Service 963350
{
Packit Service 963350
  g_return_val_if_fail (features != NULL, FALSE);
Packit Service 963350
  g_return_val_if_fail (feature != NULL, FALSE);
Packit Service 963350
Packit Service 963350
  return gst_caps_features_contains_id (features,
Packit Service 963350
      g_quark_from_string (feature));
Packit Service 963350
}
Packit Service 963350
Packit Service 963350
/**
Packit Service 963350
 * gst_caps_features_contains_id:
Packit Service 963350
 * @features: a #GstCapsFeatures.
Packit Service 963350
 * @feature: a feature
Packit Service 963350
 *
Packit Service 963350
 * Check if @features contains @feature.
Packit Service 963350
 *
Packit Service 963350
 * Returns: %TRUE if @features contains @feature.
Packit Service 963350
 *
Packit Service 963350
 * Since: 1.2
Packit Service 963350
 */
Packit Service 963350
gboolean
Packit Service 963350
gst_caps_features_contains_id (const GstCapsFeatures * features, GQuark feature)
Packit Service 963350
{
Packit Service 963350
  guint i, n;
Packit Service 963350
Packit Service 963350
  g_return_val_if_fail (features != NULL, FALSE);
Packit Service 963350
  g_return_val_if_fail (feature != 0, FALSE);
Packit Service 963350
Packit Service 963350
  if (features->is_any)
Packit Service 963350
    return TRUE;
Packit Service 963350
Packit Service 963350
  n = features->array->len;
Packit Service 963350
  if (n == 0)
Packit Service 963350
    return feature == _gst_caps_feature_memory_system_memory;
Packit Service 963350
Packit Service 963350
  for (i = 0; i < n; i++) {
Packit Service 963350
    if (gst_caps_features_get_nth_id (features, i) == feature)
Packit Service 963350
      return TRUE;
Packit Service 963350
  }
Packit Service 963350
Packit Service 963350
  return FALSE;
Packit Service 963350
}
Packit Service 963350
Packit Service 963350
/**
Packit Service 963350
 * gst_caps_features_is_equal:
Packit Service 963350
 * @features1: a #GstCapsFeatures.
Packit Service 963350
 * @features2: a #GstCapsFeatures.
Packit Service 963350
 *
Packit Service 963350
 * Check if @features1 and @features2 are equal.
Packit Service 963350
 *
Packit Service 963350
 * Returns: %TRUE if @features1 and @features2 are equal.
Packit Service 963350
 *
Packit Service 963350
 * Since: 1.2
Packit Service 963350
 */
Packit Service 963350
gboolean
Packit Service 963350
gst_caps_features_is_equal (const GstCapsFeatures * features1,
Packit Service 963350
    const GstCapsFeatures * features2)
Packit Service 963350
{
Packit Service 963350
  guint i, n;
Packit Service 963350
Packit Service 963350
  g_return_val_if_fail (features1 != NULL, FALSE);
Packit Service 963350
  g_return_val_if_fail (features2 != NULL, FALSE);
Packit Service 963350
Packit Service 963350
  if (features1->is_any || features2->is_any)
Packit Service 963350
    return TRUE;
Packit Service 963350
Packit Service 963350
  /* Check for the sysmem==empty case */
Packit Service 963350
  if (features1->array->len == 0 && features2->array->len == 0)
Packit Service 963350
    return TRUE;
Packit Service 963350
  if (features1->array->len == 0 && features2->array->len == 1
Packit Service 963350
      && gst_caps_features_contains_id (features2,
Packit Service 963350
          _gst_caps_feature_memory_system_memory))
Packit Service 963350
    return TRUE;
Packit Service 963350
  if (features2->array->len == 0 && features1->array->len == 1
Packit Service 963350
      && gst_caps_features_contains_id (features1,
Packit Service 963350
          _gst_caps_feature_memory_system_memory))
Packit Service 963350
    return TRUE;
Packit Service 963350
Packit Service 963350
  if (features1->array->len != features2->array->len)
Packit Service 963350
    return FALSE;
Packit Service 963350
Packit Service 963350
  n = features1->array->len;
Packit Service 963350
  for (i = 0; i < n; i++)
Packit Service 963350
    if (!gst_caps_features_contains_id (features2,
Packit Service 963350
            gst_caps_features_get_nth_id (features1, i)))
Packit Service 963350
      return FALSE;
Packit Service 963350
Packit Service 963350
  return TRUE;
Packit Service 963350
}
Packit Service 963350
Packit Service 963350
/**
Packit Service 963350
 * gst_caps_features_is_any:
Packit Service 963350
 * @features: a #GstCapsFeatures.
Packit Service 963350
 *
Packit Service 963350
 * Check if @features is %GST_CAPS_FEATURES_ANY.
Packit Service 963350
 *
Packit Service 963350
 * Returns: %TRUE if @features is %GST_CAPS_FEATURES_ANY.
Packit Service 963350
 *
Packit Service 963350
 * Since: 1.2
Packit Service 963350
 */
Packit Service 963350
gboolean
Packit Service 963350
gst_caps_features_is_any (const GstCapsFeatures * features)
Packit Service 963350
{
Packit Service 963350
  g_return_val_if_fail (features != NULL, FALSE);
Packit Service 963350
Packit Service 963350
  return features->is_any;
Packit Service 963350
}
Packit Service 963350
Packit Service 963350
/**
Packit Service 963350
 * gst_caps_features_add:
Packit Service 963350
 * @features: a #GstCapsFeatures.
Packit Service 963350
 * @feature: a feature.
Packit Service 963350
 *
Packit Service 963350
 * Adds @feature to @features.
Packit Service 963350
 *
Packit Service 963350
 * Since: 1.2
Packit Service 963350
 */
Packit Service 963350
void
Packit Service 963350
gst_caps_features_add (GstCapsFeatures * features, const gchar * feature)
Packit Service 963350
{
Packit Service 963350
  g_return_if_fail (features != NULL);
Packit Service 963350
  g_return_if_fail (IS_MUTABLE (features));
Packit Service 963350
  g_return_if_fail (feature != NULL);
Packit Service 963350
  g_return_if_fail (!features->is_any);
Packit Service 963350
Packit Service 963350
  gst_caps_features_add_id (features, g_quark_from_string (feature));
Packit Service 963350
}
Packit Service 963350
Packit Service 963350
/**
Packit Service 963350
 * gst_caps_features_add_id:
Packit Service 963350
 * @features: a #GstCapsFeatures.
Packit Service 963350
 * @feature: a feature.
Packit Service 963350
 *
Packit Service 963350
 * Adds @feature to @features.
Packit Service 963350
 *
Packit Service 963350
 * Since: 1.2
Packit Service 963350
 */
Packit Service 963350
void
Packit Service 963350
gst_caps_features_add_id (GstCapsFeatures * features, GQuark feature)
Packit Service 963350
{
Packit Service 963350
  g_return_if_fail (features != NULL);
Packit Service 963350
  g_return_if_fail (IS_MUTABLE (features));
Packit Service 963350
  g_return_if_fail (feature != 0);
Packit Service 963350
  g_return_if_fail (!features->is_any);
Packit Service 963350
Packit Service 963350
  if (!gst_caps_feature_name_is_valid (g_quark_to_string (feature))) {
Packit Service 963350
    g_warning ("Invalid caps feature name: %s", g_quark_to_string (feature));
Packit Service 963350
    return;
Packit Service 963350
  }
Packit Service 963350
Packit Service 963350
  /* If features is empty it will contain sysmem, however
Packit Service 963350
   * we want to add it explicitely if it is tried to be
Packit Service 963350
   * added as first features
Packit Service 963350
   */
Packit Service 963350
  if (features->array->len > 0
Packit Service 963350
      && gst_caps_features_contains_id (features, feature))
Packit Service 963350
    return;
Packit Service 963350
Packit Service 963350
  g_array_append_val (features->array, feature);
Packit Service 963350
}
Packit Service 963350
Packit Service 963350
/**
Packit Service 963350
 * gst_caps_features_remove:
Packit Service 963350
 * @features: a #GstCapsFeatures.
Packit Service 963350
 * @feature: a feature.
Packit Service 963350
 *
Packit Service 963350
 * Removes @feature from @features.
Packit Service 963350
 *
Packit Service 963350
 * Since: 1.2
Packit Service 963350
 */
Packit Service 963350
void
Packit Service 963350
gst_caps_features_remove (GstCapsFeatures * features, const gchar * feature)
Packit Service 963350
{
Packit Service 963350
  g_return_if_fail (features != NULL);
Packit Service 963350
  g_return_if_fail (IS_MUTABLE (features));
Packit Service 963350
  g_return_if_fail (feature != NULL);
Packit Service 963350
Packit Service 963350
  gst_caps_features_remove_id (features, g_quark_from_string (feature));
Packit Service 963350
}
Packit Service 963350
Packit Service 963350
/**
Packit Service 963350
 * gst_caps_features_remove_id:
Packit Service 963350
 * @features: a #GstCapsFeatures.
Packit Service 963350
 * @feature: a feature.
Packit Service 963350
 *
Packit Service 963350
 * Removes @feature from @features.
Packit Service 963350
 *
Packit Service 963350
 * Since: 1.2
Packit Service 963350
 */
Packit Service 963350
void
Packit Service 963350
gst_caps_features_remove_id (GstCapsFeatures * features, GQuark feature)
Packit Service 963350
{
Packit Service 963350
  guint i, n;
Packit Service 963350
Packit Service 963350
  g_return_if_fail (features != NULL);
Packit Service 963350
  g_return_if_fail (IS_MUTABLE (features));
Packit Service 963350
  g_return_if_fail (feature != 0);
Packit Service 963350
Packit Service 963350
  n = features->array->len;
Packit Service 963350
  for (i = 0; i < n; i++) {
Packit Service 963350
    GQuark quark = gst_caps_features_get_nth_id (features, i);
Packit Service 963350
Packit Service 963350
    if (quark == feature) {
Packit Service 963350
      g_array_remove_index_fast (features->array, i);
Packit Service 963350
      return;
Packit Service 963350
    }
Packit Service 963350
  }
Packit Service 963350
}
Packit Service 963350
Packit Service 963350
static void
Packit Service 963350
gst_caps_features_transform_to_string (const GValue * src_value,
Packit Service 963350
    GValue * dest_value)
Packit Service 963350
{
Packit Service 963350
  g_return_if_fail (src_value != NULL);
Packit Service 963350
  g_return_if_fail (dest_value != NULL);
Packit Service 963350
Packit Service 963350
  dest_value->data[0].v_pointer =
Packit Service 963350
      gst_caps_features_to_string (src_value->data[0].v_pointer);
Packit Service 963350
}