mirror of
				https://gitlab.freedesktop.org/wlroots/wlroots.git
				synced 2025-11-03 09:01:40 -05:00 
			
		
		
		
	Merge pull request #751 from emersion/rotated-surface-damage
Fix damage tracking for rotated surfaces
This commit is contained in:
		
						commit
						891610081f
					
				
					 6 changed files with 86 additions and 28 deletions
				
			
		| 
						 | 
					@ -66,8 +66,10 @@ struct roots_config {
 | 
				
			||||||
	struct wl_list bindings;
 | 
						struct wl_list bindings;
 | 
				
			||||||
	struct wl_list keyboards;
 | 
						struct wl_list keyboards;
 | 
				
			||||||
	struct wl_list cursors;
 | 
						struct wl_list cursors;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	char *config_path;
 | 
						char *config_path;
 | 
				
			||||||
	char *startup_cmd;
 | 
						char *startup_cmd;
 | 
				
			||||||
 | 
						bool debug_damage_tracking;
 | 
				
			||||||
};
 | 
					};
 | 
				
			||||||
 | 
					
 | 
				
			||||||
/**
 | 
					/**
 | 
				
			||||||
| 
						 | 
					
 | 
				
			||||||
| 
						 | 
					@ -27,7 +27,7 @@ void wlr_box_transform(const struct wlr_box *box,
 | 
				
			||||||
	struct wlr_box *dest);
 | 
						struct wlr_box *dest);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
/**
 | 
					/**
 | 
				
			||||||
 * Creates the smallest box that contains a rotated box.
 | 
					 * Creates the smallest box that contains the box rotated about its center.
 | 
				
			||||||
 */
 | 
					 */
 | 
				
			||||||
void wlr_box_rotated_bounds(const struct wlr_box *box, float rotation,
 | 
					void wlr_box_rotated_bounds(const struct wlr_box *box, float rotation,
 | 
				
			||||||
	struct wlr_box *dest);
 | 
						struct wlr_box *dest);
 | 
				
			||||||
| 
						 | 
					
 | 
				
			||||||
| 
						 | 
					@ -26,4 +26,11 @@ void wlr_region_transform(pixman_region32_t *dst, pixman_region32_t *src,
 | 
				
			||||||
void wlr_region_expand(pixman_region32_t *dst, pixman_region32_t *src,
 | 
					void wlr_region_expand(pixman_region32_t *dst, pixman_region32_t *src,
 | 
				
			||||||
	int distance);
 | 
						int distance);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					/*
 | 
				
			||||||
 | 
					 * Builds the smallest possible region that contains the region rotated about
 | 
				
			||||||
 | 
					 * the point (ox, oy).
 | 
				
			||||||
 | 
					 */
 | 
				
			||||||
 | 
					void wlr_region_rotated_bounds(pixman_region32_t *dst, pixman_region32_t *src,
 | 
				
			||||||
 | 
						float rotation, int ox, int oy);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
#endif
 | 
					#endif
 | 
				
			||||||
| 
						 | 
					
 | 
				
			||||||
| 
						 | 
					@ -25,7 +25,9 @@ static void usage(const char *name, int ret) {
 | 
				
			||||||
		"                (default: rootston.ini).\n"
 | 
							"                (default: rootston.ini).\n"
 | 
				
			||||||
		"                See `rootston.ini.example` for config\n"
 | 
							"                See `rootston.ini.example` for config\n"
 | 
				
			||||||
		"                file documentation.\n"
 | 
							"                file documentation.\n"
 | 
				
			||||||
		" -E <COMMAND>   Command that will be ran at startup.\n" , name);
 | 
							" -E <COMMAND>   Command that will be ran at startup.\n"
 | 
				
			||||||
 | 
							" -D             Enable damage tracking debugging.\n",
 | 
				
			||||||
 | 
							name);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	exit(ret);
 | 
						exit(ret);
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
| 
						 | 
					@ -394,7 +396,7 @@ struct roots_config *roots_config_create_from_args(int argc, char *argv[]) {
 | 
				
			||||||
	wl_list_init(&config->bindings);
 | 
						wl_list_init(&config->bindings);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	int c;
 | 
						int c;
 | 
				
			||||||
	while ((c = getopt(argc, argv, "C:E:h")) != -1) {
 | 
						while ((c = getopt(argc, argv, "C:E:hD")) != -1) {
 | 
				
			||||||
		switch (c) {
 | 
							switch (c) {
 | 
				
			||||||
		case 'C':
 | 
							case 'C':
 | 
				
			||||||
			config->config_path = strdup(optarg);
 | 
								config->config_path = strdup(optarg);
 | 
				
			||||||
| 
						 | 
					@ -402,6 +404,9 @@ struct roots_config *roots_config_create_from_args(int argc, char *argv[]) {
 | 
				
			||||||
		case 'E':
 | 
							case 'E':
 | 
				
			||||||
			config->startup_cmd = strdup(optarg);
 | 
								config->startup_cmd = strdup(optarg);
 | 
				
			||||||
			break;
 | 
								break;
 | 
				
			||||||
 | 
							case 'D':
 | 
				
			||||||
 | 
								config->debug_damage_tracking = true;
 | 
				
			||||||
 | 
								break;
 | 
				
			||||||
		case 'h':
 | 
							case 'h':
 | 
				
			||||||
		case '?':
 | 
							case '?':
 | 
				
			||||||
			usage(argv[0], c != 'h');
 | 
								usage(argv[0], c != 'h');
 | 
				
			||||||
| 
						 | 
					
 | 
				
			||||||
| 
						 | 
					@ -486,6 +486,10 @@ static void render_output(struct roots_output *output) {
 | 
				
			||||||
		goto renderer_end;
 | 
							goto renderer_end;
 | 
				
			||||||
	}
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						if (server->config->debug_damage_tracking) {
 | 
				
			||||||
 | 
							wlr_renderer_clear(renderer, (float[]){1, 1, 0, 0});
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	int nrects;
 | 
						int nrects;
 | 
				
			||||||
	pixman_box32_t *rects = pixman_region32_rectangles(&damage, &nrects);
 | 
						pixman_box32_t *rects = pixman_region32_rectangles(&damage, &nrects);
 | 
				
			||||||
	for (int i = 0; i < nrects; ++i) {
 | 
						for (int i = 0; i < nrects; ++i) {
 | 
				
			||||||
| 
						 | 
					@ -667,7 +671,9 @@ static void damage_from_surface(struct wlr_surface *surface,
 | 
				
			||||||
	surface_intersect_output(surface, output->desktop->layout,
 | 
						surface_intersect_output(surface, output->desktop->layout,
 | 
				
			||||||
		wlr_output, lx, ly, rotation, &box);
 | 
							wlr_output, lx, ly, rotation, &box);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	if (rotation == 0) {
 | 
						int center_x = box.x + box.width/2;
 | 
				
			||||||
 | 
						int center_y = box.y + box.height/2;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	pixman_region32_t damage;
 | 
						pixman_region32_t damage;
 | 
				
			||||||
	pixman_region32_init(&damage);
 | 
						pixman_region32_init(&damage);
 | 
				
			||||||
	pixman_region32_copy(&damage, &surface->current->surface_damage);
 | 
						pixman_region32_copy(&damage, &surface->current->surface_damage);
 | 
				
			||||||
| 
						 | 
					@ -679,20 +685,9 @@ static void damage_from_surface(struct wlr_surface *surface,
 | 
				
			||||||
			ceil(wlr_output->scale) - surface->current->scale);
 | 
								ceil(wlr_output->scale) - surface->current->scale);
 | 
				
			||||||
	}
 | 
						}
 | 
				
			||||||
	pixman_region32_translate(&damage, box.x, box.y);
 | 
						pixman_region32_translate(&damage, box.x, box.y);
 | 
				
			||||||
 | 
						wlr_region_rotated_bounds(&damage, &damage, rotation, center_x, center_y);
 | 
				
			||||||
	wlr_output_damage_add(output->damage, &damage);
 | 
						wlr_output_damage_add(output->damage, &damage);
 | 
				
			||||||
	pixman_region32_fini(&damage);
 | 
						pixman_region32_fini(&damage);
 | 
				
			||||||
	} else {
 | 
					 | 
				
			||||||
		pixman_box32_t *extents =
 | 
					 | 
				
			||||||
			pixman_region32_extents(&surface->current->surface_damage);
 | 
					 | 
				
			||||||
		struct wlr_box damage_box = {
 | 
					 | 
				
			||||||
			.x = box.x + extents->x1 * wlr_output->scale,
 | 
					 | 
				
			||||||
			.y = box.y + extents->y1 * wlr_output->scale,
 | 
					 | 
				
			||||||
			.width = (extents->x2 - extents->x1) * wlr_output->scale,
 | 
					 | 
				
			||||||
			.height = (extents->y2 - extents->y1) * wlr_output->scale,
 | 
					 | 
				
			||||||
		};
 | 
					 | 
				
			||||||
		wlr_box_rotated_bounds(&damage_box, rotation, &damage_box);
 | 
					 | 
				
			||||||
		wlr_output_damage_add_box(output->damage, &damage_box);
 | 
					 | 
				
			||||||
	}
 | 
					 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
void output_damage_from_view(struct roots_output *output,
 | 
					void output_damage_from_view(struct roots_output *output,
 | 
				
			||||||
| 
						 | 
					
 | 
				
			||||||
| 
						 | 
					@ -128,3 +128,52 @@ void wlr_region_expand(pixman_region32_t *dst, pixman_region32_t *src,
 | 
				
			||||||
	pixman_region32_init_rects(dst, dst_rects, nrects);
 | 
						pixman_region32_init_rects(dst, dst_rects, nrects);
 | 
				
			||||||
	free(dst_rects);
 | 
						free(dst_rects);
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					void wlr_region_rotated_bounds(pixman_region32_t *dst, pixman_region32_t *src,
 | 
				
			||||||
 | 
							float rotation, int ox, int oy) {
 | 
				
			||||||
 | 
						if (rotation == 0) {
 | 
				
			||||||
 | 
							pixman_region32_copy(dst, src);
 | 
				
			||||||
 | 
							return;
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						int nrects;
 | 
				
			||||||
 | 
						pixman_box32_t *src_rects = pixman_region32_rectangles(src, &nrects);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						pixman_box32_t *dst_rects = malloc(nrects * sizeof(pixman_box32_t));
 | 
				
			||||||
 | 
						if (dst_rects == NULL) {
 | 
				
			||||||
 | 
							return;
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						for (int i = 0; i < nrects; ++i) {
 | 
				
			||||||
 | 
							double x1 = src_rects[i].x1 - ox;
 | 
				
			||||||
 | 
							double y1 = src_rects[i].y1 - oy;
 | 
				
			||||||
 | 
							double x2 = src_rects[i].x2 - ox;
 | 
				
			||||||
 | 
							double y2 = src_rects[i].y2 - oy;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
							double rx1 = x1 * cos(rotation) - y1 * sin(rotation);
 | 
				
			||||||
 | 
							double ry1 = x1 * sin(rotation) + y1 * cos(rotation);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
							double rx2 = x2 * cos(rotation) - y1 * sin(rotation);
 | 
				
			||||||
 | 
							double ry2 = x2 * sin(rotation) + y1 * cos(rotation);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
							double rx3 = x2 * cos(rotation) - y2 * sin(rotation);
 | 
				
			||||||
 | 
							double ry3 = x2 * sin(rotation) + y2 * cos(rotation);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
							double rx4 = x1 * cos(rotation) - y2 * sin(rotation);
 | 
				
			||||||
 | 
							double ry4 = x1 * sin(rotation) + y2 * cos(rotation);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
							x1 = fmin(fmin(rx1, rx2), fmin(rx3, rx4));
 | 
				
			||||||
 | 
							y1 = fmin(fmin(ry1, ry2), fmin(ry3, ry4));
 | 
				
			||||||
 | 
							x2 = fmax(fmax(rx1, rx2), fmax(rx3, rx4));
 | 
				
			||||||
 | 
							y2 = fmax(fmax(ry1, ry2), fmax(ry3, ry4));
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
							dst_rects[i].x1 = floor(ox + x1);
 | 
				
			||||||
 | 
							dst_rects[i].x2 = ceil(ox + x2);
 | 
				
			||||||
 | 
							dst_rects[i].y1 = floor(oy + y1);
 | 
				
			||||||
 | 
							dst_rects[i].y2 = ceil(oy + y2);
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						pixman_region32_fini(dst);
 | 
				
			||||||
 | 
						pixman_region32_init_rects(dst, dst_rects, nrects);
 | 
				
			||||||
 | 
						free(dst_rects);
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
| 
						 | 
					
 | 
				
			||||||
		Loading…
	
	Add table
		Add a link
		
	
		Reference in a new issue