diff options
author | Pekka Paalanen <ppaalanen@gmail.com> | 2012-01-24 14:47:37 +0200 |
---|---|---|
committer | Pekka Paalanen <ppaalanen@gmail.com> | 2012-01-27 16:15:02 +0200 |
commit | 0e151bb487ab55bef96feeec7623b791c316fc91 (patch) | |
tree | 1968eac9048bbf723cc4a1327a84bcf8dafbf2bb | |
parent | 3df327f4c99afdc97a0850f80f2d3b9538797722 (diff) |
compositor: honour repaint regions with transformed surfaces
Previously, if a surface was transformed, it repainted as a whole,
regardless of the computed repaint region. As damage regions determine
repaint regions and whether a surface is considered for drawing at all,
this lead to disappearing surfaces if all surfaces were considered
transformed. Also transparent transformed surfaces were redrawn without
the surfaces below being properly redrawn, leading to alpha-saturation.
Fix that by making texture_region() use the proper global-to-surface
coordinate transformation for texture coordinates. This makes it
possible to call texture_region() also for transformed surfaces.
As texture coordinates may now lie outside the valid texture image, the
fragment shader is modified to check the fragment texture coordinates.
The special path texture_transformed_surface() is no longer used and is
removed.
This change fixes many of the rendering artifacts related to transformed
surfaces.
Signed-off-by: Pekka Paalanen <ppaalanen@gmail.com>
-rw-r--r-- | src/compositor.c | 119 | ||||
-rw-r--r-- | src/compositor.h | 1 |
2 files changed, 50 insertions, 70 deletions
diff --git a/src/compositor.c b/src/compositor.c index d8d78ce..f2fa789 100644 --- a/src/compositor.c +++ b/src/compositor.c @@ -289,12 +289,10 @@ weston_surface_to_global(struct weston_surface *surface, } } -WL_EXPORT void -weston_surface_from_global(struct weston_surface *surface, - int32_t x, int32_t y, int32_t *sx, int32_t *sy) +static void +surface_from_global_float(struct weston_surface *surface, + int32_t x, int32_t y, GLfloat *sx, GLfloat *sy) { - weston_surface_update_transform(surface); - if (surface->transform.enabled) { struct weston_vector v = { { x, y, 0.0f, 1.0f } }; @@ -309,8 +307,8 @@ weston_surface_from_global(struct weston_surface *surface, return; } - *sx = floorf(v.f[0] / v.f[3] - surface->x); - *sy = floorf(v.f[1] / v.f[3] - surface->y); + *sx = v.f[0] / v.f[3] - surface->x; + *sy = v.f[1] / v.f[3] - surface->y; } else { *sx = x - surface->x; *sy = y - surface->y; @@ -318,6 +316,19 @@ weston_surface_from_global(struct weston_surface *surface, } WL_EXPORT void +weston_surface_from_global(struct weston_surface *surface, + int32_t x, int32_t y, int32_t *sx, int32_t *sy) +{ + GLfloat sxf, syf; + + weston_surface_update_transform(surface); + + surface_from_global_float(surface, x, y, &sxf, &syf); + *sx = floorf(sxf); + *sy = floorf(syf); +} + +WL_EXPORT void weston_surface_damage_rectangle(struct weston_surface *surface, int32_t x, int32_t y, int32_t width, int32_t height) @@ -534,6 +545,7 @@ texture_region(struct weston_surface *es, pixman_region32_t *region) { struct weston_compositor *ec = es->compositor; GLfloat *v, inv_width, inv_height; + GLfloat sx, sy; pixman_box32_t *rectangles; unsigned int *p; int i, n; @@ -545,25 +557,33 @@ texture_region(struct weston_surface *es, pixman_region32_t *region) inv_height = 1.0 / es->height; for (i = 0; i < n; i++, v += 16, p += 6) { + surface_from_global_float(es, rectangles[i].x1, + rectangles[i].y1, &sx, &sy); v[ 0] = rectangles[i].x1; v[ 1] = rectangles[i].y1; - v[ 2] = (GLfloat) (rectangles[i].x1 - es->x) * inv_width; - v[ 3] = (GLfloat) (rectangles[i].y1 - es->y) * inv_height; + v[ 2] = sx * inv_width; + v[ 3] = sy * inv_height; + surface_from_global_float(es, rectangles[i].x1, + rectangles[i].y2, &sx, &sy); v[ 4] = rectangles[i].x1; v[ 5] = rectangles[i].y2; - v[ 6] = v[ 2]; - v[ 7] = (GLfloat) (rectangles[i].y2 - es->y) * inv_height; + v[ 6] = sx * inv_width; + v[ 7] = sy * inv_height; + surface_from_global_float(es, rectangles[i].x2, + rectangles[i].y1, &sx, &sy); v[ 8] = rectangles[i].x2; v[ 9] = rectangles[i].y1; - v[10] = (GLfloat) (rectangles[i].x2 - es->x) * inv_width; - v[11] = v[ 3]; + v[10] = sx * inv_width; + v[11] = sy * inv_height; + surface_from_global_float(es, rectangles[i].x2, + rectangles[i].y2, &sx, &sy); v[12] = rectangles[i].x2; v[13] = rectangles[i].y2; - v[14] = v[10]; - v[15] = v[ 7]; + v[14] = sx * inv_width; + v[15] = sy * inv_height; p[0] = i * 4 + 0; p[1] = i * 4 + 1; @@ -576,56 +596,6 @@ texture_region(struct weston_surface *es, pixman_region32_t *region) return n; } -static void -transform_vertex(struct weston_surface *surface, - GLfloat sx, GLfloat sy, - GLfloat tex_x, GLfloat tex_y, GLfloat *r) -{ - /* surface->transform.enabled must always be 1 here. */ - - struct weston_vector v = { { sx, sy, 0.0f, 1.0f } }; - - v.f[0] += surface->x; - v.f[1] += surface->y; - - weston_matrix_transform(&surface->transform.matrix, &v); - - if (fabsf(v.f[3]) < 1e-6) { - fprintf(stderr, "warning: numerical instability in " - "transform_vertex(), divisor = %g\n", v.f[3]); - } - - r[ 0] = v.f[0] / v.f[3]; - r[ 1] = v.f[1] / v.f[3]; - r[ 2] = tex_x; - r[ 3] = tex_y; -} - -static int -texture_transformed_surface(struct weston_surface *es) -{ - struct weston_compositor *ec = es->compositor; - GLfloat *v; - unsigned int *p; - - v = wl_array_add(&ec->vertices, 16 * sizeof *v); - p = wl_array_add(&ec->indices, 6 * sizeof *p); - - transform_vertex(es, 0, 0, 0.0, 0.0, &v[0]); - transform_vertex(es, 0, es->height, 0.0, 1.0, &v[4]); - transform_vertex(es, es->width, 0, 1.0, 0.0, &v[8]); - transform_vertex(es, es->width, es->height, 1.0, 1.0, &v[12]); - - p[0] = 0; - p[1] = 1; - p[2] = 2; - p[3] = 2; - p[4] = 1; - p[5] = 3; - - return 1; -} - WL_EXPORT void weston_surface_draw(struct weston_surface *es, struct weston_output *output) { @@ -672,14 +642,17 @@ weston_surface_draw(struct weston_surface *es, struct weston_output *output) ec->current_alpha = es->alpha; } + if (es->shader->texwidth_uniform != GL_NONE) + glUniform1f(es->shader->texwidth_uniform, + (GLfloat)es->width / es->pitch); + weston_surface_update_transform(es); - if (es->transform.enabled) { + if (es->transform.enabled) filter = GL_LINEAR; - n = texture_transformed_surface(es); - } else { + else filter = GL_NEAREST; - n = texture_region(es, &repaint); - } + + n = texture_region(es, &repaint); glBindTexture(GL_TEXTURE_2D, es->texture); glTexParameteri(GL_TEXTURE_2D, GL_TEXTURE_MIN_FILTER, filter); @@ -1717,8 +1690,12 @@ static const char texture_fragment_shader[] = "varying vec2 v_texcoord;\n" "uniform sampler2D tex;\n" "uniform float alpha;\n" + "uniform float texwidth;\n" "void main()\n" "{\n" + " if (v_texcoord.x < 0.0 || v_texcoord.x > texwidth ||\n" + " v_texcoord.y < 0.0 || v_texcoord.y > 1.0)\n" + " discard;\n" " gl_FragColor = texture2D(tex, v_texcoord)\n;" " gl_FragColor = alpha * gl_FragColor;\n" "}\n"; @@ -1780,6 +1757,8 @@ weston_shader_init(struct weston_shader *shader, shader->proj_uniform = glGetUniformLocation(shader->program, "proj"); shader->tex_uniform = glGetUniformLocation(shader->program, "tex"); shader->alpha_uniform = glGetUniformLocation(shader->program, "alpha"); + shader->texwidth_uniform = glGetUniformLocation(shader->program, + "texwidth"); return 0; } diff --git a/src/compositor.h b/src/compositor.h index be1740b..d8e9ab8 100644 --- a/src/compositor.h +++ b/src/compositor.h @@ -102,6 +102,7 @@ struct weston_shader { GLuint tex_uniform; GLuint alpha_uniform; GLuint color_uniform; + GLuint texwidth_uniform; }; struct weston_animation { |