From 94847fa6c4ed8bfea6b81498c68a2791fd5514fc 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 --- src/codecs/hdr/decoder.rs | 64 ++++++++++++++++++++++++--------------- src/codecs/openexr.rs | 5 ++- src/color.rs | 1 - 3 files changed, 41 insertions(+), 29 deletions(-) diff --git a/src/codecs/hdr/decoder.rs b/src/codecs/hdr/decoder.rs index 8329d577e1..c07dad6ec8 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, ))), @@ -444,13 +437,28 @@ impl HdrDecoder { /// Consumes decoder and returns a vector of transformed pixels pub fn read_image_transform T>( + self, + f: F, + output_slice: &mut [T], + ) -> ImageResult<()> { + // a bit hacky, but this function was in the public API and + // thus needs to preserve it's signature. + self.read_image_transform_flat(move |v| [f(v)], output_slice) + } + + /// Consumes decoder and returns a vector of transformed pixels + fn read_image_transform_flat< + T: Send, + F: Send + Sync + Fn(Rgbe8Pixel) -> [T; N], + const N: usize, + >( mut self, f: F, output_slice: &mut [T], ) -> ImageResult<()> { assert_eq!( output_slice.len(), - self.width as usize * self.height as usize + self.width as usize * self.height as usize * N ); // Don't read anything if image is empty @@ -458,15 +466,17 @@ impl HdrDecoder { return Ok(()); } - let chunks_iter = output_slice.chunks_mut(self.width as usize); + 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.iter_mut().zip(buf.iter()) { - *dst = f(pix); + for (dst, &pix) in chunk.chunks_mut(N).zip(buf.iter()) { + for (d, s) in dst.iter_mut().zip(IntoIterator::into_iter(f(pix))) { + *d = s; + } } } Ok(()) @@ -475,17 +485,22 @@ impl HdrDecoder { /// 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 +986,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 }