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 "alloc-util.h"
29 #include "bus-error.h"
31 #include "cgroup-util.h"
35 #include "terminal-util.h"
36 #include "udev-util.h"
37 #include "user-util.h"
39 int manager_add_device(Manager
*m
, const char *sysfs
, bool master
, Device
**_device
) {
45 d
= hashmap_get(m
->devices
, sysfs
);
47 /* we support adding master-flags, but not removing them */
48 d
->master
= d
->master
|| master
;
50 d
= device_new(m
, sysfs
, master
);
61 int manager_add_seat(Manager
*m
, const char *id
, Seat
**_seat
) {
67 s
= hashmap_get(m
->seats
, id
);
80 int manager_add_session(Manager
*m
, const char *id
, Session
**_session
) {
86 s
= hashmap_get(m
->sessions
, id
);
88 s
= session_new(m
, id
);
99 int manager_add_user(Manager
*m
, uid_t uid
, gid_t gid
, const char *name
, User
**_user
) {
105 u
= hashmap_get(m
->users
, UID_TO_PTR(uid
));
107 u
= user_new(m
, uid
, gid
, name
);
118 int manager_add_user_by_name(Manager
*m
, const char *name
, User
**_user
) {
126 r
= get_user_creds(&name
, &uid
, &gid
, NULL
, NULL
);
130 return manager_add_user(m
, uid
, gid
, name
, _user
);
133 int manager_add_user_by_uid(Manager
*m
, uid_t uid
, User
**_user
) {
141 return errno
? -errno
: -ENOENT
;
143 return manager_add_user(m
, uid
, p
->pw_gid
, p
->pw_name
, _user
);
146 int manager_add_inhibitor(Manager
*m
, const char* id
, Inhibitor
**_inhibitor
) {
152 i
= hashmap_get(m
->inhibitors
, id
);
160 i
= inhibitor_new(m
, id
);
170 int manager_add_button(Manager
*m
, const char *name
, Button
**_button
) {
176 b
= hashmap_get(m
->buttons
, name
);
178 b
= button_new(m
, name
);
189 int manager_process_seat_device(Manager
*m
, struct udev_device
*d
) {
195 if (streq_ptr(udev_device_get_action(d
), "remove")) {
197 device
= hashmap_get(m
->devices
, udev_device_get_syspath(d
));
201 seat_add_to_gc_queue(device
->seat
);
209 sn
= udev_device_get_property_value(d
, "ID_SEAT");
213 if (!seat_name_is_valid(sn
)) {
214 log_warning("Device with invalid seat name %s found, ignoring.", sn
);
218 seat
= hashmap_get(m
->seats
, sn
);
219 master
= udev_device_has_tag(d
, "master-of-seat");
221 /* Ignore non-master devices for unknown seats */
222 if (!master
&& !seat
)
225 r
= manager_add_device(m
, udev_device_get_syspath(d
), master
, &device
);
230 r
= manager_add_seat(m
, sn
, &seat
);
239 device_attach(device
, seat
);
246 int manager_process_button_device(Manager
*m
, struct udev_device
*d
) {
253 if (streq_ptr(udev_device_get_action(d
), "remove")) {
255 b
= hashmap_get(m
->buttons
, udev_device_get_sysname(d
));
264 r
= manager_add_button(m
, udev_device_get_sysname(d
), &b
);
268 sn
= udev_device_get_property_value(d
, "ID_SEAT");
272 button_set_seat(b
, sn
);
279 int manager_get_session_by_pid(Manager
*m
, pid_t pid
, Session
**session
) {
280 _cleanup_free_
char *unit
= NULL
;
289 r
= cg_pid_get_unit(pid
, &unit
);
293 s
= hashmap_get(m
->session_units
, unit
);
302 int manager_get_user_by_pid(Manager
*m
, pid_t pid
, User
**user
) {
303 _cleanup_free_
char *unit
= NULL
;
313 r
= cg_pid_get_slice(pid
, &unit
);
317 u
= hashmap_get(m
->user_units
, unit
);
325 int manager_get_idle_hint(Manager
*m
, dual_timestamp
*t
) {
328 dual_timestamp ts
= DUAL_TIMESTAMP_NULL
;
333 idle_hint
= !manager_is_inhibited(m
, INHIBIT_IDLE
, INHIBIT_BLOCK
, t
, false, false, 0, NULL
);
335 HASHMAP_FOREACH(s
, m
->sessions
, i
) {
339 ih
= session_get_idle_hint(s
, &k
);
345 if (k
.monotonic
< ts
.monotonic
)
351 } else if (idle_hint
) {
353 if (k
.monotonic
> ts
.monotonic
)
364 bool manager_shall_kill(Manager
*m
, const char *user
) {
368 if (!m
->kill_user_processes
)
371 if (strv_contains(m
->kill_exclude_users
, user
))
374 if (strv_isempty(m
->kill_only_users
))
377 return strv_contains(m
->kill_only_users
, user
);
380 static int vt_is_busy(unsigned int vtnr
) {
381 struct vt_stat vt_stat
;
383 _cleanup_close_
int fd
;
387 /* We explicitly open /dev/tty1 here instead of /dev/tty0. If
388 * we'd open the latter we'd open the foreground tty which
389 * hence would be unconditionally busy. By opening /dev/tty1
390 * we avoid this. Since tty1 is special and needs to be an
391 * explicitly loaded getty or DM this is safe. */
393 fd
= open_terminal("/dev/tty1", O_RDWR
|O_NOCTTY
|O_CLOEXEC
);
397 if (ioctl(fd
, VT_GETSTATE
, &vt_stat
) < 0)
400 r
= !!(vt_stat
.v_state
& (1 << vtnr
));
405 int manager_spawn_autovt(Manager
*m
, unsigned int vtnr
) {
406 _cleanup_bus_error_free_ sd_bus_error error
= SD_BUS_ERROR_NULL
;
407 char name
[sizeof("autovt@tty.service") + DECIMAL_STR_MAX(unsigned int)];
413 if (vtnr
> m
->n_autovts
&&
414 vtnr
!= m
->reserve_vt
)
417 if (vtnr
!= m
->reserve_vt
) {
418 /* If this is the reserved TTY, we'll start the getty
419 * on it in any case, but otherwise only if it is not
422 r
= vt_is_busy(vtnr
);
429 snprintf(name
, sizeof(name
), "autovt@tty%u.service", vtnr
);
430 r
= sd_bus_call_method(
432 "org.freedesktop.systemd1",
433 "/org/freedesktop/systemd1",
434 "org.freedesktop.systemd1.Manager",
440 log_error("Failed to start %s: %s", name
, bus_error_message(&error
, r
));
445 static bool manager_is_docked(Manager
*m
) {
449 HASHMAP_FOREACH(b
, m
->buttons
, i
)
456 static int manager_count_external_displays(Manager
*m
) {
457 _cleanup_udev_enumerate_unref_
struct udev_enumerate
*e
= NULL
;
458 struct udev_list_entry
*item
= NULL
, *first
= NULL
;
462 e
= udev_enumerate_new(m
->udev
);
466 r
= udev_enumerate_add_match_subsystem(e
, "drm");
470 r
= udev_enumerate_scan_devices(e
);
474 first
= udev_enumerate_get_list_entry(e
);
475 udev_list_entry_foreach(item
, first
) {
476 _cleanup_udev_device_unref_
struct udev_device
*d
= NULL
;
477 struct udev_device
*p
;
478 const char *status
, *enabled
, *dash
, *nn
, *i
;
479 bool external
= false;
481 d
= udev_device_new_from_syspath(m
->udev
, udev_list_entry_get_name(item
));
485 p
= udev_device_get_parent(d
);
489 /* If the parent shares the same subsystem as the
490 * device we are looking at then it is a connector,
491 * which is what we are interested in. */
492 if (!streq_ptr(udev_device_get_subsystem(p
), "drm"))
495 nn
= udev_device_get_sysname(d
);
499 /* Ignore internal displays: the type is encoded in
500 * the sysfs name, as the second dash seperated item
501 * (the first is the card name, the last the connector
502 * number). We implement a whitelist of external
503 * displays here, rather than a whitelist, to ensure
504 * we don't block suspends too eagerly. */
505 dash
= strchr(nn
, '-');
510 FOREACH_STRING(i
, "VGA-", "DVI-I-", "DVI-D-", "DVI-A-"
511 "Composite-", "SVIDEO-", "Component-",
512 "DIN-", "DP-", "HDMI-A-", "HDMI-B-", "TV-") {
514 if (startswith(dash
, i
)) {
522 /* Ignore ports that are not enabled */
523 enabled
= udev_device_get_sysattr_value(d
, "enabled");
526 if (!streq_ptr(enabled
, "enabled"))
529 /* We count any connector which is not explicitly
530 * "disconnected" as connected. */
531 status
= udev_device_get_sysattr_value(d
, "status");
532 if (!streq_ptr(status
, "disconnected"))
539 bool manager_is_docked_or_external_displays(Manager
*m
) {
542 /* If we are docked don't react to lid closing */
543 if (manager_is_docked(m
)) {
544 log_debug("System is docked.");
548 /* If we have more than one display connected,
549 * assume that we are docked. */
550 n
= manager_count_external_displays(m
);
552 log_warning_errno(n
, "Display counting failed: %m");
554 log_debug("External (%i) displays connected.", n
);