blob: d87eae38125aa244aeaacadd35bc742554c9e407 [file] [log] [blame]
Lloyd Pique70d91362018-10-18 16:02:55 -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
Lloyd Piqueab039b52019-02-13 14:22:42 -080017#include <compositionengine/CompositionRefreshArgs.h>
18#include <compositionengine/LayerFE.h>
Lloyd Piquede196652020-01-22 17:29:58 -080019#include <compositionengine/LayerFECompositionState.h>
Lloyd Piquec7b0c752019-03-07 20:59:59 -080020#include <compositionengine/OutputLayer.h>
Lloyd Pique70d91362018-10-18 16:02:55 -070021#include <compositionengine/impl/CompositionEngine.h>
Lloyd Pique45a165a2018-10-19 11:54:47 -070022#include <compositionengine/impl/Display.h>
Leon Scroggins III2f60d732022-09-12 14:42:38 -040023#include <ui/DisplayMap.h>
Lloyd Pique3b5a69e2020-01-16 17:51:01 -080024
Lloyd Piqueb97e04f2018-10-18 17:07:05 -070025#include <renderengine/RenderEngine.h>
Lloyd Piqueab039b52019-02-13 14:22:42 -080026#include <utils/Trace.h>
Lloyd Pique70d91362018-10-18 16:02:55 -070027
Lloyd Pique3b5a69e2020-01-16 17:51:01 -080028// TODO(b/129481165): remove the #pragma below and fix conversion issues
29#pragma clang diagnostic push
30#pragma clang diagnostic ignored "-Wconversion"
31
Lloyd Pique441d5042018-10-18 16:49:51 -070032#include "DisplayHardware/HWComposer.h"
33
Lloyd Pique3b5a69e2020-01-16 17:51:01 -080034// TODO(b/129481165): remove the #pragma below and fix conversion issues
35#pragma clang diagnostic pop // ignored "-Wconversion"
36
Lloyd Pique70d91362018-10-18 16:02:55 -070037namespace android::compositionengine {
38
39CompositionEngine::~CompositionEngine() = default;
40
41namespace impl {
42
43std::unique_ptr<compositionengine::CompositionEngine> createCompositionEngine() {
44 return std::make_unique<CompositionEngine>();
45}
46
47CompositionEngine::CompositionEngine() = default;
48CompositionEngine::~CompositionEngine() = default;
49
Lloyd Pique45a165a2018-10-19 11:54:47 -070050std::shared_ptr<compositionengine::Display> CompositionEngine::createDisplay(
Lloyd Piquea38ea7e2019-04-16 18:10:26 -070051 const DisplayCreationArgs& args) {
52 return compositionengine::impl::createDisplay(*this, args);
Lloyd Pique45a165a2018-10-19 11:54:47 -070053}
54
Lloyd Piquede196652020-01-22 17:29:58 -080055std::unique_ptr<compositionengine::LayerFECompositionState>
56CompositionEngine::createLayerFECompositionState() {
57 return std::make_unique<compositionengine::LayerFECompositionState>();
Lloyd Piquefeb73d72018-12-04 17:23:44 -080058}
59
Lloyd Pique441d5042018-10-18 16:49:51 -070060HWComposer& CompositionEngine::getHwComposer() const {
61 return *mHwComposer.get();
62}
63
64void CompositionEngine::setHwComposer(std::unique_ptr<HWComposer> hwComposer) {
65 mHwComposer = std::move(hwComposer);
66}
67
Lloyd Piqueb97e04f2018-10-18 17:07:05 -070068renderengine::RenderEngine& CompositionEngine::getRenderEngine() const {
Patrick Williams0a525a42022-10-26 20:20:50 +000069 return *mRenderEngine;
Lloyd Piqueb97e04f2018-10-18 17:07:05 -070070}
71
Patrick Williams0a525a42022-10-26 20:20:50 +000072void CompositionEngine::setRenderEngine(renderengine::RenderEngine* renderEngine) {
73 mRenderEngine = renderEngine;
Lloyd Piqueb97e04f2018-10-18 17:07:05 -070074}
75
Patrick Williams74c0bf62022-11-02 23:59:26 +000076TimeStats* CompositionEngine::getTimeStats() const {
77 return mTimeStats.get();
Alec Mourie4034bb2019-11-19 12:45:54 -080078}
79
80void CompositionEngine::setTimeStats(const std::shared_ptr<TimeStats>& timeStats) {
81 mTimeStats = timeStats;
82}
83
Lloyd Piqueab039b52019-02-13 14:22:42 -080084bool CompositionEngine::needsAnotherUpdate() const {
85 return mNeedsAnotherUpdate;
86}
87
88nsecs_t CompositionEngine::getLastFrameRefreshTimestamp() const {
89 return mRefreshStartTime;
90}
91
Leon Scroggins III2f60d732022-09-12 14:42:38 -040092namespace {
Leon Scroggins IIIab551a32023-11-22 10:08:25 -050093void offloadOutputs(Outputs& outputs) {
94 if (!FlagManager::getInstance().multithreaded_present() || outputs.size() < 2) {
95 return;
Leon Scroggins III2f60d732022-09-12 14:42:38 -040096 }
97
Leon Scroggins IIIab551a32023-11-22 10:08:25 -050098 ui::PhysicalDisplayVector<compositionengine::Output*> outputsToOffload;
99 for (const auto& output : outputs) {
100 if (!ftl::Optional(output->getDisplayId()).and_then(HalDisplayId::tryCast)) {
101 // Not HWC-enabled, so it is always client-composited. No need to offload.
102 continue;
103 }
Leon Scroggins IIIcbc929d2023-12-01 16:21:37 -0500104 if (!output->getState().isEnabled) {
105 continue;
106 }
Leon Scroggins IIIab551a32023-11-22 10:08:25 -0500107
108 // Only run present in multiple threads if all HWC-enabled displays
109 // being refreshed support it.
110 if (!output->supportsOffloadPresent()) {
111 return;
112 }
113 outputsToOffload.push_back(output.get());
Leon Scroggins III2f60d732022-09-12 14:42:38 -0400114 }
Leon Scroggins IIIab551a32023-11-22 10:08:25 -0500115
116 if (outputsToOffload.size() < 2) {
117 return;
118 }
119
120 // Leave the last eligible display on the main thread, which will
121 // allow it to run concurrently without an extra thread hop.
122 outputsToOffload.pop_back();
123
124 for (compositionengine::Output* output : outputsToOffload) {
125 output->offloadPresentNextFrame();
126 }
Leon Scroggins III2f60d732022-09-12 14:42:38 -0400127}
128} // namespace
129
Lloyd Piqued7b429f2019-03-07 21:11:02 -0800130void CompositionEngine::present(CompositionRefreshArgs& args) {
Lloyd Piquec29e4c62019-03-07 21:48:19 -0800131 ATRACE_CALL();
132 ALOGV(__FUNCTION__);
133
134 preComposition(args);
135
136 {
137 // latchedLayers is used to track the set of front-end layer state that
138 // has been latched across all outputs for the prepare step, and is not
139 // needed for anything else.
140 LayerFESet latchedLayers;
141
142 for (const auto& output : args.outputs) {
143 output->prepare(args, latchedLayers);
144 }
Lloyd Pique3eb1b212019-03-07 21:15:40 -0800145 }
146
Leon Scroggins III2f60d732022-09-12 14:42:38 -0400147 // Offloading the HWC call for `present` allows us to simultaneously call it
148 // on multiple displays. This is desirable because these calls block and can
149 // be slow.
Leon Scroggins IIIab551a32023-11-22 10:08:25 -0500150 offloadOutputs(args.outputs);
Leon Scroggins III2f60d732022-09-12 14:42:38 -0400151
152 ui::DisplayVector<ftl::Future<std::monostate>> presentFutures;
Lloyd Pique3eb1b212019-03-07 21:15:40 -0800153 for (const auto& output : args.outputs) {
Leon Scroggins III2f60d732022-09-12 14:42:38 -0400154 presentFutures.push_back(output->present(args));
155 }
156
157 {
158 ATRACE_NAME("Waiting on HWC");
159 for (auto& future : presentFutures) {
160 // TODO(b/185536303): Call ftl::Future::wait() once it exists, since
161 // we do not need the return value of get().
162 future.get();
163 }
Lloyd Piqued7b429f2019-03-07 21:11:02 -0800164 }
165}
166
Lloyd Piquec7b0c752019-03-07 20:59:59 -0800167void CompositionEngine::updateCursorAsync(CompositionRefreshArgs& args) {
Lloyd Piquec7b0c752019-03-07 20:59:59 -0800168
169 for (const auto& output : args.outputs) {
Lloyd Pique01c77c12019-04-17 12:48:32 -0700170 for (auto* layer : output->getOutputLayersOrderedByZ()) {
Lloyd Piquec7b0c752019-03-07 20:59:59 -0800171 if (layer->isHardwareCursor()) {
Lloyd Piquec7b0c752019-03-07 20:59:59 -0800172 layer->writeCursorPositionToHWC();
173 }
174 }
175 }
176}
177
Lloyd Piqueab039b52019-02-13 14:22:42 -0800178void CompositionEngine::preComposition(CompositionRefreshArgs& args) {
179 ATRACE_CALL();
180 ALOGV(__FUNCTION__);
181
182 bool needsAnotherUpdate = false;
183
184 mRefreshStartTime = systemTime(SYSTEM_TIME_MONOTONIC);
185
186 for (auto& layer : args.layers) {
Vishnu Nairba354102022-08-01 00:14:18 +0000187 if (layer->onPreComposition(mRefreshStartTime, args.updatingOutputGeometryThisFrame)) {
Lloyd Piqueab039b52019-02-13 14:22:42 -0800188 needsAnotherUpdate = true;
189 }
190 }
191
192 mNeedsAnotherUpdate = needsAnotherUpdate;
193}
194
Vishnu Nair0a4fb002022-08-08 02:40:42 +0000195FeatureFlags CompositionEngine::getFeatureFlags() const {
196 return {};
197}
198
Lloyd Piquec3cb7292019-05-17 15:25:14 -0700199void CompositionEngine::dump(std::string&) const {
200 // The base class has no state to dump, but derived classes might.
201}
202
Lloyd Piqueab039b52019-02-13 14:22:42 -0800203void CompositionEngine::setNeedsAnotherUpdateForTest(bool value) {
204 mNeedsAnotherUpdate = value;
205}
206
Lloyd Pique70d91362018-10-18 16:02:55 -0700207} // namespace impl
208} // namespace android::compositionengine