summaryrefslogtreecommitdiffstats
path: root/ext/resindvd/rsnparsetter.c
diff options
context:
space:
mode:
Diffstat (limited to 'ext/resindvd/rsnparsetter.c')
-rw-r--r--ext/resindvd/rsnparsetter.c405
1 files changed, 405 insertions, 0 deletions
diff --git a/ext/resindvd/rsnparsetter.c b/ext/resindvd/rsnparsetter.c
new file mode 100644
index 00000000..65de2ac3
--- /dev/null
+++ b/ext/resindvd/rsnparsetter.c
@@ -0,0 +1,405 @@
+/*
+ * Copyright (C) 2008 Jan Schmidt <thaytan@noraisin.net>
+ */
+
+#ifdef HAVE_CONFIG_H
+# include <config.h>
+#endif
+
+#include <gst/gst.h>
+#include <gst/video/video.h>
+#include <string.h>
+
+#include "rsnparsetter.h"
+#include "rsnwrappedbuffer.h"
+
+GST_DEBUG_CATEGORY_STATIC (rsn_parsetter_debug);
+#define GST_CAT_DEFAULT rsn_parsetter_debug
+
+static GstStaticPadTemplate sink_factory = GST_STATIC_PAD_TEMPLATE ("sink",
+ GST_PAD_SINK,
+ GST_PAD_ALWAYS,
+ GST_STATIC_CAPS ("video/x-raw-rgb; video/x-raw-yuv")
+ );
+
+static GstStaticPadTemplate src_factory = GST_STATIC_PAD_TEMPLATE ("src",
+ GST_PAD_SRC,
+ GST_PAD_ALWAYS,
+ GST_STATIC_CAPS ("video/x-raw-rgb; video/x-raw-yuv")
+ );
+
+static void rsn_parsetter_register_extra (GType rsn_parsetter_type);
+
+GST_BOILERPLATE_FULL (RsnParSetter, rsn_parsetter, GstElement,
+ GST_TYPE_ELEMENT, rsn_parsetter_register_extra);
+
+static void rsn_parsetter_finalize (GObject * object);
+static GstFlowReturn rsn_parsetter_chain (GstPad * pad, GstBuffer * buf);
+static gboolean rsn_parsetter_sink_event (GstPad * pad, GstEvent * event);
+static gboolean rsn_parsetter_sink_setcaps (GstPad * pad, GstCaps * caps);
+static GstFlowReturn rsn_parsetter_sink_bufferalloc (GstPad * pad,
+ guint64 offset, guint size, GstCaps * caps, GstBuffer ** buf);
+
+static GstCaps *rsn_parsetter_src_getcaps (GstPad * pad);
+static GstCaps *rsn_parsetter_convert_caps (RsnParSetter * parset,
+ GstCaps * caps, gboolean widescreen);
+static gboolean rsn_parsetter_check_caps (RsnParSetter * parset,
+ GstCaps * caps);
+
+static void
+rsn_parsetter_register_extra (GType rsn_parsetter_type)
+{
+ GST_DEBUG_CATEGORY_INIT (rsn_parsetter_debug, "rsnparsetter", 0,
+ "Resin DVD aspect ratio adjuster");
+}
+
+static void
+rsn_parsetter_base_init (gpointer gclass)
+{
+ static GstElementDetails element_details = {
+ "Resin Aspect Ratio Setter",
+ "Filter/Video",
+ "Overrides caps on video buffers to force a particular display ratio",
+ "Jan Schmidt <thaytan@noraisin.net>"
+ };
+ GstElementClass *element_class = GST_ELEMENT_CLASS (gclass);
+
+ gst_element_class_add_pad_template (element_class,
+ gst_static_pad_template_get (&src_factory));
+ gst_element_class_add_pad_template (element_class,
+ gst_static_pad_template_get (&sink_factory));
+ gst_element_class_set_details (element_class, &element_details);
+}
+
+static void
+rsn_parsetter_class_init (RsnParSetterClass * klass)
+{
+ GObjectClass *gobject_class;
+ GstElementClass *gstelement_class;
+
+ gobject_class = (GObjectClass *) klass;
+ gstelement_class = (GstElementClass *) klass;
+
+ gobject_class->finalize = rsn_parsetter_finalize;
+}
+
+static void
+rsn_parsetter_init (RsnParSetter * parset, RsnParSetterClass * gclass)
+{
+ parset->sinkpad = gst_pad_new_from_static_template (&sink_factory, "sink");
+ gst_pad_set_getcaps_function (parset->sinkpad,
+ GST_DEBUG_FUNCPTR (gst_pad_proxy_getcaps));
+ gst_pad_set_chain_function (parset->sinkpad,
+ GST_DEBUG_FUNCPTR (rsn_parsetter_chain));
+ gst_pad_set_event_function (parset->sinkpad,
+ GST_DEBUG_FUNCPTR (rsn_parsetter_sink_event));
+ gst_pad_set_setcaps_function (parset->sinkpad,
+ GST_DEBUG_FUNCPTR (rsn_parsetter_sink_setcaps));
+ gst_pad_set_bufferalloc_function (parset->sinkpad,
+ GST_DEBUG_FUNCPTR (rsn_parsetter_sink_bufferalloc));
+ gst_element_add_pad (GST_ELEMENT (parset), parset->sinkpad);
+
+ parset->srcpad = gst_pad_new_from_static_template (&src_factory, "src");
+ gst_pad_set_getcaps_function (parset->srcpad,
+ GST_DEBUG_FUNCPTR (rsn_parsetter_src_getcaps));
+ gst_element_add_pad (GST_ELEMENT (parset), parset->srcpad);
+
+ parset->caps_lock = g_mutex_new ();
+}
+
+static void
+rsn_parsetter_finalize (GObject * object)
+{
+ RsnParSetter *parset = RSN_PARSETTER (object);
+
+ gst_caps_replace (&parset->outcaps, NULL);
+ gst_caps_replace (&parset->in_caps_last, NULL);
+ gst_caps_replace (&parset->in_caps_converted, NULL);
+
+ g_mutex_free (parset->caps_lock);
+
+ G_OBJECT_CLASS (parent_class)->finalize (object);
+}
+
+static GstFlowReturn
+rsn_parsetter_chain (GstPad * pad, GstBuffer * buf)
+{
+ RsnParSetter *parset = RSN_PARSETTER (GST_OBJECT_PARENT (pad));
+
+ /* If this is a buffer we wrapped up earlier, unwrap it now */
+ if (RSN_IS_WRAPPEDBUFFER (buf)) {
+ RsnWrappedBuffer *wrap_buf = RSN_WRAPPEDBUFFER (buf);
+
+ if (wrap_buf->owner == GST_ELEMENT (parset)) {
+ buf = rsn_wrappedbuffer_unwrap_and_unref (wrap_buf);
+ GST_DEBUG_OBJECT (parset, "Unwrapping %p yields buffer %p with caps %"
+ GST_PTR_FORMAT, wrap_buf, buf, GST_BUFFER_CAPS (buf));
+ }
+ }
+
+ if (parset->outcaps != GST_BUFFER_CAPS (buf)) {
+ if (parset->override_outcaps == FALSE &&
+ gst_caps_is_equal (parset->outcaps, GST_BUFFER_CAPS (buf))) {
+ /* Just update our output caps var */
+ gst_caps_replace (&parset->outcaps, GST_BUFFER_CAPS (buf));
+ goto out;
+ }
+
+ /* Replace the caps on the output buffer */
+ buf = gst_buffer_make_metadata_writable (buf);
+ gst_buffer_set_caps (buf, parset->outcaps);
+
+ GST_DEBUG_OBJECT (parset,
+ "Replacing caps on buffer %p with caps %" GST_PTR_FORMAT,
+ buf, parset->outcaps);
+ }
+
+out:
+ return gst_pad_push (parset->srcpad, buf);
+}
+
+static gboolean
+rsn_parsetter_sink_event (GstPad * pad, GstEvent * event)
+{
+ RsnParSetter *parset = RSN_PARSETTER (gst_pad_get_parent (pad));
+ const GstStructure *structure = gst_event_get_structure (event);
+
+ if (structure != NULL &&
+ gst_structure_has_name (structure, "application/x-gst-dvd")) {
+ const char *type = gst_structure_get_string (structure, "event");
+ if (type == NULL)
+ goto out;
+
+ if (strcmp (type, "dvd-video-format") == 0) {
+ gboolean is_widescreen;
+
+ gst_structure_get_boolean (structure, "video-widescreen", &is_widescreen);
+
+ GST_DEBUG_OBJECT (parset, "Video is %s",
+ parset->is_widescreen ? "16:9" : "4:3");
+
+ g_mutex_lock (parset->caps_lock);
+ if (parset->is_widescreen != is_widescreen) {
+ /* Force caps check */
+ gst_caps_replace (&parset->in_caps_last, NULL);
+ gst_caps_replace (&parset->in_caps_converted, NULL);
+ }
+ parset->is_widescreen = is_widescreen;
+
+ /* FIXME: Added for testing: */
+ // parset->is_widescreen = FALSE;
+
+ g_mutex_unlock (parset->caps_lock);
+ }
+ }
+
+out:
+ gst_object_unref (GST_OBJECT (parset));
+ return gst_pad_event_default (pad, event);
+}
+
+static GstCaps *
+rsn_parsetter_src_getcaps (GstPad * pad)
+{
+ RsnParSetter *parset = RSN_PARSETTER (gst_pad_get_parent (pad));
+ GstCaps *ret;
+ const GstCaps *templ_caps = gst_pad_get_pad_template_caps (pad);
+
+ ret = gst_pad_peer_get_caps (parset->sinkpad);
+ if (ret == NULL)
+ ret = gst_caps_copy (templ_caps);
+ else {
+ GstCaps *temp;
+ temp = gst_caps_intersect (templ_caps, ret);
+ gst_caps_unref (ret);
+ ret = rsn_parsetter_convert_caps (parset, temp, parset->is_widescreen);
+ gst_caps_unref (temp);
+ }
+
+ gst_object_unref (parset);
+ return ret;
+}
+
+static gboolean
+rsn_parsetter_check_caps (RsnParSetter * parset, GstCaps * caps)
+{
+ GstStructure *s;
+ gint width, height;
+ gint par_n, par_d;
+ guint dar_n, dar_d;
+ gboolean ret = FALSE;
+
+ g_mutex_lock (parset->caps_lock);
+
+ if (caps == parset->in_caps_last ||
+ gst_caps_is_equal (caps, parset->in_caps_last)) {
+ ret = parset->in_caps_was_ok;
+ goto out;
+ }
+
+ /* Calculate the DAR from the incoming caps, and return TRUE if it matches
+ * the required DAR, FALSE if not */
+ s = gst_caps_get_structure (caps, 0);
+ if (s == NULL)
+ goto out;
+
+ if (!gst_structure_get_int (s, "width", &width) ||
+ !gst_structure_get_int (s, "height", &height))
+ goto out;
+
+ if (!gst_structure_get_fraction (s, "pixel-aspect-ratio", &par_n, &par_d))
+ par_n = par_d = 1;
+
+ if (!gst_video_calculate_display_ratio (&dar_n, &dar_d, width, height,
+ par_n, par_d, 1, 1))
+ goto out;
+
+ GST_DEBUG_OBJECT (parset,
+ "Incoming video caps now: w %d h %d PAR %d/%d = DAR %d/%d",
+ width, height, par_n, par_d, dar_n, dar_d);
+
+ if (parset->is_widescreen) {
+ if (dar_n == 16 && dar_d == 9)
+ ret = TRUE;
+ } else {
+ if (dar_n == 4 && dar_d == 3)
+ ret = TRUE;
+ }
+
+ gst_caps_replace (&parset->in_caps_last, caps);
+ gst_caps_replace (&parset->in_caps_converted, NULL);
+ parset->in_caps_was_ok = ret;
+
+out:
+ g_mutex_unlock (parset->caps_lock);
+ return ret;
+}
+
+static GstCaps *
+rsn_parsetter_convert_caps (RsnParSetter * parset, GstCaps * caps,
+ gboolean widescreen)
+{
+ /* Duplicate the given caps, with a PAR that provides the desired DAR */
+ GstCaps *outcaps;
+ GstStructure *s;
+ gint width, height;
+ gint par_n, par_d;
+ guint dar_n, dar_d;
+ GValue par = { 0, };
+
+ g_mutex_lock (parset->caps_lock);
+ if (caps == parset->in_caps_last && parset->in_caps_converted) {
+ outcaps = gst_caps_ref (parset->in_caps_converted);
+ goto out;
+ }
+
+ outcaps = gst_caps_copy (caps);
+
+ /* Calculate the DAR from the incoming caps, and return TRUE if it matches
+ * the required DAR, FALSE if not */
+ s = gst_caps_get_structure (outcaps, 0);
+ if (s == NULL)
+ goto out;
+
+ if (!gst_structure_get_int (s, "width", &width) ||
+ !gst_structure_get_int (s, "height", &height))
+ goto out;
+
+ if (widescreen) {
+ dar_n = 16;
+ dar_d = 9;
+ } else {
+ dar_n = 4;
+ dar_d = 3;
+ }
+
+ par_n = dar_n * height;
+ par_d = dar_d * width;
+
+ g_value_init (&par, GST_TYPE_FRACTION);
+ gst_value_set_fraction (&par, par_n, par_d);
+ gst_structure_set_value (s, "pixel-aspect-ratio", &par);
+ g_value_unset (&par);
+
+ gst_caps_replace (&parset->in_caps_converted, outcaps);
+out:
+ g_mutex_unlock (parset->caps_lock);
+ return outcaps;
+}
+
+static gboolean
+rsn_parsetter_sink_setcaps (GstPad * pad, GstCaps * caps)
+{
+ /* Check the new incoming caps against our current DAR, and mark
+ * whether the buffers will need adjusting */
+ RsnParSetter *parset = RSN_PARSETTER (gst_pad_get_parent (pad));
+
+ if (rsn_parsetter_check_caps (parset, caps)) {
+ parset->override_outcaps = FALSE;
+ gst_caps_replace (&parset->outcaps, caps);
+ } else {
+ GstCaps *override_caps = rsn_parsetter_convert_caps (parset, caps,
+ parset->is_widescreen);
+ if (parset->outcaps)
+ gst_caps_unref (parset->outcaps);
+ parset->outcaps = override_caps;
+
+ parset->override_outcaps = TRUE;
+ }
+
+ GST_DEBUG_OBJECT (parset, "caps changed: need_override now = %d",
+ parset->override_outcaps);
+
+ gst_object_unref (parset);
+ return TRUE;
+}
+
+static GstFlowReturn
+rsn_parsetter_sink_bufferalloc (GstPad * pad, guint64 offset, guint size,
+ GstCaps * caps, GstBuffer ** buf)
+{
+ RsnParSetter *parset = RSN_PARSETTER (gst_pad_get_parent (pad));
+ GstFlowReturn ret;
+
+ GST_LOG_OBJECT (parset, "Entering bufferalloc");
+
+ if (rsn_parsetter_check_caps (parset, caps)) {
+ ret = gst_pad_alloc_buffer (parset->srcpad, offset, size, caps, buf);
+ GST_LOG_OBJECT (parset, "Not wrapping buf %p", *buf);
+ } else {
+ /* Allocate and wrap a downstream buffer */
+ GstBuffer *orig_buf;
+ GstBuffer *outbuf;
+ GstCaps *override_caps = rsn_parsetter_convert_caps (parset, caps,
+ parset->is_widescreen);
+
+ ret = gst_pad_alloc_buffer (parset->srcpad, offset, size,
+ override_caps, &orig_buf);
+ gst_caps_unref (override_caps);
+
+ if (ret != GST_FLOW_OK)
+ return ret;
+
+ outbuf = (GstBuffer *) rsn_wrapped_buffer_new (orig_buf);
+ if (!outbuf) {
+ /* FIXME: Throw error */
+ return GST_FLOW_ERROR;
+ }
+
+ rsn_wrapped_buffer_set_owner (RSN_WRAPPEDBUFFER (outbuf),
+ GST_ELEMENT (parset));
+
+ gst_buffer_set_caps (outbuf, caps);
+
+ GST_LOG_OBJECT (parset,
+ "Wrapped ds buf %p with caps %" GST_PTR_FORMAT
+ " into new buf %p with caps %" GST_PTR_FORMAT,
+ orig_buf, GST_BUFFER_CAPS (orig_buf), outbuf, GST_BUFFER_CAPS (outbuf));
+
+ *buf = outbuf;
+ }
+
+ gst_object_unref (GST_OBJECT (parset));
+
+ return ret;
+}