public inbox for linux-kernel@vger.kernel.org
 help / color / mirror / Atom feed
* [patch] Fix epoll delayed initialization bug ...
@ 2005-09-16 23:35 Davide Libenzi
  2005-09-16 23:50 ` Andrew Morton
  2005-09-16 23:59 ` Roland Dreier
  0 siblings, 2 replies; 7+ messages in thread
From: Davide Libenzi @ 2005-09-16 23:35 UTC (permalink / raw)
  To: Linux Kernel Mailing List; +Cc: Andrew Morton


Al found a potential problem in epoll_create(), where the 
file->private_data member was set after fd_install(). This is obviously 
wrong since another thread might do a close() on that fd# before we set 
the file->private_data member. This goes over 2.6.13 and passes a few 
basic tests I've done here.


Signed-off-by: Davide Libenzi <davidel@xmailserver.org>


- Davide



diff -Nru linux-2.6.13.vanilla/fs/eventpoll.c linux-2.6.13/fs/eventpoll.c
--- linux-2.6.13.vanilla/fs/eventpoll.c	2005-09-16 15:20:46.000000000 -0700
+++ linux-2.6.13/fs/eventpoll.c	2005-09-16 15:21:08.000000000 -0700
@@ -231,8 +231,9 @@

  static void ep_poll_safewake_init(struct poll_safewake *psw);
  static void ep_poll_safewake(struct poll_safewake *psw, wait_queue_head_t *wq);
-static int ep_getfd(int *efd, struct inode **einode, struct file **efile);
-static int ep_file_init(struct file *file);
+static int ep_getfd(int *efd, struct inode **einode, struct file **efile,
+		    struct eventpoll *ep);
+static int ep_alloc(struct eventpoll **pep);
  static void ep_free(struct eventpoll *ep);
  static struct epitem *ep_find(struct eventpoll *ep, struct file *file, int fd);
  static void ep_use_epitem(struct epitem *epi);
@@ -501,38 +502,37 @@
  asmlinkage long sys_epoll_create(int size)
  {
  	int error, fd;
+	struct eventpoll *ep;
  	struct inode *inode;
  	struct file *file;

  	DNPRINTK(3, (KERN_INFO "[%p] eventpoll: sys_epoll_create(%d)\n",
  		     current, size));

-	/* Sanity check on the size parameter */
+	/*
+	 * Sanity check on the size parameter, and create the internal data
+	 * structure ( "struct eventpoll" ).
+	 */
  	error = -EINVAL;
-	if (size <= 0)
+	if (size <= 0 || (error = ep_alloc(&ep)))
  		goto eexit_1;

  	/*
  	 * Creates all the items needed to setup an eventpoll file. That is,
  	 * a file structure, and inode and a free file descriptor.
  	 */
-	error = ep_getfd(&fd, &inode, &file);
-	if (error)
-		goto eexit_1;
-
-	/* Setup the file internal data structure ( "struct eventpoll" ) */
-	error = ep_file_init(file);
+	error = ep_getfd(&fd, &inode, &file, ep);
  	if (error)
  		goto eexit_2;

-
  	DNPRINTK(3, (KERN_INFO "[%p] eventpoll: sys_epoll_create(%d) = %d\n",
  		     current, size, fd));

  	return fd;

  eexit_2:
-	sys_close(fd);
+	ep_free(ep);
+	kfree(ep);
  eexit_1:
  	DNPRINTK(3, (KERN_INFO "[%p] eventpoll: sys_epoll_create(%d) = %d\n",
  		     current, size, error));
@@ -706,7 +706,8 @@
  /*
   * Creates the file descriptor to be used by the epoll interface.
   */
-static int ep_getfd(int *efd, struct inode **einode, struct file **efile)
+static int ep_getfd(int *efd, struct inode **einode, struct file **efile,
+		    struct eventpoll *ep)
  {
  	struct qstr this;
  	char name[32];
@@ -756,7 +757,7 @@
  	file->f_op = &eventpoll_fops;
  	file->f_mode = FMODE_READ;
  	file->f_version = 0;
-	file->private_data = NULL;
+	file->private_data = ep;

  	/* Install the new setup file into the allocated fd. */
  	fd_install(fd, file);
@@ -777,7 +778,7 @@
  }


-static int ep_file_init(struct file *file)
+static int ep_alloc(struct eventpoll **pep)
  {
  	struct eventpoll *ep;

@@ -792,9 +793,9 @@
  	INIT_LIST_HEAD(&ep->rdllist);
  	ep->rbr = RB_ROOT;

-	file->private_data = ep;
+	*pep = ep;

-	DNPRINTK(3, (KERN_INFO "[%p] eventpoll: ep_file_init() ep=%p\n",
+	DNPRINTK(3, (KERN_INFO "[%p] eventpoll: ep_alloc() ep=%p\n",
  		     current, ep));
  	return 0;
  }

^ permalink raw reply	[flat|nested] 7+ messages in thread

end of thread, other threads:[~2005-09-17  5:47 UTC | newest]

Thread overview: 7+ messages (download: mbox.gz follow: Atom feed
-- links below jump to the message on this page --
2005-09-16 23:35 [patch] Fix epoll delayed initialization bug Davide Libenzi
2005-09-16 23:50 ` Andrew Morton
2005-09-16 23:56   ` Davide Libenzi
2005-09-17  0:05     ` Andrew Morton
2005-09-17  2:37   ` Paul Jackson
2005-09-16 23:59 ` Roland Dreier
2005-09-17  5:47   ` Al Viro

This is a public inbox, see mirroring instructions
for how to clone and mirror all data and code used for this inbox