Coverage Report

Created: 2026-06-07 07:20

next uncovered line (L), next uncovered region (R), next uncovered branch (B)
/src/libjxl/lib/jxl/dec_frame.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/dec_frame.h"
7
8
#include <jxl/decode.h>
9
#include <jxl/memory_manager.h>
10
#include <jxl/types.h>
11
12
#include <algorithm>
13
#include <cstddef>
14
#include <cstdint>
15
#include <cstdlib>
16
#include <memory>
17
#include <utility>
18
#include <vector>
19
20
#include "lib/jxl/ac_context.h"
21
#include "lib/jxl/ac_strategy.h"
22
#include "lib/jxl/base/bits.h"
23
#include "lib/jxl/base/common.h"
24
#include "lib/jxl/base/compiler_specific.h"
25
#include "lib/jxl/base/data_parallel.h"
26
#include "lib/jxl/base/printf_macros.h"
27
#include "lib/jxl/base/rect.h"
28
#include "lib/jxl/base/span.h"
29
#include "lib/jxl/base/status.h"
30
#include "lib/jxl/chroma_from_luma.h"
31
#include "lib/jxl/coeff_order.h"
32
#include "lib/jxl/coeff_order_fwd.h"
33
#include "lib/jxl/common.h"  // kMaxNumPasses
34
#include "lib/jxl/compressed_dc.h"
35
#include "lib/jxl/dct_util.h"
36
#include "lib/jxl/dec_ans.h"
37
#include "lib/jxl/dec_bit_reader.h"
38
#include "lib/jxl/dec_cache.h"
39
#include "lib/jxl/dec_group.h"
40
#include "lib/jxl/dec_modular.h"
41
#include "lib/jxl/dec_noise.h"
42
#include "lib/jxl/dec_patch_dictionary.h"
43
#include "lib/jxl/entropy_coder.h"
44
#include "lib/jxl/epf.h"
45
#include "lib/jxl/fields.h"
46
#include "lib/jxl/frame_dimensions.h"
47
#include "lib/jxl/frame_header.h"
48
#include "lib/jxl/image_bundle.h"
49
#include "lib/jxl/image_metadata.h"
50
#include "lib/jxl/image_ops.h"
51
#include "lib/jxl/jpeg/jpeg_data.h"
52
#include "lib/jxl/loop_filter.h"
53
#include "lib/jxl/passes_state.h"
54
#include "lib/jxl/quant_weights.h"
55
#include "lib/jxl/quantizer.h"
56
#include "lib/jxl/render_pipeline/render_pipeline.h"
57
#include "lib/jxl/splines.h"
58
#include "lib/jxl/toc.h"
59
60
namespace jxl {
61
62
namespace {
63
Status DecodeGlobalDCInfo(BitReader* reader, bool is_jpeg,
64
7.54k
                          PassesDecoderState* state, ThreadPool* pool) {
65
7.54k
  JXL_RETURN_IF_ERROR(state->shared_storage.quantizer.Decode(reader));
66
67
7.53k
  JXL_RETURN_IF_ERROR(DecodeBlockCtxMap(state->memory_manager(), reader,
68
7.53k
                                        &state->shared_storage.block_ctx_map));
69
70
7.48k
  JXL_RETURN_IF_ERROR(state->shared_storage.cmap.DecodeDC(reader));
71
72
  // Pre-compute info for decoding a group.
73
7.46k
  if (is_jpeg) {
74
0
    state->shared_storage.quantizer.ClearDCMul();  // Don't dequant DC
75
0
  }
76
77
7.46k
  state->shared_storage.ac_strategy.FillInvalid();
78
7.46k
  return true;
79
7.48k
}
80
}  // namespace
81
82
Status DecodeFrame(PassesDecoderState* dec_state, ThreadPool* JXL_RESTRICT pool,
83
                   const uint8_t* next_in, size_t avail_in,
84
                   FrameHeader* frame_header, ImageBundle* decoded,
85
                   const CodecMetadata& metadata,
86
846
                   bool use_slow_rendering_pipeline) {
87
846
  FrameDecoder frame_decoder(dec_state, metadata, pool,
88
846
                             use_slow_rendering_pipeline);
89
90
846
  BitReader reader(Bytes(next_in, avail_in));
91
846
  JXL_RETURN_IF_ERROR(frame_decoder.InitFrame(&reader, decoded,
92
846
                                              /*is_preview=*/false));
93
846
  JXL_RETURN_IF_ERROR(frame_decoder.InitFrameOutput());
94
846
  if (frame_header) {
95
0
    *frame_header = frame_decoder.GetFrameHeader();
96
0
  }
97
98
846
  JXL_RETURN_IF_ERROR(reader.AllReadsWithinBounds());
99
846
  size_t header_bytes = reader.TotalBitsConsumed() / kBitsPerByte;
100
846
  JXL_RETURN_IF_ERROR(reader.Close());
101
102
846
  size_t processed_bytes = header_bytes;
103
846
  Status close_ok = true;
104
846
  std::vector<std::unique_ptr<BitReader>> section_readers;
105
846
  {
106
846
    std::vector<std::unique_ptr<BitReaderScopedCloser>> section_closers;
107
846
    std::vector<FrameDecoder::SectionInfo> section_info;
108
846
    std::vector<FrameDecoder::SectionStatus> section_status;
109
846
    size_t pos = header_bytes;
110
846
    size_t index = 0;
111
846
    for (auto toc_entry : frame_decoder.Toc()) {
112
846
      JXL_RETURN_IF_ERROR(pos + toc_entry.size <= avail_in);
113
846
      auto br = make_unique<BitReader>(Bytes(next_in + pos, toc_entry.size));
114
846
      section_info.emplace_back(
115
846
          FrameDecoder::SectionInfo{br.get(), toc_entry.id, index++});
116
846
      section_closers.emplace_back(
117
846
          make_unique<BitReaderScopedCloser>(*br, close_ok));
118
846
      section_readers.emplace_back(std::move(br));
119
846
      pos += toc_entry.size;
120
846
    }
121
846
    section_status.resize(section_info.size());
122
846
    JXL_RETURN_IF_ERROR(frame_decoder.ProcessSections(
123
846
        section_info.data(), section_info.size(), section_status.data()));
124
1.69k
    for (size_t i = 0; i < section_status.size(); i++) {
125
846
      JXL_RETURN_IF_ERROR(section_status[i] == FrameDecoder::kDone);
126
846
      processed_bytes += frame_decoder.Toc()[i].size;
127
846
    }
128
846
  }
129
846
  JXL_RETURN_IF_ERROR(close_ok);
130
846
  JXL_RETURN_IF_ERROR(frame_decoder.FinalizeFrame());
131
846
  decoded->SetDecodedBytes(processed_bytes);
132
846
  return true;
133
846
}
134
135
Status FrameDecoder::InitFrame(BitReader* JXL_RESTRICT br, ImageBundle* decoded,
136
56.0k
                               bool is_preview) {
137
56.0k
  decoded_ = decoded;
138
56.0k
  JXL_ENSURE(is_finalized_);
139
56.0k
  JxlMemoryManager* memory_manager = decoded_->memory_manager();
140
141
  // Reset the dequantization matrices to their default values.
142
56.0k
  dec_state_->shared_storage.matrices = DequantMatrices();
143
144
56.0k
  frame_header_.nonserialized_is_preview = is_preview;
145
56.0k
  JXL_ENSURE(frame_header_.nonserialized_metadata != nullptr);
146
56.0k
  JXL_RETURN_IF_ERROR(ReadFrameHeader(br, &frame_header_));
147
53.0k
  frame_dim_ = frame_header_.ToFrameDimensions();
148
53.0k
  JXL_DEBUG_V(2, "FrameHeader: %s", frame_header_.DebugString().c_str());
149
150
53.0k
  const size_t num_passes = frame_header_.passes.num_passes;
151
53.0k
  const size_t num_groups = frame_dim_.num_groups;
152
153
  // If the previous frame was not a kRegularFrame, `decoded` may have different
154
  // dimensions; must reset to avoid errors.
155
53.0k
  decoded->RemoveColor();
156
53.0k
  decoded->ClearExtraChannels();
157
158
53.0k
  decoded->duration = frame_header_.animation_frame.duration;
159
160
53.0k
  if (!frame_header_.nonserialized_is_preview &&
161
52.8k
      (frame_header_.is_last || frame_header_.animation_frame.duration > 0) &&
162
10.3k
      (frame_header_.frame_type == kRegularFrame ||
163
10.3k
       frame_header_.frame_type == kSkipProgressive)) {
164
10.3k
    ++dec_state_->visible_frame_index;
165
10.3k
    dec_state_->nonvisible_frame_index = 0;
166
42.7k
  } else {
167
42.7k
    ++dec_state_->nonvisible_frame_index;
168
42.7k
  }
169
170
  // Read TOC.
171
53.0k
  const size_t toc_entries =
172
53.0k
      NumTocEntries(num_groups, frame_dim_.num_dc_groups, num_passes);
173
53.0k
  std::vector<uint32_t> sizes;
174
53.0k
  std::vector<coeff_order_t> permutation;
175
53.0k
  JXL_RETURN_IF_ERROR(
176
53.0k
      ReadToc(memory_manager, toc_entries, br, &sizes, &permutation));
177
49.0k
  bool have_permutation = !permutation.empty();
178
49.0k
  toc_.resize(toc_entries);
179
49.0k
  section_sizes_sum_ = 0;
180
139k
  for (size_t i = 0; i < toc_entries; ++i) {
181
90.2k
    toc_[i].size = sizes[i];
182
90.2k
    size_t index = have_permutation ? permutation[i] : i;
183
90.2k
    toc_[index].id = i;
184
90.2k
    if (section_sizes_sum_ + toc_[i].size < section_sizes_sum_) {
185
0
      return JXL_FAILURE("group offset overflow");
186
0
    }
187
90.2k
    section_sizes_sum_ += toc_[i].size;
188
90.2k
  }
189
190
49.0k
  if (JXL_DEBUG_V_LEVEL >= 3) {
191
0
    for (size_t i = 0; i < toc_entries; ++i) {
192
0
      JXL_DEBUG_V(3, "TOC entry %" PRIuS " size %" PRIuS " id %" PRIuS "", i,
193
0
                  toc_[i].size, toc_[i].id);
194
0
    }
195
0
  }
196
197
49.0k
  JXL_ENSURE((br->TotalBitsConsumed() % kBitsPerByte) == 0);
198
49.0k
  const size_t group_codes_begin = br->TotalBitsConsumed() / kBitsPerByte;
199
49.0k
  JXL_ENSURE(!toc_.empty());
200
201
  // Overflow check.
202
49.0k
  if (group_codes_begin + section_sizes_sum_ < group_codes_begin) {
203
0
    return JXL_FAILURE("Invalid group codes");
204
0
  }
205
206
49.0k
  if (!frame_header_.chroma_subsampling.Is444() &&
207
6.58k
      !(frame_header_.flags & FrameHeader::kSkipAdaptiveDCSmoothing) &&
208
6.22k
      frame_header_.encoding == FrameEncoding::kVarDCT) {
209
10
    return JXL_FAILURE(
210
10
        "Non-444 chroma subsampling is not allowed when adaptive DC "
211
10
        "smoothing is enabled");
212
10
  }
213
49.0k
  return true;
214
49.0k
}
215
216
41.3k
Status FrameDecoder::InitFrameOutput() {
217
41.3k
  JXL_RETURN_IF_ERROR(
218
41.3k
      InitializePassesSharedState(frame_header_, &dec_state_->shared_storage));
219
41.3k
  JXL_RETURN_IF_ERROR(dec_state_->Init(frame_header_));
220
41.3k
  modular_frame_decoder_.Init(frame_dim_);
221
222
41.3k
  if (decoded_->IsJPEG()) {
223
0
    if (frame_header_.encoding == FrameEncoding::kModular) {
224
0
      return JXL_FAILURE("Cannot output JPEG from Modular");
225
0
    }
226
0
    jpeg::JPEGData* jpeg_data = decoded_->jpeg_data.get();
227
0
    size_t num_components = jpeg_data->components.size();
228
0
    if (num_components != 1 && num_components != 3) {
229
0
      return JXL_FAILURE("Invalid number of components");
230
0
    }
231
0
    if (frame_header_.nonserialized_metadata->m.xyb_encoded) {
232
0
      return JXL_FAILURE("Cannot decode to JPEG an XYB image");
233
0
    }
234
0
    auto jpeg_c_map = JpegOrder(ColorTransform::kYCbCr, num_components == 1);
235
0
    decoded_->jpeg_data->width = frame_dim_.xsize;
236
0
    decoded_->jpeg_data->height = frame_dim_.ysize;
237
0
    for (size_t c = 0; c < num_components; c++) {
238
0
      auto& component = jpeg_data->components[jpeg_c_map[c]];
239
0
      component.width_in_blocks =
240
0
          frame_dim_.xsize_blocks >> frame_header_.chroma_subsampling.HShift(c);
241
0
      component.height_in_blocks =
242
0
          frame_dim_.ysize_blocks >> frame_header_.chroma_subsampling.VShift(c);
243
0
      component.h_samp_factor =
244
0
          1 << frame_header_.chroma_subsampling.RawHShift(c);
245
0
      component.v_samp_factor =
246
0
          1 << frame_header_.chroma_subsampling.RawVShift(c);
247
0
      size_t num_blocks = static_cast<size_t>(component.width_in_blocks) *
248
0
                          component.height_in_blocks;
249
0
      component.coeffs.resize(num_blocks * jxl::kDCTBlockSize);
250
0
    }
251
0
  }
252
253
  // Clear the state.
254
41.3k
  decoded_dc_global_ = false;
255
41.3k
  decoded_ac_global_ = false;
256
41.3k
  is_finalized_ = false;
257
41.3k
  finalized_dc_ = false;
258
41.3k
  num_sections_done_ = 0;
259
41.3k
  decoded_dc_groups_.clear();
260
41.3k
  decoded_dc_groups_.resize(frame_dim_.num_dc_groups);
261
41.3k
  decoded_passes_per_ac_group_.clear();
262
41.3k
  decoded_passes_per_ac_group_.resize(frame_dim_.num_groups, 0);
263
41.3k
  processed_section_.clear();
264
41.3k
  processed_section_.resize(toc_.size());
265
41.3k
  allocated_ = false;
266
41.3k
  return true;
267
41.3k
}
268
269
40.8k
Status FrameDecoder::ProcessDCGlobal(BitReader* br) {
270
40.8k
  PassesSharedState& shared = dec_state_->shared_storage;
271
40.8k
  JxlMemoryManager* memory_manager = shared.memory_manager;
272
40.8k
  if (frame_header_.flags & FrameHeader::kPatches) {
273
1.44k
    bool uses_extra_channels = false;
274
1.44k
    JXL_RETURN_IF_ERROR(shared.image_features.patches.Decode(
275
1.44k
        memory_manager, br, frame_dim_.xsize_padded, frame_dim_.ysize_padded,
276
1.44k
        shared.metadata->m.num_extra_channels, &uses_extra_channels));
277
1.37k
    if (uses_extra_channels && frame_header_.upsampling != 1) {
278
39
      for (size_t ecups : frame_header_.extra_channel_upsampling) {
279
39
        if (ecups != frame_header_.upsampling) {
280
1
          return JXL_FAILURE(
281
1
              "Cannot use extra channels in patches if color channels are "
282
1
              "subsampled differently from extra channels");
283
1
        }
284
39
      }
285
21
    }
286
39.3k
  } else {
287
39.3k
    shared.image_features.patches.Clear();
288
39.3k
  }
289
40.7k
  shared.image_features.splines.Clear();
290
40.7k
  if (frame_header_.flags & FrameHeader::kSplines) {
291
4.54k
    JXL_RETURN_IF_ERROR(shared.image_features.splines.Decode(
292
4.54k
        memory_manager, br, frame_dim_.xsize * frame_dim_.ysize));
293
4.54k
  }
294
40.6k
  if (frame_header_.flags & FrameHeader::kNoise) {
295
5.15k
    JXL_RETURN_IF_ERROR(DecodeNoise(br, &shared.image_features.noise_params));
296
5.15k
  }
297
40.6k
  JXL_RETURN_IF_ERROR(dec_state_->shared_storage.matrices.DecodeDC(br));
298
299
40.2k
  if (frame_header_.encoding == FrameEncoding::kVarDCT) {
300
7.54k
    JXL_RETURN_IF_ERROR(
301
7.54k
        jxl::DecodeGlobalDCInfo(br, decoded_->IsJPEG(), dec_state_, pool_));
302
7.54k
  }
303
  // Splines' draw cache uses the color correlation map.
304
40.1k
  if (frame_header_.flags & FrameHeader::kSplines) {
305
4.35k
    JXL_RETURN_IF_ERROR(shared.image_features.splines.InitializeDrawCache(
306
4.35k
        frame_dim_.xsize_upsampled, frame_dim_.ysize_upsampled,
307
4.35k
        dec_state_->shared->cmap.base()));
308
4.35k
  }
309
40.0k
  Status dec_status = modular_frame_decoder_.DecodeGlobalInfo(
310
40.0k
      br, frame_header_, /*allow_truncated_group=*/false);
311
40.0k
  if (dec_status.IsFatalError()) return dec_status;
312
38.0k
  if (dec_status) {
313
37.6k
    decoded_dc_global_ = true;
314
37.6k
  }
315
38.0k
  return dec_status;
316
40.0k
}
317
318
37.8k
Status FrameDecoder::ProcessDCGroup(size_t dc_group_id, BitReader* br) {
319
37.8k
  const size_t gx = dc_group_id % frame_dim_.xsize_dc_groups;
320
37.8k
  const size_t gy = dc_group_id / frame_dim_.xsize_dc_groups;
321
37.8k
  const LoopFilter& lf = frame_header_.loop_filter;
322
37.8k
  if (frame_header_.encoding == FrameEncoding::kVarDCT &&
323
7.37k
      !(frame_header_.flags & FrameHeader::kUseDcFrame)) {
324
7.37k
    JXL_RETURN_IF_ERROR(modular_frame_decoder_.DecodeVarDCTDC(
325
7.37k
        frame_header_, dc_group_id, br, dec_state_));
326
7.37k
  }
327
36.6k
  const Rect mrect(gx * frame_dim_.dc_group_dim, gy * frame_dim_.dc_group_dim,
328
36.6k
                   frame_dim_.dc_group_dim, frame_dim_.dc_group_dim);
329
36.6k
  JXL_RETURN_IF_ERROR(modular_frame_decoder_.DecodeGroup(
330
36.6k
      frame_header_, mrect, br, 3, 1000,
331
36.6k
      ModularStreamId::ModularDC(dc_group_id),
332
36.6k
      /*zerofill=*/false, nullptr, nullptr,
333
36.6k
      /*allow_truncated=*/false));
334
36.6k
  if (frame_header_.encoding == FrameEncoding::kVarDCT) {
335
6.14k
    JXL_RETURN_IF_ERROR(modular_frame_decoder_.DecodeAcMetadata(
336
6.14k
        frame_header_, dc_group_id, br, dec_state_));
337
30.4k
  } else if (lf.epf_iters > 0) {
338
11.0k
    FillImage(kInvSigmaNum / lf.epf_sigma_for_modular, &dec_state_->sigma);
339
11.0k
  }
340
35.6k
  decoded_dc_groups_[dc_group_id] = JXL_TRUE;
341
35.6k
  return true;
342
36.6k
}
343
344
35.4k
Status FrameDecoder::FinalizeDC() {
345
  // Do Adaptive DC smoothing if enabled. This *must* happen between all the
346
  // ProcessDCGroup and ProcessACGroup.
347
35.4k
  JxlMemoryManager* memory_manager = dec_state_->memory_manager();
348
35.4k
  if (frame_header_.encoding == FrameEncoding::kVarDCT &&
349
5.15k
      !(frame_header_.flags & FrameHeader::kSkipAdaptiveDCSmoothing) &&
350
4.63k
      !(frame_header_.flags & FrameHeader::kUseDcFrame)) {
351
4.62k
    JXL_RETURN_IF_ERROR(AdaptiveDCSmoothing(
352
4.62k
        memory_manager, dec_state_->shared->quantizer.MulDC(),
353
4.62k
        &dec_state_->shared_storage.dc_storage, pool_));
354
4.62k
  }
355
356
35.4k
  finalized_dc_ = true;
357
35.4k
  return true;
358
35.4k
}
359
360
35.4k
Status FrameDecoder::AllocateOutput() {
361
35.4k
  if (allocated_) return true;
362
35.4k
  modular_frame_decoder_.MaybeDropFullImage();
363
35.4k
  decoded_->origin = frame_header_.frame_origin;
364
35.4k
  JXL_RETURN_IF_ERROR(
365
35.4k
      dec_state_->InitForAC(frame_header_.passes.num_passes, nullptr));
366
35.4k
  allocated_ = true;
367
35.4k
  return true;
368
35.4k
}
369
370
35.3k
Status FrameDecoder::ProcessACGlobal(BitReader* br) {
371
35.3k
  JXL_ENSURE(finalized_dc_);
372
35.3k
  JxlMemoryManager* memory_manager = dec_state_->memory_manager();
373
374
  // Decode AC group.
375
35.3k
  if (frame_header_.encoding == FrameEncoding::kVarDCT) {
376
5.14k
    JXL_RETURN_IF_ERROR(dec_state_->shared_storage.matrices.Decode(
377
5.14k
        memory_manager, br, &modular_frame_decoder_));
378
4.76k
    JXL_RETURN_IF_ERROR(dec_state_->shared_storage.matrices.EnsureComputed(
379
4.76k
        memory_manager, dec_state_->used_acs));
380
381
4.71k
    size_t num_histo_bits =
382
4.71k
        CeilLog2Nonzero(dec_state_->shared->frame_dim.num_groups);
383
4.71k
    dec_state_->shared_storage.num_histograms =
384
4.71k
        1 + br->ReadBits(num_histo_bits);
385
386
4.71k
    JXL_DEBUG_V(3,
387
4.71k
                "Processing AC global with %d passes and %" PRIuS
388
4.71k
                " sets of histograms",
389
4.71k
                frame_header_.passes.num_passes,
390
4.71k
                dec_state_->shared_storage.num_histograms);
391
392
4.71k
    dec_state_->code.resize(kMaxNumPasses);
393
4.71k
    dec_state_->context_map.resize(kMaxNumPasses);
394
    // Read coefficient orders and histograms.
395
4.71k
    size_t max_num_bits_ac = 0;
396
9.47k
    for (size_t i = 0; i < frame_header_.passes.num_passes; i++) {
397
4.89k
      uint16_t used_orders = U32Coder::Read(kOrderEnc, br);
398
4.89k
      JXL_RETURN_IF_ERROR(DecodeCoeffOrders(
399
4.89k
          memory_manager, used_orders, dec_state_->used_acs,
400
4.89k
          &dec_state_->shared_storage
401
4.89k
               .coeff_orders[i * dec_state_->shared_storage.coeff_order_size],
402
4.89k
          br));
403
4.82k
      size_t num_contexts =
404
4.82k
          dec_state_->shared->num_histograms *
405
4.82k
          dec_state_->shared_storage.block_ctx_map.NumACContexts();
406
4.82k
      JXL_RETURN_IF_ERROR(DecodeHistograms(memory_manager, br, num_contexts,
407
4.82k
                                           &dec_state_->code[i],
408
4.82k
                                           &dec_state_->context_map[i]));
409
      // Add extra values to enable the cheat in hot loop of DecodeACVarBlock.
410
4.75k
      dec_state_->context_map[i].resize(
411
4.75k
          num_contexts + kZeroDensityContextLimit - kZeroDensityContextCount);
412
4.75k
      max_num_bits_ac =
413
4.75k
          std::max(max_num_bits_ac, dec_state_->code[i].max_num_bits);
414
4.75k
    }
415
4.58k
    max_num_bits_ac += CeilLog2Nonzero(frame_header_.passes.num_passes);
416
    // 16-bit buffer for decoding to JPEG are not implemented.
417
    // TODO(veluca): figure out the exact limit - 16 should still work with
418
    // 16-bit buffers, but we are excluding it for safety.
419
4.58k
    bool use_16_bit = max_num_bits_ac < 16 && !decoded_->IsJPEG();
420
4.58k
    bool store = frame_header_.passes.num_passes > 1;
421
4.58k
    size_t xs = store ? kGroupDim * kGroupDim : 0;
422
4.58k
    size_t ys = store ? frame_dim_.num_groups : 0;
423
4.58k
    if (use_16_bit) {
424
3.88k
      JXL_ASSIGN_OR_RETURN(dec_state_->coefficients,
425
3.88k
                           ACImageT<int16_t>::Make(memory_manager, xs, ys));
426
3.88k
    } else {
427
697
      JXL_ASSIGN_OR_RETURN(dec_state_->coefficients,
428
697
                           ACImageT<int32_t>::Make(memory_manager, xs, ys));
429
697
    }
430
4.58k
    if (store) {
431
36
      dec_state_->coefficients->ZeroFill();
432
36
    }
433
4.58k
  }
434
435
  // Set JPEG decoding data.
436
34.8k
  if (decoded_->IsJPEG()) {
437
0
    decoded_->color_transform = frame_header_.color_transform;
438
0
    decoded_->chroma_subsampling = frame_header_.chroma_subsampling;
439
0
    const std::vector<QuantEncoding>& qe =
440
0
        dec_state_->shared_storage.matrices.encodings();
441
0
    if (qe.empty() || qe[0].mode != QuantEncoding::Mode::kQuantModeRAW ||
442
0
        std::abs(qe[0].qraw.qtable_den - 1.f / (8 * 255)) > 1e-8f) {
443
0
      return JXL_FAILURE(
444
0
          "Quantization table is not a JPEG quantization table.");
445
0
    }
446
0
    jpeg::JPEGData* jpeg_data = decoded_->jpeg_data.get();
447
0
    size_t num_components = jpeg_data->components.size();
448
0
    bool is_gray = (num_components == 1);
449
0
    JXL_ENSURE(frame_header_.color_transform != ColorTransform::kXYB);
450
0
    auto jpeg_c_map = JpegOrder(frame_header_.color_transform, is_gray);
451
0
    size_t qt_set = 0;
452
0
    JXL_ENSURE(num_components <= 3);
453
0
    JXL_ENSURE(qe[0].qraw.qtable->size() == 3 * 8 * 8);
454
0
    int* qtable = qe[0].qraw.qtable->data();
455
0
    for (size_t c = 0; c < num_components; c++) {
456
      // TODO(eustas): why 1-st quant table for gray?
457
0
      size_t quant_c = is_gray ? 1 : c;
458
0
      size_t qpos = jpeg_data->components[jpeg_c_map[c]].quant_idx;
459
0
      JXL_ENSURE(qpos != jpeg_data->quant.size());
460
0
      qt_set |= 1 << qpos;
461
0
      for (size_t x = 0; x < 8; x++) {
462
0
        for (size_t y = 0; y < 8; y++) {
463
0
          jpeg_data->quant[qpos].values[x * 8 + y] =
464
0
              qtable[quant_c * 64 + y * 8 + x];
465
0
        }
466
0
      }
467
0
    }
468
0
    for (size_t i = 0; i < jpeg_data->quant.size(); i++) {
469
0
      if (qt_set & (1 << i)) continue;
470
0
      if (i == 0) return JXL_FAILURE("First quant table unused.");
471
      // Unused quant table is set to copy of previous quant table
472
0
      for (size_t j = 0; j < 64; j++) {
473
0
        jpeg_data->quant[i].values[j] = jpeg_data->quant[i - 1].values[j];
474
0
      }
475
0
    }
476
0
  }
477
34.8k
  decoded_ac_global_ = true;
478
34.8k
  return true;
479
34.8k
}
480
481
Status FrameDecoder::ProcessACGroup(size_t ac_group_id, PassesReaders& br,
482
                                    size_t num_passes, size_t thread,
483
37.4k
                                    bool force_draw, bool dc_only) {
484
37.4k
  size_t group_dim = frame_dim_.group_dim;
485
37.4k
  const size_t gx = ac_group_id % frame_dim_.xsize_groups;
486
37.4k
  const size_t gy = ac_group_id / frame_dim_.xsize_groups;
487
37.4k
  const size_t x = gx * group_dim;
488
37.4k
  const size_t y = gy * group_dim;
489
37.4k
  JxlMemoryManager* memory_manager = dec_state_->memory_manager();
490
37.4k
  JXL_DEBUG_V(3,
491
37.4k
              "Processing AC group %" PRIuS "(%" PRIuS ",%" PRIuS
492
37.4k
              ") group_dim: %" PRIuS " decoded passes: %u new passes: %" PRIuS,
493
37.4k
              ac_group_id, gx, gy, group_dim,
494
37.4k
              decoded_passes_per_ac_group_[ac_group_id], num_passes);
495
496
37.4k
  RenderPipelineInput render_pipeline_input =
497
37.4k
      dec_state_->render_pipeline->GetInputBuffers(ac_group_id, thread);
498
499
37.4k
  bool should_run_pipeline = true;
500
501
37.4k
  if (frame_header_.encoding == FrameEncoding::kVarDCT) {
502
5.17k
    JXL_RETURN_IF_ERROR(group_dec_caches_[thread].InitOnce(
503
5.17k
        memory_manager, frame_header_.passes.num_passes, dec_state_->used_acs));
504
5.17k
    JXL_RETURN_IF_ERROR(DecodeGroup(
505
5.17k
        frame_header_, br.data(), num_passes, ac_group_id, dec_state_,
506
5.17k
        &group_dec_caches_[thread], thread, render_pipeline_input,
507
5.17k
        decoded_->jpeg_data.get(), decoded_passes_per_ac_group_[ac_group_id],
508
5.17k
        force_draw, dc_only, &should_run_pipeline));
509
5.17k
  }
510
511
  // don't limit to image dimensions here (is done in DecodeGroup)
512
37.2k
  const Rect mrect(x, y, group_dim, group_dim);
513
37.2k
  bool modular_ready = false;
514
37.2k
  size_t pass0 = decoded_passes_per_ac_group_[ac_group_id];
515
37.2k
  size_t pass1 =
516
37.2k
      force_draw ? frame_header_.passes.num_passes : pass0 + num_passes;
517
76.8k
  for (size_t i = pass0; i < pass1; ++i) {
518
39.6k
    int minShift;
519
39.6k
    int maxShift;
520
39.6k
    frame_header_.passes.GetDownsamplingBracket(i, minShift, maxShift);
521
39.6k
    bool modular_pass_ready = true;
522
39.6k
    JXL_DEBUG_V(2, "Decoding modular in group %d pass %d",
523
39.6k
                static_cast<int>(ac_group_id), static_cast<int>(i));
524
39.6k
    if (i < pass0 + num_passes) {  // i.e. i - pass0 < num_passes
525
39.6k
      BitReader* r = br[i - pass0];
526
39.6k
      JXL_ENSURE(r);
527
39.6k
      JXL_DEBUG_V(2, "Bit reader position: %" PRIuS " / %" PRIuS,
528
39.6k
                  r->TotalBitsConsumed(), r->TotalBytes() * kBitsPerByte);
529
39.6k
      JXL_RETURN_IF_ERROR(modular_frame_decoder_.DecodeGroup(
530
39.6k
          frame_header_, mrect, r, minShift, maxShift,
531
39.6k
          ModularStreamId::ModularAC(ac_group_id, i),
532
39.6k
          /*zerofill=*/false, dec_state_, &render_pipeline_input,
533
39.6k
          /*allow_truncated=*/false, &modular_pass_ready));
534
39.6k
    } else {
535
0
      JXL_RETURN_IF_ERROR(modular_frame_decoder_.DecodeGroup(
536
0
          frame_header_, mrect, nullptr, minShift, maxShift,
537
0
          ModularStreamId::ModularAC(ac_group_id, i), /*zerofill=*/true,
538
0
          dec_state_, &render_pipeline_input,
539
0
          /*allow_truncated=*/false, &modular_pass_ready));
540
0
    }
541
39.5k
    if (modular_pass_ready) modular_ready = true;
542
39.5k
  }
543
37.1k
  decoded_passes_per_ac_group_[ac_group_id] += num_passes;
544
545
37.1k
  const bool render_noise =
546
37.1k
      ((frame_header_.flags & FrameHeader::kNoise) != 0) &&
547
6.93k
      (frame_header_.dc_level == 0);
548
37.1k
  if (render_noise) {
549
5.21k
    PrepareNoiseInput(*dec_state_, frame_dim_, frame_header_, ac_group_id,
550
5.21k
                      thread);
551
5.21k
  }
552
553
37.1k
  if (!modular_frame_decoder_.UsesFullImage() && !decoded_->IsJPEG()) {
554
5.96k
    if (should_run_pipeline && modular_ready) {
555
5.54k
      JXL_RETURN_IF_ERROR(render_pipeline_input.Done());
556
5.54k
    } else if (force_draw) {
557
0
      return JXL_FAILURE("Modular group decoding failed.");
558
0
    }
559
5.96k
  }
560
37.1k
  return true;
561
37.1k
}
562
563
void FrameDecoder::MarkSections(const SectionInfo* sections, size_t num,
564
35.0k
                                const SectionStatus* section_status) {
565
35.0k
  num_sections_done_ += num;
566
83.1k
  for (size_t i = 0; i < num; i++) {
567
48.0k
    if (section_status[i] != SectionStatus::kDone) {
568
1.85k
      processed_section_[sections[i].id] = JXL_FALSE;
569
1.85k
      num_sections_done_--;
570
1.85k
    }
571
48.0k
  }
572
35.0k
}
573
574
Status FrameDecoder::ProcessSections(const SectionInfo* sections, size_t num,
575
41.9k
                                     SectionStatus* section_status) {
576
41.9k
  if (num == 0) return true;  // Nothing to process
577
40.8k
  std::fill(section_status, section_status + num, SectionStatus::kSkipped);
578
40.8k
  size_t dc_global_sec = num;
579
40.8k
  size_t ac_global_sec = num;
580
40.8k
  std::vector<size_t> dc_group_sec(frame_dim_.num_dc_groups, num);
581
40.8k
  std::vector<std::vector<size_t>> ac_group_sec(
582
40.8k
      frame_dim_.num_groups,
583
40.8k
      std::vector<size_t>(frame_header_.passes.num_passes, num));
584
  // This keeps track of the number of ac passes we want to process during this
585
  // call of ProcessSections.
586
40.8k
  std::vector<size_t> desired_num_ac_passes(frame_dim_.num_groups);
587
40.8k
  bool single_section =
588
40.8k
      frame_dim_.num_groups == 1 && frame_header_.passes.num_passes == 1;
589
40.8k
  if (single_section) {
590
38.2k
    JXL_ENSURE(num == 1);
591
38.2k
    JXL_ENSURE(sections[0].id == 0);
592
38.2k
    if (processed_section_[0] == JXL_FALSE) {
593
38.2k
      processed_section_[0] = JXL_TRUE;
594
38.2k
      ac_group_sec[0].resize(1);
595
38.2k
      dc_global_sec = ac_global_sec = dc_group_sec[0] = ac_group_sec[0][0] = 0;
596
38.2k
      desired_num_ac_passes[0] = 1;
597
38.2k
    } else {
598
0
      section_status[0] = SectionStatus::kDuplicate;
599
0
    }
600
38.2k
  } else {
601
2.64k
    size_t ac_global_index = frame_dim_.num_dc_groups + 1;
602
19.9k
    for (size_t i = 0; i < num; i++) {
603
17.3k
      JXL_ENSURE(sections[i].id < processed_section_.size());
604
17.3k
      if (processed_section_[sections[i].id]) {
605
0
        section_status[i] = SectionStatus::kDuplicate;
606
0
        continue;
607
0
      }
608
17.3k
      if (sections[i].id == 0) {
609
2.60k
        dc_global_sec = i;
610
14.7k
      } else if (sections[i].id < ac_global_index) {
611
2.86k
        dc_group_sec[sections[i].id - 1] = i;
612
11.8k
      } else if (sections[i].id == ac_global_index) {
613
2.44k
        ac_global_sec = i;
614
9.41k
      } else {
615
9.41k
        size_t ac_idx = sections[i].id - ac_global_index - 1;
616
9.41k
        size_t acg = ac_idx % frame_dim_.num_groups;
617
9.41k
        size_t acp = ac_idx / frame_dim_.num_groups;
618
9.41k
        if (acp >= frame_header_.passes.num_passes) {
619
0
          return JXL_FAILURE("Invalid section ID");
620
0
        }
621
9.41k
        ac_group_sec[acg][acp] = i;
622
9.41k
      }
623
17.3k
      processed_section_[sections[i].id] = JXL_TRUE;
624
17.3k
    }
625
    // Count number of new passes per group.
626
18.3k
    for (size_t g = 0; g < ac_group_sec.size(); g++) {
627
15.6k
      size_t j = 0;
628
24.8k
      for (; j + decoded_passes_per_ac_group_[g] <
629
24.8k
             frame_header_.passes.num_passes;
630
20.0k
           j++) {
631
20.0k
        if (ac_group_sec[g][j + decoded_passes_per_ac_group_[g]] == num) {
632
10.8k
          break;
633
10.8k
        }
634
20.0k
      }
635
15.6k
      desired_num_ac_passes[g] = j;
636
15.6k
    }
637
2.64k
  }
638
40.8k
  if (dc_global_sec != num) {
639
40.8k
    Status dc_global_status = ProcessDCGlobal(sections[dc_global_sec].br);
640
40.8k
    if (dc_global_status.IsFatalError()) return dc_global_status;
641
38.1k
    if (dc_global_status) {
642
37.6k
      section_status[dc_global_sec] = SectionStatus::kDone;
643
37.6k
    } else {
644
477
      section_status[dc_global_sec] = SectionStatus::kPartial;
645
477
    }
646
38.1k
  }
647
648
38.2k
  if (decoded_dc_global_) {
649
37.6k
    const auto process_section = [this, &dc_group_sec, &num, &sections,
650
37.6k
                                  &section_status](size_t i,
651
38.1k
                                                   size_t thread) -> Status {
652
38.1k
      if (dc_group_sec[i] != num) {
653
37.8k
        JXL_RETURN_IF_ERROR(ProcessDCGroup(i, sections[dc_group_sec[i]].br));
654
35.6k
        section_status[dc_group_sec[i]] = SectionStatus::kDone;
655
35.6k
      }
656
35.9k
      return true;
657
38.1k
    };
658
37.6k
    JXL_RETURN_IF_ERROR(RunOnPool(pool_, 0, dc_group_sec.size(),
659
37.6k
                                  ThreadPool::NoInit, process_section,
660
37.6k
                                  "DecodeDCGroup"));
661
37.6k
  }
662
663
35.9k
  const bool render_noise =
664
35.9k
      ((frame_header_.flags & FrameHeader::kNoise) != 0) &&
665
4.76k
      (frame_header_.dc_level == 0);
666
35.9k
  if (!HasDcGroupToDecode() && !finalized_dc_) {
667
35.4k
    PassesDecoderState::PipelineOptions pipeline_options;
668
35.4k
    pipeline_options.use_slow_render_pipeline = use_slow_rendering_pipeline_;
669
35.4k
    pipeline_options.coalescing = coalescing_;
670
35.4k
    pipeline_options.render_spotcolors = render_spotcolors_;
671
35.4k
    pipeline_options.render_noise = render_noise;
672
35.4k
    JXL_RETURN_IF_ERROR(dec_state_->PreparePipeline(
673
35.4k
        frame_header_, &frame_header_.nonserialized_metadata->m, decoded_,
674
35.4k
        pipeline_options));
675
35.4k
    JXL_RETURN_IF_ERROR(FinalizeDC());
676
35.4k
    JXL_RETURN_IF_ERROR(AllocateOutput());
677
35.4k
    if (progressive_detail_ >= JxlProgressiveDetail::kDC) {
678
0
      MarkSections(sections, num, section_status);
679
0
      return true;
680
0
    }
681
35.4k
  }
682
683
35.9k
  if (finalized_dc_ && ac_global_sec != num && !decoded_ac_global_) {
684
35.3k
    JXL_RETURN_IF_ERROR(ProcessACGlobal(sections[ac_global_sec].br));
685
34.8k
    section_status[ac_global_sec] = SectionStatus::kDone;
686
34.8k
  }
687
688
35.3k
  if (progressive_detail_ >= JxlProgressiveDetail::kLastPasses) {
689
    // Mark that we only want the next progression pass.
690
0
    size_t target_complete_passes = NextNumPassesToPause();
691
0
    for (size_t i = 0; i < ac_group_sec.size(); i++) {
692
0
      desired_num_ac_passes[i] =
693
0
          std::min(desired_num_ac_passes[i],
694
0
                   target_complete_passes - decoded_passes_per_ac_group_[i]);
695
0
    }
696
0
  }
697
698
35.3k
  if (decoded_ac_global_) {
699
    // Mark all the AC groups that we received as not complete yet.
700
75.0k
    for (size_t i = 0; i < ac_group_sec.size(); i++) {
701
40.2k
      if (desired_num_ac_passes[i] != 0) {
702
37.6k
        dec_state_->render_pipeline->ClearDone(i);
703
37.6k
      }
704
40.2k
    }
705
706
34.8k
    const auto prepare_storage = [this](size_t num_threads) -> Status {
707
34.8k
      JXL_RETURN_IF_ERROR(
708
34.8k
          PrepareStorage(num_threads, decoded_passes_per_ac_group_.size()));
709
34.8k
      return true;
710
34.8k
    };
711
34.8k
    const auto process_group = [this, &ac_group_sec, &desired_num_ac_passes,
712
34.8k
                                &num, &sections, &section_status](
713
39.2k
                                   size_t g, size_t thread) -> Status {
714
39.2k
      if (desired_num_ac_passes[g] == 0) {
715
        // no new AC pass, nothing to do
716
1.81k
        return true;
717
1.81k
      }
718
37.4k
      size_t first_pass = decoded_passes_per_ac_group_[g];
719
37.4k
      PassesReaders readers = {};
720
77.4k
      for (size_t i = 0; i < desired_num_ac_passes[g]; i++) {
721
39.9k
        JXL_ENSURE(ac_group_sec[g][first_pass + i] != num);
722
39.9k
        readers[i] = sections[ac_group_sec[g][first_pass + i]].br;
723
39.9k
      }
724
37.4k
      JXL_RETURN_IF_ERROR(ProcessACGroup(
725
37.4k
          g, readers, desired_num_ac_passes[g], GetStorageLocation(thread, g),
726
37.4k
          /*force_draw=*/false, /*dc_only=*/false));
727
76.6k
      for (size_t i = 0; i < desired_num_ac_passes[g]; i++) {
728
39.5k
        section_status[ac_group_sec[g][first_pass + i]] = SectionStatus::kDone;
729
39.5k
      }
730
37.1k
      return true;
731
37.4k
    };
732
34.8k
    JXL_RETURN_IF_ERROR(RunOnPool(pool_, 0, ac_group_sec.size(),
733
34.8k
                                  prepare_storage, process_group,
734
34.8k
                                  "DecodeGroup"));
735
34.8k
  }
736
737
35.0k
  MarkSections(sections, num, section_status);
738
35.0k
  return true;
739
35.3k
}
740
741
0
Status FrameDecoder::Flush() {
742
0
  bool has_blending = frame_header_.blending_info.mode != BlendMode::kReplace ||
743
0
                      frame_header_.custom_size_or_origin;
744
0
  for (const auto& blending_info_ec :
745
0
       frame_header_.extra_channel_blending_info) {
746
0
    if (blending_info_ec.mode != BlendMode::kReplace) has_blending = true;
747
0
  }
748
  // No early Flush() if blending is enabled.
749
0
  if (has_blending && !is_finalized_) {
750
0
    return false;
751
0
  }
752
  // No early Flush() - nothing to do - if the frame is a kSkipProgressive
753
  // frame.
754
0
  if (frame_header_.frame_type == FrameType::kSkipProgressive &&
755
0
      !is_finalized_) {
756
0
    return true;
757
0
  }
758
0
  if (decoded_->IsJPEG()) {
759
    // Nothing to do.
760
0
    return true;
761
0
  }
762
0
  JXL_RETURN_IF_ERROR(AllocateOutput());
763
764
0
  uint32_t completely_decoded_ac_pass = *std::min_element(
765
0
      decoded_passes_per_ac_group_.begin(), decoded_passes_per_ac_group_.end());
766
0
  if (completely_decoded_ac_pass < frame_header_.passes.num_passes) {
767
    // We don't have all AC yet: force a draw of all the missing areas.
768
    // Mark all sections as not complete.
769
0
    for (size_t i = 0; i < decoded_passes_per_ac_group_.size(); i++) {
770
0
      if (decoded_passes_per_ac_group_[i] < frame_header_.passes.num_passes) {
771
0
        dec_state_->render_pipeline->ClearDone(i);
772
0
      }
773
0
    }
774
0
    const auto prepare_storage = [this](const size_t num_threads) -> Status {
775
0
      JXL_RETURN_IF_ERROR(
776
0
          PrepareStorage(num_threads, decoded_passes_per_ac_group_.size()));
777
0
      return true;
778
0
    };
779
0
    const auto process_group = [this](const uint32_t g,
780
0
                                      size_t thread) -> Status {
781
0
      if (decoded_passes_per_ac_group_[g] == frame_header_.passes.num_passes) {
782
        // This group was drawn already, nothing to do.
783
0
        return true;
784
0
      }
785
0
      PassesReaders readers = {};
786
0
      JXL_RETURN_IF_ERROR(ProcessACGroup(
787
0
          g, readers, /*num_passes=*/0, GetStorageLocation(thread, g),
788
0
          /*force_draw=*/true, /*dc_only=*/!decoded_ac_global_));
789
0
      return true;
790
0
    };
791
0
    JXL_RETURN_IF_ERROR(RunOnPool(pool_, 0, decoded_passes_per_ac_group_.size(),
792
0
                                  prepare_storage, process_group,
793
0
                                  "ForceDrawGroup"));
794
0
  }
795
796
  // undo global modular transforms and copy int pixel buffers to float ones
797
0
  JXL_RETURN_IF_ERROR(modular_frame_decoder_.FinalizeDecoding(
798
0
      frame_header_, dec_state_, pool_, is_finalized_));
799
800
0
  return true;
801
0
}
802
803
48.1k
int FrameDecoder::SavedAs(const FrameHeader& header) {
804
48.1k
  if (header.frame_type == FrameType::kDCFrame) {
805
    // bits 16, 32, 64, 128 for DC level
806
12.3k
    return 16 << (header.dc_level - 1);
807
35.7k
  } else if (header.CanBeReferenced()) {
808
    // bits 1, 2, 4 and 8 for the references
809
28.9k
    return 1 << header.save_as_reference;
810
28.9k
  }
811
812
6.82k
  return 0;
813
48.1k
}
814
815
33.2k
bool FrameDecoder::HasEverything() const {
816
33.2k
  if (!decoded_dc_global_) return false;
817
33.2k
  if (!decoded_ac_global_) return false;
818
33.2k
  if (HasDcGroupToDecode()) return false;
819
34.9k
  for (const auto& nb_passes : decoded_passes_per_ac_group_) {
820
34.9k
    if (nb_passes < frame_header_.passes.num_passes) return false;
821
34.9k
  }
822
33.2k
  return true;
823
33.2k
}
824
825
33.2k
int FrameDecoder::References() const {
826
33.2k
  if (is_finalized_) {
827
0
    return 0;
828
0
  }
829
33.2k
  if (!HasEverything()) return 0;
830
831
33.2k
  int result = 0;
832
833
  // Blending
834
33.2k
  if (frame_header_.frame_type == FrameType::kRegularFrame ||
835
20.0k
      frame_header_.frame_type == FrameType::kSkipProgressive) {
836
20.0k
    bool cropped = frame_header_.custom_size_or_origin;
837
20.0k
    if (cropped || frame_header_.blending_info.mode != BlendMode::kReplace) {
838
9.63k
      result |= (1 << frame_header_.blending_info.source);
839
9.63k
    }
840
20.0k
    const auto& extra = frame_header_.extra_channel_blending_info;
841
20.0k
    for (const auto& ecbi : extra) {
842
8.70k
      if (cropped || ecbi.mode != BlendMode::kReplace) {
843
4.31k
        result |= (1 << ecbi.source);
844
4.31k
      }
845
8.70k
    }
846
20.0k
  }
847
848
  // Patches
849
33.2k
  if (frame_header_.flags & FrameHeader::kPatches) {
850
1.05k
    result |= dec_state_->shared->image_features.patches.GetReferences();
851
1.05k
  }
852
853
  // DC Level
854
33.2k
  if (frame_header_.flags & FrameHeader::kUseDcFrame) {
855
    // Reads from the next dc level
856
37
    int dc_level = frame_header_.dc_level + 1;
857
    // bits 16, 32, 64, 128 for DC level
858
37
    result |= (16 << (dc_level - 1));
859
37
  }
860
861
33.2k
  return result;
862
33.2k
}
863
864
34.0k
Status FrameDecoder::FinalizeFrame() {
865
34.0k
  if (is_finalized_) {
866
0
    return JXL_FAILURE("FinalizeFrame called multiple times");
867
0
  }
868
34.0k
  is_finalized_ = true;
869
34.0k
  if (decoded_->IsJPEG()) {
870
    // Nothing to do.
871
0
    return true;
872
0
  }
873
874
  // undo global modular transforms and copy int pixel buffers to float ones
875
34.0k
  JXL_RETURN_IF_ERROR(
876
34.0k
      modular_frame_decoder_.FinalizeDecoding(frame_header_, dec_state_, pool_,
877
34.0k
                                              /*inplace=*/true));
878
879
34.0k
  if (frame_header_.CanBeReferenced()) {
880
20.5k
    auto& info = dec_state_->shared_storage
881
20.5k
                     .reference_frames[frame_header_.save_as_reference];
882
20.5k
    *info.frame = std::move(dec_state_->frame_storage_for_referencing);
883
20.5k
    info.ib_is_in_xyb = frame_header_.save_before_color_transform;
884
20.5k
  }
885
34.0k
  return true;
886
34.0k
}
887
888
}  // namespace jxl