Blame gst/gsttocsetter.c

Packit Service 963350
/* GStreamer
Packit Service 963350
 * Copyright (C) 2010, 2012 Alexander Saprykin <xelfium@gmail.com>
Packit Service 963350
 *
Packit Service 963350
 * gsttocsetter.c: interface for TOC setting on elements
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:gsttocsetter
Packit Service 963350
 * @title: GstTocSetter
Packit Service 963350
 * @short_description: Element interface that allows setting and retrieval
Packit Service 963350
 *                     of the TOC
Packit Service 963350
 *
Packit Service 963350
 * Element interface that allows setting of the TOC.
Packit Service 963350
 *
Packit Service 963350
 * Elements that support some kind of chapters or editions (or tracks like in
Packit Service 963350
 * the FLAC cue sheet) will implement this interface.
Packit Service 963350
 *
Packit Service 963350
 * If you just want to retrieve the TOC in your application then all you
Packit Service 963350
 * need to do is watch for TOC messages on your pipeline's bus (or you can
Packit Service 963350
 * perform TOC query). This interface is only for setting TOC data, not for
Packit Service 963350
 * extracting it. To set TOC from the application, find proper tocsetter element
Packit Service 963350
 * and set TOC using gst_toc_setter_set_toc().
Packit Service 963350
 *
Packit Service 963350
 * Elements implementing the #GstTocSetter interface can extend existing TOC
Packit Service 963350
 * by getting extend UID for that (you can use gst_toc_find_entry() to retrieve it)
Packit Service 963350
 * with any TOC entries received from downstream.
Packit Service 963350
 */
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 "gst_private.h"
Packit Service 963350
#include "gsttocsetter.h"
Packit Service 963350
#include <gobject/gvaluecollector.h>
Packit Service 963350
#include <string.h>
Packit Service 963350
Packit Service 963350
static GQuark gst_toc_key;
Packit Service 963350
Packit Service 963350
G_DEFINE_INTERFACE_WITH_CODE (GstTocSetter, gst_toc_setter, GST_TYPE_ELEMENT,
Packit Service 963350
    gst_toc_key = g_quark_from_static_string ("gst-toc-setter-data"););
Packit Service 963350
Packit Service 963350
static void
Packit Service 963350
gst_toc_setter_default_init (GstTocSetterInterface * klass)
Packit Service 963350
{
Packit Service 963350
  /* nothing to do here, it's a dummy interface */
Packit Service 963350
}
Packit Service 963350
Packit Service 963350
typedef struct
Packit Service 963350
{
Packit Service 963350
  GstToc *toc;
Packit Service 963350
  GMutex lock;
Packit Service 963350
} GstTocData;
Packit Service 963350
Packit Service 963350
static void
Packit Service 963350
gst_toc_data_free (gpointer p)
Packit Service 963350
{
Packit Service 963350
  GstTocData *data = (GstTocData *) p;
Packit Service 963350
Packit Service 963350
  if (data->toc)
Packit Service 963350
    gst_toc_unref (data->toc);
Packit Service 963350
Packit Service 963350
  g_mutex_clear (&data->lock);
Packit Service 963350
Packit Service 963350
  g_slice_free (GstTocData, data);
Packit Service 963350
}
Packit Service 963350
Packit Service 963350
static GstTocData *
Packit Service 963350
gst_toc_setter_get_data (GstTocSetter * setter)
Packit Service 963350
{
Packit Service 963350
  GstTocData *data;
Packit Service 963350
Packit Service 963350
  data = g_object_get_qdata (G_OBJECT (setter), gst_toc_key);
Packit Service 963350
  if (!data) {
Packit Service 963350
    static GMutex create_mutex;
Packit Service 963350
Packit Service 963350
    /* make sure no other thread is creating a GstTocData at the same time */
Packit Service 963350
    g_mutex_lock (&create_mutex);
Packit Service 963350
    data = g_object_get_qdata (G_OBJECT (setter), gst_toc_key);
Packit Service 963350
    if (!data) {
Packit Service 963350
      data = g_slice_new (GstTocData);
Packit Service 963350
      g_mutex_init (&data->lock);
Packit Service 963350
      data->toc = NULL;
Packit Service 963350
      g_object_set_qdata_full (G_OBJECT (setter), gst_toc_key, data,
Packit Service 963350
          gst_toc_data_free);
Packit Service 963350
    }
Packit Service 963350
    g_mutex_unlock (&create_mutex);
Packit Service 963350
  }
Packit Service 963350
Packit Service 963350
  return data;
Packit Service 963350
}
Packit Service 963350
Packit Service 963350
/**
Packit Service 963350
 * gst_toc_setter_reset:
Packit Service 963350
 * @setter: a #GstTocSetter.
Packit Service 963350
 *
Packit Service 963350
 * Reset the internal TOC. Elements should call this from within the
Packit Service 963350
 * state-change handler.
Packit Service 963350
 */
Packit Service 963350
void
Packit Service 963350
gst_toc_setter_reset (GstTocSetter * setter)
Packit Service 963350
{
Packit Service 963350
  g_return_if_fail (GST_IS_TOC_SETTER (setter));
Packit Service 963350
Packit Service 963350
  gst_toc_setter_set_toc (setter, NULL);
Packit Service 963350
}
Packit Service 963350
Packit Service 963350
/**
Packit Service 963350
 * gst_toc_setter_get_toc:
Packit Service 963350
 * @setter: a #GstTocSetter.
Packit Service 963350
 *
Packit Service 963350
 * Return current TOC the setter uses. The TOC should not be
Packit Service 963350
 * modified without making it writable first.
Packit Service 963350
 *
Packit Service 963350
 * Returns: (transfer full) (nullable): TOC set, or %NULL. Unref with
Packit Service 963350
 *     gst_toc_unref() when no longer needed
Packit Service 963350
 */
Packit Service 963350
GstToc *
Packit Service 963350
gst_toc_setter_get_toc (GstTocSetter * setter)
Packit Service 963350
{
Packit Service 963350
  GstTocData *data;
Packit Service 963350
  GstToc *ret = NULL;
Packit Service 963350
Packit Service 963350
  g_return_val_if_fail (GST_IS_TOC_SETTER (setter), NULL);
Packit Service 963350
Packit Service 963350
  data = gst_toc_setter_get_data (setter);
Packit Service 963350
  g_mutex_lock (&data->lock);
Packit Service 963350
Packit Service 963350
  if (data->toc != NULL)
Packit Service 963350
    ret = gst_toc_ref (data->toc);
Packit Service 963350
Packit Service 963350
  g_mutex_unlock (&data->lock);
Packit Service 963350
Packit Service 963350
  return ret;
Packit Service 963350
}
Packit Service 963350
Packit Service 963350
/**
Packit Service 963350
 * gst_toc_setter_set_toc:
Packit Service 963350
 * @setter: a #GstTocSetter.
Packit Service 963350
 * @toc: (allow-none): a #GstToc to set.
Packit Service 963350
 *
Packit Service 963350
 * Set the given TOC on the setter. Previously set TOC will be
Packit Service 963350
 * unreffed before setting a new one.
Packit Service 963350
 */
Packit Service 963350
void
Packit Service 963350
gst_toc_setter_set_toc (GstTocSetter * setter, GstToc * toc)
Packit Service 963350
{
Packit Service 963350
  GstTocData *data;
Packit Service 963350
Packit Service 963350
  g_return_if_fail (GST_IS_TOC_SETTER (setter));
Packit Service 963350
Packit Service 963350
  data = gst_toc_setter_get_data (setter);
Packit Service 963350
Packit Service 963350
  g_mutex_lock (&data->lock);
Packit Service 963350
Packit Service 963350
  if (data->toc != toc) {
Packit Service 963350
    if (data->toc)
Packit Service 963350
      gst_toc_unref (data->toc);
Packit Service 963350
Packit Service 963350
    data->toc = (toc) ? gst_toc_ref (toc) : NULL;
Packit Service 963350
  }
Packit Service 963350
Packit Service 963350
  g_mutex_unlock (&data->lock);
Packit Service 963350
}