1 /*
2 * Copyright 2018 Google Inc.
3 *
4 * Use of this source code is governed by a BSD-style license that can be
5 * found in the LICENSE file.
6 */
7
8 #include "src/core/SkColorSpacePriv.h"
9 #include "src/core/SkColorSpaceXformSteps.h"
10 #include "tests/Test.h"
11
DEF_TEST(SkColorSpaceXformSteps,r)12 DEF_TEST(SkColorSpaceXformSteps, r) {
13 auto srgb = SkColorSpace::MakeSRGB(),
14 adobe = SkColorSpace::MakeRGB(SkNamedTransferFn::k2Dot2, SkNamedGamut::kAdobeRGB),
15 srgb22 = SkColorSpace::MakeRGB(SkNamedTransferFn::k2Dot2, SkNamedGamut::kSRGB),
16 srgb1 = srgb ->makeLinearGamma(),
17 adobe1 = adobe->makeLinearGamma();
18
19 auto premul = kPremul_SkAlphaType,
20 opaque = kOpaque_SkAlphaType,
21 unpremul = kUnpremul_SkAlphaType;
22
23 struct Test {
24 sk_sp<SkColorSpace> src, dst;
25 SkAlphaType srcAT, dstAT;
26
27 bool unpremul;
28 bool linearize;
29 bool gamut_transform;
30 bool encode;
31 bool premul;
32
33 };
34 Test tests[] = {
35 // The general case is converting between two color spaces with different gamuts
36 // and different transfer functions. There's no optimization possible here.
37 { adobe, srgb, premul, premul,
38 true, // src is encoded as f(s)*a,a, so we unpremul to f(s),a before linearizing.
39 true, // linearize to s,a
40 true, // transform s to dst gamut, s'
41 true, // encode with dst transfer function, g(s'), a
42 true, // premul to g(s')*a, a
43 },
44 // All the same going the other direction.
45 { srgb, adobe, premul, premul, true,true,true,true,true },
46
47 // If the src alpha type is unpremul, we'll not need that initial unpremul step.
48 { adobe, srgb, unpremul, premul, false,true,true,true,true },
49 { srgb, adobe, unpremul, premul, false,true,true,true,true },
50
51 // If opaque, we need neither the initial unpremul, nor the premul later.
52 { adobe, srgb, opaque, premul, false,true,true,true,false },
53 { srgb, adobe, opaque, premul, false,true,true,true,false },
54
55
56 // Now let's go between sRGB and sRGB with a 2.2 gamma, the gamut staying the same.
57 { srgb, srgb22, premul, premul,
58 true, // we need to linearize, so we need to unpremul
59 true, // we need to encode to 2.2 gamma, so we need to get linear
60 false, // no need to change gamut
61 true, // linear -> gamma 2.2
62 true, // premul going into the blend
63 },
64 // Same sort of logic in the other direction.
65 { srgb22, srgb, premul, premul, true,true,false,true,true },
66
67 // As in the general case, when we change the alpha type unpremul and premul steps drop out.
68 { srgb, srgb22, unpremul, premul, false,true,false,true,true },
69 { srgb22, srgb, unpremul, premul, false,true,false,true,true },
70 { srgb, srgb22, opaque, premul, false,true,false,true,false },
71 { srgb22, srgb, opaque, premul, false,true,false,true,false },
72
73 // Let's look at the special case of completely matching color spaces.
74 // We should be ready to go into the blend without any fuss.
75 { srgb, srgb, premul, premul, false,false,false,false,false },
76 { srgb, srgb, unpremul, premul, false,false,false,false,true },
77 { srgb, srgb, opaque, premul, false,false,false,false,false },
78
79 // We can drop out the linearize step when the source is already linear.
80 { srgb1, adobe, premul, premul, true,false,true,true,true },
81 { srgb1, srgb, premul, premul, true,false,false,true,true },
82 // And we can drop the encode step when the destination is linear.
83 { adobe, srgb1, premul, premul, true,true,true,false,true },
84 { srgb, srgb1, premul, premul, true,true,false,false,true },
85
86 // Here's an interesting case where only gamut transform is needed.
87 { adobe1, srgb1, premul, premul, false,false,true,false,false },
88 { adobe1, srgb1, opaque, premul, false,false,true,false,false },
89 { adobe1, srgb1, unpremul, premul, false,false,true,false, true },
90
91 // Just finishing up with something to produce each other possible output.
92 // Nothing terribly interesting in these eight.
93 { srgb, srgb1, opaque, premul, false, true,false,false,false },
94 { srgb, srgb1, unpremul, premul, false, true,false,false, true },
95 { srgb, adobe1, opaque, premul, false, true, true,false,false },
96 { srgb, adobe1, unpremul, premul, false, true, true,false, true },
97 { srgb1, srgb, opaque, premul, false,false,false, true,false },
98 { srgb1, srgb, unpremul, premul, false,false,false, true, true },
99 { srgb1, adobe, opaque, premul, false,false, true, true,false },
100 { srgb1, adobe, unpremul, premul, false,false, true, true, true },
101
102 // Now test non-premul outputs.
103 { srgb , srgb , premul, unpremul, true,false,false,false,false },
104 { srgb , srgb1 , premul, unpremul, true, true,false,false,false },
105 { srgb1, adobe1, premul, unpremul, true,false, true,false,false },
106 { srgb , adobe1, premul, unpremul, true, true, true,false,false },
107 { srgb1, srgb , premul, unpremul, true,false,false, true,false },
108 { srgb , srgb22, premul, unpremul, true, true,false, true,false },
109 { srgb1, adobe , premul, unpremul, true,false, true, true,false },
110 { srgb , adobe , premul, unpremul, true, true, true, true,false },
111
112 // Opaque outputs are treated as the same alpha type as the source input.
113 // TODO: we'd really like to have a good way of explaining why we think this is useful.
114 { srgb , srgb , premul, opaque, false,false,false,false,false },
115 { srgb , srgb1 , premul, opaque, true, true,false,false, true },
116 { srgb1, adobe1, premul, opaque, false,false, true,false,false },
117 { srgb , adobe1, premul, opaque, true, true, true,false, true },
118 { srgb1, srgb , premul, opaque, true,false,false, true, true },
119 { srgb , srgb22, premul, opaque, true, true,false, true, true },
120 { srgb1, adobe , premul, opaque, true,false, true, true, true },
121 { srgb , adobe , premul, opaque, true, true, true, true, true },
122
123 { srgb , srgb , unpremul, opaque, false,false,false,false,false },
124 { srgb , srgb1 , unpremul, opaque, false, true,false,false,false },
125 { srgb1, adobe1, unpremul, opaque, false,false, true,false,false },
126 { srgb , adobe1, unpremul, opaque, false, true, true,false,false },
127 { srgb1, srgb , unpremul, opaque, false,false,false, true,false },
128 { srgb , srgb22, unpremul, opaque, false, true,false, true,false },
129 { srgb1, adobe , unpremul, opaque, false,false, true, true,false },
130 { srgb , adobe , unpremul, opaque, false, true, true, true,false },
131 };
132
133 uint32_t tested = 0x00000000;
134 for (const Test& t : tests) {
135 SkColorSpaceXformSteps steps{t.src.get(), t.srcAT,
136 t.dst.get(), t.dstAT};
137 REPORTER_ASSERT(r, steps.flags.unpremul == t.unpremul);
138 REPORTER_ASSERT(r, steps.flags.linearize == t.linearize);
139 REPORTER_ASSERT(r, steps.flags.gamut_transform == t.gamut_transform);
140 REPORTER_ASSERT(r, steps.flags.encode == t.encode);
141 REPORTER_ASSERT(r, steps.flags.premul == t.premul);
142
143 uint32_t bits = (uint32_t)t.unpremul << 0
144 | (uint32_t)t.linearize << 1
145 | (uint32_t)t.gamut_transform << 2
146 | (uint32_t)t.encode << 3
147 | (uint32_t)t.premul << 4;
148 tested |= (1<<bits);
149 }
150
151 // We'll check our test cases cover all 2^5 == 32 possible outputs.
152 for (uint32_t t = 0; t < 32; t++) {
153 if (tested & (1<<t)) {
154 continue;
155 }
156
157 // There are a couple impossible outputs, so consider those bits tested.
158 //
159 // Unpremul then premul should be optimized away to a noop, so 0b10001 isn't possible.
160 // A gamut transform in the middle is fine too, so 0b10101 isn't possible either.
161 if (t == 0b10001 || t == 0b10101) {
162 continue;
163 }
164
165 ERRORF(r, "{ xxx, yyy, at, %s,%s,%s,%s,%s }, not covered",
166 (t& 1) ? " true" : "false",
167 (t& 2) ? " true" : "false",
168 (t& 4) ? " true" : "false",
169 (t& 8) ? " true" : "false",
170 (t&16) ? " true" : "false");
171 }
172 }
173