• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
1 /*
2  * Copyright 2012 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 "tests/Test.h"
9 
10 // This test is specific to the GPU backend.
11 #include "include/core/SkCanvas.h"
12 #include "include/core/SkSurface.h"
13 #include "include/gpu/GrContext.h"
14 #include "include/private/SkTo.h"
15 #include "src/gpu/GrContextPriv.h"
16 #include "src/gpu/GrProxyProvider.h"
17 #include "src/gpu/GrResourceProvider.h"
18 #include "src/gpu/GrSurfaceContext.h"
19 #include "src/gpu/GrSurfaceProxy.h"
20 #include "src/gpu/GrTextureProxy.h"
21 #include "tools/gpu/ProxyUtils.h"
22 
23 // This was made indivisible by 4 to ensure we test setting GL_PACK_ALIGNMENT properly.
24 static const int X_SIZE = 13;
25 static const int Y_SIZE = 13;
26 
validate_alpha_data(skiatest::Reporter * reporter,int w,int h,const uint8_t * actual,size_t actualRowBytes,const uint8_t * expected,SkString extraMsg,GrColorType colorType)27 static void validate_alpha_data(skiatest::Reporter* reporter, int w, int h, const uint8_t* actual,
28                                 size_t actualRowBytes, const uint8_t* expected, SkString extraMsg,
29                                 GrColorType colorType) {
30     for (int y = 0; y < h; ++y) {
31         for (int x = 0; x < w; ++x) {
32             uint8_t a = actual[y * actualRowBytes + x];
33             uint8_t e = expected[y * w + x];
34             if (GrColorType::kRGBA_1010102 == colorType) {
35                 // This config only preserves two bits of alpha
36                 a >>= 6;
37                 e >>= 6;
38             }
39             if (e != a) {
40                 ERRORF(reporter,
41                        "Failed alpha readback. Expected: 0x%02x, Got: 0x%02x at (%d,%d), %s",
42                        e, a, x, y, extraMsg.c_str());
43                 return;
44             }
45         }
46     }
47 }
48 
DEF_GPUTEST_FOR_RENDERING_CONTEXTS(ReadWriteAlpha,reporter,ctxInfo)49 DEF_GPUTEST_FOR_RENDERING_CONTEXTS(ReadWriteAlpha, reporter, ctxInfo) {
50     GrContext* context = ctxInfo.grContext();
51     GrProxyProvider* proxyProvider = context->priv().proxyProvider();
52 
53     unsigned char alphaData[X_SIZE * Y_SIZE];
54 
55     static const int kClearValue = 0x2;
56 
57     bool match;
58     static const size_t kRowBytes[] = {0, X_SIZE, X_SIZE + 1, 2 * X_SIZE - 1};
59     {
60         GrSurfaceDesc desc;
61         desc.fConfig    = kAlpha_8_GrPixelConfig;    // it is a single channel texture
62         desc.fWidth     = X_SIZE;
63         desc.fHeight    = Y_SIZE;
64 
65         // We are initializing the texture with zeros here
66         memset(alphaData, 0, X_SIZE * Y_SIZE);
67 
68         const SkImageInfo ii = SkImageInfo::MakeA8(X_SIZE, Y_SIZE);
69 
70         SkPixmap pixmap(ii, alphaData, ii.minRowBytes());
71         sk_sp<SkImage> alphaImg = SkImage::MakeRasterCopy(pixmap);
72         sk_sp<GrTextureProxy> proxy = proxyProvider->createTextureProxy(
73                 alphaImg, 1, SkBudgeted::kNo, SkBackingFit::kExact);
74         if (!proxy) {
75             ERRORF(reporter, "Could not create alpha texture.");
76             return;
77         }
78         sk_sp<GrSurfaceContext> sContext(context->priv().makeWrappedSurfaceContext(
79                 std::move(proxy), GrColorType::kAlpha_8, kPremul_SkAlphaType));
80 
81         sk_sp<SkSurface> surf(SkSurface::MakeRenderTarget(context, SkBudgeted::kNo, ii));
82 
83         // create a distinctive texture
84         for (int y = 0; y < Y_SIZE; ++y) {
85             for (int x = 0; x < X_SIZE; ++x) {
86                 alphaData[y * X_SIZE + x] = y*X_SIZE+x;
87             }
88         }
89 
90         for (auto rowBytes : kRowBytes) {
91 
92             // upload the texture (do per-rowbytes iteration because we may overwrite below).
93             bool result = sContext->writePixels(ii, alphaData, 0, {0, 0});
94             REPORTER_ASSERT(reporter, result, "Initial A8 writePixels failed");
95 
96             size_t nonZeroRowBytes = rowBytes ? rowBytes : X_SIZE;
97             size_t bufLen = nonZeroRowBytes * Y_SIZE;
98             std::unique_ptr<uint8_t[]> readback(new uint8_t[bufLen]);
99             // clear readback to something non-zero so we can detect readback failures
100             memset(readback.get(), kClearValue, bufLen);
101 
102             // read the texture back
103             result = sContext->readPixels(ii, readback.get(), rowBytes, {0, 0});
104             // We don't require reading from kAlpha_8 to be supported. TODO: At least make this work
105             // when kAlpha_8 is renderable.
106             if (!result) {
107                 continue;
108             }
109             REPORTER_ASSERT(reporter, result, "Initial A8 readPixels failed");
110 
111             // make sure the original & read back versions match
112             SkString msg;
113             msg.printf("rb:%d A8", SkToU32(rowBytes));
114             validate_alpha_data(reporter, X_SIZE, Y_SIZE, readback.get(), nonZeroRowBytes,
115                                 alphaData, msg, GrColorType::kAlpha_8);
116 
117             // Now try writing to a single channel surface (if we could create one).
118             if (surf) {
119                 SkCanvas* canvas = surf->getCanvas();
120 
121                 SkPaint paint;
122 
123                 const SkRect rect = SkRect::MakeLTRB(-10, -10, X_SIZE + 10, Y_SIZE + 10);
124 
125                 paint.setColor(SK_ColorWHITE);
126 
127                 canvas->drawRect(rect, paint);
128 
129                 // Workaround for a bug in old GCC/glibc used in our Chromecast toolchain:
130                 // error: call to '__warn_memset_zero_len' declared with attribute warning:
131                 //        memset used with constant zero length parameter; this could be due
132                 //        to transposed parameters
133                 // See also: https://gcc.gnu.org/bugzilla/show_bug.cgi?id=61294
134                 if (bufLen > 0) {
135                     memset(readback.get(), kClearValue, bufLen);
136                 }
137                 result = surf->readPixels(ii, readback.get(), nonZeroRowBytes, 0, 0);
138                 REPORTER_ASSERT(reporter, result, "A8 readPixels after clear failed");
139 
140                 match = true;
141                 for (int y = 0; y < Y_SIZE && match; ++y) {
142                     for (int x = 0; x < X_SIZE && match; ++x) {
143                         uint8_t rbValue = readback.get()[y * nonZeroRowBytes + x];
144                         if (0xFF != rbValue) {
145                             ERRORF(reporter,
146                                    "Failed alpha readback after clear. Expected: 0xFF, Got: 0x%02x"
147                                    " at (%d,%d), rb:%d", rbValue, x, y, SkToU32(rowBytes));
148                             match = false;
149                         }
150                     }
151                 }
152             }
153         }
154     }
155 
156     static constexpr struct {
157         GrColorType fColorType;
158         SkAlphaType fAlphaType;
159     } kInfos[] = {
160             {GrColorType::kRGBA_8888,      kPremul_SkAlphaType},
161             {GrColorType::kBGRA_8888,      kPremul_SkAlphaType},
162             {GrColorType::kRGBA_8888_SRGB, kPremul_SkAlphaType},
163             {GrColorType::kRGBA_1010102,   kPremul_SkAlphaType},
164     };
165 
166     for (int y = 0; y < Y_SIZE; ++y) {
167         for (int x = 0; x < X_SIZE; ++x) {
168             alphaData[y * X_SIZE + x] = y*X_SIZE+x;
169         }
170     }
171 
172     const SkImageInfo dstInfo = SkImageInfo::Make(X_SIZE, Y_SIZE,
173                                                   kAlpha_8_SkColorType,
174                                                   kPremul_SkAlphaType);
175 
176     // Attempt to read back just alpha from a RGBA/BGRA texture. Once with a texture-only src and
177     // once with a render target.
178     for (auto info : kInfos) {
179         for (auto renderable : {GrRenderable::kNo, GrRenderable::kYes}) {
180             uint32_t rgbaData[X_SIZE * Y_SIZE];
181             // Make the alpha channel of the rgba texture come from alphaData.
182             for (int y = 0; y < Y_SIZE; ++y) {
183                 for (int x = 0; x < X_SIZE; ++x) {
184                     rgbaData[y * X_SIZE + x] = GrColorPackRGBA(6, 7, 8, alphaData[y * X_SIZE + x]);
185                 }
186             }
187 
188             auto origin = GrRenderable::kYes == renderable ? kBottomLeft_GrSurfaceOrigin
189                                                            : kTopLeft_GrSurfaceOrigin;
190             auto proxy = sk_gpu_test::MakeTextureProxyFromData(
191                     context, renderable, X_SIZE, Y_SIZE, info.fColorType, info.fAlphaType, origin,
192                     rgbaData, 0);
193             if (!proxy) {
194                 continue;
195             }
196 
197             sk_sp<GrSurfaceContext> sContext = context->priv().makeWrappedSurfaceContext(
198                     std::move(proxy), info.fColorType, kPremul_SkAlphaType);
199 
200             for (auto rowBytes : kRowBytes) {
201                 size_t nonZeroRowBytes = rowBytes ? rowBytes : X_SIZE;
202 
203                 std::unique_ptr<uint8_t[]> readback(new uint8_t[nonZeroRowBytes * Y_SIZE]);
204                 // Clear so we don't accidentally see values from previous iteration.
205                 memset(readback.get(), kClearValue, nonZeroRowBytes * Y_SIZE);
206 
207                 // read the texture back
208                 bool result = sContext->readPixels(dstInfo, readback.get(), rowBytes, {0, 0});
209                 REPORTER_ASSERT(reporter, result, "8888 readPixels failed");
210 
211                 // make sure the original & read back versions match
212                 SkString msg;
213                 msg.printf("rt:%d, rb:%d 8888", GrRenderable::kYes == renderable,
214                                                 SkToU32(rowBytes));
215                 validate_alpha_data(reporter, X_SIZE, Y_SIZE, readback.get(), nonZeroRowBytes,
216                                     alphaData, msg, info.fColorType);
217             }
218         }
219     }
220 }
221