Coverage Report

Created: 2026-06-07 07:20

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.14k
               size_t width) {
36
5.14k
  size_t height = (size + width - 1) / width;  // amount of rows of output
37
5.14k
  PaddedBytes result(memory_manager);
38
5.14k
  JXL_ASSIGN_OR_RETURN(result,
39
5.14k
                       PaddedBytes::WithInitialSpace(memory_manager, size));
40
  // i = output index, j input index
41
5.14k
  size_t s = 0;
42
5.14k
  size_t j = 0;
43
73.7k
  for (size_t i = 0; i < size; i++) {
44
68.6k
    result[i] = data[j];
45
68.6k
    j += height;
46
68.6k
    if (j >= size) j = ++s;
47
68.6k
  }
48
49
73.7k
  for (size_t i = 0; i < size; i++) {
50
68.6k
    data[i] = result[i];
51
68.6k
  }
52
5.14k
  return true;
53
5.14k
}
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.8k
  for (size_t i = 0; i < 9; ++i) {
67
24.8k
    if (*pos >= inputSize) {
68
114
      return JXL_FAILURE("DecodeVarInt: truncated input");
69
114
    }
70
24.6k
    const uint8_t byte = input[(*pos)++];
71
24.6k
    ret |= static_cast<uint64_t>(byte & 0x7F) << (7 * i);
72
24.6k
    if ((byte & 0x80) == 0) {
73
19.0k
      *out = ret;
74
19.0k
      return true;
75
19.0k
    }
76
24.6k
  }
77
75
  if (*pos >= inputSize) {
78
1
    return JXL_FAILURE("DecodeVarInt: truncated input (10th byte)");
79
1
  }
80
74
  const uint8_t byte = input[(*pos)++];
81
74
  if ((byte & 0x80) != 0) {
82
69
    return JXL_FAILURE("DecodeVarInt: varint exceeds 10 bytes");
83
69
  }
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.01k
  return true;
116
1.02k
}
117
118
// Decodes the result of PredictICC back to a valid ICC profile.
119
868
Status UnpredictICC(const uint8_t* enc, size_t size, PaddedBytes* result) {
120
868
  if (!result->empty()) return JXL_FAILURE("result must be empty initially");
121
868
  JxlMemoryManager* memory_manager = result->memory_manager();
122
868
  size_t pos = 0;
123
868
  uint64_t osize;
124
868
  JXL_RETURN_IF_ERROR(DecodeVarInt(enc, size, &pos, &osize));  // Output size
125
849
  JXL_RETURN_IF_ERROR(CheckIs32Bit(osize));
126
847
  uint64_t csize;
127
847
  JXL_RETURN_IF_ERROR(DecodeVarInt(enc, size, &pos, &csize));  // Commands size
128
  // Every command is translated to at least one byte.
129
833
  JXL_RETURN_IF_ERROR(CheckIs32Bit(csize));
130
831
  size_t cpos = pos;  // pos in commands stream
131
831
  JXL_RETURN_IF_ERROR(CheckOutOfBounds(pos, csize, size));
132
818
  size_t commands_end = cpos + csize;
133
818
  pos = commands_end;  // pos in data stream
134
135
  // Header
136
818
  PaddedBytes header{memory_manager};
137
818
  JXL_RETURN_IF_ERROR(header.append(ICCInitialHeaderPrediction(osize)));
138
80.5k
  for (size_t i = 0; i <= kICCHeaderSize; i++) {
139
80.5k
    if (result->size() == osize) {
140
201
      if (cpos != commands_end) return JXL_FAILURE("Not all commands used");
141
138
      if (pos != size) return JXL_FAILURE("Not all data used");
142
20
      return true;  // Valid end
143
138
    }
144
80.3k
    if (i == kICCHeaderSize) break;  // Done
145
79.7k
    ICCPredictHeader(result->data(), result->size(), header.data(), i);
146
79.7k
    if (pos >= size) return JXL_FAILURE("Out of bounds");
147
79.7k
    JXL_RETURN_IF_ERROR(result->push_back(enc[pos++] + header[i]));
148
79.7k
  }
149
604
  if (cpos >= commands_end) return JXL_FAILURE("Out of bounds");
150
151
  // Tag list
152
599
  uint64_t numtags;
153
599
  JXL_RETURN_IF_ERROR(DecodeVarInt(enc, commands_end, &cpos, &numtags));
154
155
591
  if (numtags != 0) {
156
399
    numtags--;
157
399
    JXL_RETURN_IF_ERROR(CheckIs32Bit(numtags));
158
398
    JXL_RETURN_IF_ERROR(AppendUint32(numtags, result));
159
398
    uint64_t prevtagstart = kICCHeaderSize + numtags * 12;
160
398
    uint64_t prevtagsize = 0;
161
11.9k
    for (;;) {
162
11.9k
      if (result->size() > osize) return JXL_FAILURE("Invalid result size");
163
11.9k
      if (cpos > commands_end) return JXL_FAILURE("Out of bounds");
164
11.9k
      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
215
        break;
170
11.6k
      } 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.83k
      } else if (tagcode == kCommandTagXYZ) {
177
919
        tag = kRxyzTag;
178
7.91k
      } else {
179
7.91k
        if (tagcode - kCommandTagStringFirst >= kNumTagStrings) {
180
25
          return JXL_FAILURE("Unknown tagcode");
181
25
        }
182
7.88k
        tag = *kTagStrings[tagcode - kCommandTagStringFirst];
183
7.88k
      }
184
11.6k
      JXL_RETURN_IF_ERROR(AppendKeyword(tag, result));
185
186
11.6k
      uint64_t tagstart;
187
11.6k
      uint64_t tagsize = prevtagsize;
188
11.6k
      if (tag == kRxyzTag || tag == kGxyzTag || tag == kBxyzTag ||
189
8.27k
          tag == kKxyzTag || tag == kWtptTag || tag == kBkptTag ||
190
6.42k
          tag == kLumiTag) {
191
6.42k
        tagsize = 20;
192
6.42k
      }
193
194
11.6k
      if (command & kFlagBitOffset) {
195
2.07k
        JXL_RETURN_IF_ERROR(DecodeVarInt(enc, commands_end, &cpos, &tagstart));
196
9.58k
      } else {
197
9.58k
        JXL_RETURN_IF_ERROR(CheckIs32Bit(prevtagstart));
198
9.58k
        tagstart = prevtagstart + prevtagsize;
199
9.58k
      }
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.73k
        JXL_RETURN_IF_ERROR(DecodeVarInt(enc, commands_end, &cpos, &tagsize));
204
2.73k
      }
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
914
        JXL_RETURN_IF_ERROR(CheckIs32Bit(tagstart + tagsize * 2));
221
913
        JXL_RETURN_IF_ERROR(AppendKeyword(kGxyzTag, result));
222
913
        JXL_RETURN_IF_ERROR(AppendUint32(tagstart + tagsize, result));
223
913
        JXL_RETURN_IF_ERROR(AppendUint32(tagsize, result));
224
913
        JXL_RETURN_IF_ERROR(AppendKeyword(kBxyzTag, result));
225
913
        JXL_RETURN_IF_ERROR(AppendUint32(tagstart + tagsize * 2, result));
226
913
        JXL_RETURN_IF_ERROR(AppendUint32(tagsize, result));
227
913
      }
228
11.6k
    }
229
398
  }
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.69k
      uint64_t num;
239
1.69k
      JXL_RETURN_IF_ERROR(DecodeVarInt(enc, commands_end, &cpos, &num));
240
1.68k
      JXL_RETURN_IF_ERROR(CheckOutOfBounds(pos, num, size));
241
11.3k
      for (size_t i = 0; i < num; i++) {
242
9.62k
        JXL_RETURN_IF_ERROR(result->push_back(enc[pos++]));
243
9.62k
      }
244
10.3k
    } else if (command == kCommandShuffle2 || command == kCommandShuffle4) {
245
3.28k
      uint64_t num;
246
3.28k
      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.4k
      for (size_t i = 0; i < num; i++) {
252
24.1k
        shuffled[i] = enc[pos + i];
253
24.1k
      }
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.4k
      for (size_t i = 0; i < num; i++) {
260
24.1k
        JXL_RETURN_IF_ERROR(result->push_back(shuffled[i]));
261
24.1k
        pos++;
262
24.1k
      }
263
7.01k
    } 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
38
        return JXL_FAILURE("Invalid stride");
287
38
      }
288
289
3.93k
      uint64_t num;
290
3.93k
      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.0k
      for (size_t i = 0; i < num; i++) {
298
64.1k
        shuffled[i] = enc[pos + i];
299
64.1k
      }
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.0k
      for (size_t i = 0; i < num; i++) {
307
64.1k
        uint8_t predicted = LinearPredictICCValue(result->data(), start, i,
308
64.1k
                                                  stride, width, order);
309
64.1k
        JXL_RETURN_IF_ERROR(result->push_back(predicted + shuffled[i]));
310
64.1k
      }
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.52k
    } else if (command >= kCommandTypeStartFirst &&
322
2.46k
               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.49k
        JXL_RETURN_IF_ERROR(result->push_back(0));
327
9.49k
      }
328
2.37k
    } else {
329
152
      return JXL_FAILURE("Unknown command");
330
152
    }
331
12.0k
  }
332
333
131
  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.89k
Status ICCReader::Init(BitReader* reader) {
340
2.89k
  JXL_RETURN_IF_ERROR(CheckEOI(reader));
341
2.89k
  JxlMemoryManager* memory_manager = decompressed_.memory_manager();
342
2.89k
  used_bits_base_ = reader->TotalBitsConsumed();
343
2.89k
  if (bits_to_skip_ == 0) {
344
2.44k
    enc_size_ = U64Coder::Read(reader);
345
2.44k
    if (enc_size_ > 268435456) {
346
      // Avoid too large memory allocation for invalid file.
347
62
      return JXL_FAILURE("Too large encoded profile");
348
62
    }
349
2.38k
    JXL_RETURN_IF_ERROR(DecodeHistograms(
350
2.38k
        memory_manager, reader, kNumICCContexts, &code_, &context_map_));
351
3.78k
    JXL_ASSIGN_OR_RETURN(ans_reader_, ANSSymbolReader::Create(&code_, reader));
352
3.78k
    i_ = 0;
353
3.78k
    JXL_RETURN_IF_ERROR(
354
3.78k
        decompressed_.resize(std::min<size_t>(i_ + 0x400, enc_size_)));
355
5.39k
    for (; i_ < std::min<size_t>(2, enc_size_); i_++) {
356
3.50k
      decompressed_[i_] = ans_reader_.ReadHybridUint(
357
3.50k
          ICCANSContext(i_, i_ > 0 ? decompressed_[i_ - 1] : 0,
358
3.50k
                        i_ > 1 ? decompressed_[i_ - 2] : 0),
359
3.50k
          reader, context_map_);
360
3.50k
    }
361
1.89k
    if (enc_size_ > kPreambleSize) {
362
30.2k
      for (; i_ < kPreambleSize; i_++) {
363
28.6k
        decompressed_[i_] = ans_reader_.ReadHybridUint(
364
28.6k
            ICCANSContext(i_, decompressed_[i_ - 1], decompressed_[i_ - 2]),
365
28.6k
            reader, context_map_);
366
28.6k
      }
367
1.59k
      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
448
    reader->SkipBits(bits_to_skip_);
373
448
  }
374
1.77k
  return true;
375
2.89k
}
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.8k
  auto save = [&]() {
381
13.8k
    ans_reader_.Save(checkpoint.get());
382
13.8k
    bits_to_skip_ = reader->TotalBitsConsumed() - used_bits_base_;
383
13.8k
    saved_i = i_;
384
13.8k
  };
385
1.36k
  save();
386
13.8k
  auto check_and_restore = [&]() -> Status {
387
13.8k
    Status status = CheckEOI(reader);
388
13.8k
    if (!status) {
389
      // not enough bytes.
390
492
      ans_reader_.Restore(*checkpoint);
391
492
      i_ = saved_i;
392
492
      return status;
393
492
    }
394
13.3k
    return true;
395
13.8k
  };
396
6.75M
  for (; i_ < enc_size_; i_++) {
397
6.75M
    if (i_ % ANSSymbolReader::kMaxCheckpointInterval == 0 && i_ > 0) {
398
12.7k
      JXL_RETURN_IF_ERROR(check_and_restore());
399
12.4k
      save();
400
12.4k
      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.4k
      JXL_RETURN_IF_ERROR(
406
12.4k
          decompressed_.resize(std::min<size_t>(i_ + 0x400, enc_size_)));
407
12.4k
    }
408
6.75M
    JXL_ENSURE(i_ >= 2);
409
6.75M
    decompressed_[i_] = ans_reader_.ReadHybridUint(
410
6.75M
        ICCANSContext(i_, decompressed_[i_ - 1], decompressed_[i_ - 2]), reader,
411
6.75M
        context_map_);
412
6.75M
  }
413
1.05k
  JXL_RETURN_IF_ERROR(check_and_restore());
414
868
  bits_to_skip_ = reader->TotalBitsConsumed() - used_bits_base_;
415
868
  if (!ans_reader_.CheckANSFinalState()) {
416
0
    return JXL_FAILURE("Corrupted ICC profile");
417
0
  }
418
419
868
  icc->clear();
420
868
  return UnpredictICC(decompressed_.data(), decompressed_.size(), icc);
421
868
}
422
423
18.3k
Status ICCReader::CheckEOI(BitReader* reader) {
424
18.3k
  if (reader->AllReadsWithinBounds()) return true;
425
980
  return JXL_NOT_ENOUGH_BYTES("Not enough bytes for reading ICC profile");
426
18.3k
}
427
428
}  // namespace jxl