2017-12-19 16:15:16 +00:00
|
|
|
/*
|
|
|
|
* Copyright 2017 Google Inc.
|
|
|
|
*
|
|
|
|
* Use of this source code is governed by a BSD-style license that can be
|
|
|
|
* found in the LICENSE file.
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include "StatsLayer.h"
|
|
|
|
|
|
|
|
#include "SkCanvas.h"
|
2019-01-08 14:38:02 +00:00
|
|
|
#include "SkFont.h"
|
2017-12-19 16:15:16 +00:00
|
|
|
#include "SkString.h"
|
2019-03-04 16:00:10 +00:00
|
|
|
#include "SkSurface.h"
|
2017-12-19 16:15:16 +00:00
|
|
|
#include "SkTime.h"
|
|
|
|
|
|
|
|
StatsLayer::StatsLayer()
|
2019-03-21 18:43:00 +00:00
|
|
|
: fCurrentMeasurement(-1)
|
|
|
|
, fLastTotalBegin(0)
|
2017-12-19 16:15:16 +00:00
|
|
|
, fCumulativeMeasurementTime(0)
|
2018-07-24 22:01:53 +00:00
|
|
|
, fCumulativeMeasurementCount(0)
|
2019-03-21 18:43:00 +00:00
|
|
|
, fDisplayScale(1.0f) {
|
|
|
|
memset(fTotalTimes, 0, sizeof(fTotalTimes));
|
|
|
|
}
|
2017-12-19 16:15:16 +00:00
|
|
|
|
|
|
|
void StatsLayer::resetMeasurements() {
|
|
|
|
for (int i = 0; i < fTimers.count(); ++i) {
|
|
|
|
memset(fTimers[i].fTimes, 0, sizeof(fTimers[i].fTimes));
|
|
|
|
}
|
2019-03-21 18:43:00 +00:00
|
|
|
memset(fTotalTimes, 0, sizeof(fTotalTimes));
|
|
|
|
fCurrentMeasurement = -1;
|
|
|
|
fLastTotalBegin = 0;
|
2017-12-19 16:15:16 +00:00
|
|
|
fCumulativeMeasurementTime = 0;
|
|
|
|
fCumulativeMeasurementCount = 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
StatsLayer::Timer StatsLayer::addTimer(const char* label, SkColor color, SkColor labelColor) {
|
|
|
|
Timer newTimer = fTimers.count();
|
|
|
|
TimerData& newData = fTimers.push_back();
|
|
|
|
memset(newData.fTimes, 0, sizeof(newData.fTimes));
|
|
|
|
newData.fLabel = label;
|
|
|
|
newData.fColor = color;
|
|
|
|
newData.fLabelColor = labelColor ? labelColor : color;
|
|
|
|
return newTimer;
|
|
|
|
}
|
|
|
|
|
|
|
|
void StatsLayer::beginTiming(Timer timer) {
|
2019-03-21 18:43:00 +00:00
|
|
|
if (fCurrentMeasurement >= 0) {
|
|
|
|
fTimers[timer].fTimes[fCurrentMeasurement] -= SkTime::GetMSecs();
|
|
|
|
}
|
2017-12-19 16:15:16 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void StatsLayer::endTiming(Timer timer) {
|
2019-03-21 18:43:00 +00:00
|
|
|
if (fCurrentMeasurement >= 0) {
|
|
|
|
fTimers[timer].fTimes[fCurrentMeasurement] += SkTime::GetMSecs();
|
|
|
|
}
|
2017-12-19 16:15:16 +00:00
|
|
|
}
|
|
|
|
|
2019-03-21 18:43:00 +00:00
|
|
|
void StatsLayer::onPrePaint() {
|
|
|
|
if (fCurrentMeasurement >= 0) {
|
|
|
|
fTotalTimes[fCurrentMeasurement] = SkTime::GetMSecs() - fLastTotalBegin;
|
|
|
|
fCumulativeMeasurementTime += fTotalTimes[fCurrentMeasurement];
|
|
|
|
fCumulativeMeasurementCount++;
|
|
|
|
}
|
|
|
|
fCurrentMeasurement = (fCurrentMeasurement + 1) & (kMeasurementCount - 1);
|
|
|
|
SkASSERT(fCurrentMeasurement >= 0 && fCurrentMeasurement < kMeasurementCount);
|
|
|
|
fLastTotalBegin = SkTime::GetMSecs();
|
2017-12-19 16:15:16 +00:00
|
|
|
}
|
|
|
|
|
2019-03-04 16:00:10 +00:00
|
|
|
void StatsLayer::onPaint(SkSurface* surface) {
|
2019-03-21 18:43:00 +00:00
|
|
|
int nextMeasurement = (fCurrentMeasurement + 1) & (kMeasurementCount - 1);
|
2017-12-19 16:15:16 +00:00
|
|
|
for (int i = 0; i < fTimers.count(); ++i) {
|
2019-03-21 18:43:00 +00:00
|
|
|
fTimers[i].fTimes[nextMeasurement] = 0;
|
2017-12-19 16:15:16 +00:00
|
|
|
}
|
|
|
|
|
2018-05-07 20:30:01 +00:00
|
|
|
#ifdef SK_BUILD_FOR_ANDROID
|
|
|
|
// Scale up the stats overlay on Android devices
|
|
|
|
static constexpr SkScalar kScale = 1.5;
|
|
|
|
#else
|
2018-07-24 22:01:53 +00:00
|
|
|
SkScalar kScale = fDisplayScale;
|
2018-05-07 20:30:01 +00:00
|
|
|
#endif
|
|
|
|
|
2017-12-19 16:15:16 +00:00
|
|
|
// Now draw everything
|
|
|
|
static const float kPixelPerMS = 2.0f;
|
|
|
|
static const int kDisplayWidth = 192;
|
|
|
|
static const int kGraphHeight = 100;
|
|
|
|
static const int kTextHeight = 60;
|
|
|
|
static const int kDisplayHeight = kGraphHeight + kTextHeight;
|
|
|
|
static const int kDisplayPadding = 10;
|
|
|
|
static const int kGraphPadding = 3;
|
|
|
|
static const SkScalar kBaseMS = 1000.f / 60.f; // ms/frame to hit 60 fps
|
|
|
|
|
2019-03-04 16:00:10 +00:00
|
|
|
auto canvas = surface->getCanvas();
|
2017-12-19 16:15:16 +00:00
|
|
|
SkISize canvasSize = canvas->getBaseLayerSize();
|
|
|
|
SkRect rect = SkRect::MakeXYWH(SkIntToScalar(canvasSize.fWidth-kDisplayWidth-kDisplayPadding),
|
|
|
|
SkIntToScalar(kDisplayPadding),
|
|
|
|
SkIntToScalar(kDisplayWidth), SkIntToScalar(kDisplayHeight));
|
|
|
|
SkPaint paint;
|
|
|
|
canvas->save();
|
|
|
|
|
2018-05-07 20:30:01 +00:00
|
|
|
// Scale the canvas while keeping the right edge in place.
|
|
|
|
canvas->concat(SkMatrix::MakeRectToRect(SkRect::Make(canvasSize),
|
|
|
|
SkRect::MakeXYWH(canvasSize.width() * (1 - kScale),
|
|
|
|
0,
|
|
|
|
canvasSize.width() * kScale,
|
|
|
|
canvasSize.height() * kScale),
|
|
|
|
SkMatrix::kFill_ScaleToFit));
|
|
|
|
|
2017-12-19 16:15:16 +00:00
|
|
|
paint.setColor(SK_ColorBLACK);
|
|
|
|
canvas->drawRect(rect, paint);
|
|
|
|
// draw the 16ms line
|
|
|
|
paint.setColor(SK_ColorLTGRAY);
|
|
|
|
canvas->drawLine(rect.fLeft, rect.fBottom - kBaseMS*kPixelPerMS,
|
|
|
|
rect.fRight, rect.fBottom - kBaseMS*kPixelPerMS, paint);
|
|
|
|
paint.setColor(SK_ColorRED);
|
|
|
|
paint.setStyle(SkPaint::kStroke_Style);
|
|
|
|
canvas->drawRect(rect, paint);
|
|
|
|
paint.setStyle(SkPaint::kFill_Style);
|
|
|
|
|
|
|
|
int x = SkScalarTruncToInt(rect.fLeft) + kGraphPadding;
|
|
|
|
const int xStep = 3;
|
2019-03-21 18:43:00 +00:00
|
|
|
int i = nextMeasurement;
|
2017-12-19 16:15:16 +00:00
|
|
|
SkTDArray<double> sumTimes;
|
|
|
|
sumTimes.setCount(fTimers.count());
|
|
|
|
memset(sumTimes.begin(), 0, sumTimes.count() * sizeof(double));
|
|
|
|
int count = 0;
|
2019-03-21 18:43:00 +00:00
|
|
|
double totalTime = 0;
|
|
|
|
int totalCount = 0;
|
2017-12-19 16:15:16 +00:00
|
|
|
do {
|
|
|
|
int startY = SkScalarTruncToInt(rect.fBottom);
|
|
|
|
double inc = 0;
|
|
|
|
for (int timer = 0; timer < fTimers.count(); ++timer) {
|
|
|
|
int height = (int)(fTimers[timer].fTimes[i] * kPixelPerMS + 0.5);
|
|
|
|
int endY = SkTMax(startY - height, kDisplayPadding + kTextHeight);
|
|
|
|
paint.setColor(fTimers[timer].fColor);
|
|
|
|
canvas->drawLine(SkIntToScalar(x), SkIntToScalar(startY),
|
|
|
|
SkIntToScalar(x), SkIntToScalar(endY), paint);
|
|
|
|
startY = endY;
|
|
|
|
inc += fTimers[timer].fTimes[i];
|
|
|
|
sumTimes[timer] += fTimers[timer].fTimes[i];
|
|
|
|
}
|
|
|
|
|
2019-03-21 18:43:00 +00:00
|
|
|
int height = (int)(fTotalTimes[i] * kPixelPerMS + 0.5);
|
|
|
|
height = SkTMax(0, height - (SkScalarTruncToInt(rect.fBottom) - startY));
|
|
|
|
int endY = SkTMax(startY - height, kDisplayPadding + kTextHeight);
|
|
|
|
paint.setColor(SK_ColorWHITE);
|
|
|
|
canvas->drawLine(SkIntToScalar(x), SkIntToScalar(startY),
|
|
|
|
SkIntToScalar(x), SkIntToScalar(endY), paint);
|
|
|
|
totalTime += fTotalTimes[i];
|
|
|
|
if (fTotalTimes[i] > 0) {
|
|
|
|
++totalCount;
|
|
|
|
}
|
|
|
|
|
2017-12-19 16:15:16 +00:00
|
|
|
if (inc > 0) {
|
|
|
|
++count;
|
|
|
|
}
|
|
|
|
|
|
|
|
i++;
|
|
|
|
i &= (kMeasurementCount - 1); // fast mod
|
|
|
|
x += xStep;
|
2019-03-21 18:43:00 +00:00
|
|
|
} while (i != nextMeasurement);
|
2017-12-19 16:15:16 +00:00
|
|
|
|
2019-01-08 14:38:02 +00:00
|
|
|
SkFont font(nullptr, 16);
|
2017-12-19 16:15:16 +00:00
|
|
|
paint.setColor(SK_ColorWHITE);
|
2019-03-21 18:43:00 +00:00
|
|
|
double time = totalTime / SkTMax(1, totalCount);
|
2019-01-08 14:38:02 +00:00
|
|
|
double measure = fCumulativeMeasurementTime / SkTMax(1, fCumulativeMeasurementCount);
|
|
|
|
canvas->drawString(SkStringPrintf("%4.3f ms -> %4.3f ms", time, measure),
|
|
|
|
rect.fLeft + 3, rect.fTop + 14, font, paint);
|
2017-12-19 16:15:16 +00:00
|
|
|
|
|
|
|
for (int timer = 0; timer < fTimers.count(); ++timer) {
|
|
|
|
paint.setColor(fTimers[timer].fLabelColor);
|
2019-01-08 14:38:02 +00:00
|
|
|
canvas->drawString(SkStringPrintf("%s: %4.3f ms", fTimers[timer].fLabel.c_str(),
|
|
|
|
sumTimes[timer] / SkTMax(1, count)),
|
|
|
|
rect.fLeft + 3, rect.fTop + 28 + (14 * timer), font, paint);
|
2017-12-19 16:15:16 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
canvas->restore();
|
|
|
|
}
|