Coverage Report

Created: 2026-06-10 06:56

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
143k
{
45
143k
    const TxfmInfo *const t_dim = &dav1d_txfm_dimensions[from];
46
143k
    const int is_split = (from == (int) TX_4X4 || depth > 1) ? 0 :
47
143k
        (tx_masks[depth] >> (y_off * 4 + x_off)) & 1;
48
49
143k
    if (is_split) {
50
11.4k
        const enum RectTxfmSize sub = t_dim->sub;
51
11.4k
        const int htw4 = t_dim->w >> 1, hth4 = t_dim->h >> 1;
52
53
11.4k
        decomp_tx(txa, sub, depth + 1, y_off * 2 + 0, x_off * 2 + 0, tx_masks);
54
11.4k
        if (t_dim->w >= t_dim->h)
55
9.34k
            decomp_tx((uint8_t(*)[2][32][32]) &txa[0][0][0][htw4],
56
9.34k
                      sub, depth + 1, y_off * 2 + 0, x_off * 2 + 1, tx_masks);
57
11.4k
        if (t_dim->h >= t_dim->w) {
58
7.59k
            decomp_tx((uint8_t(*)[2][32][32]) &txa[0][0][hth4][0],
59
7.59k
                      sub, depth + 1, y_off * 2 + 1, x_off * 2 + 0, tx_masks);
60
7.59k
            if (t_dim->w >= t_dim->h)
61
5.49k
                decomp_tx((uint8_t(*)[2][32][32]) &txa[0][0][hth4][htw4],
62
5.49k
                          sub, depth + 1, y_off * 2 + 1, x_off * 2 + 1, tx_masks);
63
7.59k
        }
64
131k
    } else {
65
131k
        const int lw = imin(2, t_dim->lw), lh = imin(2, t_dim->lh);
66
67
131k
#define set_ctx(rep_macro) \
68
566k
        for (int y = 0; y < t_dim->h; y++) { \
69
434k
            rep_macro(txa[0][0][y], 0, lw); \
70
434k
            rep_macro(txa[1][0][y], 0, lh); \
71
434k
            txa[0][1][y][0] = t_dim->w; \
72
434k
        }
73
131k
        case_set_upto16(t_dim->lw);
74
131k
#undef set_ctx
75
131k
        dav1d_memset_pow2[t_dim->lw](txa[1][1][0], t_dim->h);
76
131k
    }
77
143k
}
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
101k
{
86
101k
    const TxfmInfo *const t_dim = &dav1d_txfm_dimensions[max_tx];
87
101k
    int y, x;
88
89
101k
    ALIGN_STK_16(uint8_t, txa, 2 /* edge */, [2 /* txsz, step */][32 /* y */][32 /* x */]);
90
204k
    for (int y_off = 0, y = 0; y < h4; y += t_dim->h, y_off++)
91
212k
        for (int x_off = 0, x = 0; x < w4; x += t_dim->w, x_off++)
92
109k
            decomp_tx((uint8_t(*)[2][32][32]) &txa[0][0][y][x],
93
109k
                      max_tx, 0, y_off, x_off, tx_masks);
94
95
    // left block edge
96
101k
    unsigned mask = 1U << by4;
97
450k
    for (y = 0; y < h4; y++, mask <<= 1) {
98
348k
        const int sidx = mask >= 0x10000;
99
348k
        const unsigned smask = mask >> (sidx << 4);
100
348k
        masks[0][bx4][imin(txa[0][0][y][0], l[y])][sidx] |= smask;
101
348k
    }
102
103
    // top block edge
104
460k
    for (x = 0, mask = 1U << bx4; x < w4; x++, mask <<= 1) {
105
359k
        const int sidx = mask >= 0x10000;
106
359k
        const unsigned smask = mask >> (sidx << 4);
107
359k
        masks[1][by4][imin(txa[1][0][0][x], a[x])][sidx] |= smask;
108
359k
    }
109
110
101k
    if (!skip) {
111
        // inner (tx) left|right edges
112
254k
        for (y = 0, mask = 1U << by4; y < h4; y++, mask <<= 1) {
113
194k
            const int sidx = mask >= 0x10000U;
114
194k
            const unsigned smask = mask >> (sidx << 4);
115
194k
            int ltx = txa[0][0][y][0];
116
194k
            int step = txa[0][1][y][0];
117
231k
            for (x = step; x < w4; x += step) {
118
37.2k
                const int rtx = txa[0][0][y][x];
119
37.2k
                masks[0][bx4 + x][imin(rtx, ltx)][sidx] |= smask;
120
37.2k
                ltx = rtx;
121
37.2k
                step = txa[0][1][y][x];
122
37.2k
            }
123
194k
        }
124
125
        //            top
126
        // inner (tx) --- edges
127
        //           bottom
128
288k
        for (x = 0, mask = 1U << bx4; x < w4; x++, mask <<= 1) {
129
229k
            const int sidx = mask >= 0x10000U;
130
229k
            const unsigned smask = mask >> (sidx << 4);
131
229k
            int ttx = txa[1][0][0][x];
132
229k
            int step = txa[1][1][0][x];
133
262k
            for (y = step; y < h4; y += step) {
134
33.4k
                const int btx = txa[1][0][y][x];
135
33.4k
                masks[1][by4 + y][imin(ttx, btx)][sidx] |= smask;
136
33.4k
                ttx = btx;
137
33.4k
                step = txa[1][1][y][x];
138
33.4k
            }
139
229k
        }
140
59.6k
    }
141
142
450k
    for (y = 0; y < h4; y++)
143
348k
        l[y] = txa[0][0][y][w4 - 1];
144
101k
    memcpy(a, txa[1][0][h4 - 1], w4);
145
101k
}
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
259k
{
153
259k
    const TxfmInfo *const t_dim = &dav1d_txfm_dimensions[tx];
154
259k
    const int twl4 = t_dim->lw, thl4 = t_dim->lh;
155
259k
    const int twl4c = imin(2, twl4), thl4c = imin(2, thl4);
156
259k
    int y, x;
157
158
    // left block edge
159
259k
    unsigned mask = 1U << by4;
160
1.15M
    for (y = 0; y < h4; y++, mask <<= 1) {
161
898k
        const int sidx = mask >= 0x10000;
162
898k
        const unsigned smask = mask >> (sidx << 4);
163
898k
        masks[0][bx4][imin(twl4c, l[y])][sidx] |= smask;
164
898k
    }
165
166
    // top block edge
167
1.18M
    for (x = 0, mask = 1U << bx4; x < w4; x++, mask <<= 1) {
168
925k
        const int sidx = mask >= 0x10000;
169
925k
        const unsigned smask = mask >> (sidx << 4);
170
925k
        masks[1][by4][imin(thl4c, a[x])][sidx] |= smask;
171
925k
    }
172
173
    // inner (tx) left|right edges
174
259k
    const int hstep = t_dim->w;
175
259k
    unsigned t = 1U << by4;
176
259k
    unsigned inner = (unsigned) ((((uint64_t) t) << h4) - t);
177
259k
    unsigned inner1 = inner & 0xffff, inner2 = inner >> 16;
178
329k
    for (x = hstep; x < w4; x += hstep) {
179
70.3k
        if (inner1) masks[0][bx4 + x][twl4c][0] |= inner1;
180
70.3k
        if (inner2) masks[0][bx4 + x][twl4c][1] |= inner2;
181
70.3k
    }
182
183
    //            top
184
    // inner (tx) --- edges
185
    //           bottom
186
259k
    const int vstep = t_dim->h;
187
259k
    t = 1U << bx4;
188
259k
    inner = (unsigned) ((((uint64_t) t) << w4) - t);
189
259k
    inner1 = inner & 0xffff;
190
259k
    inner2 = inner >> 16;
191
321k
    for (y = vstep; y < h4; y += vstep) {
192
61.7k
        if (inner1) masks[1][by4 + y][thl4c][0] |= inner1;
193
61.7k
        if (inner2) masks[1][by4 + y][thl4c][1] |= inner2;
194
61.7k
    }
195
196
259k
    dav1d_memset_likely_pow2(a, thl4c, w4);
197
259k
    dav1d_memset_likely_pow2(l, twl4c, h4);
198
259k
}
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
287k
{
208
287k
    const TxfmInfo *const t_dim = &dav1d_txfm_dimensions[tx];
209
287k
    const int twl4 = t_dim->lw, thl4 = t_dim->lh;
210
287k
    const int twl4c = !!twl4, thl4c = !!thl4;
211
287k
    int y, x;
212
287k
    const int vbits = 4 - ss_ver, hbits = 4 - ss_hor;
213
287k
    const int vmask = 16 >> ss_ver, hmask = 16 >> ss_hor;
214
287k
    const unsigned vmax = 1 << vmask, hmax = 1 << hmask;
215
216
    // left block edge
217
287k
    unsigned mask = 1U << cby4;
218
1.19M
    for (y = 0; y < ch4; y++, mask <<= 1) {
219
909k
        const int sidx = mask >= vmax;
220
909k
        const unsigned smask = mask >> (sidx << vbits);
221
909k
        masks[0][cbx4][imin(twl4c, l[y])][sidx] |= smask;
222
909k
    }
223
224
    // top block edge
225
1.21M
    for (x = 0, mask = 1U << cbx4; x < cw4; x++, mask <<= 1) {
226
927k
        const int sidx = mask >= hmax;
227
927k
        const unsigned smask = mask >> (sidx << hbits);
228
927k
        masks[1][cby4][imin(thl4c, a[x])][sidx] |= smask;
229
927k
    }
230
231
287k
    if (!skip_inter) {
232
        // inner (tx) left|right edges
233
253k
        const int hstep = t_dim->w;
234
253k
        unsigned t = 1U << cby4;
235
253k
        unsigned inner = (unsigned) ((((uint64_t) t) << ch4) - t);
236
253k
        unsigned inner1 = inner & ((1 << vmask) - 1), inner2 = inner >> vmask;
237
285k
        for (x = hstep; x < cw4; x += hstep) {
238
31.8k
            if (inner1) masks[0][cbx4 + x][twl4c][0] |= inner1;
239
31.8k
            if (inner2) masks[0][cbx4 + x][twl4c][1] |= inner2;
240
31.8k
        }
241
242
        //            top
243
        // inner (tx) --- edges
244
        //           bottom
245
253k
        const int vstep = t_dim->h;
246
253k
        t = 1U << cbx4;
247
253k
        inner = (unsigned) ((((uint64_t) t) << cw4) - t);
248
253k
        inner1 = inner & ((1 << hmask) - 1), inner2 = inner >> hmask;
249
292k
        for (y = vstep; y < ch4; y += vstep) {
250
38.3k
            if (inner1) masks[1][cby4 + y][thl4c][0] |= inner1;
251
38.3k
            if (inner2) masks[1][cby4 + y][thl4c][1] |= inner2;
252
38.3k
        }
253
253k
    }
254
255
287k
    dav1d_memset_likely_pow2(a, thl4c, cw4);
256
287k
    dav1d_memset_likely_pow2(l, twl4c, ch4);
257
287k
}
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
327k
{
272
327k
    const uint8_t *const b_dim = dav1d_block_dimensions[bs];
273
327k
    const int bw4 = imin(iw - bx, b_dim[0]);
274
327k
    const int bh4 = imin(ih - by, b_dim[1]);
275
327k
    const int bx4 = bx & 31;
276
327k
    const int by4 = by & 31;
277
327k
    assert(bw4 >= 0 && bh4 >= 0);
278
279
327k
    if (bw4 && bh4) {
280
258k
        uint8_t (*level_cache_ptr)[4] = level_cache + by * b4_stride + bx;
281
1.15M
        for (int y = 0; y < bh4; y++) {
282
4.95M
            for (int x = 0; x < bw4; x++) {
283
4.06M
                level_cache_ptr[x][0] = filter_level[0][0][0];
284
4.06M
                level_cache_ptr[x][1] = filter_level[1][0][0];
285
4.06M
            }
286
898k
            level_cache_ptr += b4_stride;
287
898k
        }
288
289
258k
        mask_edges_intra(lflvl->filter_y, by4, bx4, bw4, bh4, ytx, ay, ly);
290
258k
    }
291
292
327k
    if (!auv) return;
293
294
228k
    const int ss_ver = layout == DAV1D_PIXEL_LAYOUT_I420;
295
228k
    const int ss_hor = layout != DAV1D_PIXEL_LAYOUT_I444;
296
228k
    const int cbw4 = imin(((iw + ss_hor) >> ss_hor) - (bx >> ss_hor),
297
228k
                          (b_dim[0] + ss_hor) >> ss_hor);
298
228k
    const int cbh4 = imin(((ih + ss_ver) >> ss_ver) - (by >> ss_ver),
299
228k
                          (b_dim[1] + ss_ver) >> ss_ver);
300
228k
    assert(cbw4 >= 0 && cbh4 >= 0);
301
302
228k
    if (!cbw4 || !cbh4) return;
303
304
201k
    const int cbx4 = bx4 >> ss_hor;
305
201k
    const int cby4 = by4 >> ss_ver;
306
307
201k
    uint8_t (*level_cache_ptr)[4] =
308
201k
        level_cache + (by >> ss_ver) * b4_stride + (bx >> ss_hor);
309
856k
    for (int y = 0; y < cbh4; y++) {
310
3.33M
        for (int x = 0; x < cbw4; x++) {
311
2.67M
            level_cache_ptr[x][2] = filter_level[2][0][0];
312
2.67M
            level_cache_ptr[x][3] = filter_level[3][0][0];
313
2.67M
        }
314
654k
        level_cache_ptr += b4_stride;
315
654k
    }
316
317
201k
    mask_edges_chroma(lflvl->filter_uv, cby4, cbx4, cbw4, cbh4, 0, uvtx,
318
201k
                      auv, luv, ss_hor, ss_ver);
319
201k
}
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
105k
{
335
105k
    const uint8_t *const b_dim = dav1d_block_dimensions[bs];
336
105k
    const int bw4 = imin(iw - bx, b_dim[0]);
337
105k
    const int bh4 = imin(ih - by, b_dim[1]);
338
105k
    const int bx4 = bx & 31;
339
105k
    const int by4 = by & 31;
340
105k
    assert(bw4 >= 0 && bh4 >= 0);
341
342
105k
    if (bw4 && bh4) {
343
101k
        uint8_t (*level_cache_ptr)[4] = level_cache + by * b4_stride + bx;
344
449k
        for (int y = 0; y < bh4; y++) {
345
2.14M
            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
348k
            level_cache_ptr += b4_stride;
350
348k
        }
351
352
101k
        mask_edges_inter(lflvl->filter_y, by4, bx4, bw4, bh4, skip,
353
101k
                         max_ytx, tx_masks, ay, ly);
354
101k
    }
355
356
105k
    if (!auv) return;
357
358
85.0k
    const int ss_ver = layout == DAV1D_PIXEL_LAYOUT_I420;
359
85.0k
    const int ss_hor = layout != DAV1D_PIXEL_LAYOUT_I444;
360
85.0k
    const int cbw4 = imin(((iw + ss_hor) >> ss_hor) - (bx >> ss_hor),
361
85.0k
                          (b_dim[0] + ss_hor) >> ss_hor);
362
85.0k
    const int cbh4 = imin(((ih + ss_ver) >> ss_ver) - (by >> ss_ver),
363
85.0k
                          (b_dim[1] + ss_ver) >> ss_ver);
364
85.0k
    assert(cbw4 >= 0 && cbh4 >= 0);
365
366
85.0k
    if (!cbw4 || !cbh4) return;
367
368
83.2k
    const int cbx4 = bx4 >> ss_hor;
369
83.2k
    const int cby4 = by4 >> ss_ver;
370
371
83.2k
    uint8_t (*level_cache_ptr)[4] =
372
83.2k
        level_cache + (by >> ss_ver) * b4_stride + (bx >> ss_hor);
373
337k
    for (int y = 0; y < cbh4; y++) {
374
1.53M
        for (int x = 0; x < cbw4; x++) {
375
1.28M
            level_cache_ptr[x][2] = filter_level[2][0][0];
376
1.28M
            level_cache_ptr[x][3] = filter_level[3][0][0];
377
1.28M
        }
378
254k
        level_cache_ptr += b4_stride;
379
254k
    }
380
381
83.2k
    mask_edges_chroma(lflvl->filter_uv, cby4, cbx4, cbw4, cbh4, skip, uvtx,
382
83.2k
                      auv, luv, ss_hor, ss_ver);
383
83.2k
}
384
385
22.6k
void dav1d_calc_eih(Av1FilterLUT *const lim_lut, const int filter_sharpness) {
386
    // set E/I/H values from loopfilter level
387
22.6k
    const int sharp = filter_sharpness;
388
1.47M
    for (int level = 0; level < 64; level++) {
389
1.44M
        int limit = level;
390
391
1.44M
        if (sharp > 0) {
392
667k
            limit >>= (sharp + 3) >> 2;
393
667k
            limit = imin(limit, 9 - sharp);
394
667k
        }
395
1.44M
        limit = imax(limit, 1);
396
397
1.44M
        lim_lut->i[level] = limit;
398
1.44M
        lim_lut->e[level] = 2 * (level + 2) + limit;
399
1.44M
    }
400
22.6k
    lim_lut->sharp[0] = (sharp + 3) >> 2;
401
22.6k
    lim_lut->sharp[1] = sharp ? 9 - sharp : 0xff;
402
22.6k
}
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
254k
{
409
254k
    const int base = iclip(iclip(base_lvl + lf_delta, 0, 63) + seg_delta, 0, 63);
410
411
254k
    if (!mr_delta) {
412
113k
        memset(lflvl_values, base, sizeof(*lflvl_values) * 8);
413
141k
    } else {
414
141k
        const int sh = base >= 32;
415
141k
        lflvl_values[0][0] = lflvl_values[0][1] =
416
141k
            iclip(base + (mr_delta->ref_delta[0] * (1 << sh)), 0, 63);
417
1.12M
        for (int r = 1; r < 8; r++) {
418
2.96M
            for (int m = 0; m < 2; m++) {
419
1.97M
                const int delta =
420
1.97M
                    mr_delta->mode_delta[m] + mr_delta->ref_delta[r];
421
1.97M
                lflvl_values[r][m] = iclip(base + (delta * (1 << sh)), 0, 63);
422
1.97M
            }
423
988k
        }
424
141k
    }
425
254k
}
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
146k
{
432
146k
    if (!base_lvl)
433
37.5k
        memset(lflvl_values, 0, sizeof(*lflvl_values) * 8);
434
108k
    else
435
108k
        calc_lf_value(lflvl_values, base_lvl, lf_delta, seg_delta, mr_delta);
436
146k
}
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
29.4k
{
442
29.4k
    const int n_seg = hdr->segmentation.enabled ? 8 : 1;
443
444
29.4k
    if (!hdr->loopfilter.level_y[0] && !hdr->loopfilter.level_y[1]) {
445
8.94k
        memset(lflvl_values, 0, sizeof(*lflvl_values) * n_seg);
446
8.94k
        return;
447
8.94k
    }
448
449
20.5k
    const Dav1dLoopfilterModeRefDeltas *const mr_deltas =
450
20.5k
        hdr->loopfilter.mode_ref_delta_enabled ?
451
20.5k
        &hdr->loopfilter.mode_ref_deltas : NULL;
452
93.8k
    for (int s = 0; s < n_seg; s++) {
453
73.3k
        const Dav1dSegmentationData *const segd =
454
73.3k
            hdr->segmentation.enabled ? &hdr->segmentation.seg_data.d[s] : NULL;
455
456
73.3k
        calc_lf_value(lflvl_values[s][0], hdr->loopfilter.level_y[0],
457
73.3k
                      lf_delta[0], segd ? segd->delta_lf_y_v : 0, mr_deltas);
458
73.3k
        calc_lf_value(lflvl_values[s][1], hdr->loopfilter.level_y[1],
459
73.3k
                      lf_delta[hdr->delta.lf.multi ? 1 : 0],
460
73.3k
                      segd ? segd->delta_lf_y_h : 0, mr_deltas);
461
73.3k
        calc_lf_value_chroma(lflvl_values[s][2], hdr->loopfilter.level_u,
462
73.3k
                             lf_delta[hdr->delta.lf.multi ? 2 : 0],
463
73.3k
                             segd ? segd->delta_lf_u : 0, mr_deltas);
464
73.3k
        calc_lf_value_chroma(lflvl_values[s][3], hdr->loopfilter.level_v,
465
73.3k
                             lf_delta[hdr->delta.lf.multi ? 3 : 0],
466
73.3k
                             segd ? segd->delta_lf_v : 0, mr_deltas);
467
73.3k
    }
468
20.5k
}