mirror of
https://gitlab.freedesktop.org/gstreamer/gstreamer.git
synced 2024-12-23 08:46:40 +00:00
Port dvdsubdec to 0.10
Original commit message from CVS: * configure.ac: * gst/dvdsub/Makefile.am: * gst/dvdsub/gstdvdsubdec.c: (gst_dvd_sub_dec_base_init), (gst_dvd_sub_dec_class_init), (gst_dvd_sub_dec_init), (gst_dvd_sub_dec_finalize), (gst_dvd_sub_dec_src_event), (gst_dvd_sub_dec_get_event_delay), (gst_dvd_sub_dec_parse_subpic), (gst_get_nibble), (gst_setup_palette), (gst_get_rle_code), (gst_draw_rle_line), (gst_dvd_sub_dec_merge_title), (gst_send_empty_fill), (gst_send_subtitle_frame), (gst_dvd_sub_dec_advance_time), (gst_dvd_sub_dec_chain), (gst_dvd_sub_dec_sink_event), (gst_dvd_sub_dec_handle_dvd_event), (plugin_init): * gst/dvdsub/gstdvdsubdec.h: Port dvdsubdec to 0.10
This commit is contained in:
parent
1d66c5e4e9
commit
56f0cea8e6
5 changed files with 1115 additions and 0 deletions
17
ChangeLog
17
ChangeLog
|
@ -1,3 +1,20 @@
|
|||
2006-02-22 Tim-Philipp Müller <tim at centricular dot net>
|
||||
|
||||
* configure.ac:
|
||||
* gst/dvdsub/Makefile.am:
|
||||
* gst/dvdsub/gstdvdsubdec.c: (gst_dvd_sub_dec_base_init),
|
||||
(gst_dvd_sub_dec_class_init), (gst_dvd_sub_dec_init),
|
||||
(gst_dvd_sub_dec_finalize), (gst_dvd_sub_dec_src_event),
|
||||
(gst_dvd_sub_dec_get_event_delay), (gst_dvd_sub_dec_parse_subpic),
|
||||
(gst_get_nibble), (gst_setup_palette), (gst_get_rle_code),
|
||||
(gst_draw_rle_line), (gst_dvd_sub_dec_merge_title),
|
||||
(gst_send_empty_fill), (gst_send_subtitle_frame),
|
||||
(gst_dvd_sub_dec_advance_time), (gst_dvd_sub_dec_chain),
|
||||
(gst_dvd_sub_dec_sink_event), (gst_dvd_sub_dec_handle_dvd_event),
|
||||
(plugin_init):
|
||||
* gst/dvdsub/gstdvdsubdec.h:
|
||||
Port dvdsubdec to 0.10
|
||||
|
||||
2006-02-21 Jan Schmidt <thaytan@mad.scientist.com>
|
||||
|
||||
* gst/mpegstream/gstdvddemux.c: (gst_dvd_demux_plugin_init):
|
||||
|
|
|
@ -210,6 +210,7 @@ dnl these are all the gst plug-ins, compilable without additional libs
|
|||
GST_PLUGINS_ALL="\
|
||||
asfdemux \
|
||||
dvdlpcmdec \
|
||||
dvdsub \
|
||||
iec958 \
|
||||
mpegaudioparse \
|
||||
mpegstream \
|
||||
|
@ -393,6 +394,7 @@ Makefile
|
|||
gst/Makefile
|
||||
gst/asfdemux/Makefile
|
||||
gst/dvdlpcmdec/Makefile
|
||||
gst/dvdsub/Makefile
|
||||
gst/iec958/Makefile
|
||||
gst/mpegaudioparse/Makefile
|
||||
gst/mpegstream/Makefile
|
||||
|
|
9
gst/dvdsub/Makefile.am
Normal file
9
gst/dvdsub/Makefile.am
Normal file
|
@ -0,0 +1,9 @@
|
|||
|
||||
plugin_LTLIBRARIES = libgstdvdsub.la
|
||||
|
||||
libgstdvdsub_la_SOURCES = gstdvdsubdec.c
|
||||
libgstdvdsub_la_CFLAGS = $(GST_CFLAGS)
|
||||
libgstdvdsub_la_LIBADD =
|
||||
libgstdvdsub_la_LDFLAGS = $(GST_PLUGIN_LDFLAGS)
|
||||
|
||||
noinst_HEADERS = gstdvdsubdec.h
|
992
gst/dvdsub/gstdvdsubdec.c
Normal file
992
gst/dvdsub/gstdvdsubdec.c
Normal file
|
@ -0,0 +1,992 @@
|
|||
/* GStreamer
|
||||
* Copyright (C) <2005> Jan Schmidt <jan@fluendo.com>
|
||||
* Copyright (C) <2002> Wim Taymans <wim@fluendo.com>
|
||||
*
|
||||
* 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.
|
||||
*/
|
||||
|
||||
/* TODO: liboil-ise code, esp. use _splat() family of functions */
|
||||
|
||||
#ifdef HAVE_CONFIG_H
|
||||
#include "config.h"
|
||||
#endif
|
||||
|
||||
#include "gstdvdsubdec.h"
|
||||
#include <string.h>
|
||||
|
||||
GST_BOILERPLATE (GstDvdSubDec, gst_dvd_sub_dec, GstElement, GST_TYPE_ELEMENT)
|
||||
|
||||
static gboolean gst_dvd_sub_dec_src_event (GstPad * srcpad,
|
||||
GstEvent * event);
|
||||
static GstFlowReturn gst_dvd_sub_dec_chain (GstPad * pad, GstBuffer * buf);
|
||||
|
||||
static gboolean gst_dvd_sub_dec_handle_dvd_event (GstDvdSubDec * dec,
|
||||
GstEvent * event);
|
||||
static void gst_dvd_sub_dec_finalize (GObject * gobject);
|
||||
static void gst_setup_palette (GstDvdSubDec * dec);
|
||||
static void gst_dvd_sub_dec_merge_title (GstDvdSubDec * dec,
|
||||
GstBuffer * buf);
|
||||
static GstClockTime gst_dvd_sub_dec_get_event_delay (GstDvdSubDec * dec);
|
||||
static gboolean gst_dvd_sub_dec_sink_event (GstPad * pad,
|
||||
GstEvent * event);
|
||||
|
||||
static GstFlowReturn gst_send_subtitle_frame (GstDvdSubDec * dec,
|
||||
GstClockTime end_ts);
|
||||
|
||||
static GstElementDetails gst_dvd_sub_dec_details = {
|
||||
"DVD subtitle Decoder",
|
||||
"Codec/Decoder/Video",
|
||||
"Decodes DVD subtitles into AYUV video frames",
|
||||
"Wim Taymans <wim.taymans@chello.be>, "
|
||||
"Jan Schmidt <thaytan@mad.scientist.com>"
|
||||
};
|
||||
|
||||
static GstStaticPadTemplate src_template = GST_STATIC_PAD_TEMPLATE ("src",
|
||||
GST_PAD_SRC,
|
||||
GST_PAD_ALWAYS,
|
||||
GST_STATIC_CAPS ("video/x-raw-yuv, format = (fourcc) AYUV, "
|
||||
"width = (int) 720, height = (int) 576, framerate = (fraction) 0/1")
|
||||
);
|
||||
|
||||
static GstStaticPadTemplate subtitle_template = GST_STATIC_PAD_TEMPLATE ("sink",
|
||||
GST_PAD_SINK,
|
||||
GST_PAD_ALWAYS,
|
||||
GST_STATIC_CAPS ("video/x-dvd-subpicture")
|
||||
);
|
||||
|
||||
GST_DEBUG_CATEGORY_STATIC (gst_dvd_sub_dec_debug);
|
||||
#define GST_CAT_DEFAULT (gst_dvd_sub_dec_debug)
|
||||
|
||||
enum
|
||||
{
|
||||
SPU_FORCE_DISPLAY = 0x00,
|
||||
SPU_SHOW = 0x01,
|
||||
SPU_HIDE = 0x02,
|
||||
SPU_SET_PALETTE = 0x03,
|
||||
SPU_SET_ALPHA = 0x04,
|
||||
SPU_SET_SIZE = 0x05,
|
||||
SPU_SET_OFFSETS = 0x06,
|
||||
SPU_WIPE = 0x07,
|
||||
SPU_END = 0xff
|
||||
};
|
||||
|
||||
static const guint32 default_clut[16] = {
|
||||
0xb48080, 0x248080, 0x628080, 0xd78080,
|
||||
0x808080, 0x808080, 0x808080, 0x808080,
|
||||
0x808080, 0x808080, 0x808080, 0x808080,
|
||||
0x808080, 0x808080, 0x808080, 0x808080
|
||||
};
|
||||
|
||||
typedef struct RLE_state
|
||||
{
|
||||
gint id;
|
||||
gint aligned;
|
||||
gint offset[2];
|
||||
gint hl_left;
|
||||
gint hl_right;
|
||||
|
||||
guchar *target;
|
||||
|
||||
guchar next;
|
||||
}
|
||||
RLE_state;
|
||||
|
||||
static void
|
||||
gst_dvd_sub_dec_base_init (gpointer klass)
|
||||
{
|
||||
GstElementClass *element_class = GST_ELEMENT_CLASS (klass);
|
||||
|
||||
gst_element_class_add_pad_template (element_class,
|
||||
gst_static_pad_template_get (&src_template));
|
||||
gst_element_class_add_pad_template (element_class,
|
||||
gst_static_pad_template_get (&subtitle_template));
|
||||
|
||||
gst_element_class_set_details (element_class, &gst_dvd_sub_dec_details);
|
||||
}
|
||||
|
||||
static void
|
||||
gst_dvd_sub_dec_class_init (GstDvdSubDecClass * klass)
|
||||
{
|
||||
GObjectClass *gobject_class;
|
||||
GstElementClass *gstelement_class;
|
||||
|
||||
gobject_class = (GObjectClass *) klass;
|
||||
gstelement_class = (GstElementClass *) klass;
|
||||
|
||||
gobject_class->finalize = gst_dvd_sub_dec_finalize;
|
||||
}
|
||||
|
||||
static void
|
||||
gst_dvd_sub_dec_init (GstDvdSubDec * dec, GstDvdSubDecClass * klass)
|
||||
{
|
||||
GstPadTemplate *tmpl;
|
||||
|
||||
dec->sinkpad =
|
||||
gst_pad_new_from_template (gst_static_pad_template_get
|
||||
(&subtitle_template), "sink");
|
||||
gst_pad_set_chain_function (dec->sinkpad,
|
||||
GST_DEBUG_FUNCPTR (gst_dvd_sub_dec_chain));
|
||||
gst_pad_set_event_function (dec->sinkpad,
|
||||
GST_DEBUG_FUNCPTR (gst_dvd_sub_dec_sink_event));
|
||||
gst_element_add_pad (GST_ELEMENT (dec), dec->sinkpad);
|
||||
|
||||
tmpl = gst_static_pad_template_get (&src_template);
|
||||
dec->srcpad = gst_pad_new_from_template (tmpl, "src");
|
||||
gst_pad_set_event_function (dec->srcpad,
|
||||
GST_DEBUG_FUNCPTR (gst_dvd_sub_dec_src_event));
|
||||
gst_pad_use_fixed_caps (dec->srcpad);
|
||||
gst_pad_set_caps (dec->srcpad, gst_pad_template_get_caps (tmpl));
|
||||
gst_object_unref (tmpl);
|
||||
gst_element_add_pad (GST_ELEMENT (dec), dec->srcpad);
|
||||
|
||||
/* FIXME: aren't there more possible sizes? (tpm) */
|
||||
dec->in_width = 720;
|
||||
dec->in_height = 576;
|
||||
|
||||
dec->partialbuf = NULL;
|
||||
dec->have_title = FALSE;
|
||||
dec->parse_pos = NULL;
|
||||
dec->forced_display = FALSE;
|
||||
dec->visible = FALSE;
|
||||
|
||||
memset (dec->menu_index, 0, sizeof (dec->menu_index));
|
||||
memset (dec->menu_alpha, 0, sizeof (dec->menu_alpha));
|
||||
memset (dec->subtitle_index, 0, sizeof (dec->subtitle_index));
|
||||
memset (dec->subtitle_alpha, 0, sizeof (dec->subtitle_alpha));
|
||||
memcpy (dec->current_clut, default_clut, sizeof (guint32) * 16);
|
||||
|
||||
gst_setup_palette (dec);
|
||||
|
||||
dec->next_ts = 0;
|
||||
dec->next_event_ts = GST_CLOCK_TIME_NONE;
|
||||
|
||||
dec->out_buffer = NULL;
|
||||
dec->buf_dirty = TRUE;
|
||||
}
|
||||
|
||||
static void
|
||||
gst_dvd_sub_dec_finalize (GObject * gobject)
|
||||
{
|
||||
GstDvdSubDec *dec = GST_DVD_SUB_DEC (gobject);
|
||||
|
||||
if (dec->partialbuf) {
|
||||
gst_buffer_unref (dec->partialbuf);
|
||||
dec->partialbuf = NULL;
|
||||
}
|
||||
|
||||
G_OBJECT_CLASS (parent_class)->finalize (gobject);
|
||||
}
|
||||
|
||||
static gboolean
|
||||
gst_dvd_sub_dec_src_event (GstPad * pad, GstEvent * event)
|
||||
{
|
||||
GstDvdSubDec *dec = GST_DVD_SUB_DEC (gst_pad_get_parent (pad));
|
||||
gboolean res = FALSE;
|
||||
|
||||
switch (GST_EVENT_TYPE (event)) {
|
||||
default:
|
||||
res = gst_pad_event_default (pad, event);
|
||||
break;
|
||||
}
|
||||
|
||||
gst_object_unref (dec);
|
||||
return res;
|
||||
}
|
||||
|
||||
static GstClockTime
|
||||
gst_dvd_sub_dec_get_event_delay (GstDvdSubDec * dec)
|
||||
{
|
||||
guchar *start = GST_BUFFER_DATA (dec->partialbuf);
|
||||
guchar *buf;
|
||||
guint16 ticks;
|
||||
GstClockTime event_delay;
|
||||
|
||||
/* If starting a new buffer, follow the first DCSQ ptr */
|
||||
if (dec->parse_pos == start) {
|
||||
buf = dec->parse_pos + dec->data_size;
|
||||
} else {
|
||||
buf = dec->parse_pos;
|
||||
}
|
||||
|
||||
ticks = GST_READ_UINT16_BE (buf);
|
||||
event_delay = gst_util_uint64_scale (ticks, 1024 * GST_SECOND, 90000);
|
||||
|
||||
GST_DEBUG_OBJECT (dec, "returning delay %" GST_TIME_FORMAT " from offset %u",
|
||||
GST_TIME_ARGS (event_delay), (guint) (buf - dec->parse_pos));
|
||||
|
||||
return event_delay;
|
||||
}
|
||||
|
||||
/*
|
||||
* Parse the next event time in the current subpicture buffer, stopping
|
||||
* when time advances to the next state.
|
||||
*/
|
||||
static void
|
||||
gst_dvd_sub_dec_parse_subpic (GstDvdSubDec * dec)
|
||||
{
|
||||
#define PARSE_BYTES_NEEDED(x) if ((buf+(x)) >= end) \
|
||||
{ GST_WARNING("Subtitle stream broken parsing %c", *buf); \
|
||||
broken = TRUE; break; }
|
||||
|
||||
guchar *start = GST_BUFFER_DATA (dec->partialbuf);
|
||||
guchar *buf;
|
||||
guchar *end;
|
||||
gboolean broken = FALSE;
|
||||
gboolean last_seq = FALSE;
|
||||
guchar *next_seq = NULL;
|
||||
guint event_time;
|
||||
|
||||
/* nothing to do if we finished this buffer already */
|
||||
if (dec->parse_pos == NULL)
|
||||
return;
|
||||
|
||||
g_return_if_fail (dec->packet_size >= 4);
|
||||
|
||||
end = start + dec->packet_size;
|
||||
if (dec->parse_pos == start) {
|
||||
buf = dec->parse_pos + dec->data_size;
|
||||
} else {
|
||||
buf = dec->parse_pos;
|
||||
}
|
||||
|
||||
g_assert (buf >= start && buf < end);
|
||||
|
||||
/* If the next control sequence is at the current offset, this is
|
||||
* the last one */
|
||||
next_seq = start + GST_READ_UINT16_BE (buf + 2);
|
||||
last_seq = (next_seq == buf);
|
||||
buf += 4;
|
||||
|
||||
while ((buf < end) && (!broken)) {
|
||||
switch (*buf) {
|
||||
case SPU_FORCE_DISPLAY: /* Forced display menu subtitle */
|
||||
dec->forced_display = TRUE;
|
||||
dec->buf_dirty = TRUE;
|
||||
GST_DEBUG_OBJECT (dec, "SPU FORCE_DISPLAY");
|
||||
buf++;
|
||||
break;
|
||||
case SPU_SHOW: /* Show the subtitle in this packet */
|
||||
dec->visible = TRUE;
|
||||
dec->buf_dirty = TRUE;
|
||||
GST_DEBUG_OBJECT (dec, "SPU SHOW at %" GST_TIME_FORMAT,
|
||||
GST_TIME_ARGS (dec->next_event_ts));
|
||||
buf++;
|
||||
break;
|
||||
case SPU_HIDE:
|
||||
/* 02 ff (ff) is the end of the packet, hide the subpicture */
|
||||
dec->visible = FALSE;
|
||||
dec->buf_dirty = TRUE;
|
||||
|
||||
GST_DEBUG_OBJECT (dec, "SPU HIDE at %" GST_TIME_FORMAT,
|
||||
GST_TIME_ARGS (dec->next_event_ts));
|
||||
buf++;
|
||||
break;
|
||||
case SPU_SET_PALETTE: /* palette */
|
||||
PARSE_BYTES_NEEDED (3);
|
||||
|
||||
GST_DEBUG_OBJECT (dec, "SPU SET_PALETTE");
|
||||
|
||||
dec->subtitle_index[3] = buf[1] >> 4;
|
||||
dec->subtitle_index[2] = buf[1] & 0xf;
|
||||
dec->subtitle_index[1] = buf[2] >> 4;
|
||||
dec->subtitle_index[0] = buf[2] & 0xf;
|
||||
gst_setup_palette (dec);
|
||||
|
||||
dec->buf_dirty = TRUE;
|
||||
buf += 3;
|
||||
break;
|
||||
case SPU_SET_ALPHA: /* transparency palette */
|
||||
PARSE_BYTES_NEEDED (3);
|
||||
|
||||
GST_DEBUG_OBJECT (dec, "SPU SET_ALPHA");
|
||||
|
||||
dec->subtitle_alpha[3] = buf[1] >> 4;
|
||||
dec->subtitle_alpha[2] = buf[1] & 0xf;
|
||||
dec->subtitle_alpha[1] = buf[2] >> 4;
|
||||
dec->subtitle_alpha[0] = buf[2] & 0xf;
|
||||
gst_setup_palette (dec);
|
||||
|
||||
dec->buf_dirty = TRUE;
|
||||
buf += 3;
|
||||
break;
|
||||
case SPU_SET_SIZE: /* image coordinates */
|
||||
PARSE_BYTES_NEEDED (7);
|
||||
|
||||
dec->left =
|
||||
CLAMP ((((guint) buf[1]) << 4) | (buf[2] >> 4), 0,
|
||||
(dec->in_width - 1));
|
||||
dec->top =
|
||||
CLAMP ((((guint) buf[4]) << 4) | (buf[5] >> 4), 0,
|
||||
(dec->in_height - 1));
|
||||
dec->right =
|
||||
CLAMP ((((buf[2] & 0x0f) << 8) | buf[3]), 0, (dec->in_width - 1));
|
||||
dec->bottom =
|
||||
CLAMP ((((buf[5] & 0x0f) << 8) | buf[6]), 0, (dec->in_height - 1));
|
||||
|
||||
GST_DEBUG_OBJECT (dec, "SPU SET_SIZE left %d, top %d, right %d, "
|
||||
"bottom %d", dec->left, dec->top, dec->right, dec->bottom);
|
||||
|
||||
dec->buf_dirty = TRUE;
|
||||
buf += 7;
|
||||
break;
|
||||
case SPU_SET_OFFSETS: /* image 1 / image 2 offsets */
|
||||
PARSE_BYTES_NEEDED (5);
|
||||
|
||||
dec->offset[0] = (((guint) buf[1]) << 8) | buf[2];
|
||||
dec->offset[1] = (((guint) buf[3]) << 8) | buf[4];
|
||||
GST_DEBUG_OBJECT (dec, "Offset1 %d, Offset2 %d",
|
||||
dec->offset[0], dec->offset[1]);
|
||||
|
||||
dec->buf_dirty = TRUE;
|
||||
buf += 5;
|
||||
break;
|
||||
case SPU_WIPE:
|
||||
{
|
||||
guint length;
|
||||
|
||||
PARSE_BYTES_NEEDED (3);
|
||||
|
||||
GST_WARNING_OBJECT (dec, "SPU_WIPE not yet implemented");
|
||||
|
||||
length = (buf[1] << 8) | (buf[2]);
|
||||
buf += 1 + length;
|
||||
|
||||
dec->buf_dirty = TRUE;
|
||||
break;
|
||||
}
|
||||
case SPU_END:
|
||||
buf = (last_seq) ? end : next_seq;
|
||||
|
||||
/* Start a new control sequence */
|
||||
if (buf + 4 < end) {
|
||||
gint ticks = GST_READ_UINT16_BE (buf);
|
||||
|
||||
event_time = gst_util_uint64_scale (ticks, 1024 * GST_SECOND, 90000);
|
||||
|
||||
GST_DEBUG_OBJECT (dec,
|
||||
"Next DCSQ at offset %d, delay %g secs (%d ticks)", buf - start,
|
||||
(gdouble) event_time / GST_SECOND, ticks);
|
||||
|
||||
dec->parse_pos = buf;
|
||||
if (event_time > 0) {
|
||||
dec->next_event_ts += event_time;
|
||||
|
||||
GST_LOG_OBJECT (dec, "Exiting parse loop with time %g",
|
||||
(gdouble) dec->next_event_ts / GST_SECOND);
|
||||
return;
|
||||
}
|
||||
} else {
|
||||
dec->parse_pos = NULL;
|
||||
dec->next_event_ts = GST_CLOCK_TIME_NONE;
|
||||
GST_LOG_OBJECT (dec, "Finished all cmds. Exiting parse loop");
|
||||
return;
|
||||
}
|
||||
default:
|
||||
GST_ERROR
|
||||
("Invalid sequence in subtitle packet header (%.2x). Skipping",
|
||||
*buf);
|
||||
broken = TRUE;
|
||||
dec->parse_pos = NULL;
|
||||
break;
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
static inline int
|
||||
gst_get_nibble (guchar * buffer, RLE_state * state)
|
||||
{
|
||||
if (state->aligned) {
|
||||
state->next = buffer[state->offset[state->id]++];
|
||||
state->aligned = 0;
|
||||
return state->next >> 4;
|
||||
} else {
|
||||
state->aligned = 1;
|
||||
return state->next & 0xf;
|
||||
}
|
||||
}
|
||||
|
||||
/* Premultiply the current lookup table into the "target" cache */
|
||||
static void
|
||||
gst_setup_palette (GstDvdSubDec * dec)
|
||||
{
|
||||
gint i;
|
||||
guint32 col;
|
||||
YUVA_val *target = dec->palette_cache;
|
||||
YUVA_val *target2 = dec->hl_palette_cache;
|
||||
|
||||
for (i = 0; i < 4; i++, target2++, target++) {
|
||||
col = dec->current_clut[dec->subtitle_index[i]];
|
||||
target->Y = (col >> 16) & 0xff;
|
||||
target->V = (col >> 8) & 0xff;
|
||||
target->U = col & 0xff;
|
||||
target->A = dec->subtitle_alpha[i] * 0xff / 0xf;
|
||||
|
||||
col = dec->current_clut[dec->menu_index[i]];
|
||||
target2->Y = (col >> 16) & 0xff;
|
||||
target2->V = (col >> 8) & 0xff;
|
||||
target2->U = col & 0xff;
|
||||
target2->A = dec->menu_alpha[i] * 0xff / 0xf;
|
||||
}
|
||||
}
|
||||
|
||||
static inline guint
|
||||
gst_get_rle_code (guchar * buffer, RLE_state * state)
|
||||
{
|
||||
gint code;
|
||||
|
||||
code = gst_get_nibble (buffer, state);
|
||||
if (code < 0x4) { /* 4 .. f */
|
||||
code = (code << 4) | gst_get_nibble (buffer, state);
|
||||
if (code < 0x10) { /* 1x .. 3x */
|
||||
code = (code << 4) | gst_get_nibble (buffer, state);
|
||||
if (code < 0x40) { /* 04x .. 0fx */
|
||||
code = (code << 4) | gst_get_nibble (buffer, state);
|
||||
}
|
||||
}
|
||||
}
|
||||
return code;
|
||||
}
|
||||
|
||||
#define DRAW_RUN(target,len,c) \
|
||||
G_STMT_START { \
|
||||
if ((c)->A) { \
|
||||
gint i; \
|
||||
for (i = 0; i < (len); i++) { \
|
||||
*(target)++ = (c)->A; \
|
||||
*(target)++ = (c)->Y; \
|
||||
*(target)++ = (c)->U; \
|
||||
*(target)++ = (c)->V; \
|
||||
} \
|
||||
} else { \
|
||||
(target) += 4 * (len); \
|
||||
} \
|
||||
} G_STMT_END
|
||||
|
||||
/*
|
||||
* This function steps over each run-length segment, drawing
|
||||
* into the YUVA buffers as it goes. UV are composited and then output
|
||||
* at half width/height
|
||||
*/
|
||||
static void
|
||||
gst_draw_rle_line (GstDvdSubDec * dec, guchar * buffer, RLE_state * state)
|
||||
{
|
||||
gint length, colourid;
|
||||
guint code;
|
||||
gint x, right;
|
||||
guchar *target;
|
||||
|
||||
target = state->target;
|
||||
|
||||
x = dec->left;
|
||||
right = dec->right + 1;
|
||||
|
||||
while (x < right) {
|
||||
gboolean in_hl;
|
||||
const YUVA_val *colour_entry;
|
||||
|
||||
code = gst_get_rle_code (buffer, state);
|
||||
length = code >> 2;
|
||||
colourid = code & 3;
|
||||
colour_entry = dec->palette_cache + colourid;
|
||||
|
||||
/* Length = 0 implies fill to the end of the line */
|
||||
/* Restrict the colour run to the end of the line */
|
||||
if (length == 0 || x + length > right)
|
||||
length = right - x;
|
||||
|
||||
/* Check if this run of colour touches the highlight region */
|
||||
in_hl = ((x <= state->hl_right) && (x + length) >= state->hl_left);
|
||||
if (in_hl) {
|
||||
gint run;
|
||||
|
||||
/* Draw to the left of the highlight */
|
||||
if (x <= state->hl_left) {
|
||||
run = MIN (length, state->hl_left - x + 1);
|
||||
|
||||
DRAW_RUN (target, run, colour_entry);
|
||||
length -= run;
|
||||
x += run;
|
||||
}
|
||||
|
||||
/* Draw across the highlight region */
|
||||
if (x <= state->hl_right) {
|
||||
const YUVA_val *hl_colour = dec->hl_palette_cache + colourid;
|
||||
|
||||
run = MIN (length, state->hl_right - x + 1);
|
||||
|
||||
DRAW_RUN (target, run, hl_colour);
|
||||
length -= run;
|
||||
x += run;
|
||||
}
|
||||
}
|
||||
|
||||
/* Draw the rest of the run */
|
||||
if (length > 0) {
|
||||
DRAW_RUN (target, length, colour_entry);
|
||||
x += length;
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
/*
|
||||
* Decode the RLE subtitle image and blend with the current
|
||||
* frame buffer.
|
||||
*/
|
||||
static void
|
||||
gst_dvd_sub_dec_merge_title (GstDvdSubDec * dec, GstBuffer * buf)
|
||||
{
|
||||
gint y;
|
||||
gint Y_stride = 4 * dec->in_width;
|
||||
guchar *buffer = GST_BUFFER_DATA (dec->partialbuf);
|
||||
|
||||
gint hl_top, hl_bottom;
|
||||
gint last_y;
|
||||
RLE_state state;
|
||||
|
||||
GST_DEBUG_OBJECT (dec, "Merging subtitle on frame at time %" GST_TIME_FORMAT,
|
||||
GST_TIME_ARGS (GST_BUFFER_TIMESTAMP (buf)));
|
||||
|
||||
state.id = 0;
|
||||
state.aligned = 1;
|
||||
state.next = 0;
|
||||
state.offset[0] = dec->offset[0];
|
||||
state.offset[1] = dec->offset[1];
|
||||
|
||||
if (dec->current_button) {
|
||||
hl_top = dec->hl_top;
|
||||
hl_bottom = dec->hl_bottom;
|
||||
} else {
|
||||
hl_top = -1;
|
||||
hl_bottom = -1;
|
||||
}
|
||||
last_y = MIN (dec->bottom, dec->in_height);
|
||||
|
||||
y = dec->top;
|
||||
state.target = GST_BUFFER_DATA (buf) + 4 * dec->left + (y * Y_stride);
|
||||
|
||||
/* Now draw scanlines until we hit last_y or end of RLE data */
|
||||
for (; ((state.offset[1] < dec->data_size + 2) && (y <= last_y)); y++) {
|
||||
/* Set up to draw the highlight if we're in the right scanlines */
|
||||
if (y > hl_bottom || y < hl_top) {
|
||||
state.hl_left = -1;
|
||||
state.hl_right = -1;
|
||||
} else {
|
||||
state.hl_left = dec->hl_left;
|
||||
state.hl_right = dec->hl_right;
|
||||
}
|
||||
gst_draw_rle_line (dec, buffer, &state);
|
||||
|
||||
state.target += Y_stride;
|
||||
|
||||
/* Realign the RLE state for the next line */
|
||||
if (!state.aligned)
|
||||
gst_get_nibble (buffer, &state);
|
||||
state.id = !state.id;
|
||||
}
|
||||
}
|
||||
|
||||
static void
|
||||
gst_send_empty_fill (GstDvdSubDec * dec, GstClockTime ts)
|
||||
{
|
||||
if (dec->next_ts < ts) {
|
||||
GST_LOG_OBJECT (dec, "Sending newsegment update to advance time to %"
|
||||
GST_TIME_FORMAT, GST_TIME_ARGS (ts));
|
||||
|
||||
gst_pad_push_event (dec->srcpad,
|
||||
gst_event_new_new_segment (TRUE, 1.0, GST_FORMAT_TIME, ts, -1, ts));
|
||||
}
|
||||
dec->next_ts = ts;
|
||||
}
|
||||
|
||||
static GstFlowReturn
|
||||
gst_send_subtitle_frame (GstDvdSubDec * dec, GstClockTime end_ts)
|
||||
{
|
||||
GstFlowReturn flow;
|
||||
GstBuffer *out_buf;
|
||||
gint x, y;
|
||||
|
||||
g_assert (dec->have_title);
|
||||
g_assert (dec->next_ts <= end_ts);
|
||||
|
||||
/* Check if we need to redraw the output buffer */
|
||||
if (dec->buf_dirty) {
|
||||
if (dec->out_buffer) {
|
||||
gst_buffer_unref (dec->out_buffer);
|
||||
dec->out_buffer = NULL;
|
||||
}
|
||||
|
||||
flow = gst_pad_alloc_buffer_and_set_caps (dec->srcpad, 0,
|
||||
4 * dec->in_width * dec->in_height, GST_PAD_CAPS (dec->srcpad),
|
||||
&out_buf);
|
||||
|
||||
if (flow != GST_FLOW_OK) {
|
||||
GST_DEBUG_OBJECT (dec, "alloc buffer failed: flow = %s",
|
||||
gst_flow_get_name (flow));
|
||||
goto out;
|
||||
}
|
||||
|
||||
/* Clear the buffer */
|
||||
/* FIXME - move this into the buffer rendering code */
|
||||
for (y = 0; y < dec->in_height; y++) {
|
||||
guchar *line = GST_BUFFER_DATA (out_buf) + 4 * dec->in_width * y;
|
||||
|
||||
for (x = 0; x < dec->in_width; x++) {
|
||||
line[0] = 0; /* A */
|
||||
line[1] = 16; /* Y */
|
||||
line[2] = 128; /* U */
|
||||
line[3] = 128; /* V */
|
||||
|
||||
line += 4;
|
||||
}
|
||||
}
|
||||
|
||||
/* FIXME: do we really want to honour the forced_display flag
|
||||
* for subtitles streans? */
|
||||
if (dec->visible || dec->forced_display) {
|
||||
gst_dvd_sub_dec_merge_title (dec, out_buf);
|
||||
}
|
||||
|
||||
dec->out_buffer = out_buf;
|
||||
dec->buf_dirty = FALSE;
|
||||
}
|
||||
|
||||
out_buf = gst_buffer_create_sub (dec->out_buffer, 0,
|
||||
GST_BUFFER_SIZE (dec->out_buffer));
|
||||
|
||||
GST_BUFFER_TIMESTAMP (out_buf) = dec->next_ts;
|
||||
GST_BUFFER_DURATION (out_buf) = GST_CLOCK_DIFF (end_ts, dec->next_ts);
|
||||
|
||||
GST_DEBUG_OBJECT (dec, "Sending subtitle buffer with ts %"
|
||||
GST_TIME_FORMAT ", dur %" G_GINT64_FORMAT,
|
||||
GST_TIME_ARGS (GST_BUFFER_TIMESTAMP (out_buf)),
|
||||
GST_BUFFER_DURATION (out_buf));
|
||||
|
||||
gst_buffer_set_caps (out_buf, GST_PAD_CAPS (dec->srcpad));
|
||||
|
||||
flow = gst_pad_push (dec->srcpad, out_buf);
|
||||
|
||||
out:
|
||||
|
||||
dec->next_ts = end_ts;
|
||||
return flow;
|
||||
}
|
||||
|
||||
/* Walk time forward, processing any subtitle events as needed. */
|
||||
static GstFlowReturn
|
||||
gst_dvd_sub_dec_advance_time (GstDvdSubDec * dec, GstClockTime new_ts)
|
||||
{
|
||||
GstFlowReturn ret = GST_FLOW_OK;
|
||||
|
||||
GST_LOG_OBJECT (dec, "Advancing time to %" GST_TIME_FORMAT,
|
||||
GST_TIME_ARGS (new_ts));
|
||||
|
||||
if (!dec->have_title) {
|
||||
gst_send_empty_fill (dec, new_ts);
|
||||
return ret;
|
||||
}
|
||||
|
||||
while (dec->next_ts < new_ts) {
|
||||
GstClockTime next_ts = new_ts;
|
||||
|
||||
if (GST_CLOCK_TIME_IS_VALID (dec->next_event_ts) &&
|
||||
dec->next_event_ts < next_ts) {
|
||||
/* We might need to process the subtitle cmd queue */
|
||||
next_ts = dec->next_event_ts;
|
||||
}
|
||||
|
||||
/*
|
||||
* Now, either output a filler or a frame spanning
|
||||
* dec->next_ts to next_ts
|
||||
*/
|
||||
if (dec->visible || dec->forced_display) {
|
||||
ret = gst_send_subtitle_frame (dec, next_ts);
|
||||
} else {
|
||||
gst_send_empty_fill (dec, next_ts);
|
||||
}
|
||||
|
||||
/*
|
||||
* and then process some subtitle cmds if we need
|
||||
*/
|
||||
if (next_ts == dec->next_event_ts)
|
||||
gst_dvd_sub_dec_parse_subpic (dec);
|
||||
}
|
||||
|
||||
return ret;
|
||||
}
|
||||
|
||||
static GstFlowReturn
|
||||
gst_dvd_sub_dec_chain (GstPad * pad, GstBuffer * buf)
|
||||
{
|
||||
GstFlowReturn ret = GST_FLOW_OK;
|
||||
GstDvdSubDec *dec;
|
||||
guint8 *data;
|
||||
glong size = 0;
|
||||
|
||||
dec = GST_DVD_SUB_DEC (GST_PAD_PARENT (pad));
|
||||
|
||||
GST_DEBUG_OBJECT (dec, "Have buffer of size %d, ts %"
|
||||
GST_TIME_FORMAT ", dur %" G_GINT64_FORMAT, GST_BUFFER_SIZE (buf),
|
||||
GST_TIME_ARGS (GST_BUFFER_TIMESTAMP (buf)), GST_BUFFER_DURATION (buf));
|
||||
|
||||
if (GST_CLOCK_TIME_IS_VALID (GST_BUFFER_TIMESTAMP (buf))) {
|
||||
if (!GST_CLOCK_TIME_IS_VALID (dec->next_ts)) {
|
||||
dec->next_ts = GST_BUFFER_TIMESTAMP (buf);
|
||||
}
|
||||
|
||||
/* Move time forward to the start of the new buffer */
|
||||
ret = gst_dvd_sub_dec_advance_time (dec, GST_BUFFER_TIMESTAMP (buf));
|
||||
}
|
||||
|
||||
if (dec->have_title) {
|
||||
gst_buffer_unref (dec->partialbuf);
|
||||
dec->partialbuf = NULL;
|
||||
dec->have_title = FALSE;
|
||||
}
|
||||
|
||||
GST_DEBUG_OBJECT (dec, "Got subtitle buffer, pts %" GST_TIME_FORMAT,
|
||||
GST_TIME_ARGS (GST_BUFFER_TIMESTAMP (buf)));
|
||||
|
||||
/* deal with partial frame from previous buffer */
|
||||
if (dec->partialbuf) {
|
||||
GstBuffer *merge;
|
||||
|
||||
merge = gst_buffer_join (dec->partialbuf, buf);
|
||||
dec->partialbuf = merge;
|
||||
} else {
|
||||
dec->partialbuf = buf;
|
||||
}
|
||||
|
||||
data = GST_BUFFER_DATA (dec->partialbuf);
|
||||
size = GST_BUFFER_SIZE (dec->partialbuf);
|
||||
|
||||
if (size > 4) {
|
||||
dec->packet_size = GST_READ_UINT16_BE (data);
|
||||
|
||||
if (dec->packet_size == size) {
|
||||
GST_LOG_OBJECT (dec, "Subtitle packet size %d, current size %ld",
|
||||
dec->packet_size, size);
|
||||
|
||||
dec->data_size = GST_READ_UINT16_BE (data + 2);
|
||||
|
||||
/* Reset parameters for a new subtitle buffer */
|
||||
dec->parse_pos = data;
|
||||
dec->forced_display = FALSE;
|
||||
dec->visible = FALSE;
|
||||
|
||||
dec->have_title = TRUE;
|
||||
dec->next_event_ts = GST_BUFFER_TIMESTAMP (dec->partialbuf);
|
||||
|
||||
if (!GST_CLOCK_TIME_IS_VALID (dec->next_event_ts))
|
||||
dec->next_event_ts = dec->next_ts;
|
||||
|
||||
dec->next_event_ts += gst_dvd_sub_dec_get_event_delay (dec);
|
||||
}
|
||||
}
|
||||
|
||||
return ret;
|
||||
}
|
||||
|
||||
static gboolean
|
||||
gst_dvd_sub_dec_sink_event (GstPad * pad, GstEvent * event)
|
||||
{
|
||||
GstDvdSubDec *dec = GST_DVD_SUB_DEC (gst_pad_get_parent (pad));
|
||||
gboolean ret = FALSE;
|
||||
|
||||
GST_LOG_OBJECT (dec, "Handling %s event", GST_EVENT_TYPE_NAME (event));
|
||||
|
||||
switch (GST_EVENT_TYPE (event)) {
|
||||
case GST_EVENT_CUSTOM_DOWNSTREAM:{
|
||||
GstClockTime ts = GST_EVENT_TIMESTAMP (event);
|
||||
|
||||
GST_LOG_OBJECT (dec, "DVD event on subtitle pad with timestamp %llu",
|
||||
GST_EVENT_TIMESTAMP (event));
|
||||
|
||||
if (GST_CLOCK_TIME_IS_VALID (ts))
|
||||
gst_dvd_sub_dec_advance_time (dec, ts);
|
||||
|
||||
if (!gst_dvd_sub_dec_handle_dvd_event (dec, event)) {
|
||||
ret = gst_pad_event_default (pad, event);
|
||||
break;
|
||||
}
|
||||
// gst_dvd_sub_dec_advance_time (dec, dec->next_ts + GST_SECOND / 30.0);
|
||||
|
||||
gst_event_unref (event);
|
||||
ret = TRUE;
|
||||
break;
|
||||
}
|
||||
case GST_EVENT_NEWSEGMENT:{
|
||||
gboolean update;
|
||||
GstFormat format;
|
||||
gint64 start, stop, pos;
|
||||
|
||||
gst_event_parse_new_segment (event, &update, NULL, &format, &start,
|
||||
&stop, &pos);
|
||||
|
||||
if (update) {
|
||||
/* update ... advance time */
|
||||
if (GST_CLOCK_TIME_IS_VALID (pos)) {
|
||||
GST_DEBUG_OBJECT (dec, "Got segment update, advancing time from %"
|
||||
GST_TIME_FORMAT " to %" GST_TIME_FORMAT,
|
||||
GST_TIME_ARGS (dec->next_ts), GST_TIME_ARGS (pos));
|
||||
|
||||
gst_dvd_sub_dec_advance_time (dec, pos);
|
||||
} else {
|
||||
GST_WARNING_OBJECT (dec, "Got segment update with invalid position");
|
||||
}
|
||||
gst_event_unref (event);
|
||||
ret = TRUE;
|
||||
} else {
|
||||
/* not just an update ... */
|
||||
|
||||
/* Turn off forced highlight display */
|
||||
// dec->forced_display = 0;
|
||||
// dec->current_button = 0;
|
||||
if (dec->partialbuf) {
|
||||
gst_buffer_unref (dec->partialbuf);
|
||||
dec->partialbuf = NULL;
|
||||
dec->have_title = FALSE;
|
||||
}
|
||||
|
||||
if (GST_CLOCK_TIME_IS_VALID (pos))
|
||||
dec->next_ts = pos;
|
||||
else
|
||||
dec->next_ts = GST_CLOCK_TIME_NONE;
|
||||
|
||||
GST_DEBUG_OBJECT (dec, "Got newsegment, new time = %"
|
||||
GST_TIME_FORMAT, GST_TIME_ARGS (dec->next_ts));
|
||||
|
||||
ret = gst_pad_event_default (pad, event);
|
||||
}
|
||||
break;
|
||||
}
|
||||
case GST_EVENT_FLUSH_START:{
|
||||
/* Turn off forced highlight display */
|
||||
dec->forced_display = 0;
|
||||
dec->current_button = 0;
|
||||
|
||||
if (dec->partialbuf) {
|
||||
gst_buffer_unref (dec->partialbuf);
|
||||
dec->partialbuf = NULL;
|
||||
dec->have_title = FALSE;
|
||||
}
|
||||
|
||||
ret = gst_pad_event_default (pad, event);
|
||||
break;
|
||||
}
|
||||
default:{
|
||||
ret = gst_pad_event_default (pad, event);
|
||||
break;
|
||||
}
|
||||
}
|
||||
gst_object_unref (dec);
|
||||
return ret;
|
||||
}
|
||||
|
||||
static gboolean
|
||||
gst_dvd_sub_dec_handle_dvd_event (GstDvdSubDec * dec, GstEvent * event)
|
||||
{
|
||||
GstStructure *structure;
|
||||
|
||||
structure = (GstStructure *) gst_event_get_structure (event);
|
||||
|
||||
if (gst_structure_has_name (structure, "dvd-spu-highlight")) {
|
||||
gint button;
|
||||
gint palette, sx, sy, ex, ey;
|
||||
gint i;
|
||||
|
||||
/* Details for the highlight region to display */
|
||||
if (!gst_structure_get_int (structure, "button", &button) ||
|
||||
!gst_structure_get_int (structure, "palette", &palette) ||
|
||||
!gst_structure_get_int (structure, "sx", &sx) ||
|
||||
!gst_structure_get_int (structure, "sy", &sy) ||
|
||||
!gst_structure_get_int (structure, "ex", &ex) ||
|
||||
!gst_structure_get_int (structure, "ey", &ey)) {
|
||||
GST_ERROR_OBJECT (dec, "Invalid dvd-spu-highlight event received");
|
||||
return TRUE;
|
||||
}
|
||||
dec->current_button = button;
|
||||
dec->hl_left = sx;
|
||||
dec->hl_top = sy;
|
||||
dec->hl_right = ex;
|
||||
dec->hl_bottom = ey;
|
||||
for (i = 0; i < 4; i++) {
|
||||
dec->menu_alpha[i] = ((guint32) (palette) >> (i * 4)) & 0x0f;
|
||||
dec->menu_index[i] = ((guint32) (palette) >> (16 + (i * 4))) & 0x0f;
|
||||
}
|
||||
|
||||
GST_DEBUG_OBJECT (dec, "New button activated highlight=(%d,%d) to (%d,%d) "
|
||||
"palette 0x%x", sx, sy, ex, ey, palette);
|
||||
gst_setup_palette (dec);
|
||||
|
||||
dec->buf_dirty = TRUE;
|
||||
} else if (gst_structure_has_name (structure, "dvd-spu-clut-change")) {
|
||||
/* Take a copy of the colour table */
|
||||
gchar name[16];
|
||||
int i;
|
||||
gint value;
|
||||
|
||||
GST_LOG_OBJECT (dec, "New colour table recieved");
|
||||
for (i = 0; i < 16; i++) {
|
||||
g_snprintf (name, sizeof (name), "clut%02d", i);
|
||||
if (!gst_structure_get_int (structure, name, &value)) {
|
||||
GST_ERROR_OBJECT (dec, "dvd-spu-clut-change event did not "
|
||||
"contain %s field", name);
|
||||
break;
|
||||
}
|
||||
dec->current_clut[i] = (guint32) (value);
|
||||
}
|
||||
|
||||
gst_setup_palette (dec);
|
||||
|
||||
dec->buf_dirty = TRUE;
|
||||
} else if (gst_structure_has_name (structure, "dvd-spu-stream-change")
|
||||
|| gst_structure_has_name (structure, "dvd-spu-reset-highlight")) {
|
||||
/* Turn off forced highlight display */
|
||||
dec->current_button = 0;
|
||||
|
||||
GST_LOG_OBJECT (dec, "Clearing button state");
|
||||
dec->buf_dirty = TRUE;
|
||||
} else if (gst_structure_has_name (structure, "dvd-spu-still-frame")) {
|
||||
/* Handle a still frame */
|
||||
GST_LOG_OBJECT (dec, "Received still frame notification");
|
||||
} else {
|
||||
/* Ignore all other unknown events */
|
||||
GST_LOG_OBJECT (dec, "Ignoring other custom event %s",
|
||||
gst_structure_get_name (structure));
|
||||
return FALSE;
|
||||
}
|
||||
return TRUE;
|
||||
}
|
||||
|
||||
static gboolean
|
||||
plugin_init (GstPlugin * plugin)
|
||||
{
|
||||
if (!gst_element_register (plugin, "dvdsubdec", GST_RANK_PRIMARY,
|
||||
GST_TYPE_DVD_SUB_DEC)) {
|
||||
return FALSE;
|
||||
}
|
||||
|
||||
GST_DEBUG_CATEGORY_INIT (gst_dvd_sub_dec_debug, "dvdsubdec", 0,
|
||||
"DVD subtitle decoder element");
|
||||
|
||||
return TRUE;
|
||||
}
|
||||
|
||||
GST_PLUGIN_DEFINE (GST_VERSION_MAJOR,
|
||||
GST_VERSION_MINOR,
|
||||
"dvdsub",
|
||||
"Decode DVD subtitles to AYUV video frames", plugin_init,
|
||||
VERSION, "LGPL", GST_PACKAGE, GST_ORIGIN)
|
95
gst/dvdsub/gstdvdsubdec.h
Normal file
95
gst/dvdsub/gstdvdsubdec.h
Normal file
|
@ -0,0 +1,95 @@
|
|||
/* GStreamer
|
||||
* Copyright (C) <2005> Jan Schmidt <jan@fluendo.com>
|
||||
* Copyright (C) <2002> Wim Taymans <wim@fluendo.com>
|
||||
*
|
||||
* 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 <gst/gst.h>
|
||||
|
||||
#define GST_TYPE_DVD_SUB_DEC (gst_dvd_sub_dec_get_type())
|
||||
#define GST_DVD_SUB_DEC(obj) (G_TYPE_CHECK_INSTANCE_CAST((obj),GST_TYPE_DVD_SUB_DEC,GstDvdSubDec))
|
||||
#define GST_DVD_SUB_DEC_CLASS(klass) (G_TYPE_CHECK_CLASS_CAST((klass),GST_TYPE_DVD_SUB_DEC,GstDvdSubDecClass))
|
||||
#define GST_IS_DVD_SUB_DEC(obj) (G_TYPE_CHECK_INSTANCE_TYPE((obj),GST_TYPE_DVD_SUB_DEC))
|
||||
#define GST_IS_DVD_SUB_DEC_CLASS(klass) (G_TYPE_CHECK_CLASS_TYPE((klass),GST_TYPE_DVD_SUB_DEC))
|
||||
|
||||
typedef struct _GstDvdSubDec GstDvdSubDec;
|
||||
typedef struct _GstDvdSubDecClass GstDvdSubDecClass;
|
||||
|
||||
/* Hold premultimplied colour values */
|
||||
typedef struct YUVA_val
|
||||
{
|
||||
guchar Y;
|
||||
guchar U;
|
||||
guchar V;
|
||||
guchar A;
|
||||
} YUVA_val;
|
||||
|
||||
struct _GstDvdSubDec
|
||||
{
|
||||
GstElement element;
|
||||
|
||||
GstPad *sinkpad;
|
||||
GstPad *srcpad;
|
||||
|
||||
gint in_width, in_height;
|
||||
|
||||
/* Collect together subtitle buffers until we have a full control sequence */
|
||||
GstBuffer *partialbuf;
|
||||
gboolean have_title;
|
||||
|
||||
guchar subtitle_index[4];
|
||||
guchar menu_index[4];
|
||||
guchar subtitle_alpha[4];
|
||||
guchar menu_alpha[4];
|
||||
|
||||
guint32 current_clut[16];
|
||||
YUVA_val palette_cache[4];
|
||||
YUVA_val hl_palette_cache[4];
|
||||
|
||||
GstClockTime next_ts;
|
||||
|
||||
/*
|
||||
* State info for the current subpicture
|
||||
* buffer
|
||||
*/
|
||||
guchar *parse_pos;
|
||||
|
||||
guint16 packet_size;
|
||||
guint16 data_size;
|
||||
|
||||
gint offset[2];
|
||||
|
||||
gboolean forced_display;
|
||||
gboolean visible;
|
||||
|
||||
gint left, top, right, bottom;
|
||||
gint hl_left, hl_top, hl_right, hl_bottom;
|
||||
|
||||
gint current_button;
|
||||
|
||||
GstClockTime next_event_ts;
|
||||
|
||||
GstBuffer *out_buffer;
|
||||
gboolean buf_dirty;
|
||||
};
|
||||
|
||||
struct _GstDvdSubDecClass
|
||||
{
|
||||
GstElementClass parent_class;
|
||||
};
|
||||
|
||||
GType gst_dvd_sub_dec_get_type (void);
|
Loading…
Reference in a new issue