mirror of
https://gitlab.freedesktop.org/gstreamer/gstreamer.git
synced 2025-01-12 18:35:35 +00:00
gst/matroska/: Ported matroska demuxer to 0.9.
Original commit message from CVS: * gst/matroska/Makefile.am: * gst/matroska/ebml-read.c: * gst/matroska/ebml-read.h: * gst/matroska/matroska-demux.c: * gst/matroska/matroska-demux.h: * gst/matroska/matroska.c: (plugin_init): Ported matroska demuxer to 0.9.
This commit is contained in:
parent
2b6838d77c
commit
66413b5f00
7 changed files with 1719 additions and 1270 deletions
|
@ -1,3 +1,12 @@
|
||||||
|
2005-10-18 Tim-Philipp Müller <tim at centricular dot net>
|
||||||
|
* gst/matroska/Makefile.am:
|
||||||
|
* gst/matroska/ebml-read.c:
|
||||||
|
* gst/matroska/ebml-read.h:
|
||||||
|
* gst/matroska/matroska-demux.c:
|
||||||
|
* gst/matroska/matroska-demux.h:
|
||||||
|
* gst/matroska/matroska.c: (plugin_init):
|
||||||
|
Ported matroska demuxer to 0.9.
|
||||||
|
|
||||||
2005-10-18 Michal Benes <michal dot benes at xeris dot cz>
|
2005-10-18 Michal Benes <michal dot benes at xeris dot cz>
|
||||||
|
|
||||||
Reviewed by: Tim-Philipp Müller <tim at centricular dot net>
|
Reviewed by: Tim-Philipp Müller <tim at centricular dot net>
|
||||||
|
|
|
@ -1,13 +1,12 @@
|
||||||
plugin_LTLIBRARIES = libgstmatroska.la
|
plugin_LTLIBRARIES = libgstmatroska.la
|
||||||
|
|
||||||
libgstmatroska_la_SOURCES = \
|
libgstmatroska_la_SOURCES = \
|
||||||
|
ebml-read.c \
|
||||||
ebml-write.c \
|
ebml-write.c \
|
||||||
matroska.c \
|
matroska.c \
|
||||||
|
matroska-demux.c \
|
||||||
matroska-mux.c
|
matroska-mux.c
|
||||||
|
|
||||||
# ebml-read.c
|
|
||||||
# matroska-demux.c
|
|
||||||
|
|
||||||
noinst_HEADERS = \
|
noinst_HEADERS = \
|
||||||
ebml-ids.h \
|
ebml-ids.h \
|
||||||
ebml-read.h \
|
ebml-read.h \
|
||||||
|
|
|
@ -31,23 +31,24 @@
|
||||||
GST_DEBUG_CATEGORY_STATIC (ebmlread_debug);
|
GST_DEBUG_CATEGORY_STATIC (ebmlread_debug);
|
||||||
#define GST_CAT_DEFAULT ebmlread_debug
|
#define GST_CAT_DEFAULT ebmlread_debug
|
||||||
|
|
||||||
enum
|
|
||||||
{
|
|
||||||
/* FILL ME */
|
|
||||||
LAST_SIGNAL
|
|
||||||
};
|
|
||||||
|
|
||||||
static void gst_ebml_read_class_init (GstEbmlReadClass * klass);
|
static void gst_ebml_read_class_init (GstEbmlReadClass * klass);
|
||||||
static void gst_ebml_read_init (GstEbmlRead * ebml);
|
static void gst_ebml_read_init (GstEbmlRead * ebml);
|
||||||
static GstStateChangeReturn gst_ebml_read_change_state (GstElement * element,
|
static GstStateChangeReturn gst_ebml_read_change_state (GstElement * element,
|
||||||
GstStateChange transition);
|
GstStateChange transition);
|
||||||
|
|
||||||
static GstElementClass *parent_class = NULL;
|
/* convenience functions */
|
||||||
|
static gboolean gst_ebml_read_peek_bytes (GstEbmlRead * ebml, guint size,
|
||||||
|
GstBuffer ** p_buf);
|
||||||
|
static gboolean gst_ebml_read_pull_bytes (GstEbmlRead * ebml, guint size,
|
||||||
|
GstBuffer ** p_buf);
|
||||||
|
|
||||||
|
|
||||||
|
static GstElementClass *parent_class; /* NULL */
|
||||||
|
|
||||||
GType
|
GType
|
||||||
gst_ebml_read_get_type (void)
|
gst_ebml_read_get_type (void)
|
||||||
{
|
{
|
||||||
static GType gst_ebml_read_type = 0;
|
static GType gst_ebml_read_type; /* 0 */
|
||||||
|
|
||||||
if (!gst_ebml_read_type) {
|
if (!gst_ebml_read_type) {
|
||||||
static const GTypeInfo gst_ebml_read_info = {
|
static const GTypeInfo gst_ebml_read_info = {
|
||||||
|
@ -75,111 +76,58 @@ gst_ebml_read_class_init (GstEbmlReadClass * klass)
|
||||||
{
|
{
|
||||||
GstElementClass *gstelement_class = (GstElementClass *) klass;
|
GstElementClass *gstelement_class = (GstElementClass *) klass;
|
||||||
|
|
||||||
parent_class = g_type_class_ref (GST_TYPE_ELEMENT);
|
parent_class = g_type_class_peek_parent (klass);
|
||||||
|
|
||||||
GST_DEBUG_CATEGORY_INIT (ebmlread_debug, "ebmlread",
|
GST_DEBUG_CATEGORY_INIT (ebmlread_debug, "ebmlread",
|
||||||
0, "EBML stream helper class");
|
0, "EBML stream helper class");
|
||||||
|
|
||||||
gstelement_class->change_state = gst_ebml_read_change_state;
|
gstelement_class->change_state =
|
||||||
|
GST_DEBUG_FUNCPTR (gst_ebml_read_change_state);
|
||||||
}
|
}
|
||||||
|
|
||||||
static void
|
static void
|
||||||
gst_ebml_read_init (GstEbmlRead * ebml)
|
gst_ebml_read_init (GstEbmlRead * ebml)
|
||||||
{
|
{
|
||||||
ebml->sinkpad = NULL;
|
ebml->sinkpad = NULL;
|
||||||
ebml->bs = NULL;
|
|
||||||
ebml->level = NULL;
|
ebml->level = NULL;
|
||||||
ebml->id_cache = 0;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
static GstStateChangeReturn
|
static GstStateChangeReturn
|
||||||
gst_ebml_read_change_state (GstElement * element, GstStateChange transition)
|
gst_ebml_read_change_state (GstElement * element, GstStateChange transition)
|
||||||
{
|
{
|
||||||
|
GstStateChangeReturn ret;
|
||||||
GstEbmlRead *ebml = GST_EBML_READ (element);
|
GstEbmlRead *ebml = GST_EBML_READ (element);
|
||||||
|
|
||||||
switch (transition) {
|
switch (transition) {
|
||||||
case GST_STATE_CHANGE_READY_TO_PAUSED:
|
case GST_STATE_CHANGE_READY_TO_PAUSED:
|
||||||
if (!ebml->sinkpad)
|
if (!ebml->sinkpad) {
|
||||||
return GST_STATE_CHANGE_FAILURE;
|
g_return_val_if_reached (GST_STATE_CHANGE_FAILURE);
|
||||||
ebml->bs = gst_bytestream_new (ebml->sinkpad);
|
|
||||||
break;
|
|
||||||
case GST_STATE_CHANGE_PAUSED_TO_READY:
|
|
||||||
gst_bytestream_destroy (ebml->bs);
|
|
||||||
while (ebml->level) {
|
|
||||||
GstEbmlLevel *level = ebml->level->data;
|
|
||||||
|
|
||||||
ebml->level = g_list_remove (ebml->level, level);
|
|
||||||
g_free (level);
|
|
||||||
}
|
}
|
||||||
break;
|
break;
|
||||||
default:
|
default:
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (GST_ELEMENT_CLASS (parent_class)->change_state)
|
ret = GST_ELEMENT_CLASS (parent_class)->change_state (element, transition);
|
||||||
return GST_ELEMENT_CLASS (parent_class)->change_state (element, transition);
|
|
||||||
|
|
||||||
return GST_STATE_CHANGE_SUCCESS;
|
|
||||||
}
|
|
||||||
|
|
||||||
/*
|
|
||||||
* Event handler. Basic:
|
|
||||||
* - EOS: end-of-file, stop processing, forward EOS.
|
|
||||||
* - Interrupt: stop processing.
|
|
||||||
* - Discont: shouldn't be handled here but in the seek handler. Error.
|
|
||||||
* - Flush: ignore, since we check for flush flags manually. Don't forward.
|
|
||||||
* - Others: warn, ignore.
|
|
||||||
* Return value indicates whether to continue processing.
|
|
||||||
*/
|
|
||||||
|
|
||||||
static gboolean
|
|
||||||
gst_ebml_read_use_event (GstEbmlRead * ebml, GstEvent * event)
|
|
||||||
{
|
|
||||||
if (!event) {
|
|
||||||
GST_ELEMENT_ERROR (ebml, RESOURCE, READ, (NULL), (NULL));
|
|
||||||
return FALSE;
|
|
||||||
}
|
|
||||||
|
|
||||||
GST_LOG ("Received event of type %d", GST_EVENT_TYPE (event));
|
|
||||||
|
|
||||||
switch (GST_EVENT_TYPE (event)) {
|
|
||||||
case GST_EVENT_EOS:
|
|
||||||
gst_pad_event_default (ebml->sinkpad, event);
|
|
||||||
return FALSE;
|
|
||||||
|
|
||||||
case GST_EVENT_INTERRUPT:
|
|
||||||
gst_event_unref (event);
|
|
||||||
return FALSE;
|
|
||||||
|
|
||||||
case GST_EVENT_DISCONTINUOUS:
|
|
||||||
GST_WARNING_OBJECT (ebml, "Unexpected discont - might lose sync");
|
|
||||||
gst_event_unref (event);
|
|
||||||
return TRUE;
|
|
||||||
|
|
||||||
case GST_EVENT_FLUSH:
|
|
||||||
gst_event_unref (event);
|
|
||||||
return TRUE;
|
|
||||||
|
|
||||||
|
switch (transition) {
|
||||||
|
case GST_STATE_CHANGE_PAUSED_TO_READY:
|
||||||
|
{
|
||||||
|
g_list_foreach (ebml->level, (GFunc) g_free, NULL);
|
||||||
|
g_list_free (ebml->level);
|
||||||
|
ebml->level = NULL;
|
||||||
|
if (ebml->cached_buffer) {
|
||||||
|
gst_buffer_unref (ebml->cached_buffer);
|
||||||
|
ebml->cached_buffer = NULL;
|
||||||
|
}
|
||||||
|
ebml->offset = 0;
|
||||||
|
break;
|
||||||
|
}
|
||||||
default:
|
default:
|
||||||
GST_WARNING ("don't know how to handle event %d", GST_EVENT_TYPE (event));
|
break;
|
||||||
gst_pad_event_default (ebml->sinkpad, event);
|
|
||||||
return FALSE;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/* happy */
|
return ret;
|
||||||
g_assert_not_reached ();
|
|
||||||
return FALSE;
|
|
||||||
}
|
|
||||||
|
|
||||||
static gboolean
|
|
||||||
gst_ebml_read_handle_event (GstEbmlRead * ebml)
|
|
||||||
{
|
|
||||||
GstEvent *event = NULL;
|
|
||||||
guint32 remaining;
|
|
||||||
|
|
||||||
gst_bytestream_get_status (ebml->bs, &remaining, &event);
|
|
||||||
|
|
||||||
return gst_ebml_read_use_event (ebml, event);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/*
|
/*
|
||||||
|
@ -193,7 +141,7 @@ static guint
|
||||||
gst_ebml_read_element_level_up (GstEbmlRead * ebml)
|
gst_ebml_read_element_level_up (GstEbmlRead * ebml)
|
||||||
{
|
{
|
||||||
guint num = 0;
|
guint num = 0;
|
||||||
guint64 pos = gst_bytestream_tell (ebml->bs);
|
guint64 pos = ebml->offset;
|
||||||
|
|
||||||
while (ebml->level != NULL) {
|
while (ebml->level != NULL) {
|
||||||
GList *last = g_list_last (ebml->level);
|
GList *last = g_list_last (ebml->level);
|
||||||
|
@ -203,67 +151,129 @@ gst_ebml_read_element_level_up (GstEbmlRead * ebml)
|
||||||
ebml->level = g_list_remove (ebml->level, level);
|
ebml->level = g_list_remove (ebml->level, level);
|
||||||
g_free (level);
|
g_free (level);
|
||||||
num++;
|
num++;
|
||||||
} else
|
} else {
|
||||||
break;
|
break;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
return num;
|
return num;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Calls pull_range for (offset,size) without advancing our offset
|
||||||
|
*/
|
||||||
|
static gboolean
|
||||||
|
gst_ebml_read_peek_bytes (GstEbmlRead * ebml, guint size, GstBuffer ** p_buf)
|
||||||
|
{
|
||||||
|
GstFlowReturn ret;
|
||||||
|
|
||||||
|
/* Caching here actually makes much less difference than one would expect.
|
||||||
|
* We do it mainly to avoid pulling buffers of 1 byte all the time */
|
||||||
|
if (ebml->cached_buffer) {
|
||||||
|
guint64 cache_offset = GST_BUFFER_OFFSET (ebml->cached_buffer);
|
||||||
|
guint cache_size = GST_BUFFER_SIZE (ebml->cached_buffer);
|
||||||
|
|
||||||
|
if (cache_offset <= ebml->offset &&
|
||||||
|
(ebml->offset + size) < (cache_offset + cache_size)) {
|
||||||
|
*p_buf = gst_buffer_create_sub (ebml->cached_buffer,
|
||||||
|
ebml->offset - cache_offset, size);
|
||||||
|
return TRUE;
|
||||||
|
}
|
||||||
|
gst_buffer_unref (ebml->cached_buffer);
|
||||||
|
ebml->cached_buffer = NULL;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (gst_pad_pull_range (ebml->sinkpad, ebml->offset, MAX (size, 64 * 1024),
|
||||||
|
&ebml->cached_buffer) == GST_FLOW_OK &&
|
||||||
|
GST_BUFFER_SIZE (ebml->cached_buffer) >= size) {
|
||||||
|
*p_buf = gst_buffer_create_sub (ebml->cached_buffer, 0, size);
|
||||||
|
return TRUE;
|
||||||
|
}
|
||||||
|
|
||||||
|
ret = gst_pad_pull_range (ebml->sinkpad, ebml->offset, size, p_buf);
|
||||||
|
if (ret != GST_FLOW_OK) {
|
||||||
|
GST_DEBUG ("pull_range returned %d", ret);
|
||||||
|
return FALSE;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (GST_BUFFER_SIZE (*p_buf) < size) {
|
||||||
|
GST_WARNING_OBJECT (ebml, "Dropping short buffer at offset %"
|
||||||
|
G_GUINT64_FORMAT ": wanted %u bytes, got %u bytes", ebml->offset,
|
||||||
|
size, GST_BUFFER_SIZE (*p_buf));
|
||||||
|
gst_buffer_unref (*p_buf);
|
||||||
|
*p_buf = NULL;
|
||||||
|
return FALSE;
|
||||||
|
}
|
||||||
|
|
||||||
|
return TRUE;
|
||||||
|
}
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Calls pull_range for (offset,size) and advances our offset by size
|
||||||
|
*/
|
||||||
|
static gboolean
|
||||||
|
gst_ebml_read_pull_bytes (GstEbmlRead * ebml, guint size, GstBuffer ** p_buf)
|
||||||
|
{
|
||||||
|
if (!gst_ebml_read_peek_bytes (ebml, size, p_buf))
|
||||||
|
return FALSE;
|
||||||
|
|
||||||
|
ebml->offset += size;
|
||||||
|
return TRUE;
|
||||||
|
}
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* Read: the element content data ID.
|
* Read: the element content data ID.
|
||||||
* Return: the number of bytes read or -1 on error.
|
* Return: FALSE on error.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
static gint
|
static gboolean
|
||||||
gst_ebml_read_element_id (GstEbmlRead * ebml, guint32 * id, guint * level_up)
|
gst_ebml_read_element_id (GstEbmlRead * ebml, guint32 * id, guint * level_up)
|
||||||
{
|
{
|
||||||
guint8 *data;
|
GstBuffer *buf;
|
||||||
gint len_mask = 0x80, read = 1, n = 1;
|
gint len_mask = 0x80, read = 1, n = 1;
|
||||||
guint32 total;
|
guint32 total;
|
||||||
|
guint8 b;
|
||||||
|
|
||||||
if (ebml->id_cache) {
|
if (!gst_ebml_read_peek_bytes (ebml, 1, &buf))
|
||||||
*id = ebml->id_cache;
|
return FALSE;
|
||||||
if (level_up)
|
|
||||||
*level_up = 0;
|
|
||||||
|
|
||||||
return 0;
|
b = GST_READ_UINT8 (GST_BUFFER_DATA (buf));
|
||||||
}
|
gst_buffer_unref (buf);
|
||||||
|
|
||||||
|
total = (guint32) b;
|
||||||
|
|
||||||
while (gst_bytestream_peek_bytes (ebml->bs, &data, 1) != 1) {
|
|
||||||
if (!gst_ebml_read_handle_event (ebml))
|
|
||||||
return -1;
|
|
||||||
}
|
|
||||||
total = data[0];
|
|
||||||
while (read <= 4 && !(total & len_mask)) {
|
while (read <= 4 && !(total & len_mask)) {
|
||||||
read++;
|
read++;
|
||||||
len_mask >>= 1;
|
len_mask >>= 1;
|
||||||
}
|
}
|
||||||
if (read > 4) {
|
if (read > 4) {
|
||||||
guint64 pos = gst_bytestream_tell (ebml->bs);
|
guint64 pos = ebml->offset;
|
||||||
|
|
||||||
GST_ELEMENT_ERROR (ebml, STREAM, DEMUX, (NULL),
|
GST_ELEMENT_ERROR (ebml, STREAM, DEMUX, (NULL),
|
||||||
("Invalid EBML ID size tag (0x%x) at position %llu (0x%llx)",
|
("Invalid EBML ID size tag (0x%x) at position %llu (0x%llx)",
|
||||||
data[0], pos, pos));
|
(guint) b, pos, pos));
|
||||||
return -1;
|
return FALSE;
|
||||||
}
|
}
|
||||||
|
|
||||||
while (gst_bytestream_peek_bytes (ebml->bs, &data, read) != read) {
|
if (!gst_ebml_read_peek_bytes (ebml, read, &buf))
|
||||||
if (!gst_ebml_read_handle_event (ebml))
|
return FALSE;
|
||||||
return -1;
|
|
||||||
}
|
|
||||||
while (n < read)
|
|
||||||
total = (total << 8) | data[n++];
|
|
||||||
|
|
||||||
*id = ebml->id_cache = total;
|
while (n < read) {
|
||||||
|
b = GST_READ_UINT8 (GST_BUFFER_DATA (buf) + n);
|
||||||
|
total = (total << 8) | b;
|
||||||
|
++n;
|
||||||
|
}
|
||||||
|
|
||||||
|
*id = total;
|
||||||
|
|
||||||
/* level */
|
/* level */
|
||||||
if (level_up)
|
if (level_up)
|
||||||
*level_up = gst_ebml_read_element_level_up (ebml);
|
*level_up = gst_ebml_read_element_level_up (ebml);
|
||||||
|
|
||||||
gst_bytestream_flush_fast (ebml->bs, read);
|
gst_buffer_unref (buf);
|
||||||
|
|
||||||
return 0;
|
ebml->offset += read;
|
||||||
|
return TRUE;
|
||||||
}
|
}
|
||||||
|
|
||||||
/*
|
/*
|
||||||
|
@ -274,141 +284,112 @@ gst_ebml_read_element_id (GstEbmlRead * ebml, guint32 * id, guint * level_up)
|
||||||
static gint
|
static gint
|
||||||
gst_ebml_read_element_length (GstEbmlRead * ebml, guint64 * length)
|
gst_ebml_read_element_length (GstEbmlRead * ebml, guint64 * length)
|
||||||
{
|
{
|
||||||
guint8 *data;
|
GstBuffer *buf;
|
||||||
gint len_mask = 0x80, read = 1, n = 1, num_ffs = 0;
|
gint len_mask = 0x80, read = 1, n = 1, num_ffs = 0;
|
||||||
guint64 total;
|
guint64 total;
|
||||||
|
guint8 b;
|
||||||
|
|
||||||
|
if (!gst_ebml_read_peek_bytes (ebml, 1, &buf))
|
||||||
|
return -1;
|
||||||
|
|
||||||
|
b = GST_READ_UINT8 (GST_BUFFER_DATA (buf));
|
||||||
|
gst_buffer_unref (buf);
|
||||||
|
|
||||||
|
total = (guint64) b;
|
||||||
|
|
||||||
while (gst_bytestream_peek_bytes (ebml->bs, &data, 1) != 1) {
|
|
||||||
if (!gst_ebml_read_handle_event (ebml))
|
|
||||||
return -1;
|
|
||||||
}
|
|
||||||
total = data[0];
|
|
||||||
while (read <= 8 && !(total & len_mask)) {
|
while (read <= 8 && !(total & len_mask)) {
|
||||||
read++;
|
read++;
|
||||||
len_mask >>= 1;
|
len_mask >>= 1;
|
||||||
}
|
}
|
||||||
if (read > 8) {
|
if (read > 8) {
|
||||||
guint64 pos = gst_bytestream_tell (ebml->bs);
|
guint64 pos = ebml->offset;
|
||||||
|
|
||||||
GST_ELEMENT_ERROR (ebml, STREAM, DEMUX, (NULL),
|
GST_ELEMENT_ERROR (ebml, STREAM, DEMUX, (NULL),
|
||||||
("Invalid EBML length size tag (0x%x) at position %llu (0x%llx)",
|
("Invalid EBML length size tag (0x%x) at position %llu (0x%llx)",
|
||||||
data[0], pos, pos));
|
(guint) b, pos, pos));
|
||||||
return -1;
|
return -1;
|
||||||
}
|
}
|
||||||
|
|
||||||
if ((total &= (len_mask - 1)) == len_mask - 1)
|
if ((total &= (len_mask - 1)) == len_mask - 1)
|
||||||
num_ffs++;
|
num_ffs++;
|
||||||
while (gst_bytestream_peek_bytes (ebml->bs, &data, read) != read) {
|
|
||||||
if (!gst_ebml_read_handle_event (ebml))
|
if (!gst_ebml_read_peek_bytes (ebml, read, &buf))
|
||||||
return -1;
|
return -1;
|
||||||
}
|
|
||||||
while (n < read) {
|
while (n < read) {
|
||||||
if (data[n] == 0xff)
|
guint8 b = GST_READ_UINT8 (GST_BUFFER_DATA (buf) + n);
|
||||||
|
|
||||||
|
if (b == 0xff)
|
||||||
num_ffs++;
|
num_ffs++;
|
||||||
total = (total << 8) | data[n];
|
total = (total << 8) | b;
|
||||||
n++;
|
++n;
|
||||||
}
|
}
|
||||||
|
gst_buffer_unref (buf);
|
||||||
|
|
||||||
if (read == num_ffs)
|
if (read == num_ffs)
|
||||||
*length = G_MAXUINT64;
|
*length = G_MAXUINT64;
|
||||||
else
|
else
|
||||||
*length = total;
|
*length = total;
|
||||||
|
|
||||||
|
ebml->offset += read;
|
||||||
|
|
||||||
return read;
|
return read;
|
||||||
}
|
}
|
||||||
|
|
||||||
/*
|
|
||||||
* Read: the actual data.
|
|
||||||
* Return: the data, as a GstBuffer.
|
|
||||||
*/
|
|
||||||
|
|
||||||
static GstBuffer *
|
|
||||||
gst_ebml_read_element_data (GstEbmlRead * ebml, guint64 length)
|
|
||||||
{
|
|
||||||
GstBuffer *buf = NULL;
|
|
||||||
|
|
||||||
if (gst_bytestream_peek (ebml->bs, &buf, length) != length) {
|
|
||||||
if (!gst_ebml_read_handle_event (ebml))
|
|
||||||
return NULL;
|
|
||||||
}
|
|
||||||
|
|
||||||
gst_bytestream_flush_fast (ebml->bs, length);
|
|
||||||
|
|
||||||
return buf;
|
|
||||||
}
|
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* Return: the ID of the next element.
|
* Return: the ID of the next element.
|
||||||
* Level_up contains the amount of levels that this
|
* Level_up contains the amount of levels that this
|
||||||
* next element lies higher than the previous one.
|
* next element lies higher than the previous one.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
guint32
|
gboolean
|
||||||
gst_ebml_peek_id (GstEbmlRead * ebml, guint * level_up)
|
gst_ebml_peek_id (GstEbmlRead * ebml, guint * level_up, guint32 * id)
|
||||||
{
|
{
|
||||||
guint32 id;
|
guint64 off;
|
||||||
|
|
||||||
g_assert (level_up);
|
g_assert (level_up);
|
||||||
|
|
||||||
if (gst_ebml_read_element_id (ebml, &id, level_up) < 0)
|
off = ebml->offset; /* save offset */
|
||||||
return 0;
|
|
||||||
|
|
||||||
return id;
|
if (!gst_ebml_read_element_id (ebml, id, level_up))
|
||||||
|
return FALSE;
|
||||||
|
|
||||||
|
ebml->offset = off; /* restore offset */
|
||||||
|
return TRUE;
|
||||||
|
}
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Return the length of the stream in bytes
|
||||||
|
*/
|
||||||
|
|
||||||
|
gint64
|
||||||
|
gst_ebml_read_get_length (GstEbmlRead * ebml)
|
||||||
|
{
|
||||||
|
GstFormat fmt = GST_FORMAT_BYTES;
|
||||||
|
gint64 pos, end;
|
||||||
|
|
||||||
|
if (!gst_pad_query_position (GST_PAD_PEER (ebml->sinkpad), &fmt, &pos, &end))
|
||||||
|
g_return_val_if_reached (0); ///// FIXME /////////
|
||||||
|
|
||||||
|
if (fmt != GST_FORMAT_BYTES || end < 0)
|
||||||
|
g_return_val_if_reached (0); ///// FIXME /////////
|
||||||
|
|
||||||
|
return end;
|
||||||
}
|
}
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* Seek to a given offset.
|
* Seek to a given offset.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
GstEvent *
|
gboolean
|
||||||
gst_ebml_read_seek (GstEbmlRead * ebml, guint64 offset)
|
gst_ebml_read_seek (GstEbmlRead * ebml, guint64 offset)
|
||||||
{
|
{
|
||||||
guint32 remaining;
|
if (offset >= gst_ebml_read_get_length (ebml))
|
||||||
GstEvent *event = NULL;
|
return FALSE;
|
||||||
guchar *data;
|
|
||||||
|
|
||||||
/* first, flush remaining buffers */
|
ebml->offset = offset;
|
||||||
gst_bytestream_get_status (ebml->bs, &remaining, &event);
|
|
||||||
if (event) {
|
|
||||||
GST_WARNING ("Unexpected event before seek");
|
|
||||||
if (!gst_ebml_read_use_event (ebml, event))
|
|
||||||
return NULL;
|
|
||||||
event = NULL;
|
|
||||||
}
|
|
||||||
if (remaining)
|
|
||||||
gst_bytestream_flush_fast (ebml->bs, remaining);
|
|
||||||
|
|
||||||
/* now seek */
|
return TRUE;
|
||||||
if (!gst_bytestream_seek (ebml->bs, offset, GST_SEEK_METHOD_SET)) {
|
|
||||||
GST_ELEMENT_ERROR (ebml, RESOURCE, SEEK, (NULL),
|
|
||||||
("Seek to position %llu (0x%llx) failed", offset, offset));
|
|
||||||
return NULL;
|
|
||||||
}
|
|
||||||
|
|
||||||
while (!event) {
|
|
||||||
/* and now, peek a new byte. This will fail because there's a
|
|
||||||
* pending event. Then, take the event and return it. */
|
|
||||||
if (gst_bytestream_peek_bytes (ebml->bs, &data, 1)) {
|
|
||||||
GST_WARNING ("Unexpected data after seek - this means seek failed");
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
|
|
||||||
/* get the discont event and return */
|
|
||||||
gst_bytestream_get_status (ebml->bs, &remaining, &event);
|
|
||||||
if (!event) {
|
|
||||||
GST_WARNING ("No discontinuity event after seek - seek failed");
|
|
||||||
break;
|
|
||||||
} else if (GST_EVENT_TYPE (event) != GST_EVENT_DISCONTINUOUS) {
|
|
||||||
if (!gst_ebml_read_use_event (ebml, event))
|
|
||||||
return NULL;
|
|
||||||
event = NULL;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
/* at the end of a seek, we have no cached ID anymore */
|
|
||||||
ebml->id_cache = 0;
|
|
||||||
|
|
||||||
return event;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/*
|
/*
|
||||||
|
@ -418,74 +399,16 @@ gst_ebml_read_seek (GstEbmlRead * ebml, guint64 offset)
|
||||||
gboolean
|
gboolean
|
||||||
gst_ebml_read_skip (GstEbmlRead * ebml)
|
gst_ebml_read_skip (GstEbmlRead * ebml)
|
||||||
{
|
{
|
||||||
gint bytes;
|
|
||||||
guint32 id, remaining;
|
|
||||||
guint64 length;
|
guint64 length;
|
||||||
GstEvent *event;
|
|
||||||
|
|
||||||
if ((bytes = gst_ebml_read_element_id (ebml, &id, NULL)) < 0)
|
|
||||||
return FALSE;
|
|
||||||
gst_bytestream_flush_fast (ebml->bs, bytes);
|
|
||||||
|
|
||||||
if ((bytes = gst_ebml_read_element_length (ebml, &length)) < 0)
|
|
||||||
return FALSE;
|
|
||||||
gst_bytestream_flush_fast (ebml->bs, bytes);
|
|
||||||
|
|
||||||
/* do we have enough bytes left to skip? */
|
|
||||||
gst_bytestream_get_status (ebml->bs, &remaining, &event);
|
|
||||||
if (event) {
|
|
||||||
g_warning ("Unexpected event before skip");
|
|
||||||
if (!gst_ebml_read_use_event (ebml, event))
|
|
||||||
return FALSE;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (remaining >= length) {
|
|
||||||
ebml->id_cache = 0;
|
|
||||||
gst_bytestream_flush_fast (ebml->bs, length);
|
|
||||||
|
|
||||||
return TRUE;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (!(event = gst_ebml_read_seek (ebml,
|
|
||||||
gst_bytestream_tell (ebml->bs) + length))) {
|
|
||||||
return FALSE;
|
|
||||||
}
|
|
||||||
|
|
||||||
gst_event_unref (event);
|
|
||||||
ebml->id_cache = 0;
|
|
||||||
|
|
||||||
return TRUE;
|
|
||||||
}
|
|
||||||
|
|
||||||
/*
|
|
||||||
* Make sure there's one whole next chunk available
|
|
||||||
* for uninterrupted reading.
|
|
||||||
*/
|
|
||||||
|
|
||||||
gboolean
|
|
||||||
gst_ebml_read_reserve (GstEbmlRead * ebml)
|
|
||||||
{
|
|
||||||
guint64 length;
|
|
||||||
gint bytes;
|
|
||||||
guint8 *data;
|
|
||||||
guint32 id;
|
guint32 id;
|
||||||
|
|
||||||
/* make sure the ID is cached */
|
if (!gst_ebml_read_element_id (ebml, &id, NULL))
|
||||||
if (gst_ebml_read_element_id (ebml, &id, NULL) < 0)
|
|
||||||
return FALSE;
|
return FALSE;
|
||||||
|
|
||||||
/* now we can peek the length */
|
if (gst_ebml_read_element_length (ebml, &length) < 0)
|
||||||
if ((bytes = gst_ebml_read_element_length (ebml, &length)) < 0)
|
|
||||||
return FALSE;
|
return FALSE;
|
||||||
length += bytes;
|
|
||||||
|
|
||||||
/* now check if we have that amount available */
|
ebml->offset += length;
|
||||||
while (gst_bytestream_peek_bytes (ebml->bs, &data, length) != length) {
|
|
||||||
if (!gst_ebml_read_handle_event (ebml))
|
|
||||||
return FALSE;
|
|
||||||
}
|
|
||||||
|
|
||||||
/* yes, available */
|
|
||||||
return TRUE;
|
return TRUE;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -496,25 +419,24 @@ gst_ebml_read_reserve (GstEbmlRead * ebml)
|
||||||
gboolean
|
gboolean
|
||||||
gst_ebml_read_buffer (GstEbmlRead * ebml, guint32 * id, GstBuffer ** buf)
|
gst_ebml_read_buffer (GstEbmlRead * ebml, guint32 * id, GstBuffer ** buf)
|
||||||
{
|
{
|
||||||
gint bytes;
|
|
||||||
guint64 length;
|
guint64 length;
|
||||||
|
|
||||||
if ((bytes = gst_ebml_read_element_id (ebml, id, NULL)) < 0)
|
if (!gst_ebml_read_element_id (ebml, id, NULL))
|
||||||
return FALSE;
|
return FALSE;
|
||||||
|
|
||||||
gst_bytestream_flush_fast (ebml->bs, bytes);
|
if (gst_ebml_read_element_length (ebml, &length) < 0)
|
||||||
|
|
||||||
if ((bytes = gst_ebml_read_element_length (ebml, &length)) < 0)
|
|
||||||
return FALSE;
|
return FALSE;
|
||||||
|
|
||||||
gst_bytestream_flush_fast (ebml->bs, bytes);
|
|
||||||
ebml->id_cache = 0;
|
|
||||||
if (length == 0) {
|
if (length == 0) {
|
||||||
*buf = gst_buffer_new ();
|
*buf = gst_buffer_new ();
|
||||||
return TRUE;
|
return TRUE;
|
||||||
}
|
}
|
||||||
|
|
||||||
return ((*buf = gst_ebml_read_element_data (ebml, length)) != NULL);
|
*buf = NULL;
|
||||||
|
if (!gst_ebml_read_pull_bytes (ebml, (guint) length, buf))
|
||||||
|
return FALSE;
|
||||||
|
|
||||||
|
return TRUE;
|
||||||
}
|
}
|
||||||
|
|
||||||
/*
|
/*
|
||||||
|
@ -686,7 +608,17 @@ gst_ebml_read_ascii (GstEbmlRead * ebml, guint32 * id, gchar ** str)
|
||||||
gboolean
|
gboolean
|
||||||
gst_ebml_read_utf8 (GstEbmlRead * ebml, guint32 * id, gchar ** str)
|
gst_ebml_read_utf8 (GstEbmlRead * ebml, guint32 * id, gchar ** str)
|
||||||
{
|
{
|
||||||
return gst_ebml_read_ascii (ebml, id, str);
|
gboolean ret;
|
||||||
|
guint64 oldoff = ebml->offset;
|
||||||
|
|
||||||
|
ret = gst_ebml_read_ascii (ebml, id, str);
|
||||||
|
|
||||||
|
if (str != NULL && *str != NULL && **str != '\0' &&
|
||||||
|
!g_utf8_validate (*str, -1, NULL)) {
|
||||||
|
GST_WARNING ("Invalid UTF-8 string at offset %" G_GUINT64_FORMAT, oldoff);
|
||||||
|
}
|
||||||
|
|
||||||
|
return ret;
|
||||||
}
|
}
|
||||||
|
|
||||||
/*
|
/*
|
||||||
|
@ -712,24 +644,20 @@ gst_ebml_read_date (GstEbmlRead * ebml, guint32 * id, gint64 * date)
|
||||||
gboolean
|
gboolean
|
||||||
gst_ebml_read_master (GstEbmlRead * ebml, guint32 * id)
|
gst_ebml_read_master (GstEbmlRead * ebml, guint32 * id)
|
||||||
{
|
{
|
||||||
gint bytes;
|
|
||||||
guint64 length;
|
|
||||||
GstEbmlLevel *level;
|
GstEbmlLevel *level;
|
||||||
|
guint64 length;
|
||||||
|
|
||||||
if ((bytes = gst_ebml_read_element_id (ebml, id, NULL)) < 0)
|
if (!gst_ebml_read_element_id (ebml, id, NULL))
|
||||||
return FALSE;
|
return FALSE;
|
||||||
gst_bytestream_flush_fast (ebml->bs, bytes);
|
|
||||||
|
|
||||||
if ((bytes = gst_ebml_read_element_length (ebml, &length)) < 0)
|
if (gst_ebml_read_element_length (ebml, &length) < 0)
|
||||||
return FALSE;
|
return FALSE;
|
||||||
gst_bytestream_flush_fast (ebml->bs, bytes);
|
|
||||||
|
|
||||||
/* remember level */
|
/* remember level */
|
||||||
level = g_new (GstEbmlLevel, 1);
|
level = g_new (GstEbmlLevel, 1);
|
||||||
level->start = gst_bytestream_tell (ebml->bs);
|
level->start = ebml->offset;
|
||||||
level->length = length;
|
level->length = length;
|
||||||
ebml->level = g_list_append (ebml->level, level);
|
ebml->level = g_list_append (ebml->level, level);
|
||||||
ebml->id_cache = 0;
|
|
||||||
|
|
||||||
return TRUE;
|
return TRUE;
|
||||||
}
|
}
|
||||||
|
@ -772,18 +700,20 @@ gst_ebml_read_header (GstEbmlRead * ebml, gchar ** doctype, guint * version)
|
||||||
if (version)
|
if (version)
|
||||||
*version = 1;
|
*version = 1;
|
||||||
|
|
||||||
if (!(id = gst_ebml_peek_id (ebml, &level_up)))
|
if (!gst_ebml_peek_id (ebml, &level_up, &id))
|
||||||
return FALSE;
|
return FALSE;
|
||||||
|
|
||||||
|
GST_DEBUG_OBJECT (ebml, "id: %08x", GST_READ_UINT32_BE (&id));
|
||||||
|
|
||||||
if (level_up != 0 || id != GST_EBML_ID_HEADER) {
|
if (level_up != 0 || id != GST_EBML_ID_HEADER) {
|
||||||
GST_ELEMENT_ERROR (ebml, STREAM, WRONG_TYPE, (NULL), (NULL));
|
GST_ELEMENT_ERROR (ebml, STREAM, WRONG_TYPE, (NULL), (NULL));
|
||||||
return FALSE;
|
return FALSE;
|
||||||
}
|
}
|
||||||
if (!gst_ebml_read_master (ebml, &id))
|
if (!gst_ebml_read_master (ebml, &id))
|
||||||
return FALSE;
|
return FALSE;
|
||||||
g_assert (id == GST_EBML_ID_HEADER);
|
|
||||||
|
|
||||||
while (TRUE) {
|
while (TRUE) {
|
||||||
if (!(id = gst_ebml_peek_id (ebml, &level_up)))
|
if (!gst_ebml_peek_id (ebml, &level_up, &id))
|
||||||
return FALSE;
|
return FALSE;
|
||||||
|
|
||||||
/* end-of-header */
|
/* end-of-header */
|
||||||
|
|
|
@ -22,9 +22,7 @@
|
||||||
#ifndef __GST_EBML_READ_H__
|
#ifndef __GST_EBML_READ_H__
|
||||||
#define __GST_EBML_READ_H__
|
#define __GST_EBML_READ_H__
|
||||||
|
|
||||||
#include <glib.h>
|
|
||||||
#include <gst/gst.h>
|
#include <gst/gst.h>
|
||||||
#include <gst/bytestream/bytestream.h>
|
|
||||||
|
|
||||||
G_BEGIN_DECLS
|
G_BEGIN_DECLS
|
||||||
|
|
||||||
|
@ -42,64 +40,77 @@ G_BEGIN_DECLS
|
||||||
(G_TYPE_INSTANCE_GET_CLASS ((obj), GST_TYPE_EBML_READ, GstEbmlReadClass))
|
(G_TYPE_INSTANCE_GET_CLASS ((obj), GST_TYPE_EBML_READ, GstEbmlReadClass))
|
||||||
|
|
||||||
typedef struct _GstEbmlLevel {
|
typedef struct _GstEbmlLevel {
|
||||||
guint64 start,
|
guint64 start;
|
||||||
length;
|
guint64 length;
|
||||||
} GstEbmlLevel;
|
} GstEbmlLevel;
|
||||||
|
|
||||||
typedef struct _GstEbmlRead {
|
typedef struct _GstEbmlRead {
|
||||||
GstElement parent;
|
GstElement parent;
|
||||||
|
|
||||||
|
GstBuffer *cached_buffer;
|
||||||
|
|
||||||
GstPad *sinkpad;
|
GstPad *sinkpad;
|
||||||
GstByteStream *bs;
|
guint64 offset;
|
||||||
|
|
||||||
GList *level;
|
GList *level;
|
||||||
|
|
||||||
/* cache of ID (peeking) */
|
|
||||||
guint32 id_cache;
|
|
||||||
} GstEbmlRead;
|
} GstEbmlRead;
|
||||||
|
|
||||||
typedef struct _GstEbmlReadClass {
|
typedef struct _GstEbmlReadClass {
|
||||||
GstElementClass parent;
|
GstElementClass parent;
|
||||||
} GstEbmlReadClass;
|
} GstEbmlReadClass;
|
||||||
|
|
||||||
GType gst_ebml_read_get_type (void);
|
GType gst_ebml_read_get_type (void);
|
||||||
|
|
||||||
guint32 gst_ebml_peek_id (GstEbmlRead *ebml,
|
gboolean gst_ebml_peek_id (GstEbmlRead *ebml,
|
||||||
guint *level_up);
|
guint *level_up,
|
||||||
GstEvent *gst_ebml_read_seek (GstEbmlRead *ebml,
|
guint32 *id);
|
||||||
guint64 offset);
|
|
||||||
gboolean gst_ebml_read_skip (GstEbmlRead *ebml);
|
gboolean gst_ebml_read_seek (GstEbmlRead *ebml,
|
||||||
gboolean gst_ebml_read_reserve (GstEbmlRead *ebml);
|
guint64 offset);
|
||||||
gboolean gst_ebml_read_buffer (GstEbmlRead *ebml,
|
|
||||||
guint32 *id,
|
gint64 gst_ebml_read_get_length (GstEbmlRead *ebml);
|
||||||
GstBuffer **buf);
|
|
||||||
gboolean gst_ebml_read_uint (GstEbmlRead *ebml,
|
gboolean gst_ebml_read_skip (GstEbmlRead *ebml);
|
||||||
guint32 *id,
|
|
||||||
guint64 *num);
|
gboolean gst_ebml_read_buffer (GstEbmlRead *ebml,
|
||||||
gboolean gst_ebml_read_sint (GstEbmlRead *ebml,
|
guint32 *id,
|
||||||
guint32 *id,
|
GstBuffer **buf);
|
||||||
gint64 *num);
|
|
||||||
gboolean gst_ebml_read_float (GstEbmlRead *ebml,
|
gboolean gst_ebml_read_uint (GstEbmlRead *ebml,
|
||||||
guint32 *id,
|
guint32 *id,
|
||||||
gdouble *num);
|
guint64 *num);
|
||||||
gboolean gst_ebml_read_ascii (GstEbmlRead *ebml,
|
|
||||||
guint32 *id,
|
gboolean gst_ebml_read_sint (GstEbmlRead *ebml,
|
||||||
gchar **str);
|
guint32 *id,
|
||||||
gboolean gst_ebml_read_utf8 (GstEbmlRead *ebml,
|
gint64 *num);
|
||||||
guint32 *id,
|
|
||||||
gchar **str);
|
gboolean gst_ebml_read_float (GstEbmlRead *ebml,
|
||||||
gboolean gst_ebml_read_date (GstEbmlRead *ebml,
|
guint32 *id,
|
||||||
guint32 *id,
|
gdouble *num);
|
||||||
gint64 *date);
|
|
||||||
gboolean gst_ebml_read_master (GstEbmlRead *ebml,
|
gboolean gst_ebml_read_ascii (GstEbmlRead *ebml,
|
||||||
guint32 *id);
|
guint32 *id,
|
||||||
gboolean gst_ebml_read_binary (GstEbmlRead *ebml,
|
gchar **str);
|
||||||
guint32 *id,
|
|
||||||
guint8 **binary,
|
gboolean gst_ebml_read_utf8 (GstEbmlRead *ebml,
|
||||||
guint64 *length);
|
guint32 *id,
|
||||||
gboolean gst_ebml_read_header (GstEbmlRead *read,
|
gchar **str);
|
||||||
gchar **doctype,
|
|
||||||
guint *version);
|
gboolean gst_ebml_read_date (GstEbmlRead *ebml,
|
||||||
|
guint32 *id,
|
||||||
|
gint64 *date);
|
||||||
|
|
||||||
|
gboolean gst_ebml_read_master (GstEbmlRead *ebml,
|
||||||
|
guint32 *id);
|
||||||
|
|
||||||
|
gboolean gst_ebml_read_binary (GstEbmlRead *ebml,
|
||||||
|
guint32 *id,
|
||||||
|
guint8 **binary,
|
||||||
|
guint64 *length);
|
||||||
|
|
||||||
|
gboolean gst_ebml_read_header (GstEbmlRead *read,
|
||||||
|
gchar **doctype,
|
||||||
|
guint *version);
|
||||||
|
|
||||||
G_END_DECLS
|
G_END_DECLS
|
||||||
|
|
||||||
|
|
File diff suppressed because it is too large
Load diff
|
@ -23,7 +23,6 @@
|
||||||
#define __GST_MATROSKA_DEMUX_H__
|
#define __GST_MATROSKA_DEMUX_H__
|
||||||
|
|
||||||
#include <gst/gst.h>
|
#include <gst/gst.h>
|
||||||
#include <gst/bytestream/bytestream.h>
|
|
||||||
|
|
||||||
#include "ebml-read.h"
|
#include "ebml-read.h"
|
||||||
#include "matroska-ids.h"
|
#include "matroska-ids.h"
|
||||||
|
@ -56,50 +55,57 @@ typedef struct _GstMatroskaDemuxIndex {
|
||||||
} GstMatroskaDemuxIndex;
|
} GstMatroskaDemuxIndex;
|
||||||
|
|
||||||
typedef struct _GstMatroskaDemux {
|
typedef struct _GstMatroskaDemux {
|
||||||
GstEbmlRead parent;
|
GstEbmlRead parent;
|
||||||
|
|
||||||
/* pads */
|
/* pads */
|
||||||
GstPad *sinkpad;
|
GstPad *sinkpad;
|
||||||
GstMatroskaTrackContext *src[GST_MATROSKA_DEMUX_MAX_STREAMS];
|
GstMatroskaTrackContext *src[GST_MATROSKA_DEMUX_MAX_STREAMS];
|
||||||
guint num_streams,
|
GstClock *clock;
|
||||||
num_v_streams, num_a_streams, num_t_streams;
|
guint num_streams;
|
||||||
GstClock *clock;
|
guint num_v_streams;
|
||||||
|
guint num_a_streams;
|
||||||
|
guint num_t_streams;
|
||||||
|
|
||||||
/* metadata */
|
/* metadata */
|
||||||
gchar *muxing_app, *writing_app;
|
gchar *muxing_app;
|
||||||
gint64 created;
|
gchar *writing_app;
|
||||||
|
gint64 created;
|
||||||
|
|
||||||
/* state */
|
/* state */
|
||||||
GstMatroskaDemuxState state;
|
GstMatroskaDemuxState state;
|
||||||
guint level_up;
|
guint level_up;
|
||||||
|
|
||||||
/* did we parse metadata/cues already? */
|
/* did we parse metadata/cues already? */
|
||||||
gboolean metadata_parsed,
|
gboolean metadata_parsed;
|
||||||
index_parsed;
|
gboolean index_parsed;
|
||||||
|
|
||||||
/* start-of-segment */
|
/* start-of-segment */
|
||||||
guint64 segment_start;
|
guint64 ebml_segment_start;
|
||||||
|
|
||||||
/* a cue (index) table */
|
/* a cue (index) table */
|
||||||
GstMatroskaIndex *index;
|
GstMatroskaIndex *index;
|
||||||
guint num_indexes;
|
guint num_indexes;
|
||||||
|
|
||||||
/* timescale in the file */
|
/* timescale in the file */
|
||||||
guint64 time_scale;
|
guint64 time_scale;
|
||||||
|
|
||||||
/* length, position (time, ns) */
|
/* length, position (time, ns) */
|
||||||
guint64 duration, pos;
|
guint64 pos;
|
||||||
|
guint64 duration;
|
||||||
|
|
||||||
/* a possible pending seek */
|
/* a possible pending seek */
|
||||||
guint64 seek_pending;
|
gboolean seek_pending;
|
||||||
|
|
||||||
|
gdouble segment_rate;
|
||||||
|
gint64 segment_start;
|
||||||
|
gint64 segment_stop;
|
||||||
|
gboolean segment_play;
|
||||||
} GstMatroskaDemux;
|
} GstMatroskaDemux;
|
||||||
|
|
||||||
typedef struct _GstMatroskaDemuxClass {
|
typedef struct _GstMatroskaDemuxClass {
|
||||||
GstEbmlReadClass parent;
|
GstEbmlReadClass parent;
|
||||||
} GstMatroskaDemuxClass;
|
} GstMatroskaDemuxClass;
|
||||||
|
|
||||||
GType gst_matroska_demux_get_type (void);
|
|
||||||
|
|
||||||
gboolean gst_matroska_demux_plugin_init (GstPlugin *plugin);
|
gboolean gst_matroska_demux_plugin_init (GstPlugin *plugin);
|
||||||
|
|
||||||
G_END_DECLS
|
G_END_DECLS
|
||||||
|
|
|
@ -23,13 +23,13 @@
|
||||||
#include "config.h"
|
#include "config.h"
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
/* #include "matroska-demux.h" */
|
#include "matroska-demux.h"
|
||||||
#include "matroska-mux.h"
|
#include "matroska-mux.h"
|
||||||
|
|
||||||
static gboolean
|
static gboolean
|
||||||
plugin_init (GstPlugin * plugin)
|
plugin_init (GstPlugin * plugin)
|
||||||
{
|
{
|
||||||
return /* gst_matroska_demux_plugin_init (plugin) && */
|
return gst_matroska_demux_plugin_init (plugin) &&
|
||||||
gst_matroska_mux_plugin_init (plugin);
|
gst_matroska_mux_plugin_init (plugin);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
Loading…
Reference in a new issue