Coverage Report

Created: 2026-01-20 07:37

next uncovered line (L), next uncovered region (R), next uncovered branch (B)
/src/libjxl/lib/jxl/image.h
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
#ifndef LIB_JXL_IMAGE_H_
7
#define LIB_JXL_IMAGE_H_
8
9
// SIMD/multicore-friendly planar image representation with row accessors.
10
11
#if defined(ADDRESS_SANITIZER) || defined(MEMORY_SANITIZER) || \
12
    defined(THREAD_SANITIZER)
13
#include <cinttypes>  // PRIu64
14
#endif
15
16
#include <jxl/memory_manager.h>
17
18
#include <algorithm>
19
#include <cstddef>
20
#include <cstdint>
21
#include <cstring>
22
#include <utility>  // std::move
23
24
#include "lib/jxl/base/compiler_specific.h"
25
#include "lib/jxl/base/status.h"
26
#include "lib/jxl/memory_manager_internal.h"
27
28
namespace jxl {
29
30
// DO NOT use PlaneBase outside of image.{h|cc}
31
namespace detail {
32
33
// Type-independent parts of Plane<> - reduces code duplication and facilitates
34
// moving member function implementations to cc file.
35
struct PlaneBase {
36
  PlaneBase()
37
16.3M
      : xsize_(0),
38
16.3M
        ysize_(0),
39
16.3M
        orig_xsize_(0),
40
16.3M
        orig_ysize_(0),
41
16.3M
        bytes_per_row_(0),
42
16.3M
        sizeof_t_(0) {}
43
44
  // Copy construction/assignment is forbidden to avoid inadvertent copies,
45
  // which can be very expensive. Use CopyImageTo() instead.
46
  PlaneBase(const PlaneBase& other) = delete;
47
  PlaneBase& operator=(const PlaneBase& other) = delete;
48
49
  // Move constructor (required for returning Image from function)
50
46.1M
  PlaneBase(PlaneBase&& other) noexcept = default;
51
52
  // Move assignment (required for std::vector)
53
10.0M
  PlaneBase& operator=(PlaneBase&& other) noexcept = default;
54
55
83.3M
  ~PlaneBase() = default;
56
57
  void Swap(PlaneBase& other);
58
59
  // Useful for pre-allocating image with some padding for alignment purposes
60
  // and later reporting the actual valid dimensions. May also be used to
61
  // un-shrink the image. Caller is responsible for ensuring xsize/ysize are <=
62
  // the original dimensions.
63
166k
  Status ShrinkTo(const size_t xsize, const size_t ysize) {
64
166k
    JXL_ENSURE(xsize <= orig_xsize_);
65
166k
    JXL_ENSURE(ysize <= orig_ysize_);
66
166k
    xsize_ = static_cast<uint32_t>(xsize);
67
166k
    ysize_ = static_cast<uint32_t>(ysize);
68
    // NOTE: we can't recompute bytes_per_row for more compact storage and
69
    // better locality because that would invalidate the image contents.
70
166k
    return true;
71
166k
  }
72
73
  // How many pixels.
74
513M
  JXL_INLINE size_t xsize() const { return xsize_; }
75
2.42G
  JXL_INLINE size_t ysize() const { return ysize_; }
76
77
  // NOTE: do not use this for copying rows - the valid xsize may be much less.
78
1.05G
  JXL_INLINE size_t bytes_per_row() const { return bytes_per_row_; }
79
80
296k
  JXL_INLINE JxlMemoryManager* memory_manager() const {
81
296k
    return bytes_.memory_manager();
82
296k
  }
83
84
  // Raw access to byte contents, for interfacing with other libraries.
85
  // Unsigned char instead of char to avoid surprises (sign extension).
86
46.8M
  JXL_INLINE uint8_t* bytes() {
87
46.8M
    uint8_t* p = bytes_.address<uint8_t>();
88
46.8M
    return static_cast<uint8_t * JXL_RESTRICT>(JXL_ASSUME_ALIGNED(p, 64));
89
46.8M
  }
90
1.00G
  JXL_INLINE const uint8_t* bytes() const {
91
1.00G
    const uint8_t* p = bytes_.address<uint8_t>();
92
1.00G
    return static_cast<const uint8_t * JXL_RESTRICT>(JXL_ASSUME_ALIGNED(p, 64));
93
1.00G
  }
94
95
 protected:
96
  PlaneBase(uint32_t xsize, uint32_t ysize, size_t sizeof_t);
97
  Status Allocate(JxlMemoryManager* memory_manager, size_t pre_padding);
98
99
  // Returns pointer to the start of a row.
100
5.93G
  JXL_INLINE void* VoidRow(const size_t y) const {
101
5.93G
    JXL_DASSERT(y < ysize_);
102
5.93G
    uint8_t* row = bytes_.address<uint8_t>() + y * bytes_per_row_;
103
5.93G
    return JXL_ASSUME_ALIGNED(row, 64);
104
5.93G
  }
105
106
  // (Members are non-const to enable assignment during move-assignment.)
107
  uint32_t xsize_;  // In valid pixels, not including any padding.
108
  uint32_t ysize_;
109
  uint32_t orig_xsize_;
110
  uint32_t orig_ysize_;
111
  size_t bytes_per_row_;  // Includes padding.
112
  AlignedMemory bytes_;
113
  size_t sizeof_t_;
114
};
115
116
}  // namespace detail
117
118
// Single channel, aligned rows separated by padding. T must be POD.
119
//
120
// 'Single channel' (one 2D array per channel) simplifies vectorization
121
// (repeating the same operation on multiple adjacent components) without the
122
// complexity of a hybrid layout (8 R, 8 G, 8 B, ...). In particular, clients
123
// can easily iterate over all components in a row and Image requires no
124
// knowledge of the pixel format beyond the component type "T".
125
//
126
// 'Aligned' means each row is aligned to the L1 cache line size. This prevents
127
// false sharing between two threads operating on adjacent rows.
128
//
129
// 'Padding' is still relevant because vectors could potentially be larger than
130
// a cache line. By rounding up row sizes to the vector size, we allow
131
// reading/writing ALIGNED vectors whose first lane is a valid sample. This
132
// avoids needing a separate loop to handle remaining unaligned lanes.
133
//
134
// This image layout could also be achieved with a vector and a row accessor
135
// function, but a class wrapper with support for "deleter" allows wrapping
136
// existing memory allocated by clients without copying the pixels. It also
137
// provides convenient accessors for xsize/ysize, which shortens function
138
// argument lists. Supports move-construction so it can be stored in containers.
139
template <typename ComponentType>
140
class Plane : public detail::PlaneBase {
141
 public:
142
  using T = ComponentType;
143
  static constexpr size_t kNumPlanes = 1;
144
145
16.3M
  Plane() = default;
jxl::Plane<float>::Plane()
Line
Count
Source
145
12.3M
  Plane() = default;
jxl::Plane<int>::Plane()
Line
Count
Source
145
2.56M
  Plane() = default;
jxl::Plane<unsigned char>::Plane()
Line
Count
Source
145
760k
  Plane() = default;
jxl::Plane<signed char>::Plane()
Line
Count
Source
145
703k
  Plane() = default;
Unexecuted instantiation: jxl::Plane<hwy::float16_t>::Plane()
Unexecuted instantiation: jxl::Plane<unsigned int>::Plane()
jxl::Plane<short>::Plane()
Line
Count
Source
145
36.6k
  Plane() = default;
146
147
  static StatusOr<Plane> Create(JxlMemoryManager* memory_manager,
148
                                const size_t xsize, const size_t ysize,
149
20.7M
                                const size_t pre_padding = 0) {
150
20.7M
    static_assert(
151
20.7M
        sizeof(T) == 1 || sizeof(T) == 2 || sizeof(T) == 4 || sizeof(T) == 8,
152
20.7M
        "Only 1/2/4/8-byte samples are supported");
153
20.7M
    uint32_t xsize32 = static_cast<uint32_t>(xsize);
154
20.7M
    uint32_t ysize32 = static_cast<uint32_t>(ysize);
155
20.7M
    JXL_ENSURE(xsize32 == xsize);
156
20.7M
    JXL_ENSURE(ysize32 == ysize);
157
20.7M
    Plane plane(xsize32, ysize32, sizeof(T));
158
20.7M
    JXL_RETURN_IF_ERROR(plane.Allocate(memory_manager, pre_padding));
159
20.7M
    return plane;
160
20.7M
  }
jxl::Plane<float>::Create(JxlMemoryManagerStruct*, unsigned long, unsigned long, unsigned long)
Line
Count
Source
149
18.7M
                                const size_t pre_padding = 0) {
150
18.7M
    static_assert(
151
18.7M
        sizeof(T) == 1 || sizeof(T) == 2 || sizeof(T) == 4 || sizeof(T) == 8,
152
18.7M
        "Only 1/2/4/8-byte samples are supported");
153
18.7M
    uint32_t xsize32 = static_cast<uint32_t>(xsize);
154
18.7M
    uint32_t ysize32 = static_cast<uint32_t>(ysize);
155
18.7M
    JXL_ENSURE(xsize32 == xsize);
156
18.7M
    JXL_ENSURE(ysize32 == ysize);
157
18.7M
    Plane plane(xsize32, ysize32, sizeof(T));
158
18.7M
    JXL_RETURN_IF_ERROR(plane.Allocate(memory_manager, pre_padding));
159
18.7M
    return plane;
160
18.7M
  }
jxl::Plane<int>::Create(JxlMemoryManagerStruct*, unsigned long, unsigned long, unsigned long)
Line
Count
Source
149
1.26M
                                const size_t pre_padding = 0) {
150
1.26M
    static_assert(
151
1.26M
        sizeof(T) == 1 || sizeof(T) == 2 || sizeof(T) == 4 || sizeof(T) == 8,
152
1.26M
        "Only 1/2/4/8-byte samples are supported");
153
1.26M
    uint32_t xsize32 = static_cast<uint32_t>(xsize);
154
1.26M
    uint32_t ysize32 = static_cast<uint32_t>(ysize);
155
1.26M
    JXL_ENSURE(xsize32 == xsize);
156
1.26M
    JXL_ENSURE(ysize32 == ysize);
157
1.26M
    Plane plane(xsize32, ysize32, sizeof(T));
158
1.26M
    JXL_RETURN_IF_ERROR(plane.Allocate(memory_manager, pre_padding));
159
1.26M
    return plane;
160
1.26M
  }
jxl::Plane<unsigned char>::Create(JxlMemoryManagerStruct*, unsigned long, unsigned long, unsigned long)
Line
Count
Source
149
455k
                                const size_t pre_padding = 0) {
150
455k
    static_assert(
151
455k
        sizeof(T) == 1 || sizeof(T) == 2 || sizeof(T) == 4 || sizeof(T) == 8,
152
455k
        "Only 1/2/4/8-byte samples are supported");
153
455k
    uint32_t xsize32 = static_cast<uint32_t>(xsize);
154
455k
    uint32_t ysize32 = static_cast<uint32_t>(ysize);
155
455k
    JXL_ENSURE(xsize32 == xsize);
156
455k
    JXL_ENSURE(ysize32 == ysize);
157
455k
    Plane plane(xsize32, ysize32, sizeof(T));
158
455k
    JXL_RETURN_IF_ERROR(plane.Allocate(memory_manager, pre_padding));
159
455k
    return plane;
160
455k
  }
jxl::Plane<signed char>::Create(JxlMemoryManagerStruct*, unsigned long, unsigned long, unsigned long)
Line
Count
Source
149
294k
                                const size_t pre_padding = 0) {
150
294k
    static_assert(
151
294k
        sizeof(T) == 1 || sizeof(T) == 2 || sizeof(T) == 4 || sizeof(T) == 8,
152
294k
        "Only 1/2/4/8-byte samples are supported");
153
294k
    uint32_t xsize32 = static_cast<uint32_t>(xsize);
154
294k
    uint32_t ysize32 = static_cast<uint32_t>(ysize);
155
294k
    JXL_ENSURE(xsize32 == xsize);
156
294k
    JXL_ENSURE(ysize32 == ysize);
157
294k
    Plane plane(xsize32, ysize32, sizeof(T));
158
294k
    JXL_RETURN_IF_ERROR(plane.Allocate(memory_manager, pre_padding));
159
294k
    return plane;
160
294k
  }
Unexecuted instantiation: jxl::Plane<hwy::float16_t>::Create(JxlMemoryManagerStruct*, unsigned long, unsigned long, unsigned long)
Unexecuted instantiation: jxl::Plane<unsigned int>::Create(JxlMemoryManagerStruct*, unsigned long, unsigned long, unsigned long)
jxl::Plane<short>::Create(JxlMemoryManagerStruct*, unsigned long, unsigned long, unsigned long)
Line
Count
Source
149
9.16k
                                const size_t pre_padding = 0) {
150
9.16k
    static_assert(
151
9.16k
        sizeof(T) == 1 || sizeof(T) == 2 || sizeof(T) == 4 || sizeof(T) == 8,
152
9.16k
        "Only 1/2/4/8-byte samples are supported");
153
9.16k
    uint32_t xsize32 = static_cast<uint32_t>(xsize);
154
9.16k
    uint32_t ysize32 = static_cast<uint32_t>(ysize);
155
9.16k
    JXL_ENSURE(xsize32 == xsize);
156
9.16k
    JXL_ENSURE(ysize32 == ysize);
157
9.16k
    Plane plane(xsize32, ysize32, sizeof(T));
158
9.16k
    JXL_RETURN_IF_ERROR(plane.Allocate(memory_manager, pre_padding));
159
9.16k
    return plane;
160
9.16k
  }
161
162
279M
  JXL_INLINE T* Row(const size_t y) { return static_cast<T*>(VoidRow(y)); }
jxl::Plane<unsigned char>::Row(unsigned long)
Line
Count
Source
162
10.9M
  JXL_INLINE T* Row(const size_t y) { return static_cast<T*>(VoidRow(y)); }
jxl::Plane<float>::Row(unsigned long)
Line
Count
Source
162
230M
  JXL_INLINE T* Row(const size_t y) { return static_cast<T*>(VoidRow(y)); }
jxl::Plane<int>::Row(unsigned long)
Line
Count
Source
162
36.4M
  JXL_INLINE T* Row(const size_t y) { return static_cast<T*>(VoidRow(y)); }
jxl::Plane<signed char>::Row(unsigned long)
Line
Count
Source
162
1.67M
  JXL_INLINE T* Row(const size_t y) { return static_cast<T*>(VoidRow(y)); }
Unexecuted instantiation: jxl::Plane<hwy::float16_t>::Row(unsigned long)
Unexecuted instantiation: jxl::Plane<unsigned int>::Row(unsigned long)
Unexecuted instantiation: jxl::Plane<short>::Row(unsigned long)
163
164
  // Returns pointer to const (see above).
165
3.70G
  JXL_INLINE const T* Row(const size_t y) const {
166
3.70G
    return static_cast<const T*>(VoidRow(y));
167
3.70G
  }
jxl::Plane<int>::Row(unsigned long) const
Line
Count
Source
165
98.1M
  JXL_INLINE const T* Row(const size_t y) const {
166
98.1M
    return static_cast<const T*>(VoidRow(y));
167
98.1M
  }
jxl::Plane<float>::Row(unsigned long) const
Line
Count
Source
165
3.61G
  JXL_INLINE const T* Row(const size_t y) const {
166
3.61G
    return static_cast<const T*>(VoidRow(y));
167
3.61G
  }
Unexecuted instantiation: jxl::Plane<unsigned char>::Row(unsigned long) const
168
169
  // Documents that the access is const.
170
1.94G
  JXL_INLINE const T* ConstRow(const size_t y) const {
171
1.94G
    return static_cast<const T*>(VoidRow(y));
172
1.94G
  }
jxl::Plane<unsigned char>::ConstRow(unsigned long) const
Line
Count
Source
170
59.1M
  JXL_INLINE const T* ConstRow(const size_t y) const {
171
59.1M
    return static_cast<const T*>(VoidRow(y));
172
59.1M
  }
jxl::Plane<float>::ConstRow(unsigned long) const
Line
Count
Source
170
1.87G
  JXL_INLINE const T* ConstRow(const size_t y) const {
171
1.87G
    return static_cast<const T*>(VoidRow(y));
172
1.87G
  }
jxl::Plane<int>::ConstRow(unsigned long) const
Line
Count
Source
170
2.69M
  JXL_INLINE const T* ConstRow(const size_t y) const {
171
2.69M
    return static_cast<const T*>(VoidRow(y));
172
2.69M
  }
jxl::Plane<signed char>::ConstRow(unsigned long) const
Line
Count
Source
170
2.60M
  JXL_INLINE const T* ConstRow(const size_t y) const {
171
2.60M
    return static_cast<const T*>(VoidRow(y));
172
2.60M
  }
173
174
  // Returns number of pixels (some of which are padding) per row. Useful for
175
  // computing other rows via pointer arithmetic. WARNING: this must
176
  // NOT be used to determine xsize.
177
10.1M
  JXL_INLINE ptrdiff_t PixelsPerRow() const {
178
10.1M
    return static_cast<ptrdiff_t>(bytes_per_row_ / sizeof(T));
179
10.1M
  }
jxl::Plane<unsigned char>::PixelsPerRow() const
Line
Count
Source
177
193k
  JXL_INLINE ptrdiff_t PixelsPerRow() const {
178
193k
    return static_cast<ptrdiff_t>(bytes_per_row_ / sizeof(T));
179
193k
  }
jxl::Plane<int>::PixelsPerRow() const
Line
Count
Source
177
7.04M
  JXL_INLINE ptrdiff_t PixelsPerRow() const {
178
7.04M
    return static_cast<ptrdiff_t>(bytes_per_row_ / sizeof(T));
179
7.04M
  }
jxl::Plane<float>::PixelsPerRow() const
Line
Count
Source
177
2.92M
  JXL_INLINE ptrdiff_t PixelsPerRow() const {
178
2.92M
    return static_cast<ptrdiff_t>(bytes_per_row_ / sizeof(T));
179
2.92M
  }
Unexecuted instantiation: jxl::Plane<short>::PixelsPerRow() const
180
181
 private:
182
  Plane(uint32_t xsize, uint32_t ysize, size_t sizeof_t)
183
20.7M
      : detail::PlaneBase(xsize, ysize, sizeof_t) {}
jxl::Plane<int>::Plane(unsigned int, unsigned int, unsigned long)
Line
Count
Source
183
1.26M
      : detail::PlaneBase(xsize, ysize, sizeof_t) {}
jxl::Plane<unsigned char>::Plane(unsigned int, unsigned int, unsigned long)
Line
Count
Source
183
455k
      : detail::PlaneBase(xsize, ysize, sizeof_t) {}
jxl::Plane<float>::Plane(unsigned int, unsigned int, unsigned long)
Line
Count
Source
183
18.7M
      : detail::PlaneBase(xsize, ysize, sizeof_t) {}
jxl::Plane<signed char>::Plane(unsigned int, unsigned int, unsigned long)
Line
Count
Source
183
294k
      : detail::PlaneBase(xsize, ysize, sizeof_t) {}
Unexecuted instantiation: jxl::Plane<hwy::float16_t>::Plane(unsigned int, unsigned int, unsigned long)
Unexecuted instantiation: jxl::Plane<unsigned int>::Plane(unsigned int, unsigned int, unsigned long)
jxl::Plane<short>::Plane(unsigned int, unsigned int, unsigned long)
Line
Count
Source
183
9.16k
      : detail::PlaneBase(xsize, ysize, sizeof_t) {}
184
};
185
186
using ImageSB = Plane<int8_t>;
187
using ImageB = Plane<uint8_t>;
188
using ImageS = Plane<int16_t>;  // signed integer or half-float
189
using ImageU = Plane<uint16_t>;
190
using ImageI = Plane<int32_t>;
191
using ImageF = Plane<float>;
192
using ImageD = Plane<double>;
193
194
// Currently, we abuse Image to either refer to an image that owns its storage
195
// or one that doesn't. In similar vein, we abuse Image* function parameters to
196
// either mean "assign to me" or "fill the provided image with data".
197
// Hopefully, the "assign to me" meaning will go away and most images in the
198
// codebase will not be backed by own storage. When this happens we can redesign
199
// Image to be a non-storage-holding view class and introduce BackedImage in
200
// those places that actually need it.
201
202
// NOTE: we can't use Image as a view because invariants are violated
203
// (alignment and the presence of padding before/after each "row").
204
205
// A bundle of 3 same-sized images. Typically constructed by moving from three
206
// rvalue references to Image. To overwrite an existing Image3 using
207
// single-channel producers, we also need access to Image*. Constructing
208
// temporary non-owning Image pointing to one plane of an existing Image3 risks
209
// dangling references, especially if the wrapper is moved. Therefore, we
210
// store an array of Image (which are compact enough that size is not a concern)
211
// and provide Plane+Row accessors.
212
template <typename ComponentType>
213
class Image3 {
214
 public:
215
  using T = ComponentType;
216
  using PlaneT = jxl::Plane<T>;
217
  static constexpr size_t kNumPlanes = 3;
218
219
3.26M
  Image3() : planes_{PlaneT(), PlaneT(), PlaneT()} {}
jxl::Image3<float>::Image3()
Line
Count
Source
219
2.51M
  Image3() : planes_{PlaneT(), PlaneT(), PlaneT()} {}
jxl::Image3<int>::Image3()
Line
Count
Source
219
745k
  Image3() : planes_{PlaneT(), PlaneT(), PlaneT()} {}
jxl::Image3<short>::Image3()
Line
Count
Source
219
3.05k
  Image3() : planes_{PlaneT(), PlaneT(), PlaneT()} {}
220
221
  // Copy construction/assignment is forbidden to avoid inadvertent copies,
222
  // which can be very expensive. Use CopyImageTo instead.
223
  Image3(const Image3& other) = delete;
224
  Image3& operator=(const Image3& other) = delete;
225
226
653k
  Image3(Image3&& other) noexcept {
227
2.61M
    for (size_t i = 0; i < kNumPlanes; i++) {
228
1.96M
      planes_[i] = std::move(other.planes_[i]);
229
1.96M
    }
230
653k
  }
jxl::Image3<float>::Image3(jxl::Image3<float>&&)
Line
Count
Source
226
621k
  Image3(Image3&& other) noexcept {
227
2.48M
    for (size_t i = 0; i < kNumPlanes; i++) {
228
1.86M
      planes_[i] = std::move(other.planes_[i]);
229
1.86M
    }
230
621k
  }
jxl::Image3<int>::Image3(jxl::Image3<int>&&)
Line
Count
Source
226
26.2k
  Image3(Image3&& other) noexcept {
227
105k
    for (size_t i = 0; i < kNumPlanes; i++) {
228
78.8k
      planes_[i] = std::move(other.planes_[i]);
229
78.8k
    }
230
26.2k
  }
jxl::Image3<short>::Image3(jxl::Image3<short>&&)
Line
Count
Source
226
6.11k
  Image3(Image3&& other) noexcept {
227
24.4k
    for (size_t i = 0; i < kNumPlanes; i++) {
228
18.3k
      planes_[i] = std::move(other.planes_[i]);
229
18.3k
    }
230
6.11k
  }
Unexecuted instantiation: jxl::Image3<unsigned char>::Image3(jxl::Image3<unsigned char>&&)
231
517k
  Image3& operator=(Image3&& other) noexcept {
232
2.06M
    for (size_t i = 0; i < kNumPlanes; i++) {
233
1.55M
      planes_[i] = std::move(other.planes_[i]);
234
1.55M
    }
235
517k
    return *this;
236
517k
  }
jxl::Image3<float>::operator=(jxl::Image3<float>&&)
Line
Count
Source
231
500k
  Image3& operator=(Image3&& other) noexcept {
232
2.00M
    for (size_t i = 0; i < kNumPlanes; i++) {
233
1.50M
      planes_[i] = std::move(other.planes_[i]);
234
1.50M
    }
235
500k
    return *this;
236
500k
  }
jxl::Image3<int>::operator=(jxl::Image3<int>&&)
Line
Count
Source
231
13.1k
  Image3& operator=(Image3&& other) noexcept {
232
52.5k
    for (size_t i = 0; i < kNumPlanes; i++) {
233
39.4k
      planes_[i] = std::move(other.planes_[i]);
234
39.4k
    }
235
13.1k
    return *this;
236
13.1k
  }
jxl::Image3<short>::operator=(jxl::Image3<short>&&)
Line
Count
Source
231
3.05k
  Image3& operator=(Image3&& other) noexcept {
232
12.2k
    for (size_t i = 0; i < kNumPlanes; i++) {
233
9.16k
      planes_[i] = std::move(other.planes_[i]);
234
9.16k
    }
235
3.05k
    return *this;
236
3.05k
  }
237
238
  static StatusOr<Image3> Create(JxlMemoryManager* memory_manager,
239
313k
                                 const size_t xsize, const size_t ysize) {
240
313k
    JXL_ASSIGN_OR_RETURN(PlaneT plane0,
241
313k
                         PlaneT::Create(memory_manager, xsize, ysize));
242
313k
    JXL_ASSIGN_OR_RETURN(PlaneT plane1,
243
313k
                         PlaneT::Create(memory_manager, xsize, ysize));
244
313k
    JXL_ASSIGN_OR_RETURN(PlaneT plane2,
245
313k
                         PlaneT::Create(memory_manager, xsize, ysize));
246
313k
    return Image3(std::move(plane0), std::move(plane1), std::move(plane2));
247
313k
  }
jxl::Image3<float>::Create(JxlMemoryManagerStruct*, unsigned long, unsigned long)
Line
Count
Source
239
297k
                                 const size_t xsize, const size_t ysize) {
240
297k
    JXL_ASSIGN_OR_RETURN(PlaneT plane0,
241
297k
                         PlaneT::Create(memory_manager, xsize, ysize));
242
297k
    JXL_ASSIGN_OR_RETURN(PlaneT plane1,
243
297k
                         PlaneT::Create(memory_manager, xsize, ysize));
244
297k
    JXL_ASSIGN_OR_RETURN(PlaneT plane2,
245
297k
                         PlaneT::Create(memory_manager, xsize, ysize));
246
297k
    return Image3(std::move(plane0), std::move(plane1), std::move(plane2));
247
297k
  }
jxl::Image3<int>::Create(JxlMemoryManagerStruct*, unsigned long, unsigned long)
Line
Count
Source
239
13.1k
                                 const size_t xsize, const size_t ysize) {
240
13.1k
    JXL_ASSIGN_OR_RETURN(PlaneT plane0,
241
13.1k
                         PlaneT::Create(memory_manager, xsize, ysize));
242
13.1k
    JXL_ASSIGN_OR_RETURN(PlaneT plane1,
243
13.1k
                         PlaneT::Create(memory_manager, xsize, ysize));
244
13.1k
    JXL_ASSIGN_OR_RETURN(PlaneT plane2,
245
13.1k
                         PlaneT::Create(memory_manager, xsize, ysize));
246
13.1k
    return Image3(std::move(plane0), std::move(plane1), std::move(plane2));
247
13.1k
  }
jxl::Image3<short>::Create(JxlMemoryManagerStruct*, unsigned long, unsigned long)
Line
Count
Source
239
3.05k
                                 const size_t xsize, const size_t ysize) {
240
3.05k
    JXL_ASSIGN_OR_RETURN(PlaneT plane0,
241
3.05k
                         PlaneT::Create(memory_manager, xsize, ysize));
242
3.05k
    JXL_ASSIGN_OR_RETURN(PlaneT plane1,
243
3.05k
                         PlaneT::Create(memory_manager, xsize, ysize));
244
3.05k
    JXL_ASSIGN_OR_RETURN(PlaneT plane2,
245
3.05k
                         PlaneT::Create(memory_manager, xsize, ysize));
246
3.05k
    return Image3(std::move(plane0), std::move(plane1), std::move(plane2));
247
3.05k
  }
Unexecuted instantiation: jxl::Image3<unsigned char>::Create(JxlMemoryManagerStruct*, unsigned long, unsigned long)
248
249
  // Returns row pointer; usage: PlaneRow(idx_plane, y)[x] = val.
250
46.8M
  JXL_INLINE T* PlaneRow(const size_t c, const size_t y) {
251
    // Custom implementation instead of calling planes_[c].Row ensures only a
252
    // single multiplication is needed for PlaneRow(0..2, y).
253
46.8M
    PlaneRowBoundsCheck(c, y);
254
46.8M
    const size_t row_offset = y * planes_[0].bytes_per_row();
255
46.8M
    void* row = planes_[c].bytes() + row_offset;
256
46.8M
    return static_cast<T * JXL_RESTRICT>(JXL_ASSUME_ALIGNED(row, 64));
257
46.8M
  }
jxl::Image3<int>::PlaneRow(unsigned long, unsigned long)
Line
Count
Source
250
1.42M
  JXL_INLINE T* PlaneRow(const size_t c, const size_t y) {
251
    // Custom implementation instead of calling planes_[c].Row ensures only a
252
    // single multiplication is needed for PlaneRow(0..2, y).
253
1.42M
    PlaneRowBoundsCheck(c, y);
254
1.42M
    const size_t row_offset = y * planes_[0].bytes_per_row();
255
1.42M
    void* row = planes_[c].bytes() + row_offset;
256
1.42M
    return static_cast<T * JXL_RESTRICT>(JXL_ASSUME_ALIGNED(row, 64));
257
1.42M
  }
jxl::Image3<float>::PlaneRow(unsigned long, unsigned long)
Line
Count
Source
250
45.3M
  JXL_INLINE T* PlaneRow(const size_t c, const size_t y) {
251
    // Custom implementation instead of calling planes_[c].Row ensures only a
252
    // single multiplication is needed for PlaneRow(0..2, y).
253
45.3M
    PlaneRowBoundsCheck(c, y);
254
45.3M
    const size_t row_offset = y * planes_[0].bytes_per_row();
255
45.3M
    void* row = planes_[c].bytes() + row_offset;
256
45.3M
    return static_cast<T * JXL_RESTRICT>(JXL_ASSUME_ALIGNED(row, 64));
257
45.3M
  }
jxl::Image3<short>::PlaneRow(unsigned long, unsigned long)
Line
Count
Source
250
189
  JXL_INLINE T* PlaneRow(const size_t c, const size_t y) {
251
    // Custom implementation instead of calling planes_[c].Row ensures only a
252
    // single multiplication is needed for PlaneRow(0..2, y).
253
189
    PlaneRowBoundsCheck(c, y);
254
189
    const size_t row_offset = y * planes_[0].bytes_per_row();
255
189
    void* row = planes_[c].bytes() + row_offset;
256
189
    return static_cast<T * JXL_RESTRICT>(JXL_ASSUME_ALIGNED(row, 64));
257
189
  }
Unexecuted instantiation: jxl::Image3<unsigned char>::PlaneRow(unsigned long, unsigned long)
258
259
  // Returns const row pointer; usage: val = PlaneRow(idx_plane, y)[x].
260
1.00G
  JXL_INLINE const T* PlaneRow(const size_t c, const size_t y) const {
261
1.00G
    PlaneRowBoundsCheck(c, y);
262
1.00G
    const size_t row_offset = y * planes_[0].bytes_per_row();
263
1.00G
    const void* row = planes_[c].bytes() + row_offset;
264
1.00G
    return static_cast<const T * JXL_RESTRICT>(JXL_ASSUME_ALIGNED(row, 64));
265
1.00G
  }
jxl::Image3<int>::PlaneRow(unsigned long, unsigned long) const
Line
Count
Source
260
1.22M
  JXL_INLINE const T* PlaneRow(const size_t c, const size_t y) const {
261
1.22M
    PlaneRowBoundsCheck(c, y);
262
1.22M
    const size_t row_offset = y * planes_[0].bytes_per_row();
263
1.22M
    const void* row = planes_[c].bytes() + row_offset;
264
1.22M
    return static_cast<const T * JXL_RESTRICT>(JXL_ASSUME_ALIGNED(row, 64));
265
1.22M
  }
jxl::Image3<float>::PlaneRow(unsigned long, unsigned long) const
Line
Count
Source
260
1.00G
  JXL_INLINE const T* PlaneRow(const size_t c, const size_t y) const {
261
1.00G
    PlaneRowBoundsCheck(c, y);
262
1.00G
    const size_t row_offset = y * planes_[0].bytes_per_row();
263
1.00G
    const void* row = planes_[c].bytes() + row_offset;
264
1.00G
    return static_cast<const T * JXL_RESTRICT>(JXL_ASSUME_ALIGNED(row, 64));
265
1.00G
  }
Unexecuted instantiation: jxl::Image3<short>::PlaneRow(unsigned long, unsigned long) const
Unexecuted instantiation: jxl::Image3<unsigned char>::PlaneRow(unsigned long, unsigned long) const
266
267
  // Returns const row pointer, even if called from a non-const Image3.
268
1.00G
  JXL_INLINE const T* ConstPlaneRow(const size_t c, const size_t y) const {
269
1.00G
    PlaneRowBoundsCheck(c, y);
270
1.00G
    return PlaneRow(c, y);
271
1.00G
  }
jxl::Image3<float>::ConstPlaneRow(unsigned long, unsigned long) const
Line
Count
Source
268
1.00G
  JXL_INLINE const T* ConstPlaneRow(const size_t c, const size_t y) const {
269
1.00G
    PlaneRowBoundsCheck(c, y);
270
1.00G
    return PlaneRow(c, y);
271
1.00G
  }
jxl::Image3<int>::ConstPlaneRow(unsigned long, unsigned long) const
Line
Count
Source
268
1.19M
  JXL_INLINE const T* ConstPlaneRow(const size_t c, const size_t y) const {
269
1.19M
    PlaneRowBoundsCheck(c, y);
270
1.19M
    return PlaneRow(c, y);
271
1.19M
  }
Unexecuted instantiation: jxl::Image3<unsigned char>::ConstPlaneRow(unsigned long, unsigned long) const
272
273
637k
  JXL_INLINE const PlaneT& Plane(size_t idx) const { return planes_[idx]; }
274
275
170k
  JXL_INLINE PlaneT& Plane(size_t idx) { return planes_[idx]; }
jxl::Image3<float>::Plane(unsigned long)
Line
Count
Source
275
170k
  JXL_INLINE PlaneT& Plane(size_t idx) { return planes_[idx]; }
Unexecuted instantiation: jxl::Image3<int>::Plane(unsigned long)
Unexecuted instantiation: jxl::Image3<short>::Plane(unsigned long)
276
277
6.72k
  void Swap(Image3& other) {
278
26.9k
    for (size_t c = 0; c < 3; ++c) {
279
20.1k
      other.planes_[c].Swap(planes_[c]);
280
20.1k
    }
281
6.72k
  }
282
283
  // Useful for pre-allocating image with some padding for alignment purposes
284
  // and later reporting the actual valid dimensions. May also be used to
285
  // un-shrink the image. Caller is responsible for ensuring xsize/ysize are <=
286
  // the original dimensions.
287
17.6k
  Status ShrinkTo(const size_t xsize, const size_t ysize) {
288
52.8k
    for (PlaneT& plane : planes_) {
289
52.8k
      JXL_RETURN_IF_ERROR(plane.ShrinkTo(xsize, ysize));
290
52.8k
    }
291
17.6k
    return true;
292
17.6k
  }
293
294
  // Sizes of all three images are guaranteed to be equal.
295
41.4k
  JXL_INLINE JxlMemoryManager* memory_manager() const {
296
41.4k
    return planes_[0].memory_manager();
297
41.4k
  }
jxl::Image3<float>::memory_manager() const
Line
Count
Source
295
41.4k
  JXL_INLINE JxlMemoryManager* memory_manager() const {
296
41.4k
    return planes_[0].memory_manager();
297
41.4k
  }
Unexecuted instantiation: jxl::Image3<unsigned char>::memory_manager() const
298
117M
  JXL_INLINE size_t xsize() const { return planes_[0].xsize(); }
jxl::Image3<float>::xsize() const
Line
Count
Source
298
117M
  JXL_INLINE size_t xsize() const { return planes_[0].xsize(); }
jxl::Image3<int>::xsize() const
Line
Count
Source
298
60.7k
  JXL_INLINE size_t xsize() const { return planes_[0].xsize(); }
jxl::Image3<short>::xsize() const
Line
Count
Source
298
5.94k
  JXL_INLINE size_t xsize() const { return planes_[0].xsize(); }
Unexecuted instantiation: jxl::Image3<unsigned char>::xsize() const
299
2.08G
  JXL_INLINE size_t ysize() const { return planes_[0].ysize(); }
jxl::Image3<float>::ysize() const
Line
Count
Source
299
2.08G
  JXL_INLINE size_t ysize() const { return planes_[0].ysize(); }
jxl::Image3<int>::ysize() const
Line
Count
Source
299
3.84M
  JXL_INLINE size_t ysize() const { return planes_[0].ysize(); }
jxl::Image3<short>::ysize() const
Line
Count
Source
299
387
  JXL_INLINE size_t ysize() const { return planes_[0].ysize(); }
Unexecuted instantiation: jxl::Image3<unsigned char>::ysize() const
300
  // Returns offset [bytes] from one row to the next row of the same plane.
301
  // WARNING: this must NOT be used to determine xsize, nor for copying rows -
302
  // the valid xsize may be much less.
303
  JXL_INLINE size_t bytes_per_row() const { return planes_[0].bytes_per_row(); }
304
  // Returns number of pixels (some of which are padding) per row. Useful for
305
  // computing other rows via pointer arithmetic. WARNING: this must NOT be used
306
  // to determine xsize.
307
1.80M
  JXL_INLINE ptrdiff_t PixelsPerRow() const { return planes_[0].PixelsPerRow(); }
jxl::Image3<int>::PixelsPerRow() const
Line
Count
Source
307
34.0k
  JXL_INLINE ptrdiff_t PixelsPerRow() const { return planes_[0].PixelsPerRow(); }
jxl::Image3<float>::PixelsPerRow() const
Line
Count
Source
307
1.77M
  JXL_INLINE ptrdiff_t PixelsPerRow() const { return planes_[0].PixelsPerRow(); }
Unexecuted instantiation: jxl::Image3<short>::PixelsPerRow() const
308
309
 private:
310
313k
  Image3(PlaneT&& plane0, PlaneT&& plane1, PlaneT&& plane2) {
311
313k
    planes_[0] = std::move(plane0);
312
313k
    planes_[1] = std::move(plane1);
313
313k
    planes_[2] = std::move(plane2);
314
313k
  }
jxl::Image3<float>::Image3(jxl::Plane<float>&&, jxl::Plane<float>&&, jxl::Plane<float>&&)
Line
Count
Source
310
297k
  Image3(PlaneT&& plane0, PlaneT&& plane1, PlaneT&& plane2) {
311
297k
    planes_[0] = std::move(plane0);
312
297k
    planes_[1] = std::move(plane1);
313
297k
    planes_[2] = std::move(plane2);
314
297k
  }
jxl::Image3<int>::Image3(jxl::Plane<int>&&, jxl::Plane<int>&&, jxl::Plane<int>&&)
Line
Count
Source
310
13.1k
  Image3(PlaneT&& plane0, PlaneT&& plane1, PlaneT&& plane2) {
311
13.1k
    planes_[0] = std::move(plane0);
312
13.1k
    planes_[1] = std::move(plane1);
313
13.1k
    planes_[2] = std::move(plane2);
314
13.1k
  }
jxl::Image3<short>::Image3(jxl::Plane<short>&&, jxl::Plane<short>&&, jxl::Plane<short>&&)
Line
Count
Source
310
3.05k
  Image3(PlaneT&& plane0, PlaneT&& plane1, PlaneT&& plane2) {
311
3.05k
    planes_[0] = std::move(plane0);
312
3.05k
    planes_[1] = std::move(plane1);
313
3.05k
    planes_[2] = std::move(plane2);
314
3.05k
  }
Unexecuted instantiation: jxl::Image3<unsigned char>::Image3(jxl::Plane<unsigned char>&&, jxl::Plane<unsigned char>&&, jxl::Plane<unsigned char>&&)
315
316
2.06G
  void PlaneRowBoundsCheck(const size_t c, const size_t y) const {
317
2.06G
    JXL_DASSERT(c < kNumPlanes && y < ysize());
318
2.06G
  }
jxl::Image3<int>::PlaneRowBoundsCheck(unsigned long, unsigned long) const
Line
Count
Source
316
3.84M
  void PlaneRowBoundsCheck(const size_t c, const size_t y) const {
317
3.84M
    JXL_DASSERT(c < kNumPlanes && y < ysize());
318
3.84M
  }
jxl::Image3<float>::PlaneRowBoundsCheck(unsigned long, unsigned long) const
Line
Count
Source
316
2.06G
  void PlaneRowBoundsCheck(const size_t c, const size_t y) const {
317
2.06G
    JXL_DASSERT(c < kNumPlanes && y < ysize());
318
2.06G
  }
jxl::Image3<short>::PlaneRowBoundsCheck(unsigned long, unsigned long) const
Line
Count
Source
316
189
  void PlaneRowBoundsCheck(const size_t c, const size_t y) const {
317
189
    JXL_DASSERT(c < kNumPlanes && y < ysize());
318
189
  }
Unexecuted instantiation: jxl::Image3<unsigned char>::PlaneRowBoundsCheck(unsigned long, unsigned long) const
319
320
  PlaneT planes_[kNumPlanes];
321
};
322
323
using Image3B = Image3<uint8_t>;
324
using Image3S = Image3<int16_t>;
325
using Image3U = Image3<uint16_t>;
326
using Image3I = Image3<int32_t>;
327
using Image3F = Image3<float>;
328
using Image3D = Image3<double>;
329
330
}  // namespace jxl
331
332
#endif  // LIB_JXL_IMAGE_H_