Commit 1c23de63 authored by Linus Torvalds's avatar Linus Torvalds

Merge tag 'ext4_for_linus_stable' of git://git.kernel.org/pub/scm/linux/kernel/git/tytso/ext4

Pull ext4 fixes from Ted Ts'o:
 "Fix a memory leak on an error path, and two races when modifying
  inodes relating to the inline_data and metadata checksum features"

* tag 'ext4_for_linus_stable' of git://git.kernel.org/pub/scm/linux/kernel/git/tytso/ext4:
  ext4: fix two spelling nits
  ext4: lock the xattr block before checksuming it
  jbd2: don't leak memory if setting up journal fails
  ext4: mark inode dirty after converting inline directory
parents a643f905 d67d64f4
...@@ -1169,10 +1169,9 @@ static int ext4_finish_convert_inline_dir(handle_t *handle, ...@@ -1169,10 +1169,9 @@ static int ext4_finish_convert_inline_dir(handle_t *handle,
set_buffer_uptodate(dir_block); set_buffer_uptodate(dir_block);
err = ext4_handle_dirty_dirent_node(handle, inode, dir_block); err = ext4_handle_dirty_dirent_node(handle, inode, dir_block);
if (err) if (err)
goto out;
set_buffer_verified(dir_block);
out:
return err; return err;
set_buffer_verified(dir_block);
return ext4_mark_inode_dirty(handle, inode);
} }
static int ext4_convert_inline_data_nolock(handle_t *handle, static int ext4_convert_inline_data_nolock(handle_t *handle,
......
...@@ -5400,7 +5400,7 @@ int ext4_getattr(const struct path *path, struct kstat *stat, ...@@ -5400,7 +5400,7 @@ int ext4_getattr(const struct path *path, struct kstat *stat,
* If there is inline data in the inode, the inode will normally not * If there is inline data in the inode, the inode will normally not
* have data blocks allocated (it may have an external xattr block). * have data blocks allocated (it may have an external xattr block).
* Report at least one sector for such files, so tools like tar, rsync, * Report at least one sector for such files, so tools like tar, rsync,
* others doen't incorrectly think the file is completely sparse. * others don't incorrectly think the file is completely sparse.
*/ */
if (unlikely(ext4_has_inline_data(inode))) if (unlikely(ext4_has_inline_data(inode)))
stat->blocks += (stat->size + 511) >> 9; stat->blocks += (stat->size + 511) >> 9;
......
...@@ -511,7 +511,7 @@ mext_check_arguments(struct inode *orig_inode, ...@@ -511,7 +511,7 @@ mext_check_arguments(struct inode *orig_inode,
if ((orig_start & ~(PAGE_MASK >> orig_inode->i_blkbits)) != if ((orig_start & ~(PAGE_MASK >> orig_inode->i_blkbits)) !=
(donor_start & ~(PAGE_MASK >> orig_inode->i_blkbits))) { (donor_start & ~(PAGE_MASK >> orig_inode->i_blkbits))) {
ext4_debug("ext4 move extent: orig and donor's start " ext4_debug("ext4 move extent: orig and donor's start "
"offset are not alligned [ino:orig %lu, donor %lu]\n", "offsets are not aligned [ino:orig %lu, donor %lu]\n",
orig_inode->i_ino, donor_inode->i_ino); orig_inode->i_ino, donor_inode->i_ino);
return -EINVAL; return -EINVAL;
} }
......
...@@ -131,31 +131,26 @@ static __le32 ext4_xattr_block_csum(struct inode *inode, ...@@ -131,31 +131,26 @@ static __le32 ext4_xattr_block_csum(struct inode *inode,
} }
static int ext4_xattr_block_csum_verify(struct inode *inode, static int ext4_xattr_block_csum_verify(struct inode *inode,
sector_t block_nr, struct buffer_head *bh)
struct ext4_xattr_header *hdr)
{
if (ext4_has_metadata_csum(inode->i_sb) &&
(hdr->h_checksum != ext4_xattr_block_csum(inode, block_nr, hdr)))
return 0;
return 1;
}
static void ext4_xattr_block_csum_set(struct inode *inode,
sector_t block_nr,
struct ext4_xattr_header *hdr)
{ {
if (!ext4_has_metadata_csum(inode->i_sb)) struct ext4_xattr_header *hdr = BHDR(bh);
return; int ret = 1;
hdr->h_checksum = ext4_xattr_block_csum(inode, block_nr, hdr); if (ext4_has_metadata_csum(inode->i_sb)) {
lock_buffer(bh);
ret = (hdr->h_checksum == ext4_xattr_block_csum(inode,
bh->b_blocknr, hdr));
unlock_buffer(bh);
}
return ret;
} }
static inline int ext4_handle_dirty_xattr_block(handle_t *handle, static void ext4_xattr_block_csum_set(struct inode *inode,
struct inode *inode,
struct buffer_head *bh) struct buffer_head *bh)
{ {
ext4_xattr_block_csum_set(inode, bh->b_blocknr, BHDR(bh)); if (ext4_has_metadata_csum(inode->i_sb))
return ext4_handle_dirty_metadata(handle, inode, bh); BHDR(bh)->h_checksum = ext4_xattr_block_csum(inode,
bh->b_blocknr, BHDR(bh));
} }
static inline const struct xattr_handler * static inline const struct xattr_handler *
...@@ -233,7 +228,7 @@ ext4_xattr_check_block(struct inode *inode, struct buffer_head *bh) ...@@ -233,7 +228,7 @@ ext4_xattr_check_block(struct inode *inode, struct buffer_head *bh)
if (BHDR(bh)->h_magic != cpu_to_le32(EXT4_XATTR_MAGIC) || if (BHDR(bh)->h_magic != cpu_to_le32(EXT4_XATTR_MAGIC) ||
BHDR(bh)->h_blocks != cpu_to_le32(1)) BHDR(bh)->h_blocks != cpu_to_le32(1))
return -EFSCORRUPTED; return -EFSCORRUPTED;
if (!ext4_xattr_block_csum_verify(inode, bh->b_blocknr, BHDR(bh))) if (!ext4_xattr_block_csum_verify(inode, bh))
return -EFSBADCRC; return -EFSBADCRC;
error = ext4_xattr_check_names(BFIRST(bh), bh->b_data + bh->b_size, error = ext4_xattr_check_names(BFIRST(bh), bh->b_data + bh->b_size,
bh->b_data); bh->b_data);
...@@ -618,23 +613,22 @@ ext4_xattr_release_block(handle_t *handle, struct inode *inode, ...@@ -618,23 +613,22 @@ ext4_xattr_release_block(handle_t *handle, struct inode *inode,
} }
} }
ext4_xattr_block_csum_set(inode, bh);
/* /*
* Beware of this ugliness: Releasing of xattr block references * Beware of this ugliness: Releasing of xattr block references
* from different inodes can race and so we have to protect * from different inodes can race and so we have to protect
* from a race where someone else frees the block (and releases * from a race where someone else frees the block (and releases
* its journal_head) before we are done dirtying the buffer. In * its journal_head) before we are done dirtying the buffer. In
* nojournal mode this race is harmless and we actually cannot * nojournal mode this race is harmless and we actually cannot
* call ext4_handle_dirty_xattr_block() with locked buffer as * call ext4_handle_dirty_metadata() with locked buffer as
* that function can call sync_dirty_buffer() so for that case * that function can call sync_dirty_buffer() so for that case
* we handle the dirtying after unlocking the buffer. * we handle the dirtying after unlocking the buffer.
*/ */
if (ext4_handle_valid(handle)) if (ext4_handle_valid(handle))
error = ext4_handle_dirty_xattr_block(handle, inode, error = ext4_handle_dirty_metadata(handle, inode, bh);
bh);
unlock_buffer(bh); unlock_buffer(bh);
if (!ext4_handle_valid(handle)) if (!ext4_handle_valid(handle))
error = ext4_handle_dirty_xattr_block(handle, inode, error = ext4_handle_dirty_metadata(handle, inode, bh);
bh);
if (IS_SYNC(inode)) if (IS_SYNC(inode))
ext4_handle_sync(handle); ext4_handle_sync(handle);
dquot_free_block(inode, EXT4_C2B(EXT4_SB(inode->i_sb), 1)); dquot_free_block(inode, EXT4_C2B(EXT4_SB(inode->i_sb), 1));
...@@ -863,11 +857,12 @@ ext4_xattr_block_set(handle_t *handle, struct inode *inode, ...@@ -863,11 +857,12 @@ ext4_xattr_block_set(handle_t *handle, struct inode *inode,
ext4_xattr_cache_insert(ext4_mb_cache, ext4_xattr_cache_insert(ext4_mb_cache,
bs->bh); bs->bh);
} }
ext4_xattr_block_csum_set(inode, bs->bh);
unlock_buffer(bs->bh); unlock_buffer(bs->bh);
if (error == -EFSCORRUPTED) if (error == -EFSCORRUPTED)
goto bad_block; goto bad_block;
if (!error) if (!error)
error = ext4_handle_dirty_xattr_block(handle, error = ext4_handle_dirty_metadata(handle,
inode, inode,
bs->bh); bs->bh);
if (error) if (error)
...@@ -967,8 +962,9 @@ ext4_xattr_block_set(handle_t *handle, struct inode *inode, ...@@ -967,8 +962,9 @@ ext4_xattr_block_set(handle_t *handle, struct inode *inode,
ce->e_reusable = 0; ce->e_reusable = 0;
ea_bdebug(new_bh, "reusing; refcount now=%d", ea_bdebug(new_bh, "reusing; refcount now=%d",
ref); ref);
ext4_xattr_block_csum_set(inode, new_bh);
unlock_buffer(new_bh); unlock_buffer(new_bh);
error = ext4_handle_dirty_xattr_block(handle, error = ext4_handle_dirty_metadata(handle,
inode, inode,
new_bh); new_bh);
if (error) if (error)
...@@ -1020,11 +1016,12 @@ ext4_xattr_block_set(handle_t *handle, struct inode *inode, ...@@ -1020,11 +1016,12 @@ ext4_xattr_block_set(handle_t *handle, struct inode *inode,
goto getblk_failed; goto getblk_failed;
} }
memcpy(new_bh->b_data, s->base, new_bh->b_size); memcpy(new_bh->b_data, s->base, new_bh->b_size);
ext4_xattr_block_csum_set(inode, new_bh);
set_buffer_uptodate(new_bh); set_buffer_uptodate(new_bh);
unlock_buffer(new_bh); unlock_buffer(new_bh);
ext4_xattr_cache_insert(ext4_mb_cache, new_bh); ext4_xattr_cache_insert(ext4_mb_cache, new_bh);
error = ext4_handle_dirty_xattr_block(handle, error = ext4_handle_dirty_metadata(handle, inode,
inode, new_bh); new_bh);
if (error) if (error)
goto cleanup; goto cleanup;
} }
......
...@@ -1125,10 +1125,8 @@ static journal_t *journal_init_common(struct block_device *bdev, ...@@ -1125,10 +1125,8 @@ static journal_t *journal_init_common(struct block_device *bdev,
/* Set up a default-sized revoke table for the new mount. */ /* Set up a default-sized revoke table for the new mount. */
err = jbd2_journal_init_revoke(journal, JOURNAL_REVOKE_DEFAULT_HASH); err = jbd2_journal_init_revoke(journal, JOURNAL_REVOKE_DEFAULT_HASH);
if (err) { if (err)
kfree(journal); goto err_cleanup;
return NULL;
}
spin_lock_init(&journal->j_history_lock); spin_lock_init(&journal->j_history_lock);
...@@ -1145,23 +1143,25 @@ static journal_t *journal_init_common(struct block_device *bdev, ...@@ -1145,23 +1143,25 @@ static journal_t *journal_init_common(struct block_device *bdev,
journal->j_wbufsize = n; journal->j_wbufsize = n;
journal->j_wbuf = kmalloc_array(n, sizeof(struct buffer_head *), journal->j_wbuf = kmalloc_array(n, sizeof(struct buffer_head *),
GFP_KERNEL); GFP_KERNEL);
if (!journal->j_wbuf) { if (!journal->j_wbuf)
kfree(journal); goto err_cleanup;
return NULL;
}
bh = getblk_unmovable(journal->j_dev, start, journal->j_blocksize); bh = getblk_unmovable(journal->j_dev, start, journal->j_blocksize);
if (!bh) { if (!bh) {
pr_err("%s: Cannot get buffer for journal superblock\n", pr_err("%s: Cannot get buffer for journal superblock\n",
__func__); __func__);
kfree(journal->j_wbuf); goto err_cleanup;
kfree(journal);
return NULL;
} }
journal->j_sb_buffer = bh; journal->j_sb_buffer = bh;
journal->j_superblock = (journal_superblock_t *)bh->b_data; journal->j_superblock = (journal_superblock_t *)bh->b_data;
return journal; return journal;
err_cleanup:
kfree(journal->j_wbuf);
jbd2_journal_destroy_revoke(journal);
kfree(journal);
return NULL;
} }
/* jbd2_journal_init_dev and jbd2_journal_init_inode: /* jbd2_journal_init_dev and jbd2_journal_init_inode:
......
...@@ -280,6 +280,7 @@ int jbd2_journal_init_revoke(journal_t *journal, int hash_size) ...@@ -280,6 +280,7 @@ int jbd2_journal_init_revoke(journal_t *journal, int hash_size)
fail1: fail1:
jbd2_journal_destroy_revoke_table(journal->j_revoke_table[0]); jbd2_journal_destroy_revoke_table(journal->j_revoke_table[0]);
journal->j_revoke_table[0] = NULL;
fail0: fail0:
return -ENOMEM; return -ENOMEM;
} }
......
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