/* * Copyright (c) 2016, Alliance for Open Media. All rights reserved * * This source code is subject to the terms of the BSD 2 Clause License and * the Alliance for Open Media Patent License 1.0. If the BSD 2 Clause License * was not distributed with this source code in the LICENSE file, you can * obtain it at www.aomedia.org/license/software. If the Alliance for Open * Media Patent License 1.0 was not distributed with this source code in the * PATENTS file, you can obtain it at www.aomedia.org/license/patent. */ #ifndef AOM_AOM_DSP_BUF_ANS_H_ #define AOM_AOM_DSP_BUF_ANS_H_ // Buffered forward ANS writer. // Symbols are written to the writer in forward (decode) order and serialized // backwards due to ANS's stack like behavior. #include <assert.h> #include "config/aom_config.h" #include "aom/aom_integer.h" #include "aom_dsp/ans.h" #include "aom_dsp/answriter.h" #ifdef __cplusplus extern "C" { #endif // __cplusplus #define ANS_METHOD_RABS 0 #define ANS_METHOD_RANS 1 struct buffered_ans_symbol { unsigned int method : 1; // one of ANS_METHOD_RABS or ANS_METHOD_RANS // TODO(aconverse): Should be possible to write this in terms of start for ABS unsigned int val_start : RANS_PROB_BITS; // Boolean value for ABS // start in symbol cycle for Rans unsigned int prob : RANS_PROB_BITS; // Probability of this symbol }; struct BufAnsCoder { struct aom_internal_error_info *error; struct buffered_ans_symbol *buf; struct AnsCoder ans; int size; int offset; int output_bytes; #if ANS_MAX_SYMBOLS int window_size; #endif int pos; // Dummy variable to store the output buffer after closing uint8_t allow_update_cdf; }; // Allocate a buffered ANS coder to store size symbols. // When ANS_MAX_SYMBOLS is turned on, the size is the fixed size of each ANS // partition. // When ANS_MAX_SYMBOLS is turned off, size is merely an initial hint and the // buffer will grow on demand void aom_buf_ans_alloc(struct BufAnsCoder *c, struct aom_internal_error_info *error); void aom_buf_ans_free(struct BufAnsCoder *c); #if !ANS_MAX_SYMBOLS void aom_buf_ans_grow(struct BufAnsCoder *c); #endif void aom_buf_ans_flush(struct BufAnsCoder *const c); static INLINE void buf_ans_write_init(struct BufAnsCoder *const c, uint8_t *const output_buffer) { c->offset = 0; c->output_bytes = 0; ans_write_init(&c->ans, output_buffer); } static INLINE void buf_rabs_write(struct BufAnsCoder *const c, uint8_t val, AnsP8 prob) { assert(c->offset <= c->size); #if !ANS_MAX_SYMBOLS if (c->offset == c->size) { aom_buf_ans_grow(c); } #endif c->buf[c->offset].method = ANS_METHOD_RABS; c->buf[c->offset].val_start = val; c->buf[c->offset].prob = prob; ++c->offset; #if ANS_MAX_SYMBOLS if (c->offset == c->size) aom_buf_ans_flush(c); #endif } // Buffer one symbol for encoding using rANS. // cum_prob: The cumulative probability before this symbol (the offset of // the symbol in the symbol cycle) // prob: The probability of this symbol (l_s from the paper) // RANS_PRECISION takes the place of m from the paper. static INLINE void buf_rans_write(struct BufAnsCoder *const c, aom_cdf_prob cum_prob, aom_cdf_prob prob) { assert(c->offset <= c->size); #if !ANS_MAX_SYMBOLS if (c->offset == c->size) { aom_buf_ans_grow(c); } #endif c->buf[c->offset].method = ANS_METHOD_RANS; c->buf[c->offset].val_start = cum_prob; c->buf[c->offset].prob = prob; ++c->offset; #if ANS_MAX_SYMBOLS if (c->offset == c->size) aom_buf_ans_flush(c); #endif } static INLINE void buf_rabs_write_bit(struct BufAnsCoder *c, int bit) { buf_rabs_write(c, bit, 128); } static INLINE void buf_rabs_write_literal(struct BufAnsCoder *c, int literal, int bits) { int bit; assert(bits < 31); for (bit = bits - 1; bit >= 0; bit--) buf_rabs_write_bit(c, 1 & (literal >> bit)); } static INLINE int buf_ans_write_end(struct BufAnsCoder *const c) { assert(c->offset == 0); return c->output_bytes; } #ifdef __cplusplus } // extern "C" #endif // __cplusplus #endif // AOM_AOM_DSP_BUF_ANS_H_