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 Avoid the prospect of callout ...



details:   https://anonhg.NetBSD.org/src/rev/447d76b69fd2
branches:  trunk
changeset: 993064:447d76b69fd2
user:      riastradh <riastradh%NetBSD.org@localhost>
date:      Mon Aug 27 15:05:01 2018 +0000

description:
Avoid the prospect of callout calls piling up.

Don't ever callout_schedule the callout while an existing call may be
in progress.

Echo some cases from cancel_delayed_work in flush_delayed_work.

diffstat:

 sys/external/bsd/common/include/linux/workqueue.h |    3 +-
 sys/external/bsd/common/linux/linux_work.c        |  136 +++++++++++++++------
 2 files changed, 99 insertions(+), 40 deletions(-)

diffs (275 lines):

diff -r 1541887edda7 -r 447d76b69fd2 sys/external/bsd/common/include/linux/workqueue.h
--- a/sys/external/bsd/common/include/linux/workqueue.h Mon Aug 27 15:04:45 2018 +0000
+++ b/sys/external/bsd/common/include/linux/workqueue.h Mon Aug 27 15:05:01 2018 +0000
@@ -1,4 +1,4 @@
-/*     $NetBSD: workqueue.h,v 1.11 2018/08/27 15:03:20 riastradh Exp $ */
+/*     $NetBSD: workqueue.h,v 1.12 2018/08/27 15:05:01 riastradh Exp $ */
 
 /*-
  * Copyright (c) 2013, 2018 The NetBSD Foundation, Inc.
@@ -72,6 +72,7 @@
        struct work_struct              work; /* Linux API name */
        struct callout                  dw_callout;
        TAILQ_ENTRY(delayed_work)       dw_entry;
+       int                             dw_resched;
        enum {
                DELAYED_WORK_IDLE,
                DELAYED_WORK_SCHEDULED,
diff -r 1541887edda7 -r 447d76b69fd2 sys/external/bsd/common/linux/linux_work.c
--- a/sys/external/bsd/common/linux/linux_work.c        Mon Aug 27 15:04:45 2018 +0000
+++ b/sys/external/bsd/common/linux/linux_work.c        Mon Aug 27 15:05:01 2018 +0000
@@ -1,4 +1,4 @@
-/*     $NetBSD: linux_work.c,v 1.34 2018/08/27 15:04:45 riastradh Exp $        */
+/*     $NetBSD: linux_work.c,v 1.35 2018/08/27 15:05:01 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.34 2018/08/27 15:04:45 riastradh Exp $");
+__KERNEL_RCSID(0, "$NetBSD: linux_work.c,v 1.35 2018/08/27 15:05:01 riastradh Exp $");
 
 #include <sys/types.h>
 #include <sys/atomic.h>
@@ -314,7 +314,10 @@
                cv_broadcast(&wq->wq_cv);
                break;
        case DELAYED_WORK_RESCHEDULED:
+               KASSERT(dw->dw_resched >= 0);
+               callout_schedule(&dw->dw_callout, dw->dw_resched);
                dw->dw_state = DELAYED_WORK_SCHEDULED;
+               dw->dw_resched = -1;
                break;
        case DELAYED_WORK_CANCELLED:
                cancel_delayed_work_done(wq, dw);
@@ -546,6 +549,7 @@
 
        INIT_WORK(&dw->work, fn);
        dw->dw_state = DELAYED_WORK_IDLE;
+       dw->dw_resched = -1;
 
        /*
         * Defer callout_init until we are going to schedule the
@@ -599,6 +603,7 @@
 
        TAILQ_REMOVE(&wq->wq_delayed, dw, dw_entry);
        callout_destroy(&dw->dw_callout);
+       dw->dw_resched = -1;
        dw->dw_state = DELAYED_WORK_IDLE;
 }
 
@@ -723,8 +728,7 @@
                                 * cancelled.  Reschedule it.
                                 */
                                dw->dw_state = DELAYED_WORK_RESCHEDULED;
-                               callout_schedule(&dw->dw_callout,
-                                   MIN(INT_MAX, ticks));
+                               dw->dw_resched = MIN(INT_MAX, ticks);
                                break;
                        default:
                                panic("invalid delayed work state: %d",
@@ -867,17 +871,12 @@
                                         * the lock.
                                         */
                                } else {
-                                       /*
-                                        * Schedule callout and tell
-                                        * the instance that's running
-                                        * now that it's been
-                                        * rescheduled.
-                                        */
+                                       /* Ask the callout to reschedule.  */
                                        dw->dw_state = DELAYED_WORK_RESCHEDULED;
-                                       callout_schedule(&dw->dw_callout,
-                                           MIN(INT_MAX, ticks));
+                                       dw->dw_resched = MIN(INT_MAX, ticks);
                                }
                        } else {
+                               /* We stopped the callout before it began.  */
                                if (ticks == 0) {
                                        /*
                                         * Run immediately: destroy the
@@ -901,12 +900,31 @@
                        timer_modified = true;
                        break;
                case DELAYED_WORK_RESCHEDULED:
+                       /*
+                        * Someone rescheduled it after the callout
+                        * started but before the poor thing even had a
+                        * chance to acquire the lock.
+                        */
+                       if (ticks == 0) {
+                               /*
+                                * We can just switch back to
+                                * DELAYED_WORK_SCHEDULED so that the
+                                * callout will queue the work as soon
+                                * as it gets the lock.
+                                */
+                               dw->dw_state = DELAYED_WORK_SCHEDULED;
+                               dw->dw_resched = -1;
+                       } else {
+                               /* Change the rescheduled time.  */
+                               dw->dw_resched = ticks;
+                       }
+                       timer_modified = true;
+                       break;
                case DELAYED_WORK_CANCELLED:
                        /*
-                        * Someone modified the timer _again_, or
-                        * cancelled it, after the callout started but
-                        * before the poor thing even had a chance to
-                        * acquire the lock.
+                        * Someone cancelled it after the callout
+                        * started but before the poor thing even had a
+                        * chance to acquire the lock.
                         */
                        if (ticks == 0) {
                                /*
@@ -918,9 +936,8 @@
                                dw->dw_state = DELAYED_WORK_SCHEDULED;
                        } else {
                                /* Reschedule it.  */
-                               callout_schedule(&dw->dw_callout,
-                                   MIN(INT_MAX, ticks));
                                dw->dw_state = DELAYED_WORK_RESCHEDULED;
+                               dw->dw_resched = MIN(INT_MAX, ticks);
                        }
                        timer_modified = true;
                        break;
@@ -949,18 +966,36 @@
        } else {
                switch (dw->dw_state) {
                case DELAYED_WORK_IDLE:
-                       if (wq->wq_current_work == &dw->work) {
+                       /*
+                        * It is either on the queue or already running
+                        * or both.
+                        */
+                       if (wq->wq_current_work != &dw->work ||
+                           wq->wq_requeued) {
                                /*
-                                * Too late, it's already running.  If
-                                * it's been requeued, tough -- it'll
-                                * run again.
+                                * It is on the queue, and it may or
+                                * may not be running.  Remove it from
+                                * the queue.
+                                */
+                               TAILQ_REMOVE(&wq->wq_queue, &dw->work,
+                                   work_entry);
+                               if (wq->wq_current_work == &dw->work) {
+                                       /*
+                                        * If it is running, then it
+                                        * must have been requeued in
+                                        * this case, so mark it no
+                                        * longer requeued.
+                                        */
+                                       KASSERT(wq->wq_requeued);
+                                       wq->wq_requeued = false;
+                               }
+                               cancelled_p = true;
+                       } else {
+                               /*
+                                * Too late, it's already running, but
+                                * at least it hasn't been requeued.
                                 */
                                cancelled_p = false;
-                       } else {
-                               /* Got in before it started.  Remove it.  */
-                               TAILQ_REMOVE(&wq->wq_queue, &dw->work,
-                                   work_entry);
-                               cancelled_p = true;
                        }
                        break;
                case DELAYED_WORK_SCHEDULED:
@@ -986,6 +1021,7 @@
                         * already fired.  We must ask it to cancel.
                         */
                        dw->dw_state = DELAYED_WORK_CANCELLED;
+                       dw->dw_resched = -1;
                        cancelled_p = true;
                        break;
                case DELAYED_WORK_CANCELLED:
@@ -1023,6 +1059,10 @@
        } else {
                switch (dw->dw_state) {
                case DELAYED_WORK_IDLE:
+                       /*
+                        * It is either on the queue or already running
+                        * or both.
+                        */
                        if (wq->wq_current_work == &dw->work) {
                                /*
                                 * Too late, it's already running.
@@ -1055,7 +1095,7 @@
                         * cancelled it in that case.
                         *
                         * If we stopped the callout before it started,
-                        * however, then destroy the callout and
+                        * then we must destroy the callout and
                         * dissociate it from the workqueue ourselves.
                         */
                        dw->dw_state = DELAYED_WORK_CANCELLED;
@@ -1070,6 +1110,7 @@
                         * wait for it to complete.
                         */
                        dw->dw_state = DELAYED_WORK_CANCELLED;
+                       dw->dw_resched = -1;
                        (void)callout_halt(&dw->dw_callout, &wq->wq_lock);
                        cancelled_p = true;
                        break;
@@ -1178,24 +1219,41 @@
                        flush_workqueue_locked(wq);
                        break;
                case DELAYED_WORK_SCHEDULED:
-               case DELAYED_WORK_RESCHEDULED:
-               case DELAYED_WORK_CANCELLED:
                        /*
-                        * The callout is still scheduled to run.
-                        * Notify it that we are cancelling, and try to
-                        * stop the callout before it runs.
+                        * If it is scheduled, mark it cancelled and
+                        * try to stop the callout before it starts.
                         *
-                        * If we do stop the callout, we are now
-                        * responsible for dissociating the work from
-                        * the queue.
+                        * If it's too late and the callout has already
+                        * begun to execute, we must wait for it to
+                        * complete.  But we got in soon enough to ask
+                        * the callout not to run.
                         *
-                        * Otherwise, wait for it to complete and
-                        * dissociate itself -- it will not put itself
-                        * on the workqueue once it is cancelled.
+                        * If we stopped the callout before it started,
+                        * then we must destroy the callout and
+                        * dissociate it from the workqueue ourselves.
                         */
                        dw->dw_state = DELAYED_WORK_CANCELLED;
                        if (!callout_halt(&dw->dw_callout, &wq->wq_lock))
                                cancel_delayed_work_done(wq, dw);
+                       break;
+               case DELAYED_WORK_RESCHEDULED:
+                       /*
+                        * If it is being rescheduled, the callout has
+                        * already fired.  We must ask it to cancel and
+                        * wait for it to complete.
+                        */
+                       dw->dw_state = DELAYED_WORK_CANCELLED;
+                       dw->dw_resched = -1;
+                       (void)callout_halt(&dw->dw_callout, &wq->wq_lock);
+                       break;
+               case DELAYED_WORK_CANCELLED:
+                       /*
+                        * If it is being cancelled, the callout has
+                        * already fired.  We need only wait for it to
+                        * complete.
+                        */
+                       (void)callout_halt(&dw->dw_callout, &wq->wq_lock);
+                       break;
                default:
                        panic("invalid delayed work state: %d",
                            dw->dw_state);



Home | Main Index | Thread Index | Old Index