linux: improve kernel feature detection
Do not check for minimum kernel and glibc versions, just check that the kernel headers export the syscall number and invoke the syscall directly. Effectively bypasses glibc.
This commit is contained in:
parent
a4c8ffa3c2
commit
6b3075cd73
@ -735,8 +735,8 @@ int uv__accept(int sockfd, struct sockaddr* saddr, socklen_t slen) {
|
|||||||
assert(sockfd >= 0);
|
assert(sockfd >= 0);
|
||||||
|
|
||||||
while (1) {
|
while (1) {
|
||||||
#if HAVE_ACCEPT4
|
#if HAVE_SYS_ACCEPT4
|
||||||
peerfd = accept4(sockfd, saddr, &slen, SOCK_NONBLOCK | SOCK_CLOEXEC);
|
peerfd = sys_accept4(sockfd, saddr, &slen, SOCK_NONBLOCK | SOCK_CLOEXEC);
|
||||||
|
|
||||||
if (peerfd != -1)
|
if (peerfd != -1)
|
||||||
break;
|
break;
|
||||||
|
|||||||
@ -32,30 +32,69 @@
|
|||||||
#endif
|
#endif
|
||||||
|
|
||||||
#undef HAVE_FUTIMES
|
#undef HAVE_FUTIMES
|
||||||
#undef HAVE_PIPE2
|
|
||||||
#undef HAVE_ACCEPT4
|
|
||||||
#undef HAVE_KQUEUE
|
#undef HAVE_KQUEUE
|
||||||
#undef HAVE_PORTS_FS
|
#undef HAVE_PORTS_FS
|
||||||
|
|
||||||
#if defined(__linux__)
|
#if defined(__linux__)
|
||||||
|
|
||||||
#include <linux/version.h>
|
# undef HAVE_SYS_UTIMESAT
|
||||||
#include <features.h>
|
# undef HAVE_SYS_PIPE2
|
||||||
|
# undef HAVE_SYS_ACCEPT4
|
||||||
|
|
||||||
/* futimes() requires linux >= 2.6.22 and glib >= 2.6 */
|
# undef _GNU_SOURCE
|
||||||
#if LINUX_VERSION_CODE >= 0x20616 && __GLIBC_PREREQ(2, 6)
|
# define _GNU_SOURCE
|
||||||
#define HAVE_FUTIMES 1
|
|
||||||
#endif
|
|
||||||
|
|
||||||
/* pipe2() requires linux >= 2.6.27 and glibc >= 2.9 */
|
# include <linux/version.h>
|
||||||
#if LINUX_VERSION_CODE >= 0x2061B && __GLIBC_PREREQ(2, 9)
|
# include <sys/syscall.h>
|
||||||
#define HAVE_PIPE2 1
|
# include <features.h>
|
||||||
#endif
|
# include <unistd.h>
|
||||||
|
|
||||||
/* accept4() requires linux >= 2.6.28 and glib >= 2.10 */
|
# if __NR_utimensat
|
||||||
#if LINUX_VERSION_CODE >= 0x2061C && __GLIBC_PREREQ(2, 10)
|
# define HAVE_SYS_UTIMESAT 1
|
||||||
#define HAVE_ACCEPT4 1
|
# endif
|
||||||
#endif
|
# if __NR_pipe2
|
||||||
|
# define HAVE_SYS_PIPE2 1
|
||||||
|
# endif
|
||||||
|
# if __NR_accept4
|
||||||
|
# define HAVE_SYS_ACCEPT4 1
|
||||||
|
# endif
|
||||||
|
|
||||||
|
# if HAVE_SYS_UTIMESAT
|
||||||
|
inline static int sys_utimesat(int dirfd,
|
||||||
|
const char* path,
|
||||||
|
const struct timespec times[2],
|
||||||
|
int flags)
|
||||||
|
{
|
||||||
|
return syscall(__NR_utimensat, dirfd, path, times, flags);
|
||||||
|
}
|
||||||
|
inline static int sys_futimes(int fd, const struct timeval times[2])
|
||||||
|
{
|
||||||
|
struct timespec ts[2];
|
||||||
|
ts[0].tv_sec = times[0].tv_sec, ts[0].tv_nsec = times[0].tv_usec * 1000;
|
||||||
|
ts[1].tv_sec = times[1].tv_sec, ts[1].tv_nsec = times[1].tv_usec * 1000;
|
||||||
|
return sys_utimesat(fd, NULL, ts, 0);
|
||||||
|
}
|
||||||
|
# undef HAVE_FUTIMES
|
||||||
|
# define HAVE_FUTIMES 1
|
||||||
|
# define futimes(fd, times) sys_futimes(fd, times)
|
||||||
|
# endif /* HAVE_SYS_FUTIMESAT */
|
||||||
|
|
||||||
|
# if HAVE_SYS_PIPE2
|
||||||
|
inline static int sys_pipe2(int pipefd[2], int flags)
|
||||||
|
{
|
||||||
|
return syscall(__NR_pipe2, pipefd, flags);
|
||||||
|
}
|
||||||
|
# endif /* HAVE_SYS_PIPE2 */
|
||||||
|
|
||||||
|
# if HAVE_SYS_ACCEPT4
|
||||||
|
inline static int sys_accept4(int fd,
|
||||||
|
struct sockaddr* addr,
|
||||||
|
socklen_t* addrlen,
|
||||||
|
int flags)
|
||||||
|
{
|
||||||
|
return syscall(__NR_accept4, fd, addr, addrlen, flags);
|
||||||
|
}
|
||||||
|
# endif /* HAVE_SYS_ACCEPT4 */
|
||||||
|
|
||||||
#endif /* __linux__ */
|
#endif /* __linux__ */
|
||||||
|
|
||||||
|
|||||||
@ -104,7 +104,7 @@ static int uv__make_socketpair(int fds[2], int flags) {
|
|||||||
|
|
||||||
|
|
||||||
static int uv__make_pipe(int fds[2], int flags) {
|
static int uv__make_pipe(int fds[2], int flags) {
|
||||||
#if HAVE_PIPE2
|
#if HAVE_SYS_PIPE2
|
||||||
int fl;
|
int fl;
|
||||||
|
|
||||||
fl = O_CLOEXEC;
|
fl = O_CLOEXEC;
|
||||||
@ -112,7 +112,7 @@ static int uv__make_pipe(int fds[2], int flags) {
|
|||||||
if (flags & UV__F_NONBLOCK)
|
if (flags & UV__F_NONBLOCK)
|
||||||
fl |= O_NONBLOCK;
|
fl |= O_NONBLOCK;
|
||||||
|
|
||||||
if (pipe2(fds, fl) == 0)
|
if (sys_pipe2(fds, fl) == 0)
|
||||||
return 0;
|
return 0;
|
||||||
|
|
||||||
if (errno != ENOSYS)
|
if (errno != ENOSYS)
|
||||||
|
|||||||
Loading…
Reference in New Issue
Block a user