Skip to content

Commit 77d2720

Browse files
Ulrich Dreppertorvalds
authored andcommitted
flag parameters: NONBLOCK in socket and socketpair
This patch introduces support for the SOCK_NONBLOCK flag in socket, socketpair, and paccept. To do this the internal function sock_attach_fd gets an additional parameter which it uses to set the appropriate flag for the file descriptor. Given that in modern, scalable programs almost all socket connections are non-blocking and the minimal additional cost for the new functionality I see no reason not to add this code. The following test must be adjusted for architectures other than x86 and x86-64 and in case the syscall numbers changed. ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ #include <fcntl.h> #include <pthread.h> #include <stdio.h> #include <unistd.h> #include <netinet/in.h> #include <sys/socket.h> #include <sys/syscall.h> #ifndef __NR_paccept # ifdef __x86_64__ # define __NR_paccept 288 # elif defined __i386__ # define SYS_PACCEPT 18 # define USE_SOCKETCALL 1 # else # error "need __NR_paccept" # endif #endif #ifdef USE_SOCKETCALL # define paccept(fd, addr, addrlen, mask, flags) \ ({ long args[6] = { \ (long) fd, (long) addr, (long) addrlen, (long) mask, 8, (long) flags }; \ syscall (__NR_socketcall, SYS_PACCEPT, args); }) #else # define paccept(fd, addr, addrlen, mask, flags) \ syscall (__NR_paccept, fd, addr, addrlen, mask, 8, flags) #endif #define PORT 57392 #define SOCK_NONBLOCK O_NONBLOCK static pthread_barrier_t b; static void * tf (void *arg) { pthread_barrier_wait (&b); int s = socket (AF_INET, SOCK_STREAM, 0); struct sockaddr_in sin; sin.sin_family = AF_INET; sin.sin_addr.s_addr = htonl (INADDR_LOOPBACK); sin.sin_port = htons (PORT); connect (s, (const struct sockaddr *) &sin, sizeof (sin)); close (s); pthread_barrier_wait (&b); pthread_barrier_wait (&b); s = socket (AF_INET, SOCK_STREAM, 0); sin.sin_port = htons (PORT); connect (s, (const struct sockaddr *) &sin, sizeof (sin)); close (s); pthread_barrier_wait (&b); return NULL; } int main (void) { int fd; fd = socket (PF_INET, SOCK_STREAM, 0); if (fd == -1) { puts ("socket(0) failed"); return 1; } int fl = fcntl (fd, F_GETFL); if (fl == -1) { puts ("fcntl failed"); return 1; } if (fl & O_NONBLOCK) { puts ("socket(0) set non-blocking mode"); return 1; } close (fd); fd = socket (PF_INET, SOCK_STREAM|SOCK_NONBLOCK, 0); if (fd == -1) { puts ("socket(SOCK_NONBLOCK) failed"); return 1; } fl = fcntl (fd, F_GETFL); if (fl == -1) { puts ("fcntl failed"); return 1; } if ((fl & O_NONBLOCK) == 0) { puts ("socket(SOCK_NONBLOCK) does not set non-blocking mode"); return 1; } close (fd); int fds[2]; if (socketpair (PF_UNIX, SOCK_STREAM, 0, fds) == -1) { puts ("socketpair(0) failed"); return 1; } for (int i = 0; i < 2; ++i) { fl = fcntl (fds[i], F_GETFL); if (fl == -1) { puts ("fcntl failed"); return 1; } if (fl & O_NONBLOCK) { printf ("socketpair(0) set non-blocking mode for fds[%d]\n", i); return 1; } close (fds[i]); } if (socketpair (PF_UNIX, SOCK_STREAM|SOCK_NONBLOCK, 0, fds) == -1) { puts ("socketpair(SOCK_NONBLOCK) failed"); return 1; } for (int i = 0; i < 2; ++i) { fl = fcntl (fds[i], F_GETFL); if (fl == -1) { puts ("fcntl failed"); return 1; } if ((fl & O_NONBLOCK) == 0) { printf ("socketpair(SOCK_NONBLOCK) does not set non-blocking mode for fds[%d]\n", i); return 1; } close (fds[i]); } pthread_barrier_init (&b, NULL, 2); struct sockaddr_in sin; pthread_t th; if (pthread_create (&th, NULL, tf, NULL) != 0) { puts ("pthread_create failed"); return 1; } int s = socket (AF_INET, SOCK_STREAM, 0); int reuse = 1; setsockopt (s, SOL_SOCKET, SO_REUSEADDR, &reuse, sizeof (reuse)); sin.sin_family = AF_INET; sin.sin_addr.s_addr = htonl (INADDR_LOOPBACK); sin.sin_port = htons (PORT); bind (s, (struct sockaddr *) &sin, sizeof (sin)); listen (s, SOMAXCONN); pthread_barrier_wait (&b); int s2 = paccept (s, NULL, 0, NULL, 0); if (s2 < 0) { puts ("paccept(0) failed"); return 1; } fl = fcntl (s2, F_GETFL); if (fl & O_NONBLOCK) { puts ("paccept(0) set non-blocking mode"); return 1; } close (s2); close (s); pthread_barrier_wait (&b); s = socket (AF_INET, SOCK_STREAM, 0); sin.sin_port = htons (PORT); setsockopt (s, SOL_SOCKET, SO_REUSEADDR, &reuse, sizeof (reuse)); bind (s, (struct sockaddr *) &sin, sizeof (sin)); listen (s, SOMAXCONN); pthread_barrier_wait (&b); s2 = paccept (s, NULL, 0, NULL, SOCK_NONBLOCK); if (s2 < 0) { puts ("paccept(SOCK_NONBLOCK) failed"); return 1; } fl = fcntl (s2, F_GETFL); if ((fl & O_NONBLOCK) == 0) { puts ("paccept(SOCK_NONBLOCK) does not set non-blocking mode"); return 1; } close (s2); close (s); pthread_barrier_wait (&b); puts ("OK"); return 0; } ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ Signed-off-by: Ulrich Drepper <drepper@redhat.com> Acked-by: Davide Libenzi <davidel@xmailserver.org> Cc: Michael Kerrisk <mtk.manpages@googlemail.com> Cc: "David S. Miller" <davem@davemloft.net> Signed-off-by: Andrew Morton <akpm@linux-foundation.org> Signed-off-by: Linus Torvalds <torvalds@linux-foundation.org>
1 parent 99829b8 commit 77d2720

File tree

2 files changed

+11
-11
lines changed

2 files changed

+11
-11
lines changed

include/linux/net.h

Lines changed: 1 addition & 1 deletion
Original file line numberDiff line numberDiff line change
@@ -20,7 +20,7 @@
2020

2121
#include <linux/wait.h>
2222
#include <linux/socket.h>
23-
#include <linux/fcntl.h> /* For O_CLOEXEC */
23+
#include <linux/fcntl.h> /* For O_CLOEXEC and O_NONBLOCK */
2424
#include <asm/socket.h>
2525

2626
struct poll_table_struct;

net/socket.c

Lines changed: 10 additions & 10 deletions
Original file line numberDiff line numberDiff line change
@@ -369,7 +369,7 @@ static int sock_alloc_fd(struct file **filep, int flags)
369369
return fd;
370370
}
371371

372-
static int sock_attach_fd(struct socket *sock, struct file *file)
372+
static int sock_attach_fd(struct socket *sock, struct file *file, int flags)
373373
{
374374
struct dentry *dentry;
375375
struct qstr name = { .name = "" };
@@ -391,7 +391,7 @@ static int sock_attach_fd(struct socket *sock, struct file *file)
391391
init_file(file, sock_mnt, dentry, FMODE_READ | FMODE_WRITE,
392392
&socket_file_ops);
393393
SOCK_INODE(sock)->i_fop = &socket_file_ops;
394-
file->f_flags = O_RDWR;
394+
file->f_flags = O_RDWR | (flags & O_NONBLOCK);
395395
file->f_pos = 0;
396396
file->private_data = sock;
397397

@@ -404,7 +404,7 @@ int sock_map_fd(struct socket *sock, int flags)
404404
int fd = sock_alloc_fd(&newfile, flags);
405405

406406
if (likely(fd >= 0)) {
407-
int err = sock_attach_fd(sock, newfile);
407+
int err = sock_attach_fd(sock, newfile, flags);
408408

409409
if (unlikely(err < 0)) {
410410
put_filp(newfile);
@@ -1223,7 +1223,7 @@ asmlinkage long sys_socket(int family, int type, int protocol)
12231223
int flags;
12241224

12251225
flags = type & ~SOCK_TYPE_MASK;
1226-
if (flags & ~SOCK_CLOEXEC)
1226+
if (flags & ~(SOCK_CLOEXEC | SOCK_NONBLOCK))
12271227
return -EINVAL;
12281228
type &= SOCK_TYPE_MASK;
12291229

@@ -1234,7 +1234,7 @@ asmlinkage long sys_socket(int family, int type, int protocol)
12341234
if (retval < 0)
12351235
goto out;
12361236

1237-
retval = sock_map_fd(sock, flags & O_CLOEXEC);
1237+
retval = sock_map_fd(sock, flags & (O_CLOEXEC | O_NONBLOCK));
12381238
if (retval < 0)
12391239
goto out_release;
12401240

@@ -1260,7 +1260,7 @@ asmlinkage long sys_socketpair(int family, int type, int protocol,
12601260
int flags;
12611261

12621262
flags = type & ~SOCK_TYPE_MASK;
1263-
if (flags & ~SOCK_CLOEXEC)
1263+
if (flags & ~(SOCK_CLOEXEC | SOCK_NONBLOCK))
12641264
return -EINVAL;
12651265
type &= SOCK_TYPE_MASK;
12661266

@@ -1298,12 +1298,12 @@ asmlinkage long sys_socketpair(int family, int type, int protocol,
12981298
goto out_release_both;
12991299
}
13001300

1301-
err = sock_attach_fd(sock1, newfile1);
1301+
err = sock_attach_fd(sock1, newfile1, flags & O_NONBLOCK);
13021302
if (unlikely(err < 0)) {
13031303
goto out_fd2;
13041304
}
13051305

1306-
err = sock_attach_fd(sock2, newfile2);
1306+
err = sock_attach_fd(sock2, newfile2, flags & O_NONBLOCK);
13071307
if (unlikely(err < 0)) {
13081308
fput(newfile1);
13091309
goto out_fd1;
@@ -1429,7 +1429,7 @@ long do_accept(int fd, struct sockaddr __user *upeer_sockaddr,
14291429
int err, len, newfd, fput_needed;
14301430
struct sockaddr_storage address;
14311431

1432-
if (flags & ~SOCK_CLOEXEC)
1432+
if (flags & ~(SOCK_CLOEXEC | SOCK_NONBLOCK))
14331433
return -EINVAL;
14341434

14351435
if (SOCK_NONBLOCK != O_NONBLOCK && (flags & SOCK_NONBLOCK))
@@ -1459,7 +1459,7 @@ long do_accept(int fd, struct sockaddr __user *upeer_sockaddr,
14591459
goto out_put;
14601460
}
14611461

1462-
err = sock_attach_fd(newsock, newfile);
1462+
err = sock_attach_fd(newsock, newfile, flags & O_NONBLOCK);
14631463
if (err < 0)
14641464
goto out_fd_simple;
14651465

0 commit comments

Comments
 (0)