blob: 34e527cdc62640933728a00f7ab0589b8772f8f7 [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
Dominik Laskowski98041832019-08-01 18:35:59 -070037class DispSync;
38class FenceTime;
39struct DisplayStateInfo;
Ana Krulece588e312018-09-18 12:32:24 -070040
Ana Krulec98b5b242018-08-10 15:03:23 -070041class Scheduler {
42public:
Ana Krulec8d3e4f32019-03-05 10:40:33 -080043 using RefreshRateType = scheduler::RefreshRateConfigs::RefreshRateType;
Dominik Laskowski98041832019-08-01 18:35:59 -070044 using ConfigEvent = scheduler::RefreshRateConfigEvent;
45
Alec Mouri7f015182019-07-11 13:56:22 -070046 using GetCurrentRefreshRateTypeCallback = std::function<RefreshRateType()>;
Ana Krulec8d3e4f32019-03-05 10:40:33 -080047 using ChangeRefreshRateCallback = std::function<void(RefreshRateType, ConfigEvent)>;
Ana Krulecc2870422019-01-29 19:00:58 -080048 using GetVsyncPeriod = std::function<nsecs_t()>;
Ana Krulec7d1d6832018-12-27 11:10:09 -080049
Dominik Laskowski98041832019-08-01 18:35:59 -070050 // Indicates whether to start the transaction early, or at vsync time.
Ana Krulec7ecce8c2018-10-12 13:44:41 -070051 enum class TransactionStart { EARLY, NORMAL };
52
Dominik Laskowski98041832019-08-01 18:35:59 -070053 Scheduler(impl::EventControlThread::SetVSyncEnabledFunction,
54 const scheduler::RefreshRateConfigs&);
Ana Krulece588e312018-09-18 12:32:24 -070055
Ana Krulec0c8cd522018-08-31 12:27:28 -070056 virtual ~Scheduler();
Ana Krulec98b5b242018-08-10 15:03:23 -070057
Dominik Laskowski98041832019-08-01 18:35:59 -070058 DispSync& getPrimaryDispSync();
Ana Krulece588e312018-09-18 12:32:24 -070059
Dominik Laskowski98041832019-08-01 18:35:59 -070060 using ConnectionHandle = scheduler::ConnectionHandle;
61 ConnectionHandle createConnection(const char* connectionName, nsecs_t phaseOffsetNs,
62 nsecs_t offsetThresholdForNextVsync, ResyncCallback,
63 impl::EventThread::InterceptVSyncsCallback);
Ana Krulec98b5b242018-08-10 15:03:23 -070064
Dominik Laskowski98041832019-08-01 18:35:59 -070065 sp<IDisplayEventConnection> createDisplayEventConnection(ConnectionHandle, ResyncCallback,
66 ISurfaceComposer::ConfigChanged);
Ana Krulece588e312018-09-18 12:32:24 -070067
Dominik Laskowski98041832019-08-01 18:35:59 -070068 EventThread* getEventThread(ConnectionHandle);
69 sp<EventThreadConnection> getEventConnection(ConnectionHandle);
Kevin DuBois413287f2019-02-25 08:46:47 -080070
Dominik Laskowski98041832019-08-01 18:35:59 -070071 void onHotplugReceived(ConnectionHandle, PhysicalDisplayId, bool connected);
72 void onConfigChanged(ConnectionHandle, PhysicalDisplayId, int32_t configId);
Ana Krulec98b5b242018-08-10 15:03:23 -070073
Dominik Laskowski98041832019-08-01 18:35:59 -070074 void onScreenAcquired(ConnectionHandle);
75 void onScreenReleased(ConnectionHandle);
Ana Krulece588e312018-09-18 12:32:24 -070076
Dominik Laskowski98041832019-08-01 18:35:59 -070077 // Modifies phase offset in the event thread.
78 void setPhaseOffset(ConnectionHandle, nsecs_t phaseOffset);
Ana Krulec98b5b242018-08-10 15:03:23 -070079
Ana Krulece588e312018-09-18 12:32:24 -070080 void getDisplayStatInfo(DisplayStatInfo* stats);
81
82 void enableHardwareVsync();
83 void disableHardwareVsync(bool makeUnavailable);
Dominik Laskowski98041832019-08-01 18:35:59 -070084
Alec Mourif8e689c2019-05-20 18:32:22 -070085 // Resyncs the scheduler to hardware vsync.
86 // If makeAvailable is true, then hardware vsync will be turned on.
87 // Otherwise, if hardware vsync is not already enabled then this method will
88 // no-op.
89 // The period is the vsync period from the current display configuration.
Ana Krulecc2870422019-01-29 19:00:58 -080090 void resyncToHardwareVsync(bool makeAvailable, nsecs_t period);
Dominik Laskowski98041832019-08-01 18:35:59 -070091 ResyncCallback makeResyncCallback(GetVsyncPeriod&&);
92
Alec Mourif8e689c2019-05-20 18:32:22 -070093 // Passes a vsync sample to DispSync. periodFlushed will be true if
94 // DispSync detected that the vsync period changed, and false otherwise.
Dominik Laskowski98041832019-08-01 18:35:59 -070095 void addResyncSample(nsecs_t timestamp, bool* periodFlushed);
96 void addPresentFence(const std::shared_ptr<FenceTime>&);
Ana Krulece588e312018-09-18 12:32:24 -070097 void setIgnorePresentFences(bool ignore);
Ady Abraham8fe11022019-06-12 17:11:12 -070098 nsecs_t getDispSyncExpectedPresentTime();
Ady Abraham09bd3922019-04-08 10:44:56 -070099 // Registers the layer in the scheduler, and returns the handle for future references.
Ady Abraham8f1ee7f2019-04-05 10:32:50 -0700100 std::unique_ptr<scheduler::LayerHistory::LayerHandle> registerLayer(std::string const& name,
101 int windowType);
102
Ady Abraham09bd3922019-04-08 10:44:56 -0700103 // Stores present time for a layer.
Ady Abrahama315ce72019-04-24 14:35:20 -0700104 void addLayerPresentTimeAndHDR(
Ady Abraham09bd3922019-04-08 10:44:56 -0700105 const std::unique_ptr<scheduler::LayerHistory::LayerHandle>& layerHandle,
Ady Abrahama315ce72019-04-24 14:35:20 -0700106 nsecs_t presentTime, bool isHDR);
107 // Stores visibility for a layer.
108 void setLayerVisibility(
109 const std::unique_ptr<scheduler::LayerHistory::LayerHandle>& layerHandle, bool visible);
Ady Abraham09bd3922019-04-08 10:44:56 -0700110 // Updates FPS based on the most content presented.
111 void updateFpsBasedOnContent();
Ady Abraham97d04232019-03-05 19:48:12 -0800112
Dominik Laskowski98041832019-08-01 18:35:59 -0700113 // Called by Scheduler to change refresh rate.
114 void setChangeRefreshRateCallback(ChangeRefreshRateCallback&&);
Ady Abraham09bd3922019-04-08 10:44:56 -0700115
Dominik Laskowski98041832019-08-01 18:35:59 -0700116 void setGetCurrentRefreshRateTypeCallback(GetCurrentRefreshRateTypeCallback&&);
117 void setGetVsyncPeriodCallback(GetVsyncPeriod&&);
118
119 bool isIdleTimerEnabled() const { return mIdleTimer.has_value(); }
Ady Abraham8532d012019-05-08 14:50:56 -0700120 void resetIdleTimer();
121
122 // Function that resets the touch timer.
123 void notifyTouchEvent();
124
Ady Abraham6fe2c172019-07-12 12:37:57 -0700125 void setDisplayPowerState(bool normal);
126
Dominik Laskowski98041832019-08-01 18:35:59 -0700127 void dump(std::string&) const;
128 void dump(ConnectionHandle, std::string&) const;
Ana Krulec0c8cd522018-08-31 12:27:28 -0700129
Daniel Solomon0f0ddc12019-08-19 19:31:09 -0700130 // Get the appropriate refresh type for current conditions.
131 RefreshRateType getPreferredRefreshRateType();
132
Ana Krulec98b5b242018-08-10 15:03:23 -0700133private:
Ana Krulecafb45842019-02-13 13:33:03 -0800134 friend class TestableScheduler;
135
Ana Krulecfefd6ae2019-02-13 17:53:08 -0800136 // In order to make sure that the features don't override themselves, we need a state machine
137 // to keep track which feature requested the config change.
Dominik Laskowskidd252cd2019-07-26 09:10:16 -0700138 enum class ContentDetectionState { Off, On };
139 enum class TimerState { Reset, Expired };
140 enum class TouchState { Inactive, Active };
Ana Krulecfefd6ae2019-02-13 17:53:08 -0800141
Dominik Laskowski7c9dbf92019-08-01 17:57:31 -0700142 // Used by tests to inject mocks.
143 Scheduler(std::unique_ptr<DispSync>, std::unique_ptr<EventControlThread>,
144 const scheduler::RefreshRateConfigs&);
145
Dominik Laskowskiccf37d72019-02-01 16:47:58 -0800146 // Creates a connection on the given EventThread and forwards the given callbacks.
Dominik Laskowski98041832019-08-01 18:35:59 -0700147 std::unique_ptr<EventThread> makeEventThread(const char* connectionName, nsecs_t phaseOffsetNs,
148 nsecs_t offsetThresholdForNextVsync,
149 impl::EventThread::InterceptVSyncsCallback&&);
150
151 // Create a connection on the given EventThread and forward the resync callback.
152 ConnectionHandle createConnection(std::unique_ptr<EventThread>, ResyncCallback&&);
Ady Abraham0f4a1b12019-06-04 16:04:04 -0700153 sp<EventThreadConnection> createConnectionInternal(EventThread*, ResyncCallback&&,
154 ISurfaceComposer::ConfigChanged);
Dominik Laskowskiccf37d72019-02-01 16:47:58 -0800155
Dominik Laskowski3a80a382019-07-25 11:16:07 -0700156 // Update feature state machine to given state when corresponding timer resets or expires.
157 void kernelIdleTimerCallback(TimerState);
158 void idleTimerCallback(TimerState);
159 void touchTimerCallback(TimerState);
160 void displayPowerTimerCallback(TimerState);
161
Ady Abraham6fe2c172019-07-12 12:37:57 -0700162 // handles various timer features to change the refresh rate.
163 template <class T>
164 void handleTimerStateChanged(T* currentState, T newState, bool eventOnContentDetection);
Dominik Laskowski98041832019-08-01 18:35:59 -0700165
166 void setVsyncPeriod(nsecs_t period);
167
Ady Abraham09bd3922019-04-08 10:44:56 -0700168 RefreshRateType calculateRefreshRateType() REQUIRES(mFeatureStateLock);
Dominik Laskowski98041832019-08-01 18:35:59 -0700169 // Acquires a lock and calls the ChangeRefreshRateCallback with given parameters.
170 void changeRefreshRate(RefreshRateType, ConfigEvent);
Ana Krulec7ab56032018-11-02 20:51:06 +0100171
Dominik Laskowski98041832019-08-01 18:35:59 -0700172 // Stores EventThread associated with a given VSyncSource, and an initial EventThreadConnection.
173 struct Connection {
174 sp<EventThreadConnection> connection;
175 std::unique_ptr<EventThread> thread;
176 };
Ady Abraham09bd3922019-04-08 10:44:56 -0700177
Dominik Laskowski98041832019-08-01 18:35:59 -0700178 ConnectionHandle::Id mNextConnectionHandleId = 0;
179 std::unordered_map<ConnectionHandle, Connection> mConnections;
Ana Krulece588e312018-09-18 12:32:24 -0700180
181 std::mutex mHWVsyncLock;
Dominik Laskowski98041832019-08-01 18:35:59 -0700182 bool mPrimaryHWVsyncEnabled GUARDED_BY(mHWVsyncLock) = false;
183 bool mHWVsyncAvailable GUARDED_BY(mHWVsyncLock) = false;
184
185 // Stores per-display state about VSYNC.
186 struct VsyncState {
187 explicit VsyncState(Scheduler& scheduler) : scheduler(scheduler) {}
188
189 void resync(const GetVsyncPeriod&);
190
191 Scheduler& scheduler;
192 std::atomic<nsecs_t> lastResyncTime = 0;
193 };
194
Ana Krulecc2870422019-01-29 19:00:58 -0800195 const std::shared_ptr<VsyncState> mPrimaryVsyncState{std::make_shared<VsyncState>(*this)};
Ana Krulece588e312018-09-18 12:32:24 -0700196
197 std::unique_ptr<DispSync> mPrimaryDispSync;
198 std::unique_ptr<EventControlThread> mEventControlThread;
Ana Krulec7ab56032018-11-02 20:51:06 +0100199
Ady Abraham09bd3922019-04-08 10:44:56 -0700200 // Historical information about individual layers. Used for predicting the refresh rate.
201 scheduler::LayerHistory mLayerHistory;
Ana Krulecfb772822018-11-30 10:44:07 +0100202
Dominik Laskowski98041832019-08-01 18:35:59 -0700203 // Whether to use idle timer callbacks that support the kernel timer.
204 const bool mSupportKernelTimer;
Ana Krulec7d1d6832018-12-27 11:10:09 -0800205
Dominik Laskowski98041832019-08-01 18:35:59 -0700206 // Timer that records time between requests for next vsync.
207 std::optional<scheduler::OneShotTimer> mIdleTimer;
Ady Abraham8532d012019-05-08 14:50:56 -0700208 // Timer used to monitor touch events.
Dominik Laskowski98041832019-08-01 18:35:59 -0700209 std::optional<scheduler::OneShotTimer> mTouchTimer;
Ady Abraham6fe2c172019-07-12 12:37:57 -0700210 // Timer used to monitor display power mode.
Dominik Laskowski98041832019-08-01 18:35:59 -0700211 std::optional<scheduler::OneShotTimer> mDisplayPowerTimer;
Ady Abraham6fe2c172019-07-12 12:37:57 -0700212
Ana Krulec7d1d6832018-12-27 11:10:09 -0800213 std::mutex mCallbackLock;
Alec Mouri7f015182019-07-11 13:56:22 -0700214 GetCurrentRefreshRateTypeCallback mGetCurrentRefreshRateTypeCallback GUARDED_BY(mCallbackLock);
Ana Krulec8d3e4f32019-03-05 10:40:33 -0800215 ChangeRefreshRateCallback mChangeRefreshRateCallback GUARDED_BY(mCallbackLock);
Alec Mouridc28b372019-04-18 21:17:13 -0700216 GetVsyncPeriod mGetVsyncPeriod GUARDED_BY(mCallbackLock);
Ana Krulecfefd6ae2019-02-13 17:53:08 -0800217
218 // In order to make sure that the features don't override themselves, we need a state machine
219 // to keep track which feature requested the config change.
220 std::mutex mFeatureStateLock;
Dominik Laskowskidd252cd2019-07-26 09:10:16 -0700221
222 struct {
223 ContentDetectionState contentDetection = ContentDetectionState::Off;
224 TimerState idleTimer = TimerState::Reset;
225 TouchState touch = TouchState::Inactive;
226 TimerState displayPowerTimer = TimerState::Expired;
227
228 RefreshRateType refreshRateType = RefreshRateType::DEFAULT;
229 uint32_t contentRefreshRate = 0;
230
231 bool isHDRContent = false;
232 bool isDisplayPowerStateNormal = true;
233 } mFeatures GUARDED_BY(mFeatureStateLock);
Ady Abraham09bd3922019-04-08 10:44:56 -0700234
235 const scheduler::RefreshRateConfigs& mRefreshRateConfigs;
Ady Abrahama315ce72019-04-24 14:35:20 -0700236
237 // Global config to force HDR content to work on DEFAULT refreshRate
Ady Abraham8444c362019-05-30 17:59:36 -0700238 static constexpr bool mForceHDRContentToDefaultRefreshRate = false;
Ana Krulec98b5b242018-08-10 15:03:23 -0700239};
240
Ana Krulec7ab56032018-11-02 20:51:06 +0100241} // namespace android