blob: b7960f7f76b0e4bc047a20c99d6bf431371bdc74 [file] [log] [blame]
Brian Osman56a24812017-12-19 11:15:16 -05001/*
2* Copyright 2017 Google Inc.
3*
4* Use of this source code is governed by a BSD-style license that can be
5* found in the LICENSE file.
6*/
7
8#include "StatsLayer.h"
9
10#include "SkCanvas.h"
Hal Canarydf2d27e2019-01-08 09:38:02 -050011#include "SkFont.h"
Brian Osman56a24812017-12-19 11:15:16 -050012#include "SkString.h"
Robert Phillips9882dae2019-03-04 11:00:10 -050013#include "SkSurface.h"
Brian Osman56a24812017-12-19 11:15:16 -050014#include "SkTime.h"
15
16StatsLayer::StatsLayer()
17 : fCurrentMeasurement(0)
18 , fCumulativeMeasurementTime(0)
Brian Osmanb63f6002018-07-24 18:01:53 -040019 , fCumulativeMeasurementCount(0)
20 , fDisplayScale(1.0f) {}
Brian Osman56a24812017-12-19 11:15:16 -050021
22void StatsLayer::resetMeasurements() {
23 for (int i = 0; i < fTimers.count(); ++i) {
24 memset(fTimers[i].fTimes, 0, sizeof(fTimers[i].fTimes));
25 }
26 fCurrentMeasurement = 0;
27 fCumulativeMeasurementTime = 0;
28 fCumulativeMeasurementCount = 0;
29}
30
31StatsLayer::Timer StatsLayer::addTimer(const char* label, SkColor color, SkColor labelColor) {
32 Timer newTimer = fTimers.count();
33 TimerData& newData = fTimers.push_back();
34 memset(newData.fTimes, 0, sizeof(newData.fTimes));
35 newData.fLabel = label;
36 newData.fColor = color;
37 newData.fLabelColor = labelColor ? labelColor : color;
38 return newTimer;
39}
40
41void StatsLayer::beginTiming(Timer timer) {
42 fTimers[timer].fTimes[fCurrentMeasurement] -= SkTime::GetMSecs();
43}
44
45void StatsLayer::endTiming(Timer timer) {
46 fTimers[timer].fTimes[fCurrentMeasurement] += SkTime::GetMSecs();
47}
48
49double StatsLayer::getLastTime(Timer timer) {
50 int idx = (fCurrentMeasurement + (kMeasurementCount - 1)) & (kMeasurementCount - 1);
51 return fTimers[timer].fTimes[idx];
52}
53
Robert Phillips9882dae2019-03-04 11:00:10 -050054void StatsLayer::onPaint(SkSurface* surface) {
Brian Osman56a24812017-12-19 11:15:16 -050055 // Advance our timing bookkeeping
56 for (int i = 0; i < fTimers.count(); ++i) {
57 fCumulativeMeasurementTime += fTimers[i].fTimes[fCurrentMeasurement];
58 }
59 fCumulativeMeasurementCount++;
60 fCurrentMeasurement = (fCurrentMeasurement + 1) & (kMeasurementCount - 1);
61 SkASSERT(fCurrentMeasurement < kMeasurementCount);
62 for (int i = 0; i < fTimers.count(); ++i) {
63 fTimers[i].fTimes[fCurrentMeasurement] = 0;
64 }
65
Florin Malita4e9e3252018-05-07 16:30:01 -040066#ifdef SK_BUILD_FOR_ANDROID
67 // Scale up the stats overlay on Android devices
68 static constexpr SkScalar kScale = 1.5;
69#else
Brian Osmanb63f6002018-07-24 18:01:53 -040070 SkScalar kScale = fDisplayScale;
Florin Malita4e9e3252018-05-07 16:30:01 -040071#endif
72
Brian Osman56a24812017-12-19 11:15:16 -050073 // Now draw everything
74 static const float kPixelPerMS = 2.0f;
75 static const int kDisplayWidth = 192;
76 static const int kGraphHeight = 100;
77 static const int kTextHeight = 60;
78 static const int kDisplayHeight = kGraphHeight + kTextHeight;
79 static const int kDisplayPadding = 10;
80 static const int kGraphPadding = 3;
81 static const SkScalar kBaseMS = 1000.f / 60.f; // ms/frame to hit 60 fps
82
Robert Phillips9882dae2019-03-04 11:00:10 -050083 auto canvas = surface->getCanvas();
Brian Osman56a24812017-12-19 11:15:16 -050084 SkISize canvasSize = canvas->getBaseLayerSize();
85 SkRect rect = SkRect::MakeXYWH(SkIntToScalar(canvasSize.fWidth-kDisplayWidth-kDisplayPadding),
86 SkIntToScalar(kDisplayPadding),
87 SkIntToScalar(kDisplayWidth), SkIntToScalar(kDisplayHeight));
88 SkPaint paint;
89 canvas->save();
90
Florin Malita4e9e3252018-05-07 16:30:01 -040091 // Scale the canvas while keeping the right edge in place.
92 canvas->concat(SkMatrix::MakeRectToRect(SkRect::Make(canvasSize),
93 SkRect::MakeXYWH(canvasSize.width() * (1 - kScale),
94 0,
95 canvasSize.width() * kScale,
96 canvasSize.height() * kScale),
97 SkMatrix::kFill_ScaleToFit));
98
Brian Osman56a24812017-12-19 11:15:16 -050099 paint.setColor(SK_ColorBLACK);
100 canvas->drawRect(rect, paint);
101 // draw the 16ms line
102 paint.setColor(SK_ColorLTGRAY);
103 canvas->drawLine(rect.fLeft, rect.fBottom - kBaseMS*kPixelPerMS,
104 rect.fRight, rect.fBottom - kBaseMS*kPixelPerMS, paint);
105 paint.setColor(SK_ColorRED);
106 paint.setStyle(SkPaint::kStroke_Style);
107 canvas->drawRect(rect, paint);
108 paint.setStyle(SkPaint::kFill_Style);
109
110 int x = SkScalarTruncToInt(rect.fLeft) + kGraphPadding;
111 const int xStep = 3;
112 int i = fCurrentMeasurement;
113 double ms = 0;
114 SkTDArray<double> sumTimes;
115 sumTimes.setCount(fTimers.count());
116 memset(sumTimes.begin(), 0, sumTimes.count() * sizeof(double));
117 int count = 0;
118 do {
119 int startY = SkScalarTruncToInt(rect.fBottom);
120 double inc = 0;
121 for (int timer = 0; timer < fTimers.count(); ++timer) {
122 int height = (int)(fTimers[timer].fTimes[i] * kPixelPerMS + 0.5);
123 int endY = SkTMax(startY - height, kDisplayPadding + kTextHeight);
124 paint.setColor(fTimers[timer].fColor);
125 canvas->drawLine(SkIntToScalar(x), SkIntToScalar(startY),
126 SkIntToScalar(x), SkIntToScalar(endY), paint);
127 startY = endY;
128 inc += fTimers[timer].fTimes[i];
129 sumTimes[timer] += fTimers[timer].fTimes[i];
130 }
131
132 if (inc > 0) {
133 ms += inc;
134 ++count;
135 }
136
137 i++;
138 i &= (kMeasurementCount - 1); // fast mod
139 x += xStep;
140 } while (i != fCurrentMeasurement);
141
Hal Canarydf2d27e2019-01-08 09:38:02 -0500142 SkFont font(nullptr, 16);
Brian Osman56a24812017-12-19 11:15:16 -0500143 paint.setColor(SK_ColorWHITE);
Hal Canarydf2d27e2019-01-08 09:38:02 -0500144 double time = ms / SkTMax(1, count);
145 double measure = fCumulativeMeasurementTime / SkTMax(1, fCumulativeMeasurementCount);
146 canvas->drawString(SkStringPrintf("%4.3f ms -> %4.3f ms", time, measure),
147 rect.fLeft + 3, rect.fTop + 14, font, paint);
Brian Osman56a24812017-12-19 11:15:16 -0500148
149 for (int timer = 0; timer < fTimers.count(); ++timer) {
Brian Osman56a24812017-12-19 11:15:16 -0500150 paint.setColor(fTimers[timer].fLabelColor);
Hal Canarydf2d27e2019-01-08 09:38:02 -0500151 canvas->drawString(SkStringPrintf("%s: %4.3f ms", fTimers[timer].fLabel.c_str(),
152 sumTimes[timer] / SkTMax(1, count)),
153 rect.fLeft + 3, rect.fTop + 28 + (14 * timer), font, paint);
Brian Osman56a24812017-12-19 11:15:16 -0500154 }
155
156 canvas->restore();
157}