xref: /aosp_15_r20/frameworks/native/services/surfaceflinger/FrameTracker.cpp (revision 38e8c45f13ce32b0dcecb25141ffecaf386fa17f)
1*38e8c45fSAndroid Build Coastguard Worker /*
2*38e8c45fSAndroid Build Coastguard Worker  * Copyright (C) 2012 The Android Open Source Project
3*38e8c45fSAndroid Build Coastguard Worker  *
4*38e8c45fSAndroid Build Coastguard Worker  * Licensed under the Apache License, Version 2.0 (the "License");
5*38e8c45fSAndroid Build Coastguard Worker  * you may not use this file except in compliance with the License.
6*38e8c45fSAndroid Build Coastguard Worker  * You may obtain a copy of the License at
7*38e8c45fSAndroid Build Coastguard Worker  *
8*38e8c45fSAndroid Build Coastguard Worker  *      http://www.apache.org/licenses/LICENSE-2.0
9*38e8c45fSAndroid Build Coastguard Worker  *
10*38e8c45fSAndroid Build Coastguard Worker  * Unless required by applicable law or agreed to in writing, software
11*38e8c45fSAndroid Build Coastguard Worker  * distributed under the License is distributed on an "AS IS" BASIS,
12*38e8c45fSAndroid Build Coastguard Worker  * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13*38e8c45fSAndroid Build Coastguard Worker  * See the License for the specific language governing permissions and
14*38e8c45fSAndroid Build Coastguard Worker  * limitations under the License.
15*38e8c45fSAndroid Build Coastguard Worker  */
16*38e8c45fSAndroid Build Coastguard Worker 
17*38e8c45fSAndroid Build Coastguard Worker // TODO(b/129481165): remove the #pragma below and fix conversion issues
18*38e8c45fSAndroid Build Coastguard Worker #pragma clang diagnostic push
19*38e8c45fSAndroid Build Coastguard Worker #pragma clang diagnostic ignored "-Wconversion"
20*38e8c45fSAndroid Build Coastguard Worker 
21*38e8c45fSAndroid Build Coastguard Worker #include <inttypes.h>
22*38e8c45fSAndroid Build Coastguard Worker 
23*38e8c45fSAndroid Build Coastguard Worker #include <android-base/stringprintf.h>
24*38e8c45fSAndroid Build Coastguard Worker #include <android/log.h>
25*38e8c45fSAndroid Build Coastguard Worker 
26*38e8c45fSAndroid Build Coastguard Worker #include <ui/FrameStats.h>
27*38e8c45fSAndroid Build Coastguard Worker 
28*38e8c45fSAndroid Build Coastguard Worker #include "FrameTracker.h"
29*38e8c45fSAndroid Build Coastguard Worker 
30*38e8c45fSAndroid Build Coastguard Worker namespace android {
31*38e8c45fSAndroid Build Coastguard Worker 
FrameTracker()32*38e8c45fSAndroid Build Coastguard Worker FrameTracker::FrameTracker() : mOffset(0), mNumFences(0), mDisplayPeriod(0) {}
33*38e8c45fSAndroid Build Coastguard Worker 
setDesiredPresentTime(nsecs_t presentTime)34*38e8c45fSAndroid Build Coastguard Worker void FrameTracker::setDesiredPresentTime(nsecs_t presentTime) {
35*38e8c45fSAndroid Build Coastguard Worker     Mutex::Autolock lock(mMutex);
36*38e8c45fSAndroid Build Coastguard Worker     mFrameRecords[mOffset].desiredPresentTime = presentTime;
37*38e8c45fSAndroid Build Coastguard Worker }
38*38e8c45fSAndroid Build Coastguard Worker 
setFrameReadyTime(nsecs_t readyTime)39*38e8c45fSAndroid Build Coastguard Worker void FrameTracker::setFrameReadyTime(nsecs_t readyTime) {
40*38e8c45fSAndroid Build Coastguard Worker     Mutex::Autolock lock(mMutex);
41*38e8c45fSAndroid Build Coastguard Worker     mFrameRecords[mOffset].frameReadyTime = readyTime;
42*38e8c45fSAndroid Build Coastguard Worker }
43*38e8c45fSAndroid Build Coastguard Worker 
setFrameReadyFence(std::shared_ptr<FenceTime> && readyFence)44*38e8c45fSAndroid Build Coastguard Worker void FrameTracker::setFrameReadyFence(
45*38e8c45fSAndroid Build Coastguard Worker         std::shared_ptr<FenceTime>&& readyFence) {
46*38e8c45fSAndroid Build Coastguard Worker     Mutex::Autolock lock(mMutex);
47*38e8c45fSAndroid Build Coastguard Worker     mFrameRecords[mOffset].frameReadyFence = std::move(readyFence);
48*38e8c45fSAndroid Build Coastguard Worker     mNumFences++;
49*38e8c45fSAndroid Build Coastguard Worker }
50*38e8c45fSAndroid Build Coastguard Worker 
setActualPresentTime(nsecs_t presentTime)51*38e8c45fSAndroid Build Coastguard Worker void FrameTracker::setActualPresentTime(nsecs_t presentTime) {
52*38e8c45fSAndroid Build Coastguard Worker     Mutex::Autolock lock(mMutex);
53*38e8c45fSAndroid Build Coastguard Worker     mFrameRecords[mOffset].actualPresentTime = presentTime;
54*38e8c45fSAndroid Build Coastguard Worker }
55*38e8c45fSAndroid Build Coastguard Worker 
setActualPresentFence(const std::shared_ptr<FenceTime> & readyFence)56*38e8c45fSAndroid Build Coastguard Worker void FrameTracker::setActualPresentFence(const std::shared_ptr<FenceTime>& readyFence) {
57*38e8c45fSAndroid Build Coastguard Worker     Mutex::Autolock lock(mMutex);
58*38e8c45fSAndroid Build Coastguard Worker     mFrameRecords[mOffset].actualPresentFence = readyFence;
59*38e8c45fSAndroid Build Coastguard Worker     mNumFences++;
60*38e8c45fSAndroid Build Coastguard Worker }
61*38e8c45fSAndroid Build Coastguard Worker 
setDisplayRefreshPeriod(nsecs_t displayPeriod)62*38e8c45fSAndroid Build Coastguard Worker void FrameTracker::setDisplayRefreshPeriod(nsecs_t displayPeriod) {
63*38e8c45fSAndroid Build Coastguard Worker     Mutex::Autolock lock(mMutex);
64*38e8c45fSAndroid Build Coastguard Worker     mDisplayPeriod = displayPeriod;
65*38e8c45fSAndroid Build Coastguard Worker }
66*38e8c45fSAndroid Build Coastguard Worker 
advanceFrame()67*38e8c45fSAndroid Build Coastguard Worker void FrameTracker::advanceFrame() {
68*38e8c45fSAndroid Build Coastguard Worker     Mutex::Autolock lock(mMutex);
69*38e8c45fSAndroid Build Coastguard Worker 
70*38e8c45fSAndroid Build Coastguard Worker     // Advance to the next frame.
71*38e8c45fSAndroid Build Coastguard Worker     mOffset = (mOffset+1) % NUM_FRAME_RECORDS;
72*38e8c45fSAndroid Build Coastguard Worker     mFrameRecords[mOffset].desiredPresentTime = INT64_MAX;
73*38e8c45fSAndroid Build Coastguard Worker     mFrameRecords[mOffset].frameReadyTime = INT64_MAX;
74*38e8c45fSAndroid Build Coastguard Worker     mFrameRecords[mOffset].actualPresentTime = INT64_MAX;
75*38e8c45fSAndroid Build Coastguard Worker 
76*38e8c45fSAndroid Build Coastguard Worker     if (mFrameRecords[mOffset].frameReadyFence != nullptr) {
77*38e8c45fSAndroid Build Coastguard Worker         // We're clobbering an unsignaled fence, so we need to decrement the
78*38e8c45fSAndroid Build Coastguard Worker         // fence count.
79*38e8c45fSAndroid Build Coastguard Worker         mFrameRecords[mOffset].frameReadyFence = nullptr;
80*38e8c45fSAndroid Build Coastguard Worker         mNumFences--;
81*38e8c45fSAndroid Build Coastguard Worker     }
82*38e8c45fSAndroid Build Coastguard Worker 
83*38e8c45fSAndroid Build Coastguard Worker     if (mFrameRecords[mOffset].actualPresentFence != nullptr) {
84*38e8c45fSAndroid Build Coastguard Worker         // We're clobbering an unsignaled fence, so we need to decrement the
85*38e8c45fSAndroid Build Coastguard Worker         // fence count.
86*38e8c45fSAndroid Build Coastguard Worker         mFrameRecords[mOffset].actualPresentFence = nullptr;
87*38e8c45fSAndroid Build Coastguard Worker         mNumFences--;
88*38e8c45fSAndroid Build Coastguard Worker     }
89*38e8c45fSAndroid Build Coastguard Worker }
90*38e8c45fSAndroid Build Coastguard Worker 
clearStats()91*38e8c45fSAndroid Build Coastguard Worker void FrameTracker::clearStats() {
92*38e8c45fSAndroid Build Coastguard Worker     Mutex::Autolock lock(mMutex);
93*38e8c45fSAndroid Build Coastguard Worker     for (size_t i = 0; i < NUM_FRAME_RECORDS; i++) {
94*38e8c45fSAndroid Build Coastguard Worker         mFrameRecords[i].desiredPresentTime = 0;
95*38e8c45fSAndroid Build Coastguard Worker         mFrameRecords[i].frameReadyTime = 0;
96*38e8c45fSAndroid Build Coastguard Worker         mFrameRecords[i].actualPresentTime = 0;
97*38e8c45fSAndroid Build Coastguard Worker         mFrameRecords[i].frameReadyFence.reset();
98*38e8c45fSAndroid Build Coastguard Worker         mFrameRecords[i].actualPresentFence.reset();
99*38e8c45fSAndroid Build Coastguard Worker     }
100*38e8c45fSAndroid Build Coastguard Worker     mNumFences = 0;
101*38e8c45fSAndroid Build Coastguard Worker     mFrameRecords[mOffset].desiredPresentTime = INT64_MAX;
102*38e8c45fSAndroid Build Coastguard Worker     mFrameRecords[mOffset].frameReadyTime = INT64_MAX;
103*38e8c45fSAndroid Build Coastguard Worker     mFrameRecords[mOffset].actualPresentTime = INT64_MAX;
104*38e8c45fSAndroid Build Coastguard Worker }
105*38e8c45fSAndroid Build Coastguard Worker 
getStats(FrameStats * outStats) const106*38e8c45fSAndroid Build Coastguard Worker void FrameTracker::getStats(FrameStats* outStats) const {
107*38e8c45fSAndroid Build Coastguard Worker     Mutex::Autolock lock(mMutex);
108*38e8c45fSAndroid Build Coastguard Worker     processFencesLocked();
109*38e8c45fSAndroid Build Coastguard Worker 
110*38e8c45fSAndroid Build Coastguard Worker     outStats->refreshPeriodNano = mDisplayPeriod;
111*38e8c45fSAndroid Build Coastguard Worker 
112*38e8c45fSAndroid Build Coastguard Worker     const size_t offset = mOffset;
113*38e8c45fSAndroid Build Coastguard Worker     for (size_t i = 1; i < NUM_FRAME_RECORDS; i++) {
114*38e8c45fSAndroid Build Coastguard Worker         const size_t index = (offset + i) % NUM_FRAME_RECORDS;
115*38e8c45fSAndroid Build Coastguard Worker 
116*38e8c45fSAndroid Build Coastguard Worker         // Skip frame records with no data (if buffer not yet full).
117*38e8c45fSAndroid Build Coastguard Worker         if (mFrameRecords[index].desiredPresentTime == 0) {
118*38e8c45fSAndroid Build Coastguard Worker             continue;
119*38e8c45fSAndroid Build Coastguard Worker         }
120*38e8c45fSAndroid Build Coastguard Worker 
121*38e8c45fSAndroid Build Coastguard Worker         nsecs_t desiredPresentTimeNano = mFrameRecords[index].desiredPresentTime;
122*38e8c45fSAndroid Build Coastguard Worker         outStats->desiredPresentTimesNano.push_back(desiredPresentTimeNano);
123*38e8c45fSAndroid Build Coastguard Worker 
124*38e8c45fSAndroid Build Coastguard Worker         nsecs_t actualPresentTimeNano = mFrameRecords[index].actualPresentTime;
125*38e8c45fSAndroid Build Coastguard Worker         outStats->actualPresentTimesNano.push_back(actualPresentTimeNano);
126*38e8c45fSAndroid Build Coastguard Worker 
127*38e8c45fSAndroid Build Coastguard Worker         nsecs_t frameReadyTimeNano = mFrameRecords[index].frameReadyTime;
128*38e8c45fSAndroid Build Coastguard Worker         outStats->frameReadyTimesNano.push_back(frameReadyTimeNano);
129*38e8c45fSAndroid Build Coastguard Worker     }
130*38e8c45fSAndroid Build Coastguard Worker }
131*38e8c45fSAndroid Build Coastguard Worker 
processFencesLocked() const132*38e8c45fSAndroid Build Coastguard Worker void FrameTracker::processFencesLocked() const {
133*38e8c45fSAndroid Build Coastguard Worker     FrameRecord* records = const_cast<FrameRecord*>(mFrameRecords);
134*38e8c45fSAndroid Build Coastguard Worker     int& numFences = const_cast<int&>(mNumFences);
135*38e8c45fSAndroid Build Coastguard Worker 
136*38e8c45fSAndroid Build Coastguard Worker     for (int i = 1; i < NUM_FRAME_RECORDS && numFences > 0; i++) {
137*38e8c45fSAndroid Build Coastguard Worker         size_t idx = (mOffset + NUM_FRAME_RECORDS - i) % NUM_FRAME_RECORDS;
138*38e8c45fSAndroid Build Coastguard Worker 
139*38e8c45fSAndroid Build Coastguard Worker         const std::shared_ptr<FenceTime>& rfence = records[idx].frameReadyFence;
140*38e8c45fSAndroid Build Coastguard Worker         if (rfence != nullptr) {
141*38e8c45fSAndroid Build Coastguard Worker             records[idx].frameReadyTime = rfence->getSignalTime();
142*38e8c45fSAndroid Build Coastguard Worker             if (records[idx].frameReadyTime < INT64_MAX) {
143*38e8c45fSAndroid Build Coastguard Worker                 records[idx].frameReadyFence = nullptr;
144*38e8c45fSAndroid Build Coastguard Worker                 numFences--;
145*38e8c45fSAndroid Build Coastguard Worker             }
146*38e8c45fSAndroid Build Coastguard Worker         }
147*38e8c45fSAndroid Build Coastguard Worker 
148*38e8c45fSAndroid Build Coastguard Worker         const std::shared_ptr<FenceTime>& pfence =
149*38e8c45fSAndroid Build Coastguard Worker                 records[idx].actualPresentFence;
150*38e8c45fSAndroid Build Coastguard Worker         if (pfence != nullptr) {
151*38e8c45fSAndroid Build Coastguard Worker             records[idx].actualPresentTime = pfence->getSignalTime();
152*38e8c45fSAndroid Build Coastguard Worker             if (records[idx].actualPresentTime < INT64_MAX) {
153*38e8c45fSAndroid Build Coastguard Worker                 records[idx].actualPresentFence = nullptr;
154*38e8c45fSAndroid Build Coastguard Worker                 numFences--;
155*38e8c45fSAndroid Build Coastguard Worker             }
156*38e8c45fSAndroid Build Coastguard Worker         }
157*38e8c45fSAndroid Build Coastguard Worker     }
158*38e8c45fSAndroid Build Coastguard Worker }
159*38e8c45fSAndroid Build Coastguard Worker 
isFrameValidLocked(size_t idx) const160*38e8c45fSAndroid Build Coastguard Worker bool FrameTracker::isFrameValidLocked(size_t idx) const {
161*38e8c45fSAndroid Build Coastguard Worker     return mFrameRecords[idx].actualPresentTime > 0 &&
162*38e8c45fSAndroid Build Coastguard Worker             mFrameRecords[idx].actualPresentTime < INT64_MAX;
163*38e8c45fSAndroid Build Coastguard Worker }
164*38e8c45fSAndroid Build Coastguard Worker 
dumpStats(std::string & result) const165*38e8c45fSAndroid Build Coastguard Worker void FrameTracker::dumpStats(std::string& result) const {
166*38e8c45fSAndroid Build Coastguard Worker     Mutex::Autolock lock(mMutex);
167*38e8c45fSAndroid Build Coastguard Worker     processFencesLocked();
168*38e8c45fSAndroid Build Coastguard Worker 
169*38e8c45fSAndroid Build Coastguard Worker     const size_t o = mOffset;
170*38e8c45fSAndroid Build Coastguard Worker     for (size_t i = 1; i < NUM_FRAME_RECORDS; i++) {
171*38e8c45fSAndroid Build Coastguard Worker         const size_t index = (o+i) % NUM_FRAME_RECORDS;
172*38e8c45fSAndroid Build Coastguard Worker         base::StringAppendF(&result, "%" PRId64 "\t%" PRId64 "\t%" PRId64 "\n",
173*38e8c45fSAndroid Build Coastguard Worker                             mFrameRecords[index].desiredPresentTime,
174*38e8c45fSAndroid Build Coastguard Worker                             mFrameRecords[index].actualPresentTime,
175*38e8c45fSAndroid Build Coastguard Worker                             mFrameRecords[index].frameReadyTime);
176*38e8c45fSAndroid Build Coastguard Worker     }
177*38e8c45fSAndroid Build Coastguard Worker     result.append("\n");
178*38e8c45fSAndroid Build Coastguard Worker }
179*38e8c45fSAndroid Build Coastguard Worker 
180*38e8c45fSAndroid Build Coastguard Worker } // namespace android
181*38e8c45fSAndroid Build Coastguard Worker 
182*38e8c45fSAndroid Build Coastguard Worker // TODO(b/129481165): remove the #pragma below and fix conversion issues
183*38e8c45fSAndroid Build Coastguard Worker #pragma clang diagnostic pop // ignored "-Wconversion"
184