1 /*
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 #include "Benchmark.h"
9 #include "SkCanvas.h"
10 #include "SkPaint.h"
11
12 #include <ctype.h>
13
14 /** This benchmark tests rendering rotated rectangles. It can optionally apply AA and/or change the
15 paint color between each rect in different ways using the ColorType enum. The xfermode used can
16 be specified as well.
17 */
18
19 enum ColorType {
20 kConstantOpaque_ColorType,
21 kConstantTransparent_ColorType,
22 kChangingOpaque_ColorType,
23 kChangingTransparent_ColorType,
24 kAlternatingOpaqueAndTransparent_ColorType,
25 };
26
start_color(ColorType ct)27 static inline SkColor start_color(ColorType ct) {
28 switch (ct) {
29 case kConstantOpaque_ColorType:
30 case kChangingOpaque_ColorType:
31 case kAlternatingOpaqueAndTransparent_ColorType:
32 return 0xFFA07040;
33 case kConstantTransparent_ColorType:
34 case kChangingTransparent_ColorType:
35 return 0x80A07040;
36 }
37 SkFAIL("Shouldn't reach here.");
38 return 0;
39 }
40
advance_color(SkColor old,ColorType ct,int step)41 static inline SkColor advance_color(SkColor old, ColorType ct, int step) {
42 if (kAlternatingOpaqueAndTransparent_ColorType == ct) {
43 ct = (step & 0x1) ? kChangingOpaque_ColorType : kChangingTransparent_ColorType ;
44 }
45 switch (ct) {
46 case kConstantOpaque_ColorType:
47 case kConstantTransparent_ColorType:
48 return old;
49 case kChangingOpaque_ColorType:
50 return 0xFF000000 | (old + 0x00010307);
51 case kChangingTransparent_ColorType:
52 return (0x00FFFFFF & (old + 0x00010307)) | 0x80000000;
53 case kAlternatingOpaqueAndTransparent_ColorType:
54 SkFAIL("Can't get here");
55 }
56 SkFAIL("Shouldn't reach here.");
57 return 0;
58 }
59
to_lower(const char * str)60 static SkString to_lower(const char* str) {
61 SkString lower(str);
62 for (size_t i = 0; i < lower.size(); i++) {
63 lower[i] = tolower(lower[i]);
64 }
65 return lower;
66 }
67
68 class RotRectBench: public Benchmark {
69 public:
RotRectBench(bool aa,ColorType ct,SkXfermode::Mode mode)70 RotRectBench(bool aa, ColorType ct, SkXfermode::Mode mode)
71 : fAA(aa)
72 , fColorType(ct)
73 , fMode(mode) {
74 this->makeName();
75 }
76
77 protected:
onGetName()78 const char* onGetName() override { return fName.c_str(); }
79
onDraw(const int loops,SkCanvas * canvas)80 void onDraw(const int loops, SkCanvas* canvas) override {
81 SkPaint paint;
82 paint.setAntiAlias(fAA);
83 paint.setXfermodeMode(fMode);
84 SkColor color = start_color(fColorType);
85
86 int w = this->getSize().x();
87 int h = this->getSize().y();
88
89 static const SkScalar kRectW = 25.1f;
90 static const SkScalar kRectH = 25.9f;
91
92 SkMatrix rotate;
93 // This value was chosen so that we frequently hit the axis-aligned case.
94 rotate.setRotate(30.f, kRectW / 2, kRectH / 2);
95 SkMatrix m = rotate;
96
97 SkScalar tx = 0, ty = 0;
98
99 for (int i = 0; i < loops; ++i) {
100 canvas->save();
101 canvas->translate(tx, ty);
102 canvas->concat(m);
103 paint.setColor(color);
104 color = advance_color(color, fColorType, i);
105
106 canvas->drawRect(SkRect::MakeWH(kRectW, kRectH), paint);
107 canvas->restore();
108
109 tx += kRectW + 2;
110 if (tx > w) {
111 tx = 0;
112 ty += kRectH + 2;
113 if (ty > h) {
114 ty = 0;
115 }
116 }
117
118 m.postConcat(rotate);
119 }
120 }
121
122 private:
makeName()123 void makeName() {
124 fName = "rotated_rects";
125 if (fAA) {
126 fName.append("_aa");
127 } else {
128 fName.append("_bw");
129 }
130 switch (fColorType) {
131 case kConstantOpaque_ColorType:
132 fName.append("_same_opaque");
133 break;
134 case kConstantTransparent_ColorType:
135 fName.append("_same_transparent");
136 break;
137 case kChangingOpaque_ColorType:
138 fName.append("_changing_opaque");
139 break;
140 case kChangingTransparent_ColorType:
141 fName.append("_changing_transparent");
142 break;
143 case kAlternatingOpaqueAndTransparent_ColorType:
144 fName.append("_alternating_transparent_and_opaque");
145 break;
146 }
147 fName.appendf("_%s", to_lower(SkXfermode::ModeName(fMode)).c_str());
148 }
149
150 bool fAA;
151 ColorType fColorType;
152 SkXfermode::Mode fMode;
153 SkString fName;
154
155 typedef Benchmark INHERITED;
156 };
157
158 // Choose kSrcOver because it always allows coverage and alpha to be conflated. kSrc only allows
159 // conflation when opaque, and kDarken because it isn't possilbe with standard GL blending.
160 DEF_BENCH(return new RotRectBench(true, kConstantOpaque_ColorType, SkXfermode::kSrcOver_Mode);)
161 DEF_BENCH(return new RotRectBench(true, kConstantTransparent_ColorType, SkXfermode::kSrcOver_Mode);)
162 DEF_BENCH(return new RotRectBench(true, kChangingOpaque_ColorType, SkXfermode::kSrcOver_Mode);)
163 DEF_BENCH(return new RotRectBench(true, kChangingTransparent_ColorType, SkXfermode::kSrcOver_Mode);)
164 DEF_BENCH(return new RotRectBench(true, kAlternatingOpaqueAndTransparent_ColorType, SkXfermode::kSrcOver_Mode);)
165
166 DEF_BENCH(return new RotRectBench(false, kConstantOpaque_ColorType, SkXfermode::kSrcOver_Mode);)
167 DEF_BENCH(return new RotRectBench(false, kConstantTransparent_ColorType, SkXfermode::kSrcOver_Mode);)
168 DEF_BENCH(return new RotRectBench(false, kChangingOpaque_ColorType, SkXfermode::kSrcOver_Mode);)
169 DEF_BENCH(return new RotRectBench(false, kChangingTransparent_ColorType, SkXfermode::kSrcOver_Mode);)
170 DEF_BENCH(return new RotRectBench(false, kAlternatingOpaqueAndTransparent_ColorType, SkXfermode::kSrcOver_Mode);)
171
172 DEF_BENCH(return new RotRectBench(true, kConstantOpaque_ColorType, SkXfermode::kSrc_Mode);)
173 DEF_BENCH(return new RotRectBench(true, kConstantTransparent_ColorType, SkXfermode::kSrc_Mode);)
174 DEF_BENCH(return new RotRectBench(true, kChangingOpaque_ColorType, SkXfermode::kSrc_Mode);)
175 DEF_BENCH(return new RotRectBench(true, kChangingTransparent_ColorType, SkXfermode::kSrc_Mode);)
176 DEF_BENCH(return new RotRectBench(true, kAlternatingOpaqueAndTransparent_ColorType, SkXfermode::kSrc_Mode);)
177
178 DEF_BENCH(return new RotRectBench(false, kConstantOpaque_ColorType, SkXfermode::kSrc_Mode);)
179 DEF_BENCH(return new RotRectBench(false, kConstantTransparent_ColorType, SkXfermode::kSrc_Mode);)
180 DEF_BENCH(return new RotRectBench(false, kChangingOpaque_ColorType, SkXfermode::kSrc_Mode);)
181 DEF_BENCH(return new RotRectBench(false, kChangingTransparent_ColorType, SkXfermode::kSrc_Mode);)
182 DEF_BENCH(return new RotRectBench(false, kAlternatingOpaqueAndTransparent_ColorType, SkXfermode::kSrc_Mode);)
183
184 DEF_BENCH(return new RotRectBench(true, kConstantOpaque_ColorType, SkXfermode::kDarken_Mode);)
185 DEF_BENCH(return new RotRectBench(true, kConstantTransparent_ColorType, SkXfermode::kDarken_Mode);)
186 DEF_BENCH(return new RotRectBench(true, kChangingOpaque_ColorType, SkXfermode::kDarken_Mode);)
187 DEF_BENCH(return new RotRectBench(true, kChangingTransparent_ColorType, SkXfermode::kDarken_Mode);)
188 DEF_BENCH(return new RotRectBench(true, kAlternatingOpaqueAndTransparent_ColorType, SkXfermode::kDarken_Mode);)
189
190 DEF_BENCH(return new RotRectBench(false, kConstantOpaque_ColorType, SkXfermode::kDarken_Mode);)
191 DEF_BENCH(return new RotRectBench(false, kConstantTransparent_ColorType, SkXfermode::kDarken_Mode);)
192 DEF_BENCH(return new RotRectBench(false, kChangingOpaque_ColorType, SkXfermode::kDarken_Mode);)
193 DEF_BENCH(return new RotRectBench(false, kChangingTransparent_ColorType, SkXfermode::kDarken_Mode);)
194 DEF_BENCH(return new RotRectBench(false, kAlternatingOpaqueAndTransparent_ColorType, SkXfermode::kDarken_Mode);)
195