blob: 66df9dca206abb25a186b9a043a5b8f6be0556c7 [file] [log] [blame]
Ady Abraham09bd3922019-04-08 10:44:56 -07001/*
2 * Copyright 2019 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#pragma once
18
19#include <cinttypes>
20#include <cstdint>
21#include <deque>
22#include <mutex>
23#include <numeric>
24#include <string>
25
26#include <log/log.h>
27
28#include <utils/Mutex.h>
29#include <utils/Timers.h>
30
31#include "SchedulerUtils.h"
32
33namespace android {
34namespace scheduler {
35
36/*
37 * This class represents information about individial layers.
38 */
39class LayerInfo {
40 /**
41 * Struct that keeps the information about the refresh rate for last
42 * HISTORY_SIZE frames. This is used to better determine the refresh rate
43 * for individual layers.
44 */
45 class RefreshRateHistory {
46 public:
47 explicit RefreshRateHistory(nsecs_t minRefreshDuration)
48 : mMinRefreshDuration(minRefreshDuration) {}
Ady Abraham187d2d82019-07-10 16:18:48 -070049 void insertRefreshRate(int refreshRate) {
Ady Abraham09bd3922019-04-08 10:44:56 -070050 mElements.push_back(refreshRate);
51 if (mElements.size() > HISTORY_SIZE) {
52 mElements.pop_front();
53 }
54 }
55
56 float getRefreshRateAvg() const {
Ady Abraham187d2d82019-07-10 16:18:48 -070057 if (mElements.empty()) {
58 return 1e9f / mMinRefreshDuration;
Ady Abraham09bd3922019-04-08 10:44:56 -070059 }
60
Ady Abraham187d2d82019-07-10 16:18:48 -070061 return scheduler::calculate_mean(mElements);
Ady Abraham09bd3922019-04-08 10:44:56 -070062 }
Ady Abraham187d2d82019-07-10 16:18:48 -070063
Ady Abraham09bd3922019-04-08 10:44:56 -070064 void clearHistory() { mElements.clear(); }
65
66 private:
67 std::deque<nsecs_t> mElements;
68 static constexpr size_t HISTORY_SIZE = 30;
69 const nsecs_t mMinRefreshDuration;
70 };
71
72 /**
73 * Struct that keeps the information about the present time for last
74 * HISTORY_SIZE frames. This is used to better determine whether the given layer
75 * is still relevant and it's refresh rate should be considered.
76 */
77 class PresentTimeHistory {
78 public:
79 void insertPresentTime(nsecs_t presentTime) {
80 mElements.push_back(presentTime);
81 if (mElements.size() > HISTORY_SIZE) {
82 mElements.pop_front();
83 }
84 }
85
86 // Checks whether the present time that was inserted HISTORY_SIZE ago is within a
87 // certain threshold: TIME_EPSILON_NS.
88 bool isRelevant() const {
Ady Abraham616b8322019-06-12 13:30:07 -070089 if (mElements.size() < 2) {
90 return false;
Ady Abraham09bd3922019-04-08 10:44:56 -070091 }
Ady Abraham616b8322019-06-12 13:30:07 -070092
93 // The layer had to publish at least HISTORY_SIZE or HISTORY_TIME of updates
94 if (mElements.size() != HISTORY_SIZE &&
95 mElements.at(mElements.size() - 1) - mElements.at(0) < HISTORY_TIME.count()) {
96 return false;
97 }
98
Ana Krulecad083c42019-06-26 16:28:08 -070099 // The last update should not be older than OBSOLETE_TIME_EPSILON_NS nanoseconds.
100 const int64_t obsoleteEpsilon =
101 systemTime() - scheduler::OBSOLETE_TIME_EPSILON_NS.count();
Ady Abraham616b8322019-06-12 13:30:07 -0700102 if (mElements.at(mElements.size() - 1) < obsoleteEpsilon) {
103 return false;
104 }
105
106 return true;
Ady Abraham09bd3922019-04-08 10:44:56 -0700107 }
108
Ana Krulecad083c42019-06-26 16:28:08 -0700109 bool isLowActivityLayer() const {
110 // We want to make sure that we received more than two frames from the layer
111 // in order to check low activity.
112 if (mElements.size() < 2) {
113 return false;
114 }
115
116 const int64_t obsoleteEpsilon =
117 systemTime() - scheduler::LOW_ACTIVITY_EPSILON_NS.count();
118 // Check the frame before last to determine whether there is low activity.
119 // If that frame is older than LOW_ACTIVITY_EPSILON_NS, the layer is sending
120 // infrequent updates.
121 if (mElements.at(mElements.size() - 2) < obsoleteEpsilon) {
122 return true;
123 }
124
125 return false;
126 }
127
Ady Abraham09bd3922019-04-08 10:44:56 -0700128 void clearHistory() { mElements.clear(); }
129
130 private:
131 std::deque<nsecs_t> mElements;
132 static constexpr size_t HISTORY_SIZE = 10;
Ady Abraham616b8322019-06-12 13:30:07 -0700133 static constexpr std::chrono::nanoseconds HISTORY_TIME = 500ms;
Ady Abraham09bd3922019-04-08 10:44:56 -0700134 };
135
136public:
Ana Krulecad083c42019-06-26 16:28:08 -0700137 LayerInfo(const std::string name, float minRefreshRate, float maxRefreshRate);
Ady Abraham09bd3922019-04-08 10:44:56 -0700138 ~LayerInfo();
139
140 LayerInfo(const LayerInfo&) = delete;
141 LayerInfo& operator=(const LayerInfo&) = delete;
142
143 // Records the last requested oresent time. It also stores information about when
144 // the layer was last updated. If the present time is farther in the future than the
145 // updated time, the updated time is the present time.
146 void setLastPresentTime(nsecs_t lastPresentTime);
147
Ady Abrahama315ce72019-04-24 14:35:20 -0700148 void setHDRContent(bool isHdr) {
149 std::lock_guard lock(mLock);
150 mIsHDR = isHdr;
151 }
152
153 void setVisibility(bool visible) {
154 std::lock_guard lock(mLock);
155 mIsVisible = visible;
156 }
157
Ady Abraham09bd3922019-04-08 10:44:56 -0700158 // Checks the present time history to see whether the layer is relevant.
159 bool isRecentlyActive() const {
160 std::lock_guard lock(mLock);
161 return mPresentTimeHistory.isRelevant();
162 }
163
164 // Calculate the average refresh rate.
165 float getDesiredRefreshRate() const {
166 std::lock_guard lock(mLock);
Ana Krulecad083c42019-06-26 16:28:08 -0700167
168 if (mPresentTimeHistory.isLowActivityLayer()) {
169 return 1e9f / mLowActivityRefreshDuration;
170 }
Ady Abraham09bd3922019-04-08 10:44:56 -0700171 return mRefreshRateHistory.getRefreshRateAvg();
172 }
173
Ady Abrahama315ce72019-04-24 14:35:20 -0700174 bool getHDRContent() {
175 std::lock_guard lock(mLock);
176 return mIsHDR;
177 }
178
179 bool isVisible() {
180 std::lock_guard lock(mLock);
181 return mIsVisible;
182 }
183
Ady Abraham09bd3922019-04-08 10:44:56 -0700184 // Return the last updated time. If the present time is farther in the future than the
185 // updated time, the updated time is the present time.
186 nsecs_t getLastUpdatedTime() {
187 std::lock_guard lock(mLock);
188 return mLastUpdatedTime;
189 }
190
191 std::string getName() const { return mName; }
192
193 void clearHistory() {
194 std::lock_guard lock(mLock);
195 mRefreshRateHistory.clearHistory();
196 mPresentTimeHistory.clearHistory();
197 }
198
199private:
200 const std::string mName;
201 const nsecs_t mMinRefreshDuration;
Ana Krulecad083c42019-06-26 16:28:08 -0700202 const nsecs_t mLowActivityRefreshDuration;
Ady Abraham09bd3922019-04-08 10:44:56 -0700203 mutable std::mutex mLock;
204 nsecs_t mLastUpdatedTime GUARDED_BY(mLock) = 0;
205 nsecs_t mLastPresentTime GUARDED_BY(mLock) = 0;
206 RefreshRateHistory mRefreshRateHistory GUARDED_BY(mLock);
207 PresentTimeHistory mPresentTimeHistory GUARDED_BY(mLock);
Ady Abrahama315ce72019-04-24 14:35:20 -0700208 bool mIsHDR GUARDED_BY(mLock) = false;
209 bool mIsVisible GUARDED_BY(mLock) = false;
Ady Abraham09bd3922019-04-08 10:44:56 -0700210};
211
212} // namespace scheduler
213} // namespace android