diff options
author | Darrick J. Wong <darrick.wong@oracle.com> | 2013-02-21 19:42:55 -0500 |
---|---|---|
committer | Linus Torvalds <torvalds@linux-foundation.org> | 2013-02-21 20:22:20 -0500 |
commit | ffecfd1a72fccfcee3dabb99b9ecba9735318f90 (patch) | |
tree | a5c3caf67249ec811a2c2c95678d9349fd8e2412 /mm/bounce.c | |
parent | 13575ca14fcdacd1ad914d00bc63eb4d96280986 (diff) |
block: optionally snapshot page contents to provide stable pages during write
This provides a band-aid to provide stable page writes on jbd without
needing to backport the fixed locking and page writeback bit handling
schemes of jbd2. The band-aid works by using bounce buffers to snapshot
page contents instead of waiting.
For those wondering about the ext3 bandage -- fixing the jbd locking
(which was done as part of ext4dev years ago) is a lot of surgery, and
setting PG_writeback on data pages when we actually hold the page lock
dropped ext3 performance by nearly an order of magnitude. If we're
going to migrate iscsi and raid to use stable page writes, the
complaints about high latency will likely return. We might as well
centralize their page snapshotting thing to one place.
Signed-off-by: Darrick J. Wong <darrick.wong@oracle.com>
Tested-by: Andy Lutomirski <luto@amacapital.net>
Cc: Adrian Hunter <adrian.hunter@intel.com>
Cc: Artem Bityutskiy <dedekind1@gmail.com>
Reviewed-by: Jan Kara <jack@suse.cz>
Cc: Joel Becker <jlbec@evilplan.org>
Cc: Mark Fasheh <mfasheh@suse.com>
Cc: Steven Whitehouse <swhiteho@redhat.com>
Cc: Jens Axboe <axboe@kernel.dk>
Cc: Eric Van Hensbergen <ericvh@gmail.com>
Cc: Ron Minnich <rminnich@sandia.gov>
Cc: Latchesar Ionkov <lucho@ionkov.net>
Signed-off-by: Andrew Morton <akpm@linux-foundation.org>
Signed-off-by: Linus Torvalds <torvalds@linux-foundation.org>
Diffstat (limited to 'mm/bounce.c')
-rw-r--r-- | mm/bounce.c | 48 |
1 files changed, 44 insertions, 4 deletions
diff --git a/mm/bounce.c b/mm/bounce.c index 042086775561..5f8901768602 100644 --- a/mm/bounce.c +++ b/mm/bounce.c | |||
@@ -178,8 +178,45 @@ static void bounce_end_io_read_isa(struct bio *bio, int err) | |||
178 | __bounce_end_io_read(bio, isa_page_pool, err); | 178 | __bounce_end_io_read(bio, isa_page_pool, err); |
179 | } | 179 | } |
180 | 180 | ||
181 | #ifdef CONFIG_NEED_BOUNCE_POOL | ||
182 | static int must_snapshot_stable_pages(struct request_queue *q, struct bio *bio) | ||
183 | { | ||
184 | struct page *page; | ||
185 | struct backing_dev_info *bdi; | ||
186 | struct address_space *mapping; | ||
187 | struct bio_vec *from; | ||
188 | int i; | ||
189 | |||
190 | if (bio_data_dir(bio) != WRITE) | ||
191 | return 0; | ||
192 | |||
193 | if (!bdi_cap_stable_pages_required(&q->backing_dev_info)) | ||
194 | return 0; | ||
195 | |||
196 | /* | ||
197 | * Based on the first page that has a valid mapping, decide whether or | ||
198 | * not we have to employ bounce buffering to guarantee stable pages. | ||
199 | */ | ||
200 | bio_for_each_segment(from, bio, i) { | ||
201 | page = from->bv_page; | ||
202 | mapping = page_mapping(page); | ||
203 | if (!mapping) | ||
204 | continue; | ||
205 | bdi = mapping->backing_dev_info; | ||
206 | return mapping->host->i_sb->s_flags & MS_SNAP_STABLE; | ||
207 | } | ||
208 | |||
209 | return 0; | ||
210 | } | ||
211 | #else | ||
212 | static int must_snapshot_stable_pages(struct request_queue *q, struct bio *bio) | ||
213 | { | ||
214 | return 0; | ||
215 | } | ||
216 | #endif /* CONFIG_NEED_BOUNCE_POOL */ | ||
217 | |||
181 | static void __blk_queue_bounce(struct request_queue *q, struct bio **bio_orig, | 218 | static void __blk_queue_bounce(struct request_queue *q, struct bio **bio_orig, |
182 | mempool_t *pool) | 219 | mempool_t *pool, int force) |
183 | { | 220 | { |
184 | struct page *page; | 221 | struct page *page; |
185 | struct bio *bio = NULL; | 222 | struct bio *bio = NULL; |
@@ -192,7 +229,7 @@ static void __blk_queue_bounce(struct request_queue *q, struct bio **bio_orig, | |||
192 | /* | 229 | /* |
193 | * is destination page below bounce pfn? | 230 | * is destination page below bounce pfn? |
194 | */ | 231 | */ |
195 | if (page_to_pfn(page) <= queue_bounce_pfn(q)) | 232 | if (page_to_pfn(page) <= queue_bounce_pfn(q) && !force) |
196 | continue; | 233 | continue; |
197 | 234 | ||
198 | /* | 235 | /* |
@@ -270,6 +307,7 @@ static void __blk_queue_bounce(struct request_queue *q, struct bio **bio_orig, | |||
270 | 307 | ||
271 | void blk_queue_bounce(struct request_queue *q, struct bio **bio_orig) | 308 | void blk_queue_bounce(struct request_queue *q, struct bio **bio_orig) |
272 | { | 309 | { |
310 | int must_bounce; | ||
273 | mempool_t *pool; | 311 | mempool_t *pool; |
274 | 312 | ||
275 | /* | 313 | /* |
@@ -278,13 +316,15 @@ void blk_queue_bounce(struct request_queue *q, struct bio **bio_orig) | |||
278 | if (!bio_has_data(*bio_orig)) | 316 | if (!bio_has_data(*bio_orig)) |
279 | return; | 317 | return; |
280 | 318 | ||
319 | must_bounce = must_snapshot_stable_pages(q, *bio_orig); | ||
320 | |||
281 | /* | 321 | /* |
282 | * for non-isa bounce case, just check if the bounce pfn is equal | 322 | * for non-isa bounce case, just check if the bounce pfn is equal |
283 | * to or bigger than the highest pfn in the system -- in that case, | 323 | * to or bigger than the highest pfn in the system -- in that case, |
284 | * don't waste time iterating over bio segments | 324 | * don't waste time iterating over bio segments |
285 | */ | 325 | */ |
286 | if (!(q->bounce_gfp & GFP_DMA)) { | 326 | if (!(q->bounce_gfp & GFP_DMA)) { |
287 | if (queue_bounce_pfn(q) >= blk_max_pfn) | 327 | if (queue_bounce_pfn(q) >= blk_max_pfn && !must_bounce) |
288 | return; | 328 | return; |
289 | pool = page_pool; | 329 | pool = page_pool; |
290 | } else { | 330 | } else { |
@@ -295,7 +335,7 @@ void blk_queue_bounce(struct request_queue *q, struct bio **bio_orig) | |||
295 | /* | 335 | /* |
296 | * slow path | 336 | * slow path |
297 | */ | 337 | */ |
298 | __blk_queue_bounce(q, bio_orig, pool); | 338 | __blk_queue_bounce(q, bio_orig, pool, must_bounce); |
299 | } | 339 | } |
300 | 340 | ||
301 | EXPORT_SYMBOL(blk_queue_bounce); | 341 | EXPORT_SYMBOL(blk_queue_bounce); |