ksft_test_result_pass("Setting/clearing PR_SET_MEMORY_MERGE works\n");
 }
 
+static int test_child_ksm(void)
+{
+       const unsigned int size = 2 * MiB;
+       char *map;
+
+       /* Test if KSM is enabled for the process. */
+       if (prctl(PR_GET_MEMORY_MERGE, 0, 0, 0, 0) != 1)
+               return -1;
+
+       /* Test if merge could really happen. */
+       map = __mmap_and_merge_range(0xcf, size, PROT_READ | PROT_WRITE, KSM_MERGE_NONE);
+       if (map == MAP_MERGE_FAIL)
+               return -2;
+       else if (map == MAP_MERGE_SKIP)
+               return -3;
+
+       munmap(map, size);
+       return 0;
+}
+
+static void test_child_ksm_err(int status)
+{
+       if (status == -1)
+               ksft_test_result_fail("unexpected PR_GET_MEMORY_MERGE result in child\n");
+       else if (status == -2)
+               ksft_test_result_fail("Merge in child failed\n");
+       else if (status == -3)
+               ksft_test_result_skip("Merge in child skipped\n");
+}
+
 /* Verify that prctl ksm flag is inherited. */
 static void test_prctl_fork(void)
 {
 
        child_pid = fork();
        if (!child_pid) {
-               exit(prctl(PR_GET_MEMORY_MERGE, 0, 0, 0, 0));
+               exit(test_child_ksm());
        } else if (child_pid < 0) {
                ksft_test_result_fail("fork() failed\n");
                return;
        if (waitpid(child_pid, &status, 0) < 0) {
                ksft_test_result_fail("waitpid() failed\n");
                return;
-       } else if (WEXITSTATUS(status) != 1) {
-               ksft_test_result_fail("unexpected PR_GET_MEMORY_MERGE result in child\n");
+       }
+
+       status = WEXITSTATUS(status);
+       if (status) {
+               test_child_ksm_err(status);
                return;
        }
 
        ksft_test_result_pass("PR_SET_MEMORY_MERGE value is inherited\n");
 }
 
-static int ksm_fork_exec_child(void)
-{
-       /* Test if KSM is enabled for the process. */
-       return prctl(PR_GET_MEMORY_MERGE, 0, 0, 0, 0) == 1;
-}
-
 static void test_prctl_fork_exec(void)
 {
        int ret, status;
                if (WIFEXITED(status)) {
                        status = WEXITSTATUS(status);
                        if (status) {
-                               ksft_test_result_fail("KSM not enabled\n");
+                               test_child_ksm_err(status);
                                return;
                        }
                } else {
        int err;
 
        if (argc > 1 && !strcmp(argv[1], FORK_EXEC_CHILD_PRG_NAME)) {
-               exit(ksm_fork_exec_child() == 1 ? 0 : 1);
+               exit(test_child_ksm());
        }
 
 #ifdef __NR_userfaultfd