// Copyright (c) Team CharLS. // SPDX-License-Identifier: BSD-3-Clause #pragma once #include "jpegls_error.h" #ifdef __cplusplus #include #else #include #endif #ifdef __cplusplus struct charls_jpegls_encoder; extern "C" { #else typedef struct charls_jpegls_encoder charls_jpegls_encoder; #endif // The following functions define the public C API of the CharLS library. // The C++ API is defined after the C API. /// /// Creates a JPEG-LS encoder instance, when finished with the instance destroy it with the function /// charls_jpegls_encoder_destroy. /// /// A reference to a new created encoder instance, or a null pointer when the creation fails. CHARLS_CHECK_RETURN CHARLS_RET_MAY_BE_NULL CHARLS_API_IMPORT_EXPORT charls_jpegls_encoder* CHARLS_API_CALLING_CONVENTION charls_jpegls_encoder_create(CHARLS_C_VOID) CHARLS_NOEXCEPT; /// /// Destroys a JPEG-LS encoder instance created with charls_jpegls_encoder_create and releases all internal resources /// attached to it. /// /// Instance to destroy. If a null pointer is passed as argument, no action occurs. CHARLS_API_IMPORT_EXPORT void CHARLS_API_CALLING_CONVENTION charls_jpegls_encoder_destroy(IN_OPT_ const charls_jpegls_encoder* encoder) CHARLS_NOEXCEPT; /// /// Configures the frame that needs to be encoded. This information will be written to the Start of Frame segment. /// /// Reference to the encoder instance. /// Information about the frame that needs to be encoded. /// The result of the operation: success or a failure code. CHARLS_API_IMPORT_EXPORT charls_jpegls_errc CHARLS_API_CALLING_CONVENTION charls_jpegls_encoder_set_frame_info( IN_ charls_jpegls_encoder* encoder, IN_ const charls_frame_info* frame_info) CHARLS_NOEXCEPT CHARLS_ATTRIBUTE((nonnull)); /// /// Configures the NEAR parameter the encoder should use. A value of 0 means lossless, 0 is also the default. /// /// Reference to the encoder instance. /// Value of the NEAR parameter. /// The result of the operation: success or a failure code. CHARLS_API_IMPORT_EXPORT charls_jpegls_errc CHARLS_API_CALLING_CONVENTION charls_jpegls_encoder_set_near_lossless( IN_ charls_jpegls_encoder* encoder, int32_t near_lossless) CHARLS_NOEXCEPT CHARLS_ATTRIBUTE((nonnull)); /// /// Configures the interleave mode the encoder should use. The default is none. /// The encoder expects the input buffer in the same format as the interleave mode. /// /// Reference to the encoder instance. /// Value of the interleave mode. /// The result of the operation: success or a failure code. CHARLS_API_IMPORT_EXPORT charls_jpegls_errc CHARLS_API_CALLING_CONVENTION charls_jpegls_encoder_set_interleave_mode( IN_ charls_jpegls_encoder* encoder, charls_interleave_mode interleave_mode) CHARLS_NOEXCEPT CHARLS_ATTRIBUTE((nonnull)); /// /// Configures the preset coding parameters the encoder should use. /// If not set the encoder will use the default preset coding parameters as defined by the JPEG-LS standard. /// Only when the coding parameters are different then the default parameters, they will be written to the /// JPEG-LS stream during the encode phase. /// /// Reference to the encoder instance. /// Reference to the preset coding parameters. /// The result of the operation: success or a failure code. CHARLS_API_IMPORT_EXPORT charls_jpegls_errc CHARLS_API_CALLING_CONVENTION charls_jpegls_encoder_set_preset_coding_parameters( IN_ charls_jpegls_encoder* encoder, IN_ const charls_jpegls_pc_parameters* preset_coding_parameters) CHARLS_NOEXCEPT CHARLS_ATTRIBUTE((nonnull)); /// /// Configures the HP color transformation the encoder should use. /// If not set the encoder will use no color transformation. /// Color transformations are a HP extension and not defined by the JPEG-LS standard and can only be set for 3 component /// encodings. /// /// Reference to the encoder instance. /// The color transformation parameters. /// The result of the operation: success or a failure code. CHARLS_API_IMPORT_EXPORT charls_jpegls_errc CHARLS_API_CALLING_CONVENTION charls_jpegls_encoder_set_color_transformation( IN_ charls_jpegls_encoder* encoder, charls_color_transformation color_transformation) CHARLS_NOEXCEPT CHARLS_ATTRIBUTE((nonnull)); /// /// Returns the size in bytes, that the encoder expects are needed to hold the encoded image. /// /// /// Size for dynamic extras like SPIFF entries and other tables are not included in this size. /// /// Reference to the encoder instance. /// Reference to the size that will be set when the functions returns. /// The result of the operation: success or a failure code. CHARLS_API_IMPORT_EXPORT charls_jpegls_errc CHARLS_API_CALLING_CONVENTION charls_jpegls_encoder_get_estimated_destination_size(IN_ const charls_jpegls_encoder* encoder, OUT_ size_t* size_in_bytes) CHARLS_NOEXCEPT CHARLS_ATTRIBUTE((nonnull)); /// /// Set the reference to the destination buffer that will contain the encoded JPEG-LS byte stream data after encoding. /// This buffer needs to remain valid during the encoding process. /// /// Reference to the encoder instance. /// Reference to the start of the destination buffer. /// Size of the destination buffer in bytes. /// The result of the operation: success or a failure code. CHARLS_API_IMPORT_EXPORT charls_jpegls_errc CHARLS_API_CALLING_CONVENTION charls_jpegls_encoder_set_destination_buffer( IN_ charls_jpegls_encoder* encoder, OUT_WRITES_BYTES_(destination_size_bytes) void* destination_buffer, size_t destination_size_bytes) CHARLS_NOEXCEPT CHARLS_ATTRIBUTE((nonnull)); /// /// Writes a standard SPIFF header to the destination. The additional values are computed from the current encoder settings. /// A SPIFF header is optional, but recommended for standalone JPEG-LS files. /// /// Reference to the encoder instance. /// The color space of the image. /// The resolution units of the next 2 parameters. /// The vertical resolution. /// The horizontal resolution. /// The result of the operation: success or a failure code. CHARLS_API_IMPORT_EXPORT charls_jpegls_errc CHARLS_API_CALLING_CONVENTION charls_jpegls_encoder_write_standard_spiff_header( IN_ charls_jpegls_encoder* encoder, charls_spiff_color_space color_space, charls_spiff_resolution_units resolution_units, uint32_t vertical_resolution, uint32_t horizontal_resolution) CHARLS_NOEXCEPT CHARLS_ATTRIBUTE((nonnull)); /// /// Writes a SPIFF header to the destination. /// /// Reference to the encoder instance. /// Reference to a SPIFF header that will be written to the destination. /// The result of the operation: success or a failure code. CHARLS_API_IMPORT_EXPORT charls_jpegls_errc CHARLS_API_CALLING_CONVENTION charls_jpegls_encoder_write_spiff_header( IN_ charls_jpegls_encoder* encoder, IN_ const charls_spiff_header* spiff_header) CHARLS_NOEXCEPT CHARLS_ATTRIBUTE((nonnull)); /// /// Writes a SPIFF directory entry to the destination. /// /// /// Function should be called after writing a SPIFF header. /// /// Reference to the encoder instance. /// The entry tag of the directory entry. /// The entry data of the directory entry. /// The size in bytes of the directory entry [0-65528]. /// The result of the operation: success or a failure code. CHARLS_API_IMPORT_EXPORT charls_jpegls_errc CHARLS_API_CALLING_CONVENTION charls_jpegls_encoder_write_spiff_entry( IN_ charls_jpegls_encoder* encoder, uint32_t entry_tag, IN_READS_BYTES_(entry_data_size_bytes) const void* entry_data, size_t entry_data_size_bytes) CHARLS_NOEXCEPT; /// /// Encodes the passed buffer with the source image data to the destination. /// /// Reference to the encoder instance. /// Byte array that holds the image data that needs to be encoded. /// Length of the array in bytes. /// /// The number of bytes from one row of pixels in memory to the next row of pixels in memory. /// Stride is sometimes called pitch. If padding bytes are present, the stride is wider than the width of the image. /// /// The result of the operation: success or a failure code. CHARLS_API_IMPORT_EXPORT charls_jpegls_errc CHARLS_API_CALLING_CONVENTION charls_jpegls_encoder_encode_from_buffer( IN_ charls_jpegls_encoder* encoder, IN_READS_BYTES_(source_size_bytes) const void* source_buffer, size_t source_size_bytes, uint32_t stride) CHARLS_NOEXCEPT CHARLS_ATTRIBUTE((nonnull)); /// /// Returns the size in bytes, that are written to the destination. /// /// Reference to the encoder instance. /// Reference to the size that will be set when the functions returns. /// The result of the operation: success or a failure code. CHARLS_API_IMPORT_EXPORT charls_jpegls_errc CHARLS_API_CALLING_CONVENTION charls_jpegls_encoder_get_bytes_written( IN_ const charls_jpegls_encoder* encoder, OUT_ size_t* bytes_written) CHARLS_NOEXCEPT CHARLS_ATTRIBUTE((nonnull)); // Note: The 4 methods below are considered obsolete and will be removed in the next major update. /// /// Encodes a byte array with pixel data to a JPEG-LS encoded (compressed) byte array. /// /// This method is considered obsolete and will be removed in the next major update. /// Byte array that holds the encoded bytes when the function returns. /// /// Length of the array in bytes. If the array is too small the function will return an error. /// /// /// This parameter will hold the number of bytes written to the destination byte array. Cannot be NULL. /// /// Byte array that holds the pixels that should be encoded. /// Length of the array in bytes. /// Parameter object that describes the pixel data and how to encode it. /// /// Character array of at least 256 characters or NULL. Hold the error message when a failure occurs, empty otherwise. /// CHARLS_DEPRECATED CHARLS_API_IMPORT_EXPORT CharlsApiResultType CHARLS_API_CALLING_CONVENTION JpegLsEncode(OUT_WRITES_BYTES_(destination_length) void* destination, size_t destination_length, OUT_ size_t* bytes_written, IN_READS_BYTES_(source_length) const void* source, size_t source_length, IN_ const struct JlsParameters* params, OUT_OPT_ char* error_message) CHARLS_ATTRIBUTE((nonnull(1, 3, 4, 6))); #ifdef __cplusplus } // extern "C" namespace charls { /// /// JPEG-LS encoder class that encapsulates the C ABI interface calls and provide a native C++ interface. /// class jpegls_encoder final { public: /// /// Encoded pixel data in 1 simple operation into a JPEG-LS encoded buffer. /// /// Source container with the pixel data bytes that need to be encoded. /// Information about the frame that needs to be encoded. /// Configures the interleave mode the encoder should use. /// Container with the JPEG-LS encoded bytes. template static auto encode(const Container& source, const charls::frame_info& info, const charls::interleave_mode interleave_mode = charls::interleave_mode::none) { jpegls_encoder encoder; encoder.frame_info(info).interleave_mode(interleave_mode); Container destination(encoder.estimated_destination_size()); encoder.destination(destination); const size_t bytes_written{encoder.encode(source)}; destination.resize(bytes_written); return destination; } jpegls_encoder() = default; ~jpegls_encoder() = default; jpegls_encoder(const jpegls_encoder&) = delete; jpegls_encoder(jpegls_encoder&&) noexcept = default; jpegls_encoder& operator=(const jpegls_encoder&) = delete; jpegls_encoder& operator=(jpegls_encoder&&) noexcept = default; /// /// Configures the frame that needs to be encoded. /// This information will be written to the Start of Frame (SOF) segment during the encode phase. /// /// Information about the frame that needs to be encoded. jpegls_encoder& frame_info(const frame_info& frame_info) { check_jpegls_errc(charls_jpegls_encoder_set_frame_info(encoder_.get(), &frame_info)); return *this; } /// /// Configures the NEAR parameter the encoder should use. A value of 0 means lossless, this is also the default. /// /// Value of the NEAR parameter. jpegls_encoder& near_lossless(const int32_t near_lossless) { check_jpegls_errc(charls_jpegls_encoder_set_near_lossless(encoder_.get(), near_lossless)); return *this; } /// /// Configures the interleave mode the encoder should use. The default is none. /// The encoder expects the input buffer in the same format as the interleave mode. /// /// Value of the interleave mode. jpegls_encoder& interleave_mode(const interleave_mode interleave_mode) { check_jpegls_errc(charls_jpegls_encoder_set_interleave_mode(encoder_.get(), interleave_mode)); return *this; } /// /// Configures the preset coding parameters the encoder should use. /// If not set the encoder will use the default preset coding parameters as defined by the JPEG-LS standard. /// Only when the coding parameters are different then the default parameters, they will be written to the /// JPEG-LS stream during the encode phase. /// /// Reference to the preset coding parameters. jpegls_encoder& preset_coding_parameters(const jpegls_pc_parameters& preset_coding_parameters) { check_jpegls_errc(charls_jpegls_encoder_set_preset_coding_parameters(encoder_.get(), &preset_coding_parameters)); return *this; } /// /// Configures the HP color transformation the encoder should use. /// If not set the encoder will use no color transformation. /// Color transformations are a HP extension and not defined by the JPEG-LS standard /// and can only be set for 3 component encodings. /// /// The color transformation parameters. jpegls_encoder& color_transformation(const color_transformation color_transformation) { check_jpegls_errc(charls_jpegls_encoder_set_color_transformation(encoder_.get(), color_transformation)); return *this; } /// /// Returns the size in bytes, that the encoder expects are needed to hold the encoded image. /// /// /// Size for dynamic extras like SPIFF entries and other tables are not included in this size. /// /// The estimated size in bytes needed to hold the encoded image. CHARLS_NO_DISCARD size_t estimated_destination_size() const { size_t size_in_bytes; check_jpegls_errc(charls_jpegls_encoder_get_estimated_destination_size(encoder_.get(), &size_in_bytes)); return size_in_bytes; } /// /// Set the reference to the destination buffer that will contain the encoded JPEG-LS byte stream data after encoding. /// This buffer needs to remain valid during the encoding process. /// /// Reference to the start of the destination buffer. /// Size of the destination buffer in bytes. jpegls_encoder& destination(OUT_WRITES_BYTES_(destination_size_bytes) void* destination_buffer, const size_t destination_size_bytes) { check_jpegls_errc( charls_jpegls_encoder_set_destination_buffer(encoder_.get(), destination_buffer, destination_size_bytes)); return *this; } /// /// Set the the container that will contain the encoded JPEG-LS byte stream data after encoding. /// This container needs to remain valid during the encoding process. /// /// /// The STL like container, that supports the functions data() and size() and the typedef value_type. /// template jpegls_encoder& destination(OUT_ Container& destination_container) { return destination(destination_container.data(), destination_container.size() * sizeof(ValueType)); } /// /// Writes a standard SPIFF header to the destination. The additional values are computed from the current encoder /// settings. /// /// The color space of the image. /// The resolution units of the next 2 parameters. /// The vertical resolution. /// The horizontal resolution. jpegls_encoder& write_standard_spiff_header(const spiff_color_space color_space, const spiff_resolution_units resolution_units = spiff_resolution_units::aspect_ratio, const uint32_t vertical_resolution = 1, const uint32_t horizontal_resolution = 1) { check_jpegls_errc(charls_jpegls_encoder_write_standard_spiff_header(encoder_.get(), color_space, resolution_units, vertical_resolution, horizontal_resolution)); return *this; } /// /// Writes a SPIFF header to the destination. /// /// Reference to a SPIFF header that will be written to the destination. jpegls_encoder& write_spiff_header(const spiff_header& header) { check_jpegls_errc(charls_jpegls_encoder_write_spiff_header(encoder_.get(), &header)); return *this; } /// /// Writes a SPIFF directory entry to the destination. /// /// The entry tag of the directory entry. /// The entry data of the directory entry. /// The size in bytes of the directory entry [0-65528]. template jpegls_encoder& write_spiff_entry(const IntDerivedType entry_tag, IN_READS_BYTES_(entry_data_size_bytes) const void* entry_data, const size_t entry_data_size_bytes) { check_jpegls_errc(charls_jpegls_encoder_write_spiff_entry(encoder_.get(), static_cast(entry_tag), entry_data, entry_data_size_bytes)); return *this; } /// /// Encodes the passed buffer with the source image data to the destination. /// /// Byte array that holds the image data that needs to be encoded. /// Length of the array in bytes. /// /// The number of bytes from one row of pixels in memory to the next row of pixels in memory. /// Stride is sometimes called pitch. If padding bytes are present, the stride is wider than the width of the image. /// /// The number of bytes written to the destination. size_t encode(IN_READS_BYTES_(source_size_bytes) const void* source_buffer, const size_t source_size_bytes, const uint32_t stride = 0) const { check_jpegls_errc( charls_jpegls_encoder_encode_from_buffer(encoder_.get(), source_buffer, source_size_bytes, stride)); return bytes_written(); } /// /// Encodes the passed STL like container with the source image data to the destination. /// /// Container that holds the image data that needs to be encoded. /// /// The number of bytes from one row of pixels in memory to the next row of pixels in memory. /// Stride is sometimes called pitch. If padding bytes are present, the stride is wider than the width of the image. /// /// The number of bytes written to the destination. template size_t encode(const Container& source_container, const uint32_t stride = 0) const { return encode(source_container.data(), source_container.size() * sizeof(ValueType), stride); } /// /// Returns the size in bytes, that are written to the destination. /// /// The bytes written. CHARLS_NO_DISCARD size_t bytes_written() const { size_t bytes_written; check_jpegls_errc(charls_jpegls_encoder_get_bytes_written(encoder_.get(), &bytes_written)); return bytes_written; } private: CHARLS_NO_DISCARD static charls_jpegls_encoder* create_encoder() { charls_jpegls_encoder* encoder{charls_jpegls_encoder_create()}; if (!encoder) throw std::bad_alloc(); return encoder; } static void destroy_encoder(IN_OPT_ const charls_jpegls_encoder* encoder) noexcept { charls_jpegls_encoder_destroy(encoder); } std::unique_ptr encoder_{create_encoder(), &destroy_encoder}; }; } // namespace charls #endif