aboutsummaryrefslogtreecommitdiffstats
path: root/fs/ext4
diff options
context:
space:
mode:
authorTheodore Ts'o <tytso@mit.edu>2014-03-24 14:43:12 -0400
committerTheodore Ts'o <tytso@mit.edu>2014-03-24 14:43:12 -0400
commit5f16f3225b06242a9ee876f07c1c9b6ed36a22b6 (patch)
treecebecf399c2d9ef5b0a24902638cb37d2e9e092b /fs/ext4
parented3654eb981fd44694b4d2a636e13f998bc10e7f (diff)
ext4: atomically set inode->i_flags in ext4_set_inode_flags()
Use cmpxchg() to atomically set i_flags instead of clearing out the S_IMMUTABLE, S_APPEND, etc. flags and then setting them from the EXT4_IMMUTABLE_FL, EXT4_APPEND_FL flags, since this opens up a race where an immutable file has the immutable flag cleared for a brief window of time. Reported-by: John Sullivan <jsrhbz@kanargh.force9.co.uk> Signed-off-by: "Theodore Ts'o" <tytso@mit.edu> Cc: stable@kernel.org
Diffstat (limited to 'fs/ext4')
-rw-r--r--fs/ext4/inode.c14
1 files changed, 8 insertions, 6 deletions
diff --git a/fs/ext4/inode.c b/fs/ext4/inode.c
index b5e182acf9b9..df067c3c6c93 100644
--- a/fs/ext4/inode.c
+++ b/fs/ext4/inode.c
@@ -3938,18 +3938,20 @@ int ext4_get_inode_loc(struct inode *inode, struct ext4_iloc *iloc)
3938void ext4_set_inode_flags(struct inode *inode) 3938void ext4_set_inode_flags(struct inode *inode)
3939{ 3939{
3940 unsigned int flags = EXT4_I(inode)->i_flags; 3940 unsigned int flags = EXT4_I(inode)->i_flags;
3941 unsigned int new_fl = 0;
3941 3942
3942 inode->i_flags &= ~(S_SYNC|S_APPEND|S_IMMUTABLE|S_NOATIME|S_DIRSYNC);
3943 if (flags & EXT4_SYNC_FL) 3943 if (flags & EXT4_SYNC_FL)
3944 inode->i_flags |= S_SYNC; 3944 new_fl |= S_SYNC;
3945 if (flags & EXT4_APPEND_FL) 3945 if (flags & EXT4_APPEND_FL)
3946 inode->i_flags |= S_APPEND; 3946 new_fl |= S_APPEND;
3947 if (flags & EXT4_IMMUTABLE_FL) 3947 if (flags & EXT4_IMMUTABLE_FL)
3948 inode->i_flags |= S_IMMUTABLE; 3948 new_fl |= S_IMMUTABLE;
3949 if (flags & EXT4_NOATIME_FL) 3949 if (flags & EXT4_NOATIME_FL)
3950 inode->i_flags |= S_NOATIME; 3950 new_fl |= S_NOATIME;
3951 if (flags & EXT4_DIRSYNC_FL) 3951 if (flags & EXT4_DIRSYNC_FL)
3952 inode->i_flags |= S_DIRSYNC; 3952 new_fl |= S_DIRSYNC;
3953 inode_set_flags(inode, new_fl,
3954 S_SYNC|S_APPEND|S_IMMUTABLE|S_NOATIME|S_DIRSYNC);
3953} 3955}
3954 3956
3955/* Propagate flags from i_flags to EXT4_I(inode)->i_flags */ 3957/* Propagate flags from i_flags to EXT4_I(inode)->i_flags */