aboutsummaryrefslogtreecommitdiffstats
path: root/fs/sync.c
diff options
context:
space:
mode:
Diffstat (limited to 'fs/sync.c')
-rw-r--r--fs/sync.c85
1 files changed, 85 insertions, 0 deletions
diff --git a/fs/sync.c b/fs/sync.c
index be0798cc33d..d5fa7b79982 100644
--- a/fs/sync.c
+++ b/fs/sync.c
@@ -18,6 +18,91 @@
18#define VALID_FLAGS (SYNC_FILE_RANGE_WAIT_BEFORE|SYNC_FILE_RANGE_WRITE| \ 18#define VALID_FLAGS (SYNC_FILE_RANGE_WAIT_BEFORE|SYNC_FILE_RANGE_WRITE| \
19 SYNC_FILE_RANGE_WAIT_AFTER) 19 SYNC_FILE_RANGE_WAIT_AFTER)
20 20
21/*
22 * Do the filesystem syncing work. For simple filesystems sync_inodes_sb(sb, 0)
23 * just dirties buffers with inodes so we have to submit IO for these buffers
24 * via __sync_blockdev(). This also speeds up the wait == 1 case since in that
25 * case write_inode() functions do sync_dirty_buffer() and thus effectively
26 * write one block at a time.
27 */
28static int __fsync_super(struct super_block *sb, int wait)
29{
30 vfs_dq_sync(sb);
31 sync_inodes_sb(sb, wait);
32 lock_super(sb);
33 if (sb->s_dirt && sb->s_op->write_super)
34 sb->s_op->write_super(sb);
35 unlock_super(sb);
36 if (sb->s_op->sync_fs)
37 sb->s_op->sync_fs(sb, wait);
38 return __sync_blockdev(sb->s_bdev, wait);
39}
40
41/*
42 * Write out and wait upon all dirty data associated with this
43 * superblock. Filesystem data as well as the underlying block
44 * device. Takes the superblock lock.
45 */
46int fsync_super(struct super_block *sb)
47{
48 int ret;
49
50 ret = __fsync_super(sb, 0);
51 if (ret < 0)
52 return ret;
53 return __fsync_super(sb, 1);
54}
55EXPORT_SYMBOL_GPL(fsync_super);
56
57/*
58 * Sync all the data for all the filesystems (called by sys_sync() and
59 * emergency sync)
60 *
61 * This operation is careful to avoid the livelock which could easily happen
62 * if two or more filesystems are being continuously dirtied. s_need_sync
63 * is used only here. We set it against all filesystems and then clear it as
64 * we sync them. So redirtied filesystems are skipped.
65 *
66 * But if process A is currently running sync_filesystems and then process B
67 * calls sync_filesystems as well, process B will set all the s_need_sync
68 * flags again, which will cause process A to resync everything. Fix that with
69 * a local mutex.
70 */
71static void sync_filesystems(int wait)
72{
73 struct super_block *sb;
74 static DEFINE_MUTEX(mutex);
75
76 mutex_lock(&mutex); /* Could be down_interruptible */
77 spin_lock(&sb_lock);
78 list_for_each_entry(sb, &super_blocks, s_list) {
79 if (sb->s_flags & MS_RDONLY)
80 continue;
81 sb->s_need_sync = 1;
82 }
83
84restart:
85 list_for_each_entry(sb, &super_blocks, s_list) {
86 if (!sb->s_need_sync)
87 continue;
88 sb->s_need_sync = 0;
89 if (sb->s_flags & MS_RDONLY)
90 continue; /* hm. Was remounted r/o meanwhile */
91 sb->s_count++;
92 spin_unlock(&sb_lock);
93 down_read(&sb->s_umount);
94 if (sb->s_root)
95 __fsync_super(sb, wait);
96 up_read(&sb->s_umount);
97 /* restart only when sb is no longer on the list */
98 spin_lock(&sb_lock);
99 if (__put_super_and_need_restart(sb))
100 goto restart;
101 }
102 spin_unlock(&sb_lock);
103 mutex_unlock(&mutex);
104}
105
21SYSCALL_DEFINE0(sync) 106SYSCALL_DEFINE0(sync)
22{ 107{
23 sync_filesystems(0); 108 sync_filesystems(0);