Re: [PATCH 2/8] ceph: single workqueue for inode related works

[Date Prev][Date Next][Thread Prev][Thread Next][Date Index][Thread Index]

 



On Tue, May 28, 2019 at 10:27 PM Jeff Layton <jlayton@xxxxxxxxxx> wrote:
>
> On Thu, 2019-05-23 at 16:13 +0800, Yan, Zheng wrote:
> > We have three workqueue for inode works. Later patch will introduce
> > one more work for inode. It's not good to introcuce more workqueue
> > and add more 'struct work_struct' to 'struct ceph_inode_info'.
> >
> > Signed-off-by: "Yan, Zheng" <zyan@xxxxxxxxxx>
> > ---
> >  fs/ceph/file.c  |   2 +-
> >  fs/ceph/inode.c | 124 ++++++++++++++++++++++--------------------------
> >  fs/ceph/super.c |  28 +++--------
> >  fs/ceph/super.h |  17 ++++---
> >  4 files changed, 74 insertions(+), 97 deletions(-)
> >
> > diff --git a/fs/ceph/file.c b/fs/ceph/file.c
> > index ccc054794542..b7be02dfb897 100644
> > --- a/fs/ceph/file.c
> > +++ b/fs/ceph/file.c
> > @@ -790,7 +790,7 @@ static void ceph_aio_complete_req(struct ceph_osd_request *req)
> >               if (aio_work) {
> >                       INIT_WORK(&aio_work->work, ceph_aio_retry_work);
> >                       aio_work->req = req;
> > -                     queue_work(ceph_inode_to_client(inode)->wb_wq,
> > +                     queue_work(ceph_inode_to_client(inode)->inode_wq,
> >                                  &aio_work->work);
> >                       return;
> >               }
> > diff --git a/fs/ceph/inode.c b/fs/ceph/inode.c
> > index 6eabcdb321cb..d9ff349821f0 100644
> > --- a/fs/ceph/inode.c
> > +++ b/fs/ceph/inode.c
> > @@ -33,9 +33,7 @@
> >
> >  static const struct inode_operations ceph_symlink_iops;
> >
> > -static void ceph_invalidate_work(struct work_struct *work);
> > -static void ceph_writeback_work(struct work_struct *work);
> > -static void ceph_vmtruncate_work(struct work_struct *work);
> > +static void ceph_inode_work(struct work_struct *work);
> >
> >  /*
> >   * find or create an inode, given the ceph ino number
> > @@ -509,10 +507,8 @@ struct inode *ceph_alloc_inode(struct super_block *sb)
> >       INIT_LIST_HEAD(&ci->i_snap_realm_item);
> >       INIT_LIST_HEAD(&ci->i_snap_flush_item);
> >
> > -     INIT_WORK(&ci->i_wb_work, ceph_writeback_work);
> > -     INIT_WORK(&ci->i_pg_inv_work, ceph_invalidate_work);
> > -
> > -     INIT_WORK(&ci->i_vmtruncate_work, ceph_vmtruncate_work);
> > +     INIT_WORK(&ci->i_work, ceph_inode_work);
> > +     ci->i_work_mask = 0;
> >
> >       ceph_fscache_inode_init(ci);
> >
> > @@ -1750,51 +1746,62 @@ bool ceph_inode_set_size(struct inode *inode, loff_t size)
> >   */
> >  void ceph_queue_writeback(struct inode *inode)
> >  {
> > +     struct ceph_inode_info *ci = ceph_inode(inode);
> > +     set_bit(CEPH_I_WORK_WRITEBACK, &ci->i_work_mask);
> > +
> >       ihold(inode);
> > -     if (queue_work(ceph_inode_to_client(inode)->wb_wq,
> > -                    &ceph_inode(inode)->i_wb_work)) {
> > +     if (queue_work(ceph_inode_to_client(inode)->inode_wq,
> > +                    &ci->i_work)) {
> >               dout("ceph_queue_writeback %p\n", inode);
> >       } else {
> > -             dout("ceph_queue_writeback %p failed\n", inode);
> > +             dout("ceph_queue_writeback %p already queued, mask=%lx\n",
> > +                  inode, ci->i_work_mask);
> >               iput(inode);
> >       }
> >  }
> >
> > -static void ceph_writeback_work(struct work_struct *work)
> > -{
> > -     struct ceph_inode_info *ci = container_of(work, struct ceph_inode_info,
> > -                                               i_wb_work);
> > -     struct inode *inode = &ci->vfs_inode;
> > -
> > -     dout("writeback %p\n", inode);
> > -     filemap_fdatawrite(&inode->i_data);
> > -     iput(inode);
> > -}
> > -
> >  /*
> >   * queue an async invalidation
> >   */
> >  void ceph_queue_invalidate(struct inode *inode)
> >  {
> > +     struct ceph_inode_info *ci = ceph_inode(inode);
> > +     set_bit(CEPH_I_WORK_INVALIDATE_PAGES, &ci->i_work_mask);
> > +
> >       ihold(inode);
> > -     if (queue_work(ceph_inode_to_client(inode)->pg_inv_wq,
> > -                    &ceph_inode(inode)->i_pg_inv_work)) {
> > +     if (queue_work(ceph_inode_to_client(inode)->inode_wq,
> > +                    &ceph_inode(inode)->i_work)) {
> >               dout("ceph_queue_invalidate %p\n", inode);
> >       } else {
> > -             dout("ceph_queue_invalidate %p failed\n", inode);
> > +             dout("ceph_queue_invalidate %p already queued, mask=%lx\n",
> > +                  inode, ci->i_work_mask);
> >               iput(inode);
> >       }
> >  }
> >
> >  /*
> > - * Invalidate inode pages in a worker thread.  (This can't be done
> > - * in the message handler context.)
> > + * Queue an async vmtruncate.  If we fail to queue work, we will handle
> > + * the truncation the next time we call __ceph_do_pending_vmtruncate.
> >   */
> > -static void ceph_invalidate_work(struct work_struct *work)
> > +void ceph_queue_vmtruncate(struct inode *inode)
> >  {
> > -     struct ceph_inode_info *ci = container_of(work, struct ceph_inode_info,
> > -                                               i_pg_inv_work);
> > -     struct inode *inode = &ci->vfs_inode;
> > +     struct ceph_inode_info *ci = ceph_inode(inode);
> > +     set_bit(CEPH_I_WORK_VMTRUNCATE, &ci->i_work_mask);
> > +
> > +     ihold(inode);
> > +     if (queue_work(ceph_inode_to_client(inode)->inode_wq,
> > +                    &ci->i_work)) {
> > +             dout("ceph_queue_vmtruncate %p\n", inode);
> > +     } else {
> > +             dout("ceph_queue_vmtruncate %p already queued, mask=%lx\n",
> > +                  inode, ci->i_work_mask);
> > +             iput(inode);
> > +     }
> > +}
> > +
> > +static void ceph_do_invalidate_pages(struct inode *inode)
> > +{
> > +     struct ceph_inode_info *ci = ceph_inode(inode);
> >       struct ceph_fs_client *fsc = ceph_inode_to_client(inode);
> >       u32 orig_gen;
> >       int check = 0;
> > @@ -1846,44 +1853,6 @@ static void ceph_invalidate_work(struct work_struct *work)
> >  out:
> >       if (check)
> >               ceph_check_caps(ci, 0, NULL);
> > -     iput(inode);
> > -}
> > -
> > -
> > -/*
> > - * called by trunc_wq;
> > - *
> > - * We also truncate in a separate thread as well.
> > - */
> > -static void ceph_vmtruncate_work(struct work_struct *work)
> > -{
> > -     struct ceph_inode_info *ci = container_of(work, struct ceph_inode_info,
> > -                                               i_vmtruncate_work);
> > -     struct inode *inode = &ci->vfs_inode;
> > -
> > -     dout("vmtruncate_work %p\n", inode);
> > -     __ceph_do_pending_vmtruncate(inode);
> > -     iput(inode);
> > -}
> > -
> > -/*
> > - * Queue an async vmtruncate.  If we fail to queue work, we will handle
> > - * the truncation the next time we call __ceph_do_pending_vmtruncate.
> > - */
> > -void ceph_queue_vmtruncate(struct inode *inode)
> > -{
> > -     struct ceph_inode_info *ci = ceph_inode(inode);
> > -
> > -     ihold(inode);
> > -
> > -     if (queue_work(ceph_sb_to_client(inode->i_sb)->trunc_wq,
> > -                    &ci->i_vmtruncate_work)) {
> > -             dout("ceph_queue_vmtruncate %p\n", inode);
> > -     } else {I
> > -             dout("ceph_queue_vmtruncate %p failed, pending=%d\n",
> > -                  inode, ci->i_truncate_pending);
> > -             iput(inode);
> > -     }
> >  }
> >
> >  /*
> > @@ -1947,6 +1916,25 @@ void __ceph_do_pending_vmtruncate(struct inode *inode)
> >       wake_up_all(&ci->i_cap_wq);
> >  }
> >
> > +static void ceph_inode_work(struct work_struct *work)
> > +{
> > +     struct ceph_inode_info *ci = container_of(work, struct ceph_inode_info,
> > +                                              i_work);
> > +     struct inode *inode = &ci->vfs_inode;
> > +
> > +     if (test_and_clear_bit(CEPH_I_WORK_WRITEBACK, &ci->i_work_mask)) {
> > +             dout("writeback %p\n", inode);
> > +             filemap_fdatawrite(&inode->i_data);
> > +     }
> > +     if (test_and_clear_bit(CEPH_I_WORK_INVALIDATE_PAGES, &ci->i_work_mask))
> > +             ceph_do_invalidate_pages(inode);
> > +
> > +     if (test_and_clear_bit(CEPH_I_WORK_VMTRUNCATE, &ci->i_work_mask))
> > +             __ceph_do_pending_vmtruncate(inode);
> > +
> > +     iput(inode);
> > +}
> > +
>
> I like this overall.
>
> It _probably_ doesn't matter, but the above change means that these
> operations are now serialized for a particular inode, whereas they could
> run parallel before.
>

I had thought about it. I didn't see any reason they must be run parallel.

Regards
Yan, Zheng

> >  /*
> >   * symlinks
> >   */
> > diff --git a/fs/ceph/super.c b/fs/ceph/super.c
> > index afc4c5d008d4..b1ee41372e85 100644
> > --- a/fs/ceph/super.c
> > +++ b/fs/ceph/super.c
> > @@ -671,18 +671,12 @@ static struct ceph_fs_client *create_fs_client(struct ceph_mount_options *fsopt,
> >        * The number of concurrent works can be high but they don't need
> >        * to be processed in parallel, limit concurrency.
> >        */
> > -     fsc->wb_wq = alloc_workqueue("ceph-writeback", 0, 1);
> > -     if (!fsc->wb_wq)
> > +     fsc->inode_wq = alloc_workqueue("ceph-inode", WQ_UNBOUND, 0);
> > +     if (!fsc->inode_wq)
> >               goto fail_client;
> > -     fsc->pg_inv_wq = alloc_workqueue("ceph-pg-invalid", 0, 1);
> > -     if (!fsc->pg_inv_wq)
> > -             goto fail_wb_wq;
> > -     fsc->trunc_wq = alloc_workqueue("ceph-trunc", 0, 1);
> > -     if (!fsc->trunc_wq)
> > -             goto fail_pg_inv_wq;
> >       fsc->cap_wq = alloc_workqueue("ceph-cap", 0, 1);
> >       if (!fsc->cap_wq)
> > -             goto fail_trunc_wq;
> > +             goto fail_inode_wq;
> >
> >       /* set up mempools */
> >       err = -ENOMEM;
> > @@ -696,12 +690,8 @@ static struct ceph_fs_client *create_fs_client(struct ceph_mount_options *fsopt,
> >
> >  fail_cap_wq:
> >       destroy_workqueue(fsc->cap_wq);
> > -fail_trunc_wq:
> > -     destroy_workqueue(fsc->trunc_wq);
> > -fail_pg_inv_wq:
> > -     destroy_workqueue(fsc->pg_inv_wq);
> > -fail_wb_wq:
> > -     destroy_workqueue(fsc->wb_wq);
> > +fail_inode_wq:
> > +     destroy_workqueue(fsc->inode_wq);
> >  fail_client:
> >       ceph_destroy_client(fsc->client);
> >  fail:
> > @@ -714,9 +704,7 @@ static struct ceph_fs_client *create_fs_client(struct ceph_mount_options *fsopt,
> >
> >  static void flush_fs_workqueues(struct ceph_fs_client *fsc)
> >  {
> > -     flush_workqueue(fsc->wb_wq);
> > -     flush_workqueue(fsc->pg_inv_wq);
> > -     flush_workqueue(fsc->trunc_wq);
> > +     flush_workqueue(fsc->inode_wq);
> >       flush_workqueue(fsc->cap_wq);
> >  }
> >
> > @@ -724,9 +712,7 @@ static void destroy_fs_client(struct ceph_fs_client *fsc)
> >  {
> >       dout("destroy_fs_client %p\n", fsc);
> >
> > -     destroy_workqueue(fsc->wb_wq);
> > -     destroy_workqueue(fsc->pg_inv_wq);
> > -     destroy_workqueue(fsc->trunc_wq);
> > +     destroy_workqueue(fsc->inode_wq);
> >       destroy_workqueue(fsc->cap_wq);
> >
> >       mempool_destroy(fsc->wb_pagevec_pool);
> > diff --git a/fs/ceph/super.h b/fs/ceph/super.h
> > index c84135bb72c6..234610ce4155 100644
> > --- a/fs/ceph/super.h
> > +++ b/fs/ceph/super.h
> > @@ -109,9 +109,7 @@ struct ceph_fs_client {
> >       mempool_t *wb_pagevec_pool;
> >       atomic_long_t writeback_count;
> >
> > -     struct workqueue_struct *wb_wq;
> > -     struct workqueue_struct *pg_inv_wq;
> > -     struct workqueue_struct *trunc_wq;
> > +     struct workqueue_struct *inode_wq;
> >       struct workqueue_struct *cap_wq;
> >
> >  #ifdef CONFIG_DEBUG_FS
> > @@ -388,10 +386,8 @@ struct ceph_inode_info {
> >       struct list_head i_snap_flush_item;
> >       struct timespec64 i_snap_btime;
> >
> > -     struct work_struct i_wb_work;  /* writeback work */
> > -     struct work_struct i_pg_inv_work;  /* page invalidation work */
> > -
> > -     struct work_struct i_vmtruncate_work;
> > +     struct work_struct i_work;
> > +     unsigned long  i_work_mask;
> >
> >  #ifdef CONFIG_CEPH_FSCACHE
> >       struct fscache_cookie *fscache;
> > @@ -513,6 +509,13 @@ static inline struct inode *ceph_find_inode(struct super_block *sb,
> >  #define CEPH_I_ERROR_FILELOCK        (1 << 12) /* have seen file lock errors */
> >
> >
> > +/*
> > + * Masks of ceph inode work.
> > + */
> > +#define CEPH_I_WORK_WRITEBACK                0 /* writeback */
> > +#define CEPH_I_WORK_INVALIDATE_PAGES 1 /* invalidate pages */
> > +#define CEPH_I_WORK_VMTRUNCATE               2 /* vmtruncate */
> > +
> >  /*
> >   * We set the ERROR_WRITE bit when we start seeing write errors on an inode
> >   * and then clear it when they start succeeding. Note that we do a lockless
>
> Reviewed-by: Jeff Layton <jlayton@xxxxxxxxxx>
>



[Index of Archives]     [CEPH Users]     [Ceph Large]     [Information on CEPH]     [Linux BTRFS]     [Linux USB Devel]     [Video for Linux]     [Linux Audio Users]     [Yosemite News]     [Linux Kernel]     [Linux SCSI]

  Powered by Linux