1 Star 0 Fork 2

HoperunHarmony/stress-ng

加入 Gitee
与超过 1200万 开发者一起发现、参与优秀开源项目,私有仓库也完全免费 :)
免费加入
克隆/下载
stress-stackmmap.c 7.25 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-put.h"
#if defined(HAVE_UCONTEXT_H)
#include <ucontext.h>
#else
UNEXPECTED
#endif
static const stress_help_t help[] = {
{ NULL, "stackmmap N", "start N workers exercising a filebacked stack" },
{ NULL, "stackmmap-ops N", "stop after N bogo stackmmap operations" },
{ NULL, NULL, NULL }
};
#if defined(HAVE_SWAPCONTEXT) && \
defined(HAVE_UCONTEXT_H)
#define MMAPSTACK_SIZE (256 * KB)
static ucontext_t c_main, c_test;
static uint8_t *stack_mmap; /* mmap'd stack */
static uintptr_t page_mask;
static size_t page_size;
/*
* push values onto file backed mmap'd stack and
* force msync on the map'd region if page boundary
* has changed
*/
static void stress_stackmmap_push_msync(void)
{
void *addr = (void *)(((uintptr_t)&addr) & page_mask);
static void *laddr;
uint32_t waste[2];
/*
* Ensure something is written to the stack that
* won't get optimized away
*/
waste[0] = stress_mwc32();
stress_uint32_put(waste[0]);
waste[1] = stress_mwc32();
stress_uint32_put(waste[1]);
stress_uint64_put((uint64_t)(intptr_t)&waste);
if (addr != laddr) {
(void)shim_msync(addr, page_size,
(stress_mwc8() & 1) ? MS_ASYNC : MS_SYNC);
laddr = addr;
}
if (keep_stressing_flag())
stress_stackmmap_push_msync();
stress_uint64_put((uint64_t)waste[1]);
}
/*
* start the push here
*/
static void stress_stackmmap_push_start(void)
{
stress_stackmmap_push_msync();
}
/*
* stress_stackmmap
* stress a file memory map'd stack
*/
static int stress_stackmmap(const stress_args_t *args)
{
int fd, ret;
volatile int rc = EXIT_FAILURE; /* could be clobbered */
char filename[PATH_MAX];
uint8_t *stack_sig;
struct sigaction new_action;
page_size = args->page_size;
page_mask = ~(page_size - 1);
/* Create file back'd mmaping for the stack */
ret = stress_temp_dir_mk_args(args);
if (ret < 0)
return exit_status(-ret);
(void)stress_temp_filename_args(args,
filename, sizeof(filename), stress_mwc32());
fd = open(filename, O_SYNC | O_RDWR | O_CREAT, S_IRUSR | S_IWUSR);
if (fd < 0) {
pr_fail("%s: open %s mmap'd stack file failed, errno=%d (%s)\n",
args->name, filename, errno, strerror(errno));
goto tidy_dir;
}
(void)shim_unlink(filename);
if (ftruncate(fd, MMAPSTACK_SIZE) < 0) {
pr_fail("%s: ftruncate failed, errno=%d (%s)\n",
args->name, errno, strerror(errno));
(void)close(fd);
goto tidy_dir;
}
stack_sig = (uint8_t *)mmap(NULL, STRESS_SIGSTKSZ,
PROT_READ | PROT_WRITE, MAP_SHARED | MAP_ANONYMOUS, -1, 0);
if (stack_sig == MAP_FAILED) {
pr_inf_skip("%s: skipping stressor, cannot mmap signal stack, "
"errno=%d (%s)\n",
args->name, errno, strerror(errno));
rc = EXIT_NO_RESOURCE;
(void)close(fd);
goto tidy_dir;
}
stack_mmap = (uint8_t *)mmap(NULL, MMAPSTACK_SIZE,
PROT_READ | PROT_WRITE, MAP_SHARED, fd, 0);
if (stack_mmap == MAP_FAILED) {
if (errno == ENXIO) {
pr_inf_skip("%s: skipping stressor, mmap not possible on file %s\n",
args->name, filename);
rc = EXIT_NO_RESOURCE;
(void)close(fd);
goto tidy_stack_sig;
}
pr_fail("%s: mmap failed, errno=%d (%s)\n",
args->name, errno, strerror(errno));
(void)close(fd);
goto tidy_stack_sig;
}
(void)close(fd);
if (shim_madvise((void *)stack_mmap, MMAPSTACK_SIZE, MADV_RANDOM) < 0) {
pr_dbg("%s: madvise failed: errno=%d (%s)\n",
args->name, errno, strerror(errno));
}
(void)memset(stack_mmap, 0, MMAPSTACK_SIZE);
#if defined(HAVE_MPROTECT)
/*
* Make ends of stack inaccessible
*/
(void)mprotect((void *)stack_mmap, page_size, PROT_NONE);
(void)mprotect((void *)(stack_mmap + MMAPSTACK_SIZE - page_size), page_size, PROT_NONE);
#else
UNEXPECTED
#endif
(void)memset(&c_test, 0, sizeof(c_test));
if (getcontext(&c_test) < 0) {
pr_fail("%s: getcontext failed, errno=%d (%s)\n",
args->name, errno, strerror(errno));
goto tidy_mmap;
}
c_test.uc_stack.ss_sp = (void *)(stack_mmap + page_size);
c_test.uc_stack.ss_size = MMAPSTACK_SIZE - (page_size * 2);
c_test.uc_link = &c_main;
stress_set_proc_state(args->name, STRESS_STATE_RUN);
/*
* set jmp handler to jmp back into the loop on a full
* stack segfault. Use swapcontext to jump into a
* new context using the new mmap'd stack
*/
do {
pid_t pid;
again:
if (!keep_stressing_flag())
break;
pid = fork();
if (pid < 0) {
if (stress_redo_fork(errno))
goto again;
if (!keep_stressing(args))
goto finish;
pr_err("%s: fork failed: errno=%d (%s)\n",
args->name, errno, strerror(errno));
} else if (pid > 0) {
int status, waitret;
/* Parent, wait for child */
(void)setpgid(pid, g_pgrp);
waitret = shim_waitpid(pid, &status, 0);
if (waitret < 0) {
if (errno != EINTR)
pr_dbg("%s: waitpid(): errno=%d (%s)\n",
args->name, errno, strerror(errno));
(void)kill(pid, SIGTERM);
(void)stress_killpid(pid);
(void)shim_waitpid(pid, &status, 0);
}
} else if (pid == 0) {
/* Child */
(void)setpgid(0, g_pgrp);
stress_parent_died_alarm();
(void)sched_settings_apply(true);
/* Make sure this is killable by OOM killer */
stress_set_oom_adjustment(args->name, true);
/*
* We need to handle SEGV signals when we
* hit the end of the mmap'd stack; however
* an alternative signal handling stack
* is required because we ran out of stack
*/
(void)memset(&new_action, 0, sizeof new_action);
new_action.sa_handler = stress_sig_handler_exit;
(void)sigemptyset(&new_action.sa_mask);
new_action.sa_flags = SA_ONSTACK;
if (sigaction(SIGSEGV, &new_action, NULL) < 0)
_exit(EXIT_FAILURE);
/*
* We need an alternative signal stack
* to handle segfaults on an overrun
* mmap'd stack
*/
if (stress_sigaltstack(stack_sig, STRESS_SIGSTKSZ) < 0)
_exit(EXIT_FAILURE);
(void)makecontext(&c_test, stress_stackmmap_push_start, 0);
(void)swapcontext(&c_main, &c_test);
_exit(0);
}
inc_counter(args);
} while (keep_stressing(args));
finish:
rc = EXIT_SUCCESS;
tidy_mmap:
stress_set_proc_state(args->name, STRESS_STATE_DEINIT);
(void)munmap((void *)stack_mmap, MMAPSTACK_SIZE);
tidy_stack_sig:
(void)munmap((void *)stack_sig, STRESS_SIGSTKSZ);
tidy_dir:
stress_set_proc_state(args->name, STRESS_STATE_DEINIT);
(void)stress_temp_dir_rm_args(args);
return rc;
}
stressor_info_t stress_stackmmap_info = {
.stressor = stress_stackmmap,
.class = CLASS_VM | CLASS_MEMORY,
.help = help
};
#else
stressor_info_t stress_stackmmap_info = {
.stressor = stress_not_implemented,
.class = CLASS_VM | CLASS_MEMORY,
.help = help
};
#endif
马建仓 AI 助手
尝试更多
代码解读
代码找茬
代码优化
1
https://gitee.com/hoperun_harmony/stress-ng.git
git@gitee.com:hoperun_harmony/stress-ng.git
hoperun_harmony
stress-ng
stress-ng
master

搜索帮助