mirror of
				https://github.com/RGBCube/serenity
				synced 2025-10-25 20:42:38 +00:00 
			
		
		
		
	 24ae35086d
			
		
	
	
		24ae35086d
		
	
	
	
	
		
			
			Errors are now deferred until `finish_decode()` is finished, meaning branches to return errors only need to occur at the end of a ranged decode. If VPX_DEBUG is enabled, a debug message will be printed immediately when an overread occurs. Average decoding times for `Tests/LibGfx/test-inputs/4.webp` improve by about 4.7% with this change, absolute decode times changing from 27.4ms±1.1ms down to 26.1ms±1.0ms.
		
			
				
	
	
		
			137 lines
		
	
	
	
		
			4.4 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
			
		
		
	
	
			137 lines
		
	
	
	
		
			4.4 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
| /*
 | |
|  * 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 {};
 | |
| }
 | |
| 
 | |
| }
 |