43810
|
1 /* Session management module for systems which understand the X Session
|
|
2 management protocol.
|
100951
|
3 Copyright (C) 2002, 2003, 2004, 2005, 2006, 2007, 2008, 2009
|
79759
|
4 Free Software Foundation, Inc.
|
43810
|
5
|
|
6 This file is part of GNU Emacs.
|
|
7
|
94963
|
8 GNU Emacs is free software: you can redistribute it and/or modify
|
43810
|
9 it under the terms of the GNU General Public License as published by
|
94963
|
10 the Free Software Foundation, either version 3 of the License, or
|
|
11 (at your option) any later version.
|
43810
|
12
|
|
13 GNU Emacs is distributed in the hope that it will be useful,
|
|
14 but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
15 MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
16 GNU General Public License for more details.
|
|
17
|
|
18 You should have received a copy of the GNU General Public License
|
94963
|
19 along with GNU Emacs. If not, see <http://www.gnu.org/licenses/>. */
|
43810
|
20
|
|
21 #include <config.h>
|
|
22
|
|
23 #ifdef HAVE_X_SM
|
|
24
|
|
25 #include <X11/SM/SMlib.h>
|
52298
1a4bd2bec861
Create and initialize a client leader window so session management
Jan Djärv <jan.h.d@swipnet.se>
diff
changeset
|
26 #include <X11/Xlib.h>
|
1a4bd2bec861
Create and initialize a client leader window so session management
Jan Djärv <jan.h.d@swipnet.se>
diff
changeset
|
27 #include <X11/Xutil.h>
|
1a4bd2bec861
Create and initialize a client leader window so session management
Jan Djärv <jan.h.d@swipnet.se>
diff
changeset
|
28
|
43810
|
29 #ifdef HAVE_UNISTD_H
|
|
30 #include <unistd.h>
|
|
31 #endif
|
|
32
|
|
33 #include <sys/param.h>
|
43848
|
34 #include <stdio.h>
|
105669
|
35 #include <setjmp.h>
|
43810
|
36
|
65764
375ab086d366
* image.c (slurp_file, xbm_read_bitmap_data): Cast to the correct
Dan Nicolaescu <dann@ics.uci.edu>
diff
changeset
|
37 #include "lisp.h"
|
43810
|
38 #include "systime.h"
|
|
39 #include "sysselect.h"
|
83004
7900111db01c
Converted display hooks to be display-local. Plus many bugfixes.
Karoly Lorentey <lorentey@elte.hu>
diff
changeset
|
40 #include "frame.h"
|
43810
|
41 #include "termhooks.h"
|
43848
|
42 #include "termopts.h"
|
52298
1a4bd2bec861
Create and initialize a client leader window so session management
Jan Djärv <jan.h.d@swipnet.se>
diff
changeset
|
43 #include "xterm.h"
|
43810
|
44
|
|
45 /* The user login name. */
|
|
46
|
|
47 extern Lisp_Object Vuser_login_name;
|
|
48
|
45803
|
49 /* This is the event used when SAVE_SESSION_EVENT occurs. */
|
43810
|
50
|
|
51 static struct input_event emacs_event;
|
|
52
|
45803
|
53 /* The descriptor that we use to check for data from the session manager. */
|
43810
|
54
|
|
55 static int ice_fd = -1;
|
|
56
|
45803
|
57 /* A flag that says if we are in shutdown interactions or not. */
|
43810
|
58
|
|
59 static int doing_interact = False;
|
|
60
|
57704
|
61 /* The session manager object for the session manager connection. */
|
43810
|
62
|
|
63 static SmcConn smc_conn;
|
|
64
|
57704
|
65 /* The client session id for this session. */
|
|
66
|
43810
|
67 static char *client_id;
|
|
68
|
57704
|
69 /* The full path name to the Emacs program. */
|
|
70
|
43810
|
71 static char *emacs_program;
|
|
72
|
57704
|
73 /* The client session id for this session as a lisp object. */
|
43810
|
74
|
|
75 Lisp_Object Vx_session_id;
|
|
76
|
|
77 /* The id we had the previous session. This is only available if we
|
57704
|
78 have been started by the session manager with SMID_OPT. */
|
43810
|
79
|
|
80 Lisp_Object Vx_session_previous_id;
|
|
81
|
|
82 /* The option we tell the session manager to start Emacs with when
|
57704
|
83 restarting Emacs. The client_id is appended. */
|
43810
|
84
|
|
85 #define SMID_OPT "--smid="
|
|
86
|
|
87
|
50016
|
88 /* The option to start Emacs without the splash screen when
|
57704
|
89 restarting Emacs. */
|
50016
|
90
|
|
91 #define NOSPLASH_OPT "--no-splash"
|
|
92
|
|
93
|
43810
|
94 /* Handle any messages from the session manager. If no connection is
|
|
95 open to a session manager, just return 0.
|
54172
|
96 Otherwise returns 1 if SAVE_SESSION_EVENT is stored in buffer BUFP. */
|
57704
|
97
|
43810
|
98 int
|
54172
|
99 x_session_check_input (bufp)
|
43810
|
100 struct input_event *bufp;
|
|
101 {
|
|
102 SELECT_TYPE read_fds;
|
|
103 EMACS_TIME tmout;
|
47942
|
104
|
43810
|
105 if (ice_fd == -1) return 0;
|
47942
|
106
|
43810
|
107 FD_ZERO (&read_fds);
|
|
108 FD_SET (ice_fd, &read_fds);
|
47942
|
109
|
43810
|
110 tmout.tv_sec = 0;
|
|
111 tmout.tv_usec = 0;
|
47942
|
112
|
43810
|
113 /* Reset this so wo can check kind after callbacks have been called by
|
|
114 IceProcessMessages. The smc_interact_CB sets the kind to
|
45803
|
115 SAVE_SESSION_EVENT, but we don't know beforehand if that callback
|
57704
|
116 will be called. */
|
45803
|
117 emacs_event.kind = NO_EVENT;
|
43810
|
118
|
|
119 if (select (ice_fd+1, &read_fds,
|
|
120 (SELECT_TYPE *)0, (SELECT_TYPE *)0, &tmout) < 0)
|
|
121 {
|
|
122 ice_fd = -1;
|
|
123 return 0;
|
|
124 }
|
47942
|
125
|
43810
|
126
|
|
127 if (FD_ISSET (ice_fd, &read_fds))
|
|
128 IceProcessMessages (SmcGetIceConnection (smc_conn),
|
|
129 (IceReplyWaitInfo *)0, (Bool *)0);
|
|
130
|
47942
|
131
|
43810
|
132 /* Check if smc_interact_CB was called and we shall generate a
|
57704
|
133 SAVE_SESSION_EVENT. */
|
54172
|
134 if (emacs_event.kind == NO_EVENT)
|
|
135 return 0;
|
43810
|
136
|
54172
|
137 bcopy (&emacs_event, bufp, sizeof (struct input_event));
|
|
138 return 1;
|
43810
|
139 }
|
|
140
|
57704
|
141 /* Return non-zero if we have a connection to a session manager. */
|
|
142
|
43810
|
143 int
|
|
144 x_session_have_connection ()
|
|
145 {
|
|
146 return ice_fd != -1;
|
|
147 }
|
|
148
|
|
149 /* This is called when the session manager says it is OK to interact with the
|
45803
|
150 user. Here we set the kind to SAVE_SESSION_EVENT so an event is generated.
|
57704
|
151 Then lisp code can interact with the user. */
|
|
152
|
43810
|
153 static void
|
|
154 smc_interact_CB (smcConn, clientData)
|
|
155 SmcConn smcConn;
|
|
156 SmPointer clientData;
|
|
157 {
|
|
158 doing_interact = True;
|
45803
|
159 emacs_event.kind = SAVE_SESSION_EVENT;
|
43810
|
160 }
|
|
161
|
49421
|
162 /* This is called when the session manager tells us to save ourselves.
|
43810
|
163 We set the required properties so the session manager can restart us,
|
|
164 plus the current working directory property (not mandatory) so we
|
|
165 are started in the correct directory.
|
|
166
|
|
167 If this is a shutdown and we can request to interact with the user,
|
57704
|
168 we do so, because we don't know what the lisp code might do. */
|
|
169
|
43810
|
170 static void
|
|
171 smc_save_yourself_CB (smcConn,
|
|
172 clientData,
|
|
173 saveType,
|
|
174 shutdown,
|
|
175 interactStyle,
|
|
176 fast)
|
|
177 SmcConn smcConn;
|
|
178 SmPointer clientData;
|
|
179 int saveType;
|
|
180 Bool shutdown;
|
|
181 int interactStyle;
|
|
182 Bool fast;
|
|
183 {
|
|
184 #define NR_PROPS 5
|
47942
|
185
|
43810
|
186 SmProp *props[NR_PROPS];
|
|
187 SmProp prop_ptr[NR_PROPS];
|
47942
|
188
|
43810
|
189 SmPropValue values[20];
|
|
190 int val_idx = 0;
|
|
191 int props_idx = 0;
|
47942
|
192
|
65412
|
193 char *cwd = NULL;
|
43810
|
194 char *smid_opt;
|
|
195
|
57704
|
196 /* How to start a new instance of Emacs. */
|
43810
|
197 props[props_idx] = &prop_ptr[props_idx];
|
|
198 props[props_idx]->name = SmCloneCommand;
|
|
199 props[props_idx]->type = SmLISTofARRAY8;
|
|
200 props[props_idx]->num_vals = 1;
|
|
201 props[props_idx]->vals = &values[val_idx++];
|
|
202 props[props_idx]->vals[0].length = strlen (emacs_program);
|
|
203 props[props_idx]->vals[0].value = emacs_program;
|
|
204 ++props_idx;
|
|
205
|
57704
|
206 /* The name of the program. */
|
43810
|
207 props[props_idx] = &prop_ptr[props_idx];
|
|
208 props[props_idx]->name = SmProgram;
|
|
209 props[props_idx]->type = SmARRAY8;
|
|
210 props[props_idx]->num_vals = 1;
|
|
211 props[props_idx]->vals = &values[val_idx++];
|
46370
40db0673e6f0
Most uses of XSTRING combined with STRING_BYTES or indirection changed to
Ken Raeburn <raeburn@raeburn.org>
diff
changeset
|
212 props[props_idx]->vals[0].length = strlen (SDATA (Vinvocation_name));
|
40db0673e6f0
Most uses of XSTRING combined with STRING_BYTES or indirection changed to
Ken Raeburn <raeburn@raeburn.org>
diff
changeset
|
213 props[props_idx]->vals[0].value = SDATA (Vinvocation_name);
|
43810
|
214 ++props_idx;
|
47942
|
215
|
57704
|
216 /* How to restart Emacs (i.e.: /path/to/emacs --smid=xxxx --no-splash). */
|
43810
|
217 props[props_idx] = &prop_ptr[props_idx];
|
|
218 props[props_idx]->name = SmRestartCommand;
|
|
219 props[props_idx]->type = SmLISTofARRAY8;
|
57704
|
220 props[props_idx]->num_vals = 3; /* /path/to/emacs, --smid=xxx --no-splash */
|
43810
|
221 props[props_idx]->vals = &values[val_idx];
|
|
222 props[props_idx]->vals[0].length = strlen (emacs_program);
|
|
223 props[props_idx]->vals[0].value = emacs_program;
|
|
224
|
|
225 smid_opt = xmalloc (strlen (SMID_OPT) + strlen (client_id) + 1);
|
|
226 strcpy (smid_opt, SMID_OPT);
|
|
227 strcat (smid_opt, client_id);
|
47942
|
228
|
43810
|
229 props[props_idx]->vals[1].length = strlen (smid_opt);
|
|
230 props[props_idx]->vals[1].value = smid_opt;
|
50016
|
231
|
|
232 props[props_idx]->vals[2].length = strlen (NOSPLASH_OPT);
|
|
233 props[props_idx]->vals[2].value = NOSPLASH_OPT;
|
|
234 val_idx += 3;
|
43810
|
235 ++props_idx;
|
|
236
|
57704
|
237 /* User id. */
|
43810
|
238 props[props_idx] = &prop_ptr[props_idx];
|
|
239 props[props_idx]->name = SmUserID;
|
|
240 props[props_idx]->type = SmARRAY8;
|
|
241 props[props_idx]->num_vals = 1;
|
|
242 props[props_idx]->vals = &values[val_idx++];
|
46370
40db0673e6f0
Most uses of XSTRING combined with STRING_BYTES or indirection changed to
Ken Raeburn <raeburn@raeburn.org>
diff
changeset
|
243 props[props_idx]->vals[0].length = strlen (SDATA (Vuser_login_name));
|
40db0673e6f0
Most uses of XSTRING combined with STRING_BYTES or indirection changed to
Ken Raeburn <raeburn@raeburn.org>
diff
changeset
|
244 props[props_idx]->vals[0].value = SDATA (Vuser_login_name);
|
43810
|
245 ++props_idx;
|
|
246
|
65412
|
247 cwd = get_current_dir_name ();
|
|
248
|
|
249 if (cwd)
|
43810
|
250 {
|
|
251 props[props_idx] = &prop_ptr[props_idx];
|
|
252 props[props_idx]->name = SmCurrentDirectory;
|
|
253 props[props_idx]->type = SmARRAY8;
|
|
254 props[props_idx]->num_vals = 1;
|
|
255 props[props_idx]->vals = &values[val_idx++];
|
|
256 props[props_idx]->vals[0].length = strlen (cwd);
|
|
257 props[props_idx]->vals[0].value = cwd;
|
|
258 ++props_idx;
|
|
259 }
|
47942
|
260
|
|
261
|
43810
|
262 SmcSetProperties (smcConn, props_idx, props);
|
|
263
|
|
264 xfree (smid_opt);
|
|
265
|
95479
|
266 free (cwd);
|
65412
|
267
|
57704
|
268 /* See if we maybe shall interact with the user. */
|
43810
|
269 if (interactStyle != SmInteractStyleAny
|
|
270 || ! shutdown
|
|
271 || saveType == SmSaveLocal
|
|
272 || ! SmcInteractRequest (smcConn, SmDialogNormal, smc_interact_CB, 0))
|
|
273 {
|
57704
|
274 /* No interaction, we are done saving ourself. */
|
43810
|
275 SmcSaveYourselfDone (smcConn, True);
|
|
276 }
|
|
277 }
|
|
278
|
57704
|
279 /* According to the SM specification, this shall close the connection. */
|
|
280
|
43810
|
281 static void
|
|
282 smc_die_CB (smcConn, clientData)
|
|
283 SmcConn smcConn;
|
|
284 SmPointer clientData;
|
|
285 {
|
|
286 SmcCloseConnection (smcConn, 0, 0);
|
|
287 ice_fd = -1;
|
|
288 }
|
|
289
|
|
290 /* We don't use the next two but they are mandatory, leave them empty.
|
|
291 According to the SM specification, we should not interact with the
|
|
292 user between smc_save_yourself_CB is called and until smc_save_complete_CB
|
|
293 is called. It seems like a lot of job to implement this and it doesn't
|
57704
|
294 even seem necessary. */
|
|
295
|
43810
|
296 static void
|
|
297 smc_save_complete_CB (smcConn, clientData)
|
|
298 SmcConn smcConn;
|
|
299 SmPointer clientData;
|
|
300 {
|
|
301 /* Empty */
|
|
302 }
|
|
303
|
|
304 static void
|
|
305 smc_shutdown_cancelled_CB (smcConn, clientData)
|
|
306 SmcConn smcConn;
|
|
307 SmPointer clientData;
|
|
308 {
|
|
309 /* Empty */
|
|
310 }
|
|
311
|
49421
|
312 /* Error handlers for SM and ICE. We don't want to exit Emacs just
|
57704
|
313 because there is some error in the session management. */
|
|
314
|
43810
|
315 static void
|
|
316 smc_error_handler (smcConn,
|
|
317 swap,
|
|
318 offendingMinorOpcode,
|
|
319 offendingSequence,
|
|
320 errorClass,
|
|
321 severity,
|
|
322 values)
|
|
323 SmcConn smcConn;
|
|
324 Bool swap;
|
|
325 int offendingMinorOpcode;
|
|
326 unsigned long offendingSequence;
|
|
327 int errorClass;
|
|
328 int severity;
|
|
329 SmPointer values;
|
|
330 {
|
57704
|
331 /* Empty */
|
43810
|
332 }
|
|
333
|
|
334 static void
|
|
335 ice_error_handler (iceConn,
|
|
336 swap,
|
|
337 offendingMinorOpcode,
|
|
338 offendingSequence,
|
|
339 errorClass,
|
|
340 severity,
|
|
341 values)
|
|
342 IceConn iceConn;
|
|
343 Bool swap;
|
|
344 int offendingMinorOpcode;
|
|
345 unsigned long offendingSequence;
|
|
346 int errorClass;
|
|
347 int severity;
|
|
348 IcePointer values;
|
|
349 {
|
57704
|
350 /* Empty */
|
43810
|
351 }
|
|
352
|
|
353
|
|
354 static void
|
|
355 ice_io_error_handler (iceConn)
|
|
356 IceConn iceConn;
|
|
357 {
|
57704
|
358 /* Connection probably gone. */
|
43810
|
359 ice_fd = -1;
|
|
360 }
|
|
361
|
|
362 /* This is called when the ICE connection is created or closed. The SM library
|
57704
|
363 uses ICE as it transport protocol. */
|
|
364
|
43810
|
365 static void
|
|
366 ice_conn_watch_CB (iceConn, clientData, opening, watchData)
|
|
367 IceConn iceConn;
|
|
368 IcePointer clientData;
|
|
369 Bool opening;
|
|
370 IcePointer *watchData;
|
|
371 {
|
|
372 if (! opening)
|
|
373 {
|
|
374 ice_fd = -1;
|
|
375 return;
|
|
376 }
|
47942
|
377
|
43810
|
378 ice_fd = IceConnectionNumber (iceConn);
|
|
379 #ifdef F_SETOWN
|
|
380 fcntl (ice_fd, F_SETOWN, getpid ());
|
|
381 #endif /* ! defined (F_SETOWN) */
|
|
382
|
|
383 #ifdef SIGIO
|
|
384 if (interrupt_input)
|
|
385 init_sigio (ice_fd);
|
|
386 #endif /* ! defined (SIGIO) */
|
|
387 }
|
|
388
|
52298
1a4bd2bec861
Create and initialize a client leader window so session management
Jan Djärv <jan.h.d@swipnet.se>
diff
changeset
|
389 /* Create the client leader window. */
|
57704
|
390
|
52298
1a4bd2bec861
Create and initialize a client leader window so session management
Jan Djärv <jan.h.d@swipnet.se>
diff
changeset
|
391 static void
|
1a4bd2bec861
Create and initialize a client leader window so session management
Jan Djärv <jan.h.d@swipnet.se>
diff
changeset
|
392 create_client_leader_window (dpyinfo, client_id)
|
1a4bd2bec861
Create and initialize a client leader window so session management
Jan Djärv <jan.h.d@swipnet.se>
diff
changeset
|
393 struct x_display_info *dpyinfo;
|
1a4bd2bec861
Create and initialize a client leader window so session management
Jan Djärv <jan.h.d@swipnet.se>
diff
changeset
|
394 char *client_id;
|
1a4bd2bec861
Create and initialize a client leader window so session management
Jan Djärv <jan.h.d@swipnet.se>
diff
changeset
|
395 {
|
1a4bd2bec861
Create and initialize a client leader window so session management
Jan Djärv <jan.h.d@swipnet.se>
diff
changeset
|
396 Window w;
|
1a4bd2bec861
Create and initialize a client leader window so session management
Jan Djärv <jan.h.d@swipnet.se>
diff
changeset
|
397 XClassHint class_hints;
|
1a4bd2bec861
Create and initialize a client leader window so session management
Jan Djärv <jan.h.d@swipnet.se>
diff
changeset
|
398 Atom sm_id;
|
1a4bd2bec861
Create and initialize a client leader window so session management
Jan Djärv <jan.h.d@swipnet.se>
diff
changeset
|
399
|
1a4bd2bec861
Create and initialize a client leader window so session management
Jan Djärv <jan.h.d@swipnet.se>
diff
changeset
|
400 w = XCreateSimpleWindow (dpyinfo->display,
|
1a4bd2bec861
Create and initialize a client leader window so session management
Jan Djärv <jan.h.d@swipnet.se>
diff
changeset
|
401 dpyinfo->root_window,
|
1a4bd2bec861
Create and initialize a client leader window so session management
Jan Djärv <jan.h.d@swipnet.se>
diff
changeset
|
402 -1, -1, 1, 1,
|
1a4bd2bec861
Create and initialize a client leader window so session management
Jan Djärv <jan.h.d@swipnet.se>
diff
changeset
|
403 CopyFromParent, CopyFromParent, CopyFromParent);
|
1a4bd2bec861
Create and initialize a client leader window so session management
Jan Djärv <jan.h.d@swipnet.se>
diff
changeset
|
404
|
1a4bd2bec861
Create and initialize a client leader window so session management
Jan Djärv <jan.h.d@swipnet.se>
diff
changeset
|
405 class_hints.res_name = (char *) SDATA (Vx_resource_name);
|
1a4bd2bec861
Create and initialize a client leader window so session management
Jan Djärv <jan.h.d@swipnet.se>
diff
changeset
|
406 class_hints.res_class = (char *) SDATA (Vx_resource_class);
|
1a4bd2bec861
Create and initialize a client leader window so session management
Jan Djärv <jan.h.d@swipnet.se>
diff
changeset
|
407 XSetClassHint (dpyinfo->display, w, &class_hints);
|
1a4bd2bec861
Create and initialize a client leader window so session management
Jan Djärv <jan.h.d@swipnet.se>
diff
changeset
|
408 XStoreName (dpyinfo->display, w, class_hints.res_name);
|
1a4bd2bec861
Create and initialize a client leader window so session management
Jan Djärv <jan.h.d@swipnet.se>
diff
changeset
|
409
|
1a4bd2bec861
Create and initialize a client leader window so session management
Jan Djärv <jan.h.d@swipnet.se>
diff
changeset
|
410 sm_id = XInternAtom (dpyinfo->display, "SM_CLIENT_ID", False);
|
1a4bd2bec861
Create and initialize a client leader window so session management
Jan Djärv <jan.h.d@swipnet.se>
diff
changeset
|
411 XChangeProperty (dpyinfo->display, w, sm_id, XA_STRING, 8, PropModeReplace,
|
1a4bd2bec861
Create and initialize a client leader window so session management
Jan Djärv <jan.h.d@swipnet.se>
diff
changeset
|
412 client_id, strlen (client_id));
|
1a4bd2bec861
Create and initialize a client leader window so session management
Jan Djärv <jan.h.d@swipnet.se>
diff
changeset
|
413
|
1a4bd2bec861
Create and initialize a client leader window so session management
Jan Djärv <jan.h.d@swipnet.se>
diff
changeset
|
414 dpyinfo->client_leader_window = w;
|
1a4bd2bec861
Create and initialize a client leader window so session management
Jan Djärv <jan.h.d@swipnet.se>
diff
changeset
|
415 }
|
1a4bd2bec861
Create and initialize a client leader window so session management
Jan Djärv <jan.h.d@swipnet.se>
diff
changeset
|
416
|
57704
|
417 /* Try to open a connection to the session manager. */
|
|
418
|
43810
|
419 void
|
52298
1a4bd2bec861
Create and initialize a client leader window so session management
Jan Djärv <jan.h.d@swipnet.se>
diff
changeset
|
420 x_session_initialize (dpyinfo)
|
1a4bd2bec861
Create and initialize a client leader window so session management
Jan Djärv <jan.h.d@swipnet.se>
diff
changeset
|
421 struct x_display_info *dpyinfo;
|
43810
|
422 {
|
|
423 #define SM_ERRORSTRING_LEN 512
|
|
424 char errorstring[SM_ERRORSTRING_LEN];
|
|
425 char* previous_id = NULL;
|
|
426 SmcCallbacks callbacks;
|
|
427 int name_len = 0;
|
47942
|
428
|
43810
|
429 /* Check if we where started by the session manager. If so, we will
|
57704
|
430 have a previous id. */
|
43810
|
431 if (! EQ (Vx_session_previous_id, Qnil) && STRINGP (Vx_session_previous_id))
|
46370
40db0673e6f0
Most uses of XSTRING combined with STRING_BYTES or indirection changed to
Ken Raeburn <raeburn@raeburn.org>
diff
changeset
|
432 previous_id = SDATA (Vx_session_previous_id);
|
43810
|
433
|
57704
|
434 /* Construct the path to the Emacs program. */
|
43810
|
435 if (! EQ (Vinvocation_directory, Qnil))
|
46370
40db0673e6f0
Most uses of XSTRING combined with STRING_BYTES or indirection changed to
Ken Raeburn <raeburn@raeburn.org>
diff
changeset
|
436 name_len += strlen (SDATA (Vinvocation_directory));
|
40db0673e6f0
Most uses of XSTRING combined with STRING_BYTES or indirection changed to
Ken Raeburn <raeburn@raeburn.org>
diff
changeset
|
437 name_len += strlen (SDATA (Vinvocation_name));
|
43810
|
438
|
|
439 /* This malloc will not be freed, but it is only done once, and hopefully
|
57704
|
440 not very large */
|
43810
|
441 emacs_program = xmalloc (name_len + 1);
|
|
442 emacs_program[0] = '\0';
|
|
443
|
|
444 if (! EQ (Vinvocation_directory, Qnil))
|
46370
40db0673e6f0
Most uses of XSTRING combined with STRING_BYTES or indirection changed to
Ken Raeburn <raeburn@raeburn.org>
diff
changeset
|
445 strcpy (emacs_program, SDATA (Vinvocation_directory));
|
40db0673e6f0
Most uses of XSTRING combined with STRING_BYTES or indirection changed to
Ken Raeburn <raeburn@raeburn.org>
diff
changeset
|
446 strcat (emacs_program, SDATA (Vinvocation_name));
|
47942
|
447
|
43810
|
448 /* The SM protocol says all callbacks are mandatory, so set up all
|
57704
|
449 here and in the mask passed to SmcOpenConnection. */
|
43810
|
450 callbacks.save_yourself.callback = smc_save_yourself_CB;
|
|
451 callbacks.save_yourself.client_data = 0;
|
|
452 callbacks.die.callback = smc_die_CB;
|
|
453 callbacks.die.client_data = 0;
|
|
454 callbacks.save_complete.callback = smc_save_complete_CB;
|
|
455 callbacks.save_complete.client_data = 0;
|
|
456 callbacks.shutdown_cancelled.callback = smc_shutdown_cancelled_CB;
|
|
457 callbacks.shutdown_cancelled.client_data = 0;
|
|
458
|
57704
|
459 /* Set error handlers. */
|
43810
|
460 SmcSetErrorHandler (smc_error_handler);
|
|
461 IceSetErrorHandler (ice_error_handler);
|
|
462 IceSetIOErrorHandler (ice_io_error_handler);
|
|
463
|
57704
|
464 /* Install callback for when connection status changes. */
|
43810
|
465 IceAddConnectionWatch (ice_conn_watch_CB, 0);
|
|
466
|
|
467 /* Open the connection to the session manager. A failure is not
|
49421
|
468 critical, it usually means that no session manager is running.
|
57704
|
469 The errorstring is here for debugging. */
|
43810
|
470 smc_conn = SmcOpenConnection (NULL, NULL, 1, 0,
|
|
471 (SmcSaveYourselfProcMask|
|
|
472 SmcDieProcMask|
|
|
473 SmcSaveCompleteProcMask|
|
|
474 SmcShutdownCancelledProcMask),
|
|
475 &callbacks,
|
|
476 previous_id,
|
|
477 &client_id,
|
|
478 SM_ERRORSTRING_LEN,
|
|
479 errorstring);
|
|
480
|
|
481 if (smc_conn != 0)
|
52298
1a4bd2bec861
Create and initialize a client leader window so session management
Jan Djärv <jan.h.d@swipnet.se>
diff
changeset
|
482 {
|
1a4bd2bec861
Create and initialize a client leader window so session management
Jan Djärv <jan.h.d@swipnet.se>
diff
changeset
|
483 Vx_session_id = make_string (client_id, strlen (client_id));
|
1a4bd2bec861
Create and initialize a client leader window so session management
Jan Djärv <jan.h.d@swipnet.se>
diff
changeset
|
484
|
1a4bd2bec861
Create and initialize a client leader window so session management
Jan Djärv <jan.h.d@swipnet.se>
diff
changeset
|
485 #ifdef USE_GTK
|
1a4bd2bec861
Create and initialize a client leader window so session management
Jan Djärv <jan.h.d@swipnet.se>
diff
changeset
|
486 /* GTK creats a leader window by itself, but we need to tell
|
1a4bd2bec861
Create and initialize a client leader window so session management
Jan Djärv <jan.h.d@swipnet.se>
diff
changeset
|
487 it about our client_id. */
|
1a4bd2bec861
Create and initialize a client leader window so session management
Jan Djärv <jan.h.d@swipnet.se>
diff
changeset
|
488 gdk_set_sm_client_id (client_id);
|
1a4bd2bec861
Create and initialize a client leader window so session management
Jan Djärv <jan.h.d@swipnet.se>
diff
changeset
|
489 #else
|
1a4bd2bec861
Create and initialize a client leader window so session management
Jan Djärv <jan.h.d@swipnet.se>
diff
changeset
|
490 create_client_leader_window (dpyinfo, client_id);
|
1a4bd2bec861
Create and initialize a client leader window so session management
Jan Djärv <jan.h.d@swipnet.se>
diff
changeset
|
491 #endif
|
1a4bd2bec861
Create and initialize a client leader window so session management
Jan Djärv <jan.h.d@swipnet.se>
diff
changeset
|
492 }
|
43810
|
493 }
|
|
494
|
83350
e7d5238afe52
Work around crashes in X session management after normal shutdown of X server.
Karoly Lorentey <lorentey@elte.hu>
diff
changeset
|
495 /* Ensure that the session manager is not contacted again. */
|
e7d5238afe52
Work around crashes in X session management after normal shutdown of X server.
Karoly Lorentey <lorentey@elte.hu>
diff
changeset
|
496
|
e7d5238afe52
Work around crashes in X session management after normal shutdown of X server.
Karoly Lorentey <lorentey@elte.hu>
diff
changeset
|
497 void
|
e7d5238afe52
Work around crashes in X session management after normal shutdown of X server.
Karoly Lorentey <lorentey@elte.hu>
diff
changeset
|
498 x_session_close ()
|
e7d5238afe52
Work around crashes in X session management after normal shutdown of X server.
Karoly Lorentey <lorentey@elte.hu>
diff
changeset
|
499 {
|
e7d5238afe52
Work around crashes in X session management after normal shutdown of X server.
Karoly Lorentey <lorentey@elte.hu>
diff
changeset
|
500 ice_fd = -1;
|
e7d5238afe52
Work around crashes in X session management after normal shutdown of X server.
Karoly Lorentey <lorentey@elte.hu>
diff
changeset
|
501 }
|
e7d5238afe52
Work around crashes in X session management after normal shutdown of X server.
Karoly Lorentey <lorentey@elte.hu>
diff
changeset
|
502
|
43810
|
503
|
|
504 DEFUN ("handle-save-session", Fhandle_save_session,
|
|
505 Shandle_save_session, 1, 1, "e",
|
|
506 doc: /* Handle the save_yourself event from a session manager.
|
47942
|
507 A session manager can tell Emacs that the window system is shutting down
|
43810
|
508 by sending Emacs a save_yourself message. Emacs executes this function when
|
|
509 such an event occurs. This function then executes `emacs-session-save'.
|
|
510 After that, this function informs the session manager that it can continue
|
|
511 or abort shutting down the window system depending on the return value
|
|
512 from `emacs-session-save' If the return value is non-nil the session manager
|
|
513 is told to abort the window system shutdown.
|
|
514
|
|
515 Do not call this function yourself. */)
|
|
516 (event)
|
|
517 Lisp_Object event;
|
|
518 {
|
|
519 /* Check doing_interact so that we don't do anything if someone called
|
|
520 this at the wrong time. */
|
|
521 if (doing_interact)
|
|
522 {
|
|
523 Bool cancel_shutdown = False;
|
|
524
|
|
525 cancel_shutdown = ! EQ (call0 (intern ("emacs-session-save")), Qnil);
|
|
526
|
|
527 SmcInteractDone (smc_conn, cancel_shutdown);
|
|
528 SmcSaveYourselfDone (smc_conn, True);
|
|
529
|
|
530 doing_interact = False;
|
|
531 }
|
43919
833c0585a612
Added return value to Fhandle_save_session to avoid compiler warning.
Jan Djärv <jan.h.d@swipnet.se>
diff
changeset
|
532
|
833c0585a612
Added return value to Fhandle_save_session to avoid compiler warning.
Jan Djärv <jan.h.d@swipnet.se>
diff
changeset
|
533 return Qnil;
|
43810
|
534 }
|
|
535
|
|
536
|
|
537 /***********************************************************************
|
|
538 Initialization
|
|
539 ***********************************************************************/
|
|
540 void
|
|
541 syms_of_xsmfns ()
|
|
542 {
|
|
543 DEFVAR_LISP ("x-session-id", &Vx_session_id,
|
|
544 doc: /* The session id Emacs got from the session manager for this session.
|
|
545 Changing the value does not change the session id used by Emacs.
|
|
546 The value is nil if no session manager is running.
|
|
547 See also `x-session-previous-id', `emacs-save-session-functions',
|
|
548 `emacs-session-save' and `emacs-session-restore'." */);
|
|
549 Vx_session_id = Qnil;
|
|
550
|
|
551 DEFVAR_LISP ("x-session-previous-id", &Vx_session_previous_id,
|
|
552 doc: /* The previous session id Emacs got from session manager.
|
47942
|
553 If Emacs is running on a window system that has a session manager, the
|
73801
|
554 session manager gives Emacs a session id. It is feasible for Emacs Lisp
|
47942
|
555 code to use the session id to save configuration in, for example, a file
|
|
556 with a file name based on the session id. If Emacs is running when the
|
|
557 window system is shut down, the session manager remembers that Emacs was
|
43810
|
558 running and saves the session id Emacs had.
|
|
559
|
47942
|
560 When the window system is started again, the session manager restarts
|
|
561 Emacs and hands Emacs the session id it had the last time it was
|
|
562 running. This is now the previous session id and the value of this
|
|
563 variable. If configuration was saved in a file as stated above, the
|
43810
|
564 previous session id shall be used to reconstruct the file name.
|
|
565
|
47942
|
566 The session id Emacs has while it is running is in the variable
|
43810
|
567 `x-session-id'. The value of this variable and `x-session-id' may be the
|
|
568 same, depending on how the session manager works.
|
|
569
|
|
570 See also `emacs-save-session-functions', `emacs-session-save' and
|
|
571 `emacs-session-restore'." */);
|
|
572 Vx_session_previous_id = Qnil;
|
47942
|
573
|
43810
|
574 defsubr (&Shandle_save_session);
|
|
575 }
|
|
576
|
|
577 #endif /* HAVE_X_SM */
|
52401
|
578
|
|
579 /* arch-tag: 56a2c58c-adfa-430a-b772-130abd29fd2e
|
|
580 (do not change this comment) */
|