2019-11-08 23:17:16 +01:00
|
|
|
// Copyright (c) Team CharLS.
|
|
|
|
// SPDX-License-Identifier: BSD-3-Clause
|
2015-05-01 11:00:13 +02:00
|
|
|
|
2018-10-11 21:12:39 +02:00
|
|
|
#include <charls/charls.h>
|
2015-05-01 11:00:13 +02:00
|
|
|
|
2019-11-17 14:48:15 +01:00
|
|
|
#include <assert.h>
|
2020-04-26 12:00:45 +02:00
|
|
|
// ReSharper disable once CppUnusedIncludeDirective
|
2019-11-17 14:48:15 +01:00
|
|
|
#include <errno.h>
|
|
|
|
#include <stdbool.h>
|
2014-12-30 12:20:15 +01:00
|
|
|
#include <stdint.h>
|
|
|
|
#include <stdio.h>
|
2019-11-17 14:48:15 +01:00
|
|
|
#include <stdlib.h>
|
2020-04-26 12:00:45 +02:00
|
|
|
#include <string.h>
|
2014-12-30 12:20:15 +01:00
|
|
|
|
2020-07-31 14:43:05 +02:00
|
|
|
const static size_t bytes_per_rgb_pixel = 3;
|
|
|
|
|
2019-11-17 14:48:15 +01:00
|
|
|
typedef struct
|
|
|
|
{
|
2018-10-30 21:57:22 +01:00
|
|
|
uint8_t magic[2]; // the magic number used to identify the BMP file:
|
|
|
|
// 0x42 0x4D (Hex code points for B and M).
|
|
|
|
// The following entries are possible:
|
|
|
|
// BM - Windows 3.1x, 95, NT, ... etc
|
|
|
|
// BA - OS/2 Bitmap Array
|
|
|
|
// CI - OS/2 Color Icon
|
|
|
|
// CP - OS/2 Color Pointer
|
|
|
|
// IC - OS/2 Icon
|
|
|
|
// PT - OS/2 Pointer.
|
|
|
|
uint32_t file_size; // the size of the BMP file in bytes
|
|
|
|
uint32_t reserved; // reserved.
|
|
|
|
uint32_t offset; // the offset, i.e. starting address, of the byte where the bitmap data can be found.
|
2014-12-30 12:20:15 +01:00
|
|
|
} bmp_header_t;
|
|
|
|
|
2019-11-17 14:48:15 +01:00
|
|
|
typedef struct
|
|
|
|
{
|
2018-10-30 21:57:22 +01:00
|
|
|
uint32_t header_size; // the size of this header (40 bytes)
|
|
|
|
uint32_t width; // the bitmap width in pixels
|
2020-04-26 12:00:45 +02:00
|
|
|
int32_t height; // the bitmap height in pixels
|
2018-10-30 21:57:22 +01:00
|
|
|
uint16_t number_planes; // the number of color planes being used. Must be set to 1
|
|
|
|
uint16_t depth; // the number of bits per pixel,which is the color depth of the image.
|
|
|
|
// Typical values are 1, 4, 8, 16, 24 and 32.
|
|
|
|
uint32_t compress_type; // the compression method being used.
|
|
|
|
uint32_t bmp_byte_size; // the image size. This is the size of the raw bitmap
|
|
|
|
// data (see below), and should not be confused with the file size.
|
|
|
|
uint32_t horizontal_resolution; // the horizontal resolution of the image. (pixel per meter)
|
|
|
|
uint32_t vertical_resolution; // the vertical resolution of the image. (pixel per meter)
|
|
|
|
uint32_t number_colors; // the number of colors in the color palette, or 0 to default to 2^depth.
|
2020-12-30 16:50:55 +01:00
|
|
|
uint32_t number_important_colors; // the number of important colors used, or 0 when every color is important generally
|
|
|
|
// ignored.
|
2014-12-30 12:20:15 +01:00
|
|
|
} bmp_dib_header_t;
|
|
|
|
|
|
|
|
|
2019-11-17 14:48:15 +01:00
|
|
|
static bool bmp_read_header(FILE* fp, bmp_header_t* header)
|
2014-12-30 12:20:15 +01:00
|
|
|
{
|
2015-05-01 11:00:13 +02:00
|
|
|
assert(fp);
|
|
|
|
assert(header);
|
|
|
|
|
2018-10-30 21:57:22 +01:00
|
|
|
return fread(&header->magic, sizeof header->magic, 1, fp) &&
|
|
|
|
fread(&header->file_size, sizeof header->file_size, 1, fp) &&
|
|
|
|
fread(&header->reserved, sizeof header->reserved, 1, fp) &&
|
2020-12-30 16:50:55 +01:00
|
|
|
fread(&header->offset, sizeof header->offset, 1, fp) && header->magic[0] == 0x42 && header->magic[1] == 0x4D;
|
2014-12-30 12:20:15 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2019-11-17 14:48:15 +01:00
|
|
|
static bool bmp_read_dib_header(FILE* fp, bmp_dib_header_t* header)
|
2014-12-30 12:20:15 +01:00
|
|
|
{
|
2015-05-01 11:00:13 +02:00
|
|
|
assert(fp);
|
|
|
|
assert(header);
|
|
|
|
|
2020-12-30 16:50:55 +01:00
|
|
|
return fread(&header->header_size, sizeof header->header_size, 1, fp) && header->header_size >= 40 &&
|
|
|
|
fread(&header->width, sizeof header->width, 1, fp) && fread(&header->height, sizeof header->height, 1, fp) &&
|
2018-10-30 21:57:22 +01:00
|
|
|
fread(&header->number_planes, sizeof header->number_planes, 1, fp) &&
|
|
|
|
fread(&header->depth, sizeof header->depth, 1, fp) &&
|
2020-08-06 18:11:34 +02:00
|
|
|
fread(&header->compress_type, sizeof header->compress_type, 1, fp) &&
|
|
|
|
fread(&header->bmp_byte_size, sizeof header->bmp_byte_size, 1, fp) &&
|
|
|
|
fread(&header->horizontal_resolution, sizeof header->horizontal_resolution, 1, fp) &&
|
|
|
|
fread(&header->vertical_resolution, sizeof header->vertical_resolution, 1, fp) &&
|
|
|
|
fread(&header->number_colors, sizeof header->number_colors, 1, fp) &&
|
|
|
|
fread(&header->number_important_colors, sizeof header->number_important_colors, 1, fp);
|
2014-12-30 12:20:15 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2020-07-31 14:43:05 +02:00
|
|
|
static void* bmp_read_pixel_data(FILE* fp, const uint32_t offset, const bmp_dib_header_t* dib_header, size_t* stride)
|
2014-12-30 12:20:15 +01:00
|
|
|
{
|
2015-05-01 11:00:13 +02:00
|
|
|
assert(fp);
|
2020-07-31 14:43:05 +02:00
|
|
|
assert(dib_header);
|
|
|
|
assert(stride);
|
|
|
|
assert(dib_header->compress_type == 0);
|
|
|
|
assert(dib_header->depth == 24);
|
2015-05-01 11:00:13 +02:00
|
|
|
|
2020-07-31 14:43:05 +02:00
|
|
|
if (fseek(fp, (long)offset, SEEK_SET))
|
2015-04-26 23:22:07 +02:00
|
|
|
return NULL;
|
|
|
|
|
2020-07-31 14:43:05 +02:00
|
|
|
// The BMP format requires that the size of each row is rounded up to a multiple of 4 bytes by padding.
|
|
|
|
*stride = ((dib_header->width * bytes_per_rgb_pixel) + 3) / 4 * 4;
|
2019-12-17 14:16:43 +01:00
|
|
|
|
2020-07-31 14:43:05 +02:00
|
|
|
const size_t buffer_size = (size_t)dib_header->height * *stride;
|
|
|
|
void* buffer = malloc(buffer_size);
|
|
|
|
if (!buffer)
|
|
|
|
return NULL;
|
|
|
|
|
|
|
|
if (fread(buffer, buffer_size, 1, fp) != 1)
|
|
|
|
{
|
|
|
|
free(buffer);
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
return buffer;
|
2015-04-26 23:22:07 +02:00
|
|
|
}
|
|
|
|
|
2020-04-26 12:00:45 +02:00
|
|
|
|
2021-12-20 20:34:00 +01:00
|
|
|
static void* handle_encoder_failure(const charls_jpegls_errc error, const char* step, const charls_jpegls_encoder* encoder,
|
2020-12-30 16:50:55 +01:00
|
|
|
void* buffer)
|
2019-11-08 21:56:38 +01:00
|
|
|
{
|
|
|
|
printf("Failed to %s: %i, %s\n", step, error, charls_get_error_message(error));
|
|
|
|
charls_jpegls_encoder_destroy(encoder);
|
|
|
|
free(buffer);
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
2015-04-26 23:22:07 +02:00
|
|
|
|
2020-07-31 14:43:05 +02:00
|
|
|
static void convert_bgr_to_rgb(uint8_t* triplet_buffer, const size_t width, const size_t height, const size_t stride)
|
2020-04-26 12:00:45 +02:00
|
|
|
{
|
2020-07-31 14:43:05 +02:00
|
|
|
for (size_t line = 0; line < height; ++line)
|
2020-04-26 12:00:45 +02:00
|
|
|
{
|
2020-07-31 14:43:05 +02:00
|
|
|
const size_t line_start = line * stride;
|
|
|
|
for (size_t pixel = 0; pixel < width; ++pixel)
|
|
|
|
{
|
|
|
|
const size_t column = pixel * bytes_per_rgb_pixel;
|
|
|
|
|
|
|
|
const size_t a = line_start + column;
|
|
|
|
const size_t b = line_start + column + 2;
|
|
|
|
|
|
|
|
const uint8_t temp = triplet_buffer[a];
|
|
|
|
triplet_buffer[a] = triplet_buffer[b];
|
|
|
|
triplet_buffer[b] = temp;
|
|
|
|
}
|
2020-04-26 12:00:45 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2020-12-30 16:50:55 +01:00
|
|
|
static void triplet_to_planar(const uint8_t* triplet_buffer, uint8_t* planar_buffer, const size_t width, const size_t height,
|
|
|
|
const size_t stride)
|
2020-04-26 12:00:45 +02:00
|
|
|
{
|
2020-07-31 14:43:05 +02:00
|
|
|
const size_t byte_count_plane = width * height;
|
|
|
|
size_t plane_column = 0;
|
|
|
|
|
|
|
|
for (size_t line = 0; line < height; ++line)
|
2020-04-26 12:00:45 +02:00
|
|
|
{
|
2020-07-31 14:43:05 +02:00
|
|
|
const size_t line_start = line * stride;
|
|
|
|
for (size_t pixel = 0; pixel < width; ++pixel)
|
|
|
|
{
|
2025-02-13 23:23:04 +01:00
|
|
|
const size_t column = line_start + (pixel * bytes_per_rgb_pixel);
|
2020-07-31 14:43:05 +02:00
|
|
|
|
|
|
|
planar_buffer[plane_column] = triplet_buffer[column];
|
2025-02-13 23:23:04 +01:00
|
|
|
planar_buffer[plane_column + (1 * byte_count_plane)] = triplet_buffer[column + 1];
|
|
|
|
planar_buffer[plane_column + (2 * byte_count_plane)] = triplet_buffer[column + 2];
|
2020-07-31 14:43:05 +02:00
|
|
|
++plane_column;
|
|
|
|
}
|
2020-04-26 12:00:45 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2020-12-30 16:50:55 +01:00
|
|
|
static bool convert_bottom_up_to_top_down(uint8_t* triplet_buffer, const size_t width, const size_t height,
|
|
|
|
const size_t stride)
|
2020-04-26 12:00:45 +02:00
|
|
|
{
|
2020-07-31 14:43:05 +02:00
|
|
|
const size_t row_length = width * bytes_per_rgb_pixel;
|
|
|
|
void* temp_row = malloc(row_length);
|
2020-04-26 12:00:45 +02:00
|
|
|
if (!temp_row)
|
|
|
|
return false;
|
|
|
|
|
|
|
|
for (size_t i = 0; i < height / 2; ++i)
|
|
|
|
{
|
|
|
|
memcpy(temp_row, &triplet_buffer[i * stride], row_length);
|
|
|
|
const size_t bottom_row = height - i - 1;
|
|
|
|
memcpy(&triplet_buffer[i * stride], &triplet_buffer[bottom_row * stride], row_length);
|
|
|
|
memcpy(&triplet_buffer[bottom_row * stride], temp_row, row_length);
|
|
|
|
}
|
|
|
|
|
2020-05-23 18:56:45 +02:00
|
|
|
free(temp_row);
|
|
|
|
|
2020-04-26 12:00:45 +02:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2020-12-30 16:50:55 +01:00
|
|
|
static void* encode_bmp_to_jpegls(const void* pixel_data, const size_t stride, const bmp_dib_header_t* header,
|
|
|
|
const charls_interleave_mode interleave_mode, const int near_lossless,
|
|
|
|
size_t* bytes_written)
|
2015-04-26 23:22:07 +02:00
|
|
|
{
|
2020-09-25 22:40:39 +02:00
|
|
|
assert(header->depth == 24 && "This function only supports 24-bit BMP pixel data.");
|
|
|
|
assert(header->compress_type == 0 && "Data needs to be stored by pixel as RGB.");
|
|
|
|
assert(header->width > 0 && "0 width not supported, may cause 0 byte malloc");
|
|
|
|
assert(header->height > 0 && "0 and negative height not supported, may cause 0 byte malloc");
|
2015-05-01 11:00:13 +02:00
|
|
|
|
2019-11-08 21:56:38 +01:00
|
|
|
charls_jpegls_encoder* encoder = charls_jpegls_encoder_create();
|
|
|
|
if (!encoder)
|
2018-10-30 22:56:09 +01:00
|
|
|
{
|
2019-11-08 21:56:38 +01:00
|
|
|
printf("Failed to create JPEG-LS encoder\n");
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
2019-11-17 14:48:15 +01:00
|
|
|
charls_frame_info frame_info = {.bits_per_sample = 8, .component_count = 3};
|
2019-11-08 21:56:38 +01:00
|
|
|
frame_info.width = header->width;
|
2020-07-31 14:43:05 +02:00
|
|
|
frame_info.height = (uint32_t)header->height;
|
2019-11-08 21:56:38 +01:00
|
|
|
charls_jpegls_errc error = charls_jpegls_encoder_set_frame_info(encoder, &frame_info);
|
|
|
|
if (error)
|
2015-05-01 11:00:13 +02:00
|
|
|
{
|
2019-11-08 21:56:38 +01:00
|
|
|
return handle_encoder_failure(error, "set frame_info", encoder, NULL);
|
2015-05-01 11:00:13 +02:00
|
|
|
}
|
|
|
|
|
2020-04-26 12:00:45 +02:00
|
|
|
error = charls_jpegls_encoder_set_interleave_mode(encoder, interleave_mode);
|
2019-11-08 21:56:38 +01:00
|
|
|
if (error)
|
|
|
|
{
|
2020-04-26 12:00:45 +02:00
|
|
|
return handle_encoder_failure(error, "set near interleave mode", encoder, NULL);
|
2019-11-08 21:56:38 +01:00
|
|
|
}
|
|
|
|
|
2020-04-26 12:00:45 +02:00
|
|
|
error = charls_jpegls_encoder_set_near_lossless(encoder, near_lossless);
|
2019-11-08 21:56:38 +01:00
|
|
|
if (error)
|
|
|
|
{
|
2020-04-26 12:00:45 +02:00
|
|
|
return handle_encoder_failure(error, "set near lossless", encoder, NULL);
|
2019-11-08 21:56:38 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
size_t encoded_buffer_size;
|
|
|
|
error = charls_jpegls_encoder_get_estimated_destination_size(encoder, &encoded_buffer_size);
|
|
|
|
if (error)
|
|
|
|
{
|
|
|
|
return handle_encoder_failure(error, "get estimated destination size", encoder, NULL);
|
|
|
|
}
|
|
|
|
|
|
|
|
void* encoded_buffer = malloc(encoded_buffer_size);
|
2020-02-02 14:22:50 +01:00
|
|
|
if (!encoded_buffer)
|
|
|
|
{
|
|
|
|
return handle_encoder_failure(error, "malloc failed", encoder, NULL);
|
|
|
|
}
|
|
|
|
|
2019-11-08 21:56:38 +01:00
|
|
|
error = charls_jpegls_encoder_set_destination_buffer(encoder, encoded_buffer, encoded_buffer_size);
|
|
|
|
if (error)
|
|
|
|
{
|
|
|
|
return handle_encoder_failure(error, "set destination buffer", encoder, encoded_buffer);
|
|
|
|
}
|
|
|
|
|
2020-08-06 18:11:34 +02:00
|
|
|
// The resolution in BMP files is often 0 to indicate that no resolution has been defined.
|
|
|
|
// The SPIFF header specification requires however that VRES and HRES are never 0.
|
|
|
|
// The ISO 10918-3 recommendation for these cases is to define that the pixels should be interpreted as a square.
|
|
|
|
if (header->vertical_resolution < 100 || header->horizontal_resolution < 100)
|
|
|
|
{
|
2020-12-30 16:50:55 +01:00
|
|
|
error = charls_jpegls_encoder_write_standard_spiff_header(encoder, CHARLS_SPIFF_COLOR_SPACE_RGB,
|
|
|
|
CHARLS_SPIFF_RESOLUTION_UNITS_ASPECT_RATIO, 1, 1);
|
2020-08-06 18:11:34 +02:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2020-12-30 16:50:55 +01:00
|
|
|
error = charls_jpegls_encoder_write_standard_spiff_header(
|
|
|
|
encoder, CHARLS_SPIFF_COLOR_SPACE_RGB, CHARLS_SPIFF_RESOLUTION_UNITS_DOTS_PER_CENTIMETER,
|
|
|
|
header->vertical_resolution / 100, header->horizontal_resolution / 100);
|
2020-08-06 18:11:34 +02:00
|
|
|
}
|
|
|
|
|
2019-11-17 14:48:15 +01:00
|
|
|
if (error)
|
|
|
|
{
|
|
|
|
return handle_encoder_failure(error, "write_standard_spiff_header", encoder, encoded_buffer);
|
|
|
|
}
|
|
|
|
|
2020-04-26 12:00:45 +02:00
|
|
|
if (interleave_mode == CHARLS_INTERLEAVE_MODE_NONE)
|
|
|
|
{
|
2020-07-31 14:43:05 +02:00
|
|
|
const size_t pixel_data_size = (size_t)header->height * header->width * bytes_per_rgb_pixel;
|
2020-04-26 12:00:45 +02:00
|
|
|
void* planar_pixel_data = malloc(pixel_data_size);
|
|
|
|
if (!planar_pixel_data)
|
|
|
|
{
|
|
|
|
return handle_encoder_failure(CHARLS_JPEGLS_ERRC_NOT_ENOUGH_MEMORY, "malloc", encoder, encoded_buffer);
|
|
|
|
}
|
|
|
|
|
2020-07-31 14:43:05 +02:00
|
|
|
triplet_to_planar(pixel_data, planar_pixel_data, header->width, (size_t)header->height, stride);
|
2020-04-26 12:00:45 +02:00
|
|
|
error = charls_jpegls_encoder_encode_from_buffer(encoder, planar_pixel_data, pixel_data_size, 0);
|
|
|
|
free(planar_pixel_data);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2020-07-31 14:43:05 +02:00
|
|
|
const size_t pixel_data_size = (size_t)header->height * stride;
|
|
|
|
error = charls_jpegls_encoder_encode_from_buffer(encoder, pixel_data, pixel_data_size, (uint32_t)stride);
|
2020-04-26 12:00:45 +02:00
|
|
|
}
|
|
|
|
|
2019-11-08 21:56:38 +01:00
|
|
|
if (error)
|
|
|
|
{
|
|
|
|
return handle_encoder_failure(error, "encode", encoder, encoded_buffer);
|
|
|
|
}
|
|
|
|
|
|
|
|
error = charls_jpegls_encoder_get_bytes_written(encoder, bytes_written);
|
|
|
|
if (error)
|
|
|
|
{
|
|
|
|
return handle_encoder_failure(error, "get bytes written", encoder, encoded_buffer);
|
|
|
|
}
|
|
|
|
|
|
|
|
charls_jpegls_encoder_destroy(encoder);
|
|
|
|
|
2015-05-01 11:00:13 +02:00
|
|
|
return encoded_buffer;
|
|
|
|
}
|
2014-12-30 12:20:15 +01:00
|
|
|
|
|
|
|
|
2020-04-03 23:38:51 +02:00
|
|
|
static bool save_jpegls_file(const char* filename, const void* buffer, const size_t buffer_size)
|
2015-05-01 11:00:13 +02:00
|
|
|
{
|
|
|
|
assert(filename);
|
|
|
|
assert(buffer);
|
|
|
|
assert(buffer_size);
|
|
|
|
|
|
|
|
bool result = false;
|
2019-11-17 14:48:15 +01:00
|
|
|
FILE* stream = fopen(filename, "wb");
|
2015-05-01 11:00:13 +02:00
|
|
|
if (stream)
|
|
|
|
{
|
|
|
|
result = fwrite(buffer, buffer_size, 1, stream);
|
2022-07-14 22:10:39 +02:00
|
|
|
result = result && fclose(stream) == 0;
|
2015-05-01 11:00:13 +02:00
|
|
|
}
|
2014-12-30 12:20:15 +01:00
|
|
|
|
2015-05-01 11:00:13 +02:00
|
|
|
return result;
|
2015-04-26 23:22:07 +02:00
|
|
|
}
|
2014-12-30 12:20:15 +01:00
|
|
|
|
2015-04-26 23:22:07 +02:00
|
|
|
|
2020-04-26 12:00:45 +02:00
|
|
|
typedef struct
|
|
|
|
{
|
2020-07-31 15:30:58 +02:00
|
|
|
const char* input_filename;
|
|
|
|
const char* output_filename;
|
2020-04-26 12:00:45 +02:00
|
|
|
charls_interleave_mode interleave_mode;
|
|
|
|
int near_lossless;
|
|
|
|
} options_t;
|
|
|
|
|
|
|
|
|
|
|
|
static bool parse_command_line_options(const int argc, char* argv[], options_t* options)
|
2015-04-26 23:22:07 +02:00
|
|
|
{
|
2019-11-17 14:48:15 +01:00
|
|
|
if (argc < 3)
|
|
|
|
{
|
2020-12-30 16:50:55 +01:00
|
|
|
printf("Usage: <input-filename> <output-filename> [interleave-mode (none, line or sample), default = none] "
|
|
|
|
"[near-lossless, default=0 (lossless)]\n");
|
2020-04-26 12:00:45 +02:00
|
|
|
return false;
|
2015-04-26 23:22:07 +02:00
|
|
|
}
|
|
|
|
|
2020-07-31 15:30:58 +02:00
|
|
|
options->input_filename = argv[1];
|
|
|
|
options->output_filename = argv[2];
|
2020-04-26 12:00:45 +02:00
|
|
|
|
2018-10-30 22:56:09 +01:00
|
|
|
if (argc > 3)
|
|
|
|
{
|
2020-07-31 15:30:58 +02:00
|
|
|
if (strcmp(argv[3], "none") == 0)
|
2020-04-26 12:00:45 +02:00
|
|
|
{
|
2020-07-31 15:30:58 +02:00
|
|
|
options->interleave_mode = CHARLS_INTERLEAVE_MODE_NONE;
|
|
|
|
}
|
|
|
|
else if (strcmp(argv[3], "line") == 0)
|
|
|
|
{
|
|
|
|
options->interleave_mode = CHARLS_INTERLEAVE_MODE_LINE;
|
|
|
|
}
|
|
|
|
else if (strcmp(argv[3], "sample") == 0)
|
|
|
|
{
|
|
|
|
options->interleave_mode = CHARLS_INTERLEAVE_MODE_SAMPLE;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
printf("Argument interleave-mode needs to be: none, line or sample\n");
|
2020-04-26 12:00:45 +02:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
options->interleave_mode = CHARLS_INTERLEAVE_MODE_NONE;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (argc > 4)
|
|
|
|
{
|
|
|
|
options->near_lossless = (int)strtol(argv[4], NULL, 10);
|
|
|
|
if (options->near_lossless < 0 || options->near_lossless > 255)
|
2018-10-30 22:56:09 +01:00
|
|
|
{
|
2020-07-31 15:30:58 +02:00
|
|
|
printf("Argument near-lossless needs to be in the range [0,255]\n");
|
2020-04-26 12:00:45 +02:00
|
|
|
return false;
|
2018-10-30 22:56:09 +01:00
|
|
|
}
|
|
|
|
}
|
2020-04-26 12:00:45 +02:00
|
|
|
else
|
|
|
|
{
|
|
|
|
options->near_lossless = 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
2018-10-30 22:56:09 +01:00
|
|
|
|
2020-04-26 12:00:45 +02:00
|
|
|
|
|
|
|
int main(const int argc, char* argv[])
|
|
|
|
{
|
|
|
|
options_t options;
|
|
|
|
if (!parse_command_line_options(argc, argv, &options))
|
|
|
|
return EXIT_FAILURE;
|
|
|
|
|
2020-07-31 15:30:58 +02:00
|
|
|
FILE* input_stream = fopen(options.input_filename, "rb");
|
2018-10-30 21:57:22 +01:00
|
|
|
if (!input_stream)
|
2015-04-26 23:22:07 +02:00
|
|
|
{
|
2020-07-31 15:30:58 +02:00
|
|
|
printf("Failed to open file: %s, errno: %d\n", options.input_filename, errno);
|
2018-10-30 21:57:22 +01:00
|
|
|
return EXIT_FAILURE;
|
|
|
|
}
|
2015-04-26 23:22:07 +02:00
|
|
|
|
2018-10-30 21:57:22 +01:00
|
|
|
bmp_header_t header;
|
|
|
|
bmp_dib_header_t dib_header;
|
|
|
|
if (!bmp_read_header(input_stream, &header) || !bmp_read_dib_header(input_stream, &dib_header))
|
|
|
|
{
|
|
|
|
printf("Failed to read the BMP info from the file: %s\n", argv[1]);
|
2022-07-14 22:10:39 +02:00
|
|
|
(void) fclose(input_stream);
|
2018-10-30 21:57:22 +01:00
|
|
|
return EXIT_FAILURE;
|
2015-04-26 23:22:07 +02:00
|
|
|
}
|
2018-10-30 21:57:22 +01:00
|
|
|
|
|
|
|
if (dib_header.compress_type != 0 || dib_header.depth != 24)
|
2015-04-26 23:22:07 +02:00
|
|
|
{
|
2018-10-30 21:57:22 +01:00
|
|
|
printf("Can only convert uncompressed 24 bits BMP files");
|
2022-07-14 22:10:39 +02:00
|
|
|
(void) fclose(input_stream);
|
2018-10-30 21:57:22 +01:00
|
|
|
return EXIT_FAILURE;
|
2014-12-30 12:20:15 +01:00
|
|
|
}
|
|
|
|
|
2020-08-06 18:11:34 +02:00
|
|
|
if (dib_header.width == 0 || dib_header.height == 0)
|
|
|
|
{
|
|
|
|
printf("Can only process an image that is 1 x 1 or bigger");
|
2022-07-14 22:10:39 +02:00
|
|
|
(void) fclose(input_stream);
|
2020-08-06 18:11:34 +02:00
|
|
|
return EXIT_FAILURE;
|
|
|
|
}
|
|
|
|
|
2020-07-31 14:43:05 +02:00
|
|
|
size_t stride;
|
|
|
|
void* pixel_data = bmp_read_pixel_data(input_stream, header.offset, &dib_header, &stride);
|
2022-07-14 22:10:39 +02:00
|
|
|
(void ) fclose(input_stream);
|
2018-10-30 21:57:22 +01:00
|
|
|
|
|
|
|
if (!pixel_data)
|
|
|
|
{
|
|
|
|
printf("Failed to read the BMP pixel data from the file: %s\n", argv[1]);
|
|
|
|
return EXIT_FAILURE;
|
|
|
|
}
|
|
|
|
|
2020-07-31 14:43:05 +02:00
|
|
|
// Pixels in the BMP file format are stored bottom up (when the height parameter is positive), JPEG-LS requires top down.
|
2020-04-26 12:00:45 +02:00
|
|
|
if (dib_header.height > 0)
|
|
|
|
{
|
2020-07-31 14:43:05 +02:00
|
|
|
if (!convert_bottom_up_to_top_down(pixel_data, dib_header.width, (size_t)dib_header.height, stride))
|
2020-04-26 12:00:45 +02:00
|
|
|
{
|
|
|
|
printf("Failed to convert the pixels from bottom up to top down\n");
|
2025-02-13 23:23:04 +01:00
|
|
|
free(pixel_data);
|
2020-04-26 12:00:45 +02:00
|
|
|
return EXIT_FAILURE;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
dib_header.height = abs(dib_header.height);
|
|
|
|
}
|
|
|
|
|
2020-07-31 14:43:05 +02:00
|
|
|
// Pixels in the BMP file format are stored as BGR. JPEG-LS (SPIFF header) only supports the RGB color model.
|
2020-12-30 16:50:55 +01:00
|
|
|
// Note: without the optional SPIFF header no color information is stored in the JPEG-LS file and the common assumption
|
|
|
|
// is RGB.
|
2020-07-31 14:43:05 +02:00
|
|
|
convert_bgr_to_rgb(pixel_data, dib_header.width, (size_t)dib_header.height, stride);
|
|
|
|
|
2018-10-30 21:57:22 +01:00
|
|
|
size_t encoded_size;
|
2020-12-30 16:50:55 +01:00
|
|
|
void* encoded_data =
|
|
|
|
encode_bmp_to_jpegls(pixel_data, stride, &dib_header, options.interleave_mode, options.near_lossless, &encoded_size);
|
2018-10-30 21:57:22 +01:00
|
|
|
free(pixel_data);
|
|
|
|
if (!encoded_data)
|
|
|
|
return EXIT_FAILURE; // error already printed.
|
|
|
|
|
2020-07-31 15:30:58 +02:00
|
|
|
const bool result = save_jpegls_file(options.output_filename, encoded_data, encoded_size);
|
2018-10-30 21:57:22 +01:00
|
|
|
free(encoded_data);
|
|
|
|
if (!result)
|
|
|
|
{
|
2020-07-31 15:30:58 +02:00
|
|
|
printf("Failed to write encoded data to the file: %s\n", options.output_filename);
|
2018-10-30 21:57:22 +01:00
|
|
|
return EXIT_FAILURE;
|
|
|
|
}
|
|
|
|
|
|
|
|
return EXIT_SUCCESS;
|
2020-03-18 08:34:55 +01:00
|
|
|
}
|