| /* |
| * Copyright (C) 2006 Aurelien Jacobs <aurel@gnuage.org> |
| * |
| * This file is part of FFmpeg. |
| * |
| * FFmpeg is free software; you can redistribute it and/or |
| * modify it under the terms of the GNU Lesser General Public |
| * License as published by the Free Software Foundation; either |
| * version 2.1 of the License, or (at your option) any later version. |
| * |
| * FFmpeg is distributed in the hope that it will be useful, |
| * but WITHOUT ANY WARRANTY; without even the implied warranty of |
| * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU |
| * Lesser General Public License for more details. |
| * |
| * You should have received a copy of the GNU Lesser General Public |
| * License along with FFmpeg; if not, write to the Free Software |
| * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA |
| */ |
| |
| /** |
| * @file |
| * Common VP5-VP9 range decoder stuff |
| */ |
| |
| #ifndef AVCODEC_VPX_RAC_H |
| #define AVCODEC_VPX_RAC_H |
| |
| #include <stdint.h> |
| |
| #include "config.h" |
| #include "libavutil/attributes.h" |
| #include "bytestream.h" |
| |
| typedef struct VPXRangeCoder { |
| int high; |
| int bits; /* stored negated (i.e. negative "bits" is a positive number of |
| bits left) in order to eliminate a negate in cache refilling */ |
| const uint8_t *buffer; |
| const uint8_t *end; |
| unsigned int code_word; |
| int end_reached; |
| } VPXRangeCoder; |
| |
| extern const uint8_t ff_vpx_norm_shift[256]; |
| int ff_vpx_init_range_decoder(VPXRangeCoder *c, const uint8_t *buf, int buf_size); |
| |
| /** |
| * returns 1 if the end of the stream has been reached, 0 otherwise. |
| */ |
| static av_always_inline int vpx_rac_is_end(VPXRangeCoder *c) |
| { |
| if (c->end <= c->buffer && c->bits >= 0) |
| c->end_reached ++; |
| return c->end_reached > 10; |
| } |
| |
| static av_always_inline unsigned int vpx_rac_renorm(VPXRangeCoder *c) |
| { |
| int shift = ff_vpx_norm_shift[c->high]; |
| int bits = c->bits; |
| unsigned int code_word = c->code_word; |
| |
| c->high <<= shift; |
| code_word <<= shift; |
| bits += shift; |
| if(bits >= 0 && c->buffer < c->end) { |
| code_word |= bytestream_get_be16(&c->buffer) << bits; |
| bits -= 16; |
| } |
| c->bits = bits; |
| return code_word; |
| } |
| |
| #if ARCH_ARM |
| #include "arm/vpx_arith.h" |
| #elif ARCH_X86 |
| #include "x86/vpx_arith.h" |
| #endif |
| |
| #ifndef vpx_rac_get_prob |
| #define vpx_rac_get_prob vpx_rac_get_prob |
| static av_always_inline int vpx_rac_get_prob(VPXRangeCoder *c, uint8_t prob) |
| { |
| unsigned int code_word = vpx_rac_renorm(c); |
| unsigned int low = 1 + (((c->high - 1) * prob) >> 8); |
| unsigned int low_shift = low << 16; |
| int bit = code_word >= low_shift; |
| |
| c->high = bit ? c->high - low : low; |
| c->code_word = bit ? code_word - low_shift : code_word; |
| |
| return bit; |
| } |
| #endif |
| |
| #ifndef vpx_rac_get_prob_branchy |
| // branchy variant, to be used where there's a branch based on the bit decoded |
| static av_always_inline int vpx_rac_get_prob_branchy(VPXRangeCoder *c, int prob) |
| { |
| unsigned long code_word = vpx_rac_renorm(c); |
| unsigned low = 1 + (((c->high - 1) * prob) >> 8); |
| unsigned low_shift = low << 16; |
| |
| if (code_word >= low_shift) { |
| c->high -= low; |
| c->code_word = code_word - low_shift; |
| return 1; |
| } |
| |
| c->high = low; |
| c->code_word = code_word; |
| return 0; |
| } |
| #endif |
| |
| static av_always_inline int vpx_rac_get(VPXRangeCoder *c) |
| { |
| unsigned int code_word = vpx_rac_renorm(c); |
| /* equiprobable */ |
| int low = (c->high + 1) >> 1; |
| unsigned int low_shift = low << 16; |
| int bit = code_word >= low_shift; |
| if (bit) { |
| c->high -= low; |
| code_word -= low_shift; |
| } else { |
| c->high = low; |
| } |
| |
| c->code_word = code_word; |
| return bit; |
| } |
| |
| #endif /* AVCODEC_VPX_RAC_H */ |