Skip to content

Commit 5f16f32

Browse files
committed
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
1 parent ed3654e commit 5f16f32

File tree

3 files changed

+42
-6
lines changed

3 files changed

+42
-6
lines changed

fs/ext4/inode.c

Lines changed: 8 additions & 6 deletions
Original file line numberDiff line numberDiff line change
@@ -3938,18 +3938,20 @@ int ext4_get_inode_loc(struct inode *inode, struct ext4_iloc *iloc)
39383938
void ext4_set_inode_flags(struct inode *inode)
39393939
{
39403940
unsigned int flags = EXT4_I(inode)->i_flags;
3941+
unsigned int new_fl = 0;
39413942

3942-
inode->i_flags &= ~(S_SYNC|S_APPEND|S_IMMUTABLE|S_NOATIME|S_DIRSYNC);
39433943
if (flags & EXT4_SYNC_FL)
3944-
inode->i_flags |= S_SYNC;
3944+
new_fl |= S_SYNC;
39453945
if (flags & EXT4_APPEND_FL)
3946-
inode->i_flags |= S_APPEND;
3946+
new_fl |= S_APPEND;
39473947
if (flags & EXT4_IMMUTABLE_FL)
3948-
inode->i_flags |= S_IMMUTABLE;
3948+
new_fl |= S_IMMUTABLE;
39493949
if (flags & EXT4_NOATIME_FL)
3950-
inode->i_flags |= S_NOATIME;
3950+
new_fl |= S_NOATIME;
39513951
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);
39533955
}
39543956

39553957
/* Propagate flags from i_flags to EXT4_I(inode)->i_flags */

fs/inode.c

Lines changed: 31 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -1899,3 +1899,34 @@ void inode_dio_done(struct inode *inode)
18991899
wake_up_bit(&inode->i_state, __I_DIO_WAKEUP);
19001900
}
19011901
EXPORT_SYMBOL(inode_dio_done);
1902+
1903+
/*
1904+
* inode_set_flags - atomically set some inode flags
1905+
*
1906+
* Note: the caller should be holding i_mutex, or else be sure that
1907+
* they have exclusive access to the inode structure (i.e., while the
1908+
* inode is being instantiated). The reason for the cmpxchg() loop
1909+
* --- which wouldn't be necessary if all code paths which modify
1910+
* i_flags actually followed this rule, is that there is at least one
1911+
* code path which doesn't today --- for example,
1912+
* __generic_file_aio_write() calls file_remove_suid() without holding
1913+
* i_mutex --- so we use cmpxchg() out of an abundance of caution.
1914+
*
1915+
* In the long run, i_mutex is overkill, and we should probably look
1916+
* at using the i_lock spinlock to protect i_flags, and then make sure
1917+
* it is so documented in include/linux/fs.h and that all code follows
1918+
* the locking convention!!
1919+
*/
1920+
void inode_set_flags(struct inode *inode, unsigned int flags,
1921+
unsigned int mask)
1922+
{
1923+
unsigned int old_flags, new_flags;
1924+
1925+
WARN_ON_ONCE(flags & ~mask);
1926+
do {
1927+
old_flags = ACCESS_ONCE(inode->i_flags);
1928+
new_flags = (old_flags & ~mask) | flags;
1929+
} while (unlikely(cmpxchg(&inode->i_flags, old_flags,
1930+
new_flags) != old_flags));
1931+
}
1932+
EXPORT_SYMBOL(inode_set_flags);

include/linux/fs.h

Lines changed: 3 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -2556,6 +2556,9 @@ static inline ssize_t blockdev_direct_IO(int rw, struct kiocb *iocb,
25562556
void inode_dio_wait(struct inode *inode);
25572557
void inode_dio_done(struct inode *inode);
25582558

2559+
extern void inode_set_flags(struct inode *inode, unsigned int flags,
2560+
unsigned int mask);
2561+
25592562
extern const struct file_operations generic_ro_fops;
25602563

25612564
#define special_file(m) (S_ISCHR(m)||S_ISBLK(m)||S_ISFIFO(m)||S_ISSOCK(m))

0 commit comments

Comments
 (0)