• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
1 #include "gm.h"
2 #include "SkColorPriv.h"
3 #include "SkGraphics.h"
4 #include "SkImageDecoder.h"
5 #include "SkImageEncoder.h"
6 #include "SkPicture.h"
7 #include "SkStream.h"
8 #include "SkRefCnt.h"
9 
10 #include "GrContext.h"
11 #include "SkGpuCanvas.h"
12 #include "SkGpuDevice.h"
13 #include "SkEGLContext.h"
14 #include "SkDevice.h"
15 
16 #ifdef SK_SUPPORT_PDF
17     #include "SkPDFDevice.h"
18     #include "SkPDFDocument.h"
19 #endif
20 
21 using namespace skiagm;
22 
23 // need to explicitly declare this, or we get some weird infinite loop llist
24 template GMRegistry* GMRegistry::gHead;
25 
26 class Iter {
27 public:
Iter()28     Iter() {
29         fReg = GMRegistry::Head();
30     }
31 
next()32     GM* next() {
33         if (fReg) {
34             GMRegistry::Factory fact = fReg->factory();
35             fReg = fReg->next();
36             return fact(0);
37         }
38         return NULL;
39     }
40 
Count()41     static int Count() {
42         const GMRegistry* reg = GMRegistry::Head();
43         int count = 0;
44         while (reg) {
45             count += 1;
46             reg = reg->next();
47         }
48         return count;
49     }
50 
51 private:
52     const GMRegistry* fReg;
53 };
54 
make_name(const char shortName[],const char configName[])55 static SkString make_name(const char shortName[], const char configName[]) {
56     SkString name(shortName);
57     name.appendf("_%s", configName);
58     return name;
59 }
60 
make_filename(const char path[],const char pathSuffix[],const SkString & name,const char suffix[])61 static SkString make_filename(const char path[],
62                               const char pathSuffix[],
63                               const SkString& name,
64                               const char suffix[]) {
65     SkString filename(path);
66     if (filename.endsWith("/")) {
67         filename.remove(filename.size() - 1, 1);
68     }
69     filename.append(pathSuffix);
70     filename.append("/");
71     filename.appendf("%s.%s", name.c_str(), suffix);
72     return filename;
73 }
74 
75 /* since PNG insists on unpremultiplying our alpha, we take no precision chances
76     and force all pixels to be 100% opaque, otherwise on compare we may not get
77     a perfect match.
78  */
force_all_opaque(const SkBitmap & bitmap)79 static void force_all_opaque(const SkBitmap& bitmap) {
80     SkAutoLockPixels lock(bitmap);
81     for (int y = 0; y < bitmap.height(); y++) {
82         for (int x = 0; x < bitmap.width(); x++) {
83             *bitmap.getAddr32(x, y) |= (SK_A32_MASK << SK_A32_SHIFT);
84         }
85     }
86 }
87 
write_bitmap(const SkString & path,const SkBitmap & bitmap)88 static bool write_bitmap(const SkString& path, const SkBitmap& bitmap) {
89     SkBitmap copy;
90     bitmap.copyTo(&copy, SkBitmap::kARGB_8888_Config);
91     force_all_opaque(copy);
92     return SkImageEncoder::EncodeFile(path.c_str(), copy,
93                                       SkImageEncoder::kPNG_Type, 100);
94 }
95 
compute_diff_pmcolor(SkPMColor c0,SkPMColor c1)96 static inline SkPMColor compute_diff_pmcolor(SkPMColor c0, SkPMColor c1) {
97     int dr = SkGetPackedR32(c0) - SkGetPackedR32(c1);
98     int dg = SkGetPackedG32(c0) - SkGetPackedG32(c1);
99     int db = SkGetPackedB32(c0) - SkGetPackedB32(c1);
100     return SkPackARGB32(0xFF, SkAbs32(dr), SkAbs32(dg), SkAbs32(db));
101 }
102 
compute_diff(const SkBitmap & target,const SkBitmap & base,SkBitmap * diff)103 static void compute_diff(const SkBitmap& target, const SkBitmap& base,
104                          SkBitmap* diff) {
105     SkAutoLockPixels alp(*diff);
106 
107     const int w = target.width();
108     const int h = target.height();
109     for (int y = 0; y < h; y++) {
110         for (int x = 0; x < w; x++) {
111             SkPMColor c0 = *base.getAddr32(x, y);
112             SkPMColor c1 = *target.getAddr32(x, y);
113             SkPMColor d = 0;
114             if (c0 != c1) {
115                 d = compute_diff_pmcolor(c0, c1);
116             }
117             *diff->getAddr32(x, y) = d;
118         }
119     }
120 }
121 
compare(const SkBitmap & target,const SkBitmap & base,const SkString & name,const char * renderModeDescriptor,SkBitmap * diff)122 static bool compare(const SkBitmap& target, const SkBitmap& base,
123                     const SkString& name, const char* renderModeDescriptor,
124                     SkBitmap* diff) {
125     SkBitmap copy;
126     const SkBitmap* bm = &target;
127     if (target.config() != SkBitmap::kARGB_8888_Config) {
128         target.copyTo(&copy, SkBitmap::kARGB_8888_Config);
129         bm = &copy;
130     }
131 
132     force_all_opaque(*bm);
133 
134     const int w = bm->width();
135     const int h = bm->height();
136     if (w != base.width() || h != base.height()) {
137         SkDebugf(
138 "---- %s dimensions mismatch for %s base [%d %d] current [%d %d]\n",
139                  renderModeDescriptor, name.c_str(),
140                  base.width(), base.height(), w, h);
141         return false;
142     }
143 
144     SkAutoLockPixels bmLock(*bm);
145     SkAutoLockPixels baseLock(base);
146 
147     for (int y = 0; y < h; y++) {
148         for (int x = 0; x < w; x++) {
149             SkPMColor c0 = *base.getAddr32(x, y);
150             SkPMColor c1 = *bm->getAddr32(x, y);
151             if (c0 != c1) {
152                 SkDebugf(
153 "----- %s pixel mismatch for %s at [%d %d] base 0x%08X current 0x%08X\n",
154                          renderModeDescriptor, name.c_str(), x, y, c0, c1);
155 
156                 if (diff) {
157                     diff->setConfig(SkBitmap::kARGB_8888_Config, w, h);
158                     diff->allocPixels();
159                     compute_diff(*bm, base, diff);
160                 }
161                 return false;
162             }
163         }
164     }
165 
166     // they're equal
167     return true;
168 }
169 
write_pdf(const SkString & path,const SkDynamicMemoryWStream & pdf)170 static bool write_pdf(const SkString& path, const SkDynamicMemoryWStream& pdf) {
171     SkFILEWStream stream(path.c_str());
172     return stream.write(pdf.getStream(), pdf.getOffset());
173 }
174 
175 enum Backend {
176   kRaster_Backend,
177   kGPU_Backend,
178   kPDF_Backend,
179 };
180 
181 struct ConfigData {
182     SkBitmap::Config    fConfig;
183     Backend             fBackend;
184     const char*         fName;
185 };
186 
187 /// Returns true if processing should continue, false to skip the
188 /// remainder of this config for this GM.
189 //@todo thudson 22 April 2011 - could refactor this to take in
190 // a factory to generate the context, always call readPixels()
191 // (logically a noop for rasters, if wasted time), and thus collapse the
192 // GPU special case and also let this be used for SkPicture testing.
setup_bitmap(const ConfigData & gRec,SkISize & size,SkBitmap * bitmap)193 static void setup_bitmap(const ConfigData& gRec, SkISize& size,
194                          SkBitmap* bitmap) {
195     bitmap->setConfig(gRec.fConfig, size.width(), size.height());
196     bitmap->allocPixels();
197     bitmap->eraseColor(0);
198 }
199 
200 // Returns true if the test should continue, false if the test should
201 // halt.
generate_image(GM * gm,const ConfigData & gRec,GrContext * context,SkBitmap & bitmap)202 static bool generate_image(GM* gm, const ConfigData& gRec,
203                            GrContext* context,
204                            SkBitmap& bitmap) {
205     SkISize size (gm->getISize());
206     setup_bitmap(gRec, size, &bitmap);
207     SkCanvas canvas(bitmap);
208 
209     if (gRec.fBackend == kRaster_Backend) {
210         gm->draw(&canvas);
211     } else {  // GPU
212         if (NULL == context) {
213             return false;
214         }
215         SkGpuCanvas gc(context,
216                        SkGpuDevice::Current3DApiRenderTarget());
217         gc.setDevice(gc.createDevice(bitmap.config(),
218                                      bitmap.width(),
219                                      bitmap.height(),
220                                      bitmap.isOpaque(),
221                                      false))->unref();
222         gm->draw(&gc);
223         gc.readPixels(&bitmap); // overwrite our previous allocation
224     }
225     return true;
226 }
227 
generate_image_from_picture(GM * gm,const ConfigData & gRec,SkPicture * pict,SkBitmap * bitmap)228 static void generate_image_from_picture(GM* gm, const ConfigData& gRec,
229                                         SkPicture* pict, SkBitmap* bitmap) {
230     SkISize size = gm->getISize();
231     setup_bitmap(gRec, size, bitmap);
232     SkCanvas canvas(*bitmap);
233     canvas.drawPicture(*pict);
234 }
235 
generate_pdf(GM * gm,SkDynamicMemoryWStream & pdf)236 static void generate_pdf(GM* gm, SkDynamicMemoryWStream& pdf) {
237 #ifdef SK_SUPPORT_PDF
238     SkISize size = gm->getISize();
239     SkMatrix identity;
240     identity.reset();
241     SkPDFDevice* dev = new SkPDFDevice(size, size, identity);
242     SkAutoUnref aur(dev);
243 
244     SkCanvas c(dev);
245     gm->draw(&c);
246 
247     SkPDFDocument doc;
248     doc.appendPage(dev);
249     doc.emitPDF(&pdf);
250 #endif
251 }
252 
write_reference_image(const ConfigData & gRec,const char writePath[],const char renderModeDescriptor[],const SkString & name,SkBitmap & bitmap,SkDynamicMemoryWStream * pdf)253 static bool write_reference_image(const ConfigData& gRec,
254                                   const char writePath [],
255                                   const char renderModeDescriptor [],
256                                   const SkString& name,
257                                   SkBitmap& bitmap,
258                                   SkDynamicMemoryWStream* pdf) {
259     SkString path;
260     bool success = false;
261     if (gRec.fBackend != kPDF_Backend) {
262         path = make_filename(writePath, renderModeDescriptor, name, "png");
263         success = write_bitmap(path, bitmap);
264     } else if (pdf) {
265         path = make_filename(writePath, renderModeDescriptor, name, "pdf");
266         success = write_pdf(path, *pdf);
267     }
268     if (!success) {
269         fprintf(stderr, "FAILED to write %s\n", path.c_str());
270     }
271     return success;
272 }
273 
compare_to_reference_image(const char readPath[],const SkString & name,SkBitmap & bitmap,const char diffPath[],const char renderModeDescriptor[])274 static bool compare_to_reference_image(const char readPath [],
275                                        const SkString& name,
276                                        SkBitmap &bitmap,
277                                        const char diffPath [],
278                                        const char renderModeDescriptor []) {
279     SkString path = make_filename(readPath, "", name, "png");
280     SkBitmap orig;
281     bool success = SkImageDecoder::DecodeFile(path.c_str(), &orig,
282                         SkBitmap::kARGB_8888_Config,
283                         SkImageDecoder::kDecodePixels_Mode, NULL);
284     if (success) {
285         SkBitmap diffBitmap;
286         success = compare(bitmap, orig, name, renderModeDescriptor,
287                           diffPath ? &diffBitmap : NULL);
288         if (!success && diffPath) {
289             SkString diffName = make_filename(diffPath, "", name, ".diff.png");
290             fprintf(stderr, "Writing %s\n", diffName.c_str());
291             write_bitmap(diffName, diffBitmap);
292         }
293     } else {
294         fprintf(stderr, "FAILED to read %s\n", path.c_str());
295     }
296     return success;
297 }
298 
handle_test_results(GM * gm,const ConfigData & gRec,const char writePath[],const char readPath[],const char diffPath[],const char renderModeDescriptor[],SkBitmap & bitmap,SkDynamicMemoryWStream * pdf)299 static bool handle_test_results(GM* gm,
300                                 const ConfigData& gRec,
301                                 const char writePath [],
302                                 const char readPath [],
303                                 const char diffPath [],
304                                 const char renderModeDescriptor [],
305                                 SkBitmap& bitmap,
306                                 SkDynamicMemoryWStream* pdf) {
307     SkString name = make_name(gm->shortName(), gRec.fName);
308 
309     if (writePath) {
310         write_reference_image(gRec, writePath, renderModeDescriptor,
311                               name, bitmap, pdf);
312     // TODO: Figure out a way to compare PDFs.
313     } else if (readPath && gRec.fBackend != kPDF_Backend) {
314         return compare_to_reference_image(readPath, name, bitmap,
315                                    diffPath, renderModeDescriptor);
316     }
317     return true;
318 }
319 
generate_new_picture(GM * gm)320 static SkPicture* generate_new_picture(GM* gm) {
321     // Pictures are refcounted so must be on heap
322     SkPicture* pict = new SkPicture;
323     SkCanvas* cv = pict->beginRecording(1000, 1000);
324     gm->draw(cv);
325     pict->endRecording();
326 
327     return pict;
328 }
329 
stream_to_new_picture(const SkPicture & src)330 static SkPicture* stream_to_new_picture(const SkPicture& src) {
331 
332     // To do in-memory commiunications with a stream, we need to:
333     // * create a dynamic memory stream
334     // * copy it into a buffer
335     // * create a read stream from it
336     // ?!?!
337 
338     SkDynamicMemoryWStream storage;
339     src.serialize(&storage);
340 
341     int streamSize = storage.getOffset();
342     SkAutoMalloc dstStorage(streamSize);
343     void* dst = dstStorage.get();
344     //char* dst = new char [streamSize];
345     //@todo thudson 22 April 2011 when can we safely delete [] dst?
346     storage.copyTo(dst);
347     SkMemoryStream pictReadback(dst, streamSize);
348     SkPicture* retval = new SkPicture (&pictReadback);
349     return retval;
350 }
351 
352 // Test: draw into a bitmap or pdf.
353 // Depending on flags, possibly compare to an expected image
354 // and possibly output a diff image if it fails to match.
test_drawing(GM * gm,const ConfigData & gRec,const char writePath[],const char readPath[],const char diffPath[],GrContext * context)355 static bool test_drawing(GM* gm,
356                          const ConfigData& gRec,
357                          const char writePath [],
358                          const char readPath [],
359                          const char diffPath [],
360                          GrContext* context) {
361     SkBitmap bitmap;
362     SkDynamicMemoryWStream pdf;
363 
364     if (gRec.fBackend == kRaster_Backend ||
365             gRec.fBackend == kGPU_Backend) {
366         // Early exit if we can't generate the image, but this is
367         // expected in some cases, so don't report a test failure.
368         if (!generate_image(gm, gRec, context, bitmap)) {
369             return true;
370         }
371     }
372     // TODO: Figure out a way to compare PDFs.
373     if (gRec.fBackend == kPDF_Backend && writePath) {
374         generate_pdf(gm, pdf);
375     }
376     return handle_test_results(gm, gRec, writePath, readPath, diffPath,
377                         "", bitmap, &pdf);
378 }
379 
test_picture_playback(GM * gm,const ConfigData & gRec,const char readPath[],const char diffPath[])380 static bool test_picture_playback(GM* gm,
381                                   const ConfigData& gRec,
382                                   const char readPath [],
383                                   const char diffPath []) {
384     SkPicture* pict = generate_new_picture(gm);
385     SkAutoUnref aur(pict);
386 
387     if (kRaster_Backend == gRec.fBackend) {
388         SkBitmap bitmap;
389         generate_image_from_picture(gm, gRec, pict, &bitmap);
390         return handle_test_results(gm, gRec, NULL, readPath, diffPath,
391                             "-replay", bitmap, NULL);
392     }
393     return true;
394 }
395 
test_picture_serialization(GM * gm,const ConfigData & gRec,const char readPath[],const char diffPath[])396 static bool test_picture_serialization(GM* gm,
397                                        const ConfigData& gRec,
398                                        const char readPath [],
399                                        const char diffPath []) {
400     SkPicture* pict = generate_new_picture(gm);
401     SkAutoUnref aurp(pict);
402     SkPicture* repict = stream_to_new_picture(*pict);
403     SkAutoUnref aurr(repict);
404 
405     if (kRaster_Backend == gRec.fBackend) {
406         SkBitmap bitmap;
407         generate_image_from_picture(gm, gRec, repict, &bitmap);
408         return handle_test_results(gm, gRec, NULL, readPath, diffPath,
409                             "-serialize", bitmap, NULL);
410     }
411     return true;
412 }
413 
usage(const char * argv0)414 static void usage(const char * argv0) {
415     SkDebugf("%s [-w writePath] [-r readPath] [-d diffPath]\n", argv0);
416     SkDebugf("    [--replay] [--serialize]\n");
417     SkDebugf("    writePath: directory to write rendered images in.\n");
418     SkDebugf(
419 "    readPath: directory to read reference images from;\n"
420 "        reports if any pixels mismatch between reference and new images\n");
421     SkDebugf("    diffPath: directory to write difference images in.\n");
422     SkDebugf("    --replay: exercise SkPicture replay.\n");
423     SkDebugf(
424 "    --serialize: exercise SkPicture serialization & deserialization.\n");
425 }
426 
427 static const ConfigData gRec[] = {
428     { SkBitmap::kARGB_8888_Config, kRaster_Backend, "8888" },
429     { SkBitmap::kARGB_4444_Config, kRaster_Backend, "4444" },
430     { SkBitmap::kRGB_565_Config,   kRaster_Backend, "565" },
431     { SkBitmap::kARGB_8888_Config, kGPU_Backend,    "gpu" },
432 #ifdef SK_SUPPORT_PDF
433     { SkBitmap::kARGB_8888_Config, kPDF_Backend,    "pdf" },
434 #endif
435 };
436 
main(int argc,char * const argv[])437 int main(int argc, char * const argv[]) {
438     SkAutoGraphics ag;
439 
440     const char* writePath = NULL;   // if non-null, where we write the originals
441     const char* readPath = NULL;    // if non-null, were we read from to compare
442     const char* diffPath = NULL;    // if non-null, where we write our diffs (from compare)
443 
444     bool doReplay = true;
445     bool doSerialize = false;
446     const char* const commandName = argv[0];
447     char* const* stop = argv + argc;
448     for (++argv; argv < stop; ++argv) {
449         if (strcmp(*argv, "-w") == 0) {
450             argv++;
451             if (argv < stop && **argv) {
452                 writePath = *argv;
453             }
454         } else if (strcmp(*argv, "-r") == 0) {
455             argv++;
456             if (argv < stop && **argv) {
457                 readPath = *argv;
458             }
459         } else if (strcmp(*argv, "-d") == 0) {
460             argv++;
461             if (argv < stop && **argv) {
462                 diffPath = *argv;
463             }
464         } else if (strcmp(*argv, "--noreplay") == 0) {
465             doReplay = false;
466         } else if (strcmp(*argv, "--serialize") == 0) {
467             doSerialize = true;
468         } else {
469           usage(commandName);
470           return -1;
471         }
472     }
473     if (argv != stop) {
474       usage(commandName);
475       return -1;
476     }
477 
478     // setup a GL context for drawing offscreen
479     GrContext* context = NULL;
480     SkEGLContext eglContext;
481     if (eglContext.init(1024, 1024)) {
482         context = GrContext::CreateGLShaderContext();
483     }
484 
485     Iter iter;
486     GM* gm;
487 
488     if (readPath) {
489         fprintf(stderr, "reading from %s\n", readPath);
490     } else if (writePath) {
491         fprintf(stderr, "writing to %s\n", writePath);
492     }
493 
494     // Accumulate success of all tests so we can flag error in any
495     // one with the return value.
496     bool overallSuccess = true;
497     while ((gm = iter.next()) != NULL) {
498         SkISize size = gm->getISize();
499         SkDebugf("drawing... %s [%d %d]\n", gm->shortName(),
500                  size.width(), size.height());
501 
502         for (size_t i = 0; i < SK_ARRAY_COUNT(gRec); i++) {
503             bool testSuccess = test_drawing(gm, gRec[i],
504                          writePath, readPath, diffPath, context);
505             overallSuccess &= testSuccess;
506 
507             if (doReplay && testSuccess) {
508                 testSuccess = test_picture_playback(gm, gRec[i],
509                                       readPath, diffPath);
510                 overallSuccess &= testSuccess;
511             }
512 
513             if (doSerialize && testSuccess) {
514                 overallSuccess &= test_picture_serialization(gm, gRec[i],
515                                            readPath, diffPath);
516             }
517         }
518         SkDELETE(gm);
519     }
520     if (false == overallSuccess) {
521         return -1;
522     }
523     return 0;
524 }
525 
526 ///////////////////////////////////////////////////////////////////////////////
527 
528 using namespace skiagm;
529 
GM()530 GM::GM() {}
~GM()531 GM::~GM() {}
532 
draw(SkCanvas * canvas)533 void GM::draw(SkCanvas* canvas) {
534     this->onDraw(canvas);
535 }
536