blob: ec20610f81e6339ccdda4ecd0534079d04f1de71 [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>
Dominik Laskowski62eff352021-12-06 09:59:41 -080020#include <cstdint>
Kevin DuBois413287f2019-02-25 08:46:47 -080021#include <functional>
Dominik Laskowski756b7892021-08-04 12:53:59 -070022#include <future>
Ana Krulec98b5b242018-08-10 15:03:23 -070023#include <memory>
Dominik Laskowski98041832019-08-01 18:35:59 -070024#include <mutex>
Dominik Laskowski98041832019-08-01 18:35:59 -070025#include <unordered_map>
ramindani69b58e82022-09-26 16:48:36 -070026#include <utility>
Ana Krulec98b5b242018-08-10 15:03:23 -070027
Ady Abrahamdec1a412020-01-24 10:23:50 -080028// TODO(b/129481165): remove the #pragma below and fix conversion issues
29#pragma clang diagnostic push
30#pragma clang diagnostic ignored "-Wconversion"
Marin Shalamanovbed7fd32020-12-21 20:02:20 +010031#pragma clang diagnostic ignored "-Wextra"
Dominik Laskowskidcb38bb2019-01-25 02:35:50 -080032#include <ui/GraphicTypes.h>
Marin Shalamanovbed7fd32020-12-21 20:02:20 +010033#pragma clang diagnostic pop // ignored "-Wconversion -Wextra"
Ana Krulec98b5b242018-08-10 15:03:23 -070034
Dominik Laskowski596a2562022-10-28 11:26:12 -040035#include <ftl/fake_guard.h>
36#include <ftl/optional.h>
Dominik Laskowski068173d2021-08-11 17:22:59 -070037#include <scheduler/Features.h>
Dominik Laskowskib418dd72023-06-13 17:31:04 -040038#include <scheduler/FrameTargeter.h>
Dominik Laskowski5d164f22022-07-07 07:56:07 -070039#include <scheduler/Time.h>
Dominik Laskowski1c99a002023-01-20 17:10:36 -050040#include <scheduler/VsyncConfig.h>
Dominik Laskowski01602522022-10-07 19:02:28 -040041#include <ui/DisplayId.h>
Dominik Laskowski6b049ff2023-01-29 15:46:45 -050042#include <ui/DisplayMap.h>
Dominik Laskowski068173d2021-08-11 17:22:59 -070043
Dominik Laskowski530d6bd2022-10-10 16:55:54 -040044#include "Display/DisplayModeRequest.h"
Ana Krulec98b5b242018-08-10 15:03:23 -070045#include "EventThread.h"
Andy Yu2ae6b6b2021-11-18 14:51:06 -080046#include "FrameRateOverrideMappings.h"
Leon Scroggins IIIc275df42023-02-07 16:40:21 -050047#include "ISchedulerCallback.h"
Ana Krulec3084c052018-11-21 20:27:17 +010048#include "LayerHistory.h"
Dominik Laskowski756b7892021-08-04 12:53:59 -070049#include "MessageQueue.h"
Ana Krulecf2c006d2019-06-21 15:37:07 -070050#include "OneShotTimer.h"
Dominik Laskowskid82e0f02022-10-26 15:23:04 -040051#include "RefreshRateSelector.h"
Dominik Laskowski03cfce82022-11-02 12:13:29 -040052#include "Utils/Dumper.h"
Dominik Laskowski1c99a002023-01-20 17:10:36 -050053#include "VsyncModulator.h"
Ana Krulec98b5b242018-08-10 15:03:23 -070054
Ady Abraham822ecbd2023-07-07 16:16:09 -070055#include <FrontEnd/LayerHierarchy.h>
56
Dominik Laskowski62eff352021-12-06 09:59:41 -080057namespace android::scheduler {
58
59// Opaque handle to scheduler connection.
60struct ConnectionHandle {
61 using Id = std::uintptr_t;
62 static constexpr Id INVALID_ID = static_cast<Id>(-1);
63
64 Id id = INVALID_ID;
65
66 explicit operator bool() const { return id != INVALID_ID; }
67};
68
69inline bool operator==(ConnectionHandle lhs, ConnectionHandle rhs) {
70 return lhs.id == rhs.id;
71}
72
73} // namespace android::scheduler
74
75namespace std {
76
77template <>
78struct hash<android::scheduler::ConnectionHandle> {
79 size_t operator()(android::scheduler::ConnectionHandle handle) const {
80 return hash<android::scheduler::ConnectionHandle::Id>()(handle.id);
81 }
82};
83
84} // namespace std
85
Ana Krulec98b5b242018-08-10 15:03:23 -070086namespace android {
87
Dominik Laskowski98041832019-08-01 18:35:59 -070088class FenceTime;
Ana Krulece588e312018-09-18 12:32:24 -070089
Adithya Srinivasan5f683cf2020-09-15 14:21:04 -070090namespace frametimeline {
91class TokenManager;
92} // namespace frametimeline
93
Dominik Laskowski068173d2021-08-11 17:22:59 -070094namespace scheduler {
95
Dominik Laskowskid82e0f02022-10-26 15:23:04 -040096using GlobalSignals = RefreshRateSelector::GlobalSignals;
ramindani69b58e82022-09-26 16:48:36 -070097
Dominik Laskowskic404cb42023-03-03 19:57:53 -050098class VsyncSchedule;
99
Leon Scroggins IIIdb16a2b2023-02-06 17:50:05 -0500100class Scheduler : android::impl::MessageQueue {
Patrick Williams83f36b22022-09-14 17:57:35 +0000101 using Impl = android::impl::MessageQueue;
Dominik Laskowski756b7892021-08-04 12:53:59 -0700102
Ana Krulec98b5b242018-08-10 15:03:23 -0700103public:
Dominik Laskowski1c99a002023-01-20 17:10:36 -0500104 Scheduler(ICompositor&, ISchedulerCallback&, FeatureFlags, sp<VsyncModulator>);
Dominik Laskowski83bd7712022-01-07 14:30:53 -0800105 virtual ~Scheduler();
106
107 void startTimers();
Dominik Laskowski83bd7712022-01-07 14:30:53 -0800108
Leon Scroggins III1af0fb62023-03-02 14:21:44 -0500109 // TODO(b/241285191): Remove this API by promoting pacesetter in onScreen{Acquired,Released}.
110 void setPacesetterDisplay(std::optional<PhysicalDisplayId>) REQUIRES(kMainThreadContext)
Dominik Laskowski596a2562022-10-28 11:26:12 -0400111 EXCLUDES(mDisplayLock);
Dominik Laskowskib5a094b2022-10-27 12:00:12 -0400112
Dominik Laskowski596a2562022-10-28 11:26:12 -0400113 using RefreshRateSelectorPtr = std::shared_ptr<RefreshRateSelector>;
114
Dominik Laskowskic404cb42023-03-03 19:57:53 -0500115 using ConstVsyncSchedulePtr = std::shared_ptr<const VsyncSchedule>;
116 using VsyncSchedulePtr = std::shared_ptr<VsyncSchedule>;
117
Dominik Laskowski596a2562022-10-28 11:26:12 -0400118 void registerDisplay(PhysicalDisplayId, RefreshRateSelectorPtr) REQUIRES(kMainThreadContext)
119 EXCLUDES(mDisplayLock);
120 void unregisterDisplay(PhysicalDisplayId) REQUIRES(kMainThreadContext) EXCLUDES(mDisplayLock);
Dominik Laskowski01602522022-10-07 19:02:28 -0400121
Dominik Laskowski83bd7712022-01-07 14:30:53 -0800122 void run();
Ana Krulec98b5b242018-08-10 15:03:23 -0700123
Dominik Laskowski756b7892021-08-04 12:53:59 -0700124 using Impl::initVsync;
Dominik Laskowski756b7892021-08-04 12:53:59 -0700125
126 using Impl::getScheduledFrameTime;
127 using Impl::setDuration;
128
Dominik Laskowskif11728a2022-07-28 13:07:42 -0700129 using Impl::scheduleConfigure;
Dominik Laskowski46f3e3b2021-08-10 11:44:24 -0700130 using Impl::scheduleFrame;
Dominik Laskowski756b7892021-08-04 12:53:59 -0700131
132 // Schedule an asynchronous or synchronous task on the main thread.
133 template <typename F, typename T = std::invoke_result_t<F>>
134 [[nodiscard]] std::future<T> schedule(F&& f) {
135 auto [task, future] = makeTask(std::move(f));
136 postMessage(std::move(task));
137 return std::move(future);
138 }
Dominik Laskowski9c93d602021-10-07 19:38:26 -0700139
Chavi Weingarten076acac2023-01-19 17:20:43 +0000140 template <typename F, typename T = std::invoke_result_t<F>>
141 [[nodiscard]] std::future<T> scheduleDelayed(F&& f, nsecs_t uptimeDelay) {
142 auto [task, future] = makeTask(std::move(f));
143 postMessageDelayed(std::move(task), uptimeDelay);
144 return std::move(future);
145 }
146
Dominik Laskowski1c99a002023-01-20 17:10:36 -0500147 enum class Cycle {
148 Render, // Surface rendering.
149 LastComposite // Ahead of display compositing by one refresh period.
150 };
151
152 ConnectionHandle createEventThread(Cycle, frametimeline::TokenManager*,
153 std::chrono::nanoseconds workDuration,
154 std::chrono::nanoseconds readyDuration);
Ana Krulec98b5b242018-08-10 15:03:23 -0700155
Ady Abraham62f216c2020-10-13 19:07:23 -0700156 sp<IDisplayEventConnection> createDisplayEventConnection(
Rachel Lee2248f522023-01-27 16:45:23 -0800157 ConnectionHandle, EventRegistrationFlags eventRegistration = {},
Ady Abraham822ecbd2023-07-07 16:16:09 -0700158 const sp<IBinder>& layerHandle = nullptr) EXCLUDES(mChoreographerLock);
Ana Krulece588e312018-09-18 12:32:24 -0700159
Dominik Laskowski98041832019-08-01 18:35:59 -0700160 sp<EventThreadConnection> getEventConnection(ConnectionHandle);
Kevin DuBois413287f2019-02-25 08:46:47 -0800161
Dominik Laskowski98041832019-08-01 18:35:59 -0700162 void onHotplugReceived(ConnectionHandle, PhysicalDisplayId, bool connected);
Ady Abrahamace3d052022-11-17 16:25:05 -0800163 void onPrimaryDisplayModeChanged(ConnectionHandle, const FrameRateMode&) EXCLUDES(mPolicyLock);
164 void onNonPrimaryDisplayModeChanged(ConnectionHandle, const FrameRateMode&);
Dominik Laskowskie99b98c2023-02-02 12:37:23 -0500165
166 void enableSyntheticVsync(bool = true) REQUIRES(kMainThreadContext);
Ana Krulece588e312018-09-18 12:32:24 -0700167
Ady Abraham62a0be22020-12-08 16:54:10 -0800168 void onFrameRateOverridesChanged(ConnectionHandle, PhysicalDisplayId)
Andy Yu2ae6b6b2021-11-18 14:51:06 -0800169 EXCLUDES(mConnectionsLock);
Ady Abraham62f216c2020-10-13 19:07:23 -0700170
Ady Abraham9c53ee72020-07-22 21:16:18 -0700171 // Modifies work duration in the event thread.
172 void setDuration(ConnectionHandle, std::chrono::nanoseconds workDuration,
173 std::chrono::nanoseconds readyDuration);
Ana Krulec98b5b242018-08-10 15:03:23 -0700174
Dominik Laskowski14956dc2023-02-22 13:43:57 -0500175 VsyncModulator& vsyncModulator() { return *mVsyncModulator; }
Dominik Laskowski1c99a002023-01-20 17:10:36 -0500176
Leon Scroggins III1af0fb62023-03-02 14:21:44 -0500177 // In some cases, we should only modulate for the pacesetter display. In those
Leon Scroggins III67388622023-02-06 20:36:20 -0500178 // cases, the caller should pass in the relevant display, and the method
Leon Scroggins III1af0fb62023-03-02 14:21:44 -0500179 // will no-op if it's not the pacesetter. Other cases are not specific to a
Leon Scroggins III67388622023-02-06 20:36:20 -0500180 // display.
Dominik Laskowski1c99a002023-01-20 17:10:36 -0500181 template <typename... Args,
182 typename Handler = std::optional<VsyncConfig> (VsyncModulator::*)(Args...)>
Leon Scroggins III67388622023-02-06 20:36:20 -0500183 void modulateVsync(std::optional<PhysicalDisplayId> id, Handler handler, Args... args) {
184 if (id) {
185 std::scoped_lock lock(mDisplayLock);
186 ftl::FakeGuard guard(kMainThreadContext);
Leon Scroggins III1af0fb62023-03-02 14:21:44 -0500187 if (id != mPacesetterDisplayId) {
Leon Scroggins III67388622023-02-06 20:36:20 -0500188 return;
189 }
190 }
191
Dominik Laskowski1c99a002023-01-20 17:10:36 -0500192 if (const auto config = (*mVsyncModulator.*handler)(args...)) {
Leon Scroggins III1af0fb62023-03-02 14:21:44 -0500193 setVsyncConfig(*config, getPacesetterVsyncPeriod());
Dominik Laskowski1c99a002023-01-20 17:10:36 -0500194 }
195 }
196
197 void setVsyncConfigSet(const VsyncConfigSet&, Period vsyncPeriod);
198
Ady Abrahamace3d052022-11-17 16:25:05 -0800199 // Sets the render rate for the scheduler to run at.
Leon Scroggins III67388622023-02-06 20:36:20 -0500200 void setRenderRate(PhysicalDisplayId, Fps);
Ady Abrahamace3d052022-11-17 16:25:05 -0800201
Leon Scroggins III4235ea02023-04-17 15:14:20 -0400202 void enableHardwareVsync(PhysicalDisplayId) REQUIRES(kMainThreadContext);
203 void disableHardwareVsync(PhysicalDisplayId, bool disallow) REQUIRES(kMainThreadContext);
Dominik Laskowski98041832019-08-01 18:35:59 -0700204
Alec Mourif8e689c2019-05-20 18:32:22 -0700205 // Resyncs the scheduler to hardware vsync.
Leon Scroggins IIIc275df42023-02-07 16:40:21 -0500206 // If allowToEnable is true, then hardware vsync will be turned on.
Alec Mourif8e689c2019-05-20 18:32:22 -0700207 // Otherwise, if hardware vsync is not already enabled then this method will
208 // no-op.
Leon Scroggins III67388622023-02-06 20:36:20 -0500209 // If refreshRate is nullopt, use the existing refresh rate of the display.
210 void resyncToHardwareVsync(PhysicalDisplayId id, bool allowToEnable,
211 std::optional<Fps> refreshRate = std::nullopt)
212 EXCLUDES(mDisplayLock) {
213 std::scoped_lock lock(mDisplayLock);
214 ftl::FakeGuard guard(kMainThreadContext);
215 resyncToHardwareVsyncLocked(id, allowToEnable, refreshRate);
216 }
Dominik Laskowski596a2562022-10-28 11:26:12 -0400217 void resync() EXCLUDES(mDisplayLock);
Ady Abrahame9befd72022-02-24 17:24:44 -0800218 void forceNextResync() { mLastResyncTime = 0; }
Dominik Laskowski98041832019-08-01 18:35:59 -0700219
Leon Scroggins IIIc275df42023-02-07 16:40:21 -0500220 // Passes a vsync sample to VsyncController. Returns true if
221 // VsyncController detected that the vsync period changed and false
222 // otherwise.
Leon Scroggins III67388622023-02-06 20:36:20 -0500223 bool addResyncSample(PhysicalDisplayId, nsecs_t timestamp,
224 std::optional<nsecs_t> hwcVsyncPeriod);
Dominik Laskowskiec0eac22023-01-28 16:16:19 -0500225 void addPresentFence(PhysicalDisplayId, std::shared_ptr<FenceTime>)
Leon Scroggins III4235ea02023-04-17 15:14:20 -0400226 REQUIRES(kMainThreadContext);
Ady Abraham8f1ee7f2019-04-05 10:32:50 -0700227
Dominik Laskowskif7a09ed2019-10-07 13:54:18 -0700228 // Layers are registered on creation, and unregistered when the weak reference expires.
229 void registerLayer(Layer*);
Vishnu Nairef68d6d2023-02-28 06:18:27 +0000230 void recordLayerHistory(int32_t id, const LayerProps& layerProps, nsecs_t presentTime,
231 LayerHistory::LayerUpdateType) EXCLUDES(mDisplayLock);
Marin Shalamanova7fe3042021-01-29 21:02:08 +0100232 void setModeChangePending(bool pending);
Andy Labrada096227e2022-06-15 16:58:11 +0000233 void setDefaultFrameRateCompatibility(Layer*);
Ady Abrahambdda8f02021-04-01 16:06:11 -0700234 void deregisterLayer(Layer*);
Ady Abraham822ecbd2023-07-07 16:16:09 -0700235 void onLayerDestroyed(Layer*) EXCLUDES(mChoreographerLock);
Dominik Laskowskif7a09ed2019-10-07 13:54:18 -0700236
Dominik Laskowski49cea512019-11-12 14:13:23 -0800237 // Detects content using layer history, and selects a matching refresh rate.
Ady Abraham822ecbd2023-07-07 16:16:09 -0700238 void chooseRefreshRateForContent(const surfaceflinger::frontend::LayerHierarchy*,
239 bool updateAttachedChoreographer) EXCLUDES(mDisplayLock);
Ady Abraham97d04232019-03-05 19:48:12 -0800240
Ady Abraham8532d012019-05-08 14:50:56 -0700241 void resetIdleTimer();
242
Dominik Laskowski8da6b0e2021-05-12 15:34:13 -0700243 // Indicates that touch interaction is taking place.
244 void onTouchHint();
Ady Abraham8532d012019-05-08 14:50:56 -0700245
Leon Scroggins III67388622023-02-06 20:36:20 -0500246 void setDisplayPowerMode(PhysicalDisplayId, hal::PowerMode powerMode)
247 REQUIRES(kMainThreadContext);
Ady Abraham6fe2c172019-07-12 12:37:57 -0700248
Dominik Laskowskic404cb42023-03-03 19:57:53 -0500249 ConstVsyncSchedulePtr getVsyncSchedule(std::optional<PhysicalDisplayId> = std::nullopt) const
250 EXCLUDES(mDisplayLock);
251
252 VsyncSchedulePtr getVsyncSchedule(std::optional<PhysicalDisplayId> idOpt = std::nullopt)
253 EXCLUDES(mDisplayLock) {
254 return std::const_pointer_cast<VsyncSchedule>(std::as_const(*this).getVsyncSchedule(idOpt));
Leon Scroggins III67388622023-02-06 20:36:20 -0500255 }
Ady Abraham55fa7272020-09-30 19:19:27 -0700256
Dominik Laskowskiec0eac22023-01-28 16:16:19 -0500257 TimePoint expectedPresentTimeForPacesetter() const EXCLUDES(mDisplayLock) {
258 std::scoped_lock lock(mDisplayLock);
259 return pacesetterDisplayLocked()
260 .transform([](const Display& display) {
261 return display.targeterPtr->target().expectedPresentTime();
262 })
263 .value_or(TimePoint());
264 }
Dominik Laskowskib418dd72023-06-13 17:31:04 -0400265
Leon Scroggins IIIdb16a2b2023-02-06 17:50:05 -0500266 // Returns true if a given vsync timestamp is considered valid vsync
267 // for a given uid
Dominik Laskowskib418dd72023-06-13 17:31:04 -0400268 bool isVsyncValid(TimePoint expectedVsyncTime, uid_t uid) const;
Leon Scroggins IIIdb16a2b2023-02-06 17:50:05 -0500269
Leon Scroggins III67388622023-02-06 20:36:20 -0500270 bool isVsyncInPhase(TimePoint expectedVsyncTime, Fps frameRate) const;
Huihong Luo1768cb02022-10-11 11:10:34 -0700271
Dominik Laskowski03cfce82022-11-02 12:13:29 -0400272 void dump(utils::Dumper&) const;
Dominik Laskowski98041832019-08-01 18:35:59 -0700273 void dump(ConnectionHandle, std::string&) const;
Leon Scroggins III67388622023-02-06 20:36:20 -0500274 void dumpVsync(std::string&) const EXCLUDES(mDisplayLock);
Ana Krulec0c8cd522018-08-31 12:27:28 -0700275
Leon Scroggins III1af0fb62023-03-02 14:21:44 -0500276 // Returns the preferred refresh rate and frame rate for the pacesetter display.
Dominik Laskowskifc378b02022-12-02 14:56:05 -0500277 FrameRateMode getPreferredDisplayMode();
Daniel Solomon0f0ddc12019-08-19 19:31:09 -0700278
Ady Abraham3a77a7b2019-12-02 18:46:59 -0800279 // Notifies the scheduler about a refresh rate timeline change.
Peiyong Line9d809e2020-04-14 13:10:48 -0700280 void onNewVsyncPeriodChangeTimeline(const hal::VsyncPeriodChangeTimeline& timeline);
Ady Abraham3a77a7b2019-12-02 18:46:59 -0800281
Dominik Laskowskidd5827a2022-03-17 12:44:23 -0700282 // Notifies the scheduler post composition. Returns if recomposite is needed.
283 bool onPostComposition(nsecs_t presentTime);
Ady Abraham3a77a7b2019-12-02 18:46:59 -0800284
Ady Abraham8a82ba62020-01-17 12:43:17 -0800285 // Notifies the scheduler when the display size has changed. Called from SF's main thread
Ady Abraham7825c682021-05-17 15:12:14 -0700286 void onActiveDisplayAreaChanged(uint32_t displayArea);
Ady Abraham8a82ba62020-01-17 12:43:17 -0800287
Ady Abraham62a0be22020-12-08 16:54:10 -0800288 // Stores the preferred refresh rate that an app should run at.
289 // FrameRateOverride.refreshRateHz == 0 means no preference.
Andy Yu2ae6b6b2021-11-18 14:51:06 -0800290 void setPreferredRefreshRateForUid(FrameRateOverride);
291
292 void setGameModeRefreshRateForUid(FrameRateOverride);
293
Alec Mouri7d436ec2021-01-27 20:40:50 -0800294 // Retrieves the overridden refresh rate for a given uid.
Dominik Laskowski596a2562022-10-28 11:26:12 -0400295 std::optional<Fps> getFrameRateOverride(uid_t) const EXCLUDES(mDisplayLock);
Ady Abraham3efa3942021-06-24 19:01:25 -0700296
Leon Scroggins III1af0fb62023-03-02 14:21:44 -0500297 Period getPacesetterVsyncPeriod() const EXCLUDES(mDisplayLock) {
298 return pacesetterSelectorPtr()->getActiveMode().fps.getPeriod();
Ady Abraham3efa3942021-06-24 19:01:25 -0700299 }
Ady Abraham62a0be22020-12-08 16:54:10 -0800300
Leon Scroggins III1af0fb62023-03-02 14:21:44 -0500301 Fps getPacesetterRefreshRate() const EXCLUDES(mDisplayLock) {
302 return pacesetterSelectorPtr()->getActiveMode().fps;
Huihong Luo1768cb02022-10-11 11:10:34 -0700303 }
304
Nathaniel Nifong1303d912021-10-06 09:41:24 -0400305 // Returns the framerate of the layer with the given sequence ID
306 float getLayerFramerate(nsecs_t now, int32_t id) const {
307 return mLayerHistory.getLayerFramerate(now, id);
308 }
309
Ady Abraham33a386b2023-07-18 15:37:11 -0700310 bool updateFrameRateOverrides(GlobalSignals, Fps displayRefreshRate) EXCLUDES(mPolicyLock);
311
Arthur Hungc70bee22023-06-02 01:35:52 +0000312 // Returns true if the small dirty detection is enabled.
313 bool supportSmallDirtyDetection() const {
314 return mFeatures.test(Feature::kSmallDirtyContentDetection);
315 }
316
317 // Returns true if the dirty area is less than threshold.
318 bool isSmallDirtyArea(uint32_t dirtyArea) const {
319 return mLayerHistory.isSmallDirtyArea(dirtyArea);
320 }
321
Ana Krulec98b5b242018-08-10 15:03:23 -0700322private:
Ana Krulecafb45842019-02-13 13:33:03 -0800323 friend class TestableScheduler;
324
Dominik Laskowskidd252cd2019-07-26 09:10:16 -0700325 enum class ContentDetectionState { Off, On };
326 enum class TimerState { Reset, Expired };
327 enum class TouchState { Inactive, Active };
Ana Krulecfefd6ae2019-02-13 17:53:08 -0800328
Dominik Laskowski08fbd852022-07-14 08:53:42 -0700329 // impl::MessageQueue overrides:
Dominik Laskowskiec0eac22023-01-28 16:16:19 -0500330 void onFrameSignal(ICompositor&, VsyncId, TimePoint expectedVsyncTime) override
331 REQUIRES(kMainThreadContext, mDisplayLock);
Dominik Laskowski08fbd852022-07-14 08:53:42 -0700332
Dominik Laskowski6505f792019-09-18 11:10:05 -0700333 // Create a connection on the given EventThread.
Steven Thomas2bbaabe2019-08-28 16:08:35 -0700334 ConnectionHandle createConnection(std::unique_ptr<EventThread>);
Dominik Laskowskiccf37d72019-02-01 16:47:58 -0800335
Dominik Laskowski3a80a382019-07-25 11:16:07 -0700336 // Update feature state machine to given state when corresponding timer resets or expires.
Dominik Laskowski596a2562022-10-28 11:26:12 -0400337 void kernelIdleTimerCallback(TimerState) EXCLUDES(mDisplayLock);
Dominik Laskowski3a80a382019-07-25 11:16:07 -0700338 void idleTimerCallback(TimerState);
339 void touchTimerCallback(TimerState);
340 void displayPowerTimerCallback(TimerState);
341
Dominik Laskowski66295432023-03-14 12:25:36 -0400342 // VsyncSchedule delegate.
343 void onHardwareVsyncRequest(PhysicalDisplayId, bool enable);
344
Leon Scroggins III67388622023-02-06 20:36:20 -0500345 void resyncToHardwareVsyncLocked(PhysicalDisplayId, bool allowToEnable,
346 std::optional<Fps> refreshRate = std::nullopt)
347 REQUIRES(kMainThreadContext, mDisplayLock);
348 void resyncAllToHardwareVsync(bool allowToEnable) EXCLUDES(mDisplayLock);
Dominik Laskowski1c99a002023-01-20 17:10:36 -0500349 void setVsyncConfig(const VsyncConfig&, Period vsyncPeriod);
Dominik Laskowski98041832019-08-01 18:35:59 -0700350
Leon Scroggins III1af0fb62023-03-02 14:21:44 -0500351 // Chooses a pacesetter among the registered displays, unless `pacesetterIdOpt` is specified.
352 // The new `mPacesetterDisplayId` is never `std::nullopt`.
353 void promotePacesetterDisplay(std::optional<PhysicalDisplayId> pacesetterIdOpt = std::nullopt)
Leon Scroggins III6fc45192023-03-16 12:13:28 -0400354 REQUIRES(kMainThreadContext) EXCLUDES(mDisplayLock);
355
356 // Changes to the displays (e.g. registering and unregistering) must be made
357 // while mDisplayLock is locked, and the new pacesetter then must be promoted while
358 // mDisplayLock is still locked. However, a new pacesetter means that
359 // MessageQueue and EventThread need to use the new pacesetter's
360 // VsyncSchedule, and this must happen while mDisplayLock is *not* locked,
361 // or else we may deadlock with EventThread.
362 std::shared_ptr<VsyncSchedule> promotePacesetterDisplayLocked(
363 std::optional<PhysicalDisplayId> pacesetterIdOpt = std::nullopt)
Dominik Laskowski596a2562022-10-28 11:26:12 -0400364 REQUIRES(kMainThreadContext, mDisplayLock);
Leon Scroggins39d25342023-04-19 17:11:01 +0000365 void applyNewVsyncSchedule(std::shared_ptr<VsyncSchedule>) EXCLUDES(mDisplayLock);
Dominik Laskowski596a2562022-10-28 11:26:12 -0400366
Leon Scroggins III1af0fb62023-03-02 14:21:44 -0500367 // Blocks until the pacesetter's idle timer thread exits. `mDisplayLock` must not be locked by
368 // the caller on the main thread to avoid deadlock, since the timer thread locks it before exit.
369 void demotePacesetterDisplay() REQUIRES(kMainThreadContext) EXCLUDES(mDisplayLock, mPolicyLock);
Dominik Laskowski596a2562022-10-28 11:26:12 -0400370
Dominik Laskowskic404cb42023-03-03 19:57:53 -0500371 void registerDisplayInternal(PhysicalDisplayId, RefreshRateSelectorPtr, VsyncSchedulePtr)
372 REQUIRES(kMainThreadContext) EXCLUDES(mDisplayLock);
Leon Scroggins III67388622023-02-06 20:36:20 -0500373
Dominik Laskowski0c41ffa2021-12-24 16:45:12 -0800374 struct Policy;
375
376 // Sets the S state of the policy to the T value under mPolicyLock, and chooses a display mode
377 // that fulfills the new policy if the state changed. Returns the signals that were considered.
378 template <typename S, typename T>
379 GlobalSignals applyPolicy(S Policy::*, T&&) EXCLUDES(mPolicyLock);
380
Dominik Laskowski530d6bd2022-10-10 16:55:54 -0400381 struct DisplayModeChoice {
Ady Abrahamace3d052022-11-17 16:25:05 -0800382 DisplayModeChoice(FrameRateMode mode, GlobalSignals consideredSignals)
383 : mode(std::move(mode)), consideredSignals(consideredSignals) {}
Dominik Laskowski530d6bd2022-10-10 16:55:54 -0400384
Ady Abrahamace3d052022-11-17 16:25:05 -0800385 FrameRateMode mode;
Dominik Laskowski530d6bd2022-10-10 16:55:54 -0400386 GlobalSignals consideredSignals;
387
388 bool operator==(const DisplayModeChoice& other) const {
Ady Abrahamace3d052022-11-17 16:25:05 -0800389 return mode == other.mode && consideredSignals == other.consideredSignals;
Dominik Laskowski530d6bd2022-10-10 16:55:54 -0400390 }
391
392 // For tests.
393 friend std::ostream& operator<<(std::ostream& stream, const DisplayModeChoice& choice) {
Ady Abrahamace3d052022-11-17 16:25:05 -0800394 return stream << '{' << to_string(*choice.mode.modePtr) << " considering "
Dominik Laskowski530d6bd2022-10-10 16:55:54 -0400395 << choice.consideredSignals.toString().c_str() << '}';
396 }
397 };
398
Dominik Laskowski6b049ff2023-01-29 15:46:45 -0500399 using DisplayModeChoiceMap = ui::PhysicalDisplayMap<PhysicalDisplayId, DisplayModeChoice>;
Dominik Laskowski596a2562022-10-28 11:26:12 -0400400
401 // See mDisplayLock for thread safety.
402 DisplayModeChoiceMap chooseDisplayModes() const
403 REQUIRES(mPolicyLock, mDisplayLock, kMainThreadContext);
ramindani69b58e82022-09-26 16:48:36 -0700404
Dominik Laskowski95df6a12022-10-07 18:11:07 -0400405 GlobalSignals makeGlobalSignals() const REQUIRES(mPolicyLock);
ramindani69b58e82022-09-26 16:48:36 -0700406
Ady Abraham33a386b2023-07-18 15:37:11 -0700407 bool updateFrameRateOverridesLocked(GlobalSignals, Fps displayRefreshRate)
408 REQUIRES(mPolicyLock);
Ady Abraham822ecbd2023-07-07 16:16:09 -0700409 void updateAttachedChoreographers(const surfaceflinger::frontend::LayerHierarchy&,
410 Fps displayRefreshRate);
411 int updateAttachedChoreographersInternal(const surfaceflinger::frontend::LayerHierarchy&,
412 Fps displayRefreshRate, int parentDivisor);
413 void updateAttachedChoreographersFrameRate(const surfaceflinger::frontend::RequestedLayerState&,
414 Fps fps) EXCLUDES(mChoreographerLock);
Ana Krulec7ab56032018-11-02 20:51:06 +0100415
Dominik Laskowski596a2562022-10-28 11:26:12 -0400416 void dispatchCachedReportedMode() REQUIRES(mPolicyLock) EXCLUDES(mDisplayLock);
Ady Abraham62a0be22020-12-08 16:54:10 -0800417
Leon Scroggins IIIdb16a2b2023-02-06 17:50:05 -0500418 android::impl::EventThread::ThrottleVsyncCallback makeThrottleVsyncCallback() const;
419 android::impl::EventThread::GetVsyncPeriodFunction makeGetVsyncPeriodFunction() const;
420
Dominik Laskowski98041832019-08-01 18:35:59 -0700421 // Stores EventThread associated with a given VSyncSource, and an initial EventThreadConnection.
422 struct Connection {
423 sp<EventThreadConnection> connection;
424 std::unique_ptr<EventThread> thread;
425 };
Ady Abraham09bd3922019-04-08 10:44:56 -0700426
Dominik Laskowski98041832019-08-01 18:35:59 -0700427 ConnectionHandle::Id mNextConnectionHandleId = 0;
Ana Krulec6ddd2612020-09-24 13:06:33 -0700428 mutable std::mutex mConnectionsLock;
429 std::unordered_map<ConnectionHandle, Connection> mConnections GUARDED_BY(mConnectionsLock);
Ana Krulece588e312018-09-18 12:32:24 -0700430
Dominik Laskowski1c99a002023-01-20 17:10:36 -0500431 ConnectionHandle mAppConnectionHandle;
432 ConnectionHandle mSfConnectionHandle;
433
Steven Thomas2bbaabe2019-08-28 16:08:35 -0700434 std::atomic<nsecs_t> mLastResyncTime = 0;
Ana Krulece588e312018-09-18 12:32:24 -0700435
Dominik Laskowski068173d2021-08-11 17:22:59 -0700436 const FeatureFlags mFeatures;
Ana Krulec7ab56032018-11-02 20:51:06 +0100437
Dominik Laskowski1c99a002023-01-20 17:10:36 -0500438 // Shifts the VSYNC phase during certain transactions and refresh rate changes.
439 const sp<VsyncModulator> mVsyncModulator;
440
Dominik Laskowski49cea512019-11-12 14:13:23 -0800441 // Used to choose refresh rate if content detection is enabled.
Dominik Laskowski9c93d602021-10-07 19:38:26 -0700442 LayerHistory mLayerHistory;
Ana Krulecfb772822018-11-30 10:44:07 +0100443
Ady Abraham8532d012019-05-08 14:50:56 -0700444 // Timer used to monitor touch events.
Dominik Laskowski03cfce82022-11-02 12:13:29 -0400445 ftl::Optional<OneShotTimer> mTouchTimer;
Ady Abraham6fe2c172019-07-12 12:37:57 -0700446 // Timer used to monitor display power mode.
Dominik Laskowski03cfce82022-11-02 12:13:29 -0400447 ftl::Optional<OneShotTimer> mDisplayPowerTimer;
Ady Abraham6fe2c172019-07-12 12:37:57 -0700448
Ady Abraham3a77a7b2019-12-02 18:46:59 -0800449 ISchedulerCallback& mSchedulerCallback;
Ana Krulecfefd6ae2019-02-13 17:53:08 -0800450
Dominik Laskowski596a2562022-10-28 11:26:12 -0400451 // mDisplayLock may be locked while under mPolicyLock.
Dominik Laskowski068173d2021-08-11 17:22:59 -0700452 mutable std::mutex mPolicyLock;
Dominik Laskowskidd252cd2019-07-26 09:10:16 -0700453
Dominik Laskowski596a2562022-10-28 11:26:12 -0400454 // Only required for reads outside kMainThreadContext. kMainThreadContext is the only writer, so
455 // must lock for writes but not reads. See also mPolicyLock for locking order.
456 mutable std::mutex mDisplayLock;
457
Dominik Laskowskiec0eac22023-01-28 16:16:19 -0500458 using FrameTargeterPtr = std::unique_ptr<FrameTargeter>;
459
Dominik Laskowskic404cb42023-03-03 19:57:53 -0500460 struct Display {
Dominik Laskowskiec0eac22023-01-28 16:16:19 -0500461 Display(PhysicalDisplayId displayId, RefreshRateSelectorPtr selectorPtr,
462 VsyncSchedulePtr schedulePtr, FeatureFlags features)
463 : displayId(displayId),
464 selectorPtr(std::move(selectorPtr)),
465 schedulePtr(std::move(schedulePtr)),
466 targeterPtr(std::make_unique<
467 FrameTargeter>(displayId,
468 features.test(Feature::kBackpressureGpuComposition))) {}
469
470 const PhysicalDisplayId displayId;
Dominik Laskowski596a2562022-10-28 11:26:12 -0400471
Dominik Laskowskic404cb42023-03-03 19:57:53 -0500472 // Effectively const except in move constructor.
473 RefreshRateSelectorPtr selectorPtr;
474 VsyncSchedulePtr schedulePtr;
Dominik Laskowskiec0eac22023-01-28 16:16:19 -0500475 FrameTargeterPtr targeterPtr;
Dominik Laskowski66295432023-03-14 12:25:36 -0400476
477 hal::PowerMode powerMode = hal::PowerMode::OFF;
Dominik Laskowskic404cb42023-03-03 19:57:53 -0500478 };
479
480 using DisplayRef = std::reference_wrapper<Display>;
481 using ConstDisplayRef = std::reference_wrapper<const Display>;
482
Dominik Laskowski6b049ff2023-01-29 15:46:45 -0500483 ui::PhysicalDisplayMap<PhysicalDisplayId, Display> mDisplays GUARDED_BY(mDisplayLock)
Dominik Laskowskic404cb42023-03-03 19:57:53 -0500484 GUARDED_BY(kMainThreadContext);
Leon Scroggins III67388622023-02-06 20:36:20 -0500485
Leon Scroggins III1af0fb62023-03-02 14:21:44 -0500486 ftl::Optional<PhysicalDisplayId> mPacesetterDisplayId GUARDED_BY(mDisplayLock)
Dominik Laskowski596a2562022-10-28 11:26:12 -0400487 GUARDED_BY(kMainThreadContext);
488
Dominik Laskowskic404cb42023-03-03 19:57:53 -0500489 ftl::Optional<DisplayRef> pacesetterDisplayLocked() REQUIRES(mDisplayLock) {
490 return static_cast<const Scheduler*>(this)->pacesetterDisplayLocked().transform(
491 [](const Display& display) { return std::ref(const_cast<Display&>(display)); });
492 }
493
494 ftl::Optional<ConstDisplayRef> pacesetterDisplayLocked() const REQUIRES(mDisplayLock) {
495 ftl::FakeGuard guard(kMainThreadContext);
496 return mPacesetterDisplayId.and_then([this](PhysicalDisplayId pacesetterId)
497 REQUIRES(mDisplayLock, kMainThreadContext) {
498 return mDisplays.get(pacesetterId);
499 });
500 }
501
Leon Scroggins III1af0fb62023-03-02 14:21:44 -0500502 RefreshRateSelectorPtr pacesetterSelectorPtr() const EXCLUDES(mDisplayLock) {
Dominik Laskowski596a2562022-10-28 11:26:12 -0400503 std::scoped_lock lock(mDisplayLock);
Leon Scroggins III1af0fb62023-03-02 14:21:44 -0500504 return pacesetterSelectorPtrLocked();
Dominik Laskowski596a2562022-10-28 11:26:12 -0400505 }
506
Leon Scroggins III1af0fb62023-03-02 14:21:44 -0500507 RefreshRateSelectorPtr pacesetterSelectorPtrLocked() const REQUIRES(mDisplayLock) {
Dominik Laskowski596a2562022-10-28 11:26:12 -0400508 ftl::FakeGuard guard(kMainThreadContext);
Dominik Laskowskic404cb42023-03-03 19:57:53 -0500509 return pacesetterDisplayLocked()
510 .transform([](const Display& display) { return display.selectorPtr; })
511 .or_else([] { return std::optional<RefreshRateSelectorPtr>(nullptr); })
Dominik Laskowski80872bd2022-11-15 11:34:33 -0500512 .value();
Dominik Laskowski596a2562022-10-28 11:26:12 -0400513 }
ramindani69b58e82022-09-26 16:48:36 -0700514
Dominik Laskowskic404cb42023-03-03 19:57:53 -0500515 ConstVsyncSchedulePtr getVsyncScheduleLocked(
516 std::optional<PhysicalDisplayId> = std::nullopt) const REQUIRES(mDisplayLock);
517
518 VsyncSchedulePtr getVsyncScheduleLocked(std::optional<PhysicalDisplayId> idOpt = std::nullopt)
519 REQUIRES(mDisplayLock) {
Leon Scroggins III67388622023-02-06 20:36:20 -0500520 return std::const_pointer_cast<VsyncSchedule>(
521 static_cast<const Scheduler*>(this)->getVsyncScheduleLocked(idOpt));
522 }
523
Dominik Laskowski0c41ffa2021-12-24 16:45:12 -0800524 struct Policy {
Dominik Laskowski068173d2021-08-11 17:22:59 -0700525 // Policy for choosing the display mode.
526 LayerHistory::Summary contentRequirements;
Dominik Laskowskidd252cd2019-07-26 09:10:16 -0700527 TimerState idleTimer = TimerState::Reset;
528 TouchState touch = TouchState::Inactive;
529 TimerState displayPowerTimer = TimerState::Expired;
Rachel Lee6a9731d2022-06-06 17:08:14 -0700530 hal::PowerMode displayPowerMode = hal::PowerMode::ON;
Ady Abrahamdfd62162020-06-10 16:11:56 -0700531
Dominik Laskowski068173d2021-08-11 17:22:59 -0700532 // Chosen display mode.
Ady Abrahamace3d052022-11-17 16:25:05 -0800533 ftl::Optional<FrameRateMode> modeOpt;
Dominik Laskowski068173d2021-08-11 17:22:59 -0700534
Marin Shalamanova7fe3042021-01-29 21:02:08 +0100535 struct ModeChangedParams {
Ady Abrahamdfd62162020-06-10 16:11:56 -0700536 ConnectionHandle handle;
Ady Abrahamace3d052022-11-17 16:25:05 -0800537 FrameRateMode mode;
Ady Abrahamdfd62162020-06-10 16:11:56 -0700538 };
539
Dominik Laskowski068173d2021-08-11 17:22:59 -0700540 // Parameters for latest dispatch of mode change event.
Marin Shalamanova7fe3042021-01-29 21:02:08 +0100541 std::optional<ModeChangedParams> cachedModeChangedParams;
Dominik Laskowski068173d2021-08-11 17:22:59 -0700542 } mPolicy GUARDED_BY(mPolicyLock);
Ady Abraham09bd3922019-04-08 10:44:56 -0700543
Ady Abraham822ecbd2023-07-07 16:16:09 -0700544 std::mutex mChoreographerLock;
545
546 struct AttachedChoreographers {
547 Fps frameRate;
548 std::unordered_set<wp<EventThreadConnection>, WpHash> connections;
549 };
550 // Map keyed by layer ID (sequence) to choreographer connections.
551 std::unordered_map<int32_t, AttachedChoreographers> mAttachedChoreographers
552 GUARDED_BY(mChoreographerLock);
553
Ady Abraham3a77a7b2019-12-02 18:46:59 -0800554 std::mutex mVsyncTimelineLock;
Peiyong Line9d809e2020-04-14 13:10:48 -0700555 std::optional<hal::VsyncPeriodChangeTimeline> mLastVsyncPeriodChangeTimeline
Ady Abraham3a77a7b2019-12-02 18:46:59 -0800556 GUARDED_BY(mVsyncTimelineLock);
557 static constexpr std::chrono::nanoseconds MAX_VSYNC_APPLIED_TIME = 200ms;
Ady Abraham8735eac2020-08-12 16:35:04 -0700558
Andy Yu2ae6b6b2021-11-18 14:51:06 -0800559 FrameRateOverrideMappings mFrameRateOverrideMappings;
Ana Krulec98b5b242018-08-10 15:03:23 -0700560};
561
Dominik Laskowski068173d2021-08-11 17:22:59 -0700562} // namespace scheduler
Ana Krulec7ab56032018-11-02 20:51:06 +0100563} // namespace android