summaryrefslogtreecommitdiffstats
path: root/ext/swfdec
diff options
context:
space:
mode:
Diffstat (limited to 'ext/swfdec')
-rw-r--r--ext/swfdec/gstswfdec.c350
-rw-r--r--ext/swfdec/gstswfdec.h26
2 files changed, 150 insertions, 226 deletions
diff --git a/ext/swfdec/gstswfdec.c b/ext/swfdec/gstswfdec.c
index ef14d1b4..929e14db 100644
--- a/ext/swfdec/gstswfdec.c
+++ b/ext/swfdec/gstswfdec.c
@@ -24,6 +24,7 @@
#include "gstswfdec.h"
#include <string.h>
#include <gst/video/video.h>
+#include <swfdec_buffer.h>
/* elementfactory information */
static GstElementDetails gst_swfdec_details =
@@ -83,18 +84,11 @@ static void gst_swfdec_set_property (GObject * object, guint prop_id,
static void gst_swfdec_get_property (GObject * object, guint prop_id,
GValue * value, GParamSpec * pspec);
-#if 0
static gboolean gst_swfdec_src_event (GstPad * pad, GstEvent * event);
-#endif
static gboolean gst_swfdec_src_query (GstPad * pad, GstQueryType type,
GstFormat * format, gint64 * value);
-
-#if 0
-static gboolean gst_swfdec_convert_sink (GstPad * pad, GstFormat src_format,
- gint64 src_value, GstFormat * dest_format, gint64 * dest_value);
-static gboolean gst_swfdec_convert_src (GstPad * pad, GstFormat src_format,
- gint64 src_value, GstFormat * dest_format, gint64 * dest_value);
-#endif
+static const GstQueryType *gst_swfdec_get_query_types (GstPad * pad);
+static const GstEventMask *gst_swfdec_get_event_masks (GstPad * pad);
static GstElementStateReturn gst_swfdec_change_state (GstElement * element);
@@ -193,11 +187,8 @@ gst_swfdec_video_link (GstPad * pad, const GstCaps * caps)
gst_structure_get_int (structure, "width", &width);
gst_structure_get_int (structure, "height", &height);
- if (swfdec->height == height && swfdec->width == width) {
- return GST_PAD_LINK_OK;
- }
-
- ret = swfdec_decoder_set_image_size (swfdec->state, width, height);
+ g_print ("setting size to %d x %d\n", width, height);
+ ret = swfdec_decoder_set_image_size (swfdec->decoder, width, height);
if (ret == SWF_OK) {
swfdec->width = width;
swfdec->height = height;
@@ -209,23 +200,22 @@ gst_swfdec_video_link (GstPad * pad, const GstCaps * caps)
}
static void
-copy_image (void *dest, void *src, int width, int height)
+gst_swfdec_buffer_free (SwfdecBuffer * buf, void *priv)
{
- guint8 *d = dest;
- guint8 *s = src;
- int x, y;
-
- for (y = 0; y < height; y++) {
- for (x = 0; x < width; x++) {
- d[0] = s[2];
- d[1] = s[1];
- d[2] = s[0];
- d[3] = 0;
- d += 4;
- s += 3;
- }
- }
+ gst_buffer_unref (GST_BUFFER (priv));
+}
+
+static SwfdecBuffer *
+gst_swfdec_buffer_to_swf (GstBuffer * buffer)
+{
+ SwfdecBuffer *sbuf;
+
+ sbuf = swfdec_buffer_new_with_data (GST_BUFFER_DATA (buffer),
+ GST_BUFFER_SIZE (buffer));
+ sbuf->free = gst_swfdec_buffer_free;
+ sbuf->priv = buffer;
+ return sbuf;
}
static void
@@ -243,16 +233,17 @@ gst_swfdec_loop (GstElement * element)
if (!swfdec->videopad) {
}
- ret = swfdec_decoder_parse (swfdec->state);
+ ret = swfdec_decoder_parse (swfdec->decoder);
if (ret == SWF_NEEDBITS) {
buf = GST_BUFFER (gst_pad_pull (swfdec->sinkpad));
if (GST_IS_EVENT (buf)) {
switch (GST_EVENT_TYPE (buf)) {
case GST_EVENT_EOS:
+ swfdec_decoder_eof (swfdec->decoder);
GST_DEBUG ("got eos");
break;
default:
- GST_DEBUG ("got event");
+ GST_DEBUG ("got other event");
break;
}
@@ -260,22 +251,20 @@ gst_swfdec_loop (GstElement * element)
if (!GST_BUFFER_DATA (buf)) {
GST_DEBUG ("expected non-null buffer");
}
- ret = swfdec_decoder_addbits (swfdec->state,
- GST_BUFFER_DATA (buf), GST_BUFFER_SIZE (buf));
+ swfdec_decoder_add_buffer (swfdec->decoder,
+ gst_swfdec_buffer_to_swf (buf));
}
- }
-
- if (ret == SWF_CHANGE) {
+ } else if (ret == SWF_CHANGE) {
GstCaps *caps;
GstPadLinkReturn link_ret;
- swfdec_decoder_get_image_size (swfdec->state,
+ swfdec_decoder_get_image_size (swfdec->decoder,
&swfdec->width, &swfdec->height);
- swfdec_decoder_get_rate (swfdec->state, &swfdec->rate);
+ swfdec_decoder_get_rate (swfdec->decoder, &swfdec->rate);
swfdec->interval = GST_SECOND / swfdec->rate;
caps = gst_caps_copy (gst_pad_get_pad_template_caps (swfdec->videopad));
- swfdec_decoder_get_rate (swfdec->state, &swfdec->frame_rate);
+ swfdec_decoder_get_rate (swfdec->decoder, &swfdec->frame_rate);
gst_caps_set_simple (caps,
"framerate", G_TYPE_DOUBLE, swfdec->frame_rate,
"height", G_TYPE_INT, swfdec->height,
@@ -288,51 +277,61 @@ gst_swfdec_loop (GstElement * element)
return;
}
swfdec->have_format = TRUE;
+ } else if (ret == SWF_EOF) {
+ SwfdecBuffer *audio_buffer;
+ SwfdecBuffer *video_buffer;
+ GstBuffer *videobuf;
+ GstBuffer *audiobuf;
+ gboolean ret;
- return;
- }
+ ret = swfdec_render_iterate (swfdec->decoder);
+ if (!ret) {
+ gst_pad_push (swfdec->videopad, GST_DATA (gst_event_new (GST_EVENT_EOS)));
+ gst_pad_push (swfdec->audiopad, GST_DATA (gst_event_new (GST_EVENT_EOS)));
- if (ret == SWF_IMAGE) {
- GstBuffer *newbuf = NULL;
- unsigned char *data;
- int len;
+ gst_element_set_eos (GST_ELEMENT (swfdec));
- /* video stuff */
- //newbuf = gst_buffer_new();
- //GST_BUFFER_SIZE(newbuf) = swfdec->width * swfdec->height * 3;
+ return;
+ }
- newbuf = gst_pad_alloc_buffer (swfdec->videopad, GST_BUFFER_OFFSET_NONE,
- swfdec->width * 4 * swfdec->height);
+ if (swfdec->send_discont) {
+ GstEvent *event;
- swfdec_decoder_get_image (swfdec->state, &data);
- copy_image (GST_BUFFER_DATA (newbuf), data, swfdec->width, swfdec->height);
- free (data);
- //GST_BUFFER_DATA(newbuf) = data;
+ swfdec->timestamp = swfdec_render_get_frame_index (swfdec->decoder) *
+ swfdec->interval;
- swfdec->timestamp += swfdec->interval;
- GST_BUFFER_TIMESTAMP (newbuf) = swfdec->timestamp;
+ GST_DEBUG ("sending discont %" G_GINT64_FORMAT, swfdec->timestamp);
- gst_pad_push (swfdec->videopad, GST_DATA (newbuf));
+ event = gst_event_new_discontinuous (FALSE, GST_FORMAT_TIME,
+ swfdec->timestamp, GST_FORMAT_UNDEFINED);
+ gst_pad_push (swfdec->videopad, GST_DATA (event));
- /* audio stuff */
+ event = gst_event_new_discontinuous (FALSE, GST_FORMAT_TIME,
+ swfdec->timestamp, GST_FORMAT_UNDEFINED);
+ gst_pad_push (swfdec->audiopad, GST_DATA (event));
- data = swfdec_decoder_get_sound_chunk (swfdec->state, &len);
- while (data) {
- newbuf = gst_buffer_new ();
+ swfdec->send_discont = FALSE;
+ }
- GST_BUFFER_SIZE (newbuf) = len;
- GST_BUFFER_DATA (newbuf) = data;
- GST_BUFFER_TIMESTAMP (newbuf) = swfdec->timestamp;
+ GST_DEBUG ("pushing image/sound %" G_GINT64_FORMAT, swfdec->timestamp);
- gst_pad_push (swfdec->audiopad, GST_DATA (newbuf));
+ video_buffer = swfdec_render_get_image (swfdec->decoder);
+ videobuf = gst_buffer_new ();
+ GST_BUFFER_DATA (videobuf) = video_buffer->data;
+ GST_BUFFER_SIZE (videobuf) = video_buffer->length;
+ GST_BUFFER_TIMESTAMP (videobuf) = swfdec->timestamp;
- data = swfdec_decoder_get_sound_chunk (swfdec->state, &len);
- }
- }
+ gst_pad_push (swfdec->videopad, GST_DATA (videobuf));
- if (ret == SWF_EOF) {
- gst_pad_push (swfdec->videopad, GST_DATA (gst_event_new (GST_EVENT_EOS)));
- gst_pad_push (swfdec->audiopad, GST_DATA (gst_event_new (GST_EVENT_EOS)));
+ audio_buffer = swfdec_render_get_audio (swfdec->decoder);
+ audiobuf = gst_buffer_new ();
+ GST_BUFFER_DATA (audiobuf) = audio_buffer->data;
+ GST_BUFFER_SIZE (audiobuf) = audio_buffer->length;
+ GST_BUFFER_TIMESTAMP (audiobuf) = swfdec->timestamp;
+
+ gst_pad_push (swfdec->audiopad, GST_DATA (audiobuf));
+
+ swfdec->timestamp += swfdec->interval;
}
}
@@ -348,27 +347,34 @@ gst_swfdec_init (GstSwfdec * swfdec)
swfdec->videopad =
gst_pad_new_from_template (gst_static_pad_template_get
(&video_template_factory), "video_00");
- gst_pad_set_query_function (swfdec->videopad,
- GST_DEBUG_FUNCPTR (gst_swfdec_src_query));
+ gst_pad_set_query_function (swfdec->videopad, gst_swfdec_src_query);
gst_pad_set_getcaps_function (swfdec->videopad, gst_swfdec_video_getcaps);
gst_pad_set_link_function (swfdec->videopad, gst_swfdec_video_link);
+ gst_pad_set_event_function (swfdec->videopad, gst_swfdec_src_event);
+ gst_pad_set_event_mask_function (swfdec->videopad,
+ gst_swfdec_get_event_masks);
+ gst_pad_set_query_type_function (swfdec->videopad,
+ gst_swfdec_get_query_types);
gst_element_add_pad (GST_ELEMENT (swfdec), swfdec->videopad);
swfdec->audiopad =
gst_pad_new_from_template (gst_static_pad_template_get
(&audio_template_factory), "audio_00");
- gst_pad_set_query_function (swfdec->audiopad,
- GST_DEBUG_FUNCPTR (gst_swfdec_src_query));
-
+ gst_pad_set_query_function (swfdec->audiopad, gst_swfdec_src_query);
+ gst_pad_set_event_function (swfdec->audiopad, gst_swfdec_src_event);
+ gst_pad_set_event_mask_function (swfdec->audiopad,
+ gst_swfdec_get_event_masks);
+ gst_pad_set_query_type_function (swfdec->audiopad,
+ gst_swfdec_get_query_types);
gst_element_add_pad (GST_ELEMENT (swfdec), swfdec->audiopad);
gst_element_set_loop_function (GST_ELEMENT (swfdec), gst_swfdec_loop);
/* initialize the swfdec decoder state */
- swfdec->state = swfdec_decoder_new ();
- g_return_if_fail (swfdec->state != NULL);
+ swfdec->decoder = swfdec_decoder_new ();
+ g_return_if_fail (swfdec->decoder != NULL);
- swfdec_decoder_set_colorspace (swfdec->state, SWF_COLORSPACE_RGB888);
+ swfdec_decoder_set_colorspace (swfdec->decoder, SWF_COLORSPACE_RGB888);
GST_FLAG_SET (GST_ELEMENT (swfdec), GST_ELEMENT_EVENT_AWARE);
@@ -378,104 +384,24 @@ gst_swfdec_init (GstSwfdec * swfdec)
static void
gst_swfdec_dispose (GObject * object)
{
- //GstSwfdec *swfdec = GST_SWFDEC (object);
+ GstSwfdec *swfdec = GST_SWFDEC (object);
- /* FIXME */
- //swfdec_decoder_free(swfdec->state);
+ swfdec_decoder_free (swfdec->decoder);
G_OBJECT_CLASS (parent_class)->dispose (object);
}
-#if 0
-static gboolean
-gst_swfdec_convert_sink (GstPad * pad, GstFormat src_format, gint64 src_value,
- GstFormat * dest_format, gint64 * dest_value)
-{
- gboolean res = TRUE;
- GstSwfdec *swfdec;
-
- swfdec = GST_SWFDEC (gst_pad_get_parent (pad));
-
- switch (src_format) {
- case GST_FORMAT_BYTES:
- switch (*dest_format) {
- case GST_FORMAT_TIME:
- default:
- res = FALSE;
- }
- break;
- case GST_FORMAT_TIME:
- switch (*dest_format) {
- case GST_FORMAT_BYTES:
- default:
- res = FALSE;
- }
- break;
- default:
- res = FALSE;
- }
- return res;
-}
-#endif
-
-#if 0
-static gboolean
-gst_swfdec_convert_src (GstPad * pad, GstFormat src_format, gint64 src_value,
- GstFormat * dest_format, gint64 * dest_value)
+static const GstQueryType *
+gst_swfdec_get_query_types (GstPad * pad)
{
- gboolean res = TRUE;
- GstSwfdec *swfdec;
-
- swfdec = GST_SWFDEC (gst_pad_get_parent (pad));
+ static const GstQueryType gst_swfdec_query_types[] = {
+ GST_QUERY_TOTAL,
+ GST_QUERY_POSITION,
+ 0
+ };
- switch (src_format) {
- case GST_FORMAT_BYTES:
- switch (*dest_format) {
- case GST_FORMAT_TIME:
- default:
- res = FALSE;
- }
- break;
- case GST_FORMAT_TIME:
- switch (*dest_format) {
- case GST_FORMAT_BYTES:
- *dest_value = src_value * 6 * (swfdec->width * swfdec->height >> 2) *
- video_rates[swfdec->decoder->frame_rate_code] / GST_SECOND;
- break;
- case GST_FORMAT_DEFAULT:
- *dest_value =
- src_value * video_rates[swfdec->decoder->frame_rate_code] /
- GST_SECOND;
- break;
- default:
- res = FALSE;
- }
- break;
- case GST_FORMAT_DEFAULT:
- switch (*dest_format) {
- case GST_FORMAT_TIME:
- if (video_rates[swfdec->decoder->frame_rate_code] != 0.0) {
- *dest_value = src_value * GST_SECOND /
- video_rates[swfdec->decoder->frame_rate_code];
- } else
- res = FALSE;
- break;
- case GST_FORMAT_BYTES:
- *dest_value = src_value * 6 * (swfdec->width * swfdec->height >> 2);
- break;
- case GST_FORMAT_DEFAULT:
- *dest_value = src_value;
- break;
- default:
- res = FALSE;
- }
- break;
- default:
- res = FALSE;
- }
- return res;
+ return gst_swfdec_query_types;
}
-#endif
static gboolean
gst_swfdec_src_query (GstPad * pad, GstQueryType type,
@@ -496,7 +422,7 @@ gst_swfdec_src_query (GstPad * pad, GstQueryType type,
int ret;
res = FALSE;
- ret = swfdec_decoder_get_n_frames (swfdec->state, &n_frames);
+ ret = swfdec_decoder_get_n_frames (swfdec->decoder, &n_frames);
if (ret == SWF_OK) {
*value = n_frames * swfdec->interval;
res = TRUE;
@@ -512,6 +438,10 @@ gst_swfdec_src_query (GstPad * pad, GstQueryType type,
case GST_QUERY_POSITION:
{
switch (*format) {
+ case GST_FORMAT_TIME:
+ *value = swfdec_render_get_frame_index (swfdec->decoder) *
+ swfdec->interval;
+ res = TRUE;
default:
res = FALSE;
break;
@@ -526,16 +456,27 @@ gst_swfdec_src_query (GstPad * pad, GstQueryType type,
return res;
}
-#if 0
+static const GstEventMask *
+gst_swfdec_get_event_masks (GstPad * pad)
+{
+ static const GstEventMask gst_swfdec_event_masks[] = {
+ {GST_EVENT_SEEK, GST_SEEK_METHOD_SET | GST_SEEK_FLAG_FLUSH},
+ {0,}
+ };
+
+ return gst_swfdec_event_masks;
+}
+
static gboolean
gst_swfdec_src_event (GstPad * pad, GstEvent * event)
{
gboolean res = TRUE;
GstSwfdec *swfdec;
- static const GstFormat formats[] = { GST_FORMAT_TIME, GST_FORMAT_BYTES };
+
+ //static const GstFormat formats[] = { GST_FORMAT_TIME, GST_FORMAT_BYTES };
#define MAX_SEEK_FORMATS 1 /* we can only do time seeking for now */
- gint i;
+ //gint i;
swfdec = GST_SWFDEC (gst_pad_get_parent (pad));
@@ -543,50 +484,30 @@ gst_swfdec_src_event (GstPad * pad, GstEvent * event)
/* the all-formats seek logic */
case GST_EVENT_SEEK:
{
- gint64 src_offset;
- gboolean flush;
- GstFormat format;
-
- format = GST_FORMAT_TIME;
-
- /* first bring the src_format to TIME */
- if (!gst_pad_convert (pad,
- GST_EVENT_SEEK_FORMAT (event), GST_EVENT_SEEK_OFFSET (event),
- &format, &src_offset)) {
- /* didn't work, probably unsupported seek format then */
- res = FALSE;
- break;
- }
-
- /* shave off the flush flag, we'll need it later */
- flush = GST_EVENT_SEEK_FLAGS (event) & GST_SEEK_FLAG_FLUSH;
+ int new_frame;
+ int ret;
+ int n_frames;
- /* assume the worst */
- res = FALSE;
+ new_frame = event->event_data.seek.offset / swfdec->interval;
+ ret = swfdec_decoder_get_n_frames (swfdec->decoder, &n_frames);
- /* while we did not exhaust our seek formats without result */
- for (i = 0; i < MAX_SEEK_FORMATS && !res; i++) {
- gint64 desired_offset;
+ if (new_frame >= 0 && new_frame < n_frames) {
+ GstEvent *event;
- format = formats[i];
+ GST_DEBUG ("seeking to frame %d\n", new_frame);
+ swfdec_render_seek (swfdec->decoder, new_frame);
- /* try to convert requested format to one we can seek with on the sinkpad */
- if (gst_pad_convert (swfdec->sinkpad, GST_FORMAT_TIME, src_offset,
- &format, &desired_offset)) {
- GstEvent *seek_event;
+ GST_DEBUG ("sending flush event\n");
+ event = gst_event_new (GST_EVENT_FLUSH);
+ gst_pad_push (swfdec->videopad, GST_DATA (event));
+ event = gst_event_new (GST_EVENT_FLUSH);
+ gst_pad_push (swfdec->audiopad, GST_DATA (event));
- /* conversion succeeded, create the seek */
- seek_event =
- gst_event_new_seek (formats[i] | GST_SEEK_METHOD_SET | flush,
- desired_offset);
- /* do the seekk */
- if (gst_pad_send_event (GST_PAD_PEER (swfdec->sinkpad), seek_event)) {
- /* seek worked, we're done, loop will exit */
- res = TRUE;
- }
- }
- /* at this point, either the seek worked or res == FALSE */
+ swfdec->send_discont = TRUE;
+ swfdec->seek_frame = new_frame;
}
+
+ res = TRUE;
break;
}
default:
@@ -596,7 +517,6 @@ gst_swfdec_src_event (GstPad * pad, GstEvent * event)
gst_event_unref (event);
return res;
}
-#endif
static GstElementStateReturn
gst_swfdec_change_state (GstElement * element)
@@ -680,6 +600,12 @@ gst_swfdec_get_property (GObject * object, guint prop_id, GValue * value,
}
}
+void
+art_warn (const char *fmt, ...)
+{
+ GST_ERROR ("caught art_warn");
+}
+
static gboolean
plugin_init (GstPlugin * plugin)
{
diff --git a/ext/swfdec/gstswfdec.h b/ext/swfdec/gstswfdec.h
index 05015682..ea8bc59f 100644
--- a/ext/swfdec/gstswfdec.h
+++ b/ext/swfdec/gstswfdec.h
@@ -24,11 +24,9 @@
#include <gst/gst.h>
#include <swfdec.h>
+#include <swfdec_render.h>
-#ifdef __cplusplus
-extern "C" {
-#endif /* __cplusplus */
-
+G_BEGIN_DECLS
#define GST_TYPE_SWFDEC \
(gst_swfdec_get_type())
@@ -44,7 +42,8 @@ extern "C" {
typedef struct _GstSwfdec GstSwfdec;
typedef struct _GstSwfdecClass GstSwfdecClass;
-struct _GstSwfdec {
+struct _GstSwfdec
+{
GstElement element;
/* pads */
@@ -52,13 +51,15 @@ struct _GstSwfdec {
GstPad *videopad;
GstPad *audiopad;
- SwfdecDecoder *state;
+ SwfdecDecoder *decoder;
gboolean closed;
- /* the timestamp of the next frame */
gboolean first;
gboolean have_format;
+ gboolean send_discont;
+ int seek_frame;
+
double rate;
gint64 timestamp;
gint64 interval;
@@ -72,16 +73,13 @@ struct _GstSwfdec {
};
-struct _GstSwfdecClass {
+struct _GstSwfdecClass
+{
GstElementClass parent_class;
};
-GType gst_swfdec_get_type(void);
-
-
-#ifdef __cplusplus
-}
-#endif /* __cplusplus */
+GType gst_swfdec_get_type (void);
+G_END_DECLS
#endif /* __GST_SWFDEC_H__ */