/* * Copyright 2019 Google LLC * * Use of this source code is governed by a BSD-style license that can be * found in the LICENSE file. */ #include "bench/Benchmark.h" #include "include/core/SkCanvas.h" #include "include/core/SkDeferredDisplayListRecorder.h" #include "include/core/SkSurfaceCharacterization.h" #include "include/gpu/GrDirectContext.h" static SkSurfaceCharacterization create_characterization(GrDirectContext* direct) { size_t maxResourceBytes = direct->getResourceCacheLimit(); if (!direct->colorTypeSupportedAsSurface(kRGBA_8888_SkColorType)) { return SkSurfaceCharacterization(); } SkImageInfo ii = SkImageInfo::Make(32, 32, kRGBA_8888_SkColorType, kPremul_SkAlphaType, nullptr); GrBackendFormat backendFormat = direct->defaultBackendFormat(kRGBA_8888_SkColorType, GrRenderable::kYes); if (!backendFormat.isValid()) { return SkSurfaceCharacterization(); } SkSurfaceProps props(0x0, kUnknown_SkPixelGeometry); SkSurfaceCharacterization c = direct->threadSafeProxy()->createCharacterization( maxResourceBytes, ii, backendFormat, 1, kTopLeft_GrSurfaceOrigin, props, false); return c; } // This benchmark tries to simulate how Viz is using SkDDLRecorders. // For each config it will create a single DDLRecorder which it reuses for all the runs // For each run it creates a DDL and stores it for later deletion. class DDLRecorderBench : public Benchmark { public: DDLRecorderBench() { } protected: bool isSuitableFor(Backend backend) override { return kGPU_Backend == backend; } const char* onGetName() override { return "DDLRecorder"; } void onDraw(int loops, SkCanvas* origCanvas) override { if (!fRecorder) { return; } SkASSERT(!fDDLs.size()); fDDLs.reserve(loops); for (int i = 0; i < loops; ++i) { SkCanvas* recordingCanvas = fRecorder->getCanvas(); recordingCanvas->drawRect(SkRect::MakeWH(32, 32), SkPaint()); fDDLs.emplace_back(fRecorder->detach()); } } private: // We create one DDLRecorder for all the timing runs and just keep reusing it void onPerCanvasPreDraw(SkCanvas* origCanvas) override { auto context = origCanvas->recordingContext()->asDirectContext(); if (!context) { return; } SkSurfaceCharacterization c = create_characterization(context); fRecorder.reset(new SkDeferredDisplayListRecorder(c)); } // We defer the clean up of the DDLs so it is done outside of the timing loop void onPostDraw(SkCanvas*) override { fDDLs.clear(); } std::unique_ptr fRecorder = nullptr; std::vector> fDDLs; typedef Benchmark INHERITED; }; DEF_BENCH(return new DDLRecorderBench();)