Bash: infinite sleep (infinite blocking)

前端 未结 11 568
日久生厌
日久生厌 2020-12-04 08:27

I use startx to start X which will evaluate my .xinitrc. In my .xinitrc I start my window manager using /usr/bin/mywm. No

相关标签:
11条回答
  • 2020-12-04 08:57

    Let me explain why sleep infinity works though it is not documented. jp48's answer is also useful.

    The most important thing: By specifying inf or infinity (both case-insensitive), you can sleep for the longest time your implementation permits (i.e. the smaller value of HUGE_VAL and TYPE_MAXIMUM(time_t)).

    Now let's dig into the details. The source code of sleep command can be read from coreutils/src/sleep.c. Essentially, the function does this:

    double s; //seconds
    xstrtod (argv[i], &p, &s, cl_strtod); //`p` is not essential (just used for error check).
    xnanosleep (s);
    

    Understanding xstrtod (argv[i], &p, &s, cl_strtod)

    xstrtod()

    According to gnulib/lib/xstrtod.c, the call of xstrtod() converts string argv[i] to a floating point value and stores it to *s, using a converting function cl_strtod().

    cl_strtod()

    As can be seen from coreutils/lib/cl-strtod.c, cl_strtod() converts a string to a floating point value, using strtod().

    strtod()

    According to man 3 strtod, strtod() converts a string to a value of type double. The manpage says

    The expected form of the (initial portion of the) string is ... or (iii) an infinity, or ...

    and an infinity is defined as

    An infinity is either "INF" or "INFINITY", disregarding case.

    Although the document tells

    If the correct value would cause overflow, plus or minus HUGE_VAL (HUGE_VALF, HUGE_VALL) is returned

    , it is not clear how an infinity is treated. So let's see the source code gnulib/lib/strtod.c. What we want to read is

    else if (c_tolower (*s) == 'i'
             && c_tolower (s[1]) == 'n'
             && c_tolower (s[2]) == 'f')
      {
        s += 3;
        if (c_tolower (*s) == 'i'
            && c_tolower (s[1]) == 'n'
            && c_tolower (s[2]) == 'i'
            && c_tolower (s[3]) == 't'
            && c_tolower (s[4]) == 'y')
          s += 5;
        num = HUGE_VAL;
        errno = saved_errno;
      }
    

    Thus, INF and INFINITY (both case-insensitive) are regarded as HUGE_VAL.

    HUGE_VAL family

    Let's use N1570 as the C standard. HUGE_VAL, HUGE_VALF and HUGE_VALL macros are defined in §7.12-3

    The macro
        HUGE_VAL
    expands to a positive double constant expression, not necessarily representable as a float. The macros
        HUGE_VALF
        HUGE_VALL
    are respectively float and long double analogs of HUGE_VAL.

    HUGE_VAL, HUGE_VALF, and HUGE_VALL can be positive infinities in an implementation that supports infinities.

    and in §7.12.1-5

    If a floating result overflows and default rounding is in effect, then the function returns the value of the macro HUGE_VAL, HUGE_VALF, or HUGE_VALL according to the return type

    Understanding xnanosleep (s)

    Now we understand all essence of xstrtod(). From the explanations above, it is crystal-clear that xnanosleep(s) we've seen first actually means xnanosleep(HUGE_VALL).

    xnanosleep()

    According to the source code gnulib/lib/xnanosleep.c, xnanosleep(s) essentially does this:

    struct timespec ts_sleep = dtotimespec (s);
    nanosleep (&ts_sleep, NULL);
    

    dtotimespec()

    This function converts an argument of type double to an object of type struct timespec. Since it is very simple, let me cite the source code gnulib/lib/dtotimespec.c. All of the comments are added by me.

    struct timespec
    dtotimespec (double sec)
    {
      if (! (TYPE_MINIMUM (time_t) < sec)) //underflow case
        return make_timespec (TYPE_MINIMUM (time_t), 0);
      else if (! (sec < 1.0 + TYPE_MAXIMUM (time_t))) //overflow case
        return make_timespec (TYPE_MAXIMUM (time_t), TIMESPEC_HZ - 1);
      else //normal case (looks complex but does nothing technical)
        {
          time_t s = sec;
          double frac = TIMESPEC_HZ * (sec - s);
          long ns = frac;
          ns += ns < frac;
          s += ns / TIMESPEC_HZ;
          ns %= TIMESPEC_HZ;
    
          if (ns < 0)
            {
              s--;
              ns += TIMESPEC_HZ;
            }
    
          return make_timespec (s, ns);
        }
    }
    

    Since time_t is defined as an integral type (see §7.27.1-3), it is natural we assume the maximum value of type time_t is smaller than HUGE_VAL (of type double), which means we enter the overflow case. (Actually this assumption is not needed since, in all cases, the procedure is essentially the same.)

    make_timespec()

    The last wall we have to climb up is make_timespec(). Very fortunately, it is so simple that citing the source code gnulib/lib/timespec.h is enough.

    _GL_TIMESPEC_INLINE struct timespec
    make_timespec (time_t s, long int ns)
    {
      struct timespec r;
      r.tv_sec = s;
      r.tv_nsec = ns;
      return r;
    }
    
    0 讨论(0)
  • 2020-12-04 08:57

    I recently had a need to do this. I came up with the following function that will allow bash to sleep forever without calling any external program:

    snore()
    {
        local IFS
        [[ -n "${_snore_fd:-}" ]] || { exec {_snore_fd}<> <(:); } 2>/dev/null ||
        {
            # workaround for MacOS and similar systems
            local fifo
            fifo=$(mktemp -u)
            mkfifo -m 700 "$fifo"
            exec {_snore_fd}<>"$fifo"
            rm "$fifo"
        }
        read ${1:+-t "$1"} -u $_snore_fd || :
    }
    

    NOTE: I previously posted a version of this that would open and close the file descriptor each time, but I found that on some systems doing this hundreds of times a second would eventually lock up. Thus the new solution keeps the file descriptor between calls to the function. Bash will clean it up on exit anyway.

    This can be called just like /bin/sleep, and it will sleep for the requested time. Called without parameters, it will hang forever.

    snore 0.1  # sleeps for 0.1 seconds
    snore 10   # sleeps for 10 seconds
    snore      # sleeps forever
    

    There's a writeup with excessive details on my blog here

    0 讨论(0)
  • 2020-12-04 08:57

    Instead of killing the window manager, try running the new one with --replace or -replace if available.

    0 讨论(0)
  • 2020-12-04 09:04

    sleep infinity does exactly what it suggests and works without cat abuse.

    0 讨论(0)
  • 2020-12-04 09:08

    TL;DR: sleep infinity actually sleeps the maximum time allowed, which is finite.

    Wondering why this is not documented anywhere, I bothered to read the sources from GNU coreutils and I found it executes roughly what follows:

    1. Use strtod from C stdlib on the first argument to convert 'infinity' to a double precision value. So, assuming IEEE 754 double precision the 64-bit positive infinity value is stored in the seconds variable.
    2. Invoke xnanosleep(seconds) (found in gnulib), this in turn invokes dtotimespec(seconds) (also in gnulib) to convert from double to struct timespec.
    3. struct timespec is just a pair of numbers: integer part (in seconds) and fractional part (in nanoseconds). Naïvely converting positive infinity to integer would result in undefined behaviour (see §6.3.1.4 from C standard), so instead it truncates to TYPE_MAXIMUM(time_t).
    4. The actual value of TYPE_MAXIMUM(time_t) is not set in the standard (even sizeof(time_t) isn't); so, for the sake of example let's pick x86-64 from a recent Linux kernel.

    This is TIME_T_MAX in the Linux kernel, which is defined (time.h) as:

    (time_t)((1UL << ((sizeof(time_t) << 3) - 1)) - 1)
    

    Note that time_t is __kernel_time_t and time_t is long; the LP64 data model is used, so sizeof(long) is 8 (64 bits).

    Which results in: TIME_T_MAX = 9223372036854775807.

    That is: sleep infinite results in an actual sleep time of 9223372036854775807 seconds (10^11 years). And for 32-bit linux systems (sizeof(long) is 4 (32 bits)): 2147483647 seconds (68 years; see also year 2038 problem).


    Edit: apparently the nanoseconds function called is not directly the syscall, but an OS-dependent wrapper (also defined in gnulib).

    There's an extra step as a result: for some systems where HAVE_BUG_BIG_NANOSLEEP is true the sleep is truncated to 24 days and then called in a loop. This is the case for some (or all?) Linux distros. Note that this wrapper may be not used if a configure-time test succeeds (source).

    In particular, that would be 24 * 24 * 60 * 60 = 2073600 seconds (plus 999999999 nanoseconds); but this is called in a loop in order to respect the specified total sleep time. Therefore the previous conclusions remain valid.


    In conclusion, the resulting sleep time is not infinite but high enough for all practical purposes, even if the resulting actual time lapse is not portable; that depends on the OS and architecture.

    To answer the original question, this is obviously good enough but if for some reason (a very resource-constrained system) you really want to avoid an useless extra countdown timer, I guess the most correct alternative is to use the cat method described in other answers.

    Edit: recent GNU coreutils versions will try to use the pause syscall (if available) instead of looping. The previous argument is no longer valid when targeting these newer versions in Linux (and possibly BSD).


    Portability

    This is an important valid concern:

    • sleep infinity is a GNU coreutils extension not contemplated in POSIX. GNU's implementation also supports a "fancy" syntax for time durations, like sleep 1h 5.2s while POSIX only allows a positive integer (e.g. sleep 0.5 is not allowed).
    • Some compatible implementations: GNU coreutils, FreeBSD (at least from version 8.2?), Busybox (requires to be compiled with options FANCY_SLEEP and FLOAT_DURATION).
    • The strtod behaviour is C and POSIX compatible (i.e. sleep("infinity", 0) is always valid in C99-conformant implementations, see §7.20.1.3).

    0 讨论(0)
提交回复
热议问题