Add support to decode JPEG-LS images with restart markers (#99)

* Add support to parse DRI segments

* Add support to decode restart markers

* Update CharLSTest to also decode color images none interleaved to ppm

* Replace static_cast<void> with std::ignore

* Update readme, changelog and refactor
This commit is contained in:
Victor Derks 2021-09-26 22:21:07 +02:00 committed by GitHub
parent 3239c6fe80
commit cb44fab24d
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23
47 changed files with 851 additions and 316 deletions

View File

@ -19,6 +19,7 @@
# -clang-diagnostic-switch-enum => Rationale: options are handled by default case
# -clang-diagnostic-exit-time-destructors => Rationale: Acceptable construction
# -clang-diagnostic-pragma-once-outside-header => Rationale: Generates false warnings for usage in header files
# -clang-diagnostic-unused-const-variable => Rationale: false warnings for constexpr in .h files
# -clang-analyzer-core.NonNullParamChecker => Rationale: cannot be effective disabled, already checked by other checkers.
# -misc-non-private-member-variables-in-classes => Rationale: design can be ok, manual review is better
# -modernize-use-trailing-return-type => Rationale: A style recommendation, this style is selected for CharLS
@ -58,6 +59,7 @@ Checks: '*,
-clang-diagnostic-switch-enum,
-clang-diagnostic-exit-time-destructors,
-clang-diagnostic-pragma-once-outside-header,
-clang-diagnostic-unused-const-variable,
-clang-analyzer-core.NonNullParamChecker,
-misc-non-private-member-variables-in-classes,
-modernize-use-trailing-return-type,

View File

@ -6,6 +6,11 @@ The format is based on [Keep a Changelog](http://keepachangelog.com/) and this p
## [Next-Release]
### Added
- The encoder API has been extended with a rewind method that can be used to re-use a configered encoder to encode multiple images in a loop.
- Added support to decode JPEG-LS images that use restart markers [#92](https://github.com/team-charls/charls/issues/92)
### Fixed
- Fixed [#84](https://github.com/team-charls/charls/issues/84), Default preset coding parameters not computed for unset values.
@ -13,7 +18,6 @@ The format is based on [Keep a Changelog](http://keepachangelog.com/) and this p
### Changed
- CMakeSettings.json has been replaced with CMakePresets.json.
- The encoder API has been extended with a rewind method that can be used to re-use a configered encoder to encode multiple images in a loop.
## [2.2.0] - 2021-1-10

View File

@ -6,6 +6,7 @@
<s:String x:Key="/Default/CodeInspection/Highlighting/InspectionSeverities/=CppClangTidyClangDiagnosticExitTimeDestructors/@EntryIndexedValue">DO_NOT_SHOW</s:String>
<s:String x:Key="/Default/CodeInspection/Highlighting/InspectionSeverities/=CppClangTidyClangDiagnosticMissingBraces/@EntryIndexedValue">DO_NOT_SHOW</s:String>
<s:String x:Key="/Default/CodeInspection/Highlighting/InspectionSeverities/=CppClangTidyClangDiagnosticSwitchEnum/@EntryIndexedValue">DO_NOT_SHOW</s:String>
<s:String x:Key="/Default/CodeInspection/Highlighting/InspectionSeverities/=CppClangTidyClangDiagnosticUnusedConstVariable/@EntryIndexedValue">DO_NOT_SHOW</s:String>
<s:String x:Key="/Default/CodeInspection/Highlighting/InspectionSeverities/=CppClangTidyCppcoreguidelinesAvoidMagicNumbers/@EntryIndexedValue">DO_NOT_SHOW</s:String>
<s:String x:Key="/Default/CodeInspection/Highlighting/InspectionSeverities/=CppClangTidyCppcoreguidelinesInitVariables/@EntryIndexedValue">DO_NOT_SHOW</s:String>
<s:String x:Key="/Default/CodeInspection/Highlighting/InspectionSeverities/=CppClangTidyCppcoreguidelinesMacroUsage/@EntryIndexedValue">DO_NOT_SHOW</s:String>
@ -46,6 +47,7 @@
<s:Boolean x:Key="/Default/UserDictionary/Words/=bugprone/@EntryIndexedValue">True</s:Boolean>
<s:Boolean x:Key="/Default/UserDictionary/Words/=CHARLS/@EntryIndexedValue">True</s:Boolean>
<s:Boolean x:Key="/Default/UserDictionary/Words/=charlstest/@EntryIndexedValue">True</s:Boolean>
<s:Boolean x:Key="/Default/UserDictionary/Words/=cmove/@EntryIndexedValue">True</s:Boolean>
<s:Boolean x:Key="/Default/UserDictionary/Words/=cmyk/@EntryIndexedValue">True</s:Boolean>
<s:Boolean x:Key="/Default/UserDictionary/Words/=cpixel/@EntryIndexedValue">True</s:Boolean>
<s:Boolean x:Key="/Default/UserDictionary/Words/=cppcoreguidelines/@EntryIndexedValue">True</s:Boolean>

View File

@ -34,8 +34,8 @@ According to preliminary test results published on <http://imagecompression.info
The following JPEG-LS options are not supported by the CharLS implementation. Most of these options are rarely used in practice.
* No support for JPEG restart markers (RST).
Restart markers make it possible to recover from corrupted JPEG files, but are seldom used for data recovery scenarios.
* No support to encode JPEG restart markers
Decoding is supported, but no recovery mechanisme is implemented for corrupted JPEG-LS files.
* No support for sub-sampled scans.
Sub-sampling is a lossly encoding mechanism and not used in lossless scenarios.
* No support for multi component frames with mixed component counts in a single scan.

View File

@ -50,6 +50,8 @@ enum charls_jpegls_errc
CHARLS_JPEGLS_ERRC_INVALID_JPEGLS_PRESET_PARAMETER_TYPE = 22,
CHARLS_JPEGLS_ERRC_JPEGLS_PRESET_EXTENDED_PARAMETER_TYPE_NOT_SUPPORTED = 23,
CHARLS_JPEGLS_ERRC_MISSING_END_OF_SPIFF_DIRECTORY = 24,
CHARLS_JPEGLS_ERRC_UNEXPECTED_RESTART_MARKER = 25,
CHARLS_JPEGLS_ERRC_RESTART_MARKER_NOT_FOUND = 26,
CHARLS_JPEGLS_ERRC_INVALID_ARGUMENT_WIDTH = 100,
CHARLS_JPEGLS_ERRC_INVALID_ARGUMENT_HEIGHT = 101,
CHARLS_JPEGLS_ERRC_INVALID_ARGUMENT_COMPONENT_COUNT = 102,
@ -285,6 +287,16 @@ enum class CHARLS_NO_DISCARD jpegls_errc
/// </summary>
missing_end_of_spiff_directory = impl::CHARLS_JPEGLS_ERRC_MISSING_END_OF_SPIFF_DIRECTORY,
/// <summary>
/// This error is returned when a restart marker is found outside the encoded entropy data.
/// </summary>
unexpected_restart_marker = impl::CHARLS_JPEGLS_ERRC_UNEXPECTED_RESTART_MARKER,
/// <summary>
/// This error is returned when an expected restart marker is not found. It may indicate data corruption in the JPEG-LS byte stream.
/// </summary>
restart_marker_not_found = impl::CHARLS_JPEGLS_ERRC_RESTART_MARKER_NOT_FOUND,
/// <summary>
/// The argument for the width parameter is outside the range [1, 65535].
/// </summary>

View File

@ -210,7 +210,7 @@ private:
component_count};
const auto codec{jls_codec_factory<encoder_strategy>().create_codec(
frame_info, {near_lossless_, interleave_mode_, color_transformation_, false}, validated_pc_parameters_)};
frame_info, {near_lossless_, 0, interleave_mode_, color_transformation_, false}, validated_pc_parameters_)};
std::unique_ptr<process_line> process_line(codec->create_process_line(source, stride));
const size_t bytes_written{codec->encode_scan(move(process_line), writer_.remaining_destination())};

View File

@ -10,6 +10,7 @@ namespace charls {
struct coding_parameters final
{
int32_t near_lossless;
uint32_t restart_interval;
charls::interleave_mode interleave_mode;
color_transformation transformation;
bool output_bgr;

View File

@ -64,7 +64,7 @@ struct jls_context final
{
b = -n + 1;
}
C = C - (C > -128);
C = C - static_cast<int16_t>(C > -128);
}
else if (b > 0)
{
@ -73,7 +73,7 @@ struct jls_context final
{
b = 0;
}
C = C + (C < 127);
C = C + static_cast<int16_t>(C < 127);
}
B = b;

View File

@ -30,7 +30,7 @@ public:
decoder_strategy& operator=(decoder_strategy&&) = delete;
virtual std::unique_ptr<process_line> create_process_line(byte_span destination, size_t stride) = 0;
virtual void set_presets(const jpegls_pc_parameters& preset_coding_parameters) = 0;
virtual void set_presets(const jpegls_pc_parameters& preset_coding_parameters, uint32_t restart_interval) = 0;
virtual void decode_scan(std::unique_ptr<process_line> output_data, const JlsRect& size, byte_span& compressed_data) = 0;
void initialize(const byte_span source)
@ -45,6 +45,15 @@ public:
make_valid();
}
void reset()
{
valid_bits_ = 0;
read_cache_ = 0;
next_ff_position_ = find_next_ff();
make_valid();
}
FORCE_INLINE void skip(const int32_t length) noexcept
{
valid_bits_ -= length;
@ -246,6 +255,14 @@ public:
return (read_value(length - 24) << 24) + read_value(24);
}
uint8_t read_byte() noexcept
{
// TODO: check end_position first.
const uint8_t value = *position_;
++position_;
return value;
}
protected:
frame_info frame_info_;
coding_parameters parameters_;

View File

@ -25,7 +25,7 @@ public:
encoder_strategy& operator=(encoder_strategy&&) = delete;
virtual std::unique_ptr<process_line> create_process_line(byte_span stream_info, size_t stride) = 0;
virtual void set_presets(const jpegls_pc_parameters& preset_coding_parameters) = 0;
virtual void set_presets(const jpegls_pc_parameters& preset_coding_parameters, uint32_t restart_interval) = 0;
virtual size_t encode_scan(std::unique_ptr<process_line> raw_data, byte_span destination) = 0;
int32_t peek_byte();

View File

@ -16,12 +16,15 @@ namespace charls {
// 0x4F - 0x6F, 0x90 - 0x93 are defined in ISO/IEC 15444-1: JPEG 2000
constexpr uint8_t jpeg_marker_start_byte{0xFF};
constexpr uint8_t jpeg_restart_marker_base{0xD0}; // RSTm: Marks the next restart interval (range is D0..D7)
constexpr uint32_t jpeg_restart_marker_range{8};
enum class jpeg_marker_code : uint8_t
{
start_of_image = 0xD8, // SOI: Marks the start of an image.
end_of_image = 0xD9, // EOI: Marks the end of an image.
start_of_scan = 0xDA, // SOS: Marks the start of scan.
start_of_image = 0xD8, // SOI: Marks the start of an image.
end_of_image = 0xD9, // EOI: Marks the end of an image.
start_of_scan = 0xDA, // SOS: Marks the start of scan.
define_restart_interval = 0xDD, // DRI: Defines the restart interval used in succeeding scans.
// The following markers are defined in ISO/IEC 10918-1 | ITU T.81.
start_of_frame_baseline_jpeg = 0xC0, // SOF_0: Marks the start of a baseline jpeg encoded frame.
@ -55,8 +58,8 @@ enum class jpeg_marker_code : uint8_t
application_data10 = 0xEA, // APP10: Application data 10.
application_data11 = 0xEB, // APP11: Application data 11.
application_data12 = 0xEC, // APP12: Application data 12: used for Picture info.
application_data13 = 0xEE, // APP13: Application data 13: used by PhotoShop IRB
application_data14 = 0xED, // APP14: Application data 14: used by Adobe
application_data13 = 0xED, // APP13: Application data 13: used by PhotoShop IRB
application_data14 = 0xEE, // APP14: Application data 14: used by Adobe
application_data15 = 0xEF, // APP15: Application data 15.
comment = 0xFE // COM: Comment block.
};

View File

@ -14,6 +14,7 @@
#include <algorithm>
#include <array>
#include <memory>
#include <tuple>
namespace charls {
@ -24,6 +25,16 @@ using std::find;
using std::unique_ptr;
using std::vector;
namespace {
constexpr bool is_restart_marker_code(const jpeg_marker_code marker_code) noexcept
{
return static_cast<uint8_t>(marker_code) >= jpeg_restart_marker_base &&
static_cast<uint8_t>(marker_code) < jpeg_restart_marker_base + jpeg_restart_marker_range;
}
} // namespace
jpeg_stream_reader::jpeg_stream_reader(const byte_span source) noexcept : source_{source}
{
}
@ -212,6 +223,7 @@ void jpeg_stream_reader::validate_marker_code(const jpeg_marker_code marker_code
return;
case jpeg_marker_code::define_restart_interval:
case jpeg_marker_code::jpegls_preset_parameters:
case jpeg_marker_code::comment:
case jpeg_marker_code::application_data0:
@ -253,6 +265,9 @@ void jpeg_stream_reader::validate_marker_code(const jpeg_marker_code marker_code
throw_jpegls_error(jpegls_errc::unexpected_end_of_image_marker);
}
if (is_restart_marker_code(marker_code))
throw_jpegls_error(jpegls_errc::unexpected_restart_marker);
throw_jpegls_error(jpegls_errc::unknown_jpeg_marker_found);
}
@ -283,6 +298,9 @@ int jpeg_stream_reader::read_marker_segment(const jpeg_marker_code marker_code,
case jpeg_marker_code::jpegls_preset_parameters:
return read_preset_parameters_segment(segment_size);
case jpeg_marker_code::define_restart_interval:
return read_define_restart_interval(segment_size);
case jpeg_marker_code::application_data0:
case jpeg_marker_code::application_data1:
case jpeg_marker_code::application_data2:
@ -422,6 +440,30 @@ int jpeg_stream_reader::read_preset_parameters_segment(const int32_t segment_siz
}
int jpeg_stream_reader::read_define_restart_interval(const int32_t segment_size)
{
// Note: The JPEG-LS standard supports a 2,3 or 4 byte restart interval (see ISO/IEC 14495-1, C.2.5)
// The original JPEG standard only supports 2 bytes (16 bit big endian).
switch (segment_size)
{
case 2:
parameters_.restart_interval = read_uint16();
return 2;
case 3:
parameters_.restart_interval = read_uint24();
return 3;
case 4:
parameters_.restart_interval = read_uint32();
return 4;
default:
throw_jpegls_error(jpegls_errc::invalid_marker_segment_size);
}
}
void jpeg_stream_reader::read_start_of_scan()
{
const int32_t segment_size{read_segment_size()};
@ -472,16 +514,24 @@ uint8_t jpeg_stream_reader::read_byte()
void jpeg_stream_reader::skip_byte()
{
static_cast<void>(read_byte());
std::ignore = read_byte();
}
uint16_t jpeg_stream_reader::read_uint16()
{
const uint16_t value = read_byte() * 256U;
const uint32_t value{read_byte() * 256U};
return static_cast<uint16_t>(value + read_byte());
}
uint32_t jpeg_stream_reader::read_uint24()
{
const uint32_t value{static_cast<uint32_t>(read_byte()) << 16U};
return value + read_uint16();
}
uint32_t jpeg_stream_reader::read_uint32()
{
uint32_t value{read_uint16()};
@ -491,6 +541,7 @@ uint32_t jpeg_stream_reader::read_uint32()
return value;
}
int32_t jpeg_stream_reader::read_segment_size()
{
const int32_t segment_size{read_uint16()};

View File

@ -60,6 +60,7 @@ public:
private:
void skip_byte();
uint16_t read_uint16();
uint32_t read_uint24();
uint32_t read_uint32();
int32_t read_segment_size();
std::vector<uint8_t> read_bytes(size_t byte_count);
@ -74,6 +75,7 @@ private:
int read_start_of_frame_segment(int32_t segment_size);
static int read_comment() noexcept;
int read_preset_parameters_segment(int32_t segment_size);
int read_define_restart_interval(int32_t segment_size);
int try_read_application_data8_segment(int32_t segment_size, spiff_header* header, bool* spiff_header_found);
int try_read_spiff_header_segment(OUT_ spiff_header& header, OUT_ bool& spiff_header_found);

View File

@ -111,7 +111,7 @@ void jpeg_stream_writer::write_start_of_frame_segment(const uint32_t width, cons
void jpeg_stream_writer::write_color_transform_segment(const color_transformation transformation)
{
array<uint8_t, 5> segment{'m', 'r', 'f', 'x', static_cast<uint8_t>(transformation)};
const array<uint8_t, 5> segment{'m', 'r', 'f', 'x', static_cast<uint8_t>(transformation)};
write_segment_header(jpeg_marker_code::application_data8, segment.size());
write_bytes(segment.data(), segment.size());

View File

@ -125,7 +125,7 @@ unique_ptr<Strategy> jls_codec_factory<Strategy>::create_codec(const frame_info&
}
}
codec->set_presets(preset_coding_parameters);
codec->set_presets(preset_coding_parameters, parameters.restart_interval);
return codec;
}

View File

@ -102,6 +102,12 @@ const char* CHARLS_API_CALLING_CONVENTION charls_get_error_message(const charls_
case jpegls_errc::missing_end_of_spiff_directory:
return "Invalid JPEG-LS stream, SPIFF header without End Of Directory (EOD) entry";
case jpegls_errc::unexpected_restart_marker:
return "Invalid JPEG-LS stream, restart (RTSm) marker found outside encoded entropy data";
case jpegls_errc::restart_marker_not_found:
return "Invalid JPEG-LS stream, missing expected restart (RTSm) marker";
case jpegls_errc::invalid_parameter_bits_per_sample:
return "Invalid JPEG-LS stream, The bit per sample (sample precision) parameter is not in the range [2, 16]";

View File

@ -8,6 +8,7 @@
#include "constants.h"
#include "context.h"
#include "context_run_mode.h"
#include "jpeg_marker_code.h"
#include "lookup_table.h"
#include "process_line.h"
@ -175,9 +176,10 @@ public:
}
private:
void set_presets(const jpegls_pc_parameters& presets) override
void set_presets(const jpegls_pc_parameters& presets, const uint32_t restart_interval) override
{
initialize_parameters(presets.threshold1, presets.threshold2, presets.threshold3, presets.reset_value);
restart_interval_ = restart_interval;
}
bool is_interleaved() noexcept
@ -475,7 +477,15 @@ private:
rect_ = rect;
Strategy::initialize(compressed_data);
do_scan();
// Process images without a restart interval, as 1 large restart interval.
if (restart_interval_ == 0)
{
restart_interval_ = frame_info().height;
}
decode_lines();
skip_bytes(compressed_data, static_cast<size_t>(Strategy::get_cur_byte_pos() - compressed_bytes));
}
@ -491,17 +501,22 @@ private:
t1_ = t1;
t2_ = t2;
t3_ = t3;
reset_threshold_ = reset_threshold;
initialize_quantization_lut();
reset_parameters();
}
void reset_parameters() noexcept
{
const jls_context context_initial_value(std::max(2, (traits_.range + 32) / 64));
for (auto& context : contexts_)
{
context = context_initial_value;
}
context_runmode_[0] = context_run_mode(0, std::max(2, (traits_.range + 32) / 64), reset_threshold);
context_runmode_[1] = context_run_mode(1, std::max(2, (traits_.range + 32) / 64), reset_threshold);
context_runmode_[0] = context_run_mode(0, std::max(2, (traits_.range + 32) / 64), reset_threshold_);
context_runmode_[1] = context_run_mode(1, std::max(2, (traits_.range + 32) / 64), reset_threshold_);
run_index_ = 0;
}
@ -515,7 +530,7 @@ private:
return frame;
}
// do_scan: Encodes or decodes a scan.
// do_scan: Encodes a scan.
// In ILV_SAMPLE mode, multiple components are handled in do_line
// In ILV_LINE mode, a call do do_line is made for every component
// In ILV_NONE mode, do_scan is called for each component
@ -564,6 +579,84 @@ private:
Strategy::end_scan();
}
void decode_lines()
{
const uint32_t pixel_stride{width_ + 4U};
const size_t component_count{
parameters().interleave_mode == interleave_mode::line ? static_cast<size_t>(frame_info().component_count) : 1U};
std::vector<pixel_type> line_buffer(static_cast<size_t>(2) * component_count * pixel_stride);
std::vector<int32_t> run_index(component_count);
for (uint32_t line{};;)
{
const uint32_t lines_in_interval{std::min(frame_info().height - line, restart_interval_)};
for (uint32_t mcu{}; mcu < lines_in_interval; ++mcu, ++line)
{
previous_line_ = &line_buffer[1];
current_line_ = &line_buffer[1 + static_cast<size_t>(component_count) * pixel_stride];
if ((line & 1) == 1)
{
std::swap(previous_line_, current_line_);
}
Strategy::on_line_begin(width_, current_line_, pixel_stride);
for (size_t component{}; component < component_count; ++component)
{
run_index_ = run_index[component];
// initialize edge pixels used for prediction
previous_line_[width_] = previous_line_[width_ - 1];
current_line_[-1] = previous_line_[0];
do_line(static_cast<pixel_type*>(nullptr)); // dummy argument for overload resolution
run_index[component] = run_index_;
previous_line_ += pixel_stride;
current_line_ += pixel_stride;
}
if (static_cast<uint32_t>(rect_.Y) <= line && line < static_cast<uint32_t>(rect_.Y + rect_.Height))
{
Strategy::on_line_end(rect_.Width,
current_line_ + rect_.X - (static_cast<size_t>(component_count) * pixel_stride),
pixel_stride);
}
}
if (line == frame_info().height)
break;
read_restart_marker();
restart_interval_counter_ = (restart_interval_counter_ + 1) % jpeg_restart_marker_range;
// After a restart marker it is required to reset the decoder.
Strategy::reset();
std::fill(line_buffer.begin(), line_buffer.end(), pixel_type{});
std::fill(run_index.begin(), run_index.end(), 0);
reset_parameters();
}
Strategy::end_scan();
}
void read_restart_marker()
{
auto byte{Strategy::read_byte()};
if (byte != jpeg_marker_start_byte)
impl::throw_jpegls_error(jpegls_errc::restart_marker_not_found);
// Read all preceding 0xFF fill values until a non 0xFF value has been found. (see T.81, B.1.1.2)
do
{
byte = Strategy::read_byte();
} while (byte == jpeg_marker_start_byte);
if (byte != jpeg_restart_marker_base + restart_interval_counter_)
impl::throw_jpegls_error(jpegls_errc::restart_marker_not_found);
}
/// <summary>Encodes/Decodes a scan line of quads in ILV_SAMPLE mode</summary>
void do_line(quad<sample_type>*)
{
@ -826,6 +919,9 @@ private:
int32_t t1_{};
int32_t t2_{};
int32_t t3_{};
int32_t reset_threshold_{};
uint32_t restart_interval_{};
uint32_t restart_interval_counter_{};
// compression context
std::array<jls_context, 365> contexts_;

View File

@ -285,7 +285,7 @@ inline void skip_bytes(byte_span& stream_info, const size_t count) noexcept
template<typename T>
std::ostream& operator<<(typename std::enable_if<std::is_enum<T>::value, std::ostream>::type& stream, const T& e)
std::ostream& operator<<(std::enable_if_t<std::is_enum<T>::value, std::ostream>& stream, const T& e)
{
return stream << static_cast<typename std::underlying_type<T>::type>(e);
}

View File

@ -7,6 +7,7 @@
#include <array>
#include <cstring>
#include <iostream>
#include <tuple>
#include <vector>
using std::array;
@ -48,7 +49,7 @@ bool verify_encoded_bytes(const void* uncompressed_data, const size_t uncompress
encoder.interleave_mode(decoder.interleave_mode());
encoder.near_lossless(decoder.near_lossless());
encoder.preset_coding_parameters(decoder.preset_coding_parameters());
static_cast<void>(encoder.encode(uncompressed_data, uncompressed_length));
std::ignore = encoder.encode(uncompressed_data, uncompressed_length);
for (size_t i{}; i != compressed_length; ++i)
{

View File

@ -50,7 +50,7 @@ void test_dicom_sample_image(const char* name)
data.erase(data.begin(), data.begin() + offset - 4);
// remove the DICOM fragment headers (in the concerned images they occur every 64k)
for (unsigned int i{}; i != data.size(); i += 64 * 1024)
for (unsigned int i{}; i < data.size(); i += 64 * 1024)
{
data.erase(data.begin() + static_cast<int>(i), data.begin() + static_cast<int>(i) + 8);
}

View File

@ -24,7 +24,7 @@ void test_jpegls_read_header(const char* filename, const int width, const int he
{
cout << "LegacyAPI JpegLsReadHeader:" << filename << "\n";
vector<uint8_t> encoded_buffer = read_file(filename);
const vector<uint8_t> encoded_buffer{read_file(filename)};
array<char, ErrorMessageSize> error_message{};
JlsParameters parameters{};

View File

@ -47,12 +47,33 @@ namespace {
constexpr ios::openmode mode_input = ios::in | ios::binary;
constexpr ios::openmode mode_output = ios::out | ios::binary;
ifstream open_input_stream(const char* filename)
{
ifstream stream;
stream.exceptions(ios::eofbit | ios::failbit | ios::badbit);
stream.open(filename, mode_input);
return stream;
}
ofstream open_output_stream(const char* filename)
{
ofstream stream;
stream.exceptions(ios::eofbit | ios::failbit | ios::badbit);
stream.open(filename, mode_output);
return stream;
}
template<typename Container>
void read(istream& input, Container& destination_container)
{
input.read(reinterpret_cast<char*>(destination_container.data()), destination_container.size());
}
size_t get_stream_length(istream& stream, const size_t end_offset = 0)
{
stream.seekg(0, ios::end);
@ -62,6 +83,34 @@ size_t get_stream_length(istream& stream, const size_t end_offset = 0)
return length;
}
template<typename SizeType>
void convert_planar_to_pixel(const size_t width, const size_t height, const void* source, void* destination) noexcept
{
const size_t stride_in_pixels = width * 3;
const auto* plane0{static_cast<const SizeType*>(source)};
const auto* plane1{static_cast<const SizeType*>(source) + (width * height)};
const auto* plane2{static_cast<const SizeType*>(source) + (width * height * 2)};
auto* pixels{static_cast<SizeType*>(destination)};
for (size_t row{}; row != height; ++row)
{
for (size_t column{}, offset = 0; column != width; ++column, offset += 3)
{
pixels[offset + 0] = plane0[column];
pixels[offset + 1] = plane1[column];
pixels[offset + 2] = plane2[column];
}
plane0 += width;
plane1 += width;
plane2 += width;
pixels += stride_in_pixels;
}
}
void test_traits16_bit()
{
const auto traits1 = default_traits<uint16_t, uint16_t>(4095, 0);
@ -149,7 +198,7 @@ vector<uint8_t> make_some_noise16_bit(const size_t length, const int bit_count,
void test_noise_image()
{
const rect_size size2 = rect_size(512, 512);
const rect_size size2{512, 512};
for (size_t bit_depth{8}; bit_depth >= 2; --bit_depth)
{
@ -200,7 +249,7 @@ void test_fail_on_too_small_output_buffer()
jpegls_encoder encoder;
encoder.destination(output_buffer);
encoder.frame_info({8, 8, 8, 1});
static_cast<void>(encoder.encode(input_buffer));
std::ignore = encoder.encode(input_buffer);
assert::is_true(false);
}
catch (const jpegls_error& e)
@ -215,7 +264,7 @@ void test_fail_on_too_small_output_buffer()
jpegls_encoder encoder;
encoder.destination(output_buffer);
encoder.frame_info({8, 8, 8, 1});
static_cast<void>(encoder.encode(input_buffer));
std::ignore = encoder.encode(input_buffer);
assert::is_true(false);
}
catch (const jpegls_error& e)
@ -238,7 +287,7 @@ void test_bgra()
void test_bgr()
{
vector<uint8_t> encoded_source = read_file("test/conformance/t8c2e3.jls");
const vector<uint8_t> encoded_source{read_file("test/conformance/t8c2e3.jls")};
jpegls_decoder decoder;
decoder.source(encoded_source);
@ -364,24 +413,21 @@ void test_decode_bit_stream_with_unknown_jpeg_marker()
void test_decode_rect()
{
vector<uint8_t> encoded_source{read_file("test/lena8b.jls")};
jpegls_decoder decoder;
decoder.source(encoded_source);
decoder.read_header();
const vector<uint8_t> encoded_source{read_file("test/lena8b.jls")};
const jpegls_decoder decoder{encoded_source, true};
vector<uint8_t> decoded_buffer(decoder.destination_size());
// ReSharper disable CppDeprecatedEntity
DISABLE_DEPRECATED_WARNING
JlsParameters params{};
constexpr JlsParameters params{};
error_code error = JpegLsDecode(decoded_buffer.data(), decoded_buffer.size(), encoded_source.data(),
encoded_source.size(), &params, nullptr);
assert::is_true(!error);
const JlsRect rect{128, 128, 256, 1};
constexpr JlsRect rect{128, 128, 256, 1};
vector<uint8_t> decoded_data(static_cast<size_t>(rect.Width) * rect.Height);
decoded_data.push_back(0x1f);
@ -403,8 +449,7 @@ void test_encode_from_stream(const char* filename, const size_t offset, const ui
const int32_t bits_per_sample, const int32_t component_count,
const interleave_mode interleave_mode, const size_t expected_length)
{
ifstream source_file{filename, ios::in | ios::binary};
assert::is_true(source_file.good());
ifstream source_file{open_input_stream(filename)};
size_t length{get_stream_length(source_file, offset)};
assert::is_true(length >= offset);
@ -424,8 +469,11 @@ void test_encode_from_stream(const char* filename, const size_t offset, const ui
}
bool decode_to_pnm(istream& input, ostream& output)
bool decode_to_pnm(const char* filename_input, const char* filename_output)
try
{
ifstream input{open_input_stream(filename_input)};
const size_t length{get_stream_length(input)};
vector<uint8_t> encoded_source(length);
read(input, encoded_source);
@ -435,8 +483,27 @@ bool decode_to_pnm(istream& input, ostream& output)
interleave_mode interleave_mode;
std::tie(frame_info, interleave_mode) = jpegls_decoder::decode(encoded_source, decoded_destination);
if (frame_info.component_count > 1 && interleave_mode == charls::interleave_mode::none)
return false; // Unsupported at the moment.
if (!(frame_info.component_count == 1 || frame_info.component_count == 3))
{
cout << "Only JPEG-LS images with component count 1 or 3 are supported to decode to pnm\n";
return false;
}
// PPM format only supports by-pixel, convert if needed.
if (interleave_mode == charls::interleave_mode::none && frame_info.component_count == 3)
{
vector<uint8_t> pixels(decoded_destination.size());
if (frame_info.bits_per_sample > 8)
{
convert_planar_to_pixel<uint16_t>(frame_info.width, frame_info.height, decoded_destination.data(), pixels.data());
}
else
{
convert_planar_to_pixel<uint8_t>(frame_info.width, frame_info.height, decoded_destination.data(), pixels.data());
}
swap(decoded_destination, pixels);
}
// PNM format requires most significant byte first (big endian).
const int max_value = (1 << frame_info.bits_per_sample) - 1;
@ -451,11 +518,18 @@ bool decode_to_pnm(istream& input, ostream& output)
}
const int magic_number = frame_info.component_count == 3 ? 6 : 5;
ofstream output{open_output_stream(filename_output)};
output << 'P' << magic_number << "\n" << frame_info.width << ' ' << frame_info.height << "\n" << max_value << "\n";
write(output, decoded_destination, decoded_destination.size());
return true;
}
catch (const system_error& error)
{
cout << "Failed to decode " << filename_input << " to " << filename_output << ", reason: " << error.what() << '\n';
return false;
}
vector<int> read_pnm_header(istream& pnm_file)
@ -492,9 +566,12 @@ vector<int> read_pnm_header(istream& pnm_file)
// into the JPEG-LS format. The 2 binary formats P5 and P6 are supported:
// Portable GrayMap: P5 = binary, extension = .pgm, 0-2^16 (gray scale)
// Portable PixMap: P6 = binary, extension.ppm, range 0-2^16 (RGB)
bool encode_pnm(istream& pnm_file, ostream& jls_file_stream)
bool encode_pnm(const char* filename_input, const char* filename_output)
try
{
vector<int> read_values{read_pnm_header(pnm_file)};
ifstream pnm_file(open_input_stream(filename_input));
const vector<int> read_values{read_pnm_header(pnm_file)};
if (read_values.size() != 4)
return false;
@ -525,21 +602,27 @@ bool encode_pnm(istream& pnm_file, ostream& jls_file_stream)
encoder.destination(destination);
const size_t bytes_encoded{encoder.encode(input_buffer)};
ofstream jls_file_stream(open_output_stream(filename_output));
write(jls_file_stream, destination, bytes_encoded);
return jls_file_stream.good();
return true;
}
catch (const system_error& error)
{
cout << "Failed to encode " << filename_input << " to " << filename_output << ", reason: " << error.what() << '\n';
return false;
}
bool compare_pnm(istream& pnm_file1, istream& pnm_file2)
{
vector<int> header1{read_pnm_header(pnm_file1)};
const vector<int> header1{read_pnm_header(pnm_file1)};
if (header1.size() != 4)
{
cout << "Cannot read header from input file 1\n";
return false;
}
vector<int> header2 = read_pnm_header(pnm_file2);
const vector<int> header2 = read_pnm_header(pnm_file2);
if (header2.size() != 4)
{
cout << "Cannot read header from input file 2\n";
@ -610,19 +693,18 @@ bool compare_pnm(istream& pnm_file1, istream& pnm_file2)
bool decode_raw(const char* filename_encoded, const char* filename_output)
try
{
try
{
const vector<uint8_t> encoded_source{read_file(filename_encoded)};
vector<uint8_t> decoded_destination;
jpegls_decoder::decode(encoded_source, decoded_destination);
write_file(filename_output, decoded_destination.data(), decoded_destination.size());
return true;
}
catch (const system_error&)
{
return false;
}
const vector<uint8_t> encoded_source{read_file(filename_encoded)};
vector<uint8_t> decoded_destination;
jpegls_decoder::decode(encoded_source, decoded_destination);
write_file(filename_output, decoded_destination.data(), decoded_destination.size());
return true;
}
catch (const system_error& error)
{
cout << "Failed to decode " << filename_encoded << " to " << filename_output << ", reason: " << error.what() << '\n';
return false;
}
@ -723,10 +805,8 @@ int main(const int argc, const char* const argv[]) // NOLINT(bugprone-exception-
cout << "Syntax: -decodetopnm input-file output-file\n";
return EXIT_FAILURE;
}
ofstream pnm_file(argv[3], mode_output);
ifstream jls_file(argv[2], mode_input);
return decode_to_pnm(jls_file, pnm_file) ? EXIT_SUCCESS : EXIT_FAILURE;
return decode_to_pnm(argv[2], argv[3]) ? EXIT_SUCCESS : EXIT_FAILURE;
}
if (str == "-encodepnm")
@ -736,10 +816,8 @@ int main(const int argc, const char* const argv[]) // NOLINT(bugprone-exception-
cout << "Syntax: -encodepnm input-file output-file\n";
return EXIT_FAILURE;
}
ifstream pnm_file(argv[2], mode_input);
ofstream jls_file(argv[3], mode_output);
return encode_pnm(pnm_file, jls_file) ? EXIT_SUCCESS : EXIT_FAILURE;
return encode_pnm(argv[2], argv[3]) ? EXIT_SUCCESS : EXIT_FAILURE;
}
if (str == "-comparepnm")

View File

@ -9,6 +9,7 @@
#include <chrono>
#include <iostream>
#include <ratio>
#include <tuple>
#include <vector>
using charls::frame_info;
@ -47,8 +48,8 @@ void test_performance(const int loop_count)
// RGBA image (This is a common PNG sample)
test_file("test/alphatest.raw", 0, rect_size(380, 287), 8, 4, false, loop_count);
const rect_size size1024 = rect_size(1024, 1024);
const rect_size size512 = rect_size(512, 512);
const rect_size size1024{1024, 1024};
const rect_size size512{512, 512};
// 16 bit mono
test_file("test/MR2_UNC", 1728, size1024, 16, 1, true, loop_count);
@ -164,7 +165,7 @@ void encode_performance_tests(const int loop_count)
encoder2.frame_info(info).interleave_mode(interleave_mode);
encoder2.destination(destination);
static_cast<void>(encoder2.encode(anymap_file.image_data()));
std::ignore = encoder2.encode(anymap_file.image_data());
}
const auto end{steady_clock::now()};

View File

@ -191,6 +191,26 @@
<FileType>Document</FileType>
<DestinationFolders>$(OutDir)DataFiles</DestinationFolders>
</CopyFileToFolders>
<CopyFileToFolders Include="test8_ilv_none_rm_7.jls">
<FileType>Document</FileType>
<DestinationFolders>$(OutDir)DataFiles</DestinationFolders>
</CopyFileToFolders>
<CopyFileToFolders Include="test8_ilv_line_rm_7.jls">
<FileType>Document</FileType>
<DestinationFolders>$(OutDir)DataFiles</DestinationFolders>
</CopyFileToFolders>
<CopyFileToFolders Include="test8_ilv_sample_rm_7.jls">
<FileType>Document</FileType>
<DestinationFolders>$(OutDir)DataFiles</DestinationFolders>
</CopyFileToFolders>
<CopyFileToFolders Include="test8_ilv_sample_rm_300.jls">
<FileType>Document</FileType>
<DestinationFolders>$(OutDir)DataFiles</DestinationFolders>
</CopyFileToFolders>
<CopyFileToFolders Include="test16_rm_5.jls">
<FileType>Document</FileType>
<DestinationFolders>$(OutDir)DataFiles</DestinationFolders>
</CopyFileToFolders>
</ItemGroup>
<Import Project="$(VCTargetsPath)\Microsoft.Cpp.targets" />
<ImportGroup Label="ExtensionTargets">

View File

@ -166,5 +166,20 @@
<CopyFileToFolders Include="2bit_parrot_150x200.pgm">
<Filter>Test Images</Filter>
</CopyFileToFolders>
<CopyFileToFolders Include="test8_ilv_line_rm_7.jls">
<Filter>Test Images</Filter>
</CopyFileToFolders>
<CopyFileToFolders Include="test8_ilv_none_rm_7.jls">
<Filter>Test Images</Filter>
</CopyFileToFolders>
<CopyFileToFolders Include="test8_ilv_sample_rm_7.jls">
<Filter>Test Images</Filter>
</CopyFileToFolders>
<CopyFileToFolders Include="test8_ilv_sample_rm_300.jls">
<Filter>Test Images</Filter>
</CopyFileToFolders>
<CopyFileToFolders Include="test16_rm_5.jls">
<Filter>Test Images</Filter>
</CopyFileToFolders>
</ItemGroup>
</Project>

View File

@ -36,7 +36,7 @@ public:
TEST_METHOD(set_source_buffer_nullptr) // NOLINT
{
const array<uint8_t, 10> buffer{};
constexpr array<uint8_t, 10> buffer{};
auto error = charls_jpegls_decoder_set_source_buffer(nullptr, buffer.data(), buffer.size());
Assert::AreEqual(jpegls_errc::invalid_argument, error);

View File

@ -46,7 +46,7 @@ public:
TEST_METHOD(set_frame_info_buffer_nullptr) // NOLINT
{
charls_frame_info frame_info{};
constexpr charls_frame_info frame_info{};
auto error = charls_jpegls_encoder_set_frame_info(nullptr, &frame_info);
Assert::AreEqual(jpegls_errc::invalid_argument, error);
@ -70,7 +70,7 @@ public:
TEST_METHOD(set_preset_coding_parameters_nullptr) // NOLINT
{
jpegls_pc_parameters parameters{};
constexpr jpegls_pc_parameters parameters{};
auto error = charls_jpegls_encoder_set_preset_coding_parameters(nullptr, &parameters);
Assert::AreEqual(jpegls_errc::invalid_argument, error);
@ -94,7 +94,7 @@ public:
charls_jpegls_encoder* const encoder = charls_jpegls_encoder_create();
charls_frame_info frame_info{1, 1, 2, 1};
constexpr charls_frame_info frame_info{1, 1, 2, 1};
error = charls_jpegls_encoder_set_frame_info(encoder, &frame_info);
Assert::AreEqual(jpegls_errc::success, error);
@ -117,7 +117,7 @@ public:
TEST_METHOD(encode_from_buffer_nullptr) // NOLINT
{
const array<uint8_t, 10> source_buffer{};
constexpr array<uint8_t, 10> source_buffer{};
auto error = charls_jpegls_encoder_encode_from_buffer(nullptr, source_buffer.data(), source_buffer.size(), 0);
Assert::AreEqual(jpegls_errc::invalid_argument, error);
@ -129,7 +129,7 @@ public:
TEST_METHOD(write_spiff_header_nullptr) // NOLINT
{
charls_spiff_header spiff_header{};
constexpr charls_spiff_header spiff_header{};
auto error = charls_jpegls_encoder_write_spiff_header(nullptr, &spiff_header);
Assert::AreEqual(jpegls_errc::invalid_argument, error);
@ -148,7 +148,7 @@ public:
TEST_METHOD(write_spiff_entry_nullptr) // NOLINT
{
const array<uint8_t, 10> entry_data{};
constexpr array<uint8_t, 10> entry_data{};
auto error = charls_jpegls_encoder_write_spiff_entry(nullptr, 5, entry_data.data(), entry_data.size());
Assert::AreEqual(jpegls_errc::invalid_argument, error);
@ -170,11 +170,11 @@ public:
auto error = charls_jpegls_encoder_set_destination_buffer(encoder, nullptr, 0);
Assert::AreEqual(jpegls_errc::success, error);
charls_frame_info frame_info{1, 1, 2, 1};
constexpr charls_frame_info frame_info{1, 1, 2, 1};
error = charls_jpegls_encoder_set_frame_info(encoder, &frame_info);
Assert::AreEqual(jpegls_errc::success, error);
array<uint8_t, 10> buffer{};
constexpr array<uint8_t, 10> buffer{};
error = charls_jpegls_encoder_encode_from_buffer(encoder, buffer.data(), buffer.size(), 0);
Assert::AreEqual(jpegls_errc::destination_buffer_too_small, error);
@ -189,7 +189,7 @@ public:
auto error = charls_jpegls_encoder_set_destination_buffer(encoder, buffer.data(), buffer.size());
Assert::AreEqual(jpegls_errc::success, error);
charls_frame_info frame_info{1, 1, 2, 1};
constexpr charls_frame_info frame_info{1, 1, 2, 1};
error = charls_jpegls_encoder_set_frame_info(encoder, &frame_info);
Assert::AreEqual(jpegls_errc::success, error);

View File

@ -9,6 +9,7 @@
#include <charls/charls.h>
#include <tuple>
#include <vector>
using Microsoft::VisualStudio::CppUnitTestFramework::Assert;
@ -32,7 +33,7 @@ public:
{
for (int blue{}; blue != 255; ++blue)
{
const transform_hp1<uint8_t> transform;
constexpr transform_hp1<uint8_t> transform{};
const auto sample = transform(red, green, blue);
const transform_hp1<uint8_t>::inverse inverse(transform);
@ -59,7 +60,7 @@ public:
{
for (int blue{}; blue != 255; ++blue)
{
const transform_hp2<uint8_t> transform;
constexpr transform_hp2<uint8_t> transform{};
const auto sample = transform(red, green, blue);
const transform_hp2<uint8_t>::inverse inverse(transform);
@ -86,7 +87,7 @@ public:
{
for (int blue{}; blue != 255; ++blue)
{
const transform_hp3<uint8_t> transformation;
constexpr transform_hp3<uint8_t> transformation{};
const auto sample = transformation(red, green, blue);
const transform_hp3<uint8_t>::inverse inverse(transformation);
const auto round_trip = inverse(sample.v1, sample.v2, sample.v3);
@ -107,19 +108,20 @@ public:
vector<uint8_t> destination(decoder.destination_size());
assert_expect_exception(jpegls_errc::bit_depth_for_transform_not_supported, [&] { decoder.decode(destination); });
assert_expect_exception(jpegls_errc::bit_depth_for_transform_not_supported,
[&decoder, &destination] { decoder.decode(destination); });
}
TEST_METHOD(encode_non_8_or_16_bit_is_not_supported) // NOLINT
{
const frame_info frame_info{2, 1, 10, 3};
constexpr frame_info frame_info{2, 1, 10, 3};
jpegls_encoder encoder;
vector<uint8_t> destination(40);
encoder.destination(destination).frame_info(frame_info).color_transformation(color_transformation::hp3);
const vector<uint8_t> source(20);
assert_expect_exception(jpegls_errc::bit_depth_for_transform_not_supported,
[&] { static_cast<void>(encoder.encode(source)); });
[&encoder, &source] { std::ignore = encoder.encode(source); });
}
};

View File

@ -89,81 +89,50 @@ public:
decompress_file("DataFiles/lena8b.jls", "DataFiles/lena8b.pgm");
}
TEST_METHOD(decompress_color_8_bit_interleave_none_lossless_restart_7) // NOLINT
{
// ISO 14495-1: official test image 1 but with restart markers.
decompress_file("DataFiles/test8_ilv_none_rm_7.jls", "DataFiles/test8.ppm", false);
}
TEST_METHOD(decompress_color_8_bit_interleave_line_lossless_restart_7) // NOLINT
{
// ISO 14495-1: official test image 2 but with restart markers.
decompress_file("DataFiles/test8_ilv_line_rm_7.jls", "DataFiles/test8.ppm", false);
}
TEST_METHOD(decompress_color_8_bit_interleave_sample_lossless_restart_7) // NOLINT
{
// ISO 14495-1: official test image 3 but with restart markers.
decompress_file("DataFiles/test8_ilv_sample_rm_7.jls", "DataFiles/test8.ppm", false);
}
TEST_METHOD(decompress_color_8_bit_interleave_sample_lossless_restart_300) // NOLINT
{
// ISO 14495-1: official test image 3 but with restart markers and restart interval 300
decompress_file("DataFiles/test8_ilv_sample_rm_300.jls", "DataFiles/test8.ppm", false);
}
TEST_METHOD(decompress_monochrome_16_bit_restart_5) // NOLINT
{
// ISO 14495-1: official test image 12 but with restart markers and restart interval 5
decompress_file("DataFiles/test16_rm_5.jls", "DataFiles/test16.pgm", false);
}
private:
static void decompress_file(const char* encoded_filename, const char* raw_filename, const bool check_encode = true)
{
vector<uint8_t> encoded_source = read_file(encoded_filename);
const vector<uint8_t> encoded_source{read_file(encoded_filename)};
const jpegls_decoder decoder{encoded_source, true};
jpegls_decoder decoder;
decoder.source(encoded_source);
decoder.read_header();
portable_anymap_file reference_file =
read_anymap_reference_file(raw_filename, decoder.interleave_mode(), decoder.frame_info());
portable_anymap_file reference_file{
read_anymap_reference_file(raw_filename, decoder.interleave_mode(), decoder.frame_info())};
test_compliance(encoded_source, reference_file.image_data(), check_encode);
test_compliance_legacy_api(encoded_source.data(), encoded_source.size(), reference_file.image_data().data(),
reference_file.image_data().size(), check_encode);
}
static void test_compliance(const vector<uint8_t>& encoded_source, const vector<uint8_t>& uncompressed_source,
const bool check_encode)
{
jpegls_decoder decoder;
decoder.source(encoded_source);
decoder.read_header();
if (check_encode)
{
Assert::IsTrue(verify_encoded_bytes(uncompressed_source, encoded_source));
}
vector<uint8_t> destination(decoder.destination_size());
decoder.decode(destination);
if (decoder.near_lossless() == 0)
{
for (size_t i{}; i != uncompressed_source.size(); ++i)
{
if (uncompressed_source[i] != destination[i]) // AreEqual is very slow, pre-test to speed up 50X
{
Assert::AreEqual(uncompressed_source[i], destination[i]);
}
}
}
else
{
const frame_info frame_info{decoder.frame_info()};
const auto near_lossless{decoder.near_lossless()};
if (frame_info.bits_per_sample <= 8)
{
for (size_t i{}; i != uncompressed_source.size(); ++i)
{
if (abs(uncompressed_source[i] - destination[i]) >
near_lossless) // AreEqual is very slow, pre-test to speed up 50X
{
Assert::AreEqual(uncompressed_source[i], destination[i]);
}
}
}
else
{
const auto* source16 = reinterpret_cast<const uint16_t*>(uncompressed_source.data());
const auto* destination16 = reinterpret_cast<const uint16_t*>(destination.data());
for (size_t i{}; i != uncompressed_source.size() / 2; ++i)
{
if (abs(source16[i] - destination16[i]) >
near_lossless) // AreEqual is very slow, pre-test to speed up 50X
{
Assert::AreEqual(static_cast<int>(source16[i]), static_cast<int>(destination16[i]));
}
}
}
}
}
static void test_compliance_legacy_api(const uint8_t* compressed_bytes, const size_t compressed_length,
const uint8_t* uncompressed_data, const size_t uncompressed_length,
const bool check_encode)
@ -233,36 +202,6 @@ private:
// ReSharper restore CppDeprecatedEntity
RESTORE_DEPRECATED_WARNING
}
static bool verify_encoded_bytes(const vector<uint8_t>& uncompressed_source, const vector<uint8_t>& encoded_source)
{
jpegls_decoder decoder;
decoder.source(encoded_source);
decoder.read_header();
jpegls_encoder encoder;
encoder.frame_info(decoder.frame_info())
.interleave_mode(decoder.interleave_mode())
.near_lossless(decoder.near_lossless())
.preset_coding_parameters(decoder.preset_coding_parameters());
vector<uint8_t> our_encoded_bytes(encoded_source.size() + 16);
encoder.destination(our_encoded_bytes);
const size_t bytes_written{encoder.encode(uncompressed_source)};
if (bytes_written != encoded_source.size())
return false;
for (size_t i{}; i != encoded_source.size(); ++i)
{
if (encoded_source[i] != our_encoded_bytes[i])
{
return false;
}
}
return true;
}
};
}} // namespace charls::test

View File

@ -26,7 +26,8 @@ public:
initialize({destination, count});
}
void set_presets(const charls::jpegls_pc_parameters& /*preset_coding_parameters*/) noexcept(false) override
void set_presets(const charls::jpegls_pc_parameters& /*preset_coding_parameters*/,
uint32_t /*restart_interval*/) noexcept(false) override
{
}
@ -63,11 +64,11 @@ public:
int bits;
};
const array<data_t, 5> in_data{{{0x00, 24}, {0xFF, 8}, {0xFFFF, 16}, {0xFFFF, 16}, {0x12345678, 31}}};
constexpr array<data_t, 5> in_data{{{0x00, 24}, {0xFF, 8}, {0xFFFF, 16}, {0xFFFF, 16}, {0x12345678, 31}}};
array<uint8_t, 100> enc_buf{};
const frame_info frame_info{};
const coding_parameters parameters{};
constexpr frame_info frame_info{};
constexpr coding_parameters parameters{};
encoder_strategy_tester encoder(frame_info, parameters);

View File

@ -29,7 +29,7 @@ std::vector<uint8_t> decode_simple_8_bit_monochrome(const std::vector<uint8_t>&
std::tie(frame_info, std::ignore) = charls::jpegls_decoder::decode(source, destination);
if (frame_info.component_count != 1 || frame_info.bits_per_sample != 8)
throw std::exception("Not a 8 bit monochrome image");
throw std::runtime_error("Not a 8 bit monochrome image");
return destination;
}
@ -40,7 +40,7 @@ std::vector<uint8_t> decode_advanced(const std::vector<uint8_t>& source)
// Standalone JPEG-LS files may have a SPIFF header (color space info, etc.)
if (decoder.spiff_header_has_value() && decoder.spiff_header().color_space != charls::spiff_color_space::grayscale)
throw std::exception("Not a grayscale image");
throw std::runtime_error("Not a grayscale image");
// After read_header() other properties can also be retrieved.
if (decoder.near_lossless() != 0)
@ -57,10 +57,10 @@ std::vector<uint8_t> decode_simple_8_bit_monochrome_legacy(const std::vector<uin
JlsParameters parameters{};
auto error = JpegLsReadHeader(source.data(), source.size(), &parameters, error_message.data());
if (error != CharlsApiResultType::OK)
throw std::exception(error_message.data());
throw std::runtime_error(error_message.data());
if (parameters.components != 1 || parameters.bitsPerSample != 8)
throw std::exception("Not a 8 bit monochrome image");
throw std::runtime_error("Not a 8 bit monochrome image");
const size_t destination_size = static_cast<size_t>(parameters.width) * static_cast<uint32_t>(parameters.height);
std::vector<uint8_t> destination(destination_size);
@ -68,7 +68,7 @@ std::vector<uint8_t> decode_simple_8_bit_monochrome_legacy(const std::vector<uin
error = JpegLsDecode(destination.data(), destination.size(), source.data(), source.size(), &parameters,
error_message.data());
if (error != CharlsApiResultType::OK)
throw std::exception(error_message.data());
throw std::runtime_error(error_message.data());
return destination;
}
@ -115,7 +115,7 @@ std::vector<uint8_t> encode_simple_8_bit_monochrome_legacy(const std::vector<uin
const auto error = JpegLsEncode(destination.data(), destination.size(), &bytes_written, source.data(), source.size(),
&parameters, error_message.data());
if (error != CharlsApiResultType::OK)
throw std::exception(error_message.data());
throw std::runtime_error(error_message.data());
destination.resize(bytes_written);
return destination;

View File

@ -15,7 +15,7 @@ public:
{
}
void set_presets(const jpegls_pc_parameters&) noexcept(false) override
void set_presets(const jpegls_pc_parameters&, uint32_t /*restart_interval*/) noexcept(false) override
{
}
@ -29,9 +29,9 @@ public:
return nullptr;
}
void initialize_forward(const byte_span info) noexcept
void initialize_forward(const byte_span destination) noexcept
{
initialize(info);
initialize(destination);
}
void append_to_bit_stream_forward(const uint32_t bits, const int32_t bit_count)

View File

@ -37,7 +37,7 @@ public:
{
const std::error_category& category{jpegls_category()};
std::string message{category.message(0)};
const std::string message{category.message(0)};
Assert::IsTrue(message.size() > 0);
}
};

View File

@ -133,7 +133,7 @@ public:
jpeg_stream_writer writer({source.data(), source.size()});
writer.write_start_of_image();
const jpegls_pc_parameters presets{1, 2, 3, 4, 5};
constexpr jpegls_pc_parameters presets{1, 2, 3, 4, 5};
writer.write_jpegls_preset_parameters_segment(presets);
writer.write_start_of_frame_segment(1, 1, 2, 1);
writer.write_start_of_scan_segment(1, 0, interleave_mode::none);
@ -375,7 +375,7 @@ public:
jpeg_stream_reader reader({writer.buffer.data(), writer.buffer.size()});
assert_expect_exception(jpegls_errc::unexpected_marker_found, [&]() { reader.read_header(); });
assert_expect_exception(jpegls_errc::unexpected_marker_found, [&reader]() { reader.read_header(); });
}
TEST_METHOD(read_header_extra_sof_should_throw) // NOLINT
@ -387,7 +387,7 @@ public:
jpeg_stream_reader reader({writer.buffer.data(), writer.buffer.size()});
assert_expect_exception(jpegls_errc::duplicate_start_of_frame_marker, [&]() { reader.read_header(); });
assert_expect_exception(jpegls_errc::duplicate_start_of_frame_marker, [&reader]() { reader.read_header(); });
}
TEST_METHOD(read_header_too_large_near_lossless_in_sos_should_throw) // NOLINT
@ -400,7 +400,7 @@ public:
jpeg_stream_reader reader({writer.buffer.data(), writer.buffer.size()});
reader.read_header();
assert_expect_exception(jpegls_errc::invalid_parameter_near_lossless, [&]() { reader.read_start_of_scan(); });
assert_expect_exception(jpegls_errc::invalid_parameter_near_lossless, [&reader]() { reader.read_start_of_scan(); });
}
TEST_METHOD(read_header_too_large_near_lossless_in_sos_should_throw2) // NOLINT
@ -418,7 +418,7 @@ public:
jpeg_stream_reader reader({writer.buffer.data(), writer.buffer.size()});
reader.read_header();
assert_expect_exception(jpegls_errc::invalid_parameter_near_lossless, [&]() { reader.read_start_of_scan(); });
assert_expect_exception(jpegls_errc::invalid_parameter_near_lossless, [&reader]() { reader.read_start_of_scan(); });
}
TEST_METHOD(read_header_with_duplicate_component_id_in_start_of_frame_segment_should_throw) // NOLINT
@ -603,6 +603,89 @@ public:
}
}
TEST_METHOD(read_header_with_define_restart_interval_16bit) // NOLINT
{
jpeg_test_stream_writer writer;
writer.write_start_of_image();
writer.write_start_of_frame_segment(512, 512, 8, 3);
writer.write_define_restart_interval(UINT16_MAX - 5, 2);
writer.write_start_of_scan_segment(0, 1, 0, interleave_mode::none);
jpeg_stream_reader reader({writer.buffer.data(), writer.buffer.size()});
reader.read_header();
Assert::AreEqual(static_cast<uint32_t>(UINT16_MAX - 5), reader.parameters().restart_interval);
}
TEST_METHOD(read_header_with_define_restart_interval_24bit) // NOLINT
{
jpeg_test_stream_writer writer;
writer.write_start_of_image();
writer.write_start_of_frame_segment(512, 512, 8, 3);
writer.write_define_restart_interval(UINT16_MAX + 5, 3);
writer.write_start_of_scan_segment(0, 1, 0, interleave_mode::none);
jpeg_stream_reader reader({writer.buffer.data(), writer.buffer.size()});
reader.read_header();
Assert::AreEqual(static_cast<uint32_t>(UINT16_MAX + 5), reader.parameters().restart_interval);
}
TEST_METHOD(read_header_with_define_restart_interval_32bit) // NOLINT
{
jpeg_test_stream_writer writer;
writer.write_start_of_image();
writer.write_start_of_frame_segment(512, 512, 8, 3);
writer.write_define_restart_interval(UINT32_MAX - 7, 4);
writer.write_start_of_scan_segment(0, 1, 0, interleave_mode::none);
jpeg_stream_reader reader({writer.buffer.data(), writer.buffer.size()});
reader.read_header();
Assert::AreEqual(UINT32_MAX - 7, reader.parameters().restart_interval);
}
TEST_METHOD(read_header_with_2_define_restart_intervals) // NOLINT
{
jpeg_test_stream_writer writer;
writer.write_start_of_image();
writer.write_define_restart_interval(UINT32_MAX, 4);
writer.write_start_of_frame_segment(512, 512, 8, 3);
writer.write_define_restart_interval(0, 3);
writer.write_start_of_scan_segment(0, 1, 0, interleave_mode::none);
jpeg_stream_reader reader({writer.buffer.data(), writer.buffer.size()});
reader.read_header();
Assert::AreEqual(0U, reader.parameters().restart_interval);
}
TEST_METHOD(read_header_with_bad_define_restart_interval) // NOLINT
{
jpeg_test_stream_writer writer;
writer.write_start_of_image();
writer.write_start_of_frame_segment(512, 512, 8, 3);
constexpr array<uint8_t, 1> buffer{};
writer.write_segment(jpeg_marker_code::define_restart_interval, buffer.data(), buffer.size());
writer.write_start_of_scan_segment(0, 1, 0, interleave_mode::none);
jpeg_stream_reader reader({writer.buffer.data(), writer.buffer.size()});
assert_expect_exception(jpegls_errc::invalid_marker_segment_size, [&reader] { reader.read_header(); });
}
TEST_METHOD(read_jpegls_stream_with_restart_marker_outside_entropy_data) // NOLINT
{
jpeg_test_stream_writer writer;
writer.write_start_of_image();
writer.write_restart_marker(0);
jpeg_stream_reader reader({writer.buffer.data(), writer.buffer.size()});
assert_expect_exception(jpegls_errc::unexpected_restart_marker, [&reader] { reader.read_header(); });
}
private:
static void read_spiff_header(const uint8_t low_version)
{

View File

@ -42,7 +42,7 @@ public:
array<uint8_t, 1> buffer{};
jpeg_stream_writer writer({buffer.data(), buffer.size()});
assert_expect_exception(jpegls_errc::destination_buffer_too_small, [&] { writer.write_start_of_image(); });
assert_expect_exception(jpegls_errc::destination_buffer_too_small, [&writer] { writer.write_start_of_image(); });
Assert::AreEqual(static_cast<size_t>(0), writer.bytes_written());
}
@ -63,7 +63,7 @@ public:
array<uint8_t, 1> buffer{};
jpeg_stream_writer writer({buffer.data(), buffer.size()});
assert_expect_exception(jpegls_errc::destination_buffer_too_small, [&] { writer.write_end_of_image(); });
assert_expect_exception(jpegls_errc::destination_buffer_too_small, [&writer] { writer.write_end_of_image(); });
Assert::AreEqual(static_cast<size_t>(0), writer.bytes_written());
}
@ -131,7 +131,7 @@ public:
Assert::AreEqual(static_cast<uint8_t>(0), buffer[28]);
Assert::AreEqual(static_cast<uint8_t>(96), buffer[29]);
// header.horizontal_resolution = 1024;
// header.horizontal_resolution = 1024
Assert::AreEqual(static_cast<uint8_t>(0), buffer[30]);
Assert::AreEqual(static_cast<uint8_t>(0), buffer[31]);
Assert::AreEqual(static_cast<uint8_t>(4), buffer[32]);
@ -155,7 +155,7 @@ public:
1024};
assert_expect_exception(jpegls_errc::destination_buffer_too_small,
[&] { writer.write_spiff_header_segment(header); });
[&writer, &header] { writer.write_spiff_header_segment(header); });
Assert::AreEqual(static_cast<size_t>(0), writer.bytes_written());
}
@ -192,7 +192,7 @@ public:
array<uint8_t, 10> buffer{};
jpeg_stream_writer writer{{buffer.data(), buffer.size()}};
array<uint8_t, 2> data{0x77, 0x66};
const array<uint8_t, 2> data{0x77, 0x66};
writer.write_spiff_directory_entry(2, data.data(), data.size());
@ -275,14 +275,14 @@ public:
Assert::AreEqual(buffer.size(), writer.bytes_written());
Assert::AreEqual(static_cast<uint8_t>(16), buffer[4]);
Assert::AreEqual(static_cast<uint8_t>(UINT8_MAX), buffer[9]);
Assert::AreEqual(UINT8_MAX, buffer[9]);
Assert::AreEqual(static_cast<uint8_t>(UINT8_MAX), buffer[buffer.size() - 3]); // Last component index.
Assert::AreEqual(UINT8_MAX, buffer[buffer.size() - 3]); // Last component index.
}
TEST_METHOD(write_color_transform_segment) // NOLINT
{
const color_transformation transformation = color_transformation::hp1;
constexpr color_transformation transformation = color_transformation::hp1;
array<uint8_t, 9> buffer{};
jpeg_stream_writer writer({buffer.data(), buffer.size()});
@ -301,7 +301,7 @@ public:
TEST_METHOD(write_jpegls_extended_parameters_marker_and_serialize) // NOLINT
{
const jpegls_pc_parameters presets{2, 1, 2, 3, 7};
constexpr jpegls_pc_parameters presets{2, 1, 2, 3, 7};
array<uint8_t, 15> buffer{};
jpeg_stream_writer writer({buffer.data(), buffer.size()});

View File

@ -82,6 +82,36 @@ public:
write_segment(jpeg_marker_code::start_of_scan, segment.data(), segment.size());
}
void write_define_restart_interval(const uint32_t restart_interval, const int size)
{
std::vector<uint8_t> segment;
switch (size)
{
case 2:
push_back(segment, static_cast<uint16_t>(restart_interval));
break;
case 3:
push_back_uint24(segment, restart_interval);
break;
case 4:
push_back(segment, restart_interval);
break;
default:
assert(false);
break;
}
write_segment(jpeg_marker_code::define_restart_interval, segment.data(), segment.size());
}
void write_restart_marker(const uint8_t interval_index)
{
write_marker(static_cast<jpeg_marker_code>(jpeg_restart_marker_base + interval_index));
}
void write_segment(const jpeg_marker_code marker_code, const void* data, const size_t data_size)
{
write_marker(marker_code);
@ -119,6 +149,14 @@ public:
int componentIdOverride{};
uint8_t mapping_table_selector{};
std::vector<uint8_t> buffer;
private:
static void push_back_uint24(std::vector<uint8_t>& values, const uint32_t value)
{
values.push_back(static_cast<uint8_t>(value >> 16));
values.push_back(static_cast<uint8_t>(value >> 8));
values.push_back(static_cast<uint8_t>(value));
}
};
}} // namespace charls::test

View File

@ -19,6 +19,7 @@
using Microsoft::VisualStudio::CppUnitTestFramework::Assert;
using std::array;
using std::error_code;
using std::ignore;
using std::tie;
using std::vector;
using namespace charls_test;
@ -50,7 +51,7 @@ public:
jpegls_decoder decoder2(std::move(decoder1));
jpegls_decoder decoder3;
const array<uint8_t, 10> buffer{};
constexpr array<uint8_t, 10> buffer{};
decoder3.source(buffer.data(), buffer.size());
decoder3 = std::move(decoder2);
}
@ -59,30 +60,30 @@ public:
{
jpegls_decoder decoder;
vector<uint8_t> source(2000);
const vector<uint8_t> source(2000);
decoder.source(source);
assert_expect_exception(jpegls_errc::invalid_operation, [&] { decoder.source(source); });
assert_expect_exception(jpegls_errc::invalid_operation, [&decoder, &source] { decoder.source(source); });
}
TEST_METHOD(read_spiff_header_without_source) // NOLINT
{
jpegls_decoder decoder;
assert_expect_exception(jpegls_errc::invalid_operation, [&] { decoder.read_spiff_header(); });
assert_expect_exception(jpegls_errc::invalid_operation, [&decoder] { decoder.read_spiff_header(); });
}
TEST_METHOD(destination_size_without_reading_header) // NOLINT
{
jpegls_decoder decoder;
const jpegls_decoder decoder;
assert_expect_exception(jpegls_errc::invalid_operation, [&] { static_cast<void>(decoder.destination_size()); });
assert_expect_exception(jpegls_errc::invalid_operation, [&decoder] { ignore = decoder.destination_size(); });
}
TEST_METHOD(read_header_without_source) // NOLINT
{
jpegls_decoder decoder;
assert_expect_exception(jpegls_errc::invalid_operation, [&] { decoder.read_header(); });
assert_expect_exception(jpegls_errc::invalid_operation, [&decoder] { decoder.read_header(); });
}
TEST_METHOD(read_header_from_non_jpegls_data) // NOLINT
@ -120,17 +121,17 @@ public:
TEST_METHOD(interleave_mode_without_read_header) // NOLINT
{
const vector<uint8_t> source(2000);
jpegls_decoder decoder{source, false};
const jpegls_decoder decoder{source, false};
assert_expect_exception(jpegls_errc::invalid_operation, [&] { static_cast<void>(decoder.interleave_mode()); });
assert_expect_exception(jpegls_errc::invalid_operation, [&decoder] { ignore = decoder.interleave_mode(); });
}
TEST_METHOD(near_lossless_without_read_header) // NOLINT
{
const vector<uint8_t> source(2000);
jpegls_decoder decoder{source, false};
const jpegls_decoder decoder{source, false};
assert_expect_exception(jpegls_errc::invalid_operation, [&] { static_cast<void>(decoder.near_lossless()); });
assert_expect_exception(jpegls_errc::invalid_operation, [&decoder] { ignore = decoder.near_lossless(); });
}
TEST_METHOD(preset_coding_parameters_without_read_header) // NOLINT
@ -140,8 +141,7 @@ public:
const vector<uint8_t> source(2000);
decoder.source(source);
assert_expect_exception(jpegls_errc::invalid_operation,
[&] { static_cast<void>(decoder.preset_coding_parameters()); });
assert_expect_exception(jpegls_errc::invalid_operation, [&decoder] { ignore = decoder.preset_coding_parameters(); });
}
TEST_METHOD(destination_size) // NOLINT
@ -261,10 +261,10 @@ public:
TEST_METHOD(decode_without_reading_header) // NOLINT
{
jpegls_decoder decoder;
const jpegls_decoder decoder;
vector<uint8_t> buffer(1000);
assert_expect_exception(jpegls_errc::invalid_operation, [&] { decoder.decode(buffer); });
assert_expect_exception(jpegls_errc::invalid_operation, [&decoder, &buffer] { decoder.decode(buffer); });
}
TEST_METHOD(decode_reference_to_mapping_table_selector) // NOLINT
@ -278,7 +278,7 @@ public:
jpegls_decoder decoder{writer.buffer, false};
assert_expect_exception(jpegls_errc::parameter_value_not_supported, [&] { decoder.read_header(); });
assert_expect_exception(jpegls_errc::parameter_value_not_supported, [&decoder] { decoder.read_header(); });
}
TEST_METHOD(read_spiff_header) // NOLINT
@ -327,7 +327,7 @@ public:
jpegls_decoder decoder{source, false};
error_code ec;
static_cast<void>(decoder.read_spiff_header(ec));
ignore = decoder.read_spiff_header(ec);
Assert::IsTrue(ec == jpegls_errc::jpeg_marker_start_byte_not_found);
}
@ -354,7 +354,7 @@ public:
jpegls_decoder decoder{source, true};
assert_expect_exception(jpegls_errc::invalid_operation, [&] { static_cast<void>(decoder.read_header()); });
assert_expect_exception(jpegls_errc::invalid_operation, [&decoder] { ignore = decoder.read_header(); });
}
TEST_METHOD(simple_decode) // NOLINT
@ -399,7 +399,7 @@ public:
{
const vector<uint8_t> source{read_file("ff_in_entropy_data.jls")};
jpegls_decoder decoder{source, true};
const jpegls_decoder decoder{source, true};
const auto& frame_info{decoder.frame_info()};
Assert::AreEqual(1, frame_info.component_count);
@ -409,14 +409,15 @@ public:
vector<uint8_t> destination(decoder.destination_size());
assert_expect_exception(jpegls_errc::invalid_encoded_data, [&] { static_cast<void>(decoder.decode(destination)); });
assert_expect_exception(jpegls_errc::invalid_encoded_data,
[&decoder, &destination] { decoder.decode(destination); });
}
TEST_METHOD(decode_file_with_golomb_large_then_k_max) // NOLINT
{
const vector<uint8_t> source{read_file("fuzzy_input_golomb_16.jls")};
jpegls_decoder decoder{source, true};
const jpegls_decoder decoder{source, true};
const auto& frame_info{decoder.frame_info()};
Assert::AreEqual(3, frame_info.component_count);
@ -426,9 +427,58 @@ public:
vector<uint8_t> destination(decoder.destination_size());
assert_expect_exception(jpegls_errc::invalid_encoded_data, [&] { static_cast<void>(decoder.decode(destination)); });
assert_expect_exception(jpegls_errc::invalid_encoded_data,
[&decoder, &destination] { decoder.decode(destination); });
}
TEST_METHOD(decode_file_with_missing_restart_marker) // NOLINT
{
vector<uint8_t> source{read_file("DataFiles/t8c0e0.jls")};
// Insert a DRI marker segment to trigger that restart markers are used.
jpeg_test_stream_writer stream_writer;
stream_writer.write_define_restart_interval(10, 3);
const auto it = source.begin() + 2;
source.insert(it, stream_writer.buffer.cbegin(), stream_writer.buffer.cend());
const jpegls_decoder decoder{source, true};
vector<uint8_t> destination(decoder.destination_size());
assert_expect_exception(jpegls_errc::restart_marker_not_found, [&decoder, &destination] { decoder.decode(destination); });
}
TEST_METHOD(decode_file_with_incorrect_restart_marker) // NOLINT
{
vector<uint8_t> source{read_file("DataFiles/test8_ilv_none_rm_7.jls")};
// Change the first restart marker to the second.
auto it{find_scan_header(source.begin(), source.end())};
it = find_first_restart_marker(it + 1, source.end());
++it;
*it = 0xD1;
const jpegls_decoder decoder{source, true};
vector<uint8_t> destination(decoder.destination_size());
assert_expect_exception(jpegls_errc::restart_marker_not_found, [&decoder, &destination] { decoder.decode(destination); });
}
TEST_METHOD(decode_file_with_extra_begin_bytes_for_restart_marker_code) // NOLINT
{
vector<uint8_t> source{read_file("DataFiles/test8_ilv_none_rm_7.jls")};
// Add additional 0xFF marker begin bytes
auto it{find_scan_header(source.begin(), source.end())};
it = find_first_restart_marker(it + 1, source.end());
const array<uint8_t, 7> extra_begin_bytes{0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF};
source.insert(it, extra_begin_bytes.cbegin(), extra_begin_bytes.cend());
const jpegls_decoder decoder{source, true};
portable_anymap_file reference_file{
read_anymap_reference_file("DataFiles/test8.ppm", decoder.interleave_mode(), decoder.frame_info())};
test_compliance(source, reference_file.image_data(), false);
}
private:
static vector<uint8_t>::iterator find_scan_header(const vector<uint8_t>::iterator& begin,
@ -445,6 +495,20 @@ private:
return end;
}
static vector<uint8_t>::iterator find_first_restart_marker(const vector<uint8_t>::iterator& begin,
const vector<uint8_t>::iterator& end) noexcept
{
constexpr uint8_t first_restart_marker{0xD0};
for (auto it{begin}; it != end; ++it)
{
if (*it == 0xFF && it + 1 != end && *(it + 1) == first_restart_marker)
return it;
}
return end;
}
static vector<uint8_t> create_default_pc_parameters_segment()
{
vector<uint8_t> segment;

View File

@ -14,8 +14,8 @@
using Microsoft::VisualStudio::CppUnitTestFramework::Assert;
using std::array;
using std::vector;
using std::ignore;
using std::vector;
constexpr size_t serialized_spiff_header_size = 34;
@ -54,16 +54,18 @@ public:
{
jpegls_encoder encoder;
assert_expect_exception(jpegls_errc::invalid_argument_width, [&] { encoder.frame_info({0, 1, 2, 1}); });
assert_expect_exception(jpegls_errc::invalid_argument_width, [&] { encoder.frame_info({UINT16_MAX + 1, 1, 2, 1}); });
assert_expect_exception(jpegls_errc::invalid_argument_width, [&encoder] { encoder.frame_info({0, 1, 2, 1}); });
assert_expect_exception(jpegls_errc::invalid_argument_width, [&encoder] {
encoder.frame_info({UINT16_MAX + 1, 1, 2, 1});
});
}
TEST_METHOD(frame_info_bad_height) // NOLINT
{
jpegls_encoder encoder;
assert_expect_exception(jpegls_errc::invalid_argument_height, [&] { encoder.frame_info({1, 0, 2, 1}); });
assert_expect_exception(jpegls_errc::invalid_argument_height, [&] {
assert_expect_exception(jpegls_errc::invalid_argument_height, [&encoder] { encoder.frame_info({1, 0, 2, 1}); });
assert_expect_exception(jpegls_errc::invalid_argument_height, [&encoder] {
encoder.frame_info({1, UINT16_MAX + 1, 2, 1});
});
}
@ -72,16 +74,24 @@ public:
{
jpegls_encoder encoder;
assert_expect_exception(jpegls_errc::invalid_argument_bits_per_sample, [&] { encoder.frame_info({1, 1, 1, 1}); });
assert_expect_exception(jpegls_errc::invalid_argument_bits_per_sample, [&] { encoder.frame_info({1, 1, 17, 1}); });
assert_expect_exception(jpegls_errc::invalid_argument_bits_per_sample, [&encoder] {
encoder.frame_info({1, 1, 1, 1});
});
assert_expect_exception(jpegls_errc::invalid_argument_bits_per_sample, [&encoder] {
encoder.frame_info({1, 1, 17, 1});
});
}
TEST_METHOD(frame_info_bad_component_count) // NOLINT
{
jpegls_encoder encoder;
assert_expect_exception(jpegls_errc::invalid_argument_component_count, [&] { encoder.frame_info({1, 1, 2, 0}); });
assert_expect_exception(jpegls_errc::invalid_argument_component_count, [&] { encoder.frame_info({1, 1, 2, 256}); });
assert_expect_exception(jpegls_errc::invalid_argument_component_count, [&encoder] {
encoder.frame_info({1, 1, 2, 0});
});
assert_expect_exception(jpegls_errc::invalid_argument_component_count, [&encoder] {
encoder.frame_info({1, 1, 2, 256});
});
}
TEST_METHOD(interleave_mode) // NOLINT
@ -98,9 +108,9 @@ public:
jpegls_encoder encoder;
assert_expect_exception(jpegls_errc::invalid_argument_interleave_mode,
[&] { encoder.interleave_mode(static_cast<charls::interleave_mode>(-1)); });
[&encoder] { encoder.interleave_mode(static_cast<charls::interleave_mode>(-1)); });
assert_expect_exception(jpegls_errc::invalid_argument_interleave_mode,
[&] { encoder.interleave_mode(static_cast<charls::interleave_mode>(3)); });
[&encoder] { encoder.interleave_mode(static_cast<charls::interleave_mode>(3)); });
}
TEST_METHOD(near_lossless) // NOLINT
@ -115,8 +125,8 @@ public:
{
jpegls_encoder encoder;
assert_expect_exception(jpegls_errc::invalid_argument_near_lossless, [&] { encoder.near_lossless(-1); });
assert_expect_exception(jpegls_errc::invalid_argument_near_lossless, [&] { encoder.near_lossless(256); });
assert_expect_exception(jpegls_errc::invalid_argument_near_lossless, [&encoder] { encoder.near_lossless(-1); });
assert_expect_exception(jpegls_errc::invalid_argument_near_lossless, [&encoder] { encoder.near_lossless(256); });
}
TEST_METHOD(estimated_destination_size_minimal_frame_info) // NOLINT
@ -179,7 +189,7 @@ public:
const jpegls_encoder encoder;
assert_expect_exception(jpegls_errc::invalid_operation,
[&] { static_cast<void>(encoder.estimated_destination_size()); });
[&encoder] { ignore = encoder.estimated_destination_size(); });
}
TEST_METHOD(destination) // NOLINT
@ -197,7 +207,8 @@ public:
vector<uint8_t> destination(200);
encoder.destination(destination);
assert_expect_exception(jpegls_errc::invalid_operation, [&] { encoder.destination(destination); });
assert_expect_exception(jpegls_errc::invalid_operation,
[&encoder, &destination] { encoder.destination(destination); });
}
TEST_METHOD(write_standard_spiff_header) // NOLINT
@ -237,7 +248,7 @@ public:
encoder.frame_info({1, 1, 2, 1});
assert_expect_exception(jpegls_errc::invalid_operation,
[&] { encoder.write_standard_spiff_header(spiff_color_space::cmyk); });
[&encoder] { encoder.write_standard_spiff_header(spiff_color_space::cmyk); });
}
TEST_METHOD(write_standard_spiff_header_without_frame_info) // NOLINT
@ -248,7 +259,7 @@ public:
encoder.destination(destination);
assert_expect_exception(jpegls_errc::invalid_operation,
[&] { encoder.write_standard_spiff_header(spiff_color_space::cmyk); });
[&encoder] { encoder.write_standard_spiff_header(spiff_color_space::cmyk); });
}
TEST_METHOD(write_standard_spiff_header_twice) // NOLINT
@ -262,7 +273,7 @@ public:
encoder.write_standard_spiff_header(spiff_color_space::cmyk);
assert_expect_exception(jpegls_errc::invalid_operation,
[&] { encoder.write_standard_spiff_header(spiff_color_space::cmyk); });
[&encoder] { encoder.write_standard_spiff_header(spiff_color_space::cmyk); });
}
TEST_METHOD(write_spiff_header) // NOLINT
@ -310,7 +321,8 @@ public:
spiff_header spiff_header{};
spiff_header.width = 1;
assert_expect_exception(jpegls_errc::invalid_argument_height, [&] { encoder.write_spiff_header(spiff_header); });
assert_expect_exception(jpegls_errc::invalid_argument_height,
[&encoder, &spiff_header] { encoder.write_spiff_header(spiff_header); });
Assert::AreEqual(static_cast<size_t>(0), encoder.bytes_written());
}
@ -326,7 +338,8 @@ public:
spiff_header spiff_header{};
spiff_header.height = 1;
assert_expect_exception(jpegls_errc::invalid_argument_width, [&] { encoder.write_spiff_header(spiff_header); });
assert_expect_exception(jpegls_errc::invalid_argument_width,
[&encoder, &spiff_header] { encoder.write_spiff_header(spiff_header); });
Assert::AreEqual(static_cast<size_t>(0), encoder.bytes_written());
}
@ -386,7 +399,7 @@ public:
encoder.destination(destination);
encoder.write_standard_spiff_header(spiff_color_space::cmyk);
assert_expect_exception(jpegls_errc::invalid_argument, [&] { encoder.write_spiff_entry(1, "test", 4); });
assert_expect_exception(jpegls_errc::invalid_argument, [&encoder] { encoder.write_spiff_entry(1, "test", 4); });
}
TEST_METHOD(write_spiff_entry_with_invalid_size) // NOLINT
@ -399,8 +412,8 @@ public:
encoder.destination(destination);
encoder.write_standard_spiff_header(spiff_color_space::cmyk);
assert_expect_exception(jpegls_errc::invalid_argument_spiff_entry_size, [&] {
vector<uint8_t> spiff_entry(65528 + 1);
assert_expect_exception(jpegls_errc::invalid_argument_spiff_entry_size, [&encoder] {
const vector<uint8_t> spiff_entry(65528 + 1);
encoder.write_spiff_entry(spiff_entry_tag::image_title, spiff_entry.data(), spiff_entry.size());
});
}
@ -414,8 +427,8 @@ public:
vector<uint8_t> destination(encoder.estimated_destination_size());
encoder.destination(destination);
assert_expect_exception(jpegls_errc::invalid_operation, [&] {
vector<uint8_t> spiff_entry(65528);
assert_expect_exception(jpegls_errc::invalid_operation, [&encoder] {
const vector<uint8_t> spiff_entry(65528);
encoder.write_spiff_entry(spiff_entry_tag::image_title, spiff_entry.data(), spiff_entry.size());
});
}
@ -424,7 +437,7 @@ public:
{
jpegls_encoder encoder;
const jpegls_pc_parameters pc_parameters{};
constexpr jpegls_pc_parameters pc_parameters{};
encoder.preset_coding_parameters(pc_parameters);
// No explicit test possible, code should remain stable.
@ -434,17 +447,18 @@ public:
TEST_METHOD(set_preset_coding_parameters_bad_values) // NOLINT
{
const array<uint8_t, 5> source{0, 1, 1, 1, 0};
const frame_info frame_info{5, 1, 8, 1};
constexpr frame_info frame_info{5, 1, 8, 1};
jpegls_encoder encoder;
encoder.frame_info(frame_info);
vector<uint8_t> destination(encoder.estimated_destination_size());
encoder.destination(destination);
const jpegls_pc_parameters bad_pc_parameters{1, 1, 1, 1, 1};
constexpr jpegls_pc_parameters bad_pc_parameters{1, 1, 1, 1, 1};
encoder.preset_coding_parameters(bad_pc_parameters);
assert_expect_exception(jpegls_errc::invalid_argument_jpegls_pc_parameters, [&] { ignore = encoder.encode(source); });
assert_expect_exception(jpegls_errc::invalid_argument_jpegls_pc_parameters,
[&encoder, &source] { ignore = encoder.encode(source); });
}
TEST_METHOD(encode_with_preset_coding_parameters_non_default_values) // NOLINT
@ -462,7 +476,7 @@ public:
jpegls_encoder encoder;
assert_expect_exception(jpegls_errc::invalid_argument_color_transformation,
[&] { encoder.color_transformation(static_cast<color_transformation>(100)); });
[&encoder] { encoder.color_transformation(static_cast<color_transformation>(100)); });
}
TEST_METHOD(encode_without_destination) // NOLINT
@ -471,7 +485,7 @@ public:
encoder.frame_info({1, 1, 2, 1});
vector<uint8_t> source(20);
assert_expect_exception(jpegls_errc::invalid_operation, [&] { ignore = encoder.encode(source); });
assert_expect_exception(jpegls_errc::invalid_operation, [&encoder, &source] { ignore = encoder.encode(source); });
}
TEST_METHOD(encode_without_frame_info) // NOLINT
@ -481,13 +495,13 @@ public:
vector<uint8_t> destination(20);
encoder.destination(destination);
const vector<uint8_t> source(20);
assert_expect_exception(jpegls_errc::invalid_operation, [&] { ignore = encoder.encode(source); });
assert_expect_exception(jpegls_errc::invalid_operation, [&encoder, &source] { ignore = encoder.encode(source); });
}
TEST_METHOD(encode_with_spiff_header) // NOLINT
{
const array<uint8_t, 5> source{0, 1, 2, 3, 4};
const frame_info frame_info{5, 1, 8, 1};
constexpr frame_info frame_info{5, 1, 8, 1};
jpegls_encoder encoder;
encoder.frame_info(frame_info);
@ -505,7 +519,7 @@ public:
TEST_METHOD(encode_with_color_transformation) // NOLINT
{
const array<uint8_t, 6> source{0, 1, 2, 3, 4, 5};
const frame_info frame_info{2, 1, 8, 3};
constexpr frame_info frame_info{2, 1, 8, 3};
jpegls_encoder encoder;
encoder.frame_info(frame_info);
@ -522,7 +536,7 @@ public:
TEST_METHOD(encode_16_bit) // NOLINT
{
const array<uint8_t, 6> source{0, 1, 2, 3, 4, 5};
const frame_info frame_info{3, 1, 16, 1};
constexpr frame_info frame_info{3, 1, 16, 1};
jpegls_encoder encoder;
encoder.frame_info(frame_info);
@ -540,7 +554,7 @@ public:
{
const vector<uint8_t> source{0, 1, 2, 3, 4, 5};
const frame_info frame_info{3, 1, 16, 1};
constexpr frame_info frame_info{3, 1, 16, 1};
const auto encoded{jpegls_encoder::encode(source, frame_info)};
test_by_decoding(encoded, frame_info, source.data(), source.size(), interleave_mode::none);
@ -550,7 +564,7 @@ public:
{
const array<uint8_t, 30> source{100, 100, 100, 0, 0, 0, 0, 0, 0, 0, 150, 150,
150, 0, 0, 0, 0, 0, 0, 0, 200, 200, 200};
const frame_info frame_info{3, 1, 8, 3};
constexpr frame_info frame_info{3, 1, 8, 3};
jpegls_encoder encoder;
encoder.frame_info(frame_info);
@ -567,7 +581,7 @@ public:
TEST_METHOD(encode_1_component_4_bit_with_high_bits_set) // NOLINT
{
const vector<uint8_t> source(512 * 512, 0xFF);
const frame_info frame_info{512, 512, 4, 1};
constexpr frame_info frame_info{512, 512, 4, 1};
jpegls_encoder encoder;
encoder.frame_info(frame_info);
@ -585,7 +599,7 @@ public:
TEST_METHOD(encode_1_component_12_bit_with_high_bits_set) // NOLINT
{
const vector<uint8_t> source(512 * 512 * 2, 0xFF);
const frame_info frame_info{512, 512, 12, 1};
constexpr frame_info frame_info{512, 512, 12, 1};
jpegls_encoder encoder;
encoder.frame_info(frame_info);
@ -603,7 +617,7 @@ public:
TEST_METHOD(encode_3_components_6_bit_with_high_bits_set_interleave_mode_sample) // NOLINT
{
const vector<uint8_t> source(512 * 512 * 3, 0xFF);
const frame_info frame_info{512, 512, 6, 3};
constexpr frame_info frame_info{512, 512, 6, 3};
jpegls_encoder encoder;
encoder.frame_info(frame_info).interleave_mode(interleave_mode::sample);
@ -621,7 +635,7 @@ public:
TEST_METHOD(encode_3_components_6_bit_with_high_bits_set_interleave_mode_line) // NOLINT
{
const vector<uint8_t> source(512 * 512 * 3, 0xFF);
const frame_info frame_info{512, 512, 6, 3};
constexpr frame_info frame_info{512, 512, 6, 3};
jpegls_encoder encoder;
encoder.frame_info(frame_info).interleave_mode(interleave_mode::line);
@ -639,7 +653,7 @@ public:
TEST_METHOD(encode_3_components_10_bit_with_high_bits_set_interleave_mode_sample) // NOLINT
{
const vector<uint8_t> source(512 * 512 * 2 * 3, 0xFF);
const frame_info frame_info{512, 512, 10, 3};
constexpr frame_info frame_info{512, 512, 10, 3};
jpegls_encoder encoder;
encoder.frame_info(frame_info).interleave_mode(interleave_mode::sample);
@ -657,7 +671,7 @@ public:
TEST_METHOD(encode_3_components_10_bit_with_high_bits_set_interleave_mode_line) // NOLINT
{
const vector<uint8_t> source(512 * 512 * 2 * 3, 0xFF);
const frame_info frame_info{512, 512, 10, 3};
constexpr frame_info frame_info{512, 512, 10, 3};
jpegls_encoder encoder;
encoder.frame_info(frame_info).interleave_mode(interleave_mode::line);
@ -675,7 +689,7 @@ public:
TEST_METHOD(encode_4_components_6_bit_with_high_bits_set_interleave_mode_sample) // NOLINT
{
const vector<uint8_t> source(512 * 512 * 4, 0xFF);
const frame_info frame_info{512, 512, 6, 4};
constexpr frame_info frame_info{512, 512, 6, 4};
jpegls_encoder encoder;
encoder.frame_info(frame_info).interleave_mode(interleave_mode::sample);
@ -693,7 +707,7 @@ public:
TEST_METHOD(encode_4_components_6_bit_with_high_bits_set_interleave_mode_line) // NOLINT
{
const vector<uint8_t> source(512 * 512 * 4, 0xFF);
const frame_info frame_info{512, 512, 6, 4};
constexpr frame_info frame_info{512, 512, 6, 4};
jpegls_encoder encoder;
encoder.frame_info(frame_info).interleave_mode(interleave_mode::line);
@ -711,7 +725,7 @@ public:
TEST_METHOD(encode_4_components_10_bit_with_high_bits_set_interleave_mode_sample) // NOLINT
{
const vector<uint8_t> source(512 * 512 * 2 * 4, 0xFF);
const frame_info frame_info{512, 512, 10, 4};
constexpr frame_info frame_info{512, 512, 10, 4};
jpegls_encoder encoder;
encoder.frame_info(frame_info).interleave_mode(interleave_mode::sample);
@ -729,7 +743,7 @@ public:
TEST_METHOD(encode_4_components_10_bit_with_high_bits_set_interleave_mode_line) // NOLINT
{
const vector<uint8_t> source(512 * 512 * 2 * 4, 0xFF);
const frame_info frame_info{512, 512, 10, 4};
constexpr frame_info frame_info{512, 512, 10, 4};
jpegls_encoder encoder;
encoder.frame_info(frame_info).interleave_mode(interleave_mode::line);
@ -747,7 +761,7 @@ public:
TEST_METHOD(rewind) // NOLINT
{
const array<uint8_t, 6> source{0, 1, 2, 3, 4, 5};
const frame_info frame_info{3, 1, 16, 1};
constexpr frame_info frame_info{3, 1, 16, 1};
jpegls_encoder encoder;
encoder.frame_info(frame_info);
@ -772,7 +786,7 @@ public:
TEST_METHOD(rewind_before_destination) // NOLINT
{
const array<uint8_t, 6> source{0, 1, 2, 3, 4, 5};
const frame_info frame_info{3, 1, 16, 1};
constexpr frame_info frame_info{3, 1, 16, 1};
jpegls_encoder encoder;
encoder.frame_info(frame_info);
@ -828,7 +842,7 @@ private:
static void encode_with_custom_preset_coding_parameters(const jpegls_pc_parameters& pc_parameters)
{
const array<uint8_t, 5> source{0, 1, 1, 1, 0};
const frame_info frame_info{5, 1, 8, 1};
constexpr frame_info frame_info{5, 1, 8, 1};
jpegls_encoder encoder;
encoder.frame_info(frame_info);

View File

@ -17,13 +17,12 @@ public:
{
using lossless_traits = lossless_traits<uint16_t, 12>;
const auto traits1{default_traits<uint16_t, uint16_t>(4095, 0)};
const lossless_traits traits2{};
Assert::IsTrue(traits1.limit == traits2.limit);
Assert::IsTrue(traits1.maximum_sample_value == traits2.maximum_sample_value);
Assert::IsTrue(traits1.reset_threshold == traits2.reset_threshold);
Assert::IsTrue(traits1.bits_per_pixel == traits2.bits_per_pixel);
Assert::IsTrue(traits1.quantized_bits_per_pixel == traits2.quantized_bits_per_pixel);
Assert::IsTrue(traits1.limit == lossless_traits::limit);
Assert::IsTrue(traits1.maximum_sample_value == lossless_traits::maximum_sample_value);
Assert::IsTrue(traits1.reset_threshold == lossless_traits::reset_threshold);
Assert::IsTrue(traits1.bits_per_pixel == lossless_traits::bits_per_pixel);
Assert::IsTrue(traits1.quantized_bits_per_pixel == lossless_traits::quantized_bits_per_pixel);
for (int i{-4096}; i <= 4096; ++i)
{
@ -42,13 +41,12 @@ public:
{
using lossless_traits = lossless_traits<uint8_t, 8>;
const auto traits1{default_traits<uint8_t, uint8_t>(255, 0)};
const lossless_traits traits2{};
Assert::IsTrue(traits1.limit == traits2.limit);
Assert::IsTrue(traits1.maximum_sample_value == traits2.maximum_sample_value);
Assert::IsTrue(traits1.reset_threshold == traits2.reset_threshold);
Assert::IsTrue(traits1.bits_per_pixel == traits2.bits_per_pixel);
Assert::IsTrue(traits1.quantized_bits_per_pixel == traits2.quantized_bits_per_pixel);
Assert::IsTrue(traits1.limit == lossless_traits::limit);
Assert::IsTrue(traits1.maximum_sample_value == lossless_traits::maximum_sample_value);
Assert::IsTrue(traits1.reset_threshold == lossless_traits::reset_threshold);
Assert::IsTrue(traits1.bits_per_pixel == lossless_traits::bits_per_pixel);
Assert::IsTrue(traits1.quantized_bits_per_pixel == lossless_traits::quantized_bits_per_pixel);
for (int i{-255}; i <= 255; ++i)
{

View File

@ -15,7 +15,7 @@ namespace {
/// <summary>
/// This is the original algorithm of ISO/IEC 14495-1, A.5.2, Code Segment A.11 (second else branch)
/// It will map signed values to unsiged values.
/// It will map signed values to unsigned values.
/// </summary>
int32_t map_error_value_original(const int32_t error_value) noexcept
{

BIN
unittest/test16_rm_5.jls Normal file

Binary file not shown.

After

Width:  |  Height:  |  Size: 67 KiB

Binary file not shown.

After

Width:  |  Height:  |  Size: 112 KiB

Binary file not shown.

After

Width:  |  Height:  |  Size: 117 KiB

Binary file not shown.

After

Width:  |  Height:  |  Size: 97 KiB

Binary file not shown.

After

Width:  |  Height:  |  Size: 108 KiB

View File

@ -130,7 +130,7 @@ vector<uint8_t> create_test_spiff_header(const uint8_t high_version, const uint8
buffer.push_back(0);
buffer.push_back(96);
// header.horizontal_resolution = 1024;
// header.horizontal_resolution = 1024
buffer.push_back(0);
buffer.push_back(0);
buffer.push_back(4);
@ -202,4 +202,89 @@ void test_round_trip_legacy(const vector<uint8_t>& source, const JlsParameters&
RESTORE_DEPRECATED_WARNING
}
bool verify_encoded_bytes(const vector<uint8_t>& uncompressed_source, const vector<uint8_t>& encoded_source)
{
const jpegls_decoder decoder{encoded_source, true};
jpegls_encoder encoder;
encoder.frame_info(decoder.frame_info())
.interleave_mode(decoder.interleave_mode())
.near_lossless(decoder.near_lossless())
.preset_coding_parameters(decoder.preset_coding_parameters());
vector<uint8_t> our_encoded_bytes(encoded_source.size() + 16);
encoder.destination(our_encoded_bytes);
const size_t bytes_written{encoder.encode(uncompressed_source)};
if (bytes_written != encoded_source.size())
return false;
for (size_t i{}; i != encoded_source.size(); ++i)
{
if (encoded_source[i] != our_encoded_bytes[i])
{
return false;
}
}
return true;
}
void test_compliance(const vector<uint8_t>& encoded_source, const vector<uint8_t>& uncompressed_source,
const bool check_encode)
{
const jpegls_decoder decoder{encoded_source, true};
if (check_encode)
{
Assert::IsTrue(verify_encoded_bytes(uncompressed_source, encoded_source));
}
vector<uint8_t> destination(decoder.destination_size());
decoder.decode(destination);
if (decoder.near_lossless() == 0)
{
for (size_t i{}; i != uncompressed_source.size(); ++i)
{
if (uncompressed_source[i] != destination[i]) // AreEqual is very slow, pre-test to speed up 50X
{
Assert::AreEqual(uncompressed_source[i], destination[i]);
}
}
}
else
{
const frame_info frame_info{decoder.frame_info()};
const auto near_lossless{decoder.near_lossless()};
if (frame_info.bits_per_sample <= 8)
{
for (size_t i{}; i != uncompressed_source.size(); ++i)
{
if (abs(uncompressed_source[i] - destination[i]) >
near_lossless) // AreEqual is very slow, pre-test to speed up 50X
{
Assert::AreEqual(uncompressed_source[i], destination[i]);
}
}
}
else
{
const auto* source16 = reinterpret_cast<const uint16_t*>(uncompressed_source.data());
const auto* destination16 = reinterpret_cast<const uint16_t*>(destination.data());
for (size_t i{}; i != uncompressed_source.size() / 2; ++i)
{
if (abs(source16[i] - destination16[i]) > near_lossless) // AreEqual is very slow, pre-test to speed up 50X
{
Assert::AreEqual(static_cast<int>(source16[i]), static_cast<int>(destination16[i]));
}
}
}
}
}
}} // namespace charls::test

View File

@ -52,6 +52,10 @@ std::vector<uint8_t> create_test_spiff_header(uint8_t high_version = 2, uint8_t
bool end_of_directory = true);
std::vector<uint8_t> create_noise_image_16_bit(size_t pixel_count, int bit_count, uint32_t seed);
void test_round_trip_legacy(const std::vector<uint8_t>& source, const JlsParameters& params);
bool verify_encoded_bytes(const std::vector<uint8_t>& uncompressed_source, const std::vector<uint8_t>& encoded_source);
void test_compliance(const std::vector<uint8_t>& encoded_source, const std::vector<uint8_t>& uncompressed_source,
bool check_encode);
/// <summary>
/// Computes how many bytes are needed to hold the number of bits.
@ -61,29 +65,6 @@ constexpr uint32_t bit_to_byte_count(const int32_t bit_count) noexcept
return static_cast<uint32_t>((bit_count + 7) / 8);
}
}} // namespace charls::test
// ReSharper disable CppInconsistentNaming
namespace Microsoft { namespace VisualStudio { namespace CppUnitTestFramework {
// ReSharper restore CppInconsistentNaming
template<>
inline std::wstring ToString<charls::jpegls_errc>(const charls::jpegls_errc& q)
{
RETURN_WIDE_STRING(static_cast<int>(q));
}
template<>
inline std::wstring ToString<charls::interleave_mode>(const charls::interleave_mode& q)
{
RETURN_WIDE_STRING(static_cast<int>(q));
}
}}} // namespace Microsoft::VisualStudio::CppUnitTestFramework
namespace charls { namespace test {
template<typename Functor>
void assert_expect_exception(const jpegls_errc error_value, Functor functor)
{
@ -106,4 +87,23 @@ void assert_expect_exception(const jpegls_errc error_value, Functor functor)
Microsoft::VisualStudio::CppUnitTestFramework::Assert::Fail();
}
}} // namespace charls::test
// ReSharper disable CppInconsistentNaming
namespace Microsoft { namespace VisualStudio { namespace CppUnitTestFramework {
// ReSharper restore CppInconsistentNaming
template<>
inline std::wstring ToString<charls::jpegls_errc>(const charls::jpegls_errc& q)
{
RETURN_WIDE_STRING(static_cast<int>(q));
}
template<>
inline std::wstring ToString<charls::interleave_mode>(const charls::interleave_mode& q)
{
RETURN_WIDE_STRING(static_cast<int>(q));
}
}}} // namespace Microsoft::VisualStudio::CppUnitTestFramework