blob: f4cc49b851d5e5760af84d98ad1d72003f08ac89 [file] [log] [blame]
Jamie Gennis82dbc742012-11-08 19:23:28 -08001/*
2 * Copyright (C) 2012 The Android Open Source Project
3 *
4 * Licensed under the Apache License, Version 2.0 (the "License");
5 * you may not use this file except in compliance with the License.
6 * You may obtain a copy of the License at
7 *
8 * http://www.apache.org/licenses/LICENSE-2.0
9 *
10 * Unless required by applicable law or agreed to in writing, software
11 * distributed under the License is distributed on an "AS IS" BASIS,
12 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13 * See the License for the specific language governing permissions and
14 * limitations under the License.
15 */
16
17// This is needed for stdint.h to define INT64_MAX in C++
18#define __STDC_LIMIT_MACROS
19
Greg Hackmann86efcc02014-03-07 12:44:02 -080020#include <inttypes.h>
21
Yiwei Zhang5434a782018-12-05 18:06:32 -080022#include <android-base/stringprintf.h>
Mark Salyzyna5e161b2016-09-29 08:08:05 -070023#include <android/log.h>
24#include <utils/String8.h>
Jamie Gennis6547ff42013-07-16 20:12:42 -070025
Svetoslavd85084b2014-03-20 10:28:31 -070026#include <ui/FrameStats.h>
Jamie Gennis82dbc742012-11-08 19:23:28 -080027
Jamie Gennis82dbc742012-11-08 19:23:28 -080028#include "FrameTracker.h"
Jamie Gennis6547ff42013-07-16 20:12:42 -070029#include "EventLog/EventLog.h"
Jamie Gennis82dbc742012-11-08 19:23:28 -080030
31namespace android {
32
33FrameTracker::FrameTracker() :
34 mOffset(0),
Jamie Gennis6547ff42013-07-16 20:12:42 -070035 mNumFences(0),
36 mDisplayPeriod(0) {
37 resetFrameCountersLocked();
Jamie Gennis82dbc742012-11-08 19:23:28 -080038}
39
40void FrameTracker::setDesiredPresentTime(nsecs_t presentTime) {
Jamie Gennis4b0eba92013-02-05 13:30:24 -080041 Mutex::Autolock lock(mMutex);
Jamie Gennis82dbc742012-11-08 19:23:28 -080042 mFrameRecords[mOffset].desiredPresentTime = presentTime;
43}
44
45void FrameTracker::setFrameReadyTime(nsecs_t readyTime) {
Jamie Gennis4b0eba92013-02-05 13:30:24 -080046 Mutex::Autolock lock(mMutex);
Jamie Gennis82dbc742012-11-08 19:23:28 -080047 mFrameRecords[mOffset].frameReadyTime = readyTime;
48}
49
Brian Anderson3d4039d2016-09-23 16:31:30 -070050void FrameTracker::setFrameReadyFence(
51 std::shared_ptr<FenceTime>&& readyFence) {
Jamie Gennis4b0eba92013-02-05 13:30:24 -080052 Mutex::Autolock lock(mMutex);
Brian Anderson3d4039d2016-09-23 16:31:30 -070053 mFrameRecords[mOffset].frameReadyFence = std::move(readyFence);
Jamie Gennis82dbc742012-11-08 19:23:28 -080054 mNumFences++;
55}
56
57void FrameTracker::setActualPresentTime(nsecs_t presentTime) {
Jamie Gennis4b0eba92013-02-05 13:30:24 -080058 Mutex::Autolock lock(mMutex);
Jamie Gennis82dbc742012-11-08 19:23:28 -080059 mFrameRecords[mOffset].actualPresentTime = presentTime;
60}
61
Brian Anderson3d4039d2016-09-23 16:31:30 -070062void FrameTracker::setActualPresentFence(
63 std::shared_ptr<FenceTime>&& readyFence) {
Jamie Gennis4b0eba92013-02-05 13:30:24 -080064 Mutex::Autolock lock(mMutex);
Brian Anderson3d4039d2016-09-23 16:31:30 -070065 mFrameRecords[mOffset].actualPresentFence = std::move(readyFence);
Jamie Gennis82dbc742012-11-08 19:23:28 -080066 mNumFences++;
67}
68
Jamie Gennis6547ff42013-07-16 20:12:42 -070069void FrameTracker::setDisplayRefreshPeriod(nsecs_t displayPeriod) {
70 Mutex::Autolock lock(mMutex);
71 mDisplayPeriod = displayPeriod;
72}
73
Jamie Gennis82dbc742012-11-08 19:23:28 -080074void FrameTracker::advanceFrame() {
Jamie Gennis4b0eba92013-02-05 13:30:24 -080075 Mutex::Autolock lock(mMutex);
Jamie Gennis6547ff42013-07-16 20:12:42 -070076
77 // Update the statistic to include the frame we just finished.
78 updateStatsLocked(mOffset);
79
80 // Advance to the next frame.
Jamie Gennis82dbc742012-11-08 19:23:28 -080081 mOffset = (mOffset+1) % NUM_FRAME_RECORDS;
82 mFrameRecords[mOffset].desiredPresentTime = INT64_MAX;
83 mFrameRecords[mOffset].frameReadyTime = INT64_MAX;
84 mFrameRecords[mOffset].actualPresentTime = INT64_MAX;
85
Peiyong Lin566a3b42018-01-09 18:22:43 -080086 if (mFrameRecords[mOffset].frameReadyFence != nullptr) {
Jamie Gennis82dbc742012-11-08 19:23:28 -080087 // We're clobbering an unsignaled fence, so we need to decrement the
88 // fence count.
Peiyong Lin566a3b42018-01-09 18:22:43 -080089 mFrameRecords[mOffset].frameReadyFence = nullptr;
Jamie Gennis82dbc742012-11-08 19:23:28 -080090 mNumFences--;
91 }
92
Peiyong Lin566a3b42018-01-09 18:22:43 -080093 if (mFrameRecords[mOffset].actualPresentFence != nullptr) {
Jamie Gennis82dbc742012-11-08 19:23:28 -080094 // We're clobbering an unsignaled fence, so we need to decrement the
95 // fence count.
Peiyong Lin566a3b42018-01-09 18:22:43 -080096 mFrameRecords[mOffset].actualPresentFence = nullptr;
Jamie Gennis82dbc742012-11-08 19:23:28 -080097 mNumFences--;
98 }
Jamie Gennis82dbc742012-11-08 19:23:28 -080099}
100
Svetoslavd85084b2014-03-20 10:28:31 -0700101void FrameTracker::clearStats() {
Jamie Gennis4b0eba92013-02-05 13:30:24 -0800102 Mutex::Autolock lock(mMutex);
Jamie Gennis82dbc742012-11-08 19:23:28 -0800103 for (size_t i = 0; i < NUM_FRAME_RECORDS; i++) {
104 mFrameRecords[i].desiredPresentTime = 0;
105 mFrameRecords[i].frameReadyTime = 0;
106 mFrameRecords[i].actualPresentTime = 0;
Brian Anderson3d4039d2016-09-23 16:31:30 -0700107 mFrameRecords[i].frameReadyFence.reset();
108 mFrameRecords[i].actualPresentFence.reset();
Jamie Gennis82dbc742012-11-08 19:23:28 -0800109 }
110 mNumFences = 0;
Jamie Gennis4b0eba92013-02-05 13:30:24 -0800111 mFrameRecords[mOffset].desiredPresentTime = INT64_MAX;
112 mFrameRecords[mOffset].frameReadyTime = INT64_MAX;
113 mFrameRecords[mOffset].actualPresentTime = INT64_MAX;
Jamie Gennis82dbc742012-11-08 19:23:28 -0800114}
115
Svetoslavd85084b2014-03-20 10:28:31 -0700116void FrameTracker::getStats(FrameStats* outStats) const {
117 Mutex::Autolock lock(mMutex);
118 processFencesLocked();
119
120 outStats->refreshPeriodNano = mDisplayPeriod;
121
122 const size_t offset = mOffset;
123 for (size_t i = 1; i < NUM_FRAME_RECORDS; i++) {
124 const size_t index = (offset + i) % NUM_FRAME_RECORDS;
125
126 // Skip frame records with no data (if buffer not yet full).
127 if (mFrameRecords[index].desiredPresentTime == 0) {
128 continue;
129 }
130
131 nsecs_t desiredPresentTimeNano = mFrameRecords[index].desiredPresentTime;
132 outStats->desiredPresentTimesNano.push_back(desiredPresentTimeNano);
133
134 nsecs_t actualPresentTimeNano = mFrameRecords[index].actualPresentTime;
135 outStats->actualPresentTimesNano.push_back(actualPresentTimeNano);
136
137 nsecs_t frameReadyTimeNano = mFrameRecords[index].frameReadyTime;
138 outStats->frameReadyTimesNano.push_back(frameReadyTimeNano);
139 }
140}
141
Jamie Gennis6547ff42013-07-16 20:12:42 -0700142void FrameTracker::logAndResetStats(const String8& name) {
143 Mutex::Autolock lock(mMutex);
144 logStatsLocked(name);
145 resetFrameCountersLocked();
146}
147
Jamie Gennis4b0eba92013-02-05 13:30:24 -0800148void FrameTracker::processFencesLocked() const {
Jamie Gennis82dbc742012-11-08 19:23:28 -0800149 FrameRecord* records = const_cast<FrameRecord*>(mFrameRecords);
150 int& numFences = const_cast<int&>(mNumFences);
151
152 for (int i = 1; i < NUM_FRAME_RECORDS && numFences > 0; i++) {
153 size_t idx = (mOffset+NUM_FRAME_RECORDS-i) % NUM_FRAME_RECORDS;
Jamie Gennis6547ff42013-07-16 20:12:42 -0700154 bool updated = false;
Jamie Gennis82dbc742012-11-08 19:23:28 -0800155
Brian Anderson3d4039d2016-09-23 16:31:30 -0700156 const std::shared_ptr<FenceTime>& rfence = records[idx].frameReadyFence;
Peiyong Lin566a3b42018-01-09 18:22:43 -0800157 if (rfence != nullptr) {
Jamie Gennis82dbc742012-11-08 19:23:28 -0800158 records[idx].frameReadyTime = rfence->getSignalTime();
159 if (records[idx].frameReadyTime < INT64_MAX) {
Peiyong Lin566a3b42018-01-09 18:22:43 -0800160 records[idx].frameReadyFence = nullptr;
Jamie Gennis82dbc742012-11-08 19:23:28 -0800161 numFences--;
Jamie Gennis6547ff42013-07-16 20:12:42 -0700162 updated = true;
Jamie Gennis82dbc742012-11-08 19:23:28 -0800163 }
164 }
165
Brian Anderson3d4039d2016-09-23 16:31:30 -0700166 const std::shared_ptr<FenceTime>& pfence =
167 records[idx].actualPresentFence;
Peiyong Lin566a3b42018-01-09 18:22:43 -0800168 if (pfence != nullptr) {
Jamie Gennis82dbc742012-11-08 19:23:28 -0800169 records[idx].actualPresentTime = pfence->getSignalTime();
170 if (records[idx].actualPresentTime < INT64_MAX) {
Peiyong Lin566a3b42018-01-09 18:22:43 -0800171 records[idx].actualPresentFence = nullptr;
Jamie Gennis82dbc742012-11-08 19:23:28 -0800172 numFences--;
Jamie Gennis6547ff42013-07-16 20:12:42 -0700173 updated = true;
Jamie Gennis82dbc742012-11-08 19:23:28 -0800174 }
175 }
Jamie Gennis6547ff42013-07-16 20:12:42 -0700176
177 if (updated) {
178 updateStatsLocked(idx);
179 }
Jamie Gennis82dbc742012-11-08 19:23:28 -0800180 }
181}
182
Jamie Gennis6547ff42013-07-16 20:12:42 -0700183void FrameTracker::updateStatsLocked(size_t newFrameIdx) const {
184 int* numFrames = const_cast<int*>(mNumFrames);
185
186 if (mDisplayPeriod > 0 && isFrameValidLocked(newFrameIdx)) {
187 size_t prevFrameIdx = (newFrameIdx+NUM_FRAME_RECORDS-1) %
188 NUM_FRAME_RECORDS;
189
190 if (isFrameValidLocked(prevFrameIdx)) {
191 nsecs_t newPresentTime =
192 mFrameRecords[newFrameIdx].actualPresentTime;
193 nsecs_t prevPresentTime =
194 mFrameRecords[prevFrameIdx].actualPresentTime;
195
196 nsecs_t duration = newPresentTime - prevPresentTime;
197 int numPeriods = int((duration + mDisplayPeriod/2) /
198 mDisplayPeriod);
199
200 for (int i = 0; i < NUM_FRAME_BUCKETS-1; i++) {
201 int nextBucket = 1 << (i+1);
202 if (numPeriods < nextBucket) {
203 numFrames[i]++;
204 return;
205 }
206 }
207
208 // The last duration bucket is a catch-all.
209 numFrames[NUM_FRAME_BUCKETS-1]++;
210 }
211 }
212}
213
214void FrameTracker::resetFrameCountersLocked() {
215 for (int i = 0; i < NUM_FRAME_BUCKETS; i++) {
216 mNumFrames[i] = 0;
217 }
218}
219
220void FrameTracker::logStatsLocked(const String8& name) const {
221 for (int i = 0; i < NUM_FRAME_BUCKETS; i++) {
222 if (mNumFrames[i] > 0) {
223 EventLog::logFrameDurations(name, mNumFrames, NUM_FRAME_BUCKETS);
224 return;
225 }
226 }
227}
228
229bool FrameTracker::isFrameValidLocked(size_t idx) const {
230 return mFrameRecords[idx].actualPresentTime > 0 &&
231 mFrameRecords[idx].actualPresentTime < INT64_MAX;
232}
233
Yiwei Zhang5434a782018-12-05 18:06:32 -0800234void FrameTracker::dumpStats(std::string& result) const {
Jamie Gennis4b0eba92013-02-05 13:30:24 -0800235 Mutex::Autolock lock(mMutex);
236 processFencesLocked();
Jamie Gennis82dbc742012-11-08 19:23:28 -0800237
238 const size_t o = mOffset;
239 for (size_t i = 1; i < NUM_FRAME_RECORDS; i++) {
240 const size_t index = (o+i) % NUM_FRAME_RECORDS;
Yiwei Zhang5434a782018-12-05 18:06:32 -0800241 base::StringAppendF(&result, "%" PRId64 "\t%" PRId64 "\t%" PRId64 "\n",
242 mFrameRecords[index].desiredPresentTime,
243 mFrameRecords[index].actualPresentTime,
244 mFrameRecords[index].frameReadyTime);
Jamie Gennis82dbc742012-11-08 19:23:28 -0800245 }
246 result.append("\n");
247}
248
249} // namespace android