1 Star 0 Fork 2

gaoxuelong / stress-ng

forked from HoperunHarmony / stress-ng 
加入 Gitee
与超过 1200万 开发者一起发现、参与优秀开源项目,私有仓库也完全免费 :)
免费加入
克隆/下载
stress-mlockmany.c 5.87 KB
一键复制 编辑 原始数据 按行查看 历史
Colin Ian King 提交于 2022-01-29 11:31 . core-, stress-: update copyright
/*
* 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"
#define UNSET_MLOCK_PROCS (0)
#define DEFAULT_MLOCK_PROCS (1024)
static const stress_help_t help[] = {
{ NULL, "mlockmany N", "start N workers exercising many mlock/munlock processes" },
{ NULL, "mlockmany-ops N", "stop after N mlockmany bogo operations" },
{ NULL, "mlockmany-procs N", "use N child processes to mlock regions" },
{ NULL, NULL, NULL }
};
/*
* stress_set_mlockmany_procs()
* set number of processes to spawn to mlock pages
*/
static int stress_set_mlockmany_procs(const char *opt)
{
size_t mlockmany_procs;
mlockmany_procs = (size_t)stress_get_uint64(opt);
stress_check_range("mlockmany-procs", mlockmany_procs,
1, 1000000);
return stress_set_setting("mlockmany-procs", TYPE_ID_SIZE_T, &mlockmany_procs);
}
static const stress_opt_set_func_t opt_set_funcs[] = {
{ OPT_mlockmany_procs, stress_set_mlockmany_procs },
{ 0, NULL }
};
#if defined(HAVE_MLOCK)
/*
* stress_mlockmany()
* stress by forking and exiting
*/
static int stress_mlockmany(const stress_args_t *args)
{
pid_t *pids;
int ret;
#if defined(RLIMIT_MEMLOCK)
struct rlimit rlim;
#endif
size_t mlock_size, mlockmany_procs = UNSET_MLOCK_PROCS;
(void)stress_get_setting("mlockmany-procs", &mlockmany_procs);
stress_set_oom_adjustment(args->name, true);
/* Explicitly drop capabilities, makes it more OOM-able */
ret = stress_drop_capabilities(args->name);
(void)ret;
if (mlockmany_procs == UNSET_MLOCK_PROCS) {
mlockmany_procs = args->num_instances > 0 ? DEFAULT_MLOCK_PROCS / args->num_instances : 1;
if (mlockmany_procs < 1)
mlockmany_procs = 1;
}
pids = calloc((size_t)mlockmany_procs, sizeof(*pids));
if (!pids) {
pr_inf("%s: cannot allocate pids array\n", args->name);
return EXIT_NO_RESOURCE;
}
#if defined(RLIMIT_MEMLOCK)
ret = getrlimit(RLIMIT_MEMLOCK, &rlim);
if (ret < 0) {
mlock_size = 8 * MB;
} else {
mlock_size = rlim.rlim_cur;
}
#else
mlock_size = args->page_size * 1024;
#endif
stress_set_proc_state(args->name, STRESS_STATE_RUN);
do {
unsigned int i, n;
size_t shmall, freemem, totalmem, freeswap, last_freeswap;
(void)memset(pids, 0, sizeof(*pids) * mlockmany_procs);
stress_get_memlimits(&shmall, &freemem, &totalmem, &last_freeswap);
for (n = 0; n < mlockmany_procs; n++) {
pid_t pid;
if (!keep_stressing(args))
break;
stress_get_memlimits(&shmall, &freemem, &totalmem, &freeswap);
/* We detected swap being used, bail out */
if (last_freeswap > freeswap)
break;
/* Keep track of expanding free swap space */
if (freeswap > last_freeswap)
last_freeswap = freeswap;
pid = fork();
if (pid == 0) {
void *ptr = MAP_FAILED;
size_t mmap_size = mlock_size;
(void)setpgid(0, g_pgrp);
stress_parent_died_alarm();
stress_set_oom_adjustment(args->name, true);
(void)sched_settings_apply(true);
shim_mlockall(0);
stress_get_memlimits(&shmall, &freemem, &totalmem, &freeswap);
/* We detected swap being used, bail out */
if (last_freeswap > freeswap)
_exit(0);
while (mmap_size > args->page_size) {
if (!keep_stressing(args))
_exit(0);
ptr = mmap(NULL, mmap_size, PROT_READ | PROT_WRITE,
MAP_PRIVATE | MAP_ANONYMOUS, -1, 0);
if (ptr != MAP_FAILED)
break;
mmap_size >>= 1;
}
if (ptr == MAP_FAILED)
_exit(0);
(void)stress_mincore_touch_pages(ptr, mmap_size);
mlock_size = mmap_size;
while (mlock_size > args->page_size) {
if (!keep_stressing(args))
_exit(0);
ret = shim_mlock(ptr, mlock_size);
if (ret == 0)
break;
mlock_size >>= 1;
}
for (;;) {
if (!keep_stressing(args))
goto unlock;
(void)shim_munlock(ptr, mlock_size);
if (!keep_stressing(args))
goto unmap;
(void)shim_mlock(ptr, mlock_size);
if (!keep_stressing(args))
goto unlock;
/* Try invalid sizes */
(void)shim_mlock(ptr, 0);
(void)shim_munlock(ptr, 0);
(void)shim_mlock(ptr, mlock_size << 1);
(void)shim_munlock(ptr, mlock_size << 1);
(void)shim_mlock(ptr, ~(size_t)0);
(void)shim_munlock(ptr, ~(size_t)0);
if (!keep_stressing(args))
goto unlock;
(void)shim_usleep_interruptible(10000);
}
unlock:
(void)shim_munlock(ptr, mlock_size);
unmap:
(void)munmap(ptr, mmap_size);
_exit(0);
}
if (pid > -1)
(void)setpgid(pids[n], g_pgrp);
pids[n] = pid;
if (!keep_stressing_flag())
break;
}
for (i = 0; i < n; i++) {
if (pids[i] > 0) {
int status;
/* Parent, wait for child */
(void)stress_killpid(pids[i]);
(void)shim_waitpid(pids[i], &status, 0);
inc_counter(args);
}
}
} while (keep_stressing(args));
stress_set_proc_state(args->name, STRESS_STATE_DEINIT);
free(pids);
return EXIT_SUCCESS;
}
stressor_info_t stress_mlockmany_info = {
.stressor = stress_mlockmany,
.class = CLASS_VM | CLASS_OS | CLASS_PATHOLOGICAL,
.opt_set_funcs = opt_set_funcs,
.help = help
};
#else
stressor_info_t stress_mlockmany_info = {
.stressor = stress_not_implemented,
.class = CLASS_VM | CLASS_OS | CLASS_PATHOLOGICAL,
.opt_set_funcs = opt_set_funcs,
.help = help
};
#endif
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