2012-09-06 05:54:15 +04:00
|
|
|
/*
|
|
|
|
* Copyright © 2012 Intel Corporation
|
2015-02-09 14:37:27 +03:00
|
|
|
* Copyright © 2015 Collabora, Ltd.
|
2012-09-06 05:54:15 +04:00
|
|
|
*
|
2015-06-12 01:35:43 +03:00
|
|
|
* Permission is hereby granted, free of charge, to any person obtaining
|
|
|
|
* a copy of this software and associated documentation files (the
|
|
|
|
* "Software"), to deal in the Software without restriction, including
|
|
|
|
* without limitation the rights to use, copy, modify, merge, publish,
|
|
|
|
* distribute, sublicense, and/or sell copies of the Software, and to
|
|
|
|
* permit persons to whom the Software is furnished to do so, subject to
|
|
|
|
* the following conditions:
|
2012-09-06 05:54:15 +04:00
|
|
|
*
|
2015-06-12 01:35:43 +03:00
|
|
|
* The above copyright notice and this permission notice (including the
|
|
|
|
* next paragraph) shall be included in all copies or substantial
|
|
|
|
* portions of the Software.
|
|
|
|
*
|
|
|
|
* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
|
|
|
|
* EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
|
|
|
|
* MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
|
|
|
|
* NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS
|
|
|
|
* BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN
|
|
|
|
* ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN
|
|
|
|
* CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
|
|
|
|
* SOFTWARE.
|
2012-09-06 05:54:15 +04:00
|
|
|
*/
|
|
|
|
|
2013-05-22 19:03:19 +04:00
|
|
|
#include "config.h"
|
2012-09-06 06:06:26 +04:00
|
|
|
|
2012-11-13 22:10:28 +04:00
|
|
|
#include <GLES2/gl2.h>
|
|
|
|
#include <GLES2/gl2ext.h>
|
|
|
|
|
2014-10-17 01:37:02 +04:00
|
|
|
#include <stdbool.h>
|
2012-09-06 05:59:35 +04:00
|
|
|
#include <stdlib.h>
|
2012-09-06 06:06:26 +04:00
|
|
|
#include <string.h>
|
|
|
|
#include <ctype.h>
|
2012-09-11 18:02:05 +04:00
|
|
|
#include <float.h>
|
|
|
|
#include <assert.h>
|
2012-11-08 19:20:46 +04:00
|
|
|
#include <linux/input.h>
|
2012-09-06 05:59:35 +04:00
|
|
|
|
2012-11-13 22:10:28 +04:00
|
|
|
#include "gl-renderer.h"
|
2013-09-13 06:01:21 +04:00
|
|
|
#include "vertex-clipping.h"
|
2012-11-13 22:10:28 +04:00
|
|
|
|
2015-06-16 01:37:08 +03:00
|
|
|
#include "shared/helpers.h"
|
2012-11-13 22:10:28 +04:00
|
|
|
#include "weston-egl-ext.h"
|
2012-09-06 05:54:15 +04:00
|
|
|
|
2012-11-13 22:10:29 +04:00
|
|
|
struct gl_shader {
|
2012-11-13 22:10:25 +04:00
|
|
|
GLuint program;
|
|
|
|
GLuint vertex_shader, fragment_shader;
|
|
|
|
GLint proj_uniform;
|
|
|
|
GLint tex_uniforms[3];
|
|
|
|
GLint alpha_uniform;
|
|
|
|
GLint color_uniform;
|
2013-05-17 15:00:40 +04:00
|
|
|
const char *vertex_source, *fragment_source;
|
2012-11-13 22:10:25 +04:00
|
|
|
};
|
|
|
|
|
2013-03-05 19:30:30 +04:00
|
|
|
#define BUFFER_DAMAGE_COUNT 2
|
|
|
|
|
2014-02-05 07:36:38 +04:00
|
|
|
enum gl_border_status {
|
|
|
|
BORDER_STATUS_CLEAN = 0,
|
|
|
|
BORDER_TOP_DIRTY = 1 << GL_RENDERER_BORDER_TOP,
|
|
|
|
BORDER_LEFT_DIRTY = 1 << GL_RENDERER_BORDER_LEFT,
|
|
|
|
BORDER_RIGHT_DIRTY = 1 << GL_RENDERER_BORDER_RIGHT,
|
|
|
|
BORDER_BOTTOM_DIRTY = 1 << GL_RENDERER_BORDER_BOTTOM,
|
|
|
|
BORDER_ALL_DIRTY = 0xf,
|
|
|
|
BORDER_SIZE_CHANGED = 0x10
|
|
|
|
};
|
|
|
|
|
2013-10-28 07:24:54 +04:00
|
|
|
struct gl_border_image {
|
|
|
|
GLuint tex;
|
|
|
|
int32_t width, height;
|
|
|
|
int32_t tex_width;
|
|
|
|
void *data;
|
|
|
|
};
|
|
|
|
|
2012-11-13 22:10:29 +04:00
|
|
|
struct gl_output_state {
|
2012-11-13 22:10:19 +04:00
|
|
|
EGLSurface egl_surface;
|
2013-03-05 19:30:30 +04:00
|
|
|
pixman_region32_t buffer_damage[BUFFER_DAMAGE_COUNT];
|
2014-10-10 03:39:44 +04:00
|
|
|
int buffer_damage_index;
|
2014-02-05 07:36:38 +04:00
|
|
|
enum gl_border_status border_damage[BUFFER_DAMAGE_COUNT];
|
2013-10-28 07:24:54 +04:00
|
|
|
struct gl_border_image borders[4];
|
2014-02-05 07:36:38 +04:00
|
|
|
enum gl_border_status border_status;
|
2014-10-16 19:55:21 +04:00
|
|
|
|
|
|
|
struct weston_matrix output_matrix;
|
2012-11-13 22:10:19 +04:00
|
|
|
};
|
|
|
|
|
2013-06-07 17:52:44 +04:00
|
|
|
enum buffer_type {
|
|
|
|
BUFFER_TYPE_NULL,
|
2015-02-09 14:56:56 +03:00
|
|
|
BUFFER_TYPE_SOLID, /* internal solid color surfaces without a buffer */
|
2013-06-07 17:52:44 +04:00
|
|
|
BUFFER_TYPE_SHM,
|
|
|
|
BUFFER_TYPE_EGL
|
|
|
|
};
|
|
|
|
|
2015-06-08 16:37:05 +03:00
|
|
|
struct gl_renderer;
|
|
|
|
|
|
|
|
struct egl_image {
|
|
|
|
struct gl_renderer *renderer;
|
|
|
|
EGLImageKHR image;
|
|
|
|
int refcount;
|
|
|
|
};
|
|
|
|
|
2012-11-13 22:10:29 +04:00
|
|
|
struct gl_surface_state {
|
2012-11-13 22:10:23 +04:00
|
|
|
GLfloat color[4];
|
2012-11-13 22:10:29 +04:00
|
|
|
struct gl_shader *shader;
|
2012-11-13 22:10:26 +04:00
|
|
|
|
|
|
|
GLuint textures[3];
|
|
|
|
int num_textures;
|
2013-06-07 17:52:45 +04:00
|
|
|
int needs_full_upload;
|
2012-12-04 17:58:16 +04:00
|
|
|
pixman_region32_t texture_damage;
|
2012-11-13 22:10:26 +04:00
|
|
|
|
2014-04-07 18:01:01 +04:00
|
|
|
/* These are only used by SHM surfaces to detect when we need
|
|
|
|
* to do a full upload to specify a new internal texture
|
|
|
|
* format */
|
|
|
|
GLenum gl_format;
|
|
|
|
GLenum gl_pixel_type;
|
|
|
|
|
2015-06-08 16:37:05 +03:00
|
|
|
struct egl_image* images[3];
|
2012-11-13 22:10:26 +04:00
|
|
|
GLenum target;
|
|
|
|
int num_images;
|
2012-12-04 17:58:13 +04:00
|
|
|
|
|
|
|
struct weston_buffer_reference buffer_ref;
|
2013-06-07 17:52:44 +04:00
|
|
|
enum buffer_type buffer_type;
|
2012-12-04 17:58:15 +04:00
|
|
|
int pitch; /* in pixels */
|
2013-05-22 16:41:37 +04:00
|
|
|
int height; /* in pixels */
|
2013-08-29 11:36:44 +04:00
|
|
|
int y_inverted;
|
2013-10-25 17:26:33 +04:00
|
|
|
|
|
|
|
struct weston_surface *surface;
|
|
|
|
|
|
|
|
struct wl_listener surface_destroy_listener;
|
2013-10-25 17:26:34 +04:00
|
|
|
struct wl_listener renderer_destroy_listener;
|
2012-11-13 22:10:23 +04:00
|
|
|
};
|
|
|
|
|
2012-11-13 22:10:29 +04:00
|
|
|
struct gl_renderer {
|
2012-11-08 19:20:46 +04:00
|
|
|
struct weston_renderer base;
|
|
|
|
int fragment_shader_debug;
|
2013-05-07 18:50:09 +04:00
|
|
|
int fan_debug;
|
2013-10-25 17:26:32 +04:00
|
|
|
struct weston_binding *fragment_binding;
|
|
|
|
struct weston_binding *fan_binding;
|
2012-11-08 19:20:46 +04:00
|
|
|
|
2012-11-13 22:10:19 +04:00
|
|
|
EGLDisplay egl_display;
|
|
|
|
EGLContext egl_context;
|
|
|
|
EGLConfig egl_config;
|
2012-11-13 22:10:20 +04:00
|
|
|
|
2013-05-09 06:38:05 +04:00
|
|
|
struct wl_array vertices;
|
|
|
|
struct wl_array vtxcnt;
|
|
|
|
|
2012-11-13 22:10:27 +04:00
|
|
|
PFNGLEGLIMAGETARGETTEXTURE2DOESPROC image_target_texture_2d;
|
|
|
|
PFNEGLCREATEIMAGEKHRPROC create_image;
|
|
|
|
PFNEGLDESTROYIMAGEKHRPROC destroy_image;
|
|
|
|
|
2014-02-05 07:36:39 +04:00
|
|
|
#ifdef EGL_EXT_swap_buffers_with_damage
|
|
|
|
PFNEGLSWAPBUFFERSWITHDAMAGEEXTPROC swap_buffers_with_damage;
|
|
|
|
#endif
|
|
|
|
|
2015-03-20 17:26:52 +03:00
|
|
|
PFNEGLCREATEPLATFORMWINDOWSURFACEEXTPROC create_platform_window;
|
|
|
|
|
2012-11-13 22:10:27 +04:00
|
|
|
int has_unpack_subimage;
|
|
|
|
|
|
|
|
PFNEGLBINDWAYLANDDISPLAYWL bind_display;
|
|
|
|
PFNEGLUNBINDWAYLANDDISPLAYWL unbind_display;
|
|
|
|
PFNEGLQUERYWAYLANDBUFFERWL query_buffer;
|
|
|
|
int has_bind_display;
|
|
|
|
|
|
|
|
int has_egl_image_external;
|
|
|
|
|
2013-03-05 19:30:30 +04:00
|
|
|
int has_egl_buffer_age;
|
|
|
|
|
2014-03-07 22:05:50 +04:00
|
|
|
int has_configless_context;
|
|
|
|
|
2012-11-13 22:10:29 +04:00
|
|
|
struct gl_shader texture_shader_rgba;
|
|
|
|
struct gl_shader texture_shader_rgbx;
|
|
|
|
struct gl_shader texture_shader_egl_external;
|
|
|
|
struct gl_shader texture_shader_y_uv;
|
|
|
|
struct gl_shader texture_shader_y_u_v;
|
|
|
|
struct gl_shader texture_shader_y_xuxv;
|
|
|
|
struct gl_shader invert_color_shader;
|
|
|
|
struct gl_shader solid_shader;
|
|
|
|
struct gl_shader *current_shader;
|
2013-10-25 17:26:34 +04:00
|
|
|
|
|
|
|
struct wl_signal destroy_signal;
|
2012-11-13 22:10:18 +04:00
|
|
|
};
|
|
|
|
|
2015-03-20 17:26:50 +03:00
|
|
|
static PFNEGLGETPLATFORMDISPLAYEXTPROC get_platform_display = NULL;
|
|
|
|
|
2012-11-13 22:10:29 +04:00
|
|
|
static inline struct gl_output_state *
|
2012-11-13 22:10:18 +04:00
|
|
|
get_output_state(struct weston_output *output)
|
|
|
|
{
|
2012-11-13 22:10:29 +04:00
|
|
|
return (struct gl_output_state *)output->renderer_state;
|
2012-11-13 22:10:18 +04:00
|
|
|
}
|
|
|
|
|
2013-10-25 17:26:33 +04:00
|
|
|
static int
|
|
|
|
gl_renderer_create_surface(struct weston_surface *surface);
|
|
|
|
|
2012-11-13 22:10:29 +04:00
|
|
|
static inline struct gl_surface_state *
|
2012-11-13 22:10:23 +04:00
|
|
|
get_surface_state(struct weston_surface *surface)
|
|
|
|
{
|
2013-10-25 17:26:33 +04:00
|
|
|
if (!surface->renderer_state)
|
|
|
|
gl_renderer_create_surface(surface);
|
|
|
|
|
2012-11-13 22:10:29 +04:00
|
|
|
return (struct gl_surface_state *)surface->renderer_state;
|
2012-11-13 22:10:23 +04:00
|
|
|
}
|
|
|
|
|
2012-11-13 22:10:29 +04:00
|
|
|
static inline struct gl_renderer *
|
2012-11-13 22:10:19 +04:00
|
|
|
get_renderer(struct weston_compositor *ec)
|
|
|
|
{
|
2012-11-13 22:10:29 +04:00
|
|
|
return (struct gl_renderer *)ec->renderer;
|
2012-11-13 22:10:19 +04:00
|
|
|
}
|
|
|
|
|
2015-06-08 16:37:05 +03:00
|
|
|
static struct egl_image*
|
|
|
|
egl_image_create(struct gl_renderer *gr, EGLenum target,
|
|
|
|
EGLClientBuffer buffer, const EGLint *attribs)
|
|
|
|
{
|
|
|
|
struct egl_image *img;
|
|
|
|
|
|
|
|
img = zalloc(sizeof *img);
|
|
|
|
img->renderer = gr;
|
|
|
|
img->refcount = 1;
|
|
|
|
img->image = gr->create_image(gr->egl_display, EGL_NO_CONTEXT,
|
|
|
|
target, buffer, attribs);
|
|
|
|
|
|
|
|
if (img->image == EGL_NO_IMAGE_KHR) {
|
|
|
|
free(img);
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
return img;
|
|
|
|
}
|
|
|
|
|
|
|
|
static struct egl_image*
|
|
|
|
egl_image_ref(struct egl_image *image)
|
|
|
|
{
|
|
|
|
image->refcount++;
|
|
|
|
|
|
|
|
return image;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int
|
|
|
|
egl_image_unref(struct egl_image *image)
|
|
|
|
{
|
|
|
|
struct gl_renderer *gr = image->renderer;
|
|
|
|
|
|
|
|
assert(image->refcount > 0);
|
|
|
|
|
|
|
|
image->refcount--;
|
|
|
|
if (image->refcount > 0)
|
|
|
|
return image->refcount;
|
|
|
|
|
|
|
|
gr->destroy_image(gr->egl_display, image->image);
|
|
|
|
free(image);
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2012-09-06 05:54:15 +04:00
|
|
|
static const char *
|
|
|
|
egl_error_string(EGLint code)
|
|
|
|
{
|
|
|
|
#define MYERRCODE(x) case x: return #x;
|
|
|
|
switch (code) {
|
|
|
|
MYERRCODE(EGL_SUCCESS)
|
|
|
|
MYERRCODE(EGL_NOT_INITIALIZED)
|
|
|
|
MYERRCODE(EGL_BAD_ACCESS)
|
|
|
|
MYERRCODE(EGL_BAD_ALLOC)
|
|
|
|
MYERRCODE(EGL_BAD_ATTRIBUTE)
|
|
|
|
MYERRCODE(EGL_BAD_CONTEXT)
|
|
|
|
MYERRCODE(EGL_BAD_CONFIG)
|
|
|
|
MYERRCODE(EGL_BAD_CURRENT_SURFACE)
|
|
|
|
MYERRCODE(EGL_BAD_DISPLAY)
|
|
|
|
MYERRCODE(EGL_BAD_SURFACE)
|
|
|
|
MYERRCODE(EGL_BAD_MATCH)
|
|
|
|
MYERRCODE(EGL_BAD_PARAMETER)
|
|
|
|
MYERRCODE(EGL_BAD_NATIVE_PIXMAP)
|
|
|
|
MYERRCODE(EGL_BAD_NATIVE_WINDOW)
|
|
|
|
MYERRCODE(EGL_CONTEXT_LOST)
|
|
|
|
default:
|
|
|
|
return "unknown";
|
|
|
|
}
|
|
|
|
#undef MYERRCODE
|
|
|
|
}
|
|
|
|
|
2013-10-14 16:57:11 +04:00
|
|
|
static void
|
2012-11-27 14:25:25 +04:00
|
|
|
gl_renderer_print_egl_error_state(void)
|
2012-09-06 05:54:15 +04:00
|
|
|
{
|
|
|
|
EGLint code;
|
|
|
|
|
|
|
|
code = eglGetError();
|
|
|
|
weston_log("EGL error state: %s (0x%04lx)\n",
|
|
|
|
egl_error_string(code), (long)code);
|
|
|
|
}
|
|
|
|
|
2012-09-06 05:59:35 +04:00
|
|
|
#define max(a, b) (((a) > (b)) ? (a) : (b))
|
|
|
|
#define min(a, b) (((a) > (b)) ? (b) : (a))
|
|
|
|
|
2012-09-11 18:02:05 +04:00
|
|
|
/*
|
|
|
|
* Compute the boundary vertices of the intersection of the global coordinate
|
|
|
|
* aligned rectangle 'rect', and an arbitrary quadrilateral produced from
|
|
|
|
* 'surf_rect' when transformed from surface coordinates into global coordinates.
|
|
|
|
* The vertices are written to 'ex' and 'ey', and the return value is the
|
|
|
|
* number of vertices. Vertices are produced in clockwise winding order.
|
|
|
|
* Guarantees to produce either zero vertices, or 3-8 vertices with non-zero
|
|
|
|
* polygon area.
|
|
|
|
*/
|
2012-09-06 05:59:35 +04:00
|
|
|
static int
|
Split the geometry information from weston_surface out into weston_view
The weston_surface structure is split into two structures:
* The weston_surface structure storres everything required for a
client-side or server-side surface. This includes buffers; callbacks;
backend private data; input, damage, and opaque regions; and a few other
bookkeeping bits.
* The weston_view structure represents an entity in the scenegraph and
storres all of the geometry information. This includes clip region,
alpha, position, and the transformation list as well as all of the
temporary information derived from the geometry state. Because a view,
and not a surface, is a scenegraph element, the view is what is placed
in layers and planes.
There are a few things worth noting about the surface/view split:
1. This is *not* a modification to the protocol. It is, instead, a
modification to Weston's internal scenegraph to allow a single surface
to exist in multiple places at a time. Clients are completely unaware
of how many views to a particular surface exist.
2. A view is considered a direct child of a surface and is destroyed when
the surface is destroyed. Because of this, the view.surface pointer is
always valid and non-null.
3. The compositor's surface_list is replaced with a view_list. Due to
subsurfaces, building the view list is a little more complicated than
it used to be and involves building a tree of views on the fly whenever
subsurfaces are used. However, this means that backends can remain
completely subsurface-agnostic.
4. Surfaces and views both keep track of which outputs they are on.
5. The weston_surface structure now has width and height fields. These
are populated when a new buffer is attached before surface.configure
is called. This is because there are many surface-based operations
that really require the width and height and digging through the views
didn't work well.
Signed-off-by: Jason Ekstrand <jason@jlekstrand.net>
2013-10-13 07:38:11 +04:00
|
|
|
calculate_edges(struct weston_view *ev, pixman_box32_t *rect,
|
2012-09-06 05:59:35 +04:00
|
|
|
pixman_box32_t *surf_rect, GLfloat *ex, GLfloat *ey)
|
|
|
|
{
|
2013-09-13 06:01:21 +04:00
|
|
|
|
2012-09-11 18:02:05 +04:00
|
|
|
struct clip_context ctx;
|
|
|
|
int i, n;
|
2012-09-06 05:59:35 +04:00
|
|
|
GLfloat min_x, max_x, min_y, max_y;
|
2012-09-11 18:02:05 +04:00
|
|
|
struct polygon8 surf = {
|
|
|
|
{ surf_rect->x1, surf_rect->x2, surf_rect->x2, surf_rect->x1 },
|
|
|
|
{ surf_rect->y1, surf_rect->y1, surf_rect->y2, surf_rect->y2 },
|
|
|
|
4
|
2012-09-06 05:59:35 +04:00
|
|
|
};
|
|
|
|
|
2012-09-11 18:02:05 +04:00
|
|
|
ctx.clip.x1 = rect->x1;
|
|
|
|
ctx.clip.y1 = rect->y1;
|
|
|
|
ctx.clip.x2 = rect->x2;
|
|
|
|
ctx.clip.y2 = rect->y2;
|
2012-09-06 05:59:35 +04:00
|
|
|
|
|
|
|
/* transform surface to screen space: */
|
2012-09-11 18:02:05 +04:00
|
|
|
for (i = 0; i < surf.n; i++)
|
Split the geometry information from weston_surface out into weston_view
The weston_surface structure is split into two structures:
* The weston_surface structure storres everything required for a
client-side or server-side surface. This includes buffers; callbacks;
backend private data; input, damage, and opaque regions; and a few other
bookkeeping bits.
* The weston_view structure represents an entity in the scenegraph and
storres all of the geometry information. This includes clip region,
alpha, position, and the transformation list as well as all of the
temporary information derived from the geometry state. Because a view,
and not a surface, is a scenegraph element, the view is what is placed
in layers and planes.
There are a few things worth noting about the surface/view split:
1. This is *not* a modification to the protocol. It is, instead, a
modification to Weston's internal scenegraph to allow a single surface
to exist in multiple places at a time. Clients are completely unaware
of how many views to a particular surface exist.
2. A view is considered a direct child of a surface and is destroyed when
the surface is destroyed. Because of this, the view.surface pointer is
always valid and non-null.
3. The compositor's surface_list is replaced with a view_list. Due to
subsurfaces, building the view list is a little more complicated than
it used to be and involves building a tree of views on the fly whenever
subsurfaces are used. However, this means that backends can remain
completely subsurface-agnostic.
4. Surfaces and views both keep track of which outputs they are on.
5. The weston_surface structure now has width and height fields. These
are populated when a new buffer is attached before surface.configure
is called. This is because there are many surface-based operations
that really require the width and height and digging through the views
didn't work well.
Signed-off-by: Jason Ekstrand <jason@jlekstrand.net>
2013-10-13 07:38:11 +04:00
|
|
|
weston_view_to_global_float(ev, surf.x[i], surf.y[i],
|
|
|
|
&surf.x[i], &surf.y[i]);
|
2012-09-06 05:59:35 +04:00
|
|
|
|
|
|
|
/* find bounding box: */
|
2012-09-11 18:02:05 +04:00
|
|
|
min_x = max_x = surf.x[0];
|
|
|
|
min_y = max_y = surf.y[0];
|
|
|
|
|
|
|
|
for (i = 1; i < surf.n; i++) {
|
|
|
|
min_x = min(min_x, surf.x[i]);
|
|
|
|
max_x = max(max_x, surf.x[i]);
|
|
|
|
min_y = min(min_y, surf.y[i]);
|
|
|
|
max_y = max(max_y, surf.y[i]);
|
2012-09-06 05:59:35 +04:00
|
|
|
}
|
|
|
|
|
|
|
|
/* First, simple bounding box check to discard early transformed
|
|
|
|
* surface rects that do not intersect with the clip region:
|
|
|
|
*/
|
2012-09-11 18:02:05 +04:00
|
|
|
if ((min_x >= ctx.clip.x2) || (max_x <= ctx.clip.x1) ||
|
|
|
|
(min_y >= ctx.clip.y2) || (max_y <= ctx.clip.y1))
|
2012-09-06 05:59:35 +04:00
|
|
|
return 0;
|
|
|
|
|
|
|
|
/* Simple case, bounding box edges are parallel to surface edges,
|
|
|
|
* there will be only four edges. We just need to clip the surface
|
|
|
|
* vertices to the clip rect bounds:
|
|
|
|
*/
|
Split the geometry information from weston_surface out into weston_view
The weston_surface structure is split into two structures:
* The weston_surface structure storres everything required for a
client-side or server-side surface. This includes buffers; callbacks;
backend private data; input, damage, and opaque regions; and a few other
bookkeeping bits.
* The weston_view structure represents an entity in the scenegraph and
storres all of the geometry information. This includes clip region,
alpha, position, and the transformation list as well as all of the
temporary information derived from the geometry state. Because a view,
and not a surface, is a scenegraph element, the view is what is placed
in layers and planes.
There are a few things worth noting about the surface/view split:
1. This is *not* a modification to the protocol. It is, instead, a
modification to Weston's internal scenegraph to allow a single surface
to exist in multiple places at a time. Clients are completely unaware
of how many views to a particular surface exist.
2. A view is considered a direct child of a surface and is destroyed when
the surface is destroyed. Because of this, the view.surface pointer is
always valid and non-null.
3. The compositor's surface_list is replaced with a view_list. Due to
subsurfaces, building the view list is a little more complicated than
it used to be and involves building a tree of views on the fly whenever
subsurfaces are used. However, this means that backends can remain
completely subsurface-agnostic.
4. Surfaces and views both keep track of which outputs they are on.
5. The weston_surface structure now has width and height fields. These
are populated when a new buffer is attached before surface.configure
is called. This is because there are many surface-based operations
that really require the width and height and digging through the views
didn't work well.
Signed-off-by: Jason Ekstrand <jason@jlekstrand.net>
2013-10-13 07:38:11 +04:00
|
|
|
if (!ev->transform.enabled)
|
2013-09-13 06:01:21 +04:00
|
|
|
return clip_simple(&ctx, &surf, ex, ey);
|
2012-09-06 05:59:35 +04:00
|
|
|
|
2012-09-11 18:02:05 +04:00
|
|
|
/* Transformed case: use a general polygon clipping algorithm to
|
|
|
|
* clip the surface rectangle with each side of 'rect'.
|
|
|
|
* The algorithm is Sutherland-Hodgman, as explained in
|
|
|
|
* http://www.codeguru.com/cpp/misc/misc/graphics/article.php/c8965/Polygon-Clipping.htm
|
|
|
|
* but without looking at any of that code.
|
2012-09-06 05:59:35 +04:00
|
|
|
*/
|
2013-09-13 06:01:21 +04:00
|
|
|
n = clip_transformed(&ctx, &surf, ex, ey);
|
2012-09-11 18:02:05 +04:00
|
|
|
|
|
|
|
if (n < 3)
|
|
|
|
return 0;
|
2012-09-06 05:59:35 +04:00
|
|
|
|
|
|
|
return n;
|
|
|
|
}
|
|
|
|
|
2014-10-17 01:37:02 +04:00
|
|
|
static bool
|
|
|
|
merge_down(pixman_box32_t *a, pixman_box32_t *b, pixman_box32_t *merge)
|
|
|
|
{
|
|
|
|
if (a->x1 == b->x1 && a->x2 == b->x2 && a->y1 == b->y2) {
|
|
|
|
merge->x1 = a->x1;
|
|
|
|
merge->x2 = a->x2;
|
|
|
|
merge->y1 = b->y1;
|
|
|
|
merge->y2 = a->y2;
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int
|
|
|
|
compress_bands(pixman_box32_t *inrects, int nrects,
|
|
|
|
pixman_box32_t **outrects)
|
|
|
|
{
|
|
|
|
bool merged;
|
|
|
|
pixman_box32_t *out, merge_rect;
|
|
|
|
int i, j, nout;
|
|
|
|
|
|
|
|
if (!nrects) {
|
|
|
|
*outrects = NULL;
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* nrects is an upper bound - we're not too worried about
|
|
|
|
* allocating a little extra
|
|
|
|
*/
|
|
|
|
out = malloc(sizeof(pixman_box32_t) * nrects);
|
|
|
|
out[0] = inrects[0];
|
|
|
|
nout = 1;
|
|
|
|
for (i = 1; i < nrects; i++) {
|
|
|
|
for (j = 0; j < nout; j++) {
|
|
|
|
merged = merge_down(&inrects[i], &out[j], &merge_rect);
|
|
|
|
if (merged) {
|
|
|
|
out[j] = merge_rect;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (!merged) {
|
|
|
|
out[nout] = inrects[i];
|
|
|
|
nout++;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
*outrects = out;
|
|
|
|
return nout;
|
|
|
|
}
|
|
|
|
|
2012-09-06 05:59:35 +04:00
|
|
|
static int
|
Split the geometry information from weston_surface out into weston_view
The weston_surface structure is split into two structures:
* The weston_surface structure storres everything required for a
client-side or server-side surface. This includes buffers; callbacks;
backend private data; input, damage, and opaque regions; and a few other
bookkeeping bits.
* The weston_view structure represents an entity in the scenegraph and
storres all of the geometry information. This includes clip region,
alpha, position, and the transformation list as well as all of the
temporary information derived from the geometry state. Because a view,
and not a surface, is a scenegraph element, the view is what is placed
in layers and planes.
There are a few things worth noting about the surface/view split:
1. This is *not* a modification to the protocol. It is, instead, a
modification to Weston's internal scenegraph to allow a single surface
to exist in multiple places at a time. Clients are completely unaware
of how many views to a particular surface exist.
2. A view is considered a direct child of a surface and is destroyed when
the surface is destroyed. Because of this, the view.surface pointer is
always valid and non-null.
3. The compositor's surface_list is replaced with a view_list. Due to
subsurfaces, building the view list is a little more complicated than
it used to be and involves building a tree of views on the fly whenever
subsurfaces are used. However, this means that backends can remain
completely subsurface-agnostic.
4. Surfaces and views both keep track of which outputs they are on.
5. The weston_surface structure now has width and height fields. These
are populated when a new buffer is attached before surface.configure
is called. This is because there are many surface-based operations
that really require the width and height and digging through the views
didn't work well.
Signed-off-by: Jason Ekstrand <jason@jlekstrand.net>
2013-10-13 07:38:11 +04:00
|
|
|
texture_region(struct weston_view *ev, pixman_region32_t *region,
|
2012-09-06 05:59:35 +04:00
|
|
|
pixman_region32_t *surf_region)
|
|
|
|
{
|
Split the geometry information from weston_surface out into weston_view
The weston_surface structure is split into two structures:
* The weston_surface structure storres everything required for a
client-side or server-side surface. This includes buffers; callbacks;
backend private data; input, damage, and opaque regions; and a few other
bookkeeping bits.
* The weston_view structure represents an entity in the scenegraph and
storres all of the geometry information. This includes clip region,
alpha, position, and the transformation list as well as all of the
temporary information derived from the geometry state. Because a view,
and not a surface, is a scenegraph element, the view is what is placed
in layers and planes.
There are a few things worth noting about the surface/view split:
1. This is *not* a modification to the protocol. It is, instead, a
modification to Weston's internal scenegraph to allow a single surface
to exist in multiple places at a time. Clients are completely unaware
of how many views to a particular surface exist.
2. A view is considered a direct child of a surface and is destroyed when
the surface is destroyed. Because of this, the view.surface pointer is
always valid and non-null.
3. The compositor's surface_list is replaced with a view_list. Due to
subsurfaces, building the view list is a little more complicated than
it used to be and involves building a tree of views on the fly whenever
subsurfaces are used. However, this means that backends can remain
completely subsurface-agnostic.
4. Surfaces and views both keep track of which outputs they are on.
5. The weston_surface structure now has width and height fields. These
are populated when a new buffer is attached before surface.configure
is called. This is because there are many surface-based operations
that really require the width and height and digging through the views
didn't work well.
Signed-off-by: Jason Ekstrand <jason@jlekstrand.net>
2013-10-13 07:38:11 +04:00
|
|
|
struct gl_surface_state *gs = get_surface_state(ev->surface);
|
|
|
|
struct weston_compositor *ec = ev->surface->compositor;
|
2013-05-09 06:38:05 +04:00
|
|
|
struct gl_renderer *gr = get_renderer(ec);
|
2012-09-06 05:59:35 +04:00
|
|
|
GLfloat *v, inv_width, inv_height;
|
|
|
|
unsigned int *vtxcnt, nvtx = 0;
|
|
|
|
pixman_box32_t *rects, *surf_rects;
|
2014-10-17 01:37:02 +04:00
|
|
|
pixman_box32_t *raw_rects;
|
|
|
|
int i, j, k, nrects, nsurf, raw_nrects;
|
|
|
|
bool used_band_compression;
|
|
|
|
raw_rects = pixman_region32_rectangles(region, &raw_nrects);
|
2012-09-06 05:59:35 +04:00
|
|
|
surf_rects = pixman_region32_rectangles(surf_region, &nsurf);
|
|
|
|
|
2014-10-17 01:37:02 +04:00
|
|
|
if (raw_nrects < 4) {
|
|
|
|
used_band_compression = false;
|
|
|
|
nrects = raw_nrects;
|
|
|
|
rects = raw_rects;
|
|
|
|
} else {
|
|
|
|
nrects = compress_bands(raw_rects, raw_nrects, &rects);
|
|
|
|
used_band_compression = true;
|
|
|
|
}
|
2012-09-06 05:59:35 +04:00
|
|
|
/* worst case we can have 8 vertices per rect (ie. clipped into
|
|
|
|
* an octagon):
|
|
|
|
*/
|
2013-05-09 06:38:05 +04:00
|
|
|
v = wl_array_add(&gr->vertices, nrects * nsurf * 8 * 4 * sizeof *v);
|
|
|
|
vtxcnt = wl_array_add(&gr->vtxcnt, nrects * nsurf * sizeof *vtxcnt);
|
2012-09-06 05:59:35 +04:00
|
|
|
|
2012-12-04 17:58:15 +04:00
|
|
|
inv_width = 1.0 / gs->pitch;
|
2013-05-22 16:41:37 +04:00
|
|
|
inv_height = 1.0 / gs->height;
|
2012-09-06 05:59:35 +04:00
|
|
|
|
|
|
|
for (i = 0; i < nrects; i++) {
|
|
|
|
pixman_box32_t *rect = &rects[i];
|
|
|
|
for (j = 0; j < nsurf; j++) {
|
|
|
|
pixman_box32_t *surf_rect = &surf_rects[j];
|
2012-11-28 19:10:26 +04:00
|
|
|
GLfloat sx, sy, bx, by;
|
2012-09-06 05:59:35 +04:00
|
|
|
GLfloat ex[8], ey[8]; /* edge points in screen space */
|
|
|
|
int n;
|
|
|
|
|
|
|
|
/* The transformed surface, after clipping to the clip region,
|
|
|
|
* can have as many as eight sides, emitted as a triangle-fan.
|
|
|
|
* The first vertex in the triangle fan can be chosen arbitrarily,
|
|
|
|
* since the area is guaranteed to be convex.
|
|
|
|
*
|
|
|
|
* If a corner of the transformed surface falls outside of the
|
|
|
|
* clip region, instead of emitting one vertex for the corner
|
|
|
|
* of the surface, up to two are emitted for two corresponding
|
|
|
|
* intersection point(s) between the surface and the clip region.
|
|
|
|
*
|
|
|
|
* To do this, we first calculate the (up to eight) points that
|
|
|
|
* form the intersection of the clip rect and the transformed
|
|
|
|
* surface.
|
|
|
|
*/
|
Split the geometry information from weston_surface out into weston_view
The weston_surface structure is split into two structures:
* The weston_surface structure storres everything required for a
client-side or server-side surface. This includes buffers; callbacks;
backend private data; input, damage, and opaque regions; and a few other
bookkeeping bits.
* The weston_view structure represents an entity in the scenegraph and
storres all of the geometry information. This includes clip region,
alpha, position, and the transformation list as well as all of the
temporary information derived from the geometry state. Because a view,
and not a surface, is a scenegraph element, the view is what is placed
in layers and planes.
There are a few things worth noting about the surface/view split:
1. This is *not* a modification to the protocol. It is, instead, a
modification to Weston's internal scenegraph to allow a single surface
to exist in multiple places at a time. Clients are completely unaware
of how many views to a particular surface exist.
2. A view is considered a direct child of a surface and is destroyed when
the surface is destroyed. Because of this, the view.surface pointer is
always valid and non-null.
3. The compositor's surface_list is replaced with a view_list. Due to
subsurfaces, building the view list is a little more complicated than
it used to be and involves building a tree of views on the fly whenever
subsurfaces are used. However, this means that backends can remain
completely subsurface-agnostic.
4. Surfaces and views both keep track of which outputs they are on.
5. The weston_surface structure now has width and height fields. These
are populated when a new buffer is attached before surface.configure
is called. This is because there are many surface-based operations
that really require the width and height and digging through the views
didn't work well.
Signed-off-by: Jason Ekstrand <jason@jlekstrand.net>
2013-10-13 07:38:11 +04:00
|
|
|
n = calculate_edges(ev, rect, surf_rect, ex, ey);
|
2012-09-06 05:59:35 +04:00
|
|
|
if (n < 3)
|
|
|
|
continue;
|
|
|
|
|
|
|
|
/* emit edge points: */
|
|
|
|
for (k = 0; k < n; k++) {
|
Split the geometry information from weston_surface out into weston_view
The weston_surface structure is split into two structures:
* The weston_surface structure storres everything required for a
client-side or server-side surface. This includes buffers; callbacks;
backend private data; input, damage, and opaque regions; and a few other
bookkeeping bits.
* The weston_view structure represents an entity in the scenegraph and
storres all of the geometry information. This includes clip region,
alpha, position, and the transformation list as well as all of the
temporary information derived from the geometry state. Because a view,
and not a surface, is a scenegraph element, the view is what is placed
in layers and planes.
There are a few things worth noting about the surface/view split:
1. This is *not* a modification to the protocol. It is, instead, a
modification to Weston's internal scenegraph to allow a single surface
to exist in multiple places at a time. Clients are completely unaware
of how many views to a particular surface exist.
2. A view is considered a direct child of a surface and is destroyed when
the surface is destroyed. Because of this, the view.surface pointer is
always valid and non-null.
3. The compositor's surface_list is replaced with a view_list. Due to
subsurfaces, building the view list is a little more complicated than
it used to be and involves building a tree of views on the fly whenever
subsurfaces are used. However, this means that backends can remain
completely subsurface-agnostic.
4. Surfaces and views both keep track of which outputs they are on.
5. The weston_surface structure now has width and height fields. These
are populated when a new buffer is attached before surface.configure
is called. This is because there are many surface-based operations
that really require the width and height and digging through the views
didn't work well.
Signed-off-by: Jason Ekstrand <jason@jlekstrand.net>
2013-10-13 07:38:11 +04:00
|
|
|
weston_view_from_global_float(ev, ex[k], ey[k],
|
|
|
|
&sx, &sy);
|
2012-09-06 05:59:35 +04:00
|
|
|
/* position: */
|
|
|
|
*(v++) = ex[k];
|
|
|
|
*(v++) = ey[k];
|
|
|
|
/* texcoord: */
|
Split the geometry information from weston_surface out into weston_view
The weston_surface structure is split into two structures:
* The weston_surface structure storres everything required for a
client-side or server-side surface. This includes buffers; callbacks;
backend private data; input, damage, and opaque regions; and a few other
bookkeeping bits.
* The weston_view structure represents an entity in the scenegraph and
storres all of the geometry information. This includes clip region,
alpha, position, and the transformation list as well as all of the
temporary information derived from the geometry state. Because a view,
and not a surface, is a scenegraph element, the view is what is placed
in layers and planes.
There are a few things worth noting about the surface/view split:
1. This is *not* a modification to the protocol. It is, instead, a
modification to Weston's internal scenegraph to allow a single surface
to exist in multiple places at a time. Clients are completely unaware
of how many views to a particular surface exist.
2. A view is considered a direct child of a surface and is destroyed when
the surface is destroyed. Because of this, the view.surface pointer is
always valid and non-null.
3. The compositor's surface_list is replaced with a view_list. Due to
subsurfaces, building the view list is a little more complicated than
it used to be and involves building a tree of views on the fly whenever
subsurfaces are used. However, this means that backends can remain
completely subsurface-agnostic.
4. Surfaces and views both keep track of which outputs they are on.
5. The weston_surface structure now has width and height fields. These
are populated when a new buffer is attached before surface.configure
is called. This is because there are many surface-based operations
that really require the width and height and digging through the views
didn't work well.
Signed-off-by: Jason Ekstrand <jason@jlekstrand.net>
2013-10-13 07:38:11 +04:00
|
|
|
weston_surface_to_buffer_float(ev->surface,
|
|
|
|
sx, sy,
|
2012-11-28 19:10:26 +04:00
|
|
|
&bx, &by);
|
|
|
|
*(v++) = bx * inv_width;
|
2013-08-29 11:36:44 +04:00
|
|
|
if (gs->y_inverted) {
|
|
|
|
*(v++) = by * inv_height;
|
|
|
|
} else {
|
|
|
|
*(v++) = (gs->height - by) * inv_height;
|
|
|
|
}
|
2012-09-06 05:59:35 +04:00
|
|
|
}
|
|
|
|
|
|
|
|
vtxcnt[nvtx++] = n;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-10-17 01:37:02 +04:00
|
|
|
if (used_band_compression)
|
|
|
|
free(rects);
|
2012-09-06 05:59:35 +04:00
|
|
|
return nvtx;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
Split the geometry information from weston_surface out into weston_view
The weston_surface structure is split into two structures:
* The weston_surface structure storres everything required for a
client-side or server-side surface. This includes buffers; callbacks;
backend private data; input, damage, and opaque regions; and a few other
bookkeeping bits.
* The weston_view structure represents an entity in the scenegraph and
storres all of the geometry information. This includes clip region,
alpha, position, and the transformation list as well as all of the
temporary information derived from the geometry state. Because a view,
and not a surface, is a scenegraph element, the view is what is placed
in layers and planes.
There are a few things worth noting about the surface/view split:
1. This is *not* a modification to the protocol. It is, instead, a
modification to Weston's internal scenegraph to allow a single surface
to exist in multiple places at a time. Clients are completely unaware
of how many views to a particular surface exist.
2. A view is considered a direct child of a surface and is destroyed when
the surface is destroyed. Because of this, the view.surface pointer is
always valid and non-null.
3. The compositor's surface_list is replaced with a view_list. Due to
subsurfaces, building the view list is a little more complicated than
it used to be and involves building a tree of views on the fly whenever
subsurfaces are used. However, this means that backends can remain
completely subsurface-agnostic.
4. Surfaces and views both keep track of which outputs they are on.
5. The weston_surface structure now has width and height fields. These
are populated when a new buffer is attached before surface.configure
is called. This is because there are many surface-based operations
that really require the width and height and digging through the views
didn't work well.
Signed-off-by: Jason Ekstrand <jason@jlekstrand.net>
2013-10-13 07:38:11 +04:00
|
|
|
triangle_fan_debug(struct weston_view *view, int first, int count)
|
2012-09-06 05:59:35 +04:00
|
|
|
{
|
Split the geometry information from weston_surface out into weston_view
The weston_surface structure is split into two structures:
* The weston_surface structure storres everything required for a
client-side or server-side surface. This includes buffers; callbacks;
backend private data; input, damage, and opaque regions; and a few other
bookkeeping bits.
* The weston_view structure represents an entity in the scenegraph and
storres all of the geometry information. This includes clip region,
alpha, position, and the transformation list as well as all of the
temporary information derived from the geometry state. Because a view,
and not a surface, is a scenegraph element, the view is what is placed
in layers and planes.
There are a few things worth noting about the surface/view split:
1. This is *not* a modification to the protocol. It is, instead, a
modification to Weston's internal scenegraph to allow a single surface
to exist in multiple places at a time. Clients are completely unaware
of how many views to a particular surface exist.
2. A view is considered a direct child of a surface and is destroyed when
the surface is destroyed. Because of this, the view.surface pointer is
always valid and non-null.
3. The compositor's surface_list is replaced with a view_list. Due to
subsurfaces, building the view list is a little more complicated than
it used to be and involves building a tree of views on the fly whenever
subsurfaces are used. However, this means that backends can remain
completely subsurface-agnostic.
4. Surfaces and views both keep track of which outputs they are on.
5. The weston_surface structure now has width and height fields. These
are populated when a new buffer is attached before surface.configure
is called. This is because there are many surface-based operations
that really require the width and height and digging through the views
didn't work well.
Signed-off-by: Jason Ekstrand <jason@jlekstrand.net>
2013-10-13 07:38:11 +04:00
|
|
|
struct weston_compositor *compositor = view->surface->compositor;
|
2012-11-13 22:10:29 +04:00
|
|
|
struct gl_renderer *gr = get_renderer(compositor);
|
2012-09-06 05:59:35 +04:00
|
|
|
int i;
|
|
|
|
GLushort *buffer;
|
|
|
|
GLushort *index;
|
|
|
|
int nelems;
|
|
|
|
static int color_idx = 0;
|
|
|
|
static const GLfloat color[][4] = {
|
|
|
|
{ 1.0, 0.0, 0.0, 1.0 },
|
|
|
|
{ 0.0, 1.0, 0.0, 1.0 },
|
|
|
|
{ 0.0, 0.0, 1.0, 1.0 },
|
|
|
|
{ 1.0, 1.0, 1.0, 1.0 },
|
|
|
|
};
|
|
|
|
|
|
|
|
nelems = (count - 1 + count - 2) * 2;
|
|
|
|
|
|
|
|
buffer = malloc(sizeof(GLushort) * nelems);
|
|
|
|
index = buffer;
|
|
|
|
|
|
|
|
for (i = 1; i < count; i++) {
|
|
|
|
*index++ = first;
|
|
|
|
*index++ = first + i;
|
|
|
|
}
|
|
|
|
|
|
|
|
for (i = 2; i < count; i++) {
|
|
|
|
*index++ = first + i - 1;
|
|
|
|
*index++ = first + i;
|
|
|
|
}
|
|
|
|
|
2012-11-13 22:10:25 +04:00
|
|
|
glUseProgram(gr->solid_shader.program);
|
|
|
|
glUniform4fv(gr->solid_shader.color_uniform, 1,
|
2012-09-06 05:59:35 +04:00
|
|
|
color[color_idx++ % ARRAY_LENGTH(color)]);
|
|
|
|
glDrawElements(GL_LINES, nelems, GL_UNSIGNED_SHORT, buffer);
|
2012-11-13 22:10:25 +04:00
|
|
|
glUseProgram(gr->current_shader->program);
|
2012-09-06 05:59:35 +04:00
|
|
|
free(buffer);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
Split the geometry information from weston_surface out into weston_view
The weston_surface structure is split into two structures:
* The weston_surface structure storres everything required for a
client-side or server-side surface. This includes buffers; callbacks;
backend private data; input, damage, and opaque regions; and a few other
bookkeeping bits.
* The weston_view structure represents an entity in the scenegraph and
storres all of the geometry information. This includes clip region,
alpha, position, and the transformation list as well as all of the
temporary information derived from the geometry state. Because a view,
and not a surface, is a scenegraph element, the view is what is placed
in layers and planes.
There are a few things worth noting about the surface/view split:
1. This is *not* a modification to the protocol. It is, instead, a
modification to Weston's internal scenegraph to allow a single surface
to exist in multiple places at a time. Clients are completely unaware
of how many views to a particular surface exist.
2. A view is considered a direct child of a surface and is destroyed when
the surface is destroyed. Because of this, the view.surface pointer is
always valid and non-null.
3. The compositor's surface_list is replaced with a view_list. Due to
subsurfaces, building the view list is a little more complicated than
it used to be and involves building a tree of views on the fly whenever
subsurfaces are used. However, this means that backends can remain
completely subsurface-agnostic.
4. Surfaces and views both keep track of which outputs they are on.
5. The weston_surface structure now has width and height fields. These
are populated when a new buffer is attached before surface.configure
is called. This is because there are many surface-based operations
that really require the width and height and digging through the views
didn't work well.
Signed-off-by: Jason Ekstrand <jason@jlekstrand.net>
2013-10-13 07:38:11 +04:00
|
|
|
repaint_region(struct weston_view *ev, pixman_region32_t *region,
|
2012-09-06 05:59:35 +04:00
|
|
|
pixman_region32_t *surf_region)
|
|
|
|
{
|
Split the geometry information from weston_surface out into weston_view
The weston_surface structure is split into two structures:
* The weston_surface structure storres everything required for a
client-side or server-side surface. This includes buffers; callbacks;
backend private data; input, damage, and opaque regions; and a few other
bookkeeping bits.
* The weston_view structure represents an entity in the scenegraph and
storres all of the geometry information. This includes clip region,
alpha, position, and the transformation list as well as all of the
temporary information derived from the geometry state. Because a view,
and not a surface, is a scenegraph element, the view is what is placed
in layers and planes.
There are a few things worth noting about the surface/view split:
1. This is *not* a modification to the protocol. It is, instead, a
modification to Weston's internal scenegraph to allow a single surface
to exist in multiple places at a time. Clients are completely unaware
of how many views to a particular surface exist.
2. A view is considered a direct child of a surface and is destroyed when
the surface is destroyed. Because of this, the view.surface pointer is
always valid and non-null.
3. The compositor's surface_list is replaced with a view_list. Due to
subsurfaces, building the view list is a little more complicated than
it used to be and involves building a tree of views on the fly whenever
subsurfaces are used. However, this means that backends can remain
completely subsurface-agnostic.
4. Surfaces and views both keep track of which outputs they are on.
5. The weston_surface structure now has width and height fields. These
are populated when a new buffer is attached before surface.configure
is called. This is because there are many surface-based operations
that really require the width and height and digging through the views
didn't work well.
Signed-off-by: Jason Ekstrand <jason@jlekstrand.net>
2013-10-13 07:38:11 +04:00
|
|
|
struct weston_compositor *ec = ev->surface->compositor;
|
2013-05-07 18:50:09 +04:00
|
|
|
struct gl_renderer *gr = get_renderer(ec);
|
2012-09-06 05:59:35 +04:00
|
|
|
GLfloat *v;
|
|
|
|
unsigned int *vtxcnt;
|
|
|
|
int i, first, nfans;
|
|
|
|
|
|
|
|
/* The final region to be painted is the intersection of
|
|
|
|
* 'region' and 'surf_region'. However, 'region' is in the global
|
|
|
|
* coordinates, and 'surf_region' is in the surface-local
|
|
|
|
* coordinates. texture_region() will iterate over all pairs of
|
|
|
|
* rectangles from both regions, compute the intersection
|
|
|
|
* polygon for each pair, and store it as a triangle fan if
|
2013-05-09 06:38:05 +04:00
|
|
|
* it has a non-zero area (at least 3 vertices1, actually).
|
2012-09-06 05:59:35 +04:00
|
|
|
*/
|
Split the geometry information from weston_surface out into weston_view
The weston_surface structure is split into two structures:
* The weston_surface structure storres everything required for a
client-side or server-side surface. This includes buffers; callbacks;
backend private data; input, damage, and opaque regions; and a few other
bookkeeping bits.
* The weston_view structure represents an entity in the scenegraph and
storres all of the geometry information. This includes clip region,
alpha, position, and the transformation list as well as all of the
temporary information derived from the geometry state. Because a view,
and not a surface, is a scenegraph element, the view is what is placed
in layers and planes.
There are a few things worth noting about the surface/view split:
1. This is *not* a modification to the protocol. It is, instead, a
modification to Weston's internal scenegraph to allow a single surface
to exist in multiple places at a time. Clients are completely unaware
of how many views to a particular surface exist.
2. A view is considered a direct child of a surface and is destroyed when
the surface is destroyed. Because of this, the view.surface pointer is
always valid and non-null.
3. The compositor's surface_list is replaced with a view_list. Due to
subsurfaces, building the view list is a little more complicated than
it used to be and involves building a tree of views on the fly whenever
subsurfaces are used. However, this means that backends can remain
completely subsurface-agnostic.
4. Surfaces and views both keep track of which outputs they are on.
5. The weston_surface structure now has width and height fields. These
are populated when a new buffer is attached before surface.configure
is called. This is because there are many surface-based operations
that really require the width and height and digging through the views
didn't work well.
Signed-off-by: Jason Ekstrand <jason@jlekstrand.net>
2013-10-13 07:38:11 +04:00
|
|
|
nfans = texture_region(ev, region, surf_region);
|
2012-09-06 05:59:35 +04:00
|
|
|
|
2013-05-09 06:38:05 +04:00
|
|
|
v = gr->vertices.data;
|
|
|
|
vtxcnt = gr->vtxcnt.data;
|
2012-09-06 05:59:35 +04:00
|
|
|
|
|
|
|
/* position: */
|
|
|
|
glVertexAttribPointer(0, 2, GL_FLOAT, GL_FALSE, 4 * sizeof *v, &v[0]);
|
|
|
|
glEnableVertexAttribArray(0);
|
|
|
|
|
|
|
|
/* texcoord: */
|
|
|
|
glVertexAttribPointer(1, 2, GL_FLOAT, GL_FALSE, 4 * sizeof *v, &v[2]);
|
|
|
|
glEnableVertexAttribArray(1);
|
|
|
|
|
|
|
|
for (i = 0, first = 0; i < nfans; i++) {
|
|
|
|
glDrawArrays(GL_TRIANGLE_FAN, first, vtxcnt[i]);
|
2013-05-07 18:50:09 +04:00
|
|
|
if (gr->fan_debug)
|
Split the geometry information from weston_surface out into weston_view
The weston_surface structure is split into two structures:
* The weston_surface structure storres everything required for a
client-side or server-side surface. This includes buffers; callbacks;
backend private data; input, damage, and opaque regions; and a few other
bookkeeping bits.
* The weston_view structure represents an entity in the scenegraph and
storres all of the geometry information. This includes clip region,
alpha, position, and the transformation list as well as all of the
temporary information derived from the geometry state. Because a view,
and not a surface, is a scenegraph element, the view is what is placed
in layers and planes.
There are a few things worth noting about the surface/view split:
1. This is *not* a modification to the protocol. It is, instead, a
modification to Weston's internal scenegraph to allow a single surface
to exist in multiple places at a time. Clients are completely unaware
of how many views to a particular surface exist.
2. A view is considered a direct child of a surface and is destroyed when
the surface is destroyed. Because of this, the view.surface pointer is
always valid and non-null.
3. The compositor's surface_list is replaced with a view_list. Due to
subsurfaces, building the view list is a little more complicated than
it used to be and involves building a tree of views on the fly whenever
subsurfaces are used. However, this means that backends can remain
completely subsurface-agnostic.
4. Surfaces and views both keep track of which outputs they are on.
5. The weston_surface structure now has width and height fields. These
are populated when a new buffer is attached before surface.configure
is called. This is because there are many surface-based operations
that really require the width and height and digging through the views
didn't work well.
Signed-off-by: Jason Ekstrand <jason@jlekstrand.net>
2013-10-13 07:38:11 +04:00
|
|
|
triangle_fan_debug(ev, first, vtxcnt[i]);
|
2012-09-06 05:59:35 +04:00
|
|
|
first += vtxcnt[i];
|
|
|
|
}
|
|
|
|
|
|
|
|
glDisableVertexAttribArray(1);
|
|
|
|
glDisableVertexAttribArray(0);
|
|
|
|
|
2013-05-09 06:38:05 +04:00
|
|
|
gr->vertices.size = 0;
|
|
|
|
gr->vtxcnt.size = 0;
|
2012-09-06 05:59:35 +04:00
|
|
|
}
|
|
|
|
|
2012-11-13 22:10:21 +04:00
|
|
|
static int
|
|
|
|
use_output(struct weston_output *output)
|
|
|
|
{
|
|
|
|
static int errored;
|
2012-11-13 22:10:29 +04:00
|
|
|
struct gl_output_state *go = get_output_state(output);
|
|
|
|
struct gl_renderer *gr = get_renderer(output->compositor);
|
2012-11-13 22:10:21 +04:00
|
|
|
EGLBoolean ret;
|
|
|
|
|
|
|
|
ret = eglMakeCurrent(gr->egl_display, go->egl_surface,
|
|
|
|
go->egl_surface, gr->egl_context);
|
|
|
|
|
|
|
|
if (ret == EGL_FALSE) {
|
|
|
|
if (errored)
|
|
|
|
return -1;
|
|
|
|
errored = 1;
|
|
|
|
weston_log("Failed to make EGL context current.\n");
|
2012-11-27 14:25:25 +04:00
|
|
|
gl_renderer_print_egl_error_state();
|
2012-11-13 22:10:21 +04:00
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2013-05-17 15:00:40 +04:00
|
|
|
static int
|
|
|
|
shader_init(struct gl_shader *shader, struct gl_renderer *gr,
|
|
|
|
const char *vertex_source, const char *fragment_source);
|
|
|
|
|
2012-09-06 05:59:35 +04:00
|
|
|
static void
|
2013-05-17 15:00:40 +04:00
|
|
|
use_shader(struct gl_renderer *gr, struct gl_shader *shader)
|
2012-09-06 05:59:35 +04:00
|
|
|
{
|
2013-05-17 15:00:40 +04:00
|
|
|
if (!shader->program) {
|
|
|
|
int ret;
|
|
|
|
|
|
|
|
ret = shader_init(shader, gr,
|
|
|
|
shader->vertex_source,
|
|
|
|
shader->fragment_source);
|
|
|
|
|
|
|
|
if (ret < 0)
|
|
|
|
weston_log("warning: failed to compile shader\n");
|
|
|
|
}
|
|
|
|
|
2012-11-13 22:10:25 +04:00
|
|
|
if (gr->current_shader == shader)
|
2012-09-06 05:59:35 +04:00
|
|
|
return;
|
|
|
|
glUseProgram(shader->program);
|
2012-11-13 22:10:25 +04:00
|
|
|
gr->current_shader = shader;
|
2012-09-06 05:59:35 +04:00
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
2012-11-13 22:10:29 +04:00
|
|
|
shader_uniforms(struct gl_shader *shader,
|
Split the geometry information from weston_surface out into weston_view
The weston_surface structure is split into two structures:
* The weston_surface structure storres everything required for a
client-side or server-side surface. This includes buffers; callbacks;
backend private data; input, damage, and opaque regions; and a few other
bookkeeping bits.
* The weston_view structure represents an entity in the scenegraph and
storres all of the geometry information. This includes clip region,
alpha, position, and the transformation list as well as all of the
temporary information derived from the geometry state. Because a view,
and not a surface, is a scenegraph element, the view is what is placed
in layers and planes.
There are a few things worth noting about the surface/view split:
1. This is *not* a modification to the protocol. It is, instead, a
modification to Weston's internal scenegraph to allow a single surface
to exist in multiple places at a time. Clients are completely unaware
of how many views to a particular surface exist.
2. A view is considered a direct child of a surface and is destroyed when
the surface is destroyed. Because of this, the view.surface pointer is
always valid and non-null.
3. The compositor's surface_list is replaced with a view_list. Due to
subsurfaces, building the view list is a little more complicated than
it used to be and involves building a tree of views on the fly whenever
subsurfaces are used. However, this means that backends can remain
completely subsurface-agnostic.
4. Surfaces and views both keep track of which outputs they are on.
5. The weston_surface structure now has width and height fields. These
are populated when a new buffer is attached before surface.configure
is called. This is because there are many surface-based operations
that really require the width and height and digging through the views
didn't work well.
Signed-off-by: Jason Ekstrand <jason@jlekstrand.net>
2013-10-13 07:38:11 +04:00
|
|
|
struct weston_view *view,
|
|
|
|
struct weston_output *output)
|
2012-09-06 05:59:35 +04:00
|
|
|
{
|
|
|
|
int i;
|
Split the geometry information from weston_surface out into weston_view
The weston_surface structure is split into two structures:
* The weston_surface structure storres everything required for a
client-side or server-side surface. This includes buffers; callbacks;
backend private data; input, damage, and opaque regions; and a few other
bookkeeping bits.
* The weston_view structure represents an entity in the scenegraph and
storres all of the geometry information. This includes clip region,
alpha, position, and the transformation list as well as all of the
temporary information derived from the geometry state. Because a view,
and not a surface, is a scenegraph element, the view is what is placed
in layers and planes.
There are a few things worth noting about the surface/view split:
1. This is *not* a modification to the protocol. It is, instead, a
modification to Weston's internal scenegraph to allow a single surface
to exist in multiple places at a time. Clients are completely unaware
of how many views to a particular surface exist.
2. A view is considered a direct child of a surface and is destroyed when
the surface is destroyed. Because of this, the view.surface pointer is
always valid and non-null.
3. The compositor's surface_list is replaced with a view_list. Due to
subsurfaces, building the view list is a little more complicated than
it used to be and involves building a tree of views on the fly whenever
subsurfaces are used. However, this means that backends can remain
completely subsurface-agnostic.
4. Surfaces and views both keep track of which outputs they are on.
5. The weston_surface structure now has width and height fields. These
are populated when a new buffer is attached before surface.configure
is called. This is because there are many surface-based operations
that really require the width and height and digging through the views
didn't work well.
Signed-off-by: Jason Ekstrand <jason@jlekstrand.net>
2013-10-13 07:38:11 +04:00
|
|
|
struct gl_surface_state *gs = get_surface_state(view->surface);
|
2014-10-16 19:55:21 +04:00
|
|
|
struct gl_output_state *go = get_output_state(output);
|
2012-09-06 05:59:35 +04:00
|
|
|
|
|
|
|
glUniformMatrix4fv(shader->proj_uniform,
|
2014-10-16 19:55:21 +04:00
|
|
|
1, GL_FALSE, go->output_matrix.d);
|
2012-11-13 22:10:23 +04:00
|
|
|
glUniform4fv(shader->color_uniform, 1, gs->color);
|
Split the geometry information from weston_surface out into weston_view
The weston_surface structure is split into two structures:
* The weston_surface structure storres everything required for a
client-side or server-side surface. This includes buffers; callbacks;
backend private data; input, damage, and opaque regions; and a few other
bookkeeping bits.
* The weston_view structure represents an entity in the scenegraph and
storres all of the geometry information. This includes clip region,
alpha, position, and the transformation list as well as all of the
temporary information derived from the geometry state. Because a view,
and not a surface, is a scenegraph element, the view is what is placed
in layers and planes.
There are a few things worth noting about the surface/view split:
1. This is *not* a modification to the protocol. It is, instead, a
modification to Weston's internal scenegraph to allow a single surface
to exist in multiple places at a time. Clients are completely unaware
of how many views to a particular surface exist.
2. A view is considered a direct child of a surface and is destroyed when
the surface is destroyed. Because of this, the view.surface pointer is
always valid and non-null.
3. The compositor's surface_list is replaced with a view_list. Due to
subsurfaces, building the view list is a little more complicated than
it used to be and involves building a tree of views on the fly whenever
subsurfaces are used. However, this means that backends can remain
completely subsurface-agnostic.
4. Surfaces and views both keep track of which outputs they are on.
5. The weston_surface structure now has width and height fields. These
are populated when a new buffer is attached before surface.configure
is called. This is because there are many surface-based operations
that really require the width and height and digging through the views
didn't work well.
Signed-off-by: Jason Ekstrand <jason@jlekstrand.net>
2013-10-13 07:38:11 +04:00
|
|
|
glUniform1f(shader->alpha_uniform, view->alpha);
|
2012-09-06 05:59:35 +04:00
|
|
|
|
2012-11-13 22:10:26 +04:00
|
|
|
for (i = 0; i < gs->num_textures; i++)
|
2012-09-06 05:59:35 +04:00
|
|
|
glUniform1i(shader->tex_uniforms[i], i);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
Split the geometry information from weston_surface out into weston_view
The weston_surface structure is split into two structures:
* The weston_surface structure storres everything required for a
client-side or server-side surface. This includes buffers; callbacks;
backend private data; input, damage, and opaque regions; and a few other
bookkeeping bits.
* The weston_view structure represents an entity in the scenegraph and
storres all of the geometry information. This includes clip region,
alpha, position, and the transformation list as well as all of the
temporary information derived from the geometry state. Because a view,
and not a surface, is a scenegraph element, the view is what is placed
in layers and planes.
There are a few things worth noting about the surface/view split:
1. This is *not* a modification to the protocol. It is, instead, a
modification to Weston's internal scenegraph to allow a single surface
to exist in multiple places at a time. Clients are completely unaware
of how many views to a particular surface exist.
2. A view is considered a direct child of a surface and is destroyed when
the surface is destroyed. Because of this, the view.surface pointer is
always valid and non-null.
3. The compositor's surface_list is replaced with a view_list. Due to
subsurfaces, building the view list is a little more complicated than
it used to be and involves building a tree of views on the fly whenever
subsurfaces are used. However, this means that backends can remain
completely subsurface-agnostic.
4. Surfaces and views both keep track of which outputs they are on.
5. The weston_surface structure now has width and height fields. These
are populated when a new buffer is attached before surface.configure
is called. This is because there are many surface-based operations
that really require the width and height and digging through the views
didn't work well.
Signed-off-by: Jason Ekstrand <jason@jlekstrand.net>
2013-10-13 07:38:11 +04:00
|
|
|
draw_view(struct weston_view *ev, struct weston_output *output,
|
|
|
|
pixman_region32_t *damage) /* in global coordinates */
|
2012-09-06 05:59:35 +04:00
|
|
|
{
|
Split the geometry information from weston_surface out into weston_view
The weston_surface structure is split into two structures:
* The weston_surface structure storres everything required for a
client-side or server-side surface. This includes buffers; callbacks;
backend private data; input, damage, and opaque regions; and a few other
bookkeeping bits.
* The weston_view structure represents an entity in the scenegraph and
storres all of the geometry information. This includes clip region,
alpha, position, and the transformation list as well as all of the
temporary information derived from the geometry state. Because a view,
and not a surface, is a scenegraph element, the view is what is placed
in layers and planes.
There are a few things worth noting about the surface/view split:
1. This is *not* a modification to the protocol. It is, instead, a
modification to Weston's internal scenegraph to allow a single surface
to exist in multiple places at a time. Clients are completely unaware
of how many views to a particular surface exist.
2. A view is considered a direct child of a surface and is destroyed when
the surface is destroyed. Because of this, the view.surface pointer is
always valid and non-null.
3. The compositor's surface_list is replaced with a view_list. Due to
subsurfaces, building the view list is a little more complicated than
it used to be and involves building a tree of views on the fly whenever
subsurfaces are used. However, this means that backends can remain
completely subsurface-agnostic.
4. Surfaces and views both keep track of which outputs they are on.
5. The weston_surface structure now has width and height fields. These
are populated when a new buffer is attached before surface.configure
is called. This is because there are many surface-based operations
that really require the width and height and digging through the views
didn't work well.
Signed-off-by: Jason Ekstrand <jason@jlekstrand.net>
2013-10-13 07:38:11 +04:00
|
|
|
struct weston_compositor *ec = ev->surface->compositor;
|
2012-11-13 22:10:29 +04:00
|
|
|
struct gl_renderer *gr = get_renderer(ec);
|
Split the geometry information from weston_surface out into weston_view
The weston_surface structure is split into two structures:
* The weston_surface structure storres everything required for a
client-side or server-side surface. This includes buffers; callbacks;
backend private data; input, damage, and opaque regions; and a few other
bookkeeping bits.
* The weston_view structure represents an entity in the scenegraph and
storres all of the geometry information. This includes clip region,
alpha, position, and the transformation list as well as all of the
temporary information derived from the geometry state. Because a view,
and not a surface, is a scenegraph element, the view is what is placed
in layers and planes.
There are a few things worth noting about the surface/view split:
1. This is *not* a modification to the protocol. It is, instead, a
modification to Weston's internal scenegraph to allow a single surface
to exist in multiple places at a time. Clients are completely unaware
of how many views to a particular surface exist.
2. A view is considered a direct child of a surface and is destroyed when
the surface is destroyed. Because of this, the view.surface pointer is
always valid and non-null.
3. The compositor's surface_list is replaced with a view_list. Due to
subsurfaces, building the view list is a little more complicated than
it used to be and involves building a tree of views on the fly whenever
subsurfaces are used. However, this means that backends can remain
completely subsurface-agnostic.
4. Surfaces and views both keep track of which outputs they are on.
5. The weston_surface structure now has width and height fields. These
are populated when a new buffer is attached before surface.configure
is called. This is because there are many surface-based operations
that really require the width and height and digging through the views
didn't work well.
Signed-off-by: Jason Ekstrand <jason@jlekstrand.net>
2013-10-13 07:38:11 +04:00
|
|
|
struct gl_surface_state *gs = get_surface_state(ev->surface);
|
2012-09-06 05:59:35 +04:00
|
|
|
/* repaint bounding region in global coordinates: */
|
|
|
|
pixman_region32_t repaint;
|
2015-02-18 10:48:59 +03:00
|
|
|
/* opaque region in surface coordinates: */
|
|
|
|
pixman_region32_t surface_opaque;
|
2012-09-06 05:59:35 +04:00
|
|
|
/* non-opaque region in surface coordinates: */
|
|
|
|
pixman_region32_t surface_blend;
|
|
|
|
GLint filter;
|
|
|
|
int i;
|
|
|
|
|
2013-11-19 17:22:04 +04:00
|
|
|
/* In case of a runtime switch of renderers, we may not have received
|
|
|
|
* an attach for this surface since the switch. In that case we don't
|
|
|
|
* have a valid buffer or a proper shader set up so skip rendering. */
|
|
|
|
if (!gs->shader)
|
|
|
|
return;
|
|
|
|
|
2012-09-06 05:59:35 +04:00
|
|
|
pixman_region32_init(&repaint);
|
|
|
|
pixman_region32_intersect(&repaint,
|
2015-02-19 12:15:33 +03:00
|
|
|
&ev->transform.boundingbox, damage);
|
Split the geometry information from weston_surface out into weston_view
The weston_surface structure is split into two structures:
* The weston_surface structure storres everything required for a
client-side or server-side surface. This includes buffers; callbacks;
backend private data; input, damage, and opaque regions; and a few other
bookkeeping bits.
* The weston_view structure represents an entity in the scenegraph and
storres all of the geometry information. This includes clip region,
alpha, position, and the transformation list as well as all of the
temporary information derived from the geometry state. Because a view,
and not a surface, is a scenegraph element, the view is what is placed
in layers and planes.
There are a few things worth noting about the surface/view split:
1. This is *not* a modification to the protocol. It is, instead, a
modification to Weston's internal scenegraph to allow a single surface
to exist in multiple places at a time. Clients are completely unaware
of how many views to a particular surface exist.
2. A view is considered a direct child of a surface and is destroyed when
the surface is destroyed. Because of this, the view.surface pointer is
always valid and non-null.
3. The compositor's surface_list is replaced with a view_list. Due to
subsurfaces, building the view list is a little more complicated than
it used to be and involves building a tree of views on the fly whenever
subsurfaces are used. However, this means that backends can remain
completely subsurface-agnostic.
4. Surfaces and views both keep track of which outputs they are on.
5. The weston_surface structure now has width and height fields. These
are populated when a new buffer is attached before surface.configure
is called. This is because there are many surface-based operations
that really require the width and height and digging through the views
didn't work well.
Signed-off-by: Jason Ekstrand <jason@jlekstrand.net>
2013-10-13 07:38:11 +04:00
|
|
|
pixman_region32_subtract(&repaint, &repaint, &ev->clip);
|
2012-09-06 05:59:35 +04:00
|
|
|
|
|
|
|
if (!pixman_region32_not_empty(&repaint))
|
|
|
|
goto out;
|
|
|
|
|
|
|
|
glBlendFunc(GL_ONE, GL_ONE_MINUS_SRC_ALPHA);
|
|
|
|
|
2013-05-07 18:50:09 +04:00
|
|
|
if (gr->fan_debug) {
|
2012-11-13 22:10:25 +04:00
|
|
|
use_shader(gr, &gr->solid_shader);
|
Split the geometry information from weston_surface out into weston_view
The weston_surface structure is split into two structures:
* The weston_surface structure storres everything required for a
client-side or server-side surface. This includes buffers; callbacks;
backend private data; input, damage, and opaque regions; and a few other
bookkeeping bits.
* The weston_view structure represents an entity in the scenegraph and
storres all of the geometry information. This includes clip region,
alpha, position, and the transformation list as well as all of the
temporary information derived from the geometry state. Because a view,
and not a surface, is a scenegraph element, the view is what is placed
in layers and planes.
There are a few things worth noting about the surface/view split:
1. This is *not* a modification to the protocol. It is, instead, a
modification to Weston's internal scenegraph to allow a single surface
to exist in multiple places at a time. Clients are completely unaware
of how many views to a particular surface exist.
2. A view is considered a direct child of a surface and is destroyed when
the surface is destroyed. Because of this, the view.surface pointer is
always valid and non-null.
3. The compositor's surface_list is replaced with a view_list. Due to
subsurfaces, building the view list is a little more complicated than
it used to be and involves building a tree of views on the fly whenever
subsurfaces are used. However, this means that backends can remain
completely subsurface-agnostic.
4. Surfaces and views both keep track of which outputs they are on.
5. The weston_surface structure now has width and height fields. These
are populated when a new buffer is attached before surface.configure
is called. This is because there are many surface-based operations
that really require the width and height and digging through the views
didn't work well.
Signed-off-by: Jason Ekstrand <jason@jlekstrand.net>
2013-10-13 07:38:11 +04:00
|
|
|
shader_uniforms(&gr->solid_shader, ev, output);
|
2012-09-06 05:59:35 +04:00
|
|
|
}
|
|
|
|
|
2012-11-13 22:10:25 +04:00
|
|
|
use_shader(gr, gs->shader);
|
Split the geometry information from weston_surface out into weston_view
The weston_surface structure is split into two structures:
* The weston_surface structure storres everything required for a
client-side or server-side surface. This includes buffers; callbacks;
backend private data; input, damage, and opaque regions; and a few other
bookkeeping bits.
* The weston_view structure represents an entity in the scenegraph and
storres all of the geometry information. This includes clip region,
alpha, position, and the transformation list as well as all of the
temporary information derived from the geometry state. Because a view,
and not a surface, is a scenegraph element, the view is what is placed
in layers and planes.
There are a few things worth noting about the surface/view split:
1. This is *not* a modification to the protocol. It is, instead, a
modification to Weston's internal scenegraph to allow a single surface
to exist in multiple places at a time. Clients are completely unaware
of how many views to a particular surface exist.
2. A view is considered a direct child of a surface and is destroyed when
the surface is destroyed. Because of this, the view.surface pointer is
always valid and non-null.
3. The compositor's surface_list is replaced with a view_list. Due to
subsurfaces, building the view list is a little more complicated than
it used to be and involves building a tree of views on the fly whenever
subsurfaces are used. However, this means that backends can remain
completely subsurface-agnostic.
4. Surfaces and views both keep track of which outputs they are on.
5. The weston_surface structure now has width and height fields. These
are populated when a new buffer is attached before surface.configure
is called. This is because there are many surface-based operations
that really require the width and height and digging through the views
didn't work well.
Signed-off-by: Jason Ekstrand <jason@jlekstrand.net>
2013-10-13 07:38:11 +04:00
|
|
|
shader_uniforms(gs->shader, ev, output);
|
2012-09-06 05:59:35 +04:00
|
|
|
|
Split the geometry information from weston_surface out into weston_view
The weston_surface structure is split into two structures:
* The weston_surface structure storres everything required for a
client-side or server-side surface. This includes buffers; callbacks;
backend private data; input, damage, and opaque regions; and a few other
bookkeeping bits.
* The weston_view structure represents an entity in the scenegraph and
storres all of the geometry information. This includes clip region,
alpha, position, and the transformation list as well as all of the
temporary information derived from the geometry state. Because a view,
and not a surface, is a scenegraph element, the view is what is placed
in layers and planes.
There are a few things worth noting about the surface/view split:
1. This is *not* a modification to the protocol. It is, instead, a
modification to Weston's internal scenegraph to allow a single surface
to exist in multiple places at a time. Clients are completely unaware
of how many views to a particular surface exist.
2. A view is considered a direct child of a surface and is destroyed when
the surface is destroyed. Because of this, the view.surface pointer is
always valid and non-null.
3. The compositor's surface_list is replaced with a view_list. Due to
subsurfaces, building the view list is a little more complicated than
it used to be and involves building a tree of views on the fly whenever
subsurfaces are used. However, this means that backends can remain
completely subsurface-agnostic.
4. Surfaces and views both keep track of which outputs they are on.
5. The weston_surface structure now has width and height fields. These
are populated when a new buffer is attached before surface.configure
is called. This is because there are many surface-based operations
that really require the width and height and digging through the views
didn't work well.
Signed-off-by: Jason Ekstrand <jason@jlekstrand.net>
2013-10-13 07:38:11 +04:00
|
|
|
if (ev->transform.enabled || output->zoom.active ||
|
2014-03-14 16:38:15 +04:00
|
|
|
output->current_scale != ev->surface->buffer_viewport.buffer.scale)
|
2012-09-06 05:59:35 +04:00
|
|
|
filter = GL_LINEAR;
|
|
|
|
else
|
|
|
|
filter = GL_NEAREST;
|
|
|
|
|
2012-11-13 22:10:26 +04:00
|
|
|
for (i = 0; i < gs->num_textures; i++) {
|
2012-09-06 05:59:35 +04:00
|
|
|
glActiveTexture(GL_TEXTURE0 + i);
|
2012-11-13 22:10:26 +04:00
|
|
|
glBindTexture(gs->target, gs->textures[i]);
|
|
|
|
glTexParameteri(gs->target, GL_TEXTURE_MIN_FILTER, filter);
|
|
|
|
glTexParameteri(gs->target, GL_TEXTURE_MAG_FILTER, filter);
|
2012-09-06 05:59:35 +04:00
|
|
|
}
|
|
|
|
|
|
|
|
/* blended region is whole surface minus opaque region: */
|
|
|
|
pixman_region32_init_rect(&surface_blend, 0, 0,
|
2013-12-03 07:01:53 +04:00
|
|
|
ev->surface->width, ev->surface->height);
|
2015-02-18 10:48:59 +03:00
|
|
|
if (ev->geometry.scissor_enabled)
|
|
|
|
pixman_region32_intersect(&surface_blend, &surface_blend,
|
|
|
|
&ev->geometry.scissor);
|
|
|
|
pixman_region32_subtract(&surface_blend, &surface_blend,
|
|
|
|
&ev->surface->opaque);
|
2012-09-06 05:59:35 +04:00
|
|
|
|
Split the geometry information from weston_surface out into weston_view
The weston_surface structure is split into two structures:
* The weston_surface structure storres everything required for a
client-side or server-side surface. This includes buffers; callbacks;
backend private data; input, damage, and opaque regions; and a few other
bookkeeping bits.
* The weston_view structure represents an entity in the scenegraph and
storres all of the geometry information. This includes clip region,
alpha, position, and the transformation list as well as all of the
temporary information derived from the geometry state. Because a view,
and not a surface, is a scenegraph element, the view is what is placed
in layers and planes.
There are a few things worth noting about the surface/view split:
1. This is *not* a modification to the protocol. It is, instead, a
modification to Weston's internal scenegraph to allow a single surface
to exist in multiple places at a time. Clients are completely unaware
of how many views to a particular surface exist.
2. A view is considered a direct child of a surface and is destroyed when
the surface is destroyed. Because of this, the view.surface pointer is
always valid and non-null.
3. The compositor's surface_list is replaced with a view_list. Due to
subsurfaces, building the view list is a little more complicated than
it used to be and involves building a tree of views on the fly whenever
subsurfaces are used. However, this means that backends can remain
completely subsurface-agnostic.
4. Surfaces and views both keep track of which outputs they are on.
5. The weston_surface structure now has width and height fields. These
are populated when a new buffer is attached before surface.configure
is called. This is because there are many surface-based operations
that really require the width and height and digging through the views
didn't work well.
Signed-off-by: Jason Ekstrand <jason@jlekstrand.net>
2013-10-13 07:38:11 +04:00
|
|
|
/* XXX: Should we be using ev->transform.opaque here? */
|
2015-02-18 10:48:59 +03:00
|
|
|
pixman_region32_init(&surface_opaque);
|
|
|
|
if (ev->geometry.scissor_enabled)
|
|
|
|
pixman_region32_intersect(&surface_opaque,
|
|
|
|
&ev->surface->opaque,
|
|
|
|
&ev->geometry.scissor);
|
|
|
|
else
|
|
|
|
pixman_region32_copy(&surface_opaque, &ev->surface->opaque);
|
|
|
|
|
|
|
|
if (pixman_region32_not_empty(&surface_opaque)) {
|
2012-11-13 22:10:25 +04:00
|
|
|
if (gs->shader == &gr->texture_shader_rgba) {
|
2012-09-06 05:59:35 +04:00
|
|
|
/* Special case for RGBA textures with possibly
|
|
|
|
* bad data in alpha channel: use the shader
|
|
|
|
* that forces texture alpha = 1.0.
|
|
|
|
* Xwayland surfaces need this.
|
|
|
|
*/
|
2012-11-13 22:10:25 +04:00
|
|
|
use_shader(gr, &gr->texture_shader_rgbx);
|
Split the geometry information from weston_surface out into weston_view
The weston_surface structure is split into two structures:
* The weston_surface structure storres everything required for a
client-side or server-side surface. This includes buffers; callbacks;
backend private data; input, damage, and opaque regions; and a few other
bookkeeping bits.
* The weston_view structure represents an entity in the scenegraph and
storres all of the geometry information. This includes clip region,
alpha, position, and the transformation list as well as all of the
temporary information derived from the geometry state. Because a view,
and not a surface, is a scenegraph element, the view is what is placed
in layers and planes.
There are a few things worth noting about the surface/view split:
1. This is *not* a modification to the protocol. It is, instead, a
modification to Weston's internal scenegraph to allow a single surface
to exist in multiple places at a time. Clients are completely unaware
of how many views to a particular surface exist.
2. A view is considered a direct child of a surface and is destroyed when
the surface is destroyed. Because of this, the view.surface pointer is
always valid and non-null.
3. The compositor's surface_list is replaced with a view_list. Due to
subsurfaces, building the view list is a little more complicated than
it used to be and involves building a tree of views on the fly whenever
subsurfaces are used. However, this means that backends can remain
completely subsurface-agnostic.
4. Surfaces and views both keep track of which outputs they are on.
5. The weston_surface structure now has width and height fields. These
are populated when a new buffer is attached before surface.configure
is called. This is because there are many surface-based operations
that really require the width and height and digging through the views
didn't work well.
Signed-off-by: Jason Ekstrand <jason@jlekstrand.net>
2013-10-13 07:38:11 +04:00
|
|
|
shader_uniforms(&gr->texture_shader_rgbx, ev, output);
|
2012-09-06 05:59:35 +04:00
|
|
|
}
|
|
|
|
|
Split the geometry information from weston_surface out into weston_view
The weston_surface structure is split into two structures:
* The weston_surface structure storres everything required for a
client-side or server-side surface. This includes buffers; callbacks;
backend private data; input, damage, and opaque regions; and a few other
bookkeeping bits.
* The weston_view structure represents an entity in the scenegraph and
storres all of the geometry information. This includes clip region,
alpha, position, and the transformation list as well as all of the
temporary information derived from the geometry state. Because a view,
and not a surface, is a scenegraph element, the view is what is placed
in layers and planes.
There are a few things worth noting about the surface/view split:
1. This is *not* a modification to the protocol. It is, instead, a
modification to Weston's internal scenegraph to allow a single surface
to exist in multiple places at a time. Clients are completely unaware
of how many views to a particular surface exist.
2. A view is considered a direct child of a surface and is destroyed when
the surface is destroyed. Because of this, the view.surface pointer is
always valid and non-null.
3. The compositor's surface_list is replaced with a view_list. Due to
subsurfaces, building the view list is a little more complicated than
it used to be and involves building a tree of views on the fly whenever
subsurfaces are used. However, this means that backends can remain
completely subsurface-agnostic.
4. Surfaces and views both keep track of which outputs they are on.
5. The weston_surface structure now has width and height fields. These
are populated when a new buffer is attached before surface.configure
is called. This is because there are many surface-based operations
that really require the width and height and digging through the views
didn't work well.
Signed-off-by: Jason Ekstrand <jason@jlekstrand.net>
2013-10-13 07:38:11 +04:00
|
|
|
if (ev->alpha < 1.0)
|
2012-09-06 05:59:35 +04:00
|
|
|
glEnable(GL_BLEND);
|
|
|
|
else
|
|
|
|
glDisable(GL_BLEND);
|
|
|
|
|
2015-02-18 10:48:59 +03:00
|
|
|
repaint_region(ev, &repaint, &surface_opaque);
|
2012-09-06 05:59:35 +04:00
|
|
|
}
|
|
|
|
|
|
|
|
if (pixman_region32_not_empty(&surface_blend)) {
|
2012-11-13 22:10:25 +04:00
|
|
|
use_shader(gr, gs->shader);
|
2012-09-06 05:59:35 +04:00
|
|
|
glEnable(GL_BLEND);
|
Split the geometry information from weston_surface out into weston_view
The weston_surface structure is split into two structures:
* The weston_surface structure storres everything required for a
client-side or server-side surface. This includes buffers; callbacks;
backend private data; input, damage, and opaque regions; and a few other
bookkeeping bits.
* The weston_view structure represents an entity in the scenegraph and
storres all of the geometry information. This includes clip region,
alpha, position, and the transformation list as well as all of the
temporary information derived from the geometry state. Because a view,
and not a surface, is a scenegraph element, the view is what is placed
in layers and planes.
There are a few things worth noting about the surface/view split:
1. This is *not* a modification to the protocol. It is, instead, a
modification to Weston's internal scenegraph to allow a single surface
to exist in multiple places at a time. Clients are completely unaware
of how many views to a particular surface exist.
2. A view is considered a direct child of a surface and is destroyed when
the surface is destroyed. Because of this, the view.surface pointer is
always valid and non-null.
3. The compositor's surface_list is replaced with a view_list. Due to
subsurfaces, building the view list is a little more complicated than
it used to be and involves building a tree of views on the fly whenever
subsurfaces are used. However, this means that backends can remain
completely subsurface-agnostic.
4. Surfaces and views both keep track of which outputs they are on.
5. The weston_surface structure now has width and height fields. These
are populated when a new buffer is attached before surface.configure
is called. This is because there are many surface-based operations
that really require the width and height and digging through the views
didn't work well.
Signed-off-by: Jason Ekstrand <jason@jlekstrand.net>
2013-10-13 07:38:11 +04:00
|
|
|
repaint_region(ev, &repaint, &surface_blend);
|
2012-09-06 05:59:35 +04:00
|
|
|
}
|
|
|
|
|
|
|
|
pixman_region32_fini(&surface_blend);
|
2015-02-18 10:48:59 +03:00
|
|
|
pixman_region32_fini(&surface_opaque);
|
2012-09-06 05:59:35 +04:00
|
|
|
|
|
|
|
out:
|
|
|
|
pixman_region32_fini(&repaint);
|
|
|
|
}
|
|
|
|
|
2012-09-06 05:54:15 +04:00
|
|
|
static void
|
Split the geometry information from weston_surface out into weston_view
The weston_surface structure is split into two structures:
* The weston_surface structure storres everything required for a
client-side or server-side surface. This includes buffers; callbacks;
backend private data; input, damage, and opaque regions; and a few other
bookkeeping bits.
* The weston_view structure represents an entity in the scenegraph and
storres all of the geometry information. This includes clip region,
alpha, position, and the transformation list as well as all of the
temporary information derived from the geometry state. Because a view,
and not a surface, is a scenegraph element, the view is what is placed
in layers and planes.
There are a few things worth noting about the surface/view split:
1. This is *not* a modification to the protocol. It is, instead, a
modification to Weston's internal scenegraph to allow a single surface
to exist in multiple places at a time. Clients are completely unaware
of how many views to a particular surface exist.
2. A view is considered a direct child of a surface and is destroyed when
the surface is destroyed. Because of this, the view.surface pointer is
always valid and non-null.
3. The compositor's surface_list is replaced with a view_list. Due to
subsurfaces, building the view list is a little more complicated than
it used to be and involves building a tree of views on the fly whenever
subsurfaces are used. However, this means that backends can remain
completely subsurface-agnostic.
4. Surfaces and views both keep track of which outputs they are on.
5. The weston_surface structure now has width and height fields. These
are populated when a new buffer is attached before surface.configure
is called. This is because there are many surface-based operations
that really require the width and height and digging through the views
didn't work well.
Signed-off-by: Jason Ekstrand <jason@jlekstrand.net>
2013-10-13 07:38:11 +04:00
|
|
|
repaint_views(struct weston_output *output, pixman_region32_t *damage)
|
2012-09-06 05:54:15 +04:00
|
|
|
{
|
|
|
|
struct weston_compositor *compositor = output->compositor;
|
Split the geometry information from weston_surface out into weston_view
The weston_surface structure is split into two structures:
* The weston_surface structure storres everything required for a
client-side or server-side surface. This includes buffers; callbacks;
backend private data; input, damage, and opaque regions; and a few other
bookkeeping bits.
* The weston_view structure represents an entity in the scenegraph and
storres all of the geometry information. This includes clip region,
alpha, position, and the transformation list as well as all of the
temporary information derived from the geometry state. Because a view,
and not a surface, is a scenegraph element, the view is what is placed
in layers and planes.
There are a few things worth noting about the surface/view split:
1. This is *not* a modification to the protocol. It is, instead, a
modification to Weston's internal scenegraph to allow a single surface
to exist in multiple places at a time. Clients are completely unaware
of how many views to a particular surface exist.
2. A view is considered a direct child of a surface and is destroyed when
the surface is destroyed. Because of this, the view.surface pointer is
always valid and non-null.
3. The compositor's surface_list is replaced with a view_list. Due to
subsurfaces, building the view list is a little more complicated than
it used to be and involves building a tree of views on the fly whenever
subsurfaces are used. However, this means that backends can remain
completely subsurface-agnostic.
4. Surfaces and views both keep track of which outputs they are on.
5. The weston_surface structure now has width and height fields. These
are populated when a new buffer is attached before surface.configure
is called. This is because there are many surface-based operations
that really require the width and height and digging through the views
didn't work well.
Signed-off-by: Jason Ekstrand <jason@jlekstrand.net>
2013-10-13 07:38:11 +04:00
|
|
|
struct weston_view *view;
|
2012-09-06 05:54:15 +04:00
|
|
|
|
Split the geometry information from weston_surface out into weston_view
The weston_surface structure is split into two structures:
* The weston_surface structure storres everything required for a
client-side or server-side surface. This includes buffers; callbacks;
backend private data; input, damage, and opaque regions; and a few other
bookkeeping bits.
* The weston_view structure represents an entity in the scenegraph and
storres all of the geometry information. This includes clip region,
alpha, position, and the transformation list as well as all of the
temporary information derived from the geometry state. Because a view,
and not a surface, is a scenegraph element, the view is what is placed
in layers and planes.
There are a few things worth noting about the surface/view split:
1. This is *not* a modification to the protocol. It is, instead, a
modification to Weston's internal scenegraph to allow a single surface
to exist in multiple places at a time. Clients are completely unaware
of how many views to a particular surface exist.
2. A view is considered a direct child of a surface and is destroyed when
the surface is destroyed. Because of this, the view.surface pointer is
always valid and non-null.
3. The compositor's surface_list is replaced with a view_list. Due to
subsurfaces, building the view list is a little more complicated than
it used to be and involves building a tree of views on the fly whenever
subsurfaces are used. However, this means that backends can remain
completely subsurface-agnostic.
4. Surfaces and views both keep track of which outputs they are on.
5. The weston_surface structure now has width and height fields. These
are populated when a new buffer is attached before surface.configure
is called. This is because there are many surface-based operations
that really require the width and height and digging through the views
didn't work well.
Signed-off-by: Jason Ekstrand <jason@jlekstrand.net>
2013-10-13 07:38:11 +04:00
|
|
|
wl_list_for_each_reverse(view, &compositor->view_list, link)
|
|
|
|
if (view->plane == &compositor->primary_plane)
|
|
|
|
draw_view(view, output, damage);
|
2012-09-06 05:54:15 +04:00
|
|
|
}
|
|
|
|
|
2013-10-28 07:24:54 +04:00
|
|
|
static void
|
2014-02-05 07:36:38 +04:00
|
|
|
draw_output_border_texture(struct gl_output_state *go,
|
|
|
|
enum gl_renderer_border_side side,
|
|
|
|
int32_t x, int32_t y,
|
2013-10-28 07:24:54 +04:00
|
|
|
int32_t width, int32_t height)
|
|
|
|
{
|
2014-02-05 07:36:38 +04:00
|
|
|
struct gl_border_image *img = &go->borders[side];
|
2013-10-28 07:24:54 +04:00
|
|
|
static GLushort indices [] = { 0, 1, 3, 3, 1, 2 };
|
|
|
|
|
|
|
|
if (!img->data) {
|
|
|
|
if (img->tex) {
|
|
|
|
glDeleteTextures(1, &img->tex);
|
|
|
|
img->tex = 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!img->tex) {
|
|
|
|
glGenTextures(1, &img->tex);
|
|
|
|
glBindTexture(GL_TEXTURE_2D, img->tex);
|
|
|
|
|
|
|
|
glTexParameteri(GL_TEXTURE_2D,
|
|
|
|
GL_TEXTURE_WRAP_S, GL_CLAMP_TO_EDGE);
|
|
|
|
glTexParameteri(GL_TEXTURE_2D,
|
|
|
|
GL_TEXTURE_WRAP_T, GL_CLAMP_TO_EDGE);
|
|
|
|
glTexParameteri(GL_TEXTURE_2D,
|
|
|
|
GL_TEXTURE_MIN_FILTER, GL_NEAREST);
|
|
|
|
glTexParameteri(GL_TEXTURE_2D,
|
|
|
|
GL_TEXTURE_MAG_FILTER, GL_NEAREST);
|
|
|
|
} else {
|
|
|
|
glBindTexture(GL_TEXTURE_2D, img->tex);
|
|
|
|
}
|
|
|
|
|
2014-02-05 07:36:38 +04:00
|
|
|
if (go->border_status & (1 << side)) {
|
2013-10-28 07:24:54 +04:00
|
|
|
#ifdef GL_EXT_unpack_subimage
|
|
|
|
glPixelStorei(GL_UNPACK_ROW_LENGTH_EXT, 0);
|
|
|
|
glPixelStorei(GL_UNPACK_SKIP_PIXELS_EXT, 0);
|
|
|
|
glPixelStorei(GL_UNPACK_SKIP_ROWS_EXT, 0);
|
|
|
|
#endif
|
|
|
|
glTexImage2D(GL_TEXTURE_2D, 0, GL_BGRA_EXT,
|
|
|
|
img->tex_width, img->height, 0,
|
|
|
|
GL_BGRA_EXT, GL_UNSIGNED_BYTE, img->data);
|
|
|
|
}
|
|
|
|
|
|
|
|
GLfloat texcoord[] = {
|
|
|
|
0.0f, 0.0f,
|
|
|
|
(GLfloat)img->width / (GLfloat)img->tex_width, 0.0f,
|
|
|
|
(GLfloat)img->width / (GLfloat)img->tex_width, 1.0f,
|
|
|
|
0.0f, 1.0f,
|
|
|
|
};
|
|
|
|
|
|
|
|
GLfloat verts[] = {
|
|
|
|
x, y,
|
|
|
|
x + width, y,
|
|
|
|
x + width, y + height,
|
|
|
|
x, y + height
|
|
|
|
};
|
|
|
|
|
|
|
|
glVertexAttribPointer(0, 2, GL_FLOAT, GL_FALSE, 0, verts);
|
|
|
|
glVertexAttribPointer(1, 2, GL_FLOAT, GL_FALSE, 0, texcoord);
|
|
|
|
glEnableVertexAttribArray(0);
|
|
|
|
glEnableVertexAttribArray(1);
|
|
|
|
|
|
|
|
glDrawElements(GL_TRIANGLES, 6, GL_UNSIGNED_SHORT, indices);
|
|
|
|
|
|
|
|
glDisableVertexAttribArray(1);
|
|
|
|
glDisableVertexAttribArray(0);
|
|
|
|
}
|
|
|
|
|
2014-02-05 07:36:39 +04:00
|
|
|
static int
|
|
|
|
output_has_borders(struct weston_output *output)
|
|
|
|
{
|
|
|
|
struct gl_output_state *go = get_output_state(output);
|
|
|
|
|
|
|
|
return go->borders[GL_RENDERER_BORDER_TOP].data ||
|
|
|
|
go->borders[GL_RENDERER_BORDER_RIGHT].data ||
|
|
|
|
go->borders[GL_RENDERER_BORDER_BOTTOM].data ||
|
|
|
|
go->borders[GL_RENDERER_BORDER_LEFT].data;
|
|
|
|
}
|
|
|
|
|
2013-10-28 07:24:54 +04:00
|
|
|
static void
|
2014-02-05 07:36:38 +04:00
|
|
|
draw_output_borders(struct weston_output *output,
|
|
|
|
enum gl_border_status border_status)
|
2013-10-28 07:24:54 +04:00
|
|
|
{
|
|
|
|
struct gl_output_state *go = get_output_state(output);
|
|
|
|
struct gl_renderer *gr = get_renderer(output->compositor);
|
|
|
|
struct gl_shader *shader = &gr->texture_shader_rgba;
|
2013-10-28 07:24:59 +04:00
|
|
|
struct gl_border_image *top, *bottom, *left, *right;
|
|
|
|
struct weston_matrix matrix;
|
|
|
|
int full_width, full_height;
|
|
|
|
|
2014-02-05 07:36:38 +04:00
|
|
|
if (border_status == BORDER_STATUS_CLEAN)
|
|
|
|
return; /* Clean. Nothing to do. */
|
|
|
|
|
2013-10-28 07:24:59 +04:00
|
|
|
top = &go->borders[GL_RENDERER_BORDER_TOP];
|
|
|
|
bottom = &go->borders[GL_RENDERER_BORDER_BOTTOM];
|
|
|
|
left = &go->borders[GL_RENDERER_BORDER_LEFT];
|
|
|
|
right = &go->borders[GL_RENDERER_BORDER_RIGHT];
|
|
|
|
|
|
|
|
full_width = output->current_mode->width + left->width + right->width;
|
|
|
|
full_height = output->current_mode->height + top->height + bottom->height;
|
2013-10-28 07:24:54 +04:00
|
|
|
|
|
|
|
glDisable(GL_BLEND);
|
|
|
|
use_shader(gr, shader);
|
|
|
|
|
2013-10-28 07:24:59 +04:00
|
|
|
glViewport(0, 0, full_width, full_height);
|
|
|
|
|
|
|
|
weston_matrix_init(&matrix);
|
|
|
|
weston_matrix_translate(&matrix, -full_width/2.0, -full_height/2.0, 0);
|
|
|
|
weston_matrix_scale(&matrix, 2.0/full_width, -2.0/full_height, 1);
|
|
|
|
glUniformMatrix4fv(shader->proj_uniform, 1, GL_FALSE, matrix.d);
|
2013-10-28 07:24:54 +04:00
|
|
|
|
|
|
|
glUniform1i(shader->tex_uniforms[0], 0);
|
|
|
|
glUniform1f(shader->alpha_uniform, 1);
|
|
|
|
glActiveTexture(GL_TEXTURE0);
|
|
|
|
|
2014-02-05 07:36:38 +04:00
|
|
|
if (border_status & BORDER_TOP_DIRTY)
|
|
|
|
draw_output_border_texture(go, GL_RENDERER_BORDER_TOP,
|
|
|
|
0, 0,
|
|
|
|
full_width, top->height);
|
|
|
|
if (border_status & BORDER_LEFT_DIRTY)
|
|
|
|
draw_output_border_texture(go, GL_RENDERER_BORDER_LEFT,
|
|
|
|
0, top->height,
|
|
|
|
left->width, output->current_mode->height);
|
|
|
|
if (border_status & BORDER_RIGHT_DIRTY)
|
|
|
|
draw_output_border_texture(go, GL_RENDERER_BORDER_RIGHT,
|
|
|
|
full_width - right->width, top->height,
|
|
|
|
right->width, output->current_mode->height);
|
|
|
|
if (border_status & BORDER_BOTTOM_DIRTY)
|
|
|
|
draw_output_border_texture(go, GL_RENDERER_BORDER_BOTTOM,
|
|
|
|
0, full_height - bottom->height,
|
|
|
|
full_width, bottom->height);
|
2013-10-28 07:24:54 +04:00
|
|
|
}
|
2012-11-13 22:10:20 +04:00
|
|
|
|
2014-02-05 07:36:39 +04:00
|
|
|
static void
|
|
|
|
output_get_border_damage(struct weston_output *output,
|
|
|
|
enum gl_border_status border_status,
|
|
|
|
pixman_region32_t *damage)
|
|
|
|
{
|
|
|
|
struct gl_output_state *go = get_output_state(output);
|
|
|
|
struct gl_border_image *top, *bottom, *left, *right;
|
|
|
|
int full_width, full_height;
|
|
|
|
|
|
|
|
if (border_status == BORDER_STATUS_CLEAN)
|
|
|
|
return; /* Clean. Nothing to do. */
|
|
|
|
|
|
|
|
top = &go->borders[GL_RENDERER_BORDER_TOP];
|
|
|
|
bottom = &go->borders[GL_RENDERER_BORDER_BOTTOM];
|
|
|
|
left = &go->borders[GL_RENDERER_BORDER_LEFT];
|
|
|
|
right = &go->borders[GL_RENDERER_BORDER_RIGHT];
|
|
|
|
|
|
|
|
full_width = output->current_mode->width + left->width + right->width;
|
|
|
|
full_height = output->current_mode->height + top->height + bottom->height;
|
|
|
|
if (border_status & BORDER_TOP_DIRTY)
|
|
|
|
pixman_region32_union_rect(damage, damage,
|
|
|
|
0, 0,
|
|
|
|
full_width, top->height);
|
|
|
|
if (border_status & BORDER_LEFT_DIRTY)
|
|
|
|
pixman_region32_union_rect(damage, damage,
|
|
|
|
0, top->height,
|
|
|
|
left->width, output->current_mode->height);
|
|
|
|
if (border_status & BORDER_RIGHT_DIRTY)
|
|
|
|
pixman_region32_union_rect(damage, damage,
|
|
|
|
full_width - right->width, top->height,
|
|
|
|
right->width, output->current_mode->height);
|
|
|
|
if (border_status & BORDER_BOTTOM_DIRTY)
|
|
|
|
pixman_region32_union_rect(damage, damage,
|
|
|
|
0, full_height - bottom->height,
|
|
|
|
full_width, bottom->height);
|
|
|
|
}
|
|
|
|
|
2013-03-05 19:30:30 +04:00
|
|
|
static void
|
2014-02-05 07:36:38 +04:00
|
|
|
output_get_damage(struct weston_output *output,
|
|
|
|
pixman_region32_t *buffer_damage, uint32_t *border_damage)
|
2013-03-05 19:30:30 +04:00
|
|
|
{
|
|
|
|
struct gl_output_state *go = get_output_state(output);
|
|
|
|
struct gl_renderer *gr = get_renderer(output->compositor);
|
|
|
|
EGLint buffer_age = 0;
|
|
|
|
EGLBoolean ret;
|
|
|
|
int i;
|
|
|
|
|
|
|
|
if (gr->has_egl_buffer_age) {
|
|
|
|
ret = eglQuerySurface(gr->egl_display, go->egl_surface,
|
|
|
|
EGL_BUFFER_AGE_EXT, &buffer_age);
|
|
|
|
if (ret == EGL_FALSE) {
|
|
|
|
weston_log("buffer age query failed.\n");
|
|
|
|
gl_renderer_print_egl_error_state();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-02-05 07:36:38 +04:00
|
|
|
if (buffer_age == 0 || buffer_age - 1 > BUFFER_DAMAGE_COUNT) {
|
2013-03-05 19:30:30 +04:00
|
|
|
pixman_region32_copy(buffer_damage, &output->region);
|
2014-02-05 07:36:38 +04:00
|
|
|
*border_damage = BORDER_ALL_DIRTY;
|
|
|
|
} else {
|
2013-03-05 19:30:30 +04:00
|
|
|
for (i = 0; i < buffer_age - 1; i++)
|
2014-10-10 03:39:44 +04:00
|
|
|
*border_damage |= go->border_damage[(go->buffer_damage_index + i) % BUFFER_DAMAGE_COUNT];
|
2014-02-05 07:36:38 +04:00
|
|
|
|
|
|
|
if (*border_damage & BORDER_SIZE_CHANGED) {
|
|
|
|
/* If we've had a resize, we have to do a full
|
|
|
|
* repaint. */
|
|
|
|
*border_damage |= BORDER_ALL_DIRTY;
|
|
|
|
pixman_region32_copy(buffer_damage, &output->region);
|
|
|
|
} else {
|
|
|
|
for (i = 0; i < buffer_age - 1; i++)
|
|
|
|
pixman_region32_union(buffer_damage,
|
|
|
|
buffer_damage,
|
2014-10-10 03:39:44 +04:00
|
|
|
&go->buffer_damage[(go->buffer_damage_index + i) % BUFFER_DAMAGE_COUNT]);
|
2014-02-05 07:36:38 +04:00
|
|
|
}
|
|
|
|
}
|
2013-03-05 19:30:30 +04:00
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
output_rotate_damage(struct weston_output *output,
|
2014-02-05 07:36:38 +04:00
|
|
|
pixman_region32_t *output_damage,
|
|
|
|
enum gl_border_status border_status)
|
2013-03-05 19:30:30 +04:00
|
|
|
{
|
|
|
|
struct gl_output_state *go = get_output_state(output);
|
|
|
|
struct gl_renderer *gr = get_renderer(output->compositor);
|
|
|
|
|
|
|
|
if (!gr->has_egl_buffer_age)
|
|
|
|
return;
|
|
|
|
|
2014-10-10 03:39:44 +04:00
|
|
|
go->buffer_damage_index += BUFFER_DAMAGE_COUNT - 1;
|
|
|
|
go->buffer_damage_index %= BUFFER_DAMAGE_COUNT;
|
2013-03-05 19:30:30 +04:00
|
|
|
|
2014-10-10 03:39:44 +04:00
|
|
|
pixman_region32_copy(&go->buffer_damage[go->buffer_damage_index], output_damage);
|
|
|
|
go->border_damage[go->buffer_damage_index] = border_status;
|
2013-03-05 19:30:30 +04:00
|
|
|
}
|
|
|
|
|
2015-05-15 20:12:40 +03:00
|
|
|
/* NOTE: We now allow falling back to ARGB gl visuals when XRGB is
|
|
|
|
* unavailable, so we're assuming the background has no transparency
|
|
|
|
* and that everything with a blend, like drop shadows, will have something
|
|
|
|
* opaque (like the background) drawn underneath it.
|
|
|
|
*
|
|
|
|
* Depending on the underlying hardware, violating that assumption could
|
|
|
|
* result in seeing through to another display plane.
|
|
|
|
*/
|
2012-09-06 06:49:55 +04:00
|
|
|
static void
|
2012-11-13 22:10:29 +04:00
|
|
|
gl_renderer_repaint_output(struct weston_output *output,
|
2012-09-06 05:54:15 +04:00
|
|
|
pixman_region32_t *output_damage)
|
|
|
|
{
|
2012-11-13 22:10:29 +04:00
|
|
|
struct gl_output_state *go = get_output_state(output);
|
2012-09-06 05:54:15 +04:00
|
|
|
struct weston_compositor *compositor = output->compositor;
|
2012-11-13 22:10:29 +04:00
|
|
|
struct gl_renderer *gr = get_renderer(compositor);
|
2012-09-06 05:54:15 +04:00
|
|
|
EGLBoolean ret;
|
|
|
|
static int errored;
|
2014-02-05 07:36:39 +04:00
|
|
|
#ifdef EGL_EXT_swap_buffers_with_damage
|
|
|
|
int i, nrects, buffer_height;
|
|
|
|
EGLint *egl_damage, *d;
|
|
|
|
pixman_box32_t *rects;
|
|
|
|
#endif
|
2013-03-05 19:30:30 +04:00
|
|
|
pixman_region32_t buffer_damage, total_damage;
|
2014-02-05 07:36:38 +04:00
|
|
|
enum gl_border_status border_damage = BORDER_STATUS_CLEAN;
|
2012-09-06 05:54:15 +04:00
|
|
|
|
2014-10-16 19:55:20 +04:00
|
|
|
if (use_output(output) < 0)
|
|
|
|
return;
|
|
|
|
|
2013-10-28 07:24:59 +04:00
|
|
|
/* Calculate the viewport */
|
|
|
|
glViewport(go->borders[GL_RENDERER_BORDER_LEFT].width,
|
|
|
|
go->borders[GL_RENDERER_BORDER_BOTTOM].height,
|
|
|
|
output->current_mode->width,
|
|
|
|
output->current_mode->height);
|
2012-09-06 05:54:15 +04:00
|
|
|
|
2014-10-16 19:55:21 +04:00
|
|
|
/* Calculate the global GL matrix */
|
|
|
|
go->output_matrix = output->matrix;
|
|
|
|
weston_matrix_translate(&go->output_matrix,
|
|
|
|
-(output->current_mode->width / 2.0),
|
|
|
|
-(output->current_mode->height / 2.0), 0);
|
|
|
|
weston_matrix_scale(&go->output_matrix,
|
|
|
|
2.0 / output->current_mode->width,
|
|
|
|
-2.0 / output->current_mode->height, 1);
|
|
|
|
|
2012-09-06 05:54:15 +04:00
|
|
|
/* if debugging, redraw everything outside the damage to clean up
|
|
|
|
* debug lines from the previous draw on this buffer:
|
|
|
|
*/
|
2013-05-07 18:50:09 +04:00
|
|
|
if (gr->fan_debug) {
|
2012-09-06 05:54:15 +04:00
|
|
|
pixman_region32_t undamaged;
|
|
|
|
pixman_region32_init(&undamaged);
|
|
|
|
pixman_region32_subtract(&undamaged, &output->region,
|
|
|
|
output_damage);
|
2013-05-07 18:50:09 +04:00
|
|
|
gr->fan_debug = 0;
|
Split the geometry information from weston_surface out into weston_view
The weston_surface structure is split into two structures:
* The weston_surface structure storres everything required for a
client-side or server-side surface. This includes buffers; callbacks;
backend private data; input, damage, and opaque regions; and a few other
bookkeeping bits.
* The weston_view structure represents an entity in the scenegraph and
storres all of the geometry information. This includes clip region,
alpha, position, and the transformation list as well as all of the
temporary information derived from the geometry state. Because a view,
and not a surface, is a scenegraph element, the view is what is placed
in layers and planes.
There are a few things worth noting about the surface/view split:
1. This is *not* a modification to the protocol. It is, instead, a
modification to Weston's internal scenegraph to allow a single surface
to exist in multiple places at a time. Clients are completely unaware
of how many views to a particular surface exist.
2. A view is considered a direct child of a surface and is destroyed when
the surface is destroyed. Because of this, the view.surface pointer is
always valid and non-null.
3. The compositor's surface_list is replaced with a view_list. Due to
subsurfaces, building the view list is a little more complicated than
it used to be and involves building a tree of views on the fly whenever
subsurfaces are used. However, this means that backends can remain
completely subsurface-agnostic.
4. Surfaces and views both keep track of which outputs they are on.
5. The weston_surface structure now has width and height fields. These
are populated when a new buffer is attached before surface.configure
is called. This is because there are many surface-based operations
that really require the width and height and digging through the views
didn't work well.
Signed-off-by: Jason Ekstrand <jason@jlekstrand.net>
2013-10-13 07:38:11 +04:00
|
|
|
repaint_views(output, &undamaged);
|
2013-05-07 18:50:09 +04:00
|
|
|
gr->fan_debug = 1;
|
2012-09-06 05:54:15 +04:00
|
|
|
pixman_region32_fini(&undamaged);
|
|
|
|
}
|
|
|
|
|
2013-03-05 19:30:28 +04:00
|
|
|
pixman_region32_init(&total_damage);
|
2013-03-05 19:30:30 +04:00
|
|
|
pixman_region32_init(&buffer_damage);
|
|
|
|
|
2014-02-05 07:36:38 +04:00
|
|
|
output_get_damage(output, &buffer_damage, &border_damage);
|
|
|
|
output_rotate_damage(output, output_damage, go->border_status);
|
2013-03-05 19:30:30 +04:00
|
|
|
|
|
|
|
pixman_region32_union(&total_damage, &buffer_damage, output_damage);
|
2014-02-05 07:36:38 +04:00
|
|
|
border_damage |= go->border_status;
|
2012-09-14 17:12:03 +04:00
|
|
|
|
Split the geometry information from weston_surface out into weston_view
The weston_surface structure is split into two structures:
* The weston_surface structure storres everything required for a
client-side or server-side surface. This includes buffers; callbacks;
backend private data; input, damage, and opaque regions; and a few other
bookkeeping bits.
* The weston_view structure represents an entity in the scenegraph and
storres all of the geometry information. This includes clip region,
alpha, position, and the transformation list as well as all of the
temporary information derived from the geometry state. Because a view,
and not a surface, is a scenegraph element, the view is what is placed
in layers and planes.
There are a few things worth noting about the surface/view split:
1. This is *not* a modification to the protocol. It is, instead, a
modification to Weston's internal scenegraph to allow a single surface
to exist in multiple places at a time. Clients are completely unaware
of how many views to a particular surface exist.
2. A view is considered a direct child of a surface and is destroyed when
the surface is destroyed. Because of this, the view.surface pointer is
always valid and non-null.
3. The compositor's surface_list is replaced with a view_list. Due to
subsurfaces, building the view list is a little more complicated than
it used to be and involves building a tree of views on the fly whenever
subsurfaces are used. However, this means that backends can remain
completely subsurface-agnostic.
4. Surfaces and views both keep track of which outputs they are on.
5. The weston_surface structure now has width and height fields. These
are populated when a new buffer is attached before surface.configure
is called. This is because there are many surface-based operations
that really require the width and height and digging through the views
didn't work well.
Signed-off-by: Jason Ekstrand <jason@jlekstrand.net>
2013-10-13 07:38:11 +04:00
|
|
|
repaint_views(output, &total_damage);
|
2013-03-05 19:30:28 +04:00
|
|
|
|
|
|
|
pixman_region32_fini(&total_damage);
|
2013-03-05 19:30:30 +04:00
|
|
|
pixman_region32_fini(&buffer_damage);
|
2012-09-06 05:54:15 +04:00
|
|
|
|
2014-02-05 07:36:38 +04:00
|
|
|
draw_output_borders(output, border_damage);
|
2012-11-13 22:10:20 +04:00
|
|
|
|
2012-11-16 19:23:52 +04:00
|
|
|
pixman_region32_copy(&output->previous_damage, output_damage);
|
2012-09-06 05:54:15 +04:00
|
|
|
wl_signal_emit(&output->frame_signal, output);
|
|
|
|
|
2014-02-05 07:36:39 +04:00
|
|
|
#ifdef EGL_EXT_swap_buffers_with_damage
|
|
|
|
if (gr->swap_buffers_with_damage) {
|
|
|
|
pixman_region32_init(&buffer_damage);
|
|
|
|
weston_transformed_region(output->width, output->height,
|
|
|
|
output->transform,
|
|
|
|
output->current_scale,
|
|
|
|
output_damage, &buffer_damage);
|
|
|
|
|
|
|
|
if (output_has_borders(output)) {
|
|
|
|
pixman_region32_translate(&buffer_damage,
|
|
|
|
go->borders[GL_RENDERER_BORDER_LEFT].width,
|
|
|
|
go->borders[GL_RENDERER_BORDER_TOP].height);
|
|
|
|
output_get_border_damage(output, go->border_status,
|
|
|
|
&buffer_damage);
|
|
|
|
}
|
|
|
|
|
|
|
|
rects = pixman_region32_rectangles(&buffer_damage, &nrects);
|
|
|
|
egl_damage = malloc(nrects * 4 * sizeof(EGLint));
|
|
|
|
|
|
|
|
buffer_height = go->borders[GL_RENDERER_BORDER_TOP].height +
|
|
|
|
output->current_mode->height +
|
|
|
|
go->borders[GL_RENDERER_BORDER_BOTTOM].height;
|
|
|
|
|
|
|
|
d = egl_damage;
|
|
|
|
for (i = 0; i < nrects; ++i) {
|
|
|
|
*d++ = rects[i].x1;
|
|
|
|
*d++ = buffer_height - rects[i].y2;
|
|
|
|
*d++ = rects[i].x2 - rects[i].x1;
|
|
|
|
*d++ = rects[i].y2 - rects[i].y1;
|
|
|
|
}
|
|
|
|
ret = gr->swap_buffers_with_damage(gr->egl_display,
|
|
|
|
go->egl_surface,
|
|
|
|
egl_damage, nrects);
|
|
|
|
free(egl_damage);
|
|
|
|
pixman_region32_fini(&buffer_damage);
|
|
|
|
} else {
|
|
|
|
ret = eglSwapBuffers(gr->egl_display, go->egl_surface);
|
|
|
|
}
|
|
|
|
#else /* ! defined EGL_EXT_swap_buffers_with_damage */
|
2012-11-13 22:10:19 +04:00
|
|
|
ret = eglSwapBuffers(gr->egl_display, go->egl_surface);
|
2014-02-05 07:36:39 +04:00
|
|
|
#endif
|
|
|
|
|
2012-09-06 05:54:15 +04:00
|
|
|
if (ret == EGL_FALSE && !errored) {
|
|
|
|
errored = 1;
|
|
|
|
weston_log("Failed in eglSwapBuffers.\n");
|
2012-11-27 14:25:25 +04:00
|
|
|
gl_renderer_print_egl_error_state();
|
2012-09-06 05:54:15 +04:00
|
|
|
}
|
|
|
|
|
2014-02-05 07:36:38 +04:00
|
|
|
go->border_status = BORDER_STATUS_CLEAN;
|
2012-09-06 05:54:15 +04:00
|
|
|
}
|
2012-09-06 06:06:26 +04:00
|
|
|
|
2012-11-13 22:10:21 +04:00
|
|
|
static int
|
2012-11-13 22:10:29 +04:00
|
|
|
gl_renderer_read_pixels(struct weston_output *output,
|
2012-11-13 22:10:21 +04:00
|
|
|
pixman_format_code_t format, void *pixels,
|
|
|
|
uint32_t x, uint32_t y,
|
|
|
|
uint32_t width, uint32_t height)
|
|
|
|
{
|
|
|
|
GLenum gl_format;
|
2014-04-03 04:53:59 +04:00
|
|
|
struct gl_output_state *go = get_output_state(output);
|
|
|
|
|
|
|
|
x += go->borders[GL_RENDERER_BORDER_LEFT].width;
|
|
|
|
y += go->borders[GL_RENDERER_BORDER_BOTTOM].height;
|
2012-11-13 22:10:21 +04:00
|
|
|
|
|
|
|
switch (format) {
|
|
|
|
case PIXMAN_a8r8g8b8:
|
|
|
|
gl_format = GL_BGRA_EXT;
|
|
|
|
break;
|
|
|
|
case PIXMAN_a8b8g8r8:
|
|
|
|
gl_format = GL_RGBA;
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (use_output(output) < 0)
|
|
|
|
return -1;
|
|
|
|
|
|
|
|
glPixelStorei(GL_PACK_ALIGNMENT, 1);
|
|
|
|
glReadPixels(x, y, width, height, gl_format,
|
|
|
|
GL_UNSIGNED_BYTE, pixels);
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2012-09-06 06:49:55 +04:00
|
|
|
static void
|
2012-11-13 22:10:29 +04:00
|
|
|
gl_renderer_flush_damage(struct weston_surface *surface)
|
2012-09-06 06:13:58 +04:00
|
|
|
{
|
2012-11-13 22:10:29 +04:00
|
|
|
struct gl_renderer *gr = get_renderer(surface->compositor);
|
|
|
|
struct gl_surface_state *gs = get_surface_state(surface);
|
2013-06-21 05:38:23 +04:00
|
|
|
struct weston_buffer *buffer = gs->buffer_ref.buffer;
|
Split the geometry information from weston_surface out into weston_view
The weston_surface structure is split into two structures:
* The weston_surface structure storres everything required for a
client-side or server-side surface. This includes buffers; callbacks;
backend private data; input, damage, and opaque regions; and a few other
bookkeeping bits.
* The weston_view structure represents an entity in the scenegraph and
storres all of the geometry information. This includes clip region,
alpha, position, and the transformation list as well as all of the
temporary information derived from the geometry state. Because a view,
and not a surface, is a scenegraph element, the view is what is placed
in layers and planes.
There are a few things worth noting about the surface/view split:
1. This is *not* a modification to the protocol. It is, instead, a
modification to Weston's internal scenegraph to allow a single surface
to exist in multiple places at a time. Clients are completely unaware
of how many views to a particular surface exist.
2. A view is considered a direct child of a surface and is destroyed when
the surface is destroyed. Because of this, the view.surface pointer is
always valid and non-null.
3. The compositor's surface_list is replaced with a view_list. Due to
subsurfaces, building the view list is a little more complicated than
it used to be and involves building a tree of views on the fly whenever
subsurfaces are used. However, this means that backends can remain
completely subsurface-agnostic.
4. Surfaces and views both keep track of which outputs they are on.
5. The weston_surface structure now has width and height fields. These
are populated when a new buffer is attached before surface.configure
is called. This is because there are many surface-based operations
that really require the width and height and digging through the views
didn't work well.
Signed-off-by: Jason Ekstrand <jason@jlekstrand.net>
2013-10-13 07:38:11 +04:00
|
|
|
struct weston_view *view;
|
|
|
|
int texture_used;
|
2012-11-13 22:10:26 +04:00
|
|
|
|
2013-08-07 20:55:07 +04:00
|
|
|
#ifdef GL_EXT_unpack_subimage
|
2012-09-06 06:13:58 +04:00
|
|
|
pixman_box32_t *rectangles;
|
|
|
|
void *data;
|
|
|
|
int i, n;
|
|
|
|
#endif
|
|
|
|
|
2012-12-04 17:58:16 +04:00
|
|
|
pixman_region32_union(&gs->texture_damage,
|
|
|
|
&gs->texture_damage, &surface->damage);
|
2012-11-07 14:25:13 +04:00
|
|
|
|
2012-12-04 17:58:13 +04:00
|
|
|
if (!buffer)
|
|
|
|
return;
|
|
|
|
|
2012-11-07 14:25:13 +04:00
|
|
|
/* Avoid upload, if the texture won't be used this time.
|
2012-12-04 17:58:13 +04:00
|
|
|
* We still accumulate the damage in texture_damage, and
|
|
|
|
* hold the reference to the buffer, in case the surface
|
|
|
|
* migrates back to the primary plane.
|
2012-11-07 14:25:13 +04:00
|
|
|
*/
|
Split the geometry information from weston_surface out into weston_view
The weston_surface structure is split into two structures:
* The weston_surface structure storres everything required for a
client-side or server-side surface. This includes buffers; callbacks;
backend private data; input, damage, and opaque regions; and a few other
bookkeeping bits.
* The weston_view structure represents an entity in the scenegraph and
storres all of the geometry information. This includes clip region,
alpha, position, and the transformation list as well as all of the
temporary information derived from the geometry state. Because a view,
and not a surface, is a scenegraph element, the view is what is placed
in layers and planes.
There are a few things worth noting about the surface/view split:
1. This is *not* a modification to the protocol. It is, instead, a
modification to Weston's internal scenegraph to allow a single surface
to exist in multiple places at a time. Clients are completely unaware
of how many views to a particular surface exist.
2. A view is considered a direct child of a surface and is destroyed when
the surface is destroyed. Because of this, the view.surface pointer is
always valid and non-null.
3. The compositor's surface_list is replaced with a view_list. Due to
subsurfaces, building the view list is a little more complicated than
it used to be and involves building a tree of views on the fly whenever
subsurfaces are used. However, this means that backends can remain
completely subsurface-agnostic.
4. Surfaces and views both keep track of which outputs they are on.
5. The weston_surface structure now has width and height fields. These
are populated when a new buffer is attached before surface.configure
is called. This is because there are many surface-based operations
that really require the width and height and digging through the views
didn't work well.
Signed-off-by: Jason Ekstrand <jason@jlekstrand.net>
2013-10-13 07:38:11 +04:00
|
|
|
texture_used = 0;
|
|
|
|
wl_list_for_each(view, &surface->views, surface_link) {
|
|
|
|
if (view->plane == &surface->compositor->primary_plane) {
|
|
|
|
texture_used = 1;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (!texture_used)
|
2012-11-07 14:25:13 +04:00
|
|
|
return;
|
|
|
|
|
2013-11-19 17:22:05 +04:00
|
|
|
if (!pixman_region32_not_empty(&gs->texture_damage) &&
|
|
|
|
!gs->needs_full_upload)
|
2012-12-04 17:58:13 +04:00
|
|
|
goto done;
|
2012-11-07 14:25:13 +04:00
|
|
|
|
2012-11-13 22:10:26 +04:00
|
|
|
glBindTexture(GL_TEXTURE_2D, gs->textures[0]);
|
2012-09-06 06:13:58 +04:00
|
|
|
|
2012-11-13 22:10:27 +04:00
|
|
|
if (!gr->has_unpack_subimage) {
|
2013-11-13 19:44:06 +04:00
|
|
|
wl_shm_buffer_begin_access(buffer->shm_buffer);
|
2014-04-07 18:01:01 +04:00
|
|
|
glTexImage2D(GL_TEXTURE_2D, 0, gs->gl_format,
|
2012-12-04 17:58:15 +04:00
|
|
|
gs->pitch, buffer->height, 0,
|
2014-04-07 18:01:01 +04:00
|
|
|
gs->gl_format, gs->gl_pixel_type,
|
2013-06-21 05:38:23 +04:00
|
|
|
wl_shm_buffer_get_data(buffer->shm_buffer));
|
2013-11-13 19:44:06 +04:00
|
|
|
wl_shm_buffer_end_access(buffer->shm_buffer);
|
2012-09-06 06:13:58 +04:00
|
|
|
|
2012-11-07 14:25:13 +04:00
|
|
|
goto done;
|
2012-09-06 06:13:58 +04:00
|
|
|
}
|
|
|
|
|
2013-08-07 20:55:07 +04:00
|
|
|
#ifdef GL_EXT_unpack_subimage
|
|
|
|
glPixelStorei(GL_UNPACK_ROW_LENGTH_EXT, gs->pitch);
|
2013-06-21 05:38:23 +04:00
|
|
|
data = wl_shm_buffer_get_data(buffer->shm_buffer);
|
2013-06-07 17:52:45 +04:00
|
|
|
|
|
|
|
if (gs->needs_full_upload) {
|
2013-08-07 20:55:07 +04:00
|
|
|
glPixelStorei(GL_UNPACK_SKIP_PIXELS_EXT, 0);
|
|
|
|
glPixelStorei(GL_UNPACK_SKIP_ROWS_EXT, 0);
|
2013-11-13 19:44:06 +04:00
|
|
|
wl_shm_buffer_begin_access(buffer->shm_buffer);
|
2014-04-07 18:01:01 +04:00
|
|
|
glTexImage2D(GL_TEXTURE_2D, 0, gs->gl_format,
|
2014-04-04 19:24:54 +04:00
|
|
|
gs->pitch, buffer->height, 0,
|
2014-04-07 18:01:01 +04:00
|
|
|
gs->gl_format, gs->gl_pixel_type, data);
|
2013-11-13 19:44:06 +04:00
|
|
|
wl_shm_buffer_end_access(buffer->shm_buffer);
|
2013-06-07 17:52:45 +04:00
|
|
|
goto done;
|
|
|
|
}
|
|
|
|
|
2012-12-04 17:58:16 +04:00
|
|
|
rectangles = pixman_region32_rectangles(&gs->texture_damage, &n);
|
2013-11-13 19:44:06 +04:00
|
|
|
wl_shm_buffer_begin_access(buffer->shm_buffer);
|
2012-09-06 06:13:58 +04:00
|
|
|
for (i = 0; i < n; i++) {
|
2012-11-28 19:10:26 +04:00
|
|
|
pixman_box32_t r;
|
|
|
|
|
|
|
|
r = weston_surface_to_buffer_rect(surface, rectangles[i]);
|
|
|
|
|
2013-08-07 20:55:07 +04:00
|
|
|
glPixelStorei(GL_UNPACK_SKIP_PIXELS_EXT, r.x1);
|
|
|
|
glPixelStorei(GL_UNPACK_SKIP_ROWS_EXT, r.y1);
|
2012-11-28 19:10:26 +04:00
|
|
|
glTexSubImage2D(GL_TEXTURE_2D, 0, r.x1, r.y1,
|
|
|
|
r.x2 - r.x1, r.y2 - r.y1,
|
2014-04-07 18:01:01 +04:00
|
|
|
gs->gl_format, gs->gl_pixel_type, data);
|
2012-09-06 06:13:58 +04:00
|
|
|
}
|
2013-11-13 19:44:06 +04:00
|
|
|
wl_shm_buffer_end_access(buffer->shm_buffer);
|
2012-09-06 06:13:58 +04:00
|
|
|
#endif
|
2012-11-07 14:25:13 +04:00
|
|
|
|
|
|
|
done:
|
2012-12-04 17:58:16 +04:00
|
|
|
pixman_region32_fini(&gs->texture_damage);
|
|
|
|
pixman_region32_init(&gs->texture_damage);
|
2013-06-07 17:52:45 +04:00
|
|
|
gs->needs_full_upload = 0;
|
2012-12-04 17:58:13 +04:00
|
|
|
|
|
|
|
weston_buffer_reference(&gs->buffer_ref, NULL);
|
2012-09-06 06:13:58 +04:00
|
|
|
}
|
|
|
|
|
2012-09-06 06:38:18 +04:00
|
|
|
static void
|
2012-11-13 22:10:29 +04:00
|
|
|
ensure_textures(struct gl_surface_state *gs, int num_textures)
|
2012-09-06 06:38:18 +04:00
|
|
|
{
|
|
|
|
int i;
|
|
|
|
|
2012-11-13 22:10:26 +04:00
|
|
|
if (num_textures <= gs->num_textures)
|
2012-09-06 06:38:18 +04:00
|
|
|
return;
|
|
|
|
|
2012-11-13 22:10:26 +04:00
|
|
|
for (i = gs->num_textures; i < num_textures; i++) {
|
|
|
|
glGenTextures(1, &gs->textures[i]);
|
|
|
|
glBindTexture(gs->target, gs->textures[i]);
|
|
|
|
glTexParameteri(gs->target,
|
2012-09-06 06:38:18 +04:00
|
|
|
GL_TEXTURE_WRAP_S, GL_CLAMP_TO_EDGE);
|
2012-11-13 22:10:26 +04:00
|
|
|
glTexParameteri(gs->target,
|
2012-09-06 06:38:18 +04:00
|
|
|
GL_TEXTURE_WRAP_T, GL_CLAMP_TO_EDGE);
|
|
|
|
}
|
2012-11-13 22:10:26 +04:00
|
|
|
gs->num_textures = num_textures;
|
|
|
|
glBindTexture(gs->target, 0);
|
2012-09-06 06:38:18 +04:00
|
|
|
}
|
|
|
|
|
2013-06-07 17:52:46 +04:00
|
|
|
static void
|
|
|
|
gl_renderer_attach_shm(struct weston_surface *es, struct weston_buffer *buffer,
|
|
|
|
struct wl_shm_buffer *shm_buffer)
|
|
|
|
{
|
|
|
|
struct weston_compositor *ec = es->compositor;
|
|
|
|
struct gl_renderer *gr = get_renderer(ec);
|
|
|
|
struct gl_surface_state *gs = get_surface_state(es);
|
2014-04-07 18:01:01 +04:00
|
|
|
GLenum gl_format, gl_pixel_type;
|
2013-08-06 22:05:55 +04:00
|
|
|
int pitch;
|
2013-06-07 17:52:46 +04:00
|
|
|
|
|
|
|
buffer->shm_buffer = shm_buffer;
|
|
|
|
buffer->width = wl_shm_buffer_get_width(shm_buffer);
|
|
|
|
buffer->height = wl_shm_buffer_get_height(shm_buffer);
|
|
|
|
|
2013-08-06 22:05:55 +04:00
|
|
|
switch (wl_shm_buffer_get_format(shm_buffer)) {
|
|
|
|
case WL_SHM_FORMAT_XRGB8888:
|
|
|
|
gs->shader = &gr->texture_shader_rgbx;
|
|
|
|
pitch = wl_shm_buffer_get_stride(shm_buffer) / 4;
|
2014-04-07 18:01:01 +04:00
|
|
|
gl_format = GL_BGRA_EXT;
|
|
|
|
gl_pixel_type = GL_UNSIGNED_BYTE;
|
2013-08-06 22:05:55 +04:00
|
|
|
break;
|
|
|
|
case WL_SHM_FORMAT_ARGB8888:
|
|
|
|
gs->shader = &gr->texture_shader_rgba;
|
|
|
|
pitch = wl_shm_buffer_get_stride(shm_buffer) / 4;
|
2014-04-07 18:01:01 +04:00
|
|
|
gl_format = GL_BGRA_EXT;
|
|
|
|
gl_pixel_type = GL_UNSIGNED_BYTE;
|
2013-08-06 22:05:55 +04:00
|
|
|
break;
|
|
|
|
case WL_SHM_FORMAT_RGB565:
|
|
|
|
gs->shader = &gr->texture_shader_rgbx;
|
|
|
|
pitch = wl_shm_buffer_get_stride(shm_buffer) / 2;
|
2014-04-07 18:01:01 +04:00
|
|
|
gl_format = GL_RGB;
|
|
|
|
gl_pixel_type = GL_UNSIGNED_SHORT_5_6_5;
|
2013-08-06 22:05:55 +04:00
|
|
|
break;
|
|
|
|
default:
|
2014-04-07 18:01:01 +04:00
|
|
|
weston_log("warning: unknown shm buffer format: %08x\n",
|
|
|
|
wl_shm_buffer_get_format(shm_buffer));
|
|
|
|
return;
|
2013-08-06 22:05:55 +04:00
|
|
|
}
|
|
|
|
|
2013-06-07 17:52:46 +04:00
|
|
|
/* Only allocate a texture if it doesn't match existing one.
|
|
|
|
* If a switch from DRM allocated buffer to a SHM buffer is
|
|
|
|
* happening, we need to allocate a new texture buffer. */
|
2013-08-06 22:05:55 +04:00
|
|
|
if (pitch != gs->pitch ||
|
2013-06-07 17:52:46 +04:00
|
|
|
buffer->height != gs->height ||
|
2014-04-07 18:01:01 +04:00
|
|
|
gl_format != gs->gl_format ||
|
|
|
|
gl_pixel_type != gs->gl_pixel_type ||
|
2013-06-07 17:52:46 +04:00
|
|
|
gs->buffer_type != BUFFER_TYPE_SHM) {
|
2013-08-06 22:05:55 +04:00
|
|
|
gs->pitch = pitch;
|
2013-06-07 17:52:46 +04:00
|
|
|
gs->height = buffer->height;
|
|
|
|
gs->target = GL_TEXTURE_2D;
|
2014-04-07 18:01:01 +04:00
|
|
|
gs->gl_format = gl_format;
|
|
|
|
gs->gl_pixel_type = gl_pixel_type;
|
2013-06-07 17:52:46 +04:00
|
|
|
gs->buffer_type = BUFFER_TYPE_SHM;
|
|
|
|
gs->needs_full_upload = 1;
|
2013-08-29 11:36:44 +04:00
|
|
|
gs->y_inverted = 1;
|
2013-06-07 17:52:46 +04:00
|
|
|
|
2013-10-25 17:26:33 +04:00
|
|
|
gs->surface = es;
|
|
|
|
|
2013-06-07 17:52:46 +04:00
|
|
|
ensure_textures(gs, 1);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
gl_renderer_attach_egl(struct weston_surface *es, struct weston_buffer *buffer,
|
|
|
|
uint32_t format)
|
|
|
|
{
|
|
|
|
struct weston_compositor *ec = es->compositor;
|
|
|
|
struct gl_renderer *gr = get_renderer(ec);
|
|
|
|
struct gl_surface_state *gs = get_surface_state(es);
|
|
|
|
EGLint attribs[3];
|
|
|
|
int i, num_planes;
|
|
|
|
|
|
|
|
buffer->legacy_buffer = (struct wl_buffer *)buffer->resource;
|
|
|
|
gr->query_buffer(gr->egl_display, buffer->legacy_buffer,
|
|
|
|
EGL_WIDTH, &buffer->width);
|
|
|
|
gr->query_buffer(gr->egl_display, buffer->legacy_buffer,
|
|
|
|
EGL_HEIGHT, &buffer->height);
|
2013-08-29 11:36:44 +04:00
|
|
|
gr->query_buffer(gr->egl_display, buffer->legacy_buffer,
|
|
|
|
EGL_WAYLAND_Y_INVERTED_WL, &buffer->y_inverted);
|
2013-06-07 17:52:46 +04:00
|
|
|
|
2015-06-08 16:37:05 +03:00
|
|
|
for (i = 0; i < gs->num_images; i++) {
|
|
|
|
egl_image_unref(gs->images[i]);
|
|
|
|
gs->images[i] = NULL;
|
|
|
|
}
|
2013-06-07 17:52:46 +04:00
|
|
|
gs->num_images = 0;
|
|
|
|
gs->target = GL_TEXTURE_2D;
|
|
|
|
switch (format) {
|
|
|
|
case EGL_TEXTURE_RGB:
|
|
|
|
case EGL_TEXTURE_RGBA:
|
|
|
|
default:
|
|
|
|
num_planes = 1;
|
|
|
|
gs->shader = &gr->texture_shader_rgba;
|
|
|
|
break;
|
|
|
|
case EGL_TEXTURE_EXTERNAL_WL:
|
|
|
|
num_planes = 1;
|
|
|
|
gs->target = GL_TEXTURE_EXTERNAL_OES;
|
|
|
|
gs->shader = &gr->texture_shader_egl_external;
|
|
|
|
break;
|
|
|
|
case EGL_TEXTURE_Y_UV_WL:
|
|
|
|
num_planes = 2;
|
|
|
|
gs->shader = &gr->texture_shader_y_uv;
|
|
|
|
break;
|
|
|
|
case EGL_TEXTURE_Y_U_V_WL:
|
|
|
|
num_planes = 3;
|
|
|
|
gs->shader = &gr->texture_shader_y_u_v;
|
|
|
|
break;
|
|
|
|
case EGL_TEXTURE_Y_XUXV_WL:
|
|
|
|
num_planes = 2;
|
|
|
|
gs->shader = &gr->texture_shader_y_xuxv;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
ensure_textures(gs, num_planes);
|
|
|
|
for (i = 0; i < num_planes; i++) {
|
|
|
|
attribs[0] = EGL_WAYLAND_PLANE_WL;
|
|
|
|
attribs[1] = i;
|
|
|
|
attribs[2] = EGL_NONE;
|
2015-06-08 16:37:05 +03:00
|
|
|
gs->images[i] = egl_image_create(gr,
|
2013-06-07 17:52:46 +04:00
|
|
|
EGL_WAYLAND_BUFFER_WL,
|
|
|
|
buffer->legacy_buffer,
|
|
|
|
attribs);
|
|
|
|
if (!gs->images[i]) {
|
|
|
|
weston_log("failed to create img for plane %d\n", i);
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
gs->num_images++;
|
|
|
|
|
|
|
|
glActiveTexture(GL_TEXTURE0 + i);
|
|
|
|
glBindTexture(gs->target, gs->textures[i]);
|
|
|
|
gr->image_target_texture_2d(gs->target,
|
2015-06-08 16:37:05 +03:00
|
|
|
gs->images[i]->image);
|
2013-06-07 17:52:46 +04:00
|
|
|
}
|
|
|
|
|
|
|
|
gs->pitch = buffer->width;
|
|
|
|
gs->height = buffer->height;
|
|
|
|
gs->buffer_type = BUFFER_TYPE_EGL;
|
2013-08-29 11:36:44 +04:00
|
|
|
gs->y_inverted = buffer->y_inverted;
|
2013-06-07 17:52:46 +04:00
|
|
|
}
|
|
|
|
|
2012-09-06 06:49:55 +04:00
|
|
|
static void
|
2013-06-21 05:38:23 +04:00
|
|
|
gl_renderer_attach(struct weston_surface *es, struct weston_buffer *buffer)
|
2012-09-06 06:38:18 +04:00
|
|
|
{
|
|
|
|
struct weston_compositor *ec = es->compositor;
|
2012-11-13 22:10:29 +04:00
|
|
|
struct gl_renderer *gr = get_renderer(ec);
|
|
|
|
struct gl_surface_state *gs = get_surface_state(es);
|
2013-06-21 05:38:23 +04:00
|
|
|
struct wl_shm_buffer *shm_buffer;
|
2013-06-07 17:52:46 +04:00
|
|
|
EGLint format;
|
|
|
|
int i;
|
2012-09-06 06:38:18 +04:00
|
|
|
|
2012-12-04 17:58:13 +04:00
|
|
|
weston_buffer_reference(&gs->buffer_ref, buffer);
|
|
|
|
|
2012-09-06 06:38:18 +04:00
|
|
|
if (!buffer) {
|
2012-11-13 22:10:26 +04:00
|
|
|
for (i = 0; i < gs->num_images; i++) {
|
2015-06-08 16:37:05 +03:00
|
|
|
egl_image_unref(gs->images[i]);
|
2012-11-13 22:10:26 +04:00
|
|
|
gs->images[i] = NULL;
|
2012-09-06 06:38:18 +04:00
|
|
|
}
|
2012-11-13 22:10:26 +04:00
|
|
|
gs->num_images = 0;
|
|
|
|
glDeleteTextures(gs->num_textures, gs->textures);
|
|
|
|
gs->num_textures = 0;
|
2013-06-07 17:52:44 +04:00
|
|
|
gs->buffer_type = BUFFER_TYPE_NULL;
|
2013-08-29 11:36:44 +04:00
|
|
|
gs->y_inverted = 1;
|
2012-09-06 06:38:18 +04:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2013-06-21 05:38:23 +04:00
|
|
|
shm_buffer = wl_shm_buffer_get(buffer->resource);
|
2012-09-06 06:38:18 +04:00
|
|
|
|
2013-06-07 17:52:46 +04:00
|
|
|
if (shm_buffer)
|
|
|
|
gl_renderer_attach_shm(es, buffer, shm_buffer);
|
2013-08-07 22:59:54 +04:00
|
|
|
else if (gr->query_buffer(gr->egl_display, (void *) buffer->resource,
|
2013-06-07 17:52:46 +04:00
|
|
|
EGL_TEXTURE_FORMAT, &format))
|
|
|
|
gl_renderer_attach_egl(es, buffer, format);
|
|
|
|
else {
|
2012-09-06 06:38:18 +04:00
|
|
|
weston_log("unhandled buffer type!\n");
|
2012-12-04 17:58:13 +04:00
|
|
|
weston_buffer_reference(&gs->buffer_ref, NULL);
|
2013-06-07 17:52:44 +04:00
|
|
|
gs->buffer_type = BUFFER_TYPE_NULL;
|
2013-08-29 11:36:44 +04:00
|
|
|
gs->y_inverted = 1;
|
2012-09-06 06:38:18 +04:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2012-11-13 22:10:23 +04:00
|
|
|
static void
|
2012-11-13 22:10:29 +04:00
|
|
|
gl_renderer_surface_set_color(struct weston_surface *surface,
|
2012-11-13 22:10:23 +04:00
|
|
|
float red, float green, float blue, float alpha)
|
|
|
|
{
|
2012-11-13 22:10:29 +04:00
|
|
|
struct gl_surface_state *gs = get_surface_state(surface);
|
|
|
|
struct gl_renderer *gr = get_renderer(surface->compositor);
|
2012-11-13 22:10:23 +04:00
|
|
|
|
|
|
|
gs->color[0] = red;
|
|
|
|
gs->color[1] = green;
|
|
|
|
gs->color[2] = blue;
|
|
|
|
gs->color[3] = alpha;
|
2015-02-09 14:56:56 +03:00
|
|
|
gs->buffer_type = BUFFER_TYPE_SOLID;
|
|
|
|
gs->pitch = 1;
|
|
|
|
gs->height = 1;
|
2012-11-13 22:10:23 +04:00
|
|
|
|
2012-11-13 22:10:25 +04:00
|
|
|
gs->shader = &gr->solid_shader;
|
2012-11-13 22:10:23 +04:00
|
|
|
}
|
|
|
|
|
2015-02-09 14:37:27 +03:00
|
|
|
static void
|
|
|
|
gl_renderer_surface_get_content_size(struct weston_surface *surface,
|
|
|
|
int *width, int *height)
|
|
|
|
{
|
|
|
|
struct gl_surface_state *gs = get_surface_state(surface);
|
|
|
|
|
|
|
|
if (gs->buffer_type == BUFFER_TYPE_NULL) {
|
|
|
|
*width = 0;
|
|
|
|
*height = 0;
|
|
|
|
} else {
|
|
|
|
*width = gs->pitch;
|
|
|
|
*height = gs->height;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static uint32_t
|
|
|
|
pack_color(pixman_format_code_t format, float *c)
|
|
|
|
{
|
|
|
|
uint8_t r = round(c[0] * 255.0f);
|
|
|
|
uint8_t g = round(c[1] * 255.0f);
|
|
|
|
uint8_t b = round(c[2] * 255.0f);
|
|
|
|
uint8_t a = round(c[3] * 255.0f);
|
|
|
|
|
|
|
|
switch (format) {
|
|
|
|
case PIXMAN_a8b8g8r8:
|
|
|
|
return (a << 24) | (b << 16) | (g << 8) | r;
|
|
|
|
default:
|
|
|
|
assert(0);
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static int
|
|
|
|
gl_renderer_surface_copy_content(struct weston_surface *surface,
|
|
|
|
void *target, size_t size,
|
|
|
|
int src_x, int src_y,
|
|
|
|
int width, int height)
|
|
|
|
{
|
|
|
|
static const GLfloat verts[4 * 2] = {
|
|
|
|
0.0f, 0.0f,
|
|
|
|
1.0f, 0.0f,
|
|
|
|
1.0f, 1.0f,
|
|
|
|
0.0f, 1.0f
|
|
|
|
};
|
|
|
|
static const GLfloat projmat_normal[16] = { /* transpose */
|
|
|
|
2.0f, 0.0f, 0.0f, 0.0f,
|
|
|
|
0.0f, 2.0f, 0.0f, 0.0f,
|
|
|
|
0.0f, 0.0f, 1.0f, 0.0f,
|
|
|
|
-1.0f, -1.0f, 0.0f, 1.0f
|
|
|
|
};
|
|
|
|
static const GLfloat projmat_yinvert[16] = { /* transpose */
|
|
|
|
2.0f, 0.0f, 0.0f, 0.0f,
|
|
|
|
0.0f, -2.0f, 0.0f, 0.0f,
|
|
|
|
0.0f, 0.0f, 1.0f, 0.0f,
|
|
|
|
-1.0f, 1.0f, 0.0f, 1.0f
|
|
|
|
};
|
|
|
|
const pixman_format_code_t format = PIXMAN_a8b8g8r8;
|
|
|
|
const size_t bytespp = 4; /* PIXMAN_a8b8g8r8 */
|
|
|
|
const GLenum gl_format = GL_RGBA; /* PIXMAN_a8b8g8r8 little-endian */
|
|
|
|
struct gl_renderer *gr = get_renderer(surface->compositor);
|
|
|
|
struct gl_surface_state *gs = get_surface_state(surface);
|
|
|
|
int cw, ch;
|
|
|
|
GLuint fbo;
|
|
|
|
GLuint tex;
|
|
|
|
GLenum status;
|
|
|
|
const GLfloat *proj;
|
|
|
|
int i;
|
|
|
|
|
|
|
|
gl_renderer_surface_get_content_size(surface, &cw, &ch);
|
|
|
|
|
|
|
|
switch (gs->buffer_type) {
|
|
|
|
case BUFFER_TYPE_NULL:
|
|
|
|
return -1;
|
|
|
|
case BUFFER_TYPE_SOLID:
|
|
|
|
*(uint32_t *)target = pack_color(format, gs->color);
|
|
|
|
return 0;
|
|
|
|
case BUFFER_TYPE_SHM:
|
|
|
|
gl_renderer_flush_damage(surface);
|
|
|
|
/* fall through */
|
|
|
|
case BUFFER_TYPE_EGL:
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
glGenTextures(1, &tex);
|
|
|
|
glBindTexture(GL_TEXTURE_2D, tex);
|
|
|
|
glTexImage2D(GL_TEXTURE_2D, 0, GL_RGBA, cw, ch,
|
|
|
|
0, GL_RGBA, GL_UNSIGNED_BYTE, NULL);
|
|
|
|
glBindTexture(GL_TEXTURE_2D, 0);
|
|
|
|
|
|
|
|
glGenFramebuffers(1, &fbo);
|
|
|
|
glBindFramebuffer(GL_FRAMEBUFFER, fbo);
|
|
|
|
glFramebufferTexture2D(GL_FRAMEBUFFER, GL_COLOR_ATTACHMENT0,
|
|
|
|
GL_TEXTURE_2D, tex, 0);
|
|
|
|
|
|
|
|
status = glCheckFramebufferStatus(GL_FRAMEBUFFER);
|
|
|
|
if (status != GL_FRAMEBUFFER_COMPLETE) {
|
|
|
|
weston_log("%s: fbo error: %#x\n", __func__, status);
|
|
|
|
glDeleteFramebuffers(1, &fbo);
|
|
|
|
glDeleteTextures(1, &tex);
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
glViewport(0, 0, cw, ch);
|
|
|
|
glDisable(GL_BLEND);
|
|
|
|
use_shader(gr, gs->shader);
|
|
|
|
if (gs->y_inverted)
|
|
|
|
proj = projmat_normal;
|
|
|
|
else
|
|
|
|
proj = projmat_yinvert;
|
|
|
|
|
|
|
|
glUniformMatrix4fv(gs->shader->proj_uniform, 1, GL_FALSE, proj);
|
|
|
|
glUniform1f(gs->shader->alpha_uniform, 1.0f);
|
|
|
|
|
|
|
|
for (i = 0; i < gs->num_textures; i++) {
|
|
|
|
glUniform1i(gs->shader->tex_uniforms[i], i);
|
|
|
|
|
|
|
|
glActiveTexture(GL_TEXTURE0 + i);
|
|
|
|
glBindTexture(gs->target, gs->textures[i]);
|
|
|
|
glTexParameteri(gs->target, GL_TEXTURE_MIN_FILTER, GL_NEAREST);
|
|
|
|
glTexParameteri(gs->target, GL_TEXTURE_MAG_FILTER, GL_NEAREST);
|
|
|
|
}
|
|
|
|
|
|
|
|
/* position: */
|
|
|
|
glVertexAttribPointer(0, 2, GL_FLOAT, GL_FALSE, 0, verts);
|
|
|
|
glEnableVertexAttribArray(0);
|
|
|
|
|
|
|
|
/* texcoord: */
|
|
|
|
glVertexAttribPointer(1, 2, GL_FLOAT, GL_FALSE, 0, verts);
|
|
|
|
glEnableVertexAttribArray(1);
|
|
|
|
|
|
|
|
glDrawArrays(GL_TRIANGLE_FAN, 0, 4);
|
|
|
|
|
|
|
|
glDisableVertexAttribArray(1);
|
|
|
|
glDisableVertexAttribArray(0);
|
|
|
|
|
|
|
|
glPixelStorei(GL_PACK_ALIGNMENT, bytespp);
|
|
|
|
glReadPixels(src_x, src_y, width, height, gl_format,
|
|
|
|
GL_UNSIGNED_BYTE, target);
|
|
|
|
|
|
|
|
glDeleteFramebuffers(1, &fbo);
|
|
|
|
glDeleteTextures(1, &tex);
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2013-10-25 17:26:33 +04:00
|
|
|
static void
|
2013-10-25 17:26:34 +04:00
|
|
|
surface_state_destroy(struct gl_surface_state *gs, struct gl_renderer *gr)
|
2013-10-25 17:26:33 +04:00
|
|
|
{
|
|
|
|
int i;
|
|
|
|
|
2013-10-25 17:26:34 +04:00
|
|
|
wl_list_remove(&gs->surface_destroy_listener.link);
|
|
|
|
wl_list_remove(&gs->renderer_destroy_listener.link);
|
2013-10-25 17:26:33 +04:00
|
|
|
|
|
|
|
gs->surface->renderer_state = NULL;
|
|
|
|
|
|
|
|
glDeleteTextures(gs->num_textures, gs->textures);
|
|
|
|
|
|
|
|
for (i = 0; i < gs->num_images; i++)
|
2015-06-08 16:37:05 +03:00
|
|
|
egl_image_unref(gs->images[i]);
|
2013-10-25 17:26:33 +04:00
|
|
|
|
|
|
|
weston_buffer_reference(&gs->buffer_ref, NULL);
|
|
|
|
pixman_region32_fini(&gs->texture_damage);
|
|
|
|
free(gs);
|
|
|
|
}
|
|
|
|
|
2013-10-25 17:26:34 +04:00
|
|
|
static void
|
|
|
|
surface_state_handle_surface_destroy(struct wl_listener *listener, void *data)
|
|
|
|
{
|
|
|
|
struct gl_surface_state *gs;
|
|
|
|
struct gl_renderer *gr;
|
|
|
|
|
|
|
|
gs = container_of(listener, struct gl_surface_state,
|
|
|
|
surface_destroy_listener);
|
|
|
|
|
|
|
|
gr = get_renderer(gs->surface->compositor);
|
|
|
|
|
|
|
|
surface_state_destroy(gs, gr);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
surface_state_handle_renderer_destroy(struct wl_listener *listener, void *data)
|
|
|
|
{
|
|
|
|
struct gl_surface_state *gs;
|
|
|
|
struct gl_renderer *gr;
|
|
|
|
|
|
|
|
gr = data;
|
|
|
|
|
|
|
|
gs = container_of(listener, struct gl_surface_state,
|
|
|
|
renderer_destroy_listener);
|
|
|
|
|
|
|
|
surface_state_destroy(gs, gr);
|
|
|
|
}
|
|
|
|
|
2012-11-13 22:10:23 +04:00
|
|
|
static int
|
2012-11-13 22:10:29 +04:00
|
|
|
gl_renderer_create_surface(struct weston_surface *surface)
|
2012-11-13 22:10:23 +04:00
|
|
|
{
|
2012-11-13 22:10:29 +04:00
|
|
|
struct gl_surface_state *gs;
|
2013-10-25 17:26:34 +04:00
|
|
|
struct gl_renderer *gr = get_renderer(surface->compositor);
|
2012-11-13 22:10:23 +04:00
|
|
|
|
2014-11-21 09:21:57 +03:00
|
|
|
gs = zalloc(sizeof *gs);
|
|
|
|
if (gs == NULL)
|
2012-11-13 22:10:23 +04:00
|
|
|
return -1;
|
|
|
|
|
2012-12-04 17:58:15 +04:00
|
|
|
/* A buffer is never attached to solid color surfaces, yet
|
|
|
|
* they still go through texcoord computations. Do not divide
|
|
|
|
* by zero there.
|
|
|
|
*/
|
|
|
|
gs->pitch = 1;
|
2013-08-29 11:36:44 +04:00
|
|
|
gs->y_inverted = 1;
|
2012-12-04 17:58:15 +04:00
|
|
|
|
2013-10-25 17:26:34 +04:00
|
|
|
gs->surface = surface;
|
|
|
|
|
2012-12-04 17:58:16 +04:00
|
|
|
pixman_region32_init(&gs->texture_damage);
|
2012-11-13 22:10:23 +04:00
|
|
|
surface->renderer_state = gs;
|
|
|
|
|
2013-10-25 17:26:33 +04:00
|
|
|
gs->surface_destroy_listener.notify =
|
|
|
|
surface_state_handle_surface_destroy;
|
|
|
|
wl_signal_add(&surface->destroy_signal,
|
|
|
|
&gs->surface_destroy_listener);
|
2012-11-13 22:10:23 +04:00
|
|
|
|
2013-10-25 17:26:34 +04:00
|
|
|
gs->renderer_destroy_listener.notify =
|
|
|
|
surface_state_handle_renderer_destroy;
|
|
|
|
wl_signal_add(&gr->destroy_signal,
|
|
|
|
&gs->renderer_destroy_listener);
|
|
|
|
|
2013-11-19 17:22:05 +04:00
|
|
|
if (surface->buffer_ref.buffer) {
|
|
|
|
gl_renderer_attach(surface, surface->buffer_ref.buffer);
|
|
|
|
gl_renderer_flush_damage(surface);
|
|
|
|
}
|
|
|
|
|
2013-10-25 17:26:33 +04:00
|
|
|
return 0;
|
2012-09-07 05:59:29 +04:00
|
|
|
}
|
|
|
|
|
2012-09-06 06:06:26 +04:00
|
|
|
static const char vertex_shader[] =
|
|
|
|
"uniform mat4 proj;\n"
|
|
|
|
"attribute vec2 position;\n"
|
|
|
|
"attribute vec2 texcoord;\n"
|
|
|
|
"varying vec2 v_texcoord;\n"
|
|
|
|
"void main()\n"
|
|
|
|
"{\n"
|
|
|
|
" gl_Position = proj * vec4(position, 0.0, 1.0);\n"
|
|
|
|
" v_texcoord = texcoord;\n"
|
|
|
|
"}\n";
|
|
|
|
|
|
|
|
/* Declare common fragment shader uniforms */
|
|
|
|
#define FRAGMENT_CONVERT_YUV \
|
|
|
|
" y *= alpha;\n" \
|
|
|
|
" u *= alpha;\n" \
|
|
|
|
" v *= alpha;\n" \
|
|
|
|
" gl_FragColor.r = y + 1.59602678 * v;\n" \
|
|
|
|
" gl_FragColor.g = y - 0.39176229 * u - 0.81296764 * v;\n" \
|
|
|
|
" gl_FragColor.b = y + 2.01723214 * u;\n" \
|
|
|
|
" gl_FragColor.a = alpha;\n"
|
|
|
|
|
2012-11-08 19:20:46 +04:00
|
|
|
static const char fragment_debug[] =
|
|
|
|
" gl_FragColor = vec4(0.0, 0.3, 0.0, 0.2) + gl_FragColor * 0.8;\n";
|
|
|
|
|
|
|
|
static const char fragment_brace[] =
|
|
|
|
"}\n";
|
|
|
|
|
2012-09-06 06:06:26 +04:00
|
|
|
static const char texture_fragment_shader_rgba[] =
|
|
|
|
"precision mediump float;\n"
|
|
|
|
"varying vec2 v_texcoord;\n"
|
|
|
|
"uniform sampler2D tex;\n"
|
|
|
|
"uniform float alpha;\n"
|
|
|
|
"void main()\n"
|
|
|
|
"{\n"
|
|
|
|
" gl_FragColor = alpha * texture2D(tex, v_texcoord)\n;"
|
2012-11-08 19:20:46 +04:00
|
|
|
;
|
2012-09-06 06:06:26 +04:00
|
|
|
|
|
|
|
static const char texture_fragment_shader_rgbx[] =
|
|
|
|
"precision mediump float;\n"
|
|
|
|
"varying vec2 v_texcoord;\n"
|
|
|
|
"uniform sampler2D tex;\n"
|
|
|
|
"uniform float alpha;\n"
|
|
|
|
"void main()\n"
|
|
|
|
"{\n"
|
|
|
|
" gl_FragColor.rgb = alpha * texture2D(tex, v_texcoord).rgb\n;"
|
|
|
|
" gl_FragColor.a = alpha;\n"
|
2012-11-08 19:20:46 +04:00
|
|
|
;
|
2012-09-06 06:06:26 +04:00
|
|
|
|
|
|
|
static const char texture_fragment_shader_egl_external[] =
|
|
|
|
"#extension GL_OES_EGL_image_external : require\n"
|
|
|
|
"precision mediump float;\n"
|
|
|
|
"varying vec2 v_texcoord;\n"
|
|
|
|
"uniform samplerExternalOES tex;\n"
|
|
|
|
"uniform float alpha;\n"
|
|
|
|
"void main()\n"
|
|
|
|
"{\n"
|
|
|
|
" gl_FragColor = alpha * texture2D(tex, v_texcoord)\n;"
|
2012-11-08 19:20:46 +04:00
|
|
|
;
|
2012-09-06 06:06:26 +04:00
|
|
|
|
|
|
|
static const char texture_fragment_shader_y_uv[] =
|
|
|
|
"precision mediump float;\n"
|
|
|
|
"uniform sampler2D tex;\n"
|
|
|
|
"uniform sampler2D tex1;\n"
|
|
|
|
"varying vec2 v_texcoord;\n"
|
|
|
|
"uniform float alpha;\n"
|
|
|
|
"void main() {\n"
|
|
|
|
" float y = 1.16438356 * (texture2D(tex, v_texcoord).x - 0.0625);\n"
|
|
|
|
" float u = texture2D(tex1, v_texcoord).r - 0.5;\n"
|
|
|
|
" float v = texture2D(tex1, v_texcoord).g - 0.5;\n"
|
|
|
|
FRAGMENT_CONVERT_YUV
|
2012-11-08 19:20:46 +04:00
|
|
|
;
|
2012-09-06 06:06:26 +04:00
|
|
|
|
|
|
|
static const char texture_fragment_shader_y_u_v[] =
|
|
|
|
"precision mediump float;\n"
|
|
|
|
"uniform sampler2D tex;\n"
|
|
|
|
"uniform sampler2D tex1;\n"
|
|
|
|
"uniform sampler2D tex2;\n"
|
|
|
|
"varying vec2 v_texcoord;\n"
|
|
|
|
"uniform float alpha;\n"
|
|
|
|
"void main() {\n"
|
|
|
|
" float y = 1.16438356 * (texture2D(tex, v_texcoord).x - 0.0625);\n"
|
|
|
|
" float u = texture2D(tex1, v_texcoord).x - 0.5;\n"
|
|
|
|
" float v = texture2D(tex2, v_texcoord).x - 0.5;\n"
|
|
|
|
FRAGMENT_CONVERT_YUV
|
2012-11-08 19:20:46 +04:00
|
|
|
;
|
2012-09-06 06:06:26 +04:00
|
|
|
|
|
|
|
static const char texture_fragment_shader_y_xuxv[] =
|
|
|
|
"precision mediump float;\n"
|
|
|
|
"uniform sampler2D tex;\n"
|
|
|
|
"uniform sampler2D tex1;\n"
|
|
|
|
"varying vec2 v_texcoord;\n"
|
|
|
|
"uniform float alpha;\n"
|
|
|
|
"void main() {\n"
|
|
|
|
" float y = 1.16438356 * (texture2D(tex, v_texcoord).x - 0.0625);\n"
|
|
|
|
" float u = texture2D(tex1, v_texcoord).g - 0.5;\n"
|
|
|
|
" float v = texture2D(tex1, v_texcoord).a - 0.5;\n"
|
|
|
|
FRAGMENT_CONVERT_YUV
|
2012-11-08 19:20:46 +04:00
|
|
|
;
|
2012-09-06 06:06:26 +04:00
|
|
|
|
|
|
|
static const char solid_fragment_shader[] =
|
|
|
|
"precision mediump float;\n"
|
|
|
|
"uniform vec4 color;\n"
|
|
|
|
"uniform float alpha;\n"
|
|
|
|
"void main()\n"
|
|
|
|
"{\n"
|
|
|
|
" gl_FragColor = alpha * color\n;"
|
2012-11-08 19:20:46 +04:00
|
|
|
;
|
2012-09-06 06:06:26 +04:00
|
|
|
|
|
|
|
static int
|
2012-11-08 19:20:46 +04:00
|
|
|
compile_shader(GLenum type, int count, const char **sources)
|
2012-09-06 06:06:26 +04:00
|
|
|
{
|
|
|
|
GLuint s;
|
|
|
|
char msg[512];
|
|
|
|
GLint status;
|
|
|
|
|
|
|
|
s = glCreateShader(type);
|
2012-11-08 19:20:46 +04:00
|
|
|
glShaderSource(s, count, sources, NULL);
|
2012-09-06 06:06:26 +04:00
|
|
|
glCompileShader(s);
|
|
|
|
glGetShaderiv(s, GL_COMPILE_STATUS, &status);
|
|
|
|
if (!status) {
|
|
|
|
glGetShaderInfoLog(s, sizeof msg, NULL, msg);
|
|
|
|
weston_log("shader info: %s\n", msg);
|
|
|
|
return GL_NONE;
|
|
|
|
}
|
|
|
|
|
|
|
|
return s;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int
|
2013-05-17 15:00:40 +04:00
|
|
|
shader_init(struct gl_shader *shader, struct gl_renderer *renderer,
|
2012-09-06 06:06:26 +04:00
|
|
|
const char *vertex_source, const char *fragment_source)
|
|
|
|
{
|
|
|
|
char msg[512];
|
|
|
|
GLint status;
|
2012-11-08 19:20:46 +04:00
|
|
|
int count;
|
|
|
|
const char *sources[3];
|
2012-09-06 06:06:26 +04:00
|
|
|
|
|
|
|
shader->vertex_shader =
|
2012-11-08 19:20:46 +04:00
|
|
|
compile_shader(GL_VERTEX_SHADER, 1, &vertex_source);
|
|
|
|
|
|
|
|
if (renderer->fragment_shader_debug) {
|
|
|
|
sources[0] = fragment_source;
|
|
|
|
sources[1] = fragment_debug;
|
|
|
|
sources[2] = fragment_brace;
|
|
|
|
count = 3;
|
|
|
|
} else {
|
|
|
|
sources[0] = fragment_source;
|
|
|
|
sources[1] = fragment_brace;
|
|
|
|
count = 2;
|
|
|
|
}
|
|
|
|
|
2012-09-06 06:06:26 +04:00
|
|
|
shader->fragment_shader =
|
2012-11-08 19:20:46 +04:00
|
|
|
compile_shader(GL_FRAGMENT_SHADER, count, sources);
|
2012-09-06 06:06:26 +04:00
|
|
|
|
|
|
|
shader->program = glCreateProgram();
|
|
|
|
glAttachShader(shader->program, shader->vertex_shader);
|
|
|
|
glAttachShader(shader->program, shader->fragment_shader);
|
|
|
|
glBindAttribLocation(shader->program, 0, "position");
|
|
|
|
glBindAttribLocation(shader->program, 1, "texcoord");
|
|
|
|
|
|
|
|
glLinkProgram(shader->program);
|
|
|
|
glGetProgramiv(shader->program, GL_LINK_STATUS, &status);
|
|
|
|
if (!status) {
|
|
|
|
glGetProgramInfoLog(shader->program, sizeof msg, NULL, msg);
|
|
|
|
weston_log("link info: %s\n", msg);
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
shader->proj_uniform = glGetUniformLocation(shader->program, "proj");
|
|
|
|
shader->tex_uniforms[0] = glGetUniformLocation(shader->program, "tex");
|
|
|
|
shader->tex_uniforms[1] = glGetUniformLocation(shader->program, "tex1");
|
|
|
|
shader->tex_uniforms[2] = glGetUniformLocation(shader->program, "tex2");
|
|
|
|
shader->alpha_uniform = glGetUniformLocation(shader->program, "alpha");
|
|
|
|
shader->color_uniform = glGetUniformLocation(shader->program, "color");
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2012-11-08 19:20:46 +04:00
|
|
|
static void
|
2012-11-13 22:10:29 +04:00
|
|
|
shader_release(struct gl_shader *shader)
|
2012-11-08 19:20:46 +04:00
|
|
|
{
|
|
|
|
glDeleteShader(shader->vertex_shader);
|
|
|
|
glDeleteShader(shader->fragment_shader);
|
|
|
|
glDeleteProgram(shader->program);
|
|
|
|
|
|
|
|
shader->vertex_shader = 0;
|
|
|
|
shader->fragment_shader = 0;
|
|
|
|
shader->program = 0;
|
|
|
|
}
|
|
|
|
|
2012-09-06 06:06:26 +04:00
|
|
|
static void
|
|
|
|
log_extensions(const char *name, const char *extensions)
|
|
|
|
{
|
|
|
|
const char *p, *end;
|
|
|
|
int l;
|
|
|
|
int len;
|
|
|
|
|
|
|
|
l = weston_log("%s:", name);
|
|
|
|
p = extensions;
|
|
|
|
while (*p) {
|
|
|
|
end = strchrnul(p, ' ');
|
|
|
|
len = end - p;
|
|
|
|
if (l + len > 78)
|
|
|
|
l = weston_log_continue("\n" STAMP_SPACE "%.*s",
|
|
|
|
len, p);
|
|
|
|
else
|
|
|
|
l += weston_log_continue(" %.*s", len, p);
|
|
|
|
for (p = end; isspace(*p); p++)
|
|
|
|
;
|
|
|
|
}
|
|
|
|
weston_log_continue("\n");
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
log_egl_gl_info(EGLDisplay egldpy)
|
|
|
|
{
|
|
|
|
const char *str;
|
|
|
|
|
|
|
|
str = eglQueryString(egldpy, EGL_VERSION);
|
|
|
|
weston_log("EGL version: %s\n", str ? str : "(null)");
|
|
|
|
|
|
|
|
str = eglQueryString(egldpy, EGL_VENDOR);
|
|
|
|
weston_log("EGL vendor: %s\n", str ? str : "(null)");
|
|
|
|
|
|
|
|
str = eglQueryString(egldpy, EGL_CLIENT_APIS);
|
|
|
|
weston_log("EGL client APIs: %s\n", str ? str : "(null)");
|
|
|
|
|
|
|
|
str = eglQueryString(egldpy, EGL_EXTENSIONS);
|
|
|
|
log_extensions("EGL extensions", str ? str : "(null)");
|
|
|
|
|
|
|
|
str = (char *)glGetString(GL_VERSION);
|
|
|
|
weston_log("GL version: %s\n", str ? str : "(null)");
|
|
|
|
|
|
|
|
str = (char *)glGetString(GL_SHADING_LANGUAGE_VERSION);
|
|
|
|
weston_log("GLSL version: %s\n", str ? str : "(null)");
|
|
|
|
|
|
|
|
str = (char *)glGetString(GL_VENDOR);
|
|
|
|
weston_log("GL vendor: %s\n", str ? str : "(null)");
|
|
|
|
|
|
|
|
str = (char *)glGetString(GL_RENDERER);
|
|
|
|
weston_log("GL renderer: %s\n", str ? str : "(null)");
|
|
|
|
|
|
|
|
str = (char *)glGetString(GL_EXTENSIONS);
|
|
|
|
log_extensions("GL extensions", str ? str : "(null)");
|
|
|
|
}
|
|
|
|
|
2012-10-24 10:43:05 +04:00
|
|
|
static void
|
|
|
|
log_egl_config_info(EGLDisplay egldpy, EGLConfig eglconfig)
|
|
|
|
{
|
|
|
|
EGLint r, g, b, a;
|
|
|
|
|
|
|
|
weston_log("Chosen EGL config details:\n");
|
|
|
|
|
|
|
|
weston_log_continue(STAMP_SPACE "RGBA bits");
|
|
|
|
if (eglGetConfigAttrib(egldpy, eglconfig, EGL_RED_SIZE, &r) &&
|
|
|
|
eglGetConfigAttrib(egldpy, eglconfig, EGL_GREEN_SIZE, &g) &&
|
|
|
|
eglGetConfigAttrib(egldpy, eglconfig, EGL_BLUE_SIZE, &b) &&
|
|
|
|
eglGetConfigAttrib(egldpy, eglconfig, EGL_ALPHA_SIZE, &a))
|
|
|
|
weston_log_continue(": %d %d %d %d\n", r, g, b, a);
|
|
|
|
else
|
|
|
|
weston_log_continue(" unknown\n");
|
|
|
|
|
|
|
|
weston_log_continue(STAMP_SPACE "swap interval range");
|
|
|
|
if (eglGetConfigAttrib(egldpy, eglconfig, EGL_MIN_SWAP_INTERVAL, &a) &&
|
|
|
|
eglGetConfigAttrib(egldpy, eglconfig, EGL_MAX_SWAP_INTERVAL, &b))
|
|
|
|
weston_log_continue(": %d - %d\n", a, b);
|
|
|
|
else
|
|
|
|
weston_log_continue(" unknown\n");
|
|
|
|
}
|
|
|
|
|
2015-05-15 20:12:39 +03:00
|
|
|
static int
|
|
|
|
match_config_to_visual(EGLDisplay egl_display,
|
|
|
|
EGLint visual_id,
|
|
|
|
EGLConfig *configs,
|
|
|
|
int count)
|
|
|
|
{
|
|
|
|
int i;
|
|
|
|
|
|
|
|
for (i = 0; i < count; ++i) {
|
|
|
|
EGLint id;
|
|
|
|
|
|
|
|
if (!eglGetConfigAttrib(egl_display,
|
|
|
|
configs[i], EGL_NATIVE_VISUAL_ID,
|
|
|
|
&id))
|
|
|
|
continue;
|
|
|
|
|
|
|
|
if (id == visual_id)
|
|
|
|
return i;
|
|
|
|
}
|
|
|
|
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
2014-03-07 22:05:50 +04:00
|
|
|
static int
|
|
|
|
egl_choose_config(struct gl_renderer *gr, const EGLint *attribs,
|
2015-05-15 20:12:39 +03:00
|
|
|
const EGLint *visual_id, const int n_ids,
|
2014-03-07 22:05:50 +04:00
|
|
|
EGLConfig *config_out)
|
|
|
|
{
|
|
|
|
EGLint count = 0;
|
|
|
|
EGLint matched = 0;
|
|
|
|
EGLConfig *configs;
|
2015-05-15 20:12:39 +03:00
|
|
|
int i, config_index = -1;
|
2014-03-07 22:05:50 +04:00
|
|
|
|
2015-05-20 22:57:58 +03:00
|
|
|
if (!eglGetConfigs(gr->egl_display, NULL, 0, &count) || count < 1) {
|
|
|
|
weston_log("No EGL configs to choose from.\n");
|
2014-03-07 22:05:50 +04:00
|
|
|
return -1;
|
2015-05-20 22:57:58 +03:00
|
|
|
}
|
2014-03-07 22:05:50 +04:00
|
|
|
configs = calloc(count, sizeof *configs);
|
|
|
|
if (!configs)
|
|
|
|
return -1;
|
|
|
|
|
|
|
|
if (!eglChooseConfig(gr->egl_display, attribs, configs,
|
2015-05-20 22:57:58 +03:00
|
|
|
count, &matched) || !matched) {
|
|
|
|
weston_log("No EGL configs with appropriate attributes.\n");
|
2014-03-07 22:05:50 +04:00
|
|
|
goto out;
|
2015-05-20 22:57:58 +03:00
|
|
|
}
|
2014-03-07 22:05:50 +04:00
|
|
|
|
2015-05-15 20:12:39 +03:00
|
|
|
if (!visual_id)
|
|
|
|
config_index = 0;
|
2014-03-07 22:05:50 +04:00
|
|
|
|
2015-05-15 20:12:39 +03:00
|
|
|
for (i = 0; config_index == -1 && i < n_ids; i++)
|
|
|
|
config_index = match_config_to_visual(gr->egl_display,
|
|
|
|
visual_id[i],
|
|
|
|
configs,
|
|
|
|
matched);
|
2014-03-07 22:05:50 +04:00
|
|
|
|
2015-05-15 20:12:39 +03:00
|
|
|
if (config_index != -1)
|
|
|
|
*config_out = configs[config_index];
|
2014-03-07 22:05:50 +04:00
|
|
|
|
|
|
|
out:
|
|
|
|
free(configs);
|
2015-05-15 20:12:39 +03:00
|
|
|
if (config_index == -1)
|
|
|
|
return -1;
|
|
|
|
|
2015-05-20 22:57:58 +03:00
|
|
|
if (i > 1)
|
|
|
|
weston_log("Unable to use first choice EGL config with id"
|
|
|
|
" 0x%x, succeeded with alternate id 0x%x.\n",
|
|
|
|
visual_id[0], visual_id[i - 1]);
|
2015-05-15 20:12:39 +03:00
|
|
|
return 0;
|
2014-03-07 22:05:50 +04:00
|
|
|
}
|
|
|
|
|
2013-10-28 07:24:54 +04:00
|
|
|
static void
|
|
|
|
gl_renderer_output_set_border(struct weston_output *output,
|
|
|
|
enum gl_renderer_border_side side,
|
|
|
|
int32_t width, int32_t height,
|
|
|
|
int32_t tex_width, unsigned char *data)
|
|
|
|
{
|
|
|
|
struct gl_output_state *go = get_output_state(output);
|
|
|
|
|
2014-02-05 07:36:38 +04:00
|
|
|
if (go->borders[side].width != width ||
|
|
|
|
go->borders[side].height != height)
|
|
|
|
/* In this case, we have to blow everything and do a full
|
|
|
|
* repaint. */
|
|
|
|
go->border_status |= BORDER_SIZE_CHANGED | BORDER_ALL_DIRTY;
|
|
|
|
|
|
|
|
if (data == NULL) {
|
|
|
|
width = 0;
|
|
|
|
height = 0;
|
|
|
|
}
|
|
|
|
|
2013-10-28 07:24:54 +04:00
|
|
|
go->borders[side].width = width;
|
|
|
|
go->borders[side].height = height;
|
|
|
|
go->borders[side].tex_width = tex_width;
|
|
|
|
go->borders[side].data = data;
|
2014-02-05 07:36:38 +04:00
|
|
|
go->border_status |= 1 << side;
|
2013-10-28 07:24:54 +04:00
|
|
|
}
|
|
|
|
|
2012-11-13 22:10:18 +04:00
|
|
|
static int
|
2012-11-13 22:10:29 +04:00
|
|
|
gl_renderer_setup(struct weston_compositor *ec, EGLSurface egl_surface);
|
2012-11-13 22:10:18 +04:00
|
|
|
|
2013-10-14 16:57:11 +04:00
|
|
|
static int
|
2012-11-13 22:10:29 +04:00
|
|
|
gl_renderer_output_create(struct weston_output *output,
|
2015-03-20 17:26:52 +03:00
|
|
|
EGLNativeWindowType window_for_legacy,
|
|
|
|
void *window_for_platform,
|
2014-03-07 22:05:50 +04:00
|
|
|
const EGLint *attribs,
|
2015-05-15 20:12:39 +03:00
|
|
|
const EGLint *visual_id,
|
|
|
|
int n_ids)
|
2012-11-13 22:10:18 +04:00
|
|
|
{
|
|
|
|
struct weston_compositor *ec = output->compositor;
|
2012-11-13 22:10:29 +04:00
|
|
|
struct gl_renderer *gr = get_renderer(ec);
|
2014-03-07 22:05:50 +04:00
|
|
|
struct gl_output_state *go;
|
|
|
|
EGLConfig egl_config;
|
2012-11-16 19:23:52 +04:00
|
|
|
int i;
|
2012-11-13 22:10:18 +04:00
|
|
|
|
2015-05-15 20:12:39 +03:00
|
|
|
if (egl_choose_config(gr, attribs, visual_id,
|
|
|
|
n_ids, &egl_config) == -1) {
|
2014-03-07 22:05:50 +04:00
|
|
|
weston_log("failed to choose EGL config for output\n");
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (egl_config != gr->egl_config &&
|
|
|
|
!gr->has_configless_context) {
|
|
|
|
weston_log("attempted to use a different EGL config for an "
|
|
|
|
"output but EGL_MESA_configless_context is not "
|
|
|
|
"supported\n");
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
2014-11-21 09:21:57 +03:00
|
|
|
go = zalloc(sizeof *go);
|
|
|
|
if (go == NULL)
|
2012-11-13 22:10:18 +04:00
|
|
|
return -1;
|
|
|
|
|
2015-03-20 17:26:52 +03:00
|
|
|
if (gr->create_platform_window) {
|
|
|
|
go->egl_surface =
|
|
|
|
gr->create_platform_window(gr->egl_display,
|
|
|
|
egl_config,
|
|
|
|
window_for_platform,
|
|
|
|
NULL);
|
2015-03-24 15:12:05 +03:00
|
|
|
} else {
|
2015-03-20 17:26:52 +03:00
|
|
|
go->egl_surface =
|
|
|
|
eglCreateWindowSurface(gr->egl_display,
|
|
|
|
egl_config,
|
|
|
|
window_for_legacy, NULL);
|
2015-03-24 15:12:05 +03:00
|
|
|
}
|
2012-11-13 22:10:18 +04:00
|
|
|
|
|
|
|
if (go->egl_surface == EGL_NO_SURFACE) {
|
|
|
|
weston_log("failed to create egl surface\n");
|
|
|
|
free(go);
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
2012-11-13 22:10:19 +04:00
|
|
|
if (gr->egl_context == NULL)
|
2012-11-13 22:10:29 +04:00
|
|
|
if (gl_renderer_setup(ec, go->egl_surface) < 0) {
|
2012-11-13 22:10:18 +04:00
|
|
|
free(go);
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
2013-03-05 19:30:30 +04:00
|
|
|
for (i = 0; i < BUFFER_DAMAGE_COUNT; i++)
|
2012-11-16 19:23:52 +04:00
|
|
|
pixman_region32_init(&go->buffer_damage[i]);
|
|
|
|
|
2012-11-13 22:10:18 +04:00
|
|
|
output->renderer_state = go;
|
|
|
|
|
2014-03-07 22:05:50 +04:00
|
|
|
log_egl_config_info(gr->egl_display, egl_config);
|
|
|
|
|
2012-11-13 22:10:18 +04:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2013-10-14 16:57:11 +04:00
|
|
|
static void
|
2012-11-13 22:10:29 +04:00
|
|
|
gl_renderer_output_destroy(struct weston_output *output)
|
2012-11-13 22:10:18 +04:00
|
|
|
{
|
2012-11-13 22:10:29 +04:00
|
|
|
struct gl_renderer *gr = get_renderer(output->compositor);
|
|
|
|
struct gl_output_state *go = get_output_state(output);
|
2012-11-16 19:23:52 +04:00
|
|
|
int i;
|
|
|
|
|
|
|
|
for (i = 0; i < 2; i++)
|
|
|
|
pixman_region32_fini(&go->buffer_damage[i]);
|
2012-11-13 22:10:18 +04:00
|
|
|
|
2012-11-13 22:10:19 +04:00
|
|
|
eglDestroySurface(gr->egl_display, go->egl_surface);
|
2012-11-13 22:10:18 +04:00
|
|
|
|
|
|
|
free(go);
|
|
|
|
}
|
|
|
|
|
2013-10-14 16:57:11 +04:00
|
|
|
static EGLSurface
|
2012-11-13 22:10:29 +04:00
|
|
|
gl_renderer_output_surface(struct weston_output *output)
|
2012-11-13 22:10:18 +04:00
|
|
|
{
|
|
|
|
return get_output_state(output)->egl_surface;
|
|
|
|
}
|
|
|
|
|
2013-01-08 20:09:01 +04:00
|
|
|
static void
|
2012-11-13 22:10:29 +04:00
|
|
|
gl_renderer_destroy(struct weston_compositor *ec)
|
2012-09-07 05:44:24 +04:00
|
|
|
{
|
2012-11-13 22:10:29 +04:00
|
|
|
struct gl_renderer *gr = get_renderer(ec);
|
2012-11-13 22:10:19 +04:00
|
|
|
|
2013-10-25 17:26:34 +04:00
|
|
|
wl_signal_emit(&gr->destroy_signal, gr);
|
|
|
|
|
2012-11-13 22:10:27 +04:00
|
|
|
if (gr->has_bind_display)
|
|
|
|
gr->unbind_display(gr->egl_display, ec->wl_display);
|
2012-11-13 22:10:19 +04:00
|
|
|
|
|
|
|
/* Work around crash in egl_dri2.c's dri2_make_current() - when does this apply? */
|
|
|
|
eglMakeCurrent(gr->egl_display,
|
|
|
|
EGL_NO_SURFACE, EGL_NO_SURFACE,
|
|
|
|
EGL_NO_CONTEXT);
|
|
|
|
|
|
|
|
eglTerminate(gr->egl_display);
|
|
|
|
eglReleaseThread();
|
2013-03-07 21:15:17 +04:00
|
|
|
|
2013-05-09 06:38:05 +04:00
|
|
|
wl_array_release(&gr->vertices);
|
|
|
|
wl_array_release(&gr->vtxcnt);
|
|
|
|
|
2014-02-08 23:11:24 +04:00
|
|
|
if (gr->fragment_binding)
|
|
|
|
weston_binding_destroy(gr->fragment_binding);
|
|
|
|
if (gr->fan_binding)
|
|
|
|
weston_binding_destroy(gr->fan_binding);
|
2013-10-25 17:26:32 +04:00
|
|
|
|
2013-03-07 21:15:17 +04:00
|
|
|
free(gr);
|
2012-11-13 22:10:19 +04:00
|
|
|
}
|
|
|
|
|
2015-04-08 17:02:22 +03:00
|
|
|
static void
|
|
|
|
renderer_setup_egl_client_extensions(struct gl_renderer *gr)
|
|
|
|
{
|
|
|
|
const char *extensions;
|
|
|
|
|
|
|
|
extensions = eglQueryString(EGL_NO_DISPLAY, EGL_EXTENSIONS);
|
|
|
|
if (!extensions) {
|
|
|
|
weston_log("Retrieving EGL client extension string failed.\n");
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (strstr(extensions, "EGL_EXT_platform_base"))
|
|
|
|
gr->create_platform_window =
|
|
|
|
(void *) eglGetProcAddress("eglCreatePlatformWindowSurfaceEXT");
|
|
|
|
else
|
|
|
|
weston_log("warning: EGL_EXT_platform_base not supported.\n");
|
|
|
|
}
|
|
|
|
|
2014-03-07 22:05:49 +04:00
|
|
|
static int
|
|
|
|
gl_renderer_setup_egl_extensions(struct weston_compositor *ec)
|
|
|
|
{
|
|
|
|
struct gl_renderer *gr = get_renderer(ec);
|
|
|
|
const char *extensions;
|
|
|
|
EGLBoolean ret;
|
|
|
|
|
|
|
|
gr->create_image = (void *) eglGetProcAddress("eglCreateImageKHR");
|
|
|
|
gr->destroy_image = (void *) eglGetProcAddress("eglDestroyImageKHR");
|
|
|
|
gr->bind_display =
|
|
|
|
(void *) eglGetProcAddress("eglBindWaylandDisplayWL");
|
|
|
|
gr->unbind_display =
|
|
|
|
(void *) eglGetProcAddress("eglUnbindWaylandDisplayWL");
|
|
|
|
gr->query_buffer =
|
|
|
|
(void *) eglGetProcAddress("eglQueryWaylandBufferWL");
|
|
|
|
|
|
|
|
extensions =
|
|
|
|
(const char *) eglQueryString(gr->egl_display, EGL_EXTENSIONS);
|
|
|
|
if (!extensions) {
|
|
|
|
weston_log("Retrieving EGL extension string failed.\n");
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (strstr(extensions, "EGL_WL_bind_wayland_display"))
|
|
|
|
gr->has_bind_display = 1;
|
|
|
|
if (gr->has_bind_display) {
|
|
|
|
ret = gr->bind_display(gr->egl_display, ec->wl_display);
|
|
|
|
if (!ret)
|
|
|
|
gr->has_bind_display = 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (strstr(extensions, "EGL_EXT_buffer_age"))
|
|
|
|
gr->has_egl_buffer_age = 1;
|
|
|
|
else
|
|
|
|
weston_log("warning: EGL_EXT_buffer_age not supported. "
|
|
|
|
"Performance could be affected.\n");
|
|
|
|
|
|
|
|
#ifdef EGL_EXT_swap_buffers_with_damage
|
|
|
|
if (strstr(extensions, "EGL_EXT_swap_buffers_with_damage"))
|
|
|
|
gr->swap_buffers_with_damage =
|
|
|
|
(void *) eglGetProcAddress("eglSwapBuffersWithDamageEXT");
|
|
|
|
else
|
|
|
|
weston_log("warning: EGL_EXT_swap_buffers_with_damage not "
|
|
|
|
"supported. Performance could be affected.\n");
|
|
|
|
#endif
|
|
|
|
|
2014-03-07 22:05:50 +04:00
|
|
|
#ifdef EGL_MESA_configless_context
|
|
|
|
if (strstr(extensions, "EGL_MESA_configless_context"))
|
|
|
|
gr->has_configless_context = 1;
|
|
|
|
#endif
|
|
|
|
|
2015-04-08 17:02:22 +03:00
|
|
|
renderer_setup_egl_client_extensions(gr);
|
|
|
|
|
2014-03-07 22:05:49 +04:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2013-10-14 16:57:11 +04:00
|
|
|
static const EGLint gl_renderer_opaque_attribs[] = {
|
2012-11-13 22:10:19 +04:00
|
|
|
EGL_SURFACE_TYPE, EGL_WINDOW_BIT,
|
|
|
|
EGL_RED_SIZE, 1,
|
|
|
|
EGL_GREEN_SIZE, 1,
|
|
|
|
EGL_BLUE_SIZE, 1,
|
|
|
|
EGL_ALPHA_SIZE, 0,
|
|
|
|
EGL_RENDERABLE_TYPE, EGL_OPENGL_ES2_BIT,
|
|
|
|
EGL_NONE
|
|
|
|
};
|
|
|
|
|
2013-10-14 16:57:11 +04:00
|
|
|
static const EGLint gl_renderer_alpha_attribs[] = {
|
2012-11-13 22:10:19 +04:00
|
|
|
EGL_SURFACE_TYPE, EGL_WINDOW_BIT,
|
|
|
|
EGL_RED_SIZE, 1,
|
|
|
|
EGL_GREEN_SIZE, 1,
|
|
|
|
EGL_BLUE_SIZE, 1,
|
|
|
|
EGL_ALPHA_SIZE, 1,
|
|
|
|
EGL_RENDERABLE_TYPE, EGL_OPENGL_ES2_BIT,
|
|
|
|
EGL_NONE
|
|
|
|
};
|
|
|
|
|
2015-03-20 17:26:50 +03:00
|
|
|
/** Checks whether a platform EGL client extension is supported
|
|
|
|
*
|
|
|
|
* \param ec The weston compositor
|
|
|
|
* \param extension_suffix The EGL client extension suffix
|
|
|
|
* \return 1 if supported, 0 if using fallbacks, -1 unsupported
|
|
|
|
*
|
|
|
|
* This function checks whether a specific platform_* extension is supported
|
|
|
|
* by EGL.
|
|
|
|
*
|
|
|
|
* The extension suffix should be the suffix of the platform extension (that
|
|
|
|
* specifies a <platform> argument as defined in EGL_EXT_platform_base). For
|
|
|
|
* example, passing "foo" will check whether either "EGL_KHR_platform_foo",
|
|
|
|
* "EGL_EXT_platform_foo", or "EGL_MESA_platform_foo" is supported.
|
|
|
|
*
|
|
|
|
* The return value is 1:
|
|
|
|
* - if the supplied EGL client extension is supported.
|
|
|
|
* The return value is 0:
|
|
|
|
* - if the platform_base client extension isn't supported so will
|
|
|
|
* fallback to eglGetDisplay and friends.
|
|
|
|
* The return value is -1:
|
|
|
|
* - if the supplied EGL client extension is not supported.
|
|
|
|
*/
|
2013-10-14 16:57:11 +04:00
|
|
|
static int
|
2015-03-20 17:26:50 +03:00
|
|
|
gl_renderer_supports(struct weston_compositor *ec,
|
|
|
|
const char *extension_suffix)
|
|
|
|
{
|
|
|
|
static const char *extensions = NULL;
|
|
|
|
char s[64];
|
|
|
|
|
|
|
|
if (!extensions) {
|
|
|
|
extensions = (const char *) eglQueryString(
|
|
|
|
EGL_NO_DISPLAY, EGL_EXTENSIONS);
|
|
|
|
|
|
|
|
if (!extensions)
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
log_extensions("EGL client extensions",
|
|
|
|
extensions);
|
|
|
|
}
|
|
|
|
|
2015-04-08 17:02:21 +03:00
|
|
|
if (!strstr(extensions, "EGL_EXT_platform_base"))
|
|
|
|
return 0;
|
|
|
|
|
2015-03-20 17:26:50 +03:00
|
|
|
snprintf(s, sizeof s, "EGL_KHR_platform_%s", extension_suffix);
|
|
|
|
if (strstr(extensions, s))
|
|
|
|
return 1;
|
|
|
|
|
|
|
|
snprintf(s, sizeof s, "EGL_EXT_platform_%s", extension_suffix);
|
|
|
|
if (strstr(extensions, s))
|
|
|
|
return 1;
|
|
|
|
|
|
|
|
snprintf(s, sizeof s, "EGL_MESA_platform_%s", extension_suffix);
|
|
|
|
if (strstr(extensions, s))
|
|
|
|
return 1;
|
|
|
|
|
2015-04-08 17:02:21 +03:00
|
|
|
/* at this point we definitely have some platform extensions but
|
|
|
|
* haven't found the supplied platform, so chances are it's
|
2015-03-20 17:26:50 +03:00
|
|
|
* not supported. */
|
|
|
|
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
2015-03-24 15:12:04 +03:00
|
|
|
static const char *
|
|
|
|
platform_to_extension(EGLenum platform)
|
|
|
|
{
|
|
|
|
switch (platform) {
|
|
|
|
case EGL_PLATFORM_GBM_KHR:
|
|
|
|
return "gbm";
|
|
|
|
case EGL_PLATFORM_WAYLAND_KHR:
|
|
|
|
return "wayland";
|
|
|
|
case EGL_PLATFORM_X11_KHR:
|
|
|
|
return "x11";
|
|
|
|
default:
|
|
|
|
assert(0 && "bad EGL platform enum");
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-03-20 17:26:50 +03:00
|
|
|
static int
|
|
|
|
gl_renderer_create(struct weston_compositor *ec, EGLenum platform,
|
|
|
|
void *native_window, const EGLint *attribs,
|
2015-05-15 20:12:39 +03:00
|
|
|
const EGLint *visual_id, int n_ids)
|
2012-11-13 22:10:19 +04:00
|
|
|
{
|
2012-11-13 22:10:29 +04:00
|
|
|
struct gl_renderer *gr;
|
2012-11-13 22:10:19 +04:00
|
|
|
EGLint major, minor;
|
2015-03-24 15:12:04 +03:00
|
|
|
int supports = 0;
|
|
|
|
|
|
|
|
if (platform) {
|
|
|
|
supports = gl_renderer_supports(
|
|
|
|
ec, platform_to_extension(platform));
|
|
|
|
if (supports < 0)
|
|
|
|
return -1;
|
|
|
|
}
|
2012-11-13 22:10:19 +04:00
|
|
|
|
2014-11-21 09:21:57 +03:00
|
|
|
gr = zalloc(sizeof *gr);
|
2012-11-13 22:10:19 +04:00
|
|
|
if (gr == NULL)
|
|
|
|
return -1;
|
|
|
|
|
2012-11-13 22:10:29 +04:00
|
|
|
gr->base.read_pixels = gl_renderer_read_pixels;
|
|
|
|
gr->base.repaint_output = gl_renderer_repaint_output;
|
|
|
|
gr->base.flush_damage = gl_renderer_flush_damage;
|
|
|
|
gr->base.attach = gl_renderer_attach;
|
|
|
|
gr->base.surface_set_color = gl_renderer_surface_set_color;
|
2013-01-08 20:09:01 +04:00
|
|
|
gr->base.destroy = gl_renderer_destroy;
|
2015-02-09 14:37:27 +03:00
|
|
|
gr->base.surface_get_content_size =
|
|
|
|
gl_renderer_surface_get_content_size;
|
|
|
|
gr->base.surface_copy_content = gl_renderer_surface_copy_content;
|
2015-03-24 15:12:04 +03:00
|
|
|
gr->egl_display = NULL;
|
2012-11-13 22:10:19 +04:00
|
|
|
|
2015-03-24 15:12:05 +03:00
|
|
|
/* extension_suffix is supported */
|
2015-03-24 15:12:04 +03:00
|
|
|
if (supports) {
|
|
|
|
if (!get_platform_display) {
|
|
|
|
get_platform_display = (void *) eglGetProcAddress(
|
|
|
|
"eglGetPlatformDisplayEXT");
|
|
|
|
}
|
2015-03-20 17:26:50 +03:00
|
|
|
|
2015-03-24 15:12:04 +03:00
|
|
|
/* also wrap this in the supports check because
|
|
|
|
* eglGetProcAddress can return non-NULL and still not
|
|
|
|
* support the feature at runtime, so ensure the
|
|
|
|
* appropriate extension checks have been done. */
|
|
|
|
if (get_platform_display && platform) {
|
|
|
|
gr->egl_display = get_platform_display(platform,
|
|
|
|
native_window,
|
|
|
|
NULL);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!gr->egl_display) {
|
2015-03-20 17:26:50 +03:00
|
|
|
weston_log("warning: either no EGL_EXT_platform_base "
|
|
|
|
"support or specific platform support; "
|
|
|
|
"falling back to eglGetDisplay.\n");
|
|
|
|
gr->egl_display = eglGetDisplay(native_window);
|
|
|
|
}
|
|
|
|
|
2012-11-13 22:10:19 +04:00
|
|
|
if (gr->egl_display == EGL_NO_DISPLAY) {
|
|
|
|
weston_log("failed to create display\n");
|
2015-06-11 20:14:45 +03:00
|
|
|
goto fail;
|
2012-11-13 22:10:19 +04:00
|
|
|
}
|
|
|
|
|
|
|
|
if (!eglInitialize(gr->egl_display, &major, &minor)) {
|
|
|
|
weston_log("failed to initialize display\n");
|
2015-06-11 20:14:45 +03:00
|
|
|
goto fail_with_error;
|
2012-11-13 22:10:19 +04:00
|
|
|
}
|
|
|
|
|
2015-05-15 20:12:39 +03:00
|
|
|
if (egl_choose_config(gr, attribs, visual_id,
|
|
|
|
n_ids, &gr->egl_config) < 0) {
|
2012-11-13 22:10:19 +04:00
|
|
|
weston_log("failed to choose EGL config\n");
|
2015-06-11 20:14:45 +03:00
|
|
|
goto fail;
|
2012-11-13 22:10:19 +04:00
|
|
|
}
|
|
|
|
|
|
|
|
ec->renderer = &gr->base;
|
2013-05-22 19:03:04 +04:00
|
|
|
ec->capabilities |= WESTON_CAP_ROTATION_ANY;
|
2013-05-22 19:03:05 +04:00
|
|
|
ec->capabilities |= WESTON_CAP_CAPTURE_YFLIP;
|
2015-02-18 10:48:59 +03:00
|
|
|
ec->capabilities |= WESTON_CAP_VIEW_CLIP_MASK;
|
2012-11-13 22:10:19 +04:00
|
|
|
|
2014-03-07 22:05:49 +04:00
|
|
|
if (gl_renderer_setup_egl_extensions(ec) < 0)
|
2015-06-11 20:14:45 +03:00
|
|
|
goto fail_with_error;
|
2014-03-07 22:05:49 +04:00
|
|
|
|
2013-08-06 22:05:55 +04:00
|
|
|
wl_display_add_shm_format(ec->wl_display, WL_SHM_FORMAT_RGB565);
|
|
|
|
|
2013-10-25 17:26:34 +04:00
|
|
|
wl_signal_init(&gr->destroy_signal);
|
|
|
|
|
2012-11-13 22:10:19 +04:00
|
|
|
return 0;
|
|
|
|
|
2015-06-11 20:14:45 +03:00
|
|
|
fail_with_error:
|
2012-11-27 14:25:25 +04:00
|
|
|
gl_renderer_print_egl_error_state();
|
2015-06-11 20:14:45 +03:00
|
|
|
fail:
|
2012-11-13 22:10:19 +04:00
|
|
|
free(gr);
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
2013-10-14 16:57:11 +04:00
|
|
|
static EGLDisplay
|
2012-11-13 22:10:29 +04:00
|
|
|
gl_renderer_display(struct weston_compositor *ec)
|
2012-11-13 22:10:19 +04:00
|
|
|
{
|
|
|
|
return get_renderer(ec)->egl_display;
|
2012-09-07 05:44:24 +04:00
|
|
|
}
|
|
|
|
|
2012-11-08 19:20:46 +04:00
|
|
|
static int
|
|
|
|
compile_shaders(struct weston_compositor *ec)
|
|
|
|
{
|
2012-11-13 22:10:29 +04:00
|
|
|
struct gl_renderer *gr = get_renderer(ec);
|
2012-11-13 22:10:25 +04:00
|
|
|
|
2013-05-17 15:00:40 +04:00
|
|
|
gr->texture_shader_rgba.vertex_source = vertex_shader;
|
|
|
|
gr->texture_shader_rgba.fragment_source = texture_fragment_shader_rgba;
|
|
|
|
|
|
|
|
gr->texture_shader_rgbx.vertex_source = vertex_shader;
|
|
|
|
gr->texture_shader_rgbx.fragment_source = texture_fragment_shader_rgbx;
|
|
|
|
|
|
|
|
gr->texture_shader_egl_external.vertex_source = vertex_shader;
|
|
|
|
gr->texture_shader_egl_external.fragment_source =
|
|
|
|
texture_fragment_shader_egl_external;
|
|
|
|
|
|
|
|
gr->texture_shader_y_uv.vertex_source = vertex_shader;
|
|
|
|
gr->texture_shader_y_uv.fragment_source = texture_fragment_shader_y_uv;
|
|
|
|
|
|
|
|
gr->texture_shader_y_u_v.vertex_source = vertex_shader;
|
|
|
|
gr->texture_shader_y_u_v.fragment_source =
|
|
|
|
texture_fragment_shader_y_u_v;
|
|
|
|
|
2013-11-27 19:43:51 +04:00
|
|
|
gr->texture_shader_y_xuxv.vertex_source = vertex_shader;
|
2013-05-17 15:00:40 +04:00
|
|
|
gr->texture_shader_y_xuxv.fragment_source =
|
|
|
|
texture_fragment_shader_y_xuxv;
|
|
|
|
|
|
|
|
gr->solid_shader.vertex_source = vertex_shader;
|
|
|
|
gr->solid_shader.fragment_source = solid_fragment_shader;
|
2012-11-08 19:20:46 +04:00
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
2015-07-15 21:00:36 +03:00
|
|
|
fragment_debug_binding(struct weston_keyboard *keyboard, uint32_t time,
|
|
|
|
uint32_t key, void *data)
|
2012-11-08 19:20:46 +04:00
|
|
|
{
|
|
|
|
struct weston_compositor *ec = data;
|
2012-11-13 22:10:29 +04:00
|
|
|
struct gl_renderer *gr = get_renderer(ec);
|
2012-11-08 19:20:46 +04:00
|
|
|
struct weston_output *output;
|
|
|
|
|
2012-11-13 22:10:25 +04:00
|
|
|
gr->fragment_shader_debug ^= 1;
|
2012-11-08 19:20:46 +04:00
|
|
|
|
2012-11-13 22:10:25 +04:00
|
|
|
shader_release(&gr->texture_shader_rgba);
|
|
|
|
shader_release(&gr->texture_shader_rgbx);
|
|
|
|
shader_release(&gr->texture_shader_egl_external);
|
|
|
|
shader_release(&gr->texture_shader_y_uv);
|
|
|
|
shader_release(&gr->texture_shader_y_u_v);
|
|
|
|
shader_release(&gr->texture_shader_y_xuxv);
|
|
|
|
shader_release(&gr->solid_shader);
|
2012-11-08 19:20:46 +04:00
|
|
|
|
2012-12-03 19:08:11 +04:00
|
|
|
/* Force use_shader() to call glUseProgram(), since we need to use
|
|
|
|
* the recompiled version of the shader. */
|
|
|
|
gr->current_shader = NULL;
|
|
|
|
|
2012-11-08 19:20:46 +04:00
|
|
|
wl_list_for_each(output, &ec->output_list, link)
|
|
|
|
weston_output_damage(output);
|
|
|
|
}
|
|
|
|
|
2013-05-07 18:50:09 +04:00
|
|
|
static void
|
2015-07-15 21:00:36 +03:00
|
|
|
fan_debug_repaint_binding(struct weston_keyboard *keyboard, uint32_t time,
|
|
|
|
uint32_t key, void *data)
|
2013-05-07 18:50:09 +04:00
|
|
|
{
|
|
|
|
struct weston_compositor *compositor = data;
|
|
|
|
struct gl_renderer *gr = get_renderer(compositor);
|
|
|
|
|
|
|
|
gr->fan_debug = !gr->fan_debug;
|
|
|
|
weston_compositor_damage_all(compositor);
|
|
|
|
}
|
|
|
|
|
2012-11-13 22:10:18 +04:00
|
|
|
static int
|
2012-11-13 22:10:29 +04:00
|
|
|
gl_renderer_setup(struct weston_compositor *ec, EGLSurface egl_surface)
|
2012-09-06 06:06:26 +04:00
|
|
|
{
|
2012-11-13 22:10:29 +04:00
|
|
|
struct gl_renderer *gr = get_renderer(ec);
|
2012-09-06 06:06:26 +04:00
|
|
|
const char *extensions;
|
2014-03-07 22:05:50 +04:00
|
|
|
EGLConfig context_config;
|
2012-09-07 04:51:00 +04:00
|
|
|
EGLBoolean ret;
|
2012-09-06 06:06:26 +04:00
|
|
|
|
2012-09-07 05:07:40 +04:00
|
|
|
static const EGLint context_attribs[] = {
|
|
|
|
EGL_CONTEXT_CLIENT_VERSION, 2,
|
|
|
|
EGL_NONE
|
|
|
|
};
|
|
|
|
|
|
|
|
if (!eglBindAPI(EGL_OPENGL_ES_API)) {
|
|
|
|
weston_log("failed to bind EGL_OPENGL_ES_API\n");
|
2012-11-27 14:25:25 +04:00
|
|
|
gl_renderer_print_egl_error_state();
|
2012-09-07 05:07:40 +04:00
|
|
|
return -1;
|
|
|
|
}
|
2012-10-24 10:43:05 +04:00
|
|
|
|
2014-03-07 22:05:50 +04:00
|
|
|
context_config = gr->egl_config;
|
|
|
|
|
|
|
|
#ifdef EGL_MESA_configless_context
|
|
|
|
if (gr->has_configless_context)
|
|
|
|
context_config = EGL_NO_CONFIG_MESA;
|
|
|
|
#endif
|
2012-10-24 10:43:05 +04:00
|
|
|
|
2014-03-07 22:05:50 +04:00
|
|
|
gr->egl_context = eglCreateContext(gr->egl_display, context_config,
|
2012-09-07 05:07:40 +04:00
|
|
|
EGL_NO_CONTEXT, context_attribs);
|
2012-11-13 22:10:19 +04:00
|
|
|
if (gr->egl_context == NULL) {
|
2012-09-07 05:07:40 +04:00
|
|
|
weston_log("failed to create context\n");
|
2012-11-27 14:25:25 +04:00
|
|
|
gl_renderer_print_egl_error_state();
|
2012-09-07 05:07:40 +04:00
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
2012-11-13 22:10:19 +04:00
|
|
|
ret = eglMakeCurrent(gr->egl_display, egl_surface,
|
|
|
|
egl_surface, gr->egl_context);
|
2012-09-07 04:51:00 +04:00
|
|
|
if (ret == EGL_FALSE) {
|
|
|
|
weston_log("Failed to make EGL context current.\n");
|
2012-11-27 14:25:25 +04:00
|
|
|
gl_renderer_print_egl_error_state();
|
2012-09-07 04:51:00 +04:00
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
2012-11-13 22:10:19 +04:00
|
|
|
log_egl_gl_info(gr->egl_display);
|
2012-09-06 06:06:26 +04:00
|
|
|
|
2012-11-13 22:10:27 +04:00
|
|
|
gr->image_target_texture_2d =
|
2012-09-06 06:06:26 +04:00
|
|
|
(void *) eglGetProcAddress("glEGLImageTargetTexture2DOES");
|
|
|
|
|
|
|
|
extensions = (const char *) glGetString(GL_EXTENSIONS);
|
|
|
|
if (!extensions) {
|
|
|
|
weston_log("Retrieving GL extension string failed.\n");
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!strstr(extensions, "GL_EXT_texture_format_BGRA8888")) {
|
|
|
|
weston_log("GL_EXT_texture_format_BGRA8888 not available\n");
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (strstr(extensions, "GL_EXT_read_format_bgra"))
|
2012-11-13 22:10:22 +04:00
|
|
|
ec->read_format = PIXMAN_a8r8g8b8;
|
2012-09-06 06:06:26 +04:00
|
|
|
else
|
2012-11-13 22:10:22 +04:00
|
|
|
ec->read_format = PIXMAN_a8b8g8r8;
|
2012-09-06 06:06:26 +04:00
|
|
|
|
2013-08-07 23:11:27 +04:00
|
|
|
#ifdef GL_EXT_unpack_subimage
|
2012-09-06 06:06:26 +04:00
|
|
|
if (strstr(extensions, "GL_EXT_unpack_subimage"))
|
2012-11-13 22:10:27 +04:00
|
|
|
gr->has_unpack_subimage = 1;
|
2013-08-07 23:11:27 +04:00
|
|
|
#endif
|
2012-09-06 06:06:26 +04:00
|
|
|
|
|
|
|
if (strstr(extensions, "GL_OES_EGL_image_external"))
|
2012-11-13 22:10:27 +04:00
|
|
|
gr->has_egl_image_external = 1;
|
2012-09-06 06:06:26 +04:00
|
|
|
|
|
|
|
glActiveTexture(GL_TEXTURE0);
|
|
|
|
|
2012-11-08 19:20:46 +04:00
|
|
|
if (compile_shaders(ec))
|
|
|
|
return -1;
|
|
|
|
|
2013-10-25 17:26:32 +04:00
|
|
|
gr->fragment_binding =
|
|
|
|
weston_compositor_add_debug_binding(ec, KEY_S,
|
|
|
|
fragment_debug_binding,
|
|
|
|
ec);
|
|
|
|
gr->fan_binding =
|
|
|
|
weston_compositor_add_debug_binding(ec, KEY_F,
|
|
|
|
fan_debug_repaint_binding,
|
|
|
|
ec);
|
2012-11-08 19:20:46 +04:00
|
|
|
|
2012-10-24 10:43:06 +04:00
|
|
|
weston_log("GL ES 2 renderer features:\n");
|
|
|
|
weston_log_continue(STAMP_SPACE "read-back format: %s\n",
|
2013-01-10 18:50:42 +04:00
|
|
|
ec->read_format == PIXMAN_a8r8g8b8 ? "BGRA" : "RGBA");
|
2012-10-24 10:43:06 +04:00
|
|
|
weston_log_continue(STAMP_SPACE "wl_shm sub-image to texture: %s\n",
|
2012-11-13 22:10:27 +04:00
|
|
|
gr->has_unpack_subimage ? "yes" : "no");
|
2012-10-24 10:43:06 +04:00
|
|
|
weston_log_continue(STAMP_SPACE "EGL Wayland extension: %s\n",
|
2012-11-13 22:10:27 +04:00
|
|
|
gr->has_bind_display ? "yes" : "no");
|
2012-10-24 10:43:06 +04:00
|
|
|
|
2012-11-13 22:10:19 +04:00
|
|
|
|
2012-09-06 06:06:26 +04:00
|
|
|
return 0;
|
|
|
|
}
|
2013-10-14 16:57:11 +04:00
|
|
|
|
|
|
|
WL_EXPORT struct gl_renderer_interface gl_renderer_interface = {
|
|
|
|
.opaque_attribs = gl_renderer_opaque_attribs,
|
|
|
|
.alpha_attribs = gl_renderer_alpha_attribs,
|
|
|
|
|
|
|
|
.create = gl_renderer_create,
|
|
|
|
.display = gl_renderer_display,
|
|
|
|
.output_create = gl_renderer_output_create,
|
|
|
|
.output_destroy = gl_renderer_output_destroy,
|
|
|
|
.output_surface = gl_renderer_output_surface,
|
2013-10-28 07:24:54 +04:00
|
|
|
.output_set_border = gl_renderer_output_set_border,
|
2013-10-14 16:57:11 +04:00
|
|
|
.print_egl_error_state = gl_renderer_print_egl_error_state
|
|
|
|
};
|