Coverage Report

Created: 2026-06-30 07:12

next uncovered line (L), next uncovered region (R), next uncovered branch (B)
/src/libjxl/lib/jxl/dec_cache.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_cache.h"
7
8
#include <jxl/memory_manager.h>
9
10
#include <algorithm>
11
#include <cstddef>
12
#include <cstdint>
13
#include <utility>
14
15
#include "lib/jxl/ac_strategy.h"
16
#include "lib/jxl/base/bits.h"
17
#include "lib/jxl/base/data_parallel.h"
18
#include "lib/jxl/base/status.h"
19
#include "lib/jxl/blending.h"
20
#include "lib/jxl/cms/color_encoding_cms.h"
21
#include "lib/jxl/coeff_order.h"
22
#include "lib/jxl/color_encoding_internal.h"
23
#include "lib/jxl/common.h"  // JXL_HIGH_PRECISION
24
#include "lib/jxl/frame_dimensions.h"
25
#include "lib/jxl/frame_header.h"
26
#include "lib/jxl/image.h"
27
#include "lib/jxl/image_bundle.h"
28
#include "lib/jxl/image_metadata.h"
29
#include "lib/jxl/loop_filter.h"
30
#include "lib/jxl/memory_manager_internal.h"
31
#include "lib/jxl/render_pipeline/render_pipeline.h"
32
#include "lib/jxl/render_pipeline/stage_blending.h"
33
#include "lib/jxl/render_pipeline/stage_chroma_upsampling.h"
34
#include "lib/jxl/render_pipeline/stage_cms.h"
35
#include "lib/jxl/render_pipeline/stage_epf.h"
36
#include "lib/jxl/render_pipeline/stage_from_linear.h"
37
#include "lib/jxl/render_pipeline/stage_gaborish.h"
38
#include "lib/jxl/render_pipeline/stage_noise.h"
39
#include "lib/jxl/render_pipeline/stage_patches.h"
40
#include "lib/jxl/render_pipeline/stage_splines.h"
41
#include "lib/jxl/render_pipeline/stage_spot.h"
42
#include "lib/jxl/render_pipeline/stage_to_linear.h"
43
#include "lib/jxl/render_pipeline/stage_tone_mapping.h"
44
#include "lib/jxl/render_pipeline/stage_upsampling.h"
45
#include "lib/jxl/render_pipeline/stage_write.h"
46
#include "lib/jxl/render_pipeline/stage_xyb.h"
47
#include "lib/jxl/render_pipeline/stage_ycbcr.h"
48
49
namespace jxl {
50
51
Status GroupDecCache::InitOnce(JxlMemoryManager* memory_manager,
52
28.4k
                               size_t num_passes, size_t used_acs) {
53
33.6k
  for (size_t i = 0; i < num_passes; i++) {
54
5.27k
    if (num_nzeroes[i].xsize() == 0) {
55
      // Allocate enough for a whole group - partial groups on the
56
      // right/bottom border just use a subset. The valid size is passed via
57
      // Rect.
58
59
4.66k
      JXL_ASSIGN_OR_RETURN(num_nzeroes[i],
60
4.66k
                           Image3I::Create(memory_manager, kGroupDimInBlocks,
61
4.66k
                                           kGroupDimInBlocks));
62
4.66k
    }
63
5.27k
  }
64
28.4k
  size_t max_block_area = 0;
65
66
795k
  for (uint8_t o = 0; o < AcStrategy::kNumValidStrategies; ++o) {
67
767k
    AcStrategy acs = AcStrategy::FromRawStrategy(o);
68
767k
    if ((used_acs & (1 << o)) == 0) continue;
69
635k
    size_t area =
70
635k
        acs.covered_blocks_x() * acs.covered_blocks_y() * kDCTBlockSize;
71
635k
    max_block_area = std::max(area, max_block_area);
72
635k
  }
73
74
28.4k
  if (max_block_area > max_block_area_) {
75
12.8k
    max_block_area_ = max_block_area;
76
    // We need 3x float blocks for dequantized coefficients and 1x for scratch
77
    // space for transforms.
78
12.8k
    JXL_ASSIGN_OR_RETURN(
79
12.8k
        float_memory_,
80
12.8k
        AlignedMemory::Create(memory_manager,
81
12.8k
                              max_block_area_ * 7 * sizeof(float)));
82
    // We need 3x int32 or int16 blocks for quantized coefficients.
83
12.8k
    JXL_ASSIGN_OR_RETURN(
84
12.8k
        int32_memory_,
85
12.8k
        AlignedMemory::Create(memory_manager,
86
12.8k
                              max_block_area_ * 3 * sizeof(int32_t)));
87
12.8k
    JXL_ASSIGN_OR_RETURN(
88
12.8k
        int16_memory_,
89
12.8k
        AlignedMemory::Create(memory_manager,
90
12.8k
                              max_block_area_ * 3 * sizeof(int16_t)));
91
12.8k
  }
92
93
28.4k
  dec_group_block = float_memory_.address<float>();
94
28.4k
  scratch_space = dec_group_block + max_block_area_ * 3;
95
28.4k
  dec_group_qblock = int32_memory_.address<int32_t>();
96
28.4k
  dec_group_qblock16 = int16_memory_.address<int16_t>();
97
28.4k
  return true;
98
28.4k
}
99
100
// Initialize the decoder state after all of DC is decoded.
101
35.4k
Status PassesDecoderState::InitForAC(size_t num_passes, ThreadPool* pool) {
102
35.4k
  shared_storage.coeff_order_size = 0;
103
991k
  for (uint8_t o = 0; o < AcStrategy::kNumValidStrategies; ++o) {
104
956k
    if (((1 << o) & used_acs) == 0) continue;
105
7.58k
    uint8_t ord = kStrategyOrder[o];
106
7.58k
    shared_storage.coeff_order_size =
107
7.58k
        std::max(kCoeffOrderOffset[3 * (ord + 1)] * kDCTBlockSize,
108
7.58k
                 shared_storage.coeff_order_size);
109
7.58k
  }
110
35.4k
  size_t sz = num_passes * shared_storage.coeff_order_size;
111
35.4k
  if (sz > shared_storage.coeff_orders.size()) {
112
1.27k
    shared_storage.coeff_orders.resize(sz);
113
1.27k
  }
114
35.4k
  return true;
115
35.4k
}
116
117
Status PassesDecoderState::PreparePipeline(const FrameHeader& frame_header,
118
                                           const ImageMetadata* metadata,
119
                                           ImageBundle* decoded,
120
43.6k
                                           PipelineOptions options) {
121
43.6k
  JxlMemoryManager* memory_manager = this->memory_manager();
122
43.6k
  size_t num_c = 3 + frame_header.nonserialized_metadata->m.num_extra_channels;
123
43.6k
  bool render_noise =
124
43.6k
      (options.render_noise && (frame_header.flags & FrameHeader::kNoise) != 0);
125
43.6k
  size_t num_tmp_c = render_noise ? 3 : 0;
126
127
43.6k
  if (frame_header.CanBeReferenced()) {
128
    // Necessary so that SetInputSizes() can allocate output buffers as needed.
129
21.0k
    frame_storage_for_referencing = ImageBundle(memory_manager, metadata);
130
21.0k
  }
131
132
43.6k
  RenderPipeline::Builder builder(memory_manager, num_c + num_tmp_c);
133
134
43.6k
  if (options.use_slow_render_pipeline) {
135
0
    builder.UseSimpleImplementation();
136
0
  }
137
138
43.6k
  if (!frame_header.chroma_subsampling.Is444()) {
139
24.5k
    for (size_t c = 0; c < 3; c++) {
140
18.4k
      if (frame_header.chroma_subsampling.HShift(c) != 0) {
141
7.02k
        JXL_RETURN_IF_ERROR(
142
7.02k
            builder.AddStage(GetChromaUpsamplingStage(c, /*horizontal=*/true)));
143
7.02k
      }
144
18.4k
      if (frame_header.chroma_subsampling.VShift(c) != 0) {
145
6.15k
        JXL_RETURN_IF_ERROR(builder.AddStage(
146
6.15k
            GetChromaUpsamplingStage(c, /*horizontal=*/false)));
147
6.15k
      }
148
18.4k
    }
149
6.13k
  }
150
151
43.6k
  if (frame_header.loop_filter.gab) {
152
22.3k
    JXL_RETURN_IF_ERROR(
153
22.3k
        builder.AddStage(GetGaborishStage(frame_header.loop_filter)));
154
22.3k
  }
155
156
43.6k
  {
157
43.6k
    const LoopFilter& lf = frame_header.loop_filter;
158
43.6k
    if (lf.epf_iters >= 3) {
159
509
      JXL_RETURN_IF_ERROR(
160
509
          builder.AddStage(GetEPFStage(lf, sigma, EpfStage::Zero)));
161
509
    }
162
43.6k
    if (lf.epf_iters >= 1) {
163
21.8k
      JXL_RETURN_IF_ERROR(
164
21.8k
          builder.AddStage(GetEPFStage(lf, sigma, EpfStage::One)));
165
21.8k
    }
166
43.6k
    if (lf.epf_iters >= 2) {
167
13.2k
      JXL_RETURN_IF_ERROR(
168
13.2k
          builder.AddStage(GetEPFStage(lf, sigma, EpfStage::Two)));
169
13.2k
    }
170
43.6k
  }
171
172
43.6k
  bool late_ec_upsample = frame_header.upsampling != 1;
173
43.6k
  for (auto ecups : frame_header.extra_channel_upsampling) {
174
14.8k
    if (ecups != frame_header.upsampling) {
175
      // If patches are applied, either frame_header.upsampling == 1 or
176
      // late_ec_upsample is true.
177
6.05k
      late_ec_upsample = false;
178
6.05k
    }
179
14.8k
  }
180
181
43.6k
  if (!late_ec_upsample) {
182
35.3k
    for (size_t ec = 0; ec < frame_header.extra_channel_upsampling.size();
183
22.2k
         ec++) {
184
13.0k
      if (frame_header.extra_channel_upsampling[ec] != 1) {
185
6.14k
        JXL_RETURN_IF_ERROR(builder.AddStage(GetUpsamplingStage(
186
6.14k
            memory_manager, frame_header.nonserialized_metadata->transform_data,
187
6.14k
            3 + ec,
188
6.14k
            CeilLog2Nonzero(frame_header.extra_channel_upsampling[ec]))));
189
6.14k
      }
190
13.0k
    }
191
22.2k
  }
192
193
43.6k
  if ((frame_header.flags & FrameHeader::kPatches) != 0) {
194
3.55k
    JXL_RETURN_IF_ERROR(builder.AddStage(GetPatchesStage(
195
3.55k
        &shared->image_features.patches,
196
3.55k
        &frame_header.nonserialized_metadata->m.extra_channel_info)));
197
3.55k
  }
198
43.6k
  if ((frame_header.flags & FrameHeader::kSplines) != 0) {
199
3.66k
    JXL_RETURN_IF_ERROR(
200
3.66k
        builder.AddStage(GetSplineStage(&shared->image_features.splines)));
201
3.66k
  }
202
203
43.6k
  if (frame_header.upsampling != 1) {
204
22.2k
    size_t nb_channels =
205
22.2k
        3 +
206
22.2k
        (late_ec_upsample ? frame_header.extra_channel_upsampling.size() : 0);
207
90.8k
    for (size_t c = 0; c < nb_channels; c++) {
208
68.5k
      JXL_RETURN_IF_ERROR(builder.AddStage(GetUpsamplingStage(
209
68.5k
          memory_manager, frame_header.nonserialized_metadata->transform_data,
210
68.5k
          c, CeilLog2Nonzero(frame_header.upsampling))));
211
68.5k
    }
212
22.2k
  }
213
  // Starting from this line all the stages considered to have zero xextra.
214
  // Upsampling does not have xextra as well (even if it happens before
215
  // splines/patches for EC).
216
43.6k
  if (render_noise) {
217
4.68k
    JXL_RETURN_IF_ERROR(builder.AddStage(GetConvolveNoiseStage(num_c)));
218
4.68k
    JXL_RETURN_IF_ERROR(builder.AddStage(GetAddNoiseStage(
219
4.68k
        shared->image_features.noise_params, shared->cmap.base(), num_c)));
220
4.68k
  }
221
43.6k
  if (frame_header.dc_level != 0) {
222
10.4k
    JXL_RETURN_IF_ERROR(builder.AddStage(GetWriteToImage3FStage(
223
10.4k
        memory_manager, &shared_storage.dc_frames[frame_header.dc_level - 1])));
224
10.4k
  }
225
226
43.6k
  if (frame_header.CanBeReferenced() &&
227
21.0k
      frame_header.save_before_color_transform) {
228
3.88k
    JXL_RETURN_IF_ERROR(builder.AddStage(GetWriteToImageBundleStage(
229
3.88k
        &frame_storage_for_referencing, output_encoding_info)));
230
3.88k
  }
231
232
43.6k
  bool has_alpha = false;
233
43.6k
  size_t alpha_c = 0;
234
45.0k
  for (size_t i = 0; i < metadata->extra_channel_info.size(); i++) {
235
9.16k
    if (metadata->extra_channel_info[i].type == ExtraChannel::kAlpha) {
236
7.79k
      has_alpha = true;
237
7.79k
      alpha_c = 3 + i;
238
7.79k
      break;
239
7.79k
    }
240
9.16k
  }
241
242
43.6k
  if (fast_xyb_srgb8_conversion) {
243
#if !JXL_HIGH_PRECISION
244
    JXL_ENSURE(!NeedsBlending(frame_header));
245
    JXL_ENSURE(!frame_header.CanBeReferenced() ||
246
               frame_header.save_before_color_transform);
247
    JXL_ENSURE(!options.render_spotcolors ||
248
               !metadata->Find(ExtraChannel::kSpotColor));
249
    bool is_rgba = (main_output.format.num_channels == 4);
250
    uint8_t* rgb_output = reinterpret_cast<uint8_t*>(main_output.buffer);
251
    JXL_RETURN_IF_ERROR(builder.AddStage(
252
        GetFastXYBTosRGB8Stage(rgb_output, main_output.stride, width, height,
253
                               is_rgba, has_alpha, alpha_c)));
254
#endif
255
43.6k
  } else {
256
43.6k
    bool linear = false;
257
43.6k
    if (frame_header.color_transform == ColorTransform::kYCbCr) {
258
6.20k
      JXL_RETURN_IF_ERROR(builder.AddStage(GetYCbCrStage()));
259
37.4k
    } else if (frame_header.color_transform == ColorTransform::kXYB) {
260
19.6k
      JXL_RETURN_IF_ERROR(builder.AddStage(GetXYBStage(output_encoding_info)));
261
19.6k
      if (output_encoding_info.color_encoding.GetColorSpace() !=
262
19.6k
          ColorSpace::kXYB) {
263
19.6k
        linear = true;
264
19.6k
      }
265
19.6k
    }  // Nothing to do for kNone.
266
267
43.6k
    if (options.coalescing && NeedsBlending(frame_header)) {
268
10.4k
      if (linear) {
269
4.90k
        JXL_RETURN_IF_ERROR(
270
4.90k
            builder.AddStage(GetFromLinearStage(output_encoding_info)));
271
4.90k
        linear = false;
272
4.90k
      }
273
10.4k
      JXL_RETURN_IF_ERROR(builder.AddStage(GetBlendingStage(
274
10.4k
          frame_header, this, output_encoding_info.color_encoding)));
275
10.4k
    }
276
277
43.6k
    if (options.coalescing && frame_header.CanBeReferenced() &&
278
21.0k
        !frame_header.save_before_color_transform) {
279
17.1k
      if (linear) {
280
4.07k
        JXL_RETURN_IF_ERROR(
281
4.07k
            builder.AddStage(GetFromLinearStage(output_encoding_info)));
282
4.07k
        linear = false;
283
4.07k
      }
284
17.1k
      JXL_RETURN_IF_ERROR(builder.AddStage(GetWriteToImageBundleStage(
285
17.1k
          &frame_storage_for_referencing, output_encoding_info)));
286
17.1k
    }
287
288
43.6k
    if (options.render_spotcolors &&
289
35.4k
        frame_header.nonserialized_metadata->m.Find(ExtraChannel::kSpotColor)) {
290
2.53k
      for (size_t i = 0; i < metadata->extra_channel_info.size(); i++) {
291
        // Don't use Find() because there may be multiple spot color channels.
292
2.16k
        const ExtraChannelInfo& eci = metadata->extra_channel_info[i];
293
2.16k
        if (eci.type == ExtraChannel::kSpotColor) {
294
653
          JXL_RETURN_IF_ERROR(
295
653
              builder.AddStage(GetSpotColorStage(i, eci.spot_color)));
296
653
        }
297
2.16k
      }
298
362
    }
299
300
43.6k
    auto tone_mapping_stage = GetToneMappingStage(output_encoding_info);
301
43.6k
    if (tone_mapping_stage) {
302
0
      if (!linear) {
303
0
        auto to_linear_stage = GetToLinearStage(output_encoding_info);
304
0
        if (!to_linear_stage) {
305
0
          if (!output_encoding_info.cms_set) {
306
0
            return JXL_FAILURE("Cannot tonemap this colorspace without a CMS");
307
0
          }
308
0
          auto cms_stage = GetCmsStage(output_encoding_info);
309
0
          if (cms_stage) {
310
0
            JXL_RETURN_IF_ERROR(builder.AddStage(std::move(cms_stage)));
311
0
          }
312
0
        } else {
313
0
          JXL_RETURN_IF_ERROR(builder.AddStage(std::move(to_linear_stage)));
314
0
        }
315
0
        linear = true;
316
0
      }
317
0
      JXL_RETURN_IF_ERROR(builder.AddStage(std::move(tone_mapping_stage)));
318
0
    }
319
320
43.6k
    if (linear) {
321
10.6k
      const size_t channels_src =
322
10.6k
          (output_encoding_info.orig_color_encoding.IsCMYK()
323
10.6k
               ? 4
324
10.6k
               : output_encoding_info.orig_color_encoding.Channels());
325
10.6k
      const size_t channels_dst =
326
10.6k
          output_encoding_info.color_encoding.Channels();
327
10.6k
      bool mixing_color_and_grey = (channels_dst != channels_src);
328
10.6k
      if ((output_encoding_info.color_encoding_is_original) ||
329
10.6k
          (!output_encoding_info.cms_set) || mixing_color_and_grey) {
330
        // in those cases we only need a linear stage in other cases we attempt
331
        // to obtain a cms stage: the cases are
332
        // - output_encoding_info.color_encoding_is_original: no cms stage
333
        // needed because it would be a no-op
334
        // - !output_encoding_info.cms_set: can't use the cms, so no point in
335
        // trying to add a cms stage
336
        // - mixing_color_and_grey: cms stage can't handle that
337
        // TODO(firsching): remove "mixing_color_and_grey" condition after
338
        // adding support for greyscale to cms stage.
339
10.6k
        JXL_RETURN_IF_ERROR(
340
10.6k
            builder.AddStage(GetFromLinearStage(output_encoding_info)));
341
10.6k
      } else {
342
0
        if (!output_encoding_info.linear_color_encoding.CreateICC()) {
343
0
          return JXL_FAILURE("Failed to create ICC");
344
0
        }
345
0
        auto cms_stage = GetCmsStage(output_encoding_info);
346
0
        if (cms_stage) {
347
0
          JXL_RETURN_IF_ERROR(builder.AddStage(std::move(cms_stage)));
348
0
        }
349
0
      }
350
10.6k
      linear = false;
351
33.0k
    } else {
352
33.0k
      auto cms_stage = GetCmsStage(output_encoding_info, false);
353
33.0k
      if (cms_stage) {
354
0
        JXL_RETURN_IF_ERROR(builder.AddStage(std::move(cms_stage)));
355
0
      }
356
33.0k
    }
357
43.6k
    (void)linear;
358
359
43.6k
    if (main_output.callback.IsPresent() || main_output.buffer) {
360
4.15k
      JXL_RETURN_IF_ERROR(builder.AddStage(GetWriteToOutputStage(
361
4.15k
          main_output, width, height, has_alpha, unpremul_alpha, alpha_c,
362
4.15k
          undo_orientation, extra_output, memory_manager)));
363
39.5k
    } else {
364
39.5k
      JXL_RETURN_IF_ERROR(builder.AddStage(
365
39.5k
          GetWriteToImageBundleStage(decoded, output_encoding_info)));
366
39.5k
    }
367
43.6k
  }
368
43.6k
  JXL_ASSIGN_OR_RETURN(render_pipeline,
369
43.6k
                       std::move(builder).Finalize(shared->frame_dim));
370
43.6k
  return render_pipeline->IsInitialized();
371
43.6k
}
372
373
}  // namespace jxl