blob: 730ea8f0c88a006d32cab05a17fb920ebada8447 [file] [log] [blame]
Ana Krulec98b5b242018-08-10 15:03:23 -07001/*
2 * Copyright 2018 The Android Open Source Project
3 *
4 * Licensed under the Apache License, Version 2.0 (the "License");
5 * you may not use this file except in compliance with the License.
6 * You may obtain a copy of the License at
7 *
8 * http://www.apache.org/licenses/LICENSE-2.0
9 *
10 * Unless required by applicable law or agreed to in writing, software
11 * distributed under the License is distributed on an "AS IS" BASIS,
12 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13 * See the License for the specific language governing permissions and
14 * limitations under the License.
15 */
16
17#pragma once
18
Dominik Laskowski98041832019-08-01 18:35:59 -070019#include <atomic>
Kevin DuBois413287f2019-02-25 08:46:47 -080020#include <functional>
Ana Krulec98b5b242018-08-10 15:03:23 -070021#include <memory>
Dominik Laskowski98041832019-08-01 18:35:59 -070022#include <mutex>
23#include <optional>
24#include <unordered_map>
Ana Krulec98b5b242018-08-10 15:03:23 -070025
Ady Abrahamdec1a412020-01-24 10:23:50 -080026// TODO(b/129481165): remove the #pragma below and fix conversion issues
27#pragma clang diagnostic push
28#pragma clang diagnostic ignored "-Wconversion"
Dominik Laskowskidcb38bb2019-01-25 02:35:50 -080029#include <ui/GraphicTypes.h>
Ady Abrahamdec1a412020-01-24 10:23:50 -080030#pragma clang diagnostic pop
Ana Krulec98b5b242018-08-10 15:03:23 -070031
Ana Krulece588e312018-09-18 12:32:24 -070032#include "EventControlThread.h"
Ana Krulec98b5b242018-08-10 15:03:23 -070033#include "EventThread.h"
Ana Krulec3084c052018-11-21 20:27:17 +010034#include "LayerHistory.h"
Ana Krulecf2c006d2019-06-21 15:37:07 -070035#include "OneShotTimer.h"
Ana Krulec8d3e4f32019-03-05 10:40:33 -080036#include "RefreshRateConfigs.h"
Ana Krulec434c22d2018-11-28 13:48:36 +010037#include "SchedulerUtils.h"
Ana Krulec98b5b242018-08-10 15:03:23 -070038
39namespace android {
40
Ady Abraham2139f732019-11-13 18:56:40 -080041using namespace std::chrono_literals;
Ady Abraham5def7332020-05-29 16:13:47 -070042using scheduler::LayerHistory;
Ady Abraham2139f732019-11-13 18:56:40 -080043
Dominik Laskowski98041832019-08-01 18:35:59 -070044class DispSync;
45class FenceTime;
Dominik Laskowski6505f792019-09-18 11:10:05 -070046class InjectVSyncSource;
Dominik Laskowski98041832019-08-01 18:35:59 -070047struct DisplayStateInfo;
Ana Krulece588e312018-09-18 12:32:24 -070048
Ady Abraham3a77a7b2019-12-02 18:46:59 -080049class ISchedulerCallback {
50public:
51 virtual ~ISchedulerCallback() = default;
52 virtual void changeRefreshRate(const scheduler::RefreshRateConfigs::RefreshRate&,
53 scheduler::RefreshRateConfigEvent) = 0;
54 virtual void repaintEverythingForHWC() = 0;
Ady Abrahama09852a2020-02-20 14:23:42 -080055 virtual void kernelTimerChanged(bool expired) = 0;
Ady Abraham3a77a7b2019-12-02 18:46:59 -080056};
57
Ady Abrahambf1349c2020-06-12 14:26:18 -070058class IPhaseOffsetControl {
59public:
60 virtual ~IPhaseOffsetControl() = default;
61 virtual void setPhaseOffset(scheduler::ConnectionHandle, nsecs_t phaseOffset) = 0;
62};
63
64class Scheduler : public IPhaseOffsetControl {
Ana Krulec98b5b242018-08-10 15:03:23 -070065public:
Ady Abraham2139f732019-11-13 18:56:40 -080066 using RefreshRate = scheduler::RefreshRateConfigs::RefreshRate;
Dominik Laskowski98041832019-08-01 18:35:59 -070067 using ConfigEvent = scheduler::RefreshRateConfigEvent;
68
Dominik Laskowski98041832019-08-01 18:35:59 -070069 // Indicates whether to start the transaction early, or at vsync time.
Ady Abrahambf1349c2020-06-12 14:26:18 -070070 enum class TransactionStart {
71 Early, // DEPRECATED. Start the transaction early. Times out on its own
72 EarlyStart, // Start the transaction early and keep this config until EarlyEnd
73 EarlyEnd, // End the early config started at EarlyStart
74 Normal // Start the transaction at the normal time
75 };
Ana Krulec7ecce8c2018-10-12 13:44:41 -070076
Dominik Laskowski98041832019-08-01 18:35:59 -070077 Scheduler(impl::EventControlThread::SetVSyncEnabledFunction,
Ady Abraham8a82ba62020-01-17 12:43:17 -080078 const scheduler::RefreshRateConfigs&, ISchedulerCallback& schedulerCallback,
Ana Krulec3d367c82020-02-25 15:02:01 -080079 bool useContentDetectionV2, bool useContentDetection);
Ana Krulece588e312018-09-18 12:32:24 -070080
Ana Krulec0c8cd522018-08-31 12:27:28 -070081 virtual ~Scheduler();
Ana Krulec98b5b242018-08-10 15:03:23 -070082
Dominik Laskowski98041832019-08-01 18:35:59 -070083 DispSync& getPrimaryDispSync();
Ana Krulece588e312018-09-18 12:32:24 -070084
Dominik Laskowski98041832019-08-01 18:35:59 -070085 using ConnectionHandle = scheduler::ConnectionHandle;
86 ConnectionHandle createConnection(const char* connectionName, nsecs_t phaseOffsetNs,
Dominik Laskowski98041832019-08-01 18:35:59 -070087 impl::EventThread::InterceptVSyncsCallback);
Ana Krulec98b5b242018-08-10 15:03:23 -070088
Steven Thomas2bbaabe2019-08-28 16:08:35 -070089 sp<IDisplayEventConnection> createDisplayEventConnection(ConnectionHandle,
Dominik Laskowski98041832019-08-01 18:35:59 -070090 ISurfaceComposer::ConfigChanged);
Ana Krulece588e312018-09-18 12:32:24 -070091
Dominik Laskowski98041832019-08-01 18:35:59 -070092 sp<EventThreadConnection> getEventConnection(ConnectionHandle);
Kevin DuBois413287f2019-02-25 08:46:47 -080093
Dominik Laskowski98041832019-08-01 18:35:59 -070094 void onHotplugReceived(ConnectionHandle, PhysicalDisplayId, bool connected);
Ady Abrahamdfd62162020-06-10 16:11:56 -070095 void onPrimaryDisplayConfigChanged(ConnectionHandle, PhysicalDisplayId,
96 HwcConfigIndexType configId, nsecs_t vsyncPeriod)
97 EXCLUDES(mFeatureStateLock);
98 void onNonPrimaryDisplayConfigChanged(ConnectionHandle, PhysicalDisplayId,
99 HwcConfigIndexType configId, nsecs_t vsyncPeriod);
Dominik Laskowski98041832019-08-01 18:35:59 -0700100 void onScreenAcquired(ConnectionHandle);
101 void onScreenReleased(ConnectionHandle);
Ana Krulece588e312018-09-18 12:32:24 -0700102
Dominik Laskowski98041832019-08-01 18:35:59 -0700103 // Modifies phase offset in the event thread.
Ady Abrahambf1349c2020-06-12 14:26:18 -0700104 void setPhaseOffset(ConnectionHandle, nsecs_t phaseOffset) override;
Ana Krulec98b5b242018-08-10 15:03:23 -0700105
Ana Krulece588e312018-09-18 12:32:24 -0700106 void getDisplayStatInfo(DisplayStatInfo* stats);
107
Dominik Laskowski6505f792019-09-18 11:10:05 -0700108 // Returns injector handle if injection has toggled, or an invalid handle otherwise.
109 ConnectionHandle enableVSyncInjection(bool enable);
110
111 // Returns false if injection is disabled.
Ady Abraham5facfb12020-04-22 15:18:31 -0700112 bool injectVSync(nsecs_t when, nsecs_t expectedVSyncTime);
Dominik Laskowski6505f792019-09-18 11:10:05 -0700113
Ana Krulece588e312018-09-18 12:32:24 -0700114 void enableHardwareVsync();
115 void disableHardwareVsync(bool makeUnavailable);
Dominik Laskowski98041832019-08-01 18:35:59 -0700116
Alec Mourif8e689c2019-05-20 18:32:22 -0700117 // Resyncs the scheduler to hardware vsync.
118 // If makeAvailable is true, then hardware vsync will be turned on.
119 // Otherwise, if hardware vsync is not already enabled then this method will
120 // no-op.
121 // The period is the vsync period from the current display configuration.
Ana Krulecc2870422019-01-29 19:00:58 -0800122 void resyncToHardwareVsync(bool makeAvailable, nsecs_t period);
Steven Thomas2bbaabe2019-08-28 16:08:35 -0700123 void resync();
Dominik Laskowski98041832019-08-01 18:35:59 -0700124
Alec Mourif8e689c2019-05-20 18:32:22 -0700125 // Passes a vsync sample to DispSync. periodFlushed will be true if
126 // DispSync detected that the vsync period changed, and false otherwise.
Ady Abraham5dee2f12020-02-05 17:49:47 -0800127 void addResyncSample(nsecs_t timestamp, std::optional<nsecs_t> hwcVsyncPeriod,
128 bool* periodFlushed);
Dominik Laskowski98041832019-08-01 18:35:59 -0700129 void addPresentFence(const std::shared_ptr<FenceTime>&);
Ana Krulece588e312018-09-18 12:32:24 -0700130 void setIgnorePresentFences(bool ignore);
Ady Abraham0ed31c92020-04-16 11:48:45 -0700131 nsecs_t getDispSyncExpectedPresentTime(nsecs_t now);
Ady Abraham8f1ee7f2019-04-05 10:32:50 -0700132
Dominik Laskowskif7a09ed2019-10-07 13:54:18 -0700133 // Layers are registered on creation, and unregistered when the weak reference expires.
134 void registerLayer(Layer*);
Ady Abraham5def7332020-05-29 16:13:47 -0700135 void recordLayerHistory(Layer*, nsecs_t presentTime, LayerHistory::LayerUpdateType updateType);
Ady Abraham32efd542020-05-19 17:49:26 -0700136 void setConfigChangePending(bool pending);
Dominik Laskowskif7a09ed2019-10-07 13:54:18 -0700137
Dominik Laskowski49cea512019-11-12 14:13:23 -0800138 // Detects content using layer history, and selects a matching refresh rate.
139 void chooseRefreshRateForContent();
Ady Abraham97d04232019-03-05 19:48:12 -0800140
Dominik Laskowski98041832019-08-01 18:35:59 -0700141 bool isIdleTimerEnabled() const { return mIdleTimer.has_value(); }
Ady Abraham8532d012019-05-08 14:50:56 -0700142 void resetIdleTimer();
143
144 // Function that resets the touch timer.
145 void notifyTouchEvent();
146
Ady Abraham6fe2c172019-07-12 12:37:57 -0700147 void setDisplayPowerState(bool normal);
148
Dominik Laskowski98041832019-08-01 18:35:59 -0700149 void dump(std::string&) const;
150 void dump(ConnectionHandle, std::string&) const;
Ana Krulec0c8cd522018-08-31 12:27:28 -0700151
Ady Abraham2139f732019-11-13 18:56:40 -0800152 // Get the appropriate refresh for current conditions.
153 std::optional<HwcConfigIndexType> getPreferredConfigId();
Daniel Solomon0f0ddc12019-08-19 19:31:09 -0700154
Ady Abraham3a77a7b2019-12-02 18:46:59 -0800155 // Notifies the scheduler about a refresh rate timeline change.
Peiyong Line9d809e2020-04-14 13:10:48 -0700156 void onNewVsyncPeriodChangeTimeline(const hal::VsyncPeriodChangeTimeline& timeline);
Ady Abraham3a77a7b2019-12-02 18:46:59 -0800157
158 // Notifies the scheduler when the display was refreshed
159 void onDisplayRefreshed(nsecs_t timestamp);
160
Ady Abraham8a82ba62020-01-17 12:43:17 -0800161 // Notifies the scheduler when the display size has changed. Called from SF's main thread
162 void onPrimaryDisplayAreaChanged(uint32_t displayArea);
163
Alec Mouri717bcb62020-02-10 17:07:19 -0800164 size_t getEventThreadConnectionCount(ConnectionHandle handle);
165
Ana Krulec98b5b242018-08-10 15:03:23 -0700166private:
Ana Krulecafb45842019-02-13 13:33:03 -0800167 friend class TestableScheduler;
168
Ana Krulecfefd6ae2019-02-13 17:53:08 -0800169 // In order to make sure that the features don't override themselves, we need a state machine
170 // to keep track which feature requested the config change.
Dominik Laskowskidd252cd2019-07-26 09:10:16 -0700171 enum class ContentDetectionState { Off, On };
172 enum class TimerState { Reset, Expired };
173 enum class TouchState { Inactive, Active };
Ana Krulecfefd6ae2019-02-13 17:53:08 -0800174
Dominik Laskowski7c9dbf92019-08-01 17:57:31 -0700175 // Used by tests to inject mocks.
176 Scheduler(std::unique_ptr<DispSync>, std::unique_ptr<EventControlThread>,
Ady Abraham8a82ba62020-01-17 12:43:17 -0800177 const scheduler::RefreshRateConfigs&, ISchedulerCallback& schedulerCallback,
Ana Krulec3d367c82020-02-25 15:02:01 -0800178 bool useContentDetectionV2, bool useContentDetection);
Dominik Laskowski7c9dbf92019-08-01 17:57:31 -0700179
Ady Abraham9e16a482019-12-03 17:19:41 -0800180 std::unique_ptr<VSyncSource> makePrimaryDispSyncSource(const char* name, nsecs_t phaseOffsetNs);
Dominik Laskowski98041832019-08-01 18:35:59 -0700181
Dominik Laskowski6505f792019-09-18 11:10:05 -0700182 // Create a connection on the given EventThread.
Steven Thomas2bbaabe2019-08-28 16:08:35 -0700183 ConnectionHandle createConnection(std::unique_ptr<EventThread>);
184 sp<EventThreadConnection> createConnectionInternal(EventThread*,
Ady Abraham0f4a1b12019-06-04 16:04:04 -0700185 ISurfaceComposer::ConfigChanged);
Dominik Laskowskiccf37d72019-02-01 16:47:58 -0800186
Dominik Laskowski3a80a382019-07-25 11:16:07 -0700187 // Update feature state machine to given state when corresponding timer resets or expires.
188 void kernelIdleTimerCallback(TimerState);
189 void idleTimerCallback(TimerState);
190 void touchTimerCallback(TimerState);
191 void displayPowerTimerCallback(TimerState);
192
Ady Abraham6fe2c172019-07-12 12:37:57 -0700193 // handles various timer features to change the refresh rate.
194 template <class T>
Ady Abrahamdfd62162020-06-10 16:11:56 -0700195 bool handleTimerStateChanged(T* currentState, T newState);
Dominik Laskowski98041832019-08-01 18:35:59 -0700196
197 void setVsyncPeriod(nsecs_t period);
198
Ana Krulec3803b8d2020-02-03 16:35:46 -0800199 // This function checks whether individual features that are affecting the refresh rate
200 // selection were initialized, prioritizes them, and calculates the HwcConfigIndexType
201 // for the suggested refresh rate.
Ady Abrahamdfd62162020-06-10 16:11:56 -0700202 HwcConfigIndexType calculateRefreshRateConfigIndexType(
203 scheduler::RefreshRateConfigs::GlobalSignals* consideredSignals = nullptr)
Ady Abraham1adbb722020-05-15 11:51:48 -0700204 REQUIRES(mFeatureStateLock);
Ana Krulec7ab56032018-11-02 20:51:06 +0100205
Ady Abrahamdfd62162020-06-10 16:11:56 -0700206 void dispatchCachedReportedConfig() REQUIRES(mFeatureStateLock);
207
Dominik Laskowski98041832019-08-01 18:35:59 -0700208 // Stores EventThread associated with a given VSyncSource, and an initial EventThreadConnection.
209 struct Connection {
210 sp<EventThreadConnection> connection;
211 std::unique_ptr<EventThread> thread;
212 };
Ady Abraham09bd3922019-04-08 10:44:56 -0700213
Dominik Laskowski98041832019-08-01 18:35:59 -0700214 ConnectionHandle::Id mNextConnectionHandleId = 0;
215 std::unordered_map<ConnectionHandle, Connection> mConnections;
Ana Krulece588e312018-09-18 12:32:24 -0700216
Dominik Laskowski6505f792019-09-18 11:10:05 -0700217 bool mInjectVSyncs = false;
218 InjectVSyncSource* mVSyncInjector = nullptr;
219 ConnectionHandle mInjectorConnectionHandle;
220
Ana Krulece588e312018-09-18 12:32:24 -0700221 std::mutex mHWVsyncLock;
Dominik Laskowski98041832019-08-01 18:35:59 -0700222 bool mPrimaryHWVsyncEnabled GUARDED_BY(mHWVsyncLock) = false;
223 bool mHWVsyncAvailable GUARDED_BY(mHWVsyncLock) = false;
224
Steven Thomas2bbaabe2019-08-28 16:08:35 -0700225 std::atomic<nsecs_t> mLastResyncTime = 0;
Ana Krulece588e312018-09-18 12:32:24 -0700226
Dan Stoza027d3652020-05-26 17:26:34 -0700227 // Whether to use idle timer callbacks that support the kernel timer.
228 const bool mSupportKernelTimer;
229
Ana Krulece588e312018-09-18 12:32:24 -0700230 std::unique_ptr<DispSync> mPrimaryDispSync;
231 std::unique_ptr<EventControlThread> mEventControlThread;
Ana Krulec7ab56032018-11-02 20:51:06 +0100232
Dominik Laskowski49cea512019-11-12 14:13:23 -0800233 // Used to choose refresh rate if content detection is enabled.
Ady Abraham5def7332020-05-29 16:13:47 -0700234 std::unique_ptr<LayerHistory> mLayerHistory;
Ana Krulecfb772822018-11-30 10:44:07 +0100235
Dominik Laskowski98041832019-08-01 18:35:59 -0700236 // Timer that records time between requests for next vsync.
237 std::optional<scheduler::OneShotTimer> mIdleTimer;
Ady Abraham8532d012019-05-08 14:50:56 -0700238 // Timer used to monitor touch events.
Dominik Laskowski98041832019-08-01 18:35:59 -0700239 std::optional<scheduler::OneShotTimer> mTouchTimer;
Ady Abraham6fe2c172019-07-12 12:37:57 -0700240 // Timer used to monitor display power mode.
Dominik Laskowski98041832019-08-01 18:35:59 -0700241 std::optional<scheduler::OneShotTimer> mDisplayPowerTimer;
Ady Abraham6fe2c172019-07-12 12:37:57 -0700242
Ady Abraham3a77a7b2019-12-02 18:46:59 -0800243 ISchedulerCallback& mSchedulerCallback;
Ana Krulecfefd6ae2019-02-13 17:53:08 -0800244
245 // In order to make sure that the features don't override themselves, we need a state machine
246 // to keep track which feature requested the config change.
247 std::mutex mFeatureStateLock;
Dominik Laskowskidd252cd2019-07-26 09:10:16 -0700248
249 struct {
Ady Abraham4ccdcb42020-02-11 17:34:34 -0800250 ContentDetectionState contentDetectionV1 = ContentDetectionState::Off;
Dominik Laskowskidd252cd2019-07-26 09:10:16 -0700251 TimerState idleTimer = TimerState::Reset;
252 TouchState touch = TouchState::Inactive;
253 TimerState displayPowerTimer = TimerState::Expired;
254
Ady Abraham2139f732019-11-13 18:56:40 -0800255 std::optional<HwcConfigIndexType> configId;
Ady Abraham5def7332020-05-29 16:13:47 -0700256 LayerHistory::Summary contentRequirements;
Dominik Laskowskidd252cd2019-07-26 09:10:16 -0700257
Dominik Laskowskidd252cd2019-07-26 09:10:16 -0700258 bool isDisplayPowerStateNormal = true;
Ady Abrahamdfd62162020-06-10 16:11:56 -0700259
260 // Used to cache the last parameters of onPrimaryDisplayConfigChanged
261 struct ConfigChangedParams {
262 ConnectionHandle handle;
263 PhysicalDisplayId displayId;
264 HwcConfigIndexType configId;
265 nsecs_t vsyncPeriod;
266 };
267
268 std::optional<ConfigChangedParams> cachedConfigChangedParams;
Dominik Laskowskidd252cd2019-07-26 09:10:16 -0700269 } mFeatures GUARDED_BY(mFeatureStateLock);
Ady Abraham09bd3922019-04-08 10:44:56 -0700270
271 const scheduler::RefreshRateConfigs& mRefreshRateConfigs;
Ady Abraham3a77a7b2019-12-02 18:46:59 -0800272
273 std::mutex mVsyncTimelineLock;
Peiyong Line9d809e2020-04-14 13:10:48 -0700274 std::optional<hal::VsyncPeriodChangeTimeline> mLastVsyncPeriodChangeTimeline
Ady Abraham3a77a7b2019-12-02 18:46:59 -0800275 GUARDED_BY(mVsyncTimelineLock);
276 static constexpr std::chrono::nanoseconds MAX_VSYNC_APPLIED_TIME = 200ms;
Ady Abraham8a82ba62020-01-17 12:43:17 -0800277
Ana Krulec3d367c82020-02-25 15:02:01 -0800278 // This variable indicates whether to use the content detection feature at all.
279 const bool mUseContentDetection;
280 // This variable indicates whether to use V2 version of the content detection.
Ady Abraham8a82ba62020-01-17 12:43:17 -0800281 const bool mUseContentDetectionV2;
Ana Krulec98b5b242018-08-10 15:03:23 -0700282};
283
Ana Krulec7ab56032018-11-02 20:51:06 +0100284} // namespace android