|
@@ -1358,12 +1358,14 @@ int ext4_expand_extra_isize_ea(struct inode *inode, int new_extra_isize,
|
|
|
int isize_diff; /* How much do we need to grow i_extra_isize */
|
|
|
|
|
|
down_write(&EXT4_I(inode)->xattr_sem);
|
|
|
+ /*
|
|
|
+ * Set EXT4_STATE_NO_EXPAND to avoid recursion when marking inode dirty
|
|
|
+ */
|
|
|
+ ext4_set_inode_state(inode, EXT4_STATE_NO_EXPAND);
|
|
|
retry:
|
|
|
isize_diff = new_extra_isize - EXT4_I(inode)->i_extra_isize;
|
|
|
- if (EXT4_I(inode)->i_extra_isize >= new_extra_isize) {
|
|
|
- up_write(&EXT4_I(inode)->xattr_sem);
|
|
|
- return 0;
|
|
|
- }
|
|
|
+ if (EXT4_I(inode)->i_extra_isize >= new_extra_isize)
|
|
|
+ goto out;
|
|
|
|
|
|
header = IHDR(inode, raw_inode);
|
|
|
entry = IFIRST(header);
|
|
@@ -1392,8 +1394,7 @@ retry:
|
|
|
(void *)header, total_ino,
|
|
|
inode->i_sb->s_blocksize);
|
|
|
EXT4_I(inode)->i_extra_isize = new_extra_isize;
|
|
|
- error = 0;
|
|
|
- goto cleanup;
|
|
|
+ goto out;
|
|
|
}
|
|
|
|
|
|
/*
|
|
@@ -1553,6 +1554,8 @@ retry:
|
|
|
kfree(bs);
|
|
|
}
|
|
|
brelse(bh);
|
|
|
+out:
|
|
|
+ ext4_clear_inode_state(inode, EXT4_STATE_NO_EXPAND);
|
|
|
up_write(&EXT4_I(inode)->xattr_sem);
|
|
|
return 0;
|
|
|
|
|
@@ -1564,6 +1567,10 @@ cleanup:
|
|
|
kfree(is);
|
|
|
kfree(bs);
|
|
|
brelse(bh);
|
|
|
+ /*
|
|
|
+ * We deliberately leave EXT4_STATE_NO_EXPAND set here since inode
|
|
|
+ * size expansion failed.
|
|
|
+ */
|
|
|
up_write(&EXT4_I(inode)->xattr_sem);
|
|
|
return error;
|
|
|
}
|