Flutter Engine
 
Loading...
Searching...
No Matches
android_surface_gl_skia.cc
Go to the documentation of this file.
1// Copyright 2013 The Flutter Authors. All rights reserved.
2// Use of this source code is governed by a BSD-style license that can be
3// found in the LICENSE file.
4
6
7#include <GLES/gl.h>
8#include <utility>
9
10#include "flutter/fml/logging.h"
14
15namespace flutter {
16
17namespace {
18// GL renderer string prefix used by the Android emulator GLES implementation.
19constexpr char kEmulatorRendererPrefix[] =
20 "Android Emulator OpenGL ES Translator";
21} // anonymous namespace
22
24 const std::shared_ptr<AndroidContextGLSkia>& android_context)
25 : android_context_(android_context),
26 native_window_(nullptr),
27 onscreen_surface_(nullptr),
28 offscreen_surface_(nullptr) {
29 // Acquire the offscreen surface.
30 offscreen_surface_ = android_context_->CreateOffscreenSurface();
31 if (!offscreen_surface_->IsValid()) {
32 offscreen_surface_ = nullptr;
33 }
34}
35
37
39 // When the onscreen surface is destroyed, the context and the surface
40 // instance should be deleted. Issue:
41 // https://github.com/flutter/flutter/issues/64414
42 android_context_->ClearCurrent();
43 onscreen_surface_ = nullptr;
44}
45
47 return offscreen_surface_ && android_context_->IsValid();
48}
49
51 GrDirectContext* gr_context) {
52 if (gr_context) {
53 return std::make_unique<GPUSurfaceGLSkia>(sk_ref_sp(gr_context), this,
54 true);
55 } else {
56 sk_sp<GrDirectContext> main_skia_context =
57 android_context_->GetMainSkiaContext();
58 if (!main_skia_context) {
59 main_skia_context = GPUSurfaceGLSkia::MakeGLContext(this);
60 android_context_->SetMainSkiaContext(main_skia_context);
61 }
62 return std::make_unique<GPUSurfaceGLSkia>(main_skia_context, this, true);
63 }
64}
65
68 FML_DCHECK(onscreen_surface_);
69 FML_DCHECK(native_window_);
70
71 if (size == onscreen_surface_->GetSize()) {
72 return true;
73 }
74
75 android_context_->ClearCurrent();
76
77 // Ensure the destructor is called since it destroys the `EGLSurface` before
78 // creating a new onscreen surface.
79 onscreen_surface_ = nullptr;
80 onscreen_surface_ = android_context_->CreateOnscreenSurface(native_window_);
81 if (!onscreen_surface_->IsValid()) {
82 FML_LOG(ERROR) << "Unable to create EGL window surface on resize.";
83 return false;
84 }
85 onscreen_surface_->MakeCurrent();
86 return true;
87}
88
91 auto status = offscreen_surface_->MakeCurrent();
93}
94
97 EGLBoolean result = eglMakeCurrent(eglGetCurrentDisplay(), EGL_NO_SURFACE,
98 EGL_NO_SURFACE, EGL_NO_CONTEXT);
99 return result == EGL_TRUE;
100}
101
104 const std::shared_ptr<PlatformViewAndroidJNI>& jni_facade) {
107 native_window_ = window;
108 // Ensure the destructor is called since it destroys the `EGLSurface` before
109 // creating a new onscreen surface.
110 onscreen_surface_ = nullptr;
111 // Create the onscreen surface.
112 onscreen_surface_ = android_context_->CreateOnscreenSurface(window);
113 if (!onscreen_surface_->IsValid()) {
114 return false;
115 }
116 return true;
117}
118
119std::unique_ptr<GLContextResult> AndroidSurfaceGLSkia::GLContextMakeCurrent() {
121 FML_DCHECK(onscreen_surface_);
122 auto status = onscreen_surface_->MakeCurrent();
123 auto default_context_result = std::make_unique<GLContextDefaultResult>(
125 return std::move(default_context_result);
126}
127
130 return android_context_->ClearCurrent();
131}
132
134 const {
137 res.supports_readback = true;
138 res.supports_partial_repaint = onscreen_surface_->SupportsPartialRepaint();
139 res.existing_damage = onscreen_surface_->InitialDamage();
140 // Some devices (Pixel2 XL) needs EGL_KHR_partial_update rect aligned to 4,
141 // otherwise there are glitches
142 // (https://github.com/flutter/flutter/issues/97482#)
143 // Larger alignment might also be beneficial for tile base renderers.
146
147 return res;
148}
149
151 const std::optional<DlIRect>& region) {
153 onscreen_surface_->SetDamageRegion(region);
154}
155
158 FML_DCHECK(onscreen_surface_);
159 if (present_info.presentation_time) {
160 onscreen_surface_->SetPresentationTime(*present_info.presentation_time);
161 }
162 return onscreen_surface_->SwapBuffers(present_info.frame_damage);
163}
164
167 // The default window bound framebuffer on Android.
168 return GLFBOInfo{
169 .fbo_id = 0,
170 .existing_damage = onscreen_surface_->InitialDamage(),
171 };
172}
173
174// |GPUSurfaceGLDelegate|
175sk_sp<const GrGLInterface> AndroidSurfaceGLSkia::GetGLInterface() const {
176 // This is a workaround for a bug in the Android emulator EGL/GLES
177 // implementation. Some versions of the emulator will not update the
178 // GL version string when the process switches to a new EGL context
179 // unless the EGL context is being made current for the first time.
180 // The inaccurate version string will be rejected by Skia when it
181 // tries to build the GrGLInterface. Flutter can work around this
182 // by creating a new context, making it current to force an update
183 // of the version, and then reverting to the previous context.
184 const char* gl_renderer =
185 reinterpret_cast<const char*>(glGetString(GL_RENDERER));
186 if (gl_renderer && strncmp(gl_renderer, kEmulatorRendererPrefix,
187 strlen(kEmulatorRendererPrefix)) == 0) {
188 EGLContext new_context = android_context_->CreateNewContext();
189 if (new_context != EGL_NO_CONTEXT) {
190 EGLContext old_context = eglGetCurrentContext();
191 EGLDisplay display = eglGetCurrentDisplay();
192 EGLSurface draw_surface = eglGetCurrentSurface(EGL_DRAW);
193 EGLSurface read_surface = eglGetCurrentSurface(EGL_READ);
194 [[maybe_unused]] EGLBoolean result =
195 eglMakeCurrent(display, draw_surface, read_surface, new_context);
196 FML_DCHECK(result == EGL_TRUE);
197 result = eglMakeCurrent(display, draw_surface, read_surface, old_context);
198 FML_DCHECK(result == EGL_TRUE);
199 result = eglDestroyContext(display, new_context);
200 FML_DCHECK(result == EGL_TRUE);
201 }
202 }
203
205}
206
208 if (!onscreen_surface_ || !onscreen_surface_->IsValid()) {
209 onscreen_surface_ = android_context_->CreatePbufferSurface();
210 }
211 sk_sp<GrDirectContext> main_skia_context =
212 android_context_->GetMainSkiaContext();
213 if (!main_skia_context) {
214 main_skia_context = GPUSurfaceGLSkia::MakeGLContext(this);
215 android_context_->SetMainSkiaContext(main_skia_context);
216 }
217
218 return std::make_unique<GPUSurfaceGLSkia>(main_skia_context, this, true);
219}
220
221} // namespace flutter
bool OnScreenSurfaceResize(const DlISize &size) override
bool GLContextPresent(const GLPresentInfo &present_info) override
std::unique_ptr< Surface > CreateGPUSurface(GrDirectContext *gr_context) override
virtual std::unique_ptr< Surface > CreateSnapshotSurface() override
sk_sp< const GrGLInterface > GetGLInterface() const override
AndroidSurfaceGLSkia(const std::shared_ptr< AndroidContextGLSkia > &android_context)
void GLContextSetDamageRegion(const std::optional< DlIRect > &region) override
std::unique_ptr< GLContextResult > GLContextMakeCurrent() override
SurfaceFrame::FramebufferInfo GLContextFramebufferInfo() const override
bool SetNativeWindow(fml::RefPtr< AndroidNativeWindow > window, const std::shared_ptr< PlatformViewAndroidJNI > &jni_facade) override
GLFBOInfo GLContextFBO(GLFrameInfo frame_info) const override
virtual sk_sp< const GrGLInterface > GetGLInterface() const
static sk_sp< GrDirectContext > MakeGLContext(GPUSurfaceGLDelegate *delegate)
GLFWwindow * window
Definition main.cc:60
#define FML_LOG(severity)
Definition logging.h:101
#define FML_DCHECK(condition)
Definition logging.h:122
it will be possible to load the file into Perfetto s trace viewer use test Running tests that layout and measure text will not yield consistent results across various platforms Enabling this option will make font resolution default to the Ahem test font on all disable asset Prevents usage of any non test fonts unless they were explicitly Loaded via prefetched default font Indicates whether the embedding started a prefetch of the default font manager before creating the engine run In non interactive keep the shell running after the Dart script has completed enable serial On low power devices with low core running concurrent GC tasks on threads can cause them to contend with the UI thread which could potentially lead to jank This option turns off all concurrent GC activities domain network JSON encoded network policy per domain This overrides the DisallowInsecureConnections switch Embedder can specify whether to allow or disallow insecure connections at a domain level old gen heap size
@ kFailure
Failed to make the egl context for the surface current.
std::optional< fml::TimePoint > presentation_time
const std::optional< DlIRect > & frame_damage
std::optional< DlIRect > existing_damage