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 Document workqueue semantics a...



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

description:
Document workqueue semantics as I understand it.

diffstat:

 sys/external/bsd/common/linux/linux_work.c |  199 ++++++++++++++++++++++++++++-
 1 files changed, 195 insertions(+), 4 deletions(-)

diffs (truncated from 388 to 300 lines):

diff -r 447d76b69fd2 -r d08a216e7e35 sys/external/bsd/common/linux/linux_work.c
--- a/sys/external/bsd/common/linux/linux_work.c        Mon Aug 27 15:05:01 2018 +0000
+++ b/sys/external/bsd/common/linux/linux_work.c        Mon Aug 27 15:05:16 2018 +0000
@@ -1,4 +1,4 @@
-/*     $NetBSD: linux_work.c,v 1.35 2018/08/27 15:05:01 riastradh Exp $        */
+/*     $NetBSD: linux_work.c,v 1.36 2018/08/27 15:05:16 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.35 2018/08/27 15:05:01 riastradh Exp $");
+__KERNEL_RCSID(0, "$NetBSD: linux_work.c,v 1.36 2018/08/27 15:05:16 riastradh Exp $");
 
 #include <sys/types.h>
 #include <sys/atomic.h>
@@ -80,6 +80,12 @@
 struct workqueue_struct        *system_long_wq __read_mostly;
 struct workqueue_struct        *system_power_efficient_wq __read_mostly;
 
+/*
+ * linux_workqueue_init()
+ *
+ *     Initialize the Linux workqueue subsystem.  Return 0 on success,
+ *     NetBSD error on failure.
+ */
 int
 linux_workqueue_init(void)
 {
@@ -118,6 +124,11 @@
        return error;
 }
 
+/*
+ * linux_workqueue_fini()
+ *
+ *     Destroy the Linux workqueue subsystem.  Never fails.
+ */
 void
 linux_workqueue_fini(void)
 {
@@ -132,6 +143,13 @@
  * Workqueues
  */
 
+/*
+ * alloc_ordered_workqueue(name, flags)
+ *
+ *     Create a workqueue of the given name.  No flags are currently
+ *     defined.  Return NULL on failure, pointer to struct
+ *     workqueue_struct object on success.
+ */
 struct workqueue_struct *
 alloc_ordered_workqueue(const char *name, int flags)
 {
@@ -169,6 +187,14 @@
        return NULL;
 }
 
+/*
+ * destroy_workqueue(wq)
+ *
+ *     Destroy a workqueue created with wq.  Cancel any pending
+ *     delayed work.  Wait for all queued work to complete.
+ *
+ *     May sleep.
+ */
 void
 destroy_workqueue(struct workqueue_struct *wq)
 {
@@ -237,6 +263,14 @@
  * Work thread and callout
  */
 
+/*
+ * linux_workqueue_thread(cookie)
+ *
+ *     Main function for a workqueue's worker thread.  Waits until
+ *     there is work queued, grabs a batch of work off the queue,
+ *     executes it all, bumps the generation number, and repeats,
+ *     until dying.
+ */
 static void __dead
 linux_workqueue_thread(void *cookie)
 {
@@ -295,6 +329,16 @@
        kthread_exit(0);
 }
 
+/*
+ * linux_workqueue_timeout(cookie)
+ *
+ *     Delayed work timeout callback.
+ *
+ *     - If scheduled, queue it.
+ *     - If rescheduled, callout_schedule ourselves again.
+ *     - If cancelled, destroy the callout and release the work from
+ *        the workqueue.
+ */
 static void
 linux_workqueue_timeout(void *cookie)
 {
@@ -331,6 +375,12 @@
 out:   mutex_exit(&wq->wq_lock);
 }
 
+/*
+ * current_work()
+ *
+ *     If in a workqueue worker thread, return the work it is
+ *     currently executing.  Otherwise return NULL.
+ */
 struct work_struct *
 current_work(void)
 {
@@ -352,6 +402,12 @@
  * Work
  */
 
+/*
+ * INIT_WORK(work, fn)
+ *
+ *     Initialize work for use with a workqueue to call fn in a worker
+ *     thread.  There is no corresponding destruction operation.
+ */
 void
 INIT_WORK(struct work_struct *work, void (*fn)(struct work_struct *))
 {
@@ -360,6 +416,16 @@
        work->func = fn;
 }
 
+/*
+ * acquire_work(work, wq)
+ *
+ *     Try to associate work with wq.  If work is already on a
+ *     workqueue, return that workqueue.  Otherwise, set work's queue
+ *     to wq, issue a memory barrier to match any prior release_work,
+ *     and return NULL.
+ *
+ *     Caller must hold wq's lock.
+ */
 static struct workqueue_struct *
 acquire_work(struct work_struct *work, struct workqueue_struct *wq)
 {
@@ -376,6 +442,14 @@
        return wq0;
 }
 
+/*
+ * release_work(work, wq)
+ *
+ *     Issue a memory barrier to match any subsequent acquire_work and
+ *     dissociate work from wq.
+ *
+ *     Caller must hold wq's lock and work must be associated with wq.
+ */
 static void
 release_work(struct work_struct *work, struct workqueue_struct *wq)
 {
@@ -387,6 +461,17 @@
        work->work_queue = NULL;
 }
 
+/*
+ * schedule_work(work)
+ *
+ *     If work is not already queued on system_wq, queue it to be run
+ *     by system_wq's worker thread when it next can.  True if it was
+ *     newly queued, false if it was already queued.  If the work was
+ *     already running, queue it to run again.
+ *
+ *     Caller must ensure work is not queued to run on a different
+ *     workqueue.
+ */
 bool
 schedule_work(struct work_struct *work)
 {
@@ -394,6 +479,17 @@
        return queue_work(system_wq, work);
 }
 
+/*
+ * queue_work(wq, work)
+ *
+ *     If work is not already queued on wq, queue it to be run by wq's
+ *     worker thread when it next can.  True if it was newly queued,
+ *     false if it was already queued.  If the work was already
+ *     running, queue it to run again.
+ *
+ *     Caller must ensure work is not queued to run on a different
+ *     workqueue.
+ */
 bool
 queue_work(struct workqueue_struct *wq, struct work_struct *work)
 {
@@ -443,6 +539,15 @@
        return newly_queued;
 }
 
+/*
+ * cancel_work(work)
+ *
+ *     If work was queued, remove it from the queue and return true.
+ *     If work was not queued, return false.  Note that work may
+ *     already be running; if it hasn't been requeued, then
+ *     cancel_work will return false, and either way, cancel_work will
+ *     NOT wait for the work to complete.
+ */
 bool
 cancel_work(struct work_struct *work)
 {
@@ -480,6 +585,17 @@
 out:   return cancelled_p;
 }
 
+/*
+ * cancel_work_sync(work)
+ *
+ *     If work was queued, remove it from the queue and return true.
+ *     If work was not queued, return false.  Note that work may
+ *     already be running; if it hasn't been requeued, then
+ *     cancel_work will return false; either way, if work was
+ *     currently running, wait for it to complete.
+ *
+ *     May sleep.
+ */
 bool
 cancel_work_sync(struct work_struct *work)
 {
@@ -543,6 +659,13 @@
  * Delayed work
  */
 
+/*
+ * INIT_DELAYED_WORK(dw, fn)
+ *
+ *     Initialize dw for use with a workqueue to call fn in a worker
+ *     thread after a delay.  There is no corresponding destruction
+ *     operation.
+ */
 void
 INIT_DELAYED_WORK(struct delayed_work *dw, void (*fn)(struct work_struct *))
 {
@@ -559,6 +682,17 @@
         */
 }
 
+/*
+ * schedule_delayed_work(dw, ticks)
+ *
+ *     If it is not currently scheduled, schedule dw to run after
+ *     ticks on system_wq.  If currently executing and not already
+ *     rescheduled, reschedule it.  True if it was newly scheduled,
+ *     false if it was already scheduled.
+ *
+ *     If ticks == 0, queue it to run as soon as the worker can,
+ *     without waiting for the next callout tick to run.
+ */
 bool
 schedule_delayed_work(struct delayed_work *dw, unsigned long ticks)
 {
@@ -631,8 +765,11 @@
  * 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.
+ *     ticks on wq.  If currently executing and not already
+ *     rescheduled, reschedule it.
+ *
+ *     If ticks == 0, queue it to run as soon as the worker can,
+ *     without waiting for the next callout tick to run.
  */
 bool
 queue_delayed_work(struct workqueue_struct *wq, struct delayed_work *dw,
@@ -756,6 +893,9 @@
  *     Schedule dw to run after ticks.  If currently scheduled,
  *     reschedule it.  If currently executing, reschedule it.  If
  *     ticks == 0, run without delay.
+ *
+ *     True if it modified the timer of an already scheduled work,
+ *     false if it newly scheduled the work.
  */
 bool
 mod_delayed_work(struct workqueue_struct *wq, struct delayed_work *dw,
@@ -950,6 +1090,16 @@
        return timer_modified;
 }
 
+/*
+ * cancel_delayed_work(dw)
+ *
+ *     If work was scheduled or queued, remove it from the schedule or
+ *     queue and return true.  If work was not scheduled or queued,
+ *     return false.  Note that work may already be running; if it
+ *     hasn't been rescheduled or requeued, then cancel_delayed_work
+ *     will return false, and either way, cancel_delayed_work will NOT



Home | Main Index | Thread Index | Old Index