Commit 043546e4 authored by Ira Weiny's avatar Ira Weiny Committed by Theodore Ts'o

fs/ext4: Only change S_DAX on inode load

To prevent complications with in memory inodes we only set S_DAX on
inode load.  FS_XFLAG_DAX can be changed at any time and S_DAX will
change after inode eviction and reload.

Add init bool to ext4_set_inode_flags() to indicate if the inode is
being newly initialized.

Assert that S_DAX is not set on an inode which is just being loaded.
Reviewed-by: default avatarJan Kara <jack@suse.cz>
Signed-off-by: default avatarIra Weiny <ira.weiny@intel.com>

Link: https://lore.kernel.org/r/20200528150003.828793-6-ira.weiny@intel.comSigned-off-by: default avatarTheodore Ts'o <tytso@mit.edu>
parent a8ab6d38
...@@ -2693,7 +2693,7 @@ extern int ext4_can_truncate(struct inode *inode); ...@@ -2693,7 +2693,7 @@ extern int ext4_can_truncate(struct inode *inode);
extern int ext4_truncate(struct inode *); extern int ext4_truncate(struct inode *);
extern int ext4_break_layouts(struct inode *); extern int ext4_break_layouts(struct inode *);
extern int ext4_punch_hole(struct inode *inode, loff_t offset, loff_t length); extern int ext4_punch_hole(struct inode *inode, loff_t offset, loff_t length);
extern void ext4_set_inode_flags(struct inode *); extern void ext4_set_inode_flags(struct inode *, bool init);
extern int ext4_alloc_da_blocks(struct inode *inode); extern int ext4_alloc_da_blocks(struct inode *inode);
extern void ext4_set_aops(struct inode *inode); extern void ext4_set_aops(struct inode *inode);
extern int ext4_writepage_trans_blocks(struct inode *); extern int ext4_writepage_trans_blocks(struct inode *);
......
...@@ -1116,7 +1116,7 @@ struct inode *__ext4_new_inode(handle_t *handle, struct inode *dir, ...@@ -1116,7 +1116,7 @@ struct inode *__ext4_new_inode(handle_t *handle, struct inode *dir,
ei->i_block_group = group; ei->i_block_group = group;
ei->i_last_alloc_group = ~0; ei->i_last_alloc_group = ~0;
ext4_set_inode_flags(inode); ext4_set_inode_flags(inode, true);
if (IS_DIRSYNC(inode)) if (IS_DIRSYNC(inode))
ext4_handle_sync(handle); ext4_handle_sync(handle);
if (insert_inode_locked(inode) < 0) { if (insert_inode_locked(inode) < 0) {
......
...@@ -4420,11 +4420,13 @@ static bool ext4_should_enable_dax(struct inode *inode) ...@@ -4420,11 +4420,13 @@ static bool ext4_should_enable_dax(struct inode *inode)
return false; return false;
} }
void ext4_set_inode_flags(struct inode *inode) void ext4_set_inode_flags(struct inode *inode, bool init)
{ {
unsigned int flags = EXT4_I(inode)->i_flags; unsigned int flags = EXT4_I(inode)->i_flags;
unsigned int new_fl = 0; unsigned int new_fl = 0;
WARN_ON_ONCE(IS_DAX(inode) && init);
if (flags & EXT4_SYNC_FL) if (flags & EXT4_SYNC_FL)
new_fl |= S_SYNC; new_fl |= S_SYNC;
if (flags & EXT4_APPEND_FL) if (flags & EXT4_APPEND_FL)
...@@ -4435,8 +4437,13 @@ void ext4_set_inode_flags(struct inode *inode) ...@@ -4435,8 +4437,13 @@ void ext4_set_inode_flags(struct inode *inode)
new_fl |= S_NOATIME; new_fl |= S_NOATIME;
if (flags & EXT4_DIRSYNC_FL) if (flags & EXT4_DIRSYNC_FL)
new_fl |= S_DIRSYNC; new_fl |= S_DIRSYNC;
if (ext4_should_enable_dax(inode))
/* Because of the way inode_set_flags() works we must preserve S_DAX
* here if already set. */
new_fl |= (inode->i_flags & S_DAX);
if (init && ext4_should_enable_dax(inode))
new_fl |= S_DAX; new_fl |= S_DAX;
if (flags & EXT4_ENCRYPT_FL) if (flags & EXT4_ENCRYPT_FL)
new_fl |= S_ENCRYPTED; new_fl |= S_ENCRYPTED;
if (flags & EXT4_CASEFOLD_FL) if (flags & EXT4_CASEFOLD_FL)
...@@ -4650,7 +4657,7 @@ struct inode *__ext4_iget(struct super_block *sb, unsigned long ino, ...@@ -4650,7 +4657,7 @@ struct inode *__ext4_iget(struct super_block *sb, unsigned long ino,
* not initialized on a new filesystem. */ * not initialized on a new filesystem. */
} }
ei->i_flags = le32_to_cpu(raw_inode->i_flags); ei->i_flags = le32_to_cpu(raw_inode->i_flags);
ext4_set_inode_flags(inode); ext4_set_inode_flags(inode, true);
inode->i_blocks = ext4_inode_blocks(raw_inode, ei); inode->i_blocks = ext4_inode_blocks(raw_inode, ei);
ei->i_file_acl = le32_to_cpu(raw_inode->i_file_acl_lo); ei->i_file_acl = le32_to_cpu(raw_inode->i_file_acl_lo);
if (ext4_has_feature_64bit(sb)) if (ext4_has_feature_64bit(sb))
......
...@@ -381,7 +381,8 @@ static int ext4_ioctl_setflags(struct inode *inode, ...@@ -381,7 +381,8 @@ static int ext4_ioctl_setflags(struct inode *inode,
ext4_clear_inode_flag(inode, i); ext4_clear_inode_flag(inode, i);
} }
ext4_set_inode_flags(inode); ext4_set_inode_flags(inode, false);
inode->i_ctime = current_time(inode); inode->i_ctime = current_time(inode);
err = ext4_mark_iloc_dirty(handle, inode, &iloc); err = ext4_mark_iloc_dirty(handle, inode, &iloc);
......
...@@ -1348,7 +1348,7 @@ static int ext4_set_context(struct inode *inode, const void *ctx, size_t len, ...@@ -1348,7 +1348,7 @@ static int ext4_set_context(struct inode *inode, const void *ctx, size_t len,
* Update inode->i_flags - S_ENCRYPTED will be enabled, * Update inode->i_flags - S_ENCRYPTED will be enabled,
* S_DAX may be disabled * S_DAX may be disabled
*/ */
ext4_set_inode_flags(inode); ext4_set_inode_flags(inode, false);
} }
return res; return res;
} }
...@@ -1375,7 +1375,7 @@ static int ext4_set_context(struct inode *inode, const void *ctx, size_t len, ...@@ -1375,7 +1375,7 @@ static int ext4_set_context(struct inode *inode, const void *ctx, size_t len,
* Update inode->i_flags - S_ENCRYPTED will be enabled, * Update inode->i_flags - S_ENCRYPTED will be enabled,
* S_DAX may be disabled * S_DAX may be disabled
*/ */
ext4_set_inode_flags(inode); ext4_set_inode_flags(inode, false);
res = ext4_mark_inode_dirty(handle, inode); res = ext4_mark_inode_dirty(handle, inode);
if (res) if (res)
EXT4_ERROR_INODE(inode, "Failed to mark inode dirty"); EXT4_ERROR_INODE(inode, "Failed to mark inode dirty");
......
...@@ -244,7 +244,7 @@ static int ext4_end_enable_verity(struct file *filp, const void *desc, ...@@ -244,7 +244,7 @@ static int ext4_end_enable_verity(struct file *filp, const void *desc,
if (err) if (err)
goto out_stop; goto out_stop;
ext4_set_inode_flag(inode, EXT4_INODE_VERITY); ext4_set_inode_flag(inode, EXT4_INODE_VERITY);
ext4_set_inode_flags(inode); ext4_set_inode_flags(inode, false);
err = ext4_mark_iloc_dirty(handle, inode, &iloc); err = ext4_mark_iloc_dirty(handle, inode, &iloc);
} }
out_stop: out_stop:
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment