Coverage Report

Created: 2025-12-03 06:57

next uncovered line (L), next uncovered region (R), next uncovered branch (B)
/src/libavif/ext/aom/aom_dsp/entenc.c
Line
Count
Source
1
/*
2
 * Copyright (c) 2001-2016, Alliance for Open Media. All rights reserved.
3
 *
4
 * This source code is subject to the terms of the BSD 2 Clause License and
5
 * the Alliance for Open Media Patent License 1.0. If the BSD 2 Clause License
6
 * was not distributed with this source code in the LICENSE file, you can
7
 * obtain it at www.aomedia.org/license/software. If the Alliance for Open
8
 * Media Patent License 1.0 was not distributed with this source code in the
9
 * PATENTS file, you can obtain it at www.aomedia.org/license/patent.
10
 */
11
12
#include <stdlib.h>
13
#include <string.h>
14
#include <math.h>
15
#include <assert.h>
16
#include "aom_dsp/entenc.h"
17
#include "aom_dsp/prob.h"
18
19
#if OD_MEASURE_EC_OVERHEAD
20
#if !defined(M_LOG2E)
21
#define M_LOG2E (1.4426950408889634073599246810019)
22
#endif
23
#define OD_LOG2(x) (M_LOG2E * log(x))
24
#endif  // OD_MEASURE_EC_OVERHEAD
25
26
/*A range encoder.
27
  See entdec.c and the references for implementation details \cite{Mar79,MNW98}.
28
29
  @INPROCEEDINGS{Mar79,
30
   author="Martin, G.N.N.",
31
   title="Range encoding: an algorithm for removing redundancy from a digitised
32
    message",
33
   booktitle="Video \& Data Recording Conference",
34
   year=1979,
35
   address="Southampton",
36
   month=Jul,
37
   URL="http://www.compressconsult.com/rangecoder/rngcod.pdf.gz"
38
  }
39
  @ARTICLE{MNW98,
40
   author="Alistair Moffat and Radford Neal and Ian H. Witten",
41
   title="Arithmetic Coding Revisited",
42
   journal="{ACM} Transactions on Information Systems",
43
   year=1998,
44
   volume=16,
45
   number=3,
46
   pages="256--294",
47
   month=Jul,
48
   URL="http://researchcommons.waikato.ac.nz/bitstream/handle/10289/78/content.pdf"
49
  }*/
50
51
/*Takes updated low and range values, renormalizes them so that
52
   32768 <= rng < 65536 (flushing bytes from low to the output buffer if
53
   necessary), and stores them back in the encoder context.
54
  low: The new value of low.
55
  rng: The new value of the range.*/
56
static void od_ec_enc_normalize(od_ec_enc *enc, od_ec_enc_window low,
57
2.44G
                                unsigned rng) {
58
2.44G
  int d;
59
2.44G
  int c;
60
2.44G
  int s;
61
2.44G
  if (enc->error) return;
62
2.44G
  c = enc->cnt;
63
2.44G
  assert(rng <= 65535U);
64
  /*The number of leading zeros in the 16-bit binary representation of rng.*/
65
2.44G
  d = 16 - OD_ILOG_NZ(rng);
66
2.44G
  s = c + d;
67
68
  /* We flush every time "low" cannot safely and efficiently accommodate any
69
     more data. Overall, c must not exceed 63 at the time of byte flush out. To
70
     facilitate this, "s" cannot exceed 56-bits because we have to keep 1 byte
71
     for carry. Also, we need to subtract 16 because we want to keep room for
72
     the next symbol worth "d"-bits (max 15). An alternate condition would be if
73
     (e < d), where e = number of leading zeros in "low", indicating there is
74
     not enough rooom to accommodate "rng" worth of "d"-bits in "low". However,
75
     this approach needs additional computations: (i) compute "e", (ii) push
76
     the leading 0x00's as a special case.
77
  */
78
2.44G
  if (s >= 40) {  // 56 - 16
79
64.4M
    unsigned char *out = enc->buf;
80
64.4M
    uint32_t storage = enc->storage;
81
64.4M
    uint32_t offs = enc->offs;
82
64.4M
    if (offs + 8 > storage) {
83
527
      storage = 2 * storage + 8;
84
527
      out = (unsigned char *)realloc(out, sizeof(*out) * storage);
85
527
      if (out == NULL) {
86
0
        enc->error = -1;
87
0
        return;
88
0
      }
89
527
      enc->buf = out;
90
527
      enc->storage = storage;
91
527
    }
92
    // Need to add 1 byte here since enc->cnt always counts 1 byte less
93
    // (enc->cnt = -9) to ensure correct operation
94
64.4M
    uint8_t num_bytes_ready = (s >> 3) + 1;
95
96
    // Update "c" to contain the number of non-ready bits in "low". Since "low"
97
    // has 64-bit capacity, we need to add the (64 - 40) cushion bits and take
98
    // off the number of ready bits.
99
64.4M
    c += 24 - (num_bytes_ready << 3);
100
101
    // Prepare "output" and update "low"
102
64.4M
    uint64_t output = low >> c;
103
64.4M
    low = low & (((uint64_t)1 << c) - 1);
104
105
    // Prepare data and carry mask
106
64.4M
    uint64_t mask = (uint64_t)1 << (num_bytes_ready << 3);
107
64.4M
    uint64_t carry = output & mask;
108
109
64.4M
    mask = mask - 0x01;
110
64.4M
    output = output & mask;
111
112
    // Write data in a single operation
113
64.4M
    write_enc_data_to_out_buf(out, offs, output, carry, &enc->offs,
114
64.4M
                              num_bytes_ready);
115
116
    // Update state of the encoder: enc->cnt to contain the number of residual
117
    // bits
118
64.4M
    s = c + d - 24;
119
64.4M
  }
120
2.44G
  enc->low = low << d;
121
2.44G
  enc->rng = rng << d;
122
2.44G
  enc->cnt = s;
123
2.44G
}
124
125
/*Initializes the encoder.
126
  size: The initial size of the buffer, in bytes.*/
127
249k
void od_ec_enc_init(od_ec_enc *enc, uint32_t size) {
128
249k
  od_ec_enc_reset(enc);
129
249k
  enc->buf = (unsigned char *)malloc(sizeof(*enc->buf) * size);
130
249k
  enc->storage = size;
131
249k
  if (size > 0 && enc->buf == NULL) {
132
0
    enc->storage = 0;
133
0
    enc->error = -1;
134
0
  }
135
249k
}
136
137
/*Reinitializes the encoder.*/
138
249k
void od_ec_enc_reset(od_ec_enc *enc) {
139
249k
  enc->offs = 0;
140
249k
  enc->low = 0;
141
249k
  enc->rng = 0x8000;
142
  /*This is initialized to -9 so that it crosses zero after we've accumulated
143
     one byte + one carry bit.*/
144
249k
  enc->cnt = -9;
145
249k
  enc->error = 0;
146
#if OD_MEASURE_EC_OVERHEAD
147
  enc->entropy = 0;
148
  enc->nb_symbols = 0;
149
#endif
150
249k
}
151
152
/*Frees the buffers used by the encoder.*/
153
249k
void od_ec_enc_clear(od_ec_enc *enc) { free(enc->buf); }
154
155
/*Encodes a symbol given its frequency in Q15.
156
  fl: CDF_PROB_TOP minus the cumulative frequency of all symbols that come
157
  before the one to be encoded.
158
  fh: CDF_PROB_TOP minus the cumulative frequency of all symbols up to and
159
  including the one to be encoded.*/
160
static void od_ec_encode_q15(od_ec_enc *enc, unsigned fl, unsigned fh, int s,
161
1.40G
                             int nsyms) {
162
1.40G
  od_ec_enc_window l;
163
1.40G
  unsigned r;
164
1.40G
  unsigned u;
165
1.40G
  unsigned v;
166
1.40G
  l = enc->low;
167
1.40G
  r = enc->rng;
168
1.40G
  assert(32768U <= r);
169
1.40G
  assert(fh <= fl);
170
1.40G
  assert(fl <= 32768U);
171
1.40G
  assert(7 - EC_PROB_SHIFT >= 0);
172
1.40G
  const int N = nsyms - 1;
173
1.40G
  if (fl < CDF_PROB_TOP) {
174
1.13G
    u = ((r >> 8) * (uint32_t)(fl >> EC_PROB_SHIFT) >> (7 - EC_PROB_SHIFT)) +
175
1.13G
        EC_MIN_PROB * (N - (s - 1));
176
1.13G
    v = ((r >> 8) * (uint32_t)(fh >> EC_PROB_SHIFT) >> (7 - EC_PROB_SHIFT)) +
177
1.13G
        EC_MIN_PROB * (N - (s + 0));
178
1.13G
    l += r - u;
179
1.13G
    r = u - v;
180
1.13G
  } else {
181
270M
    r -= ((r >> 8) * (uint32_t)(fh >> EC_PROB_SHIFT) >> (7 - EC_PROB_SHIFT)) +
182
270M
         EC_MIN_PROB * (N - (s + 0));
183
270M
  }
184
1.40G
  od_ec_enc_normalize(enc, l, r);
185
#if OD_MEASURE_EC_OVERHEAD
186
  enc->entropy -= OD_LOG2((double)(OD_ICDF(fh) - OD_ICDF(fl)) / CDF_PROB_TOP.);
187
  enc->nb_symbols++;
188
#endif
189
1.40G
}
190
191
/*Encode a single binary value.
192
  val: The value to encode (0 or 1).
193
  f: The probability that the val is one, scaled by 32768.*/
194
1.28G
void od_ec_encode_bool_q15(od_ec_enc *enc, int val, unsigned f) {
195
1.28G
  od_ec_enc_window l;
196
1.28G
  unsigned r;
197
1.28G
  unsigned v;
198
1.28G
  assert(0 < f);
199
1.28G
  assert(f < 32768U);
200
1.28G
  l = enc->low;
201
1.28G
  r = enc->rng;
202
1.28G
  assert(32768U <= r);
203
1.28G
  v = ((r >> 8) * (uint32_t)(f >> EC_PROB_SHIFT) >> (7 - EC_PROB_SHIFT));
204
1.28G
  v += EC_MIN_PROB;
205
1.28G
  if (val) l += r - v;
206
1.28G
  r = val ? v : r - v;
207
1.28G
  od_ec_enc_normalize(enc, l, r);
208
#if OD_MEASURE_EC_OVERHEAD
209
  enc->entropy -= OD_LOG2((double)(val ? f : (32768 - f)) / 32768.);
210
  enc->nb_symbols++;
211
#endif
212
1.28G
}
213
214
/*Encodes a symbol given a cumulative distribution function (CDF) table in Q15.
215
  s: The index of the symbol to encode.
216
  icdf: 32768 minus the CDF, such that symbol s falls in the range
217
         [s > 0 ? (32768 - icdf[s - 1]) : 0, 32768 - icdf[s]).
218
        The values must be monotonically decreasing, and icdf[nsyms - 1] must
219
         be 0.
220
  nsyms: The number of symbols in the alphabet.
221
         This should be at most 16.*/
222
void od_ec_encode_cdf_q15(od_ec_enc *enc, int s, const uint16_t *icdf,
223
1.40G
                          int nsyms) {
224
1.40G
  (void)nsyms;
225
1.40G
  assert(s >= 0);
226
1.40G
  assert(s < nsyms);
227
1.40G
  assert(icdf[nsyms - 1] == OD_ICDF(CDF_PROB_TOP));
228
1.40G
  od_ec_encode_q15(enc, s > 0 ? icdf[s - 1] : OD_ICDF(0), icdf[s], s, nsyms);
229
1.40G
}
230
231
#if OD_MEASURE_EC_OVERHEAD
232
#include <stdio.h>
233
#endif
234
235
/*Indicates that there are no more symbols to encode.
236
  All remaining output bytes are flushed to the output buffer.
237
  od_ec_enc_reset() should be called before using the encoder again.
238
  bytes: Returns the size of the encoded data in the returned buffer.
239
  Return: A pointer to the start of the final buffer, or NULL if there was an
240
           encoding error.*/
241
249k
unsigned char *od_ec_enc_done(od_ec_enc *enc, uint32_t *nbytes) {
242
249k
  unsigned char *out;
243
249k
  uint32_t storage;
244
249k
  uint32_t offs;
245
249k
  od_ec_enc_window m;
246
249k
  od_ec_enc_window e;
247
249k
  od_ec_enc_window l;
248
249k
  int c;
249
249k
  int s;
250
249k
  if (enc->error) return NULL;
251
#if OD_MEASURE_EC_OVERHEAD
252
  {
253
    uint32_t tell;
254
    /* Don't count the 1 bit we lose to raw bits as overhead. */
255
    tell = od_ec_enc_tell(enc) - 1;
256
    fprintf(stderr, "overhead: %f%%\n",
257
            100 * (tell - enc->entropy) / enc->entropy);
258
    fprintf(stderr, "efficiency: %f bits/symbol\n",
259
            (double)tell / enc->nb_symbols);
260
  }
261
#endif
262
263
249k
  l = enc->low;
264
249k
  c = enc->cnt;
265
249k
  s = 10;
266
249k
  m = 0x3FFF;
267
249k
  e = ((l + m) & ~m) | (m + 1);
268
249k
  s += c;
269
249k
  offs = enc->offs;
270
271
  /*Make sure there's enough room for the entropy-coded bits.*/
272
249k
  out = enc->buf;
273
249k
  storage = enc->storage;
274
249k
  const int s_bits = (s + 7) >> 3;
275
249k
  int b = OD_MAXI(s_bits, 0);
276
249k
  if (offs + b > storage) {
277
3
    storage = offs + b;
278
3
    out = (unsigned char *)realloc(out, sizeof(*out) * storage);
279
3
    if (out == NULL) {
280
0
      enc->error = -1;
281
0
      return NULL;
282
0
    }
283
3
    enc->buf = out;
284
3
    enc->storage = storage;
285
3
  }
286
287
  /*We output the minimum number of bits that ensures that the symbols encoded
288
     thus far will be decoded correctly regardless of the bits that follow.*/
289
249k
  if (s > 0) {
290
249k
    uint64_t n;
291
249k
    n = ((uint64_t)1 << (c + 16)) - 1;
292
909k
    do {
293
909k
      assert(offs < storage);
294
909k
      uint16_t val = (uint16_t)(e >> (c + 16));
295
909k
      out[offs] = (unsigned char)(val & 0x00FF);
296
909k
      if (val & 0x0100) {
297
75.6k
        assert(offs > 0);
298
75.6k
        propagate_carry_bwd(out, offs - 1);
299
75.6k
      }
300
909k
      offs++;
301
302
909k
      e &= n;
303
909k
      s -= 8;
304
909k
      c -= 8;
305
909k
      n >>= 8;
306
909k
    } while (s > 0);
307
249k
  }
308
249k
  *nbytes = offs;
309
310
249k
  return out;
311
249k
}
312
313
/*Returns the number of bits "used" by the encoded symbols so far.
314
  This same number can be computed in either the encoder or the decoder, and is
315
   suitable for making coding decisions.
316
  Warning: The value returned by this function can decrease compared to an
317
   earlier call, even after encoding more data, if there is an encoding error
318
   (i.e., a failure to allocate enough space for the output buffer).
319
  Return: The number of bits.
320
          This will always be slightly larger than the exact value (e.g., all
321
           rounding error is in the positive direction).*/
322
12.5M
int od_ec_enc_tell(const od_ec_enc *enc) {
323
  /*The 10 here counteracts the offset of -9 baked into cnt, and adds 1 extra
324
     bit, which we reserve for terminating the stream.*/
325
12.5M
  return (enc->cnt + 10) + enc->offs * 8;
326
12.5M
}
327
328
/*Returns the number of bits "used" by the encoded symbols so far.
329
  This same number can be computed in either the encoder or the decoder, and is
330
   suitable for making coding decisions.
331
  Warning: The value returned by this function can decrease compared to an
332
   earlier call, even after encoding more data, if there is an encoding error
333
   (i.e., a failure to allocate enough space for the output buffer).
334
  Return: The number of bits scaled by 2**OD_BITRES.
335
          This will always be slightly larger than the exact value (e.g., all
336
           rounding error is in the positive direction).*/
337
0
uint32_t od_ec_enc_tell_frac(const od_ec_enc *enc) {
338
0
  return od_ec_tell_frac(od_ec_enc_tell(enc), enc->rng);
339
0
}