Erik Andersen | e49d5ec | 2000-02-08 19:58:47 +0000 | [diff] [blame] | 1 | /* vi: set sw=4 ts=4: */ |
Eric Andersen | c499601 | 1999-10-20 22:08:37 +0000 | [diff] [blame] | 2 | /* |
Manuel Novoa III | cad5364 | 2003-03-19 09:13:01 +0000 | [diff] [blame] | 3 | * sleep implementation for busybox |
Eric Andersen | c499601 | 1999-10-20 22:08:37 +0000 | [diff] [blame] | 4 | * |
Manuel Novoa III | cad5364 | 2003-03-19 09:13:01 +0000 | [diff] [blame] | 5 | * Copyright (C) 2003 Manuel Novoa III <mjn3@codepoet.org> |
Eric Andersen | c499601 | 1999-10-20 22:08:37 +0000 | [diff] [blame] | 6 | * |
Denys Vlasenko | 0ef64bd | 2010-08-16 20:14:46 +0200 | [diff] [blame] | 7 | * Licensed under GPLv2 or later, see file LICENSE in this source tree. |
Eric Andersen | c499601 | 1999-10-20 22:08:37 +0000 | [diff] [blame] | 8 | */ |
Manuel Novoa III | cad5364 | 2003-03-19 09:13:01 +0000 | [diff] [blame] | 9 | /* Mar 16, 2003 Manuel Novoa III (mjn3@codepoet.org) |
| 10 | * |
| 11 | * Rewritten to do proper arg and error checking. |
| 12 | * Also, added a 'fancy' configuration to accept multiple args with |
| 13 | * time suffixes for seconds, minutes, hours, and days. |
| 14 | */ |
Denys Vlasenko | af3f420 | 2016-11-23 14:46:56 +0100 | [diff] [blame] | 15 | //config:config SLEEP |
Denys Vlasenko | 4eed2c6 | 2017-07-18 22:01:24 +0200 | [diff] [blame] | 16 | //config: bool "sleep (1.7 kb)" |
Denys Vlasenko | af3f420 | 2016-11-23 14:46:56 +0100 | [diff] [blame] | 17 | //config: default y |
| 18 | //config: help |
Denys Vlasenko | 72089cf | 2017-07-21 09:50:55 +0200 | [diff] [blame] | 19 | //config: sleep is used to pause for a specified number of seconds. |
| 20 | //config: It comes in 3 versions: |
| 21 | //config: - small: takes one integer parameter |
| 22 | //config: - fancy: takes multiple integer arguments with suffixes: |
| 23 | //config: sleep 1d 2h 3m 15s |
| 24 | //config: - fancy with fractional numbers: |
| 25 | //config: sleep 2.3s 4.5h sleeps for 16202.3 seconds |
| 26 | //config: Last one is "the most compatible" with coreutils sleep, |
| 27 | //config: but it adds around 1k of code. |
Denys Vlasenko | af3f420 | 2016-11-23 14:46:56 +0100 | [diff] [blame] | 28 | //config: |
| 29 | //config:config FEATURE_FANCY_SLEEP |
| 30 | //config: bool "Enable multiple arguments and s/m/h/d suffixes" |
| 31 | //config: default y |
| 32 | //config: depends on SLEEP |
| 33 | //config: help |
Denys Vlasenko | 72089cf | 2017-07-21 09:50:55 +0200 | [diff] [blame] | 34 | //config: Allow sleep to pause for specified minutes, hours, and days. |
Denys Vlasenko | af3f420 | 2016-11-23 14:46:56 +0100 | [diff] [blame] | 35 | //config: |
| 36 | //config:config FEATURE_FLOAT_SLEEP |
| 37 | //config: bool "Enable fractional arguments" |
| 38 | //config: default y |
| 39 | //config: depends on FEATURE_FANCY_SLEEP |
| 40 | //config: help |
Denys Vlasenko | 72089cf | 2017-07-21 09:50:55 +0200 | [diff] [blame] | 41 | //config: Allow for fractional numeric parameters. |
Denys Vlasenko | af3f420 | 2016-11-23 14:46:56 +0100 | [diff] [blame] | 42 | |
| 43 | /* Do not make this applet NOFORK. It breaks ^C-ing of pauses in shells */ |
| 44 | //applet:IF_SLEEP(APPLET(sleep, BB_DIR_BIN, BB_SUID_DROP)) |
| 45 | |
| 46 | //kbuild:lib-$(CONFIG_SLEEP) += sleep.o |
| 47 | |
| 48 | /* BB_AUDIT SUSv3 compliant */ |
| 49 | /* BB_AUDIT GNU issues -- fancy version matches except args must be ints. */ |
| 50 | /* http://www.opengroup.org/onlinepubs/007904975/utilities/sleep.html */ |
Manuel Novoa III | cad5364 | 2003-03-19 09:13:01 +0000 | [diff] [blame] | 51 | |
Pere Orga | 3442538 | 2011-03-31 14:43:25 +0200 | [diff] [blame] | 52 | //usage:#define sleep_trivial_usage |
| 53 | //usage: IF_FEATURE_FANCY_SLEEP("[") "N" IF_FEATURE_FANCY_SLEEP("]...") |
| 54 | //usage:#define sleep_full_usage "\n\n" |
| 55 | //usage: IF_NOT_FEATURE_FANCY_SLEEP("Pause for N seconds") |
| 56 | //usage: IF_FEATURE_FANCY_SLEEP( |
| 57 | //usage: "Pause for a time equal to the total of the args given, where each arg can\n" |
| 58 | //usage: "have an optional suffix of (s)econds, (m)inutes, (h)ours, or (d)ays") |
| 59 | //usage: |
| 60 | //usage:#define sleep_example_usage |
| 61 | //usage: "$ sleep 2\n" |
| 62 | //usage: "[2 second delay results]\n" |
| 63 | //usage: IF_FEATURE_FANCY_SLEEP( |
| 64 | //usage: "$ sleep 1d 3h 22m 8s\n" |
| 65 | //usage: "[98528 second delay results]\n") |
| 66 | |
Denis Vlasenko | b6adbf1 | 2007-05-26 19:00:18 +0000 | [diff] [blame] | 67 | #include "libbb.h" |
Eric Andersen | cc8ed39 | 1999-10-05 16:24:54 +0000 | [diff] [blame] | 68 | |
Denis Vlasenko | 97faf53 | 2008-07-15 22:01:49 +0000 | [diff] [blame] | 69 | #if ENABLE_FEATURE_FANCY_SLEEP || ENABLE_FEATURE_FLOAT_SLEEP |
Denis Vlasenko | 1385899 | 2006-10-08 12:49:22 +0000 | [diff] [blame] | 70 | static const struct suffix_mult sfx[] = { |
Manuel Novoa III | cad5364 | 2003-03-19 09:13:01 +0000 | [diff] [blame] | 71 | { "s", 1 }, |
| 72 | { "m", 60 }, |
| 73 | { "h", 60*60 }, |
| 74 | { "d", 24*60*60 }, |
Denys Vlasenko | 043b1e5 | 2009-09-06 12:47:55 +0200 | [diff] [blame] | 75 | { "", 0 } |
Manuel Novoa III | cad5364 | 2003-03-19 09:13:01 +0000 | [diff] [blame] | 76 | }; |
| 77 | #endif |
| 78 | |
Denis Vlasenko | 9b49a5e | 2007-10-11 10:05:36 +0000 | [diff] [blame] | 79 | int sleep_main(int argc, char **argv) MAIN_EXTERNALLY_VISIBLE; |
Denis Vlasenko | a60f84e | 2008-07-05 09:18:54 +0000 | [diff] [blame] | 80 | int sleep_main(int argc UNUSED_PARAM, char **argv) |
Eric Andersen | cc8ed39 | 1999-10-05 16:24:54 +0000 | [diff] [blame] | 81 | { |
Denis Vlasenko | 97faf53 | 2008-07-15 22:01:49 +0000 | [diff] [blame] | 82 | #if ENABLE_FEATURE_FLOAT_SLEEP |
Denis Vlasenko | adbb73b | 2008-07-12 17:05:14 +0000 | [diff] [blame] | 83 | double duration; |
| 84 | struct timespec ts; |
| 85 | #else |
Denis Vlasenko | 99912ca | 2007-04-10 15:43:37 +0000 | [diff] [blame] | 86 | unsigned duration; |
Denis Vlasenko | adbb73b | 2008-07-12 17:05:14 +0000 | [diff] [blame] | 87 | #endif |
Manuel Novoa III | cad5364 | 2003-03-19 09:13:01 +0000 | [diff] [blame] | 88 | |
Denis Vlasenko | 1d42665 | 2008-03-17 09:09:09 +0000 | [diff] [blame] | 89 | ++argv; |
| 90 | if (!*argv) |
| 91 | bb_show_usage(); |
| 92 | |
Denis Vlasenko | 97faf53 | 2008-07-15 22:01:49 +0000 | [diff] [blame] | 93 | #if ENABLE_FEATURE_FLOAT_SLEEP |
Manuel Novoa III | cad5364 | 2003-03-19 09:13:01 +0000 | [diff] [blame] | 94 | |
Denys Vlasenko | 0f2e278 | 2010-09-24 12:39:45 +0200 | [diff] [blame] | 95 | # if ENABLE_LOCALE_SUPPORT |
Denys Vlasenko | 04bcd99 | 2010-09-25 17:19:04 +0200 | [diff] [blame] | 96 | /* undo busybox.c setlocale */ |
| 97 | setlocale(LC_NUMERIC, "C"); |
Denys Vlasenko | 0f2e278 | 2010-09-24 12:39:45 +0200 | [diff] [blame] | 98 | # endif |
Manuel Novoa III | cad5364 | 2003-03-19 09:13:01 +0000 | [diff] [blame] | 99 | duration = 0; |
| 100 | do { |
Denis Vlasenko | adbb73b | 2008-07-12 17:05:14 +0000 | [diff] [blame] | 101 | char *arg = *argv; |
| 102 | if (strchr(arg, '.')) { |
| 103 | double d; |
Denys Vlasenko | 4813a51 | 2009-06-15 09:50:54 +0200 | [diff] [blame] | 104 | char *pp; |
Denis Vlasenko | adbb73b | 2008-07-12 17:05:14 +0000 | [diff] [blame] | 105 | int len = strspn(arg, "0123456789."); |
| 106 | char sv = arg[len]; |
| 107 | arg[len] = '\0'; |
Denys Vlasenko | 4813a51 | 2009-06-15 09:50:54 +0200 | [diff] [blame] | 108 | errno = 0; |
| 109 | d = strtod(arg, &pp); |
| 110 | if (errno || *pp) |
Denis Vlasenko | adbb73b | 2008-07-12 17:05:14 +0000 | [diff] [blame] | 111 | bb_show_usage(); |
Denys Vlasenko | 0f2e278 | 2010-09-24 12:39:45 +0200 | [diff] [blame] | 112 | arg += len; |
| 113 | *arg-- = sv; |
| 114 | sv = *arg; |
| 115 | *arg = '1'; |
| 116 | duration += d * xatoul_sfx(arg, sfx); |
| 117 | *arg = sv; |
| 118 | } else { |
Denis Vlasenko | adbb73b | 2008-07-12 17:05:14 +0000 | [diff] [blame] | 119 | duration += xatoul_sfx(arg, sfx); |
Denys Vlasenko | 0f2e278 | 2010-09-24 12:39:45 +0200 | [diff] [blame] | 120 | } |
Manuel Novoa III | cad5364 | 2003-03-19 09:13:01 +0000 | [diff] [blame] | 121 | } while (*++argv); |
| 122 | |
Denis Vlasenko | adbb73b | 2008-07-12 17:05:14 +0000 | [diff] [blame] | 123 | ts.tv_sec = MAXINT(typeof(ts.tv_sec)); |
| 124 | ts.tv_nsec = 0; |
| 125 | if (duration >= 0 && duration < ts.tv_sec) { |
| 126 | ts.tv_sec = duration; |
| 127 | ts.tv_nsec = (duration - ts.tv_sec) * 1000000000; |
| 128 | } |
| 129 | do { |
| 130 | errno = 0; |
| 131 | nanosleep(&ts, &ts); |
| 132 | } while (errno == EINTR); |
| 133 | |
| 134 | #elif ENABLE_FEATURE_FANCY_SLEEP |
| 135 | |
| 136 | duration = 0; |
| 137 | do { |
| 138 | duration += xatou_range_sfx(*argv, 0, UINT_MAX - duration, sfx); |
| 139 | } while (*++argv); |
| 140 | sleep(duration); |
| 141 | |
| 142 | #else /* simple */ |
Manuel Novoa III | cad5364 | 2003-03-19 09:13:01 +0000 | [diff] [blame] | 143 | |
Denis Vlasenko | 1d42665 | 2008-03-17 09:09:09 +0000 | [diff] [blame] | 144 | duration = xatou(*argv); |
Denis Vlasenko | adbb73b | 2008-07-12 17:05:14 +0000 | [diff] [blame] | 145 | sleep(duration); |
| 146 | // Off. If it's really needed, provide example why |
| 147 | //if (sleep(duration)) { |
| 148 | // bb_perror_nomsg_and_die(); |
| 149 | //} |
Manuel Novoa III | cad5364 | 2003-03-19 09:13:01 +0000 | [diff] [blame] | 150 | |
Denis Vlasenko | adbb73b | 2008-07-12 17:05:14 +0000 | [diff] [blame] | 151 | #endif |
Manuel Novoa III | cad5364 | 2003-03-19 09:13:01 +0000 | [diff] [blame] | 152 | |
Matt Kraai | 3e856ce | 2000-12-01 02:55:13 +0000 | [diff] [blame] | 153 | return EXIT_SUCCESS; |
Eric Andersen | cc8ed39 | 1999-10-05 16:24:54 +0000 | [diff] [blame] | 154 | } |