1 Star 0 Fork 2

gaoxuelong / stress-ng

forked from HoperunHarmony / stress-ng 
加入 Gitee
与超过 1200万 开发者一起发现、参与优秀开源项目,私有仓库也完全免费 :)
免费加入
克隆/下载
stress-fork.c 7.86 KB
一键复制 编辑 原始数据 按行查看 历史
/*
* Copyright (C) 2013-2021 Canonical, Ltd.
* Copyright (C) 2022 Colin Ian King.
*
* 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, write to the Free Software
* Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
*
*/
#include "stress-ng.h"
#include "core-pragma.h"
#define MIN_FORKS (1)
#define MAX_FORKS (16000)
#define DEFAULT_FORKS (1)
#define MIN_VFORKS (1)
#define MAX_VFORKS (16000)
#define DEFAULT_VFORKS (1)
static const stress_help_t fork_help[] = {
{ "f N","fork N", "start N workers spinning on fork() and exit()" },
{ NULL, "fork-ops N", "stop after N fork bogo operations" },
{ NULL, "fork-max P", "create P workers per iteration, default is 1" },
{ NULL, "fork-vm", "enable extra virtual memory pressure" },
{ NULL, NULL, NULL }
};
static const stress_help_t vfork_help[] = {
{ NULL, "vfork N", "start N workers spinning on vfork() and exit()" },
{ NULL, "vfork-ops N", "stop after N vfork bogo operations" },
{ NULL, "vfork-max P", "create P processes per iteration, default is 1" },
{ NULL, "vfork-vm", "enable extra virtual memory pressure" },
{ NULL, NULL, NULL }
};
#define STRESS_FORK (0)
#define STRESS_VFORK (1)
/*
* stress_set_fork_max()
* set maximum number of forks allowed
*/
static int stress_set_fork_max(const char *opt)
{
uint32_t fork_max;
fork_max = stress_get_uint32(opt);
stress_check_range("fork-max", fork_max,
MIN_FORKS, MAX_FORKS);
return stress_set_setting("fork-max", TYPE_ID_UINT32, &fork_max);
}
/*
* stress_set_fork_vm()
* set fork-vm flag on
*/
static int stress_set_fork_vm(const char *opt)
{
bool vm = true;
(void)opt;
return stress_set_setting("fork-vm", TYPE_ID_BOOL, &vm);
}
/*
* stress_set_vfork_max()
* set maximum number of vforks allowed
*/
static int stress_set_vfork_max(const char *opt)
{
uint32_t vfork_max;
vfork_max = stress_get_uint32(opt);
stress_check_range("vfork-max", vfork_max,
MIN_VFORKS, MAX_VFORKS);
return stress_set_setting("vfork-max", TYPE_ID_UINT32, &vfork_max);
}
/*
* stress_set_vfork_vm()
* set vfork-vm flag on
*/
static int stress_set_vfork_vm(const char *opt)
{
bool vm = true;
(void)opt;
return stress_set_setting("vfork-vm", TYPE_ID_BOOL, &vm);
}
typedef struct {
pid_t pid; /* Child PID */
int err; /* Saved fork errno */
} fork_info_t;
/*
* stress_fork_fn()
* stress by forking and exiting using
* fork function fork_fn (fork or vfork)
*/
static int stress_fork_fn(
const stress_args_t *args,
const int which,
const uint32_t fork_max,
const bool vm)
{
static fork_info_t info[MAX_FORKS];
int ret;
#if defined(__APPLE__)
double time_end = stress_time_now() + (double)g_opt_timeout;
#endif
stress_set_oom_adjustment(args->name, true);
/* Explicitly drop capabilities, makes it more OOM-able */
ret = stress_drop_capabilities(args->name);
(void)ret;
do {
NOCLOBBER uint32_t i, n;
NOCLOBBER char *fork_fn_name;
(void)memset(info, 0, sizeof(info));
for (n = 0; n < fork_max; n++) {
pid_t pid;
switch (which) {
case STRESS_FORK:
fork_fn_name = "fork";
pid = fork();
break;
case STRESS_VFORK:
fork_fn_name = "vfork";
pid = shim_vfork();
break;
default:
/* This should not happen */
fork_fn_name = "unknown";
pid = -1;
pr_fail("%s: bad fork/vfork function, aborting\n", args->name);
errno = ENOSYS;
break;
}
if (pid == 0) {
#if defined(HAVE_GETPGID)
const pid_t my_pid = getpid();
const pid_t my_pgid = getpgid(my_pid);
#else
UNEXPECTED
#endif
/*
* With new session and capabilities
* dropped vhangup will always fail
* but it's useful to exercise this
* to get more kernel coverage
*/
if (setsid() != (pid_t) -1)
shim_vhangup();
if (vm) {
int flags = 0;
#if defined(MADV_MERGEABLE)
flags |= MADV_MERGEABLE;
#endif
#if defined(MADV_WILLNEED)
flags |= MADV_WILLNEED;
#endif
#if defined(MADV_HUGEPAGE)
flags |= MADV_HUGEPAGE;
#endif
#if defined(MADV_RANDOM)
flags |= MADV_RANDOM;
#endif
if (flags)
stress_madvise_pid_all_pages(getpid(), flags);
}
/* exercise some setpgid calls before we die */
ret = setpgid(0, 0);
(void)ret;
#if defined(HAVE_GETPGID)
ret = setpgid(my_pid, my_pgid);
(void)ret;
#else
UNEXPECTED
#endif
/* -ve pgid is EINVAL */
ret = setpgid(0, -1);
(void)ret;
/* -ve pid is EINVAL */
ret = setpgid(-1, 0);
(void)ret;
(void)shim_sched_yield();
_exit(0);
} else if (pid < 0) {
info[n].err = errno;
}
if (pid > -1)
(void)setpgid(pid, g_pgrp);
info[n].pid = pid;
if (!keep_stressing(args))
break;
}
for (i = 0; i < n; i++) {
if (info[i].pid > 0) {
int status;
/* Parent, kill and then wait for child */
/* (void)kill(info[i].pid, SIGKILL); no need to kill */
(void)shim_waitpid(info[i].pid, &status, 0);
inc_counter(args);
}
}
for (i = 0; i < n; i++) {
if ((info[i].pid < 0) && (g_opt_flags & OPT_FLAGS_VERIFY)) {
switch (info[i].err) {
case EAGAIN:
case ENOMEM:
break;
default:
pr_fail("%s: %s failed, errno=%d (%s)\n", args->name,
fork_fn_name, info[i].err, strerror(info[i].err));
break;
}
}
}
#if defined(__APPLE__)
/*
* SIGALRMs don't get reliably delivered on OS X on
* vfork so check the time in case SIGARLM was not
* delivered.
*/
if ((which == STRESS_VFORK) && (stress_time_now() > time_end))
break;
#endif
} while (keep_stressing(args));
return EXIT_SUCCESS;
}
/*
* stress_fork()
* stress by forking and exiting
*/
static int stress_fork(const stress_args_t *args)
{
uint32_t fork_max = DEFAULT_FORKS;
int rc;
bool vm = false;
(void)stress_get_setting("fork-vm", &vm);
if (!stress_get_setting("fork-max", &fork_max)) {
if (g_opt_flags & OPT_FLAGS_MAXIMIZE)
fork_max = MAX_FORKS;
if (g_opt_flags & OPT_FLAGS_MINIMIZE)
fork_max = MIN_FORKS;
}
stress_set_proc_state(args->name, STRESS_STATE_RUN);
rc = stress_fork_fn(args, STRESS_FORK, fork_max, vm);
stress_set_proc_state(args->name, STRESS_STATE_DEINIT);
return rc;
}
/*
* stress_vfork()
* stress by vforking and exiting
*/
STRESS_PRAGMA_PUSH
STRESS_PRAGMA_WARN_OFF
static int stress_vfork(const stress_args_t *args)
{
uint32_t vfork_max = DEFAULT_VFORKS;
int rc;
bool vm = false;
(void)stress_get_setting("vfork-vm", &vm);
if (!stress_get_setting("vfork-max", &vfork_max)) {
if (g_opt_flags & OPT_FLAGS_MAXIMIZE)
vfork_max = MAX_VFORKS;
if (g_opt_flags & OPT_FLAGS_MINIMIZE)
vfork_max = MIN_VFORKS;
}
stress_set_proc_state(args->name, STRESS_STATE_RUN);
rc = stress_fork_fn(args, STRESS_VFORK, vfork_max, vm);
stress_set_proc_state(args->name, STRESS_STATE_DEINIT);
return rc;
}
STRESS_PRAGMA_POP
static const stress_opt_set_func_t fork_opt_set_funcs[] = {
{ OPT_fork_max, stress_set_fork_max },
{ OPT_fork_vm, stress_set_fork_vm },
{ 0, NULL }
};
static const stress_opt_set_func_t vfork_opt_set_funcs[] = {
{ OPT_vfork_max, stress_set_vfork_max },
{ OPT_vfork_vm, stress_set_vfork_vm },
{ 0, NULL }
};
stressor_info_t stress_fork_info = {
.stressor = stress_fork,
.class = CLASS_SCHEDULER | CLASS_OS,
.opt_set_funcs = fork_opt_set_funcs,
.verify = VERIFY_OPTIONAL,
.help = fork_help
};
stressor_info_t stress_vfork_info = {
.stressor = stress_vfork,
.class = CLASS_SCHEDULER | CLASS_OS,
.opt_set_funcs = vfork_opt_set_funcs,
.help = vfork_help
};
1
https://gitee.com/gaoxuelong/stress-ng.git
git@gitee.com:gaoxuelong/stress-ng.git
gaoxuelong
stress-ng
stress-ng
master

搜索帮助

53164aa7 5694891 3bd8fe86 5694891