]> git.ipfire.org Git - thirdparty/kernel/stable-queue.git/commitdiff
4.9-stable patches
authorGreg Kroah-Hartman <gregkh@linuxfoundation.org>
Mon, 4 Sep 2017 16:14:31 +0000 (18:14 +0200)
committerGreg Kroah-Hartman <gregkh@linuxfoundation.org>
Mon, 4 Sep 2017 16:14:31 +0000 (18:14 +0200)
added patches:
epoll-fix-race-between-ep_poll_callback-pollfree-and-ep_free-ep_remove.patch

queue-4.9/epoll-fix-race-between-ep_poll_callback-pollfree-and-ep_free-ep_remove.patch [new file with mode: 0644]
queue-4.9/series

diff --git a/queue-4.9/epoll-fix-race-between-ep_poll_callback-pollfree-and-ep_free-ep_remove.patch b/queue-4.9/epoll-fix-race-between-ep_poll_callback-pollfree-and-ep_free-ep_remove.patch
new file mode 100644 (file)
index 0000000..5944bf1
--- /dev/null
@@ -0,0 +1,116 @@
+From 138e4ad67afd5c6c318b056b4d17c17f2c0ca5c0 Mon Sep 17 00:00:00 2001
+From: Oleg Nesterov <oleg@redhat.com>
+Date: Fri, 1 Sep 2017 18:55:33 +0200
+Subject: epoll: fix race between ep_poll_callback(POLLFREE) and ep_free()/ep_remove()
+MIME-Version: 1.0
+Content-Type: text/plain; charset=UTF-8
+Content-Transfer-Encoding: 8bit
+
+From: Oleg Nesterov <oleg@redhat.com>
+
+commit 138e4ad67afd5c6c318b056b4d17c17f2c0ca5c0 upstream.
+
+The race was introduced by me in commit 971316f0503a ("epoll:
+ep_unregister_pollwait() can use the freed pwq->whead").  I did not
+realize that nothing can protect eventpoll after ep_poll_callback() sets
+->whead = NULL, only whead->lock can save us from the race with
+ep_free() or ep_remove().
+
+Move ->whead = NULL to the end of ep_poll_callback() and add the
+necessary barriers.
+
+TODO: cleanup the ewake/EPOLLEXCLUSIVE logic, it was confusing even
+before this patch.
+
+Hopefully this explains use-after-free reported by syzcaller:
+
+       BUG: KASAN: use-after-free in debug_spin_lock_before
+       ...
+        _raw_spin_lock_irqsave+0x4a/0x60 kernel/locking/spinlock.c:159
+        ep_poll_callback+0x29f/0xff0 fs/eventpoll.c:1148
+
+this is spin_lock(eventpoll->lock),
+
+       ...
+       Freed by task 17774:
+       ...
+        kfree+0xe8/0x2c0 mm/slub.c:3883
+        ep_free+0x22c/0x2a0 fs/eventpoll.c:865
+
+Fixes: 971316f0503a ("epoll: ep_unregister_pollwait() can use the freed pwq->whead")
+Reported-by: 范龙飞 <long7573@126.com>
+Signed-off-by: Oleg Nesterov <oleg@redhat.com>
+Signed-off-by: Linus Torvalds <torvalds@linux-foundation.org>
+Signed-off-by: Greg Kroah-Hartman <gregkh@linuxfoundation.org>
+
+
+---
+ fs/eventpoll.c |   42 ++++++++++++++++++++++++++----------------
+ 1 file changed, 26 insertions(+), 16 deletions(-)
+
+--- a/fs/eventpoll.c
++++ b/fs/eventpoll.c
+@@ -523,8 +523,13 @@ static void ep_remove_wait_queue(struct
+       wait_queue_head_t *whead;
+       rcu_read_lock();
+-      /* If it is cleared by POLLFREE, it should be rcu-safe */
+-      whead = rcu_dereference(pwq->whead);
++      /*
++       * If it is cleared by POLLFREE, it should be rcu-safe.
++       * If we read NULL we need a barrier paired with
++       * smp_store_release() in ep_poll_callback(), otherwise
++       * we rely on whead->lock.
++       */
++      whead = smp_load_acquire(&pwq->whead);
+       if (whead)
+               remove_wait_queue(whead, &pwq->wait);
+       rcu_read_unlock();
+@@ -1009,17 +1014,6 @@ static int ep_poll_callback(wait_queue_t
+       struct eventpoll *ep = epi->ep;
+       int ewake = 0;
+-      if ((unsigned long)key & POLLFREE) {
+-              ep_pwq_from_wait(wait)->whead = NULL;
+-              /*
+-               * whead = NULL above can race with ep_remove_wait_queue()
+-               * which can do another remove_wait_queue() after us, so we
+-               * can't use __remove_wait_queue(). whead->lock is held by
+-               * the caller.
+-               */
+-              list_del_init(&wait->task_list);
+-      }
+-
+       spin_lock_irqsave(&ep->lock, flags);
+       /*
+@@ -1101,10 +1095,26 @@ out_unlock:
+       if (pwake)
+               ep_poll_safewake(&ep->poll_wait);
+-      if (epi->event.events & EPOLLEXCLUSIVE)
+-              return ewake;
++      if (!(epi->event.events & EPOLLEXCLUSIVE))
++              ewake = 1;
++
++      if ((unsigned long)key & POLLFREE) {
++              /*
++               * If we race with ep_remove_wait_queue() it can miss
++               * ->whead = NULL and do another remove_wait_queue() after
++               * us, so we can't use __remove_wait_queue().
++               */
++              list_del_init(&wait->task_list);
++              /*
++               * ->whead != NULL protects us from the race with ep_free()
++               * or ep_remove(), ep_remove_wait_queue() takes whead->lock
++               * held by the caller. Once we nullify it, nothing protects
++               * ep/epi or even wait.
++               */
++              smp_store_release(&ep_pwq_from_wait(wait)->whead, NULL);
++      }
+-      return 1;
++      return ewake;
+ }
+ /*
index d9a11c4bf01ad8bc5bb144416ef4e08a7833ad7d..806debd39c1bded4dd58fbf859ab38baab403bc4 100644 (file)
@@ -15,3 +15,4 @@ lib-mpi-kunmap-after-finishing-accessing-buffer.patch
 xfrm-policy-check-policy-direction-value.patch
 drm-ttm-fix-accounting-error-when-fail-to-get-pages-for-pool.patch
 kvm-arm-arm64-force-reading-uncached-stage2-pgd.patch
+epoll-fix-race-between-ep_poll_callback-pollfree-and-ep_free-ep_remove.patch