| Ana Krulec | 61f86db | 2018-11-19 14:16:35 +0100 | [diff] [blame] | 1 | /* | 
|  | 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 Laskowski | f7a09ed | 2019-10-07 13:54:18 -0700 | [diff] [blame] | 19 | #include <android-base/thread_annotations.h> | 
|  | 20 | #include <utils/RefBase.h> | 
| Ana Krulec | 61f86db | 2018-11-19 14:16:35 +0100 | [diff] [blame] | 21 | #include <utils/Timers.h> | 
|  | 22 |  | 
| Nathaniel Nifong | 1303d91 | 2021-10-06 09:41:24 -0400 | [diff] [blame] | 23 | #include <map> | 
| Dominik Laskowski | f7a09ed | 2019-10-07 13:54:18 -0700 | [diff] [blame] | 24 | #include <memory> | 
|  | 25 | #include <mutex> | 
| Dominik Laskowski | 983f2b5 | 2020-06-25 16:54:06 -0700 | [diff] [blame] | 26 | #include <string> | 
| Dominik Laskowski | f7a09ed | 2019-10-07 13:54:18 -0700 | [diff] [blame] | 27 | #include <utility> | 
|  | 28 | #include <vector> | 
| Ana Krulec | 434c22d | 2018-11-28 13:48:36 +0100 | [diff] [blame] | 29 |  | 
| Ady Abraham | 8a82ba6 | 2020-01-17 12:43:17 -0800 | [diff] [blame] | 30 | #include "RefreshRateConfigs.h" | 
|  | 31 |  | 
| Ana Krulec | 61f86db | 2018-11-19 14:16:35 +0100 | [diff] [blame] | 32 | namespace android { | 
| Dominik Laskowski | f7a09ed | 2019-10-07 13:54:18 -0700 | [diff] [blame] | 33 |  | 
|  | 34 | class Layer; | 
|  | 35 |  | 
| Ady Abraham | 09bd392 | 2019-04-08 10:44:56 -0700 | [diff] [blame] | 36 | namespace scheduler { | 
| Ana Krulec | 61f86db | 2018-11-19 14:16:35 +0100 | [diff] [blame] | 37 |  | 
| Dominik Laskowski | f7a09ed | 2019-10-07 13:54:18 -0700 | [diff] [blame] | 38 | class LayerInfo; | 
|  | 39 |  | 
| Ana Krulec | 61f86db | 2018-11-19 14:16:35 +0100 | [diff] [blame] | 40 | class LayerHistory { | 
|  | 41 | public: | 
| Ady Abraham | 8a82ba6 | 2020-01-17 12:43:17 -0800 | [diff] [blame] | 42 | using LayerVoteType = RefreshRateConfigs::LayerVoteType; | 
|  | 43 |  | 
| Ady Abraham | 3efa394 | 2021-06-24 19:01:25 -0700 | [diff] [blame] | 44 | LayerHistory(); | 
| Marin Shalamanov | 1bc43ee | 2020-11-20 16:56:52 +0100 | [diff] [blame] | 45 | ~LayerHistory(); | 
| Ana Krulec | 61f86db | 2018-11-19 14:16:35 +0100 | [diff] [blame] | 46 |  | 
| Dominik Laskowski | f7a09ed | 2019-10-07 13:54:18 -0700 | [diff] [blame] | 47 | // Layers are unregistered when the weak reference expires. | 
| Marin Shalamanov | 4ad8b30 | 2020-12-11 15:50:08 +0100 | [diff] [blame] | 48 | void registerLayer(Layer*, LayerVoteType type); | 
| Ady Abraham | 8a82ba6 | 2020-01-17 12:43:17 -0800 | [diff] [blame] | 49 |  | 
|  | 50 | // Sets the display size. Client is responsible for synchronization. | 
| Marin Shalamanov | 1bc43ee | 2020-11-20 16:56:52 +0100 | [diff] [blame] | 51 | void setDisplayArea(uint32_t displayArea) { mDisplayArea = displayArea; } | 
| Ady Abraham | 09bd392 | 2019-04-08 10:44:56 -0700 | [diff] [blame] | 52 |  | 
| Marin Shalamanov | a7fe304 | 2021-01-29 21:02:08 +0100 | [diff] [blame] | 53 | // Sets whether a mode change is pending to be applied | 
|  | 54 | void setModeChangePending(bool pending) { mModeChangePending = pending; } | 
| Ady Abraham | 32efd54 | 2020-05-19 17:49:26 -0700 | [diff] [blame] | 55 |  | 
| Ady Abraham | 5def733 | 2020-05-29 16:13:47 -0700 | [diff] [blame] | 56 | // Represents which layer activity is recorded | 
|  | 57 | enum class LayerUpdateType { | 
|  | 58 | Buffer,       // a new buffer queued | 
|  | 59 | AnimationTX,  // a new transaction with eAnimation flag set | 
|  | 60 | SetFrameRate, // setFrameRate API was called | 
|  | 61 | }; | 
|  | 62 |  | 
| Dominik Laskowski | f7a09ed | 2019-10-07 13:54:18 -0700 | [diff] [blame] | 63 | // Marks the layer as active, and records the given state to its history. | 
| Marin Shalamanov | 1bc43ee | 2020-11-20 16:56:52 +0100 | [diff] [blame] | 64 | void record(Layer*, nsecs_t presentTime, nsecs_t now, LayerUpdateType updateType); | 
| Ady Abraham | a315ce7 | 2019-04-24 14:35:20 -0700 | [diff] [blame] | 65 |  | 
| Ady Abraham | 8a82ba6 | 2020-01-17 12:43:17 -0800 | [diff] [blame] | 66 | using Summary = std::vector<RefreshRateConfigs::LayerRequirement>; | 
| Ady Abraham | 09bd392 | 2019-04-08 10:44:56 -0700 | [diff] [blame] | 67 |  | 
| Dominik Laskowski | f7a09ed | 2019-10-07 13:54:18 -0700 | [diff] [blame] | 68 | // Rebuilds sets of active/inactive layers, and accumulates stats for active layers. | 
| Ady Abraham | 3efa394 | 2021-06-24 19:01:25 -0700 | [diff] [blame] | 69 | Summary summarize(const RefreshRateConfigs&, nsecs_t now); | 
| Ady Abraham | a9bf4ca | 2019-06-11 19:08:58 -0700 | [diff] [blame] | 70 |  | 
| Marin Shalamanov | 1bc43ee | 2020-11-20 16:56:52 +0100 | [diff] [blame] | 71 | void clear(); | 
| Ady Abraham | bdda8f0 | 2021-04-01 16:06:11 -0700 | [diff] [blame] | 72 |  | 
|  | 73 | void deregisterLayer(Layer*); | 
| Marin Shalamanov | 1bc43ee | 2020-11-20 16:56:52 +0100 | [diff] [blame] | 74 | std::string dump() const; | 
| Ady Abraham | 8a82ba6 | 2020-01-17 12:43:17 -0800 | [diff] [blame] | 75 |  | 
| Nathaniel Nifong | 1303d91 | 2021-10-06 09:41:24 -0400 | [diff] [blame] | 76 | // return the frames per second of the layer with the given sequence id. | 
|  | 77 | float getLayerFramerate(nsecs_t now, int32_t id) const; | 
|  | 78 |  | 
| Ady Abraham | 8a82ba6 | 2020-01-17 12:43:17 -0800 | [diff] [blame] | 79 | private: | 
| Dominik Laskowski | 068173d | 2021-08-11 17:22:59 -0700 | [diff] [blame] | 80 | friend class LayerHistoryTest; | 
|  | 81 | friend class TestableScheduler; | 
| Ady Abraham | 8a82ba6 | 2020-01-17 12:43:17 -0800 | [diff] [blame] | 82 |  | 
| Ady Abraham | bdda8f0 | 2021-04-01 16:06:11 -0700 | [diff] [blame] | 83 | using LayerPair = std::pair<Layer*, std::unique_ptr<LayerInfo>>; | 
| Nathaniel Nifong | 1303d91 | 2021-10-06 09:41:24 -0400 | [diff] [blame] | 84 | // keyed by id as returned from Layer::getSequence() | 
|  | 85 | using LayerInfos = std::unordered_map<int32_t, LayerPair>; | 
| Ady Abraham | 8a82ba6 | 2020-01-17 12:43:17 -0800 | [diff] [blame] | 86 |  | 
| Nathaniel Nifong | 1303d91 | 2021-10-06 09:41:24 -0400 | [diff] [blame] | 87 | // Iterates over layers maps moving all active layers to mActiveLayerInfos and all inactive | 
|  | 88 | // layers to mInactiveLayerInfos. | 
|  | 89 | // worst case time complexity is O(2 * inactive + active) | 
|  | 90 | void partitionLayers(nsecs_t now) REQUIRES(mLock); | 
| Ady Abraham | 8a82ba6 | 2020-01-17 12:43:17 -0800 | [diff] [blame] | 91 |  | 
| Dominik Laskowski | 0c41ffa | 2021-12-24 16:45:12 -0800 | [diff] [blame] | 92 | enum class LayerStatus { | 
| Nathaniel Nifong | 1303d91 | 2021-10-06 09:41:24 -0400 | [diff] [blame] | 93 | NotFound, | 
|  | 94 | LayerInActiveMap, | 
|  | 95 | LayerInInactiveMap, | 
| Ady Abraham | 8a82ba6 | 2020-01-17 12:43:17 -0800 | [diff] [blame] | 96 | }; | 
|  | 97 |  | 
| Nathaniel Nifong | 1303d91 | 2021-10-06 09:41:24 -0400 | [diff] [blame] | 98 | // looks up a layer by sequence id in both layerInfo maps. | 
|  | 99 | // The first element indicates if and where the item was found | 
| Dominik Laskowski | 0c41ffa | 2021-12-24 16:45:12 -0800 | [diff] [blame] | 100 | std::pair<LayerStatus, LayerPair*> findLayer(int32_t id) REQUIRES(mLock); | 
|  | 101 |  | 
|  | 102 | std::pair<LayerStatus, const LayerPair*> findLayer(int32_t id) const REQUIRES(mLock) { | 
|  | 103 | return const_cast<LayerHistory*>(this)->findLayer(id); | 
|  | 104 | } | 
| Ady Abraham | 8a82ba6 | 2020-01-17 12:43:17 -0800 | [diff] [blame] | 105 |  | 
|  | 106 | mutable std::mutex mLock; | 
|  | 107 |  | 
| Nathaniel Nifong | 1303d91 | 2021-10-06 09:41:24 -0400 | [diff] [blame] | 108 | // Partitioned into two maps to facility two kinds of retrieval: | 
|  | 109 | // 1. retrieval of a layer by id (attempt lookup in both maps) | 
|  | 110 | // 2. retrieval of all active layers (iterate that map) | 
|  | 111 | // The partitioning is allowed to become out of date but calling partitionLayers refreshes the | 
|  | 112 | // validity of each map. | 
|  | 113 | LayerInfos mActiveLayerInfos GUARDED_BY(mLock); | 
|  | 114 | LayerInfos mInactiveLayerInfos GUARDED_BY(mLock); | 
| Dominik Laskowski | f7a09ed | 2019-10-07 13:54:18 -0700 | [diff] [blame] | 115 |  | 
| Ady Abraham | 8a82ba6 | 2020-01-17 12:43:17 -0800 | [diff] [blame] | 116 | uint32_t mDisplayArea = 0; | 
|  | 117 |  | 
| Dominik Laskowski | f7a09ed | 2019-10-07 13:54:18 -0700 | [diff] [blame] | 118 | // Whether to emit systrace output and debug logs. | 
|  | 119 | const bool mTraceEnabled; | 
| Ana Krulec | c84d09b | 2019-11-02 23:10:29 +0100 | [diff] [blame] | 120 |  | 
|  | 121 | // Whether to use priority sent from WindowManager to determine the relevancy of the layer. | 
|  | 122 | const bool mUseFrameRatePriority; | 
| Ady Abraham | 32efd54 | 2020-05-19 17:49:26 -0700 | [diff] [blame] | 123 |  | 
| Marin Shalamanov | a7fe304 | 2021-01-29 21:02:08 +0100 | [diff] [blame] | 124 | // Whether a mode change is in progress or not | 
|  | 125 | std::atomic<bool> mModeChangePending = false; | 
| Ana Krulec | 61f86db | 2018-11-19 14:16:35 +0100 | [diff] [blame] | 126 | }; | 
|  | 127 |  | 
| Ady Abraham | 09bd392 | 2019-04-08 10:44:56 -0700 | [diff] [blame] | 128 | } // namespace scheduler | 
| Dominik Laskowski | f7a09ed | 2019-10-07 13:54:18 -0700 | [diff] [blame] | 129 | } // namespace android |