diff options
author | Thomas Vander Stichele <thomas@apestaart.org> | 2001-12-23 16:42:33 +0000 |
---|---|---|
committer | Thomas Vander Stichele <thomas@apestaart.org> | 2001-12-23 16:42:33 +0000 |
commit | b0f5f9a9410efd9ce64dfee67dbc950cbbb2d614 (patch) | |
tree | 67bc8451ab25d18d75c410cf8cf97e3cdce2d37c /ext/dv | |
parent | 1c3d0eb6e8f65df18350536dcd3a8e4e4eee3cdc (diff) |
adding dv, raw1934, gnomevfs, rtp
Original commit message from CVS:
adding dv, raw1934, gnomevfs, rtp
Diffstat (limited to 'ext/dv')
-rw-r--r-- | ext/dv/Makefile.am | 11 | ||||
-rw-r--r-- | ext/dv/NOTES | 13 | ||||
-rw-r--r-- | ext/dv/gstdvdec.c | 422 | ||||
-rw-r--r-- | ext/dv/gstdvdec.h | 101 |
4 files changed, 547 insertions, 0 deletions
diff --git a/ext/dv/Makefile.am b/ext/dv/Makefile.am new file mode 100644 index 00000000..e044aa45 --- /dev/null +++ b/ext/dv/Makefile.am @@ -0,0 +1,11 @@ +plugindir = $(libdir)/gst + +plugin_LTLIBRARIES = libgstdvdec.la + +libdvdec_la_SOURCES = gstdvdec.c +libdvdec_la_LIBADD = $(DV_LIBS) +libdvdec_la_CFLAGS = $(GST_CFLAGS) + +noinst_HEADERS = gstdvdec.h + +EXTRA_DIST = NOTES diff --git a/ext/dv/NOTES b/ext/dv/NOTES new file mode 100644 index 00000000..84211593 --- /dev/null +++ b/ext/dv/NOTES @@ -0,0 +1,13 @@ +Packets come from 1394 480 bytes at a time. This is not a video segment +length. This causes problems, since a packet boundary crossing a video +segment can split a video segment if we lose an iso packet. We can +recover from this, sorta, with significant changes to the parser. We have +to deal with the idea that a) some macroblocks just don't exist (we have +zero's for them) and b) when any of the 5 macroblocks doesn't exist, we +can't do pass 3. + +Since things are bitstream-based, we can deal with this, but we have to +add a layer of code that tries to save time (maybe) by not decoding things +that don't exist. Not sure how this is gonna work with the parse code +being based on video segments, and not easily splittable into +macroblock-level parsing (or is it?). diff --git a/ext/dv/gstdvdec.c b/ext/dv/gstdvdec.c new file mode 100644 index 00000000..dfd0dcd5 --- /dev/null +++ b/ext/dv/gstdvdec.c @@ -0,0 +1,422 @@ +//#define RGB + +/* Gnome-Streamer + * Copyright (C) <1999> Erik Walthinsen <omega@cse.ogi.edu> + * + * This library is free software; you can redistribute it and/or + * modify it under the terms of the GNU Library General Public + * License as published by the Free Software Foundation; either + * version 2 of the License, or (at your option) any later version. + * + * This library is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU + * Library General Public License for more details. + * + * You should have received a copy of the GNU Library General Public + * License along with this library; if not, write to the + * Free Software Foundation, Inc., 59 Temple Place - Suite 330, + * Boston, MA 02111-1307, USA. + */ + +#include <string.h> + +/* First, include the header file for the plugin, to bring in the + * object definition and other useful things. + */ +#include "gstdvdec.h" + +#define NTSC + +#ifdef NTSC +# define HEIGHT 480 +# define BUFFER 120000 +#else +# define HEIGHT 576 +# define BUFFER 144000 +#endif + +/* The ElementDetails structure gives a human-readable description + * of the plugin, as well as author and version data. + */ +static GstElementDetails dvdec_details = { + "DV (smpte314) decoder plugin", + "Decoder/DV", + "Uses libdv to decode DV video (libdv.sourceforge.net)", + VERSION, + "asdfasdf", + "(C) 2001", +}; + +/* These are the signals that this element can fire. They are zero- + * based because the numbers themselves are private to the object. + * LAST_SIGNAL is used for initialization of the signal array. + */ +enum { + ASDF, + /* FILL ME */ + LAST_SIGNAL +}; + +/* Arguments are identified the same way, but cannot be zero, so you + * must leave the ARG_0 entry in as a placeholder. + */ +enum { + ARG_0, + /* FILL ME */ +}; + +/* The PadFactory structures describe what pads the element has or + * can have. They can be quite complex, but for this dvdec plugin + * they are rather simple. + */ +GST_PADTEMPLATE_FACTORY ( sink_temp, + "sink", + GST_PAD_SINK, + GST_PAD_ALWAYS, + GST_CAPS_NEW ( + "dv_dec_sink", + "video/dv", + "format", GST_PROPS_STRING ("NTSC") + ) +) + + +#ifdef RGB +GST_PADTEMPLATE_FACTORY ( video_src_temp, + "video", + GST_PAD_SRC, + GST_PAD_ALWAYS, + GST_CAPS_NEW ( + "dv_dec_src", + "video/raw", + "format", GST_PROPS_FOURCC(GST_MAKE_FOURCC('R','G','B',' ')), + "bpp", GST_PROPS_INT(24), + "depth", GST_PROPS_INT(24), + "red_mask", GST_PROPS_INT(0x0000ff), + "green_mask", GST_PROPS_INT(0x00ff00), + "blue_mask", GST_PROPS_INT(0xff0000), + "width", GST_PROPS_INT (720), + "height", GST_PROPS_INT (HEIGHT) + ) +) +#else +GST_PADTEMPLATE_FACTORY ( video_src_temp, + "video", + GST_PAD_SRC, + GST_PAD_ALWAYS, + GST_CAPS_NEW ( + "dv_dec_src", + "video/raw", + "format", GST_PROPS_FOURCC (GST_MAKE_FOURCC ('Y','U','Y','2')), + "width", GST_PROPS_INT (720), + "height", GST_PROPS_INT (HEIGHT) + ) +) +#endif + +GST_PADTEMPLATE_FACTORY ( audio_src_temp, + "audio", + GST_PAD_SRC, + GST_PAD_ALWAYS, + GST_CAPS_NEW ( + "arts_sample", + "audio/raw", + "format", GST_PROPS_STRING ("int"), + "law", GST_PROPS_INT (0), + "depth", GST_PROPS_INT (16), + "width", GST_PROPS_INT (16), + "signed", GST_PROPS_BOOLEAN (TRUE), + "channels", GST_PROPS_INT (2), + "endianness", GST_PROPS_INT (G_LITTLE_ENDIAN) + ) +) + + +/* A number of functon prototypes are given so we can refer to them later. */ +static void gst_dvdec_class_init (GstDVDecClass *klass); +static void gst_dvdec_init (GstDVDec *dvdec); + +static void gst_dvdec_loop (GstElement *element); + +static void gst_dvdec_set_property (GObject *object, guint prop_id, const GValue *value, GParamSpec *pspec); +static void gst_dvdec_get_property (GObject *object, guint prop_id, GValue *value, GParamSpec *pspec); + +/* The parent class pointer needs to be kept around for some object + * operations. + */ +static GstElementClass *parent_class = NULL; + +/* This array holds the ids of the signals registered for this object. + * The array indexes are based on the enum up above. + */ +static guint gst_dvdec_signals[LAST_SIGNAL] = { 0 }; + +/* This function is used to register and subsequently return the type + * identifier for this object class. On first invocation, it will + * register the type, providing the name of the class, struct sizes, + * and pointers to the various functions that define the class. + */ +GType +gst_dvdec_get_type(void) +{ + static GType dvdec_type = 0; + + if (!dvdec_type) { + static const GTypeInfo dvdec_info = { + sizeof(GstDVDecClass), NULL, NULL, + (GClassInitFunc)gst_dvdec_class_init, + NULL, + NULL, + sizeof(GstDVDec), + 0, + (GInstanceInitFunc)gst_dvdec_init, + }; + dvdec_type = g_type_register_static(GST_TYPE_ELEMENT, "GstDVDec", &dvdec_info, 0); + } + return dvdec_type; +} + +/* In order to create an instance of an object, the class must be + * initialized by this function. GObject will take care of running + * it, based on the pointer to the function provided above. + */ +static void +gst_dvdec_class_init (GstDVDecClass *klass) +{ + /* Class pointers are needed to supply pointers to the private + * implementations of parent class methods. + */ + GObjectClass *gobject_class; + GstElementClass *gstelement_class; + + /* Since the dvdec class contains the parent classes, you can simply + * cast the pointer to get access to the parent classes. + */ + gobject_class = (GObjectClass*)klass; + gstelement_class = (GstElementClass*)klass; + + /* The parent class is needed for class method overrides. */ + parent_class = g_type_class_ref(GST_TYPE_ELEMENT); + + /* Here we add an argument to the object. This argument is an integer, + * and can be both read and written. + */ +// g_object_class_install_property(G_OBJECT_CLASS(klass), ARG_ACTIVE, +// g_param_spec_int("active","active","active", +// G_MININT,G_MAXINT,0,G_PARAM_READWRITE)); // CHECKME + + /* Here we add a signal to the object. This is avery useless signal + * called asdf. The signal will also pass a pointer to the listeners + * which happens to be the dvdec element itself */ + gst_dvdec_signals[ASDF] = + g_signal_new("asdf", G_TYPE_FROM_CLASS(klass), G_SIGNAL_RUN_LAST, + G_STRUCT_OFFSET (GstDVDecClass, asdf), NULL, NULL, + g_cclosure_marshal_VOID__POINTER, G_TYPE_NONE, 1, + GST_TYPE_DVDEC); + + /* The last thing is to provide the functions that implement get and set + * of arguments. + */ + gobject_class->set_property = gst_dvdec_set_property; + gobject_class->get_property = gst_dvdec_get_property; + + + // table initialization, only do once + dv_init(); +} + +/* This function is responsible for initializing a specific instance of + * the plugin. + */ +static void +gst_dvdec_init(GstDVDec *dvdec) +{ + dvdec->sinkpad = gst_pad_new_from_template (GST_PADTEMPLATE_GET(sink_temp), "sink"); + gst_element_add_pad(GST_ELEMENT(dvdec),dvdec->sinkpad); + + dvdec->videosrcpad = gst_pad_new_from_template (GST_PADTEMPLATE_GET(video_src_temp), "video"); + //gst_pad_set_caps (dvdec->videosrcpad, gst_pad_get_padtemplate_caps (dvdec->videosrcpad)); + gst_element_add_pad(GST_ELEMENT(dvdec),dvdec->videosrcpad); + +// dvdec->audiosrcpad = gst_pad_new_from_template (GST_PADTEMPLATE_GET(audio_src_temp), "audio"); +// gst_pad_set_caps (dvdec->audiosrcpad, gst_pad_get_padtemplate_caps (dvdec->audiosrcpad)); +// gst_element_add_pad(GST_ELEMENT(dvdec),dvdec->audiosrcpad); + + gst_element_set_loop_function (GST_ELEMENT(dvdec), gst_dvdec_loop); + + dvdec->decoder = dv_decoder_new(); + dvdec->decoder->quality = DV_QUALITY_BEST; + dvdec->inframe = g_malloc(BUFFER); + dvdec->pool = NULL; + dvdec->carryover = NULL; + dvdec->remaining = 0; +} + + +static void +gst_dvdec_loop (GstElement *element) +{ + GstDVDec *dvdec; + int needed; + GstBuffer *buf; + guint8 *outframe; + guint8 *outframe_ptrs[3]; + gint outframe_pitches[3]; + + dvdec = GST_DVDEC (element); + + do { + // grab an input frame + needed = BUFFER; + if (dvdec->remaining > 0) { + memcpy(&dvdec->inframe[BUFFER-needed], + GST_BUFFER_DATA(dvdec->carryover)+(GST_BUFFER_SIZE(dvdec->carryover)-dvdec->remaining), + dvdec->remaining); + dvdec->remaining = 0; + gst_buffer_unref(dvdec->carryover); + } + while (needed) { + buf = gst_pad_pull(dvdec->sinkpad); + if (needed < GST_BUFFER_SIZE(buf)) { + memcpy(&dvdec->inframe[BUFFER-needed],GST_BUFFER_DATA(buf),needed); +/***** NOTE: this is done because 1394src doesn't allow buffers to outlive the handler *****/ + dvdec->carryover = gst_buffer_copy(buf); + dvdec->remaining = GST_BUFFER_SIZE(buf) - needed; + needed = 0; + } else { + memcpy(&dvdec->inframe[BUFFER-needed],GST_BUFFER_DATA(buf),GST_BUFFER_SIZE(buf)); + needed -= GST_BUFFER_SIZE(buf); + } + gst_buffer_unref(buf); + } + + if (!GST_PAD_CAPS (dvdec->videosrcpad)) { + gst_pad_set_caps (dvdec->videosrcpad, gst_pad_get_padtemplate_caps (dvdec->videosrcpad)); + } + + if (!dvdec->pool) { + dvdec->pool = gst_pad_get_bufferpool (dvdec->videosrcpad); + } + + buf = NULL; + if (dvdec->pool) { + buf = gst_buffer_new_from_pool (dvdec->pool, 0, 0); + } + + if (!buf) { + // allocate an output frame + buf = gst_buffer_new(); +#ifdef RGB + GST_BUFFER_SIZE(buf) = (720*HEIGHT)*3; +#else + GST_BUFFER_SIZE(buf) = (720*HEIGHT)*2; +#endif + GST_BUFFER_DATA(buf) = g_malloc(GST_BUFFER_SIZE(buf)); + outframe = GST_BUFFER_DATA(buf); + } else { + outframe = GST_BUFFER_DATA (buf); + } + + outframe_ptrs[0] = outframe; + outframe_ptrs[1] = outframe_ptrs[0] + 720*HEIGHT; + outframe_ptrs[2] = outframe_ptrs[1] + 360*HEIGHT; +#ifdef RGB + outframe_pitches[0] = 720*3; +#else + outframe_pitches[0] = 720*2; // huh? +#endif + outframe_pitches[1] = HEIGHT/2; + outframe_pitches[2] = HEIGHT/2; + + // now we start decoding the frame + dv_parse_header(dvdec->decoder,dvdec->inframe); + +#ifdef RGB + dv_decode_full_frame(dvdec->decoder,dvdec->inframe,e_dv_color_rgb,outframe_ptrs,outframe_pitches); +#else + dv_decode_full_frame(dvdec->decoder,dvdec->inframe,e_dv_color_yuv,outframe_ptrs,outframe_pitches); +#endif + + gst_pad_push(dvdec->videosrcpad,buf); + + } while (!GST_ELEMENT_IS_COTHREAD_STOPPING (element)); +} + + + +/* Arguments are part of the Gtk+ object system, and these functions + * enable the element to respond to various arguments. + */ +static void +gst_dvdec_set_property (GObject *object, guint prop_id, const GValue *value, GParamSpec *pspec) +{ + GstDVDec *dvdec; + + /* It's not null if we got it, but it might not be ours */ + g_return_if_fail(GST_IS_DVDEC(object)); + + /* Get a pointer of the right type. */ + dvdec = GST_DVDEC(object); + + /* Check the argument id to see which argument we're setting. */ + switch (prop_id) { + default: + break; + } +} + +/* The set function is simply the inverse of the get fuction. */ +static void +gst_dvdec_get_property (GObject *object, guint prop_id, GValue *value, GParamSpec *pspec) +{ + GstDVDec *dvdec; + + /* It's not null if we got it, but it might not be ours */ + g_return_if_fail(GST_IS_DVDEC(object)); + dvdec = GST_DVDEC(object); + + switch (prop_id) { + default: + G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec); + break; + } +} + +/* This is the entry into the plugin itself. When the plugin loads, + * this function is called to register everything that the plugin provides. + */ +static gboolean +plugin_init (GModule *module, GstPlugin *plugin) +{ + GstElementFactory *factory; + + /* We need to create an ElementFactory for each element we provide. + * This consists of the name of the element, the GType identifier, + * and a pointer to the details structure at the top of the file. + */ + factory = gst_elementfactory_new("dvdec", GST_TYPE_DVDEC, &dvdec_details); + g_return_val_if_fail(factory != NULL, FALSE); + + /* The pad templates can be easily generated from the factories above, + * and then added to the list of padtemplates for the elementfactory. + * Note that the generated padtemplates are stored in static global + * variables, for the gst_dvdec_init function to use later on. + */ + gst_elementfactory_add_padtemplate (factory, GST_PADTEMPLATE_GET(sink_temp)); + gst_elementfactory_add_padtemplate (factory, GST_PADTEMPLATE_GET(video_src_temp)); + gst_elementfactory_add_padtemplate (factory, GST_PADTEMPLATE_GET(audio_src_temp)); + + /* The very last thing is to register the elementfactory with the plugin. */ + gst_plugin_add_feature (plugin, GST_PLUGIN_FEATURE (factory)); + + return TRUE; +} + +GstPluginDesc plugin_desc = { + GST_VERSION_MAJOR, + GST_VERSION_MINOR, + "dvdec", + plugin_init +}; + diff --git a/ext/dv/gstdvdec.h b/ext/dv/gstdvdec.h new file mode 100644 index 00000000..83a22b1e --- /dev/null +++ b/ext/dv/gstdvdec.h @@ -0,0 +1,101 @@ +/* Gnome-Streamer + * Copyright (C) <1999> Erik Walthinsen <omega@cse.ogi.edu> + * + * This library is free software; you can redistribute it and/or + * modify it under the terms of the GNU Library General Public + * License as published by the Free Software Foundation; either + * version 2 of the License, or (at your option) any later version. + * + * This library is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU + * Library General Public License for more details. + * + * You should have received a copy of the GNU Library General Public + * License along with this library; if not, write to the + * Free Software Foundation, Inc., 59 Temple Place - Suite 330, + * Boston, MA 02111-1307, USA. + */ + + +#ifndef __GST_DVDEC_H__ +#define __GST_DVDEC_H__ + +#include <gst/gst.h> + +#ifdef __cplusplus +extern "C" { +#endif /* __cplusplus */ + + +#include <libdv/dv.h> +#include <stdio.h> + + +/* This is the definition of the element's object structure. */ +typedef struct _GstDVDec GstDVDec; + +/* The structure itself is derived from GstElement, as can be seen by the + * fact that there's a complete instance of the GstElement structure at + * the beginning of the object. This allows the element to be cast to + * an Element or even an Object. + */ +struct _GstDVDec { + GstElement element; + + /* We need to keep track of our pads, so we do so here. */ + GstPad *sinkpad,*videosrcpad,*audiosrcpad; + + guint8 *inframe; // here because we allocate it once, can't be on stack + dv_decoder_t *decoder; + GstBuffer *carryover; + gint remaining; + GstBufferPool *pool; +}; + +/* The other half of the object is its class. The class also derives from + * the same parent, though it must be the class structure this time. + * Function pointers for polymophic methods and signals are placed in this + * structure. */ +typedef struct _GstDVDecClass GstDVDecClass; + +struct _GstDVDecClass { + GstElementClass parent_class; + + /* signals */ + void (*asdf) (GstElement *element, GstDVDec *dvdec); +}; + +/* Five standard preprocessing macros are used in the Gtk+ object system. + * The first uses the object's _get_type function to return the GType + * of the object. + */ +#define GST_TYPE_DVDEC \ + (gst_dvdec_get_type()) +/* The second is a checking cast to the correct type. If the object passed + * is not the right type, a warning will be generated on stderr. + */ +#define GST_DVDEC(obj) \ + (G_TYPE_CHECK_INSTANCE_CAST((obj),GST_TYPE_DVDEC,GstDVDec)) +/* The third is a checking cast of the class instead of the object. */ +#define GST_DVDEC_CLASS(klass) \ + (G_TYPE_CHECK_CLASS_CAST((klass),GST_TYPE_DVDEC,GstDVDec)) +/* The last two simply check to see if the passed pointer is an object or + * class of the correct type. */ +#define GST_IS_DVDEC(obj) \ + (G_TYPE_CHECK_INSTANCE_TYPE((obj),GST_TYPE_DVDEC)) +#define GST_IS_DVDEC_CLASS(obj) \ + (G_TYPE_CHECK_CLASS_TYPE((klass),GST_TYPE_DVDEC)) + +/* This is the only prototype needed, because it is used in the above + * GST_TYPE_DVDEC macro. + */ +GType gst_dvdec_get_type(void); + + +#ifdef __cplusplus +} +#endif /* __cplusplus */ + + +#endif /* __GST_DVDEC_H__ */ |