#![forbid(unsafe_code)]
#[cfg(feature = "alloc")]
use alloc::vec;
#[cfg(feature = "alloc")]
use alloc::vec::Vec;
use super::primitives;
use crate::bitstream::reader_reverse::ReverseBitReader;
use crate::error::DecompressError;
use crate::huffman::HuffmanDecodeEntry;
pub fn decode_single_stream(
table: &[HuffmanDecodeEntry],
table_log: u8,
data: &[u8],
output_size: usize,
) -> Result<Vec<u8>, DecompressError> {
let mut reader = ReverseBitReader::new(data).map_err(|_| DecompressError::BadHuffmanStream)?;
let mut output = Vec::with_capacity(output_size);
let tl = table_log as usize;
while output.len() + 4 <= output_size && reader.bits_remaining() >= tl + tl + tl + tl {
for _ in 0..4 {
let bits = reader.peek_bits(table_log);
let entry = primitives::huf_table_lookup(table, bits as usize);
output.push(entry.symbol);
reader.consume_bits(entry.num_bits);
}
}
while output.len() < output_size {
reader.refill();
let remaining = reader.bits_remaining();
if remaining == 0 {
return Err(DecompressError::BadHuffmanStream);
}
let bits = if remaining >= tl {
reader.peek_bits(table_log)
} else {
reader.peek_bits(remaining as u8) << (tl - remaining)
};
let entry = primitives::huf_table_lookup(table, bits as usize);
if entry.num_bits as usize > remaining {
return Err(DecompressError::BadHuffmanStream);
}
output.push(entry.symbol);
let _ = reader.read_bits(entry.num_bits)?;
}
Ok(output)
}
pub fn decode_single_stream_into(
table: &[HuffmanDecodeEntry],
table_log: u8,
data: &[u8],
output: &mut [u8],
) -> Result<(), DecompressError> {
let mut reader = ReverseBitReader::new(data).map_err(|_| DecompressError::BadHuffmanStream)?;
decode_stream_tail(table, table_log, &mut reader, output)
}
pub fn decode_single_stream_vec(
table: &[HuffmanDecodeEntry],
table_log: u8,
data: &[u8],
output_size: usize,
output: &mut Vec<u8>,
) -> Result<(), DecompressError> {
output.clear();
output.reserve(output_size);
primitives::set_vec_len(output, output_size);
let result;
#[cfg(target_arch = "x86_64")]
{
if crate::simd::cpu_tier() >= crate::simd::CpuTier::Bmi2 {
result = super::decode_4stream::decode_single_stream_bmi2_safe(
table, table_log, data, output,
);
if result.is_err() {
output.clear();
}
return result;
}
}
result = decode_single_stream_into(table, table_log, data, output);
if result.is_err() {
output.clear();
}
result
}
pub fn decode_4_streams(
table: &[HuffmanDecodeEntry],
table_log: u8,
data: &[u8],
output_size: usize,
) -> Result<Vec<u8>, DecompressError> {
let mut output = vec![0u8; output_size];
super::decode_4stream::decode_4_streams_core(table, table_log, data, output_size, &mut output)?;
Ok(output)
}
pub fn decode_4_streams_into(
table: &[HuffmanDecodeEntry],
table_log: u8,
data: &[u8],
output_size: usize,
output: &mut Vec<u8>,
) -> Result<(), DecompressError> {
output.clear();
output.reserve(output_size);
primitives::set_vec_len(output, output_size);
let result;
#[cfg(target_arch = "x86_64")]
{
if crate::simd::cpu_tier() >= crate::simd::CpuTier::Bmi2 {
result = super::decode_4stream::decode_4_streams_core_bmi2_safe(
table,
table_log,
data,
output_size,
output,
);
if result.is_err() {
output.clear();
}
return result;
}
}
result =
super::decode_4stream::decode_4_streams_core(table, table_log, data, output_size, output);
if result.is_err() {
output.clear();
}
result
}
pub(super) fn decode_stream_tail(
table: &[HuffmanDecodeEntry],
table_log: u8,
reader: &mut ReverseBitReader,
output: &mut [u8],
) -> Result<(), DecompressError> {
let output_size = output.len();
let tl = table_log as usize;
let mut pos = 0;
reader.refill();
while pos + 4 <= output_size && reader.bits_remaining() >= tl + tl + tl + tl {
for _ in 0..4 {
let bits = reader.peek_bits(table_log);
let entry = primitives::huf_table_lookup(table, bits as usize);
primitives::huf_output_write(output, pos, entry.symbol);
pos += 1;
reader.consume_bits(entry.num_bits);
}
}
while pos < output_size {
reader.refill();
let remaining = reader.bits_remaining();
if remaining == 0 {
return Err(DecompressError::BadHuffmanStream);
}
let bits = if remaining >= tl {
reader.peek_bits(table_log)
} else {
reader.peek_bits(remaining as u8) << (tl - remaining)
};
let entry = primitives::huf_table_lookup(table, bits as usize);
if entry.num_bits as usize > remaining {
return Err(DecompressError::BadHuffmanStream);
}
primitives::huf_output_write(output, pos, entry.symbol);
pos += 1;
let _ = reader.read_bits(entry.num_bits)?;
}
Ok(())
}