aboutsummaryrefslogtreecommitdiffstats
path: root/contrib/libs/cxxsupp/libcxxmsvc/include/__support/ibm/nanosleep.h
blob: 9c6b976c0003039dbb7e7795e7fbec422521e81a (plain) (blame)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
// -*- C++ -*-
//===----------------------------------------------------------------------===//
//
// Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
// See https://llvm.org/LICENSE.txt for license information.
// SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
//
//===----------------------------------------------------------------------===//

#ifndef _LIBCPP_SUPPORT_IBM_NANOSLEEP_H
#define _LIBCPP_SUPPORT_IBM_NANOSLEEP_H

#include <unistd.h>

inline int nanosleep(const struct timespec* __req, struct timespec* __rem) {
  // The nanosleep() function is not available on z/OS. Therefore, we will call
  // sleep() to sleep for whole seconds and usleep() to sleep for any remaining
  // fraction of a second. Any remaining nanoseconds will round up to the next
  // microsecond.
  if (__req->tv_sec < 0 || __req->tv_nsec < 0 || __req->tv_nsec > 999999999) {
    errno = EINVAL;
    return -1;
  }
  long __micro_sec = (__req->tv_nsec + 999) / 1000;
  time_t __sec = __req->tv_sec;
  if (__micro_sec > 999999) {
    ++__sec;
    __micro_sec -= 1000000;
  }
  __sec = static_cast<time_t>(sleep(static_cast<unsigned int>(__sec)));
  if (__sec) {
    if (__rem) {
      // Updating the remaining time to sleep in case of unsuccessful call to sleep().
      __rem->tv_sec = __sec;
      __rem->tv_nsec = __micro_sec * 1000;
    }
    errno = EINTR;
    return -1;
  }
  if (__micro_sec) {
    int __rt = usleep(static_cast<unsigned int>(__micro_sec));
    if (__rt != 0 && __rem) {
      // The usleep() does not provide the amount of remaining time upon its failure,
      // so the time slept will be ignored.
      __rem->tv_sec = 0;
      __rem->tv_nsec = __micro_sec * 1000;
      // The errno is already set.
      return -1;
    }
    return __rt;
  }
  return 0;
}

#endif // _LIBCPP_SUPPORT_IBM_NANOSLEEP_H