/* * 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 <stdio.h> #include "src/core/SkPicturePriv.h" #include "src/core/SkRecord.h" #include "src/core/SkRecordDraw.h" #include "include/core/SkTime.h" #include "tools/DumpRecord.h" namespace { class Dumper { public: explicit Dumper(SkCanvas* canvas, int count, bool timeWithCommand) : fDigits(0) , fIndent(0) , fIndex(0) , fDraw(canvas, nullptr, nullptr, 0, nullptr) , fTimeWithCommand(timeWithCommand) { while (count > 0) { count /= 10; fDigits++; } } template <typename T> void operator()(const T& command) { auto start = SkTime::GetNSecs(); fDraw(command); this->print(command, SkTime::GetNSecs() - start); } void operator()(const SkRecords::NoOp&) { // Move on without printing anything. } template <typename T> void print(const T& command, double ns) { this->printNameAndTime(command, ns); } void print(const SkRecords::Restore& command, double ns) { --fIndent; this->printNameAndTime(command, ns); } void print(const SkRecords::Save& command, double ns) { this->printNameAndTime(command, ns); ++fIndent; } void print(const SkRecords::SaveLayer& command, double ns) { this->printNameAndTime(command, ns); ++fIndent; } void print(const SkRecords::DrawPicture& command, double ns) { this->printNameAndTime(command, ns); if (auto bp = SkPicturePriv::AsSkBigPicture(command.picture)) { ++fIndent; const SkRecord& record = *bp->record(); for (int i = 0; i < record.count(); i++) { record.visit(i, *this); } --fIndent; } } #if 1 void print(const SkRecords::DrawAnnotation& command, double ns) { int us = (int)(ns * 1e-3); if (!fTimeWithCommand) { printf("%6dus ", us); } printf("%*d ", fDigits, fIndex++); for (int i = 0; i < fIndent; i++) { printf(" "); } if (fTimeWithCommand) { printf("%6dus ", us); } printf("DrawAnnotation [%g %g %g %g] %s\n", command.rect.left(), command.rect.top(), command.rect.right(), command.rect.bottom(), command.key.c_str()); } #endif private: template <typename T> void printNameAndTime(const T& command, double ns) { int us = (int)(ns * 1e-3); if (!fTimeWithCommand) { printf("%6dus ", us); } printf("%*d ", fDigits, fIndex++); for (int i = 0; i < fIndent; i++) { printf(" "); } if (fTimeWithCommand) { printf("%6dus ", us); } puts(NameOf(command)); } template <typename T> static const char* NameOf(const T&) { #define CASE(U) case SkRecords::U##_Type: return #U; switch (T::kType) { SK_RECORD_TYPES(CASE) } #undef CASE SkDEBUGFAIL("Unknown T"); return "Unknown T"; } static const char* NameOf(const SkRecords::SaveLayer&) { return "\x1b[31;1mSaveLayer\x1b[0m"; // Bold red. } int fDigits; int fIndent; int fIndex; SkRecords::Draw fDraw; const bool fTimeWithCommand; }; } // namespace void DumpRecord(const SkRecord& record, SkCanvas* canvas, bool timeWithCommand) { Dumper dumper(canvas, record.count(), timeWithCommand); for (int i = 0; i < record.count(); i++) { record.visit(i, dumper); } }