aboutsummaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
authorRoman Pen <roman.penyaev@profitbricks.com>2017-01-08 20:59:35 -0500
committerTheodore Ts'o <tytso@mit.edu>2017-01-08 20:59:35 -0500
commit2a9b8cba62c0741109c33a2be700ff3d7703a7c2 (patch)
tree668ed1047a40f1c1a37a4ad5573e626b3804a0b2
parent56735be05353b085a0862ca4c4943628df3420ca (diff)
ext4: Include forgotten start block on fallocate insert range
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> Cc: stable@vger.kernel.org
-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 3e295d3350a9..4d3014b5a3f9 100644
--- a/fs/ext4/extents.c
+++ b/fs/ext4/extents.c
@@ -5343,8 +5343,7 @@ ext4_ext_shift_extents(struct inode *inode, handle_t *handle,
5343 if (!extent) 5343 if (!extent)
5344 goto out; 5344 goto out;
5345 5345
5346 stop = le32_to_cpu(extent->ee_block) + 5346 stop = le32_to_cpu(extent->ee_block);
5347 ext4_ext_get_actual_len(extent);
5348 5347
5349 /* 5348 /*
5350 * In case of left shift, Don't start shifting extents until we make 5349 * In case of left shift, Don't start shifting extents until we make
@@ -5383,8 +5382,12 @@ ext4_ext_shift_extents(struct inode *inode, handle_t *handle,
5383 else 5382 else
5384 iterator = &stop; 5383 iterator = &stop;
5385 5384
5386 /* Its safe to start updating extents */ 5385 /*
5387 while (start < stop) { 5386 * Its safe to start updating extents. Start and stop are unsigned, so
5387 * in case of right shift if extent with 0 block is reached, iterator
5388 * becomes NULL to indicate the end of the loop.
5389 */
5390 while (iterator && start <= stop) {
5388 path = ext4_find_extent(inode, *iterator, &path, 0); 5391 path = ext4_find_extent(inode, *iterator, &path, 0);
5389 if (IS_ERR(path)) 5392 if (IS_ERR(path))
5390 return PTR_ERR(path); 5393 return PTR_ERR(path);
@@ -5412,8 +5415,11 @@ ext4_ext_shift_extents(struct inode *inode, handle_t *handle,
5412 ext4_ext_get_actual_len(extent); 5415 ext4_ext_get_actual_len(extent);
5413 } else { 5416 } else {
5414 extent = EXT_FIRST_EXTENT(path[depth].p_hdr); 5417 extent = EXT_FIRST_EXTENT(path[depth].p_hdr);
5415 *iterator = le32_to_cpu(extent->ee_block) > 0 ? 5418 if (le32_to_cpu(extent->ee_block) > 0)
5416 le32_to_cpu(extent->ee_block) - 1 : 0; 5419 *iterator = le32_to_cpu(extent->ee_block) - 1;
5420 else
5421 /* Beginning is reached, end of the loop */
5422 iterator = NULL;
5417 /* Update path extent in case we need to stop */ 5423 /* Update path extent in case we need to stop */
5418 while (le32_to_cpu(extent->ee_block) < start) 5424 while (le32_to_cpu(extent->ee_block) < start)
5419 extent++; 5425 extent++;