blob: 8ca54af13baadb4a5b7fadd13a2b747b5ffd0650 [file] [log] [blame]
John Reck23b797a2014-01-03 18:08:34 -08001/*
2 * Copyright (C) 2014 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
Chris Craik5e00c7c2016-07-06 16:10:09 -070017#pragma once
John Reck23b797a2014-01-03 18:08:34 -080018
John Reckba6adf62015-02-19 14:36:50 -080019#include "DamageAccumulator.h"
John Reckd9d7f122018-05-03 14:40:56 -070020#include "Lighting.h"
John Reckba6adf62015-02-19 14:36:50 -080021#include "FrameInfo.h"
John Reck4c9e59d2015-05-12 07:17:50 -070022#include "FrameInfoVisualizer.h"
Andres Morales910beb82016-02-02 16:19:40 -080023#include "FrameMetricsReporter.h"
Chris Craik0b7e8242015-10-28 16:50:44 -070024#include "IContextFactory.h"
Stan Iliev768e3932016-07-08 21:34:52 -040025#include "IRenderPipeline.h"
Chris Craik0b7e8242015-10-28 16:50:44 -070026#include "LayerUpdateQueue.h"
John Reckba6adf62015-02-19 14:36:50 -080027#include "RenderNode.h"
John Reckba6adf62015-02-19 14:36:50 -080028#include "renderthread/RenderTask.h"
29#include "renderthread/RenderThread.h"
John Reck1bcacfd2017-11-03 10:12:19 -070030#include "thread/Task.h"
31#include "thread/TaskProcessor.h"
John Reck998a6d82014-08-28 15:35:53 -070032
John Reck23b797a2014-01-03 18:08:34 -080033#include <EGL/egl.h>
John Reck19b6bcf2014-02-14 20:03:38 -080034#include <SkBitmap.h>
John Reckd04794a2015-05-08 10:04:36 -070035#include <SkRect.h>
John Reck1bcacfd2017-11-03 10:12:19 -070036#include <cutils/compiler.h>
John Reckf6481082016-02-02 15:18:23 -080037#include <gui/Surface.h>
John Reck1bcacfd2017-11-03 10:12:19 -070038#include <utils/Functor.h>
John Reck4f02bf42014-01-03 18:09:17 -080039
John Reck38f6c032016-03-17 10:23:49 -070040#include <functional>
John Reckba6adf62015-02-19 14:36:50 -080041#include <set>
John Reckb36016c2015-03-11 08:50:53 -070042#include <string>
Skuhneea7a7fb2015-08-28 07:10:31 -070043#include <vector>
John Reck23b797a2014-01-03 18:08:34 -080044
45namespace android {
46namespace uirenderer {
John Reck4f02bf42014-01-03 18:09:17 -080047
John Reck119907c2014-08-14 09:02:01 -070048class AnimationContext;
John Reck19b6bcf2014-02-14 20:03:38 -080049class DeferredLayerUpdater;
Stan Iliev216b1572018-03-26 14:29:50 -040050class ErrorHandler;
John Reck1949e792014-04-08 15:18:56 -070051class Layer;
Chris Craik5e00c7c2016-07-06 16:10:09 -070052class Rect;
John Reck443a7142014-09-04 17:40:05 -070053class RenderState;
John Reck4f02bf42014-01-03 18:09:17 -080054
John Reck23b797a2014-01-03 18:08:34 -080055namespace renderthread {
56
John Reck3b202512014-06-23 13:13:08 -070057class EglManager;
Stan Iliev768e3932016-07-08 21:34:52 -040058class Frame;
John Reck1125d1f2014-10-23 11:02:19 -070059
John Reck23b797a2014-01-03 18:08:34 -080060// This per-renderer class manages the bridge between the global EGL context
61// and the render surface.
John Reck119907c2014-08-14 09:02:01 -070062// TODO: Rename to Renderer or some other per-window, top-level manager
John Recke45b1fd2014-04-15 09:50:16 -070063class CanvasContext : public IFrameCallback {
John Reck23b797a2014-01-03 18:08:34 -080064public:
John Reck1bcacfd2017-11-03 10:12:19 -070065 static CanvasContext* create(RenderThread& thread, bool translucent, RenderNode* rootRenderNode,
66 IContextFactory* contextFactory);
John Recke45b1fd2014-04-15 09:50:16 -070067 virtual ~CanvasContext();
John Reck23b797a2014-01-03 18:08:34 -080068
Derek Sollenberger6a21ca52016-09-28 13:39:55 -040069 /**
70 * Update or create a layer specific for the provided RenderNode. The layer
71 * attached to the node will be specific to the RenderPipeline used by this
72 * context
73 *
74 * @return true if the layer has been created or updated
75 */
Stan Iliev216b1572018-03-26 14:29:50 -040076 bool createOrUpdateLayer(RenderNode* node, const DamageAccumulator& dmgAccumulator,
77 ErrorHandler* errorHandler) {
78 return mRenderPipeline->createOrUpdateLayer(node, dmgAccumulator, mWideColorGamut,
John Reck1a4a9812018-04-18 16:13:31 -070079 errorHandler);
Derek Sollenberger6a21ca52016-09-28 13:39:55 -040080 }
81
82 /**
Derek Sollenbergerb7d34b62016-11-04 10:46:18 -040083 * Pin any mutable images to the GPU cache. A pinned images is guaranteed to
84 * remain in the cache until it has been unpinned. We leverage this feature
85 * to avoid making a CPU copy of the pixels.
86 *
Derek Sollenberger189e8742016-11-16 16:00:17 -050087 * @return true if all images have been successfully pinned to the GPU cache
Derek Sollenbergerb7d34b62016-11-04 10:46:18 -040088 * and false otherwise (e.g. cache limits have been exceeded).
89 */
90 bool pinImages(std::vector<SkImage*>& mutableImages) {
91 return mRenderPipeline->pinImages(mutableImages);
92 }
John Reck1bcacfd2017-11-03 10:12:19 -070093 bool pinImages(LsaVector<sk_sp<Bitmap>>& images) { return mRenderPipeline->pinImages(images); }
Derek Sollenbergerb7d34b62016-11-04 10:46:18 -040094
95 /**
96 * Unpin any image that had be previously pinned to the GPU cache
97 */
98 void unpinImages() { mRenderPipeline->unpinImages(); }
99
100 /**
Derek Sollenberger6a21ca52016-09-28 13:39:55 -0400101 * Destroy any layers that have been attached to the provided RenderNode removing
102 * any state that may have been set during createOrUpdateLayer().
103 */
104 static void destroyLayer(RenderNode* node);
105
Derek Sollenbergerdaf72292016-10-25 12:09:18 -0400106 static void invokeFunctor(const RenderThread& thread, Functor* functor);
107
108 static void prepareToDraw(const RenderThread& thread, Bitmap* bitmap);
109
Derek Sollenberger0df62092016-09-27 16:04:42 -0400110 /*
111 * If Properties::isSkiaEnabled() is true then this will return the Skia
112 * grContext associated with the current RenderPipeline.
113 */
Derek Sollenberger98f75d52016-10-25 10:25:45 -0400114 GrContext* getGrContext() const { return mRenderThread.getGrContext(); }
Derek Sollenberger0df62092016-09-27 16:04:42 -0400115
John Reck1125d1f2014-10-23 11:02:19 -0700116 // Won't take effect until next EGLSurface creation
117 void setSwapBehavior(SwapBehavior swapBehavior);
118
John Reckf8441e62017-10-23 13:10:41 -0700119 void setSurface(sp<Surface>&& surface);
120 bool pauseSurface();
John Reck8afcc762016-04-13 10:24:06 -0700121 void setStopped(bool stopped);
John Reckf6481082016-02-02 15:18:23 -0800122 bool hasSurface() { return mNativeSurface.get(); }
John Reckaa95a882014-11-07 11:02:07 -0800123
John Reck1bcacfd2017-11-03 10:12:19 -0700124 void setup(float lightRadius, uint8_t ambientShadowAlpha, uint8_t spotShadowAlpha);
Alan Viverette50210d92015-05-14 18:05:36 -0700125 void setLightCenter(const Vector3& lightCenter);
John Reck63a06672014-05-07 13:45:54 -0700126 void setOpaque(bool opaque);
Romain Guy26a2b972017-04-17 09:39:51 -0700127 void setWideGamut(bool wideGamut);
John Reck8afcc762016-04-13 10:24:06 -0700128 bool makeCurrent();
John Reck1bcacfd2017-11-03 10:12:19 -0700129 void prepareTree(TreeInfo& info, int64_t* uiFrameInfo, int64_t syncQueued, RenderNode* target);
John Recke4267ea2014-06-03 15:53:15 -0700130 void draw();
John Reck2de950d2017-01-25 10:58:30 -0800131 void destroy();
John Reck23b797a2014-01-03 18:08:34 -0800132
Chris Craik0b7e8242015-10-28 16:50:44 -0700133 // IFrameCallback, Choreographer-driven frame callback entry point
Chris Craikd41c4d82015-01-05 15:51:13 -0800134 virtual void doFrame() override;
Skuhneea7a7fb2015-08-28 07:10:31 -0700135 void prepareAndDraw(RenderNode* node);
John Recke45b1fd2014-04-15 09:50:16 -0700136
John Reck2de950d2017-01-25 10:58:30 -0800137 void buildLayer(RenderNode* node);
John Reck19b6bcf2014-02-14 20:03:38 -0800138 bool copyLayerInto(DeferredLayerUpdater* layer, SkBitmap* bitmap);
John Reck998a6d82014-08-28 15:35:53 -0700139 void markLayerInUse(RenderNode* node);
John Reck19b6bcf2014-02-14 20:03:38 -0800140
John Reck2de950d2017-01-25 10:58:30 -0800141 void destroyHardwareResources();
John Reckf47a5942014-06-30 16:20:04 -0700142 static void trimMemory(RenderThread& thread, int level);
John Recke1628b72014-05-23 15:11:19 -0700143
Derek Sollenberger56ad6ec2016-07-22 12:13:32 -0400144 DeferredLayerUpdater* createTextureLayer();
John Reck1949e792014-04-08 15:18:56 -0700145
John Reckf47a5942014-06-30 16:20:04 -0700146 void stopDrawing();
John Recka5dda642014-05-22 15:43:54 -0700147 void notifyFramePending();
148
John Reck4c9e59d2015-05-12 07:17:50 -0700149 FrameInfoVisualizer& profiler() { return mProfiler; }
John Reckfe5e7b72014-05-23 17:42:28 -0700150
John Reckba6adf62015-02-19 14:36:50 -0800151 void dumpFrames(int fd);
152 void resetFrameStats();
153
John Reckdf1742e2017-01-19 15:56:21 -0800154 void setName(const std::string&& name);
John Reckb36016c2015-03-11 08:50:53 -0700155
John Reck2de950d2017-01-25 10:58:30 -0800156 void addRenderNode(RenderNode* node, bool placeFront);
157 void removeRenderNode(RenderNode* node);
Skuhneea7a7fb2015-08-28 07:10:31 -0700158
John Reck1bcacfd2017-11-03 10:12:19 -0700159 void setContentDrawBounds(const Rect& bounds) { mContentDrawBounds = bounds; }
Skuhneea7a7fb2015-08-28 07:10:31 -0700160
John Reck1bcacfd2017-11-03 10:12:19 -0700161 RenderState& getRenderState() { return mRenderThread.renderState(); }
Chris Craike2e53a72015-10-28 15:55:40 -0700162
Andres Morales910beb82016-02-02 16:19:40 -0800163 void addFrameMetricsObserver(FrameMetricsObserver* observer) {
164 if (mFrameMetricsReporter.get() == nullptr) {
165 mFrameMetricsReporter.reset(new FrameMetricsReporter());
Andres Morales06f5bc72015-12-15 15:21:31 -0800166 }
167
Andres Morales910beb82016-02-02 16:19:40 -0800168 mFrameMetricsReporter->addObserver(observer);
Andres Morales06f5bc72015-12-15 15:21:31 -0800169 }
170
Andres Morales910beb82016-02-02 16:19:40 -0800171 void removeFrameMetricsObserver(FrameMetricsObserver* observer) {
172 if (mFrameMetricsReporter.get() != nullptr) {
173 mFrameMetricsReporter->removeObserver(observer);
174 if (!mFrameMetricsReporter->hasObservers()) {
175 mFrameMetricsReporter.reset(nullptr);
Andres Morales06f5bc72015-12-15 15:21:31 -0800176 }
177 }
178 }
179
John Reck38f6c032016-03-17 10:23:49 -0700180 // Used to queue up work that needs to be completed before this frame completes
181 ANDROID_API void enqueueFrameWork(std::function<void()>&& func);
182
John Reck28912a52016-04-18 14:34:18 -0700183 ANDROID_API int64_t getFrameNumber();
184
Chris Craik06e2e9c2016-08-31 17:32:46 -0700185 void waitOnFences();
186
Stan Iliev23c38a92017-03-23 00:12:50 -0400187 IRenderPipeline* getRenderPipeline() { return mRenderPipeline.get(); }
188
John Reckcc2eee82018-05-17 10:44:00 -0700189 void addFrameCompleteListener(std::function<void(int64_t)>&& func) {
190 mFrameCompleteCallbacks.push_back(std::move(func));
191 }
192
John Reck23b797a2014-01-03 18:08:34 -0800193private:
Stan Iliev03de0742016-07-07 12:35:54 -0400194 CanvasContext(RenderThread& thread, bool translucent, RenderNode* rootRenderNode,
John Reck1bcacfd2017-11-03 10:12:19 -0700195 IContextFactory* contextFactory, std::unique_ptr<IRenderPipeline> renderPipeline);
Stan Iliev03de0742016-07-07 12:35:54 -0400196
John Recka5dda642014-05-22 15:43:54 -0700197 friend class RegisterFrameCallbackTask;
John Reck443a7142014-09-04 17:40:05 -0700198 // TODO: Replace with something better for layer & other GL object
199 // lifecycle tracking
200 friend class android::uirenderer::RenderState;
John Recka5dda642014-05-22 15:43:54 -0700201
John Reck2de950d2017-01-25 10:58:30 -0800202 void freePrefetchedLayers();
John Reck998a6d82014-08-28 15:35:53 -0700203
John Reck0def73a2016-07-01 16:19:13 -0700204 bool isSwapChainStuffed();
205
Stan Iliev768e3932016-07-08 21:34:52 -0400206 SkRect computeDirtyRect(const Frame& frame, SkRect* dirty);
207
John Reck77c40102015-10-26 15:49:47 -0700208 EGLint mLastFrameWidth = 0;
209 EGLint mLastFrameHeight = 0;
Chris Craikddf22152015-10-14 17:42:47 -0700210
John Reck4f02bf42014-01-03 18:09:17 -0800211 RenderThread& mRenderThread;
John Reckf6481082016-02-02 15:18:23 -0800212 sp<Surface> mNativeSurface;
John Reck306f3312016-06-10 16:01:55 -0700213 // stopped indicates the CanvasContext will reject actual redraw operations,
214 // and defer repaint until it is un-stopped
John Reck8afcc762016-04-13 10:24:06 -0700215 bool mStopped = false;
Leon Scroggins III4afdd1c2018-05-14 14:59:30 -0400216 // Incremented each time the CanvasContext is stopped. Used to ignore
217 // delayed messages that are triggered after stopping.
218 int mGenerationID;
John Reck306f3312016-06-10 16:01:55 -0700219 // CanvasContext is dirty if it has received an update that it has not
220 // painted onto its surface.
221 bool mIsDirty = false;
Stan Iliev768e3932016-07-08 21:34:52 -0400222 SwapBehavior mSwapBehavior = SwapBehavior::kSwap_default;
John Recke486d932015-10-28 09:21:19 -0700223 struct SwapHistory {
224 SkRect damage;
225 nsecs_t vsyncTime;
John Reck0def73a2016-07-01 16:19:13 -0700226 nsecs_t swapCompletedTime;
227 nsecs_t dequeueDuration;
228 nsecs_t queueDuration;
John Recke486d932015-10-28 09:21:19 -0700229 };
230
231 RingBuffer<SwapHistory, 3> mSwapHistory;
John Reck28912a52016-04-18 14:34:18 -0700232 int64_t mFrameNumber = -1;
John Reck4f02bf42014-01-03 18:09:17 -0800233
Chris Craik31635682016-07-19 17:59:12 -0700234 // last vsync for a dropped frame due to stuffed queue
235 nsecs_t mLastDropVsync = 0;
236
John Reck4f02bf42014-01-03 18:09:17 -0800237 bool mOpaque;
Romain Guy26a2b972017-04-17 09:39:51 -0700238 bool mWideColorGamut = false;
John Reckd9d7f122018-05-03 14:40:56 -0700239 LightInfo mLightInfo;
240 LightGeometry mLightGeometry = {{0, 0, 0}, 0};
Chris Craik98787e62015-11-13 10:55:30 -0800241
John Reckedc524c2015-03-18 15:24:33 -0700242 bool mHaveNewSurface = false;
John Recke4267ea2014-06-03 15:53:15 -0700243 DamageAccumulator mDamageAccumulator;
Chris Craik0b7e8242015-10-28 16:50:44 -0700244 LayerUpdateQueue mLayerUpdateQueue;
Chris Craik51d6a3d2014-12-22 17:16:56 -0800245 std::unique_ptr<AnimationContext> mAnimationContext;
John Recke45b1fd2014-04-15 09:50:16 -0700246
John Reck1bcacfd2017-11-03 10:12:19 -0700247 std::vector<sp<RenderNode>> mRenderNodes;
John Reckfe5e7b72014-05-23 17:42:28 -0700248
John Reckedc524c2015-03-18 15:24:33 -0700249 FrameInfo* mCurrentFrameInfo = nullptr;
John Reckb36016c2015-03-11 08:50:53 -0700250 std::string mName;
John Reckedc524c2015-03-18 15:24:33 -0700251 JankTracker mJankTracker;
John Reck4c9e59d2015-05-12 07:17:50 -0700252 FrameInfoVisualizer mProfiler;
Andres Morales910beb82016-02-02 16:19:40 -0800253 std::unique_ptr<FrameMetricsReporter> mFrameMetricsReporter;
John Reck998a6d82014-08-28 15:35:53 -0700254
John Reck51f2d602016-04-06 07:50:47 -0700255 std::set<RenderNode*> mPrefetchedLayers;
Skuhneea7a7fb2015-08-28 07:10:31 -0700256
257 // Stores the bounds of the main content.
Skuhneb8160872015-09-22 09:51:39 -0700258 Rect mContentDrawBounds;
John Reck38f6c032016-03-17 10:23:49 -0700259
260 // TODO: This is really a Task<void> but that doesn't really work
261 // when Future<> expects to be able to get/set a value
262 struct FuncTask : public Task<bool> {
263 std::function<void()> func;
264 };
265 class FuncTaskProcessor;
266
John Reck1bcacfd2017-11-03 10:12:19 -0700267 std::vector<sp<FuncTask>> mFrameFences;
268 sp<TaskProcessor<bool>> mFrameWorkProcessor;
Stan Iliev768e3932016-07-08 21:34:52 -0400269 std::unique_ptr<IRenderPipeline> mRenderPipeline;
John Reckcc2eee82018-05-17 10:44:00 -0700270
271 std::vector<std::function<void(int64_t)>> mFrameCompleteCallbacks;
John Reck23b797a2014-01-03 18:08:34 -0800272};
273
274} /* namespace renderthread */
275} /* namespace uirenderer */
276} /* namespace android */