Commit c03b45b8 authored by Miao Xie's avatar Miao Xie Committed by Theodore Ts'o

ext4, project: expand inode extra size if possible

When upgrading from old format, try to set project id
to old file first time, it will return EOVERFLOW, but if
that file is dirtied(touch etc), changing project id will
be allowed, this might be confusing for users, we could
try to expand @i_extra_isize here too.
Reported-by: default avatarZhang Yi <yi.zhang@huawei.com>
Signed-off-by: default avatarMiao Xie <miaoxie@huawei.com>
Signed-off-by: default avatarWang Shilong <wshilong@ddn.com>
Signed-off-by: default avatarTheodore Ts'o <tytso@mit.edu>
parent b640b2c5
...@@ -227,6 +227,9 @@ int ext4_reserve_inode_write(handle_t *handle, struct inode *inode, ...@@ -227,6 +227,9 @@ int ext4_reserve_inode_write(handle_t *handle, struct inode *inode,
int ext4_mark_inode_dirty(handle_t *handle, struct inode *inode); int ext4_mark_inode_dirty(handle_t *handle, struct inode *inode);
int ext4_expand_extra_isize(struct inode *inode,
unsigned int new_extra_isize,
struct ext4_iloc *iloc);
/* /*
* Wrapper functions with which ext4 calls into JBD. * Wrapper functions with which ext4 calls into JBD.
*/ */
......
...@@ -5702,6 +5702,42 @@ ext4_reserve_inode_write(handle_t *handle, struct inode *inode, ...@@ -5702,6 +5702,42 @@ ext4_reserve_inode_write(handle_t *handle, struct inode *inode,
return err; return err;
} }
static int __ext4_expand_extra_isize(struct inode *inode,
unsigned int new_extra_isize,
struct ext4_iloc *iloc,
handle_t *handle, int *no_expand)
{
struct ext4_inode *raw_inode;
struct ext4_xattr_ibody_header *header;
int error;
raw_inode = ext4_raw_inode(iloc);
header = IHDR(inode, raw_inode);
/* No extended attributes present */
if (!ext4_test_inode_state(inode, EXT4_STATE_XATTR) ||
header->h_magic != cpu_to_le32(EXT4_XATTR_MAGIC)) {
memset((void *)raw_inode + EXT4_GOOD_OLD_INODE_SIZE +
EXT4_I(inode)->i_extra_isize, 0,
new_extra_isize - EXT4_I(inode)->i_extra_isize);
EXT4_I(inode)->i_extra_isize = new_extra_isize;
return 0;
}
/* try to expand with EAs present */
error = ext4_expand_extra_isize_ea(inode, new_extra_isize,
raw_inode, handle);
if (error) {
/*
* Inode size expansion failed; don't try again
*/
*no_expand = 1;
}
return error;
}
/* /*
* Expand an inode by new_extra_isize bytes. * Expand an inode by new_extra_isize bytes.
* Returns 0 on success or negative error number on failure. * Returns 0 on success or negative error number on failure.
...@@ -5711,8 +5747,6 @@ static int ext4_try_to_expand_extra_isize(struct inode *inode, ...@@ -5711,8 +5747,6 @@ static int ext4_try_to_expand_extra_isize(struct inode *inode,
struct ext4_iloc iloc, struct ext4_iloc iloc,
handle_t *handle) handle_t *handle)
{ {
struct ext4_inode *raw_inode;
struct ext4_xattr_ibody_header *header;
int no_expand; int no_expand;
int error; int error;
...@@ -5736,32 +5770,53 @@ static int ext4_try_to_expand_extra_isize(struct inode *inode, ...@@ -5736,32 +5770,53 @@ static int ext4_try_to_expand_extra_isize(struct inode *inode,
if (ext4_write_trylock_xattr(inode, &no_expand) == 0) if (ext4_write_trylock_xattr(inode, &no_expand) == 0)
return -EBUSY; return -EBUSY;
raw_inode = ext4_raw_inode(&iloc); error = __ext4_expand_extra_isize(inode, new_extra_isize, &iloc,
handle, &no_expand);
ext4_write_unlock_xattr(inode, &no_expand);
header = IHDR(inode, raw_inode); return error;
}
/* No extended attributes present */ int ext4_expand_extra_isize(struct inode *inode,
if (!ext4_test_inode_state(inode, EXT4_STATE_XATTR) || unsigned int new_extra_isize,
header->h_magic != cpu_to_le32(EXT4_XATTR_MAGIC)) { struct ext4_iloc *iloc)
memset((void *)raw_inode + EXT4_GOOD_OLD_INODE_SIZE + {
EXT4_I(inode)->i_extra_isize, 0, handle_t *handle;
new_extra_isize - EXT4_I(inode)->i_extra_isize); int no_expand;
EXT4_I(inode)->i_extra_isize = new_extra_isize; int error, rc;
ext4_write_unlock_xattr(inode, &no_expand);
return 0; if (ext4_test_inode_state(inode, EXT4_STATE_NO_EXPAND)) {
brelse(iloc->bh);
return -EOVERFLOW;
} }
/* try to expand with EAs present */ handle = ext4_journal_start(inode, EXT4_HT_INODE,
error = ext4_expand_extra_isize_ea(inode, new_extra_isize, EXT4_DATA_TRANS_BLOCKS(inode->i_sb));
raw_inode, handle); if (IS_ERR(handle)) {
error = PTR_ERR(handle);
brelse(iloc->bh);
return error;
}
ext4_write_lock_xattr(inode, &no_expand);
BUFFER_TRACE(iloc.bh, "get_write_access");
error = ext4_journal_get_write_access(handle, iloc->bh);
if (error) { if (error) {
/* brelse(iloc->bh);
* Inode size expansion failed; don't try again goto out_stop;
*/
no_expand = 1;
} }
ext4_write_unlock_xattr(inode, &no_expand);
error = __ext4_expand_extra_isize(inode, new_extra_isize, iloc,
handle, &no_expand);
rc = ext4_mark_iloc_dirty(handle, inode, iloc);
if (!error)
error = rc;
ext4_write_unlock_xattr(inode, &no_expand);
out_stop:
ext4_journal_stop(handle);
return error; return error;
} }
......
...@@ -349,11 +349,14 @@ static int ext4_ioctl_setproject(struct file *filp, __u32 projid) ...@@ -349,11 +349,14 @@ static int ext4_ioctl_setproject(struct file *filp, __u32 projid)
raw_inode = ext4_raw_inode(&iloc); raw_inode = ext4_raw_inode(&iloc);
if (!EXT4_FITS_IN_INODE(raw_inode, ei, i_projid)) { if (!EXT4_FITS_IN_INODE(raw_inode, ei, i_projid)) {
err = -EOVERFLOW; err = ext4_expand_extra_isize(inode,
brelse(iloc.bh); EXT4_SB(sb)->s_want_extra_isize,
&iloc);
if (err)
goto out_unlock; goto out_unlock;
} } else {
brelse(iloc.bh); brelse(iloc.bh);
}
dquot_initialize(inode); dquot_initialize(inode);
......
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