diff options
Diffstat (limited to 'src/basic')
| -rw-r--r-- | src/basic/calendarspec.c | 88 | ||||
| -rw-r--r-- | src/basic/fd-util.c | 6 | ||||
| -rw-r--r-- | src/basic/fd-util.h | 1 | ||||
| -rw-r--r-- | src/basic/terminal-util.c | 4 | 
4 files changed, 71 insertions, 28 deletions
| diff --git a/src/basic/calendarspec.c b/src/basic/calendarspec.c index 6e0bab9b94..e4cfab364e 100644 --- a/src/basic/calendarspec.c +++ b/src/basic/calendarspec.c @@ -32,6 +32,8 @@  #include "parse-util.h"  #include "string-util.h" +/* Longest valid date/time range is 1970..2199 */ +#define MAX_RANGE_LEN   230  #define BITS_WEEKDAYS   127  static void free_chain(CalendarComponent *c) { @@ -200,7 +202,7 @@ static void format_weekdays(FILE *f, const CalendarSpec *c) {          };          int l, x; -        bool need_colon = false; +        bool need_comma = false;          assert(f);          assert(c); @@ -211,10 +213,10 @@ static void format_weekdays(FILE *f, const CalendarSpec *c) {                  if (c->weekdays_bits & (1 << x)) {                          if (l < 0) { -                                if (need_colon) +                                if (need_comma)                                          fputc(',', f);                                  else -                                        need_colon = true; +                                        need_comma = true;                                  fputs(days[x], f);                                  l = x; @@ -223,7 +225,7 @@ static void format_weekdays(FILE *f, const CalendarSpec *c) {                  } else if (l >= 0) {                          if (x > l + 1) { -                                fputc(x > l + 2 ? '-' : ',', f); +                                fputs(x > l + 2 ? ".." : ",", f);                                  fputs(days[x-1], f);                          } @@ -232,7 +234,7 @@ static void format_weekdays(FILE *f, const CalendarSpec *c) {          }          if (l >= 0 && x > l + 1) { -                fputc(x > l + 2 ? '-' : ',', f); +                fputs(x > l + 2 ? ".." : ",", f);                  fputs(days[x-1], f);          }  } @@ -357,6 +359,7 @@ static int parse_weekdays(const char **p, CalendarSpec *c) {                          skip = strlen(day_nr[i].name);                          if ((*p)[skip] != '-' && +                            (*p)[skip] != '.' &&                              (*p)[skip] != ',' &&                              (*p)[skip] != ' ' &&                              (*p)[skip] != 0) @@ -394,7 +397,18 @@ static int parse_weekdays(const char **p, CalendarSpec *c) {                          return 0;                  } -                if (**p == '-') { +                if (**p == '.') { +                        if (l >= 0) +                                return -EINVAL; + +                        if ((*p)[1] != '.') +                                return -EINVAL; + +                        l = day_nr[i].nr; +                        *p += 1; + +                /* Support ranges with "-" for backwards compatibility */ +                } else if (**p == '-') {                          if (l >= 0)                                  return -EINVAL; @@ -448,8 +462,26 @@ static int parse_component_decimal(const char **p, bool usec, unsigned long *res          return 0;  } +static int const_chain(int value, CalendarComponent **c) { +        CalendarComponent *cc = NULL; + +        assert(c); + +        cc = new0(CalendarComponent, 1); +        if (!cc) +                return -ENOMEM; + +        cc->value = value; +        cc->repeat = 0; +        cc->next = *c; + +        *c = cc; + +        return 0; +} +  static int prepend_component(const char **p, bool usec, CalendarComponent **c) { -        unsigned long value, repeat = 0; +        unsigned long i, value, range_end, range_inc, repeat = 0;          CalendarComponent *cc;          int r;          const char *e; @@ -471,6 +503,30 @@ static int prepend_component(const char **p, bool usec, CalendarComponent **c) {                  if (repeat == 0)                          return -ERANGE; +        } else if (e[0] == '.' && e[1] == '.') { +                e += 2; +                r = parse_component_decimal(&e, usec, &range_end); +                if (r < 0) +                        return r; + +                if (value >= range_end) +                        return -EINVAL; + +                range_inc = usec ? USEC_PER_SEC : 1; + +                /* Don't allow impossibly large ranges... */ +                if (range_end - value >= MAX_RANGE_LEN * range_inc) +                        return -EINVAL; + +                /* ...or ranges with only a single element */ +                if (range_end - value < range_inc) +                        return -EINVAL; + +                for (i = value; i <= range_end; i += range_inc) { +                        r = const_chain(i, c); +                        if (r < 0) +                                return r; +                }          }          if (*e != 0 && *e != ' ' && *e != ',' && *e != '-' && *e != ':') @@ -495,24 +551,6 @@ static int prepend_component(const char **p, bool usec, CalendarComponent **c) {          return 0;  } -static int const_chain(int value, CalendarComponent **c) { -        CalendarComponent *cc = NULL; - -        assert(c); - -        cc = new0(CalendarComponent, 1); -        if (!cc) -                return -ENOMEM; - -        cc->value = value; -        cc->repeat = 0; -        cc->next = *c; - -        *c = cc; - -        return 0; -} -  static int parse_chain(const char **p, bool usec, CalendarComponent **c) {          const char *t;          CalendarComponent *cc = NULL; diff --git a/src/basic/fd-util.c b/src/basic/fd-util.c index 8b466cff15..5c820332a5 100644 --- a/src/basic/fd-util.c +++ b/src/basic/fd-util.c @@ -186,6 +186,12 @@ int fd_cloexec(int fd, bool cloexec) {          return 0;  } +void stdio_unset_cloexec(void) { +        fd_cloexec(STDIN_FILENO, false); +        fd_cloexec(STDOUT_FILENO, false); +        fd_cloexec(STDERR_FILENO, false); +} +  _pure_ static bool fd_in_set(int fd, const int fdset[], unsigned n_fdset) {          unsigned i; diff --git a/src/basic/fd-util.h b/src/basic/fd-util.h index b86e41698a..34b98d4aec 100644 --- a/src/basic/fd-util.h +++ b/src/basic/fd-util.h @@ -63,6 +63,7 @@ DEFINE_TRIVIAL_CLEANUP_FUNC(DIR*, closedir);  int fd_nonblock(int fd, bool nonblock);  int fd_cloexec(int fd, bool cloexec); +void stdio_unset_cloexec(void);  int close_all_fds(const int except[], unsigned n_except); diff --git a/src/basic/terminal-util.c b/src/basic/terminal-util.c index d8cca55378..df56d85317 100644 --- a/src/basic/terminal-util.c +++ b/src/basic/terminal-util.c @@ -888,9 +888,7 @@ int make_stdio(int fd) {          /* Explicitly unset O_CLOEXEC, since if fd was < 3, then           * dup2() was a NOP and the bit hence possibly set. */ -        fd_cloexec(STDIN_FILENO, false); -        fd_cloexec(STDOUT_FILENO, false); -        fd_cloexec(STDERR_FILENO, false); +        stdio_unset_cloexec();          return 0;  } | 
