diff --git a/src/fd_util.c b/src/fd_util.c index e385be0ce..d8fe166d7 100644 --- a/src/fd_util.c +++ b/src/fd_util.c @@ -174,6 +174,30 @@ pipe_cloexec_nonblock(int fd[2]) #endif } +#ifndef WIN32 + +int +socketpair_cloexec(int domain, int type, int protocol, int sv[2]) +{ + int ret; + +#ifdef SOCK_CLOEXEC + ret = socketpair(domain, type | SOCK_CLOEXEC, protocol, sv); + if (ret >= 0 || errno != EINVAL) + return ret; +#endif + + ret = socketpair(domain, type, protocol, sv); + if (ret >= 0) { + fd_set_cloexec(sv[0], true); + fd_set_cloexec(sv[1], true); + } + + return ret; +} + +#endif + int socket_cloexec_nonblock(int domain, int type, int protocol) { diff --git a/src/fd_util.h b/src/fd_util.h index b704d3d2e..b49a51373 100644 --- a/src/fd_util.h +++ b/src/fd_util.h @@ -65,6 +65,17 @@ pipe_cloexec(int fd[2]); int pipe_cloexec_nonblock(int fd[2]); +#ifndef WIN32 + +/** + * Wrapper for socketpair(), which sets the CLOEXEC flag (atomically + * if supported by the OS). + */ +int +socketpair_cloexec(int domain, int type, int protocol, int sv[2]); + +#endif + /** * Wrapper for socket(), which sets the CLOEXEC and the NONBLOCK flag * (atomically if supported by the OS).