/*
* Copyright (c) 2017 Fujitsu Ltd.
* Ported: 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 about the race in autogroup, this test
* can crash the buggy kernel, and the bug has been fixed in:
*
* commit 18f649ef344127ef6de23a5a4272dbe2fdb73dde
* Author: Oleg Nesterov <oleg@redhat.com>
* Date: Mon Nov 14 19:46:09 2016 +0100
*
* sched/autogroup: Fix autogroup_move_group() to never skip sched_move_task()
*/
#include <sys/types.h>
#include <unistd.h>
#include <stdlib.h>
#include "tst_test.h"
#define LOOPS 1000
#define PATH_AUTOGROUP "/proc/sys/kernel/sched_autogroup_enabled"
static int orig_autogroup = -1;
static void do_test(void)
{
int i;
if (!SAFE_FORK()) {
SAFE_FILE_PRINTF(PATH_AUTOGROUP, "%d", 1);
SAFE_SETSID();
if (SAFE_FORK())
pause();
SAFE_KILL(getppid(), SIGKILL);
usleep(1000);
// The child has gone, the grandchild runs with kref == 1
SAFE_FILE_PRINTF(PATH_AUTOGROUP, "%d", 0);
SAFE_SETSID();
// runs with the freed ag/tg
for (i = 0; i < LOOPS; i++)
usleep(10);
TST_CHECKPOINT_WAKE(0);
exit(0);
}
SAFE_WAIT(NULL); // destroy the child's ag/tg
TST_CHECKPOINT_WAIT(0);
tst_res(TPASS, "Bug not reproduced");
}
static void setup(void)
{
if (access(PATH_AUTOGROUP, F_OK))
tst_brk(TCONF, "autogroup not supported");
SAFE_FILE_SCANF(PATH_AUTOGROUP, "%d", &orig_autogroup);
}
static void cleanup(void)
{
if (orig_autogroup != -1)
SAFE_FILE_PRINTF(PATH_AUTOGROUP, "%d", orig_autogroup);
}
static struct tst_test test = {
.forks_child = 1,
.needs_root = 1,
.needs_checkpoints = 1,
.setup = setup,
.cleanup = cleanup,
.test_all = do_test,
};