/* sleep - delay for a specified amount of time. Copyright (C) 84, 1991-1997, 1999 Free Software Foundation, Inc. This program is free software; you can redistribute it and/or modify it under the terms of the GNU General Public License as published by the Free Software Foundation; either version 2, or (at your option) any later version. This program is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for more details. You should have received a copy of the GNU General Public License along with this program; if not, write to the Free Software Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA. */ #include #include #include #include #include #include #define USE_CLOCK_GETTIME (defined CLOCK_REALTIME && HAVE_CLOCK_GETTIME) #if ! USE_CLOCK_GETTIME # include #endif #ifndef TIME_T_MAX # define TIME_T_MAX TYPE_MAXIMUM (time_t) #endif #include "system.h" #include "error.h" #include "long-options.h" #include "xstrtod.h" /* The official name of this program (e.g., no `g' prefix). */ #define PROGRAM_NAME "sleep" #define AUTHORS "Jim Meyering" /* The name by which this program was run. */ char *program_name; static struct option const long_options[] = { {0, 0, 0, 0} }; void usage (int status) { if (status != 0) fprintf (stderr, _("Try `%s --help' for more information.\n"), program_name); else { printf (_("\ Usage: %s NUMBER[SUFFIX]...\n\ or: %s OPTION\n\ Pause for NUMBER seconds.\n\ SUFFIX may be s for seconds (the default), m for minutes,\n\ h for hours or d for days.\n\ \n\ --help display this help and exit\n\ --version output version information and exit\n\ "), program_name, program_name); puts (_("\nReport bugs to .")); } exit (status); } /* FIXME: describe */ static int apply_suffix (double *s, char suffix_char) { unsigned int multiplier; assert (*s <= TIME_T_MAX); switch (suffix_char) { case 0: case 's': multiplier = 1; break; case 'm': multiplier = 60; break; case 'h': multiplier = 60 * 60; break; case 'd': multiplier = 60 * 60 * 24; break; default: multiplier = 0; } if (multiplier == 0) return 1; *s *= multiplier; return 0; } /* Subtract the `struct timespec' values X and Y, storing the difference in DIFF. Return 1 if the difference is negative, otherwise 0. From the GNU libc manual. */ static int timespec_subtract (struct timespec *diff, const struct timespec *x, struct timespec *y) { /* Perform the carry for the later subtraction by updating Y. */ if (x->tv_nsec < y->tv_nsec) { int nsec = (y->tv_nsec - x->tv_nsec) / 1000000000 + 1; y->tv_nsec -= 1000000000 * nsec; y->tv_sec += nsec; } if (1000000000 < x->tv_nsec - y->tv_nsec) { int nsec = (y->tv_nsec - x->tv_nsec) / 1000000000; y->tv_nsec += 1000000000 * nsec; y->tv_sec -= nsec; } /* Compute the time remaining to wait. `tv_nsec' is certainly positive. */ diff->tv_sec = x->tv_sec - y->tv_sec; diff->tv_nsec = x->tv_nsec - y->tv_nsec; /* Return 1 if result is negative. */ return x->tv_sec < y->tv_sec; } static void clock_get_realtime (struct timespec *ts) { int fail; #if USE_CLOCK_GETTIME fail = clock_gettime (CLOCK_REALTIME, &ts); #else struct timeval tv; fail = gettimeofday (&tv, NULL); if (!fail) { ts->tv_sec = tv.tv_sec; ts->tv_nsec = 1000 * tv.tv_usec; } #endif if (fail) error (1, errno, _("cannot read realtime clock")); } int main (int argc, char **argv) { int i; double seconds = 0.0; int c; int fail = 0; struct timespec ts_start; struct timespec ts_stop; struct timespec ts_sleep; /* Record start time. */ clock_get_realtime (&ts_start); program_name = argv[0]; setlocale (LC_ALL, ""); bindtextdomain (PACKAGE, LOCALEDIR); textdomain (PACKAGE); parse_long_options (argc, argv, PROGRAM_NAME, GNU_PACKAGE, VERSION, AUTHORS, usage); while ((c = getopt_long (argc, argv, "", long_options, NULL)) != -1) { switch (c) { case 0: break; default: usage (1); } } if (argc == 1) { error (0, 0, _("too few arguments")); usage (1); } for (i = optind; i < argc; i++) { double s; const char *p; if (xstrtod (argv[i], &p, &s) /* No negative intervals. */ || s < 0 /* S must fit in a time_t. */ || TIME_T_MAX < s /* No extra chars after the number and an optional s,m,h,d char. */ || (*p && *(p+1)) /* Check any suffix char and update S based on the suffix. */ || apply_suffix (&s, *p) /* Make sure the sum fits in a time_t. */ || TIME_T_MAX < (seconds += s) ) { error (0, 0, _("invalid time interval `%s'"), argv[i]); fail = 1; } } if (fail) usage (1); /* Add this here so we end up rounding to the nearest nanosecond. This ensures that that tv_nsec will be no larger than 999,999,999. */ seconds += .0000000005; /* Separate whole seconds from nanoseconds. */ ts_sleep.tv_sec = seconds; ts_sleep.tv_nsec = (seconds - ts_sleep.tv_sec) * 1000000000; ts_stop.tv_sec = ts_start.tv_sec + ts_sleep.tv_sec; ts_stop.tv_nsec = ts_start.tv_nsec + ts_sleep.tv_nsec; if (1000000000 <= ts_stop.tv_nsec) { ++ts_stop.tv_sec; ts_stop.tv_nsec -= 1000000000; } while (1) { struct timespec remaining; struct timespec ts_now; int negative; int interrupted = nanosleep (&ts_sleep, &remaining); if (!interrupted) break; clock_get_realtime (&ts_now); negative = timespec_subtract (&ts_sleep, &ts_stop, &ts_now); if (negative) break; } exit (0); }