aboutsummaryrefslogtreecommitdiffstats
path: root/libswscale/slice.c
diff options
context:
space:
mode:
authorPedro Arthur <bygrandao@gmail.com>2016-06-08 21:38:49 -0300
committerPedro Arthur <bygrandao@gmail.com>2016-06-10 09:45:24 -0300
commite616e9a4b8fde2686fcd283be3929eee3fb2be91 (patch)
tree13482750833a40d52d37532e53321977999bc046 /libswscale/slice.c
parentb5deacfb1fece3406ef0bb790c1614a7096513b4 (diff)
downloadffmpeg-e616e9a4b8fde2686fcd283be3929eee3fb2be91.tar.gz
swscale: fix ring buffer size when scaling slices of a frame
The ring buffer size should be able to store input lines when there is not enough lines to output a single line.
Diffstat (limited to 'libswscale/slice.c')
-rw-r--r--libswscale/slice.c47
1 files changed, 45 insertions, 2 deletions
diff --git a/libswscale/slice.c b/libswscale/slice.c
index 0159a739c3..e14456f75f 100644
--- a/libswscale/slice.c
+++ b/libswscale/slice.c
@@ -210,6 +210,42 @@ static void fill_ones(SwsSlice *s, int n, int is16bit)
}
}
+/*
+ Calculates the minimum ring buffer size, it should be able to store vFilterSize
+ more n lines where n is the max difference between each adjacent slice which
+ outputs a line.
+ The n lines are needed only when there is not enough src lines to output a single
+ dst line, then we should buffer these lines to process them on the next call to scale.
+*/
+static void get_min_buffer_size(SwsContext *c, int *out_lum_size, int *out_chr_size)
+{
+ int lumY;
+ int dstH = c->dstH;
+ int chrDstH = c->chrDstH;
+ int *lumFilterPos = c->vLumFilterPos;
+ int *chrFilterPos = c->vChrFilterPos;
+ int lumFilterSize = c->vLumFilterSize;
+ int chrFilterSize = c->vChrFilterSize;
+ int chrSubSample = c->chrSrcVSubSample;
+
+ *out_lum_size = lumFilterSize;
+ *out_chr_size = chrFilterSize;
+
+ for (lumY = 0; lumY < dstH; lumY++) {
+ int chrY = (int64_t)lumY * chrDstH / dstH;
+ int nextSlice = FFMAX(lumFilterPos[lumY] + lumFilterSize - 1,
+ ((chrFilterPos[chrY] + chrFilterSize - 1)
+ << chrSubSample));
+
+ nextSlice >>= chrSubSample;
+ nextSlice <<= chrSubSample;
+ (*out_lum_size) = FFMAX((*out_lum_size), nextSlice - lumFilterPos[lumY]);
+ (*out_chr_size) = FFMAX((*out_chr_size), (nextSlice >> chrSubSample) - chrFilterPos[chrY]);
+ }
+}
+
+
+
int ff_init_filters(SwsContext * c)
{
int i;
@@ -226,6 +262,13 @@ int ff_init_filters(SwsContext * c)
uint32_t * pal = usePal(c->srcFormat) ? c->pal_yuv : (uint32_t*)c->input_rgb2yuv_table;
int res = 0;
+ int lumBufSize;
+ int chrBufSize;
+
+ get_min_buffer_size(c, &lumBufSize, &chrBufSize);
+ lumBufSize = FFMAX(lumBufSize, c->vLumFilterSize + MAX_LINES_AHEAD);
+ chrBufSize = FFMAX(chrBufSize, c->vChrFilterSize + MAX_LINES_AHEAD);
+
if (c->dstBpc == 16)
dst_stride <<= 1;
@@ -248,13 +291,13 @@ int ff_init_filters(SwsContext * c)
res = alloc_slice(&c->slice[0], c->srcFormat, c->srcH, c->chrSrcH, c->chrSrcHSubSample, c->chrSrcVSubSample, 0);
if (res < 0) goto cleanup;
for (i = 1; i < c->numSlice-2; ++i) {
- res = alloc_slice(&c->slice[i], c->srcFormat, c->vLumFilterSize + MAX_LINES_AHEAD, c->vChrFilterSize + MAX_LINES_AHEAD, c->chrSrcHSubSample, c->chrSrcVSubSample, 0);
+ res = alloc_slice(&c->slice[i], c->srcFormat, lumBufSize, chrBufSize, c->chrSrcHSubSample, c->chrSrcVSubSample, 0);
if (res < 0) goto cleanup;
res = alloc_lines(&c->slice[i], FFALIGN(c->srcW*2+78, 16), c->srcW);
if (res < 0) goto cleanup;
}
// horizontal scaler output
- res = alloc_slice(&c->slice[i], c->srcFormat, c->vLumFilterSize + MAX_LINES_AHEAD, c->vChrFilterSize + MAX_LINES_AHEAD, c->chrDstHSubSample, c->chrDstVSubSample, 1);
+ res = alloc_slice(&c->slice[i], c->srcFormat, lumBufSize, chrBufSize, c->chrDstHSubSample, c->chrDstVSubSample, 1);
if (res < 0) goto cleanup;
res = alloc_lines(&c->slice[i], dst_stride, c->dstW);
if (res < 0) goto cleanup;