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