blob: 99c4daaba5e31a974b8ab4274460e2dd51fc8cb4 [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
Mark Salyzyna5e161b2016-09-29 08:08:05 -070022#include <android/log.h>
23#include <utils/String8.h>
Jamie Gennis6547ff42013-07-16 20:12:42 -070024
Svetoslavd85084b2014-03-20 10:28:31 -070025#include <ui/FrameStats.h>
Jamie Gennis82dbc742012-11-08 19:23:28 -080026
Jamie Gennis82dbc742012-11-08 19:23:28 -080027#include "FrameTracker.h"
Jamie Gennis6547ff42013-07-16 20:12:42 -070028#include "EventLog/EventLog.h"
Jamie Gennis82dbc742012-11-08 19:23:28 -080029
30namespace android {
31
32FrameTracker::FrameTracker() :
33 mOffset(0),
Jamie Gennis6547ff42013-07-16 20:12:42 -070034 mNumFences(0),
35 mDisplayPeriod(0) {
36 resetFrameCountersLocked();
Jamie Gennis82dbc742012-11-08 19:23:28 -080037}
38
39void FrameTracker::setDesiredPresentTime(nsecs_t presentTime) {
Jamie Gennis4b0eba92013-02-05 13:30:24 -080040 Mutex::Autolock lock(mMutex);
Jamie Gennis82dbc742012-11-08 19:23:28 -080041 mFrameRecords[mOffset].desiredPresentTime = presentTime;
42}
43
44void FrameTracker::setFrameReadyTime(nsecs_t readyTime) {
Jamie Gennis4b0eba92013-02-05 13:30:24 -080045 Mutex::Autolock lock(mMutex);
Jamie Gennis82dbc742012-11-08 19:23:28 -080046 mFrameRecords[mOffset].frameReadyTime = readyTime;
47}
48
Brian Anderson3d4039d2016-09-23 16:31:30 -070049void FrameTracker::setFrameReadyFence(
50 std::shared_ptr<FenceTime>&& readyFence) {
Jamie Gennis4b0eba92013-02-05 13:30:24 -080051 Mutex::Autolock lock(mMutex);
Brian Anderson3d4039d2016-09-23 16:31:30 -070052 mFrameRecords[mOffset].frameReadyFence = std::move(readyFence);
Jamie Gennis82dbc742012-11-08 19:23:28 -080053 mNumFences++;
54}
55
56void FrameTracker::setActualPresentTime(nsecs_t presentTime) {
Jamie Gennis4b0eba92013-02-05 13:30:24 -080057 Mutex::Autolock lock(mMutex);
Jamie Gennis82dbc742012-11-08 19:23:28 -080058 mFrameRecords[mOffset].actualPresentTime = presentTime;
59}
60
Brian Anderson3d4039d2016-09-23 16:31:30 -070061void FrameTracker::setActualPresentFence(
62 std::shared_ptr<FenceTime>&& readyFence) {
Jamie Gennis4b0eba92013-02-05 13:30:24 -080063 Mutex::Autolock lock(mMutex);
Brian Anderson3d4039d2016-09-23 16:31:30 -070064 mFrameRecords[mOffset].actualPresentFence = std::move(readyFence);
Jamie Gennis82dbc742012-11-08 19:23:28 -080065 mNumFences++;
66}
67
Jamie Gennis6547ff42013-07-16 20:12:42 -070068void FrameTracker::setDisplayRefreshPeriod(nsecs_t displayPeriod) {
69 Mutex::Autolock lock(mMutex);
70 mDisplayPeriod = displayPeriod;
71}
72
Jamie Gennis82dbc742012-11-08 19:23:28 -080073void FrameTracker::advanceFrame() {
Jamie Gennis4b0eba92013-02-05 13:30:24 -080074 Mutex::Autolock lock(mMutex);
Jamie Gennis6547ff42013-07-16 20:12:42 -070075
76 // Update the statistic to include the frame we just finished.
77 updateStatsLocked(mOffset);
78
79 // Advance to the next frame.
Jamie Gennis82dbc742012-11-08 19:23:28 -080080 mOffset = (mOffset+1) % NUM_FRAME_RECORDS;
81 mFrameRecords[mOffset].desiredPresentTime = INT64_MAX;
82 mFrameRecords[mOffset].frameReadyTime = INT64_MAX;
83 mFrameRecords[mOffset].actualPresentTime = INT64_MAX;
84
85 if (mFrameRecords[mOffset].frameReadyFence != NULL) {
86 // We're clobbering an unsignaled fence, so we need to decrement the
87 // fence count.
88 mFrameRecords[mOffset].frameReadyFence = NULL;
89 mNumFences--;
90 }
91
92 if (mFrameRecords[mOffset].actualPresentFence != NULL) {
93 // We're clobbering an unsignaled fence, so we need to decrement the
94 // fence count.
95 mFrameRecords[mOffset].actualPresentFence = NULL;
96 mNumFences--;
97 }
Jamie Gennis82dbc742012-11-08 19:23:28 -080098}
99
Svetoslavd85084b2014-03-20 10:28:31 -0700100void FrameTracker::clearStats() {
Jamie Gennis4b0eba92013-02-05 13:30:24 -0800101 Mutex::Autolock lock(mMutex);
Jamie Gennis82dbc742012-11-08 19:23:28 -0800102 for (size_t i = 0; i < NUM_FRAME_RECORDS; i++) {
103 mFrameRecords[i].desiredPresentTime = 0;
104 mFrameRecords[i].frameReadyTime = 0;
105 mFrameRecords[i].actualPresentTime = 0;
Brian Anderson3d4039d2016-09-23 16:31:30 -0700106 mFrameRecords[i].frameReadyFence.reset();
107 mFrameRecords[i].actualPresentFence.reset();
Jamie Gennis82dbc742012-11-08 19:23:28 -0800108 }
109 mNumFences = 0;
Jamie Gennis4b0eba92013-02-05 13:30:24 -0800110 mFrameRecords[mOffset].desiredPresentTime = INT64_MAX;
111 mFrameRecords[mOffset].frameReadyTime = INT64_MAX;
112 mFrameRecords[mOffset].actualPresentTime = INT64_MAX;
Jamie Gennis82dbc742012-11-08 19:23:28 -0800113}
114
Svetoslavd85084b2014-03-20 10:28:31 -0700115void FrameTracker::getStats(FrameStats* outStats) const {
116 Mutex::Autolock lock(mMutex);
117 processFencesLocked();
118
119 outStats->refreshPeriodNano = mDisplayPeriod;
120
121 const size_t offset = mOffset;
122 for (size_t i = 1; i < NUM_FRAME_RECORDS; i++) {
123 const size_t index = (offset + i) % NUM_FRAME_RECORDS;
124
125 // Skip frame records with no data (if buffer not yet full).
126 if (mFrameRecords[index].desiredPresentTime == 0) {
127 continue;
128 }
129
130 nsecs_t desiredPresentTimeNano = mFrameRecords[index].desiredPresentTime;
131 outStats->desiredPresentTimesNano.push_back(desiredPresentTimeNano);
132
133 nsecs_t actualPresentTimeNano = mFrameRecords[index].actualPresentTime;
134 outStats->actualPresentTimesNano.push_back(actualPresentTimeNano);
135
136 nsecs_t frameReadyTimeNano = mFrameRecords[index].frameReadyTime;
137 outStats->frameReadyTimesNano.push_back(frameReadyTimeNano);
138 }
139}
140
Jamie Gennis6547ff42013-07-16 20:12:42 -0700141void FrameTracker::logAndResetStats(const String8& name) {
142 Mutex::Autolock lock(mMutex);
143 logStatsLocked(name);
144 resetFrameCountersLocked();
145}
146
Jamie Gennis4b0eba92013-02-05 13:30:24 -0800147void FrameTracker::processFencesLocked() const {
Jamie Gennis82dbc742012-11-08 19:23:28 -0800148 FrameRecord* records = const_cast<FrameRecord*>(mFrameRecords);
149 int& numFences = const_cast<int&>(mNumFences);
150
151 for (int i = 1; i < NUM_FRAME_RECORDS && numFences > 0; i++) {
152 size_t idx = (mOffset+NUM_FRAME_RECORDS-i) % NUM_FRAME_RECORDS;
Jamie Gennis6547ff42013-07-16 20:12:42 -0700153 bool updated = false;
Jamie Gennis82dbc742012-11-08 19:23:28 -0800154
Brian Anderson3d4039d2016-09-23 16:31:30 -0700155 const std::shared_ptr<FenceTime>& rfence = records[idx].frameReadyFence;
Jamie Gennis82dbc742012-11-08 19:23:28 -0800156 if (rfence != NULL) {
157 records[idx].frameReadyTime = rfence->getSignalTime();
158 if (records[idx].frameReadyTime < INT64_MAX) {
159 records[idx].frameReadyFence = NULL;
160 numFences--;
Jamie Gennis6547ff42013-07-16 20:12:42 -0700161 updated = true;
Jamie Gennis82dbc742012-11-08 19:23:28 -0800162 }
163 }
164
Brian Anderson3d4039d2016-09-23 16:31:30 -0700165 const std::shared_ptr<FenceTime>& pfence =
166 records[idx].actualPresentFence;
Jamie Gennis82dbc742012-11-08 19:23:28 -0800167 if (pfence != NULL) {
168 records[idx].actualPresentTime = pfence->getSignalTime();
169 if (records[idx].actualPresentTime < INT64_MAX) {
170 records[idx].actualPresentFence = NULL;
171 numFences--;
Jamie Gennis6547ff42013-07-16 20:12:42 -0700172 updated = true;
Jamie Gennis82dbc742012-11-08 19:23:28 -0800173 }
174 }
Jamie Gennis6547ff42013-07-16 20:12:42 -0700175
176 if (updated) {
177 updateStatsLocked(idx);
178 }
Jamie Gennis82dbc742012-11-08 19:23:28 -0800179 }
180}
181
Jamie Gennis6547ff42013-07-16 20:12:42 -0700182void FrameTracker::updateStatsLocked(size_t newFrameIdx) const {
183 int* numFrames = const_cast<int*>(mNumFrames);
184
185 if (mDisplayPeriod > 0 && isFrameValidLocked(newFrameIdx)) {
186 size_t prevFrameIdx = (newFrameIdx+NUM_FRAME_RECORDS-1) %
187 NUM_FRAME_RECORDS;
188
189 if (isFrameValidLocked(prevFrameIdx)) {
190 nsecs_t newPresentTime =
191 mFrameRecords[newFrameIdx].actualPresentTime;
192 nsecs_t prevPresentTime =
193 mFrameRecords[prevFrameIdx].actualPresentTime;
194
195 nsecs_t duration = newPresentTime - prevPresentTime;
196 int numPeriods = int((duration + mDisplayPeriod/2) /
197 mDisplayPeriod);
198
199 for (int i = 0; i < NUM_FRAME_BUCKETS-1; i++) {
200 int nextBucket = 1 << (i+1);
201 if (numPeriods < nextBucket) {
202 numFrames[i]++;
203 return;
204 }
205 }
206
207 // The last duration bucket is a catch-all.
208 numFrames[NUM_FRAME_BUCKETS-1]++;
209 }
210 }
211}
212
213void FrameTracker::resetFrameCountersLocked() {
214 for (int i = 0; i < NUM_FRAME_BUCKETS; i++) {
215 mNumFrames[i] = 0;
216 }
217}
218
219void FrameTracker::logStatsLocked(const String8& name) const {
220 for (int i = 0; i < NUM_FRAME_BUCKETS; i++) {
221 if (mNumFrames[i] > 0) {
222 EventLog::logFrameDurations(name, mNumFrames, NUM_FRAME_BUCKETS);
223 return;
224 }
225 }
226}
227
228bool FrameTracker::isFrameValidLocked(size_t idx) const {
229 return mFrameRecords[idx].actualPresentTime > 0 &&
230 mFrameRecords[idx].actualPresentTime < INT64_MAX;
231}
232
Svetoslavd85084b2014-03-20 10:28:31 -0700233void FrameTracker::dumpStats(String8& result) const {
Jamie Gennis4b0eba92013-02-05 13:30:24 -0800234 Mutex::Autolock lock(mMutex);
235 processFencesLocked();
Jamie Gennis82dbc742012-11-08 19:23:28 -0800236
237 const size_t o = mOffset;
238 for (size_t i = 1; i < NUM_FRAME_RECORDS; i++) {
239 const size_t index = (o+i) % NUM_FRAME_RECORDS;
Greg Hackmann86efcc02014-03-07 12:44:02 -0800240 result.appendFormat("%" PRId64 "\t%" PRId64 "\t%" PRId64 "\n",
Jamie Gennis82dbc742012-11-08 19:23:28 -0800241 mFrameRecords[index].desiredPresentTime,
242 mFrameRecords[index].actualPresentTime,
243 mFrameRecords[index].frameReadyTime);
244 }
245 result.append("\n");
246}
247
248} // namespace android