On Wed 29-11-17 15:23:46, Luis R. Rodriguez wrote:
> freeze_super() holds a write lock, however we wish to also enable
> callers which already hold the write lock. To do this provide a helper
> and make freeze_super() use it. This way, all that freeze_super() does
> now is lock handling and active count management.
> 
> This change has no functional changes.
> 
> Suggested-by: Dave Chinner <da...@fromorbit.com>
> Signed-off-by: Luis R. Rodriguez <mcg...@kernel.org>

Looks good to me. You can add:

Reviewed-by: Jan Kara <j...@suse.cz>

                                                                Honza

> ---
>  fs/super.c | 100 
> +++++++++++++++++++++++++++++++++----------------------------
>  1 file changed, 55 insertions(+), 45 deletions(-)
> 
> diff --git a/fs/super.c b/fs/super.c
> index d4e33e8f1e6f..a7650ff22f0e 100644
> --- a/fs/super.c
> +++ b/fs/super.c
> @@ -1387,59 +1387,20 @@ static void sb_freeze_unlock(struct super_block *sb)
>               percpu_up_write(sb->s_writers.rw_sem + level);
>  }
>  
> -/**
> - * freeze_super - lock the filesystem and force it into a consistent state
> - * @sb: the super to lock
> - *
> - * Syncs the super to make sure the filesystem is consistent and calls the 
> fs's
> - * freeze_fs.  Subsequent calls to this without first thawing the fs will 
> return
> - * -EBUSY.
> - *
> - * During this function, sb->s_writers.frozen goes through these values:
> - *
> - * SB_UNFROZEN: File system is normal, all writes progress as usual.
> - *
> - * SB_FREEZE_WRITE: The file system is in the process of being frozen.  New
> - * writes should be blocked, though page faults are still allowed. We wait 
> for
> - * all writes to complete and then proceed to the next stage.
> - *
> - * SB_FREEZE_PAGEFAULT: Freezing continues. Now also page faults are blocked
> - * but internal fs threads can still modify the filesystem (although they
> - * should not dirty new pages or inodes), writeback can run etc. After 
> waiting
> - * for all running page faults we sync the filesystem which will clean all
> - * dirty pages and inodes (no new dirty pages or inodes can be created when
> - * sync is running).
> - *
> - * SB_FREEZE_FS: The file system is frozen. Now all internal sources of fs
> - * modification are blocked (e.g. XFS preallocation truncation on inode
> - * reclaim). This is usually implemented by blocking new transactions for
> - * filesystems that have them and need this additional guard. After all
> - * internal writers are finished we call ->freeze_fs() to finish filesystem
> - * freezing. Then we transition to SB_FREEZE_COMPLETE state. This state is
> - * mostly auxiliary for filesystems to verify they do not modify frozen fs.
> - *
> - * sb->s_writers.frozen is protected by sb->s_umount.
> - */
> -int freeze_super(struct super_block *sb)
> +/* Caller takes lock and handles active count */
> +static int freeze_locked_super(struct super_block *sb)
>  {
>       int ret;
>  
> -     atomic_inc(&sb->s_active);
> -     down_write(&sb->s_umount);
> -     if (sb->s_writers.frozen != SB_UNFROZEN) {
> -             deactivate_locked_super(sb);
> +     if (sb->s_writers.frozen != SB_UNFROZEN)
>               return -EBUSY;
> -     }
>  
> -     if (!(sb->s_flags & SB_BORN)) {
> -             up_write(&sb->s_umount);
> +     if (!(sb->s_flags & SB_BORN))
>               return 0;       /* sic - it's "nothing to do" */
> -     }
>  
>       if (sb_rdonly(sb)) {
>               /* Nothing to do really... */
>               sb->s_writers.frozen = SB_FREEZE_COMPLETE;
> -             up_write(&sb->s_umount);
>               return 0;
>       }
>  
> @@ -1468,7 +1429,6 @@ int freeze_super(struct super_block *sb)
>                       sb->s_writers.frozen = SB_UNFROZEN;
>                       sb_freeze_unlock(sb);
>                       wake_up(&sb->s_writers.wait_unfrozen);
> -                     deactivate_locked_super(sb);
>                       return ret;
>               }
>       }
> @@ -1478,9 +1438,59 @@ int freeze_super(struct super_block *sb)
>        */
>       sb->s_writers.frozen = SB_FREEZE_COMPLETE;
>       lockdep_sb_freeze_release(sb);
> -     up_write(&sb->s_umount);
>       return 0;
>  }
> +
> +/**
> + * freeze_super - lock the filesystem and force it into a consistent state
> + * @sb: the super to lock
> + *
> + * Syncs the super to make sure the filesystem is consistent and calls the 
> fs's
> + * freeze_fs.  Subsequent calls to this without first thawing the fs will 
> return
> + * -EBUSY.
> + *
> + * During this function, sb->s_writers.frozen goes through these values:
> + *
> + * SB_UNFROZEN: File system is normal, all writes progress as usual.
> + *
> + * SB_FREEZE_WRITE: The file system is in the process of being frozen.  New
> + * writes should be blocked, though page faults are still allowed. We wait 
> for
> + * all writes to complete and then proceed to the next stage.
> + *
> + * SB_FREEZE_PAGEFAULT: Freezing continues. Now also page faults are blocked
> + * but internal fs threads can still modify the filesystem (although they
> + * should not dirty new pages or inodes), writeback can run etc. After 
> waiting
> + * for all running page faults we sync the filesystem which will clean all
> + * dirty pages and inodes (no new dirty pages or inodes can be created when
> + * sync is running).
> + *
> + * SB_FREEZE_FS: The file system is frozen. Now all internal sources of fs
> + * modification are blocked (e.g. XFS preallocation truncation on inode
> + * reclaim). This is usually implemented by blocking new transactions for
> + * filesystems that have them and need this additional guard. After all
> + * internal writers are finished we call ->freeze_fs() to finish filesystem
> + * freezing. Then we transition to SB_FREEZE_COMPLETE state. This state is
> + * mostly auxiliary for filesystems to verify they do not modify frozen fs.
> + *
> + * sb->s_writers.frozen is protected by sb->s_umount.
> + */
> +int freeze_super(struct super_block *sb)
> +{
> +     int error;
> +
> +     atomic_inc(&sb->s_active);
> +
> +     down_write(&sb->s_umount);
> +     error = freeze_locked_super(sb);
> +     if (error) {
> +             deactivate_locked_super(sb);
> +             goto out;
> +     }
> +     up_write(&sb->s_umount);
> +
> +out:
> +     return error;
> +}
>  EXPORT_SYMBOL(freeze_super);
>  
>  /**
> -- 
> 2.15.0
> 
-- 
Jan Kara <j...@suse.com>
SUSE Labs, CR

Reply via email to