mirror of
https://gitlab.freedesktop.org/gstreamer/gstreamer.git
synced 2025-01-01 21:18:52 +00:00
baseparse: Fix upstream read caching
When running in pull mode (for e.g. mp3 reading), baseparse currently reads 64KB from upstream, then mp3parse consumes typically around 417/418 bytes of it. Then on the next loop, it will read a full fresh 64KB again, which is a big waste. Fix the read loop to use the available cache buffer first before going for more data, until the cache drops to < 1KB. Fixes https://gitlab.freedesktop.org/gstreamer/gstreamer/issues/518
This commit is contained in:
parent
35136dc91a
commit
e906197c62
2 changed files with 53 additions and 19 deletions
|
@ -3352,6 +3352,22 @@ done:
|
||||||
return ret;
|
return ret;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/* Return the number of bytes available in the cached
|
||||||
|
* read buffer, if any */
|
||||||
|
static guint
|
||||||
|
gst_base_parse_get_cached_available (GstBaseParse * parse)
|
||||||
|
{
|
||||||
|
if (parse->priv->cache != NULL) {
|
||||||
|
gint64 cache_offset = GST_BUFFER_OFFSET (parse->priv->cache);
|
||||||
|
gint cache_size = gst_buffer_get_size (parse->priv->cache);
|
||||||
|
|
||||||
|
if (parse->priv->offset >= cache_offset
|
||||||
|
&& parse->priv->offset < cache_offset + cache_size)
|
||||||
|
return cache_size - (parse->priv->offset - cache_offset); /* Size of the cache minus consumed */
|
||||||
|
}
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
/* pull @size bytes at current offset,
|
/* pull @size bytes at current offset,
|
||||||
* i.e. at least try to and possibly return a shorter buffer if near the end */
|
* i.e. at least try to and possibly return a shorter buffer if near the end */
|
||||||
static GstFlowReturn
|
static GstFlowReturn
|
||||||
|
@ -3373,6 +3389,9 @@ gst_base_parse_pull_range (GstBaseParse * parse, guint size,
|
||||||
*buffer = gst_buffer_copy_region (parse->priv->cache, GST_BUFFER_COPY_ALL,
|
*buffer = gst_buffer_copy_region (parse->priv->cache, GST_BUFFER_COPY_ALL,
|
||||||
parse->priv->offset - cache_offset, size);
|
parse->priv->offset - cache_offset, size);
|
||||||
GST_BUFFER_OFFSET (*buffer) = parse->priv->offset;
|
GST_BUFFER_OFFSET (*buffer) = parse->priv->offset;
|
||||||
|
GST_LOG_OBJECT (parse,
|
||||||
|
"Satisfying read request of %u bytes from cached buffer with offset %"
|
||||||
|
G_GINT64_FORMAT, size, cache_offset);
|
||||||
return GST_FLOW_OK;
|
return GST_FLOW_OK;
|
||||||
}
|
}
|
||||||
/* not enough data in the cache, free cache and get a new one */
|
/* not enough data in the cache, free cache and get a new one */
|
||||||
|
@ -3381,9 +3400,13 @@ gst_base_parse_pull_range (GstBaseParse * parse, guint size,
|
||||||
}
|
}
|
||||||
|
|
||||||
/* refill the cache */
|
/* refill the cache */
|
||||||
|
size = MAX (64 * 1024, size);
|
||||||
|
GST_LOG_OBJECT (parse,
|
||||||
|
"Reading cache buffer of %u bytes from offset %" G_GINT64_FORMAT,
|
||||||
|
size, parse->priv->offset);
|
||||||
ret =
|
ret =
|
||||||
gst_pad_pull_range (parse->sinkpad, parse->priv->offset, MAX (size,
|
gst_pad_pull_range (parse->sinkpad, parse->priv->offset, size,
|
||||||
64 * 1024), &parse->priv->cache);
|
&parse->priv->cache);
|
||||||
if (ret != GST_FLOW_OK) {
|
if (ret != GST_FLOW_OK) {
|
||||||
parse->priv->cache = NULL;
|
parse->priv->cache = NULL;
|
||||||
return ret;
|
return ret;
|
||||||
|
@ -3400,6 +3423,8 @@ gst_base_parse_pull_range (GstBaseParse * parse, guint size,
|
||||||
return GST_FLOW_OK;
|
return GST_FLOW_OK;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
GST_BUFFER_OFFSET (parse->priv->cache) = parse->priv->offset;
|
||||||
|
|
||||||
*buffer =
|
*buffer =
|
||||||
gst_buffer_copy_region (parse->priv->cache, GST_BUFFER_COPY_ALL, 0, size);
|
gst_buffer_copy_region (parse->priv->cache, GST_BUFFER_COPY_ALL, 0, size);
|
||||||
GST_BUFFER_OFFSET (*buffer) = parse->priv->offset;
|
GST_BUFFER_OFFSET (*buffer) = parse->priv->offset;
|
||||||
|
@ -3486,9 +3511,12 @@ gst_base_parse_scan_frame (GstBaseParse * parse, GstBaseParseClass * klass)
|
||||||
|
|
||||||
/* let's make this efficient for all subclass once and for all;
|
/* let's make this efficient for all subclass once and for all;
|
||||||
* maybe it does not need this much, but in the latter case, we know we are
|
* maybe it does not need this much, but in the latter case, we know we are
|
||||||
* in pull mode here and might as well try to read and supply more anyway
|
* in pull mode here and might as well try to read and supply more anyway,
|
||||||
* (so does the buffer caching mechanism) */
|
* so start with the cached buffer, or if that's shrunk below 1024 bytes,
|
||||||
fsize = 64 * 1024;
|
* pull a new cache buffer */
|
||||||
|
fsize = gst_base_parse_get_cached_available (parse);
|
||||||
|
if (fsize < 1024)
|
||||||
|
fsize = 64 * 1024;
|
||||||
|
|
||||||
while (TRUE) {
|
while (TRUE) {
|
||||||
min_size = MAX (parse->priv->min_frame_size, fsize);
|
min_size = MAX (parse->priv->min_frame_size, fsize);
|
||||||
|
@ -3516,7 +3544,8 @@ gst_base_parse_scan_frame (GstBaseParse * parse, GstBaseParseClass * klass)
|
||||||
GST_ERROR_OBJECT (parse, "Failed to detect format but draining");
|
GST_ERROR_OBJECT (parse, "Failed to detect format but draining");
|
||||||
return GST_FLOW_ERROR;
|
return GST_FLOW_ERROR;
|
||||||
} else {
|
} else {
|
||||||
fsize += 64 * 1024;
|
/* Double our frame size, or increment by at most 64KB */
|
||||||
|
fsize += MIN (fsize, 64 * 1024);
|
||||||
gst_buffer_unref (buffer);
|
gst_buffer_unref (buffer);
|
||||||
continue;
|
continue;
|
||||||
}
|
}
|
||||||
|
@ -3547,18 +3576,20 @@ gst_base_parse_scan_frame (GstBaseParse * parse, GstBaseParseClass * klass)
|
||||||
GST_LOG_OBJECT (parse, "frame finished, breaking loop");
|
GST_LOG_OBJECT (parse, "frame finished, breaking loop");
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
/* nothing flushed, no skip and draining, so nothing left to do */
|
|
||||||
if (!skip && parse->priv->drain) {
|
|
||||||
GST_LOG_OBJECT (parse, "no activity or result when draining; "
|
|
||||||
"breaking loop and marking EOS");
|
|
||||||
ret = GST_FLOW_EOS;
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
/* otherwise, get some more data
|
|
||||||
* note that is checked this does not happen indefinitely */
|
|
||||||
if (!skip) {
|
if (!skip) {
|
||||||
|
if (parse->priv->drain) {
|
||||||
|
/* nothing flushed, no skip and draining, so nothing left to do */
|
||||||
|
GST_LOG_OBJECT (parse, "no activity or result when draining; "
|
||||||
|
"breaking loop and marking EOS");
|
||||||
|
ret = GST_FLOW_EOS;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
/* otherwise, get some more data
|
||||||
|
* note that is checked this does not happen indefinitely */
|
||||||
GST_LOG_OBJECT (parse, "getting some more data");
|
GST_LOG_OBJECT (parse, "getting some more data");
|
||||||
fsize += 64 * 1024;
|
|
||||||
|
/* Double our frame size, or increment by at most 64KB */
|
||||||
|
fsize += MIN (fsize, 64 * 1024);
|
||||||
}
|
}
|
||||||
parse->priv->drain = FALSE;
|
parse->priv->drain = FALSE;
|
||||||
}
|
}
|
||||||
|
|
|
@ -464,12 +464,13 @@ _sink_chain_pull_short_read (GstPad * pad, GstObject * parent,
|
||||||
GstMapInfo map;
|
GstMapInfo map;
|
||||||
gint buffer_size = 0;
|
gint buffer_size = 0;
|
||||||
gint compare_result = 0;
|
gint compare_result = 0;
|
||||||
|
gint64 result_offset = GST_BUFFER_OFFSET (buffer);
|
||||||
|
|
||||||
gst_buffer_map (buffer, &map, GST_MAP_READ);
|
gst_buffer_map (buffer, &map, GST_MAP_READ);
|
||||||
buffer_size = map.size;
|
buffer_size = map.size;
|
||||||
|
|
||||||
fail_unless (current_offset + buffer_size <= raw_buffer_size);
|
fail_unless (result_offset + buffer_size <= raw_buffer_size);
|
||||||
compare_result = memcmp (map.data, &raw_buffer[current_offset], buffer_size);
|
compare_result = memcmp (map.data, &raw_buffer[result_offset], buffer_size);
|
||||||
fail_unless_equals_int (compare_result, 0);
|
fail_unless_equals_int (compare_result, 0);
|
||||||
|
|
||||||
gst_buffer_unmap (buffer, &map);
|
gst_buffer_unmap (buffer, &map);
|
||||||
|
@ -549,7 +550,9 @@ GST_START_TEST (parser_pull_short_read)
|
||||||
g_main_loop_run (loop);
|
g_main_loop_run (loop);
|
||||||
fail_unless_equals_int (have_eos, TRUE);
|
fail_unless_equals_int (have_eos, TRUE);
|
||||||
fail_unless_equals_int (have_data, TRUE);
|
fail_unless_equals_int (have_data, TRUE);
|
||||||
fail_unless_equals_int (buffer_pull_count, buffer_count);
|
/* If the parser asked upstream for buffers more times than buffers were
|
||||||
|
* produced, then something is wrong */
|
||||||
|
fail_unless (buffer_pull_count <= buffer_count);
|
||||||
|
|
||||||
gst_element_set_state (parsetest, GST_STATE_NULL);
|
gst_element_set_state (parsetest, GST_STATE_NULL);
|
||||||
|
|
||||||
|
|
Loading…
Reference in a new issue