Linux-HyperV List
 help / color / mirror / Atom feed
* [PATCH net] hv_netvsc: use kmap_local_page in netvsc_copy_to_send_buf
@ 2026-06-02 15:52 LeantionX
  2026-06-02 22:54 ` [EXTERNAL] " Haiyang Zhang
                   ` (2 more replies)
  0 siblings, 3 replies; 8+ messages in thread
From: LeantionX @ 2026-06-02 15:52 UTC (permalink / raw)
  To: netdev
  Cc: linux-hyperv, kys, haiyangz, wei.liu, decui, longli,
	andrew+netdev, kuba, pabeni, edumazet, davem, stable,
	linux-kernel, Anton Leontev

From: Anton Leontev <leontyevantony@gmail.com>

netvsc_copy_to_send_buf() copies skb fragment pages into the shared
VMBus send buffer using phys_to_virt() on the fragment PFN. On 32-bit
x86 with CONFIG_HIGHMEM=y, phys_to_virt() (i.e. __va()) is only valid
for LOWMEM addresses below 896 MiB. For a HIGHMEM page it returns an
address that has no kernel page table entry and lies outside the
kernel direct map, so the subsequent memcpy() faults. As this happens
on the transmit softirq path, the fault is fatal.

A HIGHMEM fragment reaches this path whenever the page backing an skb
fragment lives above the LOWMEM boundary, which is common on a 32-bit
guest with several GiB of RAM (for example when the in-kernel NFS
server splices page cache pages directly into the reply skb).

Map the fragment page on demand with kmap_local_page()/kunmap_local()
instead. Using pfn_to_page() on pb[i].pfn maps exactly the page
described by the page buffer entry. On configurations without HIGHMEM
(amd64, i386 without CONFIG_HIGHMEM) kmap_local_page() reduces to
page_address(), so this is a no-op there.

Fixes: c25aaf814a63 ("hyperv: Enable sendbuf mechanism on the send path")
Cc: stable@vger.kernel.org
Signed-off-by: Anton Leontev <leontyevantony@gmail.com>
---
 drivers/net/hyperv/netvsc.c | 5 ++++-
 1 file changed, 4 insertions(+), 1 deletion(-)

diff --git a/drivers/net/hyperv/netvsc.c b/drivers/net/hyperv/netvsc.c
index 59e95341f9b1..6984f6c97257 100644
--- a/drivers/net/hyperv/netvsc.c
+++ b/drivers/net/hyperv/netvsc.c
@@ -12,6 +12,7 @@
 #include <linux/sched.h>
 #include <linux/wait.h>
 #include <linux/mm.h>
+#include <linux/highmem.h>
 #include <linux/delay.h>
 #include <linux/io.h>
 #include <linux/slab.h>
@@ -965,11 +966,13 @@ static void netvsc_copy_to_send_buf(struct netvsc_device *net_device,
 	}
 
 	for (i = 0; i < page_count; i++) {
-		char *src = phys_to_virt(pb[i].pfn << HV_HYP_PAGE_SHIFT);
+		struct page *page = pfn_to_page(pb[i].pfn);
+		char *src = kmap_local_page(page);
 		u32 offset = pb[i].offset;
 		u32 len = pb[i].len;
 
 		memcpy(dest, (src + offset), len);
+		kunmap_local(src);
 		dest += len;
 	}
 
-- 
2.43.0


^ permalink raw reply related	[flat|nested] 8+ messages in thread

end of thread, other threads:[~2026-06-04 16:59 UTC | newest]

Thread overview: 8+ messages (download: mbox.gz follow: Atom feed
-- links below jump to the message on this page --
2026-06-02 15:52 [PATCH net] hv_netvsc: use kmap_local_page in netvsc_copy_to_send_buf LeantionX
2026-06-02 22:54 ` [EXTERNAL] " Haiyang Zhang
2026-06-03  5:29   ` Anton Leontev
2026-06-03 15:53 ` sashiko-bot
2026-06-03 16:29   ` Anton Leontev
2026-06-03 16:38 ` [PATCH net v2] " Anton Leontev
2026-06-04 16:39   ` sashiko-bot
2026-06-04 16:59     ` Anton Leontev

This is a public inbox, see mirroring instructions
for how to clone and mirror all data and code used for this inbox