1 Star 0 Fork 2

HoperunHarmony / stress-ng

加入 Gitee
与超过 1200万 开发者一起发现、参与优秀开源项目,私有仓库也完全免费 :)
免费加入
克隆/下载
stress-chmod.c 6.73 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"
#if defined(HAVE_LIBGEN_H)
#include <libgen.h>
#endif
static const stress_help_t help[] = {
{ NULL, "chmod N", "start N workers thrashing chmod file mode bits " },
{ NULL, "chmod-ops N", "stop chmod workers after N bogo operations" },
{ NULL, NULL, NULL }
};
static const mode_t modes[] = {
#if defined(S_ISUID)
S_ISUID,
#endif
#if defined(S_ISGID)
S_ISGID,
#endif
#if defined(S_ISVTX)
S_ISVTX,
#endif
#if defined(S_IRUSR)
S_IRUSR,
#endif
#if defined(S_IWUSR)
S_IWUSR,
#endif
#if defined(S_IXUSR)
S_IXUSR,
#endif
#if defined(S_IRGRP)
S_IRGRP,
#endif
#if defined(S_IWGRP)
S_IWGRP,
#endif
#if defined(S_IXGRP)
S_IXGRP,
#endif
#if defined(S_IROTH)
S_IROTH,
#endif
#if defined(S_IWOTH)
S_IWOTH,
#endif
#if defined( S_IXOTH)
S_IXOTH,
#endif
0
};
/*
* BSD systems can return EFTYPE which we can ignore
* as a "known" error on invalid chmod mode bits
*/
#if defined(EFTYPE)
#define CHECK(x) \
if ((x) && \
((errno != ENOSYS) && \
(errno != EPERM) && \
(errno != EFTYPE))) \
return -1
#else
#define CHECK(x) \
if ((x) && \
((errno != ENOSYS) && \
(errno != EPERM))) \
return -1
#endif
/*
* do_fchmod()
* fchmod the 4 different masks from a mode flag, namely:
* mode flag
* all mode flags or'd together
* inverse mode flag
* inverse all mode flags or'd together
*/
static int do_fchmod(
const int fd,
const int bad_fd,
const size_t i,
const mode_t mask,
const mode_t all_mask)
{
int ret;
CHECK(fchmod(fd, modes[i]) < 0);
CHECK(fchmod(fd, mask) < 0);
CHECK(fchmod(fd, modes[i] ^ all_mask) < 0);
CHECK(fchmod(fd, mask ^ all_mask) < 0);
/*
* Exercise bad fchmod, ignore failure
*/
ret = fchmod(bad_fd, modes[i]);
(void)ret;
return 0;
}
/*
* do_chmod()
* chmod the 4 different masks from a mode flag, namely:
* mode flag
* all mode flags or'd together
* inverse mode flag
* inverse all mode flags or'd together
*/
static int do_chmod(
const int dfd,
const int bad_fd,
const char *filebase,
const char *filename,
const char *longpath,
const size_t i,
const mode_t mask,
const mode_t all_mask)
{
int ret;
CHECK(chmod(filename, modes[i]) < 0);
CHECK(chmod(filename, mask) < 0);
CHECK(chmod(filename, modes[i] ^ all_mask) < 0);
CHECK(chmod(filename, mask ^ all_mask) < 0);
#if defined(HAVE_FCHMODAT)
if (dfd >= 0) {
CHECK(fchmodat(dfd, filebase, modes[i], 0) < 0);
CHECK(fchmodat(dfd, filebase, mask, 0) < 0);
CHECK(fchmodat(dfd, filebase, modes[i] ^ all_mask, 0) < 0);
CHECK(fchmodat(dfd, filebase, mask ^ all_mask, 0) < 0);
/*
* Exercise bad fchmodat, ignore failure
*/
ret = fchmodat(bad_fd, filebase, modes[i], 0);
(void)ret;
}
#else
(void)dfd;
(void)bad_fd;
(void)filebase;
#endif
/*
* Exercise illegal filename
*/
ret = chmod("", modes[i]);
(void)ret;
/*
* Exercise illegal overly long pathname
*/
ret = chmod(longpath, modes[i]);
(void)ret;
return 0;
}
/*
* stress_chmod
* stress chmod
*/
static int stress_chmod(const stress_args_t *args)
{
const pid_t ppid = getppid();
int fd = -1, rc = EXIT_FAILURE, retries = 0, dfd = -1;
size_t i;
const int bad_fd = stress_get_bad_fd();
mode_t all_mask = 0;
char filename[PATH_MAX], pathname[PATH_MAX], longpath[PATH_MAX + 16];
char tmp[PATH_MAX], *filebase;
/*
* Allow for multiple workers to chmod the *same* file
*/
stress_temp_dir(pathname, sizeof(pathname), args->name, ppid, 0);
if (mkdir(pathname, S_IRUSR | S_IRWXU) < 0) {
if (errno != EEXIST) {
rc = exit_status(errno);
pr_fail("%s: mkdir %s failed, errno=%d (%s)\n",
args->name, pathname, errno, strerror(errno));
return rc;
}
}
#if defined(O_DIRECTORY)
dfd = open(pathname, O_DIRECTORY | O_RDONLY);
#else
UNEXPECTED
#endif
stress_strnrnd(longpath, sizeof(longpath));
longpath[0] = '/';
(void)stress_temp_filename(filename, sizeof(filename),
args->name, ppid, 0, 0);
(void)shim_strlcpy(tmp, filename, sizeof(tmp));
filebase = basename(tmp);
if (args->instance == 0) {
if ((fd = creat(filename, S_IRUSR | S_IWUSR)) < 0) {
rc = exit_status(errno);
pr_fail("%s: create %s failed, errno=%d (%s)\n",
args->name, filename, errno, strerror(errno));
goto tidy;
}
} else {
/* Other instances must try to open the file */
for (;;) {
if ((fd = open(filename, O_RDWR, S_IRUSR | S_IWUSR)) > - 1)
break;
#if defined(__NetBSD__)
/* For some reason usleep blocks */
(void)shim_sched_yield();
#else
(void)shim_usleep(100000);
#endif
/* Timed out, then give up */
if (!keep_stressing_flag()) {
rc = EXIT_SUCCESS;
goto tidy;
}
/* Too many retries? */
if (++retries >= 10000) {
pr_err("%s: chmod: file %s took %d "
"retries to open and gave up "
"(instance %" PRIu32 ")\n",
args->name, filename, retries, args->instance);
goto tidy;
}
}
}
for (i = 0; modes[i]; i++)
all_mask |= modes[i];
stress_set_proc_state(args->name, STRESS_STATE_RUN);
do {
mode_t mask = 0;
for (i = 0; modes[i]; i++) {
mask |= modes[i];
if (do_fchmod(fd, bad_fd, i, mask, all_mask) < 0) {
pr_fail("%s: fchmod failed, errno=%d (%s)\n",
args->name, errno, strerror(errno));
}
if (do_chmod(dfd, bad_fd, filebase, filename, longpath, i, mask, all_mask) < 0) {
if ((errno == ENOENT) || (errno == ENOTDIR)) {
/*
* File was removed during test by
* another worker
*/
rc = EXIT_SUCCESS;
goto tidy;
}
pr_fail("%s: chmod %s failed, errno=%d (%s)\n",
args->name, filename, errno, strerror(errno));
}
}
inc_counter(args);
} while (keep_stressing(args));
rc = EXIT_SUCCESS;
tidy:
stress_set_proc_state(args->name, STRESS_STATE_DEINIT);
#if defined(O_DIRECTORY)
if (dfd >= 0)
(void)close(dfd);
#else
UNEXPECTED
#endif
if (fd >= 0) {
(void)fchmod(fd, 0666);
(void)close(fd);
}
(void)shim_unlink(filename);
(void)shim_rmdir(pathname);
return rc;
}
stressor_info_t stress_chmod_info = {
.stressor = stress_chmod,
.class = CLASS_FILESYSTEM | CLASS_OS,
.verify = VERIFY_ALWAYS,
.help = help
};
1
https://gitee.com/hoperun_harmony/stress-ng.git
git@gitee.com:hoperun_harmony/stress-ng.git
hoperun_harmony
stress-ng
stress-ng
master

搜索帮助