aboutsummaryrefslogtreecommitdiffstats
path: root/fs
diff options
context:
space:
mode:
authorRoman Pen <roman.penyaev@profitbricks.com>2017-01-08 20:59:35 -0500
committerGreg Kroah-Hartman <gregkh@linuxfoundation.org>2017-03-12 00:41:45 -0500
commit72ae476d0401c38c912ad740cadcc7ef302f5ef2 (patch)
tree959d4c2ec93b7b8ee4a5358f348d2e10cae5ce3c /fs
parent8ca25e39ec2d67df1f69f08c1d4ad0d6310a5c5b (diff)
ext4: Include forgotten start block on fallocate insert range
commit 2a9b8cba62c0741109c33a2be700ff3d7703a7c2 upstream. While doing 'insert range' start block should be also shifted right. The bug can be easily reproduced by the following test: ptr = malloc(4096); assert(ptr); fd = open("./ext4.file", O_CREAT | O_TRUNC | O_RDWR, 0600); assert(fd >= 0); rc = fallocate(fd, 0, 0, 8192); assert(rc == 0); for (i = 0; i < 2048; i++) *((unsigned short *)ptr + i) = 0xbeef; rc = pwrite(fd, ptr, 4096, 0); assert(rc == 4096); rc = pwrite(fd, ptr, 4096, 4096); assert(rc == 4096); for (block = 2; block < 1000; block++) { rc = fallocate(fd, FALLOC_FL_INSERT_RANGE, 4096, 4096); assert(rc == 0); for (i = 0; i < 2048; i++) *((unsigned short *)ptr + i) = block; rc = pwrite(fd, ptr, 4096, 4096); assert(rc == 4096); } Because start block is not included in the range the hole appears at the wrong offset (just after the desired offset) and the following pwrite() overwrites already existent block, keeping hole untouched. Simple way to verify wrong behaviour is to check zeroed blocks after the test: $ hexdump ./ext4.file | grep '0000 0000' The root cause of the bug is a wrong range (start, stop], where start should be inclusive, i.e. [start, stop]. This patch fixes the problem by including start into the range. But not to break left shift (range collapse) stop points to the beginning of the a block, not to the end. The other not obvious change is an iterator check on validness in a main loop. Because iterator is unsigned the following corner case should be considered with care: insert a block at 0 offset, when stop variables overflows and never becomes less than start, which is 0. To handle this special case iterator is set to NULL to indicate that end of the loop is reached. Fixes: 331573febb6a2 Signed-off-by: Roman Pen <roman.penyaev@profitbricks.com> Signed-off-by: Theodore Ts'o <tytso@mit.edu> Cc: Namjae Jeon <namjae.jeon@samsung.com> Cc: Andreas Dilger <adilger.kernel@dilger.ca> Signed-off-by: Greg Kroah-Hartman <gregkh@linuxfoundation.org>
Diffstat (limited to 'fs')
-rw-r--r--fs/ext4/extents.c18
1 files changed, 12 insertions, 6 deletions
diff --git a/fs/ext4/extents.c b/fs/ext4/extents.c
index c930a0110fb4..b4987ea2ca79 100644
--- a/fs/ext4/extents.c
+++ b/fs/ext4/extents.c
@@ -5353,8 +5353,7 @@ ext4_ext_shift_extents(struct inode *inode, handle_t *handle,
5353 if (!extent) 5353 if (!extent)
5354 goto out; 5354 goto out;
5355 5355
5356 stop = le32_to_cpu(extent->ee_block) + 5356 stop = le32_to_cpu(extent->ee_block);
5357 ext4_ext_get_actual_len(extent);
5358 5357
5359 /* 5358 /*
5360 * In case of left shift, Don't start shifting extents until we make 5359 * In case of left shift, Don't start shifting extents until we make
@@ -5393,8 +5392,12 @@ ext4_ext_shift_extents(struct inode *inode, handle_t *handle,
5393 else 5392 else
5394 iterator = &stop; 5393 iterator = &stop;
5395 5394
5396 /* Its safe to start updating extents */ 5395 /*
5397 while (start < stop) { 5396 * Its safe to start updating extents. Start and stop are unsigned, so
5397 * in case of right shift if extent with 0 block is reached, iterator
5398 * becomes NULL to indicate the end of the loop.
5399 */
5400 while (iterator && start <= stop) {
5398 path = ext4_find_extent(inode, *iterator, &path, 0); 5401 path = ext4_find_extent(inode, *iterator, &path, 0);
5399 if (IS_ERR(path)) 5402 if (IS_ERR(path))
5400 return PTR_ERR(path); 5403 return PTR_ERR(path);
@@ -5422,8 +5425,11 @@ ext4_ext_shift_extents(struct inode *inode, handle_t *handle,
5422 ext4_ext_get_actual_len(extent); 5425 ext4_ext_get_actual_len(extent);
5423 } else { 5426 } else {
5424 extent = EXT_FIRST_EXTENT(path[depth].p_hdr); 5427 extent = EXT_FIRST_EXTENT(path[depth].p_hdr);
5425 *iterator = le32_to_cpu(extent->ee_block) > 0 ? 5428 if (le32_to_cpu(extent->ee_block) > 0)
5426 le32_to_cpu(extent->ee_block) - 1 : 0; 5429 *iterator = le32_to_cpu(extent->ee_block) - 1;
5430 else
5431 /* Beginning is reached, end of the loop */
5432 iterator = NULL;
5427 /* Update path extent in case we need to stop */ 5433 /* Update path extent in case we need to stop */
5428 while (le32_to_cpu(extent->ee_block) < start) 5434 while (le32_to_cpu(extent->ee_block) < start)
5429 extent++; 5435 extent++;