2 This file is part of PulseAudio.
4 Copyright 2004-2006 Lennart Poettering
6 PulseAudio is free software; you can redistribute it and/or modify
7 it under the terms of the GNU Lesser General Public License as published
8 by the Free Software Foundation; either version 2.1 of the License,
9 or (at your option) any later version.
11 PulseAudio is distributed in the hope that it will be useful, but
12 WITHOUT ANY WARRANTY; without even the implied warranty of
13 MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
14 General Public License for more details.
16 You should have received a copy of the GNU Lesser General Public License
17 along with PulseAudio; if not, write to the Free Software
18 Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307
31 #include <X11/SM/SMlib.h>
33 #include <pulse/xmalloc.h>
34 #include <pulse/util.h>
36 #include <pulsecore/modargs.h>
37 #include <pulsecore/log.h>
38 #include <pulsecore/x11wrap.h>
40 #include "module-x11-xsmp-symdef.h"
42 PA_MODULE_AUTHOR("Lennart Poettering");
43 PA_MODULE_DESCRIPTION("X11 session management");
44 PA_MODULE_VERSION(PACKAGE_VERSION
);
45 PA_MODULE_LOAD_ONCE(false);
46 PA_MODULE_USAGE("session_manager=<session manager string> display=<X11 display>");
48 static bool ice_in_use
= false;
50 static const char* const valid_modargs
[] = {
64 static void die_cb(SmcConn connection
, SmPointer client_data
) {
65 struct userdata
*u
= client_data
;
68 pa_log_debug("Got die message from XSMP.");
70 pa_x11_wrapper_kill(u
->x11
);
72 pa_x11_wrapper_unref(u
->x11
);
75 pa_module_unload_request(u
->module
, true);
78 static void save_complete_cb(SmcConn connection
, SmPointer client_data
) {
81 static void shutdown_cancelled_cb(SmcConn connection
, SmPointer client_data
) {
82 SmcSaveYourselfDone(connection
, True
);
85 static void save_yourself_cb(SmcConn connection
, SmPointer client_data
, int save_type
, Bool _shutdown
, int interact_style
, Bool fast
) {
86 SmcSaveYourselfDone(connection
, True
);
89 static void ice_io_cb(pa_mainloop_api
*a
, pa_io_event
*e
, int fd
, pa_io_event_flags_t flags
, void *userdata
) {
90 IceConn connection
= userdata
;
92 if (IceProcessMessages(connection
, NULL
, NULL
) == IceProcessMessagesIOError
) {
93 IceSetShutdownNegotiation(connection
, False
);
94 IceCloseConnection(connection
);
98 static void new_ice_connection(IceConn connection
, IcePointer client_data
, Bool opening
, IcePointer
*watch_data
) {
99 pa_core
*c
= client_data
;
102 *watch_data
= c
->mainloop
->io_new(
104 IceConnectionNumber(connection
),
109 c
->mainloop
->io_free(*watch_data
);
112 int pa__init(pa_module
*m
) {
114 pa_modargs
*ma
= NULL
;
115 char t
[256], *vendor
, *client_id
;
116 SmcCallbacks callbacks
;
117 SmProp prop_program
, prop_user
;
118 SmProp
*prop_list
[2];
119 SmPropValue val_program
, val_user
;
122 pa_client_new_data data
;
127 pa_log("module-x11-xsmp may not be loaded twice.");
131 IceAddConnectionWatch(new_ice_connection
, m
->core
);
134 m
->userdata
= u
= pa_xnew(struct userdata
, 1);
138 u
->connection
= NULL
;
141 if (!(ma
= pa_modargs_new(m
->argument
, valid_modargs
))) {
142 pa_log("Failed to parse module arguments");
146 if (!(u
->x11
= pa_x11_wrapper_get(m
->core
, pa_modargs_get_value(ma
, "display", NULL
))))
149 e
= pa_modargs_get_value(ma
, "session_manager", NULL
);
151 if (!e
&& !getenv("SESSION_MANAGER")) {
152 pa_log("X11 session manager not running.");
156 memset(&callbacks
, 0, sizeof(callbacks
));
157 callbacks
.die
.callback
= die_cb
;
158 callbacks
.die
.client_data
= u
;
159 callbacks
.save_yourself
.callback
= save_yourself_cb
;
160 callbacks
.save_yourself
.client_data
= m
->core
;
161 callbacks
.save_complete
.callback
= save_complete_cb
;
162 callbacks
.save_complete
.client_data
= m
->core
;
163 callbacks
.shutdown_cancelled
.callback
= shutdown_cancelled_cb
;
164 callbacks
.shutdown_cancelled
.client_data
= m
->core
;
166 if (!(u
->connection
= SmcOpenConnection(
168 SmProtoMajor
, SmProtoMinor
,
169 SmcSaveYourselfProcMask
| SmcDieProcMask
| SmcSaveCompleteProcMask
| SmcShutdownCancelledProcMask
,
170 &callbacks
, NULL
, &client_id
,
173 pa_log("Failed to open connection to session manager: %s", t
);
177 prop_program
.name
= (char*) SmProgram
;
178 prop_program
.type
= (char*) SmARRAY8
;
179 val_program
.value
= (char*) PACKAGE_NAME
;
180 val_program
.length
= (int) strlen(val_program
.value
);
181 prop_program
.num_vals
= 1;
182 prop_program
.vals
= &val_program
;
183 prop_list
[0] = &prop_program
;
185 prop_user
.name
= (char*) SmUserID
;
186 prop_user
.type
= (char*) SmARRAY8
;
187 pa_get_user_name(t
, sizeof(t
));
189 val_user
.length
= (int) strlen(val_user
.value
);
190 prop_user
.num_vals
= 1;
191 prop_user
.vals
= &val_user
;
192 prop_list
[1] = &prop_user
;
194 SmcSetProperties(u
->connection
, PA_ELEMENTSOF(prop_list
), prop_list
);
196 pa_log_info("Connected to session manager '%s' as '%s'.", vendor
= SmcVendor(u
->connection
), client_id
);
198 pa_client_new_data_init(&data
);
200 data
.driver
= __FILE__
;
201 pa_proplist_setf(data
.proplist
, PA_PROP_APPLICATION_NAME
, "XSMP Session on %s as %s", vendor
, client_id
);
202 pa_proplist_sets(data
.proplist
, "xsmp.vendor", vendor
);
203 pa_proplist_sets(data
.proplist
, "xsmp.client.id", client_id
);
204 u
->client
= pa_client_new(u
->core
, &data
);
205 pa_client_new_data_done(&data
);
226 void pa__done(pa_module
*m
) {
231 if ((u
= m
->userdata
)) {
234 SmcCloseConnection(u
->connection
, 0, NULL
);
237 pa_client_free(u
->client
);
240 pa_x11_wrapper_unref(u
->x11
);
246 IceRemoveConnectionWatch(new_ice_connection
, m
->core
);