2 * Copyright (C) 2012-2013 Robert Ancell.
3 * Author: Robert Ancell <robert.ancell@canonical.com>
5 * This program is free software: you can redistribute it and/or modify it under
6 * the terms of the GNU General Public License as published by the Free Software
7 * Foundation, either version 3 of the License, or (at your option) any later
8 * version. See http://www.gnu.org/copyleft/gpl.html the full text of the
15 #include <glib/gstdio.h>
17 #include "seat-unity.h"
18 #include "configuration.h"
19 #include "unity-system-compositor.h"
20 #include "x-server-xmir.h"
24 struct SeatUnityPrivate
26 /* System compositor */
27 UnitySystemCompositor *compositor;
29 /* X server being used for XDMCP */
30 XServerXmir *xdmcp_x_server;
32 /* Next Mir ID to use for a Xmir servers */
33 gint next_x_server_id;
35 /* The currently visible session */
36 Session *active_session;
37 DisplayServer *active_display_server;
40 G_DEFINE_TYPE (SeatUnity, seat_unity, SEAT_TYPE);
42 static XServerXmir *create_x_server (Seat *seat);
45 seat_unity_setup (Seat *seat)
47 seat_set_supports_multi_session (seat, TRUE);
48 SEAT_CLASS (seat_unity_parent_class)->setup (seat);
52 check_stopped (SeatUnity *seat)
54 if (!seat->priv->compositor && !seat->priv->xdmcp_x_server)
55 SEAT_CLASS (seat_unity_parent_class)->stop (SEAT (seat));
59 xdmcp_x_server_stopped_cb (DisplayServer *display_server, Seat *seat)
61 l_debug (seat, "XDMCP X server stopped");
63 g_signal_handlers_disconnect_matched (SEAT_UNITY (seat)->priv->xdmcp_x_server, G_SIGNAL_MATCH_DATA, 0, 0, NULL, NULL, seat);
64 SEAT_UNITY (seat)->priv->xdmcp_x_server = NULL;
66 g_object_unref (display_server);
68 if (seat_get_is_stopping (seat))
69 check_stopped (SEAT_UNITY (seat));
75 compositor_ready_cb (UnitySystemCompositor *compositor, SeatUnity *seat)
77 const gchar *xdmcp_manager = NULL;
79 l_debug (seat, "Compositor ready");
81 /* If running as an XDMCP client then just start an X server */
82 xdmcp_manager = seat_get_string_property (SEAT (seat), "xdmcp-manager");
85 const gchar *key_name = NULL;
88 seat->priv->xdmcp_x_server = create_x_server (SEAT (seat));
89 x_server_local_set_xdmcp_server (X_SERVER_LOCAL (seat->priv->xdmcp_x_server), xdmcp_manager);
90 port = seat_get_integer_property (SEAT (seat), "xdmcp-port");
92 x_server_local_set_xdmcp_port (X_SERVER_LOCAL (seat->priv->xdmcp_x_server), port);
93 key_name = seat_get_string_property (SEAT (seat), "xdmcp-key");
101 path = g_build_filename (config_get_directory (config_get_instance ()), "keys.conf", NULL);
103 keys = g_key_file_new ();
104 result = g_key_file_load_from_file (keys, path, G_KEY_FILE_NONE, &error);
106 l_debug (seat, "Error getting key %s", error->message);
107 g_clear_error (&error);
113 if (g_key_file_has_key (keys, "keyring", key_name, NULL))
114 key = g_key_file_get_string (keys, "keyring", key_name, NULL);
116 l_debug (seat, "Key %s not defined", key_name);
119 x_server_local_set_xdmcp_key (X_SERVER_LOCAL (seat->priv->xdmcp_x_server), key);
124 g_key_file_free (keys);
127 g_signal_connect (seat->priv->xdmcp_x_server, DISPLAY_SERVER_SIGNAL_STOPPED, G_CALLBACK (xdmcp_x_server_stopped_cb), seat);
128 if (!display_server_start (DISPLAY_SERVER (seat->priv->xdmcp_x_server)))
129 seat_stop (SEAT (seat));
132 SEAT_CLASS (seat_unity_parent_class)->start (SEAT (seat));
136 compositor_stopped_cb (UnitySystemCompositor *compositor, SeatUnity *seat)
138 l_debug (seat, "Compositor stopped");
140 g_clear_object (&seat->priv->compositor);
142 if (seat_get_is_stopping (SEAT (seat)))
143 check_stopped (seat);
145 seat_stop (SEAT (seat));
149 seat_unity_start (Seat *seat)
154 /* Replace Plymouth if it is running */
155 if (plymouth_get_is_active () && plymouth_has_active_vt ())
157 gint active_vt = vt_get_active ();
158 if (active_vt >= vt_get_min ())
161 plymouth_quit (TRUE);
164 l_debug (seat, "Plymouth is running on VT %d, but this is less than the configured minimum of %d so not replacing it", active_vt, vt_get_min ());
166 if (plymouth_get_is_active ())
167 plymouth_quit (FALSE);
169 vt = vt_can_multi_seat () ? vt_get_unused () : 0;
172 l_debug (seat, "Failed to get a VT to run on");
176 timeout = seat_get_integer_property (SEAT (seat), "unity-compositor-timeout");
180 SEAT_UNITY (seat)->priv->compositor = unity_system_compositor_new ();
181 g_signal_connect (SEAT_UNITY (seat)->priv->compositor, DISPLAY_SERVER_SIGNAL_READY, G_CALLBACK (compositor_ready_cb), seat);
182 g_signal_connect (SEAT_UNITY (seat)->priv->compositor, DISPLAY_SERVER_SIGNAL_STOPPED, G_CALLBACK (compositor_stopped_cb), seat);
183 unity_system_compositor_set_command (SEAT_UNITY (seat)->priv->compositor, seat_get_string_property (SEAT (seat), "unity-compositor-command"));
184 unity_system_compositor_set_vt (SEAT_UNITY (seat)->priv->compositor, vt);
185 unity_system_compositor_set_timeout (SEAT_UNITY (seat)->priv->compositor, timeout);
187 return display_server_start (DISPLAY_SERVER (SEAT_UNITY (seat)->priv->compositor));
191 create_x_server (Seat *seat)
193 XServerXmir *x_server;
196 const gchar *command = NULL, *layout = NULL, *config_file = NULL;
200 l_debug (seat, "Starting X server on Unity compositor");
202 x_server = x_server_xmir_new (SEAT_UNITY (seat)->priv->compositor);
204 command = seat_get_string_property (seat, "xmir-command");
205 x_server_local_set_command (X_SERVER_LOCAL (x_server), command);
207 id = g_strdup_printf ("x-%d", SEAT_UNITY (seat)->priv->next_x_server_id);
208 SEAT_UNITY (seat)->priv->next_x_server_id++;
209 x_server_xmir_set_mir_id (x_server, id);
210 x_server_xmir_set_mir_socket (x_server, unity_system_compositor_get_socket (SEAT_UNITY (seat)->priv->compositor));
213 number = g_strdup_printf ("%d", x_server_get_display_number (X_SERVER (x_server)));
214 cookie = x_authority_new_local_cookie (number);
215 x_server_set_authority (X_SERVER (x_server), cookie);
217 g_object_unref (cookie);
219 layout = seat_get_string_property (seat, "xserver-layout");
221 x_server_local_set_layout (X_SERVER_LOCAL (x_server), layout);
223 x_server_local_set_xdg_seat (X_SERVER_LOCAL (x_server), seat_get_name (seat));
225 config_file = seat_get_string_property (seat, "xserver-config");
227 x_server_local_set_config (X_SERVER_LOCAL (x_server), config_file);
229 allow_tcp = seat_get_boolean_property (seat, "xserver-allow-tcp");
230 x_server_local_set_allow_tcp (X_SERVER_LOCAL (x_server), allow_tcp);
235 static DisplayServer *
236 seat_unity_create_display_server (Seat *seat, Session *session)
238 const gchar *session_type;
240 session_type = session_get_session_type (session);
241 if (strcmp (session_type, "x") == 0)
242 return DISPLAY_SERVER (create_x_server (seat));
243 else if (strcmp (session_type, "mir") == 0)
244 return g_object_ref (SEAT_UNITY (seat)->priv->compositor);
247 l_warning (seat, "Can't create unsupported display server '%s'", session_type);
253 seat_unity_display_server_is_used (Seat *seat, DisplayServer *display_server)
255 if (display_server == DISPLAY_SERVER (SEAT_UNITY (seat)->priv->compositor))
258 return SEAT_CLASS (seat_unity_parent_class)->display_server_is_used (seat, display_server);
261 static GreeterSession *
262 seat_unity_create_greeter_session (Seat *seat)
264 GreeterSession *greeter_session;
267 greeter_session = SEAT_CLASS (seat_unity_parent_class)->create_greeter_session (seat);
268 session_set_env (SESSION (greeter_session), "XDG_SEAT", seat_get_name (seat));
270 vt = display_server_get_vt (DISPLAY_SERVER (SEAT_UNITY (seat)->priv->compositor));
273 gchar *value = g_strdup_printf ("%d", vt);
274 session_set_env (SESSION (greeter_session), "XDG_VTNR", value);
278 return greeter_session;
282 seat_unity_create_session (Seat *seat)
287 session = SEAT_CLASS (seat_unity_parent_class)->create_session (seat);
288 session_set_env (session, "XDG_SEAT", seat_get_name (seat));
290 vt = display_server_get_vt (DISPLAY_SERVER (SEAT_UNITY (seat)->priv->compositor));
293 gchar *value = g_strdup_printf ("%d", vt);
294 session_set_env (SESSION (session), "XDG_VTNR", value);
302 get_mir_id (Session *session)
304 DisplayServer *display_server;
309 display_server = session_get_display_server (session);
310 if (IS_UNITY_SYSTEM_COMPOSITOR (display_server))
311 return session_get_env (session, "MIR_SERVER_NAME");
312 if (IS_X_SERVER_XMIR (display_server))
313 return x_server_xmir_get_mir_id (X_SERVER_XMIR (display_server));
319 seat_unity_set_active_session (Seat *s, Session *session)
321 SeatUnity *seat = SEAT_UNITY (s);
322 const gchar *old_id, *new_id;
324 old_id = get_mir_id (seat->priv->active_session);
325 new_id = get_mir_id (session);
327 g_clear_object (&seat->priv->active_session);
328 seat->priv->active_session = g_object_ref (session);
330 if (g_strcmp0 (old_id, new_id) != 0)
331 unity_system_compositor_set_active_session (seat->priv->compositor, new_id);
333 SEAT_CLASS (seat_unity_parent_class)->set_active_session (s, session);
337 seat_unity_get_active_session (Seat *seat)
339 return SEAT_UNITY (seat)->priv->active_session;
343 seat_unity_set_next_session (Seat *seat, Session *session)
345 DisplayServer *display_server;
346 const gchar *id = NULL;
351 display_server = session_get_display_server (session);
353 if (IS_X_SERVER_LOCAL (display_server))
354 id = x_server_xmir_get_mir_id (X_SERVER_XMIR (display_server));
356 id = session_get_env (session, "MIR_SERVER_NAME");
360 l_debug (seat, "Marking Mir session %s as the next session", id);
361 unity_system_compositor_set_next_session (SEAT_UNITY (seat)->priv->compositor, id);
365 l_debug (seat, "Failed to work out session ID to mark");
368 SEAT_CLASS (seat_unity_parent_class)->set_next_session (seat, session);
372 seat_unity_run_script (Seat *seat, DisplayServer *display_server, Process *script)
374 if (IS_X_SERVER_XMIR (display_server))
376 XServerXmir *x_server;
379 x_server = X_SERVER_XMIR (display_server);
380 path = x_server_local_get_authority_file_path (X_SERVER_LOCAL (x_server));
381 process_set_env (script, "DISPLAY", x_server_get_address (X_SERVER (x_server)));
382 process_set_env (script, "XAUTHORITY", path);
385 SEAT_CLASS (seat_unity_parent_class)->run_script (seat, display_server, script);
389 seat_unity_stop (Seat *seat)
391 /* Stop the compositor first */
392 if (SEAT_UNITY (seat)->priv->compositor)
393 display_server_stop (DISPLAY_SERVER (SEAT_UNITY (seat)->priv->compositor));
395 /* Stop the XDMCP X server first */
396 if (SEAT_UNITY (seat)->priv->xdmcp_x_server)
397 display_server_stop (DISPLAY_SERVER (SEAT_UNITY (seat)->priv->xdmcp_x_server));
399 check_stopped (SEAT_UNITY (seat));
403 seat_unity_init (SeatUnity *seat)
405 seat->priv = G_TYPE_INSTANCE_GET_PRIVATE (seat, SEAT_UNITY_TYPE, SeatUnityPrivate);
409 seat_unity_finalize (GObject *object)
411 SeatUnity *seat = SEAT_UNITY (object);
413 g_clear_object (&seat->priv->compositor);
414 if (seat->priv->xdmcp_x_server)
416 g_signal_handlers_disconnect_matched (seat->priv->xdmcp_x_server, G_SIGNAL_MATCH_DATA, 0, 0, NULL, NULL, seat);
417 g_object_unref (seat->priv->xdmcp_x_server);
419 g_clear_object (&seat->priv->active_session);
420 g_clear_object (&seat->priv->active_display_server);
422 G_OBJECT_CLASS (seat_unity_parent_class)->finalize (object);
426 seat_unity_class_init (SeatUnityClass *klass)
428 GObjectClass *object_class = G_OBJECT_CLASS (klass);
429 SeatClass *seat_class = SEAT_CLASS (klass);
431 object_class->finalize = seat_unity_finalize;
432 seat_class->setup = seat_unity_setup;
433 seat_class->start = seat_unity_start;
434 seat_class->create_display_server = seat_unity_create_display_server;
435 seat_class->display_server_is_used = seat_unity_display_server_is_used;
436 seat_class->create_greeter_session = seat_unity_create_greeter_session;
437 seat_class->create_session = seat_unity_create_session;
438 seat_class->set_active_session = seat_unity_set_active_session;
439 seat_class->get_active_session = seat_unity_get_active_session;
440 seat_class->set_next_session = seat_unity_set_next_session;
441 seat_class->run_script = seat_unity_run_script;
442 seat_class->stop = seat_unity_stop;
444 g_type_class_add_private (klass, sizeof (SeatUnityPrivate));