1 Star 0 Fork 2

HoperunHarmony / stress-ng

加入 Gitee
与超过 1200万 开发者一起发现、参与优秀开源项目,私有仓库也完全免费 :)
免费加入
克隆/下载
stress-softlockup.c 6.99 KB
一键复制 编辑 原始数据 按行查看 历史
/*
* Copyright (C) 2013-2021 Canonical, Ltd.
* Copyright (C) 2021-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-capabilities.h"
static const stress_help_t help[] = {
{ NULL, "softlockup N", "start N workers that cause softlockups" },
{ NULL, "softlockup-ops N", "stop after N softlockup bogo operations" },
{ NULL, NULL, NULL }
};
#if defined(HAVE_SCHED_GET_PRIORITY_MIN) && \
defined(HAVE_SCHED_SETSCHEDULER)
/*
* stress_softlockup_supported()
* check if we can run this as root
*/
static int stress_softlockup_supported(const char *name)
{
if (!stress_check_capability(SHIM_CAP_SYS_NICE)) {
pr_inf_skip("%s stressor will be skipped, "
"need to be running with CAP_SYS_NICE "
"rights for this stressor\n", name);
return -1;
}
return 0;
}
typedef struct {
const int policy;
int max_prio;
const char *name;
} stress_policy_t;
static stress_policy_t policies[] = {
#if defined(SCHED_FIFO)
{ SCHED_FIFO, 0, "SCHED_FIFO"},
#endif
#if defined(SCHED_RR)
{ SCHED_RR, 0, "SCHED_RR" },
#endif
};
static sigjmp_buf jmp_env;
/*
* stress_rlimit_handler()
* rlimit generic handler
*/
static void MLOCKED_TEXT NORETURN stress_rlimit_handler(int signum)
{
(void)signum;
keep_stressing_set_flag(false);
siglongjmp(jmp_env, 1);
}
/*
* drop_niceness();
* see how low we can go with niceness
*/
static void drop_niceness(void)
{
int nice_val, i;
errno = 0;
nice_val = nice(0);
/* Should never fail */
if (errno)
return;
/*
* Traditionally no more than -20, but see if we
* can force it lower if we are originally running
* at nice level 19
*/
for (i = 0; i < 40; i++) {
int old_nice_val = nice_val;
errno = 0;
nice_val = nice(-1);
if (errno)
return;
if (nice_val == old_nice_val)
return;
}
}
static int stress_softlockup(const stress_args_t *args)
{
size_t policy = 0;
int max_prio = 0;
bool good_policy = false;
const bool first_instance = (args->instance == 0);
const uint32_t cpus_online = (uint32_t)stress_get_processors_online();
const uint32_t num_instances = args->num_instances;
struct sigaction old_action_xcpu;
struct sched_param param;
struct rlimit rlim;
pid_t pid;
NOCLOBBER uint64_t timeout;
const double start = stress_time_now();
timeout = g_opt_timeout;
(void)memset(&param, 0, sizeof(param));
if (SIZEOF_ARRAY(policies) == 0) {
if (first_instance) {
pr_inf_skip("%s: no scheduling policies "
"available, skipping test\n",
args->name);
}
return EXIT_NOT_IMPLEMENTED;
}
/* Get the max priorities for each sched policy */
for (policy = 0; policy < SIZEOF_ARRAY(policies); policy++) {
const int prio = sched_get_priority_max(policies[policy].policy);
policies[policy].max_prio = prio;
good_policy |= (prio >= 0);
if (max_prio < prio)
max_prio = prio;
}
/*
* We may have a kernel that does not support these sched
* policies, so check for this
*/
if (!good_policy) {
if (first_instance) {
pr_inf_skip("%s: cannot get valid maximum priorities for the "
"scheduling policies, skipping test\n",
args->name);
}
return EXIT_NOT_IMPLEMENTED;
}
if ((max_prio < 1) && (args->instance == 0)) {
pr_inf("%s: running with a low maximum priority of %d\n",
args->name, max_prio);
}
if ((num_instances < cpus_online) && (first_instance)) {
pr_inf("%s: for best results, run with at least %d instances "
"of this stressor\n", args->name, cpus_online);
}
stress_set_proc_state(args->name, STRESS_STATE_RUN);
again:
pid = fork();
if (pid < 0) {
if (stress_redo_fork(errno))
goto again;
if (!keep_stressing(args))
goto finish;
pr_inf("%s: cannot fork, errno=%d (%s)\n",
args->name, errno, strerror(errno));
return EXIT_NO_RESOURCE;
} else if (pid == 0) {
const pid_t mypid = getpid();
#if defined(HAVE_ATOMIC)
uint32_t count;
#endif
int ret;
int rc = EXIT_FAILURE;
#if defined(HAVE_ATOMIC)
__sync_fetch_and_add(&g_shared->softlockup_count, 1);
/*
* Wait until all instances have reached this point
*/
do {
if ((stress_time_now() - start) > (double)timeout)
goto tidy_ok;
(void)usleep(50000);
__atomic_load(&g_shared->softlockup_count, &count, __ATOMIC_RELAXED);
} while (keep_stressing(args) && count < num_instances);
#endif
/*
* We run the stressor as a child so that
* if we hit the hard time limits the child is
* terminated with a SIGKILL and we can
* catch that with the parent
*/
rlim.rlim_cur = timeout;
rlim.rlim_max = timeout;
(void)setrlimit(RLIMIT_CPU, &rlim);
#if defined(RLIMIT_RTTIME)
rlim.rlim_cur = 1000000 * timeout;
rlim.rlim_max = 1000000 * timeout;
(void)setrlimit(RLIMIT_RTTIME, &rlim);
#endif
if (stress_sighandler(args->name, SIGXCPU, stress_rlimit_handler, &old_action_xcpu) < 0)
goto tidy;
ret = sigsetjmp(jmp_env, 1);
if (ret)
goto tidy_ok;
drop_niceness();
policy = 0;
do {
/*
* Note: Re-setting the scheduler policy on Linux
* puts the runnable process always onto the front
* of the scheduling list.
*/
param.sched_priority = policies[policy].max_prio;
ret = sched_setscheduler(mypid, policies[policy].policy, &param);
if (ret < 0) {
if (errno != EPERM) {
pr_fail("%s: sched_setscheduler "
"failed: errno=%d (%s) "
"for scheduler policy %s\n",
args->name, errno, strerror(errno),
policies[policy].name);
}
}
drop_niceness();
policy++;
policy %= SIZEOF_ARRAY(policies);
inc_counter(args);
/* Ensure we NEVER spin forever */
if ((stress_time_now() - start) > (double)timeout)
break;
} while (keep_stressing(args));
tidy_ok:
rc = EXIT_SUCCESS;
tidy:
(void)fflush(stdout);
_exit(rc);
} else {
int status;
param.sched_priority = policies[0].max_prio;
(void)sched_setscheduler(args->pid, policies[0].policy, &param);
(void)pause();
(void)kill(pid, SIGKILL);
#if defined(HAVE_ATOMIC)
__sync_fetch_and_sub(&g_shared->softlockup_count, 1);
#endif
(void)shim_waitpid(pid, &status, 0);
}
finish:
stress_set_proc_state(args->name, STRESS_STATE_DEINIT);
return EXIT_SUCCESS;
}
stressor_info_t stress_softlockup_info = {
.stressor = stress_softlockup,
.supported = stress_softlockup_supported,
.class = CLASS_SCHEDULER,
.help = help
};
#else
stressor_info_t stress_softlockup_info = {
.stressor = stress_not_implemented,
.class = CLASS_SCHEDULER,
.help = help
};
#endif
1
https://gitee.com/hoperun_harmony/stress-ng.git
git@gitee.com:hoperun_harmony/stress-ng.git
hoperun_harmony
stress-ng
stress-ng
master

搜索帮助