1cb93a386Sopenharmony_ci/*
2cb93a386Sopenharmony_ci * Copyright 2017 Google Inc.
3cb93a386Sopenharmony_ci *
4cb93a386Sopenharmony_ci * Use of this source code is governed by a BSD-style license that can be
5cb93a386Sopenharmony_ci * found in the LICENSE file.
6cb93a386Sopenharmony_ci */
7cb93a386Sopenharmony_ci
8cb93a386Sopenharmony_ci// This is a GPU-backend specific test.
9cb93a386Sopenharmony_ci
10cb93a386Sopenharmony_ci#include "tests/Test.h"
11cb93a386Sopenharmony_ci
12cb93a386Sopenharmony_ciusing namespace sk_gpu_test;
13cb93a386Sopenharmony_ci
14cb93a386Sopenharmony_ci#include "tools/gpu/GrContextFactory.h"
15cb93a386Sopenharmony_ci
16cb93a386Sopenharmony_ci#include "include/core/SkBitmap.h"
17cb93a386Sopenharmony_ci#include "include/core/SkCanvas.h"
18cb93a386Sopenharmony_ci#include "include/core/SkSurface.h"
19cb93a386Sopenharmony_ci#include "include/gpu/GrDirectContext.h"
20cb93a386Sopenharmony_ci#include "src/core/SkImagePriv.h"
21cb93a386Sopenharmony_ci
22cb93a386Sopenharmony_cistatic bool surface_is_expected_color(SkSurface* surf, const SkImageInfo& ii, SkColor color) {
23cb93a386Sopenharmony_ci    SkBitmap bm;
24cb93a386Sopenharmony_ci    bm.allocPixels(ii);
25cb93a386Sopenharmony_ci
26cb93a386Sopenharmony_ci    surf->readPixels(bm, 0, 0);
27cb93a386Sopenharmony_ci
28cb93a386Sopenharmony_ci    for (int y = 0; y < bm.height(); ++y) {
29cb93a386Sopenharmony_ci        for (int x = 0; x < bm.width(); ++x) {
30cb93a386Sopenharmony_ci            if (bm.getColor(x, y) != color) {
31cb93a386Sopenharmony_ci                return false;
32cb93a386Sopenharmony_ci            }
33cb93a386Sopenharmony_ci        }
34cb93a386Sopenharmony_ci    }
35cb93a386Sopenharmony_ci
36cb93a386Sopenharmony_ci    return true;
37cb93a386Sopenharmony_ci}
38cb93a386Sopenharmony_ci
39cb93a386Sopenharmony_cistatic void basic_test(skiatest::Reporter* reporter, GrRecordingContext* rContext) {
40cb93a386Sopenharmony_ci    const SkImageInfo ii = SkImageInfo::Make(64, 64, kN32_SkColorType, kPremul_SkAlphaType);
41cb93a386Sopenharmony_ci
42cb93a386Sopenharmony_ci    SkBitmap bm;
43cb93a386Sopenharmony_ci    bm.allocPixels(ii);
44cb93a386Sopenharmony_ci
45cb93a386Sopenharmony_ci    SkCanvas bmCanvas(bm);
46cb93a386Sopenharmony_ci    bmCanvas.clear(SK_ColorRED);
47cb93a386Sopenharmony_ci
48cb93a386Sopenharmony_ci    // We start off with the raster image being all red.
49cb93a386Sopenharmony_ci    sk_sp<SkImage> img = SkMakeImageFromRasterBitmap(bm, kNever_SkCopyPixelsMode);
50cb93a386Sopenharmony_ci
51cb93a386Sopenharmony_ci    sk_sp<SkSurface> gpuSurface = SkSurface::MakeRenderTarget(rContext, SkBudgeted::kYes, ii);
52cb93a386Sopenharmony_ci    SkCanvas* canvas = gpuSurface->getCanvas();
53cb93a386Sopenharmony_ci
54cb93a386Sopenharmony_ci    // w/o pinning - the gpu draw always reflects the current state of the underlying bitmap
55cb93a386Sopenharmony_ci    {
56cb93a386Sopenharmony_ci        canvas->drawImage(img, 0, 0);
57cb93a386Sopenharmony_ci        REPORTER_ASSERT(reporter, surface_is_expected_color(gpuSurface.get(), ii, SK_ColorRED));
58cb93a386Sopenharmony_ci
59cb93a386Sopenharmony_ci        bmCanvas.clear(SK_ColorGREEN);
60cb93a386Sopenharmony_ci
61cb93a386Sopenharmony_ci        canvas->drawImage(img, 0, 0);
62cb93a386Sopenharmony_ci        REPORTER_ASSERT(reporter, surface_is_expected_color(gpuSurface.get(), ii, SK_ColorGREEN));
63cb93a386Sopenharmony_ci    }
64cb93a386Sopenharmony_ci
65cb93a386Sopenharmony_ci    // w/ pinning - the gpu draw is stuck at the pinned state
66cb93a386Sopenharmony_ci    {
67cb93a386Sopenharmony_ci        SkImage_pinAsTexture(img.get(), rContext); // pin at blue
68cb93a386Sopenharmony_ci
69cb93a386Sopenharmony_ci        canvas->drawImage(img, 0, 0);
70cb93a386Sopenharmony_ci        REPORTER_ASSERT(reporter, surface_is_expected_color(gpuSurface.get(), ii, SK_ColorGREEN));
71cb93a386Sopenharmony_ci
72cb93a386Sopenharmony_ci        bmCanvas.clear(SK_ColorBLUE);
73cb93a386Sopenharmony_ci
74cb93a386Sopenharmony_ci        canvas->drawImage(img, 0, 0);
75cb93a386Sopenharmony_ci        REPORTER_ASSERT(reporter, surface_is_expected_color(gpuSurface.get(), ii, SK_ColorGREEN));
76cb93a386Sopenharmony_ci
77cb93a386Sopenharmony_ci        SkImage_unpinAsTexture(img.get(), rContext);
78cb93a386Sopenharmony_ci    }
79cb93a386Sopenharmony_ci
80cb93a386Sopenharmony_ci    // once unpinned local changes will be picked up
81cb93a386Sopenharmony_ci    {
82cb93a386Sopenharmony_ci        canvas->drawImage(img, 0, 0);
83cb93a386Sopenharmony_ci        REPORTER_ASSERT(reporter, surface_is_expected_color(gpuSurface.get(), ii, SK_ColorBLUE));
84cb93a386Sopenharmony_ci    }
85cb93a386Sopenharmony_ci}
86cb93a386Sopenharmony_ci
87cb93a386Sopenharmony_ci// Deleting the context while there are still pinned images shouldn't result in a crash.
88cb93a386Sopenharmony_cistatic void cleanup_test(skiatest::Reporter* reporter) {
89cb93a386Sopenharmony_ci
90cb93a386Sopenharmony_ci    const SkImageInfo ii = SkImageInfo::Make(64, 64, kN32_SkColorType, kPremul_SkAlphaType);
91cb93a386Sopenharmony_ci
92cb93a386Sopenharmony_ci    SkBitmap bm;
93cb93a386Sopenharmony_ci    bm.allocPixels(ii);
94cb93a386Sopenharmony_ci
95cb93a386Sopenharmony_ci    SkCanvas bmCanvas(bm);
96cb93a386Sopenharmony_ci    bmCanvas.clear(SK_ColorRED);
97cb93a386Sopenharmony_ci
98cb93a386Sopenharmony_ci    GrMockOptions options;
99cb93a386Sopenharmony_ci    sk_sp<GrDirectContext> mockContext = GrDirectContext::MakeMock(&options);
100cb93a386Sopenharmony_ci
101cb93a386Sopenharmony_ci    for (int i = 0; i < GrContextFactory::kContextTypeCnt; ++i) {
102cb93a386Sopenharmony_ci        GrContextFactory::ContextType ctxType = (GrContextFactory::ContextType) i;
103cb93a386Sopenharmony_ci
104cb93a386Sopenharmony_ci        {
105cb93a386Sopenharmony_ci            sk_sp<SkImage> img;
106cb93a386Sopenharmony_ci            GrDirectContext* dContext = nullptr;
107cb93a386Sopenharmony_ci
108cb93a386Sopenharmony_ci            {
109cb93a386Sopenharmony_ci                GrContextFactory testFactory;
110cb93a386Sopenharmony_ci                ContextInfo info = testFactory.getContextInfo(ctxType);
111cb93a386Sopenharmony_ci                dContext = info.directContext();
112cb93a386Sopenharmony_ci                if (!dContext) {
113cb93a386Sopenharmony_ci                    continue;
114cb93a386Sopenharmony_ci                }
115cb93a386Sopenharmony_ci
116cb93a386Sopenharmony_ci                img = SkMakeImageFromRasterBitmap(bm, kNever_SkCopyPixelsMode);
117cb93a386Sopenharmony_ci                if (!SkImage_pinAsTexture(img.get(), dContext)) {
118cb93a386Sopenharmony_ci                    continue;
119cb93a386Sopenharmony_ci                }
120cb93a386Sopenharmony_ci                // Pinning on a second context should be blocked.
121cb93a386Sopenharmony_ci                REPORTER_ASSERT(reporter, !SkImage_pinAsTexture(img.get(), mockContext.get()));
122cb93a386Sopenharmony_ci            }
123cb93a386Sopenharmony_ci
124cb93a386Sopenharmony_ci            // The context used to pin the image is gone at this point!
125cb93a386Sopenharmony_ci            // "context" isn't technically used in this call but it can't be null!
126cb93a386Sopenharmony_ci            // We don't really want to support this use case but it currently happens.
127cb93a386Sopenharmony_ci            SkImage_unpinAsTexture(img.get(), dContext);
128cb93a386Sopenharmony_ci        }
129cb93a386Sopenharmony_ci    }
130cb93a386Sopenharmony_ci}
131cb93a386Sopenharmony_ci
132cb93a386Sopenharmony_ciDEF_GPUTEST_FOR_RENDERING_CONTEXTS(PinnedImageTest, reporter, ctxInfo) {
133cb93a386Sopenharmony_ci    basic_test(reporter, ctxInfo.directContext());
134cb93a386Sopenharmony_ci    cleanup_test(reporter);
135cb93a386Sopenharmony_ci}
136