10 #include "flutter/common/constants.h"
11 #include "flutter/shell/platform/embedder/embedder.h"
17 "attribute vec2 position;\n"
18 "attribute vec2 in_texcoord;\n"
19 "uniform vec2 offset;\n"
20 "uniform vec2 scale;\n"
21 "varying vec2 texcoord;\n"
24 " gl_Position = vec4(offset + position * scale, 0, 1);\n"
25 " texcoord = in_texcoord;\n"
31 "precision mediump float;\n"
34 "uniform sampler2D texture;\n"
35 "varying vec2 texcoord;\n"
38 " gl_FragColor = texture2D(texture, texcoord);\n"
84 fl_compositor_get_type())
87 static gchar* get_shader_log(GLuint shader) {
91 glGetShaderiv(shader, GL_INFO_LOG_LENGTH, &log_length);
93 log =
static_cast<gchar*
>(g_malloc(log_length + 1));
94 glGetShaderInfoLog(shader, log_length,
nullptr, log);
104 glGetProgramiv(program, GL_INFO_LOG_LENGTH, &log_length);
106 log =
static_cast<gchar*
>(g_malloc(log_length + 1));
107 glGetProgramInfoLog(program, log_length,
nullptr, log);
115 "Failed to setup compositor shaders, unable to make OpenGL context "
120 GLuint vertex_shader = glCreateShader(GL_VERTEX_SHADER);
122 glCompileShader(vertex_shader);
123 GLint vertex_compile_status;
124 glGetShaderiv(vertex_shader, GL_COMPILE_STATUS, &vertex_compile_status);
125 if (vertex_compile_status == GL_FALSE) {
126 g_autofree gchar* shader_log = get_shader_log(vertex_shader);
127 g_warning(
"Failed to compile vertex shader: %s", shader_log);
130 GLuint fragment_shader = glCreateShader(GL_FRAGMENT_SHADER);
132 glCompileShader(fragment_shader);
133 GLint fragment_compile_status;
134 glGetShaderiv(fragment_shader, GL_COMPILE_STATUS, &fragment_compile_status);
135 if (fragment_compile_status == GL_FALSE) {
136 g_autofree gchar* shader_log = get_shader_log(fragment_shader);
137 g_warning(
"Failed to compile fragment shader: %s", shader_log);
140 self->program = glCreateProgram();
141 glAttachShader(self->program, vertex_shader);
142 glAttachShader(self->program, fragment_shader);
143 glLinkProgram(self->program);
146 glGetProgramiv(self->program, GL_LINK_STATUS, &link_status);
147 if (link_status == GL_FALSE) {
149 g_warning(
"Failed to link program: %s", program_log);
152 self->offset_location = glGetUniformLocation(self->program,
"offset");
153 self->scale_location = glGetUniformLocation(self->program,
"scale");
155 glDeleteShader(vertex_shader);
156 glDeleteShader(fragment_shader);
162 GLfloat vertex_data[] = {-1, -1, 0, 0, 1, 1, 1, 1, -1, 1, 0, 1,
163 -1, -1, 0, 0, 1, -1, 1, 0, 1, 1, 1, 1};
165 glGenBuffers(1, &self->vertex_buffer);
166 glBindBuffer(GL_ARRAY_BUFFER, self->vertex_buffer);
167 glBufferData(GL_ARRAY_BUFFER,
sizeof(vertex_data), vertex_data,
174 "Failed to cleanup compositor shaders, unable to make OpenGL context "
179 if (self->program != 0) {
180 glDeleteProgram(self->program);
182 if (self->vertex_buffer != 0) {
183 glDeleteBuffers(1, &self->vertex_buffer);
188 FlFramebuffer* framebuffer,
195 glUniform2f(self->offset_location, (2 * x /
width) - 1.0,
196 (2 * y /
width) - 1.0);
197 glUniform2f(self->scale_location, texture_width /
width,
203 glDrawArrays(GL_TRIANGLES, 0, 6);
207 const FlutterLayer**
layers,
209 FlCompositorOpenGL*
self = FL_COMPOSITOR_OPENGL(compositor);
211 g_mutex_lock(&self->frame_mutex);
213 g_mutex_unlock(&self->frame_mutex);
217 GLint general_format = GL_RGBA;
218 if (epoxy_has_gl_extension(
"GL_EXT_texture_format_BGRA8888")) {
219 general_format = GL_BGRA_EXT;
225 GLint saved_texture_binding;
226 glGetIntegerv(GL_TEXTURE_BINDING_2D, &saved_texture_binding);
227 GLint saved_vao_binding;
228 glGetIntegerv(GL_VERTEX_ARRAY_BINDING, &saved_vao_binding);
229 GLint saved_array_buffer_binding;
230 glGetIntegerv(GL_ARRAY_BUFFER_BINDING, &saved_array_buffer_binding);
231 GLint saved_draw_framebuffer_binding;
232 glGetIntegerv(GL_DRAW_FRAMEBUFFER_BINDING, &saved_draw_framebuffer_binding);
233 GLint saved_read_framebuffer_binding;
234 glGetIntegerv(GL_READ_FRAMEBUFFER_BINDING, &saved_read_framebuffer_binding);
235 GLint saved_current_program;
236 glGetIntegerv(GL_CURRENT_PROGRAM, &saved_current_program);
237 GLboolean saved_scissor_test = glIsEnabled(GL_SCISSOR_TEST);
238 GLboolean saved_blend = glIsEnabled(GL_BLEND);
240 glGetIntegerv(GL_BLEND_SRC_RGB, &saved_src_rgb);
241 GLint saved_src_alpha;
242 glGetIntegerv(GL_BLEND_SRC_ALPHA, &saved_src_alpha);
244 glGetIntegerv(GL_BLEND_DST_RGB, &saved_dst_rgb);
245 GLint saved_dst_alpha;
246 glGetIntegerv(GL_BLEND_DST_ALPHA, &saved_dst_alpha);
251 if (self->framebuffer ==
nullptr ||
254 g_clear_object(&self->framebuffer);
259 if (!self->shareable) {
261 self->pixels =
static_cast<uint8_t*
>(realloc(self->pixels, data_length));
265 self->had_first_frame =
true;
270 glGenVertexArrays(1, &vao);
271 glBindVertexArray(vao);
272 glBindBuffer(GL_ARRAY_BUFFER, self->vertex_buffer);
273 GLint position_location = glGetAttribLocation(self->program,
"position");
274 glEnableVertexAttribArray(position_location);
275 glVertexAttribPointer(position_location, 2, GL_FLOAT, GL_FALSE,
276 sizeof(GLfloat) * 4, 0);
277 GLint texcoord_location = glGetAttribLocation(self->program,
"in_texcoord");
278 glEnableVertexAttribArray(texcoord_location);
279 glVertexAttribPointer(texcoord_location, 2, GL_FLOAT, GL_FALSE,
281 reinterpret_cast<void*
>(
sizeof(GLfloat) * 2));
284 glBlendFunc(GL_SRC_ALPHA, GL_ONE_MINUS_SRC_ALPHA);
286 glUseProgram(self->program);
291 glDisable(GL_SCISSOR_TEST);
293 glBindFramebuffer(GL_DRAW_FRAMEBUFFER,
295 gboolean first_layer =
TRUE;
297 const FlutterLayer* layer =
layers[
i];
298 switch (layer->type) {
299 case kFlutterLayerContentTypeBackingStore: {
300 const FlutterBackingStore* backing_store = layer->backing_store;
301 FlFramebuffer* framebuffer =
302 FL_FRAMEBUFFER(backing_store->open_gl.framebuffer.user_data);
303 glBindFramebuffer(GL_READ_FRAMEBUFFER,
308 glBlitFramebuffer(layer->offset.x, layer->offset.y, layer->size.width,
309 layer->size.height, layer->offset.x,
310 layer->offset.y, layer->size.width,
311 layer->size.height, GL_COLOR_BUFFER_BIT,
319 case kFlutterLayerContentTypePlatformView: {
325 glBindFramebuffer(GL_READ_FRAMEBUFFER, 0);
328 glDeleteVertexArrays(1, &vao);
335 if (saved_scissor_test) {
336 glEnable(GL_SCISSOR_TEST);
338 glDisable(GL_SCISSOR_TEST);
341 glBindTexture(GL_TEXTURE_2D, saved_texture_binding);
342 glBindVertexArray(saved_vao_binding);
343 glBindBuffer(GL_ARRAY_BUFFER, saved_array_buffer_binding);
344 glBindFramebuffer(GL_DRAW_FRAMEBUFFER, saved_draw_framebuffer_binding);
345 glUseProgram(saved_current_program);
346 glBlendFuncSeparate(saved_src_rgb, saved_dst_rgb, saved_src_alpha,
349 if (!self->shareable) {
350 glBindFramebuffer(GL_READ_FRAMEBUFFER,
352 glReadPixels(0, 0,
width,
height, GL_RGBA, GL_UNSIGNED_BYTE, self->pixels);
353 glBindFramebuffer(GL_READ_FRAMEBUFFER, 0);
355 glBindFramebuffer(GL_READ_FRAMEBUFFER, saved_read_framebuffer_binding);
357 g_mutex_unlock(&self->frame_mutex);
367 FlCompositorOpenGL*
self = FL_COMPOSITOR_OPENGL(compositor);
369 g_mutex_lock(&self->frame_mutex);
370 if (self->framebuffer ==
nullptr) {
371 g_mutex_unlock(&self->frame_mutex);
376 gint scale_factor = gdk_window_get_scale_factor(
window);
377 size_t width = gdk_window_get_width(
window) * scale_factor;
378 size_t height = gdk_window_get_height(
window) * scale_factor;
381 g_mutex_unlock(&self->frame_mutex);
383 g_mutex_lock(&self->frame_mutex);
392 GLint saved_texture_binding;
393 glGetIntegerv(GL_TEXTURE_BINDING_2D, &saved_texture_binding);
398 glTexImage2D(GL_TEXTURE_2D, 0, GL_RGBA,
width,
height, 0, GL_RGBA,
399 GL_UNSIGNED_BYTE, self->pixels);
406 glBindTexture(GL_TEXTURE_2D, saved_texture_binding);
411 g_mutex_unlock(&self->frame_mutex);
417 FlCompositorOpenGL*
self = FL_COMPOSITOR_OPENGL(
object);
421 g_clear_object(&self->task_runner);
422 g_clear_object(&self->opengl_manager);
423 g_clear_object(&self->framebuffer);
424 g_clear_pointer(&self->pixels, g_free);
425 g_mutex_clear(&self->frame_mutex);
427 G_OBJECT_CLASS(fl_compositor_opengl_parent_class)->dispose(
object);
431 FL_COMPOSITOR_CLASS(klass)->present_layers =
439 g_mutex_init(&self->frame_mutex);
445 FlCompositorOpenGL*
self = FL_COMPOSITOR_OPENGL(
446 g_object_new(fl_compositor_opengl_get_type(),
nullptr));
448 self->task_runner = FL_TASK_RUNNER(g_object_ref(task_runner));
450 self->opengl_manager = FL_OPENGL_MANAGER(g_object_ref(
opengl_manager));
static void fl_compositor_opengl_dispose(GObject *object)
FlCompositorOpenGL * fl_compositor_opengl_new(FlTaskRunner *task_runner, FlOpenGLManager *opengl_manager, gboolean shareable)
static gboolean fl_compositor_opengl_present_layers(FlCompositor *compositor, const FlutterLayer **layers, size_t layers_count)
static const char * fragment_shader_src
static void fl_compositor_opengl_init(FlCompositorOpenGL *self)
static void cleanup_shader(FlCompositorOpenGL *self)
static gchar * get_program_log(GLuint program)
static void setup_shader(FlCompositorOpenGL *self)
G_DEFINE_TYPE(FlCompositorOpenGL, fl_compositor_opengl, fl_compositor_get_type()) static gchar *get_shader_log(GLuint shader)
static void fl_compositor_opengl_class_init(FlCompositorOpenGLClass *klass)
static void composite_layer(FlCompositorOpenGL *self, FlFramebuffer *framebuffer, double x, double y, int width, int height)
static const char * vertex_shader_src
static gboolean fl_compositor_opengl_render(FlCompositor *compositor, cairo_t *cr, GdkWindow *window)
G_BEGIN_DECLS FlOpenGLManager gboolean shareable
G_BEGIN_DECLS FlOpenGLManager * opengl_manager
const FlutterLayer size_t layers_count
const FlutterLayer ** layers
g_autoptr(GMutexLocker) locker
fl_task_runner_stop_wait(self->task_runner)
size_t fl_framebuffer_get_height(FlFramebuffer *self)
GLuint fl_framebuffer_get_id(FlFramebuffer *self)
gboolean fl_framebuffer_get_shareable(FlFramebuffer *self)
size_t fl_framebuffer_get_width(FlFramebuffer *self)
GLuint fl_framebuffer_get_texture_id(FlFramebuffer *self)
FlFramebuffer * fl_framebuffer_create_sibling(FlFramebuffer *self)
FlFramebuffer * fl_framebuffer_new(GLint format, size_t width, size_t height, gboolean shareable)
gboolean fl_opengl_manager_make_current(FlOpenGLManager *self)
void fl_task_runner_wait(FlTaskRunner *self)
FlCompositor parent_instance
bool blocking_main_thread
FlOpenGLManager * opengl_manager
FlFramebuffer * framebuffer
FlTaskRunner * task_runner