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