1*49cdfc7eSAndroid Build Coastguard Worker // SPDX-License-Identifier: GPL-2.0-or-later
2*49cdfc7eSAndroid Build Coastguard Worker /*
3*49cdfc7eSAndroid Build Coastguard Worker * Copyright (c) 2019 Cyril Hrubis <[email protected]>
4*49cdfc7eSAndroid Build Coastguard Worker */
5*49cdfc7eSAndroid Build Coastguard Worker
6*49cdfc7eSAndroid Build Coastguard Worker /*
7*49cdfc7eSAndroid Build Coastguard Worker * We are testing mbind() EIO error.
8*49cdfc7eSAndroid Build Coastguard Worker *
9*49cdfc7eSAndroid Build Coastguard Worker * We first fault a allocated page, then attempt to mbind it to a different node.
10*49cdfc7eSAndroid Build Coastguard Worker *
11*49cdfc7eSAndroid Build Coastguard Worker * This is a regression test for:
12*49cdfc7eSAndroid Build Coastguard Worker *
13*49cdfc7eSAndroid Build Coastguard Worker * a7f40cfe3b7a mm: mempolicy: make mbind() return -EIO when MPOL_MF_STRICT is specified
14*49cdfc7eSAndroid Build Coastguard Worker *
15*49cdfc7eSAndroid Build Coastguard Worker */
16*49cdfc7eSAndroid Build Coastguard Worker
17*49cdfc7eSAndroid Build Coastguard Worker #include <errno.h>
18*49cdfc7eSAndroid Build Coastguard Worker #include "config.h"
19*49cdfc7eSAndroid Build Coastguard Worker #ifdef HAVE_NUMA_H
20*49cdfc7eSAndroid Build Coastguard Worker # include <numa.h>
21*49cdfc7eSAndroid Build Coastguard Worker # include <numaif.h>
22*49cdfc7eSAndroid Build Coastguard Worker #endif
23*49cdfc7eSAndroid Build Coastguard Worker #include "tst_test.h"
24*49cdfc7eSAndroid Build Coastguard Worker #include "tst_numa.h"
25*49cdfc7eSAndroid Build Coastguard Worker
26*49cdfc7eSAndroid Build Coastguard Worker #ifdef HAVE_NUMA_V2
27*49cdfc7eSAndroid Build Coastguard Worker
28*49cdfc7eSAndroid Build Coastguard Worker static size_t page_size;
29*49cdfc7eSAndroid Build Coastguard Worker static struct tst_nodemap *nodes;
30*49cdfc7eSAndroid Build Coastguard Worker
setup(void)31*49cdfc7eSAndroid Build Coastguard Worker static void setup(void)
32*49cdfc7eSAndroid Build Coastguard Worker {
33*49cdfc7eSAndroid Build Coastguard Worker page_size = getpagesize();
34*49cdfc7eSAndroid Build Coastguard Worker
35*49cdfc7eSAndroid Build Coastguard Worker nodes = tst_get_nodemap(TST_NUMA_MEM, 2 * page_size / 1024);
36*49cdfc7eSAndroid Build Coastguard Worker if (nodes->cnt <= 1)
37*49cdfc7eSAndroid Build Coastguard Worker tst_brk(TCONF, "Test requires at least two NUMA memory nodes");
38*49cdfc7eSAndroid Build Coastguard Worker }
39*49cdfc7eSAndroid Build Coastguard Worker
cleanup(void)40*49cdfc7eSAndroid Build Coastguard Worker static void cleanup(void)
41*49cdfc7eSAndroid Build Coastguard Worker {
42*49cdfc7eSAndroid Build Coastguard Worker tst_nodemap_free(nodes);
43*49cdfc7eSAndroid Build Coastguard Worker }
44*49cdfc7eSAndroid Build Coastguard Worker
verify_policy(int mode)45*49cdfc7eSAndroid Build Coastguard Worker static void verify_policy(int mode)
46*49cdfc7eSAndroid Build Coastguard Worker {
47*49cdfc7eSAndroid Build Coastguard Worker struct bitmask *bm = numa_allocate_nodemask();
48*49cdfc7eSAndroid Build Coastguard Worker unsigned int i;
49*49cdfc7eSAndroid Build Coastguard Worker void *ptr;
50*49cdfc7eSAndroid Build Coastguard Worker unsigned long size = page_size;
51*49cdfc7eSAndroid Build Coastguard Worker int node = 0;
52*49cdfc7eSAndroid Build Coastguard Worker
53*49cdfc7eSAndroid Build Coastguard Worker ptr = tst_numa_map(NULL, size);
54*49cdfc7eSAndroid Build Coastguard Worker tst_nodemap_reset_counters(nodes);
55*49cdfc7eSAndroid Build Coastguard Worker tst_numa_fault(ptr, size);
56*49cdfc7eSAndroid Build Coastguard Worker tst_nodemap_count_pages(nodes, ptr, size);
57*49cdfc7eSAndroid Build Coastguard Worker tst_nodemap_print_counters(nodes);
58*49cdfc7eSAndroid Build Coastguard Worker
59*49cdfc7eSAndroid Build Coastguard Worker for (i = 0; i < nodes->cnt; i++) {
60*49cdfc7eSAndroid Build Coastguard Worker if (!nodes->counters[i]) {
61*49cdfc7eSAndroid Build Coastguard Worker node = nodes->map[i];
62*49cdfc7eSAndroid Build Coastguard Worker tst_res(TINFO, "Attempting to bind to node %i", node);
63*49cdfc7eSAndroid Build Coastguard Worker numa_bitmask_setbit(bm, node);
64*49cdfc7eSAndroid Build Coastguard Worker break;
65*49cdfc7eSAndroid Build Coastguard Worker }
66*49cdfc7eSAndroid Build Coastguard Worker }
67*49cdfc7eSAndroid Build Coastguard Worker
68*49cdfc7eSAndroid Build Coastguard Worker TEST(mbind(ptr, size, mode, bm->maskp, bm->size + 1, MPOL_MF_STRICT));
69*49cdfc7eSAndroid Build Coastguard Worker
70*49cdfc7eSAndroid Build Coastguard Worker tst_numa_unmap(ptr, size);
71*49cdfc7eSAndroid Build Coastguard Worker numa_free_nodemask(bm);
72*49cdfc7eSAndroid Build Coastguard Worker
73*49cdfc7eSAndroid Build Coastguard Worker if (TST_RET != -1) {
74*49cdfc7eSAndroid Build Coastguard Worker tst_res(TFAIL,
75*49cdfc7eSAndroid Build Coastguard Worker "mbind(%s, MPOL_MF_STRICT) node %u returned %li, expected -1",
76*49cdfc7eSAndroid Build Coastguard Worker tst_mempolicy_mode_name(mode), node, TST_RET);
77*49cdfc7eSAndroid Build Coastguard Worker return;
78*49cdfc7eSAndroid Build Coastguard Worker }
79*49cdfc7eSAndroid Build Coastguard Worker
80*49cdfc7eSAndroid Build Coastguard Worker if (TST_ERR == EIO) {
81*49cdfc7eSAndroid Build Coastguard Worker tst_res(TPASS | TTERRNO,
82*49cdfc7eSAndroid Build Coastguard Worker "mbind(%s, MPOL_MF_STRICT) node %u",
83*49cdfc7eSAndroid Build Coastguard Worker tst_mempolicy_mode_name(mode), node);
84*49cdfc7eSAndroid Build Coastguard Worker } else {
85*49cdfc7eSAndroid Build Coastguard Worker tst_res(TFAIL | TTERRNO,
86*49cdfc7eSAndroid Build Coastguard Worker "mbind(%s, MPOL_MF_STRICT) node %u expected EIO",
87*49cdfc7eSAndroid Build Coastguard Worker tst_mempolicy_mode_name(mode), node);
88*49cdfc7eSAndroid Build Coastguard Worker }
89*49cdfc7eSAndroid Build Coastguard Worker }
90*49cdfc7eSAndroid Build Coastguard Worker
91*49cdfc7eSAndroid Build Coastguard Worker static const int modes[] = {
92*49cdfc7eSAndroid Build Coastguard Worker MPOL_PREFERRED,
93*49cdfc7eSAndroid Build Coastguard Worker MPOL_BIND,
94*49cdfc7eSAndroid Build Coastguard Worker MPOL_INTERLEAVE,
95*49cdfc7eSAndroid Build Coastguard Worker };
96*49cdfc7eSAndroid Build Coastguard Worker
verify_mbind(unsigned int n)97*49cdfc7eSAndroid Build Coastguard Worker static void verify_mbind(unsigned int n)
98*49cdfc7eSAndroid Build Coastguard Worker {
99*49cdfc7eSAndroid Build Coastguard Worker verify_policy(modes[n]);
100*49cdfc7eSAndroid Build Coastguard Worker }
101*49cdfc7eSAndroid Build Coastguard Worker
102*49cdfc7eSAndroid Build Coastguard Worker static struct tst_test test = {
103*49cdfc7eSAndroid Build Coastguard Worker .setup = setup,
104*49cdfc7eSAndroid Build Coastguard Worker .cleanup = cleanup,
105*49cdfc7eSAndroid Build Coastguard Worker .test = verify_mbind,
106*49cdfc7eSAndroid Build Coastguard Worker .tcnt = ARRAY_SIZE(modes),
107*49cdfc7eSAndroid Build Coastguard Worker .tags = (const struct tst_tag[]) {
108*49cdfc7eSAndroid Build Coastguard Worker {"linux-git", "a7f40cfe3b7a"},
109*49cdfc7eSAndroid Build Coastguard Worker {}
110*49cdfc7eSAndroid Build Coastguard Worker }
111*49cdfc7eSAndroid Build Coastguard Worker };
112*49cdfc7eSAndroid Build Coastguard Worker
113*49cdfc7eSAndroid Build Coastguard Worker #else
114*49cdfc7eSAndroid Build Coastguard Worker
115*49cdfc7eSAndroid Build Coastguard Worker TST_TEST_TCONF(NUMA_ERROR_MSG);
116*49cdfc7eSAndroid Build Coastguard Worker
117*49cdfc7eSAndroid Build Coastguard Worker #endif /* HAVE_NUMA_H */
118