Blame sysdeps/posix/sleep.c

Packit 6c4009
/* Sleep for a given number of seconds.  POSIX.1 version.
Packit 6c4009
   Copyright (C) 1991-2018 Free Software Foundation, Inc.
Packit 6c4009
   This file is part of the GNU C Library.
Packit 6c4009
Packit 6c4009
   The GNU C Library is free software; you can redistribute it and/or
Packit 6c4009
   modify it under the terms of the GNU Lesser General Public
Packit 6c4009
   License as published by the Free Software Foundation; either
Packit 6c4009
   version 2.1 of the License, or (at your option) any later version.
Packit 6c4009
Packit 6c4009
   The GNU C Library is distributed in the hope that it will be useful,
Packit 6c4009
   but WITHOUT ANY WARRANTY; without even the implied warranty of
Packit 6c4009
   MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
Packit 6c4009
   Lesser General Public License for more details.
Packit 6c4009
Packit 6c4009
   You should have received a copy of the GNU Lesser General Public
Packit 6c4009
   License along with the GNU C Library; if not, see
Packit 6c4009
   <http://www.gnu.org/licenses/>.  */
Packit 6c4009
Packit 6c4009
#include <time.h>
Packit 6c4009
#include <unistd.h>
Packit 6c4009
#include <errno.h>
Packit 6c4009
#include <sys/param.h>
Packit 6c4009
Packit 6c4009
Packit 6c4009
/* Make the process sleep for SECONDS seconds, or until a signal arrives
Packit 6c4009
   and is not ignored.  The function returns the number of seconds less
Packit 6c4009
   than SECONDS which it actually slept (zero if it slept the full time).
Packit 6c4009
   If a signal handler does a `longjmp' or modifies the handling of the
Packit 6c4009
   SIGALRM signal while inside `sleep' call, the handling of the SIGALRM
Packit 6c4009
   signal afterwards is undefined.  There is no return value to indicate
Packit 6c4009
   error, but if `sleep' returns SECONDS, it probably didn't work.  */
Packit 6c4009
unsigned int
Packit 6c4009
__sleep (unsigned int seconds)
Packit 6c4009
{
Packit 6c4009
  int save_errno = errno;
Packit 6c4009
Packit 6c4009
  const unsigned int max
Packit 6c4009
    = (unsigned int) (((unsigned long int) (~((time_t) 0))) >> 1);
Packit 6c4009
  struct timespec ts = { 0, 0 };
Packit 6c4009
  do
Packit 6c4009
    {
Packit 6c4009
      if (sizeof (ts.tv_sec) <= sizeof (seconds))
Packit 6c4009
        {
Packit 6c4009
          /* Since SECONDS is unsigned assigning the value to .tv_sec can
Packit 6c4009
             overflow it.  In this case we have to wait in steps.  */
Packit 6c4009
          ts.tv_sec += MIN (seconds, max);
Packit 6c4009
          seconds -= (unsigned int) ts.tv_sec;
Packit 6c4009
        }
Packit 6c4009
      else
Packit 6c4009
        {
Packit 6c4009
          ts.tv_sec = (time_t) seconds;
Packit 6c4009
          seconds = 0;
Packit 6c4009
        }
Packit 6c4009
Packit 6c4009
      if (__nanosleep (&ts, &ts) < 0)
Packit 6c4009
        /* We were interrupted.
Packit 6c4009
           Return the number of (whole) seconds we have not yet slept.  */
Packit 6c4009
        return seconds + ts.tv_sec;
Packit 6c4009
    }
Packit 6c4009
  while (seconds > 0);
Packit 6c4009
Packit 6c4009
  __set_errno (save_errno);
Packit 6c4009
Packit 6c4009
  return 0;
Packit 6c4009
}
Packit 6c4009
weak_alias (__sleep, sleep)