blob: 93d0313e535f4219807dd75092a214e8be078406 [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
Ady Abrahamb0dbdaa2020-01-06 16:19:42 -080017// TODO(b/129481165): remove the #pragma below and fix conversion issues
18#pragma clang diagnostic push
19#pragma clang diagnostic ignored "-Wconversion"
20
Greg Hackmann86efcc02014-03-07 12:44:02 -080021#include <inttypes.h>
22
Yiwei Zhang5434a782018-12-05 18:06:32 -080023#include <android-base/stringprintf.h>
Mark Salyzyna5e161b2016-09-29 08:08:05 -070024#include <android/log.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"
29
30namespace android {
31
Ben Widawsky31135462024-10-08 11:31:20 -070032FrameTracker::FrameTracker() : mOffset(0), mNumFences(0), mDisplayPeriod(0) {}
Jamie Gennis82dbc742012-11-08 19:23:28 -080033
34void FrameTracker::setDesiredPresentTime(nsecs_t presentTime) {
Jamie Gennis4b0eba92013-02-05 13:30:24 -080035 Mutex::Autolock lock(mMutex);
Jamie Gennis82dbc742012-11-08 19:23:28 -080036 mFrameRecords[mOffset].desiredPresentTime = presentTime;
37}
38
39void FrameTracker::setFrameReadyTime(nsecs_t readyTime) {
Jamie Gennis4b0eba92013-02-05 13:30:24 -080040 Mutex::Autolock lock(mMutex);
Jamie Gennis82dbc742012-11-08 19:23:28 -080041 mFrameRecords[mOffset].frameReadyTime = readyTime;
42}
43
Brian Anderson3d4039d2016-09-23 16:31:30 -070044void FrameTracker::setFrameReadyFence(
45 std::shared_ptr<FenceTime>&& readyFence) {
Jamie Gennis4b0eba92013-02-05 13:30:24 -080046 Mutex::Autolock lock(mMutex);
Brian Anderson3d4039d2016-09-23 16:31:30 -070047 mFrameRecords[mOffset].frameReadyFence = std::move(readyFence);
Jamie Gennis82dbc742012-11-08 19:23:28 -080048 mNumFences++;
49}
50
51void FrameTracker::setActualPresentTime(nsecs_t presentTime) {
Jamie Gennis4b0eba92013-02-05 13:30:24 -080052 Mutex::Autolock lock(mMutex);
Jamie Gennis82dbc742012-11-08 19:23:28 -080053 mFrameRecords[mOffset].actualPresentTime = presentTime;
54}
55
Ady Abraham6c1b7ac2021-03-31 16:56:03 -070056void FrameTracker::setActualPresentFence(const std::shared_ptr<FenceTime>& readyFence) {
Jamie Gennis4b0eba92013-02-05 13:30:24 -080057 Mutex::Autolock lock(mMutex);
Ady Abraham6c1b7ac2021-03-31 16:56:03 -070058 mFrameRecords[mOffset].actualPresentFence = readyFence;
Jamie Gennis82dbc742012-11-08 19:23:28 -080059 mNumFences++;
60}
61
Jamie Gennis6547ff42013-07-16 20:12:42 -070062void FrameTracker::setDisplayRefreshPeriod(nsecs_t displayPeriod) {
63 Mutex::Autolock lock(mMutex);
64 mDisplayPeriod = displayPeriod;
65}
66
Jamie Gennis82dbc742012-11-08 19:23:28 -080067void FrameTracker::advanceFrame() {
Jamie Gennis4b0eba92013-02-05 13:30:24 -080068 Mutex::Autolock lock(mMutex);
Jamie Gennis6547ff42013-07-16 20:12:42 -070069
Jamie Gennis6547ff42013-07-16 20:12:42 -070070 // Advance to the next frame.
Jamie Gennis82dbc742012-11-08 19:23:28 -080071 mOffset = (mOffset+1) % NUM_FRAME_RECORDS;
72 mFrameRecords[mOffset].desiredPresentTime = INT64_MAX;
73 mFrameRecords[mOffset].frameReadyTime = INT64_MAX;
74 mFrameRecords[mOffset].actualPresentTime = INT64_MAX;
75
Peiyong Lin566a3b42018-01-09 18:22:43 -080076 if (mFrameRecords[mOffset].frameReadyFence != nullptr) {
Jamie Gennis82dbc742012-11-08 19:23:28 -080077 // We're clobbering an unsignaled fence, so we need to decrement the
78 // fence count.
Peiyong Lin566a3b42018-01-09 18:22:43 -080079 mFrameRecords[mOffset].frameReadyFence = nullptr;
Jamie Gennis82dbc742012-11-08 19:23:28 -080080 mNumFences--;
81 }
82
Peiyong Lin566a3b42018-01-09 18:22:43 -080083 if (mFrameRecords[mOffset].actualPresentFence != nullptr) {
Jamie Gennis82dbc742012-11-08 19:23:28 -080084 // We're clobbering an unsignaled fence, so we need to decrement the
85 // fence count.
Peiyong Lin566a3b42018-01-09 18:22:43 -080086 mFrameRecords[mOffset].actualPresentFence = nullptr;
Jamie Gennis82dbc742012-11-08 19:23:28 -080087 mNumFences--;
88 }
Jamie Gennis82dbc742012-11-08 19:23:28 -080089}
90
Svetoslavd85084b2014-03-20 10:28:31 -070091void FrameTracker::clearStats() {
Jamie Gennis4b0eba92013-02-05 13:30:24 -080092 Mutex::Autolock lock(mMutex);
Jamie Gennis82dbc742012-11-08 19:23:28 -080093 for (size_t i = 0; i < NUM_FRAME_RECORDS; i++) {
94 mFrameRecords[i].desiredPresentTime = 0;
95 mFrameRecords[i].frameReadyTime = 0;
96 mFrameRecords[i].actualPresentTime = 0;
Brian Anderson3d4039d2016-09-23 16:31:30 -070097 mFrameRecords[i].frameReadyFence.reset();
98 mFrameRecords[i].actualPresentFence.reset();
Jamie Gennis82dbc742012-11-08 19:23:28 -080099 }
100 mNumFences = 0;
Jamie Gennis4b0eba92013-02-05 13:30:24 -0800101 mFrameRecords[mOffset].desiredPresentTime = INT64_MAX;
102 mFrameRecords[mOffset].frameReadyTime = INT64_MAX;
103 mFrameRecords[mOffset].actualPresentTime = INT64_MAX;
Jamie Gennis82dbc742012-11-08 19:23:28 -0800104}
105
Svetoslavd85084b2014-03-20 10:28:31 -0700106void FrameTracker::getStats(FrameStats* outStats) const {
107 Mutex::Autolock lock(mMutex);
108 processFencesLocked();
109
110 outStats->refreshPeriodNano = mDisplayPeriod;
111
112 const size_t offset = mOffset;
113 for (size_t i = 1; i < NUM_FRAME_RECORDS; i++) {
114 const size_t index = (offset + i) % NUM_FRAME_RECORDS;
115
116 // Skip frame records with no data (if buffer not yet full).
117 if (mFrameRecords[index].desiredPresentTime == 0) {
118 continue;
119 }
120
121 nsecs_t desiredPresentTimeNano = mFrameRecords[index].desiredPresentTime;
122 outStats->desiredPresentTimesNano.push_back(desiredPresentTimeNano);
123
124 nsecs_t actualPresentTimeNano = mFrameRecords[index].actualPresentTime;
125 outStats->actualPresentTimesNano.push_back(actualPresentTimeNano);
126
127 nsecs_t frameReadyTimeNano = mFrameRecords[index].frameReadyTime;
128 outStats->frameReadyTimesNano.push_back(frameReadyTimeNano);
129 }
130}
131
Jamie Gennis4b0eba92013-02-05 13:30:24 -0800132void FrameTracker::processFencesLocked() const {
Jamie Gennis82dbc742012-11-08 19:23:28 -0800133 FrameRecord* records = const_cast<FrameRecord*>(mFrameRecords);
134 int& numFences = const_cast<int&>(mNumFences);
135
136 for (int i = 1; i < NUM_FRAME_RECORDS && numFences > 0; i++) {
Ben Widawsky31135462024-10-08 11:31:20 -0700137 size_t idx = (mOffset + NUM_FRAME_RECORDS - i) % NUM_FRAME_RECORDS;
Jamie Gennis82dbc742012-11-08 19:23:28 -0800138
Brian Anderson3d4039d2016-09-23 16:31:30 -0700139 const std::shared_ptr<FenceTime>& rfence = records[idx].frameReadyFence;
Peiyong Lin566a3b42018-01-09 18:22:43 -0800140 if (rfence != nullptr) {
Jamie Gennis82dbc742012-11-08 19:23:28 -0800141 records[idx].frameReadyTime = rfence->getSignalTime();
142 if (records[idx].frameReadyTime < INT64_MAX) {
Peiyong Lin566a3b42018-01-09 18:22:43 -0800143 records[idx].frameReadyFence = nullptr;
Jamie Gennis82dbc742012-11-08 19:23:28 -0800144 numFences--;
145 }
146 }
147
Brian Anderson3d4039d2016-09-23 16:31:30 -0700148 const std::shared_ptr<FenceTime>& pfence =
149 records[idx].actualPresentFence;
Peiyong Lin566a3b42018-01-09 18:22:43 -0800150 if (pfence != nullptr) {
Jamie Gennis82dbc742012-11-08 19:23:28 -0800151 records[idx].actualPresentTime = pfence->getSignalTime();
152 if (records[idx].actualPresentTime < INT64_MAX) {
Peiyong Lin566a3b42018-01-09 18:22:43 -0800153 records[idx].actualPresentFence = nullptr;
Jamie Gennis82dbc742012-11-08 19:23:28 -0800154 numFences--;
155 }
156 }
Jamie Gennis6547ff42013-07-16 20:12:42 -0700157 }
158}
159
160bool FrameTracker::isFrameValidLocked(size_t idx) const {
161 return mFrameRecords[idx].actualPresentTime > 0 &&
162 mFrameRecords[idx].actualPresentTime < INT64_MAX;
163}
164
Yiwei Zhang5434a782018-12-05 18:06:32 -0800165void FrameTracker::dumpStats(std::string& result) const {
Jamie Gennis4b0eba92013-02-05 13:30:24 -0800166 Mutex::Autolock lock(mMutex);
167 processFencesLocked();
Jamie Gennis82dbc742012-11-08 19:23:28 -0800168
169 const size_t o = mOffset;
170 for (size_t i = 1; i < NUM_FRAME_RECORDS; i++) {
171 const size_t index = (o+i) % NUM_FRAME_RECORDS;
Yiwei Zhang5434a782018-12-05 18:06:32 -0800172 base::StringAppendF(&result, "%" PRId64 "\t%" PRId64 "\t%" PRId64 "\n",
173 mFrameRecords[index].desiredPresentTime,
174 mFrameRecords[index].actualPresentTime,
175 mFrameRecords[index].frameReadyTime);
Jamie Gennis82dbc742012-11-08 19:23:28 -0800176 }
177 result.append("\n");
178}
179
180} // namespace android
Ady Abrahamb0dbdaa2020-01-06 16:19:42 -0800181
182// TODO(b/129481165): remove the #pragma below and fix conversion issues
183#pragma clang diagnostic pop // ignored "-Wconversion"