Source-Changes-HG archive

[Date Prev][Date Next][Thread Prev][Thread Next][Date Index][Thread Index][Old Index]

[src/trunk]: src/sys/external/bsd/common/linux Fix abuse of TAILQ_CONCAT.



details:   https://anonhg.NetBSD.org/src/rev/00356c2fd29f
branches:  trunk
changeset: 968952:00356c2fd29f
user:      riastradh <riastradh%NetBSD.org@localhost>
date:      Sat Feb 01 22:38:05 2020 +0000

description:
Fix abuse of TAILQ_CONCAT.

Other parts of this expect that the entries will be on wq->wq_queue
or wq->wq_dqueue, so we can't just move a batch of entries onto a
temporary queue.  Instead, use a marker node to delimit when the
batch ends.

XXX pullup

diffstat:

 sys/external/bsd/common/linux/linux_work.c |  29 +++++++++++++++--------------
 1 files changed, 15 insertions(+), 14 deletions(-)

diffs (70 lines):

diff -r 667a01a213a6 -r 00356c2fd29f sys/external/bsd/common/linux/linux_work.c
--- a/sys/external/bsd/common/linux/linux_work.c        Sat Feb 01 21:59:39 2020 +0000
+++ b/sys/external/bsd/common/linux/linux_work.c        Sat Feb 01 22:38:05 2020 +0000
@@ -1,4 +1,4 @@
-/*     $NetBSD: linux_work.c,v 1.44 2019/03/19 08:17:46 ryo Exp $      */
+/*     $NetBSD: linux_work.c,v 1.45 2020/02/01 22:38:05 riastradh Exp $        */
 
 /*-
  * Copyright (c) 2018 The NetBSD Foundation, Inc.
@@ -30,7 +30,7 @@
  */
 
 #include <sys/cdefs.h>
-__KERNEL_RCSID(0, "$NetBSD: linux_work.c,v 1.44 2019/03/19 08:17:46 ryo Exp $");
+__KERNEL_RCSID(0, "$NetBSD: linux_work.c,v 1.45 2020/02/01 22:38:05 riastradh Exp $");
 
 #include <sys/types.h>
 #include <sys/atomic.h>
@@ -348,8 +348,8 @@
 linux_workqueue_thread(void *cookie)
 {
        struct workqueue_struct *const wq = cookie;
-       struct work_head queue, dqueue;
-       struct work_head *const q[2] = { &queue, &dqueue };
+       struct work_head *const q[2] = { &wq->wq_queue, &wq->wq_dqueue };
+       struct work_struct marker, *work;
        unsigned i;
 
        lwp_setspecific(workqueue_key, wq);
@@ -368,22 +368,22 @@
                        continue;
                }
 
-               /* Grab a batch of work off the queue.  */
+               /*
+                * Start a batch of work.  Use a marker to delimit when
+                * the batch ends so we can advance the generation
+                * after the batch.
+                */
                SDT_PROBE1(sdt, linux, work, batch__start,  wq);
-               TAILQ_INIT(&queue);
-               TAILQ_INIT(&dqueue);
-               TAILQ_CONCAT(&queue, &wq->wq_queue, work_entry);
-               TAILQ_CONCAT(&dqueue, &wq->wq_dqueue, work_entry);
-
-               /* Process each work item in the batch.  */
                for (i = 0; i < 2; i++) {
-                       while (!TAILQ_EMPTY(q[i])) {
-                               struct work_struct *work = TAILQ_FIRST(q[i]);
+                       if (TAILQ_EMPTY(q[i]))
+                               continue;
+                       TAILQ_INSERT_TAIL(q[i], &marker, work_entry);
+                       while ((work = TAILQ_FIRST(q[i])) != &marker) {
                                void (*func)(struct work_struct *);
 
                                KASSERT(work_queue(work) == wq);
                                KASSERT(work_claimed(work, wq));
-                               KASSERTMSG((q[i] != &dqueue ||
+                               KASSERTMSG((q[i] != &wq->wq_dqueue ||
                                        container_of(work, struct delayed_work,
                                            work)->dw_state ==
                                        DELAYED_WORK_IDLE),
@@ -407,6 +407,7 @@
                                wq->wq_current_work = NULL;
                                cv_broadcast(&wq->wq_cv);
                        }
+                       TAILQ_REMOVE(q[i], &marker, work_entry);
                }
 
                /* Notify flush that we've completed a batch of work.  */



Home | Main Index | Thread Index | Old Index