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