| /* | 
 |  * 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. | 
 |  */ | 
 |  | 
 | #include <string.h> | 
 |  | 
 | #include "aom_dsp/buf_ans.h" | 
 | #include "aom_mem/aom_mem.h" | 
 | #include "aom/internal/aom_codec_internal.h" | 
 |  | 
 | void aom_buf_ans_alloc(struct BufAnsCoder *c, | 
 |                        struct aom_internal_error_info *error, int size) { | 
 |   c->error = error; | 
 |   c->size = size; | 
 |   assert(c->size > 1); | 
 |   AOM_CHECK_MEM_ERROR(error, c->buf, aom_malloc(c->size * sizeof(*c->buf))); | 
 |   // Initialize to overfull to trigger the assert in write. | 
 |   c->offset = c->size + 1; | 
 | } | 
 |  | 
 | void aom_buf_ans_free(struct BufAnsCoder *c) { | 
 |   aom_free(c->buf); | 
 |   c->buf = NULL; | 
 |   c->size = 0; | 
 | } | 
 |  | 
 | #if !ANS_MAX_SYMBOLS | 
 | void aom_buf_ans_grow(struct BufAnsCoder *c) { | 
 |   struct buffered_ans_symbol *new_buf = NULL; | 
 |   int new_size = c->size * 2; | 
 |   AOM_CHECK_MEM_ERROR(c->error, new_buf, | 
 |                       aom_malloc(new_size * sizeof(*new_buf))); | 
 |   memcpy(new_buf, c->buf, c->size * sizeof(*c->buf)); | 
 |   aom_free(c->buf); | 
 |   c->buf = new_buf; | 
 |   c->size = new_size; | 
 | } | 
 | #endif | 
 |  | 
 | void aom_buf_ans_flush(struct BufAnsCoder *const c) { | 
 |   int offset; | 
 | #if ANS_MAX_SYMBOLS | 
 |   if (c->offset == 0) return; | 
 | #endif | 
 |   assert(c->offset > 0); | 
 |   offset = c->offset - 1; | 
 |   // Code the first symbol such that it brings the state to the smallest normal | 
 |   // state from an initial state that would have been a subnormal/refill state. | 
 |   if (c->buf[offset].method == ANS_METHOD_RANS) { | 
 |     c->ans.state += c->buf[offset].val_start; | 
 |   } else { | 
 |     c->ans.state += c->buf[offset].val_start ? c->buf[offset].prob : 0; | 
 |   } | 
 |   for (offset = offset - 1; offset >= 0; --offset) { | 
 |     if (c->buf[offset].method == ANS_METHOD_RANS) { | 
 |       rans_write(&c->ans, c->buf[offset].val_start, c->buf[offset].prob); | 
 |     } else { | 
 |       rabs_write(&c->ans, (uint8_t)c->buf[offset].val_start, | 
 |                  (AnsP8)c->buf[offset].prob); | 
 |     } | 
 |   } | 
 |   c->offset = 0; | 
 |   c->output_bytes += ans_write_end(&c->ans); | 
 | } |