diff options
Diffstat (limited to 'src/core/load-fragment.c')
| -rw-r--r-- | src/core/load-fragment.c | 229 | 
1 files changed, 117 insertions, 112 deletions
| diff --git a/src/core/load-fragment.c b/src/core/load-fragment.c index d5cf476fda..3cb2cffedf 100644 --- a/src/core/load-fragment.c +++ b/src/core/load-fragment.c @@ -1089,46 +1089,107 @@ int config_parse_bounding_set(          return 0;  } -int config_parse_limit( -                const char *unit, -                const char *filename, -                unsigned line, -                const char *section, -                unsigned section_line, -                const char *lvalue, -                int ltype, -                const char *rvalue, -                void *data, -                void *userdata) { - -        struct rlimit **rl = data; -        rlim_t v; -        int r; - -        assert(filename); -        assert(lvalue); -        assert(rvalue); -        assert(data); +static int rlim_parse_u64(const char *val, rlim_t *res) { +        int r = 0; -        rl += ltype; - -        if (streq(rvalue, "infinity")) -                v = RLIM_INFINITY; +        if (streq(val, "infinity")) +                *res = RLIM_INFINITY;          else {                  uint64_t u;                  /* setrlimit(2) suggests rlim_t is always 64bit on Linux. */                  assert_cc(sizeof(rlim_t) == sizeof(uint64_t)); -                r = safe_atou64(rvalue, &u); +                r = safe_atou64(val, &u);                  if (r >= 0 && u >= (uint64_t) RLIM_INFINITY)                          r = -ERANGE; -                if (r < 0) { -                        log_syntax(unit, LOG_ERR, filename, line, r, "Failed to parse resource value, ignoring: %s", rvalue); -                        return 0; -                } +                if (r == 0) +                        *res = (rlim_t) u; +        } +        return r; +} + +static int rlim_parse_size(const char *val, rlim_t *res) { +        int r = 0; + +        if (streq(val, "infinity")) +                *res = RLIM_INFINITY; +        else { +                uint64_t u; + +                r = parse_size(val, 1024, &u); +                if (r >= 0 && u >= (uint64_t) RLIM_INFINITY) +                        r = -ERANGE; +                if (r == 0) +                        *res = (rlim_t) u; +        } +        return r; +} + +static int rlim_parse_sec(const char *val, rlim_t *res) { +        int r = 0; + +        if (streq(val, "infinity")) +                *res = RLIM_INFINITY; +        else { +                usec_t t; + +                r = parse_sec(val, &t); +                if (r < 0) +                        return r; +                if (t == USEC_INFINITY) +                        *res = RLIM_INFINITY; +                else +                        *res = (rlim_t) (DIV_ROUND_UP(t, USEC_PER_SEC)); + +        } +        return r; +} + +static int rlim_parse_usec(const char *val, rlim_t *res) { +        int r = 0; + +        if (streq(val, "infinity")) +                *res = RLIM_INFINITY; +        else { +                usec_t t; + +                r = parse_time(val, &t, 1); +                if (r < 0) +                        return r; +                if (t == USEC_INFINITY) +                        *res = RLIM_INFINITY; +                else +                        *res = (rlim_t) t; +        } +        return r; +} -                v = (rlim_t) u; +static int parse_rlimit_range( +                const char *unit, +                const char *filename, +                unsigned line, +                const char *value, +                struct rlimit **rl, +                int (*rlim_parser)(const char *, rlim_t *)) { + +        rlim_t soft, hard; +        _cleanup_free_ char *sword = NULL, *hword = NULL; +        int nwords, r; + +        assert(value); + +        /* <value> or <soft:hard> */ +        nwords = extract_many_words(&value, ":", EXTRACT_DONT_COALESCE_SEPARATORS, &sword, &hword, NULL); +        r = nwords < 0 ? nwords : nwords == 0 ? -EINVAL : 0; + +        if (r == 0) +                r = rlim_parser(sword, &soft); +        if (r == 0 && nwords == 2) +                r = rlim_parser(hword, &hard); +        if (r < 0) { +                log_syntax(unit, LOG_ERR, filename, line, r, "Failed to parse resource value, ignoring: %s", value); +                return 0;          }          if (!*rl) { @@ -1136,12 +1197,12 @@ int config_parse_limit(                  if (!*rl)                          return log_oom();          } - -        (*rl)->rlim_cur = (*rl)->rlim_max = v; +        (*rl)->rlim_cur = soft; +        (*rl)->rlim_max = nwords == 2 ? hard : soft;          return 0;  } -int config_parse_bytes_limit( +int config_parse_limit(                  const char *unit,                  const char *filename,                  unsigned line, @@ -1154,8 +1215,6 @@ int config_parse_bytes_limit(                  void *userdata) {          struct rlimit **rl = data; -        rlim_t bytes; -        int r;          assert(filename);          assert(lvalue); @@ -1163,31 +1222,30 @@ int config_parse_bytes_limit(          assert(data);          rl += ltype; +        return parse_rlimit_range(unit, filename, line, rvalue, rl, rlim_parse_u64); +} -        if (streq(rvalue, "infinity")) -                bytes = RLIM_INFINITY; -        else { -                uint64_t u; - -                r = parse_size(rvalue, 1024, &u); -                if (r >= 0 && u >= (uint64_t) RLIM_INFINITY) -                        r = -ERANGE; -                if (r < 0) { -                        log_syntax(unit, LOG_ERR, filename, line, r, "Failed to parse resource value, ignoring: %s", rvalue); -                        return 0; -                } +int config_parse_bytes_limit( +                const char *unit, +                const char *filename, +                unsigned line, +                const char *section, +                unsigned section_line, +                const char *lvalue, +                int ltype, +                const char *rvalue, +                void *data, +                void *userdata) { -                bytes = (rlim_t) u; -        } +        struct rlimit **rl = data; -        if (!*rl) { -                *rl = new(struct rlimit, 1); -                if (!*rl) -                        return log_oom(); -        } +        assert(filename); +        assert(lvalue); +        assert(rvalue); +        assert(data); -        (*rl)->rlim_cur = (*rl)->rlim_max = bytes; -        return 0; +        rl += ltype; +        return parse_rlimit_range(unit, filename, line, rvalue, rl, rlim_parse_size);  }  int config_parse_sec_limit( @@ -1203,8 +1261,6 @@ int config_parse_sec_limit(                  void *userdata) {          struct rlimit **rl = data; -        rlim_t seconds; -        int r;          assert(filename);          assert(lvalue); @@ -1212,35 +1268,9 @@ int config_parse_sec_limit(          assert(data);          rl += ltype; - -        if (streq(rvalue, "infinity")) -                seconds = RLIM_INFINITY; -        else { -                usec_t t; - -                r = parse_sec(rvalue, &t); -                if (r < 0) { -                        log_syntax(unit, LOG_ERR, filename, line, r, "Failed to parse resource value, ignoring: %s", rvalue); -                        return 0; -                } - -                if (t == USEC_INFINITY) -                        seconds = RLIM_INFINITY; -                else -                        seconds = (rlim_t) (DIV_ROUND_UP(t, USEC_PER_SEC)); -        } - -        if (!*rl) { -                *rl = new(struct rlimit, 1); -                if (!*rl) -                        return log_oom(); -        } - -        (*rl)->rlim_cur = (*rl)->rlim_max = seconds; -        return 0; +        return parse_rlimit_range(unit, filename, line, rvalue, rl, rlim_parse_sec);  } -  int config_parse_usec_limit(                  const char *unit,                  const char *filename, @@ -1254,8 +1284,6 @@ int config_parse_usec_limit(                  void *userdata) {          struct rlimit **rl = data; -        rlim_t useconds; -        int r;          assert(filename);          assert(lvalue); @@ -1263,33 +1291,10 @@ int config_parse_usec_limit(          assert(data);          rl += ltype; +        return parse_rlimit_range(unit, filename, line, rvalue, rl, rlim_parse_usec); +} -        if (streq(rvalue, "infinity")) -                useconds = RLIM_INFINITY; -        else { -                usec_t t; - -                r = parse_time(rvalue, &t, 1); -                if (r < 0) { -                        log_syntax(unit, LOG_ERR, filename, line, r, "Failed to parse resource value, ignoring: %s", rvalue); -                        return 0; -                } - -                if (t == USEC_INFINITY) -                        useconds = RLIM_INFINITY; -                else -                        useconds = (rlim_t) t; -        } - -        if (!*rl) { -                *rl = new(struct rlimit, 1); -                if (!*rl) -                        return log_oom(); -        } -        (*rl)->rlim_cur = (*rl)->rlim_max = useconds; -        return 0; -}  #ifdef HAVE_SYSV_COMPAT  int config_parse_sysv_priority(const char *unit, | 
