1f08c3bdfSopenharmony_ci// SPDX-License-Identifier: GPL-2.0-or-later 2f08c3bdfSopenharmony_ci/* 3f08c3bdfSopenharmony_ci * Copyright (c) 2018 Cyril Hrubis <chrubis@suse.cz> 4f08c3bdfSopenharmony_ci */ 5f08c3bdfSopenharmony_ci 6f08c3bdfSopenharmony_ci/* 7f08c3bdfSopenharmony_ci * We are testing set_mempolicy() with MPOL_INTERLEAVE. 8f08c3bdfSopenharmony_ci * 9f08c3bdfSopenharmony_ci * The test tries different subsets of memory nodes, sets the mask with 10f08c3bdfSopenharmony_ci * memopolicy, and checks that the memory was interleaved between the nodes 11f08c3bdfSopenharmony_ci * accordingly. 12f08c3bdfSopenharmony_ci */ 13f08c3bdfSopenharmony_ci 14f08c3bdfSopenharmony_ci#include <errno.h> 15f08c3bdfSopenharmony_ci#include "config.h" 16f08c3bdfSopenharmony_ci#ifdef HAVE_NUMA_V2 17f08c3bdfSopenharmony_ci# include <numa.h> 18f08c3bdfSopenharmony_ci# include <numaif.h> 19f08c3bdfSopenharmony_ci#endif 20f08c3bdfSopenharmony_ci#include "tst_test.h" 21f08c3bdfSopenharmony_ci#include "tst_numa.h" 22f08c3bdfSopenharmony_ci 23f08c3bdfSopenharmony_ci#ifdef HAVE_NUMA_V2 24f08c3bdfSopenharmony_ci 25f08c3bdfSopenharmony_ci#include "set_mempolicy.h" 26f08c3bdfSopenharmony_ci 27f08c3bdfSopenharmony_ci#define ALLOC_ON_NODE 8 28f08c3bdfSopenharmony_ci 29f08c3bdfSopenharmony_cistatic size_t page_size; 30f08c3bdfSopenharmony_cistatic struct tst_nodemap *nodes; 31f08c3bdfSopenharmony_ci 32f08c3bdfSopenharmony_cistatic void setup(void) 33f08c3bdfSopenharmony_ci{ 34f08c3bdfSopenharmony_ci page_size = getpagesize(); 35f08c3bdfSopenharmony_ci 36f08c3bdfSopenharmony_ci nodes = tst_get_nodemap(TST_NUMA_MEM, 2 * ALLOC_ON_NODE * page_size / 1024); 37f08c3bdfSopenharmony_ci if (nodes->cnt <= 1) 38f08c3bdfSopenharmony_ci tst_brk(TCONF, "Test requires at least two NUMA memory nodes"); 39f08c3bdfSopenharmony_ci} 40f08c3bdfSopenharmony_ci 41f08c3bdfSopenharmony_cistatic void cleanup(void) 42f08c3bdfSopenharmony_ci{ 43f08c3bdfSopenharmony_ci tst_nodemap_free(nodes); 44f08c3bdfSopenharmony_ci} 45f08c3bdfSopenharmony_ci 46f08c3bdfSopenharmony_cistatic void alloc_and_check(size_t size, unsigned int *exp_alloc) 47f08c3bdfSopenharmony_ci{ 48f08c3bdfSopenharmony_ci unsigned int i; 49f08c3bdfSopenharmony_ci const char *prefix = "child: "; 50f08c3bdfSopenharmony_ci 51f08c3bdfSopenharmony_ci if (SAFE_FORK()) { 52f08c3bdfSopenharmony_ci prefix = "parent: "; 53f08c3bdfSopenharmony_ci tst_reap_children(); 54f08c3bdfSopenharmony_ci } 55f08c3bdfSopenharmony_ci 56f08c3bdfSopenharmony_ci tst_nodemap_reset_counters(nodes); 57f08c3bdfSopenharmony_ci alloc_fault_count(nodes, NULL, size * page_size); 58f08c3bdfSopenharmony_ci 59f08c3bdfSopenharmony_ci for (i = 0; i < nodes->cnt; i++) { 60f08c3bdfSopenharmony_ci if (nodes->counters[i] == exp_alloc[i]) { 61f08c3bdfSopenharmony_ci tst_res(TPASS, "%sNode %u allocated %u", 62f08c3bdfSopenharmony_ci prefix, nodes->map[i], exp_alloc[i]); 63f08c3bdfSopenharmony_ci } else { 64f08c3bdfSopenharmony_ci tst_res(TFAIL, "%sNode %u allocated %u, expected %u", 65f08c3bdfSopenharmony_ci prefix, nodes->map[i], nodes->counters[i], 66f08c3bdfSopenharmony_ci exp_alloc[i]); 67f08c3bdfSopenharmony_ci } 68f08c3bdfSopenharmony_ci } 69f08c3bdfSopenharmony_ci} 70f08c3bdfSopenharmony_ci 71f08c3bdfSopenharmony_cistatic void verify_set_mempolicy(unsigned int n) 72f08c3bdfSopenharmony_ci{ 73f08c3bdfSopenharmony_ci struct bitmask *bm = numa_allocate_nodemask(); 74f08c3bdfSopenharmony_ci unsigned int exp_alloc[nodes->cnt]; 75f08c3bdfSopenharmony_ci unsigned int alloc_per_node = n ? ALLOC_ON_NODE : 2; 76f08c3bdfSopenharmony_ci unsigned int alloc_on_nodes = n ? 2 : nodes->cnt; 77f08c3bdfSopenharmony_ci unsigned int alloc_total = alloc_per_node * alloc_on_nodes; 78f08c3bdfSopenharmony_ci unsigned int i; 79f08c3bdfSopenharmony_ci 80f08c3bdfSopenharmony_ci memset(exp_alloc, 0, sizeof(exp_alloc)); 81f08c3bdfSopenharmony_ci 82f08c3bdfSopenharmony_ci for (i = 0; i < alloc_on_nodes; i++) { 83f08c3bdfSopenharmony_ci exp_alloc[i] = alloc_per_node; 84f08c3bdfSopenharmony_ci numa_bitmask_setbit(bm, nodes->map[i]); 85f08c3bdfSopenharmony_ci } 86f08c3bdfSopenharmony_ci 87f08c3bdfSopenharmony_ci TEST(set_mempolicy(MPOL_INTERLEAVE, bm->maskp, bm->size+1)); 88f08c3bdfSopenharmony_ci 89f08c3bdfSopenharmony_ci tst_res(TINFO, "Allocating on nodes 1-%u - %u pages", 90f08c3bdfSopenharmony_ci alloc_on_nodes, alloc_total); 91f08c3bdfSopenharmony_ci 92f08c3bdfSopenharmony_ci if (TST_RET) { 93f08c3bdfSopenharmony_ci tst_res(TFAIL | TTERRNO, 94f08c3bdfSopenharmony_ci "set_mempolicy(MPOL_INTERLEAVE)"); 95f08c3bdfSopenharmony_ci return; 96f08c3bdfSopenharmony_ci } 97f08c3bdfSopenharmony_ci 98f08c3bdfSopenharmony_ci tst_res(TPASS, "set_mempolicy(MPOL_INTERLEAVE)"); 99f08c3bdfSopenharmony_ci 100f08c3bdfSopenharmony_ci numa_free_nodemask(bm); 101f08c3bdfSopenharmony_ci 102f08c3bdfSopenharmony_ci alloc_and_check(alloc_total, exp_alloc); 103f08c3bdfSopenharmony_ci} 104f08c3bdfSopenharmony_ci 105f08c3bdfSopenharmony_cistatic struct tst_test test = { 106f08c3bdfSopenharmony_ci .setup = setup, 107f08c3bdfSopenharmony_ci .cleanup = cleanup, 108f08c3bdfSopenharmony_ci .test = verify_set_mempolicy, 109f08c3bdfSopenharmony_ci .tcnt = 2, 110f08c3bdfSopenharmony_ci .forks_child = 1, 111f08c3bdfSopenharmony_ci .needs_checkpoints = 1, 112f08c3bdfSopenharmony_ci}; 113f08c3bdfSopenharmony_ci 114f08c3bdfSopenharmony_ci#else 115f08c3bdfSopenharmony_ci 116f08c3bdfSopenharmony_ciTST_TEST_TCONF(NUMA_ERROR_MSG); 117f08c3bdfSopenharmony_ci 118f08c3bdfSopenharmony_ci#endif /* HAVE_NUMA_V2 */ 119