]> git.ipfire.org Git - thirdparty/systemd.git/blob - src/login/logind-seat-dbus.c
sd-bus: drop bus parameter from message callback prototype
[thirdparty/systemd.git] / src / login / logind-seat-dbus.c
1 /*-*- Mode: C; c-basic-offset: 8; indent-tabs-mode: nil -*-*/
2
3 /***
4 This file is part of systemd.
5
6 Copyright 2011 Lennart Poettering
7
8 systemd is free software; you can redistribute it and/or modify it
9 under the terms of the GNU Lesser General Public License as published by
10 the Free Software Foundation; either version 2.1 of the License, or
11 (at your option) any later version.
12
13 systemd is distributed in the hope that it will be useful, but
14 WITHOUT ANY WARRANTY; without even the implied warranty of
15 MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
16 Lesser General Public License for more details.
17
18 You should have received a copy of the GNU Lesser General Public License
19 along with systemd; If not, see <http://www.gnu.org/licenses/>.
20 ***/
21
22 #include <errno.h>
23 #include <string.h>
24
25 #include "util.h"
26 #include "bus-util.h"
27 #include "strv.h"
28 #include "bus-common-errors.h"
29 #include "bus-label.h"
30 #include "logind.h"
31 #include "logind-seat.h"
32
33 static int property_get_active_session(
34 sd_bus *bus,
35 const char *path,
36 const char *interface,
37 const char *property,
38 sd_bus_message *reply,
39 void *userdata,
40 sd_bus_error *error) {
41
42 _cleanup_free_ char *p = NULL;
43 Seat *s = userdata;
44
45 assert(bus);
46 assert(reply);
47 assert(s);
48
49 p = s->active ? session_bus_path(s->active) : strdup("/");
50 if (!p)
51 return -ENOMEM;
52
53 return sd_bus_message_append(reply, "(so)", s->active ? s->active->id : "", p);
54 }
55
56 static int property_get_can_multi_session(
57 sd_bus *bus,
58 const char *path,
59 const char *interface,
60 const char *property,
61 sd_bus_message *reply,
62 void *userdata,
63 sd_bus_error *error) {
64
65 Seat *s = userdata;
66
67 assert(bus);
68 assert(reply);
69 assert(s);
70
71 return sd_bus_message_append(reply, "b", seat_can_multi_session(s));
72 }
73
74 static int property_get_can_tty(
75 sd_bus *bus,
76 const char *path,
77 const char *interface,
78 const char *property,
79 sd_bus_message *reply,
80 void *userdata,
81 sd_bus_error *error) {
82
83 Seat *s = userdata;
84
85 assert(bus);
86 assert(reply);
87 assert(s);
88
89 return sd_bus_message_append(reply, "b", seat_can_tty(s));
90 }
91
92 static int property_get_can_graphical(
93 sd_bus *bus,
94 const char *path,
95 const char *interface,
96 const char *property,
97 sd_bus_message *reply,
98 void *userdata,
99 sd_bus_error *error) {
100
101 Seat *s = userdata;
102
103 assert(bus);
104 assert(reply);
105 assert(s);
106
107 return sd_bus_message_append(reply, "b", seat_can_graphical(s));
108 }
109
110 static int property_get_sessions(
111 sd_bus *bus,
112 const char *path,
113 const char *interface,
114 const char *property,
115 sd_bus_message *reply,
116 void *userdata,
117 sd_bus_error *error) {
118
119 Seat *s = userdata;
120 Session *session;
121 int r;
122
123 assert(bus);
124 assert(reply);
125 assert(s);
126
127 r = sd_bus_message_open_container(reply, 'a', "(so)");
128 if (r < 0)
129 return r;
130
131 LIST_FOREACH(sessions_by_seat, session, s->sessions) {
132 _cleanup_free_ char *p = NULL;
133
134 p = session_bus_path(session);
135 if (!p)
136 return -ENOMEM;
137
138 r = sd_bus_message_append(reply, "(so)", session->id, p);
139 if (r < 0)
140 return r;
141
142 }
143
144 r = sd_bus_message_close_container(reply);
145 if (r < 0)
146 return r;
147
148 return 1;
149 }
150
151 static int property_get_idle_hint(
152 sd_bus *bus,
153 const char *path,
154 const char *interface,
155 const char *property,
156 sd_bus_message *reply,
157 void *userdata,
158 sd_bus_error *error) {
159
160 Seat *s = userdata;
161
162 assert(bus);
163 assert(reply);
164 assert(s);
165
166 return sd_bus_message_append(reply, "b", seat_get_idle_hint(s, NULL) > 0);
167 }
168
169 static int property_get_idle_since_hint(
170 sd_bus *bus,
171 const char *path,
172 const char *interface,
173 const char *property,
174 sd_bus_message *reply,
175 void *userdata,
176 sd_bus_error *error) {
177
178 Seat *s = userdata;
179 dual_timestamp t;
180 uint64_t u;
181 int r;
182
183 assert(bus);
184 assert(reply);
185 assert(s);
186
187 r = seat_get_idle_hint(s, &t);
188 if (r < 0)
189 return r;
190
191 u = streq(property, "IdleSinceHint") ? t.realtime : t.monotonic;
192
193 return sd_bus_message_append(reply, "t", u);
194 }
195
196 int bus_seat_method_terminate(sd_bus_message *message, void *userdata, sd_bus_error *error) {
197 Seat *s = userdata;
198 int r;
199
200 assert(message);
201 assert(s);
202
203 r = bus_verify_polkit_async(
204 message,
205 CAP_KILL,
206 "org.freedesktop.login1.manage",
207 false,
208 UID_INVALID,
209 &s->manager->polkit_registry,
210 error);
211 if (r < 0)
212 return r;
213 if (r == 0)
214 return 1; /* Will call us back */
215
216 r = seat_stop_sessions(s, true);
217 if (r < 0)
218 return r;
219
220 return sd_bus_reply_method_return(message, NULL);
221 }
222
223 static int method_activate_session(sd_bus_message *message, void *userdata, sd_bus_error *error) {
224 Seat *s = userdata;
225 const char *name;
226 Session *session;
227 int r;
228
229 assert(message);
230 assert(s);
231
232 r = sd_bus_message_read(message, "s", &name);
233 if (r < 0)
234 return r;
235
236 session = hashmap_get(s->manager->sessions, name);
237 if (!session)
238 return sd_bus_error_setf(error, BUS_ERROR_NO_SUCH_SESSION, "No session '%s' known", name);
239
240 if (session->seat != s)
241 return sd_bus_error_setf(error, BUS_ERROR_SESSION_NOT_ON_SEAT, "Session %s not on seat %s", name, s->id);
242
243 r = session_activate(session);
244 if (r < 0)
245 return r;
246
247 return sd_bus_reply_method_return(message, NULL);
248 }
249
250 static int method_switch_to(sd_bus_message *message, void *userdata, sd_bus_error *error) {
251 Seat *s = userdata;
252 unsigned int to;
253 int r;
254
255 assert(message);
256 assert(s);
257
258 r = sd_bus_message_read(message, "u", &to);
259 if (r < 0)
260 return r;
261
262 if (to <= 0)
263 return -EINVAL;
264
265 r = seat_switch_to(s, to);
266 if (r < 0)
267 return r;
268
269 return sd_bus_reply_method_return(message, NULL);
270 }
271
272 static int method_switch_to_next(sd_bus_message *message, void *userdata, sd_bus_error *error) {
273 Seat *s = userdata;
274 int r;
275
276 assert(message);
277 assert(s);
278
279 r = seat_switch_to_next(s);
280 if (r < 0)
281 return r;
282
283 return sd_bus_reply_method_return(message, NULL);
284 }
285
286 static int method_switch_to_previous(sd_bus_message *message, void *userdata, sd_bus_error *error) {
287 Seat *s = userdata;
288 int r;
289
290 assert(message);
291 assert(s);
292
293 r = seat_switch_to_previous(s);
294 if (r < 0)
295 return r;
296
297 return sd_bus_reply_method_return(message, NULL);
298 }
299
300 const sd_bus_vtable seat_vtable[] = {
301 SD_BUS_VTABLE_START(0),
302
303 SD_BUS_PROPERTY("Id", "s", NULL, offsetof(Seat, id), SD_BUS_VTABLE_PROPERTY_CONST),
304 SD_BUS_PROPERTY("ActiveSession", "(so)", property_get_active_session, 0, SD_BUS_VTABLE_PROPERTY_EMITS_CHANGE),
305 SD_BUS_PROPERTY("CanMultiSession", "b", property_get_can_multi_session, 0, SD_BUS_VTABLE_PROPERTY_CONST),
306 SD_BUS_PROPERTY("CanTTY", "b", property_get_can_tty, 0, SD_BUS_VTABLE_PROPERTY_CONST),
307 SD_BUS_PROPERTY("CanGraphical", "b", property_get_can_graphical, 0, SD_BUS_VTABLE_PROPERTY_EMITS_CHANGE),
308 SD_BUS_PROPERTY("Sessions", "a(so)", property_get_sessions, 0, SD_BUS_VTABLE_PROPERTY_EMITS_CHANGE),
309 SD_BUS_PROPERTY("IdleHint", "b", property_get_idle_hint, 0, SD_BUS_VTABLE_PROPERTY_EMITS_CHANGE),
310 SD_BUS_PROPERTY("IdleSinceHint", "t", property_get_idle_since_hint, 0, SD_BUS_VTABLE_PROPERTY_EMITS_CHANGE),
311 SD_BUS_PROPERTY("IdleSinceHintMonotonic", "t", property_get_idle_since_hint, 0, SD_BUS_VTABLE_PROPERTY_EMITS_CHANGE),
312
313 SD_BUS_METHOD("Terminate", NULL, NULL, bus_seat_method_terminate, SD_BUS_VTABLE_UNPRIVILEGED),
314 SD_BUS_METHOD("ActivateSession", "s", NULL, method_activate_session, SD_BUS_VTABLE_UNPRIVILEGED),
315 SD_BUS_METHOD("SwitchTo", "u", NULL, method_switch_to, SD_BUS_VTABLE_UNPRIVILEGED),
316 SD_BUS_METHOD("SwitchToNext", NULL, NULL, method_switch_to_next, SD_BUS_VTABLE_UNPRIVILEGED),
317 SD_BUS_METHOD("SwitchToPrevious", NULL, NULL, method_switch_to_previous, SD_BUS_VTABLE_UNPRIVILEGED),
318
319 SD_BUS_VTABLE_END
320 };
321
322 int seat_object_find(sd_bus *bus, const char *path, const char *interface, void *userdata, void **found, sd_bus_error *error) {
323 Manager *m = userdata;
324 Seat *seat;
325 int r;
326
327 assert(bus);
328 assert(path);
329 assert(interface);
330 assert(found);
331 assert(m);
332
333 if (streq(path, "/org/freedesktop/login1/seat/self")) {
334 _cleanup_bus_creds_unref_ sd_bus_creds *creds = NULL;
335 sd_bus_message *message;
336 Session *session;
337 const char *name;
338
339 message = sd_bus_get_current_message(bus);
340 if (!message)
341 return 0;
342
343 r = sd_bus_query_sender_creds(message, SD_BUS_CREDS_SESSION|SD_BUS_CREDS_AUGMENT, &creds);
344 if (r < 0)
345 return r;
346
347 r = sd_bus_creds_get_session(creds, &name);
348 if (r < 0)
349 return r;
350
351 session = hashmap_get(m->sessions, name);
352 if (!session)
353 return 0;
354
355 seat = session->seat;
356 } else {
357 _cleanup_free_ char *e = NULL;
358 const char *p;
359
360 p = startswith(path, "/org/freedesktop/login1/seat/");
361 if (!p)
362 return 0;
363
364 e = bus_label_unescape(p);
365 if (!e)
366 return -ENOMEM;
367
368 seat = hashmap_get(m->seats, e);
369 }
370
371 if (!seat)
372 return 0;
373
374 *found = seat;
375 return 1;
376 }
377
378 char *seat_bus_path(Seat *s) {
379 _cleanup_free_ char *t = NULL;
380
381 assert(s);
382
383 t = bus_label_escape(s->id);
384 if (!t)
385 return NULL;
386
387 return strappend("/org/freedesktop/login1/seat/", t);
388 }
389
390 int seat_node_enumerator(sd_bus *bus, const char *path, void *userdata, char ***nodes, sd_bus_error *error) {
391 _cleanup_strv_free_ char **l = NULL;
392 sd_bus_message *message;
393 Manager *m = userdata;
394 Seat *seat;
395 Iterator i;
396 int r;
397
398 assert(bus);
399 assert(path);
400 assert(nodes);
401
402 HASHMAP_FOREACH(seat, m->seats, i) {
403 char *p;
404
405 p = seat_bus_path(seat);
406 if (!p)
407 return -ENOMEM;
408
409 r = strv_consume(&l, p);
410 if (r < 0)
411 return r;
412 }
413
414 message = sd_bus_get_current_message(bus);
415 if (message) {
416 _cleanup_bus_creds_unref_ sd_bus_creds *creds = NULL;
417 const char *name;
418 Session *session;
419
420 r = sd_bus_query_sender_creds(message, SD_BUS_CREDS_SESSION|SD_BUS_CREDS_AUGMENT, &creds);
421 if (r >= 0) {
422 r = sd_bus_creds_get_session(creds, &name);
423 if (r >= 0) {
424 session = hashmap_get(m->sessions, name);
425 if (session && session->seat) {
426 r = strv_extend(&l, "/org/freedesktop/login1/seat/self");
427 if (r < 0)
428 return r;
429 }
430 }
431 }
432 }
433
434 *nodes = l;
435 l = NULL;
436
437 return 1;
438 }
439
440 int seat_send_signal(Seat *s, bool new_seat) {
441 _cleanup_free_ char *p = NULL;
442
443 assert(s);
444
445 p = seat_bus_path(s);
446 if (!p)
447 return -ENOMEM;
448
449 return sd_bus_emit_signal(
450 s->manager->bus,
451 "/org/freedesktop/login1",
452 "org.freedesktop.login1.Manager",
453 new_seat ? "SeatNew" : "SeatRemoved",
454 "so", s->id, p);
455 }
456
457 int seat_send_changed(Seat *s, const char *properties, ...) {
458 _cleanup_free_ char *p = NULL;
459 char **l;
460
461 assert(s);
462
463 if (!s->started)
464 return 0;
465
466 p = seat_bus_path(s);
467 if (!p)
468 return -ENOMEM;
469
470 l = strv_from_stdarg_alloca(properties);
471
472 return sd_bus_emit_properties_changed_strv(s->manager->bus, p, "org.freedesktop.login1.Seat", l);
473 }