Refactor jpeg_stream_reader (#133)

- Use a span to manage the segment data
- Use a forward iterator
- Parse the SOS segment directly.
- Do a size check when segment size is known, then read without size checks.
- Don't use std::vector to read bytes.
This commit is contained in:
Victor Derks 2022-01-08 22:17:02 +01:00 committed by GitHub
parent b16aa20c5a
commit 2613ab21e6
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23
13 changed files with 298 additions and 238 deletions

View File

@ -108,14 +108,10 @@
<ItemDefinitionGroup Condition="'$(Configuration)'=='Debug' OR '$(Configuration)'=='Checked'">
<ClCompile>
<Optimization>Disabled</Optimization>
<PreprocessorDefinitions>_DEBUG;%(PreprocessorDefinitions)</PreprocessorDefinitions>
</ClCompile>
<Link>
<RandomizedBaseAddress>false</RandomizedBaseAddress>
</Link>
<ResourceCompile>
<PreprocessorDefinitions>_DEBUG;%(PreprocessorDefinitions)</PreprocessorDefinitions>
</ResourceCompile>
</ItemDefinitionGroup>
<PropertyGroup Condition="'$(Configuration)'=='Checked'">
@ -149,6 +145,9 @@
<!-- Mark executable image compatible with the Control-flow Enforcement Technology (CET) Shadow Stack. -->
<CETCompat>true</CETCompat>
</Link>
<ResourceCompile>
<PreprocessorDefinitions>NDEBUG;%(PreprocessorDefinitions)</PreprocessorDefinitions>
</ResourceCompile>
</ItemDefinitionGroup>
</Project>

View File

@ -5,6 +5,7 @@
<Rule Id="C26426" Action="None" />
<Rule Id="C26429" Action="None" />
<Rule Id="C26446" Action="None" />
<Rule Id="C26459" Action="None" />
<Rule Id="C26472" Action="None" />
<Rule Id="C26481" Action="None" />
<Rule Id="C26482" Action="None" />

View File

@ -9,6 +9,9 @@ C26429: Symbol 'xxx' is never tested for nullness, it can be marked as not_null
C26446: Prefer to use gsl::at() instead of unchecked subscript operator.
-> Rationale: CharLS require good performance, gsl:at() cannot be used. debug STL already checks.
C26459: You called an STL function '' with a raw pointer parameter. Consider wrapping your range in a gsl::span and pass as a span iterator (stl.1)
Rationale: gsl:span() cannot be used. Update to std:span when available (C++20).
C26472: Don't use static_cast for arithmetic conversions
-> Rationale: can only be solved with gsl::narrow_cast

View File

@ -3,6 +3,8 @@
#pragma once
#include "charls/annotations.h"
#include <cstddef>
#include <cstdint>
@ -13,15 +15,25 @@ struct byte_span final
{
byte_span() = default;
byte_span(void* data_arg, const size_t size_arg) noexcept : data{static_cast<uint8_t*>(data_arg)}, size{size_arg}
constexpr byte_span(void* data_arg, const size_t size_arg) noexcept : data{static_cast<uint8_t*>(data_arg)}, size{size_arg}
{
}
byte_span(const void* data_arg, const size_t size_arg) noexcept :
constexpr byte_span(const void* data_arg, const size_t size_arg) noexcept :
data{static_cast<uint8_t*>(const_cast<void*>(data_arg))}, size{size_arg}
{
}
CHARLS_CHECK_RETURN constexpr uint8_t* begin() const noexcept
{
return data;
}
CHARLS_CHECK_RETURN constexpr uint8_t* end() const noexcept
{
return data + size;
}
uint8_t* data{};
size_t size{};
};
@ -31,22 +43,43 @@ struct byte_span final
class const_byte_span final
{
public:
using iterator = const uint8_t*;
const_byte_span() = default;
const_byte_span(const void* data, const size_t size) noexcept : data_{static_cast<const uint8_t*>(data)}, size_{size}
constexpr const_byte_span(const void* data, const size_t size) noexcept : data_{static_cast<const uint8_t*>(data)}, size_{size}
{
}
constexpr size_t size() const noexcept
explicit constexpr const_byte_span(const uint8_t* first, const uint8_t* last) noexcept :
const_byte_span(first, last - first)
{}
CHARLS_CHECK_RETURN constexpr size_t size() const noexcept
{
return size_;
}
constexpr const uint8_t* data() const noexcept
CHARLS_CHECK_RETURN constexpr const uint8_t* data() const noexcept
{
return data_;
}
CHARLS_CHECK_RETURN constexpr iterator begin() const noexcept
{
return data_;
}
CHARLS_CHECK_RETURN constexpr iterator end() const noexcept
{
return data_ + size_;
}
CHARLS_CHECK_RETURN constexpr bool empty() const noexcept
{
return size_ == 0;
}
private:
const uint8_t* data_{};
size_t size_{};

View File

@ -23,7 +23,7 @@ VS_VERSION_INFO VERSIONINFO
FILEVERSION CHARLS_VERSION_MAJOR, CHARLS_VERSION_MINOR, CHARLS_VERSION_PATCH, 0
PRODUCTVERSION CHARLS_VERSION_MAJOR, CHARLS_VERSION_MINOR, CHARLS_VERSION_PATCH, 0
FILEFLAGSMASK VS_FFI_FILEFLAGSMASK
#ifdef _DEBUG
#ifndef NDEBUG
FILEFLAGS VS_FF_DEBUG
#else
FILEFLAGS 0x0L

View File

@ -14,7 +14,7 @@ struct charls_jpegls_decoder final
{
void source(const const_byte_span source)
{
check_argument(source.data() || source.size() == 0);
check_argument(source.data() || source.empty());
check_operation(state_ == state::initial);
reader_.source({source.data(), source.size()});
@ -41,7 +41,6 @@ struct charls_jpegls_decoder final
reader_.read_header();
}
reader_.read_start_of_scan();
state_ = state::header_read;
}
@ -114,8 +113,9 @@ struct charls_jpegls_decoder final
check_argument(destination.data || destination.size == 0);
check_operation(state_ == state::header_read);
reader_.read(destination, stride);
reader_.decode(destination, stride);
reader_.read_end_of_image();
state_ = state::completed;
}

View File

@ -130,7 +130,7 @@ struct charls_jpegls_encoder final
void write_comment(const const_byte_span comment)
{
check_argument(comment.data() || comment.size() == 0);
check_argument(comment.data() || comment.empty());
check_argument(comment.size() <= segment_max_data_size, jpegls_errc::invalid_argument_size);
check_operation(state_ >= state::destination_set && state_ < state::completed);

View File

@ -31,12 +31,12 @@ public:
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, uint32_t restart_interval) = 0;
virtual void decode_scan(std::unique_ptr<process_line> output_data, const JlsRect& size, byte_span& compressed_data) = 0;
virtual size_t decode_scan(std::unique_ptr<process_line> output_data, const JlsRect& size, const_byte_span encoded_source) = 0;
void initialize(const byte_span source)
void initialize(const const_byte_span source)
{
position_ = source.data;
end_position_ = position_ + source.size;
position_ = source.data();
end_position_ = source.end();
find_jpeg_marker_start_byte();
fill_read_cache();

View File

@ -23,7 +23,6 @@ using std::array;
using std::equal;
using std::find;
using std::unique_ptr;
using std::vector;
namespace {
@ -36,10 +35,12 @@ constexpr bool is_restart_marker_code(const jpeg_marker_code marker_code) noexce
} // namespace
void jpeg_stream_reader::source(const byte_span source) noexcept
void jpeg_stream_reader::source(const const_byte_span source) noexcept
{
ASSERT(state_ == state::before_start_of_image);
source_ = source;
position_ = source.begin();
end_position_ = source.end();
}
@ -52,6 +53,7 @@ void jpeg_stream_reader::read_header(spiff_header* header, bool* spiff_header_fo
if (UNLIKELY(read_next_marker_code() != jpeg_marker_code::start_of_image))
throw_jpegls_error(jpegls_errc::start_of_image_marker_not_found);
component_ids_.reserve(4); // expect 4 components or less.
state_ = state::header_section;
}
@ -59,45 +61,36 @@ void jpeg_stream_reader::read_header(spiff_header* header, bool* spiff_header_fo
{
const jpeg_marker_code marker_code = read_next_marker_code();
validate_marker_code(marker_code);
read_segment_size();
if (marker_code == jpeg_marker_code::start_of_scan)
{
state_ = state::scan_section;
return;
}
const int32_t segment_size{read_segment_size()};
int bytes_read;
switch (state_)
{
case state::spiff_header_section:
bytes_read = read_spiff_directory_entry(marker_code, segment_size - 2) + 2;
read_spiff_directory_entry(marker_code);
break;
default:
bytes_read = read_marker_segment(marker_code, segment_size - 2, header, spiff_header_found) + 2;
read_marker_segment(marker_code, header, spiff_header_found);
break;
}
const int padding_to_read{segment_size - bytes_read};
if (UNLIKELY(padding_to_read < 0))
throw_jpegls_error(jpegls_errc::invalid_marker_segment_size);
for (int i{}; i != padding_to_read; ++i)
{
skip_byte();
}
ASSERT(segment_data_.end() - position_ == 0); // All segment data should be processed.
if (state_ == state::header_section && spiff_header_found && *spiff_header_found)
{
state_ = state::spiff_header_section;
return;
}
if (state_ == state::bit_stream_section)
{
return;
}
}
}
void jpeg_stream_reader::read(byte_span destination, size_t stride)
void jpeg_stream_reader::decode(byte_span destination, size_t stride)
{
ASSERT(state_ == state::bit_stream_section);
@ -135,8 +128,9 @@ void jpeg_stream_reader::read(byte_span destination, size_t stride)
const unique_ptr<decoder_strategy> codec{jls_codec_factory<decoder_strategy>().create_codec(
frame_info_, parameters_, get_validated_preset_coding_parameters())};
unique_ptr<process_line> process_line(codec->create_process_line(destination, stride));
codec->decode_scan(move(process_line), rect_, source_);
skip_bytes(destination, static_cast<size_t>(bytes_per_plane));
const size_t bytes_read = codec->decode_scan(move(process_line), rect_, const_byte_span{position_, end_position_});
advance_position(bytes_read);
charls::skip_bytes(destination, static_cast<size_t>(bytes_per_plane));
state_ = state::scan_section;
if (parameters_.interleave_mode != interleave_mode::none)
@ -161,60 +155,32 @@ void jpeg_stream_reader::read_end_of_image()
}
vector<uint8_t> jpeg_stream_reader::read_bytes(const size_t byte_count)
{
vector<uint8_t> result;
result.reserve(byte_count);
for (size_t i{}; i != byte_count; ++i)
{
result.push_back(read_byte());
}
return result;
}
void jpeg_stream_reader::read_next_start_of_scan()
{
ASSERT(state_ == state::scan_section);
for (;;)
do
{
const jpeg_marker_code marker_code{read_next_marker_code()};
validate_marker_code(marker_code);
read_segment_size();
read_marker_segment(marker_code);
if (marker_code == jpeg_marker_code::start_of_scan)
{
read_start_of_scan();
return;
}
const int32_t segment_size{read_segment_size()};
const int bytes_read{read_marker_segment(marker_code, segment_size - 2) + 2};
const int padding_to_read{segment_size - bytes_read};
if (UNLIKELY(padding_to_read < 0))
throw_jpegls_error(jpegls_errc::invalid_marker_segment_size);
for (int i{}; i != padding_to_read; ++i)
{
skip_byte();
}
}
ASSERT(segment_data_.end() - position_ == 0); // All segment data should be processed.
} while (state_ == state::scan_section);
}
jpeg_marker_code jpeg_stream_reader::read_next_marker_code()
USE_DECL_ANNOTATIONS jpeg_marker_code jpeg_stream_reader::read_next_marker_code()
{
auto byte{read_byte()};
auto byte{read_byte_checked()};
if (UNLIKELY(byte != jpeg_marker_start_byte))
throw_jpegls_error(jpegls_errc::jpeg_marker_start_byte_not_found);
// Read all preceding 0xFF fill values until a non 0xFF value has been found. (see T.81, B.1.1.2)
// Read all preceding 0xFF fill values until a non 0xFF value has been found. (see ISO/IEC 10918-1, B.1.1.2)
do
{
byte = read_byte();
byte = read_byte_checked();
} while (byte == jpeg_marker_start_byte);
return static_cast<jpeg_marker_code>(byte);
@ -289,34 +255,46 @@ void jpeg_stream_reader::validate_marker_code(const jpeg_marker_code marker_code
}
jpegls_pc_parameters jpeg_stream_reader::get_validated_preset_coding_parameters() const
USE_DECL_ANNOTATIONS jpegls_pc_parameters jpeg_stream_reader::get_validated_preset_coding_parameters() const
{
jpegls_pc_parameters preset_coding_parameters;
if (UNLIKELY(!is_valid(preset_coding_parameters_, calculate_maximum_sample_value(frame_info_.bits_per_sample),
parameters_.near_lossless, &preset_coding_parameters)))
parameters_.near_lossless, &preset_coding_parameters)))
throw_jpegls_error(jpegls_errc::invalid_parameter_jpegls_pc_parameters);
return preset_coding_parameters;
}
int jpeg_stream_reader::read_marker_segment(const jpeg_marker_code marker_code, const int32_t segment_size,
spiff_header* header, bool* spiff_header_found)
void jpeg_stream_reader::read_marker_segment(const jpeg_marker_code marker_code, spiff_header* header,
bool* spiff_header_found)
{
switch (marker_code)
{
case jpeg_marker_code::start_of_frame_jpegls:
return read_start_of_frame_segment(segment_size);
read_start_of_frame_segment();
break;
case jpeg_marker_code::comment:
return read_comment(segment_size);
case jpeg_marker_code::start_of_scan:
read_start_of_scan();
break;
case jpeg_marker_code::jpegls_preset_parameters:
return read_preset_parameters_segment(segment_size);
read_preset_parameters_segment();
break;
case jpeg_marker_code::define_restart_interval:
return read_define_restart_interval(segment_size);
read_define_restart_interval();
break;
case jpeg_marker_code::application_data8:
try_read_application_data8_segment(header, spiff_header_found);
break;
case jpeg_marker_code::comment:
read_comment();
break;
case jpeg_marker_code::application_data0:
case jpeg_marker_code::application_data1:
@ -333,46 +311,43 @@ int jpeg_stream_reader::read_marker_segment(const jpeg_marker_code marker_code,
case jpeg_marker_code::application_data13:
case jpeg_marker_code::application_data14:
case jpeg_marker_code::application_data15:
return 0;
case jpeg_marker_code::application_data8:
return try_read_application_data8_segment(segment_size, header, spiff_header_found);
read_application_data();
break;
// Other tags not supported (among which DNL)
default:
ASSERT(false);
return 0;
break;
}
}
int jpeg_stream_reader::read_spiff_directory_entry(const jpeg_marker_code marker_code, const int32_t segment_size)
void jpeg_stream_reader::read_spiff_directory_entry(const jpeg_marker_code marker_code)
{
if (UNLIKELY(marker_code != jpeg_marker_code::application_data8))
throw_jpegls_error(jpegls_errc::missing_end_of_spiff_directory);
if (UNLIKELY(segment_size < 4))
throw_jpegls_error(jpegls_errc::invalid_marker_segment_size);
check_minimal_segment_size(4);
const uint32_t spiff_directory_type{read_uint32()};
if (spiff_directory_type == spiff_end_of_directory_entry_type)
{
check_segment_size(6); // 4 + 2 for dummy SOI.
state_ = state::image_section;
}
return 4;
skip_remaining_segment_data();
}
int jpeg_stream_reader::read_start_of_frame_segment(const int32_t segment_size)
void jpeg_stream_reader::read_start_of_frame_segment()
{
// A JPEG-LS Start of Frame (SOF) segment is documented in ISO/IEC 14495-1, C.2.2
// This section references ISO/IEC 10918-1, B.2.2, which defines the normal JPEG SOF,
// with some modifications.
if (UNLIKELY(segment_size < 6))
throw_jpegls_error(jpegls_errc::invalid_marker_segment_size);
check_minimal_segment_size(6);
frame_info_.bits_per_sample = read_byte();
if (UNLIKELY(frame_info_.bits_per_sample < minimum_bits_per_sample || frame_info_.bits_per_sample > maximum_bits_per_sample))
if (UNLIKELY(frame_info_.bits_per_sample < minimum_bits_per_sample ||
frame_info_.bits_per_sample > maximum_bits_per_sample))
throw_jpegls_error(jpegls_errc::invalid_parameter_bits_per_sample);
frame_info_.height = read_uint16();
@ -387,9 +362,7 @@ int jpeg_stream_reader::read_start_of_frame_segment(const int32_t segment_size)
if (UNLIKELY(frame_info_.component_count < 1))
throw_jpegls_error(jpegls_errc::invalid_parameter_component_count);
if (UNLIKELY(segment_size != 6 + (frame_info_.component_count * 3)))
throw_jpegls_error(jpegls_errc::invalid_marker_segment_size);
check_segment_size((static_cast<size_t>(frame_info_.component_count) * 3) + 6);
for (int32_t i{}; i != frame_info_.component_count; ++i)
{
// Component specification parameters
@ -403,36 +376,34 @@ int jpeg_stream_reader::read_start_of_frame_segment(const int32_t segment_size)
}
state_ = state::scan_section;
return segment_size;
}
int jpeg_stream_reader::read_comment(const int32_t segment_size) const
void jpeg_stream_reader::read_comment()
{
if (UNLIKELY(static_cast<size_t>(segment_size) > source_.size))
throw_jpegls_error(jpegls_errc::source_buffer_too_small);
if (UNLIKELY(comment_handler_ && static_cast<bool>(comment_handler_(segment_size > 0 ? source_.data : nullptr, segment_size,
comment_handler_user_context_))))
if (comment_handler_ &&
UNLIKELY(static_cast<bool>(comment_handler_(segment_data_.empty() ? nullptr : position_, segment_data_.size(),
comment_handler_user_context_))))
throw_jpegls_error(jpegls_errc::callback_failed);
return 0;
skip_remaining_segment_data();
}
int jpeg_stream_reader::read_preset_parameters_segment(const int32_t segment_size)
void jpeg_stream_reader::read_application_data() noexcept
{
if (UNLIKELY(segment_size < 1))
throw_jpegls_error(jpegls_errc::invalid_marker_segment_size);
skip_remaining_segment_data();
}
void jpeg_stream_reader::read_preset_parameters_segment()
{
check_minimal_segment_size(1);
const auto type{static_cast<jpegls_preset_parameters_type>(read_byte())};
switch (type)
{
case jpegls_preset_parameters_type::preset_coding_parameters: {
constexpr int32_t coding_parameter_segment_size = 11;
if (UNLIKELY(segment_size != coding_parameter_segment_size))
throw_jpegls_error(jpegls_errc::invalid_marker_segment_size);
case jpegls_preset_parameters_type::preset_coding_parameters:
check_segment_size(1 + (5 * sizeof(uint16_t)));
// Note: validation will be done, just before decoding as more info is needed for validation.
preset_coding_parameters_.maximum_sample_value = read_uint16();
@ -440,9 +411,7 @@ int jpeg_stream_reader::read_preset_parameters_segment(const int32_t segment_siz
preset_coding_parameters_.threshold2 = read_uint16();
preset_coding_parameters_.threshold3 = read_uint16();
preset_coding_parameters_.reset_value = read_uint16();
return coding_parameter_segment_size;
}
return;
case jpegls_preset_parameters_type::mapping_table_specification:
case jpegls_preset_parameters_type::mapping_table_continuation:
@ -464,23 +433,23 @@ 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)
void jpeg_stream_reader::read_define_restart_interval()
{
// 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)
switch (segment_data_.size())
{
case 2:
parameters_.restart_interval = read_uint16();
return 2;
break;
case 3:
parameters_.restart_interval = read_uint24();
return 3;
break;
case 4:
parameters_.restart_interval = read_uint32();
return 4;
break;
default:
throw_jpegls_error(jpegls_errc::invalid_marker_segment_size);
@ -490,18 +459,15 @@ int jpeg_stream_reader::read_define_restart_interval(const int32_t segment_size)
void jpeg_stream_reader::read_start_of_scan()
{
const int32_t segment_size{read_segment_size()};
if (UNLIKELY(segment_size < 3))
throw_jpegls_error(jpegls_errc::invalid_marker_segment_size);
const int component_count_in_scan{read_byte()};
if (UNLIKELY(component_count_in_scan != 1 && component_count_in_scan != frame_info_.component_count))
check_minimal_segment_size(1);
const size_t component_count_in_scan{read_byte()};
if (UNLIKELY(component_count_in_scan != 1 &&
component_count_in_scan != static_cast<size_t>(frame_info_.component_count)))
throw_jpegls_error(jpegls_errc::parameter_value_not_supported);
if (UNLIKELY(segment_size != 6 + (2 * component_count_in_scan)))
throw_jpegls_error(jpegls_errc::invalid_marker_segment_size);
check_segment_size((component_count_in_scan * 2) + 4);
for (int i{}; i != component_count_in_scan; ++i)
for (size_t i{}; i != component_count_in_scan; ++i)
{
skip_byte(); // Skip scan component selector
const uint8_t mapping_table_selector{read_byte()};
@ -524,63 +490,103 @@ void jpeg_stream_reader::read_start_of_scan()
}
uint8_t jpeg_stream_reader::read_byte()
USE_DECL_ANNOTATIONS uint8_t jpeg_stream_reader::read_byte_checked()
{
if (UNLIKELY(source_.size == 0))
if (UNLIKELY(position_ == end_position_))
throw_jpegls_error(jpegls_errc::source_buffer_too_small);
const uint8_t value{source_.data[0]};
skip_bytes(source_, 1);
return read_byte();
}
USE_DECL_ANNOTATIONS uint16_t jpeg_stream_reader::read_uint16_checked()
{
if (UNLIKELY(position_ + sizeof(uint16_t) > end_position_))
throw_jpegls_error(jpegls_errc::source_buffer_too_small);
return read_uint16();
}
USE_DECL_ANNOTATIONS uint8_t jpeg_stream_reader::read_byte() noexcept
{
ASSERT(position_ != end_position_);
const uint8_t value{*position_};
advance_position(1);
return value;
}
void jpeg_stream_reader::skip_byte()
void jpeg_stream_reader::skip_byte() noexcept
{
std::ignore = read_byte();
advance_position(1);
}
uint16_t jpeg_stream_reader::read_uint16()
USE_DECL_ANNOTATIONS uint16_t jpeg_stream_reader::read_uint16() noexcept
{
if (UNLIKELY(source_.size < 2))
throw_jpegls_error(jpegls_errc::source_buffer_too_small);
ASSERT(position_ + sizeof(uint16_t) <= end_position_);
const auto value{read_unaligned<uint16_t>(source_.data)};
skip_bytes(source_, 2);
return byte_swap(value);
const auto value{byte_swap(read_unaligned<uint16_t>(position_))};
advance_position(2);
return value;
}
uint32_t jpeg_stream_reader::read_uint24()
USE_DECL_ANNOTATIONS uint32_t jpeg_stream_reader::read_uint24() noexcept
{
const uint32_t value{static_cast<uint32_t>(read_byte()) << 16U};
const uint32_t value{static_cast<uint32_t>(read_uint8()) << 16U};
return value + read_uint16();
}
uint32_t jpeg_stream_reader::read_uint32()
USE_DECL_ANNOTATIONS uint32_t jpeg_stream_reader::read_uint32() noexcept
{
if (UNLIKELY(source_.size < 4))
throw_jpegls_error(jpegls_errc::source_buffer_too_small);
ASSERT(position_ + sizeof(uint32_t) <= end_position_);
const auto value{read_unaligned<uint32_t>(source_.data)};
skip_bytes(source_, 4);
return byte_swap(value);
const auto value{byte_swap(read_unaligned<uint32_t>(position_))};
advance_position(4);
return value;
}
int32_t jpeg_stream_reader::read_segment_size()
USE_DECL_ANNOTATIONS const_byte_span jpeg_stream_reader::read_bytes(const size_t byte_count) noexcept
{
const int32_t segment_size{read_uint16()};
if (UNLIKELY(segment_size < 2))
ASSERT(position_ + byte_count <= end_position_);
const const_byte_span bytes{position_, byte_count};
advance_position(byte_count);
return bytes;
}
void jpeg_stream_reader::read_segment_size()
{
constexpr size_t segment_length{2}; // The segment size also includes the length of the segment length bytes.
const size_t segment_size{read_uint16_checked()};
segment_data_ = {position_, segment_size - segment_length};
if (UNLIKELY(segment_size < segment_length || position_ + segment_data_.size() > end_position_))
throw_jpegls_error(jpegls_errc::invalid_marker_segment_size);
return segment_size;
}
int jpeg_stream_reader::try_read_application_data8_segment(const int32_t segment_size, spiff_header* header,
bool* spiff_header_found)
void jpeg_stream_reader::check_minimal_segment_size(const size_t minimum_size) const
{
if (UNLIKELY(minimum_size > segment_data_.size()))
throw_jpegls_error(jpegls_errc::invalid_marker_segment_size);
}
void jpeg_stream_reader::check_segment_size(const size_t expected_size) const
{
if (UNLIKELY(expected_size != segment_data_.size()))
throw_jpegls_error(jpegls_errc::invalid_marker_segment_size);
}
void jpeg_stream_reader::try_read_application_data8_segment(spiff_header* header, bool* spiff_header_found)
{
if (spiff_header_found)
{
@ -588,21 +594,26 @@ int jpeg_stream_reader::try_read_application_data8_segment(const int32_t segment
*spiff_header_found = false;
}
if (segment_size == 5)
return try_read_hp_color_transform_segment();
if (segment_data_.size() == 5)
{
try_read_hp_color_transform_segment();
}
else if (header && spiff_header_found && segment_data_.size() >= 30)
{
try_read_spiff_header_segment(*header, *spiff_header_found);
}
if (header && spiff_header_found && segment_size >= 30)
return try_read_spiff_header_segment(*header, *spiff_header_found);
return 0;
skip_remaining_segment_data();
}
int jpeg_stream_reader::try_read_hp_color_transform_segment()
void jpeg_stream_reader::try_read_hp_color_transform_segment()
{
ASSERT(segment_data_.size() == 5);
const array<uint8_t, 4> mrfx_tag{'m', 'r', 'f', 'x'}; // mrfx = xfrm (in big endian) = colorXFoRM
if (!equal(mrfx_tag.cbegin(), mrfx_tag.cend(), read_bytes(4).cbegin()))
return 4;
if (!equal(mrfx_tag.cbegin(), mrfx_tag.cend(), read_bytes(4).begin()))
return;
const auto transformation{read_byte()};
switch (transformation)
@ -612,7 +623,7 @@ int jpeg_stream_reader::try_read_hp_color_transform_segment()
case static_cast<uint8_t>(color_transformation::hp2):
case static_cast<uint8_t>(color_transformation::hp3):
parameters_.transformation = static_cast<color_transformation>(transformation);
return 5;
return;
case 4: // RgbAsYuvLossy (The standard lossy RGB to YCbCr transform used in JPEG.)
case 5: // Matrix (transformation is controlled using a matrix that is also stored in the segment.
@ -623,26 +634,17 @@ int jpeg_stream_reader::try_read_hp_color_transform_segment()
}
}
template<typename EnumType, EnumType Low, EnumType High>
EnumType enum_cast(uint8_t value)
USE_DECL_ANNOTATIONS void jpeg_stream_reader::try_read_spiff_header_segment(spiff_header& header, bool& spiff_header_found)
{
if (UNLIKELY(value < static_cast<uint8_t>(Low)))
throw_jpegls_error(jpegls_errc::invalid_encoded_data);
ASSERT(segment_data_.size() >= 30);
if (UNLIKELY(value > static_cast<uint8_t>(High)))
throw_jpegls_error(jpegls_errc::invalid_encoded_data);
return static_cast<EnumType>(value);
}
USE_DECL_ANNOTATIONS int jpeg_stream_reader::try_read_spiff_header_segment(spiff_header& header, bool& spiff_header_found)
{
const array<uint8_t, 6> spiff_tag{'S', 'P', 'I', 'F', 'F', 0};
if (!equal(spiff_tag.cbegin(), spiff_tag.cend(), read_bytes(6).cbegin()))
if (!equal(spiff_tag.cbegin(), spiff_tag.cend(), read_bytes(6).begin()))
{
header = {};
spiff_header_found = false;
return 6;
return;
}
const auto high_version{read_byte()};
@ -650,7 +652,7 @@ USE_DECL_ANNOTATIONS int jpeg_stream_reader::try_read_spiff_header_segment(spiff
{
header = {};
spiff_header_found = false;
return 7; // Treat unknown versions as if the SPIFF header doesn't exists.
return; // Treat unknown versions as if the SPIFF header doesn't exists.
}
skip_byte(); // low version
@ -667,7 +669,6 @@ USE_DECL_ANNOTATIONS int jpeg_stream_reader::try_read_spiff_header_segment(spiff
header.horizontal_resolution = read_uint32();
spiff_header_found = true;
return 30;
}
@ -705,7 +706,7 @@ void jpeg_stream_reader::check_interleave_mode(const interleave_mode mode) const
}
uint32_t jpeg_stream_reader::maximum_sample_value() const noexcept
USE_DECL_ANNOTATIONS uint32_t jpeg_stream_reader::maximum_sample_value() const noexcept
{
if (preset_coding_parameters_.maximum_sample_value != 0)
return static_cast<uint32_t>(preset_coding_parameters_.maximum_sample_value);
@ -714,4 +715,11 @@ uint32_t jpeg_stream_reader::maximum_sample_value() const noexcept
}
void jpeg_stream_reader::skip_remaining_segment_data() noexcept
{
const auto bytes_still_to_read{segment_data_.end() - position_};
advance_position(bytes_still_to_read);
}
} // namespace charls

View File

@ -3,12 +3,13 @@
#pragma once
#include "byte_span.h"
#include "charls/public_types.h"
#include "byte_span.h"
#include "coding_parameters.h"
#include "util.h"
#include <cstdint>
#include <vector>
namespace charls {
@ -26,7 +27,7 @@ public:
jpeg_stream_reader(jpeg_stream_reader&&) = default;
jpeg_stream_reader& operator=(jpeg_stream_reader&&) = default;
void source(byte_span source) noexcept;
void source(const_byte_span source) noexcept;
const charls::frame_info& frame_info() const noexcept
{
@ -60,39 +61,55 @@ public:
}
void read_header(spiff_header* header = nullptr, bool* spiff_header_found = nullptr);
void read(byte_span destination, size_t stride);
void decode(byte_span destination, size_t stride);
void read_end_of_image();
void read_start_of_scan();
private:
uint8_t read_byte();
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);
void advance_position(const size_t count) noexcept
{
ASSERT(position_ + count <= end_position_);
position_ += count;
}
CHARLS_CHECK_RETURN uint8_t read_byte_checked();
CHARLS_CHECK_RETURN uint16_t read_uint16_checked();
CHARLS_CHECK_RETURN uint8_t read_byte() noexcept;
void skip_byte() noexcept;
CHARLS_CHECK_RETURN uint8_t read_uint8() noexcept
{
return read_byte();
}
CHARLS_CHECK_RETURN uint16_t read_uint16() noexcept;
CHARLS_CHECK_RETURN uint32_t read_uint24() noexcept;
CHARLS_CHECK_RETURN uint32_t read_uint32() noexcept;
CHARLS_CHECK_RETURN const_byte_span read_bytes(size_t byte_count) noexcept;
void read_segment_size();
void check_minimal_segment_size(size_t minimum_size) const;
void check_segment_size(size_t expected_size) const;
void read_next_start_of_scan();
jpeg_marker_code read_next_marker_code();
CHARLS_CHECK_RETURN jpeg_marker_code read_next_marker_code();
void validate_marker_code(jpeg_marker_code marker_code) const;
jpegls_pc_parameters get_validated_preset_coding_parameters() const;
int read_marker_segment(jpeg_marker_code marker_code, int32_t segment_size, spiff_header* header = nullptr,
bool* spiff_header_found = nullptr);
int read_spiff_directory_entry(jpeg_marker_code marker_code, int32_t segment_size);
int read_start_of_frame_segment(int32_t segment_size);
int read_comment(int32_t segment_size) const;
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(CHARLS_OUT spiff_header& header, CHARLS_OUT bool& spiff_header_found);
int try_read_hp_color_transform_segment();
CHARLS_CHECK_RETURN jpegls_pc_parameters get_validated_preset_coding_parameters() const;
void read_marker_segment(jpeg_marker_code marker_code, spiff_header* header = nullptr,
bool* spiff_header_found = nullptr);
void read_spiff_directory_entry(jpeg_marker_code marker_code);
void read_start_of_frame_segment();
void read_start_of_scan();
void read_comment();
void read_application_data() noexcept;
void read_preset_parameters_segment();
void read_define_restart_interval();
void try_read_application_data8_segment(spiff_header* header, bool* spiff_header_found);
void try_read_spiff_header_segment(CHARLS_OUT spiff_header& header, CHARLS_OUT bool& spiff_header_found);
void try_read_hp_color_transform_segment();
void add_component(uint8_t component_id);
void check_parameter_coherent() const;
void check_interleave_mode(interleave_mode mode) const;
uint32_t maximum_sample_value() const noexcept;
CHARLS_CHECK_RETURN uint32_t maximum_sample_value() const noexcept;
void skip_remaining_segment_data() noexcept;
enum class state
{
@ -106,14 +123,15 @@ private:
after_end_of_image
};
byte_span source_;
const_byte_span::iterator position_{};
const_byte_span::iterator end_position_{};
const_byte_span segment_data_;
charls::frame_info frame_info_{};
coding_parameters parameters_{};
jpegls_pc_parameters preset_coding_parameters_{};
JlsRect rect_{};
std::vector<uint8_t> component_ids_;
state state_{};
at_comment_handler comment_handler_{};
void* comment_handler_user_context_{};
};

View File

@ -445,14 +445,14 @@ private:
}
// NOLINTNEXTLINE(cppcoreguidelines-explicit-virtual-functions, hicpp-use-override, modernize-use-override, clang-diagnostic-suggest-override)
void decode_scan(std::unique_ptr<process_line> process_line, const JlsRect& rect, byte_span& compressed_data)
size_t decode_scan(std::unique_ptr<process_line> process_line, const JlsRect& rect, const_byte_span encoded_source)
{
Strategy::process_line_ = std::move(process_line);
const uint8_t* compressed_bytes{compressed_data.data};
const auto* scan_begin{encoded_source.begin()};
rect_ = rect;
Strategy::initialize(compressed_data);
Strategy::initialize(encoded_source);
// Process images without a restart interval, as 1 large restart interval.
if (restart_interval_ == 0)
@ -462,7 +462,7 @@ private:
decode_lines();
skip_bytes(compressed_data, static_cast<size_t>(Strategy::get_cur_byte_pos() - compressed_bytes));
return Strategy::get_cur_byte_pos() - scan_begin;
}
// clang-format on

View File

@ -37,9 +37,10 @@ public:
return nullptr;
}
void decode_scan(unique_ptr<charls::process_line> /*output_data*/, const JlsRect& /*size*/,
charls::byte_span& /*compressed_data*/) noexcept(false) override
size_t decode_scan(unique_ptr<charls::process_line> /*process_line*/, const JlsRect& /*size*/,
charls::const_byte_span /*encoded_source*/) noexcept(false) override
{
return {};
}
int32_t read(const int32_t length)

View File

@ -28,7 +28,7 @@ public:
{
array<uint8_t, 1> buffer{};
jpeg_stream_reader reader;
reader.source({buffer.data(), 0});
reader.source({buffer.data(), static_cast<size_t>(0)});
assert_expect_exception(jpegls_errc::source_buffer_too_small, [&reader] { reader.read_header(); });
}
@ -45,7 +45,7 @@ public:
writer.write_start_of_frame_segment(1, 1, 2, 1);
writer.write_byte(extra_start_byte);
writer.write_start_of_scan_segment(0, 1, 128, interleave_mode::none);
writer.write_start_of_scan_segment(0, 1, 1, interleave_mode::none);
jpeg_stream_reader reader;
reader.source({writer.buffer.data(), writer.buffer.size()});
@ -236,9 +236,8 @@ public:
jpeg_stream_reader reader;
reader.source({writer.buffer.data(), writer.buffer.size()});
reader.read_header();
assert_expect_exception(jpegls_errc::invalid_parameter_near_lossless, [&reader] { reader.read_start_of_scan(); });
assert_expect_exception(jpegls_errc::invalid_parameter_near_lossless, [&reader] { reader.read_header(); });
}
TEST_METHOD(read_header_too_large_near_lossless_in_sos_should_throw2) // NOLINT
@ -255,9 +254,8 @@ public:
jpeg_stream_reader reader;
reader.source({writer.buffer.data(), writer.buffer.size()});
reader.read_header();
assert_expect_exception(jpegls_errc::invalid_parameter_near_lossless, [&reader] { reader.read_start_of_scan(); });
assert_expect_exception(jpegls_errc::invalid_parameter_near_lossless, [&reader] { reader.read_header(); });
}
TEST_METHOD(read_header_line_interleave_in_sos_for_single_component_should_throw) // NOLINT
@ -602,7 +600,7 @@ private:
writer.write_byte(0x02);
writer.write_start_of_frame_segment(1, 1, 2, 1);
writer.write_start_of_scan_segment(0, 1, 128, interleave_mode::none);
writer.write_start_of_scan_segment(0, 1, 1, interleave_mode::none);
jpeg_stream_reader reader;
reader.source({writer.buffer.data(), writer.buffer.size()});
@ -619,9 +617,8 @@ private:
jpeg_stream_reader reader;
reader.source({writer.buffer.data(), writer.buffer.size()});
reader.read_header();
assert_expect_exception(jpegls_errc::invalid_parameter_interleave_mode, [&reader] { reader.read_start_of_scan(); });
assert_expect_exception(jpegls_errc::invalid_parameter_interleave_mode, [&reader] { reader.read_header(); });
}
static void read_header_with_jpeg_ls_preset_parameter_with_extended_id_should_throw(const uint8_t id)