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/>.
30 #include "conf-files.h"
34 #include "path-util.h"
35 #include "string-util.h"
37 #include "sysctl-util.h"
40 static char **arg_prefixes
= NULL
;
42 static const char conf_file_dirs
[] = CONF_DIRS_NULSTR("sysctl");
44 static int apply_all(Hashmap
*sysctl_options
) {
45 char *property
, *value
;
49 HASHMAP_FOREACH_KEY(value
, property
, sysctl_options
, i
) {
52 k
= sysctl_write(property
, value
);
54 log_full_errno(k
== -ENOENT
? LOG_INFO
: LOG_WARNING
, k
,
55 "Couldn't write '%s' to '%s', ignoring: %m", value
, property
);
57 if (r
== 0 && k
!= -ENOENT
)
65 static int parse_file(Hashmap
*sysctl_options
, const char *path
, bool ignore_enoent
) {
66 _cleanup_fclose_
FILE *f
= NULL
;
71 r
= search_and_fopen_nulstr(path
, "re", NULL
, conf_file_dirs
, &f
);
73 if (ignore_enoent
&& r
== -ENOENT
)
76 return log_error_errno(r
, "Failed to open file '%s', ignoring: %m", path
);
79 log_debug("Parsing %s", path
);
81 char l
[LINE_MAX
], *p
, *value
, *new_value
, *property
, *existing
;
85 if (!fgets(l
, sizeof(l
), f
)) {
89 log_error_errno(errno
, "Failed to read file '%s', ignoring: %m", path
);
97 if (strchr(COMMENTS
"\n", *p
))
100 value
= strchr(p
, '=');
102 log_error("Line is not an assignment in file '%s': %s", path
, value
);
112 p
= sysctl_normalize(strstrip(p
));
113 value
= strstrip(value
);
115 if (!strv_isempty(arg_prefixes
)) {
117 STRV_FOREACH(i
, arg_prefixes
) {
118 t
= path_startswith(*i
, "/proc/sys/");
121 if (path_startswith(p
, t
))
129 existing
= hashmap_get2(sysctl_options
, p
, &v
);
131 if (streq(value
, existing
))
134 log_debug("Overwriting earlier assignment of %s in file '%s'.", p
, path
);
135 free(hashmap_remove(sysctl_options
, p
));
139 property
= strdup(p
);
143 new_value
= strdup(value
);
149 k
= hashmap_put(sysctl_options
, property
, new_value
);
151 log_error_errno(k
, "Failed to add sysctl variable %s to hashmap: %m", property
);
161 static void help(void) {
162 printf("%s [OPTIONS...] [CONFIGURATION FILE...]\n\n"
163 "Applies kernel sysctl settings.\n\n"
164 " -h --help Show this help\n"
165 " --version Show package version\n"
166 " --prefix=PATH Only apply rules with the specified prefix\n"
167 , program_invocation_short_name
);
170 static int parse_argv(int argc
, char *argv
[]) {
177 static const struct option options
[] = {
178 { "help", no_argument
, NULL
, 'h' },
179 { "version", no_argument
, NULL
, ARG_VERSION
},
180 { "prefix", required_argument
, NULL
, ARG_PREFIX
},
189 while ((c
= getopt_long(argc
, argv
, "h", options
, NULL
)) >= 0)
203 /* We used to require people to specify absolute paths
204 * in /proc/sys in the past. This is kinda useless, but
205 * we need to keep compatibility. We now support any
206 * sysctl name available. */
207 sysctl_normalize(optarg
);
209 if (startswith(optarg
, "/proc/sys"))
212 p
= strappend("/proc/sys/", optarg
);
216 if (strv_consume(&arg_prefixes
, p
) < 0)
226 assert_not_reached("Unhandled option");
232 int main(int argc
, char *argv
[]) {
234 Hashmap
*sysctl_options
;
236 r
= parse_argv(argc
, argv
);
238 return r
< 0 ? EXIT_FAILURE
: EXIT_SUCCESS
;
240 log_set_target(LOG_TARGET_AUTO
);
241 log_parse_environment();
246 sysctl_options
= hashmap_new(&string_hash_ops
);
247 if (!sysctl_options
) {
257 for (i
= optind
; i
< argc
; i
++) {
258 k
= parse_file(sysctl_options
, argv
[i
], false);
263 _cleanup_strv_free_
char **files
= NULL
;
266 r
= conf_files_list_nulstr(&files
, ".conf", NULL
, conf_file_dirs
);
268 log_error_errno(r
, "Failed to enumerate sysctl.d files: %m");
272 STRV_FOREACH(f
, files
) {
273 k
= parse_file(sysctl_options
, *f
, true);
279 k
= apply_all(sysctl_options
);
284 hashmap_free_free_free(sysctl_options
);
285 strv_free(arg_prefixes
);
287 return r
< 0 ? EXIT_FAILURE
: EXIT_SUCCESS
;