if (rc > 0) {
ksft_print_msg("unexpected success from execveat(2)\n");
ksft_test_result_fail("%s\n", test_name); return 1;
} if (errno != expected_errno) {
ksft_print_msg("expected errno %d (%s) not %d (%s)\n",
expected_errno, strerror(expected_errno),
errno, strerror(errno));
ksft_test_result_fail("%s\n", test_name); return 1;
}
ksft_test_result_pass("%s\n", test_name); return 0;
}
staticint check_execveat_invoked_rc(int fd, constchar *path, int flags, int expected_rc, int expected_rc2)
{ char test_name[TEST_NAME_LEN]; int status; int rc;
pid_t child; int pathlen = path ? strlen(path) : 0;
#define XX_DIR_LEN 200 staticint check_execveat_pathmax(int root_dfd, constchar *src, int is_script)
{ int fail = 0; int ii, count, len; char longname[XX_DIR_LEN + 1]; int fd;
if (*longpath == '\0') { /* Create a filename close to PATH_MAX in length */ char *cwd = getcwd(NULL, 0);
/* * Execute as a pre-opened file descriptor, which works whether this is * a script or not (because the interpreter sees a filename like * "/dev/fd/20").
*/
fd = open(longpath, O_RDONLY); if (fd > 0) {
ksft_print_msg("Invoke copy of '%s' via filename of length %zu:\n",
src, strlen(longpath));
fail += check_execveat(fd, "", AT_EMPTY_PATH);
} else {
ksft_print_msg("Failed to open length %zu filename, errno=%d (%s)\n",
strlen(longpath), errno, strerror(errno));
fail++;
}
/* * Execute as a long pathname relative to "/". If this is a script, * the interpreter will launch but fail to open the script because its * name ("/dev/fd/5/xxx....") is bigger than PATH_MAX. * * The failure code is usually 127 (POSIX: "If a command is not found, * the exit status shall be 127."), but some systems give 126 (POSIX: * "If the command name is found, but it is not an executable utility, * the exit status shall be 126."), so allow either.
*/ if (is_script) {
ksft_print_msg("Invoke script via root_dfd and relative filename\n");
fail += check_execveat_invoked_rc(root_dfd, longpath + 1, 0,
127, 126);
} else {
ksft_print_msg("Invoke exec via root_dfd and relative filename\n");
fail += check_execveat(root_dfd, longpath + 1, 0);
}
ksft_print_msg("Check execveat(AT_EMPTY_PATH)'s comm is %s\n",
expected);
ret = check_execveat_invoked_rc(fd, "", AT_EMPTY_PATH, 0, 0);
envp[1] = old_env;
argv[0] = old_argv0;
return ret;
}
staticint run_tests(void)
{ int fail = 0; char *fullname = realpath("execveat", NULL); char *fullname_script = realpath("script", NULL); char *fullname_symlink = concat(fullname, ".symlink"); int subdir_dfd = open_or_die("subdir", O_DIRECTORY|O_RDONLY); int subdir_dfd_ephemeral = open_or_die("subdir.ephemeral",
O_DIRECTORY|O_RDONLY); int dot_dfd = open_or_die(".", O_DIRECTORY|O_RDONLY); int root_dfd = open_or_die("/", O_DIRECTORY|O_RDONLY); int dot_dfd_path = open_or_die(".", O_DIRECTORY|O_RDONLY|O_PATH); int dot_dfd_cloexec = open_or_die(".", O_DIRECTORY|O_RDONLY|O_CLOEXEC); int fd = open_or_die("execveat", O_RDONLY); int fd_path = open_or_die("execveat", O_RDONLY|O_PATH); int fd_symlink = open_or_die("execveat.symlink", O_RDONLY); int fd_denatured = open_or_die("execveat.denatured", O_RDONLY); int fd_denatured_path = open_or_die("execveat.denatured",
O_RDONLY|O_PATH); int fd_script = open_or_die("script", O_RDONLY); int fd_ephemeral = open_or_die("execveat.ephemeral", O_RDONLY); int fd_ephemeral_path = open_or_die("execveat.path.ephemeral",
O_RDONLY|O_PATH); int fd_script_ephemeral = open_or_die("script.ephemeral", O_RDONLY); int fd_cloexec = open_or_die("execveat", O_RDONLY|O_CLOEXEC); int fd_script_cloexec = open_or_die("script", O_RDONLY|O_CLOEXEC);
/* Check if we have execveat at all, and bail early if not */
errno = 0;
execveat_(-1, NULL, NULL, NULL, 0); if (errno == ENOSYS) {
ksft_exit_skip( "ENOSYS calling execveat - no kernel support?\n");
}
/* Change file position to confirm it doesn't affect anything */
lseek(fd, 10, SEEK_SET);
/* Mess with executable file that's already open: */ /* fd + no path to a file that's been renamed */
rename("execveat.ephemeral", "execveat.moved");
fail += check_execveat(fd_ephemeral, "", AT_EMPTY_PATH); /* fd + no path to a file that's been deleted */
unlink("execveat.moved"); /* remove the file now fd open */
fail += check_execveat(fd_ephemeral, "", AT_EMPTY_PATH);
/* Mess with executable file that's already open with O_PATH */ /* fd + no path to a file that's been deleted */
unlink("execveat.path.ephemeral");
fail += check_execveat(fd_ephemeral_path, "", AT_EMPTY_PATH);
/* Mess with script file that's already open: */ /* fd + no path to a file that's been renamed */
rename("script.ephemeral", "script.moved");
fail += check_execveat(fd_script_ephemeral, "", AT_EMPTY_PATH); /* fd + no path to a file that's been deleted */
unlink("script.moved"); /* remove the file while fd open */
fail += check_execveat(fd_script_ephemeral, "", AT_EMPTY_PATH);
/* Rename a subdirectory in the path: */
rename("subdir.ephemeral", "subdir.moved");
fail += check_execveat(subdir_dfd_ephemeral, "../script", 0);
fail += check_execveat(subdir_dfd_ephemeral, "script", 0); /* Remove the subdir and its contents */
unlink("subdir.moved/script");
unlink("subdir.moved"); /* Shell loads via deleted subdir OK because name starts with .. */
fail += check_execveat(subdir_dfd_ephemeral, "../script", 0);
fail += check_execveat_fail(subdir_dfd_ephemeral, "script", 0, ENOENT);
int main(int argc, char **argv)
{ int ii; int rc; constchar *verbose = getenv("VERBOSE"); constchar *check_comm = getenv(CHECK_COMM);
if (argc >= 2 || check_comm) { /* * If we are invoked with an argument, or no arguments but a * command to check, don't run tests.
*/ constchar *in_test = getenv("IN_TEST");
if (verbose) {
ksft_print_msg("invoked with:\n"); for (ii = 0; ii < argc; ii++)
ksft_print_msg("\t[%d]='%s\n'", ii, argv[ii]);
}
/* If the tests wanted us to check the command, do so. */ if (check_comm) { /* TASK_COMM_LEN == 16 */ char buf[32]; int fd, ret;
fd = open("/proc/self/comm", O_RDONLY); if (fd < 0) {
ksft_perror("open() comm failed"); exit(1);
}
ret = read(fd, buf, sizeof(buf)); if (ret < 0) {
ksft_perror("read() comm failed");
close(fd); exit(1);
}
close(fd);
Die Informationen auf dieser Webseite wurden
nach bestem Wissen sorgfältig zusammengestellt. Es wird jedoch weder Vollständigkeit, noch Richtigkeit,
noch Qualität der bereit gestellten Informationen zugesichert.
Bemerkung:
Die farbliche Syntaxdarstellung und die Messung sind noch experimentell.