Coverage Report

Created: 2026-06-15 06:24

next uncovered line (L), next uncovered region (R), next uncovered branch (B)
/work/dav1d/src/lf_mask.c
Line
Count
Source
1
/*
2
 * Copyright © 2018, VideoLAN and dav1d authors
3
 * Copyright © 2018, Two Orioles, LLC
4
 * All rights reserved.
5
 *
6
 * Redistribution and use in source and binary forms, with or without
7
 * modification, are permitted provided that the following conditions are met:
8
 *
9
 * 1. Redistributions of source code must retain the above copyright notice, this
10
 *    list of conditions and the following disclaimer.
11
 *
12
 * 2. Redistributions in binary form must reproduce the above copyright notice,
13
 *    this list of conditions and the following disclaimer in the documentation
14
 *    and/or other materials provided with the distribution.
15
 *
16
 * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" AND
17
 * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED
18
 * WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
19
 * DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT OWNER OR CONTRIBUTORS BE LIABLE FOR
20
 * ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES
21
 * (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES;
22
 * LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND
23
 * ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
24
 * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS
25
 * SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
26
 */
27
28
#include "config.h"
29
30
#include <string.h>
31
32
#include "common/intops.h"
33
34
#include "src/ctx.h"
35
#include "src/levels.h"
36
#include "src/lf_mask.h"
37
#include "src/tables.h"
38
39
static void decomp_tx(uint8_t (*const txa)[2 /* txsz, step */][32 /* y */][32 /* x */],
40
                      const enum RectTxfmSize from,
41
                      const int depth,
42
                      const int y_off, const int x_off,
43
                      const uint16_t *const tx_masks)
44
169k
{
45
169k
    const TxfmInfo *const t_dim = &dav1d_txfm_dimensions[from];
46
169k
    const int is_split = (from == (int) TX_4X4 || depth > 1) ? 0 :
47
169k
        (tx_masks[depth] >> (y_off * 4 + x_off)) & 1;
48
49
169k
    if (is_split) {
50
14.8k
        const enum RectTxfmSize sub = t_dim->sub;
51
14.8k
        const int htw4 = t_dim->w >> 1, hth4 = t_dim->h >> 1;
52
53
14.8k
        decomp_tx(txa, sub, depth + 1, y_off * 2 + 0, x_off * 2 + 0, tx_masks);
54
14.8k
        if (t_dim->w >= t_dim->h)
55
11.6k
            decomp_tx((uint8_t(*)[2][32][32]) &txa[0][0][0][htw4],
56
11.6k
                      sub, depth + 1, y_off * 2 + 0, x_off * 2 + 1, tx_masks);
57
14.8k
        if (t_dim->h >= t_dim->w) {
58
11.0k
            decomp_tx((uint8_t(*)[2][32][32]) &txa[0][0][hth4][0],
59
11.0k
                      sub, depth + 1, y_off * 2 + 1, x_off * 2 + 0, tx_masks);
60
11.0k
            if (t_dim->w >= t_dim->h)
61
7.77k
                decomp_tx((uint8_t(*)[2][32][32]) &txa[0][0][hth4][htw4],
62
7.77k
                          sub, depth + 1, y_off * 2 + 1, x_off * 2 + 1, tx_masks);
63
11.0k
        }
64
154k
    } else {
65
154k
        const int lw = imin(2, t_dim->lw), lh = imin(2, t_dim->lh);
66
67
154k
#define set_ctx(rep_macro) \
68
660k
        for (int y = 0; y < t_dim->h; y++) { \
69
505k
            rep_macro(txa[0][0][y], 0, lw); \
70
505k
            rep_macro(txa[1][0][y], 0, lh); \
71
505k
            txa[0][1][y][0] = t_dim->w; \
72
505k
        }
73
154k
        case_set_upto16(t_dim->lw);
74
154k
#undef set_ctx
75
154k
        dav1d_memset_pow2[t_dim->lw](txa[1][1][0], t_dim->h);
76
154k
    }
77
169k
}
78
79
static inline void mask_edges_inter(uint16_t (*const masks)[32][3][2],
80
                                    const int by4, const int bx4,
81
                                    const int w4, const int h4, const int skip,
82
                                    const enum RectTxfmSize max_tx,
83
                                    const uint16_t *const tx_masks,
84
                                    uint8_t *const a, uint8_t *const l)
85
114k
{
86
114k
    const TxfmInfo *const t_dim = &dav1d_txfm_dimensions[max_tx];
87
114k
    int y, x;
88
89
114k
    ALIGN_STK_16(uint8_t, txa, 2 /* edge */, [2 /* txsz, step */][32 /* y */][32 /* x */]);
90
232k
    for (int y_off = 0, y = 0; y < h4; y += t_dim->h, y_off++)
91
241k
        for (int x_off = 0, x = 0; x < w4; x += t_dim->w, x_off++)
92
124k
            decomp_tx((uint8_t(*)[2][32][32]) &txa[0][0][y][x],
93
124k
                      max_tx, 0, y_off, x_off, tx_masks);
94
95
    // left block edge
96
114k
    unsigned mask = 1U << by4;
97
517k
    for (y = 0; y < h4; y++, mask <<= 1) {
98
402k
        const int sidx = mask >= 0x10000;
99
402k
        const unsigned smask = mask >> (sidx << 4);
100
402k
        masks[0][bx4][imin(txa[0][0][y][0], l[y])][sidx] |= smask;
101
402k
    }
102
103
    // top block edge
104
485k
    for (x = 0, mask = 1U << bx4; x < w4; x++, mask <<= 1) {
105
371k
        const int sidx = mask >= 0x10000;
106
371k
        const unsigned smask = mask >> (sidx << 4);
107
371k
        masks[1][by4][imin(txa[1][0][0][x], a[x])][sidx] |= smask;
108
371k
    }
109
110
114k
    if (!skip) {
111
        // inner (tx) left|right edges
112
329k
        for (y = 0, mask = 1U << by4; y < h4; y++, mask <<= 1) {
113
255k
            const int sidx = mask >= 0x10000U;
114
255k
            const unsigned smask = mask >> (sidx << 4);
115
255k
            int ltx = txa[0][0][y][0];
116
255k
            int step = txa[0][1][y][0];
117
296k
            for (x = step; x < w4; x += step) {
118
41.3k
                const int rtx = txa[0][0][y][x];
119
41.3k
                masks[0][bx4 + x][imin(rtx, ltx)][sidx] |= smask;
120
41.3k
                ltx = rtx;
121
41.3k
                step = txa[0][1][y][x];
122
41.3k
            }
123
255k
        }
124
125
        //            top
126
        // inner (tx) --- edges
127
        //           bottom
128
353k
        for (x = 0, mask = 1U << bx4; x < w4; x++, mask <<= 1) {
129
278k
            const int sidx = mask >= 0x10000U;
130
278k
            const unsigned smask = mask >> (sidx << 4);
131
278k
            int ttx = txa[1][0][0][x];
132
278k
            int step = txa[1][1][0][x];
133
316k
            for (y = step; y < h4; y += step) {
134
38.0k
                const int btx = txa[1][0][y][x];
135
38.0k
                masks[1][by4 + y][imin(ttx, btx)][sidx] |= smask;
136
38.0k
                ttx = btx;
137
38.0k
                step = txa[1][1][y][x];
138
38.0k
            }
139
278k
        }
140
74.7k
    }
141
142
517k
    for (y = 0; y < h4; y++)
143
402k
        l[y] = txa[0][0][y][w4 - 1];
144
114k
    memcpy(a, txa[1][0][h4 - 1], w4);
145
114k
}
146
147
static inline void mask_edges_intra(uint16_t (*const masks)[32][3][2],
148
                                    const int by4, const int bx4,
149
                                    const int w4, const int h4,
150
                                    const enum RectTxfmSize tx,
151
                                    uint8_t *const a, uint8_t *const l)
152
534k
{
153
534k
    const TxfmInfo *const t_dim = &dav1d_txfm_dimensions[tx];
154
534k
    const int twl4 = t_dim->lw, thl4 = t_dim->lh;
155
534k
    const int twl4c = imin(2, twl4), thl4c = imin(2, thl4);
156
534k
    int y, x;
157
158
    // left block edge
159
534k
    unsigned mask = 1U << by4;
160
4.05M
    for (y = 0; y < h4; y++, mask <<= 1) {
161
3.51M
        const int sidx = mask >= 0x10000;
162
3.51M
        const unsigned smask = mask >> (sidx << 4);
163
3.51M
        masks[0][bx4][imin(twl4c, l[y])][sidx] |= smask;
164
3.51M
    }
165
166
    // top block edge
167
3.73M
    for (x = 0, mask = 1U << bx4; x < w4; x++, mask <<= 1) {
168
3.20M
        const int sidx = mask >= 0x10000;
169
3.20M
        const unsigned smask = mask >> (sidx << 4);
170
3.20M
        masks[1][by4][imin(thl4c, a[x])][sidx] |= smask;
171
3.20M
    }
172
173
    // inner (tx) left|right edges
174
534k
    const int hstep = t_dim->w;
175
534k
    unsigned t = 1U << by4;
176
534k
    unsigned inner = (unsigned) ((((uint64_t) t) << h4) - t);
177
534k
    unsigned inner1 = inner & 0xffff, inner2 = inner >> 16;
178
902k
    for (x = hstep; x < w4; x += hstep) {
179
367k
        if (inner1) masks[0][bx4 + x][twl4c][0] |= inner1;
180
367k
        if (inner2) masks[0][bx4 + x][twl4c][1] |= inner2;
181
367k
    }
182
183
    //            top
184
    // inner (tx) --- edges
185
    //           bottom
186
534k
    const int vstep = t_dim->h;
187
534k
    t = 1U << bx4;
188
534k
    inner = (unsigned) ((((uint64_t) t) << w4) - t);
189
534k
    inner1 = inner & 0xffff;
190
534k
    inner2 = inner >> 16;
191
959k
    for (y = vstep; y < h4; y += vstep) {
192
425k
        if (inner1) masks[1][by4 + y][thl4c][0] |= inner1;
193
425k
        if (inner2) masks[1][by4 + y][thl4c][1] |= inner2;
194
425k
    }
195
196
534k
    dav1d_memset_likely_pow2(a, thl4c, w4);
197
534k
    dav1d_memset_likely_pow2(l, twl4c, h4);
198
534k
}
199
200
static void mask_edges_chroma(uint16_t (*const masks)[32][2][2],
201
                              const int cby4, const int cbx4,
202
                              const int cw4, const int ch4,
203
                              const int skip_inter,
204
                              const enum RectTxfmSize tx,
205
                              uint8_t *const a, uint8_t *const l,
206
                              const int ss_hor, const int ss_ver)
207
527k
{
208
527k
    const TxfmInfo *const t_dim = &dav1d_txfm_dimensions[tx];
209
527k
    const int twl4 = t_dim->lw, thl4 = t_dim->lh;
210
527k
    const int twl4c = !!twl4, thl4c = !!thl4;
211
527k
    int y, x;
212
527k
    const int vbits = 4 - ss_ver, hbits = 4 - ss_hor;
213
527k
    const int vmask = 16 >> ss_ver, hmask = 16 >> ss_hor;
214
527k
    const unsigned vmax = 1 << vmask, hmax = 1 << hmask;
215
216
    // left block edge
217
527k
    unsigned mask = 1U << cby4;
218
3.18M
    for (y = 0; y < ch4; y++, mask <<= 1) {
219
2.65M
        const int sidx = mask >= vmax;
220
2.65M
        const unsigned smask = mask >> (sidx << vbits);
221
2.65M
        masks[0][cbx4][imin(twl4c, l[y])][sidx] |= smask;
222
2.65M
    }
223
224
    // top block edge
225
2.89M
    for (x = 0, mask = 1U << cbx4; x < cw4; x++, mask <<= 1) {
226
2.37M
        const int sidx = mask >= hmax;
227
2.37M
        const unsigned smask = mask >> (sidx << hbits);
228
2.37M
        masks[1][cby4][imin(thl4c, a[x])][sidx] |= smask;
229
2.37M
    }
230
231
527k
    if (!skip_inter) {
232
        // inner (tx) left|right edges
233
506k
        const int hstep = t_dim->w;
234
506k
        unsigned t = 1U << cby4;
235
506k
        unsigned inner = (unsigned) ((((uint64_t) t) << ch4) - t);
236
506k
        unsigned inner1 = inner & ((1 << vmask) - 1), inner2 = inner >> vmask;
237
605k
        for (x = hstep; x < cw4; x += hstep) {
238
99.0k
            if (inner1) masks[0][cbx4 + x][twl4c][0] |= inner1;
239
99.0k
            if (inner2) masks[0][cbx4 + x][twl4c][1] |= inner2;
240
99.0k
        }
241
242
        //            top
243
        // inner (tx) --- edges
244
        //           bottom
245
506k
        const int vstep = t_dim->h;
246
506k
        t = 1U << cbx4;
247
506k
        inner = (unsigned) ((((uint64_t) t) << cw4) - t);
248
506k
        inner1 = inner & ((1 << hmask) - 1), inner2 = inner >> hmask;
249
636k
        for (y = vstep; y < ch4; y += vstep) {
250
130k
            if (inner1) masks[1][cby4 + y][thl4c][0] |= inner1;
251
130k
            if (inner2) masks[1][cby4 + y][thl4c][1] |= inner2;
252
130k
        }
253
506k
    }
254
255
527k
    dav1d_memset_likely_pow2(a, thl4c, cw4);
256
527k
    dav1d_memset_likely_pow2(l, twl4c, ch4);
257
527k
}
258
259
void dav1d_create_lf_mask_intra(Av1Filter *const lflvl,
260
                                uint8_t (*const level_cache)[4],
261
                                const ptrdiff_t b4_stride,
262
                                const uint8_t (*filter_level)[8][2],
263
                                const int bx, const int by,
264
                                const int iw, const int ih,
265
                                const enum BlockSize bs,
266
                                const enum RectTxfmSize ytx,
267
                                const enum RectTxfmSize uvtx,
268
                                const enum Dav1dPixelLayout layout,
269
                                uint8_t *const ay, uint8_t *const ly,
270
                                uint8_t *const auv, uint8_t *const luv)
271
586k
{
272
586k
    const uint8_t *const b_dim = dav1d_block_dimensions[bs];
273
586k
    const int bw4 = imin(iw - bx, b_dim[0]);
274
586k
    const int bh4 = imin(ih - by, b_dim[1]);
275
586k
    const int bx4 = bx & 31;
276
586k
    const int by4 = by & 31;
277
586k
    assert(bw4 >= 0 && bh4 >= 0);
278
279
586k
    if (bw4 && bh4) {
280
534k
        uint8_t (*level_cache_ptr)[4] = level_cache + by * b4_stride + bx;
281
4.03M
        for (int y = 0; y < bh4; y++) {
282
46.3M
            for (int x = 0; x < bw4; x++) {
283
42.8M
                level_cache_ptr[x][0] = filter_level[0][0][0];
284
42.8M
                level_cache_ptr[x][1] = filter_level[1][0][0];
285
42.8M
            }
286
3.50M
            level_cache_ptr += b4_stride;
287
3.50M
        }
288
289
534k
        mask_edges_intra(lflvl->filter_y, by4, bx4, bw4, bh4, ytx, ay, ly);
290
534k
    }
291
292
586k
    if (!auv) return;
293
294
501k
    const int ss_ver = layout == DAV1D_PIXEL_LAYOUT_I420;
295
501k
    const int ss_hor = layout != DAV1D_PIXEL_LAYOUT_I444;
296
501k
    const int cbw4 = imin(((iw + ss_hor) >> ss_hor) - (bx >> ss_hor),
297
501k
                          (b_dim[0] + ss_hor) >> ss_hor);
298
501k
    const int cbh4 = imin(((ih + ss_ver) >> ss_ver) - (by >> ss_ver),
299
501k
                          (b_dim[1] + ss_ver) >> ss_ver);
300
501k
    assert(cbw4 >= 0 && cbh4 >= 0);
301
302
501k
    if (!cbw4 || !cbh4) return;
303
304
459k
    const int cbx4 = bx4 >> ss_hor;
305
459k
    const int cby4 = by4 >> ss_ver;
306
307
459k
    uint8_t (*level_cache_ptr)[4] =
308
459k
        level_cache + (by >> ss_ver) * b4_stride + (bx >> ss_hor);
309
2.95M
    for (int y = 0; y < cbh4; y++) {
310
27.3M
        for (int x = 0; x < cbw4; x++) {
311
24.8M
            level_cache_ptr[x][2] = filter_level[2][0][0];
312
24.8M
            level_cache_ptr[x][3] = filter_level[3][0][0];
313
24.8M
        }
314
2.49M
        level_cache_ptr += b4_stride;
315
2.49M
    }
316
317
459k
    mask_edges_chroma(lflvl->filter_uv, cby4, cbx4, cbw4, cbh4, 0, uvtx,
318
459k
                      auv, luv, ss_hor, ss_ver);
319
459k
}
320
321
void dav1d_create_lf_mask_inter(Av1Filter *const lflvl,
322
                                uint8_t (*const level_cache)[4],
323
                                const ptrdiff_t b4_stride,
324
                                const uint8_t (*filter_level)[8][2],
325
                                const int bx, const int by,
326
                                const int iw, const int ih,
327
                                const int skip, const enum BlockSize bs,
328
                                const enum RectTxfmSize max_ytx,
329
                                const uint16_t *const tx_masks,
330
                                const enum RectTxfmSize uvtx,
331
                                const enum Dav1dPixelLayout layout,
332
                                uint8_t *const ay, uint8_t *const ly,
333
                                uint8_t *const auv, uint8_t *const luv)
334
125k
{
335
125k
    const uint8_t *const b_dim = dav1d_block_dimensions[bs];
336
125k
    const int bw4 = imin(iw - bx, b_dim[0]);
337
125k
    const int bh4 = imin(ih - by, b_dim[1]);
338
125k
    const int bx4 = bx & 31;
339
125k
    const int by4 = by & 31;
340
125k
    assert(bw4 >= 0 && bh4 >= 0);
341
342
125k
    if (bw4 && bh4) {
343
114k
        uint8_t (*level_cache_ptr)[4] = level_cache + by * b4_stride + bx;
344
517k
        for (int y = 0; y < bh4; y++) {
345
2.19M
            for (int x = 0; x < bw4; x++) {
346
1.79M
                level_cache_ptr[x][0] = filter_level[0][0][0];
347
1.79M
                level_cache_ptr[x][1] = filter_level[1][0][0];
348
1.79M
            }
349
402k
            level_cache_ptr += b4_stride;
350
402k
        }
351
352
114k
        mask_edges_inter(lflvl->filter_y, by4, bx4, bw4, bh4, skip,
353
114k
                         max_ytx, tx_masks, ay, ly);
354
114k
    }
355
356
125k
    if (!auv) return;
357
358
71.8k
    const int ss_ver = layout == DAV1D_PIXEL_LAYOUT_I420;
359
71.8k
    const int ss_hor = layout != DAV1D_PIXEL_LAYOUT_I444;
360
71.8k
    const int cbw4 = imin(((iw + ss_hor) >> ss_hor) - (bx >> ss_hor),
361
71.8k
                          (b_dim[0] + ss_hor) >> ss_hor);
362
71.8k
    const int cbh4 = imin(((ih + ss_ver) >> ss_ver) - (by >> ss_ver),
363
71.8k
                          (b_dim[1] + ss_ver) >> ss_ver);
364
71.8k
    assert(cbw4 >= 0 && cbh4 >= 0);
365
366
71.8k
    if (!cbw4 || !cbh4) return;
367
368
68.3k
    const int cbx4 = bx4 >> ss_hor;
369
68.3k
    const int cby4 = by4 >> ss_ver;
370
371
68.3k
    uint8_t (*level_cache_ptr)[4] =
372
68.3k
        level_cache + (by >> ss_ver) * b4_stride + (bx >> ss_hor);
373
228k
    for (int y = 0; y < cbh4; y++) {
374
543k
        for (int x = 0; x < cbw4; x++) {
375
383k
            level_cache_ptr[x][2] = filter_level[2][0][0];
376
383k
            level_cache_ptr[x][3] = filter_level[3][0][0];
377
383k
        }
378
159k
        level_cache_ptr += b4_stride;
379
159k
    }
380
381
68.3k
    mask_edges_chroma(lflvl->filter_uv, cby4, cbx4, cbw4, cbh4, skip, uvtx,
382
68.3k
                      auv, luv, ss_hor, ss_ver);
383
68.3k
}
384
385
54.0k
void dav1d_calc_eih(Av1FilterLUT *const lim_lut, const int filter_sharpness) {
386
    // set E/I/H values from loopfilter level
387
54.0k
    const int sharp = filter_sharpness;
388
3.50M
    for (int level = 0; level < 64; level++) {
389
3.45M
        int limit = level;
390
391
3.45M
        if (sharp > 0) {
392
1.49M
            limit >>= (sharp + 3) >> 2;
393
1.49M
            limit = imin(limit, 9 - sharp);
394
1.49M
        }
395
3.45M
        limit = imax(limit, 1);
396
397
3.45M
        lim_lut->i[level] = limit;
398
3.45M
        lim_lut->e[level] = 2 * (level + 2) + limit;
399
3.45M
    }
400
54.0k
    lim_lut->sharp[0] = (sharp + 3) >> 2;
401
54.0k
    lim_lut->sharp[1] = sharp ? 9 - sharp : 0xff;
402
54.0k
}
403
404
static void calc_lf_value(uint8_t (*const lflvl_values)[2],
405
                          const int base_lvl, const int lf_delta,
406
                          const int seg_delta,
407
                          const Dav1dLoopfilterModeRefDeltas *const mr_delta)
408
447k
{
409
447k
    const int base = iclip(iclip(base_lvl + lf_delta, 0, 63) + seg_delta, 0, 63);
410
411
447k
    if (!mr_delta) {
412
234k
        memset(lflvl_values, base, sizeof(*lflvl_values) * 8);
413
234k
    } else {
414
212k
        const int sh = base >= 32;
415
212k
        lflvl_values[0][0] = lflvl_values[0][1] =
416
212k
            iclip(base + (mr_delta->ref_delta[0] * (1 << sh)), 0, 63);
417
1.69M
        for (int r = 1; r < 8; r++) {
418
4.45M
            for (int m = 0; m < 2; m++) {
419
2.97M
                const int delta =
420
2.97M
                    mr_delta->mode_delta[m] + mr_delta->ref_delta[r];
421
2.97M
                lflvl_values[r][m] = iclip(base + (delta * (1 << sh)), 0, 63);
422
2.97M
            }
423
1.48M
        }
424
212k
    }
425
447k
}
426
427
static inline void calc_lf_value_chroma(uint8_t (*const lflvl_values)[2],
428
                                        const int base_lvl, const int lf_delta,
429
                                        const int seg_delta,
430
                                        const Dav1dLoopfilterModeRefDeltas *const mr_delta)
431
264k
{
432
264k
    if (!base_lvl)
433
82.0k
        memset(lflvl_values, 0, sizeof(*lflvl_values) * 8);
434
182k
    else
435
182k
        calc_lf_value(lflvl_values, base_lvl, lf_delta, seg_delta, mr_delta);
436
264k
}
437
438
void dav1d_calc_lf_values(uint8_t (*const lflvl_values)[4][8][2],
439
                          const Dav1dFrameHeader *const hdr,
440
                          const int8_t lf_delta[4])
441
63.3k
{
442
63.3k
    const int n_seg = hdr->segmentation.enabled ? 8 : 1;
443
444
63.3k
    if (!hdr->loopfilter.level_y[0] && !hdr->loopfilter.level_y[1]) {
445
20.5k
        memset(lflvl_values, 0, sizeof(*lflvl_values) * n_seg);
446
20.5k
        return;
447
20.5k
    }
448
449
42.8k
    const Dav1dLoopfilterModeRefDeltas *const mr_deltas =
450
42.8k
        hdr->loopfilter.mode_ref_delta_enabled ?
451
42.8k
        &hdr->loopfilter.mode_ref_deltas : NULL;
452
175k
    for (int s = 0; s < n_seg; s++) {
453
132k
        const Dav1dSegmentationData *const segd =
454
132k
            hdr->segmentation.enabled ? &hdr->segmentation.seg_data.d[s] : NULL;
455
456
132k
        calc_lf_value(lflvl_values[s][0], hdr->loopfilter.level_y[0],
457
132k
                      lf_delta[0], segd ? segd->delta_lf_y_v : 0, mr_deltas);
458
132k
        calc_lf_value(lflvl_values[s][1], hdr->loopfilter.level_y[1],
459
132k
                      lf_delta[hdr->delta.lf.multi ? 1 : 0],
460
132k
                      segd ? segd->delta_lf_y_h : 0, mr_deltas);
461
132k
        calc_lf_value_chroma(lflvl_values[s][2], hdr->loopfilter.level_u,
462
132k
                             lf_delta[hdr->delta.lf.multi ? 2 : 0],
463
132k
                             segd ? segd->delta_lf_u : 0, mr_deltas);
464
132k
        calc_lf_value_chroma(lflvl_values[s][3], hdr->loopfilter.level_v,
465
132k
                             lf_delta[hdr->delta.lf.multi ? 3 : 0],
466
132k
                             segd ? segd->delta_lf_v : 0, mr_deltas);
467
132k
    }
468
42.8k
}