use std::ops::Range;
use crate::decoder::DecodingError;
use super::lossless::subsample_size;
#[derive(Debug, Clone)]
pub(crate) enum TransformType {
PredictorTransform {
size_bits: u8,
predictor_data: Vec<u8>,
},
ColorTransform {
size_bits: u8,
transform_data: Vec<u8>,
},
SubtractGreen,
ColorIndexingTransform {
table_size: u16,
table_data: Vec<u8>,
},
}
pub(crate) fn apply_predictor_transform(
image_data: &mut [u8],
width: u16,
height: u16,
size_bits: u8,
predictor_data: &[u8],
) -> Result<(), DecodingError> {
let block_xsize = usize::from(subsample_size(width, size_bits));
let width = usize::from(width);
let height = usize::from(height);
image_data[3] = image_data[3].wrapping_add(255);
apply_predictor_transform_1(image_data, 4..width * 4, width);
for y in 1..height {
for i in 0..4 {
image_data[y * width * 4 + i] =
image_data[y * width * 4 + i].wrapping_add(image_data[(y - 1) * width * 4 + i]);
}
}
for y in 1..height {
for block_x in 0..block_xsize {
let block_index = (y >> size_bits) * block_xsize + block_x;
let predictor = predictor_data[block_index * 4 + 1];
let start_index = (y * width + (block_x << size_bits).max(1)) * 4;
let end_index = (y * width + ((block_x + 1) << size_bits).min(width)) * 4;
match predictor {
0 => apply_predictor_transform_0(image_data, start_index..end_index, width),
1 => apply_predictor_transform_1(image_data, start_index..end_index, width),
2 => apply_predictor_transform_2(image_data, start_index..end_index, width),
3 => apply_predictor_transform_3(image_data, start_index..end_index, width),
4 => apply_predictor_transform_4(image_data, start_index..end_index, width),
5 => apply_predictor_transform_5(image_data, start_index..end_index, width),
6 => apply_predictor_transform_6(image_data, start_index..end_index, width),
7 => apply_predictor_transform_7(image_data, start_index..end_index, width),
8 => apply_predictor_transform_8(image_data, start_index..end_index, width),
9 => apply_predictor_transform_9(image_data, start_index..end_index, width),
10 => apply_predictor_transform_10(image_data, start_index..end_index, width),
11 => apply_predictor_transform_11(image_data, start_index..end_index, width),
12 => apply_predictor_transform_12(image_data, start_index..end_index, width),
13 => apply_predictor_transform_13(image_data, start_index..end_index, width),
_ => {}
}
}
}
Ok(())
}
pub fn apply_predictor_transform_0(image_data: &mut [u8], range: Range<usize>, _width: usize) {
for i in ((range.start + 3)..range.end).step_by(4) {
image_data[i] = image_data[i].wrapping_add(0xff);
}
}
pub fn apply_predictor_transform_1(image_data: &mut [u8], range: Range<usize>, _width: usize) {
let mut prev: [u8; 4] = image_data[range.start - 4..][..4].try_into().unwrap();
for chunk in image_data[range].chunks_exact_mut(4) {
prev = [
chunk[0].wrapping_add(prev[0]),
chunk[1].wrapping_add(prev[1]),
chunk[2].wrapping_add(prev[2]),
chunk[3].wrapping_add(prev[3]),
];
chunk.copy_from_slice(&prev);
}
}
pub fn apply_predictor_transform_2(image_data: &mut [u8], range: Range<usize>, width: usize) {
for i in range {
image_data[i] = image_data[i].wrapping_add(image_data[i - width * 4]);
}
}
pub fn apply_predictor_transform_3(image_data: &mut [u8], range: Range<usize>, width: usize) {
for i in range {
image_data[i] = image_data[i].wrapping_add(image_data[i - width * 4 + 4]);
}
}
pub fn apply_predictor_transform_4(image_data: &mut [u8], range: Range<usize>, width: usize) {
for i in range {
image_data[i] = image_data[i].wrapping_add(image_data[i - width * 4 - 4]);
}
}
pub fn apply_predictor_transform_5(image_data: &mut [u8], range: Range<usize>, width: usize) {
let (old, current) = image_data[..range.end].split_at_mut(range.start);
let mut prev: [u8; 4] = old[range.start - 4..][..4].try_into().unwrap();
let top_right = &old[range.start - width * 4 + 4..];
let top = &old[range.start - width * 4..];
for ((chunk, tr), t) in current
.chunks_exact_mut(4)
.zip(top_right.chunks_exact(4))
.zip(top.chunks_exact(4))
{
prev = [
chunk[0].wrapping_add(average2(average2(prev[0], tr[0]), t[0])),
chunk[1].wrapping_add(average2(average2(prev[1], tr[1]), t[1])),
chunk[2].wrapping_add(average2(average2(prev[2], tr[2]), t[2])),
chunk[3].wrapping_add(average2(average2(prev[3], tr[3]), t[3])),
];
chunk.copy_from_slice(&prev);
}
}
pub fn apply_predictor_transform_6(image_data: &mut [u8], range: Range<usize>, width: usize) {
let (old, current) = image_data[..range.end].split_at_mut(range.start);
let mut prev: [u8; 4] = old[range.start - 4..][..4].try_into().unwrap();
let top_left = &old[range.start - width * 4 - 4..];
for (chunk, tl) in current.chunks_exact_mut(4).zip(top_left.chunks_exact(4)) {
for i in 0..4 {
chunk[i] = chunk[i].wrapping_add(average2(prev[i], tl[i]));
}
prev.copy_from_slice(chunk);
}
}
pub fn apply_predictor_transform_7(image_data: &mut [u8], range: Range<usize>, width: usize) {
let (old, current) = image_data[..range.end].split_at_mut(range.start);
let mut prev: [u8; 4] = old[range.start - 4..][..4].try_into().unwrap();
let top = &old[range.start - width * 4..][..(range.end - range.start)];
let mut current_chunks = current.chunks_exact_mut(64);
let mut top_chunks = top.chunks_exact(64);
for (current, top) in (&mut current_chunks).zip(&mut top_chunks) {
for (chunk, t) in current.chunks_exact_mut(4).zip(top.chunks_exact(4)) {
prev = [
chunk[0].wrapping_add(average2(prev[0], t[0])),
chunk[1].wrapping_add(average2(prev[1], t[1])),
chunk[2].wrapping_add(average2(prev[2], t[2])),
chunk[3].wrapping_add(average2(prev[3], t[3])),
];
chunk.copy_from_slice(&prev);
}
}
for (chunk, t) in current_chunks
.into_remainder()
.chunks_exact_mut(4)
.zip(top_chunks.remainder().chunks_exact(4))
{
prev = [
chunk[0].wrapping_add(average2(prev[0], t[0])),
chunk[1].wrapping_add(average2(prev[1], t[1])),
chunk[2].wrapping_add(average2(prev[2], t[2])),
chunk[3].wrapping_add(average2(prev[3], t[3])),
];
chunk.copy_from_slice(&prev);
}
}
pub fn apply_predictor_transform_8(image_data: &mut [u8], range: Range<usize>, width: usize) {
for i in range {
image_data[i] = image_data[i].wrapping_add(average2(
image_data[i - width * 4 - 4],
image_data[i - width * 4],
));
}
}
pub fn apply_predictor_transform_9(image_data: &mut [u8], range: Range<usize>, width: usize) {
for i in range {
image_data[i] = image_data[i].wrapping_add(average2(
image_data[i - width * 4],
image_data[i - width * 4 + 4],
));
}
}
pub fn apply_predictor_transform_10(image_data: &mut [u8], range: Range<usize>, width: usize) {
let (old, current) = image_data[..range.end].split_at_mut(range.start);
let mut prev: [u8; 4] = old[range.start - 4..][..4].try_into().unwrap();
let top_left = &old[range.start - width * 4 - 4..];
let top = &old[range.start - width * 4..];
let top_right = &old[range.start - width * 4 + 4..];
for (((chunk, tl), t), tr) in current
.chunks_exact_mut(4)
.zip(top_left.chunks_exact(4))
.zip(top.chunks_exact(4))
.zip(top_right.chunks_exact(4))
{
prev = [
chunk[0].wrapping_add(average2(average2(prev[0], tl[0]), average2(t[0], tr[0]))),
chunk[1].wrapping_add(average2(average2(prev[1], tl[1]), average2(t[1], tr[1]))),
chunk[2].wrapping_add(average2(average2(prev[2], tl[2]), average2(t[2], tr[2]))),
chunk[3].wrapping_add(average2(average2(prev[3], tl[3]), average2(t[3], tr[3]))),
];
chunk.copy_from_slice(&prev);
}
}
pub fn apply_predictor_transform_11(image_data: &mut [u8], range: Range<usize>, width: usize) {
let (old, current) = image_data[..range.end].split_at_mut(range.start);
let top = &old[range.start - width * 4..];
let mut l = [
old[range.start - 4] as i16,
old[range.start - 3] as i16,
old[range.start - 2] as i16,
old[range.start - 1] as i16,
];
let mut tl = [
old[range.start - width * 4 - 4] as i16,
old[range.start - width * 4 - 3] as i16,
old[range.start - width * 4 - 2] as i16,
old[range.start - width * 4 - 1] as i16,
];
for (chunk, top) in current.chunks_exact_mut(4).zip(top.chunks_exact(4)) {
let t = [top[0] as i16, top[1] as i16, top[2] as i16, top[3] as i16];
let mut predict_left = 0;
let mut predict_top = 0;
for i in 0..4 {
let predict = l[i] + t[i] - tl[i];
predict_left += i16::abs(predict - l[i]);
predict_top += i16::abs(predict - t[i]);
}
if predict_left < predict_top {
chunk.copy_from_slice(&[
chunk[0].wrapping_add(l[0] as u8),
chunk[1].wrapping_add(l[1] as u8),
chunk[2].wrapping_add(l[2] as u8),
chunk[3].wrapping_add(l[3] as u8),
]);
} else {
chunk.copy_from_slice(&[
chunk[0].wrapping_add(t[0] as u8),
chunk[1].wrapping_add(t[1] as u8),
chunk[2].wrapping_add(t[2] as u8),
chunk[3].wrapping_add(t[3] as u8),
]);
}
tl = t;
l = [
chunk[0] as i16,
chunk[1] as i16,
chunk[2] as i16,
chunk[3] as i16,
];
}
}
pub fn apply_predictor_transform_12(image_data: &mut [u8], range: Range<usize>, width: usize) {
let (old, current) = image_data[..range.end].split_at_mut(range.start);
let mut prev: [u8; 4] = old[range.start - 4..][..4].try_into().unwrap();
let top_left = &old[range.start - width * 4 - 4..];
let top = &old[range.start - width * 4..];
for ((chunk, tl), t) in current
.chunks_exact_mut(4)
.zip(top_left.chunks_exact(4))
.zip(top.chunks_exact(4))
{
prev = [
chunk[0].wrapping_add(clamp_add_subtract_full(
i16::from(prev[0]),
i16::from(t[0]),
i16::from(tl[0]),
)),
chunk[1].wrapping_add(clamp_add_subtract_full(
i16::from(prev[1]),
i16::from(t[1]),
i16::from(tl[1]),
)),
chunk[2].wrapping_add(clamp_add_subtract_full(
i16::from(prev[2]),
i16::from(t[2]),
i16::from(tl[2]),
)),
chunk[3].wrapping_add(clamp_add_subtract_full(
i16::from(prev[3]),
i16::from(t[3]),
i16::from(tl[3]),
)),
];
chunk.copy_from_slice(&prev);
}
}
pub fn apply_predictor_transform_13(image_data: &mut [u8], range: Range<usize>, width: usize) {
let (old, current) = image_data[..range.end].split_at_mut(range.start);
let mut prev: [u8; 4] = old[range.start - 4..][..4].try_into().unwrap();
let top_left = &old[range.start - width * 4 - 4..][..(range.end - range.start)];
let top = &old[range.start - width * 4..][..(range.end - range.start)];
for ((chunk, tl), t) in current
.chunks_exact_mut(4)
.zip(top_left.chunks_exact(4))
.zip(top.chunks_exact(4))
{
prev = [
chunk[0].wrapping_add(clamp_add_subtract_half(
(i16::from(prev[0]) + i16::from(t[0])) / 2,
i16::from(tl[0]),
)),
chunk[1].wrapping_add(clamp_add_subtract_half(
(i16::from(prev[1]) + i16::from(t[1])) / 2,
i16::from(tl[1]),
)),
chunk[2].wrapping_add(clamp_add_subtract_half(
(i16::from(prev[2]) + i16::from(t[2])) / 2,
i16::from(tl[2]),
)),
chunk[3].wrapping_add(clamp_add_subtract_half(
(i16::from(prev[3]) + i16::from(t[3])) / 2,
i16::from(tl[3]),
)),
];
chunk.copy_from_slice(&prev);
}
}
pub(crate) fn apply_color_transform(
image_data: &mut [u8],
width: u16,
size_bits: u8,
transform_data: &[u8],
) {
let block_xsize = usize::from(subsample_size(width, size_bits));
let width = usize::from(width);
for (y, row) in image_data.chunks_exact_mut(width * 4).enumerate() {
for (block_x, block) in row.chunks_mut(4 << size_bits).enumerate() {
let block_index = (y >> size_bits) * block_xsize + block_x;
let red_to_blue = transform_data[block_index * 4];
let green_to_blue = transform_data[block_index * 4 + 1];
let green_to_red = transform_data[block_index * 4 + 2];
for pixel in block.chunks_exact_mut(4) {
let green = u32::from(pixel[1]);
let mut temp_red = u32::from(pixel[0]);
let mut temp_blue = u32::from(pixel[2]);
temp_red += color_transform_delta(green_to_red as i8, green as i8);
temp_blue += color_transform_delta(green_to_blue as i8, green as i8);
temp_blue += color_transform_delta(red_to_blue as i8, temp_red as i8);
pixel[0] = (temp_red & 0xff) as u8;
pixel[2] = (temp_blue & 0xff) as u8;
}
}
}
}
pub(crate) fn apply_subtract_green_transform(image_data: &mut [u8]) {
for pixel in image_data.chunks_exact_mut(4) {
pixel[0] = pixel[0].wrapping_add(pixel[1]);
pixel[2] = pixel[2].wrapping_add(pixel[1]);
}
}
pub(crate) fn apply_color_indexing_transform(
image_data: &mut [u8],
width: u16,
height: u16,
table_size: u16,
table_data: &[u8],
) {
fn div_ceil(a: u16, b: u16) -> u16 {
let d = a / b;
let r = a % b;
if r > 0 && b > 0 {
d + 1
} else {
d
}
}
if table_size > 16 {
let mut table = table_data.chunks_exact(4).collect::<Vec<_>>();
table.resize(256, &[0; 4]);
for pixel in image_data.chunks_exact_mut(4) {
pixel.copy_from_slice(table[pixel[1] as usize]);
}
} else {
let width_bits: u8 = if table_size <= 2 {
3
} else if table_size <= 4 {
2
} else if table_size <= 16 {
1
} else {
unreachable!()
};
let bits_per_entry = 8 / (1 << width_bits);
let mask = (1 << bits_per_entry) - 1;
let table = (0..256)
.flat_map(|i| {
let mut entry = Vec::new();
for j in 0..(1 << width_bits) {
let k = i >> (j * bits_per_entry) & mask;
if k < table_size {
entry.extend_from_slice(&table_data[usize::from(k) * 4..][..4]);
} else {
entry.extend_from_slice(&[0; 4]);
}
}
entry
})
.collect::<Vec<_>>();
let table = table.chunks_exact(4 << width_bits).collect::<Vec<_>>();
let entry_size = 4 << width_bits;
let index_image_width = div_ceil(width, 1 << width_bits) as usize;
let final_entry_size = width as usize * 4 - entry_size * (index_image_width - 1);
for y in (0..height as usize).rev() {
for x in (0..index_image_width).rev() {
let input_index = y * index_image_width * 4 + x * 4 + 1;
let output_index = y * width as usize * 4 + x * entry_size;
let table_index = image_data[input_index] as usize;
if x == index_image_width - 1 {
image_data[output_index..][..final_entry_size]
.copy_from_slice(&table[table_index][..final_entry_size]);
} else {
image_data[output_index..][..entry_size].copy_from_slice(table[table_index]);
}
}
}
}
}
fn average2(a: u8, b: u8) -> u8 {
((u16::from(a) + u16::from(b)) / 2) as u8
}
fn clamp_add_subtract_full(a: i16, b: i16, c: i16) -> u8 {
#![allow(clippy::manual_clamp)]
(a + b - c).max(0).min(255) as u8
}
fn clamp_add_subtract_half(a: i16, b: i16) -> u8 {
#![allow(clippy::manual_clamp)]
(a + (a - b) / 2).max(0).min(255) as u8
}
fn color_transform_delta(t: i8, c: i8) -> u32 {
(i32::from(t) * i32::from(c)) as u32 >> 5
}
#[cfg(all(test, feature = "_benchmarks"))]
mod benches {
use rand::Rng;
use test::{black_box, Bencher};
fn measure_predictor(b: &mut Bencher, predictor: fn(&mut [u8], std::ops::Range<usize>, usize)) {
let width = 256;
let mut data = vec![0u8; width * 8];
rand::thread_rng().fill(&mut data[..]);
b.bytes = 4 * width as u64 - 4;
b.iter(|| {
predictor(
black_box(&mut data),
black_box(width * 4 + 4..width * 8),
black_box(width),
)
});
}
#[bench]
fn predictor00(b: &mut Bencher) {
measure_predictor(b, super::apply_predictor_transform_0);
}
#[bench]
fn predictor01(b: &mut Bencher) {
measure_predictor(b, super::apply_predictor_transform_1);
}
#[bench]
fn predictor02(b: &mut Bencher) {
measure_predictor(b, super::apply_predictor_transform_2);
}
#[bench]
fn predictor03(b: &mut Bencher) {
measure_predictor(b, super::apply_predictor_transform_3);
}
#[bench]
fn predictor04(b: &mut Bencher) {
measure_predictor(b, super::apply_predictor_transform_4);
}
#[bench]
fn predictor05(b: &mut Bencher) {
measure_predictor(b, super::apply_predictor_transform_5);
}
#[bench]
fn predictor06(b: &mut Bencher) {
measure_predictor(b, super::apply_predictor_transform_6);
}
#[bench]
fn predictor07(b: &mut Bencher) {
measure_predictor(b, super::apply_predictor_transform_7);
}
#[bench]
fn predictor08(b: &mut Bencher) {
measure_predictor(b, super::apply_predictor_transform_8);
}
#[bench]
fn predictor09(b: &mut Bencher) {
measure_predictor(b, super::apply_predictor_transform_9);
}
#[bench]
fn predictor10(b: &mut Bencher) {
measure_predictor(b, super::apply_predictor_transform_10);
}
#[bench]
fn predictor11(b: &mut Bencher) {
measure_predictor(b, super::apply_predictor_transform_11);
}
#[bench]
fn predictor12(b: &mut Bencher) {
measure_predictor(b, super::apply_predictor_transform_12);
}
#[bench]
fn predictor13(b: &mut Bencher) {
measure_predictor(b, super::apply_predictor_transform_13);
}
#[bench]
fn color_transform(b: &mut Bencher) {
let width = 256;
let height = 256;
let size_bits = 3;
let mut data = vec![0u8; width * height * 4];
let mut transform_data = vec![0u8; (width * height * 4) >> (size_bits * 2)];
rand::thread_rng().fill(&mut data[..]);
rand::thread_rng().fill(&mut transform_data[..]);
b.bytes = 4 * width as u64 * height as u64;
b.iter(|| {
super::apply_color_transform(
black_box(&mut data),
black_box(width as u16),
black_box(size_bits),
black_box(&transform_data),
);
});
}
#[bench]
fn subtract_green(b: &mut Bencher) {
let mut data = vec![0u8; 1024 * 4];
rand::thread_rng().fill(&mut data[..]);
b.bytes = data.len() as u64;
b.iter(|| {
super::apply_subtract_green_transform(black_box(&mut data));
});
}
}