/* * Copyright (c) 2017 Fujitsu Ltd. * Author: Guangwen Feng <fenggw-fnst@cn.fujitsu.com> * * 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. * * You should have received a copy of the GNU General Public License * along with this program, if not, see <http://www.gnu.org/licenses/>. */ /* * This is a regression test for a crash caused by memcg function * reentrant on RHEL6. When doing rmdir(), a pending signal can * interrupt the execution and lead to cgroup_clear_css_refs() * being entered repeatedly, this results in a BUG_ON(). * * This bug was introduced by following RHEL6 patch on 2.6.32-488.el6: * * [mm] memcg: fix race condition between memcg teardown and swapin * Link: https://bugzilla.redhat.com/show_bug.cgi?id=1001197 * Patch: ftp://partners.redhat.com/1c5d859a/de6aafa8185ed8fd934f2debc72b79eb/kernel-individual-patch/rhel6/v2.6.32-to-kernel-2.6.32-488.el6.tar.bz2 * 31675-mm-memcg-fix-race-condition-between-memcg-teardown-.patch * * This test can crash the buggy kernel on RHEL6.6GA, and the bug * was fixed by following patch on 2.6.32-536.el6: * * [mm] memcg: fix crash in re-entrant cgroup_clear_css_refs() * Link: https://bugzilla.redhat.com/show_bug.cgi?id=1168185 * Patch: ftp://partners.redhat.com/1c5d859a/de6aafa8185ed8fd934f2debc72b79eb/kernel-individual-patch/rhel6/v2.6.32-to-kernel-2.6.32-536.el6.tar.bz2 * 35944-mm-memcg-fix-crash-in-re-entrant-cgroup_clear_css_r.patch */ #include <errno.h> #include <unistd.h> #include <stdlib.h> #include <sys/types.h> #include <sys/mount.h> #include "tst_test.h" #define MNTPOINT "memcg" #define SUBDIR "memcg/testdir" static int mount_flag; static volatile int sigcounter; static void sighandler(int sig LTP_ATTRIBUTE_UNUSED) { sigcounter++; } static void do_child(void) { while (1) SAFE_KILL(getppid(), SIGUSR1); exit(0); } static void do_test(void) { pid_t cpid; SAFE_SIGNAL(SIGUSR1, sighandler); cpid = SAFE_FORK(); if (cpid == 0) do_child(); while (sigcounter < 50000) { if (access(SUBDIR, F_OK)) SAFE_MKDIR(SUBDIR, 0644); rmdir(SUBDIR); } SAFE_KILL(cpid, SIGKILL); SAFE_WAIT(NULL); tst_res(TPASS, "Bug not reproduced"); } static void setup(void) { int ret; SAFE_MKDIR(MNTPOINT, 0644); ret = mount("memcg", MNTPOINT, "cgroup", 0, "memory"); if (ret) { if (errno == ENOENT) tst_brk(TCONF | TERRNO, "memcg not supported"); tst_brk(TCONF | TERRNO, "mounting memcg failed"); } mount_flag = 1; } static void cleanup(void) { if (!access(SUBDIR, F_OK)) SAFE_RMDIR(SUBDIR); if (mount_flag) tst_umount(MNTPOINT); } static struct tst_test test = { .needs_root = 1, .needs_tmpdir = 1, .forks_child = 1, .min_kver = "2.6.24", .setup = setup, .cleanup = cleanup, .test_all = do_test, };