2012-06-29 14:21:22 +00:00
|
|
|
|
|
|
|
/*
|
|
|
|
* Copyright 2012 Google Inc.
|
|
|
|
*
|
|
|
|
* Use of this source code is governed by a BSD-style license that can be
|
|
|
|
* found in the LICENSE file.
|
|
|
|
*/
|
|
|
|
|
2012-08-07 16:12:23 +00:00
|
|
|
#include "SkDebugger.h"
|
2013-03-12 13:07:40 +00:00
|
|
|
#include "SkString.h"
|
|
|
|
|
2012-06-29 14:21:22 +00:00
|
|
|
|
2012-08-07 16:12:23 +00:00
|
|
|
SkDebugger::SkDebugger() {
|
|
|
|
// Create this some other dynamic way?
|
|
|
|
fDebugCanvas = new SkDebugCanvas(100, 100);
|
|
|
|
fPicture = NULL;
|
|
|
|
fPictureWidth = 0;
|
|
|
|
fPictureHeight = 0;
|
|
|
|
fIndex = 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
SkDebugger::~SkDebugger() {
|
|
|
|
// Need to inherit from SkRef object in order for following to work
|
|
|
|
SkSafeUnref(fDebugCanvas);
|
|
|
|
SkSafeUnref(fPicture);
|
|
|
|
}
|
|
|
|
|
|
|
|
void SkDebugger::loadPicture(SkPicture* picture) {
|
|
|
|
fPictureWidth = picture->width();
|
|
|
|
fPictureHeight = picture->height();
|
|
|
|
delete fDebugCanvas;
|
|
|
|
fDebugCanvas = new SkDebugCanvas(fPictureWidth, fPictureHeight);
|
|
|
|
fDebugCanvas->setBounds(fPictureWidth, fPictureHeight);
|
|
|
|
picture->draw(fDebugCanvas);
|
|
|
|
fIndex = fDebugCanvas->getSize() - 1;
|
|
|
|
SkRefCnt_SafeAssign(fPicture, picture);
|
|
|
|
}
|
|
|
|
|
2013-01-22 18:03:56 +00:00
|
|
|
SkPicture* SkDebugger::copyPicture() {
|
|
|
|
// We can't just call clone here since we want to removed the "deleted"
|
|
|
|
// commands. Playing back will strip those out.
|
|
|
|
SkPicture* newPicture = new SkPicture;
|
|
|
|
SkCanvas* canvas = newPicture->beginRecording(fPictureWidth, fPictureHeight);
|
2012-08-07 16:12:23 +00:00
|
|
|
fDebugCanvas->draw(canvas);
|
2013-01-22 18:03:56 +00:00
|
|
|
newPicture->endRecording();
|
|
|
|
return newPicture;
|
2012-06-29 14:21:22 +00:00
|
|
|
}
|
2013-03-12 13:07:40 +00:00
|
|
|
|
|
|
|
void SkDebugger::getOverviewText(const SkTDArray<double>* typeTimes,
|
|
|
|
double totTime,
|
2013-03-12 15:33:40 +00:00
|
|
|
SkString* overview,
|
|
|
|
int numRuns) {
|
2013-03-12 13:07:40 +00:00
|
|
|
const SkTDArray<SkDrawCommand*>& commands = this->getDrawCommands();
|
|
|
|
|
|
|
|
SkTDArray<int> counts;
|
|
|
|
counts.setCount(LAST_DRAWTYPE_ENUM+1);
|
|
|
|
for (int i = 0; i < LAST_DRAWTYPE_ENUM+1; ++i) {
|
|
|
|
counts[i] = 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
for (int i = 0; i < commands.count(); i++) {
|
|
|
|
counts[commands[i]->getType()]++;
|
|
|
|
}
|
|
|
|
|
|
|
|
overview->reset();
|
|
|
|
int total = 0;
|
|
|
|
#ifdef SK_DEBUG
|
|
|
|
double totPercent = 0, tempSum = 0;
|
|
|
|
#endif
|
|
|
|
for (int i = 0; i < LAST_DRAWTYPE_ENUM+1; ++i) {
|
|
|
|
if (0 == counts[i]) {
|
|
|
|
// if there were no commands of this type then they should've consumed no time
|
|
|
|
SkASSERT(NULL == typeTimes || 0.0 == (*typeTimes)[i]);
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
overview->append(SkDrawCommand::GetCommandString((DrawType) i));
|
|
|
|
overview->append(": ");
|
2013-03-12 15:33:40 +00:00
|
|
|
overview->appendS32(counts[i]);
|
2013-03-12 13:07:40 +00:00
|
|
|
if (NULL != typeTimes) {
|
|
|
|
overview->append(" - ");
|
2013-03-12 15:33:40 +00:00
|
|
|
overview->appendf("%.2f", (*typeTimes)[i]/(float)numRuns);
|
2013-03-12 13:07:40 +00:00
|
|
|
overview->append("ms");
|
|
|
|
overview->append(" - ");
|
|
|
|
double percent = 100.0*(*typeTimes)[i]/totTime;
|
2013-03-12 15:33:40 +00:00
|
|
|
overview->appendf("%.2f", percent);
|
2013-03-12 13:07:40 +00:00
|
|
|
overview->append("%");
|
|
|
|
#ifdef SK_DEBUG
|
|
|
|
totPercent += percent;
|
|
|
|
tempSum += (*typeTimes)[i];
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
overview->append("<br/>");
|
|
|
|
total += counts[i];
|
|
|
|
}
|
|
|
|
#ifdef SK_DEBUG
|
|
|
|
if (NULL != typeTimes) {
|
2013-03-13 07:01:04 +00:00
|
|
|
SkASSERT(SkScalarNearlyEqual(SkDoubleToScalar(totPercent),
|
2013-03-12 15:33:40 +00:00
|
|
|
SkDoubleToScalar(100.0)));
|
2013-03-13 07:01:04 +00:00
|
|
|
SkASSERT(SkScalarNearlyEqual(SkDoubleToScalar(tempSum),
|
2013-03-12 15:33:40 +00:00
|
|
|
SkDoubleToScalar(totTime)));
|
2013-03-12 13:07:40 +00:00
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
|
|
|
if (totTime > 0.0) {
|
|
|
|
overview->append("Total Time: ");
|
2013-03-12 15:33:40 +00:00
|
|
|
overview->appendf("%.2f", totTime/(float)numRuns);
|
2013-03-12 13:07:40 +00:00
|
|
|
overview->append("ms");
|
|
|
|
#ifdef SK_DEBUG
|
|
|
|
overview->append(" ");
|
2013-03-12 15:33:40 +00:00
|
|
|
overview->appendScalar(SkDoubleToScalar(totPercent));
|
2013-03-12 13:07:40 +00:00
|
|
|
overview->append("% ");
|
|
|
|
#endif
|
|
|
|
overview->append("<br/>");
|
|
|
|
}
|
|
|
|
|
|
|
|
SkString totalStr;
|
|
|
|
totalStr.append("Total Draw Commands: ");
|
2013-03-12 15:33:40 +00:00
|
|
|
totalStr.appendScalar(SkDoubleToScalar(total));
|
2013-03-12 13:07:40 +00:00
|
|
|
totalStr.append("<br/>");
|
|
|
|
overview->insert(0, totalStr);
|
|
|
|
|
|
|
|
overview->append("<br/>");
|
|
|
|
overview->append("SkPicture Width: ");
|
2013-03-12 15:33:40 +00:00
|
|
|
overview->appendS32(pictureWidth());
|
2013-03-12 13:07:40 +00:00
|
|
|
overview->append("px<br/>");
|
|
|
|
overview->append("SkPicture Height: ");
|
2013-03-12 15:33:40 +00:00
|
|
|
overview->appendS32(pictureHeight());
|
2013-03-12 13:07:40 +00:00
|
|
|
overview->append("px");
|
|
|
|
}
|
2013-04-12 14:50:10 +00:00
|
|
|
|
|
|
|
#include "SkImageDecoder.h"
|
|
|
|
|
|
|
|
void forceLinking();
|
|
|
|
void forceLinking() {
|
|
|
|
// This function leaks, but that is okay because it is not intended
|
|
|
|
// to be called. It is only here so that the linker will include the
|
|
|
|
// decoders.
|
|
|
|
SkDEBUGCODE(SkImageDecoder *creator = ) CreateJPEGImageDecoder();
|
|
|
|
SkASSERT(creator);
|
2013-04-15 22:05:03 +00:00
|
|
|
SkDEBUGCODE(creator = ) CreateWEBPImageDecoder();
|
|
|
|
SkASSERT(creator);
|
2013-04-12 15:04:33 +00:00
|
|
|
#if defined(SK_BUILD_FOR_UNIX) && !defined(SK_BUILD_FOR_NACL)
|
2013-04-12 14:50:10 +00:00
|
|
|
SkDEBUGCODE(creator = ) CreateGIFImageDecoder();
|
|
|
|
SkASSERT(creator);
|
|
|
|
#endif
|
|
|
|
}
|