FASYNC + dup + close gives the wrong fd in SIGIO

From: Vegard Nossum
Date: Tue Feb 03 2009 - 17:46:54 EST


Hi,

There seems to be a small deficiency in the O_ASYNC/FASYNC
implementation. When the flag is set for a file descriptor, it does
not "follow" the fd across dup()+close(), so the SIGIO handler will
get the ->si_fd for the fd at the time when O_ASYNC was set. For
example:

pipe([3, 4]) = 0
getpid() = 23982
fcntl(3, F_SETOWN, 23982) = 0
fcntl(3, F_SETSIG, 0x1d) = 0
fcntl(3, F_SETFL, O_RDONLY|O_ASYNC) = 0
dup(3) = 5
close(3) = 0
write(4, "\0"..., 1) = 1
--- SIGIO (I/O possible) @ 0 (0) ---
[...]
SIGIO for fd 3

It seems a bit obscure, I agree, but I thought I'd mention it. Maybe
the fcntl() man page could carry this extra information?

Here's my test case:

#define _GNU_SOURCE

#include <fcntl.h>
#include <signal.h>
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <unistd.h>

static int done = 0;

static void handle_sigio(int signo, siginfo_t *info, void *unused)
{
printf("SIGIO for fd %d\n", info->si_fd);
done = 1;
}

int main(void)
{
struct sigaction act;
int pipe_fd[2];

memset(&act, 0, sizeof(act));
act.sa_sigaction = &handle_sigio;
act.sa_flags = SA_SIGINFO;
sigaction(SIGIO, &act, NULL);

pipe(pipe_fd);
fcntl(pipe_fd[0], F_SETOWN, getpid());
fcntl(pipe_fd[0], F_SETSIG, SIGIO);
fcntl(pipe_fd[0], F_SETFL, O_ASYNC);
dup(pipe_fd[0]);
close(pipe_fd[0]);

write(pipe_fd[1], "", 1);

while (!done)
sleep(1);

return 0;
}



Vegard

--
"The animistic metaphor of the bug that maliciously sneaked in while
the programmer was not looking is intellectually dishonest as it
disguises that the error is the programmer's own creation."
-- E. W. Dijkstra, EWD1036
--
To unsubscribe from this list: send the line "unsubscribe linux-kernel" in
the body of a message to majordomo@xxxxxxxxxxxxxxx
More majordomo info at http://vger.kernel.org/majordomo-info.html
Please read the FAQ at http://www.tux.org/lkml/