1 /*
2 * This is a reproducer from mainline commit
3 * 9d8cebd4bcd7c3878462fdfda34bbcdeb4df7ef4:
4 *
5 * "Strangely, current mbind() doesn't merge vma with neighbor vma
6 * although it's possible. Unfortunately, many vma can reduce
7 * performance..."
8 *
9 * Copyright (C) 2010 Red Hat, Inc.
10 * This program is free software; you can redistribute it and/or
11 * modify it under the terms of version 2 of the GNU General Public
12 * License as published by the Free Software Foundation.
13 *
14 * This program is distributed in the hope that it would be useful,
15 * but WITHOUT ANY WARRANTY; without even the implied warranty of
16 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.
17 *
18 * Further, this software is distributed without any warranty that it
19 * is free of the rightful claim of any third person regarding
20 * infringement or the like. Any license provided herein, whether
21 * implied or otherwise, applies only to this software file. Patent
22 * licenses, if any, provided herein do not apply to combinations of
23 * this program with other software, or any other product whatsoever.
24 *
25 * You should have received a copy of the GNU General Public License
26 * along with this program; if not, write the Free Software
27 * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA
28 * 02110-1301, USA.
29 */
30 #include "config.h"
31 #include <sys/types.h>
32 #include <sys/mman.h>
33 #include <errno.h>
34 #if HAVE_NUMA_H
35 #include <numa.h>
36 #endif
37 #if HAVE_NUMAIF_H
38 #include <numaif.h>
39 #endif
40 #include <stdio.h>
41 #include <stdlib.h>
42 #include <string.h>
43 #include <unistd.h>
44 #include "test.h"
45 #include "safe_macros.h"
46 #include "numa_helper.h"
47
48 char *TCID = "vma02";
49 int TST_TOTAL = 1;
50
51 #if HAVE_NUMA_H && HAVE_LINUX_MEMPOLICY_H && HAVE_NUMAIF_H \
52 && HAVE_MPOL_CONSTANTS
53 #if defined(LIBNUMA_API_VERSION) && LIBNUMA_API_VERSION == 2
54 static unsigned long pagesize;
55 static int opt_node;
56 static char *opt_nodestr;
57 static option_t options[] = {
58 {"n:", &opt_node, &opt_nodestr},
59 {NULL, NULL, NULL}
60 };
61
62 static void usage(void);
63
main(int argc,char ** argv)64 int main(int argc, char **argv)
65 {
66 FILE *fp;
67 void *addr, *start, *end, *lastend;
68 int node, err, lc;
69 char buf[BUFSIZ];
70 struct bitmask *nmask = numa_allocate_nodemask();
71
72 pagesize = getpagesize();
73 tst_parse_opts(argc, argv, options, usage);
74
75 if (opt_node) {
76 node = SAFE_STRTOL(NULL, opt_nodestr, 1, LONG_MAX);
77 } else {
78 err = get_allowed_nodes(NH_MEMS | NH_MEMS, 1, &node);
79 if (err == -3)
80 tst_brkm(TCONF, NULL, "requires at least one node.");
81 else if (err < 0)
82 tst_brkm(TBROK | TERRNO, NULL, "get_allowed_nodes");
83 }
84 numa_bitmask_setbit(nmask, node);
85
86 for (lc = 0; TEST_LOOPING(lc); lc++) {
87 tst_count = 0;
88 addr = mmap(NULL, pagesize * 3, PROT_WRITE,
89 MAP_ANON | MAP_PRIVATE, 0, 0);
90 if (addr == MAP_FAILED)
91 tst_brkm(TBROK | TERRNO, NULL, "mmap");
92
93 tst_resm(TINFO, "pid = %d addr = %p", getpid(), addr);
94 /* make page populate */
95 memset(addr, 0, pagesize * 3);
96
97 /* first mbind */
98 err = mbind(addr + pagesize, pagesize, MPOL_BIND, nmask->maskp,
99 nmask->size, MPOL_MF_MOVE_ALL);
100 if (err != 0) {
101 if (errno != ENOSYS)
102 tst_brkm(TBROK | TERRNO, NULL, "mbind1");
103 else
104 tst_brkm(TCONF, NULL,
105 "mbind syscall not implemented on this system.");
106 }
107
108 /* second mbind */
109 err = mbind(addr, pagesize * 3, MPOL_DEFAULT, NULL, 0, 0);
110 if (err != 0)
111 tst_brkm(TBROK | TERRNO, NULL, "mbind2");
112
113 /* /proc/self/maps in the form of
114 "00400000-00406000 r-xp 00000000". */
115 fp = fopen("/proc/self/maps", "r");
116 if (fp == NULL)
117 tst_brkm(TBROK | TERRNO, NULL, "fopen");
118
119 while (fgets(buf, BUFSIZ, fp) != NULL) {
120 if (sscanf(buf, "%p-%p ", &start, &end) != 2)
121 continue;
122
123 if (start == addr) {
124 tst_resm(TINFO, "start = %p, end = %p",
125 start, end);
126 if (end == addr + pagesize * 3) {
127 tst_resm(TPASS, "only 1 VMA.");
128 break;
129 }
130
131 lastend = end;
132 while (fgets(buf, BUFSIZ, fp) != NULL) {
133 /* No more VMAs, break */
134 if (sscanf(buf, "%p-%p ", &start,
135 &end) != 2)
136 break;
137 tst_resm(TINFO, "start = %p, end = %p",
138 start, end);
139
140 /* more VMAs found */
141 if (start == lastend)
142 lastend = end;
143 if (end == addr + pagesize * 3) {
144 tst_resm(TFAIL,
145 ">1 unmerged VMAs.");
146 break;
147 }
148 }
149 if (end != addr + pagesize * 3)
150 tst_resm(TFAIL, "no matched VMAs.");
151 break;
152 }
153 }
154 fclose(fp);
155 if (munmap(addr, pagesize * 3) == -1)
156 tst_brkm(TWARN | TERRNO, NULL, "munmap");
157 }
158 tst_exit();
159 }
160
usage(void)161 void usage(void)
162 {
163 printf(" -n Number of NUMA nodes\n");
164 }
165 #else /* libnuma v1 */
main(void)166 int main(void)
167 {
168 tst_brkm(TCONF, NULL, "XXX: test is only supported on libnuma v2.");
169 }
170 #endif
171 #else /* no NUMA */
main(void)172 int main(void)
173 {
174 tst_brkm(TCONF, NULL, "no NUMA development packages installed.");
175 }
176 #endif
177