On Tue 18-11-14 03:37:22, Tejun Heo wrote:
> Currently, a bdi (backing_dev_info) embeds single wb (bdi_writeback)
> and the role of the separation is unclear.  For cgroup support for
> writeback IOs, a bdi will be updated to host multiple wb's where each
> wb serves writeback IOs of a different cgroup on the bdi.  To achieve
> that, a wb should carry all states necessary for servicing writeback
> IOs for a cgroup independently.
> 
> This patch moves bdi->wb_lock and ->worklist into wb.
> 
> * The lock protects bdi->worklist and bdi->wb.dwork scheduling.  While
>   moving, rename it to wb->work_lock as wb->wb_lock is confusing.
>   Also, move wb->dwork downwards so that it's colocated with the new
>   ->work_lock and ->work_list fields.
> 
> * bdi_writeback_workfn()              -> wb_workfn()
>   bdi_wakeup_thread_delayed(bdi)      -> wb_wakeup_delayed(wb)
>   bdi_wakeup_thread(bdi)              -> wb_wakeup(wb)
>   bdi_queue_work(bdi, ...)            -> wb_queue_work(wb, ...)
>   __bdi_start_writeback(bdi, ...)     -> __wb_start_writeback(wb, ...)
>   get_next_work_item(bdi)             -> get_next_work_item(wb)
> 
> * bdi_wb_shutdown() is renamed to wb_shutdown() and now takes @wb.
>   The function contained parts which belong to the containing bdi
>   rather than the wb itself - testing cap_writeback_dirty and
>   bdi_remove_from_list() invocation.  Those are moved to
>   bdi_unregister().
> 
> * bdi_wb_{init|exit}() are renamed to wb_{init|exit}().
>   Initializations of the moved bdi->wb_lock and ->work_list are
>   relocated from bdi_init() to wb_init().
> 
> * As there's still only one bdi_writeback per backing_dev_info, all
>   uses of bdi->state are mechanically replaced with bdi->wb.state
>   introducing no behavior changes.
> 
> Signed-off-by: Tejun Heo <[email protected]>
> Cc: Jens Axboe <[email protected]>
> Cc: Jan Kara <[email protected]>
> Cc: Wu Fengguang <[email protected]>
  Does this mean you want to have per-device, per-cgroup flusher workqueues?
Otherwise this change doesn't make sense...

                                                                Honza

> ---
>  fs/fs-writeback.c           | 84 
> +++++++++++++++++++++------------------------
>  include/linux/backing-dev.h | 12 +++----
>  mm/backing-dev.c            | 64 +++++++++++++++++-----------------
>  3 files changed, 77 insertions(+), 83 deletions(-)
> 
> diff --git a/fs/fs-writeback.c b/fs/fs-writeback.c
> index daa91ae..41c9f1e 100644
> --- a/fs/fs-writeback.c
> +++ b/fs/fs-writeback.c
> @@ -91,34 +91,33 @@ static inline struct inode *wb_inode(struct list_head 
> *head)
>  
>  EXPORT_TRACEPOINT_SYMBOL_GPL(wbc_writepage);
>  
> -static void bdi_wakeup_thread(struct backing_dev_info *bdi)
> +static void wb_wakeup(struct bdi_writeback *wb)
>  {
> -     spin_lock_bh(&bdi->wb_lock);
> -     if (test_bit(WB_registered, &bdi->wb.state))
> -             mod_delayed_work(bdi_wq, &bdi->wb.dwork, 0);
> -     spin_unlock_bh(&bdi->wb_lock);
> +     spin_lock_bh(&wb->work_lock);
> +     if (test_bit(WB_registered, &wb->state))
> +             mod_delayed_work(bdi_wq, &wb->dwork, 0);
> +     spin_unlock_bh(&wb->work_lock);
>  }
> -static void bdi_queue_work(struct backing_dev_info *bdi,
> -                        struct wb_writeback_work *work)
> +static void wb_queue_work(struct bdi_writeback *wb,
> +                       struct wb_writeback_work *work)
>  {
> -     trace_writeback_queue(bdi, work);
> +     trace_writeback_queue(wb->bdi, work);
>  
> -     spin_lock_bh(&bdi->wb_lock);
> -     if (!test_bit(WB_registered, &bdi->wb.state)) {
> +     spin_lock_bh(&wb->work_lock);
> +     if (!test_bit(WB_registered, &wb->state)) {
>               if (work->done)
>                       complete(work->done);
>               goto out_unlock;
>       }
> -     list_add_tail(&work->list, &bdi->work_list);
> -     mod_delayed_work(bdi_wq, &bdi->wb.dwork, 0);
> +     list_add_tail(&work->list, &wb->work_list);
> +     mod_delayed_work(bdi_wq, &wb->dwork, 0);
>  out_unlock:
> -     spin_unlock_bh(&bdi->wb_lock);
> +     spin_unlock_bh(&wb->work_lock);
>  }
>  
> -static void
> -__bdi_start_writeback(struct backing_dev_info *bdi, long nr_pages,
> -                   bool range_cyclic, enum wb_reason reason)
> +static void __wb_start_writeback(struct bdi_writeback *wb, long nr_pages,
> +                              bool range_cyclic, enum wb_reason reason)
>  {
>       struct wb_writeback_work *work;
>  
> @@ -128,8 +127,8 @@ __bdi_start_writeback(struct backing_dev_info *bdi, long 
> nr_pages,
>        */
>       work = kzalloc(sizeof(*work), GFP_ATOMIC);
>       if (!work) {
> -             trace_writeback_nowork(bdi);
> -             bdi_wakeup_thread(bdi);
> +             trace_writeback_nowork(wb->bdi);
> +             wb_wakeup(wb);
>               return;
>       }
>  
> @@ -138,7 +137,7 @@ __bdi_start_writeback(struct backing_dev_info *bdi, long 
> nr_pages,
>       work->range_cyclic = range_cyclic;
>       work->reason    = reason;
>  
> -     bdi_queue_work(bdi, work);
> +     wb_queue_work(wb, work);
>  }
>  
>  /**
> @@ -156,7 +155,7 @@ __bdi_start_writeback(struct backing_dev_info *bdi, long 
> nr_pages,
>  void bdi_start_writeback(struct backing_dev_info *bdi, long nr_pages,
>                       enum wb_reason reason)
>  {
> -     __bdi_start_writeback(bdi, nr_pages, true, reason);
> +     __wb_start_writeback(&bdi->wb, nr_pages, true, reason);
>  }
>  
>  /**
> @@ -176,7 +175,7 @@ void bdi_start_background_writeback(struct 
> backing_dev_info *bdi)
>        * writeback as soon as there is no other work to do.
>        */
>       trace_writeback_wake_background(bdi);
> -     bdi_wakeup_thread(bdi);
> +     wb_wakeup(&bdi->wb);
>  }
>  
>  /*
> @@ -848,7 +847,7 @@ static long wb_writeback(struct bdi_writeback *wb,
>                * after the other works are all done.
>                */
>               if ((work->for_background || work->for_kupdate) &&
> -                 !list_empty(&wb->bdi->work_list))
> +                 !list_empty(&wb->work_list))
>                       break;
>  
>               /*
> @@ -919,18 +918,17 @@ static long wb_writeback(struct bdi_writeback *wb,
>  /*
>   * Return the next wb_writeback_work struct that hasn't been processed yet.
>   */
> -static struct wb_writeback_work *
> -get_next_work_item(struct backing_dev_info *bdi)
> +static struct wb_writeback_work *get_next_work_item(struct bdi_writeback *wb)
>  {
>       struct wb_writeback_work *work = NULL;
>  
> -     spin_lock_bh(&bdi->wb_lock);
> -     if (!list_empty(&bdi->work_list)) {
> -             work = list_entry(bdi->work_list.next,
> +     spin_lock_bh(&wb->work_lock);
> +     if (!list_empty(&wb->work_list)) {
> +             work = list_entry(wb->work_list.next,
>                                 struct wb_writeback_work, list);
>               list_del_init(&work->list);
>       }
> -     spin_unlock_bh(&bdi->wb_lock);
> +     spin_unlock_bh(&wb->work_lock);
>       return work;
>  }
>  
> @@ -1002,14 +1000,13 @@ static long wb_check_old_data_flush(struct 
> bdi_writeback *wb)
>   */
>  static long wb_do_writeback(struct bdi_writeback *wb)
>  {
> -     struct backing_dev_info *bdi = wb->bdi;
>       struct wb_writeback_work *work;
>       long wrote = 0;
>  
>       set_bit(WB_writeback_running, &wb->state);
> -     while ((work = get_next_work_item(bdi)) != NULL) {
> +     while ((work = get_next_work_item(wb)) != NULL) {
>  
> -             trace_writeback_exec(bdi, work);
> +             trace_writeback_exec(wb->bdi, work);
>  
>               wrote += wb_writeback(wb, work);
>  
> @@ -1037,43 +1034,42 @@ static long wb_do_writeback(struct bdi_writeback *wb)
>   * Handle writeback of dirty data for the device backed by this bdi. Also
>   * reschedules periodically and does kupdated style flushing.
>   */
> -void bdi_writeback_workfn(struct work_struct *work)
> +void wb_workfn(struct work_struct *work)
>  {
>       struct bdi_writeback *wb = container_of(to_delayed_work(work),
>                                               struct bdi_writeback, dwork);
> -     struct backing_dev_info *bdi = wb->bdi;
>       long pages_written;
>  
> -     set_worker_desc("flush-%s", dev_name(bdi->dev));
> +     set_worker_desc("flush-%s", dev_name(wb->bdi->dev));
>       current->flags |= PF_SWAPWRITE;
>  
>       if (likely(!current_is_workqueue_rescuer() ||
>                  !test_bit(WB_registered, &wb->state))) {
>               /*
> -              * The normal path.  Keep writing back @bdi until its
> +              * The normal path.  Keep writing back @wb until its
>                * work_list is empty.  Note that this path is also taken
> -              * if @bdi is shutting down even when we're running off the
> +              * if @wb is shutting down even when we're running off the
>                * rescuer as work_list needs to be drained.
>                */
>               do {
>                       pages_written = wb_do_writeback(wb);
>                       trace_writeback_pages_written(pages_written);
> -             } while (!list_empty(&bdi->work_list));
> +             } while (!list_empty(&wb->work_list));
>       } else {
>               /*
>                * bdi_wq can't get enough workers and we're running off
>                * the emergency worker.  Don't hog it.  Hopefully, 1024 is
>                * enough for efficient IO.
>                */
> -             pages_written = writeback_inodes_wb(&bdi->wb, 1024,
> +             pages_written = writeback_inodes_wb(wb, 1024,
>                                                   WB_REASON_FORKER_THREAD);
>               trace_writeback_pages_written(pages_written);
>       }
>  
> -     if (!list_empty(&bdi->work_list))
> +     if (!list_empty(&wb->work_list))
>               mod_delayed_work(bdi_wq, &wb->dwork, 0);
>       else if (wb_has_dirty_io(wb) && dirty_writeback_interval)
> -             bdi_wakeup_thread_delayed(bdi);
> +             wb_wakeup_delayed(wb);
>  
>       current->flags &= ~PF_SWAPWRITE;
>  }
> @@ -1093,7 +1089,7 @@ void wakeup_flusher_threads(long nr_pages, enum 
> wb_reason reason)
>       list_for_each_entry_rcu(bdi, &bdi_list, bdi_list) {
>               if (!bdi_has_dirty_io(bdi))
>                       continue;
> -             __bdi_start_writeback(bdi, nr_pages, false, reason);
> +             __wb_start_writeback(&bdi->wb, nr_pages, false, reason);
>       }
>       rcu_read_unlock();
>  }
> @@ -1228,7 +1224,7 @@ void __mark_inode_dirty(struct inode *inode, int flags)
>                       spin_unlock(&bdi->wb.list_lock);
>  
>                       if (wakeup_bdi)
> -                             bdi_wakeup_thread_delayed(bdi);
> +                             wb_wakeup_delayed(&bdi->wb);
>                       return;
>               }
>       }
> @@ -1318,7 +1314,7 @@ void writeback_inodes_sb_nr(struct super_block *sb,
>       if (sb->s_bdi == &noop_backing_dev_info)
>               return;
>       WARN_ON(!rwsem_is_locked(&sb->s_umount));
> -     bdi_queue_work(sb->s_bdi, &work);
> +     wb_queue_work(&sb->s_bdi->wb, &work);
>       wait_for_completion(&done);
>  }
>  EXPORT_SYMBOL(writeback_inodes_sb_nr);
> @@ -1402,7 +1398,7 @@ void sync_inodes_sb(struct super_block *sb)
>               return;
>       WARN_ON(!rwsem_is_locked(&sb->s_umount));
>  
> -     bdi_queue_work(sb->s_bdi, &work);
> +     wb_queue_work(&sb->s_bdi->wb, &work);
>       wait_for_completion(&done);
>  
>       wait_sb_inodes(sb);
> diff --git a/include/linux/backing-dev.h b/include/linux/backing-dev.h
> index a077a8d..6aba0d3 100644
> --- a/include/linux/backing-dev.h
> +++ b/include/linux/backing-dev.h
> @@ -52,7 +52,6 @@ struct bdi_writeback {
>       unsigned long state;            /* Always use atomic bitops on this */
>       unsigned long last_old_flush;   /* last old data flush */
>  
> -     struct delayed_work dwork;      /* work item used for writeback */
>       struct list_head b_dirty;       /* dirty inodes */
>       struct list_head b_io;          /* parked for writeback */
>       struct list_head b_more_io;     /* parked for more writeback */
> @@ -77,6 +76,10 @@ struct bdi_writeback {
>  
>       struct fprop_local_percpu completions;
>       int dirty_exceeded;
> +
> +     spinlock_t work_lock;           /* protects work_list & dwork 
> scheduling */
> +     struct list_head work_list;
> +     struct delayed_work dwork;      /* work item used for writeback */
>  };
>  
>  struct backing_dev_info {
> @@ -92,9 +95,6 @@ struct backing_dev_info {
>       unsigned int max_ratio, max_prop_frac;
>  
>       struct bdi_writeback wb;  /* default writeback info for this bdi */
> -     spinlock_t wb_lock;       /* protects work_list & wb.dwork scheduling */
> -
> -     struct list_head work_list;
>  
>       struct device *dev;
>  
> @@ -118,9 +118,9 @@ int __must_check bdi_setup_and_register(struct 
> backing_dev_info *, char *, unsig
>  void bdi_start_writeback(struct backing_dev_info *bdi, long nr_pages,
>                       enum wb_reason reason);
>  void bdi_start_background_writeback(struct backing_dev_info *bdi);
> -void bdi_writeback_workfn(struct work_struct *work);
> +void wb_workfn(struct work_struct *work);
>  int bdi_has_dirty_io(struct backing_dev_info *bdi);
> -void bdi_wakeup_thread_delayed(struct backing_dev_info *bdi);
> +void wb_wakeup_delayed(struct bdi_writeback *wb);
>  
>  extern spinlock_t bdi_lock;
>  extern struct list_head bdi_list;
> diff --git a/mm/backing-dev.c b/mm/backing-dev.c
> index 7b9b10e..4904456 100644
> --- a/mm/backing-dev.c
> +++ b/mm/backing-dev.c
> @@ -278,7 +278,7 @@ int bdi_has_dirty_io(struct backing_dev_info *bdi)
>  }
>  
>  /*
> - * This function is used when the first inode for this bdi is marked dirty. 
> It
> + * This function is used when the first inode for this wb is marked dirty. It
>   * wakes-up the corresponding bdi thread which should then take care of the
>   * periodic background write-out of dirty inodes. Since the write-out would
>   * starts only 'dirty_writeback_interval' centisecs from now anyway, we just
> @@ -291,15 +291,15 @@ int bdi_has_dirty_io(struct backing_dev_info *bdi)
>   * We have to be careful not to postpone flush work if it is scheduled for
>   * earlier. Thus we use queue_delayed_work().
>   */
> -void bdi_wakeup_thread_delayed(struct backing_dev_info *bdi)
> +void wb_wakeup_delayed(struct bdi_writeback *wb)
>  {
>       unsigned long timeout;
>  
>       timeout = msecs_to_jiffies(dirty_writeback_interval * 10);
> -     spin_lock_bh(&bdi->wb_lock);
> -     if (test_bit(WB_registered, &bdi->wb.state))
> -             queue_delayed_work(bdi_wq, &bdi->wb.dwork, timeout);
> -     spin_unlock_bh(&bdi->wb_lock);
> +     spin_lock_bh(&wb->work_lock);
> +     if (test_bit(WB_registered, &wb->state))
> +             queue_delayed_work(bdi_wq, &wb->dwork, timeout);
> +     spin_unlock_bh(&wb->work_lock);
>  }
>  
>  /*
> @@ -352,30 +352,22 @@ EXPORT_SYMBOL(bdi_register_dev);
>  /*
>   * Remove bdi from the global list and shutdown any threads we have running
>   */
> -static void bdi_wb_shutdown(struct backing_dev_info *bdi)
> +static void wb_shutdown(struct bdi_writeback *wb)
>  {
> -     if (!bdi_cap_writeback_dirty(bdi))
> -             return;
> -
> -     /*
> -      * Make sure nobody finds us on the bdi_list anymore
> -      */
> -     bdi_remove_from_list(bdi);
> -
>       /* Make sure nobody queues further work */
> -     spin_lock_bh(&bdi->wb_lock);
> -     clear_bit(WB_registered, &bdi->wb.state);
> -     spin_unlock_bh(&bdi->wb_lock);
> +     spin_lock_bh(&wb->work_lock);
> +     clear_bit(WB_registered, &wb->state);
> +     spin_unlock_bh(&wb->work_lock);
>  
>       /*
> -      * Drain work list and shutdown the delayed_work.  At this point,
> -      * @bdi->bdi_list is empty telling bdi_Writeback_workfn() that @bdi
> -      * is dying and its work_list needs to be drained no matter what.
> +      * Drain work list and shutdown the delayed_work.  !WB_registered
> +      * tells wb_workfn() that @wb is dying and its work_list needs to
> +      * be drained no matter what.
>        */
> -     mod_delayed_work(bdi_wq, &bdi->wb.dwork, 0);
> -     flush_delayed_work(&bdi->wb.dwork);
> -     WARN_ON(!list_empty(&bdi->work_list));
> -     WARN_ON(delayed_work_pending(&bdi->wb.dwork));
> +     mod_delayed_work(bdi_wq, &wb->dwork, 0);
> +     flush_delayed_work(&wb->dwork);
> +     WARN_ON(!list_empty(&wb->work_list));
> +     WARN_ON(delayed_work_pending(&wb->dwork));
>  }
>  
>  /*
> @@ -400,7 +392,12 @@ void bdi_unregister(struct backing_dev_info *bdi)
>               trace_writeback_bdi_unregister(bdi);
>               bdi_prune_sb(bdi);
>  
> -             bdi_wb_shutdown(bdi);
> +             if (bdi_cap_writeback_dirty(bdi)) {
> +                     /* make sure nobody finds us on the bdi_list anymore */
> +                     bdi_remove_from_list(bdi);
> +                     wb_shutdown(&bdi->wb);
> +             }
> +
>               bdi_debug_unregister(bdi);
>               device_unregister(bdi->dev);
>               bdi->dev = NULL;
> @@ -413,7 +410,7 @@ EXPORT_SYMBOL(bdi_unregister);
>   */
>  #define INIT_BW              (100 << (20 - PAGE_SHIFT))
>  
> -static int bdi_wb_init(struct bdi_writeback *wb, struct backing_dev_info 
> *bdi)
> +static int wb_init(struct bdi_writeback *wb, struct backing_dev_info *bdi)
>  {
>       int i, err;
>  
> @@ -425,7 +422,6 @@ static int bdi_wb_init(struct bdi_writeback *wb, struct 
> backing_dev_info *bdi)
>       INIT_LIST_HEAD(&wb->b_io);
>       INIT_LIST_HEAD(&wb->b_more_io);
>       spin_lock_init(&wb->list_lock);
> -     INIT_DELAYED_WORK(&wb->dwork, bdi_writeback_workfn);
>  
>       wb->bw_time_stamp = jiffies;
>       wb->balanced_dirty_ratelimit = INIT_BW;
> @@ -433,6 +429,10 @@ static int bdi_wb_init(struct bdi_writeback *wb, struct 
> backing_dev_info *bdi)
>       wb->write_bandwidth = INIT_BW;
>       wb->avg_write_bandwidth = INIT_BW;
>  
> +     spin_lock_init(&wb->work_lock);
> +     INIT_LIST_HEAD(&wb->work_list);
> +     INIT_DELAYED_WORK(&wb->dwork, wb_workfn);
> +
>       err = fprop_local_init_percpu(&wb->completions, GFP_KERNEL);
>       if (err)
>               return err;
> @@ -450,7 +450,7 @@ static int bdi_wb_init(struct bdi_writeback *wb, struct 
> backing_dev_info *bdi)
>       return 0;
>  }
>  
> -static void bdi_wb_exit(struct bdi_writeback *wb)
> +static void wb_exit(struct bdi_writeback *wb)
>  {
>       int i;
>  
> @@ -471,11 +471,9 @@ int bdi_init(struct backing_dev_info *bdi)
>       bdi->min_ratio = 0;
>       bdi->max_ratio = 100;
>       bdi->max_prop_frac = FPROP_FRAC_BASE;
> -     spin_lock_init(&bdi->wb_lock);
>       INIT_LIST_HEAD(&bdi->bdi_list);
> -     INIT_LIST_HEAD(&bdi->work_list);
>  
> -     err = bdi_wb_init(&bdi->wb, bdi);
> +     err = wb_init(&bdi->wb, bdi);
>       if (err)
>               return err;
>  
> @@ -510,7 +508,7 @@ void bdi_destroy(struct backing_dev_info *bdi)
>       }
>  
>       bdi_unregister(bdi);
> -     bdi_wb_exit(&bdi->wb);
> +     wb_exit(&bdi->wb);
>  }
>  EXPORT_SYMBOL(bdi_destroy);
>  
> -- 
> 1.9.3
> 
-- 
Jan Kara <[email protected]>
SUSE Labs, CR
--
To unsubscribe from this list: send the line "unsubscribe linux-kernel" in
the body of a message to [email protected]
More majordomo info at  http://vger.kernel.org/majordomo-info.html
Please read the FAQ at  http://www.tux.org/lkml/

Reply via email to