123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137 |
- /*
- * Copyright (c) 2021, Hunter Salyer <thefalsehonesty@gmail.com>
- * Copyright (c) 2022, Gregory Bertilson <zaggy1024@gmail.com>
- *
- * SPDX-License-Identifier: BSD-2-Clause
- */
- #include <AK/BuiltinWrappers.h>
- #include <AK/Debug.h>
- #include <AK/Endian.h>
- #include "BooleanDecoder.h"
- namespace Gfx {
- // 9.2.1 Initialization process for Boolean decoder
- ErrorOr<BooleanDecoder> BooleanDecoder::initialize(ReadonlyBytes data)
- {
- if (data.size() == 0)
- return Error::from_string_literal("Size of decoder range cannot be zero");
- // NOTE: This implementation is shared between VP8 and VP9. Therefore, we do not check the
- // marker bit at the start of the range decode that is required in the VP9 specification.
- // This is instead handled by the function that instantiates all range decoders for the
- // VP9 decoder.
- // NOTE: As noted below in fill_reservoir(), we read in multi-byte-sized chunks,
- // so here we will deviate from the standard to count in bytes rather than bits.
- return BooleanDecoder { data.data(), data.size() };
- }
- // Instead of filling the value field one bit at a time as the spec suggests, we store the
- // data to be read in a reservoir of greater than one byte. This allows us to read out data
- // for the entire reservoir at once, avoiding a lot of branch misses in read_bool().
- void BooleanDecoder::fill_reservoir()
- {
- if (m_value_bits_left > 8)
- return;
- // Defer errors until the decode is finalized, so the work to check for errors and return them only has
- // to be done once. Not refilling the reservoir here will only result in reading out all zeroes until
- // the range decode is finished.
- if (m_bytes_left == 0) {
- dbgln_if(VPX_DEBUG, "BooleanDecoder has read past the end of the coded range");
- m_overread = true;
- return;
- }
- // Read the data into the most significant bits of a variable.
- auto read_size = min<size_t>(reserve_bytes, m_bytes_left);
- ValueType read_value = 0;
- memcpy(&read_value, m_data, read_size);
- read_value = AK::convert_between_host_and_big_endian(read_value);
- // Skip the number of bytes read in the data.
- m_data += read_size;
- m_bytes_left -= read_size;
- // Shift the value that was read to be less significant than the least significant bit available in the reservoir.
- read_value >>= m_value_bits_left;
- m_value |= read_value;
- m_value_bits_left += read_size * 8;
- }
- // 9.2.2 Boolean decoding process
- bool BooleanDecoder::read_bool(u8 probability)
- {
- auto split = 1u + (((m_range - 1u) * probability) >> 8u);
- // The actual value being read resides in the most significant 8 bits
- // of the value field, so we shift the split into that range for comparison.
- auto split_shifted = static_cast<ValueType>(split) << reserve_bits;
- bool return_bool;
- if (m_value < split_shifted) {
- m_range = split;
- return_bool = false;
- } else {
- m_range -= split;
- m_value -= split_shifted;
- return_bool = true;
- }
- u8 bits_to_shift_into_range = count_leading_zeroes(m_range) - ((sizeof(m_range) - 1) * 8);
- m_range <<= bits_to_shift_into_range;
- m_value <<= bits_to_shift_into_range;
- m_value_bits_left -= bits_to_shift_into_range;
- fill_reservoir();
- return return_bool;
- }
- // 9.2.4 Parsing process for read_literal
- u8 BooleanDecoder::read_literal(u8 bits)
- {
- u8 return_value = 0;
- for (size_t i = 0; i < bits; i++) {
- return_value = (2 * return_value) + read_bool(128);
- }
- return return_value;
- }
- ErrorOr<void> BooleanDecoder::finish_decode()
- {
- if (m_overread)
- return Error::from_string_literal("Range decoder was read past the end of its data");
- #if VPX_DEBUG
- // 9.2.3 Exit process for Boolean decoder
- //
- // This process is invoked when the function exit_bool( ) is called from the syntax structure.
- //
- // The padding syntax element is read using the f(BoolMaxBits) parsing process.
- //
- // It is a requirement of bitstream conformance that padding is equal to 0.
- //
- // NOTE: This requirement holds up for all of our WebP lossy test inputs, as well.
- bool padding_good = true;
- if (m_value != 0)
- padding_good = false;
- while (m_bytes_left > 0) {
- if (*m_data != 0)
- padding_good = false;
- m_data++;
- m_bytes_left--;
- }
- if (!padding_good)
- return Error::from_string_literal("Range decoder padding was non-zero");
- #endif
- return {};
- }
- }
|