| /* |
| * 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_hint) { |
| c->error = error; |
| c->size = size_hint; |
| 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; |
| } |
| |
| 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; |
| } |
| |
| 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); |
| for (offset = c->offset - 1; offset >= 0; --offset) { |
| if (c->buf[offset].method == ANS_METHOD_RANS) { |
| struct rans_sym sym; |
| sym.prob = c->buf[offset].prob; |
| sym.cum_prob = c->buf[offset].val_start; |
| rans_write(&c->ans, &sym); |
| } else { |
| uabs_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); |
| } |