diff --git a/src/filter/mod.rs b/src/filter/mod.rs index d6430ab6..c4473b2e 100644 --- a/src/filter/mod.rs +++ b/src/filter/mod.rs @@ -7,6 +7,7 @@ mod sharpen; pub use self::sharpen::*; use image::{GenericImage, GenericImageView, GrayImage, ImageBuffer, Luma, Pixel, Primitive}; +use itertools::Itertools; use crate::definitions::{Clamp, Image}; use crate::integral_image::{column_running_sum, row_running_sum}; @@ -203,51 +204,115 @@ pub fn box_filter(image: &GrayImage, x_radius: u32, y_radius: u32) -> Image(x: u32, y: u32, kernel: Kernel, f: F, image: &Image

) -> Q +where + P: Pixel, + Q: Pixel, + F: Fn(K) -> Q::Subpixel, + K: num::Num + Copy + From, +{ + let (width, height) = (image.width() as i64, image.height() as i64); + let (k_width, k_height) = (kernel.width as i64, kernel.height as i64); + let (x, y) = (i64::from(x), i64::from(y)); + + let weighted_pixels = (0..kernel.height as i64) + .cartesian_product(0..kernel.width as i64) + .map(|(k_y, k_x)| { + let kernel_weight = *kernel.at(k_x as u32, k_y as u32); + + let window_y = (y + k_y - k_height / 2).clamp(0, height - 1); + let window_x = (x + k_x - k_width / 2).clamp(0, width - 1); + + debug_assert!(image.in_bounds(window_x as u32, window_y as u32)); + + // Safety: we clamped `window_x` and `window_y` to be in bounds. + let window_pixel = unsafe { image.unsafe_get_pixel(window_x as u32, window_y as u32) }; + + //optimisation: remove allocation when `Pixel::map` allows mapping to a different + //type + #[allow(clippy::unnecessary_to_owned)] + let weighted_pixel = window_pixel + .channels() + .to_vec() + .into_iter() + .map(move |c| kernel_weight * K::from(c)); + + weighted_pixel + }); + + let final_channel_sum = weighted_pixels.fold( + //optimisation: do this without allocation when `Pixel` gains a method of constant initialization + vec![K::zero(); Q::CHANNEL_COUNT as usize], + |mut accumulator, weighted_pixel| { + for (i, weighted_subpixel) in weighted_pixel.enumerate() { + accumulator[i] = accumulator[i] + weighted_subpixel; + } + + accumulator + }, + ); + + *Q::from_slice(&final_channel_sum.into_iter().map(f).collect_vec()) +} + /// Returns 2d correlation of an image. Intermediate calculations are performed /// at type K, and the results converted to pixel Q via f. Pads by continuity. -pub fn filter(image: &Image

, kernel: Kernel, mut f: F) -> Image +pub fn filter(image: &Image

, kernel: Kernel, f: F) -> Image where P: Pixel, -

::Subpixel: Into, Q: Pixel, - F: FnMut(&mut Q::Subpixel, K), - K: num::Num + Copy, + F: Fn(K) -> Q::Subpixel, + K: num::Num + Copy + From, { - let (width, height) = image.dimensions(); - let mut out = Image::::new(width, height); - let num_channels = P::CHANNEL_COUNT as usize; - let zero = K::zero(); - let mut acc = vec![zero; num_channels]; + //TODO refactor this to use the `crate::maps` functions once that lands - let (k_width, k_height) = (kernel.width, kernel.height); + let (width, height) = image.dimensions(); - let (width, height, k_width, k_height) = - (width as i64, height as i64, k_width as i64, k_height as i64); + let mut out = Image::::new(width, height); for y in 0..height { for x in 0..width { - for k_y in 0..k_height { - let y_p = min(height - 1, max(0, y + k_y - k_height / 2)) as u32; - for k_x in 0..k_width { - let x_p = min(width - 1, max(0, x + k_x - k_width / 2)) as u32; - accumulate( - &mut acc, - unsafe { &image.unsafe_get_pixel(x_p, y_p) }, - *kernel.at(k_x as u32, k_y as u32), - ); - } - } - let out_channels = out.get_pixel_mut(x as u32, y as u32).channels_mut(); - for (a, c) in acc.iter_mut().zip(out_channels.iter_mut()) { - f(c, *a); - *a = zero; - } + out.put_pixel(x, y, filter_pixel(x, y, kernel, &f, image)); } } out } +#[cfg(feature = "rayon")] +#[cfg_attr(docsrs, doc(cfg(feature = "rayon")))] +/// Returns 2d correlation of an image. Intermediate calculations are performed +/// at type K, and the results converted to pixel Q via f. Pads by continuity. +/// This version uses rayon to parallelize the computation. +pub fn filter_parallel(image: &Image

, kernel: Kernel, f: F) -> Image +where + P: Pixel + Sync, + Q: Pixel + Send + Sync, + P::Subpixel: Sync, + Q::Subpixel: Send + Sync, + F: Fn(K) -> Q::Subpixel + Send + Sync, + K: Num + Copy + From + Sync, +{ + //TODO refactor this to use the `crate::maps` functions once that lands + + use rayon::iter::IndexedParallelIterator; + use rayon::iter::ParallelIterator; + + let (width, height) = image.dimensions(); + + let mut out: Image = Image::new(width, height); + + image + .par_enumerate_pixels() + .zip_eq(out.par_pixels_mut()) + .for_each(move |((x, y, _), output_pixel)| { + *output_pixel = filter_pixel(x, y, kernel, &f, image); + }); + + out +} + #[inline] fn gaussian(x: f32, r: f32) -> f32 { ((2.0 * f32::consts::PI).sqrt() * r).recip() * (-x.powi(2) / (2.0 * r.powi(2))).exp() @@ -321,14 +386,38 @@ where /// Returns 2d correlation of an image with a row-major kernel. Intermediate calculations are /// performed at type K, and the results clamped to subpixel type S. Pads by continuity. +/// +/// A parallelized version of this function exists with [`filter_clamped_parallel`] when +/// the crate `rayon` feature is enabled. pub fn filter_clamped(image: &Image

, kernel: Kernel) -> Image> where P::Subpixel: Into, S: Clamp + Primitive, P: WithChannel, - K: Num + Copy, + K: Num + Copy + From<

::Subpixel>, { - filter(image, kernel, |channel, acc| *channel = S::clamp(acc)) + filter(image, kernel, S::clamp) +} + +/// Returns 2d correlation of an image with a 3x3 row-major kernel. Intermediate calculations are +/// performed at type K, and the results clamped to subpixel type S. Pads by continuity. +/// This version uses rayon to parallelize the computation. +#[must_use = "the function does not modify the original image"] +#[cfg(feature = "rayon")] +#[cfg_attr(docsrs, doc(cfg(feature = "rayon")))] +pub fn filter_clamped_parallel( + image: &Image

, + kernel: Kernel, +) -> Image> +where + P: Sync, + P::Subpixel: Send + Sync, +

>::Pixel: Send + Sync, + S: Clamp + Primitive + Send + Sync, + P: WithChannel, + K: Num + Copy + Send + Sync + From, +{ + filter_parallel(image, kernel, |x| S::clamp(x)) } /// Returns horizontal correlations between an image and a 1d kernel. @@ -560,6 +649,22 @@ pub fn laplacian_filter(image: &GrayImage) -> Image> { filter_clamped(image, kernel::FOUR_LAPLACIAN_3X3) } +/// Calculates the Laplacian of an image. +/// This version uses rayon to parallelize the computation. +/// +/// The Laplacian is computed by filtering the image using the following 3x3 kernel: +/// ```notrust +/// 0, 1, 0, +/// 1, -4, 1, +/// 0, 1, 0 +/// ``` +#[must_use = "the function does not modify the original image"] +#[cfg(feature = "rayon")] +#[cfg_attr(docsrs, doc(cfg(feature = "rayon")))] +pub fn laplacian_filter_parallel(image: &GrayImage) -> Image> { + filter_clamped_parallel(image, kernel::FOUR_LAPLACIAN_3X3) +} + #[cfg(test)] mod tests { use super::*; @@ -710,7 +815,7 @@ mod tests { #[test] fn $test_name() { // I think the interesting edge cases here are determined entirely - // by the relative sizes of the kernel and the image side length, so + // by the relative sizes of the kernel and the image side length, soz // I'm just enumerating over small values instead of generating random // examples via quickcheck. for height in 0..5 { @@ -823,6 +928,31 @@ mod tests { assert_pixels_eq!(filtered, expected); } + #[test] + #[cfg(feature = "rayon")] + fn test_filter_clamped_parallel_with_results_outside_input_channel_range() { + #[rustfmt::skip] + let kernel: Kernel = Kernel::new(&[ + -1, 0, 1, + -2, 0, 2, + -1, 0, 1 + ], 3, 3); + + let image = gray_image!( + 3, 2, 1; + 6, 5, 4; + 9, 8, 7); + + let expected = gray_image!(type: i16, + -4, -8, -4; + -4, -8, -4; + -4, -8, -4 + ); + + let filtered = filter_clamped_parallel(&image, kernel); + assert_pixels_eq!(filtered, expected); + } + #[test] #[should_panic] fn test_kernel_must_be_nonempty() { @@ -838,7 +968,7 @@ mod tests { let k = &[1u8, 2u8]; let kernel = Kernel::new(k, 2, 1); - let filtered = filter(&image, kernel, |c, a| *c = a); + let filtered = filter(&image, kernel, |x| x); let expected = gray_image!( 9, 7; @@ -853,7 +983,7 @@ mod tests { let k = &[2u8]; let kernel = Kernel::new(k, 1, 1); - let filtered = filter(&image, kernel, |c, a| *c = a); + let filtered = filter(&image, kernel, |x| x); let expected = gray_image!(); assert_pixels_eq!(filtered, expected); @@ -872,8 +1002,7 @@ mod tests { 0.1, 0.2, 0.1 ]; let kernel = Kernel::new(k, 3, 3); - let filtered: Image> = - filter(&image, kernel, |c, a| *c = >::clamp(a)); + let filtered: Image> = filter(&image, kernel, >::clamp); let expected = gray_image!( 11, 7; @@ -962,8 +1091,8 @@ mod proptests { ker in arbitrary_image::>(1..20, 1..20), ) { let kernel = Kernel::new(&ker, ker.width(), ker.height()); - let out: Image> = filter(&img, kernel, |dst, src| { - *dst = src; + let out: Image> = filter(&img, kernel, |x| { + x }); assert_eq!(out.dimensions(), img.dimensions()); } @@ -1072,6 +1201,23 @@ mod benches { }); } + #[bench] + fn bench_filter_clamped_parallel_i32_filter(b: &mut Bencher) { + let image = gray_bench_image(500, 500); + #[rustfmt::skip] + let kernel: Kernel = Kernel::new(&[ + -1, 0, 1, + -2, 0, 2, + -1, 0, 1 + ], 3, 3); + + b.iter(|| { + let filtered: ImageBuffer, Vec> = + filter_clamped_parallel::<_, _, i16>(&image, kernel); + black_box(filtered); + }); + } + /// Baseline implementation of Gaussian blur is that provided by image::imageops. /// We can also use this to validate correctness of any implementations we add here. fn gaussian_baseline_rgb(image: &I, stdev: f32) -> Image> diff --git a/src/filter/sharpen.rs b/src/filter/sharpen.rs index 1f7107af..da771a28 100644 --- a/src/filter/sharpen.rs +++ b/src/filter/sharpen.rs @@ -1,3 +1,5 @@ +#[cfg(feature = "rayon")] +use super::filter_clamped_parallel; use super::{filter_clamped, gaussian_blur_f32}; use crate::{ definitions::{Clamp, Image}, @@ -13,6 +15,16 @@ pub fn sharpen3x3(image: &GrayImage) -> GrayImage { filter_clamped(image, identity_minus_laplacian) } +/// Sharpens a grayscale image by applying a 3x3 approximation to the Laplacian. +/// This version uses rayon to parallelize the computation. +#[must_use = "the function does not modify the original image"] +#[cfg(feature = "rayon")] +#[cfg_attr(docsrs, doc(cfg(feature = "rayon")))] +pub fn sharpen3x3_parallel(image: &GrayImage) -> GrayImage { + let identity_minus_laplacian = Kernel::new(&[0, -1, 0, -1, 5, -1, 0, -1, 0], 3, 3); + filter_clamped_parallel(image, identity_minus_laplacian) +} + /// Sharpens a grayscale image using a Gaussian as a low-pass filter. /// /// * `sigma` is the standard deviation of the Gaussian filter used. diff --git a/src/gradients.rs b/src/gradients.rs index 8a99ef2a..db18cce2 100644 --- a/src/gradients.rs +++ b/src/gradients.rs @@ -105,12 +105,8 @@ where ChannelMap: HasBlack, F: Fn(ChannelMap) -> Q, { - let pass1: Image> = filter(image, kernel1, |channel, acc| { - *channel = >::clamp(acc) - }); - let pass2: Image> = filter(image, kernel2, |channel, acc| { - *channel = >::clamp(acc) - }); + let pass1: Image> = filter(image, kernel1, >::clamp); + let pass2: Image> = filter(image, kernel2, >::clamp); let (width, height) = image.dimensions(); let mut out = Image::::new(width, height); @@ -330,6 +326,8 @@ where #[cfg(test)] mod tests { use crate::filter::filter_clamped; + #[cfg(feature = "rayon")] + use crate::filter::filter_clamped_parallel; use super::*; use image::{ImageBuffer, Luma}; @@ -411,6 +409,23 @@ mod tests { assert_pixels_eq!(filtered, expected); } + #[test] + #[cfg(feature = "rayon")] + fn test_horizontal_sobel_gradient_image_parallel() { + let image = gray_image!( + 3, 2, 1; + 6, 5, 4; + 9, 8, 7); + + let expected = gray_image!(type: i16, + -4, -8, -4; + -4, -8, -4; + -4, -8, -4); + + let filtered = filter_clamped_parallel(&image, kernel::SOBEL_HORIZONTAL_3X3); + assert_pixels_eq!(filtered, expected); + } + #[test] fn test_vertical_sobel_gradient_image() { let image = gray_image!( @@ -427,6 +442,23 @@ mod tests { assert_pixels_eq!(filtered, expected); } + #[test] + #[cfg(feature = "rayon")] + fn test_vertical_sobel_gradient_image_parallel() { + let image = gray_image!( + 3, 6, 9; + 2, 5, 8; + 1, 4, 7); + + let expected = gray_image!(type: i16, + -4, -4, -4; + -8, -8, -8; + -4, -4, -4); + + let filtered = filter_clamped_parallel(&image, kernel::SOBEL_VERTICAL_3X3); + assert_pixels_eq!(filtered, expected); + } + #[test] fn test_horizontal_scharr_gradient_image() { let image = gray_image!( @@ -443,6 +475,23 @@ mod tests { assert_pixels_eq!(filtered, expected); } + #[test] + #[cfg(feature = "rayon")] + fn test_horizontal_scharr_gradient_image_parallel() { + let image = gray_image!( + 3, 2, 1; + 6, 5, 4; + 9, 8, 7); + + let expected = gray_image!(type: i16, + -16, -32, -16; + -16, -32, -16; + -16, -32, -16); + + let filtered = filter_clamped_parallel(&image, kernel::SCHARR_HORIZONTAL_3X3); + assert_pixels_eq!(filtered, expected); + } + #[test] fn test_vertical_scharr_gradient_image() { let image = gray_image!( @@ -459,6 +508,23 @@ mod tests { assert_pixels_eq!(filtered, expected); } + #[test] + #[cfg(feature = "rayon")] + fn test_vertical_scharr_gradient_image_parallel() { + let image = gray_image!( + 3, 6, 9; + 2, 5, 8; + 1, 4, 7); + + let expected = gray_image!(type: i16, + -16, -16, -16; + -32, -32, -32; + -16, -16, -16); + + let filtered = filter_clamped_parallel(&image, kernel::SCHARR_VERTICAL_3X3); + assert_pixels_eq!(filtered, expected); + } + #[test] fn test_horizontal_prewitt_gradient_image() { let image = gray_image!( @@ -476,6 +542,24 @@ mod tests { } #[test] + #[cfg(feature = "rayon")] + fn test_horizontal_prewitt_gradient_image_parallel() { + let image = gray_image!( + 3, 2, 1; + 6, 5, 4; + 9, 8, 7); + + let expected = gray_image!(type: i16, + -3, -6, -3; + -3, -6, -3; + -3, -6, -3); + + let filtered = filter_clamped_parallel(&image, kernel::PREWITT_HORIZONTAL_3X3); + assert_pixels_eq!(filtered, expected); + } + + #[test] + #[cfg(feature = "rayon")] fn test_vertical_prewitt_gradient_image() { let image = gray_image!( 3, 6, 9; @@ -491,6 +575,23 @@ mod tests { assert_pixels_eq!(filtered, expected); } + #[test] + #[cfg(feature = "rayon")] + fn test_vertical_prewitt_gradient_image_parallel() { + let image = gray_image!( + 3, 6, 9; + 2, 5, 8; + 1, 4, 7); + + let expected = gray_image!(type: i16, + -3, -3, -3; + -6, -6, -6; + -3, -3, -3); + + let filtered = filter_clamped_parallel(&image, kernel::PREWITT_VERTICAL_3X3); + assert_pixels_eq!(filtered, expected); + } + #[test] fn test_horizontal_roberts_gradient_image() { let image = gray_image!( @@ -506,6 +607,24 @@ mod tests { let filtered = filter_clamped(&image, kernel::ROBERTS_HORIZONTAL_2X2); assert_pixels_eq!(filtered, expected); } + + #[test] + #[cfg(feature = "rayon")] + fn test_horizontal_roberts_gradient_image_parallel() { + let image = gray_image!( + 3, 6, 9; + 2, 5, 8; + 1, 4, 7); + + let expected = gray_image!(type: i16, + 0, -3, -3; + 1, -2, -2; + 1, -2, -2); + + let filtered = filter_clamped_parallel(&image, kernel::ROBERTS_HORIZONTAL_2X2); + assert_pixels_eq!(filtered, expected); + } + #[test] fn test_vertical_roberts_gradient_image() { let image = gray_image!( @@ -521,6 +640,23 @@ mod tests { let filtered = filter_clamped(&image, kernel::ROBERTS_VERTICAL_2X2); assert_pixels_eq!(filtered, expected); } + + #[test] + #[cfg(feature = "rayon")] + fn test_vertical_roberts_gradient_image_parallel() { + let image = gray_image!( + 3, 6, 9; + 2, 5, 8; + 1, 4, 7); + + let expected = gray_image!(type: i16, + 0, 3, 3; + 1, 4, 4; + 1, 4, 4); + + let filtered = filter_clamped_parallel(&image, kernel::ROBERTS_VERTICAL_2X2); + assert_pixels_eq!(filtered, expected); + } } #[cfg(not(miri))]