]> rtime.felk.cvut.cz Git - sojka/lightdm.git/blob - src/seat-xlocal.c
Merge with trunk
[sojka/lightdm.git] / src / seat-xlocal.c
1 /*
2  * Copyright (C) 2010-2011 Robert Ancell.
3  * Author: Robert Ancell <robert.ancell@canonical.com>
4  * 
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
9  * license.
10  */
11
12 #include <string.h>
13
14 #include "seat-xlocal.h"
15 #include "configuration.h"
16 #include "x-server-local.h"
17 #include "mir-server.h"
18 #include "plymouth.h"
19 #include "vt.h"
20
21 G_DEFINE_TYPE (SeatXLocal, seat_xlocal, SEAT_TYPE);
22
23 static gboolean
24 seat_xlocal_get_start_local_sessions (Seat *seat)
25 {
26     return !seat_get_string_property (seat, "xdmcp-manager");
27 }
28
29 static void
30 seat_xlocal_setup (Seat *seat)
31 {
32     seat_set_can_switch (seat, TRUE);
33     seat_set_share_display_server (seat, seat_get_boolean_property (seat, "xserver-share"));
34     SEAT_CLASS (seat_xlocal_parent_class)->setup (seat);
35 }
36
37 static gboolean
38 seat_xlocal_start (Seat *seat)
39 {
40    
41     return SEAT_CLASS (seat_xlocal_parent_class)->start (seat);
42 }
43
44 static void
45 x_server_ready_cb (XServerLocal *x_server, Seat *seat)
46 {
47     /* Quit Plymouth */
48     plymouth_quit (TRUE);
49 }
50
51 static void
52 x_server_transition_plymouth_cb (XServerLocal *x_server, Seat *seat)
53 {
54     /* Quit Plymouth if we didn't do the transition */
55     if (plymouth_get_is_running ())
56         plymouth_quit (FALSE);
57
58     g_signal_handlers_disconnect_matched (x_server, G_SIGNAL_MATCH_FUNC, 0, 0, NULL, x_server_transition_plymouth_cb, NULL);
59 }
60
61 static DisplayServer *
62 create_x_server (Seat *seat)
63 {
64     XServerLocal *x_server;
65     const gchar *command = NULL, *layout = NULL, *config_file = NULL, *xdmcp_manager = NULL, *key_name = NULL;
66     gboolean allow_tcp;
67     gint vt = -1, port = 0;
68
69     if (vt > 0)
70         g_debug ("Starting local X display on VT %d", vt);
71     else
72         g_debug ("Starting local X display");
73   
74     x_server = x_server_local_new ();
75
76     /* If running inside an X server use Xephyr instead */
77     if (g_getenv ("DISPLAY"))
78         command = "Xephyr";
79     if (!command)
80         command = seat_get_string_property (seat, "xserver-command");
81     if (command)
82         x_server_local_set_command (x_server, command);
83
84     /* If Plymouth is running, stop it */
85     if (plymouth_get_is_active () && plymouth_has_active_vt ())
86     {
87         gint active_vt = vt_get_active ();
88         if (active_vt >= vt_get_min ())
89         {
90             vt = active_vt;
91             g_signal_connect (x_server, "ready", G_CALLBACK (x_server_ready_cb), seat);
92             g_signal_connect (x_server, "stopped", G_CALLBACK (x_server_transition_plymouth_cb), seat);
93             plymouth_deactivate ();
94         }
95         else
96             g_debug ("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 ());
97     }
98     if (plymouth_get_is_active ())
99         plymouth_quit (FALSE);
100     if (vt < 0)
101         vt = vt_get_unused ();
102     if (vt >= 0)
103         x_server_local_set_vt (x_server, vt);
104
105     layout = seat_get_string_property (seat, "xserver-layout");
106     if (layout)
107         x_server_local_set_layout (x_server, layout);
108
109     config_file = seat_get_string_property (seat, "xserver-config");
110     if (config_file)
111         x_server_local_set_config (x_server, config_file);
112   
113     allow_tcp = seat_get_boolean_property (seat, "xserver-allow-tcp");
114     x_server_local_set_allow_tcp (x_server, allow_tcp);    
115
116     xdmcp_manager = seat_get_string_property (seat, "xdmcp-manager");
117     if (xdmcp_manager)
118         x_server_local_set_xdmcp_server (x_server, xdmcp_manager);
119
120     port = seat_get_integer_property (seat, "xdmcp-port");
121     if (port > 0)
122         x_server_local_set_xdmcp_port (x_server, port);
123
124     key_name = seat_get_string_property (seat, "xdmcp-key");
125     if (key_name)
126     {
127         gchar *dir, *path;
128         GKeyFile *keys;
129         gboolean result;
130         GError *error = NULL;
131
132         dir = config_get_string (config_get_instance (), "LightDM", "config-directory");
133         path = g_build_filename (dir, "keys.conf", NULL);
134         g_free (dir);
135
136         keys = g_key_file_new ();
137         result = g_key_file_load_from_file (keys, path, G_KEY_FILE_NONE, &error);
138         if (error)
139             g_debug ("Error getting key %s", error->message);
140         g_clear_error (&error);      
141
142         if (result)
143         {
144             gchar *key = NULL;
145
146             if (g_key_file_has_key (keys, "keyring", key_name, NULL))
147                 key = g_key_file_get_string (keys, "keyring", key_name, NULL);
148             else
149                 g_debug ("Key %s not defined", key_name);
150
151             if (key)
152                 x_server_local_set_xdmcp_key (x_server, key);
153             g_free (key);
154         }
155
156         g_free (path);
157         g_key_file_free (keys);
158     }
159
160     return DISPLAY_SERVER (x_server);
161 }
162
163 static DisplayServer *
164 create_mir_server (Seat *seat)
165 {
166     MirServer *mir_server;
167
168     mir_server = mir_server_new ();
169     mir_server_set_vt (mir_server, vt_get_unused ());
170
171     return DISPLAY_SERVER (mir_server);
172 }
173
174 static DisplayServer *
175 seat_xlocal_create_display_server (Seat *seat, const gchar *session_type)
176 {  
177     if (strcmp (session_type, "x") == 0)
178         return create_x_server (seat);
179     else if (strcmp (session_type, "mir") == 0)
180         return create_mir_server (seat);
181     else
182     {
183         g_warning ("Can't create unsupported display server '%s'", session_type);
184         return NULL;
185     }
186 }
187
188 static Greeter *
189 seat_xlocal_create_greeter_session (Seat *seat)
190 {
191     Greeter *greeter_session;
192
193     greeter_session = SEAT_CLASS (seat_xlocal_parent_class)->create_greeter_session (seat);
194     session_set_env (SESSION (greeter_session), "XDG_SEAT", "seat0");
195
196     return greeter_session;
197 }
198
199 static Session *
200 seat_xlocal_create_session (Seat *seat)
201 {
202     Session *session;
203
204     session = SEAT_CLASS (seat_xlocal_parent_class)->create_session (seat);
205     session_set_env (SESSION (session), "XDG_SEAT", "seat0");
206
207     return session;
208 }
209
210 static void
211 seat_xlocal_set_active_session (Seat *seat, Session *session)
212 {
213     gint vt = display_server_get_vt (session_get_display_server (session));
214     if (vt >= 0)
215         vt_set_active (vt);
216
217     SEAT_CLASS (seat_xlocal_parent_class)->set_active_session (seat, session);
218 }
219
220 static Session *
221 seat_xlocal_get_active_session (Seat *seat)
222 {
223     gint vt;
224     GList *link;
225
226     vt = vt_get_active ();
227     if (vt < 0)
228         return NULL;
229
230     for (link = seat_get_sessions (seat); link; link = link->next)
231     {
232         Session *session = link->data;
233         if (display_server_get_vt (session_get_display_server (session)) == vt)
234             return session;
235     }
236
237     return NULL;
238 }
239
240 static void
241 seat_xlocal_run_script (Seat *seat, DisplayServer *display_server, Process *script)
242 {
243     const gchar *path;
244     XServerLocal *x_server;
245
246     x_server = X_SERVER_LOCAL (display_server);
247     path = x_server_local_get_authority_file_path (x_server);
248     process_set_env (script, "DISPLAY", x_server_get_address (X_SERVER (x_server)));
249     process_set_env (script, "XAUTHORITY", path);
250
251     SEAT_CLASS (seat_xlocal_parent_class)->run_script (seat, display_server, script);
252 }
253
254 static void
255 seat_xlocal_init (SeatXLocal *seat)
256 {
257 }
258
259 static void
260 seat_xlocal_class_init (SeatXLocalClass *klass)
261 {
262     SeatClass *seat_class = SEAT_CLASS (klass);
263
264     seat_class->get_start_local_sessions = seat_xlocal_get_start_local_sessions;
265     seat_class->setup = seat_xlocal_setup;
266     seat_class->start = seat_xlocal_start;
267     seat_class->create_display_server = seat_xlocal_create_display_server;
268     seat_class->create_greeter_session = seat_xlocal_create_greeter_session;
269     seat_class->create_session = seat_xlocal_create_session;
270     seat_class->set_active_session = seat_xlocal_set_active_session;
271     seat_class->get_active_session = seat_xlocal_get_active_session;
272     seat_class->run_script = seat_xlocal_run_script;
273 }