diff options
Diffstat (limited to 'fs/xfs/xfs_vfsops.c')
-rw-r--r-- | fs/xfs/xfs_vfsops.c | 60 |
1 files changed, 35 insertions, 25 deletions
diff --git a/fs/xfs/xfs_vfsops.c b/fs/xfs/xfs_vfsops.c index 103500498342..42bcc0215203 100644 --- a/fs/xfs/xfs_vfsops.c +++ b/fs/xfs/xfs_vfsops.c | |||
@@ -612,7 +612,34 @@ out: | |||
612 | return XFS_ERROR(error); | 612 | return XFS_ERROR(error); |
613 | } | 613 | } |
614 | 614 | ||
615 | #define REMOUNT_READONLY_FLAGS (SYNC_REMOUNT|SYNC_ATTR|SYNC_WAIT) | 615 | STATIC int |
616 | xfs_quiesce_fs( | ||
617 | xfs_mount_t *mp) | ||
618 | { | ||
619 | int count = 0, pincount; | ||
620 | |||
621 | xfs_refcache_purge_mp(mp); | ||
622 | xfs_flush_buftarg(mp->m_ddev_targp, 0); | ||
623 | xfs_finish_reclaim_all(mp, 0); | ||
624 | |||
625 | /* This loop must run at least twice. | ||
626 | * The first instance of the loop will flush | ||
627 | * most meta data but that will generate more | ||
628 | * meta data (typically directory updates). | ||
629 | * Which then must be flushed and logged before | ||
630 | * we can write the unmount record. | ||
631 | */ | ||
632 | do { | ||
633 | xfs_syncsub(mp, SYNC_REMOUNT|SYNC_ATTR|SYNC_WAIT, 0, NULL); | ||
634 | pincount = xfs_flush_buftarg(mp->m_ddev_targp, 1); | ||
635 | if (!pincount) { | ||
636 | delay(50); | ||
637 | count++; | ||
638 | } | ||
639 | } while (count < 2); | ||
640 | |||
641 | return 0; | ||
642 | } | ||
616 | 643 | ||
617 | STATIC int | 644 | STATIC int |
618 | xfs_mntupdate( | 645 | xfs_mntupdate( |
@@ -622,8 +649,7 @@ xfs_mntupdate( | |||
622 | { | 649 | { |
623 | struct vfs *vfsp = bhvtovfs(bdp); | 650 | struct vfs *vfsp = bhvtovfs(bdp); |
624 | xfs_mount_t *mp = XFS_BHVTOM(bdp); | 651 | xfs_mount_t *mp = XFS_BHVTOM(bdp); |
625 | int pincount, error; | 652 | int error; |
626 | int count = 0; | ||
627 | 653 | ||
628 | if (args->flags & XFSMNT_NOATIME) | 654 | if (args->flags & XFSMNT_NOATIME) |
629 | mp->m_flags |= XFS_MOUNT_NOATIME; | 655 | mp->m_flags |= XFS_MOUNT_NOATIME; |
@@ -635,25 +661,7 @@ xfs_mntupdate( | |||
635 | } | 661 | } |
636 | 662 | ||
637 | if (*flags & MS_RDONLY) { | 663 | if (*flags & MS_RDONLY) { |
638 | xfs_refcache_purge_mp(mp); | 664 | xfs_quiesce_fs(mp); |
639 | xfs_flush_buftarg(mp->m_ddev_targp, 0); | ||
640 | xfs_finish_reclaim_all(mp, 0); | ||
641 | |||
642 | /* This loop must run at least twice. | ||
643 | * The first instance of the loop will flush | ||
644 | * most meta data but that will generate more | ||
645 | * meta data (typically directory updates). | ||
646 | * Which then must be flushed and logged before | ||
647 | * we can write the unmount record. | ||
648 | */ | ||
649 | do { | ||
650 | VFS_SYNC(vfsp, REMOUNT_READONLY_FLAGS, NULL, error); | ||
651 | pincount = xfs_flush_buftarg(mp->m_ddev_targp, 1); | ||
652 | if (!pincount) { | ||
653 | delay(50); | ||
654 | count++; | ||
655 | } | ||
656 | } while (count < 2); | ||
657 | 665 | ||
658 | /* Ok now write out an unmount record */ | 666 | /* Ok now write out an unmount record */ |
659 | xfs_log_unmount_write(mp); | 667 | xfs_log_unmount_write(mp); |
@@ -869,10 +877,12 @@ xfs_sync( | |||
869 | int flags, | 877 | int flags, |
870 | cred_t *credp) | 878 | cred_t *credp) |
871 | { | 879 | { |
872 | xfs_mount_t *mp; | 880 | xfs_mount_t *mp = XFS_BHVTOM(bdp); |
873 | 881 | ||
874 | mp = XFS_BHVTOM(bdp); | 882 | if (unlikely(flags == SYNC_QUIESCE)) |
875 | return (xfs_syncsub(mp, flags, 0, NULL)); | 883 | return xfs_quiesce_fs(mp); |
884 | else | ||
885 | return xfs_syncsub(mp, flags, 0, NULL); | ||
876 | } | 886 | } |
877 | 887 | ||
878 | /* | 888 | /* |