Coverage Report

Created: 2026-01-20 07:37

next uncovered line (L), next uncovered region (R), next uncovered branch (B)
/src/libjxl/lib/jxl/alpha.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/alpha.h"
7
8
#include <algorithm>
9
#include <cstring>
10
11
#include "lib/jxl/base/common.h"
12
#include "lib/jxl/base/compiler_specific.h"
13
14
namespace jxl {
15
16
5.14M
static float Clamp(float x) { return jxl::Clamp1(x, 0.0f, 1.0f); }
17
18
void PerformAlphaBlending(const AlphaBlendingInputLayer& bg,
19
                          const AlphaBlendingInputLayer& fg,
20
                          const AlphaBlendingOutput& out, size_t num_pixels,
21
1.36k
                          bool alpha_is_premultiplied, bool clamp) {
22
1.36k
  if (alpha_is_premultiplied) {
23
18.3k
    for (size_t x = 0; x < num_pixels; ++x) {
24
17.9k
      float fga = clamp ? Clamp(fg.a[x]) : fg.a[x];
25
17.9k
      out.r[x] = (fg.r[x] + bg.r[x] * (1.f - fga));
26
17.9k
      out.g[x] = (fg.g[x] + bg.g[x] * (1.f - fga));
27
17.9k
      out.b[x] = (fg.b[x] + bg.b[x] * (1.f - fga));
28
17.9k
      out.a[x] = (1.f - (1.f - fga) * (1.f - bg.a[x]));
29
17.9k
    }
30
1.01k
  } else {
31
63.5k
    for (size_t x = 0; x < num_pixels; ++x) {
32
62.5k
      float fga = clamp ? Clamp(fg.a[x]) : fg.a[x];
33
62.5k
      const float new_a = 1.f - (1.f - fga) * (1.f - bg.a[x]);
34
62.5k
      const float rnew_a = (new_a > 0 ? 1.f / new_a : 0.f);
35
62.5k
      out.r[x] = (fg.r[x] * fga + bg.r[x] * bg.a[x] * (1.f - fga)) * rnew_a;
36
62.5k
      out.g[x] = (fg.g[x] * fga + bg.g[x] * bg.a[x] * (1.f - fga)) * rnew_a;
37
62.5k
      out.b[x] = (fg.b[x] * fga + bg.b[x] * bg.a[x] * (1.f - fga)) * rnew_a;
38
62.5k
      out.a[x] = new_a;
39
62.5k
    }
40
1.01k
  }
41
1.36k
}
42
void PerformAlphaBlending(const float* bg, const float* bga, const float* fg,
43
                          const float* fga, float* out, size_t num_pixels,
44
21.0k
                          bool alpha_is_premultiplied, bool clamp) {
45
21.0k
  if (bg == bga && fg == fga) {
46
797k
    for (size_t x = 0; x < num_pixels; ++x) {
47
793k
      float fa = clamp ? Clamp(fga[x]) : fga[x];
48
793k
      out[x] = (1.f - (1.f - fa) * (1.f - bga[x]));
49
793k
    }
50
16.4k
  } else {
51
16.4k
    if (alpha_is_premultiplied) {
52
26
      for (size_t x = 0; x < num_pixels; ++x) {
53
13
        float fa = clamp ? Clamp(fga[x]) : fga[x];
54
13
        out[x] = (fg[x] + bg[x] * (1.f - fa));
55
13
      }
56
16.3k
    } else {
57
3.18M
      for (size_t x = 0; x < num_pixels; ++x) {
58
3.16M
        float fa = clamp ? Clamp(fga[x]) : fga[x];
59
3.16M
        const float new_a = 1.f - (1.f - fa) * (1.f - bga[x]);
60
3.16M
        const float rnew_a = (new_a > 0 ? 1.f / new_a : 0.f);
61
3.16M
        out[x] = (fg[x] * fa + bg[x] * bga[x] * (1.f - fa)) * rnew_a;
62
3.16M
      }
63
16.3k
    }
64
16.4k
  }
65
21.0k
}
66
67
void PerformAlphaWeightedAdd(const float* bg, const float* fg, const float* fga,
68
13.2k
                             float* out, size_t num_pixels, bool clamp) {
69
13.2k
  if (fg == fga) {
70
8.16k
    memcpy(out, bg, num_pixels * sizeof(*out));
71
8.16k
  } else if (clamp) {
72
839k
    for (size_t x = 0; x < num_pixels; ++x) {
73
835k
      out[x] = bg[x] + fg[x] * Clamp(fga[x]);
74
835k
    }
75
3.66k
  } else {
76
244k
    for (size_t x = 0; x < num_pixels; ++x) {
77
242k
      out[x] = bg[x] + fg[x] * fga[x];
78
242k
    }
79
1.43k
  }
80
13.2k
}
81
82
void PerformMulBlending(const float* bg, const float* fg, float* out,
83
21.4k
                        size_t num_pixels, bool clamp) {
84
21.4k
  if (clamp) {
85
1.44M
    for (size_t x = 0; x < num_pixels; ++x) {
86
1.43M
      out[x] = bg[x] * Clamp(fg[x]);
87
1.43M
    }
88
13.6k
  } else {
89
1.06M
    for (size_t x = 0; x < num_pixels; ++x) {
90
1.05M
      out[x] = bg[x] * fg[x];
91
1.05M
    }
92
7.80k
  }
93
21.4k
}
94
95
void PremultiplyAlpha(float* JXL_RESTRICT r, float* JXL_RESTRICT g,
96
                      float* JXL_RESTRICT b, const float* JXL_RESTRICT a,
97
0
                      size_t num_pixels) {
98
0
  for (size_t x = 0; x < num_pixels; ++x) {
99
0
    const float multiplier = std::max(kSmallAlpha, a[x]);
100
0
    r[x] *= multiplier;
101
0
    g[x] *= multiplier;
102
0
    b[x] *= multiplier;
103
0
  }
104
0
}
105
106
void UnpremultiplyAlpha(float* JXL_RESTRICT r, float* JXL_RESTRICT g,
107
                        float* JXL_RESTRICT b, const float* JXL_RESTRICT a,
108
0
                        size_t num_pixels) {
109
0
  for (size_t x = 0; x < num_pixels; ++x) {
110
0
    const float multiplier = 1.f / std::max(kSmallAlpha, a[x]);
111
0
    r[x] *= multiplier;
112
0
    g[x] *= multiplier;
113
0
    b[x] *= multiplier;
114
0
  }
115
0
}
116
117
}  // namespace jxl