blob: 8fd7e6046d74f378bcdfb73f8850b1ef641fc875 [file] [log] [blame]
Kevin DuBois1678e2c2019-08-22 12:26:24 -07001/*
2 * Copyright 2019 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
Ady Abrahame9883032023-11-20 17:54:54 -080019#include <deque>
Kevin DuBois1678e2c2019-08-22 12:26:24 -070020#include <mutex>
21#include <unordered_map>
22#include <vector>
Dominik Laskowski62eff352021-12-06 09:59:41 -080023
24#include <android-base/thread_annotations.h>
Leon Scroggins III67388622023-02-06 20:36:20 -050025#include <ui/DisplayId.h>
Dominik Laskowski62eff352021-12-06 09:59:41 -080026
Kevin DuBois1678e2c2019-08-22 12:26:24 -070027#include "VSyncTracker.h"
28
29namespace android::scheduler {
30
31class VSyncPredictor : public VSyncTracker {
32public:
33 /*
Leon Scroggins III67388622023-02-06 20:36:20 -050034 * \param [in] PhysicalDisplayid The display this corresponds to.
Ady Abrahamc585dba2023-11-15 18:41:35 -080035 * \param [in] modePtr The initial display mode
Kevin DuBois1678e2c2019-08-22 12:26:24 -070036 * \param [in] historySize The internal amount of entries to store in the model.
37 * \param [in] minimumSamplesForPrediction The minimum number of samples to collect before
38 * predicting. \param [in] outlierTolerancePercent a number 0 to 100 that will be used to filter
39 * samples that fall outlierTolerancePercent from an anticipated vsync event.
40 */
Ady Abrahamc585dba2023-11-15 18:41:35 -080041 VSyncPredictor(ftl::NonNull<DisplayModePtr> modePtr, size_t historySize,
ramindaniae645822024-01-11 10:57:29 -080042 size_t minimumSamplesForPrediction, uint32_t outlierTolerancePercent);
Kevin DuBois1678e2c2019-08-22 12:26:24 -070043 ~VSyncPredictor();
44
Ady Abraham0bb6a472020-10-12 10:22:13 -070045 bool addVsyncTimestamp(nsecs_t timestamp) final EXCLUDES(mMutex);
Ady Abraham4335afd2023-12-18 19:10:47 -080046 nsecs_t nextAnticipatedVSyncTimeFrom(nsecs_t timePoint,
47 std::optional<nsecs_t> lastVsyncOpt = {}) const final
48 EXCLUDES(mMutex);
Ady Abraham0bb6a472020-10-12 10:22:13 -070049 nsecs_t currentPeriod() const final EXCLUDES(mMutex);
Ady Abraham3db8a3c2023-11-20 17:53:47 -080050 Period minFramePeriod() const final EXCLUDES(mMutex);
Ady Abraham0bb6a472020-10-12 10:22:13 -070051 void resetModel() final EXCLUDES(mMutex);
Kevin DuBois1678e2c2019-08-22 12:26:24 -070052
Kevin DuBoisb818bfa2020-07-10 14:29:36 -070053 /* Query if the model is in need of more samples to make a prediction.
Kevin DuBois1678e2c2019-08-22 12:26:24 -070054 * \return True, if model would benefit from more samples, False if not.
55 */
Ady Abraham0bb6a472020-10-12 10:22:13 -070056 bool needsMoreSamples() const final EXCLUDES(mMutex);
Kevin DuBois1678e2c2019-08-22 12:26:24 -070057
Ady Abraham0bb6a472020-10-12 10:22:13 -070058 struct Model {
59 nsecs_t slope;
60 nsecs_t intercept;
61 };
Kevin DuBois1678e2c2019-08-22 12:26:24 -070062
Ady Abraham0bb6a472020-10-12 10:22:13 -070063 VSyncPredictor::Model getVSyncPredictionModel() const EXCLUDES(mMutex);
64
Ady Abraham5cc2e262021-03-25 13:09:17 -070065 bool isVSyncInPhase(nsecs_t timePoint, Fps frameRate) const final EXCLUDES(mMutex);
Ady Abraham0bb6a472020-10-12 10:22:13 -070066
Ady Abrahamc585dba2023-11-15 18:41:35 -080067 void setDisplayModePtr(ftl::NonNull<DisplayModePtr>) final EXCLUDES(mMutex);
68
69 void setRenderRate(Fps) final EXCLUDES(mMutex);
Ady Abrahamace3d052022-11-17 16:25:05 -080070
Ady Abrahame9883032023-11-20 17:54:54 -080071 void onFrameBegin(TimePoint expectedPresentTime, TimePoint lastConfirmedPresentTime) final
72 EXCLUDES(mMutex);
73 void onFrameMissed(TimePoint expectedPresentTime) final EXCLUDES(mMutex);
74
Ady Abraham0bb6a472020-10-12 10:22:13 -070075 void dump(std::string& result) const final EXCLUDES(mMutex);
Ady Abraham5e7371c2020-03-24 14:47:24 -070076
Kevin DuBois1678e2c2019-08-22 12:26:24 -070077private:
78 VSyncPredictor(VSyncPredictor const&) = delete;
79 VSyncPredictor& operator=(VSyncPredictor const&) = delete;
Kevin DuBoisc3e9e8e2020-01-07 09:06:52 -080080 void clearTimestamps() REQUIRES(mMutex);
Kevin DuBois1678e2c2019-08-22 12:26:24 -070081
Leon Scroggins III67388622023-02-06 20:36:20 -050082 const PhysicalDisplayId mId;
83
Kevin DuBoisecb1f0d2019-12-12 10:47:41 -080084 inline void traceInt64If(const char* name, int64_t value) const;
Ady Abrahamd9b9a042023-01-13 11:30:58 -080085 inline void traceInt64(const char* name, int64_t value) const;
Kevin DuBoisecb1f0d2019-12-12 10:47:41 -080086
Ady Abrahamf34a8132023-02-13 20:49:48 -080087 size_t next(size_t i) const REQUIRES(mMutex);
88 bool validate(nsecs_t timestamp) const REQUIRES(mMutex);
89 Model getVSyncPredictionModelLocked() const REQUIRES(mMutex);
Ady Abraham4335afd2023-12-18 19:10:47 -080090 nsecs_t snapToVsync(nsecs_t timePoint) const REQUIRES(mMutex);
91 nsecs_t snapToVsyncAlignedWithRenderRate(nsecs_t timePoint) const REQUIRES(mMutex);
Ady Abrahamf34a8132023-02-13 20:49:48 -080092 bool isVSyncInPhaseLocked(nsecs_t timePoint, unsigned divisor) const REQUIRES(mMutex);
Ady Abrahame9883032023-11-20 17:54:54 -080093 Period minFramePeriodLocked() const REQUIRES(mMutex);
94 void ensureMinFrameDurationIsKept(TimePoint, TimePoint) REQUIRES(mMutex);
Ady Abrahamf34a8132023-02-13 20:49:48 -080095
96 struct VsyncSequence {
97 nsecs_t vsyncTime;
98 int64_t seq;
99 };
100 VsyncSequence getVsyncSequenceLocked(nsecs_t timestamp) const REQUIRES(mMutex);
Ady Abrahamc585dba2023-11-15 18:41:35 -0800101 nsecs_t idealPeriod() const REQUIRES(mMutex);
Ady Abrahamf34a8132023-02-13 20:49:48 -0800102
103 bool const mTraceOn;
Kevin DuBois1678e2c2019-08-22 12:26:24 -0700104 size_t const kHistorySize;
105 size_t const kMinimumSamplesForPrediction;
106 size_t const kOutlierTolerancePercent;
Kevin DuBois1678e2c2019-08-22 12:26:24 -0700107 std::mutex mutable mMutex;
Ady Abrahamace3d052022-11-17 16:25:05 -0800108
Kevin DuBois1678e2c2019-08-22 12:26:24 -0700109 std::optional<nsecs_t> mKnownTimestamp GUARDED_BY(mMutex);
110
Ady Abraham0bb6a472020-10-12 10:22:13 -0700111 // Map between ideal vsync period and the calculated model
112 std::unordered_map<nsecs_t, Model> mutable mRateMap GUARDED_BY(mMutex);
113
Ady Abraham9c53ee72020-07-22 21:16:18 -0700114 size_t mLastTimestampIndex GUARDED_BY(mMutex) = 0;
Ady Abraham92fa2f42020-02-11 15:33:56 -0800115 std::vector<nsecs_t> mTimestamps GUARDED_BY(mMutex);
Ady Abrahamace3d052022-11-17 16:25:05 -0800116
Ady Abrahamc585dba2023-11-15 18:41:35 -0800117 ftl::NonNull<DisplayModePtr> mDisplayModePtr GUARDED_BY(mMutex);
118 std::optional<Fps> mRenderRateOpt GUARDED_BY(mMutex);
Ady Abrahamf34a8132023-02-13 20:49:48 -0800119
120 mutable std::optional<VsyncSequence> mLastVsyncSequence GUARDED_BY(mMutex);
Ady Abrahame9883032023-11-20 17:54:54 -0800121
122 std::deque<TimePoint> mPastExpectedPresentTimes GUARDED_BY(mMutex);
Ady Abraham4335afd2023-12-18 19:10:47 -0800123
124 TimePoint mLastMissedVsync GUARDED_BY(mMutex);
Kevin DuBois1678e2c2019-08-22 12:26:24 -0700125};
126
127} // namespace android::scheduler