diff options
author | Michael Niedermayer <michaelni@gmx.at> | 2013-02-16 09:03:58 +0100 |
---|---|---|
committer | Michael Niedermayer <michaelni@gmx.at> | 2013-02-16 09:09:38 +0100 |
commit | 446f7c62a2d1358a276212300d09f9af88f9a228 (patch) | |
tree | dd764077643dfd0d023a94cd82cee99823d97ebb /libavfilter/vf_yadif.c | |
parent | fab8a89abeae6bab5d2556d0116ca270ff6da5ed (diff) | |
parent | 64ed397635ef2666b0ca0c8d8c60a8bc44581d82 (diff) | |
download | ffmpeg-446f7c62a2d1358a276212300d09f9af88f9a228.tar.gz |
Merge commit '64ed397635ef2666b0ca0c8d8c60a8bc44581d82'
* commit '64ed397635ef2666b0ca0c8d8c60a8bc44581d82':
vf_yadif: fix out-of line reads
Conflicts:
libavfilter/vf_yadif.c
tests/ref/fate/filter-yadif-mode0
tests/ref/fate/filter-yadif-mode1
Merged-by: Michael Niedermayer <michaelni@gmx.at>
Diffstat (limited to 'libavfilter/vf_yadif.c')
-rw-r--r-- | libavfilter/vf_yadif.c | 125 |
1 files changed, 88 insertions, 37 deletions
diff --git a/libavfilter/vf_yadif.c b/libavfilter/vf_yadif.c index 9ac1f3e548..fb17531dbf 100644 --- a/libavfilter/vf_yadif.c +++ b/libavfilter/vf_yadif.c @@ -34,15 +34,15 @@ #define PERM_RWP AV_PERM_WRITE | AV_PERM_PRESERVE | AV_PERM_REUSE #define CHECK(j)\ - { int score = FFABS(cur[mrefs-1+(j)] - cur[prefs-1-(j)])\ + { int score = FFABS(cur[mrefs + off_left + (j)] - cur[prefs + off_left - (j)])\ + FFABS(cur[mrefs +(j)] - cur[prefs -(j)])\ - + FFABS(cur[mrefs+1+(j)] - cur[prefs+1-(j)]);\ + + FFABS(cur[mrefs + off_right + (j)] - cur[prefs + off_right - (j)]);\ if (score < spatial_score) {\ spatial_score= score;\ spatial_pred= (cur[mrefs +(j)] + cur[prefs -(j)])>>1;\ -#define FILTER \ - for (x = 0; x < w; x++) { \ +#define FILTER(start, end) \ + for (x = start; x < end; x++) { \ int c = cur[mrefs]; \ int d = (prev2[0] + next2[0])>>1; \ int e = cur[prefs]; \ @@ -51,11 +51,15 @@ int temporal_diff2 =(FFABS(next[mrefs] - c) + FFABS(next[prefs] - e) )>>1; \ int diff = FFMAX3(temporal_diff0 >> 1, temporal_diff1, temporal_diff2); \ int spatial_pred = (c+e) >> 1; \ - int spatial_score = FFABS(cur[mrefs - 1] - cur[prefs - 1]) + FFABS(c-e) \ - + FFABS(cur[mrefs + 1] - cur[prefs + 1]) - 1; \ + int off_right = (x < w - 1) ? 1 : -1;\ + int off_left = x ? -1 : 1;\ + int spatial_score = FFABS(cur[mrefs + off_left] - cur[prefs + off_left]) + FFABS(c-e) \ + + FFABS(cur[mrefs + off_right] - cur[prefs + off_right]) - 1; \ \ - CHECK(-1) CHECK(-2) }} }} \ - CHECK( 1) CHECK( 2) }} }} \ + if (x > 2 && x < w - 3) {\ + CHECK(-1) CHECK(-2) }} }} \ + CHECK( 1) CHECK( 2) }} }} \ + }\ \ if (mode < 2) { \ int b = (prev2[2 * mrefs] + next2[2 * mrefs])>>1; \ @@ -93,9 +97,34 @@ static void filter_line_c(void *dst1, uint8_t *prev2 = parity ? prev : cur ; uint8_t *next2 = parity ? cur : next; - FILTER + FILTER(0, w) } +static void filter_edges(void *dst1, void *prev1, void *cur1, void *next1, + int w, int prefs, int mrefs, int parity, int mode, + int l_edge) +{ + uint8_t *dst = dst1; + uint8_t *prev = prev1; + uint8_t *cur = cur1; + uint8_t *next = next1; + int x; + uint8_t *prev2 = parity ? prev : cur ; + uint8_t *next2 = parity ? cur : next; + + FILTER(0, l_edge) + + dst = (uint8_t*)dst1 + w - 3; + prev = (uint8_t*)prev1 + w - 3; + cur = (uint8_t*)cur1 + w - 3; + next = (uint8_t*)next1 + w - 3; + prev2 = (uint8_t*)(parity ? prev : cur); + next2 = (uint8_t*)(parity ? cur : next); + + FILTER(w - 3, w) +} + + static void filter_line_c_16bit(void *dst1, void *prev1, void *cur1, void *next1, int w, int prefs, int mrefs, int parity, @@ -111,7 +140,31 @@ static void filter_line_c_16bit(void *dst1, mrefs /= 2; prefs /= 2; - FILTER + FILTER(0, w) +} + +static void filter_edges_16bit(void *dst1, void *prev1, void *cur1, void *next1, + int w, int prefs, int mrefs, int parity, int mode, + int l_edge) +{ + uint16_t *dst = dst1; + uint16_t *prev = prev1; + uint16_t *cur = cur1; + uint16_t *next = next1; + int x; + uint16_t *prev2 = parity ? prev : cur ; + uint16_t *next2 = parity ? cur : next; + + FILTER(0, l_edge) + + dst = (uint16_t*)dst1 + w - 3; + prev = (uint16_t*)prev1 + w - 3; + cur = (uint16_t*)cur1 + w - 3; + next = (uint16_t*)next1 + w - 3; + prev2 = (uint16_t*)(parity ? prev : cur); + next2 = (uint16_t*)(parity ? cur : next); + + FILTER(w - 3, w) } static void filter(AVFilterContext *ctx, AVFilterBufferRef *dstpic, @@ -126,6 +179,7 @@ static void filter(AVFilterContext *ctx, AVFilterBufferRef *dstpic, int refs = yadif->cur->linesize[i]; int absrefs = FFABS(refs); int df = (yadif->csp->comp[i].depth_minus1 + 8) / 8; + int l_edge, l_edge_pix; if (i == 1 || i == 2) { /* Why is this not part of the per-plane description thing? */ @@ -133,11 +187,11 @@ static void filter(AVFilterContext *ctx, AVFilterBufferRef *dstpic, h >>= yadif->csp->log2_chroma_h; } - if(yadif->temp_line_size < absrefs) { - av_free(yadif->temp_line); - yadif->temp_line = av_mallocz(2*64 + 5*absrefs); - yadif->temp_line_size = absrefs; - } + /* filtering reads 3 pixels to the left/right; to avoid invalid reads, + * we need to call the c variant which avoids this for border pixels + */ + l_edge = yadif->req_align; + l_edge_pix = l_edge / df; for (y = 0; y < h; y++) { if ((y ^ parity) & 1) { @@ -146,26 +200,22 @@ static void filter(AVFilterContext *ctx, AVFilterBufferRef *dstpic, uint8_t *next = &yadif->next->data[i][y * refs]; uint8_t *dst = &dstpic->data[i][y * dstpic->linesize[i]]; int mode = y == 1 || y + 2 == h ? 2 : yadif->mode; - int prefs = y+1<h ? refs : -refs; - int mrefs = y ?-refs : refs; - - if(y<=1 || y+2>=h) { - uint8_t *tmp = yadif->temp_line + 64 + 2*absrefs; - if(mode<2) - memcpy(tmp+2*mrefs, cur+2*mrefs, w*df); - memcpy(tmp+mrefs, cur+mrefs, w*df); - memcpy(tmp , cur , w*df); - if(prefs != mrefs) { - memcpy(tmp+prefs, cur+prefs, w*df); - if(mode<2) - memcpy(tmp+2*prefs, cur+2*prefs, w*df); - } - cur = tmp; + if (yadif->req_align) { + yadif->filter_line(dst + l_edge, prev + l_edge, cur + l_edge, + next + l_edge, w - l_edge_pix - 3, + y + 1 < h ? refs : -refs, + y ? -refs : refs, + parity ^ tff, mode); + yadif->filter_edges(dst, prev, cur, next, w, + y + 1 < h ? refs : -refs, + y ? -refs : refs, + parity ^ tff, mode, l_edge_pix); + } else { + yadif->filter_line(dst, prev, cur, next + l_edge, w, + y + 1 < h ? refs : -refs, + y ? -refs : refs, + parity ^ tff, mode); } - - yadif->filter_line(dst, prev, cur, next, w, - prefs, mrefs, - parity ^ tff, mode); } else { memcpy(&dstpic->data[i][y * dstpic->linesize[i]], &yadif->cur->data[i][y * refs], w * df); @@ -333,7 +383,6 @@ static av_cold void uninit(AVFilterContext *ctx) avfilter_unref_bufferp(&yadif->prev); avfilter_unref_bufferp(&yadif->cur ); avfilter_unref_bufferp(&yadif->next); - av_freep(&yadif->temp_line); yadif->temp_line_size = 0; av_opt_free(yadif); } @@ -416,9 +465,11 @@ static int config_props(AVFilterLink *link) s->csp = av_pix_fmt_desc_get(link->format); if (s->csp->comp[0].depth_minus1 / 8 == 1) { - s->filter_line = filter_line_c_16bit; + s->filter_line = filter_line_c_16bit; + s->filter_edges = filter_edges_16bit; } else { - s->filter_line = filter_line_c; + s->filter_line = filter_line_c; + s->filter_edges = filter_edges; if (ARCH_X86) ff_yadif_init_x86(s); |