[LTP] [PATCH 09/13] testcases/kernel/mem: Move KSM bits to ksm tests

Cyril Hrubis chrubis@suse.cz
Wed Dec 18 19:45:14 CET 2024


Moves the ksm helpers into a ksm/ksm_test.h since these are not used by
any other but ksm tests.

Signed-off-by: Cyril Hrubis <chrubis@suse.cz>
---
 testcases/kernel/mem/include/mem.h    |   5 -
 testcases/kernel/mem/ksm/Makefile     |   2 +-
 testcases/kernel/mem/ksm/ksm01.c      |   2 +-
 testcases/kernel/mem/ksm/ksm02.c      |   2 +-
 testcases/kernel/mem/ksm/ksm03.c      |   2 +-
 testcases/kernel/mem/ksm/ksm04.c      |   2 +-
 testcases/kernel/mem/ksm/ksm05.c      |   2 +-
 testcases/kernel/mem/ksm/ksm06.c      |   4 +-
 testcases/kernel/mem/ksm/ksm07.c      |   3 +-
 testcases/kernel/mem/ksm/ksm_common.h |   3 +-
 testcases/kernel/mem/ksm/ksm_test.h   | 299 ++++++++++++++++++++++++++
 testcases/kernel/mem/lib/mem.c        | 287 ------------------------
 12 files changed, 312 insertions(+), 301 deletions(-)
 create mode 100644 testcases/kernel/mem/ksm/ksm_test.h

diff --git a/testcases/kernel/mem/include/mem.h b/testcases/kernel/mem/include/mem.h
index 5ebedc175..25389f34c 100644
--- a/testcases/kernel/mem/include/mem.h
+++ b/testcases/kernel/mem/include/mem.h
@@ -16,11 +16,6 @@
 
 /* KSM */
 
-void create_same_memory(int size, int num, int unit);
-void test_ksm_merge_across_nodes(unsigned long nr_pages);
-void ksm_group_check(int run, int pg_shared, int pg_sharing, int pg_volatile,
-                     int pg_unshared, int sleep_msecs, int pages_to_scan);
-
 /* HUGETLB */
 
 #define PATH_SHMMAX		"/proc/sys/kernel/shmmax"
diff --git a/testcases/kernel/mem/ksm/Makefile b/testcases/kernel/mem/ksm/Makefile
index 6aba73e61..2af02a274 100644
--- a/testcases/kernel/mem/ksm/Makefile
+++ b/testcases/kernel/mem/ksm/Makefile
@@ -7,5 +7,5 @@ LTPLIBS = numa
 ksm06: LTPLDLIBS = -lltpnuma
 
 include $(top_srcdir)/include/mk/testcases.mk
-include $(top_srcdir)/testcases/kernel/mem/include/libmem.mk
+include $(top_srcdir)/testcases/kernel/include/lib.mk
 include $(top_srcdir)/include/mk/generic_leaf_target.mk
diff --git a/testcases/kernel/mem/ksm/ksm01.c b/testcases/kernel/mem/ksm/ksm01.c
index e2d3d9e00..a22e4830f 100644
--- a/testcases/kernel/mem/ksm/ksm01.c
+++ b/testcases/kernel/mem/ksm/ksm01.c
@@ -56,7 +56,7 @@
 #include <stdlib.h>
 #include <string.h>
 #include <unistd.h>
-#include "mem.h"
+#include "tst_test.h"
 #include "ksm_common.h"
 
 static void verify_ksm(void)
diff --git a/testcases/kernel/mem/ksm/ksm02.c b/testcases/kernel/mem/ksm/ksm02.c
index a388e82ab..4d1306045 100644
--- a/testcases/kernel/mem/ksm/ksm02.c
+++ b/testcases/kernel/mem/ksm/ksm02.c
@@ -53,7 +53,7 @@
 #include <signal.h>
 #include <stdio.h>
 #include <unistd.h>
-#include "mem.h"
+#include "tst_test.h"
 #include "ksm_common.h"
 
 #ifdef HAVE_NUMA_V2
diff --git a/testcases/kernel/mem/ksm/ksm03.c b/testcases/kernel/mem/ksm/ksm03.c
index cff74700d..78844b30e 100644
--- a/testcases/kernel/mem/ksm/ksm03.c
+++ b/testcases/kernel/mem/ksm/ksm03.c
@@ -56,7 +56,7 @@
 #include <stdlib.h>
 #include <string.h>
 #include <unistd.h>
-#include "mem.h"
+#include "tst_test.h"
 #include "ksm_common.h"
 
 static void verify_ksm(void)
diff --git a/testcases/kernel/mem/ksm/ksm04.c b/testcases/kernel/mem/ksm/ksm04.c
index 31f3ff2eb..8465ee230 100644
--- a/testcases/kernel/mem/ksm/ksm04.c
+++ b/testcases/kernel/mem/ksm/ksm04.c
@@ -38,7 +38,7 @@
 #include <signal.h>
 #include <stdio.h>
 #include <unistd.h>
-#include "mem.h"
+#include "tst_test.h"
 #include "ksm_common.h"
 
 #ifdef HAVE_NUMA_V2
diff --git a/testcases/kernel/mem/ksm/ksm05.c b/testcases/kernel/mem/ksm/ksm05.c
index 1d9d9699a..025dffc09 100644
--- a/testcases/kernel/mem/ksm/ksm05.c
+++ b/testcases/kernel/mem/ksm/ksm05.c
@@ -39,7 +39,7 @@
 #include <stdlib.h>
 #include <errno.h>
 #include "tst_test.h"
-#include "mem.h"
+#include "ksm_helper.h"
 
 #ifdef HAVE_DECL_MADV_MERGEABLE
 
diff --git a/testcases/kernel/mem/ksm/ksm06.c b/testcases/kernel/mem/ksm/ksm06.c
index 80fdf1e47..09a54fd18 100644
--- a/testcases/kernel/mem/ksm/ksm06.c
+++ b/testcases/kernel/mem/ksm/ksm06.c
@@ -33,8 +33,10 @@
 #include <unistd.h>
 #include <limits.h>
 
-#include "mem.h"
+#include "tst_test.h"
 #include "tst_numa.h"
+#include "ksm_helper.h"
+#include "ksm_test.h"
 
 #ifdef HAVE_NUMA_V2
 # include <numa.h>
diff --git a/testcases/kernel/mem/ksm/ksm07.c b/testcases/kernel/mem/ksm/ksm07.c
index 619bd7b55..24b393283 100644
--- a/testcases/kernel/mem/ksm/ksm07.c
+++ b/testcases/kernel/mem/ksm/ksm07.c
@@ -23,7 +23,8 @@
  */
 
 #include <sys/wait.h>
-#include "mem.h"
+#include "tst_test.h"
+#include "ksm_helper.h"
 
 /* This test allocates one page, fills the page with a's, captures the
  * full_scan and pages_skipped counters. Then it makes sure at least 3
diff --git a/testcases/kernel/mem/ksm/ksm_common.h b/testcases/kernel/mem/ksm/ksm_common.h
index 43ea8f8c2..d677b224d 100644
--- a/testcases/kernel/mem/ksm/ksm_common.h
+++ b/testcases/kernel/mem/ksm/ksm_common.h
@@ -11,7 +11,9 @@
 #define KSM_COMMON_H__
 
 #include "tst_test.h"
+#include "ksm_helper.h"
 #include "numa_helper.h"
+#include "ksm_test.h"
 
 #define DEFAULT_MEMSIZE 128
 
@@ -67,5 +69,4 @@ static inline unsigned int get_a_numa_node(void)
 	abort();
 }
 
-
 #endif /* KSM_COMMON_H__ */
diff --git a/testcases/kernel/mem/ksm/ksm_test.h b/testcases/kernel/mem/ksm/ksm_test.h
new file mode 100644
index 000000000..ebce8197c
--- /dev/null
+++ b/testcases/kernel/mem/ksm/ksm_test.h
@@ -0,0 +1,299 @@
+// SPDX-License-Identifier: GPL-2.0-or-later
+/*
+ * Copyright (c) Linux Test Project, 2011-2021
+ * Copyright (c) Cyril Hrubis <chrubis@suse.cz> 2024
+ */
+#ifndef KSM_TEST_
+#define KSM_TEST_
+
+#include <sys/wait.h>
+
+static inline void check(char *path, long int value)
+{
+	char fullpath[BUFSIZ];
+	long actual_val;
+
+	snprintf(fullpath, BUFSIZ, PATH_KSM "%s", path);
+	SAFE_FILE_SCANF(fullpath, "%ld", &actual_val);
+
+	if (actual_val != value)
+		tst_res(TFAIL, "%s is not %ld but %ld.", path, value,
+			actual_val);
+	else
+		tst_res(TPASS, "%s is %ld.", path, actual_val);
+}
+
+static inline void final_group_check(int run, int pages_shared, int pages_sharing,
+			  int pages_volatile, int pages_unshared,
+			  int sleep_millisecs, int pages_to_scan)
+{
+	int ksm_run_orig;
+
+	tst_res(TINFO, "check!");
+	check("run", run);
+
+	/*
+	 * Temporarily stop the KSM scan during the checks: during the
+	 * KSM scan the rmap_items in the stale unstable tree of the
+	 * old pass are removed from it and are later reinserted in
+	 * the new unstable tree of the current pass. So if the checks
+	 * run in the race window between removal and re-insertion, it
+	 * can lead to unexpected false positives where page_volatile
+	 * is elevated and page_unshared is recessed.
+	 */
+	SAFE_FILE_SCANF(PATH_KSM "run", "%d", &ksm_run_orig);
+	SAFE_FILE_PRINTF(PATH_KSM "run", "0");
+
+	check("pages_shared", pages_shared);
+	check("pages_sharing", pages_sharing);
+	check("pages_volatile", pages_volatile);
+	check("pages_unshared", pages_unshared);
+	check("sleep_millisecs", sleep_millisecs);
+	check("pages_to_scan", pages_to_scan);
+
+	SAFE_FILE_PRINTF(PATH_KSM "run", "%d", ksm_run_orig);
+}
+
+static inline void ksm_group_check(int run, int pages_shared, int pages_sharing,
+		     int pages_volatile, int pages_unshared,
+		     int sleep_millisecs, int pages_to_scan)
+{
+	if (run != 1) {
+		tst_res(TFAIL, "group_check run is not 1, %d.", run);
+	} else {
+		/* wait for ksm daemon to scan all mergeable pages. */
+		wait_ksmd_full_scan();
+	}
+
+	final_group_check(run, pages_shared, pages_sharing,
+			  pages_volatile, pages_unshared,
+			  sleep_millisecs, pages_to_scan);
+}
+
+static inline void verify(char **memory, char value, int proc,
+		    int start, int end, int start2, int end2)
+{
+	int i, j;
+	void *s = NULL;
+
+	s = SAFE_MALLOC((end - start) * (end2 - start2));
+
+	tst_res(TINFO, "child %d verifies memory content.", proc);
+	memset(s, value, (end - start) * (end2 - start2));
+	if (memcmp(memory[start], s, (end - start) * (end2 - start2))
+	    != 0)
+		for (j = start; j < end; j++)
+			for (i = start2; i < end2; i++)
+				if (memory[j][i] != value)
+					tst_res(TFAIL, "child %d has %c at "
+						 "%d,%d,%d.",
+						 proc, memory[j][i], proc,
+						 j, i);
+	free(s);
+}
+
+struct ksm_merge_data {
+	char data;
+	unsigned int mergeable_size;
+};
+
+static inline void ksm_child_memset(int child_num, int size, int total_unit,
+		 struct ksm_merge_data ksm_merge_data, char **memory)
+{
+	int i = 0, j;
+	int unit = size / total_unit;
+
+	tst_res(TINFO, "child %d continues...", child_num);
+
+	if (ksm_merge_data.mergeable_size == size * TST_MB) {
+		tst_res(TINFO, "child %d allocates %d TST_MB filled with '%c'",
+			child_num, size, ksm_merge_data.data);
+
+	} else {
+		tst_res(TINFO, "child %d allocates %d TST_MB filled with '%c'"
+				" except one page with 'e'",
+				child_num, size, ksm_merge_data.data);
+	}
+
+	for (j = 0; j < total_unit; j++) {
+		for (i = 0; (unsigned int)i < unit * TST_MB; i++)
+			memory[j][i] = ksm_merge_data.data;
+	}
+
+	/* if it contains unshared page, then set 'e' char
+	 * at the end of the last page
+	 */
+	if (ksm_merge_data.mergeable_size < size * TST_MB)
+		memory[j-1][i-1] = 'e';
+}
+
+static inline void create_ksm_child(int child_num, int size, int unit,
+		       struct ksm_merge_data *ksm_merge_data)
+{
+	int j, total_unit;
+	char **memory;
+
+	/* The total units in all */
+	total_unit = size / unit;
+
+	/* Apply for the space for memory */
+	memory = SAFE_MALLOC(total_unit * sizeof(char *));
+	for (j = 0; j < total_unit; j++) {
+		memory[j] = SAFE_MMAP(NULL, unit * TST_MB, PROT_READ|PROT_WRITE,
+			MAP_ANONYMOUS|MAP_PRIVATE, -1, 0);
+#ifdef HAVE_DECL_MADV_MERGEABLE
+		if (madvise(memory[j], unit * TST_MB, MADV_MERGEABLE) == -1)
+			tst_brk(TBROK|TERRNO, "madvise");
+#endif
+	}
+
+	tst_res(TINFO, "child %d stops.", child_num);
+	if (raise(SIGSTOP) == -1)
+		tst_brk(TBROK|TERRNO, "kill");
+	fflush(stdout);
+
+	for (j = 0; j < 4; j++) {
+
+		ksm_child_memset(child_num, size, total_unit,
+				  ksm_merge_data[j], memory);
+
+		fflush(stdout);
+
+		tst_res(TINFO, "child %d stops.", child_num);
+		if (raise(SIGSTOP) == -1)
+			tst_brk(TBROK|TERRNO, "kill");
+
+		if (ksm_merge_data[j].mergeable_size < size * TST_MB) {
+			verify(memory, 'e', child_num, total_unit - 1,
+				total_unit, unit * TST_MB - 1, unit * TST_MB);
+			verify(memory, ksm_merge_data[j].data, child_num,
+				0, total_unit, 0, unit * TST_MB - 1);
+		} else {
+			verify(memory, ksm_merge_data[j].data, child_num,
+				0, total_unit, 0, unit * TST_MB);
+		}
+	}
+
+	tst_res(TINFO, "child %d finished.", child_num);
+}
+
+static inline void stop_ksm_children(int *child, int num)
+{
+	int k, status;
+
+	tst_res(TINFO, "wait for all children to stop.");
+	for (k = 0; k < num; k++) {
+		SAFE_WAITPID(child[k], &status, WUNTRACED);
+		if (!WIFSTOPPED(status))
+			tst_brk(TBROK, "child %d was not stopped", k);
+	}
+}
+
+static inline void resume_ksm_children(int *child, int num)
+{
+	int k;
+
+	tst_res(TINFO, "resume all children.");
+	for (k = 0; k < num; k++)
+		SAFE_KILL(child[k], SIGCONT);
+
+	fflush(stdout);
+}
+
+static inline void create_same_memory(int size, int num, int unit)
+{
+	int i, j, status, *child;
+	unsigned long ps, pages;
+	struct ksm_merge_data **ksm_data;
+
+	struct ksm_merge_data ksm_data0[] = {
+	       {'c', size*TST_MB}, {'c', size*TST_MB}, {'d', size*TST_MB}, {'d', size*TST_MB},
+	};
+	struct ksm_merge_data ksm_data1[] = {
+	       {'a', size*TST_MB}, {'b', size*TST_MB}, {'d', size*TST_MB}, {'d', size*TST_MB-1},
+	};
+	struct ksm_merge_data ksm_data2[] = {
+	       {'a', size*TST_MB}, {'a', size*TST_MB}, {'d', size*TST_MB}, {'d', size*TST_MB},
+	};
+
+	ps = sysconf(_SC_PAGE_SIZE);
+	pages = TST_MB / ps;
+
+	ksm_data = malloc((num - 3) * sizeof(struct ksm_merge_data *));
+	/* Since from third child, the data is same with the first child's */
+	for (i = 0; i < num - 3; i++) {
+		ksm_data[i] = malloc(4 * sizeof(struct ksm_merge_data));
+		for (j = 0; j < 4; j++) {
+			ksm_data[i][j].data = ksm_data0[j].data;
+			ksm_data[i][j].mergeable_size =
+				ksm_data0[j].mergeable_size;
+		}
+	}
+
+	child = SAFE_MALLOC(num * sizeof(int));
+
+	for (i = 0; i < num; i++) {
+		fflush(stdout);
+		switch (child[i] = SAFE_FORK()) {
+		case 0:
+			if (i == 0) {
+				create_ksm_child(i, size, unit, ksm_data0);
+				exit(0);
+			} else if (i == 1) {
+				create_ksm_child(i, size, unit, ksm_data1);
+				exit(0);
+			} else if (i == 2) {
+				create_ksm_child(i, size, unit, ksm_data2);
+				exit(0);
+			} else {
+				create_ksm_child(i, size, unit, ksm_data[i-3]);
+				exit(0);
+			}
+		}
+	}
+
+	stop_ksm_children(child, num);
+
+	tst_res(TINFO, "KSM merging...");
+	if (access(PATH_KSM "max_page_sharing", F_OK) == 0) {
+		SAFE_FILE_PRINTF(PATH_KSM "run", "2");
+		SAFE_FILE_PRINTF(PATH_KSM "max_page_sharing", "%ld", size * pages * num);
+	}
+
+	SAFE_FILE_PRINTF(PATH_KSM "run", "1");
+	SAFE_FILE_PRINTF(PATH_KSM "pages_to_scan", "%ld", size * pages * num);
+	SAFE_FILE_PRINTF(PATH_KSM "sleep_millisecs", "0");
+
+	resume_ksm_children(child, num);
+	stop_ksm_children(child, num);
+	ksm_group_check(1, 2, size * num * pages - 2, 0, 0, 0, size * pages * num);
+
+	resume_ksm_children(child, num);
+	stop_ksm_children(child, num);
+	ksm_group_check(1, 3, size * num * pages - 3, 0, 0, 0, size * pages * num);
+
+	resume_ksm_children(child, num);
+	stop_ksm_children(child, num);
+	ksm_group_check(1, 1, size * num * pages - 1, 0, 0, 0, size * pages * num);
+
+	resume_ksm_children(child, num);
+	stop_ksm_children(child, num);
+	ksm_group_check(1, 1, size * num * pages - 2, 0, 1, 0, size * pages * num);
+
+	tst_res(TINFO, "KSM unmerging...");
+	SAFE_FILE_PRINTF(PATH_KSM "run", "2");
+
+	resume_ksm_children(child, num);
+	final_group_check(2, 0, 0, 0, 0, 0, size * pages * num);
+
+	tst_res(TINFO, "stop KSM.");
+	SAFE_FILE_PRINTF(PATH_KSM "run", "0");
+	final_group_check(0, 0, 0, 0, 0, 0, size * pages * num);
+
+	while (waitpid(-1, &status, 0) > 0)
+		if (WEXITSTATUS(status) != 0)
+			tst_res(TFAIL, "child exit status is %d",
+				 WEXITSTATUS(status));
+}
+
+#endif /* KSM_TEST_ */
diff --git a/testcases/kernel/mem/lib/mem.c b/testcases/kernel/mem/lib/mem.c
index f3b844994..4e5c0b873 100644
--- a/testcases/kernel/mem/lib/mem.c
+++ b/testcases/kernel/mem/lib/mem.c
@@ -28,290 +28,3 @@
 
 /* KSM */
 
-static void check(char *path, long int value)
-{
-	char fullpath[BUFSIZ];
-	long actual_val;
-
-	snprintf(fullpath, BUFSIZ, PATH_KSM "%s", path);
-	SAFE_FILE_SCANF(fullpath, "%ld", &actual_val);
-
-	if (actual_val != value)
-		tst_res(TFAIL, "%s is not %ld but %ld.", path, value,
-			actual_val);
-	else
-		tst_res(TPASS, "%s is %ld.", path, actual_val);
-}
-
-static void final_group_check(int run, int pages_shared, int pages_sharing,
-			  int pages_volatile, int pages_unshared,
-			  int sleep_millisecs, int pages_to_scan)
-{
-	int ksm_run_orig;
-
-	tst_res(TINFO, "check!");
-	check("run", run);
-
-	/*
-	 * Temporarily stop the KSM scan during the checks: during the
-	 * KSM scan the rmap_items in the stale unstable tree of the
-	 * old pass are removed from it and are later reinserted in
-	 * the new unstable tree of the current pass. So if the checks
-	 * run in the race window between removal and re-insertion, it
-	 * can lead to unexpected false positives where page_volatile
-	 * is elevated and page_unshared is recessed.
-	 */
-	SAFE_FILE_SCANF(PATH_KSM "run", "%d", &ksm_run_orig);
-	SAFE_FILE_PRINTF(PATH_KSM "run", "0");
-
-	check("pages_shared", pages_shared);
-	check("pages_sharing", pages_sharing);
-	check("pages_volatile", pages_volatile);
-	check("pages_unshared", pages_unshared);
-	check("sleep_millisecs", sleep_millisecs);
-	check("pages_to_scan", pages_to_scan);
-
-	SAFE_FILE_PRINTF(PATH_KSM "run", "%d", ksm_run_orig);
-}
-
-void ksm_group_check(int run, int pages_shared, int pages_sharing,
-		     int pages_volatile, int pages_unshared,
-		     int sleep_millisecs, int pages_to_scan)
-{
-	if (run != 1) {
-		tst_res(TFAIL, "group_check run is not 1, %d.", run);
-	} else {
-		/* wait for ksm daemon to scan all mergeable pages. */
-		wait_ksmd_full_scan();
-	}
-
-	final_group_check(run, pages_shared, pages_sharing,
-			  pages_volatile, pages_unshared,
-			  sleep_millisecs, pages_to_scan);
-}
-
-static void verify(char **memory, char value, int proc,
-		    int start, int end, int start2, int end2)
-{
-	int i, j;
-	void *s = NULL;
-
-	s = SAFE_MALLOC((end - start) * (end2 - start2));
-
-	tst_res(TINFO, "child %d verifies memory content.", proc);
-	memset(s, value, (end - start) * (end2 - start2));
-	if (memcmp(memory[start], s, (end - start) * (end2 - start2))
-	    != 0)
-		for (j = start; j < end; j++)
-			for (i = start2; i < end2; i++)
-				if (memory[j][i] != value)
-					tst_res(TFAIL, "child %d has %c at "
-						 "%d,%d,%d.",
-						 proc, memory[j][i], proc,
-						 j, i);
-	free(s);
-}
-
-struct ksm_merge_data {
-	char data;
-	unsigned int mergeable_size;
-};
-
-static void ksm_child_memset(int child_num, int size, int total_unit,
-		 struct ksm_merge_data ksm_merge_data, char **memory)
-{
-	int i = 0, j;
-	int unit = size / total_unit;
-
-	tst_res(TINFO, "child %d continues...", child_num);
-
-	if (ksm_merge_data.mergeable_size == size * MB) {
-		tst_res(TINFO, "child %d allocates %d MB filled with '%c'",
-			child_num, size, ksm_merge_data.data);
-
-	} else {
-		tst_res(TINFO, "child %d allocates %d MB filled with '%c'"
-				" except one page with 'e'",
-				child_num, size, ksm_merge_data.data);
-	}
-
-	for (j = 0; j < total_unit; j++) {
-		for (i = 0; (unsigned int)i < unit * MB; i++)
-			memory[j][i] = ksm_merge_data.data;
-	}
-
-	/* if it contains unshared page, then set 'e' char
-	 * at the end of the last page
-	 */
-	if (ksm_merge_data.mergeable_size < size * MB)
-		memory[j-1][i-1] = 'e';
-}
-
-static void create_ksm_child(int child_num, int size, int unit,
-		       struct ksm_merge_data *ksm_merge_data)
-{
-	int j, total_unit;
-	char **memory;
-
-	/* The total units in all */
-	total_unit = size / unit;
-
-	/* Apply for the space for memory */
-	memory = SAFE_MALLOC(total_unit * sizeof(char *));
-	for (j = 0; j < total_unit; j++) {
-		memory[j] = SAFE_MMAP(NULL, unit * MB, PROT_READ|PROT_WRITE,
-			MAP_ANONYMOUS|MAP_PRIVATE, -1, 0);
-#ifdef HAVE_DECL_MADV_MERGEABLE
-		if (madvise(memory[j], unit * MB, MADV_MERGEABLE) == -1)
-			tst_brk(TBROK|TERRNO, "madvise");
-#endif
-	}
-
-	tst_res(TINFO, "child %d stops.", child_num);
-	if (raise(SIGSTOP) == -1)
-		tst_brk(TBROK|TERRNO, "kill");
-	fflush(stdout);
-
-	for (j = 0; j < 4; j++) {
-
-		ksm_child_memset(child_num, size, total_unit,
-				  ksm_merge_data[j], memory);
-
-		fflush(stdout);
-
-		tst_res(TINFO, "child %d stops.", child_num);
-		if (raise(SIGSTOP) == -1)
-			tst_brk(TBROK|TERRNO, "kill");
-
-		if (ksm_merge_data[j].mergeable_size < size * MB) {
-			verify(memory, 'e', child_num, total_unit - 1,
-				total_unit, unit * MB - 1, unit * MB);
-			verify(memory, ksm_merge_data[j].data, child_num,
-				0, total_unit, 0, unit * MB - 1);
-		} else {
-			verify(memory, ksm_merge_data[j].data, child_num,
-				0, total_unit, 0, unit * MB);
-		}
-	}
-
-	tst_res(TINFO, "child %d finished.", child_num);
-}
-
-static void stop_ksm_children(int *child, int num)
-{
-	int k, status;
-
-	tst_res(TINFO, "wait for all children to stop.");
-	for (k = 0; k < num; k++) {
-		SAFE_WAITPID(child[k], &status, WUNTRACED);
-		if (!WIFSTOPPED(status))
-			tst_brk(TBROK, "child %d was not stopped", k);
-	}
-}
-
-static void resume_ksm_children(int *child, int num)
-{
-	int k;
-
-	tst_res(TINFO, "resume all children.");
-	for (k = 0; k < num; k++)
-		SAFE_KILL(child[k], SIGCONT);
-
-	fflush(stdout);
-}
-
-void create_same_memory(int size, int num, int unit)
-{
-	int i, j, status, *child;
-	unsigned long ps, pages;
-	struct ksm_merge_data **ksm_data;
-
-	struct ksm_merge_data ksm_data0[] = {
-	       {'c', size*MB}, {'c', size*MB}, {'d', size*MB}, {'d', size*MB},
-	};
-	struct ksm_merge_data ksm_data1[] = {
-	       {'a', size*MB}, {'b', size*MB}, {'d', size*MB}, {'d', size*MB-1},
-	};
-	struct ksm_merge_data ksm_data2[] = {
-	       {'a', size*MB}, {'a', size*MB}, {'d', size*MB}, {'d', size*MB},
-	};
-
-	ps = sysconf(_SC_PAGE_SIZE);
-	pages = MB / ps;
-
-	ksm_data = malloc((num - 3) * sizeof(struct ksm_merge_data *));
-	/* Since from third child, the data is same with the first child's */
-	for (i = 0; i < num - 3; i++) {
-		ksm_data[i] = malloc(4 * sizeof(struct ksm_merge_data));
-		for (j = 0; j < 4; j++) {
-			ksm_data[i][j].data = ksm_data0[j].data;
-			ksm_data[i][j].mergeable_size =
-				ksm_data0[j].mergeable_size;
-		}
-	}
-
-	child = SAFE_MALLOC(num * sizeof(int));
-
-	for (i = 0; i < num; i++) {
-		fflush(stdout);
-		switch (child[i] = SAFE_FORK()) {
-		case 0:
-			if (i == 0) {
-				create_ksm_child(i, size, unit, ksm_data0);
-				exit(0);
-			} else if (i == 1) {
-				create_ksm_child(i, size, unit, ksm_data1);
-				exit(0);
-			} else if (i == 2) {
-				create_ksm_child(i, size, unit, ksm_data2);
-				exit(0);
-			} else {
-				create_ksm_child(i, size, unit, ksm_data[i-3]);
-				exit(0);
-			}
-		}
-	}
-
-	stop_ksm_children(child, num);
-
-	tst_res(TINFO, "KSM merging...");
-	if (access(PATH_KSM "max_page_sharing", F_OK) == 0) {
-		SAFE_FILE_PRINTF(PATH_KSM "run", "2");
-		SAFE_FILE_PRINTF(PATH_KSM "max_page_sharing", "%ld", size * pages * num);
-	}
-
-	SAFE_FILE_PRINTF(PATH_KSM "run", "1");
-	SAFE_FILE_PRINTF(PATH_KSM "pages_to_scan", "%ld", size * pages * num);
-	SAFE_FILE_PRINTF(PATH_KSM "sleep_millisecs", "0");
-
-	resume_ksm_children(child, num);
-	stop_ksm_children(child, num);
-	ksm_group_check(1, 2, size * num * pages - 2, 0, 0, 0, size * pages * num);
-
-	resume_ksm_children(child, num);
-	stop_ksm_children(child, num);
-	ksm_group_check(1, 3, size * num * pages - 3, 0, 0, 0, size * pages * num);
-
-	resume_ksm_children(child, num);
-	stop_ksm_children(child, num);
-	ksm_group_check(1, 1, size * num * pages - 1, 0, 0, 0, size * pages * num);
-
-	resume_ksm_children(child, num);
-	stop_ksm_children(child, num);
-	ksm_group_check(1, 1, size * num * pages - 2, 0, 1, 0, size * pages * num);
-
-	tst_res(TINFO, "KSM unmerging...");
-	SAFE_FILE_PRINTF(PATH_KSM "run", "2");
-
-	resume_ksm_children(child, num);
-	final_group_check(2, 0, 0, 0, 0, 0, size * pages * num);
-
-	tst_res(TINFO, "stop KSM.");
-	SAFE_FILE_PRINTF(PATH_KSM "run", "0");
-	final_group_check(0, 0, 0, 0, 0, 0, size * pages * num);
-
-	while (waitpid(-1, &status, 0) > 0)
-		if (WEXITSTATUS(status) != 0)
-			tst_res(TFAIL, "child exit status is %d",
-				 WEXITSTATUS(status));
-}
-- 
2.45.2



More information about the ltp mailing list