The patch titled Change return value from queue_work() has been added to the -mm tree. Its filename is change-return-value-from-queue_work.patch See http://www.zip.com.au/~akpm/linux/patches/stuff/added-to-mm.txt to find out what to do about this ------------------------------------------------------ Subject: Change return value from queue_work() From: Alan Stern <stern@xxxxxxxxxxxxxxxxxxx> This patch (as774) renames: queue_work() to add_work_to_q(), queue_delayed_work() to add_delayed_work_to_q(), queue_delayed_work_on() to add_delayed_work_to_q_on(). The return value is altered, so that now 0 = success and -EBUSY = failure. New routines with the original names are added, so that the majority of callers don't need any changes. The new routines call the original functions, call WARN_ON if there was a failure, and then return void. Finally, fc_queue_work(), fc_queue_devloss_work(), scsi_queue_work(), and kblockd_schedule_work() were changed to return void. Signed-off-by: Alan Stern <stern@xxxxxxxxxxxxxxxxxxx> Cc: Jens Axboe <axboe@xxxxxxx> Cc: "Brown, Len" <len.brown@xxxxxxxxx> Cc: Jeff Garzik <jeff@xxxxxxxxxx> Cc: Evgeniy Polyakov <johnpol@xxxxxxxxxxx> Cc: Markus Lidel <Markus.Lidel@xxxxxxxxxxxxxxxxx> Cc: James Bottomley <James.Bottomley@xxxxxxxxxxxx> Cc: Mark Fasheh <mark.fasheh@xxxxxxxxxx> Cc: Trond Myklebust <trond.myklebust@xxxxxxxxxx> Cc: Neil Brown <neilb@xxxxxxx> Signed-off-by: Andrew Morton <akpm@xxxxxxxx> --- block/ll_rw_blk.c | 4 - drivers/acpi/osl.c | 4 - drivers/ata/libata-core.c | 8 +- drivers/connector/connector.c | 7 +- drivers/message/i2o/i2o_block.c | 4 - drivers/scsi/hosts.c | 13 +--- drivers/scsi/scsi_transport_fc.c | 21 ++----- fs/ocfs2/cluster/tcp.c | 4 - include/linux/blkdev.h | 2 include/linux/workqueue.h | 21 +++++-- include/scsi/scsi_host.h | 2 kernel/workqueue.c | 87 ++++++++++++++++++++--------- net/sunrpc/sched.c | 2 13 files changed, 108 insertions(+), 71 deletions(-) diff -puN block/ll_rw_blk.c~change-return-value-from-queue_work block/ll_rw_blk.c --- a/block/ll_rw_blk.c~change-return-value-from-queue_work +++ a/block/ll_rw_blk.c @@ -3488,9 +3488,9 @@ void blk_rq_bio_prep(request_queue_t *q, EXPORT_SYMBOL(blk_rq_bio_prep); -int kblockd_schedule_work(struct work_struct *work) +void kblockd_schedule_work(struct work_struct *work) { - return queue_work(kblockd_workqueue, work); + queue_work(kblockd_workqueue, work); } EXPORT_SYMBOL(kblockd_schedule_work); diff -puN drivers/acpi/osl.c~change-return-value-from-queue_work drivers/acpi/osl.c --- a/drivers/acpi/osl.c~change-return-value-from-queue_work +++ a/drivers/acpi/osl.c @@ -636,9 +636,9 @@ acpi_status acpi_os_execute(acpi_execute task = (void *)(dpc + 1); INIT_WORK(task, acpi_os_execute_deferred, (void *)dpc); - if (!queue_work(kacpid_wq, task)) { + if (add_work_to_q(kacpid_wq, task) < 0) { ACPI_DEBUG_PRINT((ACPI_DB_ERROR, - "Call to queue_work() failed.\n")); + "Call to add_work_to_q() failed.\n")); kfree(dpc); status = AE_ERROR; } diff -puN drivers/ata/libata-core.c~change-return-value-from-queue_work drivers/ata/libata-core.c --- a/drivers/ata/libata-core.c~change-return-value-from-queue_work +++ a/drivers/ata/libata-core.c @@ -933,12 +933,12 @@ void ata_port_queue_task(struct ata_port PREPARE_WORK(&ap->port_task, fn, data); if (!delay) - rc = queue_work(ata_wq, &ap->port_task); + rc = add_work_to_q(ata_wq, &ap->port_task); else - rc = queue_delayed_work(ata_wq, &ap->port_task, delay); + rc = add_delayed_work_to_q(ata_wq, &ap->port_task, delay); - /* rc == 0 means that another user is using port task */ - WARN_ON(rc == 0); + /* rc < 0 means that another user is using port task */ + WARN_ON(rc < 0); } /** diff -puN drivers/connector/connector.c~change-return-value-from-queue_work drivers/connector/connector.c --- a/drivers/connector/connector.c~change-return-value-from-queue_work +++ a/drivers/connector/connector.c @@ -142,8 +142,8 @@ static int cn_call_callback(struct cn_ms __cbq->data.ddata = data; __cbq->data.destruct_data = destruct_data; - if (queue_work(dev->cbdev->cn_queue, - &__cbq->work)) + if (add_work_to_q(dev->cbdev->cn_queue, + &__cbq->work) == 0) err = 0; } else { struct work_struct *w; @@ -165,7 +165,8 @@ static int cn_call_callback(struct cn_ms w->data = d; init_timer(&w->timer); - if (queue_work(dev->cbdev->cn_queue, w)) + if (add_work_to_q(dev->cbdev->cn_queue, + w) == 0) err = 0; else { kfree(w); diff -puN drivers/message/i2o/i2o_block.c~change-return-value-from-queue_work drivers/message/i2o/i2o_block.c --- a/drivers/message/i2o/i2o_block.c~change-return-value-from-queue_work +++ a/drivers/message/i2o/i2o_block.c @@ -941,9 +941,9 @@ static void i2o_block_request_fn(struct INIT_WORK(&dreq->work, i2o_block_delayed_request_fn, dreq); - if (!queue_delayed_work(i2o_block_driver.event_queue, + if (add_delayed_work_to_q(i2o_block_driver.event_queue, &dreq->work, - I2O_BLOCK_RETRY_TIME)) + I2O_BLOCK_RETRY_TIME) < 0) kfree(dreq); else { blk_stop_queue(q); diff -puN drivers/scsi/hosts.c~change-return-value-from-queue_work drivers/scsi/hosts.c --- a/drivers/scsi/hosts.c~change-return-value-from-queue_work +++ a/drivers/scsi/hosts.c @@ -486,16 +486,11 @@ int scsi_is_host_device(const struct dev EXPORT_SYMBOL(scsi_is_host_device); /** - * scsi_queue_work - Queue work to the Scsi_Host workqueue. + * scsi_queue_work - Submit work to the Scsi_Host workqueue. * @shost: Pointer to Scsi_Host. * @work: Work to queue for execution. - * - * Return value: - * 1 - work queued for execution - * 0 - work is already queued - * -EINVAL - work queue doesn't exist **/ -int scsi_queue_work(struct Scsi_Host *shost, struct work_struct *work) +void scsi_queue_work(struct Scsi_Host *shost, struct work_struct *work) { if (unlikely(!shost->work_q)) { printk(KERN_ERR @@ -503,10 +498,10 @@ int scsi_queue_work(struct Scsi_Host *sh "when no workqueue created.\n", shost->hostt->name); dump_stack(); - return -EINVAL; + return; } - return queue_work(shost->work_q, work); + queue_work(shost->work_q, work); } EXPORT_SYMBOL_GPL(scsi_queue_work); diff -puN drivers/scsi/scsi_transport_fc.c~change-return-value-from-queue_work drivers/scsi/scsi_transport_fc.c --- a/drivers/scsi/scsi_transport_fc.c~change-return-value-from-queue_work +++ a/drivers/scsi/scsi_transport_fc.c @@ -34,7 +34,7 @@ #include <scsi/scsi_cmnd.h> #include "scsi_priv.h" -static int fc_queue_work(struct Scsi_Host *, struct work_struct *); +static void fc_queue_work(struct Scsi_Host *, struct work_struct *); /* * Redefine so that we can have same named attributes in the @@ -1309,12 +1309,8 @@ EXPORT_SYMBOL(fc_release_transport); * @shost: Pointer to Scsi_Host bound to fc_host. * @work: Work to queue for execution. * - * Return value: - * 1 - work queued for execution - * 0 - work is already queued - * -EINVAL - work queue doesn't exist **/ -static int +static void fc_queue_work(struct Scsi_Host *shost, struct work_struct *work) { if (unlikely(!fc_host_work_q(shost))) { @@ -1323,10 +1319,10 @@ fc_queue_work(struct Scsi_Host *shost, s "when no workqueue created.\n", shost->hostt->name); dump_stack(); - return -EINVAL; + return; } - return queue_work(fc_host_work_q(shost), work); + queue_work(fc_host_work_q(shost), work); } /** @@ -1352,11 +1348,8 @@ fc_flush_work(struct Scsi_Host *shost) * @shost: Pointer to Scsi_Host bound to fc_host. * @work: Work to queue for execution. * @delay: jiffies to delay the work queuing - * - * Return value: - * 0 on success / != 0 for error **/ -static int +static void fc_queue_devloss_work(struct Scsi_Host *shost, struct work_struct *work, unsigned long delay) { @@ -1366,10 +1359,10 @@ fc_queue_devloss_work(struct Scsi_Host * "when no workqueue created.\n", shost->hostt->name); dump_stack(); - return -EINVAL; + return; } - return queue_delayed_work(fc_host_devloss_work_q(shost), work, delay); + queue_delayed_work(fc_host_devloss_work_q(shost), work, delay); } /** diff -puN fs/ocfs2/cluster/tcp.c~change-return-value-from-queue_work fs/ocfs2/cluster/tcp.c --- a/fs/ocfs2/cluster/tcp.c~change-return-value-from-queue_work +++ a/fs/ocfs2/cluster/tcp.c @@ -338,7 +338,7 @@ static void o2net_sc_queue_work(struct o struct work_struct *work) { sc_get(sc); - if (!queue_work(o2net_wq, work)) + if (add_work_to_q(o2net_wq, work) < 0) sc_put(sc); } static void o2net_sc_queue_delayed_work(struct o2net_sock_container *sc, @@ -346,7 +346,7 @@ static void o2net_sc_queue_delayed_work( int delay) { sc_get(sc); - if (!queue_delayed_work(o2net_wq, work, delay)) + if (add_delayed_work_to_q(o2net_wq, work, delay) < 0) sc_put(sc); } static void o2net_sc_cancel_delayed_work(struct o2net_sock_container *sc, diff -puN include/linux/blkdev.h~change-return-value-from-queue_work include/linux/blkdev.h --- a/include/linux/blkdev.h~change-return-value-from-queue_work +++ a/include/linux/blkdev.h @@ -814,7 +814,7 @@ static inline void put_dev_sector(Sector } struct work_struct; -int kblockd_schedule_work(struct work_struct *work); +void kblockd_schedule_work(struct work_struct *work); void kblockd_flush(void); #ifdef CONFIG_LBD diff -puN include/linux/workqueue.h~change-return-value-from-queue_work include/linux/workqueue.h --- a/include/linux/workqueue.h~change-return-value-from-queue_work +++ a/include/linux/workqueue.h @@ -61,10 +61,12 @@ extern struct workqueue_struct *__create extern void destroy_workqueue(struct workqueue_struct *wq); -extern int FASTCALL(queue_work(struct workqueue_struct *wq, struct work_struct *work)); -extern int FASTCALL(queue_delayed_work(struct workqueue_struct *wq, struct work_struct *work, unsigned long delay)); -extern int queue_delayed_work_on(int cpu, struct workqueue_struct *wq, - struct work_struct *work, unsigned long delay); +extern int FASTCALL(add_work_to_q(struct workqueue_struct *wq, + struct work_struct *work)); +extern int FASTCALL(add_delayed_work_to_q(struct workqueue_struct *wq, + struct work_struct *work, unsigned long delay)); +extern int add_delayed_work_to_q_on(int cpu, struct workqueue_struct *wq, + struct work_struct *work, unsigned long delay); extern void FASTCALL(flush_workqueue(struct workqueue_struct *wq)); extern int FASTCALL(schedule_work(struct work_struct *work)); @@ -98,4 +100,15 @@ static inline int cancel_delayed_work(st return ret; } +/* + * Legacy APIs, used by old drivers that don't care about the return code. + * These routines call WARN_ON() if the underlying function fails. + */ +extern void FASTCALL(queue_work(struct workqueue_struct *wq, + struct work_struct *work)); +extern void FASTCALL(queue_delayed_work(struct workqueue_struct *wq, + struct work_struct *work, unsigned long delay)); +extern void queue_delayed_work_on(int cpu, struct workqueue_struct *wq, + struct work_struct *work, unsigned long delay); + #endif diff -puN include/scsi/scsi_host.h~change-return-value-from-queue_work include/scsi/scsi_host.h --- a/include/scsi/scsi_host.h~change-return-value-from-queue_work +++ a/include/scsi/scsi_host.h @@ -666,7 +666,7 @@ static inline int scsi_host_in_recovery( shost->tmf_in_progress; } -extern int scsi_queue_work(struct Scsi_Host *, struct work_struct *); +extern void scsi_queue_work(struct Scsi_Host *, struct work_struct *); extern void scsi_flush_work(struct Scsi_Host *); extern struct Scsi_Host *scsi_host_alloc(struct scsi_host_template *, int); diff -puN kernel/workqueue.c~change-return-value-from-queue_work kernel/workqueue.c --- a/kernel/workqueue.c~change-return-value-from-queue_work +++ a/kernel/workqueue.c @@ -80,7 +80,7 @@ static inline int is_single_threaded(str } /* Preempt must be disabled. */ -static void __queue_work(struct cpu_workqueue_struct *cwq, +static void __add_work_to_q(struct cpu_workqueue_struct *cwq, struct work_struct *work) { unsigned long flags; @@ -94,30 +94,31 @@ static void __queue_work(struct cpu_work } /** - * queue_work - queue work on a workqueue + * add_work_to_q - queue work on a workqueue * @wq: workqueue to use * @work: work to queue * - * Returns non-zero if it was successfully added. + * Returns zero if it was successfully added, -EBUSY if @work is in use. * * We queue the work to the CPU it was submitted, but there is no * guarantee that it will be processed by that CPU. */ -int fastcall queue_work(struct workqueue_struct *wq, struct work_struct *work) +int fastcall add_work_to_q(struct workqueue_struct *wq, + struct work_struct *work) { - int ret = 0, cpu = get_cpu(); + int ret = -EBUSY, cpu = get_cpu(); if (!test_and_set_bit(0, &work->pending)) { if (unlikely(is_single_threaded(wq))) cpu = singlethread_cpu; BUG_ON(!list_empty(&work->entry)); - __queue_work(per_cpu_ptr(wq->cpu_wq, cpu), work); - ret = 1; + __add_work_to_q(per_cpu_ptr(wq->cpu_wq, cpu), work); + ret = 0; } put_cpu(); return ret; } -EXPORT_SYMBOL_GPL(queue_work); +EXPORT_SYMBOL_GPL(add_work_to_q); static void delayed_work_timer_fn(unsigned long __data) { @@ -128,21 +129,21 @@ static void delayed_work_timer_fn(unsign if (unlikely(is_single_threaded(wq))) cpu = singlethread_cpu; - __queue_work(per_cpu_ptr(wq->cpu_wq, cpu), work); + __add_work_to_q(per_cpu_ptr(wq->cpu_wq, cpu), work); } /** - * queue_delayed_work - queue work on a workqueue after delay + * add_delayed_work_to_q - queue work on a workqueue after delay * @wq: workqueue to use * @work: work to queue * @delay: number of jiffies to wait before queueing * - * Returns non-zero if it was successfully added. + * Returns zero if it was successfully added, -EBUSY if @work is in use. */ -int fastcall queue_delayed_work(struct workqueue_struct *wq, +int fastcall add_delayed_work_to_q(struct workqueue_struct *wq, struct work_struct *work, unsigned long delay) { - int ret = 0; + int ret = -EBUSY; struct timer_list *timer = &work->timer; if (!test_and_set_bit(0, &work->pending)) { @@ -155,25 +156,25 @@ int fastcall queue_delayed_work(struct w timer->data = (unsigned long)work; timer->function = delayed_work_timer_fn; add_timer(timer); - ret = 1; + ret = 0; } return ret; } -EXPORT_SYMBOL_GPL(queue_delayed_work); +EXPORT_SYMBOL_GPL(add_delayed_work_to_q); /** - * queue_delayed_work_on - queue work on specific CPU after delay + * add_delayed_work_to_q_on - queue work on specific CPU after delay * @cpu: CPU number to execute work on * @wq: workqueue to use * @work: work to queue * @delay: number of jiffies to wait before queueing * - * Returns non-zero if it was successfully added. + * Returns zero if it was successfully added, -EBUSY if @work is in use. */ -int queue_delayed_work_on(int cpu, struct workqueue_struct *wq, +int add_delayed_work_to_q_on(int cpu, struct workqueue_struct *wq, struct work_struct *work, unsigned long delay) { - int ret = 0; + int ret = -EBUSY; struct timer_list *timer = &work->timer; if (!test_and_set_bit(0, &work->pending)) { @@ -186,11 +187,11 @@ int queue_delayed_work_on(int cpu, struc timer->data = (unsigned long)work; timer->function = delayed_work_timer_fn; add_timer_on(timer, cpu); - ret = 1; + ret = 0; } return ret; } -EXPORT_SYMBOL_GPL(queue_delayed_work_on); +EXPORT_SYMBOL_GPL(add_delayed_work_to_q_on); static void run_workqueue(struct cpu_workqueue_struct *cwq) { @@ -455,7 +456,7 @@ static struct workqueue_struct *keventd_ */ int fastcall schedule_work(struct work_struct *work) { - return queue_work(keventd_wq, work); + return !add_work_to_q(keventd_wq, work); } EXPORT_SYMBOL(schedule_work); @@ -469,7 +470,7 @@ EXPORT_SYMBOL(schedule_work); */ int fastcall schedule_delayed_work(struct work_struct *work, unsigned long delay) { - return queue_delayed_work(keventd_wq, work, delay); + return !add_delayed_work_to_q(keventd_wq, work, delay); } EXPORT_SYMBOL(schedule_delayed_work); @@ -485,10 +486,43 @@ EXPORT_SYMBOL(schedule_delayed_work); int schedule_delayed_work_on(int cpu, struct work_struct *work, unsigned long delay) { - return queue_delayed_work_on(cpu, keventd_wq, work, delay); + return !add_delayed_work_to_q_on(cpu, keventd_wq, work, delay); } EXPORT_SYMBOL(schedule_delayed_work_on); +/* + * Legacy API for use when the return codes aren't needed. + * These routines call WARN_ON() if the underlying function fails. + */ +void fastcall queue_work(struct workqueue_struct *wq, struct work_struct *work) +{ + int rc; + + rc = add_work_to_q(wq, work); + WARN_ON(rc < 0); +} +EXPORT_SYMBOL(queue_work); + +void fastcall queue_delayed_work(struct workqueue_struct *wq, + struct work_struct *work, unsigned long delay) +{ + int rc; + + rc = add_delayed_work_to_q(wq, work, delay); + WARN_ON(rc < 0); +} +EXPORT_SYMBOL(queue_delayed_work); + +void queue_delayed_work_on(int cpu, struct workqueue_struct *wq, + struct work_struct *work, unsigned long delay) +{ + int rc; + + rc = add_delayed_work_to_q_on(cpu, wq, work, delay); + WARN_ON(rc < 0); +} +EXPORT_SYMBOL(queue_delayed_work_on); + /** * schedule_on_each_cpu - call a function on each online CPU from keventd * @func: the function to call @@ -513,7 +547,7 @@ int schedule_on_each_cpu(void (*func)(vo mutex_lock(&workqueue_mutex); for_each_online_cpu(cpu) { INIT_WORK(per_cpu_ptr(works, cpu), func, info); - __queue_work(per_cpu_ptr(keventd_wq->cpu_wq, cpu), + __add_work_to_q(per_cpu_ptr(keventd_wq->cpu_wq, cpu), per_cpu_ptr(works, cpu)); } mutex_unlock(&workqueue_mutex); @@ -617,7 +651,8 @@ static void take_over_work(struct workqu printk("Taking work for %s\n", wq->name); work = list_entry(list.next,struct work_struct,entry); list_del(&work->entry); - __queue_work(per_cpu_ptr(wq->cpu_wq, smp_processor_id()), work); + __add_work_to_q(per_cpu_ptr(wq->cpu_wq, smp_processor_id()), + work); } spin_unlock_irq(&cwq->lock); } diff -puN net/sunrpc/sched.c~change-return-value-from-queue_work net/sunrpc/sched.c --- a/net/sunrpc/sched.c~change-return-value-from-queue_work +++ a/net/sunrpc/sched.c @@ -306,7 +306,7 @@ static void rpc_make_runnable(struct rpc int status; INIT_WORK(&task->u.tk_work, rpc_async_schedule, (void *)task); - status = queue_work(task->tk_workqueue, &task->u.tk_work); + status = add_work_to_q(task->tk_workqueue, &task->u.tk_work); if (status < 0) { printk(KERN_WARNING "RPC: failed to add task to queue: error: %d!\n", status); task->tk_status = status; _ Patches currently in -mm which might be from stern@xxxxxxxxxxxxxxxxxxx are usb-hub-driver-improve-use-of-ifdef-fix.patch block-layer-early-detection-of-medium-not-present.patch scsi-core-and-sd-early-detection-of-medium-not-present.patch sd-early-detection-of-medium-not-present.patch scsi-early-detection-of-medium-not-present-updated.patch change-return-value-from-queue_work.patch change-return-value-from-schedule_work.patch add-srcu-based-notifier-chains.patch srcu-report-out-of-memory-errors.patch srcu-report-out-of-memory-errors-fixlet.patch cpufreq-make-the-transition_notifier-chain-use-srcu.patch - To unsubscribe from this list: send the line "unsubscribe mm-commits" in the body of a message to majordomo@xxxxxxxxxxxxxxx More majordomo info at http://vger.kernel.org/majordomo-info.html