Skip to content

Commit f741bd7

Browse files
dhowellstorvalds
authored andcommitted
iov_iter: Fix iov_iter_extract_pages() with zero-sized entries
iov_iter_extract_pages() doesn't correctly handle skipping over initial zero-length entries in ITER_KVEC and ITER_BVEC-type iterators. The problem is that it accidentally reduces maxsize to 0 when it skipping and thus runs to the end of the array and returns 0. Fix this by sticking the calculated size-to-copy in a new variable rather than back in maxsize. Fixes: 7d58fe7 ("iov_iter: Add a function to extract a page list from an iterator") Signed-off-by: David Howells <dhowells@redhat.com> Reviewed-by: Christoph Hellwig <hch@lst.de> Cc: Christian Brauner <brauner@kernel.org> Cc: Jens Axboe <axboe@kernel.dk> Cc: Al Viro <viro@zeniv.linux.org.uk> Cc: David Hildenbrand <david@redhat.com> Cc: John Hubbard <jhubbard@nvidia.com> Signed-off-by: Linus Torvalds <torvalds@linux-foundation.org>
1 parent 6b8bb5b commit f741bd7

File tree

1 file changed

+15
-15
lines changed

1 file changed

+15
-15
lines changed

lib/iov_iter.c

Lines changed: 15 additions & 15 deletions
Original file line numberDiff line numberDiff line change
@@ -1654,14 +1654,14 @@ static ssize_t iov_iter_extract_bvec_pages(struct iov_iter *i,
16541654
size_t *offset0)
16551655
{
16561656
struct page **p, *page;
1657-
size_t skip = i->iov_offset, offset;
1657+
size_t skip = i->iov_offset, offset, size;
16581658
int k;
16591659

16601660
for (;;) {
16611661
if (i->nr_segs == 0)
16621662
return 0;
1663-
maxsize = min(maxsize, i->bvec->bv_len - skip);
1664-
if (maxsize)
1663+
size = min(maxsize, i->bvec->bv_len - skip);
1664+
if (size)
16651665
break;
16661666
i->iov_offset = 0;
16671667
i->nr_segs--;
@@ -1674,16 +1674,16 @@ static ssize_t iov_iter_extract_bvec_pages(struct iov_iter *i,
16741674
offset = skip % PAGE_SIZE;
16751675
*offset0 = offset;
16761676

1677-
maxpages = want_pages_array(pages, maxsize, offset, maxpages);
1677+
maxpages = want_pages_array(pages, size, offset, maxpages);
16781678
if (!maxpages)
16791679
return -ENOMEM;
16801680
p = *pages;
16811681
for (k = 0; k < maxpages; k++)
16821682
p[k] = page + k;
16831683

1684-
maxsize = min_t(size_t, maxsize, maxpages * PAGE_SIZE - offset);
1685-
iov_iter_advance(i, maxsize);
1686-
return maxsize;
1684+
size = min_t(size_t, size, maxpages * PAGE_SIZE - offset);
1685+
iov_iter_advance(i, size);
1686+
return size;
16871687
}
16881688

16891689
/*
@@ -1698,14 +1698,14 @@ static ssize_t iov_iter_extract_kvec_pages(struct iov_iter *i,
16981698
{
16991699
struct page **p, *page;
17001700
const void *kaddr;
1701-
size_t skip = i->iov_offset, offset, len;
1701+
size_t skip = i->iov_offset, offset, len, size;
17021702
int k;
17031703

17041704
for (;;) {
17051705
if (i->nr_segs == 0)
17061706
return 0;
1707-
maxsize = min(maxsize, i->kvec->iov_len - skip);
1708-
if (maxsize)
1707+
size = min(maxsize, i->kvec->iov_len - skip);
1708+
if (size)
17091709
break;
17101710
i->iov_offset = 0;
17111711
i->nr_segs--;
@@ -1717,13 +1717,13 @@ static ssize_t iov_iter_extract_kvec_pages(struct iov_iter *i,
17171717
offset = (unsigned long)kaddr & ~PAGE_MASK;
17181718
*offset0 = offset;
17191719

1720-
maxpages = want_pages_array(pages, maxsize, offset, maxpages);
1720+
maxpages = want_pages_array(pages, size, offset, maxpages);
17211721
if (!maxpages)
17221722
return -ENOMEM;
17231723
p = *pages;
17241724

17251725
kaddr -= offset;
1726-
len = offset + maxsize;
1726+
len = offset + size;
17271727
for (k = 0; k < maxpages; k++) {
17281728
size_t seg = min_t(size_t, len, PAGE_SIZE);
17291729

@@ -1737,9 +1737,9 @@ static ssize_t iov_iter_extract_kvec_pages(struct iov_iter *i,
17371737
kaddr += PAGE_SIZE;
17381738
}
17391739

1740-
maxsize = min_t(size_t, maxsize, maxpages * PAGE_SIZE - offset);
1741-
iov_iter_advance(i, maxsize);
1742-
return maxsize;
1740+
size = min_t(size_t, size, maxpages * PAGE_SIZE - offset);
1741+
iov_iter_advance(i, size);
1742+
return size;
17431743
}
17441744

17451745
/*

0 commit comments

Comments
 (0)