blob: 857fecbf5197a38cc5648b85d5573fbaa5422d2c [file] [log] [blame]
bsalomone179a912016-01-20 06:18:10 -08001/*
2 * Copyright 2016 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// This test only works with the GPU backend.
9
10#include "gm.h"
11
Greg Daniel7ef28f32017-04-20 16:41:55 +000012#include "GrBackendSurface.h"
bsalomone179a912016-01-20 06:18:10 -080013#include "GrContext.h"
Robert Phillipsf35fd8d2018-01-22 10:48:15 -050014#include "GrContextPriv.h"
robertphillips87f15c82016-05-20 11:14:33 -070015#include "GrGpu.h"
robertphillips87f15c82016-05-20 11:14:33 -070016#include "gl/GrGLContext.h"
bsalomone179a912016-01-20 06:18:10 -080017#include "SkBitmap.h"
18#include "SkGradientShader.h"
19#include "SkImage.h"
20
21namespace skiagm {
22class RectangleTexture : public GM {
23public:
24 RectangleTexture() {
25 this->setBGColor(0xFFFFFFFF);
26 }
27
28protected:
29 SkString onShortName() override {
30 return SkString("rectangle_texture");
31 }
32
Brian Salomon246bc3d2018-12-06 15:33:02 -050033 SkISize onISize() override { return SkISize::Make(1200, 500); }
bsalomone179a912016-01-20 06:18:10 -080034
35 void fillPixels(int width, int height, void *pixels) {
36 SkBitmap bmp;
37 bmp.setInfo(SkImageInfo::MakeN32(width, height, kOpaque_SkAlphaType), width * 4);
38 bmp.setPixels(pixels);
39 SkPaint paint;
40 SkCanvas canvas(bmp);
41 SkPoint pts[] = { {0, 0}, {0, SkIntToScalar(height)} };
42 SkColor colors0[] = { 0xFF1060B0 , 0xFF102030 };
reed1a9b9642016-03-13 14:13:58 -070043 paint.setShader(SkGradientShader::MakeLinear(pts, colors0, nullptr, 2,
44 SkShader::kClamp_TileMode));
bsalomone179a912016-01-20 06:18:10 -080045 canvas.drawPaint(paint);
46
Brian Salomon246bc3d2018-12-06 15:33:02 -050047 SkColor colors1[] = {0xFFA07010, 0xFFA02080};
bsalomone179a912016-01-20 06:18:10 -080048 paint.setAntiAlias(true);
reed1a9b9642016-03-13 14:13:58 -070049 paint.setShader(SkGradientShader::MakeLinear(pts, colors1, nullptr, 2,
50 SkShader::kClamp_TileMode));
bsalomone179a912016-01-20 06:18:10 -080051 canvas.drawCircle(SkIntToScalar(width) / 2, SkIntToScalar(height) / 2,
52 SkIntToScalar(width + height) / 5, paint);
53 }
54
Brian Salomon246bc3d2018-12-06 15:33:02 -050055 sk_sp<SkImage> createRectangleTextureImg(GrContext* context, GrSurfaceOrigin origin, int width,
56 int height, const uint32_t* pixels) {
bsalomone179a912016-01-20 06:18:10 -080057 if (!context) {
58 return nullptr;
59 }
Khushalc421ca12018-06-26 14:38:34 -070060 if (context->abandoned()) {
Robert Phillips6d363702018-06-25 08:53:09 -040061 return nullptr;
62 }
Robert Phillipsf35fd8d2018-01-22 10:48:15 -050063 GrGpu* gpu = context->contextPriv().getGpu();
bsalomone179a912016-01-20 06:18:10 -080064 if (!gpu) {
65 return nullptr;
66 }
67 const GrGLContext* glCtx = gpu->glContextForTesting();
68 if (!glCtx) {
69 return nullptr;
70 }
71
72 if (!(kGL_GrGLStandard == glCtx->standard() && glCtx->version() >= GR_GL_VER(3, 1)) &&
Weiliang Chen1e04b362018-07-12 17:13:35 -040073 !(glCtx->hasExtension("GL_ARB_texture_rectangle") ||
74 glCtx->hasExtension("GL_ANGLE_texture_rectangle"))) {
bsalomone179a912016-01-20 06:18:10 -080075 return nullptr;
76 }
77
78 // We will always create the GL texture as GL_RGBA, however the pixels uploaded may be
79 // be RGBA or BGRA, depending on how SkPMColor was compiled.
80 GrGLenum format;
81 if (kSkia8888_GrPixelConfig == kBGRA_8888_GrPixelConfig) {
82 format = GR_GL_BGRA;
83 } else {
84 SkASSERT(kSkia8888_GrPixelConfig == kRGBA_8888_GrPixelConfig);
85 format = GR_GL_RGBA;
86 }
87
88 const GrGLInterface* gl = glCtx->interface();
Brian Salomon246bc3d2018-12-06 15:33:02 -050089 // Useful for debugging whether errors result from use of RECTANGLE
90 // static constexpr GrGLenum kTarget = GR_GL_TEXTURE_2D;
91 static constexpr GrGLenum kTarget = GR_GL_TEXTURE_RECTANGLE;
bsalomon3c481002016-03-21 09:04:26 -070092 GrGLuint id = 0;
bsalomone179a912016-01-20 06:18:10 -080093 GR_GL_CALL(gl, GenTextures(1, &id));
Brian Salomon246bc3d2018-12-06 15:33:02 -050094 GR_GL_CALL(gl, BindTexture(kTarget, id));
95 GR_GL_CALL(gl, TexParameteri(kTarget, GR_GL_TEXTURE_MAG_FILTER, GR_GL_NEAREST));
96 GR_GL_CALL(gl, TexParameteri(kTarget, GR_GL_TEXTURE_MIN_FILTER, GR_GL_NEAREST));
97 GR_GL_CALL(gl, TexParameteri(kTarget, GR_GL_TEXTURE_WRAP_S, GR_GL_CLAMP_TO_EDGE));
98 GR_GL_CALL(gl, TexParameteri(kTarget, GR_GL_TEXTURE_WRAP_T, GR_GL_CLAMP_TO_EDGE));
99 std::unique_ptr<uint32_t[]> tempPixels;
100 if (origin == kBottomLeft_GrSurfaceOrigin) {
101 tempPixels.reset(new uint32_t[width * height]);
102 for (int y = 0; y < height; ++y) {
103 std::copy_n(pixels + width * (height - y - 1), width, tempPixels.get() + width * y);
104 }
105 pixels = tempPixels.get();
106 }
107 GR_GL_CALL(gl, TexImage2D(kTarget, 0, GR_GL_RGBA, width, height, 0, format,
108 GR_GL_UNSIGNED_BYTE, pixels));
bsalomone179a912016-01-20 06:18:10 -0800109
110 context->resetContext();
111 GrGLTextureInfo info;
112 info.fID = id;
Brian Salomon246bc3d2018-12-06 15:33:02 -0500113 info.fTarget = kTarget;
Greg Danielf5d87582017-12-18 14:48:15 -0500114 info.fFormat = GR_GL_RGBA8;
Greg Daniel7ef28f32017-04-20 16:41:55 +0000115
Greg Danielf5d87582017-12-18 14:48:15 -0500116 GrBackendTexture rectangleTex(width, height, GrMipMapped::kNo, info);
Greg Daniel7ef28f32017-04-20 16:41:55 +0000117
Brian Salomon246bc3d2018-12-06 15:33:02 -0500118 if (sk_sp<SkImage> image = SkImage::MakeFromAdoptedTexture(context, rectangleTex, origin,
Greg Danielf5d87582017-12-18 14:48:15 -0500119 kRGBA_8888_SkColorType)) {
bsalomone179a912016-01-20 06:18:10 -0800120 return image;
121 }
122 GR_GL_CALL(gl, DeleteTextures(1, &id));
123 return nullptr;
124 }
125
126 void onDraw(SkCanvas* canvas) override {
robertphillips175dd9b2016-04-28 14:32:04 -0700127 GrContext *context = canvas->getGrContext();
128 if (!context) {
bsalomone179a912016-01-20 06:18:10 -0800129 skiagm::GM::DrawGpuOnlyMessage(canvas);
130 return;
131 }
132
mtkleindbfd7ab2016-09-01 11:24:54 -0700133 constexpr int kWidth = 50;
134 constexpr int kHeight = 50;
135 constexpr SkScalar kPad = 5.f;
bsalomone179a912016-01-20 06:18:10 -0800136
137 SkPMColor pixels[kWidth * kHeight];
138 this->fillPixels(kWidth, kHeight, pixels);
bsalomone179a912016-01-20 06:18:10 -0800139
Brian Salomon246bc3d2018-12-06 15:33:02 -0500140 sk_sp<SkImage> rectImgs[] = {
141 this->createRectangleTextureImg(context, kTopLeft_GrSurfaceOrigin, kWidth, kHeight,
142 pixels),
143 this->createRectangleTextureImg(context, kBottomLeft_GrSurfaceOrigin, kWidth,
144 kHeight, pixels),
145 };
146 SkASSERT(SkToBool(rectImgs[0]) == SkToBool(rectImgs[1]));
147 if (!rectImgs[0]) {
bsalomone179a912016-01-20 06:18:10 -0800148 SkPaint paint;
Hal Canarydf2d27e2019-01-08 09:38:02 -0500149 SkFont font;
150 canvas->drawString("Could not create rectangle texture image.", 10, 100, font, paint);
bsalomone179a912016-01-20 06:18:10 -0800151 return;
152 }
153
mtkleindbfd7ab2016-09-01 11:24:54 -0700154 constexpr SkFilterQuality kQualities[] = {
Brian Salomon246bc3d2018-12-06 15:33:02 -0500155 kNone_SkFilterQuality,
156 kLow_SkFilterQuality,
157 kMedium_SkFilterQuality,
158 kHigh_SkFilterQuality,
bsalomone179a912016-01-20 06:18:10 -0800159 };
160
Brian Salomon246bc3d2018-12-06 15:33:02 -0500161 constexpr SkScalar kScales[] = {1.0f, 1.2f, 0.75f};
bsalomone179a912016-01-20 06:18:10 -0800162
163 canvas->translate(kPad, kPad);
Brian Salomon246bc3d2018-12-06 15:33:02 -0500164 for (size_t i = 0; i < SK_ARRAY_COUNT(rectImgs); ++i) {
165 for (auto s : kScales) {
166 canvas->save();
167 canvas->scale(s, s);
168 for (auto q : kQualities) {
169 // drawImage
170 SkPaint plainPaint;
171 plainPaint.setFilterQuality(q);
172 canvas->drawImage(rectImgs[i], 0, 0, &plainPaint);
173 canvas->translate(kWidth + kPad, 0);
bsalomone179a912016-01-20 06:18:10 -0800174
Brian Salomon246bc3d2018-12-06 15:33:02 -0500175 // clamp/clamp shader
176 SkPaint clampPaint;
177 clampPaint.setFilterQuality(q);
178 clampPaint.setShader(rectImgs[i]->makeShader());
179 canvas->drawRect(SkRect::MakeWH(1.5f * kWidth, 1.5f * kHeight), clampPaint);
180 canvas->translate(kWidth * 1.5f + kPad, 0);
bsalomone179a912016-01-20 06:18:10 -0800181
Brian Salomon246bc3d2018-12-06 15:33:02 -0500182 // repeat/mirror shader
183 SkPaint repeatPaint;
184 repeatPaint.setFilterQuality(q);
185 repeatPaint.setShader(rectImgs[i]->makeShader(SkShader::kRepeat_TileMode,
186 SkShader::kMirror_TileMode));
187 canvas->drawRect(SkRect::MakeWH(1.5f * kWidth, 1.5f * kHeight), repeatPaint);
188 canvas->translate(1.5f * kWidth + kPad, 0);
189
190 // drawImageRect with kStrict
191 auto srcRect = SkRect::MakeXYWH(.25f * rectImgs[i]->width(),
192 .25f * rectImgs[i]->height(),
193 .50f * rectImgs[i]->width(),
194 .50f * rectImgs[i]->height());
195 auto dstRect = SkRect::MakeXYWH(0, 0,
196 .50f * rectImgs[i]->width(),
197 .50f * rectImgs[i]->height());
198 canvas->drawImageRect(rectImgs[i], srcRect, dstRect, &plainPaint,
199 SkCanvas::kStrict_SrcRectConstraint);
200 canvas->translate(kWidth * .5f + kPad, 0);
201 }
202 canvas->restore();
203 canvas->translate(0, kPad + 1.5f * kHeight * s);
bsalomone179a912016-01-20 06:18:10 -0800204 }
bsalomone179a912016-01-20 06:18:10 -0800205 }
206 }
207
208private:
209 typedef GM INHERITED;
210};
211
212DEF_GM(return new RectangleTexture;)
213}