#jpeg #decoder #jpeg-decoder

no-std zune-jpeg

A fast, correct and safe jpeg decoder

26 releases

0.5.0-rc2 Jul 21, 2024
0.5.0-rc0 Apr 7, 2024
0.4.14 Dec 12, 2024
0.4.11 Dec 2, 2023
0.1.4 Jun 16, 2022

#16 in Images

Download history 136985/week @ 2024-09-18 150248/week @ 2024-09-25 153945/week @ 2024-10-02 146160/week @ 2024-10-09 166437/week @ 2024-10-16 167609/week @ 2024-10-23 176707/week @ 2024-10-30 177724/week @ 2024-11-06 186381/week @ 2024-11-13 189871/week @ 2024-11-20 172552/week @ 2024-11-27 203024/week @ 2024-12-04 228366/week @ 2024-12-11 186927/week @ 2024-12-18 98791/week @ 2024-12-25 169815/week @ 2025-01-01

727,718 downloads per month
Used in 318 crates (8 directly)

MIT OR Apache-2.0 OR Zlib

350KB
6K SLoC

Zune-JPEG

A fast, correct and safe jpeg decoder in pure Rust.

Usage

The library provides a simple-to-use API for jpeg decoding and an ability to add options to influence decoding.

Example

// Import the library
use zune_jpeg::JpegDecoder;
use std::fs::read;

fn main()->Result<(),DecoderErrors> {
    // load some jpeg data
    let data = read("cat.jpg").unwrap();
    // create a decoder
    let mut decoder = JpegDecoder::new(&data);
    // decode the file
    let pixels = decoder.decode()?;
}

The decoder supports more manipulations via DecoderOptions, see additional documentation in the library.

Goals

The implementation aims to have the following goals achieved, in order of importance

  1. Safety - Do not segfault on errors or invalid input. Panics are okay, but should be fixed when reported. unsafe is only used for SIMD intrinsics, and can be turned off entirely both at compile time and at runtime.
  2. Speed - Get the data as quickly as possible, which means
    1. Platform intrinsics code where justifiable
    2. Carefully written platform independent code that allows the compiler to vectorize it.
    3. Regression tests.
    4. Watch the memory usage of the program
  3. Usability - Provide utility functions like different color conversions functions.

Non-Goals

  • Bit identical results with libjpeg/libjpeg-turbo will never be an aim of this library. Jpeg is a lossy format with very few parts specified by the standard (i.e it doesn't give a reference upsampling and color conversion algorithm)

Features

  • A Pretty fast 8*8 integer IDCT.
  • Fast Huffman Decoding
  • Fast color convert functions.
  • Support for extended colorspaces like GrayScale and RGBA
  • Single-threaded decoding.
  • Support for four component JPEGs, and esoteric color schemes like CYMK
  • Support for no_std
  • BGR/BGRA decoding support.

Crate Features

feature on Capabilities
x86 yes Enables x86 specific instructions, specifically avx and sse for accelerated decoding.
std yes Enable linking to the std crate

Note that the x86 features are automatically disabled on platforms that aren't x86 during compile time hence there is no need to disable them explicitly if you are targeting such a platform.

Using in a no_std environment

The crate can be used in a no_std environment with the alloc feature.

But one is required to link to a working allocator for whatever environment the decoder will be running on

Debug vs release

The decoder heavily relies on platform specific intrinsics, namely AVX2 and SSE to gain speed-ups in decoding, but they perform poorly in debug builds. To get reasonable performance even when compiling your program in debug mode, add this to your Cargo.toml:

# `zune-jpeg` package will be always built with optimizations
[profile.dev.package.zune-jpeg]
opt-level = 3

Benchmarks

The library tries to be at fast as libjpeg-turbo while being as safe as possible. Platform specific intrinsics help get speed up intensive operations ensuring we can almost match libjpeg-turbo speeds but speeds are always +- 10 ms of this library.

For more up-to-date benchmarks, see the online repo with benchmarks here

Dependencies