diff --git a/images/mismatch_encode.jpg b/images/mismatch_encode.jpg new file mode 100644 index 00000000..46692874 Binary files /dev/null and b/images/mismatch_encode.jpg differ diff --git a/src/helpers.rs b/src/helpers.rs index 16b3042e..474f9ffc 100644 --- a/src/helpers.rs +++ b/src/helpers.rs @@ -25,11 +25,8 @@ pub const fn u32_bit_length(v: u32) -> u8 { } #[cold] -pub fn err_exit_code(_error_code: ExitCode, message: &str) -> anyhow::Result { - return Err(anyhow::Error::new(LeptonError { - exit_code: _error_code, - message: message.to_string(), - })); +pub fn err_exit_code(error_code: ExitCode, message: &str) -> anyhow::Result { + return Err(anyhow::Error::new(LeptonError::new(error_code, message))); } pub fn buffer_prefix_matches_marker( diff --git a/src/lepton_error.rs b/src/lepton_error.rs index 1debf51b..ce462061 100644 --- a/src/lepton_error.rs +++ b/src/lepton_error.rs @@ -4,16 +4,16 @@ * This software incorporates material from third parties. See NOTICE.txt for details. *--------------------------------------------------------------------------------------------*/ -use std::fmt::Display; +use std::{fmt::Display, io::ErrorKind}; #[derive(Debug, Clone, Copy, PartialEq)] #[allow(dead_code)] - +#[non_exhaustive] /// Well-defined errors for bad things that are expected to happen as part of compression/decompression pub enum ExitCode { //AssertionFailure = 1, //CodingError = 2, - //ShortRead = 3, + ShortRead = 3, Unsupported4Colors = 4, CoefficientOutOfRange = 6, StreamInconsistent = 7, @@ -23,10 +23,13 @@ pub enum ExitCode { //ThreadingPartialMcu = 12, VersionUnsupported = 13, //OnlyGarbageNoJpeg = 14, - //OsError = 33, + OsError = 33, //HeaderTooLarge = 34, //BlockOffsetOOM = 37, UnsupportedJpeg = 42, + UnsupportedJpegWithZeroIdct0 = 43, + InvalidResetCode = 44, + InvalidPadding = 45, //WrapperOutputWriteFailed = 101, BadLeptonFile = 102, @@ -48,13 +51,13 @@ impl Display for ExitCode { } /// Standard error returned by Lepton library -#[derive(Debug)] +#[derive(Debug, Clone)] pub struct LeptonError { /// standard error code - pub exit_code: ExitCode, + exit_code: ExitCode, /// diagnostic message including location. Content should not be relied on. - pub message: String, + message: String, } impl Display for LeptonError { @@ -63,4 +66,122 @@ impl Display for LeptonError { } } +impl LeptonError { + pub fn new(exit_code: ExitCode, message: &str) -> LeptonError { + LeptonError { + exit_code, + message: message.to_owned(), + } + } + + pub fn exit_code(&self) -> ExitCode { + self.exit_code + } + + pub fn message(&self) -> &str { + &self.message + } +} + impl std::error::Error for LeptonError {} + +impl From for LeptonError { + fn from(mut error: anyhow::Error) -> Self { + // first see if there is a LeptonError already inside + match error.downcast::() { + Ok(le) => { + return le; + } + Err(old_error) => { + error = old_error; + } + } + + // capture the original error string before we lose it due + // to downcasting to look for stashed LeptonErrors + let original_string = error.to_string(); + + // see if there is a LeptonError hiding inside an io error + // which happens if we cross an API boundary that returns an std::io:Error + // like Read or Write + match error.downcast::() { + Ok(ioe) => match ioe.downcast::() { + Ok(le) => { + return le; + } + Err(e) => { + return LeptonError { + exit_code: get_io_error_exit_code(&e), + message: format!("{} {}", e, original_string), + }; + } + }, + Err(_) => {} + } + + // don't know what we got, so treat it as a general failure + return LeptonError { + exit_code: ExitCode::GeneralFailure, + message: original_string, + }; + } +} + +fn get_io_error_exit_code(e: &std::io::Error) -> ExitCode { + if e.kind() == ErrorKind::UnexpectedEof { + ExitCode::ShortRead + } else { + ExitCode::OsError + } +} + +/// translates std::io::Error into LeptonError +impl From for LeptonError { + #[track_caller] + fn from(e: std::io::Error) -> Self { + match e.downcast::() { + Ok(le) => { + return le; + } + Err(e) => { + let caller = std::panic::Location::caller(); + return LeptonError { + exit_code: get_io_error_exit_code(&e), + message: format!("error {} at {}", e.to_string(), caller.to_string()), + }; + } + } + } +} + +/// translates LeptonError into std::io::Error, which involves putting into a Box and using Other +impl From for std::io::Error { + fn from(e: LeptonError) -> Self { + return std::io::Error::new(std::io::ErrorKind::Other, e); + } +} + +#[test] +fn test_error_translation() { + // test wrapping inside an io error + fn my_std_error() -> Result<(), std::io::Error> { + Err(LeptonError::new(ExitCode::SyntaxError, "test error").into()) + } + + let e: LeptonError = my_std_error().unwrap_err().into(); + assert_eq!(e.exit_code, ExitCode::SyntaxError); + assert_eq!(e.message, "test error"); + + // wrapping inside anyhow + fn my_anyhow() -> Result<(), anyhow::Error> { + Err(LeptonError::new(ExitCode::SyntaxError, "test error").into()) + } + + let e: LeptonError = my_anyhow().unwrap_err().into(); + assert_eq!(e.exit_code, ExitCode::SyntaxError); + assert_eq!(e.message, "test error"); + + // an IO error should be translated into an OsError + let e: LeptonError = std::io::Error::new(std::io::ErrorKind::NotFound, "file not found").into(); + assert_eq!(e.exit_code, ExitCode::OsError); +} diff --git a/src/lib.rs b/src/lib.rs index 55812b71..3120e59c 100644 --- a/src/lib.rs +++ b/src/lib.rs @@ -13,44 +13,25 @@ mod structs; pub mod enabled_features; pub mod lepton_error; +use anyhow::Context; pub use enabled_features::EnabledFeatures; +use helpers::here; pub use lepton_error::{ExitCode, LeptonError}; pub use metrics::Metrics; -use crate::structs::lepton_file_reader::decode_lepton_file; -use crate::structs::lepton_file_writer::{encode_lepton_wrapper, encode_lepton_wrapper_verify}; - use core::result::Result; use std::panic::catch_unwind; use std::io::{BufRead, Cursor, Seek, Write}; -/// translates internal anyhow based exception into externally visible exception -fn translate_error(e: anyhow::Error) -> LeptonError { - match e.root_cause().downcast_ref::() { - // try to extract the exit code if it was a well known error - Some(x) => { - return LeptonError { - exit_code: x.exit_code, - message: x.message.to_owned(), - }; - } - None => { - return LeptonError { - exit_code: ExitCode::GeneralFailure, - message: format!("unexpected error {0:?}", e), - }; - } - } -} - /// Decodes Lepton container and recreates the original JPEG file pub fn decode_lepton( reader: &mut R, writer: &mut W, enabled_features: &EnabledFeatures, ) -> Result { - decode_lepton_file(reader, writer, enabled_features).map_err(translate_error) + structs::lepton_file_reader::decode_lepton_file(reader, writer, enabled_features) + .map_err(LeptonError::from) } /// Encodes JPEG as compressed Lepton format. @@ -59,7 +40,8 @@ pub fn encode_lepton( writer: &mut W, enabled_features: &EnabledFeatures, ) -> Result { - encode_lepton_wrapper(reader, writer, enabled_features).map_err(translate_error) + structs::lepton_file_writer::encode_lepton_wrapper(reader, writer, enabled_features) + .map_err(LeptonError::from) } /// Compresses JPEG into Lepton format and compares input to output to verify that compression roundtrip is OK @@ -67,7 +49,8 @@ pub fn encode_lepton_verify( input_data: &[u8], enabled_features: &EnabledFeatures, ) -> Result<(Vec, Metrics), LeptonError> { - encode_lepton_wrapper_verify(input_data, enabled_features).map_err(translate_error) + structs::lepton_file_writer::encode_lepton_wrapper_verify(input_data, enabled_features) + .map_err(LeptonError::from) } /// C ABI interface for compressing image, exposed from DLL @@ -94,18 +77,12 @@ pub unsafe extern "C" fn WrapperCompressImage( features.max_threads = number_of_threads as u32; } - match encode_lepton_wrapper(&mut reader, &mut writer, &features) { - Ok(_) => {} - Err(e) => match e.root_cause().downcast_ref::() { - // try to extract the exit code if it was a well known error - Some(x) => { - return x.exit_code as i32; - } - None => { - return -1 as i32; - } - }, - } + let _metrics = match encode_lepton(&mut reader, &mut writer, &features) { + Ok(m) => m, + Err(e) => { + return e.exit_code() as i32; + } + }; *result_size = writer.position().into(); @@ -183,27 +160,25 @@ pub unsafe extern "C" fn WrapperDecompressImageEx( let mut reader = Cursor::new(input); let mut writer = Cursor::new(output); - match decode_lepton_file(&mut reader, &mut writer, &mut enabled_features) { + match decode_lepton(&mut reader, &mut writer, &mut enabled_features) { Ok(_) => { *result_size = writer.position().into(); return 0; } Err(e) => { - let exit_code = translate_error(e).exit_code; - // The retry logic below runs if the caller did not pass use_16bit_dc_estimate=true, but the decompression // encountered StreamInconsistent failure which is commonly caused by the the C++ 16 bit bug. In this case // we retry the decompression with use_16bit_dc_estimate=true. // Note that it's prefferable for the caller to pass use_16bit_dc_estimate properly and not to rely on this // retry logic, that may miss some cases leading to bad (corrupted) decompression results. - if exit_code == ExitCode::StreamInconsistent + if e.exit_code() == ExitCode::StreamInconsistent && !enabled_features.use_16bit_dc_estimate { enabled_features.use_16bit_dc_estimate = true; continue; } - return exit_code as i32; + return e.exit_code() as i32; } } } @@ -274,7 +249,7 @@ impl LeptonFileReaderContext { ) -> Result { self.reader .process_buffer(input, input_complete, writer, output_buffer_size) - .map_err(translate_error) + .map_err(LeptonError::from) } } @@ -328,7 +303,7 @@ pub unsafe extern "C" fn decompress_image( return done as i32; } Err(e) => { - return e.exit_code as i32; + return e.exit_code() as i32; } } }) { @@ -340,3 +315,68 @@ pub unsafe extern "C" fn decompress_image( } }; } + +/// used by utility to dump out the contents of a jpeg file or lepton file for debugging purposes +#[allow(dead_code)] +pub fn dump_jpeg( + input_data: &[u8], + all: bool, + enabled_features: &EnabledFeatures, +) -> Result<(), LeptonError> { + use structs::lepton_file_reader::decode_lepton_file_image; + use structs::lepton_file_writer::read_jpeg; + + let mut lh; + let block_image; + + if input_data[0] == 0xff && input_data[1] == 0xd8 { + let mut reader = Cursor::new(input_data); + + (lh, block_image) = read_jpeg(&mut reader, enabled_features, |jh| { + println!("parsed header:"); + let s = format!("{jh:?}"); + println!("{0}", s.replace("},", "},\r\n").replace("],", "],\r\n")); + }) + .map_err(LeptonError::from)?; + } else { + let mut reader = Cursor::new(input_data); + + (lh, block_image) = + decode_lepton_file_image(&mut reader, enabled_features).context(here!())?; + + loop { + println!("parsed header:"); + let s = format!("{0:?}", lh.jpeg_header); + println!("{0}", s.replace("},", "},\r\n").replace("],", "],\r\n")); + + if !lh + .advance_next_header_segment(&enabled_features) + .context(here!())? + { + break; + } + } + } + + let s = format!("{lh:?}"); + println!("{0}", s.replace("},", "},\r\n").replace("],", "],\r\n")); + + if all { + for i in 0..block_image.len() { + println!("Component {0}", i); + let image = &block_image[i]; + for dpos in 0..image.get_block_width() * image.get_original_height() { + print!("dpos={0} ", dpos); + let block = image.get_block(dpos); + + print!("{0}", block.get_transposed_from_zigzag(0)); + for i in 1..64 { + print!(",{0}", block.get_transposed_from_zigzag(i)); + } + println!(); + } + } + } + + return Ok(()); +} diff --git a/src/main.rs b/src/main.rs index bf3a7300..62dc0eac 100644 --- a/src/main.rs +++ b/src/main.rs @@ -4,37 +4,26 @@ * This software incorporates material from third parties. See NOTICE.txt for details. *--------------------------------------------------------------------------------------------*/ -mod consts; -mod enabled_features; -mod helpers; -mod jpeg_code; -mod lepton_error; -mod metrics; -mod structs; - use anyhow; -use anyhow::Context; -use helpers::err_exit_code; -use lepton_error::{ExitCode, LeptonError}; use lepton_jpeg::metrics::CpuTimeMeasure; -use log::info; +use lepton_jpeg::{ + decode_lepton, dump_jpeg, encode_lepton, encode_lepton_verify, EnabledFeatures, ExitCode, + LeptonError, Metrics, +}; +use log::{error, info}; use simple_logger::SimpleLogger; -use structs::lepton_file_reader::{decode_lepton_file, decode_lepton_file_image}; -use structs::lepton_file_writer::{encode_lepton_wrapper_verify, read_jpeg}; #[cfg(all(target_os = "windows", feature = "use_rayon"))] use thread_priority::{set_current_thread_priority, ThreadPriority, WinAPIThreadPriority}; +use std::fs::OpenOptions; use std::time::Instant; use std::{ env, - fs::{File, OpenOptions}, - io::{stdin, stdout, BufReader, Cursor, IsTerminal, Read, Seek, Write}, + fs::File, + io::{stdin, stdout, Cursor, IsTerminal, Read, Seek, Write}, time::Duration, }; -use crate::enabled_features::EnabledFeatures; -use crate::helpers::here; - fn parse_numeric_parameter(arg: &str, name: &str) -> Option { if arg.starts_with(name) { Some(arg[name.len()..].parse::().unwrap()) @@ -43,22 +32,25 @@ fn parse_numeric_parameter(arg: &str, name: &str) -> Option { } } +#[derive(Copy, Clone, Debug)] +enum FileType { + Jpeg, + Lepton, +} + // wrap main so that errors get printed nicely without a panic -fn main_with_result() -> anyhow::Result<()> { +fn main_with_result() -> Result<(), anyhow::Error> { let args: Vec = env::args().collect(); let mut filenames = Vec::new(); let mut iterations = 1; let mut dump = false; let mut all = false; + let mut verify = true; let mut overwrite = false; let mut enabled_features = EnabledFeatures::compat_lepton_vector_read(); let mut corrupt = false; - - // only output the log if we are connected to a console (otherwise if there is redirection we would corrupt the file) - if stdout().is_terminal() { - SimpleLogger::new().init().unwrap(); - } + let mut filter_level = log::LevelFilter::Info; for i in 1..args.len() { if args[i].starts_with("-") { @@ -77,6 +69,10 @@ fn main_with_result() -> anyhow::Result<()> { } else if args[i] == "-corrupt" { // randomly corrupt the files for testing corrupt = true; + } else if args[i] == "-noverify" { + verify = false; + } else if args[i] == "-quiet" { + filter_level = log::LevelFilter::Warn; } else if args[i] == "-version" { println!( "compiled library Lepton version {}, git revision: {}", @@ -136,101 +132,61 @@ fn main_with_result() -> anyhow::Result<()> { enabled_features.use_16bit_adv_predict = true; enabled_features.use_16bit_dc_estimate = true; } else { - return err_exit_code( + return Err(LeptonError::new( ExitCode::SyntaxError, format!("unknown switch {0}", args[i]).as_str(), - ); + ) + .into()); } } else { filenames.push(args[i].as_str()); } } - if dump { - let file_in = File::open(filenames[0]).unwrap(); - - let mut reader = BufReader::new(file_in); - - let mut lh; - let block_image; - - if filenames[0].to_lowercase().ends_with(".jpg") { - (lh, block_image) = read_jpeg(&mut reader, &enabled_features, |jh| { - println!("parsed header:"); - let s = format!("{jh:?}"); - println!("{0}", s.replace("},", "},\r\n").replace("],", "],\r\n")); - }) - .context(here!())?; - } else { - (lh, block_image) = - decode_lepton_file_image(&mut reader, &enabled_features).context(here!())?; - - loop { - println!("parsed header:"); - let s = format!("{0:?}", lh.jpeg_header); - println!("{0}", s.replace("},", "},\r\n").replace("],", "],\r\n")); - - if !lh - .advance_next_header_segment(&enabled_features) - .context(here!())? - { - break; - } - } - } + // only output the log if we are connected to a console (otherwise if there is redirection we would corrupt the file) + if stdout().is_terminal() { + SimpleLogger::new().with_level(filter_level).init().unwrap(); + } - let s = format!("{lh:?}"); - println!("{0}", s.replace("},", "},\r\n").replace("],", "],\r\n")); - - if all { - for i in 0..block_image.len() { - println!("Component {0}", i); - let image = &block_image[i]; - for dpos in 0..image.get_block_width() * image.get_original_height() { - print!("dpos={0} ", dpos); - let block = image.get_block(dpos); - - print!("{0}", block.get_transposed_from_zigzag(0)); - for i in 1..64 { - print!(",{0}", block.get_transposed_from_zigzag(i)); - } - println!(); - } - } - } + if dump { + let mut file_in = File::open(filenames[0]).unwrap(); + let mut contents = Vec::new(); + file_in.read_to_end(&mut contents).unwrap(); + dump_jpeg(&contents, all, &enabled_features).unwrap(); return Ok(()); } let mut input_data = Vec::new(); if filenames.len() != 2 { if stdout().is_terminal() || stdin().is_terminal() { - return err_exit_code( + return Err(LeptonError::new( ExitCode::SyntaxError, "source and destination filename are needed or input needs to be redirected", - ); + ) + .into()); } - std::io::stdin() - .read_to_end(&mut input_data) - .context(here!())?; + std::io::stdin().read_to_end(&mut input_data)?; } else { let mut file_in = File::open(filenames[0]) - .map_err(|e| LeptonError { - exit_code: ExitCode::FileNotFound, - message: e.to_string(), - }) - .context(here!())?; + .map_err(|e| LeptonError::new(ExitCode::FileNotFound, e.to_string().as_str()))?; - file_in.read_to_end(&mut input_data).context(here!())?; + file_in.read_to_end(&mut input_data)?; } if input_data.len() < 2 { - return err_exit_code(ExitCode::BadLeptonFile, "ERROR input file too small"); + return Err(LeptonError::new(ExitCode::BadLeptonFile, "ERROR input file too small").into()); } let mut metrics; let mut output_data; + let mut original_data = Vec::new(); + + // save the data if we are going to corrupt it + if corrupt { + original_data = input_data.clone(); + } let mut overall_cpu = Duration::ZERO; @@ -243,6 +199,19 @@ fn main_with_result() -> anyhow::Result<()> { r } + // see what file type we have + let file_type = if input_data[0] == 0xff && input_data[1] == 0xd8 { + FileType::Jpeg + } else if input_data[0] == 0xcf && input_data[1] == 0x84 { + FileType::Lepton + } else { + return Err(LeptonError::new( + ExitCode::BadLeptonFile, + "ERROR input file is not a valid JPEG or Lepton file", + ) + .into()); + }; + loop { let thread_cpu = CpuTimeMeasure::new(); let walltime = Instant::now(); @@ -255,31 +224,27 @@ fn main_with_result() -> anyhow::Result<()> { input_data[r] ^= 1 << bitnumber; } - if input_data[0] == 0xff && input_data[1] == 0xd8 { - // the source is a JPEG file, so run the encoder and verify the results - (output_data, metrics) = - encode_lepton_wrapper_verify(&input_data[..], &enabled_features) - .context(here!())?; - - info!( - "compressed input {0}, output {1} bytes (ratio = {2:.1}%)", - input_data.len(), - output_data.len(), - ((input_data.len() as f64) / (output_data.len() as f64) - 1.0) * 100.0 - ); - } else if input_data[0] == 0xcf && input_data[1] == 0x84 { - // the source is a lepton file, so run the decoder - let mut reader = Cursor::new(&input_data); - - output_data = Vec::with_capacity(input_data.len()); + // do the encoding/decoding, if we got an error and were corrupting the file, then restore the + // original data and continue so we can try corrupting the file in different ways + // per iteration + match do_work(file_type, verify, &input_data, &enabled_features) { + Err(e) => { + error!("error {0}", e); + + // if we corrupted the image, then restore and continue running + if corrupt { + input_data = original_data.clone(); + output_data = Vec::new(); + metrics = Metrics::default(); + } else { + return Err(e.into()); + } + } - metrics = decode_lepton_file(&mut reader, &mut output_data, &enabled_features) - .context(here!())?; - } else { - return err_exit_code( - ExitCode::BadLeptonFile, - "ERROR input file is not a valid JPEG or Lepton file", - ); + Ok((data, m)) => { + output_data = data; + metrics = m; + } } let localthread = thread_cpu.elapsed(); @@ -301,19 +266,16 @@ fn main_with_result() -> anyhow::Result<()> { } if filenames.len() != 2 { - std::io::stdout() - .write_all(&output_data[..]) - .context(here!())? + std::io::stdout().write_all(&output_data[..])? } else { let output_file: String = filenames[1].to_owned(); let mut fileout = OpenOptions::new() .write(true) .create(overwrite) .create_new(!overwrite) - .open(output_file.as_str()) - .context(here!())?; + .open(output_file.as_str())?; - fileout.write_all(&output_data[..]).context(here!())? + fileout.write_all(&output_data[..])? } if iterations > 1 { @@ -326,6 +288,47 @@ fn main_with_result() -> anyhow::Result<()> { Ok(()) } +/// does the actual encoding/decoding work +fn do_work( + file_type: FileType, + verify: bool, + input_data: &Vec, + enabled_features: &EnabledFeatures, +) -> Result<(Vec, Metrics), LeptonError> { + let metrics; + let mut output; + + match file_type { + FileType::Jpeg => { + if verify { + (output, metrics) = encode_lepton_verify(input_data, enabled_features)?; + } else { + let mut reader = Cursor::new(input_data); + output = Vec::with_capacity(input_data.len()); + let mut writer = Cursor::new(&mut output); + + metrics = encode_lepton(&mut reader, &mut writer, enabled_features)? + } + + info!( + "compressed input {0}, output {1} bytes (compression = {2:.1}%)", + input_data.len(), + output.len(), + ((input_data.len() as f64) / (output.len() as f64) - 1.0) * 100.0 + ); + } + FileType::Lepton => { + let mut reader = Cursor::new(&input_data); + + output = Vec::with_capacity(input_data.len()); + + metrics = decode_lepton(&mut reader, &mut output, &enabled_features)?; + } + } + + Ok((output, metrics)) +} + /// internal debug utility used to figure out where in the output the JPG diverged if there was a coding error writing out the JPG struct VerifyWriter { output: W, @@ -386,9 +389,11 @@ fn main() { Some(x) => { eprintln!( "error code: {0} {1} {2}", - x.exit_code, x.exit_code as i32, x.message + x.exit_code(), + x.exit_code() as i32, + x.message() ); - std::process::exit(x.exit_code as i32); + std::process::exit(x.exit_code() as i32); } None => { eprintln!("unknown error {0:?}", e); diff --git a/src/structs/bit_reader.rs b/src/structs/bit_reader.rs index 5e84a187..60437192 100644 --- a/src/structs/bit_reader.rs +++ b/src/structs/bit_reader.rs @@ -6,6 +6,7 @@ use std::io::Read; +use crate::LeptonError; use crate::{helpers::err_exit_code, jpeg_code}; use crate::lepton_error::ExitCode; @@ -116,13 +117,15 @@ impl BitReader { } else { // verify_reset_code should get called in all instances where there should be a reset code. If we find one that // is not where it is supposed to be, then we would fail to roundtrip the reset code, so just fail. - return Err(std::io::Error::new( - std::io::ErrorKind::InvalidData, + return Err(LeptonError::new( + ExitCode::InvalidResetCode, format!( "invalid reset {0:x} {1:x} code found in stream at offset {2}", 0xff, buffer[0], self.offset - ), - )); + ) + .as_str(), + ) + .into()); } } else { self.prev_offset = self.offset; @@ -190,7 +193,7 @@ impl BitReader { *pad_bit = Some(0xff); } else { return err_exit_code( - ExitCode::UnsupportedJpeg, + ExitCode::InvalidPadding, format!( "inconsistent pad bits num_bits={0} pattern={1:b}", num_bits_to_read, actual @@ -203,7 +206,7 @@ impl BitReader { // if we already saw a padding, then it should match let expected = u16::from(x) & all_one; if actual != expected { - return err_exit_code(ExitCode::UnsupportedJpeg, format!("padding of {0} bits should be set to 1 actual={1:b} expected={2:b}", num_bits_to_read, actual, expected).as_str()); + return err_exit_code(ExitCode::InvalidPadding, format!("padding of {0} bits should be set to 1 actual={1:b} expected={2:b}", num_bits_to_read, actual, expected).as_str()); } } } @@ -221,7 +224,7 @@ impl BitReader { self.inner.read_exact(&mut h)?; if h[0] != 0xff || h[1] != (jpeg_code::RST0 + (self.cpos as u8 & 7)) { return err_exit_code( - ExitCode::UnsupportedJpeg, + ExitCode::InvalidResetCode, format!( "invalid reset code {0:x} {1:x} found in stream at offset {2}", h[0], h[1], self.offset diff --git a/src/structs/jpeg_header.rs b/src/structs/jpeg_header.rs index f0285c79..44a7885a 100644 --- a/src/structs/jpeg_header.rs +++ b/src/structs/jpeg_header.rs @@ -600,7 +600,7 @@ impl JPegHeader { { if enabled_features.reject_dqts_with_zeros { - return err_exit_code(ExitCode::UnsupportedJpeg,"DQT has zero value"); + return err_exit_code(ExitCode::UnsupportedJpegWithZeroIdct0,"DQT has zero value"); } else { break; @@ -622,7 +622,7 @@ impl JPegHeader { { if enabled_features.reject_dqts_with_zeros { - return err_exit_code(ExitCode::UnsupportedJpeg,"DQT has zero value"); + return err_exit_code(ExitCode::UnsupportedJpegWithZeroIdct0,"DQT has zero value"); } else { break; @@ -918,7 +918,7 @@ impl JPegHeader { for i in [0, 1, 2, 3, 4, 5, 6, 7, 8, 16, 24, 32, 40, 48, 56] { if qtables.get_quantization_table()[i] == 0 { return err_exit_code( - ExitCode::UnsupportedJpeg, + ExitCode::UnsupportedJpegWithZeroIdct0, "Quantization table contains zero for edge which would cause a divide by zero", ); } diff --git a/tests/end_to_end.rs b/tests/end_to_end.rs index 0294434d..525efa95 100644 --- a/tests/end_to_end.rs +++ b/tests/end_to_end.rs @@ -10,7 +10,6 @@ use std::{io::Cursor, path::Path}; use std::fs::File; use std::io::Read; -use lepton_jpeg::metrics::Metrics; use lepton_jpeg::{ create_decompression_context, decompress_image, free_decompression_context, WrapperCompressImage, WrapperDecompressImage, WrapperDecompressImageEx, @@ -319,15 +318,25 @@ fn verify_encode_verify(#[values("slrcity")] file: &str) { encode_lepton_verify(&input[..], &EnabledFeatures::compat_lepton_vector_write()).unwrap(); } -fn assert_exception(expected_error: ExitCode, result: Result) { +fn assert_exception(expected_error: ExitCode, result: Result) { match result { Ok(_) => panic!("failure was expected"), Err(e) => { - assert_eq!(expected_error, e.exit_code, "unexpected error {0:?}", e); + assert_eq!(expected_error, e.exit_code(), "unexpected error {0:?}", e); } } } +#[rstest] +fn verify_encode_verify_fail(#[values("mismatch_encode")] file: &str) { + let input = read_file(file, ".jpg"); + + assert_exception( + ExitCode::VerificationContentMismatch, + encode_lepton_verify(&input[..], &EnabledFeatures::compat_lepton_vector_write()), + ); +} + /// ensures we error out if we have the progressive flag disabled #[rstest] fn verify_encode_progressive_false( @@ -371,7 +380,7 @@ fn verify_encode_image_with_zeros_in_dqt_tables() { let mut lepton = Vec::new(); assert_exception( - ExitCode::UnsupportedJpeg, + ExitCode::UnsupportedJpegWithZeroIdct0, encode_lepton( &mut Cursor::new(&input), &mut Cursor::new(&mut lepton), @@ -473,9 +482,12 @@ fn extern_interface_decompress_chunked() { #[rstest] fn verify_extern_interface_rejects_compression_of_unsupported_jpegs( - #[values("zeros_in_dqt_tables", "nonoptimalprogressive")] file: &str, + #[values( + ("zeros_in_dqt_tables", ExitCode::UnsupportedJpegWithZeroIdct0), + ("nonoptimalprogressive", ExitCode::UnsupportedJpeg))] + file: (&str, ExitCode), ) { - let input = read_file(file, ".jpg"); + let input = read_file(file.0, ".jpg"); let mut compressed = Vec::new(); compressed.resize(input.len() + 10000, 0); @@ -491,7 +503,7 @@ fn verify_extern_interface_rejects_compression_of_unsupported_jpegs( (&mut result_size) as *mut u64, ); - assert_eq!(retval, ExitCode::UnsupportedJpeg as i32); + assert_eq!(retval, file.1 as i32); } }