| /* |
| * Out Of Memory (OOM) for Memory Resource Controller |
| * |
| * The program is designed to cope with unpredictable like amount and |
| * system physical memory, swap size and other VMM technology like KSM, |
| * memcg, memory hotplug and so on which may affect the OOM |
| * behaviours. It simply increase the memory consumption 3G each time |
| * until all the available memory is consumed and OOM is triggered. |
| * |
| * 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. |
| */ |
| |
| #include "config.h" |
| #include <sys/types.h> |
| #include <sys/stat.h> |
| #include <errno.h> |
| #include <fcntl.h> |
| #include <stdio.h> |
| #if HAVE_NUMA_H |
| #include <numa.h> |
| #endif |
| |
| #include "numa_helper.h" |
| #include "mem.h" |
| |
| #ifdef HAVE_NUMA_V2 |
| |
| static int memcg_mounted; |
| |
| static void verify_oom(void) |
| { |
| #if __WORDSIZE == 32 |
| tst_brk(TCONF, "test is not designed for 32-bit system."); |
| #endif |
| |
| SAFE_FILE_PRINTF(MEMCG_PATH_NEW "/tasks", "%d", getpid()); |
| SAFE_FILE_PRINTF(MEMCG_LIMIT, "%ld", TESTMEM); |
| |
| testoom(0, 0, ENOMEM, 1); |
| |
| if (access(MEMCG_SW_LIMIT, F_OK) == -1) { |
| if (errno == ENOENT) |
| tst_res(TCONF, |
| "memcg swap accounting is disabled"); |
| else |
| tst_brk(TBROK | TERRNO, "access"); |
| } else { |
| SAFE_FILE_PRINTF(MEMCG_SW_LIMIT, "%ld", TESTMEM); |
| testoom(0, 1, ENOMEM, 1); |
| } |
| |
| /* OOM for MEMCG with mempolicy */ |
| if (is_numa(NULL, NH_MEMS, 2)) { |
| tst_res(TINFO, "OOM on MEMCG & mempolicy..."); |
| testoom(MPOL_BIND, 0, ENOMEM, 1); |
| testoom(MPOL_INTERLEAVE, 0, ENOMEM, 1); |
| testoom(MPOL_PREFERRED, 0, ENOMEM, 1); |
| } |
| } |
| |
| static void setup(void) |
| { |
| overcommit = get_sys_tune("overcommit_memory"); |
| set_sys_tune("overcommit_memory", 1, 1); |
| mount_mem("memcg", "cgroup", "memory", MEMCG_PATH, MEMCG_PATH_NEW); |
| memcg_mounted = 1; |
| } |
| |
| static void cleanup(void) |
| { |
| if (overcommit != -1) |
| set_sys_tune("overcommit_memory", overcommit, 0); |
| if (memcg_mounted) |
| umount_mem(MEMCG_PATH, MEMCG_PATH_NEW); |
| } |
| |
| static struct tst_test test = { |
| .needs_root = 1, |
| .forks_child = 1, |
| .timeout = -1, |
| .setup = setup, |
| .cleanup = cleanup, |
| .test_all = verify_oom, |
| }; |
| |
| #else |
| TST_TEST_TCONF(NUMA_ERROR_MSG); |
| #endif |