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 Handle a bazillion additional ...



details:   https://anonhg.NetBSD.org/src/rev/6461012b27ae
branches:  trunk
changeset: 364827:6461012b27ae
user:      riastradh <riastradh%NetBSD.org@localhost>
date:      Mon Aug 27 15:03:32 2018 +0000

description:
Handle a bazillion additional cases I hadn't considered.

There's a lot of duplication here.  To be deduplicated in a
subsequent commit.

diffstat:

 sys/external/bsd/common/linux/linux_work.c |  453 ++++++++++++++++++++++++----
 1 files changed, 380 insertions(+), 73 deletions(-)

diffs (truncated from 588 to 300 lines):

diff -r 91d39c63644f -r 6461012b27ae sys/external/bsd/common/linux/linux_work.c
--- a/sys/external/bsd/common/linux/linux_work.c        Mon Aug 27 15:03:20 2018 +0000
+++ b/sys/external/bsd/common/linux/linux_work.c        Mon Aug 27 15:03:32 2018 +0000
@@ -1,4 +1,4 @@
-/*     $NetBSD: linux_work.c,v 1.28 2018/08/27 15:03:20 riastradh Exp $        */
+/*     $NetBSD: linux_work.c,v 1.29 2018/08/27 15:03:32 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.28 2018/08/27 15:03:20 riastradh Exp $");
+__KERNEL_RCSID(0, "$NetBSD: linux_work.c,v 1.29 2018/08/27 15:03:32 riastradh Exp $");
 
 #include <sys/types.h>
 #include <sys/atomic.h>
@@ -65,8 +65,6 @@
                            struct workqueue_struct *);
 static void            release_work(struct work_struct *,
                            struct workqueue_struct *);
-static void            queue_delayed_work_anew(struct workqueue_struct *,
-                           struct delayed_work *, unsigned long);
 static void            cancel_delayed_work_done(struct workqueue_struct *,
                            struct delayed_work *);
 
@@ -399,11 +397,39 @@
 
        mutex_enter(&wq->wq_lock);
        if (__predict_true((wq0 = acquire_work(work, wq)) == NULL)) {
+               /*
+                * It wasn't on any workqueue at all.  Put it on this
+                * one, and signal the worker thread that there is work
+                * to do.
+                */
                TAILQ_INSERT_TAIL(&wq->wq_queue, work, work_entry);
                newly_queued = true;
+               cv_broadcast(&wq->wq_cv);
        } else {
+               /*
+                * It was on a workqueue, which had better be this one.
+                * Requeue it if it has been taken off the queue to
+                * execute and hasn't been requeued yet.  The worker
+                * thread should already be running, so no need to
+                * signal it.
+                */
                KASSERT(wq0 == wq);
-               newly_queued = false;
+               if (wq->wq_current_work == work && !wq->wq_requeued) {
+                       /*
+                        * It has been taken off the queue to execute,
+                        * and it hasn't been put back on the queue
+                        * again.  Put it back on the queue.  No need
+                        * to signal the worker thread because it will
+                        * notice when it reacquires the lock after
+                        * doing the work.
+                        */
+                       TAILQ_INSERT_TAIL(&wq->wq_queue, work, work_entry);
+                       wq->wq_requeued = true;
+                       newly_queued = true;
+               } else {
+                       /* It is still on the queue; nothing to do.  */
+                       newly_queued = false;
+               }
        }
        mutex_exit(&wq->wq_lock);
 
@@ -422,10 +448,23 @@
 
        mutex_enter(&wq->wq_lock);
        if (__predict_false(work->work_queue != wq)) {
+               /*
+                * It has finished execution or been cancelled by
+                * another thread, and has been moved off the
+                * workqueue, so it's too to cancel.
+                */
                cancelled_p = false;
        } else if (wq->wq_current_work == work) {
+               /*
+                * It has already begun execution, so it's too late to
+                * cancel now.
+                */
                cancelled_p = false;
        } else {
+               /*
+                * It is still on the queue.  Take it off the queue and
+                * report successful cancellation.
+                */
                TAILQ_REMOVE(&wq->wq_queue, work, work_entry);
                cancelled_p = true;
        }
@@ -446,13 +485,28 @@
 
        mutex_enter(&wq->wq_lock);
        if (__predict_false(work->work_queue != wq)) {
+               /*
+                * It has finished execution or been cancelled by
+                * another thread, and has been moved off the
+                * workqueue, so it's too to cancel.
+                */
                cancelled_p = false;
        } else if (wq->wq_current_work == work) {
+               /*
+                * It has already begun execution, so it's too late to
+                * cancel now.  Wait for it to complete.  Don't wait
+                * more than one generation in case it gets requeued.
+                */
+               uint64_t gen = wq->wq_gen;
                do {
                        cv_wait(&wq->wq_cv, &wq->wq_lock);
-               } while (wq->wq_current_work == work);
+               } while (wq->wq_current_work == work && wq->wq_gen == gen);
                cancelled_p = false;
        } else {
+               /*
+                * It is still on the queue.  Take it off the queue and
+                * report successful cancellation.
+                */
                TAILQ_REMOVE(&wq->wq_queue, work, work_entry);
                cancelled_p = true;
        }
@@ -487,38 +541,13 @@
        return queue_delayed_work(system_wq, dw, ticks);
 }
 
-static void
-queue_delayed_work_anew(struct workqueue_struct *wq, struct delayed_work *dw,
-    unsigned long ticks)
-{
-
-       KASSERT(mutex_owned(&wq->wq_lock));
-       KASSERT(dw->work.work_queue == wq);
-       KASSERT((dw->dw_state == DELAYED_WORK_IDLE) ||
-           (dw->dw_state == DELAYED_WORK_SCHEDULED));
-
-       if (ticks == 0) {
-               if (dw->dw_state == DELAYED_WORK_SCHEDULED) {
-                       callout_destroy(&dw->dw_callout);
-                       TAILQ_REMOVE(&wq->wq_delayed, dw, dw_entry);
-               } else {
-                       KASSERT(dw->dw_state == DELAYED_WORK_IDLE);
-               }
-               TAILQ_INSERT_TAIL(&wq->wq_queue, &dw->work, work_entry);
-               dw->dw_state = DELAYED_WORK_IDLE;
-       } else {
-               if (dw->dw_state == DELAYED_WORK_IDLE) {
-                       callout_init(&dw->dw_callout, CALLOUT_MPSAFE);
-                       callout_reset(&dw->dw_callout, MIN(INT_MAX, ticks),
-                           &linux_workqueue_timeout, dw);
-                       TAILQ_INSERT_HEAD(&wq->wq_delayed, dw, dw_entry);
-               } else {
-                       KASSERT(dw->dw_state == DELAYED_WORK_SCHEDULED);
-               }
-               dw->dw_state = DELAYED_WORK_SCHEDULED;
-       }
-}
-
+/*
+ * cancel_delayed_work_done(wq, dw)
+ *
+ *     Complete cancellation of a delayed work: transition from
+ *     DELAYED_WORK_CANCELLED to DELAYED_WORK_IDLE and off the
+ *     workqueue.  Caller must not touch dw after this returns.
+ */
 static void
 cancel_delayed_work_done(struct workqueue_struct *wq, struct delayed_work *dw)
 {
@@ -533,6 +562,13 @@
        /* Can't touch dw after this point.  */
 }
 
+/*
+ * queue_delayed_work(wq, dw, ticks)
+ *
+ *     If it is not currently scheduled, schedule dw to run after
+ *     ticks.  If currently executing and not already rescheduled,
+ *     reschedule it.  If ticks == 0, run without delay.
+ */
 bool
 queue_delayed_work(struct workqueue_struct *wq, struct delayed_work *dw,
     unsigned long ticks)
@@ -542,18 +578,130 @@
 
        mutex_enter(&wq->wq_lock);
        if (__predict_true((wq0 = acquire_work(&dw->work, wq)) == NULL)) {
+               /*
+                * It wasn't on any workqueue at all.  Schedule it to
+                * run on this one.
+                */
                KASSERT(dw->dw_state == DELAYED_WORK_IDLE);
-               queue_delayed_work_anew(wq, dw, ticks);
+               if (ticks == 0) {
+                       TAILQ_INSERT_TAIL(&wq->wq_queue, &dw->work,
+                           work_entry);
+                       cv_broadcast(&wq->wq_cv);
+               } else {
+                       /*
+                        * Initialize a callout and schedule to run
+                        * after a delay.
+                        */
+                       callout_init(&dw->dw_callout, CALLOUT_MPSAFE);
+                       callout_setfunc(&dw->dw_callout,
+                           &linux_workqueue_timeout, dw);
+                       TAILQ_INSERT_HEAD(&wq->wq_delayed, dw, dw_entry);
+                       dw->dw_state = DELAYED_WORK_SCHEDULED;
+                       callout_schedule(&dw->dw_callout, MIN(INT_MAX, ticks));
+               }
                newly_queued = true;
        } else {
+               /*
+                * It was on a workqueue, which had better be this one.
+                *
+                * - If it has already begun to run, and it is not yet
+                *   scheduled to run again, schedule it again.
+                *
+                * - If the callout is cancelled, reschedule it.
+                *
+                * - Otherwise, leave it alone.
+                */
                KASSERT(wq0 == wq);
-               newly_queued = false;
+               if (wq->wq_current_work != &dw->work || !wq->wq_requeued) {
+                       /*
+                        * It is either scheduled, on the queue but not
+                        * in progress, or in progress but not on the
+                        * queue.
+                        */
+                       switch (dw->dw_state) {
+                       case DELAYED_WORK_IDLE:
+                               /*
+                                * It is not scheduled to run, and it
+                                * is not on the queue if it is
+                                * running.
+                                */
+                               if (ticks == 0) {
+                                       /*
+                                        * If it's in progress, put it
+                                        * on the queue to run as soon
+                                        * as the worker thread gets to
+                                        * it.  No need for a wakeup
+                                        * because either the worker
+                                        * thread already knows it is
+                                        * on the queue, or will check
+                                        * once it is done executing.
+                                        */
+                                       if (wq->wq_current_work == &dw->work) {
+                                               KASSERT(!wq->wq_requeued);
+                                               TAILQ_INSERT_TAIL(&wq->wq_queue,
+                                                   &dw->work, work_entry);
+                                               wq->wq_requeued = true;
+                                       }
+                               } else {
+                                       /*
+                                        * Initialize a callout and
+                                        * schedule it to run after the
+                                        * specified delay.
+                                        */
+                                       callout_init(&dw->dw_callout,
+                                           CALLOUT_MPSAFE);
+                                       callout_reset(&dw->dw_callout,
+                                           MIN(INT_MAX, ticks),
+                                           &linux_workqueue_timeout, dw);
+                                       TAILQ_INSERT_HEAD(&wq->wq_delayed, dw,
+                                           dw_entry);
+                                       dw->dw_state = DELAYED_WORK_SCHEDULED;
+                               }
+                               break;
+                       case DELAYED_WORK_SCHEDULED:
+                       case DELAYED_WORK_RESCHEDULED:
+                               /*
+                                * It is already scheduled to run after
+                                * a delay.  Leave it be.
+                                */
+                               break;
+                       case DELAYED_WORK_CANCELLED:
+                               /*
+                                * It was scheduled and the callout has
+                                * begun to execute, but it was
+                                * cancelled.  Reschedule it.
+                                */
+                               dw->dw_state = DELAYED_WORK_RESCHEDULED;
+                               callout_schedule(&dw->dw_callout,
+                                   MIN(INT_MAX, ticks));
+                               break;
+                       default:
+                               panic("invalid delayed work state: %d",
+                                   dw->dw_state);
+                       }
+               } else {
+                       /*
+                        * It is in progress and it has been requeued.
+                        * It cannot be scheduled to run after a delay
+                        * at this point.  We just leave it be.
+                        */
+                       KASSERTMSG((dw->dw_state == DELAYED_WORK_IDLE),
+                           "delayed work %p in wrong state: %d",
+                           dw, dw->dw_state);
+               }
        }



Home | Main Index | Thread Index | Old Index