John Reck | 67b1e2b | 2020-08-26 13:17:24 -0700 | [diff] [blame] | 1 | /* |
| 2 | * Copyright 2020 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 SF_SKIAGLRENDERENGINE_H_ |
| 18 | #define SF_SKIAGLRENDERENGINE_H_ |
| 19 | |
Alec Mouri | b577745 | 2020-09-28 11:32:42 -0700 | [diff] [blame] | 20 | #include <EGL/egl.h> |
| 21 | #include <EGL/eglext.h> |
| 22 | #include <GLES2/gl2.h> |
| 23 | #include <GrDirectContext.h> |
| 24 | #include <SkSurface.h> |
| 25 | #include <android-base/thread_annotations.h> |
Alec Mouri | a90a570 | 2021-04-16 16:36:21 +0000 | [diff] [blame] | 26 | #include <renderengine/ExternalTexture.h> |
Alec Mouri | b577745 | 2020-09-28 11:32:42 -0700 | [diff] [blame] | 27 | #include <renderengine/RenderEngine.h> |
John Reck | 67b1e2b | 2020-08-26 13:17:24 -0700 | [diff] [blame] | 28 | #include <sys/types.h> |
Alec Mouri | b577745 | 2020-09-28 11:32:42 -0700 | [diff] [blame] | 29 | |
John Reck | 67b1e2b | 2020-08-26 13:17:24 -0700 | [diff] [blame] | 30 | #include <mutex> |
| 31 | #include <unordered_map> |
| 32 | |
Alec Mouri | c7f6c8b | 2020-11-09 18:35:20 -0800 | [diff] [blame] | 33 | #include "AutoBackendTexture.h" |
Alec Mouri | b577745 | 2020-09-28 11:32:42 -0700 | [diff] [blame] | 34 | #include "EGL/egl.h" |
Leon Scroggins III | 9f3072c | 2021-03-22 10:42:47 -0400 | [diff] [blame] | 35 | #include "GrContextOptions.h" |
Alec Mouri | c7f6c8b | 2020-11-09 18:35:20 -0800 | [diff] [blame] | 36 | #include "SkImageInfo.h" |
John Reck | 67b1e2b | 2020-08-26 13:17:24 -0700 | [diff] [blame] | 37 | #include "SkiaRenderEngine.h" |
Alec Mouri | c7f6c8b | 2020-11-09 18:35:20 -0800 | [diff] [blame] | 38 | #include "android-base/macros.h" |
Alec Mouri | c0aae73 | 2021-01-12 13:32:18 -0800 | [diff] [blame] | 39 | #include "debug/SkiaCapture.h" |
Lucas Dupin | f4cb4a0 | 2020-09-22 14:19:26 -0700 | [diff] [blame] | 40 | #include "filters/BlurFilter.h" |
Alec Mouri | c0aae73 | 2021-01-12 13:32:18 -0800 | [diff] [blame] | 41 | #include "filters/LinearEffect.h" |
Nader Jawad | 2dfc98b | 2021-04-08 20:35:39 -0700 | [diff] [blame] | 42 | #include "filters/StretchShaderFactory.h" |
John Reck | 67b1e2b | 2020-08-26 13:17:24 -0700 | [diff] [blame] | 43 | |
Kevin Lubick | 8e0ef68 | 2022-03-04 10:34:35 -0500 | [diff] [blame] | 44 | class SkData; |
| 45 | |
Kevin Lubick | bd9374e | 2022-04-22 15:09:19 -0400 | [diff] [blame] | 46 | struct SkPoint3; |
| 47 | |
John Reck | 67b1e2b | 2020-08-26 13:17:24 -0700 | [diff] [blame] | 48 | namespace android { |
| 49 | namespace renderengine { |
| 50 | namespace skia { |
| 51 | |
| 52 | class SkiaGLRenderEngine : public skia::SkiaRenderEngine { |
| 53 | public: |
| 54 | static std::unique_ptr<SkiaGLRenderEngine> create(const RenderEngineCreationArgs& args); |
Lucas Dupin | d508e47 | 2020-11-04 04:32:06 +0000 | [diff] [blame] | 55 | SkiaGLRenderEngine(const RenderEngineCreationArgs& args, EGLDisplay display, EGLContext ctxt, |
| 56 | EGLSurface placeholder, EGLContext protectedContext, |
John Reck | 67b1e2b | 2020-08-26 13:17:24 -0700 | [diff] [blame] | 57 | EGLSurface protectedPlaceholder); |
Alec Mouri | c0aae73 | 2021-01-12 13:32:18 -0800 | [diff] [blame] | 58 | ~SkiaGLRenderEngine() override EXCLUDES(mRenderingMutex); |
John Reck | 67b1e2b | 2020-08-26 13:17:24 -0700 | [diff] [blame] | 59 | |
Ady Abraham | fe2a6db | 2021-06-09 15:41:37 -0700 | [diff] [blame] | 60 | std::future<void> primeCache() override; |
Derek Sollenberger | d3f6065 | 2021-06-11 15:34:36 -0400 | [diff] [blame] | 61 | void cleanupPostRender() override; |
| 62 | void cleanFramebufferCache() override{}; |
Alec Mouri | d6f0946 | 2020-12-07 11:18:17 -0800 | [diff] [blame] | 63 | int getContextPriority() override; |
Lucas Dupin | d508e47 | 2020-11-04 04:32:06 +0000 | [diff] [blame] | 64 | bool isProtected() const override { return mInProtectedContext; } |
| 65 | bool supportsProtectedContent() const override; |
Derek Sollenberger | 1ec2fb5 | 2021-06-16 15:11:27 -0400 | [diff] [blame] | 66 | void useProtectedContext(bool useProtectedContext) override; |
Derek Sollenberger | b399837 | 2021-02-16 15:16:56 -0500 | [diff] [blame] | 67 | bool supportsBackgroundBlur() override { return mBlurFilter != nullptr; } |
Ady Abraham | ed3290f | 2021-05-17 15:12:14 -0700 | [diff] [blame] | 68 | void onActiveDisplaySizeChanged(ui::Size size) override; |
Nathaniel Nifong | b9f27ef | 2021-04-01 16:44:12 -0400 | [diff] [blame] | 69 | int reportShadersCompiled() override; |
John Reck | 67b1e2b | 2020-08-26 13:17:24 -0700 | [diff] [blame] | 70 | |
| 71 | protected: |
Ana Krulec | 1d12b3b | 2021-01-27 16:49:51 -0800 | [diff] [blame] | 72 | void dump(std::string& result) override; |
John Reck | 67b1e2b | 2020-08-26 13:17:24 -0700 | [diff] [blame] | 73 | size_t getMaxTextureSize() const override; |
| 74 | size_t getMaxViewportDims() const override; |
Alec Mouri | a90a570 | 2021-04-16 16:36:21 +0000 | [diff] [blame] | 75 | void mapExternalTextureBuffer(const sp<GraphicBuffer>& buffer, bool isRenderable) override; |
| 76 | void unmapExternalTextureBuffer(const sp<GraphicBuffer>& buffer) override; |
Derek Sollenberger | d3f6065 | 2021-06-11 15:34:36 -0400 | [diff] [blame] | 77 | bool canSkipPostRenderCleanup() const override; |
Sally Qi | 4cabdd0 | 2021-08-05 16:45:57 -0700 | [diff] [blame] | 78 | void drawLayersInternal(const std::shared_ptr<std::promise<RenderEngineResult>>&& resultPromise, |
| 79 | const DisplaySettings& display, |
Sally Qi | 59a9f50 | 2021-10-12 18:53:23 +0000 | [diff] [blame] | 80 | const std::vector<LayerSettings>& layers, |
Sally Qi | 4cabdd0 | 2021-08-05 16:45:57 -0700 | [diff] [blame] | 81 | const std::shared_ptr<ExternalTexture>& buffer, |
| 82 | const bool useFramebufferCache, base::unique_fd&& bufferFence) override; |
John Reck | 67b1e2b | 2020-08-26 13:17:24 -0700 | [diff] [blame] | 83 | |
| 84 | private: |
| 85 | static EGLConfig chooseEglConfig(EGLDisplay display, int format, bool logConfig); |
| 86 | static EGLContext createEglContext(EGLDisplay display, EGLConfig config, |
Alec Mouri | d6f0946 | 2020-12-07 11:18:17 -0800 | [diff] [blame] | 87 | EGLContext shareContext, |
| 88 | std::optional<ContextPriority> contextPriority, |
John Reck | 67b1e2b | 2020-08-26 13:17:24 -0700 | [diff] [blame] | 89 | Protection protection); |
Alec Mouri | d6f0946 | 2020-12-07 11:18:17 -0800 | [diff] [blame] | 90 | static std::optional<RenderEngine::ContextPriority> createContextPriority( |
| 91 | const RenderEngineCreationArgs& args); |
John Reck | 67b1e2b | 2020-08-26 13:17:24 -0700 | [diff] [blame] | 92 | static EGLSurface createPlaceholderEglPbufferSurface(EGLDisplay display, EGLConfig config, |
| 93 | int hwcFormat, Protection protection); |
Lucas Dupin | 3f11e92 | 2020-09-22 17:31:04 -0700 | [diff] [blame] | 94 | inline SkRect getSkRect(const FloatRect& layer); |
| 95 | inline SkRect getSkRect(const Rect& layer); |
Derek Sollenberger | c31985e | 2021-05-18 16:38:17 -0400 | [diff] [blame] | 96 | inline std::pair<SkRRect, SkRRect> getBoundsAndClip(const FloatRect& bounds, |
Vishnu Nair | 50c0afe | 2022-07-11 15:04:07 -0700 | [diff] [blame] | 97 | const FloatRect& crop, |
| 98 | const vec2& cornerRadius); |
Sally Qi | 59a9f50 | 2021-10-12 18:53:23 +0000 | [diff] [blame] | 99 | inline bool layerHasBlur(const LayerSettings& layer, bool colorTransformModifiesAlpha); |
Lucas Dupin | 3f11e92 | 2020-09-22 17:31:04 -0700 | [diff] [blame] | 100 | inline SkColor getSkColor(const vec4& color); |
Lucas Dupin | bb1a1d4 | 2020-09-18 15:17:02 -0700 | [diff] [blame] | 101 | inline SkM44 getSkM44(const mat4& matrix); |
Lucas Dupin | 3f11e92 | 2020-09-22 17:31:04 -0700 | [diff] [blame] | 102 | inline SkPoint3 getSkPoint3(const vec3& vector); |
Derek Sollenberger | b24258c | 2021-05-04 13:47:34 -0400 | [diff] [blame] | 103 | inline GrDirectContext* getActiveGrContext() const; |
John Reck | 67b1e2b | 2020-08-26 13:17:24 -0700 | [diff] [blame] | 104 | |
| 105 | base::unique_fd flush(); |
Alec Mouri | 4ee7b49 | 2021-08-11 10:36:55 -0700 | [diff] [blame] | 106 | // waitFence attempts to wait in the GPU, and if unable to waits on the CPU instead. |
| 107 | void waitFence(base::borrowed_fd fenceFd); |
| 108 | bool waitGpuFence(base::borrowed_fd fenceFd); |
| 109 | |
Derek Sollenberger | 3f77aa4 | 2021-02-04 11:06:34 -0500 | [diff] [blame] | 110 | void initCanvas(SkCanvas* canvas, const DisplaySettings& display); |
Derek Sollenberger | b0e764c | 2021-05-04 14:31:37 -0400 | [diff] [blame] | 111 | void drawShadow(SkCanvas* canvas, const SkRRect& casterRRect, |
Lucas Dupin | 3f11e92 | 2020-09-22 17:31:04 -0700 | [diff] [blame] | 112 | const ShadowSettings& shadowSettings); |
Alec Mouri | cdf6cbc | 2021-11-01 17:21:15 -0700 | [diff] [blame] | 113 | |
Derek Sollenberger | e2fe78c | 2021-02-23 13:22:54 -0500 | [diff] [blame] | 114 | // If requiresLinearEffect is true or the layer has a stretchEffect a new shader is returned. |
| 115 | // Otherwise it returns the input shader. |
Alec Mouri | cdf6cbc | 2021-11-01 17:21:15 -0700 | [diff] [blame] | 116 | struct RuntimeEffectShaderParameters { |
| 117 | sk_sp<SkShader> shader; |
| 118 | const LayerSettings& layer; |
| 119 | const DisplaySettings& display; |
| 120 | bool undoPremultipliedAlpha; |
| 121 | bool requiresLinearEffect; |
| 122 | float layerDimmingRatio; |
| 123 | }; |
| 124 | sk_sp<SkShader> createRuntimeEffectShader(const RuntimeEffectShaderParameters&); |
John Reck | 67b1e2b | 2020-08-26 13:17:24 -0700 | [diff] [blame] | 125 | |
| 126 | EGLDisplay mEGLDisplay; |
John Reck | 67b1e2b | 2020-08-26 13:17:24 -0700 | [diff] [blame] | 127 | EGLContext mEGLContext; |
| 128 | EGLSurface mPlaceholderSurface; |
| 129 | EGLContext mProtectedEGLContext; |
| 130 | EGLSurface mProtectedPlaceholderSurface; |
Lucas Dupin | f4cb4a0 | 2020-09-22 14:19:26 -0700 | [diff] [blame] | 131 | BlurFilter* mBlurFilter = nullptr; |
John Reck | 67b1e2b | 2020-08-26 13:17:24 -0700 | [diff] [blame] | 132 | |
Derek Sollenberger | c4a05e1 | 2021-03-24 16:45:20 -0400 | [diff] [blame] | 133 | const PixelFormat mDefaultPixelFormat; |
Alec Mouri | b577745 | 2020-09-28 11:32:42 -0700 | [diff] [blame] | 134 | const bool mUseColorManagement; |
| 135 | |
Nader Jawad | 2dfc98b | 2021-04-08 20:35:39 -0700 | [diff] [blame] | 136 | // Identifier used or various mappings of layers to various |
| 137 | // textures or shaders |
Nader Jawad | c088bdc | 2021-05-10 13:24:46 -0700 | [diff] [blame] | 138 | using GraphicBufferId = uint64_t; |
Nader Jawad | 2dfc98b | 2021-04-08 20:35:39 -0700 | [diff] [blame] | 139 | |
Alec Mouri | a90a570 | 2021-04-16 16:36:21 +0000 | [diff] [blame] | 140 | // Number of external holders of ExternalTexture references, per GraphicBuffer ID. |
Nader Jawad | c088bdc | 2021-05-10 13:24:46 -0700 | [diff] [blame] | 141 | std::unordered_map<GraphicBufferId, int32_t> mGraphicBufferExternalRefs |
Alec Mouri | c7f6c8b | 2020-11-09 18:35:20 -0800 | [diff] [blame] | 142 | GUARDED_BY(mRenderingMutex); |
Derek Sollenberger | 4500718 | 2021-06-10 14:47:21 -0400 | [diff] [blame] | 143 | // Cache of GL textures that we'll store per GraphicBuffer ID, shared between GPU contexts. |
Nader Jawad | c088bdc | 2021-05-10 13:24:46 -0700 | [diff] [blame] | 144 | std::unordered_map<GraphicBufferId, std::shared_ptr<AutoBackendTexture::LocalRef>> mTextureCache |
| 145 | GUARDED_BY(mRenderingMutex); |
Alec Mouri | 492c85c | 2021-11-19 15:58:10 -0800 | [diff] [blame] | 146 | std::unordered_map<shaders::LinearEffect, sk_sp<SkRuntimeEffect>, shaders::LinearEffectHasher> |
| 147 | mRuntimeEffects; |
Derek Sollenberger | d3f6065 | 2021-06-11 15:34:36 -0400 | [diff] [blame] | 148 | AutoBackendTexture::CleanupManager mTextureCleanupMgr GUARDED_BY(mRenderingMutex); |
Nader Jawad | 2dfc98b | 2021-04-08 20:35:39 -0700 | [diff] [blame] | 149 | |
| 150 | StretchShaderFactory mStretchShaderFactory; |
John Reck | 67b1e2b | 2020-08-26 13:17:24 -0700 | [diff] [blame] | 151 | // Mutex guarding rendering operations, so that: |
| 152 | // 1. GL operations aren't interleaved, and |
| 153 | // 2. Internal state related to rendering that is potentially modified by |
| 154 | // multiple threads is guaranteed thread-safe. |
Derek Sollenberger | d3f6065 | 2021-06-11 15:34:36 -0400 | [diff] [blame] | 155 | mutable std::mutex mRenderingMutex; |
John Reck | 67b1e2b | 2020-08-26 13:17:24 -0700 | [diff] [blame] | 156 | |
| 157 | sp<Fence> mLastDrawFence; |
| 158 | |
Lucas Dupin | d508e47 | 2020-11-04 04:32:06 +0000 | [diff] [blame] | 159 | // Graphics context used for creating surfaces and submitting commands |
John Reck | 67b1e2b | 2020-08-26 13:17:24 -0700 | [diff] [blame] | 160 | sk_sp<GrDirectContext> mGrContext; |
Lucas Dupin | d508e47 | 2020-11-04 04:32:06 +0000 | [diff] [blame] | 161 | // Same as above, but for protected content (eg. DRM) |
| 162 | sk_sp<GrDirectContext> mProtectedGrContext; |
John Reck | 67b1e2b | 2020-08-26 13:17:24 -0700 | [diff] [blame] | 163 | |
Lucas Dupin | d508e47 | 2020-11-04 04:32:06 +0000 | [diff] [blame] | 164 | bool mInProtectedContext = false; |
Ana Krulec | 70d15b1b | 2020-12-01 10:05:15 -0800 | [diff] [blame] | 165 | // Object to capture commands send to Skia. |
Alec Mouri | c0aae73 | 2021-01-12 13:32:18 -0800 | [diff] [blame] | 166 | std::unique_ptr<SkiaCapture> mCapture; |
Leon Scroggins III | 9f3072c | 2021-03-22 10:42:47 -0400 | [diff] [blame] | 167 | |
| 168 | // Implements PersistentCache as a way to monitor what SkSL shaders Skia has |
| 169 | // cached. |
| 170 | class SkSLCacheMonitor : public GrContextOptions::PersistentCache { |
| 171 | public: |
| 172 | SkSLCacheMonitor() = default; |
| 173 | ~SkSLCacheMonitor() override = default; |
| 174 | |
| 175 | sk_sp<SkData> load(const SkData& key) override; |
| 176 | |
| 177 | void store(const SkData& key, const SkData& data, const SkString& description) override; |
| 178 | |
| 179 | int shadersCachedSinceLastCall() { |
| 180 | const int shadersCachedSinceLastCall = mShadersCachedSinceLastCall; |
| 181 | mShadersCachedSinceLastCall = 0; |
| 182 | return shadersCachedSinceLastCall; |
| 183 | } |
| 184 | |
Leon Scroggins III | 45be918 | 2022-04-27 10:37:11 -0400 | [diff] [blame] | 185 | int totalShadersCompiled() const { return mTotalShadersCompiled; } |
| 186 | |
Leon Scroggins III | 9f3072c | 2021-03-22 10:42:47 -0400 | [diff] [blame] | 187 | private: |
| 188 | int mShadersCachedSinceLastCall = 0; |
Leon Scroggins III | adf18d8 | 2022-04-22 13:23:05 -0400 | [diff] [blame] | 189 | int mTotalShadersCompiled = 0; |
Leon Scroggins III | 9f3072c | 2021-03-22 10:42:47 -0400 | [diff] [blame] | 190 | }; |
| 191 | |
| 192 | SkSLCacheMonitor mSkSLCacheMonitor; |
John Reck | 67b1e2b | 2020-08-26 13:17:24 -0700 | [diff] [blame] | 193 | }; |
| 194 | |
| 195 | } // namespace skia |
| 196 | } // namespace renderengine |
| 197 | } // namespace android |
| 198 | |
Galia Peycheva | 6c46065 | 2020-11-03 19:42:42 +0100 | [diff] [blame] | 199 | #endif /* SF_GLESRENDERENGINE_H_ */ |