xref: /aosp_15_r20/external/libaom/aom_dsp/entenc.h (revision 77c1e3ccc04c968bd2bc212e87364f250e820521)
1*77c1e3ccSAndroid Build Coastguard Worker /*
2*77c1e3ccSAndroid Build Coastguard Worker  * Copyright (c) 2001-2016, Alliance for Open Media. All rights reserved.
3*77c1e3ccSAndroid Build Coastguard Worker  *
4*77c1e3ccSAndroid Build Coastguard Worker  * This source code is subject to the terms of the BSD 2 Clause License and
5*77c1e3ccSAndroid Build Coastguard Worker  * the Alliance for Open Media Patent License 1.0. If the BSD 2 Clause License
6*77c1e3ccSAndroid Build Coastguard Worker  * was not distributed with this source code in the LICENSE file, you can
7*77c1e3ccSAndroid Build Coastguard Worker  * obtain it at www.aomedia.org/license/software. If the Alliance for Open
8*77c1e3ccSAndroid Build Coastguard Worker  * Media Patent License 1.0 was not distributed with this source code in the
9*77c1e3ccSAndroid Build Coastguard Worker  * PATENTS file, you can obtain it at www.aomedia.org/license/patent.
10*77c1e3ccSAndroid Build Coastguard Worker  */
11*77c1e3ccSAndroid Build Coastguard Worker 
12*77c1e3ccSAndroid Build Coastguard Worker #ifndef AOM_AOM_DSP_ENTENC_H_
13*77c1e3ccSAndroid Build Coastguard Worker #define AOM_AOM_DSP_ENTENC_H_
14*77c1e3ccSAndroid Build Coastguard Worker #include <stddef.h>
15*77c1e3ccSAndroid Build Coastguard Worker #include "aom_dsp/entcode.h"
16*77c1e3ccSAndroid Build Coastguard Worker #include "aom_util/endian_inl.h"
17*77c1e3ccSAndroid Build Coastguard Worker 
18*77c1e3ccSAndroid Build Coastguard Worker #ifdef __cplusplus
19*77c1e3ccSAndroid Build Coastguard Worker extern "C" {
20*77c1e3ccSAndroid Build Coastguard Worker #endif
21*77c1e3ccSAndroid Build Coastguard Worker 
22*77c1e3ccSAndroid Build Coastguard Worker typedef uint64_t od_ec_enc_window;
23*77c1e3ccSAndroid Build Coastguard Worker 
24*77c1e3ccSAndroid Build Coastguard Worker typedef struct od_ec_enc od_ec_enc;
25*77c1e3ccSAndroid Build Coastguard Worker 
26*77c1e3ccSAndroid Build Coastguard Worker #define OD_MEASURE_EC_OVERHEAD (0)
27*77c1e3ccSAndroid Build Coastguard Worker 
28*77c1e3ccSAndroid Build Coastguard Worker /*The entropy encoder context.*/
29*77c1e3ccSAndroid Build Coastguard Worker struct od_ec_enc {
30*77c1e3ccSAndroid Build Coastguard Worker   /*Buffered output.
31*77c1e3ccSAndroid Build Coastguard Worker     This contains only the raw bits until the final call to od_ec_enc_done(),
32*77c1e3ccSAndroid Build Coastguard Worker      where all the arithmetic-coded data gets prepended to it.*/
33*77c1e3ccSAndroid Build Coastguard Worker   unsigned char *buf;
34*77c1e3ccSAndroid Build Coastguard Worker   /*The size of the buffer.*/
35*77c1e3ccSAndroid Build Coastguard Worker   uint32_t storage;
36*77c1e3ccSAndroid Build Coastguard Worker   /*The offset at which the next entropy-coded byte will be written.*/
37*77c1e3ccSAndroid Build Coastguard Worker   uint32_t offs;
38*77c1e3ccSAndroid Build Coastguard Worker   /*The low end of the current range.*/
39*77c1e3ccSAndroid Build Coastguard Worker   od_ec_enc_window low;
40*77c1e3ccSAndroid Build Coastguard Worker   /*The number of values in the current range.*/
41*77c1e3ccSAndroid Build Coastguard Worker   uint16_t rng;
42*77c1e3ccSAndroid Build Coastguard Worker   /*The number of bits of data in the current value.*/
43*77c1e3ccSAndroid Build Coastguard Worker   int16_t cnt;
44*77c1e3ccSAndroid Build Coastguard Worker   /*Nonzero if an error occurred.*/
45*77c1e3ccSAndroid Build Coastguard Worker   int error;
46*77c1e3ccSAndroid Build Coastguard Worker #if OD_MEASURE_EC_OVERHEAD
47*77c1e3ccSAndroid Build Coastguard Worker   double entropy;
48*77c1e3ccSAndroid Build Coastguard Worker   int nb_symbols;
49*77c1e3ccSAndroid Build Coastguard Worker #endif
50*77c1e3ccSAndroid Build Coastguard Worker };
51*77c1e3ccSAndroid Build Coastguard Worker 
52*77c1e3ccSAndroid Build Coastguard Worker /*See entenc.c for further documentation.*/
53*77c1e3ccSAndroid Build Coastguard Worker 
54*77c1e3ccSAndroid Build Coastguard Worker void od_ec_enc_init(od_ec_enc *enc, uint32_t size) OD_ARG_NONNULL(1);
55*77c1e3ccSAndroid Build Coastguard Worker void od_ec_enc_reset(od_ec_enc *enc) OD_ARG_NONNULL(1);
56*77c1e3ccSAndroid Build Coastguard Worker void od_ec_enc_clear(od_ec_enc *enc) OD_ARG_NONNULL(1);
57*77c1e3ccSAndroid Build Coastguard Worker 
58*77c1e3ccSAndroid Build Coastguard Worker void od_ec_encode_bool_q15(od_ec_enc *enc, int val, unsigned f_q15)
59*77c1e3ccSAndroid Build Coastguard Worker     OD_ARG_NONNULL(1);
60*77c1e3ccSAndroid Build Coastguard Worker void od_ec_encode_cdf_q15(od_ec_enc *enc, int s, const uint16_t *cdf, int nsyms)
61*77c1e3ccSAndroid Build Coastguard Worker     OD_ARG_NONNULL(1) OD_ARG_NONNULL(3);
62*77c1e3ccSAndroid Build Coastguard Worker 
63*77c1e3ccSAndroid Build Coastguard Worker void od_ec_enc_bits(od_ec_enc *enc, uint32_t fl, unsigned ftb)
64*77c1e3ccSAndroid Build Coastguard Worker     OD_ARG_NONNULL(1);
65*77c1e3ccSAndroid Build Coastguard Worker 
66*77c1e3ccSAndroid Build Coastguard Worker void od_ec_enc_patch_initial_bits(od_ec_enc *enc, unsigned val, int nbits)
67*77c1e3ccSAndroid Build Coastguard Worker     OD_ARG_NONNULL(1);
68*77c1e3ccSAndroid Build Coastguard Worker OD_WARN_UNUSED_RESULT unsigned char *od_ec_enc_done(od_ec_enc *enc,
69*77c1e3ccSAndroid Build Coastguard Worker                                                     uint32_t *nbytes)
70*77c1e3ccSAndroid Build Coastguard Worker     OD_ARG_NONNULL(1) OD_ARG_NONNULL(2);
71*77c1e3ccSAndroid Build Coastguard Worker 
72*77c1e3ccSAndroid Build Coastguard Worker OD_WARN_UNUSED_RESULT int od_ec_enc_tell(const od_ec_enc *enc)
73*77c1e3ccSAndroid Build Coastguard Worker     OD_ARG_NONNULL(1);
74*77c1e3ccSAndroid Build Coastguard Worker OD_WARN_UNUSED_RESULT uint32_t od_ec_enc_tell_frac(const od_ec_enc *enc)
75*77c1e3ccSAndroid Build Coastguard Worker     OD_ARG_NONNULL(1);
76*77c1e3ccSAndroid Build Coastguard Worker 
77*77c1e3ccSAndroid Build Coastguard Worker // buf is the frame bitbuffer, offs is where carry to be added
propagate_carry_bwd(unsigned char * buf,uint32_t offs)78*77c1e3ccSAndroid Build Coastguard Worker static inline void propagate_carry_bwd(unsigned char *buf, uint32_t offs) {
79*77c1e3ccSAndroid Build Coastguard Worker   uint16_t sum, carry = 1;
80*77c1e3ccSAndroid Build Coastguard Worker   do {
81*77c1e3ccSAndroid Build Coastguard Worker     sum = (uint16_t)buf[offs] + 1;
82*77c1e3ccSAndroid Build Coastguard Worker     buf[offs--] = (unsigned char)sum;
83*77c1e3ccSAndroid Build Coastguard Worker     carry = sum >> 8;
84*77c1e3ccSAndroid Build Coastguard Worker   } while (carry);
85*77c1e3ccSAndroid Build Coastguard Worker }
86*77c1e3ccSAndroid Build Coastguard Worker 
87*77c1e3ccSAndroid Build Coastguard Worker // Convert to big-endian byte order and write data to buffer adding the
88*77c1e3ccSAndroid Build Coastguard Worker // carry-bit
write_enc_data_to_out_buf(unsigned char * out,uint32_t offs,uint64_t output,uint64_t carry,uint32_t * enc_offs,uint8_t num_bytes_ready)89*77c1e3ccSAndroid Build Coastguard Worker static inline void write_enc_data_to_out_buf(unsigned char *out, uint32_t offs,
90*77c1e3ccSAndroid Build Coastguard Worker                                              uint64_t output, uint64_t carry,
91*77c1e3ccSAndroid Build Coastguard Worker                                              uint32_t *enc_offs,
92*77c1e3ccSAndroid Build Coastguard Worker                                              uint8_t num_bytes_ready) {
93*77c1e3ccSAndroid Build Coastguard Worker   const uint64_t reg = HToBE64(output << ((8 - num_bytes_ready) << 3));
94*77c1e3ccSAndroid Build Coastguard Worker   memcpy(&out[offs], &reg, 8);
95*77c1e3ccSAndroid Build Coastguard Worker   // Propagate carry backwards if exists
96*77c1e3ccSAndroid Build Coastguard Worker   if (carry) {
97*77c1e3ccSAndroid Build Coastguard Worker     assert(offs > 0);
98*77c1e3ccSAndroid Build Coastguard Worker     propagate_carry_bwd(out, offs - 1);
99*77c1e3ccSAndroid Build Coastguard Worker   }
100*77c1e3ccSAndroid Build Coastguard Worker   *enc_offs = offs + num_bytes_ready;
101*77c1e3ccSAndroid Build Coastguard Worker }
102*77c1e3ccSAndroid Build Coastguard Worker 
103*77c1e3ccSAndroid Build Coastguard Worker #ifdef __cplusplus
104*77c1e3ccSAndroid Build Coastguard Worker }  // extern "C"
105*77c1e3ccSAndroid Build Coastguard Worker #endif
106*77c1e3ccSAndroid Build Coastguard Worker 
107*77c1e3ccSAndroid Build Coastguard Worker #endif  // AOM_AOM_DSP_ENTENC_H_
108