diff options
Diffstat (limited to 'mm/bounce.c')
-rw-r--r-- | mm/bounce.c | 75 |
1 files changed, 20 insertions, 55 deletions
diff --git a/mm/bounce.c b/mm/bounce.c index a5c2ec3589cb..c9f0a4339a7d 100644 --- a/mm/bounce.c +++ b/mm/bounce.c | |||
@@ -101,7 +101,7 @@ static void copy_to_high_bio_irq(struct bio *to, struct bio *from) | |||
101 | struct bio_vec *tovec, *fromvec; | 101 | struct bio_vec *tovec, *fromvec; |
102 | int i; | 102 | int i; |
103 | 103 | ||
104 | __bio_for_each_segment(tovec, to, i, 0) { | 104 | bio_for_each_segment(tovec, to, i) { |
105 | fromvec = from->bi_io_vec + i; | 105 | fromvec = from->bi_io_vec + i; |
106 | 106 | ||
107 | /* | 107 | /* |
@@ -134,7 +134,7 @@ static void bounce_end_io(struct bio *bio, mempool_t *pool, int err) | |||
134 | /* | 134 | /* |
135 | * free up bounce indirect pages used | 135 | * free up bounce indirect pages used |
136 | */ | 136 | */ |
137 | __bio_for_each_segment(bvec, bio, i, 0) { | 137 | bio_for_each_segment_all(bvec, bio, i) { |
138 | org_vec = bio_orig->bi_io_vec + i; | 138 | org_vec = bio_orig->bi_io_vec + i; |
139 | if (bvec->bv_page == org_vec->bv_page) | 139 | if (bvec->bv_page == org_vec->bv_page) |
140 | continue; | 140 | continue; |
@@ -199,78 +199,43 @@ static int must_snapshot_stable_pages(struct request_queue *q, struct bio *bio) | |||
199 | static void __blk_queue_bounce(struct request_queue *q, struct bio **bio_orig, | 199 | static void __blk_queue_bounce(struct request_queue *q, struct bio **bio_orig, |
200 | mempool_t *pool, int force) | 200 | mempool_t *pool, int force) |
201 | { | 201 | { |
202 | struct page *page; | 202 | struct bio *bio; |
203 | struct bio *bio = NULL; | 203 | int rw = bio_data_dir(*bio_orig); |
204 | int i, rw = bio_data_dir(*bio_orig); | ||
205 | struct bio_vec *to, *from; | 204 | struct bio_vec *to, *from; |
205 | unsigned i; | ||
206 | 206 | ||
207 | bio_for_each_segment(from, *bio_orig, i) { | 207 | bio_for_each_segment(from, *bio_orig, i) |
208 | page = from->bv_page; | 208 | if (page_to_pfn(from->bv_page) > queue_bounce_pfn(q)) |
209 | goto bounce; | ||
209 | 210 | ||
210 | /* | 211 | return; |
211 | * is destination page below bounce pfn? | 212 | bounce: |
212 | */ | 213 | bio = bio_clone_bioset(*bio_orig, GFP_NOIO, fs_bio_set); |
213 | if (page_to_pfn(page) <= queue_bounce_pfn(q) && !force) | ||
214 | continue; | ||
215 | |||
216 | /* | ||
217 | * irk, bounce it | ||
218 | */ | ||
219 | if (!bio) { | ||
220 | unsigned int cnt = (*bio_orig)->bi_vcnt; | ||
221 | 214 | ||
222 | bio = bio_alloc(GFP_NOIO, cnt); | 215 | bio_for_each_segment_all(to, bio, i) { |
223 | memset(bio->bi_io_vec, 0, cnt * sizeof(struct bio_vec)); | 216 | struct page *page = to->bv_page; |
224 | } | ||
225 | |||
226 | 217 | ||
227 | to = bio->bi_io_vec + i; | 218 | if (page_to_pfn(page) <= queue_bounce_pfn(q) && !force) |
219 | continue; | ||
228 | 220 | ||
229 | to->bv_page = mempool_alloc(pool, q->bounce_gfp); | ||
230 | to->bv_len = from->bv_len; | ||
231 | to->bv_offset = from->bv_offset; | ||
232 | inc_zone_page_state(to->bv_page, NR_BOUNCE); | 221 | inc_zone_page_state(to->bv_page, NR_BOUNCE); |
222 | to->bv_page = mempool_alloc(pool, q->bounce_gfp); | ||
233 | 223 | ||
234 | if (rw == WRITE) { | 224 | if (rw == WRITE) { |
235 | char *vto, *vfrom; | 225 | char *vto, *vfrom; |
236 | 226 | ||
237 | flush_dcache_page(from->bv_page); | 227 | flush_dcache_page(page); |
228 | |||
238 | vto = page_address(to->bv_page) + to->bv_offset; | 229 | vto = page_address(to->bv_page) + to->bv_offset; |
239 | vfrom = kmap(from->bv_page) + from->bv_offset; | 230 | vfrom = kmap_atomic(page) + to->bv_offset; |
240 | memcpy(vto, vfrom, to->bv_len); | 231 | memcpy(vto, vfrom, to->bv_len); |
241 | kunmap(from->bv_page); | 232 | kunmap_atomic(vfrom); |
242 | } | 233 | } |
243 | } | 234 | } |
244 | 235 | ||
245 | /* | ||
246 | * no pages bounced | ||
247 | */ | ||
248 | if (!bio) | ||
249 | return; | ||
250 | |||
251 | trace_block_bio_bounce(q, *bio_orig); | 236 | trace_block_bio_bounce(q, *bio_orig); |
252 | 237 | ||
253 | /* | ||
254 | * at least one page was bounced, fill in possible non-highmem | ||
255 | * pages | ||
256 | */ | ||
257 | __bio_for_each_segment(from, *bio_orig, i, 0) { | ||
258 | to = bio_iovec_idx(bio, i); | ||
259 | if (!to->bv_page) { | ||
260 | to->bv_page = from->bv_page; | ||
261 | to->bv_len = from->bv_len; | ||
262 | to->bv_offset = from->bv_offset; | ||
263 | } | ||
264 | } | ||
265 | |||
266 | bio->bi_bdev = (*bio_orig)->bi_bdev; | ||
267 | bio->bi_flags |= (1 << BIO_BOUNCED); | 238 | bio->bi_flags |= (1 << BIO_BOUNCED); |
268 | bio->bi_sector = (*bio_orig)->bi_sector; | ||
269 | bio->bi_rw = (*bio_orig)->bi_rw; | ||
270 | |||
271 | bio->bi_vcnt = (*bio_orig)->bi_vcnt; | ||
272 | bio->bi_idx = (*bio_orig)->bi_idx; | ||
273 | bio->bi_size = (*bio_orig)->bi_size; | ||
274 | 239 | ||
275 | if (pool == page_pool) { | 240 | if (pool == page_pool) { |
276 | bio->bi_end_io = bounce_end_io_write; | 241 | bio->bi_end_io = bounce_end_io_write; |