2 * Copyright 2014 Google Inc.
4 * Use of this source code is governed by a BSD-style license that can be
5 * found in the LICENSE file.
9 #include "include/core/SkBlendMode.h"
10 #include "include/core/SkCanvas.h"
11 #include "include/core/SkColor.h"
12 #include "include/core/SkColorSpace.h"
13 #include "include/core/SkFont.h"
14 #include "include/core/SkImage.h"
15 #include "include/core/SkImageInfo.h"
16 #include "include/core/SkPaint.h"
17 #include "include/core/SkPoint.h"
18 #include "include/core/SkRect.h"
19 #include "include/core/SkRefCnt.h"
20 #include "include/core/SkScalar.h"
21 #include "include/core/SkShader.h"
22 #include "include/core/SkSize.h"
23 #include "include/core/SkString.h"
24 #include "include/core/SkSurface.h"
25 #include "include/core/SkSurfaceProps.h"
26 #include "include/core/SkTileMode.h"
27 #include "include/core/SkTypeface.h"
28 #include "include/core/SkTypes.h"
29 #include "include/effects/SkGradientShader.h"
30 #include "include/gpu/GrDirectContext.h"
31 #include "include/gpu/GrRecordingContext.h"
32 #include "include/utils/SkTextUtils.h"
33 #include "tools/ToolUtils.h"
34 #include "tools/gpu/BackendSurfaceFactory.h"
39 static sk_sp<SkShader> make_shader() {
42 SkPoint pts[] = { { 0, 0 }, { W, H } };
43 SkColor colors[] = { SkColorSetRGB(a, a, a), SkColorSetRGB(b, b, b) };
44 return SkGradientShader::MakeLinear(pts, colors, nullptr, 2, SkTileMode::kClamp);
47 static sk_sp<SkSurface> make_surface(GrRecordingContext* ctx,
48 const SkImageInfo& info,
49 SkPixelGeometry geo) {
50 SkSurfaceProps props(0, geo);
52 return SkSurface::MakeRenderTarget(ctx, SkBudgeted::kNo, info, 0, &props);
54 return SkSurface::MakeRaster(info, &props);
58 static void test_draw(SkCanvas* canvas, const char label[]) {
61 paint.setAntiAlias(true);
62 paint.setDither(true);
64 paint.setShader(make_shader());
65 canvas->drawRect(SkRect::MakeWH(W, H), paint);
66 paint.setShader(nullptr);
68 paint.setColor(SK_ColorWHITE);
69 SkFont font(ToolUtils::create_portable_typeface(), 32);
70 font.setEdging(SkFont::Edging::kSubpixelAntiAlias);
71 SkTextUtils::DrawString(canvas, label, W / 2, H * 3 / 4, font, paint,
72 SkTextUtils::kCenter_Align);
75 class SurfacePropsGM : public skiagm::GM {
80 SkString onShortName() override {
81 return SkString("surfaceprops");
84 SkISize onISize() override {
85 return SkISize::Make(W, H * 5);
88 void onDraw(SkCanvas* canvas) override {
89 auto ctx = canvas->recordingContext();
91 // must be opaque to have a hope of testing LCD text
92 const SkImageInfo info = SkImageInfo::MakeN32(W, H, kOpaque_SkAlphaType);
98 { kUnknown_SkPixelGeometry, "Unknown" },
99 { kRGB_H_SkPixelGeometry, "RGB_H" },
100 { kBGR_H_SkPixelGeometry, "BGR_H" },
101 { kRGB_V_SkPixelGeometry, "RGB_V" },
102 { kBGR_V_SkPixelGeometry, "BGR_V" },
107 for (const auto& rec : recs) {
108 auto surface(make_surface(ctx, info, rec.fGeo));
110 SkDebugf("failed to create surface! label: %s", rec.fLabel);
113 test_draw(surface->getCanvas(), rec.fLabel);
114 surface->draw(canvas, x, y);
120 using INHERITED = GM;
122 DEF_GM( return new SurfacePropsGM )
125 static bool equal(const SkSurfaceProps& a, const SkSurfaceProps& b) {
126 return a.flags() == b.flags() && a.pixelGeometry() == b.pixelGeometry();
130 class NewSurfaceGM : public skiagm::GM {
135 SkString onShortName() override {
136 return SkString("surfacenew");
139 SkISize onISize() override {
140 return SkISize::Make(300, 140);
143 static void drawInto(SkCanvas* canvas) {
144 canvas->drawColor(SK_ColorRED);
147 void onDraw(SkCanvas* canvas) override {
148 SkImageInfo info = SkImageInfo::MakeN32Premul(100, 100);
150 auto surf(ToolUtils::makeSurface(canvas, info, nullptr));
151 drawInto(surf->getCanvas());
153 sk_sp<SkImage> image(surf->makeImageSnapshot());
154 canvas->drawImage(image, 10, 10);
156 auto surf2(surf->makeSurface(info));
157 drawInto(surf2->getCanvas());
159 // Assert that the props were communicated transitively through the first image
160 SkASSERT(equal(surf->props(), surf2->props()));
162 sk_sp<SkImage> image2(surf2->makeImageSnapshot());
163 canvas->drawImage(image2.get(), 10 + SkIntToScalar(image->width()) + 10, 10);
167 using INHERITED = GM;
169 DEF_GM( return new NewSurfaceGM )
171 ///////////////////////////////////////////////////////////////////////////////////////////////////
173 // The GPU backend may behave differently when images are snapped from wrapped textures and
174 // render targets compared.
183 static sk_sp<SkSurface> make_surface(const SkImageInfo& ii, SkCanvas* canvas, SurfaceType type) {
184 GrDirectContext* direct = GrAsDirectContext(canvas->recordingContext());
187 return ToolUtils::makeSurface(canvas, ii);
188 case kBackendTexture:
192 return sk_gpu_test::MakeBackendTextureSurface(direct, ii, kTopLeft_GrSurfaceOrigin, 1);
193 case kBackendRenderTarget:
194 return sk_gpu_test::MakeBackendRenderTargetSurface(direct,
196 kTopLeft_GrSurfaceOrigin,
202 using MakeSurfaceFn = std::function<sk_sp<SkSurface>(const SkImageInfo&)>;
204 #define DEF_BASIC_SURFACE_TEST(name, canvas, main, W, H) \
205 DEF_SIMPLE_GM(name, canvas, W, H) { \
206 auto make = [canvas](const SkImageInfo& ii) { \
207 return make_surface(ii, canvas, SurfaceType::kManaged); \
209 main(canvas, MakeSurfaceFn(make)); \
212 #define DEF_BACKEND_SURFACE_TEST(name, canvas, main, type, W, H) \
213 DEF_SIMPLE_GM_CAN_FAIL(name, canvas, err_msg, W, H) { \
214 GrDirectContext* direct = GrAsDirectContext(canvas->recordingContext()); \
215 if (!direct || direct->abandoned()) { \
216 *err_msg = "Requires non-abandoned GrDirectContext"; \
217 return skiagm::DrawResult::kSkip; \
219 auto make = [canvas](const SkImageInfo& ii) { return make_surface(ii, canvas, type); }; \
220 main(canvas, MakeSurfaceFn(make)); \
221 return skiagm::DrawResult::kOk; \
224 #define DEF_BET_SURFACE_TEST(name, canvas, main, W, H) \
225 DEF_BACKEND_SURFACE_TEST(SK_MACRO_CONCAT(name, _bet), canvas, main, \
226 SurfaceType::kBackendTexture, W, H)
228 #define DEF_BERT_SURFACE_TEST(name, canvas, main, W, H) \
229 DEF_BACKEND_SURFACE_TEST(SK_MACRO_CONCAT(name, _bert), canvas, main, \
230 SurfaceType::kBackendRenderTarget, W, H)
232 // This makes 3 GMs from the same code, normal, wrapped backend texture, and wrapped backend
234 #define DEF_SURFACE_TESTS(name, canvas, W, H) \
235 static void SK_MACRO_CONCAT(name, _main)(SkCanvas*, const MakeSurfaceFn&); \
236 DEF_BASIC_SURFACE_TEST(name, canvas, SK_MACRO_CONCAT(name, _main), W, H) \
237 DEF_BET_SURFACE_TEST (name, canvas, SK_MACRO_CONCAT(name, _main), W, H) \
238 DEF_BERT_SURFACE_TEST (name, canvas, SK_MACRO_CONCAT(name, _main), W, H) \
239 static void SK_MACRO_CONCAT(name, _main)(SkCanvas * canvas, const MakeSurfaceFn& make)
241 DEF_SURFACE_TESTS(copy_on_write_retain, canvas, 256, 256) {
242 const SkImageInfo info = SkImageInfo::MakeN32Premul(256, 256);
243 sk_sp<SkSurface> surf = make(info);
245 surf->getCanvas()->clear(SK_ColorRED);
246 // its important that image survives longer than the next draw, so the surface will see
247 // an outstanding image, and have to decide if it should retain or discard those pixels
248 sk_sp<SkImage> image = surf->makeImageSnapshot();
250 // normally a clear+opaque should trigger the discard optimization, but since we have a clip
251 // it should not (we need the previous red pixels).
252 surf->getCanvas()->clipRect(SkRect::MakeWH(128, 256));
253 surf->getCanvas()->clear(SK_ColorBLUE);
255 // expect to see two rects: blue | red
256 canvas->drawImage(surf->makeImageSnapshot(), 0, 0);
259 // Like copy_on_write_retain but draws the snapped image back to the surface it was snapped from.
260 DEF_SURFACE_TESTS(copy_on_write_retain2, canvas, 256, 256) {
261 const SkImageInfo info = SkImageInfo::MakeN32Premul(256, 256);
262 sk_sp<SkSurface> surf = make(info);
264 surf->getCanvas()->clear(SK_ColorBLUE);
265 // its important that image survives longer than the next draw, so the surface will see
266 // an outstanding image, and have to decide if it should retain or discard those pixels
267 sk_sp<SkImage> image = surf->makeImageSnapshot();
269 surf->getCanvas()->clear(SK_ColorRED);
270 // normally a clear+opaque should trigger the discard optimization, but since we have a clip
271 // it should not (we need the previous red pixels).
272 surf->getCanvas()->clipRect(SkRect::MakeWH(128, 256));
273 surf->getCanvas()->drawImage(image, 0, 0);
275 // expect to see two rects: blue | red
276 canvas->drawImage(surf->makeImageSnapshot(), 0, 0);
279 DEF_SURFACE_TESTS(simple_snap_image, canvas, 256, 256) {
280 const SkImageInfo info = SkImageInfo::MakeN32Premul(256, 256);
281 sk_sp<SkSurface> surf = make(info);
283 surf->getCanvas()->clear(SK_ColorRED);
284 sk_sp<SkImage> image = surf->makeImageSnapshot();
285 // expect to see just red
286 canvas->drawImage(std::move(image), 0, 0);
289 // Like simple_snap_image but the surface dies before the image.
290 DEF_SURFACE_TESTS(simple_snap_image2, canvas, 256, 256) {
291 const SkImageInfo info = SkImageInfo::MakeN32Premul(256, 256);
292 sk_sp<SkSurface> surf = make(info);
294 surf->getCanvas()->clear(SK_ColorRED);
295 sk_sp<SkImage> image = surf->makeImageSnapshot();
297 // expect to see just red
298 canvas->drawImage(std::move(image), 0, 0);
301 DEF_SIMPLE_GM(snap_with_mips, canvas, 80, 75) {
302 auto ct = canvas->imageInfo().colorType() == kUnknown_SkColorType
303 ? kRGBA_8888_SkColorType
304 : canvas->imageInfo().colorType();
305 auto ii = SkImageInfo::Make({32, 32},
308 canvas->imageInfo().refColorSpace());
309 auto surface = SkSurface::MakeRaster(ii);
311 auto nextImage = [&](SkColor color) {
312 surface->getCanvas()->clear(color);
314 paint.setColor(~color | 0xFF000000);
315 surface->getCanvas()->drawRect(SkRect::MakeLTRB(surface->width() *2/5.f,
316 surface->height()*2/5.f,
317 surface->width() *3/5.f,
318 surface->height()*3/5.f),
320 return surface->makeImageSnapshot()->withDefaultMipmaps();
323 static constexpr int kPad = 8;
324 static const SkSamplingOptions kSampling{SkFilterMode::kLinear, SkMipmapMode::kLinear};
327 for (int y = 0; y < 3; ++y) {
329 SkColor kColors[] = {0xFFF0F0F0, SK_ColorBLUE};
330 for (int x = 0; x < 2; ++x) {
331 auto image = nextImage(kColors[x]);
332 canvas->drawImage(image, 0, 0, kSampling);
333 canvas->translate(ii.width() + kPad, 0);
336 canvas->translate(0, ii.width() + kPad);
337 canvas->scale(.4f, .4f);
342 DEF_SURFACE_TESTS(copy_on_write_savelayer, canvas, 256, 256) {
343 const SkImageInfo info = SkImageInfo::MakeN32Premul(256, 256);
344 sk_sp<SkSurface> surf = make(info);
345 surf->getCanvas()->clear(SK_ColorRED);
346 // its important that image survives longer than the next draw, so the surface will see
347 // an outstanding image, and have to decide if it should retain or discard those pixels
348 sk_sp<SkImage> image = surf->makeImageSnapshot();
350 // now draw into a full-screen layer. This should (a) trigger a copy-on-write, but it should
351 // not trigger discard, even tho its alpha (SK_ColorBLUE) is opaque, since it is in a layer
352 // with a non-opaque paint.
354 paint.setAlphaf(0.25f);
355 surf->getCanvas()->saveLayer({0, 0, 256, 256}, &paint);
356 surf->getCanvas()->clear(SK_ColorBLUE);
357 surf->getCanvas()->restore();
359 // expect to see two rects: blue blended on red
360 canvas->drawImage(surf->makeImageSnapshot(), 0, 0);
363 DEF_SURFACE_TESTS(surface_underdraw, canvas, 256, 256) {
364 SkImageInfo info = SkImageInfo::MakeN32Premul(256, 256, nullptr);
365 auto surf = make(info);
367 const SkIRect subset = SkIRect::MakeLTRB(180, 0, 256, 256);
371 SkPoint pts[] = {{0, 0}, {40, 50}};
372 SkColor colors[] = {SK_ColorRED, SK_ColorBLUE};
373 auto sh = SkGradientShader::MakeLinear(pts, colors, nullptr, 2, SkTileMode::kRepeat);
376 surf->getCanvas()->drawPaint(paint);
379 // save away the right-hand strip, then clear it
380 sk_sp<SkImage> saveImg = surf->makeImageSnapshot(subset);
383 paint.setBlendMode(SkBlendMode::kClear);
384 surf->getCanvas()->drawRect(SkRect::Make(subset), paint);
387 // draw the "foreground"
390 paint.setColor(SK_ColorGREEN);
391 SkRect r = { 0, 10, 256, 35 };
392 while (r.fBottom < 256) {
393 surf->getCanvas()->drawRect(r, paint);
394 r.offset(0, r.height() * 2);
400 SkPoint pts[] = {{SkIntToScalar(subset.left()), 0}, {SkIntToScalar(subset.right()), 0}};
401 SkColor colors[] = {0xFF000000, 0};
402 auto sh = SkGradientShader::MakeLinear(pts, colors, nullptr, 2, SkTileMode::kClamp);
405 paint.setBlendMode(SkBlendMode::kDstIn);
406 surf->getCanvas()->drawRect(SkRect::Make(subset), paint);
409 // restore the original strip, drawing it "under" the current foreground
412 paint.setBlendMode(SkBlendMode::kDstOver);
413 surf->getCanvas()->drawImage(saveImg,
414 SkIntToScalar(subset.left()), SkIntToScalar(subset.top()),
415 SkSamplingOptions(), &paint);
419 surf->draw(canvas, 0, 0);