blob: 36280e3888f76d439af61ace5967cd122db9fcdd [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 Laskowski5d164f22022-07-07 07:56:07 -070038#include <scheduler/Time.h>
Dominik Laskowski01602522022-10-07 19:02:28 -040039#include <ui/DisplayId.h>
Dominik Laskowski068173d2021-08-11 17:22:59 -070040
ramindani69b58e82022-09-26 16:48:36 -070041#include "Display/DisplayMap.h"
Dominik Laskowski530d6bd2022-10-10 16:55:54 -040042#include "Display/DisplayModeRequest.h"
Ana Krulec98b5b242018-08-10 15:03:23 -070043#include "EventThread.h"
Andy Yu2ae6b6b2021-11-18 14:51:06 -080044#include "FrameRateOverrideMappings.h"
Ana Krulec3084c052018-11-21 20:27:17 +010045#include "LayerHistory.h"
Dominik Laskowski756b7892021-08-04 12:53:59 -070046#include "MessageQueue.h"
Ana Krulecf2c006d2019-06-21 15:37:07 -070047#include "OneShotTimer.h"
Dominik Laskowskid82e0f02022-10-26 15:23:04 -040048#include "RefreshRateSelector.h"
Dominik Laskowski03cfce82022-11-02 12:13:29 -040049#include "Utils/Dumper.h"
Dominik Laskowski068173d2021-08-11 17:22:59 -070050#include "VsyncSchedule.h"
Ana Krulec98b5b242018-08-10 15:03:23 -070051
Dominik Laskowski62eff352021-12-06 09:59:41 -080052namespace android::scheduler {
53
54// Opaque handle to scheduler connection.
55struct ConnectionHandle {
56 using Id = std::uintptr_t;
57 static constexpr Id INVALID_ID = static_cast<Id>(-1);
58
59 Id id = INVALID_ID;
60
61 explicit operator bool() const { return id != INVALID_ID; }
62};
63
64inline bool operator==(ConnectionHandle lhs, ConnectionHandle rhs) {
65 return lhs.id == rhs.id;
66}
67
68} // namespace android::scheduler
69
70namespace std {
71
72template <>
73struct hash<android::scheduler::ConnectionHandle> {
74 size_t operator()(android::scheduler::ConnectionHandle handle) const {
75 return hash<android::scheduler::ConnectionHandle::Id>()(handle.id);
76 }
77};
78
79} // namespace std
80
Ana Krulec98b5b242018-08-10 15:03:23 -070081namespace android {
82
Dominik Laskowski98041832019-08-01 18:35:59 -070083class FenceTime;
Ana Krulece588e312018-09-18 12:32:24 -070084
Adithya Srinivasan5f683cf2020-09-15 14:21:04 -070085namespace frametimeline {
86class TokenManager;
87} // namespace frametimeline
88
Dominik Laskowski068173d2021-08-11 17:22:59 -070089namespace scheduler {
90
Dominik Laskowskid82e0f02022-10-26 15:23:04 -040091using GlobalSignals = RefreshRateSelector::GlobalSignals;
ramindani69b58e82022-09-26 16:48:36 -070092
Dominik Laskowski8b01cc02020-07-14 19:02:41 -070093struct ISchedulerCallback {
94 virtual void setVsyncEnabled(bool) = 0;
Dominik Laskowski530d6bd2022-10-10 16:55:54 -040095 virtual void requestDisplayModes(std::vector<display::DisplayModeRequest>) = 0;
Ady Abrahama09852a2020-02-20 14:23:42 -080096 virtual void kernelTimerChanged(bool expired) = 0;
Ady Abraham62a0be22020-12-08 16:54:10 -080097 virtual void triggerOnFrameRateOverridesChanged() = 0;
Dominik Laskowski8b01cc02020-07-14 19:02:41 -070098
99protected:
100 ~ISchedulerCallback() = default;
Ady Abraham3a77a7b2019-12-02 18:46:59 -0800101};
102
Patrick Williams83f36b22022-09-14 17:57:35 +0000103class Scheduler : android::impl::MessageQueue {
104 using Impl = android::impl::MessageQueue;
Dominik Laskowski756b7892021-08-04 12:53:59 -0700105
Ana Krulec98b5b242018-08-10 15:03:23 -0700106public:
Dominik Laskowski068173d2021-08-11 17:22:59 -0700107 Scheduler(ICompositor&, ISchedulerCallback&, FeatureFlags);
Dominik Laskowski83bd7712022-01-07 14:30:53 -0800108 virtual ~Scheduler();
109
110 void startTimers();
Dominik Laskowski83bd7712022-01-07 14:30:53 -0800111
Dominik Laskowski596a2562022-10-28 11:26:12 -0400112 // TODO(b/241285191): Remove this API by promoting leader in onScreen{Acquired,Released}.
113 void setLeaderDisplay(std::optional<PhysicalDisplayId>) REQUIRES(kMainThreadContext)
114 EXCLUDES(mDisplayLock);
Dominik Laskowskib5a094b2022-10-27 12:00:12 -0400115
Dominik Laskowski596a2562022-10-28 11:26:12 -0400116 using RefreshRateSelectorPtr = std::shared_ptr<RefreshRateSelector>;
117
118 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 Laskowski068173d2021-08-11 17:22:59 -0700124 void createVsyncSchedule(FeatureFlags);
Dominik Laskowski756b7892021-08-04 12:53:59 -0700125
126 using Impl::initVsync;
Dominik Laskowski756b7892021-08-04 12:53:59 -0700127
128 using Impl::getScheduledFrameTime;
129 using Impl::setDuration;
130
Dominik Laskowskif11728a2022-07-28 13:07:42 -0700131 using Impl::scheduleConfigure;
Dominik Laskowski46f3e3b2021-08-10 11:44:24 -0700132 using Impl::scheduleFrame;
Dominik Laskowski756b7892021-08-04 12:53:59 -0700133
134 // Schedule an asynchronous or synchronous task on the main thread.
135 template <typename F, typename T = std::invoke_result_t<F>>
136 [[nodiscard]] std::future<T> schedule(F&& f) {
137 auto [task, future] = makeTask(std::move(f));
138 postMessage(std::move(task));
139 return std::move(future);
140 }
Dominik Laskowski9c93d602021-10-07 19:38:26 -0700141
Chavi Weingarten076acac2023-01-19 17:20:43 +0000142 template <typename F, typename T = std::invoke_result_t<F>>
143 [[nodiscard]] std::future<T> scheduleDelayed(F&& f, nsecs_t uptimeDelay) {
144 auto [task, future] = makeTask(std::move(f));
145 postMessageDelayed(std::move(task), uptimeDelay);
146 return std::move(future);
147 }
148
Adithya Srinivasan5f683cf2020-09-15 14:21:04 -0700149 ConnectionHandle createConnection(const char* connectionName, frametimeline::TokenManager*,
Ady Abraham9c53ee72020-07-22 21:16:18 -0700150 std::chrono::nanoseconds workDuration,
Huihong Luoab8ffef2022-08-18 13:02:26 -0700151 std::chrono::nanoseconds readyDuration);
Ana Krulec98b5b242018-08-10 15:03:23 -0700152
Ady Abraham62f216c2020-10-13 19:07:23 -0700153 sp<IDisplayEventConnection> createDisplayEventConnection(
Huihong Luo1b0c49f2022-03-15 19:18:21 -0700154 ConnectionHandle, EventRegistrationFlags eventRegistration = {});
Ana Krulece588e312018-09-18 12:32:24 -0700155
Dominik Laskowski98041832019-08-01 18:35:59 -0700156 sp<EventThreadConnection> getEventConnection(ConnectionHandle);
Kevin DuBois413287f2019-02-25 08:46:47 -0800157
Dominik Laskowski98041832019-08-01 18:35:59 -0700158 void onHotplugReceived(ConnectionHandle, PhysicalDisplayId, bool connected);
Ady Abrahamace3d052022-11-17 16:25:05 -0800159 void onPrimaryDisplayModeChanged(ConnectionHandle, const FrameRateMode&) EXCLUDES(mPolicyLock);
160 void onNonPrimaryDisplayModeChanged(ConnectionHandle, const FrameRateMode&);
Dominik Laskowski98041832019-08-01 18:35:59 -0700161 void onScreenAcquired(ConnectionHandle);
162 void onScreenReleased(ConnectionHandle);
Ana Krulece588e312018-09-18 12:32:24 -0700163
Ady Abraham62a0be22020-12-08 16:54:10 -0800164 void onFrameRateOverridesChanged(ConnectionHandle, PhysicalDisplayId)
Andy Yu2ae6b6b2021-11-18 14:51:06 -0800165 EXCLUDES(mConnectionsLock);
Ady Abraham62f216c2020-10-13 19:07:23 -0700166
Ady Abraham9c53ee72020-07-22 21:16:18 -0700167 // Modifies work duration in the event thread.
168 void setDuration(ConnectionHandle, std::chrono::nanoseconds workDuration,
169 std::chrono::nanoseconds readyDuration);
Ana Krulec98b5b242018-08-10 15:03:23 -0700170
Ady Abrahamace3d052022-11-17 16:25:05 -0800171 // Sets the render rate for the scheduler to run at.
172 void setRenderRate(Fps);
173
Ana Krulece588e312018-09-18 12:32:24 -0700174 void enableHardwareVsync();
175 void disableHardwareVsync(bool makeUnavailable);
Dominik Laskowski98041832019-08-01 18:35:59 -0700176
Alec Mourif8e689c2019-05-20 18:32:22 -0700177 // Resyncs the scheduler to hardware vsync.
178 // If makeAvailable is true, then hardware vsync will be turned on.
179 // Otherwise, if hardware vsync is not already enabled then this method will
180 // no-op.
Dominik Laskowskib0054a22022-03-03 09:03:06 -0800181 void resyncToHardwareVsync(bool makeAvailable, Fps refreshRate);
Dominik Laskowski596a2562022-10-28 11:26:12 -0400182 void resync() EXCLUDES(mDisplayLock);
Ady Abrahame9befd72022-02-24 17:24:44 -0800183 void forceNextResync() { mLastResyncTime = 0; }
Dominik Laskowski98041832019-08-01 18:35:59 -0700184
Ady Abraham8cb21882020-08-26 18:22:05 -0700185 // Passes a vsync sample to VsyncController. periodFlushed will be true if
186 // VsyncController detected that the vsync period changed, and false otherwise.
Ady Abraham5dee2f12020-02-05 17:49:47 -0800187 void addResyncSample(nsecs_t timestamp, std::optional<nsecs_t> hwcVsyncPeriod,
188 bool* periodFlushed);
Dominik Laskowski068173d2021-08-11 17:22:59 -0700189 void addPresentFence(std::shared_ptr<FenceTime>);
Ady Abraham8f1ee7f2019-04-05 10:32:50 -0700190
Dominik Laskowskif7a09ed2019-10-07 13:54:18 -0700191 // Layers are registered on creation, and unregistered when the weak reference expires.
192 void registerLayer(Layer*);
Dominik Laskowski596a2562022-10-28 11:26:12 -0400193 void recordLayerHistory(Layer*, nsecs_t presentTime, LayerHistory::LayerUpdateType)
194 EXCLUDES(mDisplayLock);
Marin Shalamanova7fe3042021-01-29 21:02:08 +0100195 void setModeChangePending(bool pending);
Andy Labrada096227e2022-06-15 16:58:11 +0000196 void setDefaultFrameRateCompatibility(Layer*);
Ady Abrahambdda8f02021-04-01 16:06:11 -0700197 void deregisterLayer(Layer*);
Dominik Laskowskif7a09ed2019-10-07 13:54:18 -0700198
Dominik Laskowski49cea512019-11-12 14:13:23 -0800199 // Detects content using layer history, and selects a matching refresh rate.
Dominik Laskowski596a2562022-10-28 11:26:12 -0400200 void chooseRefreshRateForContent() EXCLUDES(mDisplayLock);
Ady Abraham97d04232019-03-05 19:48:12 -0800201
Ady Abraham8532d012019-05-08 14:50:56 -0700202 void resetIdleTimer();
203
Dominik Laskowski8da6b0e2021-05-12 15:34:13 -0700204 // Indicates that touch interaction is taking place.
205 void onTouchHint();
Ady Abraham8532d012019-05-08 14:50:56 -0700206
Rachel Lee6a9731d2022-06-06 17:08:14 -0700207 void setDisplayPowerMode(hal::PowerMode powerMode);
Ady Abraham6fe2c172019-07-12 12:37:57 -0700208
Dominik Laskowski5d164f22022-07-07 07:56:07 -0700209 VsyncSchedule& getVsyncSchedule() { return *mVsyncSchedule; }
Ady Abraham55fa7272020-09-30 19:19:27 -0700210
Ady Abraham0bb6a472020-10-12 10:22:13 -0700211 // Returns true if a given vsync timestamp is considered valid vsync
212 // for a given uid
Dominik Laskowski08fbd852022-07-14 08:53:42 -0700213 bool isVsyncValid(TimePoint expectedVsyncTimestamp, uid_t uid) const;
Ady Abraham0bb6a472020-10-12 10:22:13 -0700214
Dominik Laskowski03cfce82022-11-02 12:13:29 -0400215 void dump(utils::Dumper&) const;
Dominik Laskowski98041832019-08-01 18:35:59 -0700216 void dump(ConnectionHandle, std::string&) const;
Ady Abraham8cb21882020-08-26 18:22:05 -0700217 void dumpVsync(std::string&) const;
Ana Krulec0c8cd522018-08-31 12:27:28 -0700218
Dominik Laskowskifc378b02022-12-02 14:56:05 -0500219 // Returns the preferred refresh rate and frame rate for the leader display.
220 FrameRateMode getPreferredDisplayMode();
Daniel Solomon0f0ddc12019-08-19 19:31:09 -0700221
Ady Abraham3a77a7b2019-12-02 18:46:59 -0800222 // Notifies the scheduler about a refresh rate timeline change.
Peiyong Line9d809e2020-04-14 13:10:48 -0700223 void onNewVsyncPeriodChangeTimeline(const hal::VsyncPeriodChangeTimeline& timeline);
Ady Abraham3a77a7b2019-12-02 18:46:59 -0800224
Dominik Laskowskidd5827a2022-03-17 12:44:23 -0700225 // Notifies the scheduler post composition. Returns if recomposite is needed.
226 bool onPostComposition(nsecs_t presentTime);
Ady Abraham3a77a7b2019-12-02 18:46:59 -0800227
Ady Abraham8a82ba62020-01-17 12:43:17 -0800228 // Notifies the scheduler when the display size has changed. Called from SF's main thread
Ady Abraham7825c682021-05-17 15:12:14 -0700229 void onActiveDisplayAreaChanged(uint32_t displayArea);
Ady Abraham8a82ba62020-01-17 12:43:17 -0800230
Alec Mouri717bcb62020-02-10 17:07:19 -0800231 size_t getEventThreadConnectionCount(ConnectionHandle handle);
232
Ady Abraham62a0be22020-12-08 16:54:10 -0800233 // Stores the preferred refresh rate that an app should run at.
234 // FrameRateOverride.refreshRateHz == 0 means no preference.
Andy Yu2ae6b6b2021-11-18 14:51:06 -0800235 void setPreferredRefreshRateForUid(FrameRateOverride);
236
237 void setGameModeRefreshRateForUid(FrameRateOverride);
238
Alec Mouri7d436ec2021-01-27 20:40:50 -0800239 // Retrieves the overridden refresh rate for a given uid.
Dominik Laskowski596a2562022-10-28 11:26:12 -0400240 std::optional<Fps> getFrameRateOverride(uid_t) const EXCLUDES(mDisplayLock);
Ady Abraham3efa3942021-06-24 19:01:25 -0700241
Dominik Laskowski596a2562022-10-28 11:26:12 -0400242 nsecs_t getLeaderVsyncPeriod() const EXCLUDES(mDisplayLock) {
Ady Abrahamace3d052022-11-17 16:25:05 -0800243 return leaderSelectorPtr()->getActiveMode().fps.getPeriodNsecs();
Ady Abraham3efa3942021-06-24 19:01:25 -0700244 }
Ady Abraham62a0be22020-12-08 16:54:10 -0800245
Nathaniel Nifong1303d912021-10-06 09:41:24 -0400246 // Returns the framerate of the layer with the given sequence ID
247 float getLayerFramerate(nsecs_t now, int32_t id) const {
248 return mLayerHistory.getLayerFramerate(now, id);
249 }
250
Ana Krulec98b5b242018-08-10 15:03:23 -0700251private:
Ana Krulecafb45842019-02-13 13:33:03 -0800252 friend class TestableScheduler;
253
Dominik Laskowskidd252cd2019-07-26 09:10:16 -0700254 enum class ContentDetectionState { Off, On };
255 enum class TimerState { Reset, Expired };
256 enum class TouchState { Inactive, Active };
Ana Krulecfefd6ae2019-02-13 17:53:08 -0800257
Dominik Laskowski08fbd852022-07-14 08:53:42 -0700258 // impl::MessageQueue overrides:
259 void onFrameSignal(ICompositor&, VsyncId, TimePoint expectedVsyncTime) override;
260
Dominik Laskowski6505f792019-09-18 11:10:05 -0700261 // Create a connection on the given EventThread.
Steven Thomas2bbaabe2019-08-28 16:08:35 -0700262 ConnectionHandle createConnection(std::unique_ptr<EventThread>);
Ady Abraham62f216c2020-10-13 19:07:23 -0700263 sp<EventThreadConnection> createConnectionInternal(
Huihong Luo1b0c49f2022-03-15 19:18:21 -0700264 EventThread*, EventRegistrationFlags eventRegistration = {});
Dominik Laskowskiccf37d72019-02-01 16:47:58 -0800265
Dominik Laskowski3a80a382019-07-25 11:16:07 -0700266 // Update feature state machine to given state when corresponding timer resets or expires.
Dominik Laskowski596a2562022-10-28 11:26:12 -0400267 void kernelIdleTimerCallback(TimerState) EXCLUDES(mDisplayLock);
Dominik Laskowski3a80a382019-07-25 11:16:07 -0700268 void idleTimerCallback(TimerState);
269 void touchTimerCallback(TimerState);
270 void displayPowerTimerCallback(TimerState);
271
Dominik Laskowski98041832019-08-01 18:35:59 -0700272 void setVsyncPeriod(nsecs_t period);
273
Dominik Laskowski596a2562022-10-28 11:26:12 -0400274 // Chooses a leader among the registered displays, unless `leaderIdOpt` is specified. The new
275 // `mLeaderDisplayId` is never `std::nullopt`.
276 void promoteLeaderDisplay(std::optional<PhysicalDisplayId> leaderIdOpt = std::nullopt)
277 REQUIRES(kMainThreadContext, mDisplayLock);
278
279 // Blocks until the leader's idle timer thread exits. `mDisplayLock` must not be locked by the
280 // caller on the main thread to avoid deadlock, since the timer thread locks it before exit.
281 void demoteLeaderDisplay() REQUIRES(kMainThreadContext) EXCLUDES(mDisplayLock, mPolicyLock);
282
Dominik Laskowski0c41ffa2021-12-24 16:45:12 -0800283 struct Policy;
284
285 // Sets the S state of the policy to the T value under mPolicyLock, and chooses a display mode
286 // that fulfills the new policy if the state changed. Returns the signals that were considered.
287 template <typename S, typename T>
288 GlobalSignals applyPolicy(S Policy::*, T&&) EXCLUDES(mPolicyLock);
289
Dominik Laskowski530d6bd2022-10-10 16:55:54 -0400290 struct DisplayModeChoice {
Ady Abrahamace3d052022-11-17 16:25:05 -0800291 DisplayModeChoice(FrameRateMode mode, GlobalSignals consideredSignals)
292 : mode(std::move(mode)), consideredSignals(consideredSignals) {}
Dominik Laskowski530d6bd2022-10-10 16:55:54 -0400293
Ady Abrahamace3d052022-11-17 16:25:05 -0800294 FrameRateMode mode;
Dominik Laskowski530d6bd2022-10-10 16:55:54 -0400295 GlobalSignals consideredSignals;
296
297 bool operator==(const DisplayModeChoice& other) const {
Ady Abrahamace3d052022-11-17 16:25:05 -0800298 return mode == other.mode && consideredSignals == other.consideredSignals;
Dominik Laskowski530d6bd2022-10-10 16:55:54 -0400299 }
300
301 // For tests.
302 friend std::ostream& operator<<(std::ostream& stream, const DisplayModeChoice& choice) {
Ady Abrahamace3d052022-11-17 16:25:05 -0800303 return stream << '{' << to_string(*choice.mode.modePtr) << " considering "
Dominik Laskowski530d6bd2022-10-10 16:55:54 -0400304 << choice.consideredSignals.toString().c_str() << '}';
305 }
306 };
307
308 using DisplayModeChoiceMap = display::PhysicalDisplayMap<PhysicalDisplayId, DisplayModeChoice>;
Dominik Laskowski596a2562022-10-28 11:26:12 -0400309
310 // See mDisplayLock for thread safety.
311 DisplayModeChoiceMap chooseDisplayModes() const
312 REQUIRES(mPolicyLock, mDisplayLock, kMainThreadContext);
ramindani69b58e82022-09-26 16:48:36 -0700313
Dominik Laskowski95df6a12022-10-07 18:11:07 -0400314 GlobalSignals makeGlobalSignals() const REQUIRES(mPolicyLock);
ramindani69b58e82022-09-26 16:48:36 -0700315
Dominik Laskowskia8626ec2021-12-15 18:13:30 -0800316 bool updateFrameRateOverrides(GlobalSignals, Fps displayRefreshRate) REQUIRES(mPolicyLock);
Ana Krulec7ab56032018-11-02 20:51:06 +0100317
Dominik Laskowski596a2562022-10-28 11:26:12 -0400318 void dispatchCachedReportedMode() REQUIRES(mPolicyLock) EXCLUDES(mDisplayLock);
Ady Abraham62a0be22020-12-08 16:54:10 -0800319
Dominik Laskowski596a2562022-10-28 11:26:12 -0400320 android::impl::EventThread::ThrottleVsyncCallback makeThrottleVsyncCallback() const;
Patrick Williams83f36b22022-09-14 17:57:35 +0000321 android::impl::EventThread::GetVsyncPeriodFunction makeGetVsyncPeriodFunction() const;
Ady Abrahamdfd62162020-06-10 16:11:56 -0700322
Dominik Laskowski98041832019-08-01 18:35:59 -0700323 // Stores EventThread associated with a given VSyncSource, and an initial EventThreadConnection.
324 struct Connection {
325 sp<EventThreadConnection> connection;
326 std::unique_ptr<EventThread> thread;
327 };
Ady Abraham09bd3922019-04-08 10:44:56 -0700328
Dominik Laskowski98041832019-08-01 18:35:59 -0700329 ConnectionHandle::Id mNextConnectionHandleId = 0;
Ana Krulec6ddd2612020-09-24 13:06:33 -0700330 mutable std::mutex mConnectionsLock;
331 std::unordered_map<ConnectionHandle, Connection> mConnections GUARDED_BY(mConnectionsLock);
Ana Krulece588e312018-09-18 12:32:24 -0700332
Ady Abraham4f960d12021-10-13 16:59:49 -0700333 mutable std::mutex mHWVsyncLock;
Dominik Laskowski98041832019-08-01 18:35:59 -0700334 bool mPrimaryHWVsyncEnabled GUARDED_BY(mHWVsyncLock) = false;
335 bool mHWVsyncAvailable GUARDED_BY(mHWVsyncLock) = false;
336
Steven Thomas2bbaabe2019-08-28 16:08:35 -0700337 std::atomic<nsecs_t> mLastResyncTime = 0;
Ana Krulece588e312018-09-18 12:32:24 -0700338
Dominik Laskowski068173d2021-08-11 17:22:59 -0700339 const FeatureFlags mFeatures;
340 std::optional<VsyncSchedule> mVsyncSchedule;
Ana Krulec7ab56032018-11-02 20:51:06 +0100341
Dominik Laskowski49cea512019-11-12 14:13:23 -0800342 // Used to choose refresh rate if content detection is enabled.
Dominik Laskowski9c93d602021-10-07 19:38:26 -0700343 LayerHistory mLayerHistory;
Ana Krulecfb772822018-11-30 10:44:07 +0100344
Ady Abraham8532d012019-05-08 14:50:56 -0700345 // Timer used to monitor touch events.
Dominik Laskowski03cfce82022-11-02 12:13:29 -0400346 ftl::Optional<OneShotTimer> mTouchTimer;
Ady Abraham6fe2c172019-07-12 12:37:57 -0700347 // Timer used to monitor display power mode.
Dominik Laskowski03cfce82022-11-02 12:13:29 -0400348 ftl::Optional<OneShotTimer> mDisplayPowerTimer;
Ady Abraham6fe2c172019-07-12 12:37:57 -0700349
Ady Abraham3a77a7b2019-12-02 18:46:59 -0800350 ISchedulerCallback& mSchedulerCallback;
Ana Krulecfefd6ae2019-02-13 17:53:08 -0800351
Dominik Laskowski596a2562022-10-28 11:26:12 -0400352 // mDisplayLock may be locked while under mPolicyLock.
Dominik Laskowski068173d2021-08-11 17:22:59 -0700353 mutable std::mutex mPolicyLock;
Dominik Laskowskidd252cd2019-07-26 09:10:16 -0700354
Dominik Laskowski596a2562022-10-28 11:26:12 -0400355 // Only required for reads outside kMainThreadContext. kMainThreadContext is the only writer, so
356 // must lock for writes but not reads. See also mPolicyLock for locking order.
357 mutable std::mutex mDisplayLock;
358
359 display::PhysicalDisplayMap<PhysicalDisplayId, RefreshRateSelectorPtr> mRefreshRateSelectors
360 GUARDED_BY(mDisplayLock) GUARDED_BY(kMainThreadContext);
361
362 ftl::Optional<PhysicalDisplayId> mLeaderDisplayId GUARDED_BY(mDisplayLock)
363 GUARDED_BY(kMainThreadContext);
364
365 RefreshRateSelectorPtr leaderSelectorPtr() const EXCLUDES(mDisplayLock) {
366 std::scoped_lock lock(mDisplayLock);
367 return leaderSelectorPtrLocked();
368 }
369
370 RefreshRateSelectorPtr leaderSelectorPtrLocked() const REQUIRES(mDisplayLock) {
371 ftl::FakeGuard guard(kMainThreadContext);
372 const RefreshRateSelectorPtr noLeader;
373 return mLeaderDisplayId
374 .and_then([this](PhysicalDisplayId leaderId)
375 REQUIRES(mDisplayLock, kMainThreadContext) {
376 return mRefreshRateSelectors.get(leaderId);
377 })
378 .value_or(std::cref(noLeader));
379 }
ramindani69b58e82022-09-26 16:48:36 -0700380
Dominik Laskowski0c41ffa2021-12-24 16:45:12 -0800381 struct Policy {
Dominik Laskowski068173d2021-08-11 17:22:59 -0700382 // Policy for choosing the display mode.
383 LayerHistory::Summary contentRequirements;
Dominik Laskowskidd252cd2019-07-26 09:10:16 -0700384 TimerState idleTimer = TimerState::Reset;
385 TouchState touch = TouchState::Inactive;
386 TimerState displayPowerTimer = TimerState::Expired;
Rachel Lee6a9731d2022-06-06 17:08:14 -0700387 hal::PowerMode displayPowerMode = hal::PowerMode::ON;
Ady Abrahamdfd62162020-06-10 16:11:56 -0700388
Dominik Laskowski068173d2021-08-11 17:22:59 -0700389 // Chosen display mode.
Ady Abrahamace3d052022-11-17 16:25:05 -0800390 ftl::Optional<FrameRateMode> modeOpt;
Dominik Laskowski068173d2021-08-11 17:22:59 -0700391
Marin Shalamanova7fe3042021-01-29 21:02:08 +0100392 struct ModeChangedParams {
Ady Abrahamdfd62162020-06-10 16:11:56 -0700393 ConnectionHandle handle;
Ady Abrahamace3d052022-11-17 16:25:05 -0800394 FrameRateMode mode;
Ady Abrahamdfd62162020-06-10 16:11:56 -0700395 };
396
Dominik Laskowski068173d2021-08-11 17:22:59 -0700397 // Parameters for latest dispatch of mode change event.
Marin Shalamanova7fe3042021-01-29 21:02:08 +0100398 std::optional<ModeChangedParams> cachedModeChangedParams;
Dominik Laskowski068173d2021-08-11 17:22:59 -0700399 } mPolicy GUARDED_BY(mPolicyLock);
Ady Abraham09bd3922019-04-08 10:44:56 -0700400
Ady Abraham3a77a7b2019-12-02 18:46:59 -0800401 std::mutex mVsyncTimelineLock;
Peiyong Line9d809e2020-04-14 13:10:48 -0700402 std::optional<hal::VsyncPeriodChangeTimeline> mLastVsyncPeriodChangeTimeline
Ady Abraham3a77a7b2019-12-02 18:46:59 -0800403 GUARDED_BY(mVsyncTimelineLock);
404 static constexpr std::chrono::nanoseconds MAX_VSYNC_APPLIED_TIME = 200ms;
Ady Abraham8735eac2020-08-12 16:35:04 -0700405
Andy Yu2ae6b6b2021-11-18 14:51:06 -0800406 FrameRateOverrideMappings mFrameRateOverrideMappings;
Ady Abraham4f960d12021-10-13 16:59:49 -0700407
408 // Keeps track of whether the screen is acquired for debug
409 std::atomic<bool> mScreenAcquired = false;
Ana Krulec98b5b242018-08-10 15:03:23 -0700410};
411
Dominik Laskowski068173d2021-08-11 17:22:59 -0700412} // namespace scheduler
Ana Krulec7ab56032018-11-02 20:51:06 +0100413} // namespace android