From 2bd8bf9c67a2a59605ae022b16178260c779794f Mon Sep 17 00:00:00 2001 From: julianknodt Date: Tue, 18 Jul 2023 22:26:51 -0700 Subject: [PATCH] Remove some allocations from HDR image Switch to unsafe code instead of bytemuck --- src/codecs/hdr/decoder.rs | 90 +++++++++++++++++++++++++++++---------- src/codecs/openexr.rs | 5 +-- src/color.rs | 1 - 3 files changed, 69 insertions(+), 27 deletions(-) diff --git a/src/codecs/hdr/decoder.rs b/src/codecs/hdr/decoder.rs index 8329d577e1..5e6e2cba49 100644 --- a/src/codecs/hdr/decoder.rs +++ b/src/codecs/hdr/decoder.rs @@ -156,14 +156,7 @@ impl HdrAdapter { fn read_image_data(&mut self, buf: &mut [u8]) -> ImageResult<()> { assert_eq!(u64::try_from(buf.len()), Ok(self.total_bytes())); match self.inner.take() { - Some(decoder) => { - let img: Vec> = decoder.read_image_ldr()?; - for (i, Rgb(data)) in img.into_iter().enumerate() { - buf[(i * 3)..][..3].copy_from_slice(&data); - } - - Ok(()) - } + Some(decoder) => decoder.read_image_ldr_buf(buf), None => Err(ImageError::Parameter(ParameterError::from_kind( ParameterErrorKind::NoMoreData, ))), @@ -247,6 +240,17 @@ pub struct HdrDecoder { meta: HdrMetadata, } +impl HdrDecoder { + /// The width of this HDR decoder + pub fn width(&self) -> u32 { + self.width + } + /// The height of this HDR decoder + pub fn height(&self) -> u32 { + self.height + } +} + /// Refer to [wikipedia](https://en.wikipedia.org/wiki/RGBE_image_format) #[repr(C)] #[derive(Clone, Copy, Debug, Default, PartialEq, Eq)] @@ -443,11 +447,14 @@ impl HdrDecoder { } /// Consumes decoder and returns a vector of transformed pixels - pub fn read_image_transform T>( + pub fn read_image_transform T>( mut self, f: F, output_slice: &mut [T], - ) -> ImageResult<()> { + ) -> ImageResult<()> + where + T: Send + Copy, + { assert_eq!( output_slice.len(), self.width as usize * self.height as usize @@ -472,20 +479,58 @@ impl HdrDecoder { Ok(()) } + /// Consumes decoder and returns a vector of transformed pixels + fn read_image_transform_flat [T; N], const N: usize>( + mut self, + f: F, + output_slice: &mut [T], + ) -> ImageResult<()> + where + T: Send + Copy, + { + assert_eq!( + output_slice.len(), + self.width as usize * self.height as usize * N + ); + + // Don't read anything if image is empty + if self.width == 0 || self.height == 0 { + return Ok(()); + } + + let chunks_iter = output_slice.chunks_mut(self.width as usize * N); + + let mut buf = vec![Default::default(); self.width as usize]; + for chunk in chunks_iter { + // read_scanline overwrites the entire buffer or returns an Err, + // so not resetting the buffer here is ok. + read_scanline(&mut self.r, &mut buf[..])?; + for (dst, &pix) in chunk.chunks_mut(N).zip(buf.iter()) { + dst.copy_from_slice(&f(pix)); + } + } + Ok(()) + } + /// Consumes decoder and returns a vector of `Rgb` pixels. /// scale = 1, gamma = 2.2 pub fn read_image_ldr(self) -> ImageResult>> { - let mut ret = vec![Rgb([0, 0, 0]); self.width as usize * self.height as usize]; - self.read_image_transform(|pix| pix.to_ldr(), &mut ret[..])?; - Ok(ret) + let sz = (self.width * self.height) as usize; + let mut buf = vec![Rgb([0; 3]); sz]; + self.read_image_transform(|pix| pix.to_ldr(), &mut buf[..])?; + Ok(buf) + } + fn read_image_ldr_buf(self, buf: &mut [u8]) -> ImageResult<()> { + let sz = (self.width * self.height * 3) as usize; + assert!(buf.len() >= sz); + self.read_image_transform_flat(|pix| pix.to_ldr().0, &mut buf[..sz]) } - /// Consumes decoder and returns a vector of `Rgb` pixels. - /// pub fn read_image_hdr(self) -> ImageResult>> { - let mut ret = vec![Rgb([0.0, 0.0, 0.0]); self.width as usize * self.height as usize]; - self.read_image_transform(|pix| pix.to_hdr(), &mut ret[..])?; - Ok(ret) + let sz = (self.width * self.height) as usize; + let mut buf = vec![Rgb([0.0f32; 3]); sz]; + self.read_image_transform(|pix| pix.to_hdr(), &mut buf[..])?; + Ok(buf) } } @@ -971,15 +1016,14 @@ fn split_at_first_test() { // or return None to indicate end of file fn read_line_u8(r: &mut R) -> ::std::io::Result>> { let mut ret = Vec::with_capacity(16); - match r.read_until(b'\n', &mut ret) { - Ok(0) => Ok(None), - Ok(_) => { - if let Some(&b'\n') = ret[..].last() { + match r.read_until(b'\n', &mut ret)? { + 0 => Ok(None), + _ => { + if let Some(&b'\n') = ret.last() { let _ = ret.pop(); } Ok(Some(ret)) } - Err(err) => Err(err), } } diff --git a/src/codecs/openexr.rs b/src/codecs/openexr.rs index 52d6ba9421..724c15b2df 100644 --- a/src/codecs/openexr.rs +++ b/src/codecs/openexr.rs @@ -467,12 +467,11 @@ mod test { .clone() .join("overexposed gradient - data window equals display window.exr"); - let hdr: Vec> = crate::codecs::hdr::HdrDecoder::new(std::io::BufReader::new( + let hdr_decoder = crate::codecs::hdr::HdrDecoder::new(std::io::BufReader::new( std::fs::File::open(&reference_path).unwrap(), )) - .unwrap() - .read_image_hdr() .unwrap(); + let hdr = hdr_decoder.read_image_hdr().unwrap(); let exr_pixels: Rgb32FImage = read_as_rgb_image_from_file(exr_path).unwrap(); assert_eq!( diff --git a/src/color.rs b/src/color.rs index 57a8511f2c..8426eb546e 100644 --- a/src/color.rs +++ b/src/color.rs @@ -349,7 +349,6 @@ impl From<[T; $channels]> for $ident { Self(c) } } - )* // END Structure definitions }