Coverage Report

Created: 2025-11-16 07:22

next uncovered line (L), next uncovered region (R), next uncovered branch (B)
/src/libjxl/lib/jxl/headers.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/headers.h"
7
8
#include <cstddef>
9
#include <cstdint>
10
#include <limits>
11
12
#include "lib/jxl/base/compiler_specific.h"
13
#include "lib/jxl/base/status.h"
14
#include "lib/jxl/dec_bit_reader.h"
15
#include "lib/jxl/field_encodings.h"
16
#include "lib/jxl/fields.h"
17
#include "lib/jxl/frame_dimensions.h"
18
19
namespace jxl {
20
namespace {
21
22
struct Rational {
23
  constexpr explicit Rational(uint32_t num, uint32_t den)
24
0
      : num(num), den(den) {}
25
26
  // Returns floor(multiplicand * rational).
27
1.19M
  constexpr uint32_t MulTruncate(uint32_t multiplicand) const {
28
1.19M
    return static_cast<uint64_t>(multiplicand) * num / den;
29
1.19M
  }
30
31
  uint32_t num;
32
  uint32_t den;
33
};
34
35
1.19M
Rational FixedAspectRatios(uint32_t ratio) {
36
1.19M
  JXL_DASSERT(0 != ratio && ratio < 8);
37
  // Other candidates: 5/4, 7/5, 14/9, 16/10, 5/3, 21/9, 12/5
38
1.19M
  constexpr Rational kRatios[7] = {Rational(1, 1),    // square
39
1.19M
                                   Rational(12, 10),  //
40
1.19M
                                   Rational(4, 3),    // camera
41
1.19M
                                   Rational(3, 2),    // mobile camera
42
1.19M
                                   Rational(16, 9),   // camera/display
43
1.19M
                                   Rational(5, 4),    //
44
1.19M
                                   Rational(2, 1)};   //
45
1.19M
  return kRatios[ratio - 1];
46
1.19M
}
47
48
3.58k
uint32_t FindAspectRatio(uint32_t xsize, uint32_t ysize) {
49
26.5k
  for (uint32_t r = 1; r < 8; ++r) {
50
23.4k
    if (xsize == FixedAspectRatios(r).MulTruncate(ysize)) {
51
487
      return r;
52
487
    }
53
23.4k
  }
54
3.10k
  return 0;  // Must send xsize instead
55
3.58k
}
56
57
}  // namespace
58
59
1.98M
size_t SizeHeader::xsize() const {
60
1.98M
  if (ratio_ != 0) {
61
1.16M
    return FixedAspectRatios(ratio_).MulTruncate(
62
1.16M
        static_cast<uint32_t>(ysize()));
63
1.16M
  }
64
827k
  return small_ ? ((xsize_div8_minus_1_ + 1) * 8) : xsize_;
65
1.98M
}
66
67
3.58k
Status SizeHeader::Set(size_t xsize64, size_t ysize64) {
68
3.58k
  constexpr size_t kDimensionCap = std::numeric_limits<uint32_t>::max();
69
3.58k
  if (xsize64 > kDimensionCap || ysize64 > kDimensionCap) {
70
0
    return JXL_FAILURE("Image too large");
71
0
  }
72
3.58k
  const uint32_t xsize32 = static_cast<uint32_t>(xsize64);
73
3.58k
  const uint32_t ysize32 = static_cast<uint32_t>(ysize64);
74
3.58k
  if (xsize64 == 0 || ysize64 == 0) return JXL_FAILURE("Empty image");
75
3.58k
  ratio_ = FindAspectRatio(xsize32, ysize32);
76
3.58k
  small_ = ysize64 <= 256 && (ysize64 % kBlockDim) == 0 &&
77
1.14k
           (ratio_ != 0 || (xsize64 <= 256 && (xsize64 % kBlockDim) == 0));
78
3.58k
  if (small_) {
79
1.05k
    ysize_div8_minus_1_ = ysize32 / 8 - 1;
80
2.53k
  } else {
81
2.53k
    ysize_ = ysize32;
82
2.53k
  }
83
84
3.58k
  if (ratio_ == 0) {
85
3.10k
    if (small_) {
86
731
      xsize_div8_minus_1_ = xsize32 / 8 - 1;
87
2.37k
    } else {
88
2.37k
      xsize_ = xsize32;
89
2.37k
    }
90
3.10k
  }
91
3.58k
  JXL_ENSURE(xsize() == xsize64);
92
3.58k
  JXL_ENSURE(ysize() == ysize64);
93
3.58k
  return true;
94
3.58k
}
95
96
0
Status PreviewHeader::Set(size_t xsize64, size_t ysize64) {
97
0
  const uint32_t xsize32 = static_cast<uint32_t>(xsize64);
98
0
  const uint32_t ysize32 = static_cast<uint32_t>(ysize64);
99
0
  if (xsize64 == 0 || ysize64 == 0) return JXL_FAILURE("Empty preview");
100
0
  div8_ = (xsize64 % kBlockDim) == 0 && (ysize64 % kBlockDim) == 0;
101
0
  if (div8_) {
102
0
    ysize_div8_ = ysize32 / 8;
103
0
  } else {
104
0
    ysize_ = ysize32;
105
0
  }
106
107
0
  ratio_ = FindAspectRatio(xsize32, ysize32);
108
0
  if (ratio_ == 0) {
109
0
    if (div8_) {
110
0
      xsize_div8_ = xsize32 / 8;
111
0
    } else {
112
0
      xsize_ = xsize32;
113
0
    }
114
0
  }
115
0
  JXL_ENSURE(xsize() == xsize64);
116
0
  JXL_ENSURE(ysize() == ysize64);
117
0
  return true;
118
0
}
119
120
8.61k
size_t PreviewHeader::xsize() const {
121
8.61k
  if (ratio_ != 0) {
122
7.08k
    return FixedAspectRatios(ratio_).MulTruncate(
123
7.08k
        static_cast<uint32_t>(ysize()));
124
7.08k
  }
125
1.52k
  return div8_ ? (xsize_div8_ * 8) : xsize_;
126
8.61k
}
127
128
1.94M
SizeHeader::SizeHeader() { Bundle::Init(this); }
129
2.93M
Status SizeHeader::VisitFields(Visitor* JXL_RESTRICT visitor) {
130
2.93M
  JXL_QUIET_RETURN_IF_ERROR(visitor->Bool(false, &small_));
131
132
2.93M
  if (visitor->Conditional(small_)) {
133
2.40M
    JXL_QUIET_RETURN_IF_ERROR(visitor->Bits(5, 0, &ysize_div8_minus_1_));
134
2.40M
  }
135
2.93M
  if (visitor->Conditional(!small_)) {
136
    // (Could still be small, but non-multiple of 8.)
137
2.57M
    JXL_QUIET_RETURN_IF_ERROR(visitor->U32(BitsOffset(9, 1), BitsOffset(13, 1),
138
2.57M
                                           BitsOffset(18, 1), BitsOffset(30, 1),
139
2.57M
                                           1, &ysize_));
140
2.57M
  }
141
142
2.93M
  JXL_QUIET_RETURN_IF_ERROR(visitor->Bits(3, 0, &ratio_));
143
2.93M
  if (visitor->Conditional(ratio_ == 0 && small_)) {
144
2.19M
    JXL_QUIET_RETURN_IF_ERROR(visitor->Bits(5, 0, &xsize_div8_minus_1_));
145
2.19M
  }
146
2.93M
  if (visitor->Conditional(ratio_ == 0 && !small_)) {
147
2.18M
    JXL_QUIET_RETURN_IF_ERROR(visitor->U32(BitsOffset(9, 1), BitsOffset(13, 1),
148
2.18M
                                           BitsOffset(18, 1), BitsOffset(30, 1),
149
2.18M
                                           1, &xsize_));
150
2.18M
  }
151
152
2.93M
  return true;
153
2.93M
}
154
155
1.16M
PreviewHeader::PreviewHeader() { Bundle::Init(this); }
156
1.30M
Status PreviewHeader::VisitFields(Visitor* JXL_RESTRICT visitor) {
157
1.30M
  JXL_QUIET_RETURN_IF_ERROR(visitor->Bool(false, &div8_));
158
159
1.30M
  if (visitor->Conditional(div8_)) {
160
1.27M
    JXL_QUIET_RETURN_IF_ERROR(visitor->U32(Val(16), Val(32), BitsOffset(5, 1),
161
1.27M
                                           BitsOffset(9, 33), 1, &ysize_div8_));
162
1.27M
  }
163
1.30M
  if (visitor->Conditional(!div8_)) {
164
1.29M
    JXL_QUIET_RETURN_IF_ERROR(visitor->U32(BitsOffset(6, 1), BitsOffset(8, 65),
165
1.29M
                                           BitsOffset(10, 321),
166
1.29M
                                           BitsOffset(12, 1345), 1, &ysize_));
167
1.29M
  }
168
169
1.30M
  JXL_QUIET_RETURN_IF_ERROR(visitor->Bits(3, 0, &ratio_));
170
1.30M
  if (visitor->Conditional(ratio_ == 0 && div8_)) {
171
1.26M
    JXL_QUIET_RETURN_IF_ERROR(visitor->U32(Val(16), Val(32), BitsOffset(5, 1),
172
1.26M
                                           BitsOffset(9, 33), 1, &xsize_div8_));
173
1.26M
  }
174
1.30M
  if (visitor->Conditional(ratio_ == 0 && !div8_)) {
175
1.27M
    JXL_QUIET_RETURN_IF_ERROR(visitor->U32(BitsOffset(6, 1), BitsOffset(8, 65),
176
1.27M
                                           BitsOffset(10, 321),
177
1.27M
                                           BitsOffset(12, 1345), 1, &xsize_));
178
1.27M
  }
179
180
1.30M
  return true;
181
1.30M
}
182
183
1.16M
AnimationHeader::AnimationHeader() { Bundle::Init(this); }
184
1.28M
Status AnimationHeader::VisitFields(Visitor* JXL_RESTRICT visitor) {
185
1.28M
  JXL_QUIET_RETURN_IF_ERROR(visitor->U32(Val(100), Val(1000), BitsOffset(10, 1),
186
1.28M
                                         BitsOffset(30, 1), 1, &tps_numerator));
187
1.28M
  JXL_QUIET_RETURN_IF_ERROR(visitor->U32(Val(1), Val(1001), BitsOffset(8, 1),
188
1.28M
                                         BitsOffset(10, 1), 1,
189
1.28M
                                         &tps_denominator));
190
191
1.28M
  JXL_QUIET_RETURN_IF_ERROR(
192
1.28M
      visitor->U32(Val(0), Bits(3), Bits(16), Bits(32), 0, &num_loops));
193
194
1.27M
  JXL_QUIET_RETURN_IF_ERROR(visitor->Bool(false, &have_timecodes));
195
1.27M
  return true;
196
1.27M
}
197
198
Status ReadSizeHeader(BitReader* JXL_RESTRICT reader,
199
0
                      SizeHeader* JXL_RESTRICT size) {
200
0
  return Bundle::Read(reader, size);
201
0
}
202
203
}  // namespace jxl