blob: 2987424c9931e3de9f7cd3b95bd99b7187887527 [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
Dominik Laskowskidcb38bb2019-01-25 02:35:50 -080026#include <ui/GraphicTypes.h>
Ana Krulec98b5b242018-08-10 15:03:23 -070027
Ana Krulece588e312018-09-18 12:32:24 -070028#include "EventControlThread.h"
Ana Krulec98b5b242018-08-10 15:03:23 -070029#include "EventThread.h"
Ana Krulec3084c052018-11-21 20:27:17 +010030#include "LayerHistory.h"
Ana Krulecf2c006d2019-06-21 15:37:07 -070031#include "OneShotTimer.h"
Ana Krulec8d3e4f32019-03-05 10:40:33 -080032#include "RefreshRateConfigs.h"
Ana Krulec434c22d2018-11-28 13:48:36 +010033#include "SchedulerUtils.h"
Ana Krulec98b5b242018-08-10 15:03:23 -070034
35namespace android {
36
Ady Abraham2139f732019-11-13 18:56:40 -080037using namespace std::chrono_literals;
38
Dominik Laskowski98041832019-08-01 18:35:59 -070039class DispSync;
40class FenceTime;
Dominik Laskowski6505f792019-09-18 11:10:05 -070041class InjectVSyncSource;
Dominik Laskowski98041832019-08-01 18:35:59 -070042struct DisplayStateInfo;
Ana Krulece588e312018-09-18 12:32:24 -070043
Ady Abraham3a77a7b2019-12-02 18:46:59 -080044class ISchedulerCallback {
45public:
46 virtual ~ISchedulerCallback() = default;
47 virtual void changeRefreshRate(const scheduler::RefreshRateConfigs::RefreshRate&,
48 scheduler::RefreshRateConfigEvent) = 0;
49 virtual void repaintEverythingForHWC() = 0;
50};
51
Ana Krulec98b5b242018-08-10 15:03:23 -070052class Scheduler {
53public:
Ady Abraham2139f732019-11-13 18:56:40 -080054 using RefreshRate = scheduler::RefreshRateConfigs::RefreshRate;
Dominik Laskowski98041832019-08-01 18:35:59 -070055 using ConfigEvent = scheduler::RefreshRateConfigEvent;
56
Dominik Laskowski98041832019-08-01 18:35:59 -070057 // Indicates whether to start the transaction early, or at vsync time.
Ana Krulec7ecce8c2018-10-12 13:44:41 -070058 enum class TransactionStart { EARLY, NORMAL };
59
Dominik Laskowski98041832019-08-01 18:35:59 -070060 Scheduler(impl::EventControlThread::SetVSyncEnabledFunction,
Ady Abraham8a82ba62020-01-17 12:43:17 -080061 const scheduler::RefreshRateConfigs&, ISchedulerCallback& schedulerCallback,
62 bool useContentDetectionV2);
Ana Krulece588e312018-09-18 12:32:24 -070063
Ana Krulec0c8cd522018-08-31 12:27:28 -070064 virtual ~Scheduler();
Ana Krulec98b5b242018-08-10 15:03:23 -070065
Dominik Laskowski98041832019-08-01 18:35:59 -070066 DispSync& getPrimaryDispSync();
Ana Krulece588e312018-09-18 12:32:24 -070067
Dominik Laskowski98041832019-08-01 18:35:59 -070068 using ConnectionHandle = scheduler::ConnectionHandle;
69 ConnectionHandle createConnection(const char* connectionName, nsecs_t phaseOffsetNs,
Dominik Laskowski98041832019-08-01 18:35:59 -070070 impl::EventThread::InterceptVSyncsCallback);
Ana Krulec98b5b242018-08-10 15:03:23 -070071
Steven Thomas2bbaabe2019-08-28 16:08:35 -070072 sp<IDisplayEventConnection> createDisplayEventConnection(ConnectionHandle,
Dominik Laskowski98041832019-08-01 18:35:59 -070073 ISurfaceComposer::ConfigChanged);
Ana Krulece588e312018-09-18 12:32:24 -070074
Dominik Laskowski98041832019-08-01 18:35:59 -070075 sp<EventThreadConnection> getEventConnection(ConnectionHandle);
Kevin DuBois413287f2019-02-25 08:46:47 -080076
Dominik Laskowski98041832019-08-01 18:35:59 -070077 void onHotplugReceived(ConnectionHandle, PhysicalDisplayId, bool connected);
Alec Mouri60aee1c2019-10-28 16:18:59 -070078 void onConfigChanged(ConnectionHandle, PhysicalDisplayId, HwcConfigIndexType configId,
79 nsecs_t vsyncPeriod);
Ana Krulec98b5b242018-08-10 15:03:23 -070080
Dominik Laskowski98041832019-08-01 18:35:59 -070081 void onScreenAcquired(ConnectionHandle);
82 void onScreenReleased(ConnectionHandle);
Ana Krulece588e312018-09-18 12:32:24 -070083
Dominik Laskowski98041832019-08-01 18:35:59 -070084 // Modifies phase offset in the event thread.
85 void setPhaseOffset(ConnectionHandle, nsecs_t phaseOffset);
Ana Krulec98b5b242018-08-10 15:03:23 -070086
Ana Krulece588e312018-09-18 12:32:24 -070087 void getDisplayStatInfo(DisplayStatInfo* stats);
88
Dominik Laskowski6505f792019-09-18 11:10:05 -070089 // Returns injector handle if injection has toggled, or an invalid handle otherwise.
90 ConnectionHandle enableVSyncInjection(bool enable);
91
92 // Returns false if injection is disabled.
93 bool injectVSync(nsecs_t when);
94
Ana Krulece588e312018-09-18 12:32:24 -070095 void enableHardwareVsync();
96 void disableHardwareVsync(bool makeUnavailable);
Dominik Laskowski98041832019-08-01 18:35:59 -070097
Alec Mourif8e689c2019-05-20 18:32:22 -070098 // Resyncs the scheduler to hardware vsync.
99 // If makeAvailable is true, then hardware vsync will be turned on.
100 // Otherwise, if hardware vsync is not already enabled then this method will
101 // no-op.
102 // The period is the vsync period from the current display configuration.
Ana Krulecc2870422019-01-29 19:00:58 -0800103 void resyncToHardwareVsync(bool makeAvailable, nsecs_t period);
Steven Thomas2bbaabe2019-08-28 16:08:35 -0700104 void resync();
Dominik Laskowski98041832019-08-01 18:35:59 -0700105
Alec Mourif8e689c2019-05-20 18:32:22 -0700106 // Passes a vsync sample to DispSync. periodFlushed will be true if
107 // DispSync detected that the vsync period changed, and false otherwise.
Dominik Laskowski98041832019-08-01 18:35:59 -0700108 void addResyncSample(nsecs_t timestamp, bool* periodFlushed);
109 void addPresentFence(const std::shared_ptr<FenceTime>&);
Ana Krulece588e312018-09-18 12:32:24 -0700110 void setIgnorePresentFences(bool ignore);
Ady Abraham8fe11022019-06-12 17:11:12 -0700111 nsecs_t getDispSyncExpectedPresentTime();
Ady Abraham8f1ee7f2019-04-05 10:32:50 -0700112
Dominik Laskowskif7a09ed2019-10-07 13:54:18 -0700113 // Layers are registered on creation, and unregistered when the weak reference expires.
114 void registerLayer(Layer*);
Ady Abraham2139f732019-11-13 18:56:40 -0800115 void recordLayerHistory(Layer*, nsecs_t presentTime);
Dominik Laskowskif7a09ed2019-10-07 13:54:18 -0700116
Dominik Laskowski49cea512019-11-12 14:13:23 -0800117 // Detects content using layer history, and selects a matching refresh rate.
118 void chooseRefreshRateForContent();
Ady Abraham97d04232019-03-05 19:48:12 -0800119
Dominik Laskowski98041832019-08-01 18:35:59 -0700120 bool isIdleTimerEnabled() const { return mIdleTimer.has_value(); }
Ady Abraham8532d012019-05-08 14:50:56 -0700121 void resetIdleTimer();
122
123 // Function that resets the touch timer.
124 void notifyTouchEvent();
125
Ady Abraham6fe2c172019-07-12 12:37:57 -0700126 void setDisplayPowerState(bool normal);
127
Dominik Laskowski98041832019-08-01 18:35:59 -0700128 void dump(std::string&) const;
129 void dump(ConnectionHandle, std::string&) const;
Ana Krulec0c8cd522018-08-31 12:27:28 -0700130
Ady Abraham2139f732019-11-13 18:56:40 -0800131 // Get the appropriate refresh for current conditions.
132 std::optional<HwcConfigIndexType> getPreferredConfigId();
Daniel Solomon0f0ddc12019-08-19 19:31:09 -0700133
Ady Abraham3a77a7b2019-12-02 18:46:59 -0800134 // Notifies the scheduler about a refresh rate timeline change.
135 void onNewVsyncPeriodChangeTimeline(const HWC2::VsyncPeriodChangeTimeline& timeline);
136
137 // Notifies the scheduler when the display was refreshed
138 void onDisplayRefreshed(nsecs_t timestamp);
139
Ady Abraham8a82ba62020-01-17 12:43:17 -0800140 // Notifies the scheduler when the display size has changed. Called from SF's main thread
141 void onPrimaryDisplayAreaChanged(uint32_t displayArea);
142
Ana Krulec98b5b242018-08-10 15:03:23 -0700143private:
Ana Krulecafb45842019-02-13 13:33:03 -0800144 friend class TestableScheduler;
145
Ana Krulecfefd6ae2019-02-13 17:53:08 -0800146 // In order to make sure that the features don't override themselves, we need a state machine
147 // to keep track which feature requested the config change.
Dominik Laskowskidd252cd2019-07-26 09:10:16 -0700148 enum class ContentDetectionState { Off, On };
149 enum class TimerState { Reset, Expired };
150 enum class TouchState { Inactive, Active };
Ana Krulecfefd6ae2019-02-13 17:53:08 -0800151
Dominik Laskowski7c9dbf92019-08-01 17:57:31 -0700152 // Used by tests to inject mocks.
153 Scheduler(std::unique_ptr<DispSync>, std::unique_ptr<EventControlThread>,
Ady Abraham8a82ba62020-01-17 12:43:17 -0800154 const scheduler::RefreshRateConfigs&, ISchedulerCallback& schedulerCallback,
155 bool useContentDetectionV2);
Dominik Laskowski7c9dbf92019-08-01 17:57:31 -0700156
Ady Abraham9e16a482019-12-03 17:19:41 -0800157 std::unique_ptr<VSyncSource> makePrimaryDispSyncSource(const char* name, nsecs_t phaseOffsetNs);
Dominik Laskowski98041832019-08-01 18:35:59 -0700158
Dominik Laskowski6505f792019-09-18 11:10:05 -0700159 // Create a connection on the given EventThread.
Steven Thomas2bbaabe2019-08-28 16:08:35 -0700160 ConnectionHandle createConnection(std::unique_ptr<EventThread>);
161 sp<EventThreadConnection> createConnectionInternal(EventThread*,
Ady Abraham0f4a1b12019-06-04 16:04:04 -0700162 ISurfaceComposer::ConfigChanged);
Dominik Laskowskiccf37d72019-02-01 16:47:58 -0800163
Dominik Laskowski3a80a382019-07-25 11:16:07 -0700164 // Update feature state machine to given state when corresponding timer resets or expires.
165 void kernelIdleTimerCallback(TimerState);
166 void idleTimerCallback(TimerState);
167 void touchTimerCallback(TimerState);
168 void displayPowerTimerCallback(TimerState);
169
Ady Abraham6fe2c172019-07-12 12:37:57 -0700170 // handles various timer features to change the refresh rate.
171 template <class T>
172 void handleTimerStateChanged(T* currentState, T newState, bool eventOnContentDetection);
Dominik Laskowski98041832019-08-01 18:35:59 -0700173
174 void setVsyncPeriod(nsecs_t period);
175
Ady Abraham2139f732019-11-13 18:56:40 -0800176 HwcConfigIndexType calculateRefreshRateType() REQUIRES(mFeatureStateLock);
Ana Krulec7ab56032018-11-02 20:51:06 +0100177
Ady Abraham8a82ba62020-01-17 12:43:17 -0800178 bool layerHistoryHasClientSpecifiedFrameRate() REQUIRES(mFeatureStateLock);
179
Dominik Laskowski98041832019-08-01 18:35:59 -0700180 // Stores EventThread associated with a given VSyncSource, and an initial EventThreadConnection.
181 struct Connection {
182 sp<EventThreadConnection> connection;
183 std::unique_ptr<EventThread> thread;
184 };
Ady Abraham09bd3922019-04-08 10:44:56 -0700185
Dominik Laskowski98041832019-08-01 18:35:59 -0700186 ConnectionHandle::Id mNextConnectionHandleId = 0;
187 std::unordered_map<ConnectionHandle, Connection> mConnections;
Ana Krulece588e312018-09-18 12:32:24 -0700188
Dominik Laskowski6505f792019-09-18 11:10:05 -0700189 bool mInjectVSyncs = false;
190 InjectVSyncSource* mVSyncInjector = nullptr;
191 ConnectionHandle mInjectorConnectionHandle;
192
Ana Krulece588e312018-09-18 12:32:24 -0700193 std::mutex mHWVsyncLock;
Dominik Laskowski98041832019-08-01 18:35:59 -0700194 bool mPrimaryHWVsyncEnabled GUARDED_BY(mHWVsyncLock) = false;
195 bool mHWVsyncAvailable GUARDED_BY(mHWVsyncLock) = false;
196
Steven Thomas2bbaabe2019-08-28 16:08:35 -0700197 std::atomic<nsecs_t> mLastResyncTime = 0;
Ana Krulece588e312018-09-18 12:32:24 -0700198
199 std::unique_ptr<DispSync> mPrimaryDispSync;
200 std::unique_ptr<EventControlThread> mEventControlThread;
Ana Krulec7ab56032018-11-02 20:51:06 +0100201
Dominik Laskowski49cea512019-11-12 14:13:23 -0800202 // Used to choose refresh rate if content detection is enabled.
Ady Abrahame3ed2f92020-01-06 17:01:28 -0800203 std::unique_ptr<scheduler::LayerHistory> mLayerHistory;
Ana Krulecfb772822018-11-30 10:44:07 +0100204
Dominik Laskowski98041832019-08-01 18:35:59 -0700205 // Whether to use idle timer callbacks that support the kernel timer.
206 const bool mSupportKernelTimer;
Ana Krulec7d1d6832018-12-27 11:10:09 -0800207
Dominik Laskowski98041832019-08-01 18:35:59 -0700208 // Timer that records time between requests for next vsync.
209 std::optional<scheduler::OneShotTimer> mIdleTimer;
Ady Abraham8532d012019-05-08 14:50:56 -0700210 // Timer used to monitor touch events.
Dominik Laskowski98041832019-08-01 18:35:59 -0700211 std::optional<scheduler::OneShotTimer> mTouchTimer;
Ady Abraham6fe2c172019-07-12 12:37:57 -0700212 // Timer used to monitor display power mode.
Dominik Laskowski98041832019-08-01 18:35:59 -0700213 std::optional<scheduler::OneShotTimer> mDisplayPowerTimer;
Ady Abraham6fe2c172019-07-12 12:37:57 -0700214
Ady Abraham3a77a7b2019-12-02 18:46:59 -0800215 ISchedulerCallback& mSchedulerCallback;
Ana Krulecfefd6ae2019-02-13 17:53:08 -0800216
217 // In order to make sure that the features don't override themselves, we need a state machine
218 // to keep track which feature requested the config change.
219 std::mutex mFeatureStateLock;
Dominik Laskowskidd252cd2019-07-26 09:10:16 -0700220
221 struct {
222 ContentDetectionState contentDetection = ContentDetectionState::Off;
223 TimerState idleTimer = TimerState::Reset;
224 TouchState touch = TouchState::Inactive;
225 TimerState displayPowerTimer = TimerState::Expired;
226
Ady Abraham2139f732019-11-13 18:56:40 -0800227 std::optional<HwcConfigIndexType> configId;
Ady Abraham8a82ba62020-01-17 12:43:17 -0800228 scheduler::LayerHistory::Summary contentRequirements;
Dominik Laskowskidd252cd2019-07-26 09:10:16 -0700229
Dominik Laskowskidd252cd2019-07-26 09:10:16 -0700230 bool isDisplayPowerStateNormal = true;
231 } mFeatures GUARDED_BY(mFeatureStateLock);
Ady Abraham09bd3922019-04-08 10:44:56 -0700232
233 const scheduler::RefreshRateConfigs& mRefreshRateConfigs;
Ady Abraham3a77a7b2019-12-02 18:46:59 -0800234
235 std::mutex mVsyncTimelineLock;
236 std::optional<HWC2::VsyncPeriodChangeTimeline> mLastVsyncPeriodChangeTimeline
237 GUARDED_BY(mVsyncTimelineLock);
238 static constexpr std::chrono::nanoseconds MAX_VSYNC_APPLIED_TIME = 200ms;
Ady Abraham8a82ba62020-01-17 12:43:17 -0800239
240 const bool mUseContentDetectionV2;
Ana Krulec98b5b242018-08-10 15:03:23 -0700241};
242
Ana Krulec7ab56032018-11-02 20:51:06 +0100243} // namespace android