/* * Copyright (C) 2010-2017 Red Hat, Inc. * * This program is free software; you can redistribute it and/or modify * it under the terms of the GNU General Public License as published by * the Free Software Foundation; either version 2 of the License, or * (at your option) any later version. * * This program is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See * the GNU General Public License for more details. * * Out Of Memory when changing cpuset's mems on NUMA. There was a * problem reported upstream that the allocator may see an empty * nodemask when changing cpuset's mems. * http://lkml.org/lkml/2010/5/4/77 * http://lkml.org/lkml/2010/5/4/79 * http://lkml.org/lkml/2010/5/4/80 * This test is based on the reproducers for the above issue. */ #include "config.h" #include <stdio.h> #include <sys/wait.h> #if HAVE_NUMA_H #include <numa.h> #endif #if HAVE_NUMAIF_H #include <numaif.h> #endif #include "mem.h" #include "numa_helper.h" #ifdef HAVE_NUMA_V2 volatile int end; static int *nodes; static int nnodes; static long ncpus; static void sighandler(int signo LTP_ATTRIBUTE_UNUSED); static int mem_hog(void); static int mem_hog_cpuset(int ntasks); static long count_cpu(void); static void test_cpuset(void) { int child, i, status; unsigned long nmask[MAXNODES / BITS_PER_LONG] = { 0 }; char mems[BUFSIZ], buf[BUFSIZ]; read_cpuset_files(CPATH, "cpus", buf); write_cpuset_files(CPATH_NEW, "cpus", buf); read_cpuset_files(CPATH, "mems", mems); write_cpuset_files(CPATH_NEW, "mems", mems); SAFE_FILE_PRINTF(CPATH_NEW "/tasks", "%d", getpid()); child = SAFE_FORK(); if (child == 0) { for (i = 0; i < nnodes; i++) { if (nodes[i] >= MAXNODES) continue; set_node(nmask, nodes[i]); } if (set_mempolicy(MPOL_BIND, nmask, MAXNODES) == -1) tst_brk(TBROK | TERRNO, "set_mempolicy"); exit(mem_hog_cpuset(ncpus > 1 ? ncpus : 1)); } snprintf(buf, BUFSIZ, "%d", nodes[0]); write_cpuset_files(CPATH_NEW, "mems", buf); snprintf(buf, BUFSIZ, "%d", nodes[1]); write_cpuset_files(CPATH_NEW, "mems", buf); SAFE_WAITPID(child, &status, WUNTRACED | WCONTINUED); if (WEXITSTATUS(status) != 0) { tst_res(TFAIL, "child exit status is %d", WEXITSTATUS(status)); return; } tst_res(TPASS, "cpuset test pass"); } static void setup(void) { mount_mem("cpuset", "cpuset", NULL, CPATH, CPATH_NEW); ncpus = count_cpu(); if (get_allowed_nodes_arr(NH_MEMS | NH_CPUS, &nnodes, &nodes) < 0) tst_brk(TBROK | TERRNO, "get_allowed_nodes_arr"); if (nnodes <= 1) tst_brk(TCONF, "requires a NUMA system."); } static void cleanup(void) { umount_mem(CPATH, CPATH_NEW); } static void sighandler(int signo LTP_ATTRIBUTE_UNUSED) { end = 1; } static int mem_hog(void) { long pagesize; unsigned long *addr; int ret = 0; pagesize = getpagesize(); while (!end) { addr = SAFE_MMAP(NULL, pagesize * 10, PROT_READ | PROT_WRITE, MAP_PRIVATE | MAP_ANONYMOUS, -1, 0); memset(addr, 0xF7, pagesize * 10); SAFE_MUNMAP(addr, pagesize * 10); } return ret; } static int mem_hog_cpuset(int ntasks) { int i, status, ret = 0; struct sigaction sa; pid_t *pids; if (ntasks <= 0) tst_brk(TBROK | TERRNO, "ntasks is small."); sa.sa_handler = sighandler; if (sigemptyset(&sa.sa_mask) < 0) tst_brk(TBROK | TERRNO, "sigemptyset"); sa.sa_flags = 0; if (sigaction(SIGUSR1, &sa, NULL) < 0) tst_brk(TBROK | TERRNO, "sigaction"); pids = SAFE_MALLOC(sizeof(pid_t) * ntasks); for (i = 0; i < ntasks; i++) { switch (pids[i] = fork()) { case -1: tst_res(TFAIL | TERRNO, "fork %d", pids[i]); ret = 1; break; case 0: ret = mem_hog(); exit(ret); default: break; } } while (i--) { if (kill(pids[i], SIGUSR1) == -1) { tst_res(TFAIL | TERRNO, "kill %d", pids[i]); ret = 1; } } while (waitpid(-1, &status, WUNTRACED | WCONTINUED) > 0) { if (WIFEXITED(status)) { if (WEXITSTATUS(status) != 0) { tst_res(TFAIL, "child exit status is %d", WEXITSTATUS(status)); ret = 1; } } else if (WIFSIGNALED(status)) { tst_res(TFAIL, "child caught signal %d", WTERMSIG(status)); ret = 1; } } return ret; } static long count_cpu(void) { int ncpus = 0; while (path_exist(PATH_SYS_SYSTEM "/cpu/cpu%d", ncpus)) ncpus++; return ncpus; } static struct tst_test test = { .needs_root = 1, .setup = setup, .cleanup = cleanup, .test_all = test_cpuset, .min_kver = "2.6.32", }; #else TST_TEST_TCONF(NUMA_ERROR_MSG); #endif