blob: 3888891c1f6501c97709cf94c12f10245086919d [file] [log] [blame]
commit-bot@chromium.orgc3fe5492014-01-30 18:15:51 +00001/*
2 * Copyright 2014 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
commit-bot@chromium.orgc3fe5492014-01-30 18:15:51 +000012#include "GrContext.h"
joshualitt5478d422014-11-14 16:00:38 -080013#include "GrDefaultGeoProcFactory.h"
Robert Phillips7c525e62018-06-12 10:11:12 -040014#include "GrMemoryPool.h"
Brian Salomon576eda12017-07-14 11:30:17 -040015#include "GrOpFlushState.h"
commit-bot@chromium.orgc3fe5492014-01-30 18:15:51 +000016#include "GrPathUtils.h"
Brian Salomon576eda12017-07-14 11:30:17 -040017#include "GrRenderTargetContextPriv.h"
commit-bot@chromium.orgc3fe5492014-01-30 18:15:51 +000018#include "SkColorPriv.h"
commit-bot@chromium.orgc3fe5492014-01-30 18:15:51 +000019#include "SkGeometry.h"
Cary Clark74f623d2017-11-06 20:02:02 -050020#include "SkPointPriv.h"
commit-bot@chromium.orgc3fe5492014-01-30 18:15:51 +000021#include "SkTLList.h"
commit-bot@chromium.orgc3fe5492014-01-30 18:15:51 +000022#include "effects/GrConvexPolyEffect.h"
Brian Salomon576eda12017-07-14 11:30:17 -040023#include "ops/GrMeshDrawOp.h"
commit-bot@chromium.orgc3fe5492014-01-30 18:15:51 +000024
bsalomon342bfc22016-04-01 06:06:20 -070025/** outset rendered rect to visualize anti-aliased poly edges */
26static SkRect outset(const SkRect& unsorted) {
27 SkRect r = unsorted;
28 r.outset(5.f, 5.f);
29 return r;
30}
joshualitt95964c62015-02-11 13:45:50 -080031
bsalomon342bfc22016-04-01 06:06:20 -070032/** sorts a rect */
33static SkRect sorted_rect(const SkRect& unsorted) {
34 SkRect r = unsorted;
35 r.sort();
36 return r;
37}
38
39namespace skiagm {
Brian Salomon576eda12017-07-14 11:30:17 -040040class PolyBoundsOp : public GrMeshDrawOp {
joshualitt95964c62015-02-11 13:45:50 -080041public:
Brian Salomon25a88092016-12-01 09:36:50 -050042 DEFINE_OP_CLASS_ID
joshualitt95964c62015-02-11 13:45:50 -080043
Robert Phillips7c525e62018-06-12 10:11:12 -040044 static std::unique_ptr<GrDrawOp> Make(GrContext* context,
45 GrPaint&& paint,
46 const SkRect& rect) {
Robert Phillips9da87e02019-02-04 13:26:26 -050047 GrOpMemoryPool* pool = context->priv().opMemoryPool();
Robert Phillipsc994a932018-06-19 13:09:54 -040048
49 return pool->allocate<PolyBoundsOp>(std::move(paint), rect);
Brian Salomon576eda12017-07-14 11:30:17 -040050 }
51
Robert Phillipsb493eeb2017-09-13 13:10:52 -040052 const char* name() const override { return "PolyBoundsOp"; }
53
Brian Salomon7d94bb52018-10-12 14:37:19 -040054 void visitProxies(const VisitProxyFunc& func, VisitorType) const override {
Robert Phillipsb493eeb2017-09-13 13:10:52 -040055 fProcessors.visitProxies(func);
56 }
57
Brian Salomon576eda12017-07-14 11:30:17 -040058 FixedFunctionFlags fixedFunctionFlags() const override { return FixedFunctionFlags::kNone; }
59
Chris Dalton4b62aed2019-01-15 11:53:00 -070060 GrProcessorSet::Analysis finalize(const GrCaps& caps, const GrAppliedClip* clip) override {
61 return fProcessors.finalize(fColor, GrProcessorAnalysisCoverage::kNone, clip, false, caps,
62 &fColor);
joshualitt95964c62015-02-11 13:45:50 -080063 }
64
65private:
Robert Phillips7c525e62018-06-12 10:11:12 -040066 friend class ::GrOpMemoryPool; // for ctor
67
Brian Salomon576eda12017-07-14 11:30:17 -040068 PolyBoundsOp(GrPaint&& paint, const SkRect& rect)
69 : INHERITED(ClassID())
Brian Osmancf860852018-10-31 14:04:39 -040070 , fColor(paint.getColor4f())
Brian Salomon576eda12017-07-14 11:30:17 -040071 , fProcessors(std::move(paint))
72 , fRect(outset(rect)) {
73 this->setBounds(sorted_rect(fRect), HasAABloat::kNo, IsZeroArea::kNo);
74 }
Brian Salomon6b316e92016-12-16 09:35:49 -050075
Brian Salomon91326c32017-08-09 16:02:19 -040076 void onPrepareDraws(Target* target) override {
bsalomon342bfc22016-04-01 06:06:20 -070077 using namespace GrDefaultGeoProcFactory;
joshualitt95964c62015-02-11 13:45:50 -080078
Brian Salomon576eda12017-07-14 11:30:17 -040079 Color color(fColor);
Brian Salomon8c852be2017-01-04 10:44:42 -050080 sk_sp<GrGeometryProcessor> gp(GrDefaultGeoProcFactory::Make(
Ruiqi Maob609e6d2018-07-17 10:19:38 -040081 target->caps().shaderCaps(),
82 color,
83 Coverage::kSolid_Type,
84 LocalCoords::kUnused_Type,
85 SkMatrix::I()));
joshualitt95964c62015-02-11 13:45:50 -080086
Brian Osmanf04fb3c2018-11-12 15:34:00 -050087 SkASSERT(gp->vertexStride() == sizeof(SkPoint));
Brian Salomon7eae3e02018-08-07 14:02:38 +000088 QuadHelper helper(target, sizeof(SkPoint), 1);
89 SkPoint* verts = reinterpret_cast<SkPoint*>(helper.vertices());
bsalomonb5238a72015-05-05 07:49:49 -070090 if (!verts) {
joshualitt4b31de82015-03-05 14:33:41 -080091 return;
92 }
93
Brian Salomonec42e152018-05-18 12:52:22 -040094 SkPointPriv::SetRectTriStrip(verts, fRect, sizeof(SkPoint));
joshualitt95964c62015-02-11 13:45:50 -080095
Brian Salomon49348902018-06-26 09:12:38 -040096 auto pipe = target->makePipeline(0, std::move(fProcessors), target->detachAppliedClip());
Brian Salomon7eae3e02018-08-07 14:02:38 +000097 helper.recordDraw(target, std::move(gp), pipe.fPipeline, pipe.fFixedDynamicState);
joshualitt95964c62015-02-11 13:45:50 -080098 }
99
Brian Osmancf860852018-10-31 14:04:39 -0400100 SkPMColor4f fColor;
Brian Salomon576eda12017-07-14 11:30:17 -0400101 GrProcessorSet fProcessors;
bsalomon342bfc22016-04-01 06:06:20 -0700102 SkRect fRect;
joshualitt95964c62015-02-11 13:45:50 -0800103
Brian Salomon576eda12017-07-14 11:30:17 -0400104 typedef GrMeshDrawOp INHERITED;
joshualitt95964c62015-02-11 13:45:50 -0800105};
106
commit-bot@chromium.orgc3fe5492014-01-30 18:15:51 +0000107/**
joshualittb0a8a372014-09-23 09:50:21 -0700108 * This GM directly exercises a GrProcessor that draws convex polygons.
commit-bot@chromium.orgc3fe5492014-01-30 18:15:51 +0000109 */
110class ConvexPolyEffect : public GM {
111public:
112 ConvexPolyEffect() {
113 this->setBGColor(0xFFFFFFFF);
114 }
115
116protected:
mtklein36352bf2015-03-25 18:17:31 -0700117 SkString onShortName() override {
commit-bot@chromium.orgc3fe5492014-01-30 18:15:51 +0000118 return SkString("convex_poly_effect");
119 }
120
mtklein36352bf2015-03-25 18:17:31 -0700121 SkISize onISize() override {
tfarinaf5393182014-06-09 23:59:03 -0700122 return SkISize::Make(720, 800);
commit-bot@chromium.orgc3fe5492014-01-30 18:15:51 +0000123 }
124
mtklein36352bf2015-03-25 18:17:31 -0700125 void onOnceBeforeDraw() override {
commit-bot@chromium.orgc3fe5492014-01-30 18:15:51 +0000126 SkPath tri;
127 tri.moveTo(5.f, 5.f);
128 tri.lineTo(100.f, 20.f);
129 tri.lineTo(15.f, 100.f);
skia.committer@gmail.com4c18e9f2014-01-31 03:01:59 +0000130
commit-bot@chromium.orgc3fe5492014-01-30 18:15:51 +0000131 fPaths.addToTail(tri);
132 fPaths.addToTail(SkPath())->reverseAddPath(tri);
skia.committer@gmail.com4c18e9f2014-01-31 03:01:59 +0000133
commit-bot@chromium.orgc3fe5492014-01-30 18:15:51 +0000134 tri.close();
135 fPaths.addToTail(tri);
skia.committer@gmail.com4c18e9f2014-01-31 03:01:59 +0000136
commit-bot@chromium.orgc3fe5492014-01-30 18:15:51 +0000137 SkPath ngon;
mtkleindbfd7ab2016-09-01 11:24:54 -0700138 constexpr SkScalar kRadius = 50.f;
commit-bot@chromium.orgc3fe5492014-01-30 18:15:51 +0000139 const SkPoint center = { kRadius, kRadius };
140 for (int i = 0; i < GrConvexPolyEffect::kMaxEdges; ++i) {
141 SkScalar angle = 2 * SK_ScalarPI * i / GrConvexPolyEffect::kMaxEdges;
142 SkPoint point;
143 point.fY = SkScalarSinCos(angle, &point.fX);
144 point.scale(kRadius);
145 point = center + point;
146 if (0 == i) {
147 ngon.moveTo(point);
148 } else {
149 ngon.lineTo(point);
150 }
151 }
skia.committer@gmail.com4c18e9f2014-01-31 03:01:59 +0000152
commit-bot@chromium.orgc3fe5492014-01-30 18:15:51 +0000153 fPaths.addToTail(ngon);
154 SkMatrix scaleM;
155 scaleM.setScale(1.1f, 0.4f);
156 ngon.transform(scaleM);
157 fPaths.addToTail(ngon);
skia.committer@gmail.comf0b0cda2014-02-09 03:02:01 +0000158
bsalomon7888de02016-03-28 15:04:45 -0700159 SkPath linePath;
160 linePath.moveTo(5.f, 5.f);
161 linePath.lineTo(6.f, 6.f);
162 fPaths.addToTail(linePath);
163
commit-bot@chromium.orgf0539802014-02-08 19:31:05 +0000164 // integer edges
165 fRects.addToTail(SkRect::MakeLTRB(5.f, 1.f, 30.f, 25.f));
166 // half-integer edges
167 fRects.addToTail(SkRect::MakeLTRB(5.5f, 0.5f, 29.5f, 24.5f));
168 // vertically/horizontally thin rects that cover pixel centers
169 fRects.addToTail(SkRect::MakeLTRB(5.25f, 0.5f, 5.75f, 24.5f));
170 fRects.addToTail(SkRect::MakeLTRB(5.5f, 0.5f, 29.5f, 0.75f));
171 // vertically/horizontally thin rects that don't cover pixel centers
172 fRects.addToTail(SkRect::MakeLTRB(5.55f, 0.5f, 5.75f, 24.5f));
173 fRects.addToTail(SkRect::MakeLTRB(5.5f, .05f, 29.5f, .25f));
174 // small in x and y
175 fRects.addToTail(SkRect::MakeLTRB(5.05f, .55f, 5.45f, .85f));
176 // inverted in x and y
177 fRects.addToTail(SkRect::MakeLTRB(100.f, 50.5f, 5.f, 0.5f));
commit-bot@chromium.orgc3fe5492014-01-30 18:15:51 +0000178 }
179
mtklein36352bf2015-03-25 18:17:31 -0700180 void onDraw(SkCanvas* canvas) override {
Brian Osman11052242016-10-27 14:47:55 -0400181 GrRenderTargetContext* renderTargetContext =
182 canvas->internal_private_accessTopLayerRenderTargetContext();
183 if (!renderTargetContext) {
robertphillips175dd9b2016-04-28 14:32:04 -0700184 skiagm::GM::DrawGpuOnlyMessage(canvas);
joshualittf5883a62016-01-13 07:47:38 -0800185 return;
186 }
187
Robert Phillips7c525e62018-06-12 10:11:12 -0400188 GrContext* context = canvas->getGrContext();
189 if (!context) {
190 return;
191 }
192
commit-bot@chromium.orgc3fe5492014-01-30 18:15:51 +0000193 SkScalar y = 0;
mtkleindbfd7ab2016-09-01 11:24:54 -0700194 constexpr SkScalar kDX = 12.f;
bsalomonf045d602015-11-18 19:01:12 -0800195 for (PathList::Iter iter(fPaths, PathList::Iter::kHead_IterStart);
bsalomon49f085d2014-09-05 13:34:00 -0700196 iter.get();
commit-bot@chromium.orgc3fe5492014-01-30 18:15:51 +0000197 iter.next()) {
commit-bot@chromium.orgf0539802014-02-08 19:31:05 +0000198 const SkPath* path = iter.get();
commit-bot@chromium.orgc3fe5492014-01-30 18:15:51 +0000199 SkScalar x = 0;
200
Ethan Nicholas1706f842017-11-10 11:58:19 -0500201 for (int et = 0; et < kGrClipEdgeTypeCnt; ++et) {
robertphillips1d24b8d2015-03-26 19:57:08 -0700202 const SkMatrix m = SkMatrix::MakeTrans(x, y);
commit-bot@chromium.orgc3fe5492014-01-30 18:15:51 +0000203 SkPath p;
commit-bot@chromium.orgc3fe5492014-01-30 18:15:51 +0000204 path->transform(m, &p);
skia.committer@gmail.com4c18e9f2014-01-31 03:01:59 +0000205
Ethan Nicholas0f3c7322017-11-09 14:51:17 -0500206 GrClipEdgeType edgeType = (GrClipEdgeType) et;
Ethan Nicholaseace9352018-10-15 20:09:54 +0000207 std::unique_ptr<GrFragmentProcessor> fp(GrConvexPolyEffect::Make(edgeType, p));
joshualittb0a8a372014-09-23 09:50:21 -0700208 if (!fp) {
commit-bot@chromium.orgcabf4b22014-03-05 18:27:43 +0000209 continue;
commit-bot@chromium.orgc3fe5492014-01-30 18:15:51 +0000210 }
skia.committer@gmail.com4c18e9f2014-01-31 03:01:59 +0000211
robertphillips28a838e2016-06-23 14:07:00 -0700212 GrPaint grPaint;
Brian Osmancb3d0872018-10-16 15:19:28 -0400213 grPaint.setColor4f({ 0, 0, 0, 1.f });
Brian Salomona1633922017-01-09 11:46:10 -0500214 grPaint.setXPFactory(GrPorterDuffXPFactory::Get(SkBlendMode::kSrc));
robertphillips28a838e2016-06-23 14:07:00 -0700215 grPaint.addCoverageFragmentProcessor(std::move(fp));
joshualitt9853cce2014-11-17 14:22:48 -0800216
Brian Salomon576eda12017-07-14 11:30:17 -0400217 std::unique_ptr<GrDrawOp> op =
Robert Phillips7c525e62018-06-12 10:11:12 -0400218 PolyBoundsOp::Make(context, std::move(grPaint), p.getBounds());
Brian Salomon576eda12017-07-14 11:30:17 -0400219 renderTargetContext->priv().testingOnly_addDrawOp(std::move(op));
skia.committer@gmail.com4c18e9f2014-01-31 03:01:59 +0000220
bsalomon7888de02016-03-28 15:04:45 -0700221 x += SkScalarCeilToScalar(path->getBounds().width() + kDX);
commit-bot@chromium.orgc3fe5492014-01-30 18:15:51 +0000222 }
skia.committer@gmail.com4c18e9f2014-01-31 03:01:59 +0000223
commit-bot@chromium.orgc3fe5492014-01-30 18:15:51 +0000224 // Draw AA and non AA paths using normal API for reference.
225 canvas->save();
226 canvas->translate(x, y);
227 SkPaint paint;
228 canvas->drawPath(*path, paint);
229 canvas->translate(path->getBounds().width() + 10.f, 0);
230 paint.setAntiAlias(true);
231 canvas->drawPath(*path, paint);
232 canvas->restore();
skia.committer@gmail.com4c18e9f2014-01-31 03:01:59 +0000233
commit-bot@chromium.orgf0539802014-02-08 19:31:05 +0000234 y += SkScalarCeilToScalar(path->getBounds().height() + 20.f);
235 }
236
bsalomonf045d602015-11-18 19:01:12 -0800237 for (RectList::Iter iter(fRects, RectList::Iter::kHead_IterStart);
bsalomon49f085d2014-09-05 13:34:00 -0700238 iter.get();
commit-bot@chromium.orgf0539802014-02-08 19:31:05 +0000239 iter.next()) {
240
241 SkScalar x = 0;
242
Ethan Nicholas1706f842017-11-10 11:58:19 -0500243 for (int et = 0; et < kGrClipEdgeTypeCnt; ++et) {
commit-bot@chromium.orgd85f32c2014-02-28 14:43:26 +0000244 SkRect rect = *iter.get();
245 rect.offset(x, y);
Ethan Nicholas0f3c7322017-11-09 14:51:17 -0500246 GrClipEdgeType edgeType = (GrClipEdgeType) et;
Ethan Nicholaseace9352018-10-15 20:09:54 +0000247 std::unique_ptr<GrFragmentProcessor> fp(GrConvexPolyEffect::Make(edgeType, rect));
joshualittb0a8a372014-09-23 09:50:21 -0700248 if (!fp) {
commit-bot@chromium.orgcabf4b22014-03-05 18:27:43 +0000249 continue;
commit-bot@chromium.orgd85f32c2014-02-28 14:43:26 +0000250 }
251
robertphillips28a838e2016-06-23 14:07:00 -0700252 GrPaint grPaint;
Brian Osmancb3d0872018-10-16 15:19:28 -0400253 grPaint.setColor4f({ 0, 0, 0, 1.f });
Brian Salomona1633922017-01-09 11:46:10 -0500254 grPaint.setXPFactory(GrPorterDuffXPFactory::Get(SkBlendMode::kSrc));
robertphillips28a838e2016-06-23 14:07:00 -0700255 grPaint.addCoverageFragmentProcessor(std::move(fp));
commit-bot@chromium.orgd85f32c2014-02-28 14:43:26 +0000256
Robert Phillips7c525e62018-06-12 10:11:12 -0400257 std::unique_ptr<GrDrawOp> op = PolyBoundsOp::Make(context, std::move(grPaint),
258 rect);
Brian Salomon576eda12017-07-14 11:30:17 -0400259 renderTargetContext->priv().testingOnly_addDrawOp(std::move(op));
commit-bot@chromium.orgd85f32c2014-02-28 14:43:26 +0000260
bsalomon7888de02016-03-28 15:04:45 -0700261 x += SkScalarCeilToScalar(rect.width() + kDX);
commit-bot@chromium.orgf0539802014-02-08 19:31:05 +0000262 }
263
commit-bot@chromium.orgd85f32c2014-02-28 14:43:26 +0000264 // Draw rect without and with AA using normal API for reference
commit-bot@chromium.orgf0539802014-02-08 19:31:05 +0000265 canvas->save();
266 canvas->translate(x, y);
267 SkPaint paint;
commit-bot@chromium.orgd85f32c2014-02-28 14:43:26 +0000268 canvas->drawRect(*iter.get(), paint);
bsalomon7888de02016-03-28 15:04:45 -0700269 x += SkScalarCeilToScalar(iter.get()->width() + kDX);
commit-bot@chromium.orgf0539802014-02-08 19:31:05 +0000270 paint.setAntiAlias(true);
271 canvas->drawRect(*iter.get(), paint);
272 canvas->restore();
273
commit-bot@chromium.orgd85f32c2014-02-28 14:43:26 +0000274 y += SkScalarCeilToScalar(iter.get()->height() + 20.f);
commit-bot@chromium.orgc3fe5492014-01-30 18:15:51 +0000275 }
276 }
277
278private:
bsalomonf045d602015-11-18 19:01:12 -0800279 typedef SkTLList<SkPath, 1> PathList;
280 typedef SkTLList<SkRect, 1> RectList;
281 PathList fPaths;
282 RectList fRects;
skia.committer@gmail.comf0b0cda2014-02-09 03:02:01 +0000283
commit-bot@chromium.orgc3fe5492014-01-30 18:15:51 +0000284 typedef GM INHERITED;
285};
286
halcanary385fe4d2015-08-26 13:07:48 -0700287DEF_GM(return new ConvexPolyEffect;)
commit-bot@chromium.orgc3fe5492014-01-30 18:15:51 +0000288}