// Copyright (c) the JPEG XL Project Authors. All rights reserved. // // Use of this source code is governed by a BSD-style // license that can be found in the LICENSE file. use crate::*; use jxl_simd::{F32SimdVec, SimdDescriptor}; #[inline(always)] fn idct2d_2_2_impl(d: D, data: &mut [f32]) { assert_eq!(data.len(), 4, "Data length mismatch"); const { assert!(2usize.is_multiple_of(D::F32Vec::LEN)) }; const { assert!(2usize.is_multiple_of(D::F32Vec::LEN)) }; let data = D::F32Vec::make_array_slice_mut(data); let chunks = 2 / D::F32Vec::LEN; for i in 0..chunks { do_idct_2(d, &mut data[i..], chunks); } for i in 0..chunks { D::F32Vec::transpose_square(d, &mut data[i * 2 + i..], chunks); for j in i + 1..chunks { D::F32Vec::transpose_square(d, &mut data[j * 2 + i..], chunks); D::F32Vec::transpose_square(d, &mut data[i * 2 + j..], chunks); for k in 0..D::F32Vec::LEN { data.swap(i * 2 + j + k * chunks, j * 2 + i + k * chunks); } } do_idct_2(d, &mut data[i..], chunks); } } #[inline(always)] fn idct2d_4_4_impl(d: D, data: &mut [f32]) { assert_eq!(data.len(), 16, "Data length mismatch"); const { assert!(4usize.is_multiple_of(D::F32Vec::LEN)) }; const { assert!(4usize.is_multiple_of(D::F32Vec::LEN)) }; let data = D::F32Vec::make_array_slice_mut(data); let chunks = 4 / D::F32Vec::LEN; for i in 0..chunks { do_idct_4(d, &mut data[i..], chunks); } for i in 0..chunks { D::F32Vec::transpose_square(d, &mut data[i * 4 + i..], chunks); for j in i + 1..chunks { D::F32Vec::transpose_square(d, &mut data[j * 4 + i..], chunks); D::F32Vec::transpose_square(d, &mut data[i * 4 + j..], chunks); for k in 0..D::F32Vec::LEN { data.swap(i * 4 + j + k * chunks, j * 4 + i + k * chunks); } } do_idct_4(d, &mut data[i..], chunks); } } #[inline(always)] fn idct2d_4_8_impl(d: D, data: &mut [f32]) { assert_eq!(data.len(), 32, "Data length mismatch"); const { assert!(4usize.is_multiple_of(D::F32Vec::LEN)) }; const { assert!(8usize.is_multiple_of(D::F32Vec::LEN)) }; let data = D::F32Vec::make_array_slice_mut(data); let column_chunks = 8 / D::F32Vec::LEN; let row_chunks = 4 / D::F32Vec::LEN; for i in 0..row_chunks { for j in 0..column_chunks { D::F32Vec::transpose_square(d, &mut data[i * 8 + j..], column_chunks); } do_idct_8_rowblock(d, &mut data[i * 8..]); } for i in 0..column_chunks { for j in 0..row_chunks { D::F32Vec::transpose_square(d, &mut data[j * 8 + i..], column_chunks); } do_idct_4(d, &mut data[i..], column_chunks); } } #[inline(always)] fn idct2d_8_4_impl(d: D, data: &mut [f32]) { assert_eq!(data.len(), 32, "Data length mismatch"); const { assert!(8usize.is_multiple_of(D::F32Vec::LEN)) }; const { assert!(4usize.is_multiple_of(D::F32Vec::LEN)) }; let data = D::F32Vec::make_array_slice_mut(data); let column_chunks = 4 / D::F32Vec::LEN; let row_chunks = 8 / D::F32Vec::LEN; for i in 0..row_chunks { do_idct_4(d, &mut data[i..], row_chunks); } for i in 0..column_chunks { let tr_block = |data: &mut [::UnderlyingArray], i, j, l| { D::F32Vec::transpose_square(d, &mut data[i * 8 + j + l * column_chunks..], row_chunks) }; (0..2).for_each(|l| tr_block(data, i, i, l)); for j in i + 1..column_chunks { (0..2).for_each(|l| tr_block(data, i, j, l)); (0..2).for_each(|l| tr_block(data, j, i, l)); for l in 0..2 { for k in 0..D::F32Vec::LEN { data.swap( i * 8 + j + k * row_chunks + l * column_chunks, j * 8 + i + k * row_chunks + l * column_chunks, ); } } } do_idct_8_trh(d, &mut data[i..]); } } #[inline(always)] fn idct2d_8_8_impl(d: D, data: &mut [f32]) { assert_eq!(data.len(), 64, "Data length mismatch"); const { assert!(8usize.is_multiple_of(D::F32Vec::LEN)) }; const { assert!(8usize.is_multiple_of(D::F32Vec::LEN)) }; let data = D::F32Vec::make_array_slice_mut(data); let chunks = 8 / D::F32Vec::LEN; for i in 0..chunks { do_idct_8(d, &mut data[i..], chunks); } for i in 0..chunks { D::F32Vec::transpose_square(d, &mut data[i * 8 + i..], chunks); for j in i + 1..chunks { D::F32Vec::transpose_square(d, &mut data[j * 8 + i..], chunks); D::F32Vec::transpose_square(d, &mut data[i * 8 + j..], chunks); for k in 0..D::F32Vec::LEN { data.swap(i * 8 + j + k * chunks, j * 8 + i + k * chunks); } } do_idct_8(d, &mut data[i..], chunks); } } #[inline(always)] fn idct2d_8_16_impl(d: D, data: &mut [f32]) { assert_eq!(data.len(), 128, "Data length mismatch"); const { assert!(8usize.is_multiple_of(D::F32Vec::LEN)) }; const { assert!(16usize.is_multiple_of(D::F32Vec::LEN)) }; let data = D::F32Vec::make_array_slice_mut(data); let column_chunks = 16 / D::F32Vec::LEN; let row_chunks = 8 / D::F32Vec::LEN; for i in 0..row_chunks { for j in 0..column_chunks { D::F32Vec::transpose_square(d, &mut data[i * 16 + j..], column_chunks); } do_idct_16_rowblock(d, &mut data[i * 16..]); } for i in 0..column_chunks { for j in 0..row_chunks { D::F32Vec::transpose_square(d, &mut data[j * 16 + i..], column_chunks); } do_idct_8(d, &mut data[i..], column_chunks); } } #[inline(always)] fn idct2d_8_32_impl(d: D, data: &mut [f32]) { assert_eq!(data.len(), 256, "Data length mismatch"); const { assert!(8usize.is_multiple_of(D::F32Vec::LEN)) }; const { assert!(32usize.is_multiple_of(D::F32Vec::LEN)) }; let data = D::F32Vec::make_array_slice_mut(data); let column_chunks = 32 / D::F32Vec::LEN; let row_chunks = 8 / D::F32Vec::LEN; for i in 0..row_chunks { for j in 0..column_chunks { D::F32Vec::transpose_square(d, &mut data[i * 32 + j..], column_chunks); } do_idct_32_rowblock(d, &mut data[i * 32..]); } for i in 0..column_chunks { for j in 0..row_chunks { D::F32Vec::transpose_square(d, &mut data[j * 32 + i..], column_chunks); } do_idct_8(d, &mut data[i..], column_chunks); } } #[inline(always)] fn idct2d_16_8_impl(d: D, data: &mut [f32]) { assert_eq!(data.len(), 128, "Data length mismatch"); const { assert!(16usize.is_multiple_of(D::F32Vec::LEN)) }; const { assert!(8usize.is_multiple_of(D::F32Vec::LEN)) }; let data = D::F32Vec::make_array_slice_mut(data); let column_chunks = 8 / D::F32Vec::LEN; let row_chunks = 16 / D::F32Vec::LEN; for i in 0..row_chunks { do_idct_8(d, &mut data[i..], row_chunks); } for i in 0..column_chunks { let tr_block = |data: &mut [::UnderlyingArray], i, j, l| { D::F32Vec::transpose_square(d, &mut data[i * 16 + j + l * column_chunks..], row_chunks) }; (0..2).for_each(|l| tr_block(data, i, i, l)); for j in i + 1..column_chunks { (0..2).for_each(|l| tr_block(data, i, j, l)); (0..2).for_each(|l| tr_block(data, j, i, l)); for l in 0..2 { for k in 0..D::F32Vec::LEN { data.swap( i * 16 + j + k * row_chunks + l * column_chunks, j * 16 + i + k * row_chunks + l * column_chunks, ); } } } do_idct_16_trh(d, &mut data[i..]); } } #[inline(always)] fn idct2d_16_16_impl(d: D, data: &mut [f32]) { assert_eq!(data.len(), 256, "Data length mismatch"); const { assert!(16usize.is_multiple_of(D::F32Vec::LEN)) }; const { assert!(16usize.is_multiple_of(D::F32Vec::LEN)) }; let data = D::F32Vec::make_array_slice_mut(data); let chunks = 16 / D::F32Vec::LEN; for i in 0..chunks { do_idct_16(d, &mut data[i..], chunks); } for i in 0..chunks { D::F32Vec::transpose_square(d, &mut data[i * 16 + i..], chunks); for j in i + 1..chunks { D::F32Vec::transpose_square(d, &mut data[j * 16 + i..], chunks); D::F32Vec::transpose_square(d, &mut data[i * 16 + j..], chunks); for k in 0..D::F32Vec::LEN { data.swap(i * 16 + j + k * chunks, j * 16 + i + k * chunks); } } do_idct_16(d, &mut data[i..], chunks); } } #[inline(always)] fn idct2d_16_32_impl(d: D, data: &mut [f32]) { assert_eq!(data.len(), 512, "Data length mismatch"); const { assert!(16usize.is_multiple_of(D::F32Vec::LEN)) }; const { assert!(32usize.is_multiple_of(D::F32Vec::LEN)) }; let data = D::F32Vec::make_array_slice_mut(data); let column_chunks = 32 / D::F32Vec::LEN; let row_chunks = 16 / D::F32Vec::LEN; for i in 0..row_chunks { for j in 0..column_chunks { D::F32Vec::transpose_square(d, &mut data[i * 32 + j..], column_chunks); } do_idct_32_rowblock(d, &mut data[i * 32..]); } for i in 0..column_chunks { for j in 0..row_chunks { D::F32Vec::transpose_square(d, &mut data[j * 32 + i..], column_chunks); } do_idct_16(d, &mut data[i..], column_chunks); } } #[inline(always)] fn idct2d_32_8_impl(d: D, data: &mut [f32]) { assert_eq!(data.len(), 256, "Data length mismatch"); const { assert!(32usize.is_multiple_of(D::F32Vec::LEN)) }; const { assert!(8usize.is_multiple_of(D::F32Vec::LEN)) }; let data = D::F32Vec::make_array_slice_mut(data); let column_chunks = 8 / D::F32Vec::LEN; let row_chunks = 32 / D::F32Vec::LEN; for i in 0..row_chunks { do_idct_8(d, &mut data[i..], row_chunks); } for i in 0..column_chunks { let tr_block = |data: &mut [::UnderlyingArray], i, j, l| { D::F32Vec::transpose_square(d, &mut data[i * 32 + j + l * column_chunks..], row_chunks) }; (0..4).for_each(|l| tr_block(data, i, i, l)); for j in i + 1..column_chunks { (0..4).for_each(|l| tr_block(data, i, j, l)); (0..4).for_each(|l| tr_block(data, j, i, l)); for l in 0..4 { for k in 0..D::F32Vec::LEN { data.swap( i * 32 + j + k * row_chunks + l * column_chunks, j * 32 + i + k * row_chunks + l * column_chunks, ); } } } do_idct_32_trq(d, &mut data[i..]); } } #[inline(always)] fn idct2d_32_16_impl(d: D, data: &mut [f32]) { assert_eq!(data.len(), 512, "Data length mismatch"); const { assert!(32usize.is_multiple_of(D::F32Vec::LEN)) }; const { assert!(16usize.is_multiple_of(D::F32Vec::LEN)) }; let data = D::F32Vec::make_array_slice_mut(data); let column_chunks = 16 / D::F32Vec::LEN; let row_chunks = 32 / D::F32Vec::LEN; for i in 0..row_chunks { do_idct_16(d, &mut data[i..], row_chunks); } for i in 0..column_chunks { let tr_block = |data: &mut [::UnderlyingArray], i, j, l| { D::F32Vec::transpose_square(d, &mut data[i * 32 + j + l * column_chunks..], row_chunks) }; (0..2).for_each(|l| tr_block(data, i, i, l)); for j in i + 1..column_chunks { (0..2).for_each(|l| tr_block(data, i, j, l)); (0..2).for_each(|l| tr_block(data, j, i, l)); for l in 0..2 { for k in 0..D::F32Vec::LEN { data.swap( i * 32 + j + k * row_chunks + l * column_chunks, j * 32 + i + k * row_chunks + l * column_chunks, ); } } } do_idct_32_trh(d, &mut data[i..]); } } #[inline(always)] fn idct2d_32_32_impl(d: D, data: &mut [f32]) { assert_eq!(data.len(), 1024, "Data length mismatch"); const { assert!(32usize.is_multiple_of(D::F32Vec::LEN)) }; const { assert!(32usize.is_multiple_of(D::F32Vec::LEN)) }; let data = D::F32Vec::make_array_slice_mut(data); let chunks = 32 / D::F32Vec::LEN; for i in 0..chunks { do_idct_32(d, &mut data[i..], chunks); } for i in 0..chunks { D::F32Vec::transpose_square(d, &mut data[i * 32 + i..], chunks); for j in i + 1..chunks { D::F32Vec::transpose_square(d, &mut data[j * 32 + i..], chunks); D::F32Vec::transpose_square(d, &mut data[i * 32 + j..], chunks); for k in 0..D::F32Vec::LEN { data.swap(i * 32 + j + k * chunks, j * 32 + i + k * chunks); } } do_idct_32(d, &mut data[i..], chunks); } } #[inline(always)] #[allow(unused_variables)] pub fn idct2d_2_2(d: D, data: &mut [f32]) { let d = jxl_simd::ScalarDescriptor::new().unwrap(); idct2d_2_2_impl(d, data) } #[inline(always)] #[allow(unused_variables)] pub fn idct2d_4_4(d: D, data: &mut [f32]) { let d = d.maybe_downgrade_128bit(); idct2d_4_4_impl(d, data) } #[inline(always)] #[allow(unused_variables)] pub fn idct2d_4_8(d: D, data: &mut [f32]) { let d = d.maybe_downgrade_128bit(); idct2d_4_8_impl(d, data) } #[inline(always)] #[allow(unused_variables)] pub fn idct2d_8_4(d: D, data: &mut [f32]) { let d = d.maybe_downgrade_128bit(); idct2d_8_4_impl(d, data) } #[inline(always)] #[allow(unused_variables)] pub fn idct2d_8_8(d: D, data: &mut [f32]) { let d = d.maybe_downgrade_256bit(); idct2d_8_8_impl(d, data) } #[inline(always)] #[allow(unused_variables)] pub fn idct2d_8_16(d: D, data: &mut [f32]) { let d = d.maybe_downgrade_256bit(); idct2d_8_16_impl(d, data) } #[inline(always)] #[allow(unused_variables)] pub fn idct2d_8_32(d: D, data: &mut [f32]) { let d = d.maybe_downgrade_256bit(); idct2d_8_32_impl(d, data) } #[inline(always)] #[allow(unused_variables)] pub fn idct2d_16_8(d: D, data: &mut [f32]) { let d = d.maybe_downgrade_256bit(); idct2d_16_8_impl(d, data) } #[inline(always)] #[allow(unused_variables)] pub fn idct2d_16_16(d: D, data: &mut [f32]) { idct2d_16_16_impl(d, data) } #[inline(always)] #[allow(unused_variables)] pub fn idct2d_16_32(d: D, data: &mut [f32]) { idct2d_16_32_impl(d, data) } #[inline(always)] #[allow(unused_variables)] pub fn idct2d_32_8(d: D, data: &mut [f32]) { let d = d.maybe_downgrade_256bit(); idct2d_32_8_impl(d, data) } #[inline(always)] #[allow(unused_variables)] pub fn idct2d_32_16(d: D, data: &mut [f32]) { idct2d_32_16_impl(d, data) } #[inline(always)] #[allow(unused_variables)] pub fn idct2d_32_32(d: D, data: &mut [f32]) { idct2d_32_32_impl(d, data) }