1 /*-*- Mode: C; c-basic-offset: 8; indent-tabs-mode: nil -*-*/
4 This file is part of systemd.
6 Copyright 2011 Lennart Poettering
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.
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.
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/>.
22 #include <sys/types.h>
23 #include <sys/ioctl.h>
28 #include "bus-error.h"
30 #include "cgroup-util.h"
34 #include "terminal-util.h"
35 #include "udev-util.h"
37 int manager_add_device(Manager
*m
, const char *sysfs
, bool master
, Device
**_device
) {
43 d
= hashmap_get(m
->devices
, sysfs
);
45 /* we support adding master-flags, but not removing them */
46 d
->master
= d
->master
|| master
;
48 d
= device_new(m
, sysfs
, master
);
59 int manager_add_seat(Manager
*m
, const char *id
, Seat
**_seat
) {
65 s
= hashmap_get(m
->seats
, id
);
78 int manager_add_session(Manager
*m
, const char *id
, Session
**_session
) {
84 s
= hashmap_get(m
->sessions
, id
);
86 s
= session_new(m
, id
);
97 int manager_add_user(Manager
*m
, uid_t uid
, gid_t gid
, const char *name
, User
**_user
) {
103 u
= hashmap_get(m
->users
, UID_TO_PTR(uid
));
105 u
= user_new(m
, uid
, gid
, name
);
116 int manager_add_user_by_name(Manager
*m
, const char *name
, User
**_user
) {
124 r
= get_user_creds(&name
, &uid
, &gid
, NULL
, NULL
);
128 return manager_add_user(m
, uid
, gid
, name
, _user
);
131 int manager_add_user_by_uid(Manager
*m
, uid_t uid
, User
**_user
) {
139 return errno
? -errno
: -ENOENT
;
141 return manager_add_user(m
, uid
, p
->pw_gid
, p
->pw_name
, _user
);
144 int manager_add_inhibitor(Manager
*m
, const char* id
, Inhibitor
**_inhibitor
) {
150 i
= hashmap_get(m
->inhibitors
, id
);
158 i
= inhibitor_new(m
, id
);
168 int manager_add_button(Manager
*m
, const char *name
, Button
**_button
) {
174 b
= hashmap_get(m
->buttons
, name
);
176 b
= button_new(m
, name
);
187 int manager_process_seat_device(Manager
*m
, struct udev_device
*d
) {
193 if (streq_ptr(udev_device_get_action(d
), "remove")) {
195 device
= hashmap_get(m
->devices
, udev_device_get_syspath(d
));
199 seat_add_to_gc_queue(device
->seat
);
207 sn
= udev_device_get_property_value(d
, "ID_SEAT");
211 if (!seat_name_is_valid(sn
)) {
212 log_warning("Device with invalid seat name %s found, ignoring.", sn
);
216 seat
= hashmap_get(m
->seats
, sn
);
217 master
= udev_device_has_tag(d
, "master-of-seat");
219 /* Ignore non-master devices for unknown seats */
220 if (!master
&& !seat
)
223 r
= manager_add_device(m
, udev_device_get_syspath(d
), master
, &device
);
228 r
= manager_add_seat(m
, sn
, &seat
);
237 device_attach(device
, seat
);
244 int manager_process_button_device(Manager
*m
, struct udev_device
*d
) {
251 if (streq_ptr(udev_device_get_action(d
), "remove")) {
253 b
= hashmap_get(m
->buttons
, udev_device_get_sysname(d
));
262 r
= manager_add_button(m
, udev_device_get_sysname(d
), &b
);
266 sn
= udev_device_get_property_value(d
, "ID_SEAT");
270 button_set_seat(b
, sn
);
277 int manager_get_session_by_pid(Manager
*m
, pid_t pid
, Session
**session
) {
278 _cleanup_free_
char *unit
= NULL
;
287 r
= cg_pid_get_unit(pid
, &unit
);
291 s
= hashmap_get(m
->session_units
, unit
);
300 int manager_get_user_by_pid(Manager
*m
, pid_t pid
, User
**user
) {
301 _cleanup_free_
char *unit
= NULL
;
311 r
= cg_pid_get_slice(pid
, &unit
);
315 u
= hashmap_get(m
->user_units
, unit
);
323 int manager_get_idle_hint(Manager
*m
, dual_timestamp
*t
) {
326 dual_timestamp ts
= DUAL_TIMESTAMP_NULL
;
331 idle_hint
= !manager_is_inhibited(m
, INHIBIT_IDLE
, INHIBIT_BLOCK
, t
, false, false, 0, NULL
);
333 HASHMAP_FOREACH(s
, m
->sessions
, i
) {
337 ih
= session_get_idle_hint(s
, &k
);
343 if (k
.monotonic
< ts
.monotonic
)
349 } else if (idle_hint
) {
351 if (k
.monotonic
> ts
.monotonic
)
362 bool manager_shall_kill(Manager
*m
, const char *user
) {
366 if (!m
->kill_user_processes
)
369 if (strv_contains(m
->kill_exclude_users
, user
))
372 if (strv_isempty(m
->kill_only_users
))
375 return strv_contains(m
->kill_only_users
, user
);
378 static int vt_is_busy(unsigned int vtnr
) {
379 struct vt_stat vt_stat
;
381 _cleanup_close_
int fd
;
385 /* We explicitly open /dev/tty1 here instead of /dev/tty0. If
386 * we'd open the latter we'd open the foreground tty which
387 * hence would be unconditionally busy. By opening /dev/tty1
388 * we avoid this. Since tty1 is special and needs to be an
389 * explicitly loaded getty or DM this is safe. */
391 fd
= open_terminal("/dev/tty1", O_RDWR
|O_NOCTTY
|O_CLOEXEC
);
395 if (ioctl(fd
, VT_GETSTATE
, &vt_stat
) < 0)
398 r
= !!(vt_stat
.v_state
& (1 << vtnr
));
403 int manager_spawn_autovt(Manager
*m
, unsigned int vtnr
) {
404 _cleanup_bus_error_free_ sd_bus_error error
= SD_BUS_ERROR_NULL
;
405 char name
[sizeof("autovt@tty.service") + DECIMAL_STR_MAX(unsigned int)];
411 if (vtnr
> m
->n_autovts
&&
412 vtnr
!= m
->reserve_vt
)
415 if (vtnr
!= m
->reserve_vt
) {
416 /* If this is the reserved TTY, we'll start the getty
417 * on it in any case, but otherwise only if it is not
420 r
= vt_is_busy(vtnr
);
427 snprintf(name
, sizeof(name
), "autovt@tty%u.service", vtnr
);
428 r
= sd_bus_call_method(
430 "org.freedesktop.systemd1",
431 "/org/freedesktop/systemd1",
432 "org.freedesktop.systemd1.Manager",
438 log_error("Failed to start %s: %s", name
, bus_error_message(&error
, r
));
443 static bool manager_is_docked(Manager
*m
) {
447 HASHMAP_FOREACH(b
, m
->buttons
, i
)
454 static int manager_count_external_displays(Manager
*m
) {
455 _cleanup_udev_enumerate_unref_
struct udev_enumerate
*e
= NULL
;
456 struct udev_list_entry
*item
= NULL
, *first
= NULL
;
460 e
= udev_enumerate_new(m
->udev
);
464 r
= udev_enumerate_add_match_subsystem(e
, "drm");
468 r
= udev_enumerate_scan_devices(e
);
472 first
= udev_enumerate_get_list_entry(e
);
473 udev_list_entry_foreach(item
, first
) {
474 _cleanup_udev_device_unref_
struct udev_device
*d
= NULL
;
475 struct udev_device
*p
;
476 const char *status
, *enabled
, *dash
, *nn
, *i
;
477 bool external
= false;
479 d
= udev_device_new_from_syspath(m
->udev
, udev_list_entry_get_name(item
));
483 p
= udev_device_get_parent(d
);
487 /* If the parent shares the same subsystem as the
488 * device we are looking at then it is a connector,
489 * which is what we are interested in. */
490 if (!streq_ptr(udev_device_get_subsystem(p
), "drm"))
493 nn
= udev_device_get_sysname(d
);
497 /* Ignore internal displays: the type is encoded in
498 * the sysfs name, as the second dash seperated item
499 * (the first is the card name, the last the connector
500 * number). We implement a whitelist of external
501 * displays here, rather than a whitelist, to ensure
502 * we don't block suspends too eagerly. */
503 dash
= strchr(nn
, '-');
508 FOREACH_STRING(i
, "VGA-", "DVI-I-", "DVI-D-", "DVI-A-"
509 "Composite-", "SVIDEO-", "Component-",
510 "DIN-", "DP-", "HDMI-A-", "HDMI-B-", "TV-") {
512 if (startswith(dash
, i
)) {
520 /* Ignore ports that are not enabled */
521 enabled
= udev_device_get_sysattr_value(d
, "enabled");
524 if (!streq_ptr(enabled
, "enabled"))
527 /* We count any connector which is not explicitly
528 * "disconnected" as connected. */
529 status
= udev_device_get_sysattr_value(d
, "status");
530 if (!streq_ptr(status
, "disconnected"))
537 bool manager_is_docked_or_external_displays(Manager
*m
) {
540 /* If we are docked don't react to lid closing */
541 if (manager_is_docked(m
)) {
542 log_debug("System is docked.");
546 /* If we have more than one display connected,
547 * assume that we are docked. */
548 n
= manager_count_external_displays(m
);
550 log_warning_errno(n
, "Display counting failed: %m");
552 log_debug("External (%i) displays connected.", n
);