xref: /llvm-project/libc/src/sys/select/linux/select.cpp (revision 5ff3ff33ff930e4ec49da7910612d8a41eb068cb)
1be4e4257SSiva Chandra Reddy //===-- Linux implementation of select ------------------------------------===//
2be4e4257SSiva Chandra Reddy //
3be4e4257SSiva Chandra Reddy // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
4be4e4257SSiva Chandra Reddy // See https://llvm.org/LICENSE.txt for license information.
5be4e4257SSiva Chandra Reddy // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
6be4e4257SSiva Chandra Reddy //
7be4e4257SSiva Chandra Reddy //===----------------------------------------------------------------------===//
8be4e4257SSiva Chandra Reddy 
9be4e4257SSiva Chandra Reddy #include "src/sys/select/select.h"
10be4e4257SSiva Chandra Reddy 
11f626a350SNick Desaulniers #include "hdr/types/sigset_t.h"
12f626a350SNick Desaulniers #include "hdr/types/struct_timespec.h"
13be4e4257SSiva Chandra Reddy #include "src/__support/CPP/limits.h"
14be4e4257SSiva Chandra Reddy #include "src/__support/OSUtil/syscall.h" // For internal syscall function.
15be4e4257SSiva Chandra Reddy #include "src/__support/common.h"
16*5ff3ff33SPetr Hosek #include "src/__support/macros/config.h"
17204587a3SSiva Chandra Reddy #include "src/errno/libc_errno.h"
18f626a350SNick Desaulniers 
19be4e4257SSiva Chandra Reddy #include <stddef.h>      // For size_t
20be4e4257SSiva Chandra Reddy #include <sys/syscall.h> // For syscall numbers.
21be4e4257SSiva Chandra Reddy 
22*5ff3ff33SPetr Hosek namespace LIBC_NAMESPACE_DECL {
23be4e4257SSiva Chandra Reddy 
24be4e4257SSiva Chandra Reddy struct pselect6_sigset_t {
25be4e4257SSiva Chandra Reddy   sigset_t *ss;
26be4e4257SSiva Chandra Reddy   size_t ss_len;
27be4e4257SSiva Chandra Reddy };
28be4e4257SSiva Chandra Reddy 
29be4e4257SSiva Chandra Reddy LLVM_LIBC_FUNCTION(int, select,
30be4e4257SSiva Chandra Reddy                    (int nfds, fd_set *__restrict read_set,
31be4e4257SSiva Chandra Reddy                     fd_set *__restrict write_set, fd_set *__restrict error_set,
32be4e4257SSiva Chandra Reddy                     struct timeval *__restrict timeout)) {
33be4e4257SSiva Chandra Reddy   // Linux has a SYS_select syscall but it is not available on all
34be4e4257SSiva Chandra Reddy   // architectures. So, we use the SYS_pselect6 syscall which is more
35be4e4257SSiva Chandra Reddy   // widely available. However, SYS_pselect6 takes a struct timespec argument
36be4e4257SSiva Chandra Reddy   // instead of a struct timeval argument. Also, it takes an additional
37be4e4257SSiva Chandra Reddy   // argument which is a pointer to an object of a type defined above as
38be4e4257SSiva Chandra Reddy   // "pselect6_sigset_t".
39be4e4257SSiva Chandra Reddy   struct timespec ts {
40be4e4257SSiva Chandra Reddy     0, 0
41be4e4257SSiva Chandra Reddy   };
42be4e4257SSiva Chandra Reddy   if (timeout != nullptr) {
43be4e4257SSiva Chandra Reddy     // In general, if the tv_sec and tv_usec in |timeout| are correctly set,
44be4e4257SSiva Chandra Reddy     // then converting tv_usec to nanoseconds will not be a problem. However,
45be4e4257SSiva Chandra Reddy     // if tv_usec in |timeout| is more than a second, it can lead to overflows.
46be4e4257SSiva Chandra Reddy     // So, we detect such cases and adjust.
47be4e4257SSiva Chandra Reddy     constexpr time_t TIME_MAX = cpp::numeric_limits<time_t>::max();
48be4e4257SSiva Chandra Reddy     if ((TIME_MAX - timeout->tv_sec) < (timeout->tv_usec / 1000000)) {
49be4e4257SSiva Chandra Reddy       ts.tv_sec = TIME_MAX;
50be4e4257SSiva Chandra Reddy       ts.tv_nsec = 999999999;
51be4e4257SSiva Chandra Reddy     } else {
52be4e4257SSiva Chandra Reddy       ts.tv_sec = timeout->tv_sec + timeout->tv_usec / 1000000;
53be4e4257SSiva Chandra Reddy       ts.tv_nsec = timeout->tv_usec * 1000;
54be4e4257SSiva Chandra Reddy     }
55be4e4257SSiva Chandra Reddy   }
56be4e4257SSiva Chandra Reddy   pselect6_sigset_t pss{nullptr, sizeof(sigset_t)};
57c9783d2bSMikhail R. Gadelha #if SYS_pselect6
58b6bc9d72SGuillaume Chatelet   int ret = LIBC_NAMESPACE::syscall_impl<int>(SYS_pselect6, nfds, read_set,
59f0a3954eSMichael Jones                                               write_set, error_set, &ts, &pss);
60c9783d2bSMikhail R. Gadelha #elif defined(SYS_pselect6_time64)
61b6bc9d72SGuillaume Chatelet   int ret = LIBC_NAMESPACE::syscall_impl<int>(
62b6bc9d72SGuillaume Chatelet       SYS_pselect6_time64, nfds, read_set, write_set, error_set, &ts, &pss);
63c9783d2bSMikhail R. Gadelha #else
64c9783d2bSMikhail R. Gadelha #error "SYS_pselect6 and SYS_pselect6_time64 syscalls not available."
65c9783d2bSMikhail R. Gadelha #endif
66be4e4257SSiva Chandra Reddy   if (ret < 0) {
67204587a3SSiva Chandra Reddy     libc_errno = -ret;
68be4e4257SSiva Chandra Reddy     return -1;
69be4e4257SSiva Chandra Reddy   }
70be4e4257SSiva Chandra Reddy   return ret;
71be4e4257SSiva Chandra Reddy }
72be4e4257SSiva Chandra Reddy 
73*5ff3ff33SPetr Hosek } // namespace LIBC_NAMESPACE_DECL
74