Coverage Report

Created: 2026-03-31 06:56

next uncovered line (L), next uncovered region (R), next uncovered branch (B)
/src/libjxl/lib/jxl/modular/transform/palette.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/modular/transform/palette.h"
7
8
#include <jxl/memory_manager.h>
9
10
#include <algorithm>
11
#include <cstddef>
12
#include <cstdint>
13
#include <utility>
14
#include <vector>
15
16
#include "lib/jxl/base/common.h"
17
#include "lib/jxl/base/compiler_specific.h"
18
#include "lib/jxl/base/data_parallel.h"
19
#include "lib/jxl/base/status.h"
20
#include "lib/jxl/image.h"
21
#include "lib/jxl/modular/encoding/context_predict.h"
22
#include "lib/jxl/modular/modular_image.h"
23
#include "lib/jxl/modular/options.h"
24
#include "lib/jxl/modular/transform/transform.h"  // CheckEqualChannels
25
26
namespace jxl {
27
28
Status InvPalette(Image &input, uint32_t begin_c, uint32_t nb_colors,
29
                  uint32_t nb_deltas, Predictor predictor,
30
3.96k
                  const weighted::Header &wp_header, ThreadPool *pool) {
31
3.96k
  JxlMemoryManager *memory_manager = input.memory_manager();
32
3.96k
  if (input.nb_meta_channels < 1) {
33
0
    return JXL_FAILURE("Error: Palette transform without palette.");
34
0
  }
35
3.96k
  int nb = input.channel[0].h;
36
3.96k
  uint32_t c0 = begin_c + 1;
37
3.96k
  if (c0 >= input.channel.size()) {
38
0
    return JXL_FAILURE("Channel is out of range.");
39
0
  }
40
3.96k
  size_t w = input.channel[c0].w;
41
3.96k
  size_t h = input.channel[c0].h;
42
3.96k
  if (nb < 1) return JXL_FAILURE("Corrupted transforms");
43
8.18k
  for (int i = 1; i < nb; i++) {
44
4.21k
    JXL_ASSIGN_OR_RETURN(Channel c, Channel::Create(memory_manager, w, h,
45
4.21k
                                                    input.channel[c0].hshift,
46
4.21k
                                                    input.channel[c0].vshift));
47
4.21k
    input.channel.insert(input.channel.begin() + c0 + 1, std::move(c));
48
4.21k
  }
49
3.96k
  const Channel &palette = input.channel[0];
50
3.96k
  const pixel_type *JXL_RESTRICT p_palette = input.channel[0].Row(0);
51
3.96k
  ptrdiff_t onerow = input.channel[0].plane.PixelsPerRow();
52
3.96k
  ptrdiff_t onerow_image = input.channel[c0].plane.PixelsPerRow();
53
3.96k
  const int bit_depth = std::min(input.bitdepth, 24);
54
55
3.96k
  if (w == 0) {
56
    // Nothing to do.
57
    // Avoid touching "empty" channels with non-zero height.
58
3.94k
  } else if (nb_deltas == 0 && predictor == Predictor::Zero) {
59
1.59k
    if (nb == 1) {
60
1.01k
      const auto process_row = [&](const uint32_t task,
61
40.9k
                                   size_t /* thread */) -> Status {
62
40.9k
        const size_t y = task;
63
40.9k
        pixel_type *p = input.channel[c0].Row(y);
64
10.1M
        for (size_t x = 0; x < w; x++) {
65
10.1M
          const int index =
66
10.1M
              Clamp1<int>(p[x], 0, static_cast<pixel_type>(palette.w) - 1);
67
10.1M
          p[x] = palette_internal::GetPaletteValue(p_palette, index, /*c=*/0,
68
10.1M
                                                   /*palette_size=*/palette.w,
69
10.1M
                                                   /*onerow=*/onerow,
70
10.1M
                                                   /*bit_depth=*/bit_depth);
71
10.1M
        }
72
40.9k
        return true;
73
40.9k
      };
74
1.01k
      JXL_RETURN_IF_ERROR(RunOnPool(pool, 0, h, ThreadPool::NoInit, process_row,
75
1.01k
                                    "UndoChannelPalette"));
76
1.01k
    } else {
77
581
      const auto process_row = [&](const uint32_t task,
78
80.6k
                                   size_t /* thread */) -> Status {
79
80.6k
        const size_t y = task;
80
80.6k
        std::vector<pixel_type *> p_out(nb);
81
80.6k
        const pixel_type *p_index = input.channel[c0].Row(y);
82
354k
        for (int c = 0; c < nb; c++) p_out[c] = input.channel[c0 + c].Row(y);
83
16.2M
        for (size_t x = 0; x < w; x++) {
84
16.1M
          const int index = p_index[x];
85
68.4M
          for (int c = 0; c < nb; c++) {
86
52.2M
            p_out[c][x] = palette_internal::GetPaletteValue(
87
52.2M
                p_palette, index, /*c=*/c,
88
52.2M
                /*palette_size=*/palette.w,
89
52.2M
                /*onerow=*/onerow, /*bit_depth=*/bit_depth);
90
52.2M
          }
91
16.1M
        }
92
80.6k
        return true;
93
80.6k
      };
94
581
      JXL_RETURN_IF_ERROR(RunOnPool(pool, 0, h, ThreadPool::NoInit, process_row,
95
581
                                    "UndoPalette"));
96
581
    }
97
2.34k
  } else {
98
    // Parallelized per channel.
99
2.34k
    ImageI indices;
100
2.34k
    ImageI &plane = input.channel[c0].plane;
101
2.34k
    JXL_ASSIGN_OR_RETURN(
102
2.34k
        indices, ImageI::Create(memory_manager, plane.xsize(), plane.ysize()));
103
2.34k
    plane.Swap(indices);
104
2.34k
    if (predictor == Predictor::Weighted) {
105
526
      const auto process_channel = [&](const uint32_t c,
106
1.48k
                                       size_t /* thread */) -> Status {
107
1.48k
        Channel &channel = input.channel[c0 + c];
108
1.48k
        weighted::State wp_state(wp_header, channel.w, channel.h);
109
121k
        for (size_t y = 0; y < channel.h; y++) {
110
120k
          pixel_type *JXL_RESTRICT p = channel.Row(y);
111
120k
          const pixel_type *JXL_RESTRICT idx = indices.Row(y);
112
20.3M
          for (size_t x = 0; x < channel.w; x++) {
113
20.2M
            int index = idx[x];
114
20.2M
            pixel_type_w val = 0;
115
20.2M
            const pixel_type palette_entry = palette_internal::GetPaletteValue(
116
20.2M
                p_palette, index, /*c=*/c,
117
20.2M
                /*palette_size=*/palette.w, /*onerow=*/onerow,
118
20.2M
                /*bit_depth=*/bit_depth);
119
20.2M
            if (index < static_cast<int32_t>(nb_deltas)) {
120
6.44M
              PredictionResult pred = PredictNoTreeWP(
121
6.44M
                  channel.w, p + x, onerow_image, x, y, predictor, &wp_state);
122
6.44M
              val = pred.guess + palette_entry;
123
13.7M
            } else {
124
13.7M
              val = palette_entry;
125
13.7M
            }
126
20.2M
            p[x] = val;
127
20.2M
            wp_state.UpdateErrors(p[x], x, y, channel.w);
128
20.2M
          }
129
120k
        }
130
1.48k
        return true;
131
1.48k
      };
132
526
      JXL_RETURN_IF_ERROR(RunOnPool(pool, 0, nb, ThreadPool::NoInit,
133
526
                                    process_channel, "UndoDeltaPaletteWP"));
134
1.82k
    } else {
135
1.82k
      const auto process_channel = [&](const uint32_t c,
136
3.65k
                                       size_t /* thread */) -> Status {
137
3.65k
        Channel &channel = input.channel[c0 + c];
138
332k
        for (size_t y = 0; y < channel.h; y++) {
139
328k
          pixel_type *JXL_RESTRICT p = channel.Row(y);
140
328k
          const pixel_type *JXL_RESTRICT idx = indices.Row(y);
141
68.8M
          for (size_t x = 0; x < channel.w; x++) {
142
68.5M
            int index = idx[x];
143
68.5M
            pixel_type_w val = 0;
144
68.5M
            const pixel_type palette_entry = palette_internal::GetPaletteValue(
145
68.5M
                p_palette, index, /*c=*/c,
146
68.5M
                /*palette_size=*/palette.w,
147
68.5M
                /*onerow=*/onerow, /*bit_depth=*/bit_depth);
148
68.5M
            if (index < static_cast<int32_t>(nb_deltas)) {
149
38.8M
              PredictionResult pred = PredictNoTreeNoWP(
150
38.8M
                  channel.w, p + x, onerow_image, x, y, predictor);
151
38.8M
              val = pred.guess + palette_entry;
152
38.8M
            } else {
153
29.6M
              val = palette_entry;
154
29.6M
            }
155
68.5M
            p[x] = val;
156
68.5M
          }
157
328k
        }
158
3.65k
        return true;
159
3.65k
      };
160
1.82k
      JXL_RETURN_IF_ERROR(RunOnPool(pool, 0, nb, ThreadPool::NoInit,
161
1.82k
                                    process_channel, "UndoDeltaPaletteNoWP"));
162
1.82k
    }
163
2.34k
  }
164
3.96k
  if (c0 >= input.nb_meta_channels) {
165
    // Palette was done on normal channels
166
3.75k
    input.nb_meta_channels--;
167
3.75k
  } else {
168
    // Palette was done on metachannels
169
206
    JXL_ENSURE(static_cast<int>(input.nb_meta_channels) >= 2 - nb);
170
206
    input.nb_meta_channels -= 2 - nb;
171
206
    JXL_ENSURE(begin_c + nb - 1 < input.nb_meta_channels);
172
206
  }
173
3.96k
  input.channel.erase(input.channel.begin(), input.channel.begin() + 1);
174
3.96k
  return true;
175
3.96k
}
176
177
Status MetaPalette(Image &input, uint32_t begin_c, uint32_t end_c,
178
4.11k
                   uint32_t nb_colors, uint32_t nb_deltas, bool lossy) {
179
4.11k
  JXL_RETURN_IF_ERROR(CheckEqualChannels(input, begin_c, end_c));
180
4.08k
  JxlMemoryManager *memory_manager = input.memory_manager();
181
182
4.08k
  size_t nb = end_c - begin_c + 1;
183
4.08k
  if (begin_c >= input.nb_meta_channels) {
184
    // Palette was done on normal channels
185
3.86k
    input.nb_meta_channels++;
186
3.86k
  } else {
187
    // Palette was done on metachannels
188
219
    JXL_ENSURE(end_c < input.nb_meta_channels);
189
    // we remove nb-1 metachannels and add one
190
219
    input.nb_meta_channels += 2 - nb;
191
219
  }
192
4.08k
  input.channel.erase(input.channel.begin() + begin_c + 1,
193
4.08k
                      input.channel.begin() + end_c + 1);
194
4.08k
  JXL_ASSIGN_OR_RETURN(
195
4.08k
      Channel pch, Channel::Create(memory_manager, nb_colors + nb_deltas, nb));
196
4.08k
  pch.hshift = -1;
197
4.08k
  pch.vshift = -1;
198
4.08k
  input.channel.insert(input.channel.begin(), std::move(pch));
199
4.08k
  return true;
200
4.08k
}
201
202
}  // namespace jxl