Blame gst/gstcontext.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
 * Copyright (C) 2013 Sebastian Dröge <slomo@circular-chaos.org>
Packit Service 963350
 *
Packit Service 963350
 * gstcontext.h: Header for GstContext subsystem
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:gstcontext
Packit Service 963350
 * @title: GstContext
Packit Service 963350
 * @short_description: Lightweight objects to represent element contexts
Packit Service 963350
 * @see_also: #GstMiniObject, #GstElement
Packit Service 963350
 *
Packit Service 963350
 * #GstContext is a container object used to store contexts like a device
Packit Service 963350
 * context, a display server connection and similar concepts that should
Packit Service 963350
 * be shared between multiple elements.
Packit Service 963350
 *
Packit Service 963350
 * Applications can set a context on a complete pipeline by using
Packit Service 963350
 * gst_element_set_context(), which will then be propagated to all
Packit Service 963350
 * child elements. Elements can handle these in #GstElementClass.set_context()
Packit Service 963350
 * and merge them with the context information they already have.
Packit Service 963350
 *
Packit Service 963350
 * When an element needs a context it will do the following actions in this
Packit Service 963350
 * order until one step succeeds:
Packit Service 963350
 * 1. Check if the element already has a context
Packit Service 963350
 * 2. Query downstream with GST_QUERY_CONTEXT for the context
Packit Service 963350
 * 3. Query upstream with GST_QUERY_CONTEXT for the context
Packit Service 963350
 * 4. Post a GST_MESSAGE_NEED_CONTEXT message on the bus with the required
Packit Service 963350
 *    context types and afterwards check if a usable context was set now
Packit Service 963350
 * 5. Create a context by itself and post a GST_MESSAGE_HAVE_CONTEXT message
Packit Service 963350
 *    on the bus.
Packit Service 963350
 *
Packit Service 963350
 * Bins will catch GST_MESSAGE_NEED_CONTEXT messages and will set any previously
Packit Service 963350
 * known context on the element that asks for it if possible. Otherwise the
Packit Service 963350
 * application should provide one if it can.
Packit Service 963350
 *
Packit Service 963350
 * #GstContexts can be persistent.
Packit Service 963350
 * A persistent #GstContext is kept in elements when they reach
Packit Service 963350
 * %GST_STATE_NULL, non-persistent ones will be removed.
Packit Service 963350
 * Also, a non-persistent context won't override a previous persistent
Packit Service 963350
 * context set to an element.
Packit Service 963350
 *
Packit Service 963350
 * Since: 1.2
Packit Service 963350
 */
Packit Service 963350
Packit Service 963350
#include "gst_private.h"
Packit Service 963350
#include <string.h>
Packit Service 963350
#include "gstcontext.h"
Packit Service 963350
#include "gstquark.h"
Packit Service 963350
Packit Service 963350
struct _GstContext
Packit Service 963350
{
Packit Service 963350
  GstMiniObject mini_object;
Packit Service 963350
Packit Service 963350
  gchar *context_type;
Packit Service 963350
  GstStructure *structure;
Packit Service 963350
  gboolean persistent;
Packit Service 963350
};
Packit Service 963350
Packit Service 963350
#define GST_CONTEXT_STRUCTURE(c)  (((GstContext *)(c))->structure)
Packit Service 963350
Packit Service 963350
GType _gst_context_type = 0;
Packit Service 963350
GST_DEFINE_MINI_OBJECT_TYPE (GstContext, gst_context);
Packit Service 963350
Packit Service 963350
void
Packit Service 963350
_priv_gst_context_initialize (void)
Packit Service 963350
{
Packit Service 963350
  GST_CAT_INFO (GST_CAT_GST_INIT, "init contexts");
Packit Service 963350
Packit Service 963350
  /* the GstMiniObject types need to be class_ref'd once before it can be
Packit Service 963350
   * done from multiple threads;
Packit Service 963350
   * see http://bugzilla.gnome.org/show_bug.cgi?id=304551 */
Packit Service 963350
  gst_context_get_type ();
Packit Service 963350
Packit Service 963350
  _gst_context_type = gst_context_get_type ();
Packit Service 963350
}
Packit Service 963350
Packit Service 963350
static void
Packit Service 963350
_gst_context_free (GstContext * context)
Packit Service 963350
{
Packit Service 963350
  GstStructure *structure;
Packit Service 963350
Packit Service 963350
  g_return_if_fail (context != NULL);
Packit Service 963350
Packit Service 963350
  GST_CAT_LOG (GST_CAT_CONTEXT, "finalize context %p: %" GST_PTR_FORMAT,
Packit Service 963350
      context, GST_CONTEXT_STRUCTURE (context));
Packit Service 963350
Packit Service 963350
  structure = GST_CONTEXT_STRUCTURE (context);
Packit Service 963350
  if (structure) {
Packit Service 963350
    gst_structure_set_parent_refcount (structure, NULL);
Packit Service 963350
    gst_structure_free (structure);
Packit Service 963350
  }
Packit Service 963350
  g_free (context->context_type);
Packit Service 963350
Packit Service 963350
  g_slice_free1 (sizeof (GstContext), context);
Packit Service 963350
}
Packit Service 963350
Packit Service 963350
static void gst_context_init (GstContext * context);
Packit Service 963350
Packit Service 963350
static GstContext *
Packit Service 963350
_gst_context_copy (GstContext * context)
Packit Service 963350
{
Packit Service 963350
  GstContext *copy;
Packit Service 963350
  GstStructure *structure;
Packit Service 963350
Packit Service 963350
  GST_CAT_LOG (GST_CAT_CONTEXT, "copy context %p: %" GST_PTR_FORMAT, context,
Packit Service 963350
      GST_CONTEXT_STRUCTURE (context));
Packit Service 963350
Packit Service 963350
  copy = g_slice_new0 (GstContext);
Packit Service 963350
Packit Service 963350
  gst_context_init (copy);
Packit Service 963350
Packit Service 963350
  copy->context_type = g_strdup (context->context_type);
Packit Service 963350
Packit Service 963350
  structure = GST_CONTEXT_STRUCTURE (context);
Packit Service 963350
  GST_CONTEXT_STRUCTURE (copy) = gst_structure_copy (structure);
Packit Service 963350
  gst_structure_set_parent_refcount (GST_CONTEXT_STRUCTURE (copy),
Packit Service 963350
      &copy->mini_object.refcount);
Packit Service 963350
Packit Service 963350
  copy->persistent = context->persistent;
Packit Service 963350
Packit Service 963350
  return GST_CONTEXT_CAST (copy);
Packit Service 963350
}
Packit Service 963350
Packit Service 963350
static void
Packit Service 963350
gst_context_init (GstContext * context)
Packit Service 963350
{
Packit Service 963350
  gst_mini_object_init (GST_MINI_OBJECT_CAST (context), 0, _gst_context_type,
Packit Service 963350
      (GstMiniObjectCopyFunction) _gst_context_copy, NULL,
Packit Service 963350
      (GstMiniObjectFreeFunction) _gst_context_free);
Packit Service 963350
}
Packit Service 963350
Packit Service 963350
/**
Packit Service 963350
 * gst_context_new:
Packit Service 963350
 * @context_type: Context type
Packit Service 963350
 * @persistent: Persistent context
Packit Service 963350
 *
Packit Service 963350
 * Create a new context.
Packit Service 963350
 *
Packit Service 963350
 * Returns: (transfer full): The new context.
Packit Service 963350
 *
Packit Service 963350
 * Since: 1.2
Packit Service 963350
 */
Packit Service 963350
GstContext *
Packit Service 963350
gst_context_new (const gchar * context_type, gboolean persistent)
Packit Service 963350
{
Packit Service 963350
  GstContext *context;
Packit Service 963350
  GstStructure *structure;
Packit Service 963350
Packit Service 963350
  g_return_val_if_fail (context_type != NULL, NULL);
Packit Service 963350
Packit Service 963350
  context = g_slice_new0 (GstContext);
Packit Service 963350
Packit Service 963350
  GST_CAT_LOG (GST_CAT_CONTEXT, "creating new context %p", context);
Packit Service 963350
Packit Service 963350
  structure = gst_structure_new_id_empty (GST_QUARK (CONTEXT));
Packit Service 963350
  gst_structure_set_parent_refcount (structure, &context->mini_object.refcount);
Packit Service 963350
  gst_context_init (context);
Packit Service 963350
Packit Service 963350
  context->context_type = g_strdup (context_type);
Packit Service 963350
  GST_CONTEXT_STRUCTURE (context) = structure;
Packit Service 963350
  context->persistent = persistent;
Packit Service 963350
Packit Service 963350
  return context;
Packit Service 963350
}
Packit Service 963350
Packit Service 963350
/**
Packit Service 963350
 * gst_context_get_context_type:
Packit Service 963350
 * @context: The #GstContext.
Packit Service 963350
 *
Packit Service 963350
 * Get the type of @context.
Packit Service 963350
 *
Packit Service 963350
 * Returns: The type of the context.
Packit Service 963350
 *
Packit Service 963350
 * Since: 1.2
Packit Service 963350
 */
Packit Service 963350
const gchar *
Packit Service 963350
gst_context_get_context_type (const GstContext * context)
Packit Service 963350
{
Packit Service 963350
  g_return_val_if_fail (GST_IS_CONTEXT (context), NULL);
Packit Service 963350
Packit Service 963350
  return context->context_type;
Packit Service 963350
}
Packit Service 963350
Packit Service 963350
/**
Packit Service 963350
 * gst_context_has_context_type:
Packit Service 963350
 * @context: The #GstContext.
Packit Service 963350
 * @context_type: Context type to check.
Packit Service 963350
 *
Packit Service 963350
 * Checks if @context has @context_type.
Packit Service 963350
 *
Packit Service 963350
 * Returns: %TRUE if @context has @context_type.
Packit Service 963350
 *
Packit Service 963350
 * Since: 1.2
Packit Service 963350
 */
Packit Service 963350
gboolean
Packit Service 963350
gst_context_has_context_type (const GstContext * context,
Packit Service 963350
    const gchar * context_type)
Packit Service 963350
{
Packit Service 963350
  g_return_val_if_fail (GST_IS_CONTEXT (context), FALSE);
Packit Service 963350
  g_return_val_if_fail (context_type != NULL, FALSE);
Packit Service 963350
Packit Service 963350
  return strcmp (context->context_type, context_type) == 0;
Packit Service 963350
}
Packit Service 963350
Packit Service 963350
/**
Packit Service 963350
 * gst_context_get_structure:
Packit Service 963350
 * @context: The #GstContext.
Packit Service 963350
 *
Packit Service 963350
 * Access the structure of the context.
Packit Service 963350
 *
Packit Service 963350
 * Returns: (transfer none): The structure of the context. The structure is
Packit Service 963350
 * still owned by the context, which means that you should not modify it,
Packit Service 963350
 * free it and that the pointer becomes invalid when you free the context.
Packit Service 963350
 *
Packit Service 963350
 * Since: 1.2
Packit Service 963350
 */
Packit Service 963350
const GstStructure *
Packit Service 963350
gst_context_get_structure (const GstContext * context)
Packit Service 963350
{
Packit Service 963350
  g_return_val_if_fail (GST_IS_CONTEXT (context), NULL);
Packit Service 963350
Packit Service 963350
  return GST_CONTEXT_STRUCTURE (context);
Packit Service 963350
}
Packit Service 963350
Packit Service 963350
/**
Packit Service 963350
 * gst_context_writable_structure:
Packit Service 963350
 * @context: The #GstContext.
Packit Service 963350
 *
Packit Service 963350
 * Get a writable version of the structure.
Packit Service 963350
 *
Packit Service 963350
 * Returns: The structure of the context. The structure is still
Packit Service 963350
 * owned by the context, which means that you should not free it and
Packit Service 963350
 * that the pointer becomes invalid when you free the context.
Packit Service 963350
 * This function checks if @context is writable.
Packit Service 963350
 *
Packit Service 963350
 * Since: 1.2
Packit Service 963350
 */
Packit Service 963350
GstStructure *
Packit Service 963350
gst_context_writable_structure (GstContext * context)
Packit Service 963350
{
Packit Service 963350
  g_return_val_if_fail (GST_IS_CONTEXT (context), NULL);
Packit Service 963350
  g_return_val_if_fail (gst_context_is_writable (context), NULL);
Packit Service 963350
Packit Service 963350
  return GST_CONTEXT_STRUCTURE (context);
Packit Service 963350
}
Packit Service 963350
Packit Service 963350
/**
Packit Service 963350
 * gst_context_is_persistent:
Packit Service 963350
 * @context: The #GstContext.
Packit Service 963350
 *
Packit Service 963350
 * Check if @context is persistent.
Packit Service 963350
 *
Packit Service 963350
 * Returns: %TRUE if the context is persistent.
Packit Service 963350
 *
Packit Service 963350
 * Since: 1.2
Packit Service 963350
 */
Packit Service 963350
gboolean
Packit Service 963350
gst_context_is_persistent (const GstContext * context)
Packit Service 963350
{
Packit Service 963350
  g_return_val_if_fail (GST_IS_CONTEXT (context), FALSE);
Packit Service 963350
Packit Service 963350
  return context->persistent;
Packit Service 963350
}