1cabdff1aSopenharmony_ci/*
2cabdff1aSopenharmony_ci * VC-1 and WMV3 decoder
3cabdff1aSopenharmony_ci * Copyright (c) 2011 Mashiat Sarker Shakkhar
4cabdff1aSopenharmony_ci * Copyright (c) 2006-2007 Konstantin Shishkov
5cabdff1aSopenharmony_ci * Partly based on vc9.c (c) 2005 Anonymous, Alex Beregszaszi, Michael Niedermayer
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/**
25cabdff1aSopenharmony_ci * @file
26cabdff1aSopenharmony_ci * VC-1 and WMV3 block decoding routines
27cabdff1aSopenharmony_ci */
28cabdff1aSopenharmony_ci
29cabdff1aSopenharmony_ci#include "avcodec.h"
30cabdff1aSopenharmony_ci#include "h264chroma.h"
31cabdff1aSopenharmony_ci#include "mathops.h"
32cabdff1aSopenharmony_ci#include "mpegvideo.h"
33cabdff1aSopenharmony_ci#include "vc1.h"
34cabdff1aSopenharmony_ci
35cabdff1aSopenharmony_cistatic av_always_inline void vc1_scale_luma(uint8_t *srcY,
36cabdff1aSopenharmony_ci                                            int k, int linesize)
37cabdff1aSopenharmony_ci{
38cabdff1aSopenharmony_ci    int i, j;
39cabdff1aSopenharmony_ci    for (j = 0; j < k; j++) {
40cabdff1aSopenharmony_ci        for (i = 0; i < k; i++)
41cabdff1aSopenharmony_ci            srcY[i] = ((srcY[i] - 128) >> 1) + 128;
42cabdff1aSopenharmony_ci        srcY += linesize;
43cabdff1aSopenharmony_ci    }
44cabdff1aSopenharmony_ci}
45cabdff1aSopenharmony_ci
46cabdff1aSopenharmony_cistatic av_always_inline void vc1_scale_chroma(uint8_t *srcU, uint8_t *srcV,
47cabdff1aSopenharmony_ci                                              int k, int uvlinesize)
48cabdff1aSopenharmony_ci{
49cabdff1aSopenharmony_ci    int i, j;
50cabdff1aSopenharmony_ci    for (j = 0; j < k; j++) {
51cabdff1aSopenharmony_ci        for (i = 0; i < k; i++) {
52cabdff1aSopenharmony_ci            srcU[i] = ((srcU[i] - 128) >> 1) + 128;
53cabdff1aSopenharmony_ci            srcV[i] = ((srcV[i] - 128) >> 1) + 128;
54cabdff1aSopenharmony_ci        }
55cabdff1aSopenharmony_ci        srcU += uvlinesize;
56cabdff1aSopenharmony_ci        srcV += uvlinesize;
57cabdff1aSopenharmony_ci    }
58cabdff1aSopenharmony_ci}
59cabdff1aSopenharmony_ci
60cabdff1aSopenharmony_cistatic av_always_inline void vc1_lut_scale_luma(uint8_t *srcY,
61cabdff1aSopenharmony_ci                                                uint8_t *lut1, uint8_t *lut2,
62cabdff1aSopenharmony_ci                                                int k, int linesize)
63cabdff1aSopenharmony_ci{
64cabdff1aSopenharmony_ci    int i, j;
65cabdff1aSopenharmony_ci
66cabdff1aSopenharmony_ci    for (j = 0; j < k; j += 2) {
67cabdff1aSopenharmony_ci        for (i = 0; i < k; i++)
68cabdff1aSopenharmony_ci            srcY[i] = lut1[srcY[i]];
69cabdff1aSopenharmony_ci        srcY += linesize;
70cabdff1aSopenharmony_ci
71cabdff1aSopenharmony_ci        if (j + 1 == k)
72cabdff1aSopenharmony_ci            break;
73cabdff1aSopenharmony_ci
74cabdff1aSopenharmony_ci        for (i = 0; i < k; i++)
75cabdff1aSopenharmony_ci            srcY[i] = lut2[srcY[i]];
76cabdff1aSopenharmony_ci        srcY += linesize;
77cabdff1aSopenharmony_ci    }
78cabdff1aSopenharmony_ci}
79cabdff1aSopenharmony_ci
80cabdff1aSopenharmony_cistatic av_always_inline void vc1_lut_scale_chroma(uint8_t *srcU, uint8_t *srcV,
81cabdff1aSopenharmony_ci                                                  uint8_t *lut1, uint8_t *lut2,
82cabdff1aSopenharmony_ci                                                  int k, int uvlinesize)
83cabdff1aSopenharmony_ci{
84cabdff1aSopenharmony_ci    int i, j;
85cabdff1aSopenharmony_ci
86cabdff1aSopenharmony_ci    for (j = 0; j < k; j += 2) {
87cabdff1aSopenharmony_ci        for (i = 0; i < k; i++) {
88cabdff1aSopenharmony_ci            srcU[i] = lut1[srcU[i]];
89cabdff1aSopenharmony_ci            srcV[i] = lut1[srcV[i]];
90cabdff1aSopenharmony_ci        }
91cabdff1aSopenharmony_ci        srcU += uvlinesize;
92cabdff1aSopenharmony_ci        srcV += uvlinesize;
93cabdff1aSopenharmony_ci
94cabdff1aSopenharmony_ci        if (j + 1 == k)
95cabdff1aSopenharmony_ci            break;
96cabdff1aSopenharmony_ci
97cabdff1aSopenharmony_ci        for (i = 0; i < k; i++) {
98cabdff1aSopenharmony_ci            srcU[i] = lut2[srcU[i]];
99cabdff1aSopenharmony_ci            srcV[i] = lut2[srcV[i]];
100cabdff1aSopenharmony_ci        }
101cabdff1aSopenharmony_ci        srcU += uvlinesize;
102cabdff1aSopenharmony_ci        srcV += uvlinesize;
103cabdff1aSopenharmony_ci    }
104cabdff1aSopenharmony_ci}
105cabdff1aSopenharmony_ci
106cabdff1aSopenharmony_cistatic const uint8_t popcount4[16] = { 0, 1, 1, 2, 1, 2, 2, 3, 1, 2, 2, 3, 2, 3, 3, 4 };
107cabdff1aSopenharmony_ci
108cabdff1aSopenharmony_cistatic av_always_inline int get_luma_mv(VC1Context *v, int dir, int16_t *tx, int16_t *ty)
109cabdff1aSopenharmony_ci{
110cabdff1aSopenharmony_ci    MpegEncContext *s = &v->s;
111cabdff1aSopenharmony_ci    int idx = v->mv_f[dir][s->block_index[0] + v->blocks_off] |
112cabdff1aSopenharmony_ci             (v->mv_f[dir][s->block_index[1] + v->blocks_off] << 1) |
113cabdff1aSopenharmony_ci             (v->mv_f[dir][s->block_index[2] + v->blocks_off] << 2) |
114cabdff1aSopenharmony_ci             (v->mv_f[dir][s->block_index[3] + v->blocks_off] << 3);
115cabdff1aSopenharmony_ci    static const uint8_t index2[16] = { 0, 0, 0, 0x23, 0, 0x13, 0x03, 0, 0, 0x12, 0x02, 0, 0x01, 0, 0, 0 };
116cabdff1aSopenharmony_ci    int opp_count = popcount4[idx];
117cabdff1aSopenharmony_ci
118cabdff1aSopenharmony_ci    switch (opp_count) {
119cabdff1aSopenharmony_ci    case 0:
120cabdff1aSopenharmony_ci    case 4:
121cabdff1aSopenharmony_ci        *tx = median4(s->mv[dir][0][0], s->mv[dir][1][0], s->mv[dir][2][0], s->mv[dir][3][0]);
122cabdff1aSopenharmony_ci        *ty = median4(s->mv[dir][0][1], s->mv[dir][1][1], s->mv[dir][2][1], s->mv[dir][3][1]);
123cabdff1aSopenharmony_ci        break;
124cabdff1aSopenharmony_ci    case 1:
125cabdff1aSopenharmony_ci        *tx = mid_pred(s->mv[dir][idx < 2][0], s->mv[dir][1 + (idx < 4)][0], s->mv[dir][2 + (idx < 8)][0]);
126cabdff1aSopenharmony_ci        *ty = mid_pred(s->mv[dir][idx < 2][1], s->mv[dir][1 + (idx < 4)][1], s->mv[dir][2 + (idx < 8)][1]);
127cabdff1aSopenharmony_ci        break;
128cabdff1aSopenharmony_ci    case 3:
129cabdff1aSopenharmony_ci        *tx = mid_pred(s->mv[dir][idx > 0xd][0], s->mv[dir][1 + (idx > 0xb)][0], s->mv[dir][2 + (idx > 0x7)][0]);
130cabdff1aSopenharmony_ci        *ty = mid_pred(s->mv[dir][idx > 0xd][1], s->mv[dir][1 + (idx > 0xb)][1], s->mv[dir][2 + (idx > 0x7)][1]);
131cabdff1aSopenharmony_ci        break;
132cabdff1aSopenharmony_ci    case 2:
133cabdff1aSopenharmony_ci        *tx = (s->mv[dir][index2[idx] >> 4][0] + s->mv[dir][index2[idx] & 0xf][0]) / 2;
134cabdff1aSopenharmony_ci        *ty = (s->mv[dir][index2[idx] >> 4][1] + s->mv[dir][index2[idx] & 0xf][1]) / 2;
135cabdff1aSopenharmony_ci        break;
136cabdff1aSopenharmony_ci    }
137cabdff1aSopenharmony_ci    return opp_count;
138cabdff1aSopenharmony_ci}
139cabdff1aSopenharmony_ci
140cabdff1aSopenharmony_cistatic av_always_inline int get_chroma_mv(VC1Context *v, int dir, int16_t *tx, int16_t *ty)
141cabdff1aSopenharmony_ci{
142cabdff1aSopenharmony_ci    MpegEncContext *s = &v->s;
143cabdff1aSopenharmony_ci    int idx = !v->mb_type[0][s->block_index[0]] |
144cabdff1aSopenharmony_ci             (!v->mb_type[0][s->block_index[1]] << 1) |
145cabdff1aSopenharmony_ci             (!v->mb_type[0][s->block_index[2]] << 2) |
146cabdff1aSopenharmony_ci             (!v->mb_type[0][s->block_index[3]] << 3);
147cabdff1aSopenharmony_ci    static const uint8_t index2[16] = { 0, 0, 0, 0x01, 0, 0x02, 0x12, 0, 0, 0x03, 0x13, 0, 0x23, 0, 0, 0 };
148cabdff1aSopenharmony_ci    int valid_count = popcount4[idx];
149cabdff1aSopenharmony_ci
150cabdff1aSopenharmony_ci    switch (valid_count) {
151cabdff1aSopenharmony_ci    case 4:
152cabdff1aSopenharmony_ci        *tx = median4(s->mv[dir][0][0], s->mv[dir][1][0], s->mv[dir][2][0], s->mv[dir][3][0]);
153cabdff1aSopenharmony_ci        *ty = median4(s->mv[dir][0][1], s->mv[dir][1][1], s->mv[dir][2][1], s->mv[dir][3][1]);
154cabdff1aSopenharmony_ci        break;
155cabdff1aSopenharmony_ci    case 3:
156cabdff1aSopenharmony_ci        *tx = mid_pred(s->mv[dir][idx > 0xd][0], s->mv[dir][1 + (idx > 0xb)][0], s->mv[dir][2 + (idx > 0x7)][0]);
157cabdff1aSopenharmony_ci        *ty = mid_pred(s->mv[dir][idx > 0xd][1], s->mv[dir][1 + (idx > 0xb)][1], s->mv[dir][2 + (idx > 0x7)][1]);
158cabdff1aSopenharmony_ci        break;
159cabdff1aSopenharmony_ci    case 2:
160cabdff1aSopenharmony_ci        *tx = (s->mv[dir][index2[idx] >> 4][0] + s->mv[dir][index2[idx] & 0xf][0]) / 2;
161cabdff1aSopenharmony_ci        *ty = (s->mv[dir][index2[idx] >> 4][1] + s->mv[dir][index2[idx] & 0xf][1]) / 2;
162cabdff1aSopenharmony_ci        break;
163cabdff1aSopenharmony_ci    default:
164cabdff1aSopenharmony_ci        return 0;
165cabdff1aSopenharmony_ci    }
166cabdff1aSopenharmony_ci    return valid_count;
167cabdff1aSopenharmony_ci}
168cabdff1aSopenharmony_ci
169cabdff1aSopenharmony_ci/** Do motion compensation over 1 macroblock
170cabdff1aSopenharmony_ci * Mostly adapted hpel_motion and qpel_motion from mpegvideo.c
171cabdff1aSopenharmony_ci */
172cabdff1aSopenharmony_civoid ff_vc1_mc_1mv(VC1Context *v, int dir)
173cabdff1aSopenharmony_ci{
174cabdff1aSopenharmony_ci    MpegEncContext *s = &v->s;
175cabdff1aSopenharmony_ci    H264ChromaContext *h264chroma = &v->h264chroma;
176cabdff1aSopenharmony_ci    uint8_t *srcY, *srcU, *srcV;
177cabdff1aSopenharmony_ci    int dxy, mx, my, uvmx, uvmy, src_x, src_y, uvsrc_x, uvsrc_y;
178cabdff1aSopenharmony_ci    int v_edge_pos = s->v_edge_pos >> v->field_mode;
179cabdff1aSopenharmony_ci    int i;
180cabdff1aSopenharmony_ci    uint8_t (*luty)[256], (*lutuv)[256];
181cabdff1aSopenharmony_ci    int use_ic;
182cabdff1aSopenharmony_ci    int interlace;
183cabdff1aSopenharmony_ci    int linesize, uvlinesize;
184cabdff1aSopenharmony_ci
185cabdff1aSopenharmony_ci    if ((!v->field_mode ||
186cabdff1aSopenharmony_ci         (v->ref_field_type[dir] == 1 && v->cur_field_type == 1)) &&
187cabdff1aSopenharmony_ci        !v->s.last_picture.f->data[0])
188cabdff1aSopenharmony_ci        return;
189cabdff1aSopenharmony_ci
190cabdff1aSopenharmony_ci    linesize = s->current_picture_ptr->f->linesize[0];
191cabdff1aSopenharmony_ci    uvlinesize = s->current_picture_ptr->f->linesize[1];
192cabdff1aSopenharmony_ci
193cabdff1aSopenharmony_ci    mx = s->mv[dir][0][0];
194cabdff1aSopenharmony_ci    my = s->mv[dir][0][1];
195cabdff1aSopenharmony_ci
196cabdff1aSopenharmony_ci    // store motion vectors for further use in B-frames
197cabdff1aSopenharmony_ci    if (s->pict_type == AV_PICTURE_TYPE_P) {
198cabdff1aSopenharmony_ci        for (i = 0; i < 4; i++) {
199cabdff1aSopenharmony_ci            s->current_picture.motion_val[1][s->block_index[i] + v->blocks_off][0] = mx;
200cabdff1aSopenharmony_ci            s->current_picture.motion_val[1][s->block_index[i] + v->blocks_off][1] = my;
201cabdff1aSopenharmony_ci        }
202cabdff1aSopenharmony_ci    }
203cabdff1aSopenharmony_ci
204cabdff1aSopenharmony_ci    uvmx = (mx + ((mx & 3) == 3)) >> 1;
205cabdff1aSopenharmony_ci    uvmy = (my + ((my & 3) == 3)) >> 1;
206cabdff1aSopenharmony_ci    v->luma_mv[s->mb_x][0] = uvmx;
207cabdff1aSopenharmony_ci    v->luma_mv[s->mb_x][1] = uvmy;
208cabdff1aSopenharmony_ci
209cabdff1aSopenharmony_ci    if (v->field_mode &&
210cabdff1aSopenharmony_ci        v->cur_field_type != v->ref_field_type[dir]) {
211cabdff1aSopenharmony_ci        my   = my   - 2 + 4 * v->cur_field_type;
212cabdff1aSopenharmony_ci        uvmy = uvmy - 2 + 4 * v->cur_field_type;
213cabdff1aSopenharmony_ci    }
214cabdff1aSopenharmony_ci
215cabdff1aSopenharmony_ci    // fastuvmc shall be ignored for interlaced frame picture
216cabdff1aSopenharmony_ci    if (v->fastuvmc && (v->fcm != ILACE_FRAME)) {
217cabdff1aSopenharmony_ci        uvmx = uvmx + ((uvmx < 0) ? (uvmx & 1) : -(uvmx & 1));
218cabdff1aSopenharmony_ci        uvmy = uvmy + ((uvmy < 0) ? (uvmy & 1) : -(uvmy & 1));
219cabdff1aSopenharmony_ci    }
220cabdff1aSopenharmony_ci    if (!dir) {
221cabdff1aSopenharmony_ci        if (v->field_mode && (v->cur_field_type != v->ref_field_type[dir]) && v->second_field) {
222cabdff1aSopenharmony_ci            srcY = s->current_picture.f->data[0];
223cabdff1aSopenharmony_ci            srcU = s->current_picture.f->data[1];
224cabdff1aSopenharmony_ci            srcV = s->current_picture.f->data[2];
225cabdff1aSopenharmony_ci            luty  = v->curr_luty;
226cabdff1aSopenharmony_ci            lutuv = v->curr_lutuv;
227cabdff1aSopenharmony_ci            use_ic = *v->curr_use_ic;
228cabdff1aSopenharmony_ci            interlace = 1;
229cabdff1aSopenharmony_ci        } else {
230cabdff1aSopenharmony_ci            srcY = s->last_picture.f->data[0];
231cabdff1aSopenharmony_ci            srcU = s->last_picture.f->data[1];
232cabdff1aSopenharmony_ci            srcV = s->last_picture.f->data[2];
233cabdff1aSopenharmony_ci            luty  = v->last_luty;
234cabdff1aSopenharmony_ci            lutuv = v->last_lutuv;
235cabdff1aSopenharmony_ci            use_ic = v->last_use_ic;
236cabdff1aSopenharmony_ci            interlace = s->last_picture.f->interlaced_frame;
237cabdff1aSopenharmony_ci        }
238cabdff1aSopenharmony_ci    } else {
239cabdff1aSopenharmony_ci        srcY = s->next_picture.f->data[0];
240cabdff1aSopenharmony_ci        srcU = s->next_picture.f->data[1];
241cabdff1aSopenharmony_ci        srcV = s->next_picture.f->data[2];
242cabdff1aSopenharmony_ci        luty  = v->next_luty;
243cabdff1aSopenharmony_ci        lutuv = v->next_lutuv;
244cabdff1aSopenharmony_ci        use_ic = v->next_use_ic;
245cabdff1aSopenharmony_ci        interlace = s->next_picture.f->interlaced_frame;
246cabdff1aSopenharmony_ci    }
247cabdff1aSopenharmony_ci
248cabdff1aSopenharmony_ci    if (!srcY || !srcU) {
249cabdff1aSopenharmony_ci        av_log(v->s.avctx, AV_LOG_ERROR, "Referenced frame missing.\n");
250cabdff1aSopenharmony_ci        return;
251cabdff1aSopenharmony_ci    }
252cabdff1aSopenharmony_ci
253cabdff1aSopenharmony_ci    src_x   = s->mb_x * 16 + (mx   >> 2);
254cabdff1aSopenharmony_ci    src_y   = s->mb_y * 16 + (my   >> 2);
255cabdff1aSopenharmony_ci    uvsrc_x = s->mb_x *  8 + (uvmx >> 2);
256cabdff1aSopenharmony_ci    uvsrc_y = s->mb_y *  8 + (uvmy >> 2);
257cabdff1aSopenharmony_ci
258cabdff1aSopenharmony_ci    if (v->profile != PROFILE_ADVANCED) {
259cabdff1aSopenharmony_ci        src_x   = av_clip(  src_x, -16, s->mb_width  * 16);
260cabdff1aSopenharmony_ci        src_y   = av_clip(  src_y, -16, s->mb_height * 16);
261cabdff1aSopenharmony_ci        uvsrc_x = av_clip(uvsrc_x,  -8, s->mb_width  *  8);
262cabdff1aSopenharmony_ci        uvsrc_y = av_clip(uvsrc_y,  -8, s->mb_height *  8);
263cabdff1aSopenharmony_ci    } else {
264cabdff1aSopenharmony_ci        src_x   = av_clip(  src_x, -17, s->avctx->coded_width);
265cabdff1aSopenharmony_ci        uvsrc_x = av_clip(uvsrc_x,  -8, s->avctx->coded_width  >> 1);
266cabdff1aSopenharmony_ci        if (v->fcm == ILACE_FRAME) {
267cabdff1aSopenharmony_ci            src_y = av_clip(src_y, -18 + (src_y & 1), s->avctx->coded_height + (src_y & 1));
268cabdff1aSopenharmony_ci            uvsrc_y = av_clip(uvsrc_y, -8 + (uvsrc_y & 1), (s->avctx->coded_height >> 1) + (uvsrc_y & 1));
269cabdff1aSopenharmony_ci        } else {
270cabdff1aSopenharmony_ci            src_y = av_clip(src_y, -18, s->avctx->coded_height + 1);
271cabdff1aSopenharmony_ci            uvsrc_y = av_clip(uvsrc_y, -8, s->avctx->coded_height >> 1);
272cabdff1aSopenharmony_ci        }
273cabdff1aSopenharmony_ci    }
274cabdff1aSopenharmony_ci
275cabdff1aSopenharmony_ci    srcY += src_y   * s->linesize   + src_x;
276cabdff1aSopenharmony_ci    srcU += uvsrc_y * s->uvlinesize + uvsrc_x;
277cabdff1aSopenharmony_ci    srcV += uvsrc_y * s->uvlinesize + uvsrc_x;
278cabdff1aSopenharmony_ci
279cabdff1aSopenharmony_ci    if (v->field_mode && v->ref_field_type[dir]) {
280cabdff1aSopenharmony_ci        srcY += linesize;
281cabdff1aSopenharmony_ci        srcU += uvlinesize;
282cabdff1aSopenharmony_ci        srcV += uvlinesize;
283cabdff1aSopenharmony_ci    }
284cabdff1aSopenharmony_ci
285cabdff1aSopenharmony_ci    /* for grayscale we should not try to read from unknown area */
286cabdff1aSopenharmony_ci    if (CONFIG_GRAY && s->avctx->flags & AV_CODEC_FLAG_GRAY) {
287cabdff1aSopenharmony_ci        srcU = s->sc.edge_emu_buffer + 18 * s->linesize;
288cabdff1aSopenharmony_ci        srcV = s->sc.edge_emu_buffer + 18 * s->linesize;
289cabdff1aSopenharmony_ci    }
290cabdff1aSopenharmony_ci
291cabdff1aSopenharmony_ci    if (v->rangeredfrm || use_ic
292cabdff1aSopenharmony_ci        || s->h_edge_pos < 22 || v_edge_pos < 22
293cabdff1aSopenharmony_ci        || (unsigned)(src_x - s->mspel) > s->h_edge_pos - (mx&3) - 16 - s->mspel * 3
294cabdff1aSopenharmony_ci        || (unsigned)(src_y - 1)        > v_edge_pos    - (my&3) - 16 - 3) {
295cabdff1aSopenharmony_ci        uint8_t *ubuf = s->sc.edge_emu_buffer + 19 * s->linesize;
296cabdff1aSopenharmony_ci        uint8_t *vbuf = ubuf + 9 * s->uvlinesize;
297cabdff1aSopenharmony_ci        const int k = 17 + s->mspel * 2;
298cabdff1aSopenharmony_ci
299cabdff1aSopenharmony_ci        srcY -= s->mspel * (1 + s->linesize);
300cabdff1aSopenharmony_ci        if (interlace) {
301cabdff1aSopenharmony_ci            s->vdsp.emulated_edge_mc(s->sc.edge_emu_buffer,
302cabdff1aSopenharmony_ci                                     srcY,
303cabdff1aSopenharmony_ci                                     linesize << 1,
304cabdff1aSopenharmony_ci                                     linesize << 1,
305cabdff1aSopenharmony_ci                                     k,
306cabdff1aSopenharmony_ci                                     v->field_mode ? k : k + 1 >> 1,
307cabdff1aSopenharmony_ci                                     src_x - s->mspel,
308cabdff1aSopenharmony_ci                                     src_y - s->mspel >> !v->field_mode,
309cabdff1aSopenharmony_ci                                     s->h_edge_pos,
310cabdff1aSopenharmony_ci                                     s->v_edge_pos >> 1);
311cabdff1aSopenharmony_ci            if (!v->field_mode)
312cabdff1aSopenharmony_ci                s->vdsp.emulated_edge_mc(s->sc.edge_emu_buffer + linesize,
313cabdff1aSopenharmony_ci                                         srcY + linesize,
314cabdff1aSopenharmony_ci                                         linesize << 1,
315cabdff1aSopenharmony_ci                                         linesize << 1,
316cabdff1aSopenharmony_ci                                         k,
317cabdff1aSopenharmony_ci                                         k >> 1,
318cabdff1aSopenharmony_ci                                         src_x - s->mspel,
319cabdff1aSopenharmony_ci                                         src_y - s->mspel + 1 >> 1,
320cabdff1aSopenharmony_ci                                         s->h_edge_pos,
321cabdff1aSopenharmony_ci                                         s->v_edge_pos >> 1);
322cabdff1aSopenharmony_ci        } else
323cabdff1aSopenharmony_ci            s->vdsp.emulated_edge_mc(s->sc.edge_emu_buffer,
324cabdff1aSopenharmony_ci                                     srcY,
325cabdff1aSopenharmony_ci                                     linesize,
326cabdff1aSopenharmony_ci                                     linesize,
327cabdff1aSopenharmony_ci                                     k,
328cabdff1aSopenharmony_ci                                     v->field_mode ? (k << 1) - 1 : k,
329cabdff1aSopenharmony_ci                                     src_x - s->mspel,
330cabdff1aSopenharmony_ci                                     v->field_mode ? 2 * (src_y - s->mspel) + v->ref_field_type[dir] :
331cabdff1aSopenharmony_ci                                                     src_y - s->mspel,
332cabdff1aSopenharmony_ci                                     s->h_edge_pos,
333cabdff1aSopenharmony_ci                                     s->v_edge_pos);
334cabdff1aSopenharmony_ci        srcY = s->sc.edge_emu_buffer;
335cabdff1aSopenharmony_ci        if (interlace) {
336cabdff1aSopenharmony_ci            s->vdsp.emulated_edge_mc(ubuf,
337cabdff1aSopenharmony_ci                                     srcU,
338cabdff1aSopenharmony_ci                                     uvlinesize << 1,
339cabdff1aSopenharmony_ci                                     uvlinesize << 1,
340cabdff1aSopenharmony_ci                                     9,
341cabdff1aSopenharmony_ci                                     v->field_mode ? 9 : 5,
342cabdff1aSopenharmony_ci                                     uvsrc_x,
343cabdff1aSopenharmony_ci                                     uvsrc_y >> !v->field_mode,
344cabdff1aSopenharmony_ci                                     s->h_edge_pos >> 1,
345cabdff1aSopenharmony_ci                                     s->v_edge_pos >> 2);
346cabdff1aSopenharmony_ci            s->vdsp.emulated_edge_mc(vbuf,
347cabdff1aSopenharmony_ci                                     srcV,
348cabdff1aSopenharmony_ci                                     uvlinesize << 1,
349cabdff1aSopenharmony_ci                                     uvlinesize << 1,
350cabdff1aSopenharmony_ci                                     9,
351cabdff1aSopenharmony_ci                                     v->field_mode ? 9 : 5,
352cabdff1aSopenharmony_ci                                     uvsrc_x,
353cabdff1aSopenharmony_ci                                     uvsrc_y >> !v->field_mode,
354cabdff1aSopenharmony_ci                                     s->h_edge_pos >> 1,
355cabdff1aSopenharmony_ci                                     s->v_edge_pos >> 2);
356cabdff1aSopenharmony_ci            if (!v->field_mode) {
357cabdff1aSopenharmony_ci                s->vdsp.emulated_edge_mc(ubuf + uvlinesize,
358cabdff1aSopenharmony_ci                                         srcU + uvlinesize,
359cabdff1aSopenharmony_ci                                         uvlinesize << 1,
360cabdff1aSopenharmony_ci                                         uvlinesize << 1,
361cabdff1aSopenharmony_ci                                         9,
362cabdff1aSopenharmony_ci                                         4,
363cabdff1aSopenharmony_ci                                         uvsrc_x,
364cabdff1aSopenharmony_ci                                         uvsrc_y + 1 >> 1,
365cabdff1aSopenharmony_ci                                         s->h_edge_pos >> 1,
366cabdff1aSopenharmony_ci                                         s->v_edge_pos >> 2);
367cabdff1aSopenharmony_ci                s->vdsp.emulated_edge_mc(vbuf + uvlinesize,
368cabdff1aSopenharmony_ci                                         srcV + uvlinesize,
369cabdff1aSopenharmony_ci                                         uvlinesize << 1,
370cabdff1aSopenharmony_ci                                         uvlinesize << 1,
371cabdff1aSopenharmony_ci                                         9,
372cabdff1aSopenharmony_ci                                         4,
373cabdff1aSopenharmony_ci                                         uvsrc_x,
374cabdff1aSopenharmony_ci                                         uvsrc_y + 1 >> 1,
375cabdff1aSopenharmony_ci                                         s->h_edge_pos >> 1,
376cabdff1aSopenharmony_ci                                         s->v_edge_pos >> 2);
377cabdff1aSopenharmony_ci            }
378cabdff1aSopenharmony_ci        } else {
379cabdff1aSopenharmony_ci            s->vdsp.emulated_edge_mc(ubuf,
380cabdff1aSopenharmony_ci                                     srcU,
381cabdff1aSopenharmony_ci                                     uvlinesize,
382cabdff1aSopenharmony_ci                                     uvlinesize,
383cabdff1aSopenharmony_ci                                     9,
384cabdff1aSopenharmony_ci                                     v->field_mode ? 17 : 9,
385cabdff1aSopenharmony_ci                                     uvsrc_x,
386cabdff1aSopenharmony_ci                                     v->field_mode ? 2 * uvsrc_y + v->ref_field_type[dir] : uvsrc_y,
387cabdff1aSopenharmony_ci                                     s->h_edge_pos >> 1,
388cabdff1aSopenharmony_ci                                     s->v_edge_pos >> 1);
389cabdff1aSopenharmony_ci            s->vdsp.emulated_edge_mc(vbuf,
390cabdff1aSopenharmony_ci                                     srcV,
391cabdff1aSopenharmony_ci                                     uvlinesize,
392cabdff1aSopenharmony_ci                                     uvlinesize,
393cabdff1aSopenharmony_ci                                     9,
394cabdff1aSopenharmony_ci                                     v->field_mode ? 17 : 9,
395cabdff1aSopenharmony_ci                                     uvsrc_x,
396cabdff1aSopenharmony_ci                                     v->field_mode ? 2 * uvsrc_y + v->ref_field_type[dir] : uvsrc_y,
397cabdff1aSopenharmony_ci                                     s->h_edge_pos >> 1,
398cabdff1aSopenharmony_ci                                     s->v_edge_pos >> 1);
399cabdff1aSopenharmony_ci        }
400cabdff1aSopenharmony_ci        srcU = ubuf;
401cabdff1aSopenharmony_ci        srcV = vbuf;
402cabdff1aSopenharmony_ci        /* if we deal with range reduction we need to scale source blocks */
403cabdff1aSopenharmony_ci        if (v->rangeredfrm) {
404cabdff1aSopenharmony_ci            vc1_scale_luma(srcY, k, s->linesize);
405cabdff1aSopenharmony_ci            vc1_scale_chroma(srcU, srcV, 9, s->uvlinesize);
406cabdff1aSopenharmony_ci        }
407cabdff1aSopenharmony_ci        /* if we deal with intensity compensation we need to scale source blocks */
408cabdff1aSopenharmony_ci        if (use_ic) {
409cabdff1aSopenharmony_ci            vc1_lut_scale_luma(srcY,
410cabdff1aSopenharmony_ci                               luty[v->field_mode ? v->ref_field_type[dir] : ((0 + src_y - s->mspel) & 1)],
411cabdff1aSopenharmony_ci                               luty[v->field_mode ? v->ref_field_type[dir] : ((1 + src_y - s->mspel) & 1)],
412cabdff1aSopenharmony_ci                               k, s->linesize);
413cabdff1aSopenharmony_ci            vc1_lut_scale_chroma(srcU, srcV,
414cabdff1aSopenharmony_ci                                 lutuv[v->field_mode ? v->ref_field_type[dir] : ((0 + uvsrc_y) & 1)],
415cabdff1aSopenharmony_ci                                 lutuv[v->field_mode ? v->ref_field_type[dir] : ((1 + uvsrc_y) & 1)],
416cabdff1aSopenharmony_ci                                 9, s->uvlinesize);
417cabdff1aSopenharmony_ci        }
418cabdff1aSopenharmony_ci        srcY += s->mspel * (1 + s->linesize);
419cabdff1aSopenharmony_ci    }
420cabdff1aSopenharmony_ci
421cabdff1aSopenharmony_ci    if (s->mspel) {
422cabdff1aSopenharmony_ci        dxy = ((my & 3) << 2) | (mx & 3);
423cabdff1aSopenharmony_ci        v->vc1dsp.put_vc1_mspel_pixels_tab[0][dxy](s->dest[0], srcY, s->linesize, v->rnd);
424cabdff1aSopenharmony_ci    } else { // hpel mc - always used for luma
425cabdff1aSopenharmony_ci        dxy = (my & 2) | ((mx & 2) >> 1);
426cabdff1aSopenharmony_ci        if (!v->rnd)
427cabdff1aSopenharmony_ci            s->hdsp.put_pixels_tab[0][dxy](s->dest[0], srcY, s->linesize, 16);
428cabdff1aSopenharmony_ci        else
429cabdff1aSopenharmony_ci            s->hdsp.put_no_rnd_pixels_tab[0][dxy](s->dest[0], srcY, s->linesize, 16);
430cabdff1aSopenharmony_ci    }
431cabdff1aSopenharmony_ci
432cabdff1aSopenharmony_ci    if (CONFIG_GRAY && s->avctx->flags & AV_CODEC_FLAG_GRAY)
433cabdff1aSopenharmony_ci        return;
434cabdff1aSopenharmony_ci    /* Chroma MC always uses qpel bilinear */
435cabdff1aSopenharmony_ci    uvmx = (uvmx & 3) << 1;
436cabdff1aSopenharmony_ci    uvmy = (uvmy & 3) << 1;
437cabdff1aSopenharmony_ci    if (!v->rnd) {
438cabdff1aSopenharmony_ci        h264chroma->put_h264_chroma_pixels_tab[0](s->dest[1], srcU, s->uvlinesize, 8, uvmx, uvmy);
439cabdff1aSopenharmony_ci        h264chroma->put_h264_chroma_pixels_tab[0](s->dest[2], srcV, s->uvlinesize, 8, uvmx, uvmy);
440cabdff1aSopenharmony_ci    } else {
441cabdff1aSopenharmony_ci        v->vc1dsp.put_no_rnd_vc1_chroma_pixels_tab[0](s->dest[1], srcU, s->uvlinesize, 8, uvmx, uvmy);
442cabdff1aSopenharmony_ci        v->vc1dsp.put_no_rnd_vc1_chroma_pixels_tab[0](s->dest[2], srcV, s->uvlinesize, 8, uvmx, uvmy);
443cabdff1aSopenharmony_ci    }
444cabdff1aSopenharmony_ci    if (v->field_mode) {
445cabdff1aSopenharmony_ci        v->mv_f[dir][s->block_index[4] + v->mb_off] = v->cur_field_type != v->ref_field_type[dir];
446cabdff1aSopenharmony_ci        v->mv_f[dir][s->block_index[5] + v->mb_off] = v->cur_field_type != v->ref_field_type[dir];
447cabdff1aSopenharmony_ci    }
448cabdff1aSopenharmony_ci}
449cabdff1aSopenharmony_ci
450cabdff1aSopenharmony_ci/** Do motion compensation for 4-MV macroblock - luminance block
451cabdff1aSopenharmony_ci */
452cabdff1aSopenharmony_civoid ff_vc1_mc_4mv_luma(VC1Context *v, int n, int dir, int avg)
453cabdff1aSopenharmony_ci{
454cabdff1aSopenharmony_ci    MpegEncContext *s = &v->s;
455cabdff1aSopenharmony_ci    uint8_t *srcY;
456cabdff1aSopenharmony_ci    int dxy, mx, my, src_x, src_y;
457cabdff1aSopenharmony_ci    int off;
458cabdff1aSopenharmony_ci    int fieldmv = (v->fcm == ILACE_FRAME) ? v->blk_mv_type[s->block_index[n]] : 0;
459cabdff1aSopenharmony_ci    int v_edge_pos = s->v_edge_pos >> v->field_mode;
460cabdff1aSopenharmony_ci    uint8_t (*luty)[256];
461cabdff1aSopenharmony_ci    int use_ic;
462cabdff1aSopenharmony_ci    int interlace;
463cabdff1aSopenharmony_ci    int linesize;
464cabdff1aSopenharmony_ci
465cabdff1aSopenharmony_ci    if ((!v->field_mode ||
466cabdff1aSopenharmony_ci         (v->ref_field_type[dir] == 1 && v->cur_field_type == 1)) &&
467cabdff1aSopenharmony_ci        !v->s.last_picture.f->data[0])
468cabdff1aSopenharmony_ci        return;
469cabdff1aSopenharmony_ci
470cabdff1aSopenharmony_ci    linesize = s->current_picture_ptr->f->linesize[0];
471cabdff1aSopenharmony_ci
472cabdff1aSopenharmony_ci    mx = s->mv[dir][n][0];
473cabdff1aSopenharmony_ci    my = s->mv[dir][n][1];
474cabdff1aSopenharmony_ci
475cabdff1aSopenharmony_ci    if (!dir) {
476cabdff1aSopenharmony_ci        if (v->field_mode && (v->cur_field_type != v->ref_field_type[dir]) && v->second_field) {
477cabdff1aSopenharmony_ci            srcY = s->current_picture.f->data[0];
478cabdff1aSopenharmony_ci            luty = v->curr_luty;
479cabdff1aSopenharmony_ci            use_ic = *v->curr_use_ic;
480cabdff1aSopenharmony_ci            interlace = 1;
481cabdff1aSopenharmony_ci        } else {
482cabdff1aSopenharmony_ci            srcY = s->last_picture.f->data[0];
483cabdff1aSopenharmony_ci            luty = v->last_luty;
484cabdff1aSopenharmony_ci            use_ic = v->last_use_ic;
485cabdff1aSopenharmony_ci            interlace = s->last_picture.f->interlaced_frame;
486cabdff1aSopenharmony_ci        }
487cabdff1aSopenharmony_ci    } else {
488cabdff1aSopenharmony_ci        srcY = s->next_picture.f->data[0];
489cabdff1aSopenharmony_ci        luty = v->next_luty;
490cabdff1aSopenharmony_ci        use_ic = v->next_use_ic;
491cabdff1aSopenharmony_ci        interlace = s->next_picture.f->interlaced_frame;
492cabdff1aSopenharmony_ci    }
493cabdff1aSopenharmony_ci
494cabdff1aSopenharmony_ci    if (!srcY) {
495cabdff1aSopenharmony_ci        av_log(v->s.avctx, AV_LOG_ERROR, "Referenced frame missing.\n");
496cabdff1aSopenharmony_ci        return;
497cabdff1aSopenharmony_ci    }
498cabdff1aSopenharmony_ci
499cabdff1aSopenharmony_ci    if (v->field_mode) {
500cabdff1aSopenharmony_ci        if (v->cur_field_type != v->ref_field_type[dir])
501cabdff1aSopenharmony_ci            my = my - 2 + 4 * v->cur_field_type;
502cabdff1aSopenharmony_ci    }
503cabdff1aSopenharmony_ci
504cabdff1aSopenharmony_ci    if (s->pict_type == AV_PICTURE_TYPE_P && n == 3 && v->field_mode) {
505cabdff1aSopenharmony_ci        int opp_count = get_luma_mv(v, 0,
506cabdff1aSopenharmony_ci                                    &s->current_picture.motion_val[1][s->block_index[0] + v->blocks_off][0],
507cabdff1aSopenharmony_ci                                    &s->current_picture.motion_val[1][s->block_index[0] + v->blocks_off][1]);
508cabdff1aSopenharmony_ci        int k, f = opp_count > 2;
509cabdff1aSopenharmony_ci        for (k = 0; k < 4; k++)
510cabdff1aSopenharmony_ci            v->mv_f[1][s->block_index[k] + v->blocks_off] = f;
511cabdff1aSopenharmony_ci    }
512cabdff1aSopenharmony_ci
513cabdff1aSopenharmony_ci    if (v->fcm == ILACE_FRAME) {  // not sure if needed for other types of picture
514cabdff1aSopenharmony_ci        int qx, qy;
515cabdff1aSopenharmony_ci        int width  = s->avctx->coded_width;
516cabdff1aSopenharmony_ci        int height = s->avctx->coded_height >> 1;
517cabdff1aSopenharmony_ci        if (s->pict_type == AV_PICTURE_TYPE_P) {
518cabdff1aSopenharmony_ci            s->current_picture.motion_val[1][s->block_index[n] + v->blocks_off][0] = mx;
519cabdff1aSopenharmony_ci            s->current_picture.motion_val[1][s->block_index[n] + v->blocks_off][1] = my;
520cabdff1aSopenharmony_ci        }
521cabdff1aSopenharmony_ci        qx = (s->mb_x * 16) + (mx >> 2);
522cabdff1aSopenharmony_ci        qy = (s->mb_y *  8) + (my >> 3);
523cabdff1aSopenharmony_ci
524cabdff1aSopenharmony_ci        if (qx < -17)
525cabdff1aSopenharmony_ci            mx -= 4 * (qx + 17);
526cabdff1aSopenharmony_ci        else if (qx > width)
527cabdff1aSopenharmony_ci            mx -= 4 * (qx - width);
528cabdff1aSopenharmony_ci        if (qy < -18)
529cabdff1aSopenharmony_ci            my -= 8 * (qy + 18);
530cabdff1aSopenharmony_ci        else if (qy > height + 1)
531cabdff1aSopenharmony_ci            my -= 8 * (qy - height - 1);
532cabdff1aSopenharmony_ci    }
533cabdff1aSopenharmony_ci
534cabdff1aSopenharmony_ci    if ((v->fcm == ILACE_FRAME) && fieldmv)
535cabdff1aSopenharmony_ci        off = ((n > 1) ? s->linesize : 0) + (n & 1) * 8;
536cabdff1aSopenharmony_ci    else
537cabdff1aSopenharmony_ci        off = s->linesize * 4 * (n & 2) + (n & 1) * 8;
538cabdff1aSopenharmony_ci
539cabdff1aSopenharmony_ci    src_x = s->mb_x * 16 + (n & 1) * 8 + (mx >> 2);
540cabdff1aSopenharmony_ci    if (!fieldmv)
541cabdff1aSopenharmony_ci        src_y = s->mb_y * 16 + (n & 2) * 4 + (my >> 2);
542cabdff1aSopenharmony_ci    else
543cabdff1aSopenharmony_ci        src_y = s->mb_y * 16 + ((n > 1) ? 1 : 0) + (my >> 2);
544cabdff1aSopenharmony_ci
545cabdff1aSopenharmony_ci    if (v->profile != PROFILE_ADVANCED) {
546cabdff1aSopenharmony_ci        src_x = av_clip(src_x, -16, s->mb_width  * 16);
547cabdff1aSopenharmony_ci        src_y = av_clip(src_y, -16, s->mb_height * 16);
548cabdff1aSopenharmony_ci    } else {
549cabdff1aSopenharmony_ci        src_x = av_clip(src_x, -17, s->avctx->coded_width);
550cabdff1aSopenharmony_ci        if (v->fcm == ILACE_FRAME)
551cabdff1aSopenharmony_ci            src_y = av_clip(src_y, -18 + (src_y & 1), s->avctx->coded_height + (src_y & 1));
552cabdff1aSopenharmony_ci        else
553cabdff1aSopenharmony_ci            src_y = av_clip(src_y, -18, s->avctx->coded_height + 1);
554cabdff1aSopenharmony_ci    }
555cabdff1aSopenharmony_ci
556cabdff1aSopenharmony_ci    srcY += src_y * s->linesize + src_x;
557cabdff1aSopenharmony_ci    if (v->field_mode && v->ref_field_type[dir])
558cabdff1aSopenharmony_ci        srcY += linesize;
559cabdff1aSopenharmony_ci
560cabdff1aSopenharmony_ci    if (v->rangeredfrm || use_ic
561cabdff1aSopenharmony_ci        || s->h_edge_pos < 13 || v_edge_pos < 23
562cabdff1aSopenharmony_ci        || (unsigned)(src_x - s->mspel) > s->h_edge_pos - (mx & 3) - 8 - s->mspel * 2
563cabdff1aSopenharmony_ci        || (unsigned)(src_y - (s->mspel << fieldmv)) > v_edge_pos - (my & 3) - ((8 + s->mspel * 2) << fieldmv)) {
564cabdff1aSopenharmony_ci        const int k = 9 + s->mspel * 2;
565cabdff1aSopenharmony_ci
566cabdff1aSopenharmony_ci        srcY -= s->mspel * (1 + (s->linesize << fieldmv));
567cabdff1aSopenharmony_ci        /* check emulate edge stride and offset */
568cabdff1aSopenharmony_ci        if (interlace) {
569cabdff1aSopenharmony_ci            s->vdsp.emulated_edge_mc(s->sc.edge_emu_buffer,
570cabdff1aSopenharmony_ci                                     srcY,
571cabdff1aSopenharmony_ci                                     linesize << 1,
572cabdff1aSopenharmony_ci                                     linesize << 1,
573cabdff1aSopenharmony_ci                                     k,
574cabdff1aSopenharmony_ci                                     v->field_mode ? k : (k << fieldmv) + 1 >> 1,
575cabdff1aSopenharmony_ci                                     src_x - s->mspel,
576cabdff1aSopenharmony_ci                                     src_y - (s->mspel << fieldmv) >> !v->field_mode,
577cabdff1aSopenharmony_ci                                     s->h_edge_pos,
578cabdff1aSopenharmony_ci                                     s->v_edge_pos >> 1);
579cabdff1aSopenharmony_ci            if (!v->field_mode && !fieldmv)
580cabdff1aSopenharmony_ci                s->vdsp.emulated_edge_mc(s->sc.edge_emu_buffer + linesize,
581cabdff1aSopenharmony_ci                                         srcY + linesize,
582cabdff1aSopenharmony_ci                                         linesize << 1,
583cabdff1aSopenharmony_ci                                         linesize << 1,
584cabdff1aSopenharmony_ci                                         k,
585cabdff1aSopenharmony_ci                                         k >> 1,
586cabdff1aSopenharmony_ci                                         src_x - s->mspel,
587cabdff1aSopenharmony_ci                                         src_y - s->mspel + 1 >> 1,
588cabdff1aSopenharmony_ci                                         s->h_edge_pos,
589cabdff1aSopenharmony_ci                                         s->v_edge_pos >> 1);
590cabdff1aSopenharmony_ci        } else
591cabdff1aSopenharmony_ci            s->vdsp.emulated_edge_mc(s->sc.edge_emu_buffer,
592cabdff1aSopenharmony_ci                                     srcY,
593cabdff1aSopenharmony_ci                                     linesize,
594cabdff1aSopenharmony_ci                                     linesize,
595cabdff1aSopenharmony_ci                                     k,
596cabdff1aSopenharmony_ci                                     v->field_mode ? (k << 1) - 1 : k << fieldmv,
597cabdff1aSopenharmony_ci                                     src_x - s->mspel,
598cabdff1aSopenharmony_ci                                     v->field_mode ? 2 * (src_y - s->mspel) + v->ref_field_type[dir] :
599cabdff1aSopenharmony_ci                                                     src_y - (s->mspel << fieldmv),
600cabdff1aSopenharmony_ci                                     s->h_edge_pos,
601cabdff1aSopenharmony_ci                                     s->v_edge_pos);
602cabdff1aSopenharmony_ci        srcY = s->sc.edge_emu_buffer;
603cabdff1aSopenharmony_ci        /* if we deal with range reduction we need to scale source blocks */
604cabdff1aSopenharmony_ci        if (v->rangeredfrm) {
605cabdff1aSopenharmony_ci            vc1_scale_luma(srcY, k, s->linesize << fieldmv);
606cabdff1aSopenharmony_ci        }
607cabdff1aSopenharmony_ci        /* if we deal with intensity compensation we need to scale source blocks */
608cabdff1aSopenharmony_ci        if (use_ic) {
609cabdff1aSopenharmony_ci            vc1_lut_scale_luma(srcY,
610cabdff1aSopenharmony_ci                               luty[v->field_mode ? v->ref_field_type[dir] : (((0<<fieldmv)+src_y - (s->mspel << fieldmv)) & 1)],
611cabdff1aSopenharmony_ci                               luty[v->field_mode ? v->ref_field_type[dir] : (((1<<fieldmv)+src_y - (s->mspel << fieldmv)) & 1)],
612cabdff1aSopenharmony_ci                               k, s->linesize << fieldmv);
613cabdff1aSopenharmony_ci        }
614cabdff1aSopenharmony_ci        srcY += s->mspel * (1 + (s->linesize << fieldmv));
615cabdff1aSopenharmony_ci    }
616cabdff1aSopenharmony_ci
617cabdff1aSopenharmony_ci    if (s->mspel) {
618cabdff1aSopenharmony_ci        dxy = ((my & 3) << 2) | (mx & 3);
619cabdff1aSopenharmony_ci        if (avg)
620cabdff1aSopenharmony_ci            v->vc1dsp.avg_vc1_mspel_pixels_tab[1][dxy](s->dest[0] + off, srcY, s->linesize << fieldmv, v->rnd);
621cabdff1aSopenharmony_ci        else
622cabdff1aSopenharmony_ci            v->vc1dsp.put_vc1_mspel_pixels_tab[1][dxy](s->dest[0] + off, srcY, s->linesize << fieldmv, v->rnd);
623cabdff1aSopenharmony_ci    } else { // hpel mc - always used for luma
624cabdff1aSopenharmony_ci        dxy = (my & 2) | ((mx & 2) >> 1);
625cabdff1aSopenharmony_ci        if (!v->rnd)
626cabdff1aSopenharmony_ci            s->hdsp.put_pixels_tab[1][dxy](s->dest[0] + off, srcY, s->linesize, 8);
627cabdff1aSopenharmony_ci        else
628cabdff1aSopenharmony_ci            s->hdsp.put_no_rnd_pixels_tab[1][dxy](s->dest[0] + off, srcY, s->linesize, 8);
629cabdff1aSopenharmony_ci    }
630cabdff1aSopenharmony_ci}
631cabdff1aSopenharmony_ci
632cabdff1aSopenharmony_ci/** Do motion compensation for 4-MV macroblock - both chroma blocks
633cabdff1aSopenharmony_ci */
634cabdff1aSopenharmony_civoid ff_vc1_mc_4mv_chroma(VC1Context *v, int dir)
635cabdff1aSopenharmony_ci{
636cabdff1aSopenharmony_ci    MpegEncContext *s = &v->s;
637cabdff1aSopenharmony_ci    H264ChromaContext *h264chroma = &v->h264chroma;
638cabdff1aSopenharmony_ci    uint8_t *srcU, *srcV;
639cabdff1aSopenharmony_ci    int uvmx, uvmy, uvsrc_x, uvsrc_y;
640cabdff1aSopenharmony_ci    int16_t tx, ty;
641cabdff1aSopenharmony_ci    int chroma_ref_type;
642cabdff1aSopenharmony_ci    int v_edge_pos = s->v_edge_pos >> v->field_mode;
643cabdff1aSopenharmony_ci    uint8_t (*lutuv)[256];
644cabdff1aSopenharmony_ci    int use_ic;
645cabdff1aSopenharmony_ci    int interlace;
646cabdff1aSopenharmony_ci    int uvlinesize;
647cabdff1aSopenharmony_ci
648cabdff1aSopenharmony_ci    if (!v->field_mode && !v->s.last_picture.f->data[0])
649cabdff1aSopenharmony_ci        return;
650cabdff1aSopenharmony_ci    if (CONFIG_GRAY && s->avctx->flags & AV_CODEC_FLAG_GRAY)
651cabdff1aSopenharmony_ci        return;
652cabdff1aSopenharmony_ci
653cabdff1aSopenharmony_ci    /* calculate chroma MV vector from four luma MVs */
654cabdff1aSopenharmony_ci    if (!v->field_mode || !v->numref) {
655cabdff1aSopenharmony_ci        int valid_count = get_chroma_mv(v, dir, &tx, &ty);
656cabdff1aSopenharmony_ci        if (!valid_count) {
657cabdff1aSopenharmony_ci            s->current_picture.motion_val[1][s->block_index[0] + v->blocks_off][0] = 0;
658cabdff1aSopenharmony_ci            s->current_picture.motion_val[1][s->block_index[0] + v->blocks_off][1] = 0;
659cabdff1aSopenharmony_ci            v->luma_mv[s->mb_x][0] = v->luma_mv[s->mb_x][1] = 0;
660cabdff1aSopenharmony_ci            return; //no need to do MC for intra blocks
661cabdff1aSopenharmony_ci        }
662cabdff1aSopenharmony_ci        chroma_ref_type = v->ref_field_type[dir];
663cabdff1aSopenharmony_ci    } else {
664cabdff1aSopenharmony_ci        int opp_count = get_luma_mv(v, dir, &tx, &ty);
665cabdff1aSopenharmony_ci        chroma_ref_type = v->cur_field_type ^ (opp_count > 2);
666cabdff1aSopenharmony_ci    }
667cabdff1aSopenharmony_ci    if (v->field_mode && chroma_ref_type == 1 && v->cur_field_type == 1 && !v->s.last_picture.f->data[0])
668cabdff1aSopenharmony_ci        return;
669cabdff1aSopenharmony_ci    s->current_picture.motion_val[1][s->block_index[0] + v->blocks_off][0] = tx;
670cabdff1aSopenharmony_ci    s->current_picture.motion_val[1][s->block_index[0] + v->blocks_off][1] = ty;
671cabdff1aSopenharmony_ci
672cabdff1aSopenharmony_ci    uvlinesize = s->current_picture_ptr->f->linesize[1];
673cabdff1aSopenharmony_ci
674cabdff1aSopenharmony_ci    uvmx = (tx + ((tx & 3) == 3)) >> 1;
675cabdff1aSopenharmony_ci    uvmy = (ty + ((ty & 3) == 3)) >> 1;
676cabdff1aSopenharmony_ci
677cabdff1aSopenharmony_ci    v->luma_mv[s->mb_x][0] = uvmx;
678cabdff1aSopenharmony_ci    v->luma_mv[s->mb_x][1] = uvmy;
679cabdff1aSopenharmony_ci
680cabdff1aSopenharmony_ci    if (v->fastuvmc) {
681cabdff1aSopenharmony_ci        uvmx = uvmx + ((uvmx < 0) ? (uvmx & 1) : -(uvmx & 1));
682cabdff1aSopenharmony_ci        uvmy = uvmy + ((uvmy < 0) ? (uvmy & 1) : -(uvmy & 1));
683cabdff1aSopenharmony_ci    }
684cabdff1aSopenharmony_ci    // Field conversion bias
685cabdff1aSopenharmony_ci    if (v->cur_field_type != chroma_ref_type)
686cabdff1aSopenharmony_ci        uvmy += 2 - 4 * chroma_ref_type;
687cabdff1aSopenharmony_ci
688cabdff1aSopenharmony_ci    uvsrc_x = s->mb_x * 8 + (uvmx >> 2);
689cabdff1aSopenharmony_ci    uvsrc_y = s->mb_y * 8 + (uvmy >> 2);
690cabdff1aSopenharmony_ci
691cabdff1aSopenharmony_ci    if (v->profile != PROFILE_ADVANCED) {
692cabdff1aSopenharmony_ci        uvsrc_x = av_clip(uvsrc_x, -8, s->mb_width  * 8);
693cabdff1aSopenharmony_ci        uvsrc_y = av_clip(uvsrc_y, -8, s->mb_height * 8);
694cabdff1aSopenharmony_ci    } else {
695cabdff1aSopenharmony_ci        uvsrc_x = av_clip(uvsrc_x, -8, s->avctx->coded_width  >> 1);
696cabdff1aSopenharmony_ci        uvsrc_y = av_clip(uvsrc_y, -8, s->avctx->coded_height >> 1);
697cabdff1aSopenharmony_ci    }
698cabdff1aSopenharmony_ci
699cabdff1aSopenharmony_ci    if (!dir) {
700cabdff1aSopenharmony_ci        if (v->field_mode && (v->cur_field_type != chroma_ref_type) && v->second_field) {
701cabdff1aSopenharmony_ci            srcU = s->current_picture.f->data[1];
702cabdff1aSopenharmony_ci            srcV = s->current_picture.f->data[2];
703cabdff1aSopenharmony_ci            lutuv = v->curr_lutuv;
704cabdff1aSopenharmony_ci            use_ic = *v->curr_use_ic;
705cabdff1aSopenharmony_ci            interlace = 1;
706cabdff1aSopenharmony_ci        } else {
707cabdff1aSopenharmony_ci            srcU = s->last_picture.f->data[1];
708cabdff1aSopenharmony_ci            srcV = s->last_picture.f->data[2];
709cabdff1aSopenharmony_ci            lutuv = v->last_lutuv;
710cabdff1aSopenharmony_ci            use_ic = v->last_use_ic;
711cabdff1aSopenharmony_ci            interlace = s->last_picture.f->interlaced_frame;
712cabdff1aSopenharmony_ci        }
713cabdff1aSopenharmony_ci    } else {
714cabdff1aSopenharmony_ci        srcU = s->next_picture.f->data[1];
715cabdff1aSopenharmony_ci        srcV = s->next_picture.f->data[2];
716cabdff1aSopenharmony_ci        lutuv = v->next_lutuv;
717cabdff1aSopenharmony_ci        use_ic = v->next_use_ic;
718cabdff1aSopenharmony_ci        interlace = s->next_picture.f->interlaced_frame;
719cabdff1aSopenharmony_ci    }
720cabdff1aSopenharmony_ci
721cabdff1aSopenharmony_ci    if (!srcU) {
722cabdff1aSopenharmony_ci        av_log(v->s.avctx, AV_LOG_ERROR, "Referenced frame missing.\n");
723cabdff1aSopenharmony_ci        return;
724cabdff1aSopenharmony_ci    }
725cabdff1aSopenharmony_ci
726cabdff1aSopenharmony_ci    srcU += uvsrc_y * s->uvlinesize + uvsrc_x;
727cabdff1aSopenharmony_ci    srcV += uvsrc_y * s->uvlinesize + uvsrc_x;
728cabdff1aSopenharmony_ci
729cabdff1aSopenharmony_ci    if (v->field_mode) {
730cabdff1aSopenharmony_ci        if (chroma_ref_type) {
731cabdff1aSopenharmony_ci            srcU += uvlinesize;
732cabdff1aSopenharmony_ci            srcV += uvlinesize;
733cabdff1aSopenharmony_ci        }
734cabdff1aSopenharmony_ci    }
735cabdff1aSopenharmony_ci
736cabdff1aSopenharmony_ci    if (v->rangeredfrm || use_ic
737cabdff1aSopenharmony_ci        || s->h_edge_pos < 18 || v_edge_pos < 18
738cabdff1aSopenharmony_ci        || (unsigned)uvsrc_x > (s->h_edge_pos >> 1) - 9
739cabdff1aSopenharmony_ci        || (unsigned)uvsrc_y > (v_edge_pos    >> 1) - 9) {
740cabdff1aSopenharmony_ci        if (interlace) {
741cabdff1aSopenharmony_ci            s->vdsp.emulated_edge_mc(s->sc.edge_emu_buffer,
742cabdff1aSopenharmony_ci                                     srcU,
743cabdff1aSopenharmony_ci                                     uvlinesize << 1,
744cabdff1aSopenharmony_ci                                     uvlinesize << 1,
745cabdff1aSopenharmony_ci                                     9,
746cabdff1aSopenharmony_ci                                     v->field_mode ? 9 : 5,
747cabdff1aSopenharmony_ci                                     uvsrc_x,
748cabdff1aSopenharmony_ci                                     uvsrc_y >> !v->field_mode,
749cabdff1aSopenharmony_ci                                     s->h_edge_pos >> 1,
750cabdff1aSopenharmony_ci                                     s->v_edge_pos >> 2);
751cabdff1aSopenharmony_ci            s->vdsp.emulated_edge_mc(s->sc.edge_emu_buffer + 16,
752cabdff1aSopenharmony_ci                                     srcV,
753cabdff1aSopenharmony_ci                                     uvlinesize << 1,
754cabdff1aSopenharmony_ci                                     uvlinesize << 1,
755cabdff1aSopenharmony_ci                                     9,
756cabdff1aSopenharmony_ci                                     v->field_mode ? 9 : 5,
757cabdff1aSopenharmony_ci                                     uvsrc_x,
758cabdff1aSopenharmony_ci                                     uvsrc_y >> !v->field_mode,
759cabdff1aSopenharmony_ci                                     s->h_edge_pos >> 1,
760cabdff1aSopenharmony_ci                                     s->v_edge_pos >> 2);
761cabdff1aSopenharmony_ci            if (!v->field_mode) {
762cabdff1aSopenharmony_ci                s->vdsp.emulated_edge_mc(s->sc.edge_emu_buffer + uvlinesize,
763cabdff1aSopenharmony_ci                                         srcU + uvlinesize,
764cabdff1aSopenharmony_ci                                         uvlinesize << 1,
765cabdff1aSopenharmony_ci                                         uvlinesize << 1,
766cabdff1aSopenharmony_ci                                         9,
767cabdff1aSopenharmony_ci                                         4,
768cabdff1aSopenharmony_ci                                         uvsrc_x,
769cabdff1aSopenharmony_ci                                         uvsrc_y + 1 >> 1,
770cabdff1aSopenharmony_ci                                         s->h_edge_pos >> 1,
771cabdff1aSopenharmony_ci                                         s->v_edge_pos >> 2);
772cabdff1aSopenharmony_ci                s->vdsp.emulated_edge_mc(s->sc.edge_emu_buffer + 16 + uvlinesize,
773cabdff1aSopenharmony_ci                                         srcV + uvlinesize,
774cabdff1aSopenharmony_ci                                         uvlinesize << 1,
775cabdff1aSopenharmony_ci                                         uvlinesize << 1,
776cabdff1aSopenharmony_ci                                         9,
777cabdff1aSopenharmony_ci                                         4,
778cabdff1aSopenharmony_ci                                         uvsrc_x,
779cabdff1aSopenharmony_ci                                         uvsrc_y + 1 >> 1,
780cabdff1aSopenharmony_ci                                         s->h_edge_pos >> 1,
781cabdff1aSopenharmony_ci                                         s->v_edge_pos >> 2);
782cabdff1aSopenharmony_ci            }
783cabdff1aSopenharmony_ci        } else {
784cabdff1aSopenharmony_ci            s->vdsp.emulated_edge_mc(s->sc.edge_emu_buffer,
785cabdff1aSopenharmony_ci                                     srcU,
786cabdff1aSopenharmony_ci                                     uvlinesize,
787cabdff1aSopenharmony_ci                                     uvlinesize,
788cabdff1aSopenharmony_ci                                     9,
789cabdff1aSopenharmony_ci                                     v->field_mode ? 17 : 9,
790cabdff1aSopenharmony_ci                                     uvsrc_x,
791cabdff1aSopenharmony_ci                                     v->field_mode ? 2 * uvsrc_y + chroma_ref_type : uvsrc_y,
792cabdff1aSopenharmony_ci                                     s->h_edge_pos >> 1,
793cabdff1aSopenharmony_ci                                     s->v_edge_pos >> 1);
794cabdff1aSopenharmony_ci            s->vdsp.emulated_edge_mc(s->sc.edge_emu_buffer + 16,
795cabdff1aSopenharmony_ci                                     srcV,
796cabdff1aSopenharmony_ci                                     uvlinesize,
797cabdff1aSopenharmony_ci                                     uvlinesize,
798cabdff1aSopenharmony_ci                                     9,
799cabdff1aSopenharmony_ci                                     v->field_mode ? 17 : 9,
800cabdff1aSopenharmony_ci                                     uvsrc_x,
801cabdff1aSopenharmony_ci                                     v->field_mode ? 2 * uvsrc_y + chroma_ref_type : uvsrc_y,
802cabdff1aSopenharmony_ci                                     s->h_edge_pos >> 1,
803cabdff1aSopenharmony_ci                                     s->v_edge_pos >> 1);
804cabdff1aSopenharmony_ci        }
805cabdff1aSopenharmony_ci        srcU = s->sc.edge_emu_buffer;
806cabdff1aSopenharmony_ci        srcV = s->sc.edge_emu_buffer + 16;
807cabdff1aSopenharmony_ci
808cabdff1aSopenharmony_ci        /* if we deal with range reduction we need to scale source blocks */
809cabdff1aSopenharmony_ci        if (v->rangeredfrm) {
810cabdff1aSopenharmony_ci            vc1_scale_chroma(srcU, srcV, 9, s->uvlinesize);
811cabdff1aSopenharmony_ci        }
812cabdff1aSopenharmony_ci        /* if we deal with intensity compensation we need to scale source blocks */
813cabdff1aSopenharmony_ci        if (use_ic) {
814cabdff1aSopenharmony_ci            vc1_lut_scale_chroma(srcU, srcV,
815cabdff1aSopenharmony_ci                                 lutuv[v->field_mode ? chroma_ref_type : ((0 + uvsrc_y) & 1)],
816cabdff1aSopenharmony_ci                                 lutuv[v->field_mode ? chroma_ref_type : ((1 + uvsrc_y) & 1)],
817cabdff1aSopenharmony_ci                                 9, s->uvlinesize);
818cabdff1aSopenharmony_ci        }
819cabdff1aSopenharmony_ci    }
820cabdff1aSopenharmony_ci
821cabdff1aSopenharmony_ci    /* Chroma MC always uses qpel bilinear */
822cabdff1aSopenharmony_ci    uvmx = (uvmx & 3) << 1;
823cabdff1aSopenharmony_ci    uvmy = (uvmy & 3) << 1;
824cabdff1aSopenharmony_ci    if (!v->rnd) {
825cabdff1aSopenharmony_ci        h264chroma->put_h264_chroma_pixels_tab[0](s->dest[1], srcU, s->uvlinesize, 8, uvmx, uvmy);
826cabdff1aSopenharmony_ci        h264chroma->put_h264_chroma_pixels_tab[0](s->dest[2], srcV, s->uvlinesize, 8, uvmx, uvmy);
827cabdff1aSopenharmony_ci    } else {
828cabdff1aSopenharmony_ci        v->vc1dsp.put_no_rnd_vc1_chroma_pixels_tab[0](s->dest[1], srcU, s->uvlinesize, 8, uvmx, uvmy);
829cabdff1aSopenharmony_ci        v->vc1dsp.put_no_rnd_vc1_chroma_pixels_tab[0](s->dest[2], srcV, s->uvlinesize, 8, uvmx, uvmy);
830cabdff1aSopenharmony_ci    }
831cabdff1aSopenharmony_ci    if (v->field_mode) {
832cabdff1aSopenharmony_ci        v->mv_f[dir][s->block_index[4] + v->mb_off] = v->cur_field_type != chroma_ref_type;
833cabdff1aSopenharmony_ci        v->mv_f[dir][s->block_index[5] + v->mb_off] = v->cur_field_type != chroma_ref_type;
834cabdff1aSopenharmony_ci    }
835cabdff1aSopenharmony_ci}
836cabdff1aSopenharmony_ci
837cabdff1aSopenharmony_ci/** Do motion compensation for 4-MV interlaced frame chroma macroblock (both U and V)
838cabdff1aSopenharmony_ci */
839cabdff1aSopenharmony_civoid ff_vc1_mc_4mv_chroma4(VC1Context *v, int dir, int dir2, int avg)
840cabdff1aSopenharmony_ci{
841cabdff1aSopenharmony_ci    MpegEncContext *s = &v->s;
842cabdff1aSopenharmony_ci    H264ChromaContext *h264chroma = &v->h264chroma;
843cabdff1aSopenharmony_ci    uint8_t *srcU, *srcV;
844cabdff1aSopenharmony_ci    int uvsrc_x, uvsrc_y;
845cabdff1aSopenharmony_ci    int uvmx_field[4], uvmy_field[4];
846cabdff1aSopenharmony_ci    int i, off, tx, ty;
847cabdff1aSopenharmony_ci    int fieldmv = v->blk_mv_type[s->block_index[0]];
848cabdff1aSopenharmony_ci    static const uint8_t s_rndtblfield[16] = { 0, 0, 1, 2, 4, 4, 5, 6, 2, 2, 3, 8, 6, 6, 7, 12 };
849cabdff1aSopenharmony_ci    int v_dist = fieldmv ? 1 : 4; // vertical offset for lower sub-blocks
850cabdff1aSopenharmony_ci    int v_edge_pos = s->v_edge_pos >> 1;
851cabdff1aSopenharmony_ci    int use_ic;
852cabdff1aSopenharmony_ci    int interlace;
853cabdff1aSopenharmony_ci    int uvlinesize;
854cabdff1aSopenharmony_ci    uint8_t (*lutuv)[256];
855cabdff1aSopenharmony_ci
856cabdff1aSopenharmony_ci    if (CONFIG_GRAY && s->avctx->flags & AV_CODEC_FLAG_GRAY)
857cabdff1aSopenharmony_ci        return;
858cabdff1aSopenharmony_ci
859cabdff1aSopenharmony_ci    uvlinesize = s->current_picture_ptr->f->linesize[1];
860cabdff1aSopenharmony_ci
861cabdff1aSopenharmony_ci    for (i = 0; i < 4; i++) {
862cabdff1aSopenharmony_ci        int d = i < 2 ? dir: dir2;
863cabdff1aSopenharmony_ci        tx = s->mv[d][i][0];
864cabdff1aSopenharmony_ci        uvmx_field[i] = (tx + ((tx & 3) == 3)) >> 1;
865cabdff1aSopenharmony_ci        ty = s->mv[d][i][1];
866cabdff1aSopenharmony_ci        if (fieldmv)
867cabdff1aSopenharmony_ci            uvmy_field[i] = (ty >> 4) * 8 + s_rndtblfield[ty & 0xF];
868cabdff1aSopenharmony_ci        else
869cabdff1aSopenharmony_ci            uvmy_field[i] = (ty + ((ty & 3) == 3)) >> 1;
870cabdff1aSopenharmony_ci    }
871cabdff1aSopenharmony_ci
872cabdff1aSopenharmony_ci    for (i = 0; i < 4; i++) {
873cabdff1aSopenharmony_ci        off = (i & 1) * 4 + ((i & 2) ? v_dist * s->uvlinesize : 0);
874cabdff1aSopenharmony_ci        uvsrc_x = s->mb_x * 8 +  (i & 1) * 4           + (uvmx_field[i] >> 2);
875cabdff1aSopenharmony_ci        uvsrc_y = s->mb_y * 8 + ((i & 2) ? v_dist : 0) + (uvmy_field[i] >> 2);
876cabdff1aSopenharmony_ci        // FIXME: implement proper pull-back (see vc1cropmv.c, vc1CROPMV_ChromaPullBack())
877cabdff1aSopenharmony_ci        uvsrc_x = av_clip(uvsrc_x, -8, s->avctx->coded_width  >> 1);
878cabdff1aSopenharmony_ci        if (v->fcm == ILACE_FRAME)
879cabdff1aSopenharmony_ci            uvsrc_y = av_clip(uvsrc_y, -8 + (uvsrc_y & 1), (s->avctx->coded_height >> 1) + (uvsrc_y & 1));
880cabdff1aSopenharmony_ci        else
881cabdff1aSopenharmony_ci            uvsrc_y = av_clip(uvsrc_y, -8, s->avctx->coded_height >> 1);
882cabdff1aSopenharmony_ci        if (i < 2 ? dir : dir2) {
883cabdff1aSopenharmony_ci            srcU = s->next_picture.f->data[1];
884cabdff1aSopenharmony_ci            srcV = s->next_picture.f->data[2];
885cabdff1aSopenharmony_ci            lutuv  = v->next_lutuv;
886cabdff1aSopenharmony_ci            use_ic = v->next_use_ic;
887cabdff1aSopenharmony_ci            interlace = s->next_picture.f->interlaced_frame;
888cabdff1aSopenharmony_ci        } else {
889cabdff1aSopenharmony_ci            srcU = s->last_picture.f->data[1];
890cabdff1aSopenharmony_ci            srcV = s->last_picture.f->data[2];
891cabdff1aSopenharmony_ci            lutuv  = v->last_lutuv;
892cabdff1aSopenharmony_ci            use_ic = v->last_use_ic;
893cabdff1aSopenharmony_ci            interlace = s->last_picture.f->interlaced_frame;
894cabdff1aSopenharmony_ci        }
895cabdff1aSopenharmony_ci        if (!srcU)
896cabdff1aSopenharmony_ci            return;
897cabdff1aSopenharmony_ci        srcU += uvsrc_y * s->uvlinesize + uvsrc_x;
898cabdff1aSopenharmony_ci        srcV += uvsrc_y * s->uvlinesize + uvsrc_x;
899cabdff1aSopenharmony_ci        uvmx_field[i] = (uvmx_field[i] & 3) << 1;
900cabdff1aSopenharmony_ci        uvmy_field[i] = (uvmy_field[i] & 3) << 1;
901cabdff1aSopenharmony_ci
902cabdff1aSopenharmony_ci        if (use_ic
903cabdff1aSopenharmony_ci            || s->h_edge_pos < 10 || v_edge_pos < (5 << fieldmv)
904cabdff1aSopenharmony_ci            || (unsigned)uvsrc_x > (s->h_edge_pos >> 1) - 5
905cabdff1aSopenharmony_ci            || (unsigned)uvsrc_y > v_edge_pos - (5 << fieldmv)) {
906cabdff1aSopenharmony_ci            if (interlace) {
907cabdff1aSopenharmony_ci                s->vdsp.emulated_edge_mc(s->sc.edge_emu_buffer,
908cabdff1aSopenharmony_ci                                         srcU,
909cabdff1aSopenharmony_ci                                         uvlinesize << 1,
910cabdff1aSopenharmony_ci                                         uvlinesize << 1,
911cabdff1aSopenharmony_ci                                         5,
912cabdff1aSopenharmony_ci                                         (5 << fieldmv) + 1 >> 1,
913cabdff1aSopenharmony_ci                                         uvsrc_x,
914cabdff1aSopenharmony_ci                                         uvsrc_y >> 1,
915cabdff1aSopenharmony_ci                                         s->h_edge_pos >> 1,
916cabdff1aSopenharmony_ci                                         s->v_edge_pos >> 2);
917cabdff1aSopenharmony_ci                s->vdsp.emulated_edge_mc(s->sc.edge_emu_buffer + 16,
918cabdff1aSopenharmony_ci                                         srcV,
919cabdff1aSopenharmony_ci                                         uvlinesize << 1,
920cabdff1aSopenharmony_ci                                         uvlinesize << 1,
921cabdff1aSopenharmony_ci                                         5,
922cabdff1aSopenharmony_ci                                         (5 << fieldmv) + 1 >> 1,
923cabdff1aSopenharmony_ci                                         uvsrc_x,
924cabdff1aSopenharmony_ci                                         uvsrc_y >> 1,
925cabdff1aSopenharmony_ci                                         s->h_edge_pos >> 1,
926cabdff1aSopenharmony_ci                                         s->v_edge_pos >> 2);
927cabdff1aSopenharmony_ci                if (!fieldmv) {
928cabdff1aSopenharmony_ci                    s->vdsp.emulated_edge_mc(s->sc.edge_emu_buffer + uvlinesize,
929cabdff1aSopenharmony_ci                                             srcU + uvlinesize,
930cabdff1aSopenharmony_ci                                             uvlinesize << 1,
931cabdff1aSopenharmony_ci                                             uvlinesize << 1,
932cabdff1aSopenharmony_ci                                             5,
933cabdff1aSopenharmony_ci                                             2,
934cabdff1aSopenharmony_ci                                             uvsrc_x,
935cabdff1aSopenharmony_ci                                             uvsrc_y + 1 >> 1,
936cabdff1aSopenharmony_ci                                             s->h_edge_pos >> 1,
937cabdff1aSopenharmony_ci                                             s->v_edge_pos >> 2);
938cabdff1aSopenharmony_ci                    s->vdsp.emulated_edge_mc(s->sc.edge_emu_buffer + 16 + uvlinesize,
939cabdff1aSopenharmony_ci                                             srcV + uvlinesize,
940cabdff1aSopenharmony_ci                                             uvlinesize << 1,
941cabdff1aSopenharmony_ci                                             uvlinesize << 1,
942cabdff1aSopenharmony_ci                                             5,
943cabdff1aSopenharmony_ci                                             2,
944cabdff1aSopenharmony_ci                                             uvsrc_x,
945cabdff1aSopenharmony_ci                                             uvsrc_y + 1 >> 1,
946cabdff1aSopenharmony_ci                                             s->h_edge_pos >> 1,
947cabdff1aSopenharmony_ci                                             s->v_edge_pos >> 2);
948cabdff1aSopenharmony_ci                }
949cabdff1aSopenharmony_ci            } else {
950cabdff1aSopenharmony_ci                s->vdsp.emulated_edge_mc(s->sc.edge_emu_buffer,
951cabdff1aSopenharmony_ci                                         srcU,
952cabdff1aSopenharmony_ci                                         uvlinesize,
953cabdff1aSopenharmony_ci                                         uvlinesize,
954cabdff1aSopenharmony_ci                                         5,
955cabdff1aSopenharmony_ci                                         5 << fieldmv,
956cabdff1aSopenharmony_ci                                         uvsrc_x,
957cabdff1aSopenharmony_ci                                         uvsrc_y,
958cabdff1aSopenharmony_ci                                         s->h_edge_pos >> 1,
959cabdff1aSopenharmony_ci                                         s->v_edge_pos >> 1);
960cabdff1aSopenharmony_ci                s->vdsp.emulated_edge_mc(s->sc.edge_emu_buffer + 16,
961cabdff1aSopenharmony_ci                                         srcV,
962cabdff1aSopenharmony_ci                                         uvlinesize,
963cabdff1aSopenharmony_ci                                         uvlinesize,
964cabdff1aSopenharmony_ci                                         5,
965cabdff1aSopenharmony_ci                                         5 << fieldmv,
966cabdff1aSopenharmony_ci                                         uvsrc_x,
967cabdff1aSopenharmony_ci                                         uvsrc_y,
968cabdff1aSopenharmony_ci                                         s->h_edge_pos >> 1,
969cabdff1aSopenharmony_ci                                         s->v_edge_pos >> 1);
970cabdff1aSopenharmony_ci            }
971cabdff1aSopenharmony_ci            srcU = s->sc.edge_emu_buffer;
972cabdff1aSopenharmony_ci            srcV = s->sc.edge_emu_buffer + 16;
973cabdff1aSopenharmony_ci
974cabdff1aSopenharmony_ci            /* if we deal with intensity compensation we need to scale source blocks */
975cabdff1aSopenharmony_ci            if (use_ic) {
976cabdff1aSopenharmony_ci                vc1_lut_scale_chroma(srcU, srcV,
977cabdff1aSopenharmony_ci                                     lutuv[(uvsrc_y + (0 << fieldmv)) & 1],
978cabdff1aSopenharmony_ci                                     lutuv[(uvsrc_y + (1 << fieldmv)) & 1],
979cabdff1aSopenharmony_ci                                     5, s->uvlinesize << fieldmv);
980cabdff1aSopenharmony_ci            }
981cabdff1aSopenharmony_ci        }
982cabdff1aSopenharmony_ci        if (avg) {
983cabdff1aSopenharmony_ci            if (!v->rnd) {
984cabdff1aSopenharmony_ci                h264chroma->avg_h264_chroma_pixels_tab[1](s->dest[1] + off, srcU, s->uvlinesize << fieldmv, 4, uvmx_field[i], uvmy_field[i]);
985cabdff1aSopenharmony_ci                h264chroma->avg_h264_chroma_pixels_tab[1](s->dest[2] + off, srcV, s->uvlinesize << fieldmv, 4, uvmx_field[i], uvmy_field[i]);
986cabdff1aSopenharmony_ci            } else {
987cabdff1aSopenharmony_ci                v->vc1dsp.avg_no_rnd_vc1_chroma_pixels_tab[1](s->dest[1] + off, srcU, s->uvlinesize << fieldmv, 4, uvmx_field[i], uvmy_field[i]);
988cabdff1aSopenharmony_ci                v->vc1dsp.avg_no_rnd_vc1_chroma_pixels_tab[1](s->dest[2] + off, srcV, s->uvlinesize << fieldmv, 4, uvmx_field[i], uvmy_field[i]);
989cabdff1aSopenharmony_ci            }
990cabdff1aSopenharmony_ci        } else {
991cabdff1aSopenharmony_ci            if (!v->rnd) {
992cabdff1aSopenharmony_ci                h264chroma->put_h264_chroma_pixels_tab[1](s->dest[1] + off, srcU, s->uvlinesize << fieldmv, 4, uvmx_field[i], uvmy_field[i]);
993cabdff1aSopenharmony_ci                h264chroma->put_h264_chroma_pixels_tab[1](s->dest[2] + off, srcV, s->uvlinesize << fieldmv, 4, uvmx_field[i], uvmy_field[i]);
994cabdff1aSopenharmony_ci            } else {
995cabdff1aSopenharmony_ci                v->vc1dsp.put_no_rnd_vc1_chroma_pixels_tab[1](s->dest[1] + off, srcU, s->uvlinesize << fieldmv, 4, uvmx_field[i], uvmy_field[i]);
996cabdff1aSopenharmony_ci                v->vc1dsp.put_no_rnd_vc1_chroma_pixels_tab[1](s->dest[2] + off, srcV, s->uvlinesize << fieldmv, 4, uvmx_field[i], uvmy_field[i]);
997cabdff1aSopenharmony_ci            }
998cabdff1aSopenharmony_ci        }
999cabdff1aSopenharmony_ci    }
1000cabdff1aSopenharmony_ci}
1001cabdff1aSopenharmony_ci
1002cabdff1aSopenharmony_ci/** Motion compensation for direct or interpolated blocks in B-frames
1003cabdff1aSopenharmony_ci */
1004cabdff1aSopenharmony_civoid ff_vc1_interp_mc(VC1Context *v)
1005cabdff1aSopenharmony_ci{
1006cabdff1aSopenharmony_ci    MpegEncContext *s = &v->s;
1007cabdff1aSopenharmony_ci    H264ChromaContext *h264chroma = &v->h264chroma;
1008cabdff1aSopenharmony_ci    uint8_t *srcY, *srcU, *srcV;
1009cabdff1aSopenharmony_ci    int dxy, mx, my, uvmx, uvmy, src_x, src_y, uvsrc_x, uvsrc_y;
1010cabdff1aSopenharmony_ci    int v_edge_pos = s->v_edge_pos >> v->field_mode;
1011cabdff1aSopenharmony_ci    int use_ic = v->next_use_ic;
1012cabdff1aSopenharmony_ci    int interlace;
1013cabdff1aSopenharmony_ci    int linesize, uvlinesize;
1014cabdff1aSopenharmony_ci
1015cabdff1aSopenharmony_ci    if (!v->field_mode && !v->s.next_picture.f->data[0])
1016cabdff1aSopenharmony_ci        return;
1017cabdff1aSopenharmony_ci
1018cabdff1aSopenharmony_ci    linesize = s->current_picture_ptr->f->linesize[0];
1019cabdff1aSopenharmony_ci    uvlinesize = s->current_picture_ptr->f->linesize[1];
1020cabdff1aSopenharmony_ci
1021cabdff1aSopenharmony_ci    mx   = s->mv[1][0][0];
1022cabdff1aSopenharmony_ci    my   = s->mv[1][0][1];
1023cabdff1aSopenharmony_ci    uvmx = (mx + ((mx & 3) == 3)) >> 1;
1024cabdff1aSopenharmony_ci    uvmy = (my + ((my & 3) == 3)) >> 1;
1025cabdff1aSopenharmony_ci    if (v->field_mode && v->cur_field_type != v->ref_field_type[1]) {
1026cabdff1aSopenharmony_ci        my   = my   - 2 + 4 * v->cur_field_type;
1027cabdff1aSopenharmony_ci        uvmy = uvmy - 2 + 4 * v->cur_field_type;
1028cabdff1aSopenharmony_ci    }
1029cabdff1aSopenharmony_ci    if (v->fastuvmc) {
1030cabdff1aSopenharmony_ci        uvmx = uvmx + ((uvmx < 0) ? -(uvmx & 1) : (uvmx & 1));
1031cabdff1aSopenharmony_ci        uvmy = uvmy + ((uvmy < 0) ? -(uvmy & 1) : (uvmy & 1));
1032cabdff1aSopenharmony_ci    }
1033cabdff1aSopenharmony_ci    srcY = s->next_picture.f->data[0];
1034cabdff1aSopenharmony_ci    srcU = s->next_picture.f->data[1];
1035cabdff1aSopenharmony_ci    srcV = s->next_picture.f->data[2];
1036cabdff1aSopenharmony_ci
1037cabdff1aSopenharmony_ci    interlace = s->next_picture.f->interlaced_frame;
1038cabdff1aSopenharmony_ci
1039cabdff1aSopenharmony_ci    src_x   = s->mb_x * 16 + (mx   >> 2);
1040cabdff1aSopenharmony_ci    src_y   = s->mb_y * 16 + (my   >> 2);
1041cabdff1aSopenharmony_ci    uvsrc_x = s->mb_x *  8 + (uvmx >> 2);
1042cabdff1aSopenharmony_ci    uvsrc_y = s->mb_y *  8 + (uvmy >> 2);
1043cabdff1aSopenharmony_ci
1044cabdff1aSopenharmony_ci    if (v->profile != PROFILE_ADVANCED) {
1045cabdff1aSopenharmony_ci        src_x   = av_clip(  src_x, -16, s->mb_width  * 16);
1046cabdff1aSopenharmony_ci        src_y   = av_clip(  src_y, -16, s->mb_height * 16);
1047cabdff1aSopenharmony_ci        uvsrc_x = av_clip(uvsrc_x,  -8, s->mb_width  *  8);
1048cabdff1aSopenharmony_ci        uvsrc_y = av_clip(uvsrc_y,  -8, s->mb_height *  8);
1049cabdff1aSopenharmony_ci    } else {
1050cabdff1aSopenharmony_ci        src_x   = av_clip(  src_x, -17, s->avctx->coded_width);
1051cabdff1aSopenharmony_ci        uvsrc_x = av_clip(uvsrc_x,  -8, s->avctx->coded_width  >> 1);
1052cabdff1aSopenharmony_ci        if (v->fcm == ILACE_FRAME) {
1053cabdff1aSopenharmony_ci            src_y = av_clip(src_y, -18 + (src_y & 1), s->avctx->coded_height + (src_y & 1));
1054cabdff1aSopenharmony_ci            uvsrc_y = av_clip(uvsrc_y, -8 + (uvsrc_y & 1), (s->avctx->coded_height >> 1) + (uvsrc_y & 1));
1055cabdff1aSopenharmony_ci        } else {
1056cabdff1aSopenharmony_ci            src_y = av_clip(src_y, -18, s->avctx->coded_height + 1);
1057cabdff1aSopenharmony_ci            uvsrc_y = av_clip(uvsrc_y,  -8, s->avctx->coded_height >> 1);
1058cabdff1aSopenharmony_ci        }
1059cabdff1aSopenharmony_ci    }
1060cabdff1aSopenharmony_ci
1061cabdff1aSopenharmony_ci    srcY += src_y   * s->linesize   + src_x;
1062cabdff1aSopenharmony_ci    srcU += uvsrc_y * s->uvlinesize + uvsrc_x;
1063cabdff1aSopenharmony_ci    srcV += uvsrc_y * s->uvlinesize + uvsrc_x;
1064cabdff1aSopenharmony_ci
1065cabdff1aSopenharmony_ci    if (v->field_mode && v->ref_field_type[1]) {
1066cabdff1aSopenharmony_ci        srcY += linesize;
1067cabdff1aSopenharmony_ci        srcU += uvlinesize;
1068cabdff1aSopenharmony_ci        srcV += uvlinesize;
1069cabdff1aSopenharmony_ci    }
1070cabdff1aSopenharmony_ci
1071cabdff1aSopenharmony_ci    /* for grayscale we should not try to read from unknown area */
1072cabdff1aSopenharmony_ci    if (CONFIG_GRAY && s->avctx->flags & AV_CODEC_FLAG_GRAY) {
1073cabdff1aSopenharmony_ci        srcU = s->sc.edge_emu_buffer + 18 * s->linesize;
1074cabdff1aSopenharmony_ci        srcV = s->sc.edge_emu_buffer + 18 * s->linesize;
1075cabdff1aSopenharmony_ci    }
1076cabdff1aSopenharmony_ci
1077cabdff1aSopenharmony_ci    if (v->rangeredfrm || s->h_edge_pos < 22 || v_edge_pos < 22 || use_ic
1078cabdff1aSopenharmony_ci        || (unsigned)(src_x - 1) > s->h_edge_pos - (mx & 3) - 16 - 3
1079cabdff1aSopenharmony_ci        || (unsigned)(src_y - 1) > v_edge_pos    - (my & 3) - 16 - 3) {
1080cabdff1aSopenharmony_ci        uint8_t *ubuf = s->sc.edge_emu_buffer + 19 * s->linesize;
1081cabdff1aSopenharmony_ci        uint8_t *vbuf = ubuf + 9 * s->uvlinesize;
1082cabdff1aSopenharmony_ci        const int k = 17 + s->mspel * 2;
1083cabdff1aSopenharmony_ci
1084cabdff1aSopenharmony_ci        srcY -= s->mspel * (1 + s->linesize);
1085cabdff1aSopenharmony_ci        if (interlace) {
1086cabdff1aSopenharmony_ci            s->vdsp.emulated_edge_mc(s->sc.edge_emu_buffer,
1087cabdff1aSopenharmony_ci                                     srcY,
1088cabdff1aSopenharmony_ci                                     linesize << 1,
1089cabdff1aSopenharmony_ci                                     linesize << 1,
1090cabdff1aSopenharmony_ci                                     k,
1091cabdff1aSopenharmony_ci                                     v->field_mode ? k : (k + 1 >> 1),
1092cabdff1aSopenharmony_ci                                     src_x - s->mspel,
1093cabdff1aSopenharmony_ci                                     src_y - s->mspel >> !v->field_mode,
1094cabdff1aSopenharmony_ci                                     s->h_edge_pos,
1095cabdff1aSopenharmony_ci                                     s->v_edge_pos >> 1);
1096cabdff1aSopenharmony_ci            if (!v->field_mode)
1097cabdff1aSopenharmony_ci                s->vdsp.emulated_edge_mc(s->sc.edge_emu_buffer + linesize,
1098cabdff1aSopenharmony_ci                                         srcY + linesize,
1099cabdff1aSopenharmony_ci                                         linesize << 1,
1100cabdff1aSopenharmony_ci                                         linesize << 1,
1101cabdff1aSopenharmony_ci                                         k,
1102cabdff1aSopenharmony_ci                                         k >> 1,
1103cabdff1aSopenharmony_ci                                         src_x - s->mspel,
1104cabdff1aSopenharmony_ci                                         src_y - s->mspel + 1 >> 1,
1105cabdff1aSopenharmony_ci                                         s->h_edge_pos,
1106cabdff1aSopenharmony_ci                                         s->v_edge_pos >> 1);
1107cabdff1aSopenharmony_ci        } else
1108cabdff1aSopenharmony_ci            s->vdsp.emulated_edge_mc(s->sc.edge_emu_buffer,
1109cabdff1aSopenharmony_ci                                     srcY,
1110cabdff1aSopenharmony_ci                                     linesize,
1111cabdff1aSopenharmony_ci                                     linesize,
1112cabdff1aSopenharmony_ci                                     k,
1113cabdff1aSopenharmony_ci                                     v->field_mode ? (k << 1) - 1 : k,
1114cabdff1aSopenharmony_ci                                     src_x - s->mspel,
1115cabdff1aSopenharmony_ci                                     v->field_mode ? 2 * (src_y - s->mspel) + v->ref_field_type[1] :
1116cabdff1aSopenharmony_ci                                                     src_y - s->mspel,
1117cabdff1aSopenharmony_ci                                     s->h_edge_pos,
1118cabdff1aSopenharmony_ci                                     s->v_edge_pos);
1119cabdff1aSopenharmony_ci        srcY = s->sc.edge_emu_buffer;
1120cabdff1aSopenharmony_ci        if (interlace) {
1121cabdff1aSopenharmony_ci            s->vdsp.emulated_edge_mc(ubuf,
1122cabdff1aSopenharmony_ci                                     srcU,
1123cabdff1aSopenharmony_ci                                     uvlinesize << 1,
1124cabdff1aSopenharmony_ci                                     uvlinesize << 1,
1125cabdff1aSopenharmony_ci                                     9,
1126cabdff1aSopenharmony_ci                                     v->field_mode ? 9 : 5,
1127cabdff1aSopenharmony_ci                                     uvsrc_x,
1128cabdff1aSopenharmony_ci                                     uvsrc_y >> !v->field_mode,
1129cabdff1aSopenharmony_ci                                     s->h_edge_pos >> 1,
1130cabdff1aSopenharmony_ci                                     s->v_edge_pos >> 2);
1131cabdff1aSopenharmony_ci            s->vdsp.emulated_edge_mc(vbuf,
1132cabdff1aSopenharmony_ci                                     srcV,
1133cabdff1aSopenharmony_ci                                     uvlinesize << 1,
1134cabdff1aSopenharmony_ci                                     uvlinesize << 1,
1135cabdff1aSopenharmony_ci                                     9,
1136cabdff1aSopenharmony_ci                                     v->field_mode ? 9 : 5,
1137cabdff1aSopenharmony_ci                                     uvsrc_x,
1138cabdff1aSopenharmony_ci                                     uvsrc_y >> !v->field_mode,
1139cabdff1aSopenharmony_ci                                     s->h_edge_pos >> 1,
1140cabdff1aSopenharmony_ci                                     s->v_edge_pos >> 2);
1141cabdff1aSopenharmony_ci            if (!v->field_mode) {
1142cabdff1aSopenharmony_ci                s->vdsp.emulated_edge_mc(ubuf + uvlinesize,
1143cabdff1aSopenharmony_ci                                         srcU + uvlinesize,
1144cabdff1aSopenharmony_ci                                         uvlinesize << 1,
1145cabdff1aSopenharmony_ci                                         uvlinesize << 1,
1146cabdff1aSopenharmony_ci                                         9,
1147cabdff1aSopenharmony_ci                                         4,
1148cabdff1aSopenharmony_ci                                         uvsrc_x,
1149cabdff1aSopenharmony_ci                                         uvsrc_y + 1 >> 1,
1150cabdff1aSopenharmony_ci                                         s->h_edge_pos >> 1,
1151cabdff1aSopenharmony_ci                                         s->v_edge_pos >> 2);
1152cabdff1aSopenharmony_ci                s->vdsp.emulated_edge_mc(vbuf + uvlinesize,
1153cabdff1aSopenharmony_ci                                         srcV + uvlinesize,
1154cabdff1aSopenharmony_ci                                         uvlinesize << 1,
1155cabdff1aSopenharmony_ci                                         uvlinesize << 1,
1156cabdff1aSopenharmony_ci                                         9,
1157cabdff1aSopenharmony_ci                                         4,
1158cabdff1aSopenharmony_ci                                         uvsrc_x,
1159cabdff1aSopenharmony_ci                                         uvsrc_y + 1 >> 1,
1160cabdff1aSopenharmony_ci                                         s->h_edge_pos >> 1,
1161cabdff1aSopenharmony_ci                                         s->v_edge_pos >> 2);
1162cabdff1aSopenharmony_ci            }
1163cabdff1aSopenharmony_ci        } else {
1164cabdff1aSopenharmony_ci            s->vdsp.emulated_edge_mc(ubuf,
1165cabdff1aSopenharmony_ci                                     srcU,
1166cabdff1aSopenharmony_ci                                     uvlinesize,
1167cabdff1aSopenharmony_ci                                     uvlinesize,
1168cabdff1aSopenharmony_ci                                     9,
1169cabdff1aSopenharmony_ci                                     v->field_mode ? 17 : 9,
1170cabdff1aSopenharmony_ci                                     uvsrc_x,
1171cabdff1aSopenharmony_ci                                     v->field_mode ? 2 * uvsrc_y + v->ref_field_type[1] : uvsrc_y,
1172cabdff1aSopenharmony_ci                                     s->h_edge_pos >> 1,
1173cabdff1aSopenharmony_ci                                     s->v_edge_pos >> 1);
1174cabdff1aSopenharmony_ci            s->vdsp.emulated_edge_mc(vbuf,
1175cabdff1aSopenharmony_ci                                     srcV,
1176cabdff1aSopenharmony_ci                                     uvlinesize,
1177cabdff1aSopenharmony_ci                                     uvlinesize,
1178cabdff1aSopenharmony_ci                                     9,
1179cabdff1aSopenharmony_ci                                     v->field_mode ? 17 : 9,
1180cabdff1aSopenharmony_ci                                     uvsrc_x,
1181cabdff1aSopenharmony_ci                                     v->field_mode ? 2 * uvsrc_y + v->ref_field_type[1] : uvsrc_y,
1182cabdff1aSopenharmony_ci                                     s->h_edge_pos >> 1,
1183cabdff1aSopenharmony_ci                                     s->v_edge_pos >> 1);
1184cabdff1aSopenharmony_ci        }
1185cabdff1aSopenharmony_ci        srcU = ubuf;
1186cabdff1aSopenharmony_ci        srcV = vbuf;
1187cabdff1aSopenharmony_ci        /* if we deal with range reduction we need to scale source blocks */
1188cabdff1aSopenharmony_ci        if (v->rangeredfrm) {
1189cabdff1aSopenharmony_ci            vc1_scale_luma(srcY, k, s->linesize);
1190cabdff1aSopenharmony_ci            vc1_scale_chroma(srcU, srcV, 9, s->uvlinesize);
1191cabdff1aSopenharmony_ci        }
1192cabdff1aSopenharmony_ci
1193cabdff1aSopenharmony_ci        if (use_ic) {
1194cabdff1aSopenharmony_ci            uint8_t (*luty )[256] = v->next_luty;
1195cabdff1aSopenharmony_ci            uint8_t (*lutuv)[256] = v->next_lutuv;
1196cabdff1aSopenharmony_ci            vc1_lut_scale_luma(srcY,
1197cabdff1aSopenharmony_ci                               luty[v->field_mode ? v->ref_field_type[1] : ((0+src_y - s->mspel) & 1)],
1198cabdff1aSopenharmony_ci                               luty[v->field_mode ? v->ref_field_type[1] : ((1+src_y - s->mspel) & 1)],
1199cabdff1aSopenharmony_ci                               k, s->linesize);
1200cabdff1aSopenharmony_ci            vc1_lut_scale_chroma(srcU, srcV,
1201cabdff1aSopenharmony_ci                                 lutuv[v->field_mode ? v->ref_field_type[1] : ((0+uvsrc_y) & 1)],
1202cabdff1aSopenharmony_ci                                 lutuv[v->field_mode ? v->ref_field_type[1] : ((1+uvsrc_y) & 1)],
1203cabdff1aSopenharmony_ci                                 9, s->uvlinesize);
1204cabdff1aSopenharmony_ci        }
1205cabdff1aSopenharmony_ci        srcY += s->mspel * (1 + s->linesize);
1206cabdff1aSopenharmony_ci    }
1207cabdff1aSopenharmony_ci
1208cabdff1aSopenharmony_ci    if (s->mspel) {
1209cabdff1aSopenharmony_ci        dxy = ((my & 3) << 2) | (mx & 3);
1210cabdff1aSopenharmony_ci        v->vc1dsp.avg_vc1_mspel_pixels_tab[0][dxy](s->dest[0], srcY, s->linesize, v->rnd);
1211cabdff1aSopenharmony_ci    } else { // hpel mc
1212cabdff1aSopenharmony_ci        dxy = (my & 2) | ((mx & 2) >> 1);
1213cabdff1aSopenharmony_ci
1214cabdff1aSopenharmony_ci        if (!v->rnd)
1215cabdff1aSopenharmony_ci            s->hdsp.avg_pixels_tab[0][dxy](s->dest[0], srcY, s->linesize, 16);
1216cabdff1aSopenharmony_ci        else
1217cabdff1aSopenharmony_ci            s->hdsp.avg_no_rnd_pixels_tab[dxy](s->dest[0], srcY, s->linesize, 16);
1218cabdff1aSopenharmony_ci    }
1219cabdff1aSopenharmony_ci
1220cabdff1aSopenharmony_ci    if (CONFIG_GRAY && s->avctx->flags & AV_CODEC_FLAG_GRAY)
1221cabdff1aSopenharmony_ci        return;
1222cabdff1aSopenharmony_ci    /* Chroma MC always uses qpel bilinear */
1223cabdff1aSopenharmony_ci    uvmx = (uvmx & 3) << 1;
1224cabdff1aSopenharmony_ci    uvmy = (uvmy & 3) << 1;
1225cabdff1aSopenharmony_ci    if (!v->rnd) {
1226cabdff1aSopenharmony_ci        h264chroma->avg_h264_chroma_pixels_tab[0](s->dest[1], srcU, s->uvlinesize, 8, uvmx, uvmy);
1227cabdff1aSopenharmony_ci        h264chroma->avg_h264_chroma_pixels_tab[0](s->dest[2], srcV, s->uvlinesize, 8, uvmx, uvmy);
1228cabdff1aSopenharmony_ci    } else {
1229cabdff1aSopenharmony_ci        v->vc1dsp.avg_no_rnd_vc1_chroma_pixels_tab[0](s->dest[1], srcU, s->uvlinesize, 8, uvmx, uvmy);
1230cabdff1aSopenharmony_ci        v->vc1dsp.avg_no_rnd_vc1_chroma_pixels_tab[0](s->dest[2], srcV, s->uvlinesize, 8, uvmx, uvmy);
1231cabdff1aSopenharmony_ci    }
1232cabdff1aSopenharmony_ci}
1233