/* * 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 #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) { c->error = error; 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); }