xref: /OK3568_Linux_fs/buildroot/package/weston/0013-Support-setting-output-flow-direction.patch (revision 4882a59341e53eb6f0b4789bf948001014eff981)
1*4882a593SmuzhiyunFrom b342bc6139ea8a765084eacfc0ca365561fcdd30 Mon Sep 17 00:00:00 2001
2*4882a593SmuzhiyunFrom: Jeffy Chen <jeffy.chen@rock-chips.com>
3*4882a593SmuzhiyunDate: Wed, 27 Apr 2022 18:05:28 +0800
4*4882a593SmuzhiyunSubject: [PATCH 13/93] Support setting output flow direction
5*4882a593Smuzhiyun
6*4882a593SmuzhiyunSet env "WESTON_OUTPUT_FLOW" to change output flow direction:
7*4882a593Smuzhiyunhorizontal:
8*4882a593Smuzhiyun  Place outputs horizontal.
9*4882a593Smuzhiyunvertical:
10*4882a593Smuzhiyun  Place outputs vertical.
11*4882a593Smuzhiyunsame-as:
12*4882a593Smuzhiyun  Place outputs at (0,0).
13*4882a593Smuzhiyun
14*4882a593SmuzhiyunSigned-off-by: Jeffy Chen <jeffy.chen@rock-chips.com>
15*4882a593Smuzhiyun---
16*4882a593Smuzhiyun clients/desktop-shell.c       | 20 +++++++++++++++
17*4882a593Smuzhiyun compositor/main.c             | 11 ++++++++
18*4882a593Smuzhiyun desktop-shell/shell.c         | 17 ++++++++++---
19*4882a593Smuzhiyun include/libweston/libweston.h | 11 ++++++++
20*4882a593Smuzhiyun libweston/compositor.c        | 48 +++++++++++++++--------------------
21*4882a593Smuzhiyun 5 files changed, 77 insertions(+), 30 deletions(-)
22*4882a593Smuzhiyun
23*4882a593Smuzhiyundiff --git a/clients/desktop-shell.c b/clients/desktop-shell.c
24*4882a593Smuzhiyunindex b450a01..6eb9775 100644
25*4882a593Smuzhiyun--- a/clients/desktop-shell.c
26*4882a593Smuzhiyun+++ b/clients/desktop-shell.c
27*4882a593Smuzhiyun@@ -135,6 +135,8 @@ struct output {
28*4882a593Smuzhiyun 	int y;
29*4882a593Smuzhiyun 	struct panel *panel;
30*4882a593Smuzhiyun 	struct background *background;
31*4882a593Smuzhiyun+
32*4882a593Smuzhiyun+	struct desktop *desktop;
33*4882a593Smuzhiyun };
34*4882a593Smuzhiyun
35*4882a593Smuzhiyun struct panel_launcher {
36*4882a593Smuzhiyun@@ -1227,10 +1229,27 @@ output_handle_geometry(void *data,
37*4882a593Smuzhiyun                        int transform)
38*4882a593Smuzhiyun {
39*4882a593Smuzhiyun 	struct output *output = data;
40*4882a593Smuzhiyun+	struct desktop *desktop = output->desktop;
41*4882a593Smuzhiyun+	struct wl_surface *surface;
42*4882a593Smuzhiyun
43*4882a593Smuzhiyun 	output->x = x;
44*4882a593Smuzhiyun 	output->y = y;
45*4882a593Smuzhiyun
46*4882a593Smuzhiyun+	if (y && output->panel) {
47*4882a593Smuzhiyun+		/* HACK: Re-set the panel to destroy it */
48*4882a593Smuzhiyun+		surface = window_get_wl_surface(output->panel->window);
49*4882a593Smuzhiyun+		weston_desktop_shell_set_panel(desktop->shell,
50*4882a593Smuzhiyun+					       output->output, surface);
51*4882a593Smuzhiyun+	}
52*4882a593Smuzhiyun+
53*4882a593Smuzhiyun+	if (!y && desktop->want_panel && !output->panel) {
54*4882a593Smuzhiyun+		/* based on output_init() */
55*4882a593Smuzhiyun+		output->panel = panel_create(desktop, output);
56*4882a593Smuzhiyun+		surface = window_get_wl_surface(output->panel->window);
57*4882a593Smuzhiyun+		weston_desktop_shell_set_panel(desktop->shell,
58*4882a593Smuzhiyun+					       output->output, surface);
59*4882a593Smuzhiyun+	}
60*4882a593Smuzhiyun+
61*4882a593Smuzhiyun 	if (output->panel)
62*4882a593Smuzhiyun 		window_set_buffer_transform(output->panel->window, transform);
63*4882a593Smuzhiyun 	if (output->background)
64*4882a593Smuzhiyun@@ -1300,6 +1319,7 @@ create_output(struct desktop *desktop, uint32_t id)
65*4882a593Smuzhiyun 	if (!output)
66*4882a593Smuzhiyun 		return;
67*4882a593Smuzhiyun
68*4882a593Smuzhiyun+	output->desktop = desktop;
69*4882a593Smuzhiyun 	output->output =
70*4882a593Smuzhiyun 		display_bind(desktop->display, id, &wl_output_interface, 2);
71*4882a593Smuzhiyun 	output->server_output_id = id;
72*4882a593Smuzhiyundiff --git a/compositor/main.c b/compositor/main.c
73*4882a593Smuzhiyunindex 631355e..57a52b6 100644
74*4882a593Smuzhiyun--- a/compositor/main.c
75*4882a593Smuzhiyun+++ b/compositor/main.c
76*4882a593Smuzhiyun@@ -3546,6 +3546,7 @@ wet_main(int argc, char *argv[], const struct weston_testsuite_data *test_data)
77*4882a593Smuzhiyun 	struct weston_log_subscriber *flight_rec = NULL;
78*4882a593Smuzhiyun 	sigset_t mask;
79*4882a593Smuzhiyun 	struct sigaction action;
80*4882a593Smuzhiyun+	char *buf;
81*4882a593Smuzhiyun
82*4882a593Smuzhiyun 	bool wait_for_debugger = false;
83*4882a593Smuzhiyun 	struct wl_protocol_logger *protologger = NULL;
84*4882a593Smuzhiyun@@ -3699,6 +3700,16 @@ wet_main(int argc, char *argv[], const struct weston_testsuite_data *test_data)
85*4882a593Smuzhiyun 		goto out;
86*4882a593Smuzhiyun 	}
87*4882a593Smuzhiyun
88*4882a593Smuzhiyun+	wet.compositor->output_flow = WESTON_OUTPUT_FLOW_HORIZONTAL;
89*4882a593Smuzhiyun+
90*4882a593Smuzhiyun+	buf = getenv("WESTON_OUTPUT_FLOW");
91*4882a593Smuzhiyun+	if (buf) {
92*4882a593Smuzhiyun+		if (!strcmp(buf, "vertical"))
93*4882a593Smuzhiyun+			wet.compositor->output_flow = WESTON_OUTPUT_FLOW_VERTICAL;
94*4882a593Smuzhiyun+		else if (!strcmp(buf, "same-as"))
95*4882a593Smuzhiyun+			wet.compositor->output_flow = WESTON_OUTPUT_FLOW_SAME_AS;
96*4882a593Smuzhiyun+	}
97*4882a593Smuzhiyun+
98*4882a593Smuzhiyun 	protocol_scope =
99*4882a593Smuzhiyun 		weston_log_ctx_add_log_scope(log_ctx, "proto",
100*4882a593Smuzhiyun 					     "Wayland protocol dump for all clients.\n",
101*4882a593Smuzhiyundiff --git a/desktop-shell/shell.c b/desktop-shell/shell.c
102*4882a593Smuzhiyunindex 7363ee3..79e20ad 100644
103*4882a593Smuzhiyun--- a/desktop-shell/shell.c
104*4882a593Smuzhiyun+++ b/desktop-shell/shell.c
105*4882a593Smuzhiyun@@ -1020,7 +1020,7 @@ constrain_position(struct weston_move_grab *move, int *cx, int *cy)
106*4882a593Smuzhiyun 	y = wl_fixed_to_int(pointer->y + move->dy);
107*4882a593Smuzhiyun
108*4882a593Smuzhiyun 	if (shsurf->shell->panel_position ==
109*4882a593Smuzhiyun-	    WESTON_DESKTOP_SHELL_PANEL_POSITION_TOP) {
110*4882a593Smuzhiyun+	    WESTON_DESKTOP_SHELL_PANEL_POSITION_TOP && !surface->output->y) {
111*4882a593Smuzhiyun 		get_output_work_area(shsurf->shell, surface->output, &area);
112*4882a593Smuzhiyun 		geometry =
113*4882a593Smuzhiyun 			weston_desktop_surface_get_geometry(shsurf->desktop_surface);
114*4882a593Smuzhiyun@@ -2658,6 +2658,18 @@ desktop_shell_set_panel(struct wl_client *client,
115*4882a593Smuzhiyun 		wl_resource_get_user_data(surface_resource);
116*4882a593Smuzhiyun 	struct weston_view *view, *next;
117*4882a593Smuzhiyun 	struct shell_output *sh_output;
118*4882a593Smuzhiyun+	struct weston_output *output;
119*4882a593Smuzhiyun+
120*4882a593Smuzhiyun+	output = weston_head_from_resource(output_resource)->output;
121*4882a593Smuzhiyun+	sh_output = find_shell_output_from_weston_output(shell, output);
122*4882a593Smuzhiyun+
123*4882a593Smuzhiyun+	if (surface == sh_output->panel_surface) {
124*4882a593Smuzhiyun+		/* HACK: Re-set to destroy output panel */
125*4882a593Smuzhiyun+		weston_desktop_shell_send_configure(resource, 0,
126*4882a593Smuzhiyun+						    surface_resource,
127*4882a593Smuzhiyun+						    0, 0);
128*4882a593Smuzhiyun+		return;
129*4882a593Smuzhiyun+	}
130*4882a593Smuzhiyun
131*4882a593Smuzhiyun 	if (surface->committed) {
132*4882a593Smuzhiyun 		wl_resource_post_error(surface_resource,
133*4882a593Smuzhiyun@@ -2673,10 +2685,9 @@ desktop_shell_set_panel(struct wl_client *client,
134*4882a593Smuzhiyun 	surface->committed = panel_committed;
135*4882a593Smuzhiyun 	surface->committed_private = shell;
136*4882a593Smuzhiyun 	weston_surface_set_label_func(surface, panel_get_label);
137*4882a593Smuzhiyun-	surface->output = weston_head_from_resource(output_resource)->output;
138*4882a593Smuzhiyun+	surface->output = output;
139*4882a593Smuzhiyun 	weston_view_set_output(view, surface->output);
140*4882a593Smuzhiyun
141*4882a593Smuzhiyun-	sh_output = find_shell_output_from_weston_output(shell, surface->output);
142*4882a593Smuzhiyun 	if (sh_output->panel_surface) {
143*4882a593Smuzhiyun 		/* The output already has a panel, tell our helper
144*4882a593Smuzhiyun 		 * there is no need for another one. */
145*4882a593Smuzhiyundiff --git a/include/libweston/libweston.h b/include/libweston/libweston.h
146*4882a593Smuzhiyunindex a3a23bc..103c293 100644
147*4882a593Smuzhiyun--- a/include/libweston/libweston.h
148*4882a593Smuzhiyun+++ b/include/libweston/libweston.h
149*4882a593Smuzhiyun@@ -1184,6 +1184,12 @@ struct weston_color_manager;
150*4882a593Smuzhiyun struct weston_dmabuf_feedback;
151*4882a593Smuzhiyun struct weston_dmabuf_feedback_format_table;
152*4882a593Smuzhiyun
153*4882a593Smuzhiyun+enum weston_output_flow {
154*4882a593Smuzhiyun+	WESTON_OUTPUT_FLOW_HORIZONTAL,
155*4882a593Smuzhiyun+	WESTON_OUTPUT_FLOW_VERTICAL,
156*4882a593Smuzhiyun+	WESTON_OUTPUT_FLOW_SAME_AS,
157*4882a593Smuzhiyun+};
158*4882a593Smuzhiyun+
159*4882a593Smuzhiyun /** Main object, container-like structure which aggregates all other objects.
160*4882a593Smuzhiyun  *
161*4882a593Smuzhiyun  * \ingroup compositor
162*4882a593Smuzhiyun@@ -1317,6 +1323,8 @@ struct weston_compositor {
163*4882a593Smuzhiyun 	/* One-time warning about a view appearing in the layer list when it
164*4882a593Smuzhiyun 	 * or its surface are not mapped. */
165*4882a593Smuzhiyun 	bool warned_about_unmapped_surface_or_view;
166*4882a593Smuzhiyun+
167*4882a593Smuzhiyun+	enum weston_output_flow output_flow;
168*4882a593Smuzhiyun };
169*4882a593Smuzhiyun
170*4882a593Smuzhiyun struct weston_solid_buffer_values {
171*4882a593Smuzhiyun@@ -2347,6 +2355,9 @@ struct weston_color_profile *
172*4882a593Smuzhiyun weston_compositor_load_icc_file(struct weston_compositor *compositor,
173*4882a593Smuzhiyun 				const char *path);
174*4882a593Smuzhiyun
175*4882a593Smuzhiyun+void
176*4882a593Smuzhiyun+weston_compositor_reflow_outputs(struct weston_compositor *compositor);
177*4882a593Smuzhiyun+
178*4882a593Smuzhiyun #ifdef  __cplusplus
179*4882a593Smuzhiyun }
180*4882a593Smuzhiyun #endif
181*4882a593Smuzhiyundiff --git a/libweston/compositor.c b/libweston/compositor.c
182*4882a593Smuzhiyunindex 55751a7..23316ef 100644
183*4882a593Smuzhiyun--- a/libweston/compositor.c
184*4882a593Smuzhiyun+++ b/libweston/compositor.c
185*4882a593Smuzhiyun@@ -258,10 +258,6 @@ weston_mode_switch_finish(struct weston_output *output,
186*4882a593Smuzhiyun 					       mode_changed, scale_changed);
187*4882a593Smuzhiyun }
188*4882a593Smuzhiyun
189*4882a593Smuzhiyun-static void
190*4882a593Smuzhiyun-weston_compositor_reflow_outputs(struct weston_compositor *compositor,
191*4882a593Smuzhiyun-				struct weston_output *resized_output, int delta_width);
192*4882a593Smuzhiyun-
193*4882a593Smuzhiyun /**
194*4882a593Smuzhiyun  * \ingroup output
195*4882a593Smuzhiyun  */
196*4882a593Smuzhiyun@@ -272,7 +268,6 @@ weston_output_mode_set_native(struct weston_output *output,
197*4882a593Smuzhiyun {
198*4882a593Smuzhiyun 	int ret;
199*4882a593Smuzhiyun 	int mode_changed = 0, scale_changed = 0;
200*4882a593Smuzhiyun-	int32_t old_width;
201*4882a593Smuzhiyun
202*4882a593Smuzhiyun 	if (!output->switch_mode)
203*4882a593Smuzhiyun 		return -1;
204*4882a593Smuzhiyun@@ -288,14 +283,13 @@ weston_output_mode_set_native(struct weston_output *output,
205*4882a593Smuzhiyun 		}
206*4882a593Smuzhiyun 	}
207*4882a593Smuzhiyun
208*4882a593Smuzhiyun-	old_width = output->width;
209*4882a593Smuzhiyun 	output->native_mode = mode;
210*4882a593Smuzhiyun 	output->native_scale = scale;
211*4882a593Smuzhiyun
212*4882a593Smuzhiyun 	weston_mode_switch_finish(output, mode_changed, scale_changed);
213*4882a593Smuzhiyun
214*4882a593Smuzhiyun 	if (mode_changed || scale_changed) {
215*4882a593Smuzhiyun-		weston_compositor_reflow_outputs(output->compositor, output, output->width - old_width);
216*4882a593Smuzhiyun+		weston_compositor_reflow_outputs(output->compositor);
217*4882a593Smuzhiyun
218*4882a593Smuzhiyun 		wl_signal_emit(&output->compositor->output_resized_signal, output);
219*4882a593Smuzhiyun 	}
220*4882a593Smuzhiyun@@ -6468,30 +6462,29 @@ weston_head_get_destroy_listener(struct weston_head *head,
221*4882a593Smuzhiyun static void
222*4882a593Smuzhiyun weston_output_set_position(struct weston_output *output, int x, int y);
223*4882a593Smuzhiyun
224*4882a593Smuzhiyun-/* Move other outputs when one is resized so the space remains contiguous. */
225*4882a593Smuzhiyun-static void
226*4882a593Smuzhiyun-weston_compositor_reflow_outputs(struct weston_compositor *compositor,
227*4882a593Smuzhiyun-				struct weston_output *resized_output, int delta_width)
228*4882a593Smuzhiyun+WL_EXPORT void
229*4882a593Smuzhiyun+weston_compositor_reflow_outputs(struct weston_compositor *compositor)
230*4882a593Smuzhiyun {
231*4882a593Smuzhiyun 	struct weston_output *output;
232*4882a593Smuzhiyun-	bool start_resizing = false;
233*4882a593Smuzhiyun+	int x, y, next_x, next_y;
234*4882a593Smuzhiyun
235*4882a593Smuzhiyun 	if (compositor->output_flow_dirty)
236*4882a593Smuzhiyun 		return;
237*4882a593Smuzhiyun
238*4882a593Smuzhiyun-	if (!delta_width)
239*4882a593Smuzhiyun-		return;
240*4882a593Smuzhiyun-
241*4882a593Smuzhiyun+	next_x = next_y = 0;
242*4882a593Smuzhiyun 	wl_list_for_each(output, &compositor->output_list, link) {
243*4882a593Smuzhiyun-		if (output == resized_output) {
244*4882a593Smuzhiyun-			start_resizing = true;
245*4882a593Smuzhiyun+		if (output->destroying)
246*4882a593Smuzhiyun 			continue;
247*4882a593Smuzhiyun-		}
248*4882a593Smuzhiyun
249*4882a593Smuzhiyun-		if (start_resizing) {
250*4882a593Smuzhiyun-			weston_output_set_position(output, output->x + delta_width, output->y);
251*4882a593Smuzhiyun-			output->dirty = 1;
252*4882a593Smuzhiyun-		}
253*4882a593Smuzhiyun+		x = next_x;
254*4882a593Smuzhiyun+		y = next_y;
255*4882a593Smuzhiyun+
256*4882a593Smuzhiyun+		if (compositor->output_flow == WESTON_OUTPUT_FLOW_HORIZONTAL)
257*4882a593Smuzhiyun+			next_x += output->width;
258*4882a593Smuzhiyun+		else if (compositor->output_flow == WESTON_OUTPUT_FLOW_VERTICAL)
259*4882a593Smuzhiyun+			next_y += output->height;
260*4882a593Smuzhiyun+
261*4882a593Smuzhiyun+		weston_output_set_position(output, x, y);
262*4882a593Smuzhiyun 	}
263*4882a593Smuzhiyun }
264*4882a593Smuzhiyun
265*4882a593Smuzhiyun@@ -6583,6 +6576,8 @@ weston_output_transform_scale_init(struct weston_output *output, uint32_t transf
266*4882a593Smuzhiyun static void
267*4882a593Smuzhiyun weston_output_init_geometry(struct weston_output *output, int x, int y)
268*4882a593Smuzhiyun {
269*4882a593Smuzhiyun+	output->dirty = 1;
270*4882a593Smuzhiyun+
271*4882a593Smuzhiyun 	output->x = x;
272*4882a593Smuzhiyun 	output->y = y;
273*4882a593Smuzhiyun
274*4882a593Smuzhiyun@@ -6616,8 +6611,6 @@ weston_output_set_position(struct weston_output *output, int x, int y)
275*4882a593Smuzhiyun
276*4882a593Smuzhiyun 	weston_output_init_geometry(output, x, y);
277*4882a593Smuzhiyun
278*4882a593Smuzhiyun-	output->dirty = 1;
279*4882a593Smuzhiyun-
280*4882a593Smuzhiyun 	/* Move views on this output. */
281*4882a593Smuzhiyun 	wl_signal_emit(&output->compositor->output_moved_signal, output);
282*4882a593Smuzhiyun
283*4882a593Smuzhiyun@@ -6909,7 +6902,7 @@ weston_compositor_remove_output(struct weston_output *output)
284*4882a593Smuzhiyun
285*4882a593Smuzhiyun 	weston_presentation_feedback_discard_list(&output->feedback_list);
286*4882a593Smuzhiyun
287*4882a593Smuzhiyun-	weston_compositor_reflow_outputs(compositor, output, -output->width);
288*4882a593Smuzhiyun+	weston_compositor_reflow_outputs(compositor);
289*4882a593Smuzhiyun
290*4882a593Smuzhiyun 	wl_list_remove(&output->link);
291*4882a593Smuzhiyun 	wl_list_insert(compositor->pending_output_list.prev, &output->link);
292*4882a593Smuzhiyun@@ -6983,7 +6976,7 @@ weston_output_set_transform(struct weston_output *output,
293*4882a593Smuzhiyun
294*4882a593Smuzhiyun 	weston_output_init_geometry(output, output->x, output->y);
295*4882a593Smuzhiyun
296*4882a593Smuzhiyun-	output->dirty = 1;
297*4882a593Smuzhiyun+	weston_compositor_reflow_outputs(output->compositor);
298*4882a593Smuzhiyun
299*4882a593Smuzhiyun 	/* Notify clients of the change for output transform. */
300*4882a593Smuzhiyun 	wl_list_for_each(head, &output->head_list, output_link) {
301*4882a593Smuzhiyun@@ -7346,7 +7339,6 @@ weston_output_enable(struct weston_output *output)
302*4882a593Smuzhiyun 	/* Make sure we have a transform set */
303*4882a593Smuzhiyun 	assert(output->transform != UINT32_MAX);
304*4882a593Smuzhiyun
305*4882a593Smuzhiyun-	output->dirty = 1;
306*4882a593Smuzhiyun 	output->original_scale = output->scale;
307*4882a593Smuzhiyun
308*4882a593Smuzhiyun 	wl_signal_init(&output->frame_signal);
309*4882a593Smuzhiyun@@ -7385,6 +7377,8 @@ weston_output_enable(struct weston_output *output)
310*4882a593Smuzhiyun 		   output->name, head_names);
311*4882a593Smuzhiyun 	free(head_names);
312*4882a593Smuzhiyun
313*4882a593Smuzhiyun+	weston_compositor_reflow_outputs(output->compositor);
314*4882a593Smuzhiyun+
315*4882a593Smuzhiyun 	return 0;
316*4882a593Smuzhiyun }
317*4882a593Smuzhiyun
318*4882a593Smuzhiyun--
319*4882a593Smuzhiyun2.20.1
320*4882a593Smuzhiyun
321