]> git.ipfire.org Git - thirdparty/kernel/linux.git/commitdiff
iov_iter: fix copy_page_to_iter_nofault()
authorOmar Sandoval <osandov@fb.com>
Fri, 17 Nov 2023 21:38:46 +0000 (13:38 -0800)
committerChristian Brauner <brauner@kernel.org>
Sat, 18 Nov 2023 15:42:07 +0000 (16:42 +0100)
The recent conversion to inline functions made two mistakes:

1. It tries to copy the full amount requested (bytes), not just what's
   available in the kmap'd page (n).
2. It's not applying the offset in the first page.

Note that copy_page_to_iter_nofault() is only used by /proc/kcore. This
was detected by drgn's test suite.

Fixes: f1982740f5e7 ("iov_iter: Convert iterate*() to inline funcs")
Signed-off-by: Omar Sandoval <osandov@fb.com>
Link: https://lore.kernel.org/r/c1616e06b5248013cbbb1881bb4fef85a7a69ccb.1700257019.git.osandov@fb.com
Acked-by: David Howells <dhowells@redhat.com>
Signed-off-by: Christian Brauner <brauner@kernel.org>
lib/iov_iter.c

index de7d11cf4c6356deccc37f180fa992dbe4d4c7b0..8ff6824a100539a7894db06edf342c3c362099b8 100644 (file)
@@ -409,7 +409,7 @@ size_t copy_page_to_iter_nofault(struct page *page, unsigned offset, size_t byte
                void *kaddr = kmap_local_page(page);
                size_t n = min(bytes, (size_t)PAGE_SIZE - offset);
 
-               n = iterate_and_advance(i, bytes, kaddr,
+               n = iterate_and_advance(i, n, kaddr + offset,
                                        copy_to_user_iter_nofault,
                                        memcpy_to_iter);
                kunmap_local(kaddr);