| Jamie Gennis | 82dbc74 | 2012-11-08 19:23:28 -0800 | [diff] [blame] | 1 | /* | 
 | 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 Abraham | b0dbdaa | 2020-01-06 16:19:42 -0800 | [diff] [blame] | 17 | // TODO(b/129481165): remove the #pragma below and fix conversion issues | 
 | 18 | #pragma clang diagnostic push | 
 | 19 | #pragma clang diagnostic ignored "-Wconversion" | 
 | 20 |  | 
| Jamie Gennis | 82dbc74 | 2012-11-08 19:23:28 -0800 | [diff] [blame] | 21 | // This is needed for stdint.h to define INT64_MAX in C++ | 
 | 22 | #define __STDC_LIMIT_MACROS | 
 | 23 |  | 
| Greg Hackmann | 86efcc0 | 2014-03-07 12:44:02 -0800 | [diff] [blame] | 24 | #include <inttypes.h> | 
 | 25 |  | 
| Yiwei Zhang | 5434a78 | 2018-12-05 18:06:32 -0800 | [diff] [blame] | 26 | #include <android-base/stringprintf.h> | 
| Mark Salyzyn | a5e161b | 2016-09-29 08:08:05 -0700 | [diff] [blame] | 27 | #include <android/log.h> | 
| Jamie Gennis | 6547ff4 | 2013-07-16 20:12:42 -0700 | [diff] [blame] | 28 |  | 
| Svetoslav | d85084b | 2014-03-20 10:28:31 -0700 | [diff] [blame] | 29 | #include <ui/FrameStats.h> | 
| Jamie Gennis | 82dbc74 | 2012-11-08 19:23:28 -0800 | [diff] [blame] | 30 |  | 
| Jamie Gennis | 82dbc74 | 2012-11-08 19:23:28 -0800 | [diff] [blame] | 31 | #include "FrameTracker.h" | 
| Jamie Gennis | 6547ff4 | 2013-07-16 20:12:42 -0700 | [diff] [blame] | 32 | #include "EventLog/EventLog.h" | 
| Jamie Gennis | 82dbc74 | 2012-11-08 19:23:28 -0800 | [diff] [blame] | 33 |  | 
 | 34 | namespace android { | 
 | 35 |  | 
 | 36 | FrameTracker::FrameTracker() : | 
 | 37 |         mOffset(0), | 
| Jamie Gennis | 6547ff4 | 2013-07-16 20:12:42 -0700 | [diff] [blame] | 38 |         mNumFences(0), | 
 | 39 |         mDisplayPeriod(0) { | 
 | 40 |     resetFrameCountersLocked(); | 
| Jamie Gennis | 82dbc74 | 2012-11-08 19:23:28 -0800 | [diff] [blame] | 41 | } | 
 | 42 |  | 
 | 43 | void FrameTracker::setDesiredPresentTime(nsecs_t presentTime) { | 
| Jamie Gennis | 4b0eba9 | 2013-02-05 13:30:24 -0800 | [diff] [blame] | 44 |     Mutex::Autolock lock(mMutex); | 
| Jamie Gennis | 82dbc74 | 2012-11-08 19:23:28 -0800 | [diff] [blame] | 45 |     mFrameRecords[mOffset].desiredPresentTime = presentTime; | 
 | 46 | } | 
 | 47 |  | 
 | 48 | void FrameTracker::setFrameReadyTime(nsecs_t readyTime) { | 
| Jamie Gennis | 4b0eba9 | 2013-02-05 13:30:24 -0800 | [diff] [blame] | 49 |     Mutex::Autolock lock(mMutex); | 
| Jamie Gennis | 82dbc74 | 2012-11-08 19:23:28 -0800 | [diff] [blame] | 50 |     mFrameRecords[mOffset].frameReadyTime = readyTime; | 
 | 51 | } | 
 | 52 |  | 
| Brian Anderson | 3d4039d | 2016-09-23 16:31:30 -0700 | [diff] [blame] | 53 | void FrameTracker::setFrameReadyFence( | 
 | 54 |         std::shared_ptr<FenceTime>&& readyFence) { | 
| Jamie Gennis | 4b0eba9 | 2013-02-05 13:30:24 -0800 | [diff] [blame] | 55 |     Mutex::Autolock lock(mMutex); | 
| Brian Anderson | 3d4039d | 2016-09-23 16:31:30 -0700 | [diff] [blame] | 56 |     mFrameRecords[mOffset].frameReadyFence = std::move(readyFence); | 
| Jamie Gennis | 82dbc74 | 2012-11-08 19:23:28 -0800 | [diff] [blame] | 57 |     mNumFences++; | 
 | 58 | } | 
 | 59 |  | 
 | 60 | void FrameTracker::setActualPresentTime(nsecs_t presentTime) { | 
| Jamie Gennis | 4b0eba9 | 2013-02-05 13:30:24 -0800 | [diff] [blame] | 61 |     Mutex::Autolock lock(mMutex); | 
| Jamie Gennis | 82dbc74 | 2012-11-08 19:23:28 -0800 | [diff] [blame] | 62 |     mFrameRecords[mOffset].actualPresentTime = presentTime; | 
 | 63 | } | 
 | 64 |  | 
| Brian Anderson | 3d4039d | 2016-09-23 16:31:30 -0700 | [diff] [blame] | 65 | void FrameTracker::setActualPresentFence( | 
 | 66 |         std::shared_ptr<FenceTime>&& readyFence) { | 
| Jamie Gennis | 4b0eba9 | 2013-02-05 13:30:24 -0800 | [diff] [blame] | 67 |     Mutex::Autolock lock(mMutex); | 
| Brian Anderson | 3d4039d | 2016-09-23 16:31:30 -0700 | [diff] [blame] | 68 |     mFrameRecords[mOffset].actualPresentFence = std::move(readyFence); | 
| Jamie Gennis | 82dbc74 | 2012-11-08 19:23:28 -0800 | [diff] [blame] | 69 |     mNumFences++; | 
 | 70 | } | 
 | 71 |  | 
| Jamie Gennis | 6547ff4 | 2013-07-16 20:12:42 -0700 | [diff] [blame] | 72 | void FrameTracker::setDisplayRefreshPeriod(nsecs_t displayPeriod) { | 
 | 73 |     Mutex::Autolock lock(mMutex); | 
 | 74 |     mDisplayPeriod = displayPeriod; | 
 | 75 | } | 
 | 76 |  | 
| Jamie Gennis | 82dbc74 | 2012-11-08 19:23:28 -0800 | [diff] [blame] | 77 | void FrameTracker::advanceFrame() { | 
| Jamie Gennis | 4b0eba9 | 2013-02-05 13:30:24 -0800 | [diff] [blame] | 78 |     Mutex::Autolock lock(mMutex); | 
| Jamie Gennis | 6547ff4 | 2013-07-16 20:12:42 -0700 | [diff] [blame] | 79 |  | 
 | 80 |     // Update the statistic to include the frame we just finished. | 
 | 81 |     updateStatsLocked(mOffset); | 
 | 82 |  | 
 | 83 |     // Advance to the next frame. | 
| Jamie Gennis | 82dbc74 | 2012-11-08 19:23:28 -0800 | [diff] [blame] | 84 |     mOffset = (mOffset+1) % NUM_FRAME_RECORDS; | 
 | 85 |     mFrameRecords[mOffset].desiredPresentTime = INT64_MAX; | 
 | 86 |     mFrameRecords[mOffset].frameReadyTime = INT64_MAX; | 
 | 87 |     mFrameRecords[mOffset].actualPresentTime = INT64_MAX; | 
 | 88 |  | 
| Peiyong Lin | 566a3b4 | 2018-01-09 18:22:43 -0800 | [diff] [blame] | 89 |     if (mFrameRecords[mOffset].frameReadyFence != nullptr) { | 
| Jamie Gennis | 82dbc74 | 2012-11-08 19:23:28 -0800 | [diff] [blame] | 90 |         // We're clobbering an unsignaled fence, so we need to decrement the | 
 | 91 |         // fence count. | 
| Peiyong Lin | 566a3b4 | 2018-01-09 18:22:43 -0800 | [diff] [blame] | 92 |         mFrameRecords[mOffset].frameReadyFence = nullptr; | 
| Jamie Gennis | 82dbc74 | 2012-11-08 19:23:28 -0800 | [diff] [blame] | 93 |         mNumFences--; | 
 | 94 |     } | 
 | 95 |  | 
| Peiyong Lin | 566a3b4 | 2018-01-09 18:22:43 -0800 | [diff] [blame] | 96 |     if (mFrameRecords[mOffset].actualPresentFence != nullptr) { | 
| Jamie Gennis | 82dbc74 | 2012-11-08 19:23:28 -0800 | [diff] [blame] | 97 |         // We're clobbering an unsignaled fence, so we need to decrement the | 
 | 98 |         // fence count. | 
| Peiyong Lin | 566a3b4 | 2018-01-09 18:22:43 -0800 | [diff] [blame] | 99 |         mFrameRecords[mOffset].actualPresentFence = nullptr; | 
| Jamie Gennis | 82dbc74 | 2012-11-08 19:23:28 -0800 | [diff] [blame] | 100 |         mNumFences--; | 
 | 101 |     } | 
| Jamie Gennis | 82dbc74 | 2012-11-08 19:23:28 -0800 | [diff] [blame] | 102 | } | 
 | 103 |  | 
| Svetoslav | d85084b | 2014-03-20 10:28:31 -0700 | [diff] [blame] | 104 | void FrameTracker::clearStats() { | 
| Jamie Gennis | 4b0eba9 | 2013-02-05 13:30:24 -0800 | [diff] [blame] | 105 |     Mutex::Autolock lock(mMutex); | 
| Jamie Gennis | 82dbc74 | 2012-11-08 19:23:28 -0800 | [diff] [blame] | 106 |     for (size_t i = 0; i < NUM_FRAME_RECORDS; i++) { | 
 | 107 |         mFrameRecords[i].desiredPresentTime = 0; | 
 | 108 |         mFrameRecords[i].frameReadyTime = 0; | 
 | 109 |         mFrameRecords[i].actualPresentTime = 0; | 
| Brian Anderson | 3d4039d | 2016-09-23 16:31:30 -0700 | [diff] [blame] | 110 |         mFrameRecords[i].frameReadyFence.reset(); | 
 | 111 |         mFrameRecords[i].actualPresentFence.reset(); | 
| Jamie Gennis | 82dbc74 | 2012-11-08 19:23:28 -0800 | [diff] [blame] | 112 |     } | 
 | 113 |     mNumFences = 0; | 
| Jamie Gennis | 4b0eba9 | 2013-02-05 13:30:24 -0800 | [diff] [blame] | 114 |     mFrameRecords[mOffset].desiredPresentTime = INT64_MAX; | 
 | 115 |     mFrameRecords[mOffset].frameReadyTime = INT64_MAX; | 
 | 116 |     mFrameRecords[mOffset].actualPresentTime = INT64_MAX; | 
| Jamie Gennis | 82dbc74 | 2012-11-08 19:23:28 -0800 | [diff] [blame] | 117 | } | 
 | 118 |  | 
| Svetoslav | d85084b | 2014-03-20 10:28:31 -0700 | [diff] [blame] | 119 | void FrameTracker::getStats(FrameStats* outStats) const { | 
 | 120 |     Mutex::Autolock lock(mMutex); | 
 | 121 |     processFencesLocked(); | 
 | 122 |  | 
 | 123 |     outStats->refreshPeriodNano = mDisplayPeriod; | 
 | 124 |  | 
 | 125 |     const size_t offset = mOffset; | 
 | 126 |     for (size_t i = 1; i < NUM_FRAME_RECORDS; i++) { | 
 | 127 |         const size_t index = (offset + i) % NUM_FRAME_RECORDS; | 
 | 128 |  | 
 | 129 |         // Skip frame records with no data (if buffer not yet full). | 
 | 130 |         if (mFrameRecords[index].desiredPresentTime == 0) { | 
 | 131 |             continue; | 
 | 132 |         } | 
 | 133 |  | 
 | 134 |         nsecs_t desiredPresentTimeNano = mFrameRecords[index].desiredPresentTime; | 
 | 135 |         outStats->desiredPresentTimesNano.push_back(desiredPresentTimeNano); | 
 | 136 |  | 
 | 137 |         nsecs_t actualPresentTimeNano = mFrameRecords[index].actualPresentTime; | 
 | 138 |         outStats->actualPresentTimesNano.push_back(actualPresentTimeNano); | 
 | 139 |  | 
 | 140 |         nsecs_t frameReadyTimeNano = mFrameRecords[index].frameReadyTime; | 
 | 141 |         outStats->frameReadyTimesNano.push_back(frameReadyTimeNano); | 
 | 142 |     } | 
 | 143 | } | 
 | 144 |  | 
| Dominik Laskowski | 87a07e4 | 2019-10-10 20:38:02 -0700 | [diff] [blame] | 145 | void FrameTracker::logAndResetStats(const std::string_view& name) { | 
| Jamie Gennis | 6547ff4 | 2013-07-16 20:12:42 -0700 | [diff] [blame] | 146 |     Mutex::Autolock lock(mMutex); | 
 | 147 |     logStatsLocked(name); | 
 | 148 |     resetFrameCountersLocked(); | 
 | 149 | } | 
 | 150 |  | 
| Jamie Gennis | 4b0eba9 | 2013-02-05 13:30:24 -0800 | [diff] [blame] | 151 | void FrameTracker::processFencesLocked() const { | 
| Jamie Gennis | 82dbc74 | 2012-11-08 19:23:28 -0800 | [diff] [blame] | 152 |     FrameRecord* records = const_cast<FrameRecord*>(mFrameRecords); | 
 | 153 |     int& numFences = const_cast<int&>(mNumFences); | 
 | 154 |  | 
 | 155 |     for (int i = 1; i < NUM_FRAME_RECORDS && numFences > 0; i++) { | 
 | 156 |         size_t idx = (mOffset+NUM_FRAME_RECORDS-i) % NUM_FRAME_RECORDS; | 
| Jamie Gennis | 6547ff4 | 2013-07-16 20:12:42 -0700 | [diff] [blame] | 157 |         bool updated = false; | 
| Jamie Gennis | 82dbc74 | 2012-11-08 19:23:28 -0800 | [diff] [blame] | 158 |  | 
| Brian Anderson | 3d4039d | 2016-09-23 16:31:30 -0700 | [diff] [blame] | 159 |         const std::shared_ptr<FenceTime>& rfence = records[idx].frameReadyFence; | 
| Peiyong Lin | 566a3b4 | 2018-01-09 18:22:43 -0800 | [diff] [blame] | 160 |         if (rfence != nullptr) { | 
| Jamie Gennis | 82dbc74 | 2012-11-08 19:23:28 -0800 | [diff] [blame] | 161 |             records[idx].frameReadyTime = rfence->getSignalTime(); | 
 | 162 |             if (records[idx].frameReadyTime < INT64_MAX) { | 
| Peiyong Lin | 566a3b4 | 2018-01-09 18:22:43 -0800 | [diff] [blame] | 163 |                 records[idx].frameReadyFence = nullptr; | 
| Jamie Gennis | 82dbc74 | 2012-11-08 19:23:28 -0800 | [diff] [blame] | 164 |                 numFences--; | 
| Jamie Gennis | 6547ff4 | 2013-07-16 20:12:42 -0700 | [diff] [blame] | 165 |                 updated = true; | 
| Jamie Gennis | 82dbc74 | 2012-11-08 19:23:28 -0800 | [diff] [blame] | 166 |             } | 
 | 167 |         } | 
 | 168 |  | 
| Brian Anderson | 3d4039d | 2016-09-23 16:31:30 -0700 | [diff] [blame] | 169 |         const std::shared_ptr<FenceTime>& pfence = | 
 | 170 |                 records[idx].actualPresentFence; | 
| Peiyong Lin | 566a3b4 | 2018-01-09 18:22:43 -0800 | [diff] [blame] | 171 |         if (pfence != nullptr) { | 
| Jamie Gennis | 82dbc74 | 2012-11-08 19:23:28 -0800 | [diff] [blame] | 172 |             records[idx].actualPresentTime = pfence->getSignalTime(); | 
 | 173 |             if (records[idx].actualPresentTime < INT64_MAX) { | 
| Peiyong Lin | 566a3b4 | 2018-01-09 18:22:43 -0800 | [diff] [blame] | 174 |                 records[idx].actualPresentFence = nullptr; | 
| Jamie Gennis | 82dbc74 | 2012-11-08 19:23:28 -0800 | [diff] [blame] | 175 |                 numFences--; | 
| Jamie Gennis | 6547ff4 | 2013-07-16 20:12:42 -0700 | [diff] [blame] | 176 |                 updated = true; | 
| Jamie Gennis | 82dbc74 | 2012-11-08 19:23:28 -0800 | [diff] [blame] | 177 |             } | 
 | 178 |         } | 
| Jamie Gennis | 6547ff4 | 2013-07-16 20:12:42 -0700 | [diff] [blame] | 179 |  | 
 | 180 |         if (updated) { | 
 | 181 |             updateStatsLocked(idx); | 
 | 182 |         } | 
| Jamie Gennis | 82dbc74 | 2012-11-08 19:23:28 -0800 | [diff] [blame] | 183 |     } | 
 | 184 | } | 
 | 185 |  | 
| Jamie Gennis | 6547ff4 | 2013-07-16 20:12:42 -0700 | [diff] [blame] | 186 | void FrameTracker::updateStatsLocked(size_t newFrameIdx) const { | 
 | 187 |     int* numFrames = const_cast<int*>(mNumFrames); | 
 | 188 |  | 
 | 189 |     if (mDisplayPeriod > 0 && isFrameValidLocked(newFrameIdx)) { | 
 | 190 |         size_t prevFrameIdx = (newFrameIdx+NUM_FRAME_RECORDS-1) % | 
 | 191 |                 NUM_FRAME_RECORDS; | 
 | 192 |  | 
 | 193 |         if (isFrameValidLocked(prevFrameIdx)) { | 
 | 194 |             nsecs_t newPresentTime = | 
 | 195 |                     mFrameRecords[newFrameIdx].actualPresentTime; | 
 | 196 |             nsecs_t prevPresentTime = | 
 | 197 |                     mFrameRecords[prevFrameIdx].actualPresentTime; | 
 | 198 |  | 
 | 199 |             nsecs_t duration = newPresentTime - prevPresentTime; | 
 | 200 |             int numPeriods = int((duration + mDisplayPeriod/2) / | 
 | 201 |                     mDisplayPeriod); | 
 | 202 |  | 
 | 203 |             for (int i = 0; i < NUM_FRAME_BUCKETS-1; i++) { | 
 | 204 |                 int nextBucket = 1 << (i+1); | 
 | 205 |                 if (numPeriods < nextBucket) { | 
 | 206 |                     numFrames[i]++; | 
 | 207 |                     return; | 
 | 208 |                 } | 
 | 209 |             } | 
 | 210 |  | 
 | 211 |             // The last duration bucket is a catch-all. | 
 | 212 |             numFrames[NUM_FRAME_BUCKETS-1]++; | 
 | 213 |         } | 
 | 214 |     } | 
 | 215 | } | 
 | 216 |  | 
 | 217 | void FrameTracker::resetFrameCountersLocked() { | 
 | 218 |     for (int i = 0; i < NUM_FRAME_BUCKETS; i++) { | 
 | 219 |         mNumFrames[i] = 0; | 
 | 220 |     } | 
 | 221 | } | 
 | 222 |  | 
| Dominik Laskowski | 87a07e4 | 2019-10-10 20:38:02 -0700 | [diff] [blame] | 223 | void FrameTracker::logStatsLocked(const std::string_view& name) const { | 
| Jamie Gennis | 6547ff4 | 2013-07-16 20:12:42 -0700 | [diff] [blame] | 224 |     for (int i = 0; i < NUM_FRAME_BUCKETS; i++) { | 
 | 225 |         if (mNumFrames[i] > 0) { | 
 | 226 |             EventLog::logFrameDurations(name, mNumFrames, NUM_FRAME_BUCKETS); | 
 | 227 |             return; | 
 | 228 |         } | 
 | 229 |     } | 
 | 230 | } | 
 | 231 |  | 
 | 232 | bool FrameTracker::isFrameValidLocked(size_t idx) const { | 
 | 233 |     return mFrameRecords[idx].actualPresentTime > 0 && | 
 | 234 |             mFrameRecords[idx].actualPresentTime < INT64_MAX; | 
 | 235 | } | 
 | 236 |  | 
| Yiwei Zhang | 5434a78 | 2018-12-05 18:06:32 -0800 | [diff] [blame] | 237 | void FrameTracker::dumpStats(std::string& result) const { | 
| Jamie Gennis | 4b0eba9 | 2013-02-05 13:30:24 -0800 | [diff] [blame] | 238 |     Mutex::Autolock lock(mMutex); | 
 | 239 |     processFencesLocked(); | 
| Jamie Gennis | 82dbc74 | 2012-11-08 19:23:28 -0800 | [diff] [blame] | 240 |  | 
 | 241 |     const size_t o = mOffset; | 
 | 242 |     for (size_t i = 1; i < NUM_FRAME_RECORDS; i++) { | 
 | 243 |         const size_t index = (o+i) % NUM_FRAME_RECORDS; | 
| Yiwei Zhang | 5434a78 | 2018-12-05 18:06:32 -0800 | [diff] [blame] | 244 |         base::StringAppendF(&result, "%" PRId64 "\t%" PRId64 "\t%" PRId64 "\n", | 
 | 245 |                             mFrameRecords[index].desiredPresentTime, | 
 | 246 |                             mFrameRecords[index].actualPresentTime, | 
 | 247 |                             mFrameRecords[index].frameReadyTime); | 
| Jamie Gennis | 82dbc74 | 2012-11-08 19:23:28 -0800 | [diff] [blame] | 248 |     } | 
 | 249 |     result.append("\n"); | 
 | 250 | } | 
 | 251 |  | 
 | 252 | } // namespace android | 
| Ady Abraham | b0dbdaa | 2020-01-06 16:19:42 -0800 | [diff] [blame] | 253 |  | 
 | 254 | // TODO(b/129481165): remove the #pragma below and fix conversion issues | 
 | 255 | #pragma clang diagnostic pop // ignored "-Wconversion" |