]> git.ipfire.org Git - thirdparty/kernel/linux.git/commitdiff
mshv: Fix error handling in mshv_region_pin
authorStanislav Kinsburskii <skinsburskii@linux.microsoft.com>
Tue, 17 Mar 2026 15:04:55 +0000 (15:04 +0000)
committerWei Liu <wei.liu@kernel.org>
Wed, 18 Mar 2026 16:18:49 +0000 (16:18 +0000)
The current error handling has two issues:

First, pin_user_pages_fast() can return a short pin count (less than
requested but greater than zero) when it cannot pin all requested pages.
This is treated as success, leading to partially pinned regions being
used, which causes memory corruption.

Second, when an error occurs mid-loop, already pinned pages from the
current batch are not properly accounted for before calling
mshv_region_invalidate_pages(), causing a page reference leak.

Treat short pins as errors and fix partial batch accounting before
cleanup.

Signed-off-by: Stanislav Kinsburskii <skinsburskii@linux.microsoft.com>
Reviewed-by: Michael Kelley <mhklinux@outlook.com>
Signed-off-by: Wei Liu <wei.liu@kernel.org>
drivers/hv/mshv_regions.c

index c28aac0726deb7c7d68ef3226fe932832eacb72b..fdffd4f002f6fdc4bc435426b81a075dc7c9c836 100644 (file)
@@ -314,15 +314,17 @@ int mshv_region_pin(struct mshv_mem_region *region)
                ret = pin_user_pages_fast(userspace_addr, nr_pages,
                                          FOLL_WRITE | FOLL_LONGTERM,
                                          pages);
-               if (ret < 0)
+               if (ret != nr_pages)
                        goto release_pages;
        }
 
        return 0;
 
 release_pages:
+       if (ret > 0)
+               done_count += ret;
        mshv_region_invalidate_pages(region, 0, done_count);
-       return ret;
+       return ret < 0 ? ret : -ENOMEM;
 }
 
 static int mshv_region_chunk_unmap(struct mshv_mem_region *region,