Commit d031a886 authored by Andreas Gruenbacher's avatar Andreas Gruenbacher

gfs2: Fix filesystem block deallocation for short writes

When a write cannot be carried out in full, gfs2_iomap_end() releases
blocks that have been allocated for this write but haven't been used.

To compute the end of the allocation, gfs2_iomap_end() incorrectly
rounded the end of the attempted write down to the next block boundary
to arrive at the end of the allocation.  It would have to round up, but
the end of the allocation is also available as iomap->offset +
iomap->length, so just use that instead.

In addition, use round_up() for computing the start of the unused range.

Fixes: 64bc06bb ("gfs2: iomap buffered write support")
Signed-off-by: default avatarAndreas Gruenbacher <agruenba@redhat.com>
parent 4a2316a1
...@@ -1153,13 +1153,12 @@ static int gfs2_iomap_end(struct inode *inode, loff_t pos, loff_t length, ...@@ -1153,13 +1153,12 @@ static int gfs2_iomap_end(struct inode *inode, loff_t pos, loff_t length,
if (length != written && (iomap->flags & IOMAP_F_NEW)) { if (length != written && (iomap->flags & IOMAP_F_NEW)) {
/* Deallocate blocks that were just allocated. */ /* Deallocate blocks that were just allocated. */
loff_t blockmask = i_blocksize(inode) - 1; loff_t hstart = round_up(pos + written, i_blocksize(inode));
loff_t end = (pos + length) & ~blockmask; loff_t hend = iomap->offset + iomap->length;
pos = (pos + written + blockmask) & ~blockmask; if (hstart < hend) {
if (pos < end) { truncate_pagecache_range(inode, hstart, hend - 1);
truncate_pagecache_range(inode, pos, end - 1); punch_hole(ip, hstart, hend - hstart);
punch_hole(ip, pos, end - pos);
} }
} }
......
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