1*c0909341SAndroid Build Coastguard Worker /*
2*c0909341SAndroid Build Coastguard Worker * Copyright © 2018, VideoLAN and dav1d authors
3*c0909341SAndroid Build Coastguard Worker * Copyright © 2018, Two Orioles, LLC
4*c0909341SAndroid Build Coastguard Worker * All rights reserved.
5*c0909341SAndroid Build Coastguard Worker *
6*c0909341SAndroid Build Coastguard Worker * Redistribution and use in source and binary forms, with or without
7*c0909341SAndroid Build Coastguard Worker * modification, are permitted provided that the following conditions are met:
8*c0909341SAndroid Build Coastguard Worker *
9*c0909341SAndroid Build Coastguard Worker * 1. Redistributions of source code must retain the above copyright notice, this
10*c0909341SAndroid Build Coastguard Worker * list of conditions and the following disclaimer.
11*c0909341SAndroid Build Coastguard Worker *
12*c0909341SAndroid Build Coastguard Worker * 2. Redistributions in binary form must reproduce the above copyright notice,
13*c0909341SAndroid Build Coastguard Worker * this list of conditions and the following disclaimer in the documentation
14*c0909341SAndroid Build Coastguard Worker * and/or other materials provided with the distribution.
15*c0909341SAndroid Build Coastguard Worker *
16*c0909341SAndroid Build Coastguard Worker * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" AND
17*c0909341SAndroid Build Coastguard Worker * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED
18*c0909341SAndroid Build Coastguard Worker * WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
19*c0909341SAndroid Build Coastguard Worker * DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT OWNER OR CONTRIBUTORS BE LIABLE FOR
20*c0909341SAndroid Build Coastguard Worker * ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES
21*c0909341SAndroid Build Coastguard Worker * (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES;
22*c0909341SAndroid Build Coastguard Worker * LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND
23*c0909341SAndroid Build Coastguard Worker * ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
24*c0909341SAndroid Build Coastguard Worker * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS
25*c0909341SAndroid Build Coastguard Worker * SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
26*c0909341SAndroid Build Coastguard Worker */
27*c0909341SAndroid Build Coastguard Worker
28*c0909341SAndroid Build Coastguard Worker #include "config.h"
29*c0909341SAndroid Build Coastguard Worker
30*c0909341SAndroid Build Coastguard Worker #include <string.h>
31*c0909341SAndroid Build Coastguard Worker
32*c0909341SAndroid Build Coastguard Worker #include "common/intops.h"
33*c0909341SAndroid Build Coastguard Worker
34*c0909341SAndroid Build Coastguard Worker #include "src/cdef_apply.h"
35*c0909341SAndroid Build Coastguard Worker
36*c0909341SAndroid Build Coastguard Worker enum Backup2x8Flags {
37*c0909341SAndroid Build Coastguard Worker BACKUP_2X8_Y = 1 << 0,
38*c0909341SAndroid Build Coastguard Worker BACKUP_2X8_UV = 1 << 1,
39*c0909341SAndroid Build Coastguard Worker };
40*c0909341SAndroid Build Coastguard Worker
backup2lines(pixel * const dst[3],pixel * const src[3],const ptrdiff_t stride[2],const enum Dav1dPixelLayout layout)41*c0909341SAndroid Build Coastguard Worker static void backup2lines(pixel *const dst[3], /*const*/ pixel *const src[3],
42*c0909341SAndroid Build Coastguard Worker const ptrdiff_t stride[2],
43*c0909341SAndroid Build Coastguard Worker const enum Dav1dPixelLayout layout)
44*c0909341SAndroid Build Coastguard Worker {
45*c0909341SAndroid Build Coastguard Worker const ptrdiff_t y_stride = PXSTRIDE(stride[0]);
46*c0909341SAndroid Build Coastguard Worker if (y_stride < 0)
47*c0909341SAndroid Build Coastguard Worker pixel_copy(dst[0] + y_stride, src[0] + 7 * y_stride, -2 * y_stride);
48*c0909341SAndroid Build Coastguard Worker else
49*c0909341SAndroid Build Coastguard Worker pixel_copy(dst[0], src[0] + 6 * y_stride, 2 * y_stride);
50*c0909341SAndroid Build Coastguard Worker
51*c0909341SAndroid Build Coastguard Worker if (layout != DAV1D_PIXEL_LAYOUT_I400) {
52*c0909341SAndroid Build Coastguard Worker const ptrdiff_t uv_stride = PXSTRIDE(stride[1]);
53*c0909341SAndroid Build Coastguard Worker if (uv_stride < 0) {
54*c0909341SAndroid Build Coastguard Worker const int uv_off = layout == DAV1D_PIXEL_LAYOUT_I420 ? 3 : 7;
55*c0909341SAndroid Build Coastguard Worker pixel_copy(dst[1] + uv_stride, src[1] + uv_off * uv_stride, -2 * uv_stride);
56*c0909341SAndroid Build Coastguard Worker pixel_copy(dst[2] + uv_stride, src[2] + uv_off * uv_stride, -2 * uv_stride);
57*c0909341SAndroid Build Coastguard Worker } else {
58*c0909341SAndroid Build Coastguard Worker const int uv_off = layout == DAV1D_PIXEL_LAYOUT_I420 ? 2 : 6;
59*c0909341SAndroid Build Coastguard Worker pixel_copy(dst[1], src[1] + uv_off * uv_stride, 2 * uv_stride);
60*c0909341SAndroid Build Coastguard Worker pixel_copy(dst[2], src[2] + uv_off * uv_stride, 2 * uv_stride);
61*c0909341SAndroid Build Coastguard Worker }
62*c0909341SAndroid Build Coastguard Worker }
63*c0909341SAndroid Build Coastguard Worker }
64*c0909341SAndroid Build Coastguard Worker
backup2x8(pixel dst[3][8][2],pixel * const src[3],const ptrdiff_t src_stride[2],int x_off,const enum Dav1dPixelLayout layout,const enum Backup2x8Flags flag)65*c0909341SAndroid Build Coastguard Worker static void backup2x8(pixel dst[3][8][2],
66*c0909341SAndroid Build Coastguard Worker /*const*/ pixel *const src[3],
67*c0909341SAndroid Build Coastguard Worker const ptrdiff_t src_stride[2], int x_off,
68*c0909341SAndroid Build Coastguard Worker const enum Dav1dPixelLayout layout,
69*c0909341SAndroid Build Coastguard Worker const enum Backup2x8Flags flag)
70*c0909341SAndroid Build Coastguard Worker {
71*c0909341SAndroid Build Coastguard Worker ptrdiff_t y_off = 0;
72*c0909341SAndroid Build Coastguard Worker if (flag & BACKUP_2X8_Y) {
73*c0909341SAndroid Build Coastguard Worker for (int y = 0; y < 8; y++, y_off += PXSTRIDE(src_stride[0]))
74*c0909341SAndroid Build Coastguard Worker pixel_copy(dst[0][y], &src[0][y_off + x_off - 2], 2);
75*c0909341SAndroid Build Coastguard Worker }
76*c0909341SAndroid Build Coastguard Worker
77*c0909341SAndroid Build Coastguard Worker if (layout == DAV1D_PIXEL_LAYOUT_I400 || !(flag & BACKUP_2X8_UV))
78*c0909341SAndroid Build Coastguard Worker return;
79*c0909341SAndroid Build Coastguard Worker
80*c0909341SAndroid Build Coastguard Worker const int ss_ver = layout == DAV1D_PIXEL_LAYOUT_I420;
81*c0909341SAndroid Build Coastguard Worker const int ss_hor = layout != DAV1D_PIXEL_LAYOUT_I444;
82*c0909341SAndroid Build Coastguard Worker
83*c0909341SAndroid Build Coastguard Worker x_off >>= ss_hor;
84*c0909341SAndroid Build Coastguard Worker y_off = 0;
85*c0909341SAndroid Build Coastguard Worker for (int y = 0; y < (8 >> ss_ver); y++, y_off += PXSTRIDE(src_stride[1])) {
86*c0909341SAndroid Build Coastguard Worker pixel_copy(dst[1][y], &src[1][y_off + x_off - 2], 2);
87*c0909341SAndroid Build Coastguard Worker pixel_copy(dst[2][y], &src[2][y_off + x_off - 2], 2);
88*c0909341SAndroid Build Coastguard Worker }
89*c0909341SAndroid Build Coastguard Worker }
90*c0909341SAndroid Build Coastguard Worker
adjust_strength(const int strength,const unsigned var)91*c0909341SAndroid Build Coastguard Worker static int adjust_strength(const int strength, const unsigned var) {
92*c0909341SAndroid Build Coastguard Worker if (!var) return 0;
93*c0909341SAndroid Build Coastguard Worker const int i = var >> 6 ? imin(ulog2(var >> 6), 12) : 0;
94*c0909341SAndroid Build Coastguard Worker return (strength * (4 + i) + 8) >> 4;
95*c0909341SAndroid Build Coastguard Worker }
96*c0909341SAndroid Build Coastguard Worker
bytefn(dav1d_cdef_brow)97*c0909341SAndroid Build Coastguard Worker void bytefn(dav1d_cdef_brow)(Dav1dTaskContext *const tc,
98*c0909341SAndroid Build Coastguard Worker pixel *const p[3],
99*c0909341SAndroid Build Coastguard Worker const Av1Filter *const lflvl,
100*c0909341SAndroid Build Coastguard Worker const int by_start, const int by_end,
101*c0909341SAndroid Build Coastguard Worker const int sbrow_start, const int sby)
102*c0909341SAndroid Build Coastguard Worker {
103*c0909341SAndroid Build Coastguard Worker Dav1dFrameContext *const f = (Dav1dFrameContext *)tc->f;
104*c0909341SAndroid Build Coastguard Worker const int bitdepth_min_8 = BITDEPTH == 8 ? 0 : f->cur.p.bpc - 8;
105*c0909341SAndroid Build Coastguard Worker const Dav1dDSPContext *const dsp = f->dsp;
106*c0909341SAndroid Build Coastguard Worker enum CdefEdgeFlags edges = CDEF_HAVE_BOTTOM | (by_start > 0 ? CDEF_HAVE_TOP : 0);
107*c0909341SAndroid Build Coastguard Worker pixel *ptrs[3] = { p[0], p[1], p[2] };
108*c0909341SAndroid Build Coastguard Worker const int sbsz = 16;
109*c0909341SAndroid Build Coastguard Worker const int sb64w = f->sb128w << 1;
110*c0909341SAndroid Build Coastguard Worker const int damping = f->frame_hdr->cdef.damping + bitdepth_min_8;
111*c0909341SAndroid Build Coastguard Worker const enum Dav1dPixelLayout layout = f->cur.p.layout;
112*c0909341SAndroid Build Coastguard Worker const int uv_idx = DAV1D_PIXEL_LAYOUT_I444 - layout;
113*c0909341SAndroid Build Coastguard Worker const int ss_ver = layout == DAV1D_PIXEL_LAYOUT_I420;
114*c0909341SAndroid Build Coastguard Worker const int ss_hor = layout != DAV1D_PIXEL_LAYOUT_I444;
115*c0909341SAndroid Build Coastguard Worker static const uint8_t uv_dirs[2][8] = { { 0, 1, 2, 3, 4, 5, 6, 7 },
116*c0909341SAndroid Build Coastguard Worker { 7, 0, 2, 4, 5, 6, 6, 6 } };
117*c0909341SAndroid Build Coastguard Worker const uint8_t *uv_dir = uv_dirs[layout == DAV1D_PIXEL_LAYOUT_I422];
118*c0909341SAndroid Build Coastguard Worker const int have_tt = f->c->n_tc > 1;
119*c0909341SAndroid Build Coastguard Worker const int sb128 = f->seq_hdr->sb128;
120*c0909341SAndroid Build Coastguard Worker const int resize = f->frame_hdr->width[0] != f->frame_hdr->width[1];
121*c0909341SAndroid Build Coastguard Worker const ptrdiff_t y_stride = PXSTRIDE(f->cur.stride[0]);
122*c0909341SAndroid Build Coastguard Worker const ptrdiff_t uv_stride = PXSTRIDE(f->cur.stride[1]);
123*c0909341SAndroid Build Coastguard Worker
124*c0909341SAndroid Build Coastguard Worker for (int bit = 0, by = by_start; by < by_end; by += 2, edges |= CDEF_HAVE_TOP) {
125*c0909341SAndroid Build Coastguard Worker const int tf = tc->top_pre_cdef_toggle;
126*c0909341SAndroid Build Coastguard Worker const int by_idx = (by & 30) >> 1;
127*c0909341SAndroid Build Coastguard Worker if (by + 2 >= f->bh) edges &= ~CDEF_HAVE_BOTTOM;
128*c0909341SAndroid Build Coastguard Worker
129*c0909341SAndroid Build Coastguard Worker if ((!have_tt || sbrow_start || by + 2 < by_end) &&
130*c0909341SAndroid Build Coastguard Worker edges & CDEF_HAVE_BOTTOM)
131*c0909341SAndroid Build Coastguard Worker {
132*c0909341SAndroid Build Coastguard Worker // backup pre-filter data for next iteration
133*c0909341SAndroid Build Coastguard Worker pixel *const cdef_top_bak[3] = {
134*c0909341SAndroid Build Coastguard Worker f->lf.cdef_line[!tf][0] + have_tt * sby * 4 * y_stride,
135*c0909341SAndroid Build Coastguard Worker f->lf.cdef_line[!tf][1] + have_tt * sby * 8 * uv_stride,
136*c0909341SAndroid Build Coastguard Worker f->lf.cdef_line[!tf][2] + have_tt * sby * 8 * uv_stride
137*c0909341SAndroid Build Coastguard Worker };
138*c0909341SAndroid Build Coastguard Worker backup2lines(cdef_top_bak, ptrs, f->cur.stride, layout);
139*c0909341SAndroid Build Coastguard Worker }
140*c0909341SAndroid Build Coastguard Worker
141*c0909341SAndroid Build Coastguard Worker ALIGN_STK_16(pixel, lr_bak, 2 /* idx */, [3 /* plane */][8 /* y */][2 /* x */]);
142*c0909341SAndroid Build Coastguard Worker pixel *iptrs[3] = { ptrs[0], ptrs[1], ptrs[2] };
143*c0909341SAndroid Build Coastguard Worker edges &= ~CDEF_HAVE_LEFT;
144*c0909341SAndroid Build Coastguard Worker edges |= CDEF_HAVE_RIGHT;
145*c0909341SAndroid Build Coastguard Worker enum Backup2x8Flags prev_flag = 0;
146*c0909341SAndroid Build Coastguard Worker for (int sbx = 0, last_skip = 1; sbx < sb64w; sbx++, edges |= CDEF_HAVE_LEFT) {
147*c0909341SAndroid Build Coastguard Worker const int sb128x = sbx >> 1;
148*c0909341SAndroid Build Coastguard Worker const int sb64_idx = ((by & sbsz) >> 3) + (sbx & 1);
149*c0909341SAndroid Build Coastguard Worker const int cdef_idx = lflvl[sb128x].cdef_idx[sb64_idx];
150*c0909341SAndroid Build Coastguard Worker if (cdef_idx == -1 ||
151*c0909341SAndroid Build Coastguard Worker (!f->frame_hdr->cdef.y_strength[cdef_idx] &&
152*c0909341SAndroid Build Coastguard Worker !f->frame_hdr->cdef.uv_strength[cdef_idx]))
153*c0909341SAndroid Build Coastguard Worker {
154*c0909341SAndroid Build Coastguard Worker last_skip = 1;
155*c0909341SAndroid Build Coastguard Worker goto next_sb;
156*c0909341SAndroid Build Coastguard Worker }
157*c0909341SAndroid Build Coastguard Worker
158*c0909341SAndroid Build Coastguard Worker // Create a complete 32-bit mask for the sb row ahead of time.
159*c0909341SAndroid Build Coastguard Worker const uint16_t (*noskip_row)[2] = &lflvl[sb128x].noskip_mask[by_idx];
160*c0909341SAndroid Build Coastguard Worker const unsigned noskip_mask = (unsigned) noskip_row[0][1] << 16 |
161*c0909341SAndroid Build Coastguard Worker noskip_row[0][0];
162*c0909341SAndroid Build Coastguard Worker
163*c0909341SAndroid Build Coastguard Worker const int y_lvl = f->frame_hdr->cdef.y_strength[cdef_idx];
164*c0909341SAndroid Build Coastguard Worker const int uv_lvl = f->frame_hdr->cdef.uv_strength[cdef_idx];
165*c0909341SAndroid Build Coastguard Worker const enum Backup2x8Flags flag = !!y_lvl + (!!uv_lvl << 1);
166*c0909341SAndroid Build Coastguard Worker
167*c0909341SAndroid Build Coastguard Worker const int y_pri_lvl = (y_lvl >> 2) << bitdepth_min_8;
168*c0909341SAndroid Build Coastguard Worker int y_sec_lvl = y_lvl & 3;
169*c0909341SAndroid Build Coastguard Worker y_sec_lvl += y_sec_lvl == 3;
170*c0909341SAndroid Build Coastguard Worker y_sec_lvl <<= bitdepth_min_8;
171*c0909341SAndroid Build Coastguard Worker
172*c0909341SAndroid Build Coastguard Worker const int uv_pri_lvl = (uv_lvl >> 2) << bitdepth_min_8;
173*c0909341SAndroid Build Coastguard Worker int uv_sec_lvl = uv_lvl & 3;
174*c0909341SAndroid Build Coastguard Worker uv_sec_lvl += uv_sec_lvl == 3;
175*c0909341SAndroid Build Coastguard Worker uv_sec_lvl <<= bitdepth_min_8;
176*c0909341SAndroid Build Coastguard Worker
177*c0909341SAndroid Build Coastguard Worker pixel *bptrs[3] = { iptrs[0], iptrs[1], iptrs[2] };
178*c0909341SAndroid Build Coastguard Worker for (int bx = sbx * sbsz; bx < imin((sbx + 1) * sbsz, f->bw);
179*c0909341SAndroid Build Coastguard Worker bx += 2, edges |= CDEF_HAVE_LEFT)
180*c0909341SAndroid Build Coastguard Worker {
181*c0909341SAndroid Build Coastguard Worker if (bx + 2 >= f->bw) edges &= ~CDEF_HAVE_RIGHT;
182*c0909341SAndroid Build Coastguard Worker
183*c0909341SAndroid Build Coastguard Worker // check if this 8x8 block had any coded coefficients; if not,
184*c0909341SAndroid Build Coastguard Worker // go to the next block
185*c0909341SAndroid Build Coastguard Worker const uint32_t bx_mask = 3U << (bx & 30);
186*c0909341SAndroid Build Coastguard Worker if (!(noskip_mask & bx_mask)) {
187*c0909341SAndroid Build Coastguard Worker last_skip = 1;
188*c0909341SAndroid Build Coastguard Worker goto next_b;
189*c0909341SAndroid Build Coastguard Worker }
190*c0909341SAndroid Build Coastguard Worker const int do_left = last_skip ? flag : (prev_flag ^ flag) & flag;
191*c0909341SAndroid Build Coastguard Worker prev_flag = flag;
192*c0909341SAndroid Build Coastguard Worker if (do_left && edges & CDEF_HAVE_LEFT) {
193*c0909341SAndroid Build Coastguard Worker // we didn't backup the prefilter data because it wasn't
194*c0909341SAndroid Build Coastguard Worker // there, so do it here instead
195*c0909341SAndroid Build Coastguard Worker backup2x8(lr_bak[bit], bptrs, f->cur.stride, 0, layout, do_left);
196*c0909341SAndroid Build Coastguard Worker }
197*c0909341SAndroid Build Coastguard Worker if (edges & CDEF_HAVE_RIGHT) {
198*c0909341SAndroid Build Coastguard Worker // backup pre-filter data for next iteration
199*c0909341SAndroid Build Coastguard Worker backup2x8(lr_bak[!bit], bptrs, f->cur.stride, 8, layout, flag);
200*c0909341SAndroid Build Coastguard Worker }
201*c0909341SAndroid Build Coastguard Worker
202*c0909341SAndroid Build Coastguard Worker int dir;
203*c0909341SAndroid Build Coastguard Worker unsigned variance;
204*c0909341SAndroid Build Coastguard Worker if (y_pri_lvl || uv_pri_lvl)
205*c0909341SAndroid Build Coastguard Worker dir = dsp->cdef.dir(bptrs[0], f->cur.stride[0],
206*c0909341SAndroid Build Coastguard Worker &variance HIGHBD_CALL_SUFFIX);
207*c0909341SAndroid Build Coastguard Worker
208*c0909341SAndroid Build Coastguard Worker const pixel *top, *bot;
209*c0909341SAndroid Build Coastguard Worker ptrdiff_t offset;
210*c0909341SAndroid Build Coastguard Worker
211*c0909341SAndroid Build Coastguard Worker if (!have_tt) goto st_y;
212*c0909341SAndroid Build Coastguard Worker if (sbrow_start && by == by_start) {
213*c0909341SAndroid Build Coastguard Worker if (resize) {
214*c0909341SAndroid Build Coastguard Worker offset = (sby - 1) * 4 * y_stride + bx * 4;
215*c0909341SAndroid Build Coastguard Worker top = &f->lf.cdef_lpf_line[0][offset];
216*c0909341SAndroid Build Coastguard Worker } else {
217*c0909341SAndroid Build Coastguard Worker offset = (sby * (4 << sb128) - 4) * y_stride + bx * 4;
218*c0909341SAndroid Build Coastguard Worker top = &f->lf.lr_lpf_line[0][offset];
219*c0909341SAndroid Build Coastguard Worker }
220*c0909341SAndroid Build Coastguard Worker bot = bptrs[0] + 8 * y_stride;
221*c0909341SAndroid Build Coastguard Worker } else if (!sbrow_start && by + 2 >= by_end) {
222*c0909341SAndroid Build Coastguard Worker top = &f->lf.cdef_line[tf][0][sby * 4 * y_stride + bx * 4];
223*c0909341SAndroid Build Coastguard Worker if (resize) {
224*c0909341SAndroid Build Coastguard Worker offset = (sby * 4 + 2) * y_stride + bx * 4;
225*c0909341SAndroid Build Coastguard Worker bot = &f->lf.cdef_lpf_line[0][offset];
226*c0909341SAndroid Build Coastguard Worker } else {
227*c0909341SAndroid Build Coastguard Worker const int line = sby * (4 << sb128) + 4 * sb128 + 2;
228*c0909341SAndroid Build Coastguard Worker offset = line * y_stride + bx * 4;
229*c0909341SAndroid Build Coastguard Worker bot = &f->lf.lr_lpf_line[0][offset];
230*c0909341SAndroid Build Coastguard Worker }
231*c0909341SAndroid Build Coastguard Worker } else {
232*c0909341SAndroid Build Coastguard Worker st_y:;
233*c0909341SAndroid Build Coastguard Worker offset = sby * 4 * y_stride;
234*c0909341SAndroid Build Coastguard Worker top = &f->lf.cdef_line[tf][0][have_tt * offset + bx * 4];
235*c0909341SAndroid Build Coastguard Worker bot = bptrs[0] + 8 * y_stride;
236*c0909341SAndroid Build Coastguard Worker }
237*c0909341SAndroid Build Coastguard Worker if (y_pri_lvl) {
238*c0909341SAndroid Build Coastguard Worker const int adj_y_pri_lvl = adjust_strength(y_pri_lvl, variance);
239*c0909341SAndroid Build Coastguard Worker if (adj_y_pri_lvl || y_sec_lvl)
240*c0909341SAndroid Build Coastguard Worker dsp->cdef.fb[0](bptrs[0], f->cur.stride[0], lr_bak[bit][0],
241*c0909341SAndroid Build Coastguard Worker top, bot, adj_y_pri_lvl, y_sec_lvl,
242*c0909341SAndroid Build Coastguard Worker dir, damping, edges HIGHBD_CALL_SUFFIX);
243*c0909341SAndroid Build Coastguard Worker } else if (y_sec_lvl)
244*c0909341SAndroid Build Coastguard Worker dsp->cdef.fb[0](bptrs[0], f->cur.stride[0], lr_bak[bit][0],
245*c0909341SAndroid Build Coastguard Worker top, bot, 0, y_sec_lvl, 0, damping,
246*c0909341SAndroid Build Coastguard Worker edges HIGHBD_CALL_SUFFIX);
247*c0909341SAndroid Build Coastguard Worker
248*c0909341SAndroid Build Coastguard Worker if (!uv_lvl) goto skip_uv;
249*c0909341SAndroid Build Coastguard Worker assert(layout != DAV1D_PIXEL_LAYOUT_I400);
250*c0909341SAndroid Build Coastguard Worker
251*c0909341SAndroid Build Coastguard Worker const int uvdir = uv_pri_lvl ? uv_dir[dir] : 0;
252*c0909341SAndroid Build Coastguard Worker for (int pl = 1; pl <= 2; pl++) {
253*c0909341SAndroid Build Coastguard Worker if (!have_tt) goto st_uv;
254*c0909341SAndroid Build Coastguard Worker if (sbrow_start && by == by_start) {
255*c0909341SAndroid Build Coastguard Worker if (resize) {
256*c0909341SAndroid Build Coastguard Worker offset = (sby - 1) * 4 * uv_stride + (bx * 4 >> ss_hor);
257*c0909341SAndroid Build Coastguard Worker top = &f->lf.cdef_lpf_line[pl][offset];
258*c0909341SAndroid Build Coastguard Worker } else {
259*c0909341SAndroid Build Coastguard Worker const int line = sby * (4 << sb128) - 4;
260*c0909341SAndroid Build Coastguard Worker offset = line * uv_stride + (bx * 4 >> ss_hor);
261*c0909341SAndroid Build Coastguard Worker top = &f->lf.lr_lpf_line[pl][offset];
262*c0909341SAndroid Build Coastguard Worker }
263*c0909341SAndroid Build Coastguard Worker bot = bptrs[pl] + (8 >> ss_ver) * uv_stride;
264*c0909341SAndroid Build Coastguard Worker } else if (!sbrow_start && by + 2 >= by_end) {
265*c0909341SAndroid Build Coastguard Worker const ptrdiff_t top_offset = sby * 8 * uv_stride +
266*c0909341SAndroid Build Coastguard Worker (bx * 4 >> ss_hor);
267*c0909341SAndroid Build Coastguard Worker top = &f->lf.cdef_line[tf][pl][top_offset];
268*c0909341SAndroid Build Coastguard Worker if (resize) {
269*c0909341SAndroid Build Coastguard Worker offset = (sby * 4 + 2) * uv_stride + (bx * 4 >> ss_hor);
270*c0909341SAndroid Build Coastguard Worker bot = &f->lf.cdef_lpf_line[pl][offset];
271*c0909341SAndroid Build Coastguard Worker } else {
272*c0909341SAndroid Build Coastguard Worker const int line = sby * (4 << sb128) + 4 * sb128 + 2;
273*c0909341SAndroid Build Coastguard Worker offset = line * uv_stride + (bx * 4 >> ss_hor);
274*c0909341SAndroid Build Coastguard Worker bot = &f->lf.lr_lpf_line[pl][offset];
275*c0909341SAndroid Build Coastguard Worker }
276*c0909341SAndroid Build Coastguard Worker } else {
277*c0909341SAndroid Build Coastguard Worker st_uv:;
278*c0909341SAndroid Build Coastguard Worker const ptrdiff_t offset = sby * 8 * uv_stride;
279*c0909341SAndroid Build Coastguard Worker top = &f->lf.cdef_line[tf][pl][have_tt * offset + (bx * 4 >> ss_hor)];
280*c0909341SAndroid Build Coastguard Worker bot = bptrs[pl] + (8 >> ss_ver) * uv_stride;
281*c0909341SAndroid Build Coastguard Worker }
282*c0909341SAndroid Build Coastguard Worker dsp->cdef.fb[uv_idx](bptrs[pl], f->cur.stride[1],
283*c0909341SAndroid Build Coastguard Worker lr_bak[bit][pl], top, bot,
284*c0909341SAndroid Build Coastguard Worker uv_pri_lvl, uv_sec_lvl, uvdir,
285*c0909341SAndroid Build Coastguard Worker damping - 1, edges HIGHBD_CALL_SUFFIX);
286*c0909341SAndroid Build Coastguard Worker }
287*c0909341SAndroid Build Coastguard Worker
288*c0909341SAndroid Build Coastguard Worker skip_uv:
289*c0909341SAndroid Build Coastguard Worker bit ^= 1;
290*c0909341SAndroid Build Coastguard Worker last_skip = 0;
291*c0909341SAndroid Build Coastguard Worker
292*c0909341SAndroid Build Coastguard Worker next_b:
293*c0909341SAndroid Build Coastguard Worker bptrs[0] += 8;
294*c0909341SAndroid Build Coastguard Worker bptrs[1] += 8 >> ss_hor;
295*c0909341SAndroid Build Coastguard Worker bptrs[2] += 8 >> ss_hor;
296*c0909341SAndroid Build Coastguard Worker }
297*c0909341SAndroid Build Coastguard Worker
298*c0909341SAndroid Build Coastguard Worker next_sb:
299*c0909341SAndroid Build Coastguard Worker iptrs[0] += sbsz * 4;
300*c0909341SAndroid Build Coastguard Worker iptrs[1] += sbsz * 4 >> ss_hor;
301*c0909341SAndroid Build Coastguard Worker iptrs[2] += sbsz * 4 >> ss_hor;
302*c0909341SAndroid Build Coastguard Worker }
303*c0909341SAndroid Build Coastguard Worker
304*c0909341SAndroid Build Coastguard Worker ptrs[0] += 8 * PXSTRIDE(f->cur.stride[0]);
305*c0909341SAndroid Build Coastguard Worker ptrs[1] += 8 * PXSTRIDE(f->cur.stride[1]) >> ss_ver;
306*c0909341SAndroid Build Coastguard Worker ptrs[2] += 8 * PXSTRIDE(f->cur.stride[1]) >> ss_ver;
307*c0909341SAndroid Build Coastguard Worker tc->top_pre_cdef_toggle ^= 1;
308*c0909341SAndroid Build Coastguard Worker }
309*c0909341SAndroid Build Coastguard Worker }
310