Commit 5a260029 authored by Theodore Ts'o's avatar Theodore Ts'o Committed by Kleber Sacilotto de Souza

ext4: avoid divide by zero fault when deleting corrupted inline directories

BugLink: https://bugs.launchpad.net/bugs/1798617

commit 4d982e25 upstream.

A specially crafted file system can trick empty_inline_dir() into
reading past the last valid entry in a inline directory, and then run
into the end of xattr marker. This will trigger a divide by zero
fault.  Fix this by using the size of the inline directory instead of
dir->i_size.

Also clean up error reporting in __ext4_check_dir_entry so that the
message is clearer and more understandable --- and avoids the division
by zero trap if the size passed in is zero.  (I'm not sure why we
coded it that way in the first place; printing offset % size is
actually more confusing and less useful.)

https://bugzilla.kernel.org/show_bug.cgi?id=200933Signed-off-by: default avatarTheodore Ts'o <tytso@mit.edu>
Reported-by: default avatarWen Xu <wen.xu@gatech.edu>
Cc: stable@vger.kernel.org
Signed-off-by: default avatarGreg Kroah-Hartman <gregkh@linuxfoundation.org>
Signed-off-by: default avatarStefan Bader <stefan.bader@canonical.com>
Signed-off-by: default avatarKleber Sacilotto de Souza <kleber.souza@canonical.com>
parent 1bd9ba6b
...@@ -74,7 +74,7 @@ int __ext4_check_dir_entry(const char *function, unsigned int line, ...@@ -74,7 +74,7 @@ int __ext4_check_dir_entry(const char *function, unsigned int line,
else if (unlikely(rlen < EXT4_DIR_REC_LEN(de->name_len))) else if (unlikely(rlen < EXT4_DIR_REC_LEN(de->name_len)))
error_msg = "rec_len is too small for name_len"; error_msg = "rec_len is too small for name_len";
else if (unlikely(((char *) de - buf) + rlen > size)) else if (unlikely(((char *) de - buf) + rlen > size))
error_msg = "directory entry across range"; error_msg = "directory entry overrun";
else if (unlikely(le32_to_cpu(de->inode) > else if (unlikely(le32_to_cpu(de->inode) >
le32_to_cpu(EXT4_SB(dir->i_sb)->s_es->s_inodes_count))) le32_to_cpu(EXT4_SB(dir->i_sb)->s_es->s_inodes_count)))
error_msg = "inode out of bounds"; error_msg = "inode out of bounds";
...@@ -83,18 +83,16 @@ int __ext4_check_dir_entry(const char *function, unsigned int line, ...@@ -83,18 +83,16 @@ int __ext4_check_dir_entry(const char *function, unsigned int line,
if (filp) if (filp)
ext4_error_file(filp, function, line, bh->b_blocknr, ext4_error_file(filp, function, line, bh->b_blocknr,
"bad entry in directory: %s - offset=%u(%u), " "bad entry in directory: %s - offset=%u, "
"inode=%u, rec_len=%d, name_len=%d", "inode=%u, rec_len=%d, name_len=%d, size=%d",
error_msg, (unsigned) (offset % size), error_msg, offset, le32_to_cpu(de->inode),
offset, le32_to_cpu(de->inode), rlen, de->name_len, size);
rlen, de->name_len);
else else
ext4_error_inode(dir, function, line, bh->b_blocknr, ext4_error_inode(dir, function, line, bh->b_blocknr,
"bad entry in directory: %s - offset=%u(%u), " "bad entry in directory: %s - offset=%u, "
"inode=%u, rec_len=%d, name_len=%d", "inode=%u, rec_len=%d, name_len=%d, size=%d",
error_msg, (unsigned) (offset % size), error_msg, offset, le32_to_cpu(de->inode),
offset, le32_to_cpu(de->inode), rlen, de->name_len, size);
rlen, de->name_len);
return 1; return 1;
} }
......
...@@ -1756,6 +1756,7 @@ int empty_inline_dir(struct inode *dir, int *has_inline_data) ...@@ -1756,6 +1756,7 @@ int empty_inline_dir(struct inode *dir, int *has_inline_data)
{ {
int err, inline_size; int err, inline_size;
struct ext4_iloc iloc; struct ext4_iloc iloc;
size_t inline_len;
void *inline_pos; void *inline_pos;
unsigned int offset; unsigned int offset;
struct ext4_dir_entry_2 *de; struct ext4_dir_entry_2 *de;
...@@ -1783,8 +1784,9 @@ int empty_inline_dir(struct inode *dir, int *has_inline_data) ...@@ -1783,8 +1784,9 @@ int empty_inline_dir(struct inode *dir, int *has_inline_data)
goto out; goto out;
} }
inline_len = ext4_get_inline_size(dir);
offset = EXT4_INLINE_DOTDOT_SIZE; offset = EXT4_INLINE_DOTDOT_SIZE;
while (offset < dir->i_size) { while (offset < inline_len) {
de = ext4_get_inline_entry(dir, &iloc, offset, de = ext4_get_inline_entry(dir, &iloc, offset,
&inline_pos, &inline_size); &inline_pos, &inline_size);
if (ext4_check_dir_entry(dir, NULL, de, if (ext4_check_dir_entry(dir, NULL, de,
......
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