blob: 501e86aaa459d85ce41f640ff0f46c9c826da9d7 [file] [log] [blame]
reed@android.combd700c32009-01-05 03:34:50 +00001#include "SkCanvas.h"
reed@android.comf523e252009-01-26 23:15:37 +00002#include "SkColorPriv.h"
reed@android.com3a859a02009-01-28 00:56:29 +00003#include "SkGraphics.h"
reed@android.comb398fe82009-01-07 11:47:57 +00004#include "SkImageEncoder.h"
reed@android.com6c924ad2009-03-31 03:48:49 +00005#include "SkNWayCanvas.h"
6#include "SkPicture.h"
reed@android.combd700c32009-01-05 03:34:50 +00007#include "SkString.h"
reed@android.com4bc19832009-01-19 20:08:35 +00008#include "SkTime.h"
reed@android.combd700c32009-01-05 03:34:50 +00009
10#include "SkBenchmark.h"
11
reed@android.com29348cb2009-08-04 18:17:15 +000012#ifdef ANDROID
13static void log_error(const char msg[]) { SkDebugf("%s", msg); }
14static void log_progress(const char msg[]) { SkDebugf("%s", msg); }
15#else
16static void log_error(const char msg[]) { fprintf(stderr, "%s", msg); }
17static void log_progress(const char msg[]) { printf("%s", msg); }
18#endif
19
20static void log_error(const SkString& str) { log_error(str.c_str()); }
21static void log_progress(const SkString& str) { log_progress(str.c_str()); }
22
23///////////////////////////////////////////////////////////////////////////////
24
reed@android.com6c924ad2009-03-31 03:48:49 +000025static void erase(SkBitmap& bm) {
26 if (bm.config() == SkBitmap::kA8_Config) {
27 bm.eraseColor(0);
28 } else {
29 bm.eraseColor(SK_ColorWHITE);
30 }
31}
32
33static bool equal(const SkBitmap& bm1, const SkBitmap& bm2) {
34 if (bm1.width() != bm2.width() ||
35 bm1.height() != bm2.height() ||
36 bm1.config() != bm2.config()) {
37 return false;
38 }
39
40 size_t pixelBytes = bm1.width() * bm1.bytesPerPixel();
41 for (int y = 0; y < bm1.height(); y++) {
42 if (memcmp(bm1.getAddr(0, y), bm2.getAddr(0, y), pixelBytes)) {
43 return false;
44 }
45 }
46
47 return true;
48}
49
reed@android.combd700c32009-01-05 03:34:50 +000050class Iter {
51public:
52 Iter() {
53 fBench = BenchRegistry::Head();
54 }
55
56 SkBenchmark* next() {
57 if (fBench) {
58 BenchRegistry::Factory f = fBench->factory();
59 fBench = fBench->next();
reed@android.coma70ba362009-01-22 17:41:57 +000060 return f(0);
reed@android.combd700c32009-01-05 03:34:50 +000061 }
62 return NULL;
63 }
64
65private:
66 const BenchRegistry* fBench;
67};
68
69static void make_filename(const char name[], SkString* path) {
70 path->set(name);
71 for (int i = 0; name[i]; i++) {
72 switch (name[i]) {
73 case '/':
74 case '\\':
75 case ' ':
76 case ':':
77 path->writable_str()[i] = '-';
78 break;
79 default:
80 break;
81 }
82 }
83}
84
reed@android.com4c7d3d62009-01-21 03:15:13 +000085static void saveFile(const char name[], const char config[], const char dir[],
86 const SkBitmap& bm) {
reed@android.com4c7d3d62009-01-21 03:15:13 +000087 SkBitmap copy;
88 if (!bm.copyTo(&copy, SkBitmap::kARGB_8888_Config)) {
89 return;
90 }
reed@android.comf523e252009-01-26 23:15:37 +000091
92 if (bm.config() == SkBitmap::kA8_Config) {
93 // turn alpha into gray-scale
94 size_t size = copy.getSize() >> 2;
95 SkPMColor* p = copy.getAddr32(0, 0);
96 for (size_t i = 0; i < size; i++) {
97 int c = (*p >> SK_A32_SHIFT) & 0xFF;
98 c = 255 - c;
99 c |= (c << 24) | (c << 16) | (c << 8);
100 *p++ = c | (SK_A32_MASK << SK_A32_SHIFT);
101 }
102 }
reed@android.com4c7d3d62009-01-21 03:15:13 +0000103
104 SkString str;
105 make_filename(name, &str);
106 str.appendf("_%s.png", config);
107 str.prepend(dir);
108 ::remove(str.c_str());
109 SkImageEncoder::EncodeFile(str.c_str(), copy, SkImageEncoder::kPNG_Type,
110 100);
reed@android.com4c7d3d62009-01-21 03:15:13 +0000111}
112
113static void performClip(SkCanvas* canvas, int w, int h) {
114 SkRect r;
115
116 r.set(SkIntToScalar(10), SkIntToScalar(10),
117 SkIntToScalar(w*2/3), SkIntToScalar(h*2/3));
118 canvas->clipRect(r, SkRegion::kIntersect_Op);
119
120 r.set(SkIntToScalar(w/3), SkIntToScalar(h/3),
121 SkIntToScalar(w-10), SkIntToScalar(h-10));
122 canvas->clipRect(r, SkRegion::kXOR_Op);
123}
124
125static void performRotate(SkCanvas* canvas, int w, int h) {
126 const SkScalar x = SkIntToScalar(w) / 2;
127 const SkScalar y = SkIntToScalar(h) / 2;
128
129 canvas->translate(x, y);
130 canvas->rotate(SkIntToScalar(35));
131 canvas->translate(-x, -y);
132}
133
reed@android.com387359e2009-08-04 01:51:09 +0000134static void performScale(SkCanvas* canvas, int w, int h) {
135 const SkScalar x = SkIntToScalar(w) / 2;
136 const SkScalar y = SkIntToScalar(h) / 2;
137
138 canvas->translate(x, y);
139 // just enough so we can't take the sprite case
140 canvas->scale(SK_Scalar1 * 99/100, SK_Scalar1 * 99/100);
141 canvas->translate(-x, -y);
142}
143
reed@android.com6c924ad2009-03-31 03:48:49 +0000144static void compare_pict_to_bitmap(SkPicture* pict, const SkBitmap& bm) {
145 SkBitmap bm2;
146
147 bm2.setConfig(bm.config(), bm.width(), bm.height());
148 bm2.allocPixels();
149 erase(bm2);
150
151 SkCanvas canvas(bm2);
152 canvas.drawPicture(*pict);
153
154 if (!equal(bm, bm2)) {
155 SkDebugf("----- compare_pict_to_bitmap failed\n");
156 }
157}
158
reed@android.com29348cb2009-08-04 18:17:15 +0000159static bool parse_bool_arg(char * const* argv, char* const* stop, bool* var) {
160 if (argv < stop) {
161 *var = atoi(*argv) != 0;
162 return true;
163 }
164 return false;
165}
166
reed@android.com4bc19832009-01-19 20:08:35 +0000167static const struct {
168 SkBitmap::Config fConfig;
169 const char* fName;
170} gConfigs[] = {
171 { SkBitmap::kARGB_8888_Config, "8888" },
172 { SkBitmap::kRGB_565_Config, "565", },
173 { SkBitmap::kARGB_4444_Config, "4444", },
174 { SkBitmap::kA8_Config, "A8", }
175};
176
reed@android.com4c7d3d62009-01-21 03:15:13 +0000177static int findConfig(const char config[]) {
178 for (size_t i = 0; i < SK_ARRAY_COUNT(gConfigs); i++) {
179 if (!strcmp(config, gConfigs[i].fName)) {
180 return i;
181 }
182 }
183 return -1;
184}
185
reed@android.combd700c32009-01-05 03:34:50 +0000186int main (int argc, char * const argv[]) {
reed@android.com3a859a02009-01-28 00:56:29 +0000187 SkAutoGraphics ag;
188
reed@android.com4bc19832009-01-19 20:08:35 +0000189 int repeatDraw = 1;
190 int forceAlpha = 0xFF;
191 bool forceAA = true;
reed@android.com29348cb2009-08-04 18:17:15 +0000192 bool forceFilter = false;
reed@android.com387359e2009-08-04 01:51:09 +0000193 bool doScale = false;
reed@android.com4c7d3d62009-01-21 03:15:13 +0000194 bool doRotate = false;
195 bool doClip = false;
reed@android.com6c924ad2009-03-31 03:48:49 +0000196 bool doPict = false;
reed@android.com387359e2009-08-04 01:51:09 +0000197 const char* matchStr = NULL;
reed@android.com4bc19832009-01-19 20:08:35 +0000198
reed@android.comb398fe82009-01-07 11:47:57 +0000199 SkString outDir;
reed@android.com387359e2009-08-04 01:51:09 +0000200 SkBitmap::Config outConfig = SkBitmap::kNo_Config;
201 const char* configName = "";
202 int configCount = SK_ARRAY_COUNT(gConfigs);
reed@android.combd700c32009-01-05 03:34:50 +0000203
reed@android.comb398fe82009-01-07 11:47:57 +0000204 char* const* stop = argv + argc;
205 for (++argv; argv < stop; ++argv) {
206 if (strcmp(*argv, "-o") == 0) {
207 argv++;
208 if (argv < stop && **argv) {
209 outDir.set(*argv);
210 if (outDir.c_str()[outDir.size() - 1] != '/') {
211 outDir.append("/");
212 }
213 }
reed@android.com6c924ad2009-03-31 03:48:49 +0000214 } else if (strcmp(*argv, "-pict") == 0) {
215 doPict = true;
reed@android.com4bc19832009-01-19 20:08:35 +0000216 } else if (strcmp(*argv, "-repeat") == 0) {
217 argv++;
218 if (argv < stop) {
219 repeatDraw = atoi(*argv);
220 if (repeatDraw < 1) {
221 repeatDraw = 1;
222 }
223 } else {
reed@android.com29348cb2009-08-04 18:17:15 +0000224 log_error("missing arg for -repeat\n");
reed@android.com4bc19832009-01-19 20:08:35 +0000225 return -1;
226 }
reed@android.com4c7d3d62009-01-21 03:15:13 +0000227 } else if (!strcmp(*argv, "-rotate")) {
228 doRotate = true;
reed@android.com387359e2009-08-04 01:51:09 +0000229 } else if (!strcmp(*argv, "-scale")) {
230 doScale = true;
reed@android.com4c7d3d62009-01-21 03:15:13 +0000231 } else if (!strcmp(*argv, "-clip")) {
232 doClip = true;
reed@android.com4bc19832009-01-19 20:08:35 +0000233 } else if (strcmp(*argv, "-forceAA") == 0) {
reed@android.com29348cb2009-08-04 18:17:15 +0000234 if (!parse_bool_arg(++argv, stop, &forceAA)) {
235 log_error("missing arg for -forceAA\n");
236 return -1;
237 }
238 } else if (strcmp(*argv, "-forceFilter") == 0) {
239 if (!parse_bool_arg(++argv, stop, &forceFilter)) {
240 log_error("missing arg for -forceFilter\n");
241 return -1;
242 }
reed@android.com4bc19832009-01-19 20:08:35 +0000243 } else if (strcmp(*argv, "-forceBlend") == 0) {
reed@android.com29348cb2009-08-04 18:17:15 +0000244 bool wantAlpha = false;
245 if (!parse_bool_arg(++argv, stop, &wantAlpha)) {
246 log_error("missing arg for -forceBlend\n");
247 return -1;
248 }
249 forceAlpha = wantAlpha ? 0x80 : 0xFF;
reed@android.com387359e2009-08-04 01:51:09 +0000250 } else if (strcmp(*argv, "-match") == 0) {
251 argv++;
252 if (argv < stop) {
253 matchStr = *argv;
254 } else {
reed@android.com29348cb2009-08-04 18:17:15 +0000255 log_error("missing arg for -match\n");
reed@android.com387359e2009-08-04 01:51:09 +0000256 return -1;
reed@android.com4c7d3d62009-01-21 03:15:13 +0000257 }
reed@android.com387359e2009-08-04 01:51:09 +0000258 } else if (strcmp(*argv, "-config") == 0) {
259 argv++;
260 if (argv < stop) {
261 int index = findConfig(*argv);
262 if (index >= 0) {
263 outConfig = gConfigs[index].fConfig;
264 configName = gConfigs[index].fName;
265 configCount = 1;
266 } else {
reed@android.com29348cb2009-08-04 18:17:15 +0000267 SkString str;
268 str.printf("unrecognized config %s\n", *argv);
269 log_error(str);
reed@android.com387359e2009-08-04 01:51:09 +0000270 return -1;
271 }
272 } else {
reed@android.com29348cb2009-08-04 18:17:15 +0000273 log_error("missing arg for -config\n");
reed@android.com387359e2009-08-04 01:51:09 +0000274 return -1;
275 }
276 } else {
reed@android.com29348cb2009-08-04 18:17:15 +0000277 SkString str;
278 str.printf("unrecognized arg %s\n", *argv);
279 log_error(str);
reed@android.com387359e2009-08-04 01:51:09 +0000280 return -1;
reed@android.comb398fe82009-01-07 11:47:57 +0000281 }
282 }
reed@android.com387359e2009-08-04 01:51:09 +0000283
reed@android.combd700c32009-01-05 03:34:50 +0000284 Iter iter;
285 SkBenchmark* bench;
286 while ((bench = iter.next()) != NULL) {
reed@android.comb398fe82009-01-07 11:47:57 +0000287 SkIPoint dim = bench->getSize();
288 if (dim.fX <= 0 || dim.fY <= 0) {
289 continue;
290 }
reed@android.comb398fe82009-01-07 11:47:57 +0000291
reed@android.com4bc19832009-01-19 20:08:35 +0000292 bench->setForceAlpha(forceAlpha);
293 bench->setForceAA(forceAA);
reed@android.com29348cb2009-08-04 18:17:15 +0000294 bench->setForceFilter(forceFilter);
reed@android.com4bc19832009-01-19 20:08:35 +0000295
reed@android.com387359e2009-08-04 01:51:09 +0000296 // only run benchmarks if their name contains matchStr
297 if (matchStr && strstr(bench->getName(), matchStr) == NULL) {
298 continue;
299 }
300
reed@android.com29348cb2009-08-04 18:17:15 +0000301 {
302 SkString str;
303 str.printf("running bench %16s", bench->getName());
304 log_progress(str);
305 }
reed@android.com4bc19832009-01-19 20:08:35 +0000306
307 for (int configIndex = 0; configIndex < configCount; configIndex++) {
308 if (configCount > 1) {
309 outConfig = gConfigs[configIndex].fConfig;
310 configName = gConfigs[configIndex].fName;
311 }
312
313 SkBitmap bm;
314 bm.setConfig(outConfig, dim.fX, dim.fY);
315 bm.allocPixels();
reed@android.com6c924ad2009-03-31 03:48:49 +0000316 erase(bm);
reed@android.comf523e252009-01-26 23:15:37 +0000317
reed@android.com4bc19832009-01-19 20:08:35 +0000318 SkCanvas canvas(bm);
reed@android.comf523e252009-01-26 23:15:37 +0000319
reed@android.com4c7d3d62009-01-21 03:15:13 +0000320 if (doClip) {
321 performClip(&canvas, dim.fX, dim.fY);
322 }
reed@android.com387359e2009-08-04 01:51:09 +0000323 if (doScale) {
324 performScale(&canvas, dim.fX, dim.fY);
325 }
reed@android.com4c7d3d62009-01-21 03:15:13 +0000326 if (doRotate) {
327 performRotate(&canvas, dim.fX, dim.fY);
328 }
329
reed@android.com4bc19832009-01-19 20:08:35 +0000330 SkMSec now = SkTime::GetMSecs();
331 for (int i = 0; i < repeatDraw; i++) {
reed@android.com6c924ad2009-03-31 03:48:49 +0000332 SkCanvas* c = &canvas;
333
334 SkNWayCanvas nway;
335 SkPicture* pict = NULL;
336 if (doPict) {
337 pict = new SkPicture;
338 nway.addCanvas(pict->beginRecording(bm.width(), bm.height()));
339 nway.addCanvas(&canvas);
340 c = &nway;
341 }
342
343 SkAutoCanvasRestore acr(c, true);
344 bench->draw(c);
345
346 if (pict) {
347 compare_pict_to_bitmap(pict, bm);
348 pict->unref();
349 }
reed@android.com4bc19832009-01-19 20:08:35 +0000350 }
351 if (repeatDraw > 1) {
reed@android.com29348cb2009-08-04 18:17:15 +0000352 SkString str;
353 str.printf(" %4s:%7.2f", configName,
354 (SkTime::GetMSecs() - now) / (double)repeatDraw);
355 log_progress(str);
reed@android.com4bc19832009-01-19 20:08:35 +0000356 }
reed@android.com4c7d3d62009-01-21 03:15:13 +0000357 if (outDir.size() > 0) {
358 saveFile(bench->getName(), configName, outDir.c_str(), bm);
359 }
reed@android.com4bc19832009-01-19 20:08:35 +0000360 }
reed@android.com29348cb2009-08-04 18:17:15 +0000361 log_progress("\n");
reed@android.combd700c32009-01-05 03:34:50 +0000362 }
363
364 return 0;
365}