fastimageresize

Rust library for fast image resizing with using of SIMD instructions.

Note: This library does not support converting image color spaces. If it is important for you to resize images with a non-linear color space (e.g. sRGB) correctly, then you need to convert it to a linear color space before resizing. Read more about resizing with respect to color space.

CHANGELOG

Supported pixel formats and available optimisations: - U8 - one u8 component per pixel: - native Rust-code without forced SIMD - AVX2 - U8x3 - three u8 components per pixel (e.g. RGB): - native Rust-code without forced SIMD - SSE4.1 (auto-vectorization) - AVX2 - U8x4 - four u8 components per pixel (RGBA, RGBx, CMYK and other): - native Rust-code without forced SIMD - SSE4.1 - AVX2 - I32 - one i32 component per pixel: - native Rust-code without forced SIMD - F32 - one f32 component per pixel: - native Rust-code without forced SIMD

Benchmarks

Environment: - CPU: Intel(R) Core(TM) i7-6700K CPU @ 4.00GHz - RAM: DDR4 3000 MHz - Ubuntu 20.04 (linux 5.11) - Rust 1.57.0 - fastimageresize = "0.6.0" - glassbench = "0.3.1" - rustflags = ["-C", "llvm-args=-x86-branches-within-32B-boundaries"]

Other Rust libraries used to compare of resizing speed: - image = "0.23.14" (https://crates.io/crates/image) - resize = "0.7.2" (https://crates.io/crates/resize)

Resize algorithms: - Nearest - Convolution with Bilinear filter - Convolution with CatmullRom filter - Convolution with Lanczos3 filter

Resize RGB image (U8x3) 4928x3279 => 852x567

Pipeline:

src_image => resize => dst_image

| | Nearest | Bilinear | CatmullRom | Lanczos3 | |------------|:-------:|:--------:|:----------:|:--------:| | image | 92.300 | 180.063 | 257.043 | 342.268 | | resize | 15.429 | 71.589 | 131.447 | 191.048 | | fir rust | 0.481 | 55.347 | 89.512 | 121.270 | | fir sse4.1 | - | 44.841 | 54.630 | 75.284 | | fir avx2 | - | 10.814 | 14.408 | 20.897 |

Resize RGBA image (U8x4) 4928x3279 => 852x567

Pipeline:

src_image => multiply by alpha => resize => divide by alpha => dst_image

| | Nearest | Bilinear | CatmullRom | Lanczos3 | |------------|:-------:|:--------:|:----------:|:--------:| | image | 98.598 | 177.427 | 253.149 | 336.999 | | resize | 17.988 | 79.891 | 149.178 | 218.593 | | fir rust | 11.952 | 63.214 | 89.397 | 118.714 | | fir sse4.1 | 9.169 | 20.664 | 26.442 | 34.326 | | fir avx2 | 7.353 | 15.514 | 18.936 | 24.624 |

Resize grayscale image (U8) 4928x3279 => 852x567

Pipeline:

src_image => resize => dst_image

| | Nearest | Bilinear | CatmullRom | Lanczos3 | |----------|:-------:|:--------:|:----------:|:--------:| | image | 77.209 | 128.704 | 166.905 | 210.503 | | resize | 9.694 | 24.477 | 47.722 | 80.824 | | fir rust | 0.198 | 21.457 | 24.256 | 36.893 | | fir avx2 | - | 9.758 | 7.787 | 12.099 |

Examples

Resize image

```rust use std::io::BufWriter; use std::num::NonZeroU32;

use image::codecs::png::PngEncoder; use image::io::Reader as ImageReader; use image::{ColorType, GenericImageView};

use fastimageresize as fr;

[test]

fn resizeimageexample() { // Read source image from file let img = ImageReader::open("./data/nasa-4928x3279.png") .unwrap() .decode() .unwrap(); let width = NonZeroU32::new(img.width()).unwrap(); let height = NonZeroU32::new(img.height()).unwrap(); let mut srcimage = fr::Image::fromvecu8( width, height, img.torgba8().into_raw(), fr::PixelType::U8x4, ) .unwrap();

// Create MulDiv instance
let alpha_mul_div = fr::MulDiv::default();
// Multiple RGB channels of source image by alpha channel
alpha_mul_div
    .multiply_alpha_inplace(&mut src_image.view_mut())
    .unwrap();

// Create container for data of destination image
let dst_width = NonZeroU32::new(1024).unwrap();
let dst_height = NonZeroU32::new(768).unwrap();
let mut dst_image = fr::Image::new(
  dst_width, 
  dst_height, 
  src_image.pixel_type(),
);

// Get mutable view of destination image data
let mut dst_view = dst_image.view_mut();

// Create Resizer instance and resize source image
// into buffer of destination image
let mut resizer = fr::Resizer::new(
    fr::ResizeAlg::Convolution(fr::FilterType::Lanczos3)
);
resizer.resize(&src_image.view(), &mut dst_view).unwrap();

// Divide RGB channels of destination image by alpha
alpha_mul_div.divide_alpha_inplace(&mut dst_view).unwrap();

// Write destination image as PNG-file
let mut result_buf = BufWriter::new(Vec::new());
PngEncoder::new(&mut result_buf)
    .encode(
        dst_image.buffer(),
        dst_width.get(),
        dst_height.get(),
        ColorType::Rgba8,
    )
    .unwrap();

} ```

Change CPU extensions used by resizer

```rust, ignore use fastimageresize as fr;

fn main() { le mut resizer = fr::Resizer::new( fr::ResizeAlg::Convolution(fr::FilterType::Lanczos3), ); unsafe { resizer.setcpuextensions(fr::CpuExtensions::Sse4_1); } } ```