commit-bot@chromium.org | a90c680 | 2014-04-30 13:20:45 +0000 | [diff] [blame] | 1 | #include "DMQuiltTask.h" |
| 2 | #include "DMUtil.h" |
| 3 | #include "DMWriteTask.h" |
| 4 | |
mtklein | e4636aa | 2014-07-09 13:10:58 -0700 | [diff] [blame] | 5 | #include "SkBBHFactory.h" |
commit-bot@chromium.org | a90c680 | 2014-04-30 13:20:45 +0000 | [diff] [blame] | 6 | #include "SkCommandLineFlags.h" |
| 7 | #include "SkPicture.h" |
mtklein | 406654b | 2014-09-03 15:34:37 -0700 | [diff] [blame] | 8 | #include "SkTaskGroup.h" |
commit-bot@chromium.org | a90c680 | 2014-04-30 13:20:45 +0000 | [diff] [blame] | 9 | |
mtklein | e4636aa | 2014-07-09 13:10:58 -0700 | [diff] [blame] | 10 | DEFINE_bool(quilt, true, "If true, draw GM via a picture into a quilt of small tiles and compare."); |
mtklein | 7cdc1ee | 2014-07-07 10:41:04 -0700 | [diff] [blame] | 11 | DEFINE_int32(quiltTile, 256, "Dimension of (square) quilt tile."); |
commit-bot@chromium.org | a90c680 | 2014-04-30 13:20:45 +0000 | [diff] [blame] | 12 | |
| 13 | namespace DM { |
| 14 | |
mtklein | a9ceaf5 | 2014-09-29 08:44:46 -0700 | [diff] [blame] | 15 | static const char* kBBHs[] = { "nobbh", "rtree", "tilegrid" }; |
| 16 | QuiltTask::QuiltTask(const Task& parent, skiagm::GM* gm, SkBitmap reference, QuiltTask::BBH bbh) |
commit-bot@chromium.org | a90c680 | 2014-04-30 13:20:45 +0000 | [diff] [blame] | 17 | : CpuTask(parent) |
mtklein | 5ad6ee1 | 2014-08-11 08:08:43 -0700 | [diff] [blame] | 18 | , fBBH(bbh) |
mtklein | a9ceaf5 | 2014-09-29 08:44:46 -0700 | [diff] [blame] | 19 | , fName(UnderJoin(parent.name().c_str(), kBBHs[bbh])) |
commit-bot@chromium.org | a90c680 | 2014-04-30 13:20:45 +0000 | [diff] [blame] | 20 | , fGM(gm) |
| 21 | , fReference(reference) |
| 22 | {} |
| 23 | |
| 24 | static int tiles_needed(int fullDimension, int tileDimension) { |
| 25 | return (fullDimension + tileDimension - 1) / tileDimension; |
| 26 | } |
| 27 | |
mtklein | ec2ae58 | 2014-07-01 07:46:50 -0700 | [diff] [blame] | 28 | class Tile : public SkRunnable { |
| 29 | public: |
mtklein | 7cdc1ee | 2014-07-07 10:41:04 -0700 | [diff] [blame] | 30 | Tile(int x, int y, const SkPicture& picture, SkBitmap* quilt) |
| 31 | : fX(x * FLAGS_quiltTile) |
| 32 | , fY(y * FLAGS_quiltTile) |
mtklein | ec2ae58 | 2014-07-01 07:46:50 -0700 | [diff] [blame] | 33 | , fPicture(picture) |
mtklein | 7cdc1ee | 2014-07-07 10:41:04 -0700 | [diff] [blame] | 34 | , fQuilt(quilt) {} |
mtklein | ec2ae58 | 2014-07-01 07:46:50 -0700 | [diff] [blame] | 35 | |
| 36 | virtual void run() SK_OVERRIDE { |
| 37 | SkBitmap tile; |
mtklein | 7cdc1ee | 2014-07-07 10:41:04 -0700 | [diff] [blame] | 38 | fQuilt->extractSubset(&tile, SkIRect::MakeXYWH(fX, fY, FLAGS_quiltTile, FLAGS_quiltTile)); |
mtklein | ec2ae58 | 2014-07-01 07:46:50 -0700 | [diff] [blame] | 39 | SkCanvas tileCanvas(tile); |
| 40 | |
mtklein | 7cdc1ee | 2014-07-07 10:41:04 -0700 | [diff] [blame] | 41 | tileCanvas.translate(SkIntToScalar(-fX), SkIntToScalar(-fY)); |
robertphillips | c5ba71d | 2014-09-04 08:42:50 -0700 | [diff] [blame] | 42 | fPicture.playback(&tileCanvas); |
mtklein | ec2ae58 | 2014-07-01 07:46:50 -0700 | [diff] [blame] | 43 | tileCanvas.flush(); |
| 44 | |
mtklein | ec2ae58 | 2014-07-01 07:46:50 -0700 | [diff] [blame] | 45 | delete this; |
| 46 | } |
| 47 | |
| 48 | private: |
| 49 | const int fX, fY; |
mtklein | ec2ae58 | 2014-07-01 07:46:50 -0700 | [diff] [blame] | 50 | const SkPicture& fPicture; |
mtklein | 7cdc1ee | 2014-07-07 10:41:04 -0700 | [diff] [blame] | 51 | SkBitmap* fQuilt; |
mtklein | ec2ae58 | 2014-07-01 07:46:50 -0700 | [diff] [blame] | 52 | }; |
| 53 | |
commit-bot@chromium.org | a90c680 | 2014-04-30 13:20:45 +0000 | [diff] [blame] | 54 | void QuiltTask::draw() { |
mtklein | e4636aa | 2014-07-09 13:10:58 -0700 | [diff] [blame] | 55 | SkAutoTDelete<SkBBHFactory> factory; |
mtklein | 5ad6ee1 | 2014-08-11 08:08:43 -0700 | [diff] [blame] | 56 | switch (fBBH) { |
| 57 | case kNone_BBH: break; |
| 58 | case kRTree_BBH: |
mtklein | e4636aa | 2014-07-09 13:10:58 -0700 | [diff] [blame] | 59 | factory.reset(SkNEW(SkRTreeFactory)); |
| 60 | break; |
mtklein | 5ad6ee1 | 2014-08-11 08:08:43 -0700 | [diff] [blame] | 61 | case kTileGrid_BBH: { |
mtklein | e4636aa | 2014-07-09 13:10:58 -0700 | [diff] [blame] | 62 | const SkTileGridFactory::TileGridInfo tiles = { |
| 63 | { FLAGS_quiltTile, FLAGS_quiltTile }, |
| 64 | /*overlap: */{0, 0}, |
| 65 | /*offset: */{0, 0}, |
| 66 | }; |
| 67 | factory.reset(SkNEW_ARGS(SkTileGridFactory, (tiles))); |
| 68 | break; |
| 69 | } |
mtklein | e4636aa | 2014-07-09 13:10:58 -0700 | [diff] [blame] | 70 | } |
| 71 | |
| 72 | // A couple GMs draw wrong when using a bounding box hierarchy. |
| 73 | // This almost certainly means we have a bug to fix, but for now |
| 74 | // just draw without a bounding box hierarchy. |
| 75 | if (fGM->getFlags() & skiagm::GM::kNoBBH_Flag) { |
| 76 | factory.reset(NULL); |
| 77 | } |
| 78 | |
mtklein | a9ceaf5 | 2014-09-29 08:44:46 -0700 | [diff] [blame] | 79 | SkAutoTUnref<const SkPicture> recorded(RecordPicture(fGM.get(), factory.get())); |
commit-bot@chromium.org | a90c680 | 2014-04-30 13:20:45 +0000 | [diff] [blame] | 80 | |
| 81 | SkBitmap full; |
commit-bot@chromium.org | 2664207 | 2014-05-15 17:33:31 +0000 | [diff] [blame] | 82 | AllocatePixels(fReference, &full); |
commit-bot@chromium.org | a90c680 | 2014-04-30 13:20:45 +0000 | [diff] [blame] | 83 | |
mtklein | e4636aa | 2014-07-09 13:10:58 -0700 | [diff] [blame] | 84 | if (fGM->getFlags() & skiagm::GM::kSkipTiled_Flag) { |
| 85 | // Some GMs don't draw exactly the same when tiled. Draw them in one go. |
| 86 | SkCanvas canvas(full); |
robertphillips | c5ba71d | 2014-09-04 08:42:50 -0700 | [diff] [blame] | 87 | recorded->playback(&canvas); |
mtklein | e4636aa | 2014-07-09 13:10:58 -0700 | [diff] [blame] | 88 | canvas.flush(); |
| 89 | } else { |
| 90 | // Draw tiles in parallel into the same bitmap, simulating aggressive impl-side painting. |
mtklein | 406654b | 2014-09-03 15:34:37 -0700 | [diff] [blame] | 91 | SkTaskGroup tg; |
mtklein | e4636aa | 2014-07-09 13:10:58 -0700 | [diff] [blame] | 92 | for (int y = 0; y < tiles_needed(full.height(), FLAGS_quiltTile); y++) { |
| 93 | for (int x = 0; x < tiles_needed(full.width(), FLAGS_quiltTile); x++) { |
| 94 | // Deletes itself when done. |
mtklein | 406654b | 2014-09-03 15:34:37 -0700 | [diff] [blame] | 95 | tg.add(new Tile(x, y, *recorded, &full)); |
mtklein | e4636aa | 2014-07-09 13:10:58 -0700 | [diff] [blame] | 96 | } |
commit-bot@chromium.org | a90c680 | 2014-04-30 13:20:45 +0000 | [diff] [blame] | 97 | } |
| 98 | } |
| 99 | |
commit-bot@chromium.org | a90c680 | 2014-04-30 13:20:45 +0000 | [diff] [blame] | 100 | if (!BitmapsEqual(full, fReference)) { |
| 101 | this->fail(); |
mtklein | ea65bfa | 2014-09-09 07:59:46 -0700 | [diff] [blame] | 102 | this->spawnChild(SkNEW_ARGS(WriteTask, (*this, "GM", full))); |
commit-bot@chromium.org | a90c680 | 2014-04-30 13:20:45 +0000 | [diff] [blame] | 103 | } |
| 104 | } |
| 105 | |
| 106 | bool QuiltTask::shouldSkip() const { |
| 107 | if (fGM->getFlags() & skiagm::GM::kSkipPicture_Flag) { |
| 108 | return true; |
| 109 | } |
commit-bot@chromium.org | a90c680 | 2014-04-30 13:20:45 +0000 | [diff] [blame] | 110 | return !FLAGS_quilt; |
| 111 | } |
| 112 | |
| 113 | } // namespace DM |