Add a migration test to guest_memfd_test, run for the
MMAP | INIT_SHARED configuration on systems with at least two NUMA
nodes (skipped otherwise).

Migrate every folio from node 0 to node 1 with move_pages(2) and
check both the resulting node and the data. Migrate them back and
re-check the data.

Signed-off-by: Shivank Garg <[email protected]>
---
 tools/testing/selftests/kvm/guest_memfd_test.c | 77 ++++++++++++++++++++++++++
 1 file changed, 77 insertions(+)

diff --git a/tools/testing/selftests/kvm/guest_memfd_test.c 
b/tools/testing/selftests/kvm/guest_memfd_test.c
index 
832ef4dfb99faa4411af847d21eb426c34342434..04931d3add46cb117fe5b093ed48f838cb124542
 100644
--- a/tools/testing/selftests/kvm/guest_memfd_test.c
+++ b/tools/testing/selftests/kvm/guest_memfd_test.c
@@ -76,6 +76,82 @@ static void test_mmap_supported(int fd, size_t total_size)
        kvm_munmap(mem, total_size);
 }
 
+/*
+ * Each page is filled with a distinct byte (its index). Check every byte that
+ * data is intact after migration.
+ */
+static void verify_page(const char *page, int page_idx, size_t size,
+                       const char *when)
+{
+       char expected = (char)(page_idx & 0xff);
+       size_t off;
+
+       for (off = 0; off < size; off++)
+               TEST_ASSERT(page[off] == expected,
+                           "Page %d corrupted at offset %zu %s", page_idx, 
off, when);
+}
+
+static void test_migrate_folio(int fd, size_t total_size)
+{
+       const unsigned long nodemask_0 = 1; /* nid: 0 */
+       unsigned long maxnode = BITS_PER_TYPE(nodemask_0);
+       int page_count = total_size / page_size;
+       void **addr;
+       int *status, *nodes;
+       char *mem;
+       int i;
+
+       if (!is_multi_numa_node_system())
+               return;
+
+       mem = kvm_mmap(total_size, PROT_READ | PROT_WRITE, MAP_SHARED, fd);
+
+       addr = calloc(page_count, sizeof(*addr));
+       status = calloc(page_count, sizeof(*status));
+       nodes = calloc(page_count, sizeof(*nodes));
+       TEST_ASSERT(addr && status && nodes, "Failed to allocate page arrays");
+
+       /* Allocate all folios on node 0 and fill each with a known pattern. */
+       kvm_mbind(mem, total_size, MPOL_BIND, &nodemask_0, maxnode, 0);
+       for (i = 0; i < page_count; i++) {
+               memset(mem + i * page_size, (char)(i & 0xff), page_size);
+               addr[i] = mem + i * page_size;
+       }
+
+       kvm_move_pages(0, page_count, addr, NULL, status, 0);
+       for (i = 0; i < page_count; i++)
+               TEST_ASSERT(status[i] == 0, "Page %d should be on node 0", i);
+
+       /* Migrate node 0 -> 1, then check both the location and the data. */
+       for (i = 0; i < page_count; i++)
+               nodes[i] = 1;
+       kvm_move_pages(0, page_count, addr, nodes, status, MPOL_MF_MOVE);
+
+       kvm_move_pages(0, page_count, addr, NULL, status, 0);
+       for (i = 0; i < page_count; i++)
+               TEST_ASSERT(status[i] == 1,
+                           "Page %d should be on node 1 after migration", i);
+       for (i = 0; i < page_count; i++)
+               verify_page(mem + i * page_size, i, page_size, "after 
migration");
+
+       /* Migrate back node 1 -> 0, then re-check the location and the data. */
+       for (i = 0; i < page_count; i++)
+               nodes[i] = 0;
+       kvm_move_pages(0, page_count, addr, nodes, status, MPOL_MF_MOVE);
+
+       kvm_move_pages(0, page_count, addr, NULL, status, 0);
+       for (i = 0; i < page_count; i++)
+               TEST_ASSERT(status[i] == 0,
+                           "Page %d should be on node 0 after round-trip", i);
+       for (i = 0; i < page_count; i++)
+               verify_page(mem + i * page_size, i, page_size, "after 
round-trip");
+
+       free(addr);
+       free(status);
+       free(nodes);
+       kvm_munmap(mem, total_size);
+}
+
 static void test_mbind(int fd, size_t total_size)
 {
        const unsigned long nodemask_0 = 1; /* nid: 0 */
@@ -434,6 +510,7 @@ static void __test_guest_memfd(struct kvm_vm *vm, u64 flags)
                        gmem_test(fault_overflow, vm, flags);
                        gmem_test(numa_allocation, vm, flags);
                        __gmem_test(collapse, vm, flags, pmd_size);
+                       gmem_test(migrate_folio, vm, flags);
                } else {
                        gmem_test(fault_private, vm, flags);
                }

-- 
2.43.0


Reply via email to