[patch 6/9] mm: be sure to trim blocks

Previous message: [thread] [date] [author]
Next message: [thread] [date] [author]
From: Nick Piggin
Date: Monday, January 29, 2007 - 3:32 am

If prepare_write fails with AOP_TRUNCATED_PAGE, or if commit_write fails, then
we may have failed the write operation despite prepare_write having
instantiated blocks past i_size. Fix this, and consolidate the trimming into
one place.

Signed-off-by: Nick Piggin <npiggin@suse.de>

Index: linux-2.6/mm/filemap.c
===================================================================
--- linux-2.6.orig/mm/filemap.c
+++ linux-2.6/mm/filemap.c
@@ -1911,22 +1911,9 @@ generic_file_buffered_write(struct kiocb
 		}
 
 		status = a_ops->prepare_write(file, page, offset, offset+bytes);
-		if (unlikely(status)) {
-			loff_t isize = i_size_read(inode);
+		if (unlikely(status))
+			goto fs_write_aop_error;
 
-			if (status != AOP_TRUNCATED_PAGE)
-				unlock_page(page);
-			page_cache_release(page);
-			if (status == AOP_TRUNCATED_PAGE)
-				continue;
-			/*
-			 * prepare_write() may have instantiated a few blocks
-			 * outside i_size.  Trim these off again.
-			 */
-			if (pos + bytes > isize)
-				vmtruncate(inode, isize);
-			break;
-		}
 		if (likely(nr_segs == 1))
 			copied = filemap_copy_from_user(page, offset,
 							buf, bytes);
@@ -1935,40 +1922,53 @@ generic_file_buffered_write(struct kiocb
 						cur_iov, iov_offset, bytes);
 		flush_dcache_page(page);
 		status = a_ops->commit_write(file, page, offset, offset+bytes);
-		if (status == AOP_TRUNCATED_PAGE) {
-			page_cache_release(page);
-			continue;
+		if (unlikely(status < 0))
+			goto fs_write_aop_error;
+		if (unlikely(copied != bytes)) {
+			status = -EFAULT;
+			goto fs_write_aop_error;
 		}
-		if (likely(copied > 0)) {
-			if (!status)
-				status = copied;
+		if (unlikely(status > 0)) /* filesystem did partial write */
+			copied = status;
 
-			if (status >= 0) {
-				written += status;
-				count -= status;
-				pos += status;
-				buf += status;
-				if (unlikely(nr_segs > 1)) {
-					filemap_set_next_iovec(&cur_iov,
-							&iov_offset, status);
-					if (count)
-						buf = cur_iov->iov_base +
-							iov_offset;
-				} else {
-					iov_offset += status;
-				}
+		if (likely(copied > 0)) {
+			written += copied;
+			count -= copied;
+			pos += copied;
+			buf += copied;
+			if (unlikely(nr_segs > 1)) {
+				filemap_set_next_iovec(&cur_iov,
+						&iov_offset, copied);
+				if (count)
+					buf = cur_iov->iov_base + iov_offset;
+			} else {
+				iov_offset += copied;
 			}
 		}
-		if (unlikely(copied != bytes))
-			if (status >= 0)
-				status = -EFAULT;
 		unlock_page(page);
 		mark_page_accessed(page);
 		page_cache_release(page);
-		if (status < 0)
-			break;
 		balance_dirty_pages_ratelimited(mapping);
 		cond_resched();
+		continue;
+
+fs_write_aop_error:
+		if (status != AOP_TRUNCATED_PAGE)
+			unlock_page(page);
+		page_cache_release(page);
+
+		/*
+		 * prepare_write() may have instantiated a few blocks
+		 * outside i_size.  Trim these off again. Don't need
+		 * i_size_read because we hold i_mutex.
+		 */
+		if (pos + bytes > inode->i_size)
+			vmtruncate(inode, inode->i_size);
+		if (status == AOP_TRUNCATED_PAGE)
+			continue;
+		else
+			break;
+
 	} while (count);
 	*ppos = pos;
 
-
Previous message: [thread] [date] [author]
Next message: [thread] [date] [author]

Messages in current thread:
[patch 0/9] buffered write deadlock fix, Nick Piggin, (Mon Jan 29, 3:31 am)
[patch 1/9] fs: libfs buffered write leak fix, Nick Piggin, (Mon Jan 29, 3:31 am)
[patch 4/9] mm: generic_file_buffered_write cleanup, Nick Piggin, (Mon Jan 29, 3:32 am)
[patch 5/9] mm: debug write deadlocks, Nick Piggin, (Mon Jan 29, 3:32 am)
[patch 6/9] mm: be sure to trim blocks, Nick Piggin, (Mon Jan 29, 3:32 am)
[patch 7/9] mm: cleanup pagecache insertion operations, Nick Piggin, (Mon Jan 29, 3:32 am)
[patch 9/9] mm: fix pagecache write deadlocks, Nick Piggin, (Mon Jan 29, 3:33 am)
Re: [patch 9/9] mm: fix pagecache write deadlocks, Nick Piggin, (Mon Jan 29, 4:11 am)
Re: [patch 0/9] buffered write deadlock fix, Andrew Morton, (Tue Jan 30, 1:55 pm)
Re: [patch 0/9] buffered write deadlock fix, Andrew Morton, (Tue Jan 30, 4:21 pm)
Re: [patch 0/9] buffered write deadlock fix, Nick Piggin, (Tue Jan 30, 5:32 pm)
Re: [patch 0/9] buffered write deadlock fix, Nick Piggin, (Tue Jan 30, 6:31 pm)
Re: [patch 0/9] buffered write deadlock fix, Andrew Morton, (Fri Feb 2, 4:52 pm)
Re: [patch 1/9] fs: libfs buffered write leak fix, Andrew Morton, (Fri Feb 2, 4:52 pm)
Re: [patch 9/9] mm: fix pagecache write deadlocks, Andrew Morton, (Fri Feb 2, 4:53 pm)
Re: [patch 0/9] buffered write deadlock fix, Nick Piggin, (Fri Feb 2, 6:22 pm)
Re: [patch 1/9] fs: libfs buffered write leak fix, Nick Piggin, (Fri Feb 2, 6:33 pm)
Re: [patch 9/9] mm: fix pagecache write deadlocks, Nick Piggin, (Fri Feb 2, 6:38 pm)
Re: [patch 1/9] fs: libfs buffered write leak fix, Andrew Morton, (Fri Feb 2, 6:58 pm)
Re: [patch 1/9] fs: libfs buffered write leak fix, Nick Piggin, (Fri Feb 2, 7:09 pm)
Re: [patch 1/9] fs: libfs buffered write leak fix, Andrew Morton, (Fri Feb 2, 7:19 pm)
Re: [patch 1/9] fs: libfs buffered write leak fix, Nick Piggin, (Fri Feb 2, 7:28 pm)
Re: [patch 0/9] buffered write deadlock fix, Suparna Bhattacharya, (Fri Feb 2, 11:43 pm)
Re: [patch 0/9] buffered write deadlock fix, Fengguang Wu, (Sat Feb 3, 8:31 am)
Re: [patch 1/9] fs: libfs buffered write leak fix, Nick Piggin, (Sat Feb 3, 8:55 pm)