Coverage Report

Created: 2026-05-30 06:10

next uncovered line (L), next uncovered region (R), next uncovered branch (B)
/work/dav1d/src/msac.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 <limits.h>
31
32
#include "common/intops.h"
33
34
#include "src/msac.h"
35
36
591M
#define EC_PROB_SHIFT 6
37
335M
#define EC_MIN_PROB 4  // must be <= (1<<EC_PROB_SHIFT)/16
38
39
298M
#define EC_WIN_SIZE (sizeof(ec_win) << 3)
40
41
2.02M
static inline void ctx_refill(MsacContext *const s) {
42
2.02M
    const uint8_t *buf_pos = s->buf_pos;
43
2.02M
    const uint8_t *buf_end = s->buf_end;
44
2.02M
    int c = EC_WIN_SIZE - s->cnt - 24;
45
2.02M
    ec_win dif = s->dif;
46
12.0M
    do {
47
12.0M
        if (buf_pos >= buf_end) {
48
            // set remaining bits to 1;
49
65.9k
            dif |= ~(~(ec_win)0xff << c);
50
65.9k
            break;
51
65.9k
        }
52
12.0M
        dif |= (ec_win)(*buf_pos++ ^ 0xff) << c;
53
12.0M
        c -= 8;
54
12.0M
    } while (c >= 0);
55
2.02M
    s->dif = dif;
56
2.02M
    s->cnt = EC_WIN_SIZE - c - 24;
57
2.02M
    s->buf_pos = buf_pos;
58
2.02M
}
59
60
int dav1d_msac_decode_subexp(MsacContext *const s, const int ref,
61
                             const int n, unsigned k)
62
554k
{
63
554k
    assert(n >> k == 8);
64
65
554k
    unsigned a = 0;
66
554k
    if (dav1d_msac_decode_bool_equi(s)) {
67
294k
        if (dav1d_msac_decode_bool_equi(s))
68
168k
            k += dav1d_msac_decode_bool_equi(s) + 1;
69
294k
        a = 1 << k;
70
294k
    }
71
554k
    const unsigned v = dav1d_msac_decode_bools(s, k) + a;
72
554k
    return ref * 2 <= n ? inv_recenter(ref, v) :
73
554k
                          n - 1 - inv_recenter(n - 1 - ref, v);
74
554k
}
75
76
#if !(HAVE_ASM && TRIM_DSP_FUNCTIONS && ( \
77
  ARCH_AARCH64 || \
78
  (ARCH_ARM && (defined(__ARM_NEON) || defined(__APPLE__) || defined(_WIN32))) \
79
))
80
/* Takes updated dif and range values, renormalizes them so that
81
 * 32768 <= rng < 65536 (reading more bytes from the stream into dif if
82
 * necessary), and stores them back in the decoder context.
83
 * dif: The new value of dif.
84
 * rng: The new value of the range. */
85
static inline void ctx_norm(MsacContext *const s, const ec_win dif,
86
                            const unsigned rng)
87
178M
{
88
178M
    const int d = 15 ^ (31 ^ clz(rng));
89
178M
    const int cnt = s->cnt;
90
178M
    assert(rng <= 65535U);
91
178M
    s->dif = dif << d;
92
178M
    s->rng = rng << d;
93
178M
    s->cnt = cnt - d;
94
    // unsigned compare avoids redundant refills at eob
95
178M
    if ((unsigned)cnt < (unsigned)d)
96
1.89M
        ctx_refill(s);
97
178M
}
98
99
39.7M
unsigned dav1d_msac_decode_bool_equi_c(MsacContext *const s) {
100
39.7M
    const unsigned r = s->rng;
101
39.7M
    ec_win dif = s->dif;
102
39.7M
    assert((dif >> (EC_WIN_SIZE - 16)) < r);
103
    // When the probability is 1/2, f = 16384 >> EC_PROB_SHIFT = 256 and we can
104
    // replace the multiply with a simple shift.
105
39.7M
    unsigned v = ((r >> 8) << 7) + EC_MIN_PROB;
106
39.7M
    const ec_win vw = (ec_win)v << (EC_WIN_SIZE - 16);
107
39.7M
    const unsigned ret = dif >= vw;
108
39.7M
    dif -= ret * vw;
109
39.7M
    v += ret * (r - 2 * v);
110
39.7M
    ctx_norm(s, dif, v);
111
39.7M
    return !ret;
112
39.7M
}
113
114
/* Decode a single binary value.
115
 * f: The probability that the bit is one
116
 * Return: The value decoded (0 or 1). */
117
28.9M
unsigned dav1d_msac_decode_bool_c(MsacContext *const s, const unsigned f) {
118
28.9M
    const unsigned r = s->rng;
119
28.9M
    ec_win dif = s->dif;
120
28.9M
    assert((dif >> (EC_WIN_SIZE - 16)) < r);
121
28.9M
    unsigned v = ((r >> 8) * (f >> EC_PROB_SHIFT) >> (7 - EC_PROB_SHIFT)) + EC_MIN_PROB;
122
28.9M
    const ec_win vw = (ec_win)v << (EC_WIN_SIZE - 16);
123
28.9M
    const unsigned ret = dif >= vw;
124
28.9M
    dif -= ret * vw;
125
28.9M
    v += ret * (r - 2 * v);
126
28.9M
    ctx_norm(s, dif, v);
127
28.9M
    return !ret;
128
28.9M
}
129
130
/* Decodes a symbol given an inverse cumulative distribution function (CDF)
131
 * table in Q15. */
132
unsigned dav1d_msac_decode_symbol_adapt_c(MsacContext *const s,
133
                                          uint16_t *const cdf,
134
                                          const size_t n_symbols)
135
113M
{
136
113M
    const unsigned c = s->dif >> (EC_WIN_SIZE - 16), r = s->rng >> 8;
137
113M
    unsigned u, v = s->rng, val = -1;
138
139
113M
    assert(n_symbols <= 15);
140
113M
    assert(cdf[n_symbols] <= 32);
141
142
266M
    do {
143
266M
        val++;
144
266M
        u = v;
145
266M
        v = r * (cdf[val] >> EC_PROB_SHIFT);
146
266M
        v >>= 7 - EC_PROB_SHIFT;
147
266M
        v += EC_MIN_PROB * ((unsigned)n_symbols - val);
148
266M
    } while (c < v);
149
150
113M
    assert(u <= s->rng);
151
152
113M
    ctx_norm(s, s->dif - ((ec_win)v << (EC_WIN_SIZE - 16)), u - v);
153
154
113M
    if (s->allow_update_cdf) {
155
98.6M
        const unsigned count = cdf[n_symbols];
156
98.6M
        const unsigned rate = 4 + (count >> 4) + (n_symbols > 2);
157
98.6M
        unsigned i;
158
234M
        for (i = 0; i < val; i++)
159
136M
            cdf[i] += (32768 - cdf[i]) >> rate;
160
363M
        for (; i < n_symbols; i++)
161
265M
            cdf[i] -= cdf[i] >> rate;
162
98.6M
        cdf[n_symbols] = count + (count < 32);
163
98.6M
    }
164
165
113M
    return val;
166
113M
}
167
168
unsigned dav1d_msac_decode_bool_adapt_c(MsacContext *const s,
169
                                        uint16_t *const cdf)
170
27.7M
{
171
27.7M
    const unsigned bit = dav1d_msac_decode_bool(s, *cdf);
172
173
27.7M
    if (s->allow_update_cdf) {
174
        // update_cdf() specialized for boolean CDFs
175
23.5M
        const unsigned count = cdf[1];
176
23.5M
        const int rate = 4 + (count >> 4);
177
23.5M
        if (bit)
178
13.4M
            cdf[0] += (32768 - cdf[0]) >> rate;
179
10.0M
        else
180
10.0M
            cdf[0] -= cdf[0] >> rate;
181
23.5M
        cdf[1] = count + (count < 32);
182
23.5M
    }
183
184
27.7M
    return bit;
185
27.7M
}
186
187
6.55M
unsigned dav1d_msac_decode_hi_tok_c(MsacContext *const s, uint16_t *const cdf) {
188
6.55M
    unsigned tok_br = dav1d_msac_decode_symbol_adapt4(s, cdf, 3);
189
6.55M
    unsigned tok = 3 + tok_br;
190
6.55M
    if (tok_br == 3) {
191
2.58M
        tok_br = dav1d_msac_decode_symbol_adapt4(s, cdf, 3);
192
2.58M
        tok = 6 + tok_br;
193
2.58M
        if (tok_br == 3) {
194
1.51M
            tok_br = dav1d_msac_decode_symbol_adapt4(s, cdf, 3);
195
1.51M
            tok = 9 + tok_br;
196
1.51M
            if (tok_br == 3)
197
1.09M
                tok = 12 + dav1d_msac_decode_symbol_adapt4(s, cdf, 3);
198
1.51M
        }
199
2.58M
    }
200
6.55M
    return tok;
201
6.55M
}
202
#endif
203
204
void dav1d_msac_init(MsacContext *const s, const uint8_t *const data,
205
                     const size_t sz, const int disable_cdf_update_flag)
206
137k
{
207
137k
    s->buf_pos = data;
208
137k
    s->buf_end = data + sz;
209
137k
    s->dif = 0;
210
137k
    s->rng = 0x8000;
211
137k
    s->cnt = -15;
212
137k
    s->allow_update_cdf = !disable_cdf_update_flag;
213
137k
    ctx_refill(s);
214
215
#if ARCH_X86_64 && HAVE_ASM
216
    s->symbol_adapt16 = dav1d_msac_decode_symbol_adapt_c;
217
218
    msac_init_x86(s);
219
#endif
220
137k
}