2 This file is part of systemd.
4 Copyright 2013 Lennart Poettering
6 systemd is free software; you can redistribute it and/or modify it
7 under the terms of the GNU Lesser General Public License as published by
8 the Free Software Foundation; either version 2.1 of the License, or
9 (at your option) any later version.
11 systemd 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 Lesser General Public License for more details.
16 You should have received a copy of the GNU Lesser General Public License
17 along with systemd; If not, see <http://www.gnu.org/licenses/>.
20 #include "alloc-util.h"
21 #include "bus-common-errors.h"
22 #include "bus-internal.h"
24 #include "dbus-cgroup.h"
25 #include "dbus-kill.h"
26 #include "dbus-scope.h"
27 #include "dbus-unit.h"
30 #include "selinux-access.h"
33 static int bus_scope_abandon(sd_bus_message
*message
, void *userdata
, sd_bus_error
*error
) {
40 r
= mac_selinux_unit_access_check(UNIT(s
), message
, "stop", error
);
44 r
= bus_verify_manage_units_async(UNIT(s
)->manager
, message
, error
);
48 return 1; /* No authorization for now, but the async polkit stuff will call us again when it has it */
52 return sd_bus_error_setf(error
, BUS_ERROR_SCOPE_NOT_RUNNING
, "Scope %s is not running, cannot abandon.", UNIT(s
)->id
);
56 return sd_bus_reply_method_return(message
, NULL
);
59 static BUS_DEFINE_PROPERTY_GET_ENUM(property_get_result
, scope_result
, ScopeResult
);
61 const sd_bus_vtable bus_scope_vtable
[] = {
62 SD_BUS_VTABLE_START(0),
63 SD_BUS_PROPERTY("Controller", "s", NULL
, offsetof(Scope
, controller
), SD_BUS_VTABLE_PROPERTY_CONST
),
64 SD_BUS_PROPERTY("TimeoutStopUSec", "t", bus_property_get_usec
, offsetof(Scope
, timeout_stop_usec
), SD_BUS_VTABLE_PROPERTY_CONST
),
65 SD_BUS_PROPERTY("Result", "s", property_get_result
, offsetof(Scope
, result
), SD_BUS_VTABLE_PROPERTY_EMITS_CHANGE
),
66 SD_BUS_SIGNAL("RequestStop", NULL
, 0),
67 SD_BUS_METHOD("Abandon", NULL
, NULL
, bus_scope_abandon
, SD_BUS_VTABLE_UNPRIVILEGED
),
71 static int bus_scope_set_transient_property(
74 sd_bus_message
*message
,
75 UnitSetPropertiesMode mode
,
76 sd_bus_error
*error
) {
84 if (streq(name
, "PIDs")) {
88 r
= sd_bus_message_enter_container(message
, 'a', "u");
92 while ((r
= sd_bus_message_read(message
, "u", &pid
)) > 0) {
97 if (mode
!= UNIT_CHECK
) {
98 r
= unit_watch_pid(UNIT(s
), pid
);
99 if (r
< 0 && r
!= -EEXIST
)
108 r
= sd_bus_message_exit_container(message
);
117 } else if (streq(name
, "Controller")) {
118 const char *controller
;
121 r
= sd_bus_message_read(message
, "s", &controller
);
125 if (!isempty(controller
) && !service_name_is_valid(controller
))
126 return sd_bus_error_setf(error
, SD_BUS_ERROR_INVALID_ARGS
, "Controller '%s' is not a valid bus name.", controller
);
128 if (mode
!= UNIT_CHECK
) {
129 if (isempty(controller
))
132 c
= strdup(controller
);
143 } else if (streq(name
, "TimeoutStopUSec")) {
145 if (mode
!= UNIT_CHECK
) {
146 r
= sd_bus_message_read(message
, "t", &s
->timeout_stop_usec
);
150 unit_write_drop_in_private_format(UNIT(s
), mode
, name
, "TimeoutStopSec="USEC_FMT
"us", s
->timeout_stop_usec
);
152 r
= sd_bus_message_skip(message
, "t");
163 int bus_scope_set_property(
166 sd_bus_message
*message
,
167 UnitSetPropertiesMode mode
,
168 sd_bus_error
*error
) {
177 r
= bus_cgroup_set_property(u
, &s
->cgroup_context
, name
, message
, mode
, error
);
181 if (u
->load_state
== UNIT_STUB
) {
182 /* While we are created we still accept PIDs */
184 r
= bus_scope_set_transient_property(s
, name
, message
, mode
, error
);
188 r
= bus_kill_context_set_transient_property(u
, &s
->kill_context
, name
, message
, mode
, error
);
196 int bus_scope_commit_properties(Unit
*u
) {
199 unit_update_cgroup_members_masks(u
);
200 unit_realize_cgroup(u
);
205 int bus_scope_send_request_stop(Scope
*s
) {
206 _cleanup_(sd_bus_message_unrefp
) sd_bus_message
*m
= NULL
;
207 _cleanup_free_
char *p
= NULL
;
215 p
= unit_dbus_path(UNIT(s
));
219 r
= sd_bus_message_new_signal(
220 UNIT(s
)->manager
->api_bus
,
223 "org.freedesktop.systemd1.Scope",
228 return sd_bus_send_to(UNIT(s
)->manager
->api_bus
, m
, /* s->controller */ NULL
, NULL
);