linux-kernel.vger.kernel.org archive mirror
 help / color / mirror / Atom feed
* [patch 7/10] handle concurrent block_write_full_page and set_page_dirty
@ 2002-05-05 20:57 Andrew Morton
  2002-05-05 21:58 ` Daniel Phillips
  0 siblings, 1 reply; 2+ messages in thread
From: Andrew Morton @ 2002-05-05 20:57 UTC (permalink / raw)
  To: Linus Torvalds; +Cc: lkml



set_page_dirty() runs without the page lock.  So
__block_write_full_page() needs to be able to cope with the page's
buffers being dirtied concurrently, on another CPU.

Do this with careful ordering and a test-and-set.



=====================================

--- 2.5.13/mm/page-writeback.c~writepage-versus-set_page_dirty	Sun May  5 13:32:01 2002
+++ 2.5.13-akpm/mm/page-writeback.c	Sun May  5 13:32:35 2002
@@ -480,6 +480,8 @@ int __set_page_dirty_buffers(struct page
 		do {
 			if (buffer_uptodate(bh))
 				set_buffer_dirty(bh);
+			else
+				buffer_error();
 			bh = bh->b_this_page;
 		} while (bh != head);
 	}
--- 2.5.13/fs/buffer.c~writepage-versus-set_page_dirty	Sun May  5 13:32:01 2002
+++ 2.5.13-akpm/fs/buffer.c	Sun May  5 13:32:35 2002
@@ -1288,6 +1288,16 @@ static int __block_write_full_page(struc
 					(1 << BH_Dirty)|(1 << BH_Uptodate));
 	}
 
+	/*
+	 * Be very careful.  We have no exclusion from __set_page_dirty_buffers
+	 * here, and the (potentially unmapped) buffers may become dirty at
+	 * any time.  If a buffer becomes dirty here after we've inspected it
+	 * then we just miss that fact, and the page stays dirty.
+	 *
+	 * Buffers outside i_size may be dirtied by __set_page_dirty_buffers;
+	 * handle that here by just cleaning them.
+	 */
+
 	block = page->index << (PAGE_CACHE_SHIFT - inode->i_blkbits);
 	head = page_buffers(page);
 	bh = head;
@@ -1298,15 +1308,11 @@ static int __block_write_full_page(struc
 	 */
 	do {
 		if (block > last_block) {
-			if (buffer_dirty(bh))
-				buffer_error();
+			clear_buffer_dirty(bh);
 			if (buffer_mapped(bh))
 				buffer_error();
 			/*
-			 * NOTE: this buffer can only be marked uptodate
-			 * because we know that block_write_full_page has
-			 * zeroed it out.  That seems unnecessary and may go
-			 * away.
+			 * The buffer was zeroed by block_write_full_page()
 			 */
 			set_buffer_uptodate(bh);
 		} else if (!buffer_mapped(bh) && buffer_dirty(bh)) {
@@ -1327,11 +1333,9 @@ static int __block_write_full_page(struc
 
 	do {
 		get_bh(bh);
-		if (buffer_dirty(bh)) {
+		if (buffer_mapped(bh) && buffer_dirty(bh)) {
 			lock_buffer(bh);
-			if (buffer_dirty(bh)) {
-				if (!buffer_mapped(bh))
-					buffer_error();
+			if (test_clear_buffer_dirty(bh)) {
 				if (!buffer_uptodate(bh))
 					buffer_error();
 				set_buffer_async_io(bh);
@@ -1353,7 +1357,6 @@ static int __block_write_full_page(struc
 	do {
 		struct buffer_head *next = bh->b_this_page;
 		if (buffer_async(bh)) {
-			clear_buffer_dirty(bh);
 			submit_bh(WRITE, bh);
 			nr_underway++;
 		}


-

^ permalink raw reply	[flat|nested] 2+ messages in thread

* Re: [patch 7/10] handle concurrent block_write_full_page and set_page_dirty
  2002-05-05 20:57 [patch 7/10] handle concurrent block_write_full_page and set_page_dirty Andrew Morton
@ 2002-05-05 21:58 ` Daniel Phillips
  0 siblings, 0 replies; 2+ messages in thread
From: Daniel Phillips @ 2002-05-05 21:58 UTC (permalink / raw)
  To: Andrew Morton, Linus Torvalds; +Cc: lkml

On Sunday 05 May 2002 22:57, Andrew Morton wrote:
> +	 * Buffers outside i_size may be dirtied by __set_page_dirty_buffers;
> +	 * handle that here by just cleaning them.

"Cleaning", to me at least, means writing out as a prelude to setting clean.
I'm think meant 'just setting them clean'.

-- 
Daniel

^ permalink raw reply	[flat|nested] 2+ messages in thread

end of thread, other threads:[~2002-05-05 21:59 UTC | newest]

Thread overview: 2+ messages (download: mbox.gz / follow: Atom feed)
-- links below jump to the message on this page --
2002-05-05 20:57 [patch 7/10] handle concurrent block_write_full_page and set_page_dirty Andrew Morton
2002-05-05 21:58 ` Daniel Phillips

This is a public inbox, see mirroring instructions
for how to clone and mirror all data and code used for this inbox;
as well as URLs for NNTP newsgroup(s).