1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125 126 127 128 129 130 131 132 133 134 135 136 137 138 139 140 141 142 143 144 145 146 147 148 149 150 151 152 153 154 155 156 157 158 159 160 161 162 163 164 165 166 167 168 169 170 171 172 173 174 175 176 177 178 179 180 181 182 183 184 185 186 187 188 189 190 191 192 193 194 195 196 197 198 199 200 201 202 203 204 205 206 207 208 209 210 211 212 213 214 215 216 217 218 219 220 221 222 223 224 225 226 227 228 229 230 231 232 233 234 235 236 237 238 239 240 241 242 243 244 245 246 247 248 249 250 251 252 253 254 255 256 257 258 259 260 261 262 263 264 265 266 267 268 269 270 271 272 273 274 275 276 277 278 279 280 281 282 283 284 285 286 287 288 289 290 291 292 293 294 295 296 297 298 299 300 301 302 303 304 305 306 307 308 309 310 311 312 313 314 315 316 317 318 319 320 321
|
/*
libheif example application.
MIT License
Copyright (c) 2017 struktur AG, Joachim Bauch <bauch@struktur.de>
Permission is hereby granted, free of charge, to any person obtaining a copy
of this software and associated documentation files (the "Software"), to deal
in the Software without restriction, including without limitation the rights
to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
copies of the Software, and to permit persons to whom the Software is
furnished to do so, subject to the following conditions:
The above copyright notice and this permission notice shall be included in all
copies or substantial portions of the Software.
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
SOFTWARE.
*/
#include <cerrno>
#include <cstring>
#include <vector>
#include <limits>
#include "encoder_jpeg.h"
#include "exif.h"
#include <jpeglib.h>
#include "common_utils.h"
#define JPEG_XMP_MARKER (JPEG_APP0+1) /* JPEG marker code for XMP */
#define JPEG_XMP_MARKER_ID "http://ns.adobe.com/xap/1.0/"
struct ErrorHandler
{
struct jpeg_error_mgr pub; /* "public" fields */
jmp_buf setjmp_buffer; /* for return to caller */
};
JpegEncoder::JpegEncoder(int quality) : quality_(quality)
{
if (quality_ < 0 || quality_ > 100) {
quality_ = kDefaultQuality;
}
}
void JpegEncoder::UpdateDecodingOptions(const struct heif_image_handle* handle,
struct heif_decoding_options* options) const
{
options->convert_hdr_to_8bit = 1;
}
// static
static void OnJpegError(j_common_ptr cinfo)
{
ErrorHandler* handler = reinterpret_cast<ErrorHandler*>(cinfo->err);
longjmp(handler->setjmp_buffer, 1);
}
#define MAX_BYTES_IN_MARKER 65533 /* maximum data len of a JPEG marker */
#if !defined(HAVE_JPEG_WRITE_ICC_PROFILE)
#define ICC_MARKER (JPEG_APP0 + 2) /* JPEG marker code for ICC */
#define ICC_OVERHEAD_LEN 14 /* size of non-profile data in APP2 */
#define MAX_DATA_BYTES_IN_MARKER (MAX_BYTES_IN_MARKER - ICC_OVERHEAD_LEN)
/*
* This routine writes the given ICC profile data into a JPEG file. It *must*
* be called AFTER calling jpeg_start_compress() and BEFORE the first call to
* jpeg_write_scanlines(). (This ordering ensures that the APP2 marker(s) will
* appear after the SOI and JFIF or Adobe markers, but before all else.)
*/
/* This function is copied almost as is from libjpeg-turbo */
static
void jpeg_write_icc_profile(j_compress_ptr cinfo, const JOCTET* icc_data_ptr,
unsigned int icc_data_len)
{
unsigned int num_markers; /* total number of markers we'll write */
int cur_marker = 1; /* per spec, counting starts at 1 */
unsigned int length; /* number of bytes to write in this marker */
/* Calculate the number of markers we'll need, rounding up of course */
num_markers = icc_data_len / MAX_DATA_BYTES_IN_MARKER;
if (num_markers * MAX_DATA_BYTES_IN_MARKER != icc_data_len)
num_markers++;
while (icc_data_len > 0) {
/* length of profile to put in this marker */
length = icc_data_len;
if (length > MAX_DATA_BYTES_IN_MARKER)
length = MAX_DATA_BYTES_IN_MARKER;
icc_data_len -= length;
/* Write the JPEG marker header (APP2 code and marker length) */
jpeg_write_m_header(cinfo, ICC_MARKER,
(unsigned int) (length + ICC_OVERHEAD_LEN));
/* Write the marker identifying string "ICC_PROFILE" (null-terminated). We
* code it in this less-than-transparent way so that the code works even if
* the local character set is not ASCII.
*/
jpeg_write_m_byte(cinfo, 0x49);
jpeg_write_m_byte(cinfo, 0x43);
jpeg_write_m_byte(cinfo, 0x43);
jpeg_write_m_byte(cinfo, 0x5F);
jpeg_write_m_byte(cinfo, 0x50);
jpeg_write_m_byte(cinfo, 0x52);
jpeg_write_m_byte(cinfo, 0x4F);
jpeg_write_m_byte(cinfo, 0x46);
jpeg_write_m_byte(cinfo, 0x49);
jpeg_write_m_byte(cinfo, 0x4C);
jpeg_write_m_byte(cinfo, 0x45);
jpeg_write_m_byte(cinfo, 0x0);
/* Add the sequencing info */
jpeg_write_m_byte(cinfo, cur_marker);
jpeg_write_m_byte(cinfo, (int) num_markers);
/* Add the profile data */
while (length--) {
jpeg_write_m_byte(cinfo, *icc_data_ptr);
icc_data_ptr++;
}
cur_marker++;
}
}
#endif // !defined(HAVE_JPEG_WRITE_ICC_PROFILE)
bool JpegEncoder::Encode(const struct heif_image_handle* handle,
const struct heif_image* image, const std::string& filename)
{
FILE* fp = fopen(filename.c_str(), "wb");
if (!fp) {
fprintf(stderr, "Can't open %s: %s\n", filename.c_str(), strerror(errno));
return false;
}
struct jpeg_compress_struct cinfo;
struct ErrorHandler jerr;
cinfo.err = jpeg_std_error(reinterpret_cast<struct jpeg_error_mgr*>(&jerr));
jerr.pub.error_exit = &OnJpegError;
if (setjmp(jerr.setjmp_buffer)) {
cinfo.err->output_message(reinterpret_cast<j_common_ptr>(&cinfo));
jpeg_destroy_compress(&cinfo);
fclose(fp);
return false;
}
jpeg_create_compress(&cinfo);
jpeg_stdio_dest(&cinfo, fp);
cinfo.image_width = heif_image_get_width(image, heif_channel_Y);
cinfo.image_height = heif_image_get_height(image, heif_channel_Y);
cinfo.input_components = 3;
cinfo.in_color_space = JCS_YCbCr;
jpeg_set_defaults(&cinfo);
static const boolean kForceBaseline = TRUE;
jpeg_set_quality(&cinfo, quality_, kForceBaseline);
static const boolean kWriteAllTables = TRUE;
jpeg_start_compress(&cinfo, kWriteAllTables);
// --- Write EXIF
if (handle) {
size_t exifsize = 0;
uint8_t* exifdata = GetExifMetaData(handle, &exifsize);
if (exifdata) {
if (exifsize > 4) {
static const uint8_t kExifMarker = JPEG_APP0 + 1;
uint32_t skip = four_bytes_to_uint32(exifdata[0], exifdata[1], exifdata[2], exifdata[3]);
if (skip > (exifsize - 4)) {
fprintf(stderr, "Invalid EXIF data (offset too large)\n");
free(exifdata);
jpeg_destroy_compress(&cinfo);
fclose(fp);
return false;
}
skip += 4;
uint8_t* ptr = exifdata + skip;
size_t size = exifsize - skip;
if (size > std::numeric_limits<uint32_t>::max()) {
fprintf(stderr, "EXIF larger than 4GB is not supported");
free(exifdata);
jpeg_destroy_compress(&cinfo);
fclose(fp);
return false;
}
auto size32 = static_cast<uint32_t>(size);
// libheif by default normalizes the image orientation, so that we have to set the EXIF Orientation to "Horizontal (normal)"
modify_exif_orientation_tag_if_it_exists(ptr, size32, 1);
overwrite_exif_image_size_if_it_exists(ptr, size32, cinfo.image_width, cinfo.image_height);
// We have to limit the size for the memcpy, otherwise GCC warns that we exceed the maximum size.
if (size > 0x1000000) {
size = 0x1000000;
}
std::vector<uint8_t> jpegExifMarkerData(6 + size);
memcpy(jpegExifMarkerData.data() + 6, ptr, size);
jpegExifMarkerData[0] = 'E';
jpegExifMarkerData[1] = 'x';
jpegExifMarkerData[2] = 'i';
jpegExifMarkerData[3] = 'f';
jpegExifMarkerData[4] = 0;
jpegExifMarkerData[5] = 0;
ptr = jpegExifMarkerData.data();
size = jpegExifMarkerData.size();
while (size > MAX_BYTES_IN_MARKER) {
jpeg_write_marker(&cinfo, kExifMarker, ptr,
static_cast<unsigned int>(MAX_BYTES_IN_MARKER));
ptr += MAX_BYTES_IN_MARKER;
size -= MAX_BYTES_IN_MARKER;
}
jpeg_write_marker(&cinfo, kExifMarker, ptr,
static_cast<unsigned int>(size));
}
free(exifdata);
}
}
// --- Write XMP
// spec: https://raw.githubusercontent.com/adobe/xmp-docs/master/XMPSpecifications/XMPSpecificationPart3.pdf
if (handle) {
auto xmp = get_xmp_metadata(handle);
if (xmp.size() > 65502) {
fprintf(stderr, "XMP data too large, ExtendedXMP is not supported yet.\n");
}
else if (!xmp.empty()) {
std::vector<uint8_t> xmpWithId;
xmpWithId.resize(xmp.size() + strlen(JPEG_XMP_MARKER_ID) + 1);
strcpy((char*) xmpWithId.data(), JPEG_XMP_MARKER_ID);
memcpy(xmpWithId.data() + strlen(JPEG_XMP_MARKER_ID) + 1, xmp.data(), xmp.size());
jpeg_write_marker(&cinfo, JPEG_XMP_MARKER, xmpWithId.data(), static_cast<unsigned int>(xmpWithId.size()));
}
}
// --- Write ICC
if (handle) {
size_t profile_size = heif_image_handle_get_raw_color_profile_size(handle);
if (profile_size > 0) {
uint8_t* profile_data = static_cast<uint8_t*>(malloc(profile_size));
heif_image_handle_get_raw_color_profile(handle, profile_data);
jpeg_write_icc_profile(&cinfo, profile_data, (unsigned int) profile_size);
free(profile_data);
}
if (heif_image_get_bits_per_pixel(image, heif_channel_Y) != 8) {
fprintf(stderr, "JPEG writer cannot handle image with >8 bpp.\n");
jpeg_destroy_compress(&cinfo);
fclose(fp);
return false;
}
}
size_t stride_y;
const uint8_t* row_y = heif_image_get_plane_readonly2(image, heif_channel_Y,
&stride_y);
size_t stride_u;
const uint8_t* row_u = heif_image_get_plane_readonly2(image, heif_channel_Cb,
&stride_u);
size_t stride_v;
const uint8_t* row_v = heif_image_get_plane_readonly2(image, heif_channel_Cr,
&stride_v);
JSAMPARRAY buffer = cinfo.mem->alloc_sarray(
reinterpret_cast<j_common_ptr>(&cinfo), JPOOL_IMAGE,
cinfo.image_width * cinfo.input_components, 1);
JSAMPROW row[1] = {buffer[0]};
while (cinfo.next_scanline < cinfo.image_height) {
size_t offset_y = cinfo.next_scanline * stride_y;
const uint8_t* start_y = &row_y[offset_y];
size_t offset_u = (cinfo.next_scanline / 2) * stride_u;
const uint8_t* start_u = &row_u[offset_u];
size_t offset_v = (cinfo.next_scanline / 2) * stride_v;
const uint8_t* start_v = &row_v[offset_v];
JOCTET* bufp = buffer[0];
for (JDIMENSION x = 0; x < cinfo.image_width; ++x) {
*bufp++ = start_y[x];
*bufp++ = start_u[x / 2];
*bufp++ = start_v[x / 2];
}
jpeg_write_scanlines(&cinfo, row, 1);
}
jpeg_finish_compress(&cinfo);
fclose(fp);
jpeg_destroy_compress(&cinfo);
return true;
}
|