blob: 3fdddac52a60d74519d338ee0b582ba42e484137 [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>
Dominik Laskowskifb4b7372023-11-22 09:56:54 -050036#include <ftl/non_null.h>
Dominik Laskowski596a2562022-10-28 11:26:12 -040037#include <ftl/optional.h>
Dominik Laskowski068173d2021-08-11 17:22:59 -070038#include <scheduler/Features.h>
Manasi Navare5ef1fa12024-11-07 23:49:46 +000039#include <scheduler/FrameRateMode.h>
Dominik Laskowskib418dd72023-06-13 17:31:04 -040040#include <scheduler/FrameTargeter.h>
Dominik Laskowski5d164f22022-07-07 07:56:07 -070041#include <scheduler/Time.h>
Dominik Laskowski1c99a002023-01-20 17:10:36 -050042#include <scheduler/VsyncConfig.h>
Dominik Laskowski01602522022-10-07 19:02:28 -040043#include <ui/DisplayId.h>
Dominik Laskowski6b049ff2023-01-29 15:46:45 -050044#include <ui/DisplayMap.h>
Dominik Laskowski068173d2021-08-11 17:22:59 -070045
Manasi Navare5ef1fa12024-11-07 23:49:46 +000046#include "DisplayHardware/DisplayMode.h"
Ana Krulec98b5b242018-08-10 15:03:23 -070047#include "EventThread.h"
Andy Yu2ae6b6b2021-11-18 14:51:06 -080048#include "FrameRateOverrideMappings.h"
Leon Scroggins IIIc275df42023-02-07 16:40:21 -050049#include "ISchedulerCallback.h"
Ana Krulec3084c052018-11-21 20:27:17 +010050#include "LayerHistory.h"
Dominik Laskowski756b7892021-08-04 12:53:59 -070051#include "MessageQueue.h"
Ana Krulecf2c006d2019-06-21 15:37:07 -070052#include "OneShotTimer.h"
Dominik Laskowskid82e0f02022-10-26 15:23:04 -040053#include "RefreshRateSelector.h"
Tony Huang9ac5e6e2023-08-24 09:01:44 +000054#include "SmallAreaDetectionAllowMappings.h"
Dominik Laskowski03cfce82022-11-02 12:13:29 -040055#include "Utils/Dumper.h"
Dominik Laskowski1c99a002023-01-20 17:10:36 -050056#include "VsyncModulator.h"
Ana Krulec98b5b242018-08-10 15:03:23 -070057
Ady Abraham822ecbd2023-07-07 16:16:09 -070058#include <FrontEnd/LayerHierarchy.h>
59
Ana Krulec98b5b242018-08-10 15:03:23 -070060namespace android {
61
Dominik Laskowski98041832019-08-01 18:35:59 -070062class FenceTime;
Leon Scroggins III823d4ca2023-12-12 16:57:34 -050063class TimeStats;
Ana Krulece588e312018-09-18 12:32:24 -070064
Adithya Srinivasan5f683cf2020-09-15 14:21:04 -070065namespace frametimeline {
66class TokenManager;
67} // namespace frametimeline
68
Leon Scroggins III823d4ca2023-12-12 16:57:34 -050069namespace surfaceflinger {
70class Factory;
71} // namespace surfaceflinger
72
Dominik Laskowski068173d2021-08-11 17:22:59 -070073namespace scheduler {
74
Dominik Laskowskid82e0f02022-10-26 15:23:04 -040075using GlobalSignals = RefreshRateSelector::GlobalSignals;
ramindani69b58e82022-09-26 16:48:36 -070076
Leon Scroggins III823d4ca2023-12-12 16:57:34 -050077class RefreshRateStats;
78class VsyncConfiguration;
Dominik Laskowskic404cb42023-03-03 19:57:53 -050079class VsyncSchedule;
80
Dominik Laskowski4babfc42024-02-16 12:28:40 -050081enum class Cycle {
82 Render, // Surface rendering.
83 LastComposite // Ahead of display compositing by one refresh period.
84};
85
Ady Abrahamf2851612023-09-25 17:19:00 -070086class Scheduler : public IEventThreadCallback, android::impl::MessageQueue {
Patrick Williams83f36b22022-09-14 17:57:35 +000087 using Impl = android::impl::MessageQueue;
Dominik Laskowski756b7892021-08-04 12:53:59 -070088
Ana Krulec98b5b242018-08-10 15:03:23 -070089public:
Leon Scroggins III823d4ca2023-12-12 16:57:34 -050090 Scheduler(ICompositor&, ISchedulerCallback&, FeatureFlags, surfaceflinger::Factory&,
ramindaniae645822024-01-11 10:57:29 -080091 Fps activeRefreshRate, TimeStats&);
Dominik Laskowski83bd7712022-01-07 14:30:53 -080092 virtual ~Scheduler();
93
94 void startTimers();
Dominik Laskowski83bd7712022-01-07 14:30:53 -080095
Dominik Laskowski16d1b3c2024-07-12 14:35:00 -040096 // TODO: b/241285191 - Remove this API by promoting pacesetter in onScreen{Acquired,Released}.
97 void setPacesetterDisplay(PhysicalDisplayId) REQUIRES(kMainThreadContext)
Dominik Laskowski596a2562022-10-28 11:26:12 -040098 EXCLUDES(mDisplayLock);
Dominik Laskowskib5a094b2022-10-27 12:00:12 -040099
Dominik Laskowski596a2562022-10-28 11:26:12 -0400100 using RefreshRateSelectorPtr = std::shared_ptr<RefreshRateSelector>;
101
Dominik Laskowskic404cb42023-03-03 19:57:53 -0500102 using ConstVsyncSchedulePtr = std::shared_ptr<const VsyncSchedule>;
103 using VsyncSchedulePtr = std::shared_ptr<VsyncSchedule>;
104
Dominik Laskowski3ff44c72024-06-17 11:03:30 -0400105 // After registration/unregistration, `activeDisplayId` is promoted to pacesetter. Note that the
106 // active display is never unregistered, since hotplug disconnect never happens for activatable
107 // displays, i.e. a foldable's internal displays or otherwise the (internal or external) primary
108 // display.
109 // TODO: b/255635821 - Remove active display parameters.
110 void registerDisplay(PhysicalDisplayId, RefreshRateSelectorPtr,
111 PhysicalDisplayId activeDisplayId) REQUIRES(kMainThreadContext)
Dominik Laskowski596a2562022-10-28 11:26:12 -0400112 EXCLUDES(mDisplayLock);
Dominik Laskowski3ff44c72024-06-17 11:03:30 -0400113 void unregisterDisplay(PhysicalDisplayId, PhysicalDisplayId activeDisplayId)
114 REQUIRES(kMainThreadContext) EXCLUDES(mDisplayLock);
Dominik Laskowski01602522022-10-07 19:02:28 -0400115
Dominik Laskowski83bd7712022-01-07 14:30:53 -0800116 void run();
Ana Krulec98b5b242018-08-10 15:03:23 -0700117
Leon Scroggins IIIa7be94e2024-01-23 12:24:30 -0500118 void initVsync(frametimeline::TokenManager&, std::chrono::nanoseconds workDuration);
Dominik Laskowski756b7892021-08-04 12:53:59 -0700119
Dominik Laskowski756b7892021-08-04 12:53:59 -0700120 using Impl::setDuration;
121
ramindani32a88b12024-01-31 18:45:30 -0800122 using Impl::getScheduledFrameResult;
Dominik Laskowskif11728a2022-07-28 13:07:42 -0700123 using Impl::scheduleConfigure;
Dominik Laskowski46f3e3b2021-08-10 11:44:24 -0700124 using Impl::scheduleFrame;
Dominik Laskowski756b7892021-08-04 12:53:59 -0700125
126 // Schedule an asynchronous or synchronous task on the main thread.
127 template <typename F, typename T = std::invoke_result_t<F>>
128 [[nodiscard]] std::future<T> schedule(F&& f) {
129 auto [task, future] = makeTask(std::move(f));
130 postMessage(std::move(task));
131 return std::move(future);
132 }
Dominik Laskowski9c93d602021-10-07 19:38:26 -0700133
Chavi Weingarten076acac2023-01-19 17:20:43 +0000134 template <typename F, typename T = std::invoke_result_t<F>>
135 [[nodiscard]] std::future<T> scheduleDelayed(F&& f, nsecs_t uptimeDelay) {
136 auto [task, future] = makeTask(std::move(f));
137 postMessageDelayed(std::move(task), uptimeDelay);
138 return std::move(future);
139 }
140
Dominik Laskowski4babfc42024-02-16 12:28:40 -0500141 void createEventThread(Cycle, frametimeline::TokenManager*,
142 std::chrono::nanoseconds workDuration,
143 std::chrono::nanoseconds readyDuration);
Ana Krulec98b5b242018-08-10 15:03:23 -0700144
Ady Abraham62f216c2020-10-13 19:07:23 -0700145 sp<IDisplayEventConnection> createDisplayEventConnection(
Dominik Laskowski4babfc42024-02-16 12:28:40 -0500146 Cycle, EventRegistrationFlags eventRegistration = {},
Ady Abraham822ecbd2023-07-07 16:16:09 -0700147 const sp<IBinder>& layerHandle = nullptr) EXCLUDES(mChoreographerLock);
Ana Krulece588e312018-09-18 12:32:24 -0700148
Dominik Laskowski091129a2024-02-21 14:26:03 -0500149 enum class Hotplug { Connected, Disconnected };
150 void dispatchHotplug(PhysicalDisplayId, Hotplug);
151
152 void dispatchHotplugError(int32_t errorCode);
Brian Johnson5dcd75d2023-08-15 09:36:37 -0700153
Dominik Laskowskibda52362024-08-04 00:41:46 -0400154 // Returns true if the PhysicalDisplayId is the pacesetter.
Ady Abrahamfc48edc2024-11-05 19:29:49 -0800155 bool onDisplayModeChanged(PhysicalDisplayId, const FrameRateMode&,
156 bool clearContentRequirements) EXCLUDES(mPolicyLock);
Dominik Laskowskie99b98c2023-02-02 12:37:23 -0500157
Manasi Navare5ef1fa12024-11-07 23:49:46 +0000158 void onDisplayModeRejected(PhysicalDisplayId, DisplayModeId);
159
Dominik Laskowskie99b98c2023-02-02 12:37:23 -0500160 void enableSyntheticVsync(bool = true) REQUIRES(kMainThreadContext);
Ady Abraham8e3e2ea2024-10-31 15:52:31 -0700161 void omitVsyncDispatching(bool) REQUIRES(kMainThreadContext);
Ana Krulece588e312018-09-18 12:32:24 -0700162
Dominik Laskowski4babfc42024-02-16 12:28:40 -0500163 void onHdcpLevelsChanged(Cycle, PhysicalDisplayId, int32_t, int32_t);
Huihong Luo9ebb7a72023-06-27 17:01:50 -0700164
Ady Abraham9c53ee72020-07-22 21:16:18 -0700165 // Modifies work duration in the event thread.
Dominik Laskowski4babfc42024-02-16 12:28:40 -0500166 void setDuration(Cycle, std::chrono::nanoseconds workDuration,
Ady Abraham9c53ee72020-07-22 21:16:18 -0700167 std::chrono::nanoseconds readyDuration);
Ana Krulec98b5b242018-08-10 15:03:23 -0700168
Dominik Laskowski14956dc2023-02-22 13:43:57 -0500169 VsyncModulator& vsyncModulator() { return *mVsyncModulator; }
Dominik Laskowski1c99a002023-01-20 17:10:36 -0500170
Leon Scroggins III1af0fb62023-03-02 14:21:44 -0500171 // In some cases, we should only modulate for the pacesetter display. In those
Leon Scroggins III67388622023-02-06 20:36:20 -0500172 // cases, the caller should pass in the relevant display, and the method
Leon Scroggins III1af0fb62023-03-02 14:21:44 -0500173 // will no-op if it's not the pacesetter. Other cases are not specific to a
Leon Scroggins III67388622023-02-06 20:36:20 -0500174 // display.
Dominik Laskowski1c99a002023-01-20 17:10:36 -0500175 template <typename... Args,
176 typename Handler = std::optional<VsyncConfig> (VsyncModulator::*)(Args...)>
Leon Scroggins III67388622023-02-06 20:36:20 -0500177 void modulateVsync(std::optional<PhysicalDisplayId> id, Handler handler, Args... args) {
178 if (id) {
179 std::scoped_lock lock(mDisplayLock);
180 ftl::FakeGuard guard(kMainThreadContext);
Leon Scroggins III1af0fb62023-03-02 14:21:44 -0500181 if (id != mPacesetterDisplayId) {
Leon Scroggins III67388622023-02-06 20:36:20 -0500182 return;
183 }
184 }
185
Dominik Laskowski1c99a002023-01-20 17:10:36 -0500186 if (const auto config = (*mVsyncModulator.*handler)(args...)) {
Leon Scroggins III1af0fb62023-03-02 14:21:44 -0500187 setVsyncConfig(*config, getPacesetterVsyncPeriod());
Dominik Laskowski1c99a002023-01-20 17:10:36 -0500188 }
189 }
190
Dominik Laskowskie37c04e2024-08-04 15:48:00 -0400191 void updatePhaseConfiguration(PhysicalDisplayId, Fps);
Leon Scroggins III823d4ca2023-12-12 16:57:34 -0500192
193 const VsyncConfiguration& getVsyncConfiguration() const { return *mVsyncConfiguration; }
Dominik Laskowski1c99a002023-01-20 17:10:36 -0500194
Ady Abrahamace3d052022-11-17 16:25:05 -0800195 // Sets the render rate for the scheduler to run at.
Ady Abrahamee6365b2024-03-06 14:31:45 -0800196 void setRenderRate(PhysicalDisplayId, Fps, bool applyImmediately);
Ady Abrahamace3d052022-11-17 16:25:05 -0800197
Leon Scroggins III4235ea02023-04-17 15:14:20 -0400198 void enableHardwareVsync(PhysicalDisplayId) REQUIRES(kMainThreadContext);
199 void disableHardwareVsync(PhysicalDisplayId, bool disallow) REQUIRES(kMainThreadContext);
Dominik Laskowski98041832019-08-01 18:35:59 -0700200
Alec Mourif8e689c2019-05-20 18:32:22 -0700201 // Resyncs the scheduler to hardware vsync.
Leon Scroggins IIIc275df42023-02-07 16:40:21 -0500202 // If allowToEnable is true, then hardware vsync will be turned on.
Alec Mourif8e689c2019-05-20 18:32:22 -0700203 // Otherwise, if hardware vsync is not already enabled then this method will
204 // no-op.
Ady Abrahamc585dba2023-11-15 18:41:35 -0800205 // If modePtr is nullopt, use the active display mode.
Leon Scroggins III67388622023-02-06 20:36:20 -0500206 void resyncToHardwareVsync(PhysicalDisplayId id, bool allowToEnable,
Ady Abrahamc585dba2023-11-15 18:41:35 -0800207 DisplayModePtr modePtr = nullptr) EXCLUDES(mDisplayLock) {
Leon Scroggins III67388622023-02-06 20:36:20 -0500208 std::scoped_lock lock(mDisplayLock);
209 ftl::FakeGuard guard(kMainThreadContext);
Ady Abrahamc585dba2023-11-15 18:41:35 -0800210 resyncToHardwareVsyncLocked(id, allowToEnable, modePtr);
Leon Scroggins III67388622023-02-06 20:36:20 -0500211 }
Ady Abrahame9befd72022-02-24 17:24:44 -0800212 void forceNextResync() { mLastResyncTime = 0; }
Dominik Laskowski98041832019-08-01 18:35:59 -0700213
Leon Scroggins IIIc275df42023-02-07 16:40:21 -0500214 // Passes a vsync sample to VsyncController. Returns true if
215 // VsyncController detected that the vsync period changed and false
216 // otherwise.
Leon Scroggins III67388622023-02-06 20:36:20 -0500217 bool addResyncSample(PhysicalDisplayId, nsecs_t timestamp,
218 std::optional<nsecs_t> hwcVsyncPeriod);
Dominik Laskowskiec0eac22023-01-28 16:16:19 -0500219 void addPresentFence(PhysicalDisplayId, std::shared_ptr<FenceTime>)
Leon Scroggins III4235ea02023-04-17 15:14:20 -0400220 REQUIRES(kMainThreadContext);
Ady Abraham8f1ee7f2019-04-05 10:32:50 -0700221
Dominik Laskowskif7a09ed2019-10-07 13:54:18 -0700222 // Layers are registered on creation, and unregistered when the weak reference expires.
Vishnu Nair5c61a012024-08-05 21:14:39 -0700223 void registerLayer(Layer*, FrameRateCompatibility);
Vishnu Nairef68d6d2023-02-28 06:18:27 +0000224 void recordLayerHistory(int32_t id, const LayerProps& layerProps, nsecs_t presentTime,
Vishnu Nair47b7bb42023-09-29 16:27:33 -0700225 nsecs_t now, LayerHistory::LayerUpdateType) EXCLUDES(mDisplayLock);
Marin Shalamanova7fe3042021-01-29 21:02:08 +0100226 void setModeChangePending(bool pending);
Vishnu Nair80e8cfe2023-09-29 17:03:45 -0700227 void setDefaultFrameRateCompatibility(int32_t id, scheduler::FrameRateCompatibility);
Vishnu Nair41376b62023-11-08 05:08:58 -0800228 void setLayerProperties(int32_t id, const LayerProps&);
Ady Abrahambdda8f02021-04-01 16:06:11 -0700229 void deregisterLayer(Layer*);
Ady Abraham822ecbd2023-07-07 16:16:09 -0700230 void onLayerDestroyed(Layer*) EXCLUDES(mChoreographerLock);
Dominik Laskowskif7a09ed2019-10-07 13:54:18 -0700231
Dominik Laskowski49cea512019-11-12 14:13:23 -0800232 // Detects content using layer history, and selects a matching refresh rate.
Ady Abraham822ecbd2023-07-07 16:16:09 -0700233 void chooseRefreshRateForContent(const surfaceflinger::frontend::LayerHierarchy*,
234 bool updateAttachedChoreographer) EXCLUDES(mDisplayLock);
Ady Abraham97d04232019-03-05 19:48:12 -0800235
Ady Abraham8532d012019-05-08 14:50:56 -0700236 void resetIdleTimer();
237
Dominik Laskowski8da6b0e2021-05-12 15:34:13 -0700238 // Indicates that touch interaction is taking place.
239 void onTouchHint();
Ady Abraham8532d012019-05-08 14:50:56 -0700240
Leon Scroggins III823d4ca2023-12-12 16:57:34 -0500241 void setDisplayPowerMode(PhysicalDisplayId, hal::PowerMode) REQUIRES(kMainThreadContext);
242
243 // TODO(b/255635821): Track this per display.
244 void setActiveDisplayPowerModeForRefreshRateStats(hal::PowerMode) REQUIRES(kMainThreadContext);
Ady Abraham6fe2c172019-07-12 12:37:57 -0700245
Dominik Laskowskic404cb42023-03-03 19:57:53 -0500246 ConstVsyncSchedulePtr getVsyncSchedule(std::optional<PhysicalDisplayId> = std::nullopt) const
247 EXCLUDES(mDisplayLock);
248
249 VsyncSchedulePtr getVsyncSchedule(std::optional<PhysicalDisplayId> idOpt = std::nullopt)
250 EXCLUDES(mDisplayLock) {
251 return std::const_pointer_cast<VsyncSchedule>(std::as_const(*this).getVsyncSchedule(idOpt));
Leon Scroggins III67388622023-02-06 20:36:20 -0500252 }
Ady Abraham55fa7272020-09-30 19:19:27 -0700253
Dominik Laskowskiec0eac22023-01-28 16:16:19 -0500254 TimePoint expectedPresentTimeForPacesetter() const EXCLUDES(mDisplayLock) {
255 std::scoped_lock lock(mDisplayLock);
256 return pacesetterDisplayLocked()
257 .transform([](const Display& display) {
258 return display.targeterPtr->target().expectedPresentTime();
259 })
260 .value_or(TimePoint());
261 }
Dominik Laskowskib418dd72023-06-13 17:31:04 -0400262
Leon Scroggins IIIdb16a2b2023-02-06 17:50:05 -0500263 // Returns true if a given vsync timestamp is considered valid vsync
264 // for a given uid
Dominik Laskowskib418dd72023-06-13 17:31:04 -0400265 bool isVsyncValid(TimePoint expectedVsyncTime, uid_t uid) const;
Leon Scroggins IIIdb16a2b2023-02-06 17:50:05 -0500266
Leon Scroggins III67388622023-02-06 20:36:20 -0500267 bool isVsyncInPhase(TimePoint expectedVsyncTime, Fps frameRate) const;
Huihong Luo1768cb02022-10-11 11:10:34 -0700268
Dominik Laskowski03cfce82022-11-02 12:13:29 -0400269 void dump(utils::Dumper&) const;
Dominik Laskowski4babfc42024-02-16 12:28:40 -0500270 void dump(Cycle, std::string&) const;
Leon Scroggins III67388622023-02-06 20:36:20 -0500271 void dumpVsync(std::string&) const EXCLUDES(mDisplayLock);
Ana Krulec0c8cd522018-08-31 12:27:28 -0700272
Leon Scroggins III1af0fb62023-03-02 14:21:44 -0500273 // Returns the preferred refresh rate and frame rate for the pacesetter display.
Dominik Laskowskifc378b02022-12-02 14:56:05 -0500274 FrameRateMode getPreferredDisplayMode();
Daniel Solomon0f0ddc12019-08-19 19:31:09 -0700275
Ady Abraham3a77a7b2019-12-02 18:46:59 -0800276 // Notifies the scheduler about a refresh rate timeline change.
Peiyong Line9d809e2020-04-14 13:10:48 -0700277 void onNewVsyncPeriodChangeTimeline(const hal::VsyncPeriodChangeTimeline& timeline);
Ady Abraham3a77a7b2019-12-02 18:46:59 -0800278
Leon Scroggins III5b581492023-10-31 14:29:41 -0400279 // Notifies the scheduler once the composition is presented. Returns if recomposite is needed.
280 bool onCompositionPresented(nsecs_t presentTime);
Ady Abraham3a77a7b2019-12-02 18:46:59 -0800281
Ady Abraham8a82ba62020-01-17 12:43:17 -0800282 // Notifies the scheduler when the display size has changed. Called from SF's main thread
Ady Abraham7825c682021-05-17 15:12:14 -0700283 void onActiveDisplayAreaChanged(uint32_t displayArea);
Ady Abraham8a82ba62020-01-17 12:43:17 -0800284
Ady Abraham62a0be22020-12-08 16:54:10 -0800285 // Stores the preferred refresh rate that an app should run at.
286 // FrameRateOverride.refreshRateHz == 0 means no preference.
Andy Yu2ae6b6b2021-11-18 14:51:06 -0800287 void setPreferredRefreshRateForUid(FrameRateOverride);
288
Andy Yu8c2703d2023-11-03 11:22:46 -0700289 // Stores the frame rate override that a game should run at set by game interventions.
290 // FrameRateOverride.refreshRateHz == 0 means no preference.
291 void setGameModeFrameRateForUid(FrameRateOverride) EXCLUDES(mDisplayLock);
292
293 // Stores the frame rate override that a game should run rat set by default game frame rate.
294 // FrameRateOverride.refreshRateHz == 0 means no preference, game default game frame rate is not
295 // enabled.
296 //
297 // "ro.surface_flinger.game_default_frame_rate_override" sets the frame rate value,
298 // "persist.graphics.game_default_frame_rate.enabled" controls whether this feature is enabled.
299 void setGameDefaultFrameRateForUid(FrameRateOverride) EXCLUDES(mDisplayLock);
Andy Yu2ae6b6b2021-11-18 14:51:06 -0800300
Tony Huangf3621102023-09-04 17:14:22 +0800301 void updateSmallAreaDetection(std::vector<std::pair<int32_t, float>>& uidThresholdMappings);
Tony Huang9ac5e6e2023-08-24 09:01:44 +0000302
Tony Huangf3621102023-09-04 17:14:22 +0800303 void setSmallAreaDetectionThreshold(int32_t appId, float threshold);
Tony Huang9ac5e6e2023-08-24 09:01:44 +0000304
305 // Returns true if the dirty area is less than threshold.
Tony Huangf3621102023-09-04 17:14:22 +0800306 bool isSmallDirtyArea(int32_t appId, uint32_t dirtyArea);
Tony Huang9ac5e6e2023-08-24 09:01:44 +0000307
Alec Mouri7d436ec2021-01-27 20:40:50 -0800308 // Retrieves the overridden refresh rate for a given uid.
Dominik Laskowski596a2562022-10-28 11:26:12 -0400309 std::optional<Fps> getFrameRateOverride(uid_t) const EXCLUDES(mDisplayLock);
Ady Abraham3efa3942021-06-24 19:01:25 -0700310
Leon Scroggins III1af0fb62023-03-02 14:21:44 -0500311 Period getPacesetterVsyncPeriod() const EXCLUDES(mDisplayLock) {
312 return pacesetterSelectorPtr()->getActiveMode().fps.getPeriod();
Ady Abraham3efa3942021-06-24 19:01:25 -0700313 }
Ady Abraham62a0be22020-12-08 16:54:10 -0800314
Leon Scroggins III1af0fb62023-03-02 14:21:44 -0500315 Fps getPacesetterRefreshRate() const EXCLUDES(mDisplayLock) {
316 return pacesetterSelectorPtr()->getActiveMode().fps;
Huihong Luo1768cb02022-10-11 11:10:34 -0700317 }
318
ramindani0491e642023-11-16 17:42:14 -0800319 Fps getNextFrameInterval(PhysicalDisplayId, TimePoint currentExpectedPresentTime) const
320 EXCLUDES(mDisplayLock);
321
Nathaniel Nifong1303d912021-10-06 09:41:24 -0400322 // Returns the framerate of the layer with the given sequence ID
323 float getLayerFramerate(nsecs_t now, int32_t id) const {
324 return mLayerHistory.getLayerFramerate(now, id);
325 }
326
Dominik Laskowski1d7be062024-07-24 13:59:07 -0400327 void updateFrameRateOverrides(GlobalSignals, Fps displayRefreshRate) EXCLUDES(mPolicyLock);
Ady Abraham33a386b2023-07-18 15:37:11 -0700328
Jerry Chang36678002023-11-29 16:56:17 +0000329 // Returns true if the small dirty detection is enabled for the appId.
330 bool supportSmallDirtyDetection(int32_t appId) {
331 return mFeatures.test(Feature::kSmallDirtyContentDetection) &&
332 mSmallAreaDetectionAllowMappings.getThresholdForAppId(appId).has_value();
Arthur Hungc70bee22023-06-02 01:35:52 +0000333 }
334
Alec Mouri1c7938e2023-09-22 04:17:23 +0000335 // Injects a delay that is a fraction of the predicted frame duration for the next frame.
336 void injectPacesetterDelay(float frameDurationFraction) REQUIRES(kMainThreadContext) {
337 mPacesetterFrameDurationFractionToSkip = frameDurationFraction;
338 }
339
Melody Hsue524dd92024-08-27 22:27:29 +0000340 // Propagates a flag to the EventThread indicating that buffer stuffing
341 // recovery should begin.
342 void addBufferStuffedUids(BufferStuffingMap bufferStuffedUids);
343
Ady Abraham708ebfb2024-11-14 14:42:49 -0800344 void setDebugPresentDelay(TimePoint delay) { mDebugPresentDelay = delay; }
345
Ana Krulec98b5b242018-08-10 15:03:23 -0700346private:
Ana Krulecafb45842019-02-13 13:33:03 -0800347 friend class TestableScheduler;
348
Dominik Laskowskidd252cd2019-07-26 09:10:16 -0700349 enum class ContentDetectionState { Off, On };
350 enum class TimerState { Reset, Expired };
351 enum class TouchState { Inactive, Active };
Ana Krulecfefd6ae2019-02-13 17:53:08 -0800352
Dominik Laskowski08fbd852022-07-14 08:53:42 -0700353 // impl::MessageQueue overrides:
Dominik Laskowskiec0eac22023-01-28 16:16:19 -0500354 void onFrameSignal(ICompositor&, VsyncId, TimePoint expectedVsyncTime) override
355 REQUIRES(kMainThreadContext, mDisplayLock);
Dominik Laskowski08fbd852022-07-14 08:53:42 -0700356
Dominik Laskowski4babfc42024-02-16 12:28:40 -0500357 // Used to skip event dispatch before EventThread creation during boot.
358 // TODO: b/241285191 - Reorder Scheduler initialization to avoid this.
359 bool hasEventThreads() const {
Ady Abraham16490fd2024-05-07 16:08:36 -0700360 return CC_LIKELY(
361 mRenderEventThread &&
362 (FlagManager::getInstance().deprecate_vsync_sf() || mLastCompositeEventThread));
Dominik Laskowski4babfc42024-02-16 12:28:40 -0500363 }
364
365 EventThread& eventThreadFor(Cycle cycle) const {
366 return *(cycle == Cycle::Render ? mRenderEventThread : mLastCompositeEventThread);
367 }
Dominik Laskowskiccf37d72019-02-01 16:47:58 -0800368
Dominik Laskowski3a80a382019-07-25 11:16:07 -0700369 // Update feature state machine to given state when corresponding timer resets or expires.
Dominik Laskowski596a2562022-10-28 11:26:12 -0400370 void kernelIdleTimerCallback(TimerState) EXCLUDES(mDisplayLock);
Dominik Laskowski3a80a382019-07-25 11:16:07 -0700371 void idleTimerCallback(TimerState);
372 void touchTimerCallback(TimerState);
373 void displayPowerTimerCallback(TimerState);
374
Dominik Laskowski66295432023-03-14 12:25:36 -0400375 // VsyncSchedule delegate.
376 void onHardwareVsyncRequest(PhysicalDisplayId, bool enable);
377
Leon Scroggins III67388622023-02-06 20:36:20 -0500378 void resyncToHardwareVsyncLocked(PhysicalDisplayId, bool allowToEnable,
Ady Abrahamc585dba2023-11-15 18:41:35 -0800379 DisplayModePtr modePtr = nullptr)
Leon Scroggins III67388622023-02-06 20:36:20 -0500380 REQUIRES(kMainThreadContext, mDisplayLock);
381 void resyncAllToHardwareVsync(bool allowToEnable) EXCLUDES(mDisplayLock);
Dominik Laskowski1c99a002023-01-20 17:10:36 -0500382 void setVsyncConfig(const VsyncConfig&, Period vsyncPeriod);
Dominik Laskowski98041832019-08-01 18:35:59 -0700383
Dominik Laskowskib9724192024-07-12 15:55:28 -0400384 // TODO: b/241286431 - Remove this option, which assumes that the pacesetter does not change
385 // when a (secondary) display is registered or unregistered. In the short term, this avoids
386 // a deadlock where the main thread joins with the timer thread as the timer thread waits to
387 // lock a mutex held by the main thread.
388 struct PromotionParams {
389 // Whether to stop and start the idle timer. Ignored unless connected_display flag is set.
390 bool toggleIdleTimer;
391 };
392
393 void promotePacesetterDisplay(PhysicalDisplayId pacesetterId, PromotionParams)
394 REQUIRES(kMainThreadContext) EXCLUDES(mDisplayLock);
Leon Scroggins III6fc45192023-03-16 12:13:28 -0400395
396 // Changes to the displays (e.g. registering and unregistering) must be made
397 // while mDisplayLock is locked, and the new pacesetter then must be promoted while
398 // mDisplayLock is still locked. However, a new pacesetter means that
399 // MessageQueue and EventThread need to use the new pacesetter's
400 // VsyncSchedule, and this must happen while mDisplayLock is *not* locked,
401 // or else we may deadlock with EventThread.
Dominik Laskowskib9724192024-07-12 15:55:28 -0400402 std::shared_ptr<VsyncSchedule> promotePacesetterDisplayLocked(PhysicalDisplayId pacesetterId,
403 PromotionParams)
Dominik Laskowski596a2562022-10-28 11:26:12 -0400404 REQUIRES(kMainThreadContext, mDisplayLock);
Leon Scroggins39d25342023-04-19 17:11:01 +0000405 void applyNewVsyncSchedule(std::shared_ptr<VsyncSchedule>) EXCLUDES(mDisplayLock);
Dominik Laskowski596a2562022-10-28 11:26:12 -0400406
Dominik Laskowskib9724192024-07-12 15:55:28 -0400407 // If toggleIdleTimer is true, the calling thread blocks until the pacesetter's idle timer
408 // thread exits, in which case mDisplayLock must not be locked by the caller to avoid deadlock,
409 // since the timer thread locks it before exit.
410 void demotePacesetterDisplay(PromotionParams) REQUIRES(kMainThreadContext)
411 EXCLUDES(mDisplayLock, mPolicyLock);
Dominik Laskowski596a2562022-10-28 11:26:12 -0400412
Dominik Laskowski3ff44c72024-06-17 11:03:30 -0400413 void registerDisplayInternal(PhysicalDisplayId, RefreshRateSelectorPtr, VsyncSchedulePtr,
414 PhysicalDisplayId activeDisplayId) REQUIRES(kMainThreadContext)
415 EXCLUDES(mDisplayLock);
Leon Scroggins III67388622023-02-06 20:36:20 -0500416
Dominik Laskowski0c41ffa2021-12-24 16:45:12 -0800417 struct Policy;
418
419 // Sets the S state of the policy to the T value under mPolicyLock, and chooses a display mode
420 // that fulfills the new policy if the state changed. Returns the signals that were considered.
421 template <typename S, typename T>
422 GlobalSignals applyPolicy(S Policy::*, T&&) EXCLUDES(mPolicyLock);
423
Dominik Laskowski530d6bd2022-10-10 16:55:54 -0400424 struct DisplayModeChoice {
Ady Abrahamace3d052022-11-17 16:25:05 -0800425 DisplayModeChoice(FrameRateMode mode, GlobalSignals consideredSignals)
426 : mode(std::move(mode)), consideredSignals(consideredSignals) {}
Dominik Laskowski530d6bd2022-10-10 16:55:54 -0400427
Dominik Laskowski9e88d622024-03-06 17:42:39 -0500428 static DisplayModeChoice from(RefreshRateSelector::RankedFrameRates rankedFrameRates) {
429 return {rankedFrameRates.ranking.front().frameRateMode,
430 rankedFrameRates.consideredSignals};
431 }
432
Ady Abrahamace3d052022-11-17 16:25:05 -0800433 FrameRateMode mode;
Dominik Laskowski530d6bd2022-10-10 16:55:54 -0400434 GlobalSignals consideredSignals;
435
436 bool operator==(const DisplayModeChoice& other) const {
Ady Abrahamace3d052022-11-17 16:25:05 -0800437 return mode == other.mode && consideredSignals == other.consideredSignals;
Dominik Laskowski530d6bd2022-10-10 16:55:54 -0400438 }
439
440 // For tests.
441 friend std::ostream& operator<<(std::ostream& stream, const DisplayModeChoice& choice) {
Ady Abrahamace3d052022-11-17 16:25:05 -0800442 return stream << '{' << to_string(*choice.mode.modePtr) << " considering "
Dominik Laskowski530d6bd2022-10-10 16:55:54 -0400443 << choice.consideredSignals.toString().c_str() << '}';
444 }
445 };
446
Dominik Laskowski6b049ff2023-01-29 15:46:45 -0500447 using DisplayModeChoiceMap = ui::PhysicalDisplayMap<PhysicalDisplayId, DisplayModeChoice>;
Dominik Laskowski596a2562022-10-28 11:26:12 -0400448
449 // See mDisplayLock for thread safety.
450 DisplayModeChoiceMap chooseDisplayModes() const
451 REQUIRES(mPolicyLock, mDisplayLock, kMainThreadContext);
ramindani69b58e82022-09-26 16:48:36 -0700452
Dominik Laskowski95df6a12022-10-07 18:11:07 -0400453 GlobalSignals makeGlobalSignals() const REQUIRES(mPolicyLock);
ramindani69b58e82022-09-26 16:48:36 -0700454
Ady Abraham33a386b2023-07-18 15:37:11 -0700455 bool updateFrameRateOverridesLocked(GlobalSignals, Fps displayRefreshRate)
456 REQUIRES(mPolicyLock);
Dominik Laskowski1d7be062024-07-24 13:59:07 -0400457
458 void onFrameRateOverridesChanged();
459
Ady Abraham822ecbd2023-07-07 16:16:09 -0700460 void updateAttachedChoreographers(const surfaceflinger::frontend::LayerHierarchy&,
461 Fps displayRefreshRate);
462 int updateAttachedChoreographersInternal(const surfaceflinger::frontend::LayerHierarchy&,
463 Fps displayRefreshRate, int parentDivisor);
464 void updateAttachedChoreographersFrameRate(const surfaceflinger::frontend::RequestedLayerState&,
465 Fps fps) EXCLUDES(mChoreographerLock);
Ana Krulec7ab56032018-11-02 20:51:06 +0100466
Dominik Laskowskifc94b412024-08-03 15:02:23 -0400467 void emitModeChangeIfNeeded() REQUIRES(mPolicyLock) EXCLUDES(mDisplayLock);
Ady Abraham62a0be22020-12-08 16:54:10 -0800468
Ady Abrahamf2851612023-09-25 17:19:00 -0700469 // IEventThreadCallback overrides
470 bool throttleVsync(TimePoint, uid_t) override;
Ying Weiaf854ad2024-03-16 05:24:39 +0000471 // Get frame interval
Ady Abrahamf2851612023-09-25 17:19:00 -0700472 Period getVsyncPeriod(uid_t) override EXCLUDES(mDisplayLock);
473 void resync() override EXCLUDES(mDisplayLock);
ramindaniae645822024-01-11 10:57:29 -0800474 void onExpectedPresentTimePosted(TimePoint expectedPresentTime) override EXCLUDES(mDisplayLock);
Leon Scroggins IIIdb16a2b2023-02-06 17:50:05 -0500475
Dominik Laskowski4babfc42024-02-16 12:28:40 -0500476 std::unique_ptr<EventThread> mRenderEventThread;
Dominik Laskowski4babfc42024-02-16 12:28:40 -0500477 std::unique_ptr<EventThread> mLastCompositeEventThread;
Dominik Laskowski1c99a002023-01-20 17:10:36 -0500478
Steven Thomas2bbaabe2019-08-28 16:08:35 -0700479 std::atomic<nsecs_t> mLastResyncTime = 0;
Ana Krulece588e312018-09-18 12:32:24 -0700480
Dominik Laskowski068173d2021-08-11 17:22:59 -0700481 const FeatureFlags mFeatures;
Ana Krulec7ab56032018-11-02 20:51:06 +0100482
Leon Scroggins III823d4ca2023-12-12 16:57:34 -0500483 // Stores phase offsets configured per refresh rate.
484 const std::unique_ptr<VsyncConfiguration> mVsyncConfiguration;
485
Dominik Laskowski1c99a002023-01-20 17:10:36 -0500486 // Shifts the VSYNC phase during certain transactions and refresh rate changes.
487 const sp<VsyncModulator> mVsyncModulator;
488
Leon Scroggins III823d4ca2023-12-12 16:57:34 -0500489 const std::unique_ptr<RefreshRateStats> mRefreshRateStats;
490
Dominik Laskowski49cea512019-11-12 14:13:23 -0800491 // Used to choose refresh rate if content detection is enabled.
Dominik Laskowski9c93d602021-10-07 19:38:26 -0700492 LayerHistory mLayerHistory;
Ana Krulecfb772822018-11-30 10:44:07 +0100493
Ady Abraham8532d012019-05-08 14:50:56 -0700494 // Timer used to monitor touch events.
Dominik Laskowski03cfce82022-11-02 12:13:29 -0400495 ftl::Optional<OneShotTimer> mTouchTimer;
Ady Abraham6fe2c172019-07-12 12:37:57 -0700496 // Timer used to monitor display power mode.
Dominik Laskowski03cfce82022-11-02 12:13:29 -0400497 ftl::Optional<OneShotTimer> mDisplayPowerTimer;
Ady Abraham6fe2c172019-07-12 12:37:57 -0700498
Alec Mouri1c7938e2023-09-22 04:17:23 +0000499 // Injected delay prior to compositing, for simulating jank.
500 float mPacesetterFrameDurationFractionToSkip GUARDED_BY(kMainThreadContext) = 0.f;
501
Ady Abraham3a77a7b2019-12-02 18:46:59 -0800502 ISchedulerCallback& mSchedulerCallback;
Ana Krulecfefd6ae2019-02-13 17:53:08 -0800503
Dominik Laskowski596a2562022-10-28 11:26:12 -0400504 // mDisplayLock may be locked while under mPolicyLock.
Dominik Laskowski068173d2021-08-11 17:22:59 -0700505 mutable std::mutex mPolicyLock;
Dominik Laskowskidd252cd2019-07-26 09:10:16 -0700506
Dominik Laskowski596a2562022-10-28 11:26:12 -0400507 // Only required for reads outside kMainThreadContext. kMainThreadContext is the only writer, so
508 // must lock for writes but not reads. See also mPolicyLock for locking order.
509 mutable std::mutex mDisplayLock;
510
Dominik Laskowskiec0eac22023-01-28 16:16:19 -0500511 using FrameTargeterPtr = std::unique_ptr<FrameTargeter>;
512
Dominik Laskowskic404cb42023-03-03 19:57:53 -0500513 struct Display {
Dominik Laskowskiec0eac22023-01-28 16:16:19 -0500514 Display(PhysicalDisplayId displayId, RefreshRateSelectorPtr selectorPtr,
515 VsyncSchedulePtr schedulePtr, FeatureFlags features)
516 : displayId(displayId),
517 selectorPtr(std::move(selectorPtr)),
518 schedulePtr(std::move(schedulePtr)),
Leon Scroggins III0bd0d4c2022-12-08 13:20:45 -0500519 targeterPtr(std::make_unique<FrameTargeter>(displayId, features)) {}
Dominik Laskowskiec0eac22023-01-28 16:16:19 -0500520
521 const PhysicalDisplayId displayId;
Dominik Laskowski596a2562022-10-28 11:26:12 -0400522
Dominik Laskowskic404cb42023-03-03 19:57:53 -0500523 // Effectively const except in move constructor.
524 RefreshRateSelectorPtr selectorPtr;
525 VsyncSchedulePtr schedulePtr;
Dominik Laskowskiec0eac22023-01-28 16:16:19 -0500526 FrameTargeterPtr targeterPtr;
Dominik Laskowski66295432023-03-14 12:25:36 -0400527
528 hal::PowerMode powerMode = hal::PowerMode::OFF;
Dominik Laskowskic404cb42023-03-03 19:57:53 -0500529 };
530
531 using DisplayRef = std::reference_wrapper<Display>;
532 using ConstDisplayRef = std::reference_wrapper<const Display>;
533
Dominik Laskowski6b049ff2023-01-29 15:46:45 -0500534 ui::PhysicalDisplayMap<PhysicalDisplayId, Display> mDisplays GUARDED_BY(mDisplayLock)
Dominik Laskowskic404cb42023-03-03 19:57:53 -0500535 GUARDED_BY(kMainThreadContext);
Leon Scroggins III67388622023-02-06 20:36:20 -0500536
Leon Scroggins III1af0fb62023-03-02 14:21:44 -0500537 ftl::Optional<PhysicalDisplayId> mPacesetterDisplayId GUARDED_BY(mDisplayLock)
Dominik Laskowski596a2562022-10-28 11:26:12 -0400538 GUARDED_BY(kMainThreadContext);
539
Dominik Laskowskic404cb42023-03-03 19:57:53 -0500540 ftl::Optional<DisplayRef> pacesetterDisplayLocked() REQUIRES(mDisplayLock) {
541 return static_cast<const Scheduler*>(this)->pacesetterDisplayLocked().transform(
542 [](const Display& display) { return std::ref(const_cast<Display&>(display)); });
543 }
544
545 ftl::Optional<ConstDisplayRef> pacesetterDisplayLocked() const REQUIRES(mDisplayLock) {
546 ftl::FakeGuard guard(kMainThreadContext);
547 return mPacesetterDisplayId.and_then([this](PhysicalDisplayId pacesetterId)
548 REQUIRES(mDisplayLock, kMainThreadContext) {
549 return mDisplays.get(pacesetterId);
550 });
551 }
552
Dominik Laskowskifb4b7372023-11-22 09:56:54 -0500553 // The pacesetter must exist as a precondition.
554 ftl::NonNull<const Display*> pacesetterPtrLocked() const REQUIRES(mDisplayLock) {
555 return ftl::as_non_null(&pacesetterDisplayLocked()->get());
556 }
557
Leon Scroggins III1af0fb62023-03-02 14:21:44 -0500558 RefreshRateSelectorPtr pacesetterSelectorPtr() const EXCLUDES(mDisplayLock) {
Dominik Laskowski596a2562022-10-28 11:26:12 -0400559 std::scoped_lock lock(mDisplayLock);
Leon Scroggins III1af0fb62023-03-02 14:21:44 -0500560 return pacesetterSelectorPtrLocked();
Dominik Laskowski596a2562022-10-28 11:26:12 -0400561 }
562
Leon Scroggins III1af0fb62023-03-02 14:21:44 -0500563 RefreshRateSelectorPtr pacesetterSelectorPtrLocked() const REQUIRES(mDisplayLock) {
Dominik Laskowskic404cb42023-03-03 19:57:53 -0500564 return pacesetterDisplayLocked()
565 .transform([](const Display& display) { return display.selectorPtr; })
566 .or_else([] { return std::optional<RefreshRateSelectorPtr>(nullptr); })
Dominik Laskowski80872bd2022-11-15 11:34:33 -0500567 .value();
Dominik Laskowski596a2562022-10-28 11:26:12 -0400568 }
ramindani69b58e82022-09-26 16:48:36 -0700569
Dominik Laskowskic404cb42023-03-03 19:57:53 -0500570 ConstVsyncSchedulePtr getVsyncScheduleLocked(
571 std::optional<PhysicalDisplayId> = std::nullopt) const REQUIRES(mDisplayLock);
572
573 VsyncSchedulePtr getVsyncScheduleLocked(std::optional<PhysicalDisplayId> idOpt = std::nullopt)
574 REQUIRES(mDisplayLock) {
Leon Scroggins III67388622023-02-06 20:36:20 -0500575 return std::const_pointer_cast<VsyncSchedule>(
576 static_cast<const Scheduler*>(this)->getVsyncScheduleLocked(idOpt));
577 }
578
Dominik Laskowski0c41ffa2021-12-24 16:45:12 -0800579 struct Policy {
Dominik Laskowski068173d2021-08-11 17:22:59 -0700580 // Policy for choosing the display mode.
581 LayerHistory::Summary contentRequirements;
Dominik Laskowskidd252cd2019-07-26 09:10:16 -0700582 TimerState idleTimer = TimerState::Reset;
583 TouchState touch = TouchState::Inactive;
584 TimerState displayPowerTimer = TimerState::Expired;
Rachel Lee6a9731d2022-06-06 17:08:14 -0700585 hal::PowerMode displayPowerMode = hal::PowerMode::ON;
Ady Abrahamdfd62162020-06-10 16:11:56 -0700586
Dominik Laskowski068173d2021-08-11 17:22:59 -0700587 // Chosen display mode.
Ady Abrahamace3d052022-11-17 16:25:05 -0800588 ftl::Optional<FrameRateMode> modeOpt;
Dominik Laskowski068173d2021-08-11 17:22:59 -0700589
Dominik Laskowskifc94b412024-08-03 15:02:23 -0400590 // Display mode of latest emitted event.
591 std::optional<FrameRateMode> emittedModeOpt;
Dominik Laskowski068173d2021-08-11 17:22:59 -0700592 } mPolicy GUARDED_BY(mPolicyLock);
Ady Abraham09bd3922019-04-08 10:44:56 -0700593
Ady Abraham822ecbd2023-07-07 16:16:09 -0700594 std::mutex mChoreographerLock;
595
596 struct AttachedChoreographers {
597 Fps frameRate;
598 std::unordered_set<wp<EventThreadConnection>, WpHash> connections;
599 };
600 // Map keyed by layer ID (sequence) to choreographer connections.
601 std::unordered_map<int32_t, AttachedChoreographers> mAttachedChoreographers
602 GUARDED_BY(mChoreographerLock);
603
Ady Abraham3a77a7b2019-12-02 18:46:59 -0800604 std::mutex mVsyncTimelineLock;
Peiyong Line9d809e2020-04-14 13:10:48 -0700605 std::optional<hal::VsyncPeriodChangeTimeline> mLastVsyncPeriodChangeTimeline
Ady Abraham3a77a7b2019-12-02 18:46:59 -0800606 GUARDED_BY(mVsyncTimelineLock);
607 static constexpr std::chrono::nanoseconds MAX_VSYNC_APPLIED_TIME = 200ms;
Ady Abraham8735eac2020-08-12 16:35:04 -0700608
Andy Yu2ae6b6b2021-11-18 14:51:06 -0800609 FrameRateOverrideMappings mFrameRateOverrideMappings;
Tony Huang9ac5e6e2023-08-24 09:01:44 +0000610 SmallAreaDetectionAllowMappings mSmallAreaDetectionAllowMappings;
Ady Abraham708ebfb2024-11-14 14:42:49 -0800611
612 std::atomic<std::optional<TimePoint>> mDebugPresentDelay;
Ana Krulec98b5b242018-08-10 15:03:23 -0700613};
614
Dominik Laskowski068173d2021-08-11 17:22:59 -0700615} // namespace scheduler
Ana Krulec7ab56032018-11-02 20:51:06 +0100616} // namespace android