blob: 76e8f57adc70204819408941b975394c6d5707a7 [file] [log] [blame]
Ana Krulec98b5b242018-08-10 15:03:23 -07001/*
2 * Copyright 2018 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
Dominik Laskowski98041832019-08-01 18:35:59 -070019#include <atomic>
Kevin DuBois413287f2019-02-25 08:46:47 -080020#include <functional>
Ana Krulec98b5b242018-08-10 15:03:23 -070021#include <memory>
Dominik Laskowski98041832019-08-01 18:35:59 -070022#include <mutex>
23#include <optional>
24#include <unordered_map>
Ana Krulec98b5b242018-08-10 15:03:23 -070025
Ady Abrahamdec1a412020-01-24 10:23:50 -080026// TODO(b/129481165): remove the #pragma below and fix conversion issues
27#pragma clang diagnostic push
28#pragma clang diagnostic ignored "-Wconversion"
Dominik Laskowskidcb38bb2019-01-25 02:35:50 -080029#include <ui/GraphicTypes.h>
Ady Abrahamdec1a412020-01-24 10:23:50 -080030#pragma clang diagnostic pop
Ana Krulec98b5b242018-08-10 15:03:23 -070031
Ana Krulec98b5b242018-08-10 15:03:23 -070032#include "EventThread.h"
Ana Krulec3084c052018-11-21 20:27:17 +010033#include "LayerHistory.h"
Ana Krulecf2c006d2019-06-21 15:37:07 -070034#include "OneShotTimer.h"
Ana Krulec8d3e4f32019-03-05 10:40:33 -080035#include "RefreshRateConfigs.h"
Ana Krulec434c22d2018-11-28 13:48:36 +010036#include "SchedulerUtils.h"
Ana Krulec98b5b242018-08-10 15:03:23 -070037
38namespace android {
39
Ady Abraham2139f732019-11-13 18:56:40 -080040using namespace std::chrono_literals;
Ady Abraham5def7332020-05-29 16:13:47 -070041using scheduler::LayerHistory;
Ady Abraham2139f732019-11-13 18:56:40 -080042
Dominik Laskowski98041832019-08-01 18:35:59 -070043class FenceTime;
Dominik Laskowski6505f792019-09-18 11:10:05 -070044class InjectVSyncSource;
Ady Abraham8735eac2020-08-12 16:35:04 -070045class PredictedVsyncTracer;
Ady Abraham5a858552020-03-31 17:54:56 -070046
47namespace scheduler {
Ady Abraham8cb21882020-08-26 18:22:05 -070048class VsyncController;
Ady Abraham5a858552020-03-31 17:54:56 -070049class VSyncDispatch;
50class VSyncTracker;
51} // namespace scheduler
Ana Krulece588e312018-09-18 12:32:24 -070052
Adithya Srinivasan5f683cf2020-09-15 14:21:04 -070053namespace frametimeline {
54class TokenManager;
55} // namespace frametimeline
56
Dominik Laskowski8b01cc02020-07-14 19:02:41 -070057struct ISchedulerCallback {
58 virtual void setVsyncEnabled(bool) = 0;
Ady Abraham3a77a7b2019-12-02 18:46:59 -080059 virtual void changeRefreshRate(const scheduler::RefreshRateConfigs::RefreshRate&,
60 scheduler::RefreshRateConfigEvent) = 0;
61 virtual void repaintEverythingForHWC() = 0;
Ady Abrahama09852a2020-02-20 14:23:42 -080062 virtual void kernelTimerChanged(bool expired) = 0;
Dominik Laskowski8b01cc02020-07-14 19:02:41 -070063
64protected:
65 ~ISchedulerCallback() = default;
Ady Abraham3a77a7b2019-12-02 18:46:59 -080066};
67
Dominik Laskowski08d05c22020-07-22 00:05:08 -070068class Scheduler {
Ana Krulec98b5b242018-08-10 15:03:23 -070069public:
Ady Abraham2139f732019-11-13 18:56:40 -080070 using RefreshRate = scheduler::RefreshRateConfigs::RefreshRate;
Dominik Laskowski98041832019-08-01 18:35:59 -070071 using ConfigEvent = scheduler::RefreshRateConfigEvent;
72
Dominik Laskowski8b01cc02020-07-14 19:02:41 -070073 Scheduler(const scheduler::RefreshRateConfigs&, ISchedulerCallback&);
Dominik Laskowski08d05c22020-07-22 00:05:08 -070074 ~Scheduler();
Ana Krulec98b5b242018-08-10 15:03:23 -070075
Dominik Laskowski98041832019-08-01 18:35:59 -070076 using ConnectionHandle = scheduler::ConnectionHandle;
Adithya Srinivasan5f683cf2020-09-15 14:21:04 -070077 ConnectionHandle createConnection(const char* connectionName, frametimeline::TokenManager*,
Ady Abraham9c53ee72020-07-22 21:16:18 -070078 std::chrono::nanoseconds workDuration,
79 std::chrono::nanoseconds readyDuration,
Dominik Laskowski98041832019-08-01 18:35:59 -070080 impl::EventThread::InterceptVSyncsCallback);
Ana Krulec98b5b242018-08-10 15:03:23 -070081
Ady Abraham62f216c2020-10-13 19:07:23 -070082 sp<IDisplayEventConnection> createDisplayEventConnection(
83 ConnectionHandle, ISurfaceComposer::EventRegistrationFlags eventRegistration = {});
Ana Krulece588e312018-09-18 12:32:24 -070084
Dominik Laskowski98041832019-08-01 18:35:59 -070085 sp<EventThreadConnection> getEventConnection(ConnectionHandle);
Kevin DuBois413287f2019-02-25 08:46:47 -080086
Dominik Laskowski98041832019-08-01 18:35:59 -070087 void onHotplugReceived(ConnectionHandle, PhysicalDisplayId, bool connected);
Ady Abrahamdfd62162020-06-10 16:11:56 -070088 void onPrimaryDisplayConfigChanged(ConnectionHandle, PhysicalDisplayId,
89 HwcConfigIndexType configId, nsecs_t vsyncPeriod)
90 EXCLUDES(mFeatureStateLock);
91 void onNonPrimaryDisplayConfigChanged(ConnectionHandle, PhysicalDisplayId,
92 HwcConfigIndexType configId, nsecs_t vsyncPeriod);
Dominik Laskowski98041832019-08-01 18:35:59 -070093 void onScreenAcquired(ConnectionHandle);
94 void onScreenReleased(ConnectionHandle);
Ana Krulece588e312018-09-18 12:32:24 -070095
Ady Abraham62f216c2020-10-13 19:07:23 -070096 void onFrameRateOverridesChanged(ConnectionHandle, PhysicalDisplayId,
97 std::vector<FrameRateOverride>);
98
Ady Abraham9c53ee72020-07-22 21:16:18 -070099 // Modifies work duration in the event thread.
100 void setDuration(ConnectionHandle, std::chrono::nanoseconds workDuration,
101 std::chrono::nanoseconds readyDuration);
Ana Krulec98b5b242018-08-10 15:03:23 -0700102
Ady Abraham8cb21882020-08-26 18:22:05 -0700103 void getDisplayStatInfo(DisplayStatInfo* stats, nsecs_t now);
Ana Krulece588e312018-09-18 12:32:24 -0700104
Dominik Laskowski6505f792019-09-18 11:10:05 -0700105 // Returns injector handle if injection has toggled, or an invalid handle otherwise.
106 ConnectionHandle enableVSyncInjection(bool enable);
Dominik Laskowski6505f792019-09-18 11:10:05 -0700107 // Returns false if injection is disabled.
Ady Abraham9c53ee72020-07-22 21:16:18 -0700108 bool injectVSync(nsecs_t when, nsecs_t expectedVSyncTime, nsecs_t deadlineTimestamp);
Ana Krulece588e312018-09-18 12:32:24 -0700109 void enableHardwareVsync();
110 void disableHardwareVsync(bool makeUnavailable);
Dominik Laskowski98041832019-08-01 18:35:59 -0700111
Alec Mourif8e689c2019-05-20 18:32:22 -0700112 // Resyncs the scheduler to hardware vsync.
113 // If makeAvailable is true, then hardware vsync will be turned on.
114 // Otherwise, if hardware vsync is not already enabled then this method will
115 // no-op.
116 // The period is the vsync period from the current display configuration.
Ana Krulecc2870422019-01-29 19:00:58 -0800117 void resyncToHardwareVsync(bool makeAvailable, nsecs_t period);
Steven Thomas2bbaabe2019-08-28 16:08:35 -0700118 void resync();
Dominik Laskowski98041832019-08-01 18:35:59 -0700119
Ady Abraham8cb21882020-08-26 18:22:05 -0700120 // Passes a vsync sample to VsyncController. periodFlushed will be true if
121 // VsyncController detected that the vsync period changed, and false otherwise.
Ady Abraham5dee2f12020-02-05 17:49:47 -0800122 void addResyncSample(nsecs_t timestamp, std::optional<nsecs_t> hwcVsyncPeriod,
123 bool* periodFlushed);
Dominik Laskowski98041832019-08-01 18:35:59 -0700124 void addPresentFence(const std::shared_ptr<FenceTime>&);
Ana Krulece588e312018-09-18 12:32:24 -0700125 void setIgnorePresentFences(bool ignore);
Ady Abraham8f1ee7f2019-04-05 10:32:50 -0700126
Dominik Laskowskif7a09ed2019-10-07 13:54:18 -0700127 // Layers are registered on creation, and unregistered when the weak reference expires.
128 void registerLayer(Layer*);
Ady Abraham5def7332020-05-29 16:13:47 -0700129 void recordLayerHistory(Layer*, nsecs_t presentTime, LayerHistory::LayerUpdateType updateType);
Ady Abraham32efd542020-05-19 17:49:26 -0700130 void setConfigChangePending(bool pending);
Dominik Laskowskif7a09ed2019-10-07 13:54:18 -0700131
Dominik Laskowski49cea512019-11-12 14:13:23 -0800132 // Detects content using layer history, and selects a matching refresh rate.
133 void chooseRefreshRateForContent();
Ady Abraham97d04232019-03-05 19:48:12 -0800134
Dominik Laskowski98041832019-08-01 18:35:59 -0700135 bool isIdleTimerEnabled() const { return mIdleTimer.has_value(); }
Ady Abraham8532d012019-05-08 14:50:56 -0700136 void resetIdleTimer();
137
138 // Function that resets the touch timer.
139 void notifyTouchEvent();
140
Ady Abraham6fe2c172019-07-12 12:37:57 -0700141 void setDisplayPowerState(bool normal);
142
Ady Abraham55fa7272020-09-30 19:19:27 -0700143 scheduler::VSyncDispatch& getVsyncDispatch() { return *mVsyncSchedule.dispatch; }
144
Ady Abraham0bb6a472020-10-12 10:22:13 -0700145 // Returns true if a given vsync timestamp is considered valid vsync
146 // for a given uid
147 bool isVsyncValid(nsecs_t expectedVsyncTimestamp, uid_t uid) const;
148
Dominik Laskowski98041832019-08-01 18:35:59 -0700149 void dump(std::string&) const;
150 void dump(ConnectionHandle, std::string&) const;
Ady Abraham8cb21882020-08-26 18:22:05 -0700151 void dumpVsync(std::string&) const;
Ana Krulec0c8cd522018-08-31 12:27:28 -0700152
Ady Abraham2139f732019-11-13 18:56:40 -0800153 // Get the appropriate refresh for current conditions.
154 std::optional<HwcConfigIndexType> getPreferredConfigId();
Daniel Solomon0f0ddc12019-08-19 19:31:09 -0700155
Ady Abraham3a77a7b2019-12-02 18:46:59 -0800156 // Notifies the scheduler about a refresh rate timeline change.
Peiyong Line9d809e2020-04-14 13:10:48 -0700157 void onNewVsyncPeriodChangeTimeline(const hal::VsyncPeriodChangeTimeline& timeline);
Ady Abraham3a77a7b2019-12-02 18:46:59 -0800158
159 // Notifies the scheduler when the display was refreshed
160 void onDisplayRefreshed(nsecs_t timestamp);
161
Ady Abraham8a82ba62020-01-17 12:43:17 -0800162 // Notifies the scheduler when the display size has changed. Called from SF's main thread
163 void onPrimaryDisplayAreaChanged(uint32_t displayArea);
164
Alec Mouri717bcb62020-02-10 17:07:19 -0800165 size_t getEventThreadConnectionCount(ConnectionHandle handle);
166
Ady Abraham9c53ee72020-07-22 21:16:18 -0700167 std::unique_ptr<VSyncSource> makePrimaryDispSyncSource(const char* name,
168 std::chrono::nanoseconds workDuration,
169 std::chrono::nanoseconds readyDuration,
170 bool traceVsync = true);
171
Ana Krulec98b5b242018-08-10 15:03:23 -0700172private:
Ana Krulecafb45842019-02-13 13:33:03 -0800173 friend class TestableScheduler;
174
Ana Krulecfefd6ae2019-02-13 17:53:08 -0800175 // In order to make sure that the features don't override themselves, we need a state machine
176 // to keep track which feature requested the config change.
Dominik Laskowskidd252cd2019-07-26 09:10:16 -0700177 enum class ContentDetectionState { Off, On };
178 enum class TimerState { Reset, Expired };
179 enum class TouchState { Inactive, Active };
Ana Krulecfefd6ae2019-02-13 17:53:08 -0800180
Dominik Laskowski8b01cc02020-07-14 19:02:41 -0700181 struct Options {
182 // Whether to use idle timer callbacks that support the kernel timer.
183 bool supportKernelTimer;
184 // Whether to use content detection at all.
185 bool useContentDetection;
186 // Whether to use improved content detection.
187 bool useContentDetectionV2;
Dominik Laskowski8b01cc02020-07-14 19:02:41 -0700188 };
Dominik Laskowski983f2b52020-06-25 16:54:06 -0700189
Dominik Laskowski8b01cc02020-07-14 19:02:41 -0700190 struct VsyncSchedule {
Ady Abraham8cb21882020-08-26 18:22:05 -0700191 std::unique_ptr<scheduler::VsyncController> controller;
Dominik Laskowski8b01cc02020-07-14 19:02:41 -0700192 std::unique_ptr<scheduler::VSyncTracker> tracker;
193 std::unique_ptr<scheduler::VSyncDispatch> dispatch;
194 };
195
196 // Unlike the testing constructor, this creates the VsyncSchedule, LayerHistory, and timers.
197 Scheduler(const scheduler::RefreshRateConfigs&, ISchedulerCallback&, Options);
198
199 // Used by tests to inject mocks.
200 Scheduler(VsyncSchedule, const scheduler::RefreshRateConfigs&, ISchedulerCallback&,
201 std::unique_ptr<LayerHistory>, Options);
202
Ady Abraham8cb21882020-08-26 18:22:05 -0700203 static VsyncSchedule createVsyncSchedule(bool supportKernelIdleTimer);
Dominik Laskowski983f2b52020-06-25 16:54:06 -0700204 static std::unique_ptr<LayerHistory> createLayerHistory(const scheduler::RefreshRateConfigs&,
205 bool useContentDetectionV2);
Dominik Laskowski7c9dbf92019-08-01 17:57:31 -0700206
Dominik Laskowski6505f792019-09-18 11:10:05 -0700207 // Create a connection on the given EventThread.
Steven Thomas2bbaabe2019-08-28 16:08:35 -0700208 ConnectionHandle createConnection(std::unique_ptr<EventThread>);
Ady Abraham62f216c2020-10-13 19:07:23 -0700209 sp<EventThreadConnection> createConnectionInternal(
210 EventThread*, ISurfaceComposer::EventRegistrationFlags eventRegistration = {});
Dominik Laskowskiccf37d72019-02-01 16:47:58 -0800211
Dominik Laskowski3a80a382019-07-25 11:16:07 -0700212 // Update feature state machine to given state when corresponding timer resets or expires.
213 void kernelIdleTimerCallback(TimerState);
214 void idleTimerCallback(TimerState);
215 void touchTimerCallback(TimerState);
216 void displayPowerTimerCallback(TimerState);
217
Ady Abraham6fe2c172019-07-12 12:37:57 -0700218 // handles various timer features to change the refresh rate.
219 template <class T>
Ady Abrahamdfd62162020-06-10 16:11:56 -0700220 bool handleTimerStateChanged(T* currentState, T newState);
Dominik Laskowski98041832019-08-01 18:35:59 -0700221
222 void setVsyncPeriod(nsecs_t period);
223
Ana Krulec3803b8d2020-02-03 16:35:46 -0800224 // This function checks whether individual features that are affecting the refresh rate
225 // selection were initialized, prioritizes them, and calculates the HwcConfigIndexType
226 // for the suggested refresh rate.
Ady Abrahamdfd62162020-06-10 16:11:56 -0700227 HwcConfigIndexType calculateRefreshRateConfigIndexType(
228 scheduler::RefreshRateConfigs::GlobalSignals* consideredSignals = nullptr)
Ady Abraham1adbb722020-05-15 11:51:48 -0700229 REQUIRES(mFeatureStateLock);
Ana Krulec7ab56032018-11-02 20:51:06 +0100230
Ady Abrahamdfd62162020-06-10 16:11:56 -0700231 void dispatchCachedReportedConfig() REQUIRES(mFeatureStateLock);
232
Dominik Laskowski98041832019-08-01 18:35:59 -0700233 // Stores EventThread associated with a given VSyncSource, and an initial EventThreadConnection.
234 struct Connection {
235 sp<EventThreadConnection> connection;
236 std::unique_ptr<EventThread> thread;
237 };
Ady Abraham09bd3922019-04-08 10:44:56 -0700238
Dominik Laskowski98041832019-08-01 18:35:59 -0700239 ConnectionHandle::Id mNextConnectionHandleId = 0;
Ana Krulec6ddd2612020-09-24 13:06:33 -0700240 mutable std::mutex mConnectionsLock;
241 std::unordered_map<ConnectionHandle, Connection> mConnections GUARDED_BY(mConnectionsLock);
Ana Krulece588e312018-09-18 12:32:24 -0700242
Dominik Laskowski6505f792019-09-18 11:10:05 -0700243 bool mInjectVSyncs = false;
244 InjectVSyncSource* mVSyncInjector = nullptr;
245 ConnectionHandle mInjectorConnectionHandle;
246
Ana Krulece588e312018-09-18 12:32:24 -0700247 std::mutex mHWVsyncLock;
Dominik Laskowski98041832019-08-01 18:35:59 -0700248 bool mPrimaryHWVsyncEnabled GUARDED_BY(mHWVsyncLock) = false;
249 bool mHWVsyncAvailable GUARDED_BY(mHWVsyncLock) = false;
250
Steven Thomas2bbaabe2019-08-28 16:08:35 -0700251 std::atomic<nsecs_t> mLastResyncTime = 0;
Ana Krulece588e312018-09-18 12:32:24 -0700252
Dominik Laskowski8b01cc02020-07-14 19:02:41 -0700253 const Options mOptions;
254 VsyncSchedule mVsyncSchedule;
Ana Krulec7ab56032018-11-02 20:51:06 +0100255
Dominik Laskowski49cea512019-11-12 14:13:23 -0800256 // Used to choose refresh rate if content detection is enabled.
Dominik Laskowski983f2b52020-06-25 16:54:06 -0700257 const std::unique_ptr<LayerHistory> mLayerHistory;
Ana Krulecfb772822018-11-30 10:44:07 +0100258
Dominik Laskowski98041832019-08-01 18:35:59 -0700259 // Timer that records time between requests for next vsync.
260 std::optional<scheduler::OneShotTimer> mIdleTimer;
Ady Abraham8532d012019-05-08 14:50:56 -0700261 // Timer used to monitor touch events.
Dominik Laskowski98041832019-08-01 18:35:59 -0700262 std::optional<scheduler::OneShotTimer> mTouchTimer;
Ady Abraham6fe2c172019-07-12 12:37:57 -0700263 // Timer used to monitor display power mode.
Dominik Laskowski98041832019-08-01 18:35:59 -0700264 std::optional<scheduler::OneShotTimer> mDisplayPowerTimer;
Ady Abraham6fe2c172019-07-12 12:37:57 -0700265
Ady Abraham3a77a7b2019-12-02 18:46:59 -0800266 ISchedulerCallback& mSchedulerCallback;
Ana Krulecfefd6ae2019-02-13 17:53:08 -0800267
268 // In order to make sure that the features don't override themselves, we need a state machine
269 // to keep track which feature requested the config change.
270 std::mutex mFeatureStateLock;
Dominik Laskowskidd252cd2019-07-26 09:10:16 -0700271
272 struct {
Ady Abraham4ccdcb42020-02-11 17:34:34 -0800273 ContentDetectionState contentDetectionV1 = ContentDetectionState::Off;
Dominik Laskowskidd252cd2019-07-26 09:10:16 -0700274 TimerState idleTimer = TimerState::Reset;
275 TouchState touch = TouchState::Inactive;
276 TimerState displayPowerTimer = TimerState::Expired;
277
Ady Abraham2139f732019-11-13 18:56:40 -0800278 std::optional<HwcConfigIndexType> configId;
Ady Abraham5def7332020-05-29 16:13:47 -0700279 LayerHistory::Summary contentRequirements;
Dominik Laskowskidd252cd2019-07-26 09:10:16 -0700280
Dominik Laskowskidd252cd2019-07-26 09:10:16 -0700281 bool isDisplayPowerStateNormal = true;
Ady Abrahamdfd62162020-06-10 16:11:56 -0700282
283 // Used to cache the last parameters of onPrimaryDisplayConfigChanged
284 struct ConfigChangedParams {
285 ConnectionHandle handle;
286 PhysicalDisplayId displayId;
287 HwcConfigIndexType configId;
288 nsecs_t vsyncPeriod;
289 };
290
291 std::optional<ConfigChangedParams> cachedConfigChangedParams;
Dominik Laskowskidd252cd2019-07-26 09:10:16 -0700292 } mFeatures GUARDED_BY(mFeatureStateLock);
Ady Abraham09bd3922019-04-08 10:44:56 -0700293
294 const scheduler::RefreshRateConfigs& mRefreshRateConfigs;
Ady Abraham3a77a7b2019-12-02 18:46:59 -0800295
296 std::mutex mVsyncTimelineLock;
Peiyong Line9d809e2020-04-14 13:10:48 -0700297 std::optional<hal::VsyncPeriodChangeTimeline> mLastVsyncPeriodChangeTimeline
Ady Abraham3a77a7b2019-12-02 18:46:59 -0800298 GUARDED_BY(mVsyncTimelineLock);
299 static constexpr std::chrono::nanoseconds MAX_VSYNC_APPLIED_TIME = 200ms;
Ady Abraham8735eac2020-08-12 16:35:04 -0700300
301 const std::unique_ptr<PredictedVsyncTracer> mPredictedVsyncTracer;
Ana Krulec98b5b242018-08-10 15:03:23 -0700302};
303
Ana Krulec7ab56032018-11-02 20:51:06 +0100304} // namespace android