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