2 * Copyright (C) 2010-2011 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
18 #include <glib/gstdio.h>
20 #include "x-server-xvnc.h"
21 #include "configuration.h"
22 #include "x-server-local.h"
25 struct XServerXVNCPrivate
27 /* X server process */
28 Process *x_server_process;
30 /* Command to run the X server */
34 gchar *authority_file;
36 /* File descriptor to use for standard input */
39 /* Geometry and colour depth */
40 gint width, height, depth;
42 /* TRUE when received ready signal */
46 G_DEFINE_TYPE (XServerXVNC, x_server_xvnc, X_SERVER_TYPE);
49 x_server_xvnc_new (void)
51 XServerXVNC *self = g_object_new (X_SERVER_XVNC_TYPE, NULL);
54 x_server_set_display_number (X_SERVER (self), x_server_local_get_unused_display_number ());
56 name = g_strdup_printf ("xvnc-%d", x_server_get_display_number (X_SERVER (self)));
57 display_server_set_name (DISPLAY_SERVER (self), name);
64 x_server_xvnc_set_command (XServerXVNC *server, const gchar *command)
66 g_return_if_fail (server != NULL);
67 g_free (server->priv->command);
68 server->priv->command = g_strdup (command);
72 x_server_xvnc_set_socket (XServerXVNC *server, int fd)
74 g_return_if_fail (server != NULL);
75 server->priv->socket_fd = fd;
79 x_server_xvnc_get_socket (XServerXVNC *server)
81 g_return_val_if_fail (server != NULL, 0);
82 return server->priv->socket_fd;
86 x_server_xvnc_set_geometry (XServerXVNC *server, gint width, gint height)
88 g_return_if_fail (server != NULL);
89 server->priv->width = width;
90 server->priv->height = height;
94 x_server_xvnc_set_depth (XServerXVNC *server, gint depth)
96 g_return_if_fail (server != NULL);
97 server->priv->depth = depth;
101 x_server_xvnc_get_authority_file_path (XServerXVNC *server)
103 g_return_val_if_fail (server != NULL, 0);
104 return server->priv->authority_file;
108 get_absolute_command (const gchar *command)
111 gchar *absolute_binary, *absolute_command = NULL;
113 tokens = g_strsplit (command, " ", 2);
115 absolute_binary = g_find_program_in_path (tokens[0]);
119 absolute_command = g_strjoin (" ", absolute_binary, tokens[1], NULL);
121 absolute_command = g_strdup (absolute_binary);
126 return absolute_command;
130 run_cb (Process *process, gpointer user_data)
132 XServerXVNC *server = user_data;
135 dup2 (server->priv->socket_fd, STDIN_FILENO);
136 dup2 (server->priv->socket_fd, STDOUT_FILENO);
137 close (server->priv->socket_fd);
139 /* Set SIGUSR1 to ignore so the X server can indicate it when it is ready */
140 signal (SIGUSR1, SIG_IGN);
144 got_signal_cb (Process *process, int signum, XServerXVNC *server)
146 if (signum == SIGUSR1 && !server->priv->got_signal)
148 server->priv->got_signal = TRUE;
149 l_debug (server, "Got signal from Xvnc server :%d", x_server_get_display_number (X_SERVER (server)));
151 // FIXME: Check return value
152 DISPLAY_SERVER_CLASS (x_server_xvnc_parent_class)->start (DISPLAY_SERVER (server));
157 stopped_cb (Process *process, XServerXVNC *server)
159 l_debug (server, "Xvnc server stopped");
161 g_object_unref (server->priv->x_server_process);
162 server->priv->x_server_process = NULL;
164 x_server_local_release_display_number (x_server_get_display_number (X_SERVER (server)));
166 l_debug (server, "Removing X server authority %s", server->priv->authority_file);
168 g_unlink (server->priv->authority_file);
169 g_free (server->priv->authority_file);
170 server->priv->authority_file = NULL;
172 DISPLAY_SERVER_CLASS (x_server_xvnc_parent_class)->stop (DISPLAY_SERVER (server));
176 x_server_xvnc_get_can_share (DisplayServer *server)
182 x_server_xvnc_start (DisplayServer *display_server)
184 XServerXVNC *server = X_SERVER_XVNC (display_server);
185 XAuthority *authority;
187 gchar *filename, *run_dir, *dir, *log_file, *absolute_command;
189 gchar hostname[1024], *number;
190 GError *error = NULL;
192 g_return_val_if_fail (server->priv->x_server_process == NULL, FALSE);
194 server->priv->got_signal = FALSE;
196 server->priv->x_server_process = process_new (run_cb, server);
197 process_set_clear_environment (server->priv->x_server_process, TRUE);
198 g_signal_connect (server->priv->x_server_process, PROCESS_SIGNAL_GOT_SIGNAL, G_CALLBACK (got_signal_cb), server);
199 g_signal_connect (server->priv->x_server_process, PROCESS_SIGNAL_STOPPED, G_CALLBACK (stopped_cb), server);
202 filename = g_strdup_printf ("%s.log", display_server_get_name (display_server));
203 dir = config_get_string (config_get_instance (), "LightDM", "log-directory");
204 log_file = g_build_filename (dir, filename, NULL);
205 process_set_log_file (server->priv->x_server_process, log_file, FALSE);
206 l_debug (display_server, "Logging to %s", log_file);
211 absolute_command = get_absolute_command (server->priv->command);
212 if (!absolute_command)
214 l_debug (display_server, "Can't launch X server %s, not found in path", server->priv->command);
215 stopped_cb (server->priv->x_server_process, X_SERVER_XVNC (server));
219 gethostname (hostname, 1024);
220 number = g_strdup_printf ("%d", x_server_get_display_number (X_SERVER (server)));
221 authority = x_authority_new_cookie (XAUTH_FAMILY_LOCAL, (guint8*) hostname, strlen (hostname), number);
223 x_server_set_authority (X_SERVER (server), authority);
225 run_dir = config_get_string (config_get_instance (), "LightDM", "run-directory");
226 dir = g_build_filename (run_dir, "root", NULL);
228 if (g_mkdir_with_parents (dir, S_IRWXU) < 0)
229 l_warning (display_server, "Failed to make authority directory %s: %s", dir, strerror (errno));
231 server->priv->authority_file = g_build_filename (dir, x_server_get_address (X_SERVER (server)), NULL);
234 l_debug (display_server, "Writing X server authority to %s", server->priv->authority_file);
236 x_authority_write (authority, XAUTH_WRITE_MODE_REPLACE, server->priv->authority_file, &error);
238 l_warning (display_server, "Failed to write authority: %s", error->message);
239 g_clear_error (&error);
241 command = g_string_new (absolute_command);
242 g_free (absolute_command);
244 g_string_append_printf (command, " :%d", x_server_get_display_number (X_SERVER (server)));
245 g_string_append_printf (command, " -auth %s", server->priv->authority_file);
246 g_string_append (command, " -inetd -nolisten tcp");
247 if (server->priv->width > 0 && server->priv->height > 0)
248 g_string_append_printf (command, " -geometry %dx%d", server->priv->width, server->priv->height);
249 if (server->priv->depth > 0)
250 g_string_append_printf (command, " -depth %d", server->priv->depth);
252 process_set_command (server->priv->x_server_process, command->str);
253 g_string_free (command, TRUE);
255 l_debug (display_server, "Launching Xvnc server");
257 /* Variable required for regression tests */
258 if (g_getenv ("LIGHTDM_TEST_ROOT"))
260 process_set_env (server->priv->x_server_process, "LIGHTDM_TEST_ROOT", g_getenv ("LIGHTDM_TEST_ROOT"));
261 process_set_env (server->priv->x_server_process, "LD_LIBRARY_PATH", g_getenv ("LD_LIBRARY_PATH"));
264 result = process_start (server->priv->x_server_process, FALSE);
267 l_debug (display_server, "Waiting for ready signal from Xvnc server :%d", x_server_get_display_number (X_SERVER (server)));
270 stopped_cb (server->priv->x_server_process, X_SERVER_XVNC (server));
276 x_server_xvnc_stop (DisplayServer *server)
278 process_stop (X_SERVER_XVNC (server)->priv->x_server_process);
282 x_server_xvnc_init (XServerXVNC *server)
284 server->priv = G_TYPE_INSTANCE_GET_PRIVATE (server, X_SERVER_XVNC_TYPE, XServerXVNCPrivate);
285 server->priv->command = g_strdup ("Xvnc");
286 server->priv->width = 1024;
287 server->priv->height = 768;
288 server->priv->depth = 8;
292 x_server_xvnc_finalize (GObject *object)
294 XServerXVNC *self = X_SERVER_XVNC (object);
296 g_clear_object (&self->priv->x_server_process);
297 g_free (self->priv->command);
298 g_free (self->priv->authority_file);
300 G_OBJECT_CLASS (x_server_xvnc_parent_class)->finalize (object);
304 x_server_xvnc_class_init (XServerXVNCClass *klass)
306 GObjectClass *object_class = G_OBJECT_CLASS (klass);
307 DisplayServerClass *display_server_class = DISPLAY_SERVER_CLASS (klass);
309 display_server_class->get_can_share = x_server_xvnc_get_can_share;
310 display_server_class->start = x_server_xvnc_start;
311 display_server_class->stop = x_server_xvnc_stop;
312 object_class->finalize = x_server_xvnc_finalize;
314 g_type_class_add_private (klass, sizeof (XServerXVNCPrivate));