mirror of
https://gitlab.freedesktop.org/gstreamer/gstreamer.git
synced 2025-01-17 12:55:53 +00:00
jpegdec: optimize buffer handling when parsing
Use an adapter to collect incoming data, and use adapter API to scan and peek. Fixes #583047.
This commit is contained in:
parent
a88acc848e
commit
58fbcf01e5
2 changed files with 98 additions and 165 deletions
|
@ -152,8 +152,7 @@ gst_jpeg_dec_finalize (GObject * object)
|
||||||
|
|
||||||
jpeg_destroy_decompress (&dec->cinfo);
|
jpeg_destroy_decompress (&dec->cinfo);
|
||||||
|
|
||||||
if (dec->tempbuf)
|
g_object_unref (dec->adapter);
|
||||||
gst_buffer_unref (dec->tempbuf);
|
|
||||||
|
|
||||||
G_OBJECT_CLASS (parent_class)->finalize (object);
|
G_OBJECT_CLASS (parent_class)->finalize (object);
|
||||||
}
|
}
|
||||||
|
@ -317,89 +316,34 @@ gst_jpeg_dec_init (GstJpegDec * dec)
|
||||||
|
|
||||||
/* init properties */
|
/* init properties */
|
||||||
dec->idct_method = JPEG_DEFAULT_IDCT_METHOD;
|
dec->idct_method = JPEG_DEFAULT_IDCT_METHOD;
|
||||||
}
|
|
||||||
|
|
||||||
static inline gboolean
|
dec->adapter = gst_adapter_new ();
|
||||||
is_jpeg_start_marker (const guint8 * data)
|
|
||||||
{
|
|
||||||
return (data[0] == 0xff && data[1] == 0xd8);
|
|
||||||
}
|
|
||||||
|
|
||||||
static gboolean
|
|
||||||
gst_jpeg_dec_find_jpeg_header (GstJpegDec * dec)
|
|
||||||
{
|
|
||||||
const guint8 *data;
|
|
||||||
guint size;
|
|
||||||
|
|
||||||
data = GST_BUFFER_DATA (dec->tempbuf);
|
|
||||||
size = GST_BUFFER_SIZE (dec->tempbuf);
|
|
||||||
|
|
||||||
g_return_val_if_fail (size >= 2, FALSE);
|
|
||||||
|
|
||||||
while (!is_jpeg_start_marker (data) || data[2] != 0xff) {
|
|
||||||
const guint8 *marker;
|
|
||||||
GstBuffer *tmp;
|
|
||||||
guint off;
|
|
||||||
|
|
||||||
marker = memchr (data + 1, 0xff, size - 1 - 2);
|
|
||||||
if (marker == NULL) {
|
|
||||||
off = size - 1; /* keep last byte */
|
|
||||||
} else {
|
|
||||||
off = marker - data;
|
|
||||||
}
|
|
||||||
|
|
||||||
tmp = gst_buffer_create_sub (dec->tempbuf, off, size - off);
|
|
||||||
gst_buffer_unref (dec->tempbuf);
|
|
||||||
dec->tempbuf = tmp;
|
|
||||||
|
|
||||||
data = GST_BUFFER_DATA (dec->tempbuf);
|
|
||||||
size = GST_BUFFER_SIZE (dec->tempbuf);
|
|
||||||
|
|
||||||
if (size < 2)
|
|
||||||
return FALSE; /* wait for more data */
|
|
||||||
}
|
|
||||||
|
|
||||||
return TRUE; /* got header */
|
|
||||||
}
|
}
|
||||||
|
|
||||||
static gboolean
|
static gboolean
|
||||||
gst_jpeg_dec_ensure_header (GstJpegDec * dec)
|
gst_jpeg_dec_ensure_header (GstJpegDec * dec)
|
||||||
{
|
{
|
||||||
g_return_val_if_fail (dec->tempbuf != NULL, FALSE);
|
gint av;
|
||||||
|
gint offset;
|
||||||
|
|
||||||
check_header:
|
av = gst_adapter_available (dec->adapter);
|
||||||
|
/* we expect at least 4 bytes, first of which start marker */
|
||||||
/* we need at least a start marker (0xff 0xd8)
|
offset = gst_adapter_masked_scan_uint32 (dec->adapter, 0xffffff00, 0xffd8ff00,
|
||||||
* and an end marker (0xff 0xd9) */
|
0, av);
|
||||||
if (GST_BUFFER_SIZE (dec->tempbuf) <= 4) {
|
if (G_UNLIKELY (offset < 0)) {
|
||||||
GST_DEBUG ("Not enough data");
|
GST_DEBUG_OBJECT (dec, "No JPEG header in current buffer");
|
||||||
return FALSE; /* we need more data */
|
/* not found */
|
||||||
|
if (av > 4)
|
||||||
|
gst_adapter_flush (dec->adapter, av - 4);
|
||||||
|
return FALSE;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (!is_jpeg_start_marker (GST_BUFFER_DATA (dec->tempbuf))) {
|
GST_DEBUG_OBJECT (dec, "Found JPEG header");
|
||||||
GST_DEBUG ("Not a JPEG header, resyncing to header...");
|
gst_adapter_flush (dec->adapter, offset);
|
||||||
if (!gst_jpeg_dec_find_jpeg_header (dec)) {
|
|
||||||
GST_DEBUG ("No JPEG header in current buffer");
|
|
||||||
return FALSE; /* we need more data */
|
|
||||||
}
|
|
||||||
GST_DEBUG ("Found JPEG header");
|
|
||||||
goto check_header; /* buffer might have changed */
|
|
||||||
}
|
|
||||||
|
|
||||||
return TRUE;
|
return TRUE;
|
||||||
}
|
}
|
||||||
|
|
||||||
#if 0
|
|
||||||
static gboolean
|
|
||||||
gst_jpeg_dec_have_end_marker (GstJpegDec * dec)
|
|
||||||
{
|
|
||||||
guint8 *data = GST_BUFFER_DATA (dec->tempbuf);
|
|
||||||
guint size = GST_BUFFER_SIZE (dec->tempbuf);
|
|
||||||
|
|
||||||
return (size > 2 && data && is_jpeg_end_marker (data + size - 2));
|
|
||||||
}
|
|
||||||
#endif
|
|
||||||
|
|
||||||
static inline gboolean
|
static inline gboolean
|
||||||
gst_jpeg_dec_parse_tag_has_entropy_segment (guint8 tag)
|
gst_jpeg_dec_parse_tag_has_entropy_segment (guint8 tag)
|
||||||
{
|
{
|
||||||
|
@ -408,103 +352,107 @@ gst_jpeg_dec_parse_tag_has_entropy_segment (guint8 tag)
|
||||||
return FALSE;
|
return FALSE;
|
||||||
}
|
}
|
||||||
|
|
||||||
/* returns image length in bytes if parsed
|
/* returns image length in bytes if parsed
|
||||||
* successfully, otherwise 0 */
|
* successfully, otherwise 0 */
|
||||||
static guint
|
static guint
|
||||||
gst_jpeg_dec_parse_image_data (GstJpegDec * dec)
|
gst_jpeg_dec_parse_image_data (GstJpegDec * dec)
|
||||||
{
|
{
|
||||||
guint8 *start, *data, *end;
|
|
||||||
guint size;
|
guint size;
|
||||||
gboolean resync;
|
gboolean resync;
|
||||||
|
GstAdapter *adapter = dec->adapter;
|
||||||
|
gint offset, noffset;
|
||||||
|
|
||||||
size = GST_BUFFER_SIZE (dec->tempbuf);
|
size = gst_adapter_available (adapter);
|
||||||
start = GST_BUFFER_DATA (dec->tempbuf);
|
|
||||||
end = start + size;
|
|
||||||
data = start;
|
|
||||||
|
|
||||||
g_return_val_if_fail (is_jpeg_start_marker (data), 0);
|
/* we expect at least 4 bytes, first of which start marker */
|
||||||
|
if (gst_adapter_masked_scan_uint32 (adapter, 0xffff0000, 0xffd80000, 0, 4))
|
||||||
|
return 0;
|
||||||
|
|
||||||
GST_DEBUG ("Parsing jpeg image data (%u bytes)", size);
|
GST_DEBUG ("Parsing jpeg image data (%u bytes)", size);
|
||||||
|
|
||||||
GST_DEBUG ("Parse state: offset=%d, resync=%d, entropy len=%d",
|
GST_DEBUG ("Parse state: offset=%d, resync=%d, entropy len=%d",
|
||||||
dec->parse_offset, dec->parse_resync, dec->parse_entropy_len);
|
dec->parse_offset, dec->parse_resync, dec->parse_entropy_len);
|
||||||
|
|
||||||
/* resume from state offset (also skips start marker) */
|
/* offset is 2 less than actual offset;
|
||||||
data += (dec->parse_offset ? dec->parse_offset : 2);
|
* - adapter needs at least 4 bytes for scanning,
|
||||||
|
* - start and end marker ensure at least that much
|
||||||
|
*/
|
||||||
|
/* resume from state offset */
|
||||||
|
offset = dec->parse_offset;
|
||||||
|
|
||||||
while (1) {
|
while (1) {
|
||||||
guint frame_len;
|
guint frame_len;
|
||||||
|
guint32 value;
|
||||||
|
|
||||||
/* do we need to resync? */
|
noffset =
|
||||||
resync = (*data != 0xff);
|
gst_adapter_masked_scan_uint32_peek (adapter, 0x0000ff00, 0x0000ff00,
|
||||||
if (resync) {
|
offset, size - offset, &value);
|
||||||
GST_DEBUG ("Lost sync at 0x%08" G_GINT64_MODIFIER "x, resyncing",
|
/* lost sync if 0xff marker not where expected */
|
||||||
(gint64) (data - start));
|
if ((resync = (noffset != offset))) {
|
||||||
/* at the very least we expect 0xff 0xNN, thus end-1 */
|
GST_DEBUG ("Lost sync at 0x%08x, resyncing", offset + 2);
|
||||||
while (*data != 0xff && data < end - 1)
|
|
||||||
++data;
|
|
||||||
if (G_UNLIKELY (*data != 0xff)) {
|
|
||||||
GST_DEBUG ("at end of input and no next marker found, need more data");
|
|
||||||
goto need_more_data;
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
/* may have marker, but could have been resyncng */
|
/* may have marker, but could have been resyncng */
|
||||||
resync = resync || dec->parse_resync;
|
resync = resync || dec->parse_resync;
|
||||||
/* Skip over extra 0xff */
|
/* Skip over extra 0xff */
|
||||||
while (*data == 0xff && data < end)
|
while ((noffset > 0) && ((value & 0xff) == 0xff)) {
|
||||||
++data;
|
noffset++;
|
||||||
|
noffset =
|
||||||
|
gst_adapter_masked_scan_uint32_peek (adapter, 0x0000ff00, 0x0000ff00,
|
||||||
|
noffset, size - noffset, &value);
|
||||||
|
}
|
||||||
/* enough bytes left for marker? (we need 0xNN after the 0xff) */
|
/* enough bytes left for marker? (we need 0xNN after the 0xff) */
|
||||||
if (data >= end) {
|
if (noffset < 0) {
|
||||||
GST_DEBUG ("at end of input and no EOI marker found, need more data");
|
GST_DEBUG ("at end of input and no EOI marker found, need more data");
|
||||||
goto need_more_data1;
|
goto need_more_data;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (*data == 0xd9) {
|
/* now lock on the marker we found */
|
||||||
GST_DEBUG ("0x%08" G_GINT64_MODIFIER "x: EOI marker",
|
offset = noffset;
|
||||||
(gint64) (data - start));
|
value = value & 0xff;
|
||||||
|
if (value == 0xd9) {
|
||||||
|
GST_DEBUG ("0x%08x: EOI marker", offset + 2);
|
||||||
/* clear parse state */
|
/* clear parse state */
|
||||||
dec->parse_resync = FALSE;
|
dec->parse_resync = FALSE;
|
||||||
dec->parse_offset = 0;
|
dec->parse_offset = 0;
|
||||||
return (data - start + 1);
|
return (offset + 4);
|
||||||
}
|
}
|
||||||
|
|
||||||
if (*data >= 0xd0 && *data <= 0xd7)
|
if (value >= 0xd0 && value <= 0xd7)
|
||||||
frame_len = 0;
|
frame_len = 0;
|
||||||
else if (data >= end - 2)
|
else {
|
||||||
goto need_more_data1;
|
/* peek tag and subsequent length */
|
||||||
else
|
if (offset + 2 + 4 > size)
|
||||||
frame_len = GST_READ_UINT16_BE (data + 1);
|
goto need_more_data;
|
||||||
GST_DEBUG ("0x%08" G_GINT64_MODIFIER "x: tag %02x, frame_len=%u",
|
else
|
||||||
(gint64) (data - start - 1), *data, frame_len);
|
gst_adapter_masked_scan_uint32_peek (adapter, 0x0, 0x0, offset + 2, 4,
|
||||||
|
&frame_len);
|
||||||
|
frame_len = frame_len & 0xffff;
|
||||||
|
}
|
||||||
|
GST_DEBUG ("0x%08x: tag %02x, frame_len=%u", offset + 2, value, frame_len);
|
||||||
/* the frame length includes the 2 bytes for the length; here we want at
|
/* the frame length includes the 2 bytes for the length; here we want at
|
||||||
* least 2 more bytes at the end for an end marker, thus end-2 */
|
* least 2 more bytes at the end for an end marker */
|
||||||
if (data + 1 + frame_len >= end - 2) {
|
if (offset + 2 + 2 + frame_len + 2 > size) {
|
||||||
if (resync) {
|
goto need_more_data;
|
||||||
GST_DEBUG ("not a valid sync (not enough data).");
|
|
||||||
/* Since *data != 0xff, the next iteration will go into resync again. */
|
|
||||||
continue;
|
|
||||||
}
|
|
||||||
/* theoretically we could have lost sync and not really need more
|
|
||||||
* data, but that's just tough luck and a broken image then */
|
|
||||||
GST_DEBUG ("at end of input and no EOI marker found, need more data");
|
|
||||||
goto need_more_data1;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
if (gst_jpeg_dec_parse_tag_has_entropy_segment (*data)) {
|
if (gst_jpeg_dec_parse_tag_has_entropy_segment (value)) {
|
||||||
guint8 *d2 = data + 1 + frame_len;
|
|
||||||
guint eseglen = dec->parse_entropy_len;
|
guint eseglen = dec->parse_entropy_len;
|
||||||
|
|
||||||
GST_DEBUG ("0x%08" G_GINT64_MODIFIER "x: finding entropy segment length",
|
GST_DEBUG ("0x%08x: finding entropy segment length", offset + 2);
|
||||||
(gint64) (data - start - 1));
|
noffset = offset + 2 + frame_len + dec->parse_entropy_len;
|
||||||
while (1) {
|
while (1) {
|
||||||
if (d2 + eseglen >= end - 1) {
|
noffset = gst_adapter_masked_scan_uint32_peek (adapter, 0x0000ff00,
|
||||||
|
0x0000ff00, noffset, size - noffset, &value);
|
||||||
|
if (noffset < 0) {
|
||||||
/* need more data */
|
/* need more data */
|
||||||
dec->parse_entropy_len = eseglen;
|
dec->parse_entropy_len = size - offset - 4 - frame_len - 2;
|
||||||
goto need_more_data1;
|
goto need_more_data;
|
||||||
}
|
}
|
||||||
if (d2[eseglen] == 0xff && d2[eseglen + 1] != 0x00)
|
if ((value & 0xff) != 0x00) {
|
||||||
|
eseglen = noffset - offset - frame_len - 2;
|
||||||
break;
|
break;
|
||||||
++eseglen;
|
}
|
||||||
|
noffset++;
|
||||||
}
|
}
|
||||||
dec->parse_entropy_len = 0;
|
dec->parse_entropy_len = 0;
|
||||||
frame_len += eseglen;
|
frame_len += eseglen;
|
||||||
|
@ -514,28 +462,24 @@ gst_jpeg_dec_parse_image_data (GstJpegDec * dec)
|
||||||
if (resync) {
|
if (resync) {
|
||||||
/* check if we will still be in sync if we interpret
|
/* check if we will still be in sync if we interpret
|
||||||
* this as a sync point and skip this frame */
|
* this as a sync point and skip this frame */
|
||||||
if (data[2 + frame_len] != 0xff) {
|
noffset = offset + frame_len + 2;
|
||||||
|
noffset = gst_adapter_masked_scan_uint32 (adapter, 0x0000ff00, 0x0000ff00,
|
||||||
|
noffset, 4);
|
||||||
|
if (noffset < 0) {
|
||||||
/* ignore and continue resyncing until we hit the end
|
/* ignore and continue resyncing until we hit the end
|
||||||
* of our data or find a sync point that looks okay */
|
* of our data or find a sync point that looks okay */
|
||||||
continue;
|
continue;
|
||||||
}
|
}
|
||||||
GST_DEBUG ("found sync at %p", data - size);
|
GST_DEBUG ("found sync at 0x%x", offset + 2);
|
||||||
}
|
}
|
||||||
|
|
||||||
data += 1 + frame_len;
|
offset += frame_len + 2;
|
||||||
}
|
}
|
||||||
|
|
||||||
/* EXITS */
|
/* EXITS */
|
||||||
need_more_data:
|
need_more_data:
|
||||||
{
|
{
|
||||||
dec->parse_offset = data - start;
|
dec->parse_offset = offset;
|
||||||
dec->parse_resync = resync;
|
|
||||||
return 0;
|
|
||||||
}
|
|
||||||
need_more_data1:
|
|
||||||
{
|
|
||||||
/* step back one to point at marker */
|
|
||||||
dec->parse_offset = (data - start) - 1;
|
|
||||||
dec->parse_resync = resync;
|
dec->parse_resync = resync;
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
@ -1188,17 +1132,13 @@ gst_jpeg_dec_chain (GstPad * pad, GstBuffer * buf)
|
||||||
if (GST_BUFFER_IS_DISCONT (buf)) {
|
if (GST_BUFFER_IS_DISCONT (buf)) {
|
||||||
GST_DEBUG_OBJECT (dec, "buffer has DISCONT flag set");
|
GST_DEBUG_OBJECT (dec, "buffer has DISCONT flag set");
|
||||||
dec->discont = TRUE;
|
dec->discont = TRUE;
|
||||||
if (!dec->packetized && dec->tempbuf != NULL) {
|
if (!dec->packetized && gst_adapter_available (dec->adapter)) {
|
||||||
GST_WARNING_OBJECT (dec, "DISCONT buffer in non-packetized mode, bad");
|
GST_WARNING_OBJECT (dec, "DISCONT buffer in non-packetized mode, bad");
|
||||||
gst_buffer_replace (&dec->tempbuf, NULL);
|
gst_adapter_clear (dec->adapter);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
if (dec->tempbuf) {
|
gst_adapter_push (dec->adapter, buf);
|
||||||
dec->tempbuf = gst_buffer_join (dec->tempbuf, buf);
|
|
||||||
} else {
|
|
||||||
dec->tempbuf = buf;
|
|
||||||
}
|
|
||||||
buf = NULL;
|
buf = NULL;
|
||||||
|
|
||||||
/* If we are non-packetized and know the total incoming size in bytes,
|
/* If we are non-packetized and know the total incoming size in bytes,
|
||||||
|
@ -1206,10 +1146,10 @@ gst_jpeg_dec_chain (GstPad * pad, GstBuffer * buf)
|
||||||
|
|
||||||
if (!dec->packetized && (dec->segment.format == GST_FORMAT_BYTES) &&
|
if (!dec->packetized && (dec->segment.format == GST_FORMAT_BYTES) &&
|
||||||
(dec->segment.stop != -1) &&
|
(dec->segment.stop != -1) &&
|
||||||
(GST_BUFFER_SIZE (dec->tempbuf) < dec->segment.stop)) {
|
(gst_adapter_available (dec->adapter) < dec->segment.stop)) {
|
||||||
/* We assume that non-packetized input in bytes is *one* single jpeg image */
|
/* We assume that non-packetized input in bytes is *one* single jpeg image */
|
||||||
GST_DEBUG ("Non-packetized mode. Got %d bytes, need %" G_GINT64_FORMAT,
|
GST_DEBUG ("Non-packetized mode. Got %d bytes, need %" G_GINT64_FORMAT,
|
||||||
GST_BUFFER_SIZE (dec->tempbuf), dec->segment.stop);
|
gst_adapter_available (dec->adapter), dec->segment.stop);
|
||||||
goto need_more_data;
|
goto need_more_data;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -1221,7 +1161,7 @@ gst_jpeg_dec_chain (GstPad * pad, GstBuffer * buf)
|
||||||
* do some sanity checking instead of parsing all of
|
* do some sanity checking instead of parsing all of
|
||||||
* the jpeg data */
|
* the jpeg data */
|
||||||
if (dec->packetized) {
|
if (dec->packetized) {
|
||||||
img_len = GST_BUFFER_SIZE (dec->tempbuf);
|
img_len = gst_adapter_available (dec->adapter);
|
||||||
} else {
|
} else {
|
||||||
/* Parse jpeg image to handle jpeg input that
|
/* Parse jpeg image to handle jpeg input that
|
||||||
* is not aligned to buffer boundaries */
|
* is not aligned to buffer boundaries */
|
||||||
|
@ -1235,7 +1175,7 @@ gst_jpeg_dec_chain (GstPad * pad, GstBuffer * buf)
|
||||||
if (dec->packetized && !gst_jpeg_dec_do_qos (dec, timestamp))
|
if (dec->packetized && !gst_jpeg_dec_do_qos (dec, timestamp))
|
||||||
goto skip_decoding;
|
goto skip_decoding;
|
||||||
|
|
||||||
data = (guchar *) GST_BUFFER_DATA (dec->tempbuf);
|
data = (guint8 *) gst_adapter_peek (dec->adapter, img_len);
|
||||||
GST_LOG_OBJECT (dec, "image size = %u", img_len);
|
GST_LOG_OBJECT (dec, "image size = %u", img_len);
|
||||||
|
|
||||||
dec->jsrc.pub.next_input_byte = data;
|
dec->jsrc.pub.next_input_byte = data;
|
||||||
|
@ -1454,16 +1394,7 @@ gst_jpeg_dec_chain (GstPad * pad, GstBuffer * buf)
|
||||||
|
|
||||||
skip_decoding:
|
skip_decoding:
|
||||||
done:
|
done:
|
||||||
if (GST_BUFFER_SIZE (dec->tempbuf) == img_len) {
|
gst_adapter_flush (dec->adapter, img_len);
|
||||||
gst_buffer_unref (dec->tempbuf);
|
|
||||||
dec->tempbuf = NULL;
|
|
||||||
} else {
|
|
||||||
GstBuffer *buf = gst_buffer_create_sub (dec->tempbuf, img_len,
|
|
||||||
GST_BUFFER_SIZE (dec->tempbuf) - img_len);
|
|
||||||
|
|
||||||
gst_buffer_unref (dec->tempbuf);
|
|
||||||
dec->tempbuf = buf;
|
|
||||||
}
|
|
||||||
|
|
||||||
exit:
|
exit:
|
||||||
|
|
||||||
|
@ -1598,6 +1529,10 @@ gst_jpeg_dec_sink_event (GstPad * pad, GstEvent * event)
|
||||||
GST_DEBUG_OBJECT (dec, "Aborting decompress");
|
GST_DEBUG_OBJECT (dec, "Aborting decompress");
|
||||||
jpeg_abort_decompress (&dec->cinfo);
|
jpeg_abort_decompress (&dec->cinfo);
|
||||||
gst_segment_init (&dec->segment, GST_FORMAT_UNDEFINED);
|
gst_segment_init (&dec->segment, GST_FORMAT_UNDEFINED);
|
||||||
|
gst_adapter_clear (dec->adapter);
|
||||||
|
dec->parse_offset = 0;
|
||||||
|
dec->parse_entropy_len = 0;
|
||||||
|
dec->parse_resync = FALSE;
|
||||||
gst_jpeg_dec_reset_qos (dec);
|
gst_jpeg_dec_reset_qos (dec);
|
||||||
break;
|
break;
|
||||||
case GST_EVENT_NEWSEGMENT:{
|
case GST_EVENT_NEWSEGMENT:{
|
||||||
|
@ -1700,10 +1635,7 @@ gst_jpeg_dec_change_state (GstElement * element, GstStateChange transition)
|
||||||
|
|
||||||
switch (transition) {
|
switch (transition) {
|
||||||
case GST_STATE_CHANGE_PAUSED_TO_READY:
|
case GST_STATE_CHANGE_PAUSED_TO_READY:
|
||||||
if (dec->tempbuf) {
|
gst_adapter_clear (dec->adapter);
|
||||||
gst_buffer_unref (dec->tempbuf);
|
|
||||||
dec->tempbuf = NULL;
|
|
||||||
}
|
|
||||||
gst_jpeg_dec_free_buffers (dec);
|
gst_jpeg_dec_free_buffers (dec);
|
||||||
break;
|
break;
|
||||||
default:
|
default:
|
||||||
|
|
|
@ -25,6 +25,7 @@
|
||||||
#include <setjmp.h>
|
#include <setjmp.h>
|
||||||
#include <gst/gst.h>
|
#include <gst/gst.h>
|
||||||
#include <gst/video/video.h>
|
#include <gst/video/video.h>
|
||||||
|
#include <gst/base/gstadapter.h>
|
||||||
|
|
||||||
/* this is a hack hack hack to get around jpeglib header bugs... */
|
/* this is a hack hack hack to get around jpeglib header bugs... */
|
||||||
#ifdef HAVE_STDLIB_H
|
#ifdef HAVE_STDLIB_H
|
||||||
|
@ -69,7 +70,7 @@ struct _GstJpegDec {
|
||||||
GstPad *sinkpad;
|
GstPad *sinkpad;
|
||||||
GstPad *srcpad;
|
GstPad *srcpad;
|
||||||
|
|
||||||
GstBuffer *tempbuf;
|
GstAdapter *adapter;
|
||||||
|
|
||||||
/* TRUE if each input buffer contains a whole jpeg image */
|
/* TRUE if each input buffer contains a whole jpeg image */
|
||||||
gboolean packetized;
|
gboolean packetized;
|
||||||
|
|
Loading…
Reference in a new issue