Coverage Report

Created: 2026-06-30 07:12

next uncovered line (L), next uncovered region (R), next uncovered branch (B)
/src/libjxl/lib/jxl/icc_codec.cc
Line
Count
Source
1
// Copyright (c) the JPEG XL Project Authors. All rights reserved.
2
//
3
// Use of this source code is governed by a BSD-style
4
// license that can be found in the LICENSE file.
5
6
#include "lib/jxl/icc_codec.h"
7
8
#include <jxl/memory_manager.h>
9
10
#include <algorithm>
11
#include <cstddef>
12
#include <cstdint>
13
14
#include "lib/jxl/base/common.h"
15
#include "lib/jxl/base/status.h"
16
#include "lib/jxl/dec_ans.h"
17
#include "lib/jxl/dec_bit_reader.h"
18
#include "lib/jxl/fields.h"
19
#include "lib/jxl/icc_codec_common.h"
20
#include "lib/jxl/padded_bytes.h"
21
22
namespace jxl {
23
namespace {
24
25
// Shuffles or interleaves bytes, for example with width 2, turns "ABCDabcd"
26
// into "AaBbCcDd". Transposes a matrix of ceil(size / width) columns and
27
// width rows. There are size elements, size may be < width * height, if so the
28
// last elements of the rightmost column are missing, the missing spots are
29
// transposed along with the filled spots, and the result has the missing
30
// elements at the end of the bottom row. The input is the input matrix in
31
// scanline order but with missing elements skipped (which may occur in multiple
32
// locations), the output is the result matrix in scanline order (with
33
// no need to skip missing elements as they are past the end of the data).
34
Status Shuffle(JxlMemoryManager* memory_manager, uint8_t* data, size_t size,
35
5.15k
               size_t width) {
36
5.15k
  size_t height = (size + width - 1) / width;  // amount of rows of output
37
5.15k
  PaddedBytes result(memory_manager);
38
5.15k
  JXL_ASSIGN_OR_RETURN(result,
39
5.15k
                       PaddedBytes::WithInitialSpace(memory_manager, size));
40
  // i = output index, j input index
41
5.15k
  size_t s = 0;
42
5.15k
  size_t j = 0;
43
74.1k
  for (size_t i = 0; i < size; i++) {
44
68.9k
    result[i] = data[j];
45
68.9k
    j += height;
46
68.9k
    if (j >= size) j = ++s;
47
68.9k
  }
48
49
74.1k
  for (size_t i = 0; i < size; i++) {
50
68.9k
    data[i] = result[i];
51
68.9k
  }
52
5.15k
  return true;
53
5.15k
}
54
55
// Two base-128 varints at up to 10 bytes each.
56
constexpr size_t kPreambleSize = 20;
57
58
// Decodes a base-128 unsigned varint into *out, advancing *pos by the exact
59
// number of bytes consumed. Returns an error if the input is truncated, if
60
// the terminator (top-bit-clear byte) is not seen within 10 bytes, or if the
61
// 10th byte encodes a value that does not fit in a uint64_t.
62
Status DecodeVarInt(const uint8_t* input, size_t inputSize, size_t* pos,
63
19.2k
                    uint64_t* out) {
64
19.2k
  uint64_t ret = 0;
65
  // 9 bytes cover bits 0..62; the 10th byte may only contribute bit 63.
66
24.9k
  for (size_t i = 0; i < 9; ++i) {
67
24.8k
    if (*pos >= inputSize) {
68
115
      return JXL_FAILURE("DecodeVarInt: truncated input");
69
115
    }
70
24.7k
    const uint8_t byte = input[(*pos)++];
71
24.7k
    ret |= static_cast<uint64_t>(byte & 0x7F) << (7 * i);
72
24.7k
    if ((byte & 0x80) == 0) {
73
19.1k
      *out = ret;
74
19.1k
      return true;
75
19.1k
    }
76
24.7k
  }
77
73
  if (*pos >= inputSize) {
78
1
    return JXL_FAILURE("DecodeVarInt: truncated input (10th byte)");
79
1
  }
80
72
  const uint8_t byte = input[(*pos)++];
81
72
  if ((byte & 0x80) != 0) {
82
67
    return JXL_FAILURE("DecodeVarInt: varint exceeds 10 bytes");
83
67
  }
84
5
  if ((byte & 0x7E) != 0) {
85
4
    return JXL_FAILURE("DecodeVarInt: value exceeds 2^64 - 1");
86
4
  }
87
1
  ret |= static_cast<uint64_t>(byte & 0x01) << 63;
88
1
  *out = ret;
89
1
  return true;
90
5
}
91
92
}  // namespace
93
94
// Mimics the beginning of UnpredictICC for quick validity check.
95
// At least kPreambleSize bytes of data should be valid at invocation time.
96
1.10k
Status CheckPreamble(const PaddedBytes& data, size_t enc_size) {
97
1.10k
  const uint8_t* enc = data.data();
98
1.10k
  size_t size = data.size();
99
1.10k
  size_t pos = 0;
100
1.10k
  uint64_t osize;
101
1.10k
  JXL_RETURN_IF_ERROR(DecodeVarInt(enc, size, &pos, &osize));
102
1.07k
  JXL_RETURN_IF_ERROR(CheckIs32Bit(osize));
103
1.07k
  uint64_t csize;
104
1.07k
  JXL_RETURN_IF_ERROR(DecodeVarInt(enc, size, &pos, &csize));
105
1.06k
  JXL_RETURN_IF_ERROR(CheckIs32Bit(csize));
106
1.06k
  JXL_RETURN_IF_ERROR(CheckOutOfBounds(pos, csize, size));
107
  // We expect that UnpredictICC inflates input, not the other way round.
108
1.04k
  if (osize + 65536 < enc_size) return JXL_FAILURE("Malformed ICC");
109
110
  // NB(eustas): 64 MiB ICC should be enough for everything!?
111
1.02k
  const size_t output_limit = 1 << 28;
112
1.02k
  if (output_limit && osize > output_limit) {
113
4
    return JXL_FAILURE("Decoded ICC is too large");
114
4
  }
115
1.02k
  return true;
116
1.02k
}
117
118
// Decodes the result of PredictICC back to a valid ICC profile.
119
872
Status UnpredictICC(const uint8_t* enc, size_t size, PaddedBytes* result) {
120
872
  if (!result->empty()) return JXL_FAILURE("result must be empty initially");
121
872
  JxlMemoryManager* memory_manager = result->memory_manager();
122
872
  size_t pos = 0;
123
872
  uint64_t osize;
124
872
  JXL_RETURN_IF_ERROR(DecodeVarInt(enc, size, &pos, &osize));  // Output size
125
854
  JXL_RETURN_IF_ERROR(CheckIs32Bit(osize));
126
852
  uint64_t csize;
127
852
  JXL_RETURN_IF_ERROR(DecodeVarInt(enc, size, &pos, &csize));  // Commands size
128
  // Every command is translated to at least one byte.
129
836
  JXL_RETURN_IF_ERROR(CheckIs32Bit(csize));
130
835
  size_t cpos = pos;  // pos in commands stream
131
835
  JXL_RETURN_IF_ERROR(CheckOutOfBounds(pos, csize, size));
132
822
  size_t commands_end = cpos + csize;
133
822
  pos = commands_end;  // pos in data stream
134
135
  // Header
136
822
  PaddedBytes header{memory_manager};
137
822
  JXL_RETURN_IF_ERROR(header.append(ICCInitialHeaderPrediction(osize)));
138
81.4k
  for (size_t i = 0; i <= kICCHeaderSize; i++) {
139
81.4k
    if (result->size() == osize) {
140
199
      if (cpos != commands_end) return JXL_FAILURE("Not all commands used");
141
136
      if (pos != size) return JXL_FAILURE("Not all data used");
142
20
      return true;  // Valid end
143
136
    }
144
81.2k
    if (i == kICCHeaderSize) break;  // Done
145
80.5k
    ICCPredictHeader(result->data(), result->size(), header.data(), i);
146
80.5k
    if (pos >= size) return JXL_FAILURE("Out of bounds");
147
80.5k
    JXL_RETURN_IF_ERROR(result->push_back(enc[pos++] + header[i]));
148
80.5k
  }
149
611
  if (cpos >= commands_end) return JXL_FAILURE("Out of bounds");
150
151
  // Tag list
152
606
  uint64_t numtags;
153
606
  JXL_RETURN_IF_ERROR(DecodeVarInt(enc, commands_end, &cpos, &numtags));
154
155
598
  if (numtags != 0) {
156
407
    numtags--;
157
407
    JXL_RETURN_IF_ERROR(CheckIs32Bit(numtags));
158
405
    JXL_RETURN_IF_ERROR(AppendUint32(numtags, result));
159
405
    uint64_t prevtagstart = kICCHeaderSize + numtags * 12;
160
405
    uint64_t prevtagsize = 0;
161
12.0k
    for (;;) {
162
12.0k
      if (result->size() > osize) return JXL_FAILURE("Invalid result size");
163
12.0k
      if (cpos > commands_end) return JXL_FAILURE("Out of bounds");
164
12.0k
      if (cpos == commands_end) break;  // Valid end
165
11.9k
      uint8_t command = enc[cpos++];
166
11.9k
      uint8_t tagcode = command & 63;
167
11.9k
      Tag tag;
168
11.9k
      if (tagcode == 0) {
169
220
        break;
170
11.7k
      } else if (tagcode == kCommandTagUnknown) {
171
1.03k
        JXL_RETURN_IF_ERROR(CheckOutOfBounds(pos, 4, size));
172
1.03k
        tag = DecodeKeyword(enc, size, pos);
173
1.03k
        pos += 4;
174
10.6k
      } else if (tagcode == kCommandTagTRC) {
175
1.82k
        tag = kRtrcTag;
176
8.87k
      } else if (tagcode == kCommandTagXYZ) {
177
938
        tag = kRxyzTag;
178
7.93k
      } else {
179
7.93k
        if (tagcode - kCommandTagStringFirst >= kNumTagStrings) {
180
26
          return JXL_FAILURE("Unknown tagcode");
181
26
        }
182
7.90k
        tag = *kTagStrings[tagcode - kCommandTagStringFirst];
183
7.90k
      }
184
11.7k
      JXL_RETURN_IF_ERROR(AppendKeyword(tag, result));
185
186
11.7k
      uint64_t tagstart;
187
11.7k
      uint64_t tagsize = prevtagsize;
188
11.7k
      if (tag == kRxyzTag || tag == kGxyzTag || tag == kBxyzTag ||
189
8.29k
          tag == kKxyzTag || tag == kWtptTag || tag == kBkptTag ||
190
6.45k
          tag == kLumiTag) {
191
6.45k
        tagsize = 20;
192
6.45k
      }
193
194
11.7k
      if (command & kFlagBitOffset) {
195
2.07k
        JXL_RETURN_IF_ERROR(DecodeVarInt(enc, commands_end, &cpos, &tagstart));
196
9.62k
      } else {
197
9.62k
        JXL_RETURN_IF_ERROR(CheckIs32Bit(prevtagstart));
198
9.62k
        tagstart = prevtagstart + prevtagsize;
199
9.62k
      }
200
11.6k
      JXL_RETURN_IF_ERROR(CheckIs32Bit(tagstart));
201
11.6k
      JXL_RETURN_IF_ERROR(AppendUint32(tagstart, result));
202
11.6k
      if (command & kFlagBitSize) {
203
2.74k
        JXL_RETURN_IF_ERROR(DecodeVarInt(enc, commands_end, &cpos, &tagsize));
204
2.74k
      }
205
11.6k
      JXL_RETURN_IF_ERROR(CheckIs32Bit(tagsize));
206
11.6k
      JXL_RETURN_IF_ERROR(AppendUint32(tagsize, result));
207
11.6k
      prevtagstart = tagstart;
208
11.6k
      prevtagsize = tagsize;
209
210
11.6k
      if (tagcode == kCommandTagTRC) {
211
1.81k
        JXL_RETURN_IF_ERROR(AppendKeyword(kGtrcTag, result));
212
1.81k
        JXL_RETURN_IF_ERROR(AppendUint32(tagstart, result));
213
1.81k
        JXL_RETURN_IF_ERROR(AppendUint32(tagsize, result));
214
1.81k
        JXL_RETURN_IF_ERROR(AppendKeyword(kBtrcTag, result));
215
1.81k
        JXL_RETURN_IF_ERROR(AppendUint32(tagstart, result));
216
1.81k
        JXL_RETURN_IF_ERROR(AppendUint32(tagsize, result));
217
1.81k
      }
218
219
11.6k
      if (tagcode == kCommandTagXYZ) {
220
932
        JXL_RETURN_IF_ERROR(CheckIs32Bit(tagstart + tagsize * 2));
221
931
        JXL_RETURN_IF_ERROR(AppendKeyword(kGxyzTag, result));
222
931
        JXL_RETURN_IF_ERROR(AppendUint32(tagstart + tagsize, result));
223
931
        JXL_RETURN_IF_ERROR(AppendUint32(tagsize, result));
224
931
        JXL_RETURN_IF_ERROR(AppendKeyword(kBxyzTag, result));
225
931
        JXL_RETURN_IF_ERROR(AppendUint32(tagstart + tagsize * 2, result));
226
931
        JXL_RETURN_IF_ERROR(AppendUint32(tagsize, result));
227
931
      }
228
11.6k
    }
229
405
  }
230
231
  // Main Content
232
12.1k
  for (;;) {
233
12.1k
    if (result->size() > osize) return JXL_FAILURE("Invalid result size");
234
12.1k
    if (cpos > commands_end) return JXL_FAILURE("Out of bounds");
235
12.1k
    if (cpos == commands_end) break;  // Valid end
236
12.0k
    uint8_t command = enc[cpos++];
237
12.0k
    if (command == kCommandInsert) {
238
1.68k
      uint64_t num;
239
1.68k
      JXL_RETURN_IF_ERROR(DecodeVarInt(enc, commands_end, &cpos, &num));
240
1.68k
      JXL_RETURN_IF_ERROR(CheckOutOfBounds(pos, num, size));
241
10.8k
      for (size_t i = 0; i < num; i++) {
242
9.13k
        JXL_RETURN_IF_ERROR(result->push_back(enc[pos++]));
243
9.13k
      }
244
10.3k
    } else if (command == kCommandShuffle2 || command == kCommandShuffle4) {
245
3.29k
      uint64_t num;
246
3.29k
      JXL_RETURN_IF_ERROR(DecodeVarInt(enc, commands_end, &cpos, &num));
247
3.26k
      JXL_RETURN_IF_ERROR(CheckOutOfBounds(pos, num, size));
248
3.25k
      PaddedBytes shuffled(memory_manager);
249
3.25k
      JXL_ASSIGN_OR_RETURN(shuffled,
250
3.25k
                           PaddedBytes::WithInitialSpace(memory_manager, num));
251
27.5k
      for (size_t i = 0; i < num; i++) {
252
24.2k
        shuffled[i] = enc[pos + i];
253
24.2k
      }
254
3.25k
      if (command == kCommandShuffle2) {
255
1.85k
        JXL_RETURN_IF_ERROR(Shuffle(memory_manager, shuffled.data(), num, 2));
256
1.85k
      } else if (command == kCommandShuffle4) {
257
1.39k
        JXL_RETURN_IF_ERROR(Shuffle(memory_manager, shuffled.data(), num, 4));
258
1.39k
      }
259
27.5k
      for (size_t i = 0; i < num; i++) {
260
24.2k
        JXL_RETURN_IF_ERROR(result->push_back(shuffled[i]));
261
24.2k
        pos++;
262
24.2k
      }
263
7.02k
    } else if (command == kCommandPredict) {
264
4.02k
      JXL_RETURN_IF_ERROR(CheckOutOfBounds(cpos, 2, commands_end));
265
4.00k
      uint8_t flags = enc[cpos++];
266
267
4.00k
      size_t width = (flags & 3) + 1;
268
4.00k
      if (width == 3) return JXL_FAILURE("Invalid width");
269
270
3.99k
      int order = (flags & 12) >> 2;
271
3.99k
      if (order == 3) return JXL_FAILURE("Invalid order");
272
273
3.99k
      uint64_t stride = width;
274
3.99k
      if (flags & 16) {
275
1.05k
        JXL_RETURN_IF_ERROR(DecodeVarInt(enc, commands_end, &cpos, &stride));
276
1.03k
        if (stride < width) {
277
3
          return JXL_FAILURE("Invalid stride");
278
3
        }
279
1.03k
      }
280
      // If stride * 4 >= result->size(), return failure. The check
281
      // "size == 0 || ((size - 1) >> 2) < stride" corresponds to
282
      // "stride * 4 >= size", but does not suffer from integer overflow.
283
      // This check is more strict than necessary but follows the specification
284
      // and the encoder should ensure this is followed.
285
3.96k
      if (result->empty() || ((result->size() - 1u) >> 2u) < stride) {
286
39
        return JXL_FAILURE("Invalid stride");
287
39
      }
288
289
3.92k
      uint64_t num;
290
3.92k
      JXL_RETURN_IF_ERROR(DecodeVarInt(enc, commands_end, &cpos, &num));  // in bytes
291
3.92k
      JXL_RETURN_IF_ERROR(CheckOutOfBounds(pos, num, size));
292
293
3.89k
      PaddedBytes shuffled(memory_manager);
294
3.89k
      JXL_ASSIGN_OR_RETURN(shuffled,
295
3.89k
                           PaddedBytes::WithInitialSpace(memory_manager, num));
296
297
68.2k
      for (size_t i = 0; i < num; i++) {
298
64.3k
        shuffled[i] = enc[pos + i];
299
64.3k
      }
300
3.89k
      if (width > 1) {
301
1.89k
        JXL_RETURN_IF_ERROR(
302
1.89k
            Shuffle(memory_manager, shuffled.data(), num, width));
303
1.89k
      }
304
305
3.89k
      size_t start = result->size();
306
68.2k
      for (size_t i = 0; i < num; i++) {
307
64.3k
        uint8_t predicted = LinearPredictICCValue(result->data(), start, i,
308
64.3k
                                                  stride, width, order);
309
64.3k
        JXL_RETURN_IF_ERROR(result->push_back(predicted + shuffled[i]));
310
64.3k
      }
311
3.89k
      pos += num;
312
3.89k
    } else if (command == kCommandXYZ) {
313
471
      JXL_RETURN_IF_ERROR(AppendKeyword(kXyz_Tag, result));
314
2.35k
      for (int i = 0; i < 4; i++) {
315
1.88k
        JXL_RETURN_IF_ERROR(result->push_back(0));
316
1.88k
      }
317
471
      JXL_RETURN_IF_ERROR(CheckOutOfBounds(pos, 12, size));
318
6.00k
      for (size_t i = 0; i < 12; i++) {
319
5.54k
        JXL_RETURN_IF_ERROR(result->push_back(enc[pos++]));
320
5.54k
      }
321
2.53k
    } else if (command >= kCommandTypeStartFirst &&
322
2.47k
               command < kCommandTypeStartFirst + kNumTypeStrings) {
323
2.37k
      JXL_RETURN_IF_ERROR(AppendKeyword(
324
2.37k
          *kTypeStrings[command - kCommandTypeStartFirst], result));
325
11.8k
      for (size_t i = 0; i < 4; i++) {
326
9.51k
        JXL_RETURN_IF_ERROR(result->push_back(0));
327
9.51k
      }
328
2.37k
    } else {
329
153
      return JXL_FAILURE("Unknown command");
330
153
    }
331
12.0k
  }
332
333
132
  if (pos != size) return JXL_FAILURE("Not all data used");
334
19
  if (result->size() != osize) return JXL_FAILURE("Invalid result size");
335
336
18
  return true;
337
19
}
338
339
2.90k
Status ICCReader::Init(BitReader* reader) {
340
2.90k
  JXL_RETURN_IF_ERROR(CheckEOI(reader));
341
2.90k
  JxlMemoryManager* memory_manager = decompressed_.memory_manager();
342
2.90k
  used_bits_base_ = reader->TotalBitsConsumed();
343
2.90k
  if (bits_to_skip_ == 0) {
344
2.45k
    enc_size_ = U64Coder::Read(reader);
345
2.45k
    if (enc_size_ > 268435456) {
346
      // Avoid too large memory allocation for invalid file.
347
54
      return JXL_FAILURE("Too large encoded profile");
348
54
    }
349
2.40k
    JXL_RETURN_IF_ERROR(DecodeHistograms(
350
2.40k
        memory_manager, reader, kNumICCContexts, &code_, &context_map_));
351
3.82k
    JXL_ASSIGN_OR_RETURN(ans_reader_, ANSSymbolReader::Create(&code_, reader));
352
3.82k
    i_ = 0;
353
3.82k
    JXL_RETURN_IF_ERROR(
354
3.82k
        decompressed_.resize(std::min<size_t>(i_ + 0x400, enc_size_)));
355
5.44k
    for (; i_ < std::min<size_t>(2, enc_size_); i_++) {
356
3.53k
      decompressed_[i_] = ans_reader_.ReadHybridUint(
357
3.53k
          ICCANSContext(i_, i_ > 0 ? decompressed_[i_ - 1] : 0,
358
3.53k
                        i_ > 1 ? decompressed_[i_ - 2] : 0),
359
3.53k
          reader, context_map_);
360
3.53k
    }
361
1.91k
    if (enc_size_ > kPreambleSize) {
362
30.5k
      for (; i_ < kPreambleSize; i_++) {
363
28.9k
        decompressed_[i_] = ans_reader_.ReadHybridUint(
364
28.9k
            ICCANSContext(i_, decompressed_[i_ - 1], decompressed_[i_ - 2]),
365
28.9k
            reader, context_map_);
366
28.9k
      }
367
1.60k
      JXL_RETURN_IF_ERROR(CheckEOI(reader));
368
1.10k
      JXL_RETURN_IF_ERROR(CheckPreamble(decompressed_, enc_size_));
369
1.10k
    }
370
1.32k
    bits_to_skip_ = reader->TotalBitsConsumed() - used_bits_base_;
371
1.32k
  } else {
372
447
    reader->SkipBits(bits_to_skip_);
373
447
  }
374
1.77k
  return true;
375
2.90k
}
376
377
1.36k
Status ICCReader::Process(BitReader* reader, PaddedBytes* icc) {
378
1.36k
  auto checkpoint = jxl::make_unique<ANSSymbolReader::Checkpoint>();
379
1.36k
  size_t saved_i = 0;
380
13.9k
  auto save = [&]() {
381
13.9k
    ans_reader_.Save(checkpoint.get());
382
13.9k
    bits_to_skip_ = reader->TotalBitsConsumed() - used_bits_base_;
383
13.9k
    saved_i = i_;
384
13.9k
  };
385
1.36k
  save();
386
13.9k
  auto check_and_restore = [&]() -> Status {
387
13.9k
    Status status = CheckEOI(reader);
388
13.9k
    if (!status) {
389
      // not enough bytes.
390
490
      ans_reader_.Restore(*checkpoint);
391
490
      i_ = saved_i;
392
490
      return status;
393
490
    }
394
13.4k
    return true;
395
13.9k
  };
396
6.82M
  for (; i_ < enc_size_; i_++) {
397
6.82M
    if (i_ % ANSSymbolReader::kMaxCheckpointInterval == 0 && i_ > 0) {
398
12.8k
      JXL_RETURN_IF_ERROR(check_and_restore());
399
12.5k
      save();
400
12.5k
      if ((i_ > 0) && (((i_ & 0xFFFF) == 0))) {
401
69
        float used_bytes =
402
69
            (reader->TotalBitsConsumed() - used_bits_base_) / 8.0f;
403
69
        if (i_ > used_bytes * 256) return JXL_FAILURE("Corrupted stream");
404
69
      }
405
12.5k
      JXL_RETURN_IF_ERROR(
406
12.5k
          decompressed_.resize(std::min<size_t>(i_ + 0x400, enc_size_)));
407
12.5k
    }
408
6.82M
    JXL_ENSURE(i_ >= 2);
409
6.82M
    decompressed_[i_] = ans_reader_.ReadHybridUint(
410
6.82M
        ICCANSContext(i_, decompressed_[i_ - 1], decompressed_[i_ - 2]), reader,
411
6.82M
        context_map_);
412
6.82M
  }
413
1.05k
  JXL_RETURN_IF_ERROR(check_and_restore());
414
872
  bits_to_skip_ = reader->TotalBitsConsumed() - used_bits_base_;
415
872
  if (!ans_reader_.CheckANSFinalState()) {
416
0
    return JXL_FAILURE("Corrupted ICC profile");
417
0
  }
418
419
872
  icc->clear();
420
872
  return UnpredictICC(decompressed_.data(), decompressed_.size(), icc);
421
872
}
422
423
18.4k
Status ICCReader::CheckEOI(BitReader* reader) {
424
18.4k
  if (reader->AllReadsWithinBounds()) return true;
425
996
  return JXL_NOT_ENOUGH_BYTES("Not enough bytes for reading ICC profile");
426
18.4k
}
427
428
}  // namespace jxl