blob: 2255c6624aece6053061496583a44e445c43ad4f [file] [log] [blame]
bsalomon@google.com48dd1a22011-10-31 14:18:20 +00001/*
2 * Copyright 2011 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
Mike Kleinc0bd9f92019-04-23 12:05:21 -05008#include "gm/gm.h"
Tyler Denniston45f94f82020-02-04 16:09:08 -05009#include "gm/verifiers/gmverifier.h"
Ben Wagner7fde8e12019-05-01 17:28:53 -040010#include "include/core/SkBitmap.h"
11#include "include/core/SkBlendMode.h"
12#include "include/core/SkCanvas.h"
Ben Wagner7fde8e12019-05-01 17:28:53 -040013#include "include/core/SkFont.h"
14#include "include/core/SkFontTypes.h"
15#include "include/core/SkMatrix.h"
16#include "include/core/SkPaint.h"
17#include "include/core/SkRect.h"
Mike Kleinc0bd9f92019-04-23 12:05:21 -050018#include "include/core/SkShader.h"
Ben Wagner7fde8e12019-05-01 17:28:53 -040019#include "include/core/SkTileMode.h"
20#include "include/core/SkTypeface.h"
Robert Phillipsb7bfbc22020-07-01 12:55:01 -040021#include "include/gpu/GrRecordingContext.h"
Brian Salomon8f7d9532020-12-23 09:16:59 -050022#include "src/core/SkCanvasPriv.h"
Mike Kleinc0bd9f92019-04-23 12:05:21 -050023#include "src/core/SkTraceEvent.h"
24#include "tools/ToolUtils.h"
Ben Wagner7fde8e12019-05-01 17:28:53 -040025
26#include <stdarg.h>
27
Brian Salomoneebe7352020-12-09 16:37:04 -050028class GrSurfaceDrawContext;
Ben Wagner7fde8e12019-05-01 17:28:53 -040029
bsalomon@google.com48dd1a22011-10-31 14:18:20 +000030using namespace skiagm;
31
Chris Dalton50e24d72019-02-07 16:20:09 -070032constexpr char GM::kErrorMsg_DrawSkippedGpuOnly[];
33
34static void draw_failure_message(SkCanvas* canvas, const char format[], ...) {
35 SkString failureMsg;
36
37 va_list argp;
38 va_start(argp, format);
39 failureMsg.appendVAList(format, argp);
40 va_end(argp);
41
42 constexpr SkScalar kOffset = 5.0f;
43 canvas->drawColor(SkColorSetRGB(200,0,0));
44 SkFont font;
45 SkRect bounds;
Ben Wagner51e15a62019-05-07 15:38:46 -040046 font.measureText(failureMsg.c_str(), failureMsg.size(), SkTextEncoding::kUTF8, &bounds);
Hal Canary0a7b3932019-05-02 11:31:28 -040047 SkPaint textPaint(SkColors::kWhite);
Chris Dalton50e24d72019-02-07 16:20:09 -070048 canvas->drawString(failureMsg, kOffset, bounds.height() + kOffset, font, textPaint);
49}
50
51static void draw_gpu_only_message(SkCanvas* canvas) {
52 SkBitmap bmp;
53 bmp.allocN32Pixels(128, 64);
54 SkCanvas bmpCanvas(bmp);
55 bmpCanvas.drawColor(SK_ColorWHITE);
Mike Kleinea3f0142019-03-20 11:12:10 -050056 SkFont font(ToolUtils::create_portable_typeface(), 20);
Hal Canary0a7b3932019-05-02 11:31:28 -040057 SkPaint paint(SkColors::kRed);
Chris Dalton50e24d72019-02-07 16:20:09 -070058 bmpCanvas.drawString("GPU Only", 20, 40, font, paint);
59 SkMatrix localM;
60 localM.setRotate(35.f);
61 localM.postTranslate(10.f, 0.f);
Mike Reedb41bd152020-12-12 11:18:31 -050062 paint.setShader(bmp.makeShader(SkTileMode::kMirror, SkTileMode::kMirror,
63 SkSamplingOptions(SkFilterMode::kLinear,
64 SkMipmapMode::kNearest),
65 localM));
Chris Dalton50e24d72019-02-07 16:20:09 -070066 canvas->drawPaint(paint);
67}
68
Robert Phillipsd26d25e2020-06-25 13:26:22 -040069static void handle_gm_failure(SkCanvas* canvas, DrawResult result, const SkString& errorMsg) {
70 if (DrawResult::kFail == result) {
71 draw_failure_message(canvas, "DRAW FAILED: %s", errorMsg.c_str());
72 } else if (SkString(GM::kErrorMsg_DrawSkippedGpuOnly) == errorMsg) {
73 draw_gpu_only_message(canvas);
74 } else {
75 draw_failure_message(canvas, "DRAW SKIPPED: %s", errorMsg.c_str());
76 }
77}
78
Chris Dalton3a778372019-02-07 15:23:36 -070079GM::GM(SkColor bgColor) {
commit-bot@chromium.orgb21fac12014-02-07 21:13:11 +000080 fMode = kGM_Mode;
Chris Dalton3a778372019-02-07 15:23:36 -070081 fBGColor = bgColor;
bsalomon@google.com48dd1a22011-10-31 14:18:20 +000082}
tfarina880914c2014-06-09 12:05:34 -070083
bsalomon@google.com48dd1a22011-10-31 14:18:20 +000084GM::~GM() {}
85
Robert Phillipsb87b39b2020-07-01 14:45:24 -040086DrawResult GM::gpuSetup(GrDirectContext* context, SkCanvas* canvas, SkString* errorMsg) {
Robert Phillips889d6132020-06-16 11:11:33 -040087 TRACE_EVENT1("GM", TRACE_FUNC, "name", TRACE_STR_COPY(this->getName()));
Robert Phillipse9229532020-06-26 10:10:49 -040088 if (!fGpuSetup) {
89 // When drawn in viewer, gpuSetup will be called multiple times with the same
90 // GrContext.
91 fGpuSetup = true;
92 fGpuSetupResult = this->onGpuSetup(context, errorMsg);
Robert Phillipsd26d25e2020-06-25 13:26:22 -040093 }
Robert Phillipse9229532020-06-26 10:10:49 -040094 if (DrawResult::kOk != fGpuSetupResult) {
95 handle_gm_failure(canvas, fGpuSetupResult, *errorMsg);
96 }
97
98 return fGpuSetupResult;
Robert Phillips889d6132020-06-16 11:11:33 -040099}
100
Robert Phillipsb795bea2020-06-25 12:38:53 -0400101void GM::gpuTeardown() {
102 this->onGpuTeardown();
Robert Phillipse9229532020-06-26 10:10:49 -0400103
104 // After 'gpuTeardown' a GM can be reused with a different GrContext. Reset the flag
105 // so 'onGpuSetup' will be called.
106 fGpuSetup = false;
Robert Phillipsb795bea2020-06-25 12:38:53 -0400107}
108
Chris Dalton50e24d72019-02-07 16:20:09 -0700109DrawResult GM::draw(SkCanvas* canvas, SkString* errorMsg) {
Mike Kleinb323a5e2017-07-24 15:21:31 -0400110 TRACE_EVENT1("GM", TRACE_FUNC, "name", TRACE_STR_COPY(this->getName()));
bsalomon@google.com48dd1a22011-10-31 14:18:20 +0000111 this->drawBackground(canvas);
Chris Dalton50e24d72019-02-07 16:20:09 -0700112 return this->drawContent(canvas, errorMsg);
bsalomon@google.com48dd1a22011-10-31 14:18:20 +0000113}
114
Chris Dalton50e24d72019-02-07 16:20:09 -0700115DrawResult GM::drawContent(SkCanvas* canvas, SkString* errorMsg) {
Mike Kleinb323a5e2017-07-24 15:21:31 -0400116 TRACE_EVENT0("GM", TRACE_FUNC);
Robert Phillips83b749a2020-06-25 11:41:19 -0400117 this->onceBeforeDraw();
Mike Reed36c4fb32018-12-18 11:48:01 -0500118 SkAutoCanvasRestore acr(canvas, true);
Chris Dalton50e24d72019-02-07 16:20:09 -0700119 DrawResult drawResult = this->onDraw(canvas, errorMsg);
120 if (DrawResult::kOk != drawResult) {
Robert Phillipsd26d25e2020-06-25 13:26:22 -0400121 handle_gm_failure(canvas, drawResult, *errorMsg);
Chris Dalton50e24d72019-02-07 16:20:09 -0700122 }
123 return drawResult;
bsalomon@google.com48dd1a22011-10-31 14:18:20 +0000124}
125
126void GM::drawBackground(SkCanvas* canvas) {
Mike Kleinb323a5e2017-07-24 15:21:31 -0400127 TRACE_EVENT0("GM", TRACE_FUNC);
Robert Phillips83b749a2020-06-25 11:41:19 -0400128 this->onceBeforeDraw();
Chris Dalton21ca3702019-02-01 12:15:42 -0700129 canvas->drawColor(fBGColor, SkBlendMode::kSrc);
bsalomon@google.com48dd1a22011-10-31 14:18:20 +0000130}
131
Mike Klein9707e902019-02-07 16:18:22 -0500132DrawResult GM::onDraw(SkCanvas* canvas, SkString* errorMsg) {
133 this->onDraw(canvas);
134 return DrawResult::kOk;
135}
136void GM::onDraw(SkCanvas*) { SK_ABORT("Not implemented."); }
137
138
139SkISize SimpleGM::onISize() { return fSize; }
140SkString SimpleGM::onShortName() { return fName; }
141DrawResult SimpleGM::onDraw(SkCanvas* canvas, SkString* errorMsg) {
142 return fDrawProc(canvas, errorMsg);
143}
144
145SkISize SimpleGpuGM::onISize() { return fSize; }
146SkString SimpleGpuGM::onShortName() { return fName; }
Brian Salomoneebe7352020-12-09 16:37:04 -0500147DrawResult SimpleGpuGM::onDraw(GrRecordingContext* ctx, GrSurfaceDrawContext* rtc,
Robert Phillips95c250c2020-06-29 15:36:12 -0400148 SkCanvas* canvas, SkString* errorMsg) {
Mike Klein9707e902019-02-07 16:18:22 -0500149 return fDrawProc(ctx, rtc, canvas, errorMsg);
150}
151
commit-bot@chromium.org38aeb0f2014-02-26 23:01:57 +0000152const char* GM::getName() {
bsalomon@google.com48dd1a22011-10-31 14:18:20 +0000153 if (fShortName.size() == 0) {
154 fShortName = this->onShortName();
155 }
156 return fShortName.c_str();
157}
158
159void GM::setBGColor(SkColor color) {
160 fBGColor = color;
161}
162
Hal Canary41248072019-07-11 16:32:53 -0400163bool GM::animate(double nanos) { return this->onAnimate(nanos); }
reedd9adfe62015-02-01 19:01:04 -0800164
Mike Klein9707e902019-02-07 16:18:22 -0500165bool GM::runAsBench() const { return false; }
166void GM::modifyGrContextOptions(GrContextOptions* options) {}
167
Tyler Denniston45f94f82020-02-04 16:09:08 -0500168std::unique_ptr<verifiers::VerifierList> GM::getVerifiers() const {
169 // No verifiers by default.
170 return nullptr;
171}
172
Mike Klein9707e902019-02-07 16:18:22 -0500173void GM::onOnceBeforeDraw() {}
174
Hal Canary41248072019-07-11 16:32:53 -0400175bool GM::onAnimate(double /*nanos*/) { return false; }
Hal Canaryc74a5502019-07-08 14:55:15 -0400176
177bool GM::onChar(SkUnichar uni) { return false; }
178
Mike Klein9707e902019-02-07 16:18:22 -0500179bool GM::onGetControls(SkMetaData*) { return false; }
Hal Canaryc74a5502019-07-08 14:55:15 -0400180
Mike Klein9707e902019-02-07 16:18:22 -0500181void GM::onSetControls(const SkMetaData&) {}
182
reedd9adfe62015-02-01 19:01:04 -0800183/////////////////////////////////////////////////////////////////////////////////////////////
184
reed@google.com2d6ef522012-01-03 17:20:38 +0000185void GM::drawSizeBounds(SkCanvas* canvas, SkColor color) {
Hal Canary0a7b3932019-05-02 11:31:28 -0400186 canvas->drawRect(SkRect::Make(this->getISize()), SkPaint(SkColor4f::FromColor(color)));
reed@google.com2d6ef522012-01-03 17:20:38 +0000187}
188
bsalomon@google.com48dd1a22011-10-31 14:18:20 +0000189// need to explicitly declare this, or we get some weird infinite loop llist
tfarinabcbc1782014-06-18 14:32:48 -0700190template GMRegistry* GMRegistry::gHead;
halcanaryf62c6342015-01-12 15:27:46 -0800191
Brian Salomoneebe7352020-12-09 16:37:04 -0500192DrawResult GpuGM::onDraw(GrRecordingContext* ctx, GrSurfaceDrawContext* rtc, SkCanvas* canvas,
Robert Phillips889d6132020-06-16 11:11:33 -0400193 SkString* errorMsg) {
Mike Klein9707e902019-02-07 16:18:22 -0500194 this->onDraw(ctx, rtc, canvas);
195 return DrawResult::kOk;
196}
Brian Salomoneebe7352020-12-09 16:37:04 -0500197void GpuGM::onDraw(GrRecordingContext*, GrSurfaceDrawContext*, SkCanvas*) {
Mike Klein9707e902019-02-07 16:18:22 -0500198 SK_ABORT("Not implemented.");
199}
200
Chris Dalton50e24d72019-02-07 16:20:09 -0700201DrawResult GpuGM::onDraw(SkCanvas* canvas, SkString* errorMsg) {
Robert Phillips95c250c2020-06-29 15:36:12 -0400202
203 auto ctx = canvas->recordingContext();
Brian Salomon8f7d9532020-12-23 09:16:59 -0500204 GrSurfaceDrawContext* sdc = SkCanvasPriv::TopDeviceSurfaceDrawContext(canvas);
205 if (!ctx || !sdc) {
Chris Dalton50e24d72019-02-07 16:20:09 -0700206 *errorMsg = kErrorMsg_DrawSkippedGpuOnly;
207 return DrawResult::kSkip;
Chris Dalton3a778372019-02-07 15:23:36 -0700208 }
Robert Phillips9eb00022020-06-30 15:30:12 -0400209 if (ctx->abandoned()) {
Chris Dalton50e24d72019-02-07 16:20:09 -0700210 *errorMsg = "GrContext abandoned.";
Mike Klein290690c2019-04-02 18:11:57 -0400211 return DrawResult::kSkip;
Chris Dalton3a778372019-02-07 15:23:36 -0700212 }
Brian Salomon8f7d9532020-12-23 09:16:59 -0500213 return this->onDraw(ctx, sdc, canvas, errorMsg);
halcanaryf62c6342015-01-12 15:27:46 -0800214}
Mike Kleinc9eace82018-10-31 10:49:38 -0400215
216template <typename Fn>
217static void mark(SkCanvas* canvas, SkScalar x, SkScalar y, Fn&& fn) {
218 SkPaint alpha;
219 alpha.setAlpha(0x50);
220 canvas->saveLayer(nullptr, &alpha);
221 canvas->translate(x,y);
222 canvas->scale(2,2);
223 fn();
224 canvas->restore();
225}
226
227void MarkGMGood(SkCanvas* canvas, SkScalar x, SkScalar y) {
228 mark(canvas, x,y, [&]{
Mike Kleinc9eace82018-10-31 10:49:38 -0400229 // A green circle.
Hal Canary0a7b3932019-05-02 11:31:28 -0400230 canvas->drawCircle(0, 0, 12, SkPaint(SkColor4f::FromColor(SkColorSetRGB(27, 158, 119))));
Mike Kleinc9eace82018-10-31 10:49:38 -0400231
232 // Cut out a check mark.
Hal Canary0a7b3932019-05-02 11:31:28 -0400233 SkPaint paint(SkColors::kTransparent);
Mike Kleinc9eace82018-10-31 10:49:38 -0400234 paint.setBlendMode(SkBlendMode::kSrc);
Mike Kleinc9eace82018-10-31 10:49:38 -0400235 paint.setStrokeWidth(2);
236 paint.setStyle(SkPaint::kStroke_Style);
237 canvas->drawLine(-6, 0,
238 -1, 5, paint);
239 canvas->drawLine(-1, +5,
240 +7, -5, paint);
241 });
242}
243
244void MarkGMBad(SkCanvas* canvas, SkScalar x, SkScalar y) {
245 mark(canvas, x,y, [&] {
Mike Kleinc9eace82018-10-31 10:49:38 -0400246 // A red circle.
Hal Canary0a7b3932019-05-02 11:31:28 -0400247 canvas->drawCircle(0,0, 12, SkPaint(SkColor4f::FromColor(SkColorSetRGB(231, 41, 138))));
Mike Kleinc9eace82018-10-31 10:49:38 -0400248
249 // Cut out an 'X'.
Hal Canary0a7b3932019-05-02 11:31:28 -0400250 SkPaint paint(SkColors::kTransparent);
Mike Kleinc9eace82018-10-31 10:49:38 -0400251 paint.setBlendMode(SkBlendMode::kSrc);
Mike Kleinc9eace82018-10-31 10:49:38 -0400252 paint.setStrokeWidth(2);
253 paint.setStyle(SkPaint::kStroke_Style);
254 canvas->drawLine(-5,-5,
255 +5,+5, paint);
256 canvas->drawLine(+5,-5,
257 -5,+5, paint);
258 });
259}