#![allow(
clippy::too_many_arguments,
clippy::int_plus_one,
stable_features,
unused_features
)]
#![cfg_attr(docsrs, feature(doc_cfg))]
#![cfg_attr(
all(feature = "nightly_fcma", target_arch = "aarch64"),
feature(stdarch_neon_fcma)
)]
#![cfg_attr(
all(feature = "avx512", any(target_arch = "x86", target_arch = "x86_64")),
feature(cfg_version)
)]
#![cfg_attr(
all(feature = "avx512", any(target_arch = "x86", target_arch = "x86_64")),
feature(avx512_target_feature)
)]
#![cfg_attr(
all(feature = "avx512", any(target_arch = "x86", target_arch = "x86_64")),
feature(stdarch_x86_avx512)
)]
#![cfg_attr(
all(feature = "avx512", any(target_arch = "x86", target_arch = "x86_64")),
feature(x86_amx_intrinsics)
)]
#![cfg_attr(feature = "nightly_f16", feature(f16))]
#![deny(
clippy::print_stdout,
clippy::print_stderr,
clippy::print_literal,
clippy::print_in_format_impl
)]
#![cfg_attr(
all(feature = "sve", target_arch = "aarch64"),
feature(stdarch_aarch64_sve)
)]
#[cfg(feature = "fft")]
#[cfg_attr(docsrs, doc(cfg(feature = "fft")))]
mod adaptive_blur;
#[cfg(all(target_arch = "x86_64", feature = "avx"))]
mod avx;
mod bilateral;
mod box_filter;
mod channels_configuration;
mod edge_mode;
mod fast_bilateral_filter;
#[cfg(feature = "image")]
#[cfg_attr(docsrs, doc(cfg(feature = "image")))]
mod fast_bilateral_image;
mod fast_divide;
mod fast_gaussian;
#[cfg(feature = "image")]
#[cfg_attr(docsrs, doc(cfg(feature = "image")))]
mod fast_gaussian_image;
#[cfg(feature = "image")]
#[cfg_attr(docsrs, doc(cfg(feature = "image")))]
mod fast_gaussian_image_next;
mod fast_gaussian_next;
mod filter1d;
mod filter2d;
mod gamma_curves;
mod gaussian;
#[cfg(feature = "image")]
#[cfg_attr(docsrs, doc(cfg(feature = "image")))]
mod gaussian_blur_image;
mod image;
mod image_linearization;
mod img_size;
mod laplacian;
mod lens;
mod median_blur;
mod mlaf;
mod motion_blur;
#[cfg(all(target_arch = "aarch64", feature = "neon"))]
mod neon;
mod primitives;
mod safe_math;
mod sobel;
#[cfg(any(target_arch = "x86_64", target_arch = "x86"))]
mod sse;
#[cfg(feature = "image")]
#[cfg_attr(docsrs, doc(cfg(feature = "image")))]
mod stack_blur_image;
mod stackblur;
#[cfg(all(target_arch = "aarch64", feature = "sve"))]
mod sve;
mod threading_policy;
mod to_storage;
mod unsafe_slice;
mod util;
#[cfg(all(target_arch = "wasm32", target_feature = "simd128"))]
mod wasm32;
#[cfg(feature = "fft")]
#[cfg_attr(docsrs, doc(cfg(feature = "fft")))]
pub use adaptive_blur::adaptive_blur;
pub use bilateral::{BilateralBlurParams, bilateral_filter};
pub use box_filter::{
BoxBlurParameters, CLTParameters, box_blur, box_blur_f32, box_blur_u16, gaussian_box_blur,
gaussian_box_blur_f32, gaussian_box_blur_u16, tent_blur, tent_blur_f32, tent_blur_u16,
};
pub use channels_configuration::FastBlurChannels;
pub use edge_mode::{BorderHandle, EdgeMode, EdgeMode2D, Scalar};
pub use fast_bilateral_filter::{
fast_bilateral_filter, fast_bilateral_filter_f32, fast_bilateral_filter_u16,
};
#[cfg(feature = "image")]
#[cfg_attr(docsrs, doc(cfg(feature = "image")))]
pub use fast_bilateral_image::fast_bilateral_filter_image;
#[cfg(feature = "nightly_f16")]
pub use fast_gaussian::fast_gaussian_f16;
pub use fast_gaussian::{fast_gaussian, fast_gaussian_f32, fast_gaussian_u16};
#[cfg(feature = "image")]
#[cfg_attr(docsrs, doc(cfg(feature = "image")))]
pub use fast_gaussian_image::fast_gaussian_blur_image;
#[cfg(feature = "image")]
#[cfg_attr(docsrs, doc(cfg(feature = "image")))]
pub use fast_gaussian_image_next::fast_gaussian_next_blur_image;
#[cfg(feature = "nightly_f16")]
pub use fast_gaussian_next::fast_gaussian_next_f16;
pub use fast_gaussian_next::{fast_gaussian_next, fast_gaussian_next_f32, fast_gaussian_next_u16};
pub use filter1d::{
KernelShape, filter_1d_approx, filter_1d_complex, filter_1d_complex_fixed_point,
filter_1d_exact,
};
#[cfg(feature = "fft")]
#[cfg_attr(docsrs, doc(cfg(feature = "fft")))]
pub use filter2d::{
FftNumber, fft_next_good_size, filter_2d_fft, filter_2d_fft_complex, filter_2d_rgb_fft,
filter_2d_rgb_fft_complex, filter_2d_rgba_fft, filter_2d_rgba_fft_complex,
};
pub use filter2d::{filter_2d, filter_2d_arbitrary, filter_2d_rgb, filter_2d_rgba};
pub use gamma_curves::TransferFunction;
#[cfg(feature = "nightly_f16")]
pub use gaussian::gaussian_blur_f16;
pub use gaussian::{
ConvolutionMode, GaussianBlurParams, IeeeBinaryConvolutionMode, complex_gaussian_kernel,
gaussian_blur, gaussian_blur_f32, gaussian_blur_u16, gaussian_kernel_1d,
gaussian_kernel_1d_f64, sigma_size, sigma_size_d,
};
#[cfg(feature = "image")]
#[cfg_attr(docsrs, doc(cfg(feature = "image")))]
pub use gaussian_blur_image::gaussian_blur_image;
pub use image::{BlurImage, BlurImageMut, BufferStore};
pub use img_size::ImageSize;
pub use laplacian::{laplacian, laplacian_kernel};
pub use lens::lens_kernel;
pub use median_blur::median_blur;
pub use motion_blur::{generate_motion_kernel, motion_blur};
pub use sobel::sobel;
#[cfg(feature = "image")]
#[cfg_attr(docsrs, doc(cfg(feature = "image")))]
pub use stack_blur_image::stack_blur_image;
pub use stackblur::stack_blur::stack_blur;
#[cfg(feature = "nightly_f16")]
pub use stackblur::stack_blur_f16::stack_blur_f16;
pub use stackblur::stack_blur_f32::stack_blur_f32;
pub use stackblur::stack_blur_u16;
pub use threading_policy::ThreadingPolicy;
pub use util::{BlurError, MismatchedSize};
#[derive(Copy, Clone, Default, PartialOrd, PartialEq, Debug)]
pub struct AnisotropicRadius {
pub x_axis: u32,
pub y_axis: u32,
}
impl AnisotropicRadius {
pub fn new(radius: u32) -> AnisotropicRadius {
AnisotropicRadius {
x_axis: radius,
y_axis: radius,
}
}
pub fn create(x: u32, y: u32) -> AnisotropicRadius {
AnisotropicRadius {
x_axis: x,
y_axis: y,
}
}
pub fn clamp(&self, min: u32, max: u32) -> AnisotropicRadius {
AnisotropicRadius {
x_axis: self.x_axis.clamp(min, max),
y_axis: self.y_axis.clamp(min, max),
}
}
pub fn max(&self, max: u32) -> AnisotropicRadius {
AnisotropicRadius {
x_axis: self.x_axis.max(max),
y_axis: self.y_axis.max(max),
}
}
}
#[cfg(test)]
mod tests {
#[cfg(feature = "fft")]
fn gaussian_kernel_9x9(sigma: f32) -> Vec<f32> {
let mut kernel = [[0.0f32; 9]; 9];
let mut sum = 0.0;
let radius = 4;
let two_sigma_sq = 2.0 * sigma * sigma;
let norm = 1.0 / (std::f32::consts::PI * two_sigma_sq);
for y in -radius..=radius {
for x in -radius..=radius {
let value = norm * ((-(x * x + y * y) as f32) / two_sigma_sq).exp();
kernel[(y + radius) as usize][(x + radius) as usize] = value;
sum += value;
}
}
for row in &mut kernel {
for v in row.iter_mut() {
*v /= sum;
}
}
kernel
.iter()
.flat_map(|x| x)
.map(|&x| x)
.collect::<Vec<f32>>()
}
#[cfg(feature = "fft")]
#[test]
fn test_fft_rgb() {
use super::*;
let width: usize = 188;
let height: usize = 188;
let src = vec![126u8; width * height * 3];
let src_image = BlurImage::borrow(
&src,
width as u32,
height as u32,
FastBlurChannels::Channels3,
);
let mut dst = BlurImageMut::default();
let kernel = gaussian_kernel_9x9(3.);
filter_2d_rgb_fft::<u8, f32>(
&src_image,
&mut dst,
&kernel,
KernelShape::new(9, 9),
EdgeMode::Clamp.as_2d(),
Scalar::default(),
ThreadingPolicy::Single,
)
.unwrap();
for (i, &cn) in dst.data.borrow_mut().iter().enumerate() {
let diff = (cn as i32 - 126).abs();
assert!(
diff <= 3,
"Diff expected to be less than 3 but it was {diff} at {i}"
);
}
}
}