1 /*-*- Mode: C; c-basic-offset: 8; indent-tabs-mode: nil -*-*/
4 This file is part of systemd.
6 Copyright 2010 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/>.
23 #include <dbus/dbus.h>
24 #include <sys/prctl.h>
26 #include "dbus-execute.h"
30 #include "dbus-common.h"
31 #include "syscall-list.h"
34 DEFINE_BUS_PROPERTY_APPEND_ENUM(bus_execute_append_input, exec_input, ExecInput);
35 DEFINE_BUS_PROPERTY_APPEND_ENUM(bus_execute_append_output, exec_output, ExecOutput);
37 int bus_execute_append_env_files(DBusMessageIter *i, const char *property, void *data) {
38 char **env_files = data, **j;
39 DBusMessageIter sub, sub2;
44 if (!dbus_message_iter_open_container(i, DBUS_TYPE_ARRAY, "(sb)", &sub))
47 STRV_FOREACH(j, env_files) {
48 dbus_bool_t b = false;
56 if (!dbus_message_iter_open_container(&sub, DBUS_TYPE_STRUCT, NULL, &sub2) ||
57 !dbus_message_iter_append_basic(&sub2, DBUS_TYPE_STRING, &fn) ||
58 !dbus_message_iter_append_basic(&sub2, DBUS_TYPE_BOOLEAN, &b) ||
59 !dbus_message_iter_close_container(&sub, &sub2))
63 if (!dbus_message_iter_close_container(i, &sub))
69 int bus_execute_append_oom_score_adjust(DBusMessageIter *i, const char *property, void *data) {
70 ExecContext *c = data;
77 if (c->oom_score_adjust_set)
78 n = c->oom_score_adjust;
83 if (read_one_line_file("/proc/self/oom_score_adj", &t) >= 0) {
89 if (!dbus_message_iter_append_basic(i, DBUS_TYPE_INT32, &n))
95 int bus_execute_append_nice(DBusMessageIter *i, const char *property, void *data) {
96 ExecContext *c = data;
106 n = getpriority(PRIO_PROCESS, 0);
108 if (!dbus_message_iter_append_basic(i, DBUS_TYPE_INT32, &n))
114 int bus_execute_append_ioprio(DBusMessageIter *i, const char *property, void *data) {
115 ExecContext *c = data;
125 n = ioprio_get(IOPRIO_WHO_PROCESS, 0);
127 if (!dbus_message_iter_append_basic(i, DBUS_TYPE_INT32, &n))
133 int bus_execute_append_cpu_sched_policy(DBusMessageIter *i, const char *property, void *data) {
134 ExecContext *c = data;
141 if (c->cpu_sched_set)
142 n = c->cpu_sched_policy;
144 n = sched_getscheduler(0);
146 if (!dbus_message_iter_append_basic(i, DBUS_TYPE_INT32, &n))
152 int bus_execute_append_cpu_sched_priority(DBusMessageIter *i, const char *property, void *data) {
153 ExecContext *c = data;
160 if (c->cpu_sched_set)
161 n = c->cpu_sched_priority;
163 struct sched_param p;
166 if (sched_getparam(0, &p) >= 0)
167 n = p.sched_priority;
172 if (!dbus_message_iter_append_basic(i, DBUS_TYPE_INT32, &n))
178 int bus_execute_append_affinity(DBusMessageIter *i, const char *property, void *data) {
179 ExecContext *c = data;
187 if (!dbus_message_iter_open_container(i, DBUS_TYPE_ARRAY, "y", &sub))
191 b = dbus_message_iter_append_fixed_array(&sub, DBUS_TYPE_BYTE, &c->cpuset, CPU_ALLOC_SIZE(c->cpuset_ncpus));
193 b = dbus_message_iter_append_fixed_array(&sub, DBUS_TYPE_BYTE, &c->cpuset, 0);
198 if (!dbus_message_iter_close_container(i, &sub))
204 int bus_execute_append_timer_slack_nsec(DBusMessageIter *i, const char *property, void *data) {
205 ExecContext *c = data;
212 if (c->timer_slack_nsec != (nsec_t) -1)
213 u = (uint64_t) c->timer_slack_nsec;
215 u = (uint64_t) prctl(PR_GET_TIMERSLACK);
217 if (!dbus_message_iter_append_basic(i, DBUS_TYPE_UINT64, &u))
223 int bus_execute_append_capability_bs(DBusMessageIter *i, const char *property, void *data) {
224 ExecContext *c = data;
225 uint64_t normal, inverted;
231 /* We store this negated internally, to match the kernel, but
232 * we expose it normalized. */
234 normal = *(uint64_t*) data;
237 return bus_property_append_uint64(i, property, &inverted);
240 int bus_execute_append_capabilities(DBusMessageIter *i, const char *property, void *data) {
241 ExecContext *c = data;
251 s = t = cap_to_text(c->capabilities, NULL);
258 b = dbus_message_iter_append_basic(i, DBUS_TYPE_STRING, &s);
269 int bus_execute_append_rlimits(DBusMessageIter *i, const char *property, void *data) {
270 ExecContext *c = data;
278 assert_se((r = rlimit_from_string(property)) >= 0);
281 u = (uint64_t) c->rlimit[r]->rlim_max;
288 u = (uint64_t) rl.rlim_max;
291 if (!dbus_message_iter_append_basic(i, DBUS_TYPE_UINT64, &u))
297 int bus_execute_append_command(DBusMessageIter *i, const char *property, void *data) {
298 ExecCommand *c = data;
299 DBusMessageIter sub, sub2, sub3;
304 if (!dbus_message_iter_open_container(i, DBUS_TYPE_ARRAY, "(sasbttttuii)", &sub))
307 LIST_FOREACH(command, c, c) {
310 int32_t code, status;
316 if (!dbus_message_iter_open_container(&sub, DBUS_TYPE_STRUCT, NULL, &sub2) ||
317 !dbus_message_iter_append_basic(&sub2, DBUS_TYPE_STRING, &c->path) ||
318 !dbus_message_iter_open_container(&sub2, DBUS_TYPE_ARRAY, "s", &sub3))
321 STRV_FOREACH(l, c->argv)
322 if (!dbus_message_iter_append_basic(&sub3, DBUS_TYPE_STRING, l))
325 pid = (uint32_t) c->exec_status.pid;
326 code = (int32_t) c->exec_status.code;
327 status = (int32_t) c->exec_status.status;
331 if (!dbus_message_iter_close_container(&sub2, &sub3) ||
332 !dbus_message_iter_append_basic(&sub2, DBUS_TYPE_BOOLEAN, &b) ||
333 !dbus_message_iter_append_basic(&sub2, DBUS_TYPE_UINT64, &c->exec_status.start_timestamp.realtime) ||
334 !dbus_message_iter_append_basic(&sub2, DBUS_TYPE_UINT64, &c->exec_status.start_timestamp.monotonic) ||
335 !dbus_message_iter_append_basic(&sub2, DBUS_TYPE_UINT64, &c->exec_status.exit_timestamp.realtime) ||
336 !dbus_message_iter_append_basic(&sub2, DBUS_TYPE_UINT64, &c->exec_status.exit_timestamp.monotonic) ||
337 !dbus_message_iter_append_basic(&sub2, DBUS_TYPE_UINT32, &pid) ||
338 !dbus_message_iter_append_basic(&sub2, DBUS_TYPE_INT32, &code) ||
339 !dbus_message_iter_append_basic(&sub2, DBUS_TYPE_INT32, &status))
342 if (!dbus_message_iter_close_container(&sub, &sub2))
346 if (!dbus_message_iter_close_container(i, &sub))
352 int bus_execute_append_syscall_filter(DBusMessageIter *i, const char *property, void *data) {
353 ExecContext *c = data;
361 if (!dbus_message_iter_open_container(i, DBUS_TYPE_ARRAY, "u", &sub))
364 if (c->syscall_filter)
365 b = dbus_message_iter_append_fixed_array(&sub, DBUS_TYPE_UINT32, &c->syscall_filter, (syscall_max() + 31) >> 4);
367 b = dbus_message_iter_append_fixed_array(&sub, DBUS_TYPE_UINT32, &c->syscall_filter, 0);
372 if (!dbus_message_iter_close_container(i, &sub))
378 const BusProperty bus_exec_context_properties[] = {
379 { "Environment", bus_property_append_strv, "as", offsetof(ExecContext, environment), true },
380 { "EnvironmentFiles", bus_execute_append_env_files, "a(sb)", offsetof(ExecContext, environment_files), true },
381 { "UMask", bus_property_append_mode, "u", offsetof(ExecContext, umask) },
382 { "LimitCPU", bus_execute_append_rlimits, "t", 0 },
383 { "LimitFSIZE", bus_execute_append_rlimits, "t", 0 },
384 { "LimitDATA", bus_execute_append_rlimits, "t", 0 },
385 { "LimitSTACK", bus_execute_append_rlimits, "t", 0 },
386 { "LimitCORE", bus_execute_append_rlimits, "t", 0 },
387 { "LimitRSS", bus_execute_append_rlimits, "t", 0 },
388 { "LimitNOFILE", bus_execute_append_rlimits, "t", 0 },
389 { "LimitAS", bus_execute_append_rlimits, "t", 0 },
390 { "LimitNPROC", bus_execute_append_rlimits, "t", 0 },
391 { "LimitMEMLOCK", bus_execute_append_rlimits, "t", 0 },
392 { "LimitLOCKS", bus_execute_append_rlimits, "t", 0 },
393 { "LimitSIGPENDING", bus_execute_append_rlimits, "t", 0 },
394 { "LimitMSGQUEUE", bus_execute_append_rlimits, "t", 0 },
395 { "LimitNICE", bus_execute_append_rlimits, "t", 0 },
396 { "LimitRTPRIO", bus_execute_append_rlimits, "t", 0 },
397 { "LimitRTTIME", bus_execute_append_rlimits, "t", 0 },
398 { "WorkingDirectory", bus_property_append_string, "s", offsetof(ExecContext, working_directory), true },
399 { "RootDirectory", bus_property_append_string, "s", offsetof(ExecContext, root_directory), true },
400 { "OOMScoreAdjust", bus_execute_append_oom_score_adjust, "i", 0 },
401 { "Nice", bus_execute_append_nice, "i", 0 },
402 { "IOScheduling", bus_execute_append_ioprio, "i", 0 },
403 { "CPUSchedulingPolicy", bus_execute_append_cpu_sched_policy, "i", 0 },
404 { "CPUSchedulingPriority", bus_execute_append_cpu_sched_priority, "i", 0 },
405 { "CPUAffinity", bus_execute_append_affinity, "ay", 0 },
406 { "TimerSlackNSec", bus_execute_append_timer_slack_nsec, "t", 0 },
407 { "CPUSchedulingResetOnFork", bus_property_append_bool, "b", offsetof(ExecContext, cpu_sched_reset_on_fork) },
408 { "NonBlocking", bus_property_append_bool, "b", offsetof(ExecContext, non_blocking) },
409 { "StandardInput", bus_execute_append_input, "s", offsetof(ExecContext, std_input) },
410 { "StandardOutput", bus_execute_append_output, "s", offsetof(ExecContext, std_output) },
411 { "StandardError", bus_execute_append_output, "s", offsetof(ExecContext, std_error) },
412 { "TTYPath", bus_property_append_string, "s", offsetof(ExecContext, tty_path), true },
413 { "TTYReset", bus_property_append_bool, "b", offsetof(ExecContext, tty_reset) },
414 { "TTYVHangup", bus_property_append_bool, "b", offsetof(ExecContext, tty_vhangup) },
415 { "TTYVTDisallocate", bus_property_append_bool, "b", offsetof(ExecContext, tty_vt_disallocate) },
416 { "SyslogPriority", bus_property_append_int, "i", offsetof(ExecContext, syslog_priority) },
417 { "SyslogIdentifier", bus_property_append_string, "s", offsetof(ExecContext, syslog_identifier), true },
418 { "SyslogLevelPrefix", bus_property_append_bool, "b", offsetof(ExecContext, syslog_level_prefix) },
419 { "Capabilities", bus_execute_append_capabilities, "s", 0 },
420 { "SecureBits", bus_property_append_int, "i", offsetof(ExecContext, secure_bits) },
421 { "CapabilityBoundingSet", bus_execute_append_capability_bs, "t", offsetof(ExecContext, capability_bounding_set_drop) },
422 { "User", bus_property_append_string, "s", offsetof(ExecContext, user), true },
423 { "Group", bus_property_append_string, "s", offsetof(ExecContext, group), true },
424 { "SupplementaryGroups", bus_property_append_strv, "as", offsetof(ExecContext, supplementary_groups), true },
425 { "TCPWrapName", bus_property_append_string, "s", offsetof(ExecContext, tcpwrap_name), true },
426 { "PAMName", bus_property_append_string, "s", offsetof(ExecContext, pam_name), true },
427 { "ReadWriteDirectories", bus_property_append_strv, "as", offsetof(ExecContext, read_write_dirs), true },
428 { "ReadOnlyDirectories", bus_property_append_strv, "as", offsetof(ExecContext, read_only_dirs), true },
429 { "InaccessibleDirectories", bus_property_append_strv, "as", offsetof(ExecContext, inaccessible_dirs), true },
430 { "MountFlags", bus_property_append_ul, "t", offsetof(ExecContext, mount_flags) },
431 { "PrivateTmp", bus_property_append_bool, "b", offsetof(ExecContext, private_tmp) },
432 { "PrivateNetwork", bus_property_append_bool, "b", offsetof(ExecContext, private_network) },
433 { "SameProcessGroup", bus_property_append_bool, "b", offsetof(ExecContext, same_pgrp) },
434 { "UtmpIdentifier", bus_property_append_string, "s", offsetof(ExecContext, utmp_id), true },
435 { "ControlGroupModify", bus_property_append_bool, "b", offsetof(ExecContext, control_group_modify) },
436 { "ControlGroupPersistent", bus_property_append_tristate_false, "b", offsetof(ExecContext, control_group_persistent) },
437 { "IgnoreSIGPIPE", bus_property_append_bool, "b", offsetof(ExecContext, ignore_sigpipe) },
438 { "NoNewPrivileges", bus_property_append_bool, "b", offsetof(ExecContext, no_new_privileges) },
439 { "SystemCallFilter", bus_execute_append_syscall_filter, "au", 0 },