X-Git-Url: https://www.chiark.greenend.org.uk/ucgi/~ianmdlvl/git?a=blobdiff_plain;f=src%2Fsysctl%2Fsysctl.c;h=43c765f064aa4acf92c3d553c052062fdc0b8416;hb=924bc14fef39373f4523664207007a6c82c2b2d5;hp=8868732ad1352230fbb2e5ff1b4c9d2cd4ef8202;hpb=4cf7ea556aa1e74f9b34d4467f36d46a1bb25da3;p=elogind.git diff --git a/src/sysctl/sysctl.c b/src/sysctl/sysctl.c index 8868732ad..43c765f06 100644 --- a/src/sysctl/sysctl.c +++ b/src/sysctl/sysctl.c @@ -35,70 +35,11 @@ #include "conf-files.h" #include "fileio.h" #include "build.h" +#include "sysctl-util.h" static char **arg_prefixes = NULL; -static const char conf_file_dirs[] = - "/etc/sysctl.d\0" - "/run/sysctl.d\0" - "/usr/local/lib/sysctl.d\0" - "/usr/lib/sysctl.d\0" -#ifdef HAVE_SPLIT_USR - "/lib/sysctl.d\0" -#endif - ; - -static char *normalize_sysctl(char *s) { - char *n; - - for (n = s; *n; n++) - if (*n == '.') - *n = '/'; - - return s; -} - -static int apply_sysctl(const char *property, const char *value) { - _cleanup_free_ char *p = NULL; - char *n; - int r = 0, k; - - log_debug("Setting '%s' to '%s'", property, value); - - p = new(char, sizeof("/proc/sys/") + strlen(property)); - if (!p) - return log_oom(); - - n = stpcpy(p, "/proc/sys/"); - strcpy(n, property); - - if (!strv_isempty(arg_prefixes)) { - char **i; - bool good = false; - - STRV_FOREACH(i, arg_prefixes) - if (path_startswith(p, *i)) { - good = true; - break; - } - - if (!good) { - log_debug("Skipping %s", p); - return 0; - } - } - - k = write_string_file(p, value); - if (k < 0) { - log_full(k == -ENOENT ? LOG_DEBUG : LOG_WARNING, - "Failed to write '%s' to '%s': %s", value, p, strerror(-k)); - - if (k != -ENOENT && r == 0) - r = k; - } - - return r; -} +static const char conf_file_dirs[] = CONF_DIRS_NULSTR("sysctl"); static int apply_all(Hashmap *sysctl_options) { int r = 0; @@ -110,9 +51,14 @@ static int apply_all(Hashmap *sysctl_options) { HASHMAP_FOREACH_KEY(value, property, sysctl_options, i) { int k; - k = apply_sysctl(property, value); - if (k < 0 && r == 0) - r = k; + k = sysctl_write(property, value); + if (k < 0) { + log_full(k == -ENOENT ? LOG_DEBUG : LOG_WARNING, + "Failed to write '%s' to '%s': %s", value, property, strerror(-k)); + + if (r == 0) + r = k; + } } return r; } @@ -128,11 +74,10 @@ static int parse_file(Hashmap *sysctl_options, const char *path, bool ignore_eno if (ignore_enoent && r == -ENOENT) return 0; - log_error("Failed to open file '%s', ignoring: %s", path, strerror(-r)); - return r; + return log_error_errno(r, "Failed to open file '%s', ignoring: %m", path); } - log_debug("parse: %s", path); + log_debug("Parsing %s", path); while (!feof(f)) { char l[LINE_MAX], *p, *value, *new_value, *property, *existing; void *v; @@ -142,7 +87,7 @@ static int parse_file(Hashmap *sysctl_options, const char *path, bool ignore_eno if (feof(f)) break; - log_error("Failed to read file '%s', ignoring: %m", path); + log_error_errno(errno, "Failed to read file '%s', ignoring: %m", path); return -errno; } @@ -165,15 +110,29 @@ static int parse_file(Hashmap *sysctl_options, const char *path, bool ignore_eno *value = 0; value++; - p = normalize_sysctl(strstrip(p)); + p = sysctl_normalize(strstrip(p)); value = strstrip(value); + if (!strv_isempty(arg_prefixes)) { + char **i, *t; + STRV_FOREACH(i, arg_prefixes) { + t = path_startswith(*i, "/proc/sys/"); + if (t == NULL) + t = *i; + if (path_startswith(p, t)) + goto found; + } + /* not found */ + continue; + } + +found: existing = hashmap_get2(sysctl_options, p, &v); if (existing) { if (streq(value, existing)) continue; - log_info("Overwriting earlier assignment of %s in file '%s'.", p, path); + log_debug("Overwriting earlier assignment of %s in file '%s'.", p, path); free(hashmap_remove(sysctl_options, p)); free(v); } @@ -190,7 +149,7 @@ static int parse_file(Hashmap *sysctl_options, const char *path, bool ignore_eno k = hashmap_put(sysctl_options, property, new_value); if (k < 0) { - log_error("Failed to add sysctl variable %s to hashmap: %s", property, strerror(-k)); + log_error_errno(k, "Failed to add sysctl variable %s to hashmap: %m", property); free(property); free(new_value); return k; @@ -200,16 +159,13 @@ static int parse_file(Hashmap *sysctl_options, const char *path, bool ignore_eno return r; } -static int help(void) { - +static void help(void) { printf("%s [OPTIONS...] [CONFIGURATION FILE...]\n\n" "Applies kernel sysctl settings.\n\n" " -h --help Show this help\n" " --version Show package version\n" - " --prefix=PATH Only apply rules that apply to paths with the specified prefix\n", - program_invocation_short_name); - - return 0; + " --prefix=PATH Only apply rules with the specified prefix\n" + , program_invocation_short_name); } static int parse_argv(int argc, char *argv[]) { @@ -231,12 +187,13 @@ static int parse_argv(int argc, char *argv[]) { assert(argc >= 0); assert(argv); - while ((c = getopt_long(argc, argv, "h", options, NULL)) >= 0) { + while ((c = getopt_long(argc, argv, "h", options, NULL)) >= 0) switch (c) { case 'h': - return help(); + help(); + return 0; case ARG_VERSION: puts(PACKAGE_STRING); @@ -246,11 +203,19 @@ static int parse_argv(int argc, char *argv[]) { case ARG_PREFIX: { char *p; - for (p = optarg; *p; p++) - if (*p == '.') - *p = '/'; - - if (strv_extend(&arg_prefixes, optarg) < 0) + /* We used to require people to specify absolute paths + * in /proc/sys in the past. This is kinda useless, but + * we need to keep compatibility. We now support any + * sysctl name available. */ + sysctl_normalize(optarg); + if (startswith(optarg, "/proc/sys")) + p = strdup(optarg); + else + p = strappend("/proc/sys/", optarg); + + if (!p) + return log_oom(); + if (strv_consume(&arg_prefixes, p) < 0) return log_oom(); break; @@ -262,7 +227,6 @@ static int parse_argv(int argc, char *argv[]) { default: assert_not_reached("Unhandled option"); } - } return 1; } @@ -281,7 +245,7 @@ int main(int argc, char *argv[]) { umask(0022); - sysctl_options = hashmap_new(string_hash_func, string_compare_func); + sysctl_options = hashmap_new(&string_hash_ops); if (!sysctl_options) { r = log_oom(); goto finish; @@ -303,7 +267,7 @@ int main(int argc, char *argv[]) { r = conf_files_list_nulstr(&files, ".conf", NULL, conf_file_dirs); if (r < 0) { - log_error("Failed to enumerate sysctl.d files: %s", strerror(-r)); + log_error_errno(r, "Failed to enumerate sysctl.d files: %m"); goto finish; }