From: Jens Axboe Subject: Re: [PATCH] block_write_full_page: switch synchronous writes to use WRITE_SYNC_PLUG Date: Thu, 9 Apr 2009 19:59:14 +0200 Message-ID: <20090409175914.GJ5178@kernel.dk> References: <20090407070835.GM5178@kernel.dk> <20090407002313.fcdd1da0.akpm@linux-foundation.org> <20090407075732.GO5178@kernel.dk> <20090407190913.GA31723@mit.edu> <20090407193239.GE5178@kernel.dk> <20090407214421.GA7031@mit.edu> <20090407221933.GB7031@mit.edu> <20090407160944.de3c5139.akpm@linux-foundation.org> <20090408080844.GW5178@kernel.dk> <20090408153428.6195a442.akpm@linux-foundation.org> Mime-Version: 1.0 Content-Type: text/plain; charset=us-ascii Cc: Theodore Tso , linux-kernel@vger.kernel.org, linux-ext4@vger.kernel.org, jack@suse.cz To: Andrew Morton Return-path: Received: from brick.kernel.dk ([93.163.65.50]:38886 "EHLO kernel.dk" rhost-flags-OK-OK-OK-OK) by vger.kernel.org with ESMTP id S1759082AbZDIR7P (ORCPT ); Thu, 9 Apr 2009 13:59:15 -0400 Content-Disposition: inline In-Reply-To: <20090408153428.6195a442.akpm@linux-foundation.org> Sender: linux-ext4-owner@vger.kernel.org List-ID: On Wed, Apr 08 2009, Andrew Morton wrote: > On Wed, 8 Apr 2009 10:08:44 +0200 Jens Axboe wrote: > > > > So how does WRITE_SYNC_PLUG differ from WRITE, and what effect does > > > this change have upon kernel behaviour? > > > > How about something like this. Comments welcome. > > It's lovely. > > > Should we move this to > > a dedicated header file? fs.h is amazingly cluttered as it is. > > Sometime, perhaps. > > > diff --git a/include/linux/fs.h b/include/linux/fs.h > > index 562d285..6b6597a 100644 > > --- a/include/linux/fs.h > > +++ b/include/linux/fs.h > > @@ -87,6 +87,57 @@ struct inodes_stat_t { > > */ > > #define FMODE_NOCMTIME ((__force fmode_t)2048) > > > > +/* > > + * The below are the various read and write types that we support. Some of > > + * them include behavioral modifiers that send information down to the > > + * block layer and IO scheduler. Terminology: > > + * > > + * The block layer uses device plugging to defer IO a little bit, in > > + * the hope that we will see more IO very shortly. This increases > > + * coalescing of adjacent IO and thus reduces the number of IOs we > > + * have to send to the device. It also allows for better queuing, > > + * if the IO isn't mergeable. If the caller is going to be waiting > > + * for the IO, then he must ensure that the device is unplugged so > > + * that the IO is dispatched to the driver. > > + * > > + * All IO is handled async in Linux. This is fine for background > > + * writes, but for reads or writes that someone waits for completion > > + * on, we want to notify the block layer and IO scheduler so that they > > + * know about it. That allows them to make better scheduling > > + * decisions. So when the below references 'sync' and 'async', it > > + * is referencing this priority hint. > > + * > > + * With that in mind, the available types are: > > + * > > + * READ A normal read operation. Device will be plugged. > > + * READ_SYNC A synchronous read. Device is not plugged, caller can > > + * immediately wait on this read without caring about > > + * unplugging. > > + * READA Used for read-ahead operations. Lower priority, and the > > + * block layer could (in theory) choose to ignore this > > + * request if it runs into resource problems. > > + * WRITE A normal async write. Device will be plugged. > > + * SWRITE Like WRITE, but a special case for ll_rw_block() that > > + * tells it to lock the buffer first. Normally a buffer > > + * must be locked before doing IO. > > + * WRITE_SYNC_PLUG Synchronous write. Identical to WRITE, but passes down > > + * the hint that someone will be waiting on this IO > > + * shortly. > > From the text, I'd expect WRITE_SYNC_PLUG to, err, unplug! You are still mixing up the sync hint and unplugging. I'll expand it a bit, I guess. > > + * WRITE_SYNC Like WRITE_SYNC_PLUG, but also unplugs the device > > + * immediately after submission. The write equivalent > > + * of READ_SYNC. > > But this contradicts my expectation. > > So what does WRITE_SYNC_PLUG really do dofferent from WRITE? It tells the IO scheduler that this write is really sync, not async. The key difference between the two is as written - the sync one will have someone waiting for its completion. What the IO scheduler does with the flag is its own business. In reality it means that the write is expedited, whereas async writes are done somewhat more lazily. > > + * WRITE_ODIRECT Special case write for O_DIRECT only. > > + * SWRITE_SYNC > > + * SWRITE_SYNC_PLUG Like WRITE_SYNC/WRITE_SYNC_PLUG, but locks the buffer. > > + * See SWRITE. > > + * WRITE_BARRIER Like WRITE, but tells the block layer that all > > + * previously submitted writes must be safely on storage > > + * before this one is started. Also guarantees that when > > + * this write is complete, it itself is also safely on > > + * storage. Prevents reordering of writes on both sides > > + * of this IO. > > + * > > + */ > > #define RW_MASK 1 > > #define RWA_MASK 2 > > #define READ 0 > > @@ -102,6 +153,11 @@ struct inodes_stat_t { > > (SWRITE | (1 << BIO_RW_SYNCIO) | (1 << BIO_RW_NOIDLE)) > > #define SWRITE_SYNC (SWRITE_SYNC_PLUG | (1 << BIO_RW_UNPLUG)) > > #define WRITE_BARRIER (WRITE | (1 << BIO_RW_BARRIER)) > > + > > +/* > > + * These aren't really reads or writes, they pass down information about > > + * parts of device that are now unused by the file system. > > + */ > > #define DISCARD_NOBARRIER (1 << BIO_RW_DISCARD) > > #define DISCARD_BARRIER ((1 << BIO_RW_DISCARD) | (1 << BIO_RW_BARRIER)) > -- Jens Axboe