xref: /third_party/ffmpeg/libavcodec/vp9recon.c (revision cabdff1a)
1cabdff1aSopenharmony_ci/*
2cabdff1aSopenharmony_ci * VP9 compatible video decoder
3cabdff1aSopenharmony_ci *
4cabdff1aSopenharmony_ci * Copyright (C) 2013 Ronald S. Bultje <rsbultje gmail com>
5cabdff1aSopenharmony_ci * Copyright (C) 2013 Clément Bœsch <u pkh me>
6cabdff1aSopenharmony_ci *
7cabdff1aSopenharmony_ci * This file is part of FFmpeg.
8cabdff1aSopenharmony_ci *
9cabdff1aSopenharmony_ci * FFmpeg is free software; you can redistribute it and/or
10cabdff1aSopenharmony_ci * modify it under the terms of the GNU Lesser General Public
11cabdff1aSopenharmony_ci * License as published by the Free Software Foundation; either
12cabdff1aSopenharmony_ci * version 2.1 of the License, or (at your option) any later version.
13cabdff1aSopenharmony_ci *
14cabdff1aSopenharmony_ci * FFmpeg is distributed in the hope that it will be useful,
15cabdff1aSopenharmony_ci * but WITHOUT ANY WARRANTY; without even the implied warranty of
16cabdff1aSopenharmony_ci * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
17cabdff1aSopenharmony_ci * Lesser General Public License for more details.
18cabdff1aSopenharmony_ci *
19cabdff1aSopenharmony_ci * You should have received a copy of the GNU Lesser General Public
20cabdff1aSopenharmony_ci * License along with FFmpeg; if not, write to the Free Software
21cabdff1aSopenharmony_ci * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA
22cabdff1aSopenharmony_ci */
23cabdff1aSopenharmony_ci
24cabdff1aSopenharmony_ci#include "libavutil/avassert.h"
25cabdff1aSopenharmony_ci#include "libavutil/mem_internal.h"
26cabdff1aSopenharmony_ci
27cabdff1aSopenharmony_ci#include "threadframe.h"
28cabdff1aSopenharmony_ci#include "videodsp.h"
29cabdff1aSopenharmony_ci#include "vp9data.h"
30cabdff1aSopenharmony_ci#include "vp9dec.h"
31cabdff1aSopenharmony_ci
32cabdff1aSopenharmony_cistatic av_always_inline int check_intra_mode(VP9TileData *td, int mode, uint8_t **a,
33cabdff1aSopenharmony_ci                                             uint8_t *dst_edge, ptrdiff_t stride_edge,
34cabdff1aSopenharmony_ci                                             uint8_t *dst_inner, ptrdiff_t stride_inner,
35cabdff1aSopenharmony_ci                                             uint8_t *l, int col, int x, int w,
36cabdff1aSopenharmony_ci                                             int row, int y, enum TxfmMode tx,
37cabdff1aSopenharmony_ci                                             int p, int ss_h, int ss_v, int bytesperpixel)
38cabdff1aSopenharmony_ci{
39cabdff1aSopenharmony_ci    VP9Context *s = td->s;
40cabdff1aSopenharmony_ci    int have_top = row > 0 || y > 0;
41cabdff1aSopenharmony_ci    int have_left = col > td->tile_col_start || x > 0;
42cabdff1aSopenharmony_ci    int have_right = x < w - 1;
43cabdff1aSopenharmony_ci    int bpp = s->s.h.bpp;
44cabdff1aSopenharmony_ci    static const uint8_t mode_conv[10][2 /* have_left */][2 /* have_top */] = {
45cabdff1aSopenharmony_ci        [VERT_PRED]            = { { DC_127_PRED,          VERT_PRED            },
46cabdff1aSopenharmony_ci                                   { DC_127_PRED,          VERT_PRED            } },
47cabdff1aSopenharmony_ci        [HOR_PRED]             = { { DC_129_PRED,          DC_129_PRED          },
48cabdff1aSopenharmony_ci                                   { HOR_PRED,             HOR_PRED             } },
49cabdff1aSopenharmony_ci        [DC_PRED]              = { { DC_128_PRED,          TOP_DC_PRED          },
50cabdff1aSopenharmony_ci                                   { LEFT_DC_PRED,         DC_PRED              } },
51cabdff1aSopenharmony_ci        [DIAG_DOWN_LEFT_PRED]  = { { DC_127_PRED,          DIAG_DOWN_LEFT_PRED  },
52cabdff1aSopenharmony_ci                                   { DC_127_PRED,          DIAG_DOWN_LEFT_PRED  } },
53cabdff1aSopenharmony_ci        [DIAG_DOWN_RIGHT_PRED] = { { DIAG_DOWN_RIGHT_PRED, DIAG_DOWN_RIGHT_PRED },
54cabdff1aSopenharmony_ci                                   { DIAG_DOWN_RIGHT_PRED, DIAG_DOWN_RIGHT_PRED } },
55cabdff1aSopenharmony_ci        [VERT_RIGHT_PRED]      = { { VERT_RIGHT_PRED,      VERT_RIGHT_PRED      },
56cabdff1aSopenharmony_ci                                   { VERT_RIGHT_PRED,      VERT_RIGHT_PRED      } },
57cabdff1aSopenharmony_ci        [HOR_DOWN_PRED]        = { { HOR_DOWN_PRED,        HOR_DOWN_PRED        },
58cabdff1aSopenharmony_ci                                   { HOR_DOWN_PRED,        HOR_DOWN_PRED        } },
59cabdff1aSopenharmony_ci        [VERT_LEFT_PRED]       = { { DC_127_PRED,          VERT_LEFT_PRED       },
60cabdff1aSopenharmony_ci                                   { DC_127_PRED,          VERT_LEFT_PRED       } },
61cabdff1aSopenharmony_ci        [HOR_UP_PRED]          = { { DC_129_PRED,          DC_129_PRED          },
62cabdff1aSopenharmony_ci                                   { HOR_UP_PRED,          HOR_UP_PRED          } },
63cabdff1aSopenharmony_ci        [TM_VP8_PRED]          = { { DC_129_PRED,          VERT_PRED            },
64cabdff1aSopenharmony_ci                                   { HOR_PRED,             TM_VP8_PRED          } },
65cabdff1aSopenharmony_ci    };
66cabdff1aSopenharmony_ci    static const struct {
67cabdff1aSopenharmony_ci        uint8_t needs_left:1;
68cabdff1aSopenharmony_ci        uint8_t needs_top:1;
69cabdff1aSopenharmony_ci        uint8_t needs_topleft:1;
70cabdff1aSopenharmony_ci        uint8_t needs_topright:1;
71cabdff1aSopenharmony_ci        uint8_t invert_left:1;
72cabdff1aSopenharmony_ci    } edges[N_INTRA_PRED_MODES] = {
73cabdff1aSopenharmony_ci        [VERT_PRED]            = { .needs_top  = 1 },
74cabdff1aSopenharmony_ci        [HOR_PRED]             = { .needs_left = 1 },
75cabdff1aSopenharmony_ci        [DC_PRED]              = { .needs_top  = 1, .needs_left = 1 },
76cabdff1aSopenharmony_ci        [DIAG_DOWN_LEFT_PRED]  = { .needs_top  = 1, .needs_topright = 1 },
77cabdff1aSopenharmony_ci        [DIAG_DOWN_RIGHT_PRED] = { .needs_left = 1, .needs_top = 1,
78cabdff1aSopenharmony_ci                                   .needs_topleft = 1 },
79cabdff1aSopenharmony_ci        [VERT_RIGHT_PRED]      = { .needs_left = 1, .needs_top = 1,
80cabdff1aSopenharmony_ci                                   .needs_topleft = 1 },
81cabdff1aSopenharmony_ci        [HOR_DOWN_PRED]        = { .needs_left = 1, .needs_top = 1,
82cabdff1aSopenharmony_ci                                   .needs_topleft = 1 },
83cabdff1aSopenharmony_ci        [VERT_LEFT_PRED]       = { .needs_top  = 1, .needs_topright = 1 },
84cabdff1aSopenharmony_ci        [HOR_UP_PRED]          = { .needs_left = 1, .invert_left = 1 },
85cabdff1aSopenharmony_ci        [TM_VP8_PRED]          = { .needs_left = 1, .needs_top = 1,
86cabdff1aSopenharmony_ci                                   .needs_topleft = 1 },
87cabdff1aSopenharmony_ci        [LEFT_DC_PRED]         = { .needs_left = 1 },
88cabdff1aSopenharmony_ci        [TOP_DC_PRED]          = { .needs_top  = 1 },
89cabdff1aSopenharmony_ci        [DC_128_PRED]          = { 0 },
90cabdff1aSopenharmony_ci        [DC_127_PRED]          = { 0 },
91cabdff1aSopenharmony_ci        [DC_129_PRED]          = { 0 }
92cabdff1aSopenharmony_ci    };
93cabdff1aSopenharmony_ci
94cabdff1aSopenharmony_ci    av_assert2(mode >= 0 && mode < 10);
95cabdff1aSopenharmony_ci    mode = mode_conv[mode][have_left][have_top];
96cabdff1aSopenharmony_ci    if (edges[mode].needs_top) {
97cabdff1aSopenharmony_ci        uint8_t *top, *topleft;
98cabdff1aSopenharmony_ci        int n_px_need = 4 << tx, n_px_have = (((s->cols - col) << !ss_h) - x) * 4;
99cabdff1aSopenharmony_ci        int n_px_need_tr = 0;
100cabdff1aSopenharmony_ci
101cabdff1aSopenharmony_ci        if (tx == TX_4X4 && edges[mode].needs_topright && have_right)
102cabdff1aSopenharmony_ci            n_px_need_tr = 4;
103cabdff1aSopenharmony_ci
104cabdff1aSopenharmony_ci        // if top of sb64-row, use s->intra_pred_data[] instead of
105cabdff1aSopenharmony_ci        // dst[-stride] for intra prediction (it contains pre- instead of
106cabdff1aSopenharmony_ci        // post-loopfilter data)
107cabdff1aSopenharmony_ci        if (have_top) {
108cabdff1aSopenharmony_ci            top = !(row & 7) && !y ?
109cabdff1aSopenharmony_ci                s->intra_pred_data[p] + (col * (8 >> ss_h) + x * 4) * bytesperpixel :
110cabdff1aSopenharmony_ci                y == 0 ? &dst_edge[-stride_edge] : &dst_inner[-stride_inner];
111cabdff1aSopenharmony_ci            if (have_left)
112cabdff1aSopenharmony_ci                topleft = !(row & 7) && !y ?
113cabdff1aSopenharmony_ci                    s->intra_pred_data[p] + (col * (8 >> ss_h) + x * 4) * bytesperpixel :
114cabdff1aSopenharmony_ci                    y == 0 || x == 0 ? &dst_edge[-stride_edge] :
115cabdff1aSopenharmony_ci                    &dst_inner[-stride_inner];
116cabdff1aSopenharmony_ci        }
117cabdff1aSopenharmony_ci
118cabdff1aSopenharmony_ci        if (have_top &&
119cabdff1aSopenharmony_ci            (!edges[mode].needs_topleft || (have_left && top == topleft)) &&
120cabdff1aSopenharmony_ci            (tx != TX_4X4 || !edges[mode].needs_topright || have_right) &&
121cabdff1aSopenharmony_ci            n_px_need + n_px_need_tr <= n_px_have) {
122cabdff1aSopenharmony_ci            *a = top;
123cabdff1aSopenharmony_ci        } else {
124cabdff1aSopenharmony_ci            if (have_top) {
125cabdff1aSopenharmony_ci                if (n_px_need <= n_px_have) {
126cabdff1aSopenharmony_ci                    memcpy(*a, top, n_px_need * bytesperpixel);
127cabdff1aSopenharmony_ci                } else {
128cabdff1aSopenharmony_ci#define memset_bpp(c, i1, v, i2, num) do { \
129cabdff1aSopenharmony_ci    if (bytesperpixel == 1) { \
130cabdff1aSopenharmony_ci        memset(&(c)[(i1)], (v)[(i2)], (num)); \
131cabdff1aSopenharmony_ci    } else { \
132cabdff1aSopenharmony_ci        int n, val = AV_RN16A(&(v)[(i2) * 2]); \
133cabdff1aSopenharmony_ci        for (n = 0; n < (num); n++) { \
134cabdff1aSopenharmony_ci            AV_WN16A(&(c)[((i1) + n) * 2], val); \
135cabdff1aSopenharmony_ci        } \
136cabdff1aSopenharmony_ci    } \
137cabdff1aSopenharmony_ci} while (0)
138cabdff1aSopenharmony_ci                    memcpy(*a, top, n_px_have * bytesperpixel);
139cabdff1aSopenharmony_ci                    memset_bpp(*a, n_px_have, (*a), n_px_have - 1, n_px_need - n_px_have);
140cabdff1aSopenharmony_ci                }
141cabdff1aSopenharmony_ci            } else {
142cabdff1aSopenharmony_ci#define memset_val(c, val, num) do { \
143cabdff1aSopenharmony_ci    if (bytesperpixel == 1) { \
144cabdff1aSopenharmony_ci        memset((c), (val), (num)); \
145cabdff1aSopenharmony_ci    } else { \
146cabdff1aSopenharmony_ci        int n; \
147cabdff1aSopenharmony_ci        for (n = 0; n < (num); n++) { \
148cabdff1aSopenharmony_ci            AV_WN16A(&(c)[n * 2], (val)); \
149cabdff1aSopenharmony_ci        } \
150cabdff1aSopenharmony_ci    } \
151cabdff1aSopenharmony_ci} while (0)
152cabdff1aSopenharmony_ci                memset_val(*a, (128 << (bpp - 8)) - 1, n_px_need);
153cabdff1aSopenharmony_ci            }
154cabdff1aSopenharmony_ci            if (edges[mode].needs_topleft) {
155cabdff1aSopenharmony_ci                if (have_left && have_top) {
156cabdff1aSopenharmony_ci#define assign_bpp(c, i1, v, i2) do { \
157cabdff1aSopenharmony_ci    if (bytesperpixel == 1) { \
158cabdff1aSopenharmony_ci        (c)[(i1)] = (v)[(i2)]; \
159cabdff1aSopenharmony_ci    } else { \
160cabdff1aSopenharmony_ci        AV_COPY16(&(c)[(i1) * 2], &(v)[(i2) * 2]); \
161cabdff1aSopenharmony_ci    } \
162cabdff1aSopenharmony_ci} while (0)
163cabdff1aSopenharmony_ci                    assign_bpp(*a, -1, topleft, -1);
164cabdff1aSopenharmony_ci                } else {
165cabdff1aSopenharmony_ci#define assign_val(c, i, v) do { \
166cabdff1aSopenharmony_ci    if (bytesperpixel == 1) { \
167cabdff1aSopenharmony_ci        (c)[(i)] = (v); \
168cabdff1aSopenharmony_ci    } else { \
169cabdff1aSopenharmony_ci        AV_WN16A(&(c)[(i) * 2], (v)); \
170cabdff1aSopenharmony_ci    } \
171cabdff1aSopenharmony_ci} while (0)
172cabdff1aSopenharmony_ci                    assign_val((*a), -1, (128 << (bpp - 8)) + (have_top ? +1 : -1));
173cabdff1aSopenharmony_ci                }
174cabdff1aSopenharmony_ci            }
175cabdff1aSopenharmony_ci            if (tx == TX_4X4 && edges[mode].needs_topright) {
176cabdff1aSopenharmony_ci                if (have_top && have_right &&
177cabdff1aSopenharmony_ci                    n_px_need + n_px_need_tr <= n_px_have) {
178cabdff1aSopenharmony_ci                    memcpy(&(*a)[4 * bytesperpixel], &top[4 * bytesperpixel], 4 * bytesperpixel);
179cabdff1aSopenharmony_ci                } else {
180cabdff1aSopenharmony_ci                    memset_bpp(*a, 4, *a, 3, 4);
181cabdff1aSopenharmony_ci                }
182cabdff1aSopenharmony_ci            }
183cabdff1aSopenharmony_ci        }
184cabdff1aSopenharmony_ci    }
185cabdff1aSopenharmony_ci    if (edges[mode].needs_left) {
186cabdff1aSopenharmony_ci        if (have_left) {
187cabdff1aSopenharmony_ci            int n_px_need = 4 << tx, i, n_px_have = (((s->rows - row) << !ss_v) - y) * 4;
188cabdff1aSopenharmony_ci            uint8_t *dst = x == 0 ? dst_edge : dst_inner;
189cabdff1aSopenharmony_ci            ptrdiff_t stride = x == 0 ? stride_edge : stride_inner;
190cabdff1aSopenharmony_ci
191cabdff1aSopenharmony_ci            if (edges[mode].invert_left) {
192cabdff1aSopenharmony_ci                if (n_px_need <= n_px_have) {
193cabdff1aSopenharmony_ci                    for (i = 0; i < n_px_need; i++)
194cabdff1aSopenharmony_ci                        assign_bpp(l, i, &dst[i * stride], -1);
195cabdff1aSopenharmony_ci                } else {
196cabdff1aSopenharmony_ci                    for (i = 0; i < n_px_have; i++)
197cabdff1aSopenharmony_ci                        assign_bpp(l, i, &dst[i * stride], -1);
198cabdff1aSopenharmony_ci                    memset_bpp(l, n_px_have, l, n_px_have - 1, n_px_need - n_px_have);
199cabdff1aSopenharmony_ci                }
200cabdff1aSopenharmony_ci            } else {
201cabdff1aSopenharmony_ci                if (n_px_need <= n_px_have) {
202cabdff1aSopenharmony_ci                    for (i = 0; i < n_px_need; i++)
203cabdff1aSopenharmony_ci                        assign_bpp(l, n_px_need - 1 - i, &dst[i * stride], -1);
204cabdff1aSopenharmony_ci                } else {
205cabdff1aSopenharmony_ci                    for (i = 0; i < n_px_have; i++)
206cabdff1aSopenharmony_ci                        assign_bpp(l, n_px_need - 1 - i, &dst[i * stride], -1);
207cabdff1aSopenharmony_ci                    memset_bpp(l, 0, l, n_px_need - n_px_have, n_px_need - n_px_have);
208cabdff1aSopenharmony_ci                }
209cabdff1aSopenharmony_ci            }
210cabdff1aSopenharmony_ci        } else {
211cabdff1aSopenharmony_ci            memset_val(l, (128 << (bpp - 8)) + 1, 4 << tx);
212cabdff1aSopenharmony_ci        }
213cabdff1aSopenharmony_ci    }
214cabdff1aSopenharmony_ci
215cabdff1aSopenharmony_ci    return mode;
216cabdff1aSopenharmony_ci}
217cabdff1aSopenharmony_ci
218cabdff1aSopenharmony_cistatic av_always_inline void intra_recon(VP9TileData *td, ptrdiff_t y_off,
219cabdff1aSopenharmony_ci                                         ptrdiff_t uv_off, int bytesperpixel)
220cabdff1aSopenharmony_ci{
221cabdff1aSopenharmony_ci    VP9Context *s = td->s;
222cabdff1aSopenharmony_ci    VP9Block *b = td->b;
223cabdff1aSopenharmony_ci    int row = td->row, col = td->col;
224cabdff1aSopenharmony_ci    int w4 = ff_vp9_bwh_tab[1][b->bs][0] << 1, step1d = 1 << b->tx, n;
225cabdff1aSopenharmony_ci    int h4 = ff_vp9_bwh_tab[1][b->bs][1] << 1, x, y, step = 1 << (b->tx * 2);
226cabdff1aSopenharmony_ci    int end_x = FFMIN(2 * (s->cols - col), w4);
227cabdff1aSopenharmony_ci    int end_y = FFMIN(2 * (s->rows - row), h4);
228cabdff1aSopenharmony_ci    int tx = 4 * s->s.h.lossless + b->tx, uvtx = b->uvtx + 4 * s->s.h.lossless;
229cabdff1aSopenharmony_ci    int uvstep1d = 1 << b->uvtx, p;
230cabdff1aSopenharmony_ci    uint8_t *dst = td->dst[0], *dst_r = s->s.frames[CUR_FRAME].tf.f->data[0] + y_off;
231cabdff1aSopenharmony_ci    LOCAL_ALIGNED_32(uint8_t, a_buf, [96]);
232cabdff1aSopenharmony_ci    LOCAL_ALIGNED_32(uint8_t, l, [64]);
233cabdff1aSopenharmony_ci
234cabdff1aSopenharmony_ci    for (n = 0, y = 0; y < end_y; y += step1d) {
235cabdff1aSopenharmony_ci        uint8_t *ptr = dst, *ptr_r = dst_r;
236cabdff1aSopenharmony_ci        for (x = 0; x < end_x; x += step1d, ptr += 4 * step1d * bytesperpixel,
237cabdff1aSopenharmony_ci                               ptr_r += 4 * step1d * bytesperpixel, n += step) {
238cabdff1aSopenharmony_ci            int mode = b->mode[b->bs > BS_8x8 && b->tx == TX_4X4 ?
239cabdff1aSopenharmony_ci                               y * 2 + x : 0];
240cabdff1aSopenharmony_ci            uint8_t *a = &a_buf[32];
241cabdff1aSopenharmony_ci            enum TxfmType txtp = ff_vp9_intra_txfm_type[mode];
242cabdff1aSopenharmony_ci            int eob = b->skip ? 0 : b->tx > TX_8X8 ? AV_RN16A(&td->eob[n]) : td->eob[n];
243cabdff1aSopenharmony_ci
244cabdff1aSopenharmony_ci            mode = check_intra_mode(td, mode, &a, ptr_r,
245cabdff1aSopenharmony_ci                                    s->s.frames[CUR_FRAME].tf.f->linesize[0],
246cabdff1aSopenharmony_ci                                    ptr, td->y_stride, l,
247cabdff1aSopenharmony_ci                                    col, x, w4, row, y, b->tx, 0, 0, 0, bytesperpixel);
248cabdff1aSopenharmony_ci            s->dsp.intra_pred[b->tx][mode](ptr, td->y_stride, l, a);
249cabdff1aSopenharmony_ci            if (eob)
250cabdff1aSopenharmony_ci                s->dsp.itxfm_add[tx][txtp](ptr, td->y_stride,
251cabdff1aSopenharmony_ci                                           td->block + 16 * n * bytesperpixel, eob);
252cabdff1aSopenharmony_ci        }
253cabdff1aSopenharmony_ci        dst_r += 4 * step1d * s->s.frames[CUR_FRAME].tf.f->linesize[0];
254cabdff1aSopenharmony_ci        dst   += 4 * step1d * td->y_stride;
255cabdff1aSopenharmony_ci    }
256cabdff1aSopenharmony_ci
257cabdff1aSopenharmony_ci    // U/V
258cabdff1aSopenharmony_ci    w4    >>= s->ss_h;
259cabdff1aSopenharmony_ci    end_x >>= s->ss_h;
260cabdff1aSopenharmony_ci    end_y >>= s->ss_v;
261cabdff1aSopenharmony_ci    step = 1 << (b->uvtx * 2);
262cabdff1aSopenharmony_ci    for (p = 0; p < 2; p++) {
263cabdff1aSopenharmony_ci        dst   = td->dst[1 + p];
264cabdff1aSopenharmony_ci        dst_r = s->s.frames[CUR_FRAME].tf.f->data[1 + p] + uv_off;
265cabdff1aSopenharmony_ci        for (n = 0, y = 0; y < end_y; y += uvstep1d) {
266cabdff1aSopenharmony_ci            uint8_t *ptr = dst, *ptr_r = dst_r;
267cabdff1aSopenharmony_ci            for (x = 0; x < end_x; x += uvstep1d, ptr += 4 * uvstep1d * bytesperpixel,
268cabdff1aSopenharmony_ci                                   ptr_r += 4 * uvstep1d * bytesperpixel, n += step) {
269cabdff1aSopenharmony_ci                int mode = b->uvmode;
270cabdff1aSopenharmony_ci                uint8_t *a = &a_buf[32];
271cabdff1aSopenharmony_ci                int eob = b->skip ? 0 : b->uvtx > TX_8X8 ? AV_RN16A(&td->uveob[p][n]) : td->uveob[p][n];
272cabdff1aSopenharmony_ci
273cabdff1aSopenharmony_ci                mode = check_intra_mode(td, mode, &a, ptr_r,
274cabdff1aSopenharmony_ci                                        s->s.frames[CUR_FRAME].tf.f->linesize[1],
275cabdff1aSopenharmony_ci                                        ptr, td->uv_stride, l, col, x, w4, row, y,
276cabdff1aSopenharmony_ci                                        b->uvtx, p + 1, s->ss_h, s->ss_v, bytesperpixel);
277cabdff1aSopenharmony_ci                s->dsp.intra_pred[b->uvtx][mode](ptr, td->uv_stride, l, a);
278cabdff1aSopenharmony_ci                if (eob)
279cabdff1aSopenharmony_ci                    s->dsp.itxfm_add[uvtx][DCT_DCT](ptr, td->uv_stride,
280cabdff1aSopenharmony_ci                                                    td->uvblock[p] + 16 * n * bytesperpixel, eob);
281cabdff1aSopenharmony_ci            }
282cabdff1aSopenharmony_ci            dst_r += 4 * uvstep1d * s->s.frames[CUR_FRAME].tf.f->linesize[1];
283cabdff1aSopenharmony_ci            dst   += 4 * uvstep1d * td->uv_stride;
284cabdff1aSopenharmony_ci        }
285cabdff1aSopenharmony_ci    }
286cabdff1aSopenharmony_ci}
287cabdff1aSopenharmony_ci
288cabdff1aSopenharmony_civoid ff_vp9_intra_recon_8bpp(VP9TileData *td, ptrdiff_t y_off, ptrdiff_t uv_off)
289cabdff1aSopenharmony_ci{
290cabdff1aSopenharmony_ci    intra_recon(td, y_off, uv_off, 1);
291cabdff1aSopenharmony_ci}
292cabdff1aSopenharmony_ci
293cabdff1aSopenharmony_civoid ff_vp9_intra_recon_16bpp(VP9TileData *td, ptrdiff_t y_off, ptrdiff_t uv_off)
294cabdff1aSopenharmony_ci{
295cabdff1aSopenharmony_ci    intra_recon(td, y_off, uv_off, 2);
296cabdff1aSopenharmony_ci}
297cabdff1aSopenharmony_ci
298cabdff1aSopenharmony_cistatic av_always_inline void mc_luma_unscaled(VP9TileData *td, vp9_mc_func (*mc)[2],
299cabdff1aSopenharmony_ci                                              uint8_t *dst, ptrdiff_t dst_stride,
300cabdff1aSopenharmony_ci                                              const uint8_t *ref, ptrdiff_t ref_stride,
301cabdff1aSopenharmony_ci                                              ThreadFrame *ref_frame,
302cabdff1aSopenharmony_ci                                              ptrdiff_t y, ptrdiff_t x, const VP56mv *mv,
303cabdff1aSopenharmony_ci                                              int bw, int bh, int w, int h, int bytesperpixel)
304cabdff1aSopenharmony_ci{
305cabdff1aSopenharmony_ci    VP9Context *s = td->s;
306cabdff1aSopenharmony_ci    int mx = mv->x, my = mv->y, th;
307cabdff1aSopenharmony_ci
308cabdff1aSopenharmony_ci    y += my >> 3;
309cabdff1aSopenharmony_ci    x += mx >> 3;
310cabdff1aSopenharmony_ci    ref += y * ref_stride + x * bytesperpixel;
311cabdff1aSopenharmony_ci    mx &= 7;
312cabdff1aSopenharmony_ci    my &= 7;
313cabdff1aSopenharmony_ci    // FIXME bilinear filter only needs 0/1 pixels, not 3/4
314cabdff1aSopenharmony_ci    // we use +7 because the last 7 pixels of each sbrow can be changed in
315cabdff1aSopenharmony_ci    // the longest loopfilter of the next sbrow
316cabdff1aSopenharmony_ci    th = (y + bh + 4 * !!my + 7) >> 6;
317cabdff1aSopenharmony_ci    ff_thread_await_progress(ref_frame, FFMAX(th, 0), 0);
318cabdff1aSopenharmony_ci    // The arm/aarch64 _hv filters read one more row than what actually is
319cabdff1aSopenharmony_ci    // needed, so switch to emulated edge one pixel sooner vertically
320cabdff1aSopenharmony_ci    // (!!my * 5) than horizontally (!!mx * 4).
321cabdff1aSopenharmony_ci    if (x < !!mx * 3 || y < !!my * 3 ||
322cabdff1aSopenharmony_ci        x + !!mx * 4 > w - bw || y + !!my * 5 > h - bh) {
323cabdff1aSopenharmony_ci        s->vdsp.emulated_edge_mc(td->edge_emu_buffer,
324cabdff1aSopenharmony_ci                                 ref - !!my * 3 * ref_stride - !!mx * 3 * bytesperpixel,
325cabdff1aSopenharmony_ci                                 160, ref_stride,
326cabdff1aSopenharmony_ci                                 bw + !!mx * 7, bh + !!my * 7,
327cabdff1aSopenharmony_ci                                 x - !!mx * 3, y - !!my * 3, w, h);
328cabdff1aSopenharmony_ci        ref = td->edge_emu_buffer + !!my * 3 * 160 + !!mx * 3 * bytesperpixel;
329cabdff1aSopenharmony_ci        ref_stride = 160;
330cabdff1aSopenharmony_ci    }
331cabdff1aSopenharmony_ci    mc[!!mx][!!my](dst, dst_stride, ref, ref_stride, bh, mx << 1, my << 1);
332cabdff1aSopenharmony_ci}
333cabdff1aSopenharmony_ci
334cabdff1aSopenharmony_cistatic av_always_inline void mc_chroma_unscaled(VP9TileData *td, vp9_mc_func (*mc)[2],
335cabdff1aSopenharmony_ci                                                uint8_t *dst_u, uint8_t *dst_v,
336cabdff1aSopenharmony_ci                                                ptrdiff_t dst_stride,
337cabdff1aSopenharmony_ci                                                const uint8_t *ref_u, ptrdiff_t src_stride_u,
338cabdff1aSopenharmony_ci                                                const uint8_t *ref_v, ptrdiff_t src_stride_v,
339cabdff1aSopenharmony_ci                                                ThreadFrame *ref_frame,
340cabdff1aSopenharmony_ci                                                ptrdiff_t y, ptrdiff_t x, const VP56mv *mv,
341cabdff1aSopenharmony_ci                                                int bw, int bh, int w, int h, int bytesperpixel)
342cabdff1aSopenharmony_ci{
343cabdff1aSopenharmony_ci    VP9Context *s = td->s;
344cabdff1aSopenharmony_ci    int mx = mv->x * (1 << !s->ss_h), my = mv->y * (1 << !s->ss_v), th;
345cabdff1aSopenharmony_ci
346cabdff1aSopenharmony_ci    y += my >> 4;
347cabdff1aSopenharmony_ci    x += mx >> 4;
348cabdff1aSopenharmony_ci    ref_u += y * src_stride_u + x * bytesperpixel;
349cabdff1aSopenharmony_ci    ref_v += y * src_stride_v + x * bytesperpixel;
350cabdff1aSopenharmony_ci    mx &= 15;
351cabdff1aSopenharmony_ci    my &= 15;
352cabdff1aSopenharmony_ci    // FIXME bilinear filter only needs 0/1 pixels, not 3/4
353cabdff1aSopenharmony_ci    // we use +7 because the last 7 pixels of each sbrow can be changed in
354cabdff1aSopenharmony_ci    // the longest loopfilter of the next sbrow
355cabdff1aSopenharmony_ci    th = (y + bh + 4 * !!my + 7) >> (6 - s->ss_v);
356cabdff1aSopenharmony_ci    ff_thread_await_progress(ref_frame, FFMAX(th, 0), 0);
357cabdff1aSopenharmony_ci    // The arm/aarch64 _hv filters read one more row than what actually is
358cabdff1aSopenharmony_ci    // needed, so switch to emulated edge one pixel sooner vertically
359cabdff1aSopenharmony_ci    // (!!my * 5) than horizontally (!!mx * 4).
360cabdff1aSopenharmony_ci    if (x < !!mx * 3 || y < !!my * 3 ||
361cabdff1aSopenharmony_ci        x + !!mx * 4 > w - bw || y + !!my * 5 > h - bh) {
362cabdff1aSopenharmony_ci        s->vdsp.emulated_edge_mc(td->edge_emu_buffer,
363cabdff1aSopenharmony_ci                                 ref_u - !!my * 3 * src_stride_u - !!mx * 3 * bytesperpixel,
364cabdff1aSopenharmony_ci                                 160, src_stride_u,
365cabdff1aSopenharmony_ci                                 bw + !!mx * 7, bh + !!my * 7,
366cabdff1aSopenharmony_ci                                 x - !!mx * 3, y - !!my * 3, w, h);
367cabdff1aSopenharmony_ci        ref_u = td->edge_emu_buffer + !!my * 3 * 160 + !!mx * 3 * bytesperpixel;
368cabdff1aSopenharmony_ci        mc[!!mx][!!my](dst_u, dst_stride, ref_u, 160, bh, mx, my);
369cabdff1aSopenharmony_ci
370cabdff1aSopenharmony_ci        s->vdsp.emulated_edge_mc(td->edge_emu_buffer,
371cabdff1aSopenharmony_ci                                 ref_v - !!my * 3 * src_stride_v - !!mx * 3 * bytesperpixel,
372cabdff1aSopenharmony_ci                                 160, src_stride_v,
373cabdff1aSopenharmony_ci                                 bw + !!mx * 7, bh + !!my * 7,
374cabdff1aSopenharmony_ci                                 x - !!mx * 3, y - !!my * 3, w, h);
375cabdff1aSopenharmony_ci        ref_v = td->edge_emu_buffer + !!my * 3 * 160 + !!mx * 3 * bytesperpixel;
376cabdff1aSopenharmony_ci        mc[!!mx][!!my](dst_v, dst_stride, ref_v, 160, bh, mx, my);
377cabdff1aSopenharmony_ci    } else {
378cabdff1aSopenharmony_ci        mc[!!mx][!!my](dst_u, dst_stride, ref_u, src_stride_u, bh, mx, my);
379cabdff1aSopenharmony_ci        mc[!!mx][!!my](dst_v, dst_stride, ref_v, src_stride_v, bh, mx, my);
380cabdff1aSopenharmony_ci    }
381cabdff1aSopenharmony_ci}
382cabdff1aSopenharmony_ci
383cabdff1aSopenharmony_ci#define mc_luma_dir(td, mc, dst, dst_ls, src, src_ls, tref, row, col, mv, \
384cabdff1aSopenharmony_ci                    px, py, pw, ph, bw, bh, w, h, i) \
385cabdff1aSopenharmony_ci    mc_luma_unscaled(td, s->dsp.mc, dst, dst_ls, src, src_ls, tref, row, col, \
386cabdff1aSopenharmony_ci                     mv, bw, bh, w, h, bytesperpixel)
387cabdff1aSopenharmony_ci#define mc_chroma_dir(td, mc, dstu, dstv, dst_ls, srcu, srcu_ls, srcv, srcv_ls, tref, \
388cabdff1aSopenharmony_ci                      row, col, mv, px, py, pw, ph, bw, bh, w, h, i) \
389cabdff1aSopenharmony_ci    mc_chroma_unscaled(td, s->dsp.mc, dstu, dstv, dst_ls, srcu, srcu_ls, srcv, srcv_ls, tref, \
390cabdff1aSopenharmony_ci                       row, col, mv, bw, bh, w, h, bytesperpixel)
391cabdff1aSopenharmony_ci#define SCALED 0
392cabdff1aSopenharmony_ci#define FN(x) x##_8bpp
393cabdff1aSopenharmony_ci#define BYTES_PER_PIXEL 1
394cabdff1aSopenharmony_ci#include "vp9_mc_template.c"
395cabdff1aSopenharmony_ci#undef FN
396cabdff1aSopenharmony_ci#undef BYTES_PER_PIXEL
397cabdff1aSopenharmony_ci#define FN(x) x##_16bpp
398cabdff1aSopenharmony_ci#define BYTES_PER_PIXEL 2
399cabdff1aSopenharmony_ci#include "vp9_mc_template.c"
400cabdff1aSopenharmony_ci#undef mc_luma_dir
401cabdff1aSopenharmony_ci#undef mc_chroma_dir
402cabdff1aSopenharmony_ci#undef FN
403cabdff1aSopenharmony_ci#undef BYTES_PER_PIXEL
404cabdff1aSopenharmony_ci#undef SCALED
405cabdff1aSopenharmony_ci
406cabdff1aSopenharmony_cistatic av_always_inline void mc_luma_scaled(VP9TileData *td, vp9_scaled_mc_func smc,
407cabdff1aSopenharmony_ci                                            vp9_mc_func (*mc)[2],
408cabdff1aSopenharmony_ci                                            uint8_t *dst, ptrdiff_t dst_stride,
409cabdff1aSopenharmony_ci                                            const uint8_t *ref, ptrdiff_t ref_stride,
410cabdff1aSopenharmony_ci                                            ThreadFrame *ref_frame,
411cabdff1aSopenharmony_ci                                            ptrdiff_t y, ptrdiff_t x, const VP56mv *in_mv,
412cabdff1aSopenharmony_ci                                            int px, int py, int pw, int ph,
413cabdff1aSopenharmony_ci                                            int bw, int bh, int w, int h, int bytesperpixel,
414cabdff1aSopenharmony_ci                                            const uint16_t *scale, const uint8_t *step)
415cabdff1aSopenharmony_ci{
416cabdff1aSopenharmony_ci    VP9Context *s = td->s;
417cabdff1aSopenharmony_ci    if (s->s.frames[CUR_FRAME].tf.f->width == ref_frame->f->width &&
418cabdff1aSopenharmony_ci        s->s.frames[CUR_FRAME].tf.f->height == ref_frame->f->height) {
419cabdff1aSopenharmony_ci        mc_luma_unscaled(td, mc, dst, dst_stride, ref, ref_stride, ref_frame,
420cabdff1aSopenharmony_ci                         y, x, in_mv, bw, bh, w, h, bytesperpixel);
421cabdff1aSopenharmony_ci    } else {
422cabdff1aSopenharmony_ci#define scale_mv(n, dim) (((int64_t)(n) * scale[dim]) >> 14)
423cabdff1aSopenharmony_ci    int mx, my;
424cabdff1aSopenharmony_ci    int refbw_m1, refbh_m1;
425cabdff1aSopenharmony_ci    int th;
426cabdff1aSopenharmony_ci    VP56mv mv;
427cabdff1aSopenharmony_ci
428cabdff1aSopenharmony_ci    mv.x = av_clip(in_mv->x, -(x + pw - px + 4) * 8, (s->cols * 8 - x + px + 3) * 8);
429cabdff1aSopenharmony_ci    mv.y = av_clip(in_mv->y, -(y + ph - py + 4) * 8, (s->rows * 8 - y + py + 3) * 8);
430cabdff1aSopenharmony_ci    // BUG libvpx seems to scale the two components separately. This introduces
431cabdff1aSopenharmony_ci    // rounding errors but we have to reproduce them to be exactly compatible
432cabdff1aSopenharmony_ci    // with the output from libvpx...
433cabdff1aSopenharmony_ci    mx = scale_mv(mv.x * 2, 0) + scale_mv(x * 16, 0);
434cabdff1aSopenharmony_ci    my = scale_mv(mv.y * 2, 1) + scale_mv(y * 16, 1);
435cabdff1aSopenharmony_ci
436cabdff1aSopenharmony_ci    y = my >> 4;
437cabdff1aSopenharmony_ci    x = mx >> 4;
438cabdff1aSopenharmony_ci    ref += y * ref_stride + x * bytesperpixel;
439cabdff1aSopenharmony_ci    mx &= 15;
440cabdff1aSopenharmony_ci    my &= 15;
441cabdff1aSopenharmony_ci    refbw_m1 = ((bw - 1) * step[0] + mx) >> 4;
442cabdff1aSopenharmony_ci    refbh_m1 = ((bh - 1) * step[1] + my) >> 4;
443cabdff1aSopenharmony_ci    // FIXME bilinear filter only needs 0/1 pixels, not 3/4
444cabdff1aSopenharmony_ci    // we use +7 because the last 7 pixels of each sbrow can be changed in
445cabdff1aSopenharmony_ci    // the longest loopfilter of the next sbrow
446cabdff1aSopenharmony_ci    th = (y + refbh_m1 + 4 + 7) >> 6;
447cabdff1aSopenharmony_ci    ff_thread_await_progress(ref_frame, FFMAX(th, 0), 0);
448cabdff1aSopenharmony_ci    // The arm/aarch64 _hv filters read one more row than what actually is
449cabdff1aSopenharmony_ci    // needed, so switch to emulated edge one pixel sooner vertically
450cabdff1aSopenharmony_ci    // (y + 5 >= h - refbh_m1) than horizontally (x + 4 >= w - refbw_m1).
451cabdff1aSopenharmony_ci    if (x < 3 || y < 3 || x + 4 >= w - refbw_m1 || y + 5 >= h - refbh_m1) {
452cabdff1aSopenharmony_ci        s->vdsp.emulated_edge_mc(td->edge_emu_buffer,
453cabdff1aSopenharmony_ci                                 ref - 3 * ref_stride - 3 * bytesperpixel,
454cabdff1aSopenharmony_ci                                 288, ref_stride,
455cabdff1aSopenharmony_ci                                 refbw_m1 + 8, refbh_m1 + 8,
456cabdff1aSopenharmony_ci                                 x - 3, y - 3, w, h);
457cabdff1aSopenharmony_ci        ref = td->edge_emu_buffer + 3 * 288 + 3 * bytesperpixel;
458cabdff1aSopenharmony_ci        ref_stride = 288;
459cabdff1aSopenharmony_ci    }
460cabdff1aSopenharmony_ci    smc(dst, dst_stride, ref, ref_stride, bh, mx, my, step[0], step[1]);
461cabdff1aSopenharmony_ci    }
462cabdff1aSopenharmony_ci}
463cabdff1aSopenharmony_ci
464cabdff1aSopenharmony_cistatic av_always_inline void mc_chroma_scaled(VP9TileData *td, vp9_scaled_mc_func smc,
465cabdff1aSopenharmony_ci                                              vp9_mc_func (*mc)[2],
466cabdff1aSopenharmony_ci                                              uint8_t *dst_u, uint8_t *dst_v,
467cabdff1aSopenharmony_ci                                              ptrdiff_t dst_stride,
468cabdff1aSopenharmony_ci                                              const uint8_t *ref_u, ptrdiff_t src_stride_u,
469cabdff1aSopenharmony_ci                                              const uint8_t *ref_v, ptrdiff_t src_stride_v,
470cabdff1aSopenharmony_ci                                              ThreadFrame *ref_frame,
471cabdff1aSopenharmony_ci                                              ptrdiff_t y, ptrdiff_t x, const VP56mv *in_mv,
472cabdff1aSopenharmony_ci                                              int px, int py, int pw, int ph,
473cabdff1aSopenharmony_ci                                              int bw, int bh, int w, int h, int bytesperpixel,
474cabdff1aSopenharmony_ci                                              const uint16_t *scale, const uint8_t *step)
475cabdff1aSopenharmony_ci{
476cabdff1aSopenharmony_ci    VP9Context *s = td->s;
477cabdff1aSopenharmony_ci    if (s->s.frames[CUR_FRAME].tf.f->width == ref_frame->f->width &&
478cabdff1aSopenharmony_ci        s->s.frames[CUR_FRAME].tf.f->height == ref_frame->f->height) {
479cabdff1aSopenharmony_ci        mc_chroma_unscaled(td, mc, dst_u, dst_v, dst_stride, ref_u, src_stride_u,
480cabdff1aSopenharmony_ci                           ref_v, src_stride_v, ref_frame,
481cabdff1aSopenharmony_ci                           y, x, in_mv, bw, bh, w, h, bytesperpixel);
482cabdff1aSopenharmony_ci    } else {
483cabdff1aSopenharmony_ci    int mx, my;
484cabdff1aSopenharmony_ci    int refbw_m1, refbh_m1;
485cabdff1aSopenharmony_ci    int th;
486cabdff1aSopenharmony_ci    VP56mv mv;
487cabdff1aSopenharmony_ci
488cabdff1aSopenharmony_ci    if (s->ss_h) {
489cabdff1aSopenharmony_ci        // BUG https://code.google.com/p/webm/issues/detail?id=820
490cabdff1aSopenharmony_ci        mv.x = av_clip(in_mv->x, -(x + pw - px + 4) * 16, (s->cols * 4 - x + px + 3) * 16);
491cabdff1aSopenharmony_ci        mx = scale_mv(mv.x, 0) + (scale_mv(x * 16, 0) & ~15) + (scale_mv(x * 32, 0) & 15);
492cabdff1aSopenharmony_ci    } else {
493cabdff1aSopenharmony_ci        mv.x = av_clip(in_mv->x, -(x + pw - px + 4) * 8, (s->cols * 8 - x + px + 3) * 8);
494cabdff1aSopenharmony_ci        mx = scale_mv(mv.x * 2, 0) + scale_mv(x * 16, 0);
495cabdff1aSopenharmony_ci    }
496cabdff1aSopenharmony_ci    if (s->ss_v) {
497cabdff1aSopenharmony_ci        // BUG https://code.google.com/p/webm/issues/detail?id=820
498cabdff1aSopenharmony_ci        mv.y = av_clip(in_mv->y, -(y + ph - py + 4) * 16, (s->rows * 4 - y + py + 3) * 16);
499cabdff1aSopenharmony_ci        my = scale_mv(mv.y, 1) + (scale_mv(y * 16, 1) & ~15) + (scale_mv(y * 32, 1) & 15);
500cabdff1aSopenharmony_ci    } else {
501cabdff1aSopenharmony_ci        mv.y = av_clip(in_mv->y, -(y + ph - py + 4) * 8, (s->rows * 8 - y + py + 3) * 8);
502cabdff1aSopenharmony_ci        my = scale_mv(mv.y * 2, 1) + scale_mv(y * 16, 1);
503cabdff1aSopenharmony_ci    }
504cabdff1aSopenharmony_ci#undef scale_mv
505cabdff1aSopenharmony_ci    y = my >> 4;
506cabdff1aSopenharmony_ci    x = mx >> 4;
507cabdff1aSopenharmony_ci    ref_u += y * src_stride_u + x * bytesperpixel;
508cabdff1aSopenharmony_ci    ref_v += y * src_stride_v + x * bytesperpixel;
509cabdff1aSopenharmony_ci    mx &= 15;
510cabdff1aSopenharmony_ci    my &= 15;
511cabdff1aSopenharmony_ci    refbw_m1 = ((bw - 1) * step[0] + mx) >> 4;
512cabdff1aSopenharmony_ci    refbh_m1 = ((bh - 1) * step[1] + my) >> 4;
513cabdff1aSopenharmony_ci    // FIXME bilinear filter only needs 0/1 pixels, not 3/4
514cabdff1aSopenharmony_ci    // we use +7 because the last 7 pixels of each sbrow can be changed in
515cabdff1aSopenharmony_ci    // the longest loopfilter of the next sbrow
516cabdff1aSopenharmony_ci    th = (y + refbh_m1 + 4 + 7) >> (6 - s->ss_v);
517cabdff1aSopenharmony_ci    ff_thread_await_progress(ref_frame, FFMAX(th, 0), 0);
518cabdff1aSopenharmony_ci    // The arm/aarch64 _hv filters read one more row than what actually is
519cabdff1aSopenharmony_ci    // needed, so switch to emulated edge one pixel sooner vertically
520cabdff1aSopenharmony_ci    // (y + 5 >= h - refbh_m1) than horizontally (x + 4 >= w - refbw_m1).
521cabdff1aSopenharmony_ci    if (x < 3 || y < 3 || x + 4 >= w - refbw_m1 || y + 5 >= h - refbh_m1) {
522cabdff1aSopenharmony_ci        s->vdsp.emulated_edge_mc(td->edge_emu_buffer,
523cabdff1aSopenharmony_ci                                 ref_u - 3 * src_stride_u - 3 * bytesperpixel,
524cabdff1aSopenharmony_ci                                 288, src_stride_u,
525cabdff1aSopenharmony_ci                                 refbw_m1 + 8, refbh_m1 + 8,
526cabdff1aSopenharmony_ci                                 x - 3, y - 3, w, h);
527cabdff1aSopenharmony_ci        ref_u = td->edge_emu_buffer + 3 * 288 + 3 * bytesperpixel;
528cabdff1aSopenharmony_ci        smc(dst_u, dst_stride, ref_u, 288, bh, mx, my, step[0], step[1]);
529cabdff1aSopenharmony_ci
530cabdff1aSopenharmony_ci        s->vdsp.emulated_edge_mc(td->edge_emu_buffer,
531cabdff1aSopenharmony_ci                                 ref_v - 3 * src_stride_v - 3 * bytesperpixel,
532cabdff1aSopenharmony_ci                                 288, src_stride_v,
533cabdff1aSopenharmony_ci                                 refbw_m1 + 8, refbh_m1 + 8,
534cabdff1aSopenharmony_ci                                 x - 3, y - 3, w, h);
535cabdff1aSopenharmony_ci        ref_v = td->edge_emu_buffer + 3 * 288 + 3 * bytesperpixel;
536cabdff1aSopenharmony_ci        smc(dst_v, dst_stride, ref_v, 288, bh, mx, my, step[0], step[1]);
537cabdff1aSopenharmony_ci    } else {
538cabdff1aSopenharmony_ci        smc(dst_u, dst_stride, ref_u, src_stride_u, bh, mx, my, step[0], step[1]);
539cabdff1aSopenharmony_ci        smc(dst_v, dst_stride, ref_v, src_stride_v, bh, mx, my, step[0], step[1]);
540cabdff1aSopenharmony_ci    }
541cabdff1aSopenharmony_ci    }
542cabdff1aSopenharmony_ci}
543cabdff1aSopenharmony_ci
544cabdff1aSopenharmony_ci#define mc_luma_dir(td, mc, dst, dst_ls, src, src_ls, tref, row, col, mv, \
545cabdff1aSopenharmony_ci                    px, py, pw, ph, bw, bh, w, h, i) \
546cabdff1aSopenharmony_ci    mc_luma_scaled(td, s->dsp.s##mc, s->dsp.mc, dst, dst_ls, src, src_ls, tref, row, col, \
547cabdff1aSopenharmony_ci                   mv, px, py, pw, ph, bw, bh, w, h, bytesperpixel, \
548cabdff1aSopenharmony_ci                   s->mvscale[b->ref[i]], s->mvstep[b->ref[i]])
549cabdff1aSopenharmony_ci#define mc_chroma_dir(td, mc, dstu, dstv, dst_ls, srcu, srcu_ls, srcv, srcv_ls, tref, \
550cabdff1aSopenharmony_ci                      row, col, mv, px, py, pw, ph, bw, bh, w, h, i) \
551cabdff1aSopenharmony_ci    mc_chroma_scaled(td, s->dsp.s##mc, s->dsp.mc, dstu, dstv, dst_ls, srcu, srcu_ls, srcv, srcv_ls, tref, \
552cabdff1aSopenharmony_ci                     row, col, mv, px, py, pw, ph, bw, bh, w, h, bytesperpixel, \
553cabdff1aSopenharmony_ci                     s->mvscale[b->ref[i]], s->mvstep[b->ref[i]])
554cabdff1aSopenharmony_ci#define SCALED 1
555cabdff1aSopenharmony_ci#define FN(x) x##_scaled_8bpp
556cabdff1aSopenharmony_ci#define BYTES_PER_PIXEL 1
557cabdff1aSopenharmony_ci#include "vp9_mc_template.c"
558cabdff1aSopenharmony_ci#undef FN
559cabdff1aSopenharmony_ci#undef BYTES_PER_PIXEL
560cabdff1aSopenharmony_ci#define FN(x) x##_scaled_16bpp
561cabdff1aSopenharmony_ci#define BYTES_PER_PIXEL 2
562cabdff1aSopenharmony_ci#include "vp9_mc_template.c"
563cabdff1aSopenharmony_ci#undef mc_luma_dir
564cabdff1aSopenharmony_ci#undef mc_chroma_dir
565cabdff1aSopenharmony_ci#undef FN
566cabdff1aSopenharmony_ci#undef BYTES_PER_PIXEL
567cabdff1aSopenharmony_ci#undef SCALED
568cabdff1aSopenharmony_ci
569cabdff1aSopenharmony_cistatic av_always_inline void inter_recon(VP9TileData *td, int bytesperpixel)
570cabdff1aSopenharmony_ci{
571cabdff1aSopenharmony_ci    VP9Context *s = td->s;
572cabdff1aSopenharmony_ci    VP9Block *b = td->b;
573cabdff1aSopenharmony_ci    int row = td->row, col = td->col;
574cabdff1aSopenharmony_ci
575cabdff1aSopenharmony_ci    if (s->mvscale[b->ref[0]][0] == REF_INVALID_SCALE ||
576cabdff1aSopenharmony_ci        (b->comp && s->mvscale[b->ref[1]][0] == REF_INVALID_SCALE)) {
577cabdff1aSopenharmony_ci        if (!s->td->error_info) {
578cabdff1aSopenharmony_ci            s->td->error_info = AVERROR_INVALIDDATA;
579cabdff1aSopenharmony_ci            av_log(NULL, AV_LOG_ERROR, "Bitstream not supported, "
580cabdff1aSopenharmony_ci                                       "reference frame has invalid dimensions\n");
581cabdff1aSopenharmony_ci        }
582cabdff1aSopenharmony_ci        return;
583cabdff1aSopenharmony_ci    }
584cabdff1aSopenharmony_ci
585cabdff1aSopenharmony_ci    if (s->mvscale[b->ref[0]][0] || (b->comp && s->mvscale[b->ref[1]][0])) {
586cabdff1aSopenharmony_ci        if (bytesperpixel == 1) {
587cabdff1aSopenharmony_ci            inter_pred_scaled_8bpp(td);
588cabdff1aSopenharmony_ci        } else {
589cabdff1aSopenharmony_ci            inter_pred_scaled_16bpp(td);
590cabdff1aSopenharmony_ci        }
591cabdff1aSopenharmony_ci    } else {
592cabdff1aSopenharmony_ci        if (bytesperpixel == 1) {
593cabdff1aSopenharmony_ci            inter_pred_8bpp(td);
594cabdff1aSopenharmony_ci        } else {
595cabdff1aSopenharmony_ci            inter_pred_16bpp(td);
596cabdff1aSopenharmony_ci        }
597cabdff1aSopenharmony_ci    }
598cabdff1aSopenharmony_ci
599cabdff1aSopenharmony_ci    if (!b->skip) {
600cabdff1aSopenharmony_ci        /* mostly copied intra_recon() */
601cabdff1aSopenharmony_ci
602cabdff1aSopenharmony_ci        int w4 = ff_vp9_bwh_tab[1][b->bs][0] << 1, step1d = 1 << b->tx, n;
603cabdff1aSopenharmony_ci        int h4 = ff_vp9_bwh_tab[1][b->bs][1] << 1, x, y, step = 1 << (b->tx * 2);
604cabdff1aSopenharmony_ci        int end_x = FFMIN(2 * (s->cols - col), w4);
605cabdff1aSopenharmony_ci        int end_y = FFMIN(2 * (s->rows - row), h4);
606cabdff1aSopenharmony_ci        int tx = 4 * s->s.h.lossless + b->tx, uvtx = b->uvtx + 4 * s->s.h.lossless;
607cabdff1aSopenharmony_ci        int uvstep1d = 1 << b->uvtx, p;
608cabdff1aSopenharmony_ci        uint8_t *dst = td->dst[0];
609cabdff1aSopenharmony_ci
610cabdff1aSopenharmony_ci        // y itxfm add
611cabdff1aSopenharmony_ci        for (n = 0, y = 0; y < end_y; y += step1d) {
612cabdff1aSopenharmony_ci            uint8_t *ptr = dst;
613cabdff1aSopenharmony_ci            for (x = 0; x < end_x; x += step1d,
614cabdff1aSopenharmony_ci                 ptr += 4 * step1d * bytesperpixel, n += step) {
615cabdff1aSopenharmony_ci                int eob = b->tx > TX_8X8 ? AV_RN16A(&td->eob[n]) : td->eob[n];
616cabdff1aSopenharmony_ci
617cabdff1aSopenharmony_ci                if (eob)
618cabdff1aSopenharmony_ci                    s->dsp.itxfm_add[tx][DCT_DCT](ptr, td->y_stride,
619cabdff1aSopenharmony_ci                                                  td->block + 16 * n * bytesperpixel, eob);
620cabdff1aSopenharmony_ci            }
621cabdff1aSopenharmony_ci            dst += 4 * td->y_stride * step1d;
622cabdff1aSopenharmony_ci        }
623cabdff1aSopenharmony_ci
624cabdff1aSopenharmony_ci        // uv itxfm add
625cabdff1aSopenharmony_ci        end_x >>= s->ss_h;
626cabdff1aSopenharmony_ci        end_y >>= s->ss_v;
627cabdff1aSopenharmony_ci        step = 1 << (b->uvtx * 2);
628cabdff1aSopenharmony_ci        for (p = 0; p < 2; p++) {
629cabdff1aSopenharmony_ci            dst = td->dst[p + 1];
630cabdff1aSopenharmony_ci            for (n = 0, y = 0; y < end_y; y += uvstep1d) {
631cabdff1aSopenharmony_ci                uint8_t *ptr = dst;
632cabdff1aSopenharmony_ci                for (x = 0; x < end_x; x += uvstep1d,
633cabdff1aSopenharmony_ci                     ptr += 4 * uvstep1d * bytesperpixel, n += step) {
634cabdff1aSopenharmony_ci                    int eob = b->uvtx > TX_8X8 ? AV_RN16A(&td->uveob[p][n]) : td->uveob[p][n];
635cabdff1aSopenharmony_ci
636cabdff1aSopenharmony_ci                    if (eob)
637cabdff1aSopenharmony_ci                        s->dsp.itxfm_add[uvtx][DCT_DCT](ptr, td->uv_stride,
638cabdff1aSopenharmony_ci                                                        td->uvblock[p] + 16 * n * bytesperpixel, eob);
639cabdff1aSopenharmony_ci                }
640cabdff1aSopenharmony_ci                dst += 4 * uvstep1d * td->uv_stride;
641cabdff1aSopenharmony_ci            }
642cabdff1aSopenharmony_ci        }
643cabdff1aSopenharmony_ci    }
644cabdff1aSopenharmony_ci}
645cabdff1aSopenharmony_ci
646cabdff1aSopenharmony_civoid ff_vp9_inter_recon_8bpp(VP9TileData *td)
647cabdff1aSopenharmony_ci{
648cabdff1aSopenharmony_ci    inter_recon(td, 1);
649cabdff1aSopenharmony_ci}
650cabdff1aSopenharmony_ci
651cabdff1aSopenharmony_civoid ff_vp9_inter_recon_16bpp(VP9TileData *td)
652cabdff1aSopenharmony_ci{
653cabdff1aSopenharmony_ci    inter_recon(td, 2);
654cabdff1aSopenharmony_ci}
655