• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
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