// SPDX-License-Identifier: GPL-2.0-or-later
/* Copyright (c) 2018 FUJITSU LIMITED. All rights reserved.
* Authors: Jinhui huang <huangjh.jy@cn.fujitsu.com>
*/
/* Test Description:
* Check various errnos for execveat(2):
* 1) execveat() fails and returns EBADF if dirfd is a invalid file
* descriptor.
* 2) execveat() fails and returns EINVAL if flag specified is invalid.
* 3) execveat() fails and returns ELOOP if the file identified by dirfd and
* pathname is a symbolic link and flag includes AT_SYMLINK_NOFOLLOW.
* 4) execveat() fails and returns ENOTDIR if pathname is relative and dirfd
* is a file descriptor referring to a file other than a directory.
*/
#define _GNU_SOURCE
#include "config.h"
#include <stdio.h>
#include <errno.h>
#include "tst_test.h"
#include "lapi/execveat.h"
#include "lapi/fcntl.h"
#include "execveat.h"
#define TESTDIR "testdir"
#define TEST_APP "execveat_errno"
#define TEST_SYMLINK "execveat_symlink"
#define TEST_REL_APP TESTDIR"/"TEST_APP
#define TEST_ERL_SYMLINK TESTDIR"/"TEST_SYMLINK
static int bad_fd = -1, fd;
static char app_abs_path[512], app_sym_path[512];
static struct tcase {
int *fd;
char *pathname;
int flag;
int exp_err;
} tcases[] = {
{&bad_fd, "", AT_EMPTY_PATH, EBADF},
{&fd, app_abs_path, -1, EINVAL},
{&fd, app_sym_path, AT_SYMLINK_NOFOLLOW, ELOOP},
{&fd, TEST_REL_APP, 0, ENOTDIR},
};
static void verify_execveat(unsigned int i)
{
struct tcase *tc = &tcases[i];
char *argv[2] = {TEST_APP, NULL};
pid_t pid;
pid = SAFE_FORK();
if (pid == 0) {
TEST(execveat(*tc->fd, tc->pathname, argv, environ, tc->flag));
if (tc->exp_err != TST_ERR) {
tst_res(TFAIL | TTERRNO,
"execveat() fails unexpectedly, expected: %s",
tst_strerrno(tc->exp_err));
} else {
tst_res(TPASS | TTERRNO,
"execveat() fails as expected");
}
}
}
static void setup(void)
{
char cur_dir_path[512];
check_execveat();
SAFE_MKDIR(TESTDIR, 0777);
SAFE_CP(TEST_APP, TEST_REL_APP);
SAFE_GETCWD(cur_dir_path, sizeof(cur_dir_path));
sprintf(app_abs_path, "%s/%s", cur_dir_path, TEST_REL_APP);
sprintf(app_sym_path, "%s/%s", cur_dir_path, TEST_ERL_SYMLINK);
SAFE_SYMLINK(TEST_REL_APP, TEST_ERL_SYMLINK);
fd = SAFE_OPEN(TEST_REL_APP, O_PATH);
}
static const char *const resource_files[] = {
TEST_APP,
NULL,
};
static void cleanup(void)
{
if (fd > 0)
SAFE_CLOSE(fd);
}
static struct tst_test test = {
.resource_files = resource_files,
.tcnt = ARRAY_SIZE(tcases),
.test = verify_execveat,
.child_needs_reinit = 1,
.forks_child = 1,
.cleanup = cleanup,
.setup = setup,
};