]> rtime.felk.cvut.cz Git - sojka/lightdm.git/blob - src/x-server.c
Merge with trunk
[sojka/lightdm.git] / src / x-server.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 <config.h>
13 #include <string.h>
14 #include <xcb/xcb.h>
15
16 #include "x-server.h"
17 #include "configuration.h"
18
19 struct XServerPrivate
20 {  
21     /* Host running the server */
22     gchar *hostname;
23
24     /* Display number */
25     guint number;
26
27     /* Cached server address */
28     gchar *address;
29
30     /* Authority */
31     XAuthority *authority;
32
33     /* Connection to this X server */
34     xcb_connection_t *connection;
35 };
36
37 G_DEFINE_TYPE (XServer, x_server, DISPLAY_SERVER_TYPE);
38
39 void
40 x_server_set_hostname (XServer *server, const gchar *hostname)
41 {
42     g_return_if_fail (server != NULL);
43     g_free (server->priv->hostname);
44     server->priv->hostname = g_strdup (hostname);
45     g_free (server->priv->address);
46     server->priv->address = NULL;
47 }
48
49 gchar *
50 x_server_get_hostname (XServer *server)
51 {
52     g_return_val_if_fail (server != NULL, NULL);
53     return server->priv->hostname;
54 }
55
56 void
57 x_server_set_display_number (XServer *server, guint number)
58 {
59     g_return_if_fail (server != NULL);
60     server->priv->number = number;
61     g_free (server->priv->address);
62     server->priv->address = NULL;
63 }
64
65 guint
66 x_server_get_display_number (XServer *server)
67 {
68     g_return_val_if_fail (server != NULL, 0);
69     return server->priv->number;
70 }
71
72 const gchar *
73 x_server_get_address (XServer *server)
74 {
75     g_return_val_if_fail (server != NULL, NULL);
76
77     if (!server->priv->address)
78     {
79         if (server->priv->hostname)
80             server->priv->address = g_strdup_printf("%s:%d", server->priv->hostname, server->priv->number);
81         else
82             server->priv->address = g_strdup_printf(":%d", server->priv->number);
83     }  
84
85     return server->priv->address;
86 }
87
88 void
89 x_server_set_authority (XServer *server, XAuthority *authority)
90 {
91     g_return_if_fail (server != NULL);
92
93     if (server->priv->authority)
94         g_object_unref (server->priv->authority);
95     if (authority)
96         server->priv->authority = g_object_ref (authority);
97     else
98         server->priv->authority = NULL;
99 }
100
101 XAuthority *
102 x_server_get_authority (XServer *server)
103 {
104     g_return_val_if_fail (server != NULL, NULL);
105     return server->priv->authority;
106 }
107
108 static const gchar *
109 x_server_get_session_type (DisplayServer *server)
110 {
111     return "x";
112 }
113
114 static gboolean
115 x_server_get_can_share (DisplayServer *server)
116 {
117     return TRUE;
118 }
119
120 static gboolean
121 x_server_start (DisplayServer *display_server)
122 {
123     XServer *server = X_SERVER (display_server);
124     xcb_auth_info_t *auth = NULL, a;
125
126     if (server->priv->authority)
127     {
128         a.namelen = strlen (x_authority_get_authorization_name (server->priv->authority));
129         a.name = (char *) x_authority_get_authorization_name (server->priv->authority);
130         a.datalen = x_authority_get_authorization_data_length (server->priv->authority);
131         a.data = (char *) x_authority_get_authorization_data (server->priv->authority);
132         auth = &a;
133     }
134
135     /* Open connection */  
136     l_debug (server, "Connecting to XServer %s", x_server_get_address (server));
137     server->priv->connection = xcb_connect_to_display_with_auth_info (x_server_get_address (server), auth, NULL);
138     if (xcb_connection_has_error (server->priv->connection))
139     {
140         l_debug (server, "Error connecting to XServer %s", x_server_get_address (server));
141         return FALSE;
142     }
143
144     return DISPLAY_SERVER_CLASS (x_server_parent_class)->start (display_server);
145 }
146
147 static void
148 x_server_connect_session (DisplayServer *display_server, Session *session)
149 {
150     gint vt;
151
152     session_set_env (session, "XDG_SESSION_TYPE", "x11");
153
154     display_server = session_get_display_server (session);
155
156     vt = display_server_get_vt (display_server);
157     if (vt > 0)
158     {
159         gchar *t;
160
161         t = g_strdup_printf ("/dev/tty%d", vt);
162         session_set_tty (session, t);
163         g_free (t);
164
165         t = g_strdup_printf ("%d", vt);
166         session_set_env (session, "XDG_VTNR", t);
167         g_free (t);
168     }
169     else
170         l_debug (session, "Not setting XDG_VTNR");
171
172     session_set_env (session, "DISPLAY", x_server_get_address (X_SERVER (display_server)));
173     session_set_xdisplay (session, x_server_get_address (X_SERVER (display_server)));
174     session_set_remote_host_name (session, x_server_get_hostname (X_SERVER (display_server)));
175     session_set_x_authority (session,
176                              x_server_get_authority (X_SERVER (display_server)),
177                              config_get_boolean (config_get_instance (), "LightDM", "user-authority-in-system-dir"));
178 }
179
180 static void
181 x_server_disconnect_session (DisplayServer *display_server, Session *session)
182 {
183     gint vt;
184
185     session_unset_env (session, "XDG_SESSION_TYPE");
186     vt = display_server_get_vt (display_server);
187     if (vt > 0)
188     {
189         session_set_tty (session, NULL);
190         session_unset_env (session, "XDG_VTNR");
191     }
192     session_unset_env (session, "DISPLAY");
193     session_set_xdisplay (session, NULL);
194     session_set_remote_host_name (session, NULL);
195     session_set_x_authority (session, NULL, FALSE);
196 }
197
198 void
199 x_server_init (XServer *server)
200 {
201     server->priv = G_TYPE_INSTANCE_GET_PRIVATE (server, X_SERVER_TYPE, XServerPrivate);
202 }
203
204 static void
205 x_server_finalize (GObject *object)
206 {
207     XServer *self;
208
209     self = X_SERVER (object);
210
211     g_free (self->priv->hostname);
212     g_free (self->priv->address);
213     if (self->priv->authority)
214         g_object_unref (self->priv->authority);
215     if (self->priv->connection)
216         xcb_disconnect (self->priv->connection);
217
218     G_OBJECT_CLASS (x_server_parent_class)->finalize (object);
219 }
220
221 static void
222 x_server_class_init (XServerClass *klass)
223 {
224     GObjectClass *object_class = G_OBJECT_CLASS (klass);
225     DisplayServerClass *display_server_class = DISPLAY_SERVER_CLASS (klass);
226
227     display_server_class->get_session_type = x_server_get_session_type;    
228     display_server_class->get_can_share = x_server_get_can_share;
229     display_server_class->start = x_server_start;
230     display_server_class->connect_session = x_server_connect_session;
231     display_server_class->disconnect_session = x_server_disconnect_session;
232     object_class->finalize = x_server_finalize;
233
234     g_type_class_add_private (klass, sizeof (XServerPrivate));
235 }