iov_iter: use bvec iterator to implement iterate_bvec()

bvec has one native/mature iterator for long time, so not
necessary to use the reinvented wheel for iterating bvecs
in lib/iov_iter.c.

Two ITER_BVEC test cases are run:
	- xfstest(-g auto) on loop dio/aio, no regression found
	- swap file works well under extreme stress(stress-ng --all 64 -t
	  800 -v), and lots of OOMs are triggerd, and the whole
	system still survives

Reviewed-by: Christoph Hellwig <hch@lst.de>
Signed-off-by: Ming Lei <ming.lei@canonical.com>
Tested-by: Hannes Reinecke <hare@suse.com>
Signed-off-by: Jens Axboe <axboe@fb.com>
This commit is contained in:
Ming Lei 2016-05-30 21:34:32 +08:00 committed by Jens Axboe
parent 80f162ff06
commit 1bdc76aea1

View File

@ -56,37 +56,24 @@
n = wanted; \ n = wanted; \
} }
#define iterate_bvec(i, n, __v, __p, skip, STEP) { \ #define iterate_bvec(i, n, __v, __bi, skip, STEP) { \
size_t wanted = n; \ struct bvec_iter __start; \
__p = i->bvec; \ __start.bi_size = n; \
__v.bv_len = min_t(size_t, n, __p->bv_len - skip); \ __start.bi_bvec_done = skip; \
if (likely(__v.bv_len)) { \ __start.bi_idx = 0; \
__v.bv_page = __p->bv_page; \ for_each_bvec(__v, i->bvec, __bi, __start) { \
__v.bv_offset = __p->bv_offset + skip; \ if (!__v.bv_len) \
(void)(STEP); \
skip += __v.bv_len; \
n -= __v.bv_len; \
} \
while (unlikely(n)) { \
__p++; \
__v.bv_len = min_t(size_t, n, __p->bv_len); \
if (unlikely(!__v.bv_len)) \
continue; \ continue; \
__v.bv_page = __p->bv_page; \
__v.bv_offset = __p->bv_offset; \
(void)(STEP); \ (void)(STEP); \
skip = __v.bv_len; \
n -= __v.bv_len; \
} \ } \
n = wanted; \
} }
#define iterate_all_kinds(i, n, v, I, B, K) { \ #define iterate_all_kinds(i, n, v, I, B, K) { \
size_t skip = i->iov_offset; \ size_t skip = i->iov_offset; \
if (unlikely(i->type & ITER_BVEC)) { \ if (unlikely(i->type & ITER_BVEC)) { \
const struct bio_vec *bvec; \
struct bio_vec v; \ struct bio_vec v; \
iterate_bvec(i, n, v, bvec, skip, (B)) \ struct bvec_iter __bi; \
iterate_bvec(i, n, v, __bi, skip, (B)) \
} else if (unlikely(i->type & ITER_KVEC)) { \ } else if (unlikely(i->type & ITER_KVEC)) { \
const struct kvec *kvec; \ const struct kvec *kvec; \
struct kvec v; \ struct kvec v; \
@ -104,15 +91,13 @@
if (i->count) { \ if (i->count) { \
size_t skip = i->iov_offset; \ size_t skip = i->iov_offset; \
if (unlikely(i->type & ITER_BVEC)) { \ if (unlikely(i->type & ITER_BVEC)) { \
const struct bio_vec *bvec; \ const struct bio_vec *bvec = i->bvec; \
struct bio_vec v; \ struct bio_vec v; \
iterate_bvec(i, n, v, bvec, skip, (B)) \ struct bvec_iter __bi; \
if (skip == bvec->bv_len) { \ iterate_bvec(i, n, v, __bi, skip, (B)) \
bvec++; \ i->bvec = __bvec_iter_bvec(i->bvec, __bi); \
skip = 0; \ i->nr_segs -= i->bvec - bvec; \
} \ skip = __bi.bi_bvec_done; \
i->nr_segs -= bvec - i->bvec; \
i->bvec = bvec; \
} else if (unlikely(i->type & ITER_KVEC)) { \ } else if (unlikely(i->type & ITER_KVEC)) { \
const struct kvec *kvec; \ const struct kvec *kvec; \
struct kvec v; \ struct kvec v; \