John Reck | 23b797a | 2014-01-03 18:08:34 -0800 | [diff] [blame] | 1 | /* |
| 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 | |
| 17 | #ifndef CANVASCONTEXT_H_ |
| 18 | #define CANVASCONTEXT_H_ |
| 19 | |
John Reck | ba6adf6 | 2015-02-19 14:36:50 -0800 | [diff] [blame] | 20 | #include "DamageAccumulator.h" |
John Reck | ba6adf6 | 2015-02-19 14:36:50 -0800 | [diff] [blame] | 21 | #include "FrameInfo.h" |
John Reck | 4c9e59d | 2015-05-12 07:17:50 -0700 | [diff] [blame] | 22 | #include "FrameInfoVisualizer.h" |
Andres Morales | 06f5bc7 | 2015-12-15 15:21:31 -0800 | [diff] [blame] | 23 | #include "FrameStatsReporter.h" |
Chris Craik | 0b7e824 | 2015-10-28 16:50:44 -0700 | [diff] [blame] | 24 | #include "IContextFactory.h" |
| 25 | #include "LayerUpdateQueue.h" |
John Reck | ba6adf6 | 2015-02-19 14:36:50 -0800 | [diff] [blame] | 26 | #include "RenderNode.h" |
| 27 | #include "utils/RingBuffer.h" |
| 28 | #include "renderthread/RenderTask.h" |
| 29 | #include "renderthread/RenderThread.h" |
John Reck | 998a6d8 | 2014-08-28 15:35:53 -0700 | [diff] [blame] | 30 | |
Chris Craik | 98787e6 | 2015-11-13 10:55:30 -0800 | [diff] [blame] | 31 | #if HWUI_NEW_OPS |
Chris Craik | 9e7fcfd | 2015-11-25 13:27:33 -0800 | [diff] [blame] | 32 | #include "BakedOpDispatcher.h" |
Chris Craik | 98787e6 | 2015-11-13 10:55:30 -0800 | [diff] [blame] | 33 | #include "BakedOpRenderer.h" |
| 34 | #endif |
| 35 | |
John Reck | 23b797a | 2014-01-03 18:08:34 -0800 | [diff] [blame] | 36 | #include <cutils/compiler.h> |
| 37 | #include <EGL/egl.h> |
John Reck | 19b6bcf | 2014-02-14 20:03:38 -0800 | [diff] [blame] | 38 | #include <SkBitmap.h> |
John Reck | d04794a | 2015-05-08 10:04:36 -0700 | [diff] [blame] | 39 | #include <SkRect.h> |
John Reck | 4f02bf4 | 2014-01-03 18:09:17 -0800 | [diff] [blame] | 40 | #include <utils/Functor.h> |
| 41 | |
John Reck | ba6adf6 | 2015-02-19 14:36:50 -0800 | [diff] [blame] | 42 | #include <set> |
John Reck | b36016c | 2015-03-11 08:50:53 -0700 | [diff] [blame] | 43 | #include <string> |
Skuhne | ea7a7fb | 2015-08-28 07:10:31 -0700 | [diff] [blame] | 44 | #include <vector> |
John Reck | 23b797a | 2014-01-03 18:08:34 -0800 | [diff] [blame] | 45 | |
| 46 | namespace android { |
| 47 | namespace uirenderer { |
John Reck | 4f02bf4 | 2014-01-03 18:09:17 -0800 | [diff] [blame] | 48 | |
John Reck | 119907c | 2014-08-14 09:02:01 -0700 | [diff] [blame] | 49 | class AnimationContext; |
John Reck | 19b6bcf | 2014-02-14 20:03:38 -0800 | [diff] [blame] | 50 | class DeferredLayerUpdater; |
John Reck | 4f02bf4 | 2014-01-03 18:09:17 -0800 | [diff] [blame] | 51 | class OpenGLRenderer; |
| 52 | class Rect; |
John Reck | 1949e79 | 2014-04-08 15:18:56 -0700 | [diff] [blame] | 53 | class Layer; |
John Reck | 443a714 | 2014-09-04 17:40:05 -0700 | [diff] [blame] | 54 | class RenderState; |
John Reck | 4f02bf4 | 2014-01-03 18:09:17 -0800 | [diff] [blame] | 55 | |
John Reck | 23b797a | 2014-01-03 18:08:34 -0800 | [diff] [blame] | 56 | namespace renderthread { |
| 57 | |
John Reck | 3b20251 | 2014-06-23 13:13:08 -0700 | [diff] [blame] | 58 | class EglManager; |
John Reck | 4f02bf4 | 2014-01-03 18:09:17 -0800 | [diff] [blame] | 59 | |
John Reck | 1125d1f | 2014-10-23 11:02:19 -0700 | [diff] [blame] | 60 | enum SwapBehavior { |
| 61 | kSwap_default, |
| 62 | kSwap_discardBuffer, |
| 63 | }; |
| 64 | |
John Reck | 23b797a | 2014-01-03 18:08:34 -0800 | [diff] [blame] | 65 | // This per-renderer class manages the bridge between the global EGL context |
| 66 | // and the render surface. |
John Reck | 119907c | 2014-08-14 09:02:01 -0700 | [diff] [blame] | 67 | // TODO: Rename to Renderer or some other per-window, top-level manager |
John Reck | e45b1fd | 2014-04-15 09:50:16 -0700 | [diff] [blame] | 68 | class CanvasContext : public IFrameCallback { |
John Reck | 23b797a | 2014-01-03 18:08:34 -0800 | [diff] [blame] | 69 | public: |
John Reck | 119907c | 2014-08-14 09:02:01 -0700 | [diff] [blame] | 70 | CanvasContext(RenderThread& thread, bool translucent, RenderNode* rootRenderNode, |
| 71 | IContextFactory* contextFactory); |
John Reck | e45b1fd | 2014-04-15 09:50:16 -0700 | [diff] [blame] | 72 | virtual ~CanvasContext(); |
John Reck | 23b797a | 2014-01-03 18:08:34 -0800 | [diff] [blame] | 73 | |
John Reck | 1125d1f | 2014-10-23 11:02:19 -0700 | [diff] [blame] | 74 | // Won't take effect until next EGLSurface creation |
| 75 | void setSwapBehavior(SwapBehavior swapBehavior); |
| 76 | |
Thomas Buhot | 0bcd0cb | 2015-12-04 12:18:03 +0100 | [diff] [blame] | 77 | void initialize(ANativeWindow* window); |
John Reck | a5dda64 | 2014-05-22 15:43:54 -0700 | [diff] [blame] | 78 | void updateSurface(ANativeWindow* window); |
John Reck | 01a5ea3 | 2014-12-03 13:01:07 -0800 | [diff] [blame] | 79 | bool pauseSurface(ANativeWindow* window); |
John Reck | aa95a88 | 2014-11-07 11:02:07 -0800 | [diff] [blame] | 80 | bool hasSurface() { return mNativeWindow.get(); } |
| 81 | |
Alan Viverette | 50210d9 | 2015-05-14 18:05:36 -0700 | [diff] [blame] | 82 | void setup(int width, int height, float lightRadius, |
Chris Craik | 058fc64 | 2014-07-23 18:19:28 -0700 | [diff] [blame] | 83 | uint8_t ambientShadowAlpha, uint8_t spotShadowAlpha); |
Alan Viverette | 50210d9 | 2015-05-14 18:05:36 -0700 | [diff] [blame] | 84 | void setLightCenter(const Vector3& lightCenter); |
John Reck | 63a0667 | 2014-05-07 13:45:54 -0700 | [diff] [blame] | 85 | void setOpaque(bool opaque); |
John Reck | 860d155 | 2014-04-11 19:15:05 -0700 | [diff] [blame] | 86 | void makeCurrent(); |
Skuhne | ea7a7fb | 2015-08-28 07:10:31 -0700 | [diff] [blame] | 87 | void prepareTree(TreeInfo& info, int64_t* uiFrameInfo, |
| 88 | int64_t syncQueued, RenderNode* target); |
John Reck | e4267ea | 2014-06-03 15:53:15 -0700 | [diff] [blame] | 89 | void draw(); |
John Reck | 17035b0 | 2014-09-03 07:39:53 -0700 | [diff] [blame] | 90 | void destroy(); |
John Reck | 23b797a | 2014-01-03 18:08:34 -0800 | [diff] [blame] | 91 | |
Chris Craik | 0b7e824 | 2015-10-28 16:50:44 -0700 | [diff] [blame] | 92 | // IFrameCallback, Choreographer-driven frame callback entry point |
Chris Craik | d41c4d8 | 2015-01-05 15:51:13 -0800 | [diff] [blame] | 93 | virtual void doFrame() override; |
Skuhne | ea7a7fb | 2015-08-28 07:10:31 -0700 | [diff] [blame] | 94 | void prepareAndDraw(RenderNode* node); |
John Reck | e45b1fd | 2014-04-15 09:50:16 -0700 | [diff] [blame] | 95 | |
John Reck | 3e82495 | 2014-08-20 10:08:39 -0700 | [diff] [blame] | 96 | void buildLayer(RenderNode* node); |
John Reck | 19b6bcf | 2014-02-14 20:03:38 -0800 | [diff] [blame] | 97 | bool copyLayerInto(DeferredLayerUpdater* layer, SkBitmap* bitmap); |
John Reck | 998a6d8 | 2014-08-28 15:35:53 -0700 | [diff] [blame] | 98 | void markLayerInUse(RenderNode* node); |
John Reck | 19b6bcf | 2014-02-14 20:03:38 -0800 | [diff] [blame] | 99 | |
John Reck | f47a594 | 2014-06-30 16:20:04 -0700 | [diff] [blame] | 100 | void destroyHardwareResources(); |
| 101 | static void trimMemory(RenderThread& thread, int level); |
John Reck | e1628b7 | 2014-05-23 15:11:19 -0700 | [diff] [blame] | 102 | |
John Reck | 3b20251 | 2014-06-23 13:13:08 -0700 | [diff] [blame] | 103 | static void invokeFunctor(RenderThread& thread, Functor* functor); |
John Reck | 23b797a | 2014-01-03 18:08:34 -0800 | [diff] [blame] | 104 | |
John Reck | fc53ef27 | 2014-02-11 10:40:25 -0800 | [diff] [blame] | 105 | void runWithGlContext(RenderTask* task); |
| 106 | |
John Reck | 1949e79 | 2014-04-08 15:18:56 -0700 | [diff] [blame] | 107 | Layer* createTextureLayer(); |
| 108 | |
John Reck | 3b20251 | 2014-06-23 13:13:08 -0700 | [diff] [blame] | 109 | ANDROID_API static void setTextureAtlas(RenderThread& thread, |
| 110 | const sp<GraphicBuffer>& buffer, int64_t* map, size_t mapSize); |
John Reck | 66f0be6 | 2014-05-13 13:39:31 -0700 | [diff] [blame] | 111 | |
John Reck | f47a594 | 2014-06-30 16:20:04 -0700 | [diff] [blame] | 112 | void stopDrawing(); |
John Reck | a5dda64 | 2014-05-22 15:43:54 -0700 | [diff] [blame] | 113 | void notifyFramePending(); |
| 114 | |
John Reck | 4c9e59d | 2015-05-12 07:17:50 -0700 | [diff] [blame] | 115 | FrameInfoVisualizer& profiler() { return mProfiler; } |
John Reck | fe5e7b7 | 2014-05-23 17:42:28 -0700 | [diff] [blame] | 116 | |
John Reck | ba6adf6 | 2015-02-19 14:36:50 -0800 | [diff] [blame] | 117 | void dumpFrames(int fd); |
| 118 | void resetFrameStats(); |
| 119 | |
John Reck | b36016c | 2015-03-11 08:50:53 -0700 | [diff] [blame] | 120 | void setName(const std::string&& name) { mName = name; } |
| 121 | const std::string& name() { return mName; } |
| 122 | |
John Reck | e248bd1 | 2015-08-05 13:53:53 -0700 | [diff] [blame] | 123 | void serializeDisplayListTree(); |
| 124 | |
Skuhne | ea7a7fb | 2015-08-28 07:10:31 -0700 | [diff] [blame] | 125 | void addRenderNode(RenderNode* node, bool placeFront) { |
| 126 | int pos = placeFront ? 0 : static_cast<int>(mRenderNodes.size()); |
Chris Craik | 0b7e824 | 2015-10-28 16:50:44 -0700 | [diff] [blame] | 127 | mRenderNodes.emplace(mRenderNodes.begin() + pos, node); |
Skuhne | ea7a7fb | 2015-08-28 07:10:31 -0700 | [diff] [blame] | 128 | } |
| 129 | |
| 130 | void removeRenderNode(RenderNode* node) { |
| 131 | mRenderNodes.erase(std::remove(mRenderNodes.begin(), mRenderNodes.end(), node), |
| 132 | mRenderNodes.end()); |
| 133 | } |
| 134 | |
Skuhne | b816087 | 2015-09-22 09:51:39 -0700 | [diff] [blame] | 135 | void setContentDrawBounds(int left, int top, int right, int bottom) { |
| 136 | mContentDrawBounds.set(left, top, right, bottom); |
Skuhne | ea7a7fb | 2015-08-28 07:10:31 -0700 | [diff] [blame] | 137 | } |
| 138 | |
Chris Craik | e2e53a7 | 2015-10-28 15:55:40 -0700 | [diff] [blame] | 139 | RenderState& getRenderState() { |
| 140 | return mRenderThread.renderState(); |
| 141 | } |
| 142 | |
Andres Morales | 06f5bc7 | 2015-12-15 15:21:31 -0800 | [diff] [blame] | 143 | void addFrameStatsObserver(FrameStatsObserver* observer) { |
| 144 | if (mFrameStatsReporter.get() == nullptr) { |
| 145 | mFrameStatsReporter.reset(new FrameStatsReporter()); |
| 146 | } |
| 147 | |
| 148 | mFrameStatsReporter->addObserver(observer); |
| 149 | } |
| 150 | |
| 151 | void removeFrameStatsObserver(FrameStatsObserver* observer) { |
| 152 | if (mFrameStatsReporter.get() != nullptr) { |
| 153 | mFrameStatsReporter->removeObserver(observer); |
| 154 | if (!mFrameStatsReporter->hasObservers()) { |
| 155 | mFrameStatsReporter.reset(nullptr); |
| 156 | } |
| 157 | } |
| 158 | } |
| 159 | |
| 160 | long getDroppedFrameReportCount() { |
| 161 | if (mFrameStatsReporter.get() != nullptr) { |
| 162 | return mFrameStatsReporter->getDroppedReports(); |
| 163 | } |
| 164 | |
| 165 | return 0; |
| 166 | } |
| 167 | |
John Reck | 23b797a | 2014-01-03 18:08:34 -0800 | [diff] [blame] | 168 | private: |
John Reck | a5dda64 | 2014-05-22 15:43:54 -0700 | [diff] [blame] | 169 | friend class RegisterFrameCallbackTask; |
John Reck | 443a714 | 2014-09-04 17:40:05 -0700 | [diff] [blame] | 170 | // TODO: Replace with something better for layer & other GL object |
| 171 | // lifecycle tracking |
| 172 | friend class android::uirenderer::RenderState; |
John Reck | a5dda64 | 2014-05-22 15:43:54 -0700 | [diff] [blame] | 173 | |
John Reck | a5dda64 | 2014-05-22 15:43:54 -0700 | [diff] [blame] | 174 | void setSurface(ANativeWindow* window); |
John Reck | f7d9c1d | 2014-04-09 10:01:03 -0700 | [diff] [blame] | 175 | void requireSurface(); |
John Reck | 4f02bf4 | 2014-01-03 18:09:17 -0800 | [diff] [blame] | 176 | |
John Reck | 998a6d8 | 2014-08-28 15:35:53 -0700 | [diff] [blame] | 177 | void freePrefetechedLayers(); |
| 178 | |
John Reck | 77c4010 | 2015-10-26 15:49:47 -0700 | [diff] [blame] | 179 | EGLint mLastFrameWidth = 0; |
| 180 | EGLint mLastFrameHeight = 0; |
Chris Craik | ddf2215 | 2015-10-14 17:42:47 -0700 | [diff] [blame] | 181 | |
John Reck | 4f02bf4 | 2014-01-03 18:09:17 -0800 | [diff] [blame] | 182 | RenderThread& mRenderThread; |
John Reck | 3b20251 | 2014-06-23 13:13:08 -0700 | [diff] [blame] | 183 | EglManager& mEglManager; |
John Reck | a5dda64 | 2014-05-22 15:43:54 -0700 | [diff] [blame] | 184 | sp<ANativeWindow> mNativeWindow; |
John Reck | edc524c | 2015-03-18 15:24:33 -0700 | [diff] [blame] | 185 | EGLSurface mEglSurface = EGL_NO_SURFACE; |
| 186 | bool mBufferPreserved = false; |
| 187 | SwapBehavior mSwapBehavior = kSwap_default; |
John Reck | e486d93 | 2015-10-28 09:21:19 -0700 | [diff] [blame] | 188 | struct SwapHistory { |
| 189 | SkRect damage; |
| 190 | nsecs_t vsyncTime; |
| 191 | nsecs_t swapTime; |
| 192 | }; |
| 193 | |
| 194 | RingBuffer<SwapHistory, 3> mSwapHistory; |
John Reck | 4f02bf4 | 2014-01-03 18:09:17 -0800 | [diff] [blame] | 195 | |
| 196 | bool mOpaque; |
John Reck | edc524c | 2015-03-18 15:24:33 -0700 | [diff] [blame] | 197 | OpenGLRenderer* mCanvas = nullptr; |
Chris Craik | 98787e6 | 2015-11-13 10:55:30 -0800 | [diff] [blame] | 198 | #if HWUI_NEW_OPS |
| 199 | BakedOpRenderer::LightInfo mLightInfo; |
| 200 | Vector3 mLightCenter = { 0, 0, 0 }; |
| 201 | #endif |
| 202 | |
John Reck | edc524c | 2015-03-18 15:24:33 -0700 | [diff] [blame] | 203 | bool mHaveNewSurface = false; |
John Reck | e4267ea | 2014-06-03 15:53:15 -0700 | [diff] [blame] | 204 | DamageAccumulator mDamageAccumulator; |
Chris Craik | 0b7e824 | 2015-10-28 16:50:44 -0700 | [diff] [blame] | 205 | LayerUpdateQueue mLayerUpdateQueue; |
Chris Craik | 51d6a3d | 2014-12-22 17:16:56 -0800 | [diff] [blame] | 206 | std::unique_ptr<AnimationContext> mAnimationContext; |
John Reck | e45b1fd | 2014-04-15 09:50:16 -0700 | [diff] [blame] | 207 | |
Skuhne | ea7a7fb | 2015-08-28 07:10:31 -0700 | [diff] [blame] | 208 | std::vector< sp<RenderNode> > mRenderNodes; |
John Reck | fe5e7b7 | 2014-05-23 17:42:28 -0700 | [diff] [blame] | 209 | |
John Reck | edc524c | 2015-03-18 15:24:33 -0700 | [diff] [blame] | 210 | FrameInfo* mCurrentFrameInfo = nullptr; |
John Reck | 4c9e59d | 2015-05-12 07:17:50 -0700 | [diff] [blame] | 211 | // Ring buffer large enough for 2 seconds worth of frames |
| 212 | RingBuffer<FrameInfo, 120> mFrames; |
John Reck | b36016c | 2015-03-11 08:50:53 -0700 | [diff] [blame] | 213 | std::string mName; |
John Reck | edc524c | 2015-03-18 15:24:33 -0700 | [diff] [blame] | 214 | JankTracker mJankTracker; |
John Reck | 4c9e59d | 2015-05-12 07:17:50 -0700 | [diff] [blame] | 215 | FrameInfoVisualizer mProfiler; |
Andres Morales | 06f5bc7 | 2015-12-15 15:21:31 -0800 | [diff] [blame] | 216 | std::unique_ptr<FrameStatsReporter> mFrameStatsReporter; |
John Reck | 998a6d8 | 2014-08-28 15:35:53 -0700 | [diff] [blame] | 217 | |
| 218 | std::set<RenderNode*> mPrefetechedLayers; |
Skuhne | ea7a7fb | 2015-08-28 07:10:31 -0700 | [diff] [blame] | 219 | |
| 220 | // Stores the bounds of the main content. |
Skuhne | b816087 | 2015-09-22 09:51:39 -0700 | [diff] [blame] | 221 | Rect mContentDrawBounds; |
John Reck | 23b797a | 2014-01-03 18:08:34 -0800 | [diff] [blame] | 222 | }; |
| 223 | |
| 224 | } /* namespace renderthread */ |
| 225 | } /* namespace uirenderer */ |
| 226 | } /* namespace android */ |
| 227 | #endif /* CANVASCONTEXT_H_ */ |