From b61504dae063b1a30ea5fc4c57a6f4152e80a3e7 Mon Sep 17 00:00:00 2001 From: "Sergey \"Shnatsel\" Davidoff" Date: Fri, 10 Mar 2023 18:43:57 +0100 Subject: [PATCH 01/18] Replace jpeg_decoder with zune-jpeg --- Cargo.toml | 2 + src/codecs/jpeg/decoder.rs | 1280 +++--------------------------------- 2 files changed, 85 insertions(+), 1197 deletions(-) diff --git a/Cargo.toml b/Cargo.toml index 5858e9d6ea..0dec884fcf 100644 --- a/Cargo.toml +++ b/Cargo.toml @@ -47,6 +47,8 @@ color_quant = "1.1" exr = { version = "1.5.0", optional = true } qoi = { version = "0.4", optional = true } libwebp = { package = "webp", version = "0.2.2", default-features = false, optional = true } +zune-core = { git = "https://github.com/etemesi254/zune-image" } +zune-jpeg = { git = "https://github.com/etemesi254/zune-image" } [dev-dependencies] crc32fast = "1.2.0" diff --git a/src/codecs/jpeg/decoder.rs b/src/codecs/jpeg/decoder.rs index e60b56e9ab..e0b7af2224 100644 --- a/src/codecs/jpeg/decoder.rs +++ b/src/codecs/jpeg/decoder.rs @@ -1,36 +1,47 @@ -use std::convert::TryFrom; use std::io::{self, Cursor, Read}; use std::marker::PhantomData; use std::mem; use crate::color::ColorType; use crate::error::{ - DecodingError, ImageError, ImageResult, UnsupportedError, UnsupportedErrorKind, + DecodingError, ImageError, ImageResult, UnsupportedError, UnsupportedErrorKind, LimitError, }; use crate::image::{ImageDecoder, ImageFormat}; +type ZuneColorSpace = zune_core::colorspace::ColorSpace; + /// JPEG decoder pub struct JpegDecoder { - decoder: jpeg::Decoder, - metadata: jpeg::ImageInfo, + input: Vec, + orig_color_space: ZuneColorSpace, + width: u16, + height: u16, + // For API compatibility with the previous jpeg_decoder wrapper. + // Can be removed later, which would be an API break. + phantom: PhantomData, } impl JpegDecoder { /// Create a new decoder that decodes from the stream ```r``` pub fn new(r: R) -> ImageResult> { - let mut decoder = jpeg::Decoder::new(r); - - decoder.read_info().map_err(ImageError::from_jpeg)?; - let mut metadata = decoder.info().ok_or_else(|| { - ImageError::Decoding(DecodingError::from_format_hint(ImageFormat::Jpeg.into())) - })?; - - // We convert CMYK data to RGB before returning it to the user. - if metadata.pixel_format == jpeg::PixelFormat::CMYK32 { - metadata.pixel_format = jpeg::PixelFormat::RGB24; - } - - Ok(JpegDecoder { decoder, metadata }) + let mut input = Vec::new(); + let mut r = r; + r.read_to_end(&mut input)?; + let mut decoder = zune_jpeg::JpegDecoder::new(&input); + decoder + .decode_headers() + .map_err(ImageError::from_jpeg)?; + // now that we've decoded the headers we can `.unwrap()` + // all these functions that only fail if called before decoding the headers + let (width, height) = decoder.dimensions().unwrap(); + let orig_color_space = decoder.get_output_colorspace().unwrap(); + Ok(JpegDecoder { + input, + orig_color_space, + width, + height, + phantom: PhantomData, + }) } /// Configure the decoder to scale the image during decoding. @@ -51,15 +62,7 @@ impl JpegDecoder { requested_width: u16, requested_height: u16, ) -> ImageResult<(u16, u16)> { - let result = self - .decoder - .scale(requested_width, requested_height) - .map_err(ImageError::from_jpeg)?; - - self.metadata.width = result.0; - self.metadata.height = result.1; - - Ok(result) + todo!(); // TODO: implement using subsampling } } @@ -83,1207 +86,90 @@ impl<'a, R: 'a + Read> ImageDecoder<'a> for JpegDecoder { type Reader = JpegReader; fn dimensions(&self) -> (u32, u32) { - ( - u32::from(self.metadata.width), - u32::from(self.metadata.height), - ) + (u32::from(self.width), u32::from(self.height)) } fn color_type(&self) -> ColorType { - ColorType::from_jpeg(self.metadata.pixel_format) + ColorType::from_jpeg(self.orig_color_space) } fn icc_profile(&mut self) -> Option> { - self.decoder.icc_profile().clone() + None // zune-jpeg doesn't support it } - fn into_reader(mut self) -> ImageResult { - let mut data = self.decoder.decode().map_err(ImageError::from_jpeg)?; - data = match self.decoder.info().unwrap().pixel_format { - jpeg::PixelFormat::CMYK32 => cmyk_to_rgb(&data), - _ => data, - }; - + fn into_reader(self) -> ImageResult { + let mut decoder = new_zune_decoder(&self.input, self.orig_color_space); + let data = decoder.decode().map_err(ImageError::from_jpeg)?; Ok(JpegReader(Cursor::new(data), PhantomData)) } - fn read_image(mut self, buf: &mut [u8]) -> ImageResult<()> { - assert_eq!(u64::try_from(buf.len()), Ok(self.total_bytes())); + fn read_image(self, buf: &mut [u8]) -> ImageResult<()> { + let advertised_len = self.total_bytes(); + let actual_len = buf.len() as u64; - let mut data = self.decoder.decode().map_err(ImageError::from_jpeg)?; - data = match self.decoder.info().unwrap().pixel_format { - jpeg::PixelFormat::CMYK32 => cmyk_to_rgb(&data), - _ => data, - }; + if actual_len != advertised_len { + return Err(ImageError::Decoding(DecodingError::new( + ImageFormat::Jpeg.into(), + format!( + "Length of the decoded data {actual_len}\ + doesn't match the advertised dimensions of the image\ + that imply length {advertised_len}" + ), + ))); + } + let mut decoder = new_zune_decoder(&self.input, self.orig_color_space); + let data = decoder.decode().map_err(ImageError::from_jpeg)?; buf.copy_from_slice(&data); Ok(()) } } -fn cmyk_to_rgb(input: &[u8]) -> Vec { - let count = input.len() / 4; - let mut output = vec![0; 3 * count]; - - let in_pixels = input[..4 * count].chunks_exact(4); - let out_pixels = output[..3 * count].chunks_exact_mut(3); - - for (pixel, outp) in in_pixels.zip(out_pixels) { - let c = 255 - u16::from(pixel[0]); - let m = 255 - u16::from(pixel[1]); - let y = 255 - u16::from(pixel[2]); - let k = 255 - u16::from(pixel[3]); - // CMY -> RGB - let r = (k * c) / 255; - let g = (k * m) / 255; - let b = (k * y) / 255; - - outp[0] = r as u8; - outp[1] = g as u8; - outp[2] = b as u8; +impl ColorType { + fn from_jpeg(colorspace: ZuneColorSpace) -> ColorType { + let colorspace = to_supported_color_space(colorspace); + use zune_core::colorspace::ColorSpace::*; + match colorspace { + // TODO: are these always 8-bit? I've asked at + // https://github.com/etemesi254/zune-image/discussions/99 + RGB => ColorType::Rgb8, + RGBA => ColorType::Rgba8, + Luma => ColorType::L8, + LumaA => ColorType::La8, + // to_supported_color_space() doesn't return any of the other variants + _ => unreachable!(), + } } - - output } -impl ColorType { - fn from_jpeg(pixel_format: jpeg::PixelFormat) -> ColorType { - use jpeg::PixelFormat::*; - match pixel_format { - L8 => ColorType::L8, - L16 => ColorType::L16, - RGB24 => ColorType::Rgb8, - CMYK32 => panic!(), - } +fn to_supported_color_space(orig: ZuneColorSpace) -> ZuneColorSpace { + use zune_core::colorspace::ColorSpace::*; + match orig { + RGB | RGBA | Luma | LumaA => orig, + // the rest is not supported by `image` so it will be converted to RGB during decoding + _ => RGB, } } +fn new_zune_decoder(input: &[u8], orig_color_space: ZuneColorSpace) -> zune_jpeg::JpegDecoder { + let target_color_space = to_supported_color_space(orig_color_space); + let options = + zune_core::options::DecoderOptions::default().jpeg_set_out_colorspace(target_color_space); + zune_jpeg::JpegDecoder::new_with_options(options, &input) +} + impl ImageError { - fn from_jpeg(err: jpeg::Error) -> ImageError { - use jpeg::Error::*; + fn from_jpeg(err: zune_jpeg::errors::DecodeErrors) -> ImageError { + use zune_jpeg::errors::DecodeErrors::*; match err { - err @ Format(_) => { - ImageError::Decoding(DecodingError::new(ImageFormat::Jpeg.into(), err)) - } Unsupported(desc) => ImageError::Unsupported(UnsupportedError::from_format_and_kind( ImageFormat::Jpeg.into(), UnsupportedErrorKind::GenericFeature(format!("{:?}", desc)), )), - Io(err) => ImageError::IoError(err), - Internal(err) => { - ImageError::Decoding(DecodingError::new(ImageFormat::Jpeg.into(), err)) - } - } - } -} - -#[cfg(test)] -mod tests { - #[cfg(feature = "benchmarks")] - extern crate test; - - use super::cmyk_to_rgb; - #[cfg(feature = "benchmarks")] - use test::Bencher; - - #[cfg(feature = "benchmarks")] - const W: usize = 256; - #[cfg(feature = "benchmarks")] - const H: usize = 256; - - #[test] - fn cmyk_to_rgb_correct() { - for c in 0..=255 { - for k in 0..=255 { - // Based on R = 255 * (1-C/255) * (1-K/255) - let r = (255.0 - f32::from(c)) * (255.0 - f32::from(k)) / 255.0; - let r_u8 = r as u8; - let convert_r = cmyk_to_rgb(&[c, 0, 0, k])[0]; - let convert_g = cmyk_to_rgb(&[0, c, 0, k])[1]; - let convert_b = cmyk_to_rgb(&[0, 0, c, k])[2]; - - assert_eq!( - convert_r, r_u8, - "c = {}, k = {}, cymk_to_rgb[0] = {}, should be {}", - c, k, convert_r, r_u8 - ); - assert_eq!( - convert_g, r_u8, - "m = {}, k = {}, cymk_to_rgb[1] = {}, should be {}", - c, k, convert_g, r_u8 - ); - assert_eq!( - convert_b, r_u8, - "y = {}, k = {}, cymk_to_rgb[2] = {}, should be {}", - c, k, convert_b, r_u8 - ); - } - } - } - - fn single_pix_correct(cmyk_pix: [u8; 4], rgb_pix_true: [u8; 3]) { - let rgb_pix = cmyk_to_rgb(&cmyk_pix); - assert_eq!( - rgb_pix_true[0], rgb_pix[0], - "With CMYK {:?} expected {:?}, got {:?}", - cmyk_pix, rgb_pix_true, rgb_pix - ); - assert_eq!( - rgb_pix_true[1], rgb_pix[1], - "With CMYK {:?} expected {:?}, got {:?}", - cmyk_pix, rgb_pix_true, rgb_pix - ); - assert_eq!( - rgb_pix_true[2], rgb_pix[2], - "With CMYK {:?} expected {:?}, got {:?}", - cmyk_pix, rgb_pix_true, rgb_pix - ); - } - - #[test] - fn test_assorted_colors() { - let cmyk_pixels = vec![ - [0, 51, 102, 65], - [153, 204, 0, 65], - [0, 0, 0, 67], - [0, 85, 170, 69], - [0, 0, 0, 71], - [0, 0, 0, 73], - [0, 17, 34, 75], - [51, 68, 85, 75], - [102, 119, 136, 75], - [153, 170, 187, 75], - [204, 221, 238, 75], - [0, 0, 0, 77], - [0, 0, 0, 79], - [0, 85, 170, 81], - [0, 0, 0, 83], - [0, 3, 6, 85], - [9, 12, 15, 85], - [18, 21, 24, 85], - [27, 30, 33, 85], - [36, 39, 42, 85], - [45, 48, 51, 85], - [54, 57, 60, 85], - [63, 66, 69, 85], - [72, 75, 78, 85], - [81, 84, 87, 85], - [90, 93, 96, 85], - [99, 102, 105, 85], - [108, 111, 114, 85], - [117, 120, 123, 85], - [126, 129, 132, 85], - [135, 138, 141, 85], - [144, 147, 150, 85], - [153, 156, 159, 85], - [162, 165, 168, 85], - [171, 174, 177, 85], - [180, 183, 186, 85], - [189, 192, 195, 85], - [198, 201, 204, 85], - [207, 210, 213, 85], - [216, 219, 222, 85], - [225, 228, 231, 85], - [234, 237, 240, 85], - [243, 246, 249, 85], - [252, 0, 0, 85], - [0, 85, 170, 87], - [0, 0, 0, 89], - [0, 0, 0, 91], - [0, 85, 170, 93], - [0, 51, 102, 95], - [153, 204, 0, 95], - [0, 0, 0, 97], - [0, 85, 170, 99], - [0, 0, 0, 101], - [0, 0, 0, 103], - [0, 17, 34, 105], - [51, 68, 85, 105], - [102, 119, 136, 105], - [153, 170, 187, 105], - [204, 221, 238, 105], - [0, 0, 0, 107], - [0, 0, 0, 109], - [0, 85, 170, 111], - [0, 0, 0, 113], - [0, 51, 102, 115], - [153, 204, 0, 115], - [0, 85, 170, 117], - [0, 15, 30, 119], - [45, 60, 75, 119], - [90, 105, 120, 119], - [135, 150, 165, 119], - [180, 195, 210, 119], - [225, 240, 0, 119], - [0, 0, 0, 121], - [0, 85, 170, 123], - [0, 51, 102, 125], - [153, 204, 0, 125], - [0, 0, 0, 127], - [0, 0, 0, 128], - [0, 85, 170, 129], - [0, 51, 102, 130], - [153, 204, 0, 130], - [0, 0, 0, 131], - [0, 85, 170, 132], - [0, 0, 0, 133], - [0, 0, 0, 134], - [0, 17, 34, 135], - [51, 68, 85, 135], - [102, 119, 136, 135], - [153, 170, 187, 135], - [204, 221, 238, 135], - [0, 15, 30, 136], - [45, 60, 75, 136], - [90, 105, 120, 136], - [135, 150, 165, 136], - [180, 195, 210, 136], - [225, 240, 0, 136], - [0, 0, 0, 137], - [0, 85, 170, 138], - [0, 0, 0, 139], - [0, 51, 102, 140], - [153, 204, 0, 140], - [0, 85, 170, 141], - [0, 0, 0, 142], - [0, 0, 0, 143], - [0, 85, 170, 144], - [0, 51, 102, 145], - [153, 204, 0, 145], - [0, 0, 0, 146], - [0, 85, 170, 147], - [0, 0, 0, 148], - [0, 0, 0, 149], - [0, 17, 34, 150], - [51, 68, 85, 150], - [102, 119, 136, 150], - [153, 170, 187, 150], - [204, 221, 238, 150], - [0, 0, 0, 151], - [0, 0, 0, 152], - [0, 5, 10, 153], - [15, 20, 25, 153], - [30, 35, 40, 153], - [45, 50, 55, 153], - [60, 65, 70, 153], - [75, 80, 85, 153], - [90, 95, 100, 153], - [105, 110, 115, 153], - [120, 125, 130, 153], - [135, 140, 145, 153], - [150, 155, 160, 153], - [165, 170, 175, 153], - [180, 185, 190, 153], - [195, 200, 205, 153], - [210, 215, 220, 153], - [225, 230, 235, 153], - [240, 245, 250, 153], - [0, 0, 0, 154], - [0, 51, 102, 155], - [153, 204, 0, 155], - [0, 85, 170, 156], - [0, 0, 0, 157], - [0, 0, 0, 158], - [0, 85, 170, 159], - [0, 51, 102, 160], - [153, 204, 0, 160], - [0, 0, 0, 161], - [0, 85, 170, 162], - [0, 0, 0, 163], - [0, 0, 0, 164], - [0, 17, 34, 165], - [51, 68, 85, 165], - [102, 119, 136, 165], - [153, 170, 187, 165], - [204, 221, 238, 165], - [0, 0, 0, 166], - [0, 0, 0, 167], - [0, 85, 170, 168], - [0, 0, 0, 169], - [0, 3, 6, 170], - [9, 12, 15, 170], - [18, 21, 24, 170], - [27, 30, 33, 170], - [36, 39, 42, 170], - [45, 48, 51, 170], - [54, 57, 60, 170], - [63, 66, 69, 170], - [72, 75, 78, 170], - [81, 84, 87, 170], - [90, 93, 96, 170], - [99, 102, 105, 170], - [108, 111, 114, 170], - [117, 120, 123, 170], - [126, 129, 132, 170], - [135, 138, 141, 170], - [144, 147, 150, 170], - [153, 156, 159, 170], - [162, 165, 168, 170], - [171, 174, 177, 170], - [180, 183, 186, 170], - [189, 192, 195, 170], - [198, 201, 204, 170], - [207, 210, 213, 170], - [216, 219, 222, 170], - [225, 228, 231, 170], - [234, 237, 240, 170], - [243, 246, 249, 170], - [252, 0, 0, 170], - [0, 85, 170, 171], - [0, 0, 0, 172], - [0, 0, 0, 173], - [0, 85, 170, 174], - [0, 51, 102, 175], - [153, 204, 0, 175], - [0, 0, 0, 176], - [0, 85, 170, 177], - [0, 0, 0, 178], - [0, 0, 0, 179], - [0, 17, 34, 180], - [51, 68, 85, 180], - [102, 119, 136, 180], - [153, 170, 187, 180], - [204, 221, 238, 180], - [0, 0, 0, 181], - [0, 0, 0, 182], - [0, 85, 170, 183], - [0, 0, 0, 184], - [0, 51, 102, 185], - [153, 204, 0, 185], - [0, 85, 170, 186], - [0, 15, 30, 187], - [45, 60, 75, 187], - [90, 105, 120, 187], - [135, 150, 165, 187], - [180, 195, 210, 187], - [225, 240, 0, 187], - [0, 0, 0, 188], - [0, 85, 170, 189], - [0, 51, 102, 190], - [153, 204, 0, 190], - [0, 0, 0, 191], - [0, 85, 170, 192], - [0, 0, 0, 193], - [0, 0, 0, 194], - [0, 17, 34, 195], - [51, 68, 85, 195], - [102, 119, 136, 195], - [153, 170, 187, 195], - [204, 221, 238, 195], - [0, 0, 0, 196], - [0, 0, 0, 197], - [0, 85, 170, 198], - [0, 0, 0, 199], - [0, 51, 102, 200], - [153, 204, 0, 200], - [0, 85, 170, 201], - [0, 0, 0, 202], - [0, 0, 0, 203], - [0, 5, 10, 204], - [15, 20, 25, 204], - [30, 35, 40, 204], - [45, 50, 55, 204], - [60, 65, 70, 204], - [75, 80, 85, 204], - [90, 95, 100, 204], - [105, 110, 115, 204], - [120, 125, 130, 204], - [135, 140, 145, 204], - [150, 155, 160, 204], - [165, 170, 175, 204], - [180, 185, 190, 204], - [195, 200, 205, 204], - [210, 215, 220, 204], - [225, 230, 235, 204], - [240, 245, 250, 204], - [0, 51, 102, 205], - [153, 204, 0, 205], - [0, 0, 0, 206], - [0, 85, 170, 207], - [0, 0, 0, 208], - [0, 0, 0, 209], - [0, 17, 34, 210], - [51, 68, 85, 210], - [102, 119, 136, 210], - [153, 170, 187, 210], - [204, 221, 238, 210], - [0, 0, 0, 211], - [0, 0, 0, 212], - [0, 85, 170, 213], - [0, 0, 0, 214], - [0, 51, 102, 215], - [153, 204, 0, 215], - [0, 85, 170, 216], - [0, 0, 0, 217], - [0, 0, 0, 218], - [0, 85, 170, 219], - [0, 51, 102, 220], - [153, 204, 0, 220], - [0, 15, 30, 221], - [45, 60, 75, 221], - [90, 105, 120, 221], - [135, 150, 165, 221], - [180, 195, 210, 221], - [225, 240, 0, 221], - [0, 85, 170, 222], - [0, 0, 0, 223], - [0, 0, 0, 224], - [0, 17, 34, 225], - [51, 68, 85, 225], - [102, 119, 136, 225], - [153, 170, 187, 225], - [204, 221, 238, 225], - [0, 0, 0, 226], - [0, 0, 0, 227], - [0, 85, 170, 228], - [0, 0, 0, 229], - [0, 51, 102, 230], - [153, 204, 0, 230], - [0, 85, 170, 231], - [0, 0, 0, 232], - [0, 0, 0, 233], - [0, 85, 170, 234], - [0, 51, 102, 235], - [153, 204, 0, 235], - [0, 0, 0, 236], - [0, 85, 170, 237], - [0, 15, 30, 238], - [45, 60, 75, 238], - [90, 105, 120, 238], - [135, 150, 165, 238], - [180, 195, 210, 238], - [225, 240, 0, 238], - [0, 0, 0, 239], - [0, 17, 34, 240], - [51, 68, 85, 240], - [102, 119, 136, 240], - [153, 170, 187, 240], - [204, 221, 238, 240], - [0, 0, 0, 241], - [0, 0, 0, 242], - [0, 85, 170, 243], - [0, 0, 0, 244], - [0, 51, 102, 245], - [153, 204, 0, 245], - [0, 85, 170, 246], - [0, 0, 0, 247], - [0, 0, 0, 248], - [0, 85, 170, 249], - [0, 51, 102, 250], - [153, 204, 0, 250], - [0, 0, 0, 251], - [0, 85, 170, 252], - [0, 0, 0, 253], - [0, 0, 0, 254], - [5, 15, 25, 102], - [35, 40, 45, 102], - [50, 55, 60, 102], - [65, 70, 75, 102], - [80, 85, 90, 102], - [95, 100, 105, 102], - [110, 115, 120, 102], - [125, 130, 135, 102], - [140, 145, 150, 102], - [155, 160, 165, 102], - [170, 175, 180, 102], - [185, 190, 195, 102], - [200, 205, 210, 102], - [215, 220, 225, 102], - [230, 235, 240, 102], - [245, 250, 0, 102], - [15, 45, 60, 68], - [75, 90, 105, 68], - [120, 135, 150, 68], - [165, 180, 195, 68], - [210, 225, 240, 68], - [17, 34, 51, 45], - [68, 85, 102, 45], - [119, 136, 153, 45], - [170, 187, 204, 45], - [221, 238, 0, 45], - [17, 51, 68, 60], - [85, 102, 119, 60], - [136, 153, 170, 60], - [187, 204, 221, 60], - [238, 0, 0, 60], - [17, 34, 51, 90], - [68, 85, 102, 90], - [119, 136, 153, 90], - [170, 187, 204, 90], - [221, 238, 0, 90], - [17, 34, 51, 120], - [68, 85, 102, 120], - [119, 136, 153, 120], - [170, 187, 204, 120], - [221, 238, 0, 120], - [20, 25, 30, 51], - [35, 40, 45, 51], - [50, 55, 60, 51], - [65, 70, 75, 51], - [80, 85, 90, 51], - [95, 100, 105, 51], - [110, 115, 120, 51], - [125, 130, 135, 51], - [140, 145, 150, 51], - [155, 160, 165, 51], - [170, 175, 180, 51], - [185, 190, 195, 51], - [200, 205, 210, 51], - [215, 220, 225, 51], - [230, 235, 240, 51], - [245, 250, 0, 51], - [45, 60, 75, 17], - [90, 105, 120, 17], - [135, 150, 165, 17], - [180, 195, 210, 17], - [225, 240, 0, 17], - [45, 75, 90, 34], - [105, 120, 135, 34], - [150, 165, 180, 34], - [195, 210, 225, 34], - [240, 0, 0, 34], - [51, 153, 204, 20], - [51, 102, 153, 25], - [204, 0, 0, 25], - [51, 85, 119, 30], - [136, 153, 170, 30], - [187, 204, 221, 30], - [238, 0, 0, 30], - [51, 102, 153, 35], - [204, 0, 0, 35], - [51, 102, 153, 40], - [204, 0, 0, 40], - [51, 102, 153, 50], - [204, 0, 0, 50], - [51, 102, 153, 55], - [204, 0, 0, 55], - [51, 102, 153, 70], - [204, 0, 0, 70], - [51, 102, 153, 80], - [204, 0, 0, 80], - [51, 102, 153, 100], - [204, 0, 0, 100], - [51, 102, 153, 110], - [204, 0, 0, 110], - [65, 67, 69, 0], - [71, 73, 75, 0], - [77, 79, 81, 0], - [83, 85, 87, 0], - [89, 91, 93, 0], - [95, 97, 99, 0], - [101, 103, 105, 0], - [107, 109, 111, 0], - [113, 115, 117, 0], - [119, 121, 123, 0], - [125, 127, 128, 0], - [129, 130, 131, 0], - [132, 133, 134, 0], - [135, 136, 137, 0], - [138, 139, 140, 0], - [141, 142, 143, 0], - [144, 145, 146, 0], - [147, 148, 149, 0], - [150, 151, 152, 0], - [153, 154, 155, 0], - [156, 157, 158, 0], - [159, 160, 161, 0], - [162, 163, 164, 0], - [165, 166, 167, 0], - [168, 169, 170, 0], - [171, 172, 173, 0], - [174, 175, 176, 0], - [177, 178, 179, 0], - [180, 181, 182, 0], - [183, 184, 185, 0], - [186, 187, 188, 0], - [189, 190, 191, 0], - [192, 193, 194, 0], - [195, 196, 197, 0], - [198, 199, 200, 0], - [201, 202, 203, 0], - [204, 205, 206, 0], - [207, 208, 209, 0], - [210, 211, 212, 0], - [213, 214, 215, 0], - [216, 217, 218, 0], - [219, 220, 221, 0], - [222, 223, 224, 0], - [225, 226, 227, 0], - [228, 229, 230, 0], - [231, 232, 233, 0], - [234, 235, 236, 0], - [237, 238, 239, 0], - [240, 241, 242, 0], - [243, 244, 245, 0], - [246, 247, 248, 0], - [249, 250, 251, 0], - [252, 253, 254, 0], - [68, 85, 102, 15], - [119, 136, 153, 15], - [170, 187, 204, 15], - [221, 238, 0, 15], - [85, 170, 0, 3], - [85, 170, 0, 6], - [85, 170, 0, 9], - [85, 170, 0, 12], - [85, 170, 0, 18], - [85, 170, 0, 21], - [85, 170, 0, 24], - [85, 170, 0, 27], - [85, 170, 0, 33], - [85, 170, 0, 36], - [85, 170, 0, 39], - [85, 170, 0, 42], - [85, 170, 0, 48], - [85, 170, 0, 54], - [85, 170, 0, 57], - [85, 170, 0, 63], - [85, 170, 0, 66], - [85, 170, 0, 72], - [85, 170, 0, 78], - [85, 170, 0, 84], - [85, 170, 0, 96], - [85, 170, 0, 108], - [85, 170, 0, 114], - [85, 170, 0, 126], - [102, 153, 204, 5], - [153, 204, 0, 10], - ]; - let rgb_pixels = vec![ - [190, 152, 114], - [76, 38, 190], - [188, 188, 188], - [186, 124, 62], - [184, 184, 184], - [182, 182, 182], - [180, 168, 156], - [144, 132, 120], - [108, 96, 84], - [72, 60, 48], - [36, 24, 12], - [178, 178, 178], - [176, 176, 176], - [174, 116, 58], - [172, 172, 172], - [170, 168, 166], - [164, 162, 160], - [158, 156, 154], - [152, 150, 148], - [146, 144, 142], - [140, 138, 136], - [134, 132, 130], - [128, 126, 124], - [122, 120, 118], - [116, 114, 112], - [110, 108, 106], - [104, 102, 100], - [98, 96, 94], - [92, 90, 88], - [86, 84, 82], - [80, 78, 76], - [74, 72, 70], - [68, 66, 64], - [62, 60, 58], - [56, 54, 52], - [50, 48, 46], - [44, 42, 40], - [38, 36, 34], - [32, 30, 28], - [26, 24, 22], - [20, 18, 16], - [14, 12, 10], - [8, 6, 4], - [2, 170, 170], - [168, 112, 56], - [166, 166, 166], - [164, 164, 164], - [162, 108, 54], - [160, 128, 96], - [64, 32, 160], - [158, 158, 158], - [156, 104, 52], - [154, 154, 154], - [152, 152, 152], - [150, 140, 130], - [120, 110, 100], - [90, 80, 70], - [60, 50, 40], - [30, 20, 10], - [148, 148, 148], - [146, 146, 146], - [144, 96, 48], - [142, 142, 142], - [140, 112, 84], - [56, 28, 140], - [138, 92, 46], - [136, 128, 120], - [112, 104, 96], - [88, 80, 72], - [64, 56, 48], - [40, 32, 24], - [16, 8, 136], - [134, 134, 134], - [132, 88, 44], - [130, 104, 78], - [52, 26, 130], - [128, 128, 128], - [127, 127, 127], - [126, 84, 42], - [125, 100, 75], - [50, 25, 125], - [124, 124, 124], - [123, 82, 41], - [122, 122, 122], - [121, 121, 121], - [120, 112, 104], - [96, 88, 80], - [72, 64, 56], - [48, 40, 32], - [24, 16, 8], - [119, 112, 105], - [98, 91, 84], - [77, 70, 63], - [56, 49, 42], - [35, 28, 21], - [14, 7, 119], - [118, 118, 118], - [117, 78, 39], - [116, 116, 116], - [115, 92, 69], - [46, 23, 115], - [114, 76, 38], - [113, 113, 113], - [112, 112, 112], - [111, 74, 37], - [110, 88, 66], - [44, 22, 110], - [109, 109, 109], - [108, 72, 36], - [107, 107, 107], - [106, 106, 106], - [105, 98, 91], - [84, 77, 70], - [63, 56, 49], - [42, 35, 28], - [21, 14, 7], - [104, 104, 104], - [103, 103, 103], - [102, 100, 98], - [96, 94, 92], - [90, 88, 86], - [84, 82, 80], - [78, 76, 74], - [72, 70, 68], - [66, 64, 62], - [60, 58, 56], - [54, 52, 50], - [48, 46, 44], - [42, 40, 38], - [36, 34, 32], - [30, 28, 26], - [24, 22, 20], - [18, 16, 14], - [12, 10, 8], - [6, 4, 2], - [101, 101, 101], - [100, 80, 60], - [40, 20, 100], - [99, 66, 33], - [98, 98, 98], - [97, 97, 97], - [96, 64, 32], - [95, 76, 57], - [38, 19, 95], - [94, 94, 94], - [93, 62, 31], - [92, 92, 92], - [91, 91, 91], - [90, 84, 78], - [72, 66, 60], - [54, 48, 42], - [36, 30, 24], - [18, 12, 6], - [89, 89, 89], - [88, 88, 88], - [87, 58, 29], - [86, 86, 86], - [85, 84, 83], - [82, 81, 80], - [79, 78, 77], - [76, 75, 74], - [73, 72, 71], - [70, 69, 68], - [67, 66, 65], - [64, 63, 62], - [61, 60, 59], - [58, 57, 56], - [55, 54, 53], - [52, 51, 50], - [49, 48, 47], - [46, 45, 44], - [43, 42, 41], - [40, 39, 38], - [37, 36, 35], - [34, 33, 32], - [31, 30, 29], - [28, 27, 26], - [25, 24, 23], - [22, 21, 20], - [19, 18, 17], - [16, 15, 14], - [13, 12, 11], - [10, 9, 8], - [7, 6, 5], - [4, 3, 2], - [1, 85, 85], - [84, 56, 28], - [83, 83, 83], - [82, 82, 82], - [81, 54, 27], - [80, 64, 48], - [32, 16, 80], - [79, 79, 79], - [78, 52, 26], - [77, 77, 77], - [76, 76, 76], - [75, 70, 65], - [60, 55, 50], - [45, 40, 35], - [30, 25, 20], - [15, 10, 5], - [74, 74, 74], - [73, 73, 73], - [72, 48, 24], - [71, 71, 71], - [70, 56, 42], - [28, 14, 70], - [69, 46, 23], - [68, 64, 60], - [56, 52, 48], - [44, 40, 36], - [32, 28, 24], - [20, 16, 12], - [8, 4, 68], - [67, 67, 67], - [66, 44, 22], - [65, 52, 39], - [26, 13, 65], - [64, 64, 64], - [63, 42, 21], - [62, 62, 62], - [61, 61, 61], - [60, 56, 52], - [48, 44, 40], - [36, 32, 28], - [24, 20, 16], - [12, 8, 4], - [59, 59, 59], - [58, 58, 58], - [57, 38, 19], - [56, 56, 56], - [55, 44, 33], - [22, 11, 55], - [54, 36, 18], - [53, 53, 53], - [52, 52, 52], - [51, 50, 49], - [48, 47, 46], - [45, 44, 43], - [42, 41, 40], - [39, 38, 37], - [36, 35, 34], - [33, 32, 31], - [30, 29, 28], - [27, 26, 25], - [24, 23, 22], - [21, 20, 19], - [18, 17, 16], - [15, 14, 13], - [12, 11, 10], - [9, 8, 7], - [6, 5, 4], - [3, 2, 1], - [50, 40, 30], - [20, 10, 50], - [49, 49, 49], - [48, 32, 16], - [47, 47, 47], - [46, 46, 46], - [45, 42, 39], - [36, 33, 30], - [27, 24, 21], - [18, 15, 12], - [9, 6, 3], - [44, 44, 44], - [43, 43, 43], - [42, 28, 14], - [41, 41, 41], - [40, 32, 24], - [16, 8, 40], - [39, 26, 13], - [38, 38, 38], - [37, 37, 37], - [36, 24, 12], - [35, 28, 21], - [14, 7, 35], - [34, 32, 30], - [28, 26, 24], - [22, 20, 18], - [16, 14, 12], - [10, 8, 6], - [4, 2, 34], - [33, 22, 11], - [32, 32, 32], - [31, 31, 31], - [30, 28, 26], - [24, 22, 20], - [18, 16, 14], - [12, 10, 8], - [6, 4, 2], - [29, 29, 29], - [28, 28, 28], - [27, 18, 9], - [26, 26, 26], - [25, 20, 15], - [10, 5, 25], - [24, 16, 8], - [23, 23, 23], - [22, 22, 22], - [21, 14, 7], - [20, 16, 12], - [8, 4, 20], - [19, 19, 19], - [18, 12, 6], - [17, 16, 15], - [14, 13, 12], - [11, 10, 9], - [8, 7, 6], - [5, 4, 3], - [2, 1, 17], - [16, 16, 16], - [15, 14, 13], - [12, 11, 10], - [9, 8, 7], - [6, 5, 4], - [3, 2, 1], - [14, 14, 14], - [13, 13, 13], - [12, 8, 4], - [11, 11, 11], - [10, 8, 6], - [4, 2, 10], - [9, 6, 3], - [8, 8, 8], - [7, 7, 7], - [6, 4, 2], - [5, 4, 3], - [2, 1, 5], - [4, 4, 4], - [3, 2, 1], - [2, 2, 2], - [1, 1, 1], - [150, 144, 138], - [132, 129, 126], - [123, 120, 117], - [114, 111, 108], - [105, 102, 99], - [96, 93, 90], - [87, 84, 81], - [78, 75, 72], - [69, 66, 63], - [60, 57, 54], - [51, 48, 45], - [42, 39, 36], - [33, 30, 27], - [24, 21, 18], - [15, 12, 9], - [6, 3, 153], - [176, 154, 143], - [132, 121, 110], - [99, 88, 77], - [66, 55, 44], - [33, 22, 11], - [196, 182, 168], - [154, 140, 126], - [112, 98, 84], - [70, 56, 42], - [28, 14, 210], - [182, 156, 143], - [130, 117, 104], - [91, 78, 65], - [52, 39, 26], - [13, 195, 195], - [154, 143, 132], - [121, 110, 99], - [88, 77, 66], - [55, 44, 33], - [22, 11, 165], - [126, 117, 108], - [99, 90, 81], - [72, 63, 54], - [45, 36, 27], - [18, 9, 135], - [188, 184, 180], - [176, 172, 168], - [164, 160, 156], - [152, 148, 144], - [140, 136, 132], - [128, 124, 120], - [116, 112, 108], - [104, 100, 96], - [92, 88, 84], - [80, 76, 72], - [68, 64, 60], - [56, 52, 48], - [44, 40, 36], - [32, 28, 24], - [20, 16, 12], - [8, 4, 204], - [196, 182, 168], - [154, 140, 126], - [112, 98, 84], - [70, 56, 42], - [28, 14, 238], - [182, 156, 143], - [130, 117, 104], - [91, 78, 65], - [52, 39, 26], - [13, 221, 221], - [188, 94, 47], - [184, 138, 92], - [46, 230, 230], - [180, 150, 120], - [105, 90, 75], - [60, 45, 30], - [15, 225, 225], - [176, 132, 88], - [44, 220, 220], - [172, 129, 86], - [43, 215, 215], - [164, 123, 82], - [41, 205, 205], - [160, 120, 80], - [40, 200, 200], - [148, 111, 74], - [37, 185, 185], - [140, 105, 70], - [35, 175, 175], - [124, 93, 62], - [31, 155, 155], - [116, 87, 58], - [29, 145, 145], - [190, 188, 186], - [184, 182, 180], - [178, 176, 174], - [172, 170, 168], - [166, 164, 162], - [160, 158, 156], - [154, 152, 150], - [148, 146, 144], - [142, 140, 138], - [136, 134, 132], - [130, 128, 127], - [126, 125, 124], - [123, 122, 121], - [120, 119, 118], - [117, 116, 115], - [114, 113, 112], - [111, 110, 109], - [108, 107, 106], - [105, 104, 103], - [102, 101, 100], - [99, 98, 97], - [96, 95, 94], - [93, 92, 91], - [90, 89, 88], - [87, 86, 85], - [84, 83, 82], - [81, 80, 79], - [78, 77, 76], - [75, 74, 73], - [72, 71, 70], - [69, 68, 67], - [66, 65, 64], - [63, 62, 61], - [60, 59, 58], - [57, 56, 55], - [54, 53, 52], - [51, 50, 49], - [48, 47, 46], - [45, 44, 43], - [42, 41, 40], - [39, 38, 37], - [36, 35, 34], - [33, 32, 31], - [30, 29, 28], - [27, 26, 25], - [24, 23, 22], - [21, 20, 19], - [18, 17, 16], - [15, 14, 13], - [12, 11, 10], - [9, 8, 7], - [6, 5, 4], - [3, 2, 1], - [176, 160, 144], - [128, 112, 96], - [80, 64, 48], - [32, 16, 240], - [168, 84, 252], - [166, 83, 249], - [164, 82, 246], - [162, 81, 243], - [158, 79, 237], - [156, 78, 234], - [154, 77, 231], - [152, 76, 228], - [148, 74, 222], - [146, 73, 219], - [144, 72, 216], - [142, 71, 213], - [138, 69, 207], - [134, 67, 201], - [132, 66, 198], - [128, 64, 192], - [126, 63, 189], - [122, 61, 183], - [118, 59, 177], - [114, 57, 171], - [106, 53, 159], - [98, 49, 147], - [94, 47, 141], - [86, 43, 129], - [150, 100, 50], - [98, 49, 245], - ]; - for (&cmyk_pixel, rgb_pixel) in cmyk_pixels.iter().zip(rgb_pixels) { - single_pix_correct(cmyk_pixel, rgb_pixel); - } - } - - #[cfg(feature = "benchmarks")] - #[bench] - fn bench_cmyk_to_rgb(b: &mut Bencher) { - let mut v = Vec::with_capacity((W * H * 4) as usize); - for c in 0..=255 { - for k in 0..=255 { - v.push(c as u8); - v.push(0); - v.push(0); - v.push(k as u8); - } + LargeDimensions(_) => ImageError::Limits(LimitError::from_kind( + crate::error::LimitErrorKind::DimensionError, + )), + err => ImageError::Decoding(DecodingError::new(ImageFormat::Jpeg.into(), err)), } - - b.iter(|| { - cmyk_to_rgb(&v); - }); - } - - #[cfg(feature = "benchmarks")] - #[bench] - fn bench_cmyk_to_rgb_single(b: &mut Bencher) { - b.iter(|| { - cmyk_to_rgb(&[128, 128, 128, 128]); - }); } } From f40a507ec56896bfed7b01c046d11ac4c583fab0 Mon Sep 17 00:00:00 2001 From: "Sergey \"Shnatsel\" Davidoff" Date: Mon, 13 Mar 2023 16:30:50 +0100 Subject: [PATCH 02/18] Switch to released versions of zune-* --- Cargo.toml | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/Cargo.toml b/Cargo.toml index 0dec884fcf..8319e16000 100644 --- a/Cargo.toml +++ b/Cargo.toml @@ -47,8 +47,8 @@ color_quant = "1.1" exr = { version = "1.5.0", optional = true } qoi = { version = "0.4", optional = true } libwebp = { package = "webp", version = "0.2.2", default-features = false, optional = true } -zune-core = { git = "https://github.com/etemesi254/zune-image" } -zune-jpeg = { git = "https://github.com/etemesi254/zune-image" } +zune-core = { version = "0.2.1" } +zune-jpeg = { version = "0.3.13" } [dev-dependencies] crc32fast = "1.2.0" From a837e95d436c7dcbc0c65279a80dd0d396ffc013 Mon Sep 17 00:00:00 2001 From: "Sergey \"Shnatsel\" Davidoff" Date: Mon, 13 Mar 2023 16:39:12 +0100 Subject: [PATCH 03/18] Update comment --- src/codecs/jpeg/decoder.rs | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/src/codecs/jpeg/decoder.rs b/src/codecs/jpeg/decoder.rs index e0b7af2224..62cc5e3e59 100644 --- a/src/codecs/jpeg/decoder.rs +++ b/src/codecs/jpeg/decoder.rs @@ -130,8 +130,8 @@ impl ColorType { let colorspace = to_supported_color_space(colorspace); use zune_core::colorspace::ColorSpace::*; match colorspace { - // TODO: are these always 8-bit? I've asked at - // https://github.com/etemesi254/zune-image/discussions/99 + // As of zune-jpeg 0.3.13 the output is always 8-bit, + // but support for 16-bit JPEG might be added in the future. RGB => ColorType::Rgb8, RGBA => ColorType::Rgba8, Luma => ColorType::L8, From 25ba2780b9ae01826e42392d8681fc2c2d0e0c42 Mon Sep 17 00:00:00 2001 From: "Sergey \"Shnatsel\" Davidoff" Date: Mon, 13 Mar 2023 16:40:02 +0100 Subject: [PATCH 04/18] Decode directly into the provided buffer, without a memcpy() --- src/codecs/jpeg/decoder.rs | 3 +-- 1 file changed, 1 insertion(+), 2 deletions(-) diff --git a/src/codecs/jpeg/decoder.rs b/src/codecs/jpeg/decoder.rs index 62cc5e3e59..3588e2d6ac 100644 --- a/src/codecs/jpeg/decoder.rs +++ b/src/codecs/jpeg/decoder.rs @@ -119,8 +119,7 @@ impl<'a, R: 'a + Read> ImageDecoder<'a> for JpegDecoder { } let mut decoder = new_zune_decoder(&self.input, self.orig_color_space); - let data = decoder.decode().map_err(ImageError::from_jpeg)?; - buf.copy_from_slice(&data); + decoder.decode_into(buf).map_err(ImageError::from_jpeg)?; Ok(()) } } From 30798e8bac04d57a4685f0a0d3d1cc0855e32ea0 Mon Sep 17 00:00:00 2001 From: "Sergey \"Shnatsel\" Davidoff" Date: Mon, 13 Mar 2023 16:40:42 +0100 Subject: [PATCH 05/18] cargo fmt --- src/codecs/jpeg/decoder.rs | 6 ++---- 1 file changed, 2 insertions(+), 4 deletions(-) diff --git a/src/codecs/jpeg/decoder.rs b/src/codecs/jpeg/decoder.rs index 3588e2d6ac..86a0d4b1fd 100644 --- a/src/codecs/jpeg/decoder.rs +++ b/src/codecs/jpeg/decoder.rs @@ -4,7 +4,7 @@ use std::mem; use crate::color::ColorType; use crate::error::{ - DecodingError, ImageError, ImageResult, UnsupportedError, UnsupportedErrorKind, LimitError, + DecodingError, ImageError, ImageResult, LimitError, UnsupportedError, UnsupportedErrorKind, }; use crate::image::{ImageDecoder, ImageFormat}; @@ -28,9 +28,7 @@ impl JpegDecoder { let mut r = r; r.read_to_end(&mut input)?; let mut decoder = zune_jpeg::JpegDecoder::new(&input); - decoder - .decode_headers() - .map_err(ImageError::from_jpeg)?; + decoder.decode_headers().map_err(ImageError::from_jpeg)?; // now that we've decoded the headers we can `.unwrap()` // all these functions that only fail if called before decoding the headers let (width, height) = decoder.dimensions().unwrap(); From e924ccdb23027fd18384dce61f15c1cbd3751b6d Mon Sep 17 00:00:00 2001 From: "Sergey \"Shnatsel\" Davidoff" Date: Mon, 13 Mar 2023 16:48:33 +0100 Subject: [PATCH 06/18] Remove jpeg-decoder from Cargo.toml --- Cargo.toml | 14 +++++--------- 1 file changed, 5 insertions(+), 9 deletions(-) diff --git a/Cargo.toml b/Cargo.toml index 8319e16000..0a24579570 100644 --- a/Cargo.toml +++ b/Cargo.toml @@ -35,7 +35,6 @@ byteorder = "1.3.2" num-rational = { version = "0.4", default-features = false } num-traits = "0.2.0" gif = { version = "0.12", optional = true } -jpeg = { package = "jpeg-decoder", version = "0.3.0", default-features = false, optional = true } png = { version = "0.17.6", optional = true } tiff = { version = "0.8.0", optional = true } ravif = { version = "0.11.0", optional = true } @@ -47,8 +46,8 @@ color_quant = "1.1" exr = { version = "1.5.0", optional = true } qoi = { version = "0.4", optional = true } libwebp = { package = "webp", version = "0.2.2", default-features = false, optional = true } -zune-core = { version = "0.2.1" } -zune-jpeg = { version = "0.3.13" } +zune-core = { version = "0.2.1", default-features = false, optional = true } +zune-jpeg = { version = "0.3.13", optional = true } [dev-dependencies] crc32fast = "1.2.0" @@ -56,15 +55,13 @@ num-complex = "0.4" glob = "0.3" quickcheck = "1" criterion = "0.3" -# Keep this in sync with the jpeg dependency above. This is used to enable the platform_independent -# feature when testing, so `cargo test` works correctly. -jpeg = { package = "jpeg-decoder", version = "0.3.0", default-features = false, features = ["platform_independent"] } [features] # TODO: Add "avif" to this list while preparing for 0.24.0 default = ["gif", "jpeg", "ico", "png", "pnm", "tga", "tiff", "webp", "bmp", "hdr", "dxt", "dds", "farbfeld", "jpeg_rayon", "openexr", "qoi"] ico = ["bmp", "png"] +jpeg = ["dep:zune-core", "dep:zune-jpeg"] pnm = [] tga = [] bmp = [] @@ -80,9 +77,8 @@ webp = [] # Non-default, not included in `webp`. Requires native dependency libwebp. webp-encoder = ["libwebp"] -# Enables multi-threading. -# Requires latest stable Rust. -jpeg_rayon = ["jpeg/rayon"] +# Dummy feature for backwards compatibility +jpeg_rayon = ["jpeg"] # Non-default, enables avif support. # Requires latest stable Rust. avif = ["avif-encoder"] From 270b416a85119d4ac7a5c911a81420a9997857d1 Mon Sep 17 00:00:00 2001 From: "Sergey \"Shnatsel\" Davidoff" Date: Mon, 13 Mar 2023 20:59:36 +0100 Subject: [PATCH 07/18] Don't panic when scale() is called --- src/codecs/jpeg/decoder.rs | 8 +++++--- 1 file changed, 5 insertions(+), 3 deletions(-) diff --git a/src/codecs/jpeg/decoder.rs b/src/codecs/jpeg/decoder.rs index 86a0d4b1fd..8dc8f9841c 100644 --- a/src/codecs/jpeg/decoder.rs +++ b/src/codecs/jpeg/decoder.rs @@ -57,10 +57,12 @@ impl JpegDecoder { /// applied, is returned. pub fn scale( &mut self, - requested_width: u16, - requested_height: u16, + _requested_width: u16, + _requested_height: u16, ) -> ImageResult<(u16, u16)> { - todo!(); // TODO: implement using subsampling + // zune-jpeg doesn't support this yet: + // https://github.com/etemesi254/zune-image/issues/103 + Ok((self.width, self.height)) } } From b3446b82d26798d7cad240576274ca035ea34300 Mon Sep 17 00:00:00 2001 From: "Sergey \"Shnatsel\" Davidoff" Date: Mon, 13 Mar 2023 21:48:59 +0100 Subject: [PATCH 08/18] Update comment --- src/codecs/jpeg/decoder.rs | 17 ++++------------- 1 file changed, 4 insertions(+), 13 deletions(-) diff --git a/src/codecs/jpeg/decoder.rs b/src/codecs/jpeg/decoder.rs index 8dc8f9841c..506886bd8b 100644 --- a/src/codecs/jpeg/decoder.rs +++ b/src/codecs/jpeg/decoder.rs @@ -42,19 +42,10 @@ impl JpegDecoder { }) } - /// Configure the decoder to scale the image during decoding. - /// - /// This efficiently scales the image by the smallest supported - /// scale factor that produces an image larger than or equal to - /// the requested size in at least one axis. The currently - /// implemented scale factors are 1/8, 1/4, 1/2 and 1. - /// - /// To generate a thumbnail of an exact size, pass the desired - /// size and then scale to the final size using a traditional - /// resampling algorithm. - /// - /// The size of the image to be loaded, with the scale factor - /// applied, is returned. + /// Some decoders support scaling the image during decoding, + /// but the current backend, `zune-jpeg`, doesn't, + /// so this function currently does nothing + /// and always returns the original dimensions. pub fn scale( &mut self, _requested_width: u16, From 32975385908c780a749d77d135f920f63913166a Mon Sep 17 00:00:00 2001 From: Sergey Shnatsel Davidoff Date: Tue, 14 Mar 2023 17:07:22 +0000 Subject: [PATCH 09/18] bump zune-jpeg dependency --- Cargo.toml | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/Cargo.toml b/Cargo.toml index 0a24579570..87da16c9e3 100644 --- a/Cargo.toml +++ b/Cargo.toml @@ -47,7 +47,7 @@ exr = { version = "1.5.0", optional = true } qoi = { version = "0.4", optional = true } libwebp = { package = "webp", version = "0.2.2", default-features = false, optional = true } zune-core = { version = "0.2.1", default-features = false, optional = true } -zune-jpeg = { version = "0.3.13", optional = true } +zune-jpeg = { version = "0.3.14", optional = true } [dev-dependencies] crc32fast = "1.2.0" From ef0e2c9edad6e4d4171a41d39e43a46ac6aaf2b7 Mon Sep 17 00:00:00 2001 From: Sergey Shnatsel Davidoff Date: Tue, 14 Mar 2023 17:07:51 +0000 Subject: [PATCH 10/18] Add ICC extraction support --- src/codecs/jpeg/decoder.rs | 5 ++++- 1 file changed, 4 insertions(+), 1 deletion(-) diff --git a/src/codecs/jpeg/decoder.rs b/src/codecs/jpeg/decoder.rs index 506886bd8b..12b021c86b 100644 --- a/src/codecs/jpeg/decoder.rs +++ b/src/codecs/jpeg/decoder.rs @@ -16,6 +16,7 @@ pub struct JpegDecoder { orig_color_space: ZuneColorSpace, width: u16, height: u16, + icc: Option>, // For API compatibility with the previous jpeg_decoder wrapper. // Can be removed later, which would be an API break. phantom: PhantomData, @@ -33,11 +34,13 @@ impl JpegDecoder { // all these functions that only fail if called before decoding the headers let (width, height) = decoder.dimensions().unwrap(); let orig_color_space = decoder.get_output_colorspace().unwrap(); + let icc = decoder.icc_profile(); Ok(JpegDecoder { input, orig_color_space, width, height, + icc, phantom: PhantomData, }) } @@ -85,7 +88,7 @@ impl<'a, R: 'a + Read> ImageDecoder<'a> for JpegDecoder { } fn icc_profile(&mut self) -> Option> { - None // zune-jpeg doesn't support it + self.icc.clone() } fn into_reader(self) -> ImageResult { From 52bc19e03741a23f4c41edc39d3a05d4723199d0 Mon Sep 17 00:00:00 2001 From: "Sergey \"Shnatsel\" Davidoff" Date: Tue, 14 Mar 2023 21:58:54 +0100 Subject: [PATCH 11/18] Extract the ICC profile lazily --- src/codecs/jpeg/decoder.rs | 10 ++++++---- 1 file changed, 6 insertions(+), 4 deletions(-) diff --git a/src/codecs/jpeg/decoder.rs b/src/codecs/jpeg/decoder.rs index 12b021c86b..2d0dc70040 100644 --- a/src/codecs/jpeg/decoder.rs +++ b/src/codecs/jpeg/decoder.rs @@ -16,7 +16,6 @@ pub struct JpegDecoder { orig_color_space: ZuneColorSpace, width: u16, height: u16, - icc: Option>, // For API compatibility with the previous jpeg_decoder wrapper. // Can be removed later, which would be an API break. phantom: PhantomData, @@ -34,13 +33,11 @@ impl JpegDecoder { // all these functions that only fail if called before decoding the headers let (width, height) = decoder.dimensions().unwrap(); let orig_color_space = decoder.get_output_colorspace().unwrap(); - let icc = decoder.icc_profile(); Ok(JpegDecoder { input, orig_color_space, width, height, - icc, phantom: PhantomData, }) } @@ -88,7 +85,12 @@ impl<'a, R: 'a + Read> ImageDecoder<'a> for JpegDecoder { } fn icc_profile(&mut self) -> Option> { - self.icc.clone() + let mut decoder = zune_jpeg::JpegDecoder::new(&self.input); + if let Ok(_) = decoder.decode_headers() { + decoder.icc_profile() + } else { + None + } } fn into_reader(self) -> ImageResult { From 2c6e79209e02440430a94f65b3c7669a5141e7ba Mon Sep 17 00:00:00 2001 From: "Sergey \"Shnatsel\" Davidoff" Date: Tue, 14 Mar 2023 21:18:50 +0000 Subject: [PATCH 12/18] commit review suggestion Co-authored-by: Jonathan Behrens --- src/codecs/jpeg/decoder.rs | 7 ++----- 1 file changed, 2 insertions(+), 5 deletions(-) diff --git a/src/codecs/jpeg/decoder.rs b/src/codecs/jpeg/decoder.rs index 2d0dc70040..8bc30c1c8e 100644 --- a/src/codecs/jpeg/decoder.rs +++ b/src/codecs/jpeg/decoder.rs @@ -86,11 +86,8 @@ impl<'a, R: 'a + Read> ImageDecoder<'a> for JpegDecoder { fn icc_profile(&mut self) -> Option> { let mut decoder = zune_jpeg::JpegDecoder::new(&self.input); - if let Ok(_) = decoder.decode_headers() { - decoder.icc_profile() - } else { - None - } + decoder.decode_headers().ok()?; + decoder.icc_profile() } fn into_reader(self) -> ImageResult { From f1335135dc10a5fc46a63a201b4de9e0b674d914 Mon Sep 17 00:00:00 2001 From: "Sergey \"Shnatsel\" Davidoff" Date: Tue, 14 Mar 2023 22:33:53 +0100 Subject: [PATCH 13/18] Disable JPEG reference image comparison via a hash; lossy decoders are never going to be bit-exact --- tests/reference_images.rs | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/tests/reference_images.rs b/tests/reference_images.rs index 9106b63a8c..e3b7eb1382 100644 --- a/tests/reference_images.rs +++ b/tests/reference_images.rs @@ -18,7 +18,7 @@ where { let base: PathBuf = BASE_PATH.iter().collect(); let decoders = &[ - "tga", "tiff", "png", "gif", "bmp", "ico", "jpg", "hdr", "pbm", "webp", + "tga", "tiff", "png", "gif", "bmp", "ico", "hdr", "pbm", "webp", ]; for decoder in decoders { let mut path = base.clone(); From 625b4932c339606f0ba804dd43dfd7ada6187b51 Mon Sep 17 00:00:00 2001 From: "Sergey \"Shnatsel\" Davidoff" Date: Tue, 14 Mar 2023 22:42:31 +0100 Subject: [PATCH 14/18] Sync Cargo.toml.public-private-dependencies with Cargo.toml --- Cargo.toml.public-private-dependencies | 38 ++++++++++++++++---------- 1 file changed, 24 insertions(+), 14 deletions(-) diff --git a/Cargo.toml.public-private-dependencies b/Cargo.toml.public-private-dependencies index 540949de6f..bdd4fad5d4 100644 --- a/Cargo.toml.public-private-dependencies +++ b/Cargo.toml.public-private-dependencies @@ -2,9 +2,12 @@ cargo-features = ["public-dependency"] [package] name = "image" -version = "0.24.0-alpha" +version = "0.24.5" edition = "2018" -rust-version = "1.56" +resolver = "2" + +# note: when changed, also update test runner in `.github/workflows/rust.yml` +rust-version = "1.61.0" license = "MIT" description = "Imaging library written in Rust. Provides basic filters and decoders for the most common image formats." @@ -31,20 +34,22 @@ path = "./src/lib.rs" [dependencies] bytemuck = { version = "1.7.0", features = ["extern_crate_alloc"] } # includes cast_vec byteorder = "1.3.2" -num-iter = "0.1.32" num-rational = { version = "0.4", default-features = false } num-traits = { version = "0.2.0", public = true } -gif = { version = "0.11.1", optional = true } -jpeg = { package = "jpeg-decoder", version = "0.2.1", default-features = false, optional = true } -png = { version = "0.17.0", optional = true } -tiff = { version = "0.7.1", optional = true } -ravif = { version = "0.8.0", optional = true } +gif = { version = "0.12", optional = true } +png = { version = "0.17.6", optional = true } +tiff = { version = "0.8.0", optional = true } +ravif = { version = "0.11.0", optional = true } rgb = { version = "0.8.25", optional = true } mp4parse = { version = "0.12.0", optional = true } dav1d = { version = "0.6.0", optional = true } dcv-color-primitives = { version = "0.4.0", optional = true } -exr = { version = "1.4.1", optional = true } color_quant = { version = "1.1", public = true } +exr = { version = "1.5.0", optional = true } +qoi = { version = "0.4", optional = true } +libwebp = { package = "webp", version = "0.2.2", default-features = false, optional = true } +zune-core = { version = "0.2.1", default-features = false, optional = true } +zune-jpeg = { version = "0.3.14", optional = true } [dev-dependencies] crc32fast = "1.2.0" @@ -55,22 +60,27 @@ criterion = "0.3" [features] # TODO: Add "avif" to this list while preparing for 0.24.0 -default = ["gif", "jpeg", "ico", "png", "pnm", "tga", "tiff", "webp", "bmp", "hdr", "dxt", "dds", "farbfeld", "jpeg_rayon", "openexr"] +default = ["gif", "jpeg", "ico", "png", "pnm", "tga", "tiff", "webp", "bmp", "hdr", "dxt", "dds", "farbfeld", "jpeg_rayon", "openexr", "qoi"] ico = ["bmp", "png"] +jpeg = ["dep:zune-core", "dep:zune-jpeg"] pnm = [] tga = [] -webp = [] bmp = [] hdr = [] dxt = [] dds = ["dxt"] farbfeld = [] openexr = ["exr"] +qoi = ["dep:qoi"] -# Enables multi-threading. -# Requires latest stable Rust. -jpeg_rayon = ["jpeg/rayon"] +# Enables WebP decoder support. +webp = [] +# Non-default, not included in `webp`. Requires native dependency libwebp. +webp-encoder = ["libwebp"] + +# Dummy feature for backwards compatibility +jpeg_rayon = ["jpeg"] # Non-default, enables avif support. # Requires latest stable Rust. avif = ["avif-encoder"] From fbf67e75bc153b004b318341f2245bcf4a10876a Mon Sep 17 00:00:00 2001 From: "Sergey \"Shnatsel\" Davidoff" Date: Tue, 14 Mar 2023 22:43:52 +0100 Subject: [PATCH 15/18] Drop jpeg_rayon from default features, it doesn't do anything anymore --- Cargo.toml | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/Cargo.toml b/Cargo.toml index 87da16c9e3..423856acfb 100644 --- a/Cargo.toml +++ b/Cargo.toml @@ -58,7 +58,7 @@ criterion = "0.3" [features] # TODO: Add "avif" to this list while preparing for 0.24.0 -default = ["gif", "jpeg", "ico", "png", "pnm", "tga", "tiff", "webp", "bmp", "hdr", "dxt", "dds", "farbfeld", "jpeg_rayon", "openexr", "qoi"] +default = ["gif", "jpeg", "ico", "png", "pnm", "tga", "tiff", "webp", "bmp", "hdr", "dxt", "dds", "farbfeld", "openexr", "qoi"] ico = ["bmp", "png"] jpeg = ["dep:zune-core", "dep:zune-jpeg"] From dbb0a6b17f07c0b589a98720db511b855c0a41b7 Mon Sep 17 00:00:00 2001 From: "Sergey \"Shnatsel\" Davidoff" Date: Wed, 15 Mar 2023 00:42:05 +0100 Subject: [PATCH 16/18] Add dimensions limits to the struct and pass them to the decoder; still missing an API to set them --- src/codecs/jpeg/decoder.rs | 24 ++++++++++++++++++++---- 1 file changed, 20 insertions(+), 4 deletions(-) diff --git a/src/codecs/jpeg/decoder.rs b/src/codecs/jpeg/decoder.rs index 8bc30c1c8e..b6949e787b 100644 --- a/src/codecs/jpeg/decoder.rs +++ b/src/codecs/jpeg/decoder.rs @@ -7,6 +7,7 @@ use crate::error::{ DecodingError, ImageError, ImageResult, LimitError, UnsupportedError, UnsupportedErrorKind, }; use crate::image::{ImageDecoder, ImageFormat}; +use crate::io::Limits; type ZuneColorSpace = zune_core::colorspace::ColorSpace; @@ -16,6 +17,7 @@ pub struct JpegDecoder { orig_color_space: ZuneColorSpace, width: u16, height: u16, + limits: Limits, // For API compatibility with the previous jpeg_decoder wrapper. // Can be removed later, which would be an API break. phantom: PhantomData, @@ -33,11 +35,15 @@ impl JpegDecoder { // all these functions that only fail if called before decoding the headers let (width, height) = decoder.dimensions().unwrap(); let orig_color_space = decoder.get_output_colorspace().unwrap(); + // Limits are disabled by default for backwards compatibility with jpeg_decoder + // which did not support limits, so enabling them would break crate users + let limits = Limits::no_limits(); Ok(JpegDecoder { input, orig_color_space, width, height, + limits, phantom: PhantomData, }) } @@ -91,7 +97,7 @@ impl<'a, R: 'a + Read> ImageDecoder<'a> for JpegDecoder { } fn into_reader(self) -> ImageResult { - let mut decoder = new_zune_decoder(&self.input, self.orig_color_space); + let mut decoder = new_zune_decoder(&self.input, self.orig_color_space, self.limits); let data = decoder.decode().map_err(ImageError::from_jpeg)?; Ok(JpegReader(Cursor::new(data), PhantomData)) } @@ -111,7 +117,7 @@ impl<'a, R: 'a + Read> ImageDecoder<'a> for JpegDecoder { ))); } - let mut decoder = new_zune_decoder(&self.input, self.orig_color_space); + let mut decoder = new_zune_decoder(&self.input, self.orig_color_space, self.limits); decoder.decode_into(buf).map_err(ImageError::from_jpeg)?; Ok(()) } @@ -143,10 +149,20 @@ fn to_supported_color_space(orig: ZuneColorSpace) -> ZuneColorSpace { } } -fn new_zune_decoder(input: &[u8], orig_color_space: ZuneColorSpace) -> zune_jpeg::JpegDecoder { +fn new_zune_decoder( + input: &[u8], + orig_color_space: ZuneColorSpace, + limits: Limits, +) -> zune_jpeg::JpegDecoder { let target_color_space = to_supported_color_space(orig_color_space); - let options = + let mut options = zune_core::options::DecoderOptions::default().jpeg_set_out_colorspace(target_color_space); + if let Some(max_width) = limits.max_image_width { + options = options.set_max_width(max_width as usize); // u32 to usize never truncates + }; + if let Some(max_height) = limits.max_image_height { + options = options.set_max_height(max_height as usize); // u32 to usize never truncates + }; zune_jpeg::JpegDecoder::new_with_options(options, &input) } From 6f4b8eaf7f1b626e595d0bc00f2d435124de1b22 Mon Sep 17 00:00:00 2001 From: "Sergey \"Shnatsel\" Davidoff" Date: Wed, 15 Mar 2023 00:55:05 +0100 Subject: [PATCH 17/18] Implement setting dimension limits --- src/codecs/jpeg/decoder.rs | 5 +++++ 1 file changed, 5 insertions(+) diff --git a/src/codecs/jpeg/decoder.rs b/src/codecs/jpeg/decoder.rs index b6949e787b..a51c01f77b 100644 --- a/src/codecs/jpeg/decoder.rs +++ b/src/codecs/jpeg/decoder.rs @@ -121,6 +121,11 @@ impl<'a, R: 'a + Read> ImageDecoder<'a> for JpegDecoder { decoder.decode_into(buf).map_err(ImageError::from_jpeg)?; Ok(()) } + + fn set_limits(&mut self, limits: Limits) -> ImageResult<()> { + self.limits = limits; + Ok(()) + } } impl ColorType { From 1f6e8d89cb3490d24d87c7e9d820c8a57b646a24 Mon Sep 17 00:00:00 2001 From: "Sergey \"Shnatsel\" Davidoff" Date: Wed, 15 Mar 2023 01:55:10 +0100 Subject: [PATCH 18/18] Correctly set unbounded limits --- src/codecs/jpeg/decoder.rs | 14 ++++++++------ 1 file changed, 8 insertions(+), 6 deletions(-) diff --git a/src/codecs/jpeg/decoder.rs b/src/codecs/jpeg/decoder.rs index a51c01f77b..1a1330ef7b 100644 --- a/src/codecs/jpeg/decoder.rs +++ b/src/codecs/jpeg/decoder.rs @@ -162,12 +162,14 @@ fn new_zune_decoder( let target_color_space = to_supported_color_space(orig_color_space); let mut options = zune_core::options::DecoderOptions::default().jpeg_set_out_colorspace(target_color_space); - if let Some(max_width) = limits.max_image_width { - options = options.set_max_width(max_width as usize); // u32 to usize never truncates - }; - if let Some(max_height) = limits.max_image_height { - options = options.set_max_height(max_height as usize); // u32 to usize never truncates - }; + options = options.set_max_width(match limits.max_image_width { + Some(max_width) => max_width as usize, // u32 to usize never truncates + None => usize::MAX, + }); + options = options.set_max_height(match limits.max_image_height { + Some(max_height) => max_height as usize, // u32 to usize never truncates + None => usize::MAX, + }); zune_jpeg::JpegDecoder::new_with_options(options, &input) }