| /* |
| * Copyright 2014 Google Inc. |
| * |
| * Use of this source code is governed by a BSD-style license that can be |
| * found in the LICENSE file. |
| */ |
| |
| #include "Benchmark.h" |
| #include "SkCanvas.h" |
| #include "SkColor.h" |
| #include "SkNullCanvas.h" |
| #include "SkPaint.h" |
| #include "SkPicture.h" |
| #include "SkPictureRecorder.h" |
| #include "SkString.h" |
| |
| class PictureNesting : public Benchmark { |
| public: |
| PictureNesting(const char* name, int maxLevel, int maxPictureLevel) |
| : fMaxLevel(maxLevel) |
| , fMaxPictureLevel(maxPictureLevel) { |
| fName.printf("picture_nesting_%s_%d", name, this->countPics()); |
| fPaint.setColor(SK_ColorRED); |
| fPaint.setAntiAlias(true); |
| fPaint.setStyle(SkPaint::kStroke_Style); |
| } |
| |
| protected: |
| const char* onGetName() override { |
| return fName.c_str(); |
| } |
| |
| void doDraw(SkCanvas* canvas) { |
| SkIPoint canvasSize = onGetSize(); |
| canvas->save(); |
| canvas->scale(SkIntToScalar(canvasSize.x()), SkIntToScalar(canvasSize.y())); |
| |
| SkDEBUGCODE(int pics = ) this->sierpinsky(canvas, 0, fPaint); |
| SkASSERT(pics == this->countPics()); |
| |
| canvas->restore(); |
| } |
| |
| int sierpinsky(SkCanvas* canvas, int lvl, const SkPaint& paint) { |
| if (++lvl > fMaxLevel) { |
| return 0; |
| } |
| |
| int pics = 0; |
| bool recordPicture = lvl <= fMaxPictureLevel; |
| SkPictureRecorder recorder; |
| SkCanvas* c = canvas; |
| |
| if (recordPicture) { |
| c = recorder.beginRecording(1, 1); |
| pics++; |
| } |
| |
| c->drawLine(0.5, 0, 0, 1, paint); |
| c->drawLine(0.5, 0, 1, 1, paint); |
| c->drawLine(0, 1, 1, 1, paint); |
| |
| c->save(); |
| c->scale(0.5, 0.5); |
| |
| c->translate(0, 1); |
| pics += this->sierpinsky(c, lvl, paint); |
| |
| c->translate(1, 0); |
| pics += this->sierpinsky(c, lvl, paint); |
| |
| c->translate(-0.5, -1); |
| pics += this->sierpinsky(c, lvl, paint); |
| c->restore(); |
| |
| if (recordPicture) { |
| canvas->drawPicture(recorder.finishRecordingAsPicture()); |
| } |
| |
| return pics; |
| } |
| |
| int fMaxLevel; |
| int fMaxPictureLevel; |
| |
| private: |
| int countPics() const { |
| // Solve: pics from sierpinsky |
| // f(m) = 1 + 3*f(m - 1) |
| // f(0) = 0 |
| // via "recursive function to closed form" tricks |
| // f(m) = 1/2 (3^m - 1) |
| int pics = 1; |
| for (int i = 0; i < fMaxPictureLevel; i++) { |
| pics *= 3; |
| } |
| pics--; |
| pics /= 2; |
| return pics; |
| } |
| |
| SkString fName; |
| SkPaint fPaint; |
| |
| typedef Benchmark INHERITED; |
| }; |
| |
| class PictureNestingRecording : public PictureNesting { |
| public: |
| PictureNestingRecording(int maxLevel, int maxPictureLevel) |
| : INHERITED("recording", maxLevel, maxPictureLevel) { |
| } |
| |
| protected: |
| bool isSuitableFor(Backend backend) override { |
| return backend == kNonRendering_Backend; |
| } |
| |
| void onDraw(int loops, SkCanvas*) override { |
| SkIPoint canvasSize = onGetSize(); |
| SkPictureRecorder recorder; |
| |
| for (int i = 0; i < loops; i++) { |
| SkCanvas* c = recorder.beginRecording(SkIntToScalar(canvasSize.x()), |
| SkIntToScalar(canvasSize.y())); |
| this->doDraw(c); |
| (void)recorder.finishRecordingAsPicture(); |
| } |
| } |
| |
| private: |
| typedef PictureNesting INHERITED; |
| }; |
| |
| class PictureNestingPlayback : public PictureNesting { |
| public: |
| PictureNestingPlayback(int maxLevel, int maxPictureLevel) |
| : INHERITED("playback", maxLevel, maxPictureLevel) { |
| } |
| protected: |
| void onDelayedSetup() override { |
| this->INHERITED::onDelayedSetup(); |
| |
| SkIPoint canvasSize = onGetSize(); |
| SkPictureRecorder recorder; |
| SkCanvas* c = recorder.beginRecording(SkIntToScalar(canvasSize.x()), |
| SkIntToScalar(canvasSize.y())); |
| |
| this->doDraw(c); |
| fPicture = recorder.finishRecordingAsPicture(); |
| } |
| |
| void onDraw(int loops, SkCanvas* canvas) override { |
| for (int i = 0; i < loops; i++) { |
| canvas->drawPicture(fPicture); |
| } |
| } |
| |
| private: |
| sk_sp<SkPicture> fPicture; |
| |
| typedef PictureNesting INHERITED; |
| }; |
| |
| DEF_BENCH( return new PictureNestingRecording(8, 0); ) |
| DEF_BENCH( return new PictureNestingRecording(8, 1); ) |
| DEF_BENCH( return new PictureNestingRecording(8, 2); ) |
| DEF_BENCH( return new PictureNestingRecording(8, 3); ) |
| DEF_BENCH( return new PictureNestingRecording(8, 4); ) |
| DEF_BENCH( return new PictureNestingRecording(8, 5); ) |
| DEF_BENCH( return new PictureNestingRecording(8, 6); ) |
| DEF_BENCH( return new PictureNestingRecording(8, 7); ) |
| DEF_BENCH( return new PictureNestingRecording(8, 8); ) |
| |
| DEF_BENCH( return new PictureNestingPlayback(8, 0); ) |
| DEF_BENCH( return new PictureNestingPlayback(8, 1); ) |
| DEF_BENCH( return new PictureNestingPlayback(8, 2); ) |
| DEF_BENCH( return new PictureNestingPlayback(8, 3); ) |
| DEF_BENCH( return new PictureNestingPlayback(8, 4); ) |
| DEF_BENCH( return new PictureNestingPlayback(8, 5); ) |
| DEF_BENCH( return new PictureNestingPlayback(8, 6); ) |
| DEF_BENCH( return new PictureNestingPlayback(8, 7); ) |
| DEF_BENCH( return new PictureNestingPlayback(8, 8); ) |