/*
* Copyright (c) 2014 Fujitsu Ltd.
* Author: Xiaoguang Wang <wangxg.fnst@cn.fujitsu.com>
*
* This program is free software; you can redistribute it and/or modify it
* under the terms of version 2 of the GNU General Public License as
* published by the Free Software Foundation.
*
* This program is distributed in the hope that it would be useful, but
* WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.
*
* You should have received a copy of the GNU General Public License along
* with this program; if not, write the Free Software Foundation, Inc.,
* 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA.
*/
/*
* Description:
* Verify that:
* Basic test for fcntl(2) using F_GETOWN, F_SETOWN, F_GETOWN_EX,
* F_SETOWN_EX, F_GETSIG, F_SETSIG argument.
*/
#include <stdio.h>
#include <errno.h>
#include <unistd.h>
#include <fcntl.h>
#include <string.h>
#include <signal.h>
#include <sys/types.h>
#include <sys/wait.h>
#include <pwd.h>
#include <sched.h>
#include "test.h"
#include "config.h"
#include "lapi/syscalls.h"
#include "safe_macros.h"
#include "lapi/fcntl.h"
static void setup(void);
static void cleanup(void);
static void setown_pid_test(void);
static void setown_pgrp_test(void);
#if defined(HAVE_STRUCT_F_OWNER_EX)
static int ownex_enabled;
static char *ownex_tconf_msg = "F_GETOWN_EX and F_SETOWN_EX only run on "
"kernels that are 2.6.32 and higher";
static void setownex_tid_test(void);
static void setownex_pid_test(void);
static void setownex_pgrp_test(void);
static struct f_owner_ex orig_own_ex;
#endif
static void signal_parent(void);
static void check_io_signal(char *des);
static void test_set_and_get_sig(int sig, char *des);
static pid_t pid;
static pid_t orig_pid;
static pid_t pgrp_pid;
static struct timespec timeout;
static sigset_t newset, oldset;
static int test_fd;
static int pipe_fds[2];
static void (*testfunc[])(void) = {
setown_pid_test, setown_pgrp_test,
#if defined(HAVE_STRUCT_F_OWNER_EX)
setownex_tid_test, setownex_pid_test, setownex_pgrp_test
#endif
};
char *TCID = "fcntl31";
int TST_TOTAL = ARRAY_SIZE(testfunc);
int main(int ac, char **av)
{
int lc, i;
tst_parse_opts(ac, av, NULL, NULL);
setup();
for (lc = 0; TEST_LOOPING(lc); lc++) {
tst_count = 0;
for (i = 0; i < TST_TOTAL; i++)
(*testfunc[i])();
}
cleanup();
tst_exit();
}
static void setup(void)
{
int ret;
tst_sig(FORK, DEF_HANDLER, cleanup);
TEST_PAUSE;
/* we have these tests on pipe */
SAFE_PIPE(cleanup, pipe_fds);
test_fd = pipe_fds[0];
if (fcntl(test_fd, F_SETFL, O_ASYNC) < 0)
tst_brkm(TBROK | TERRNO, cleanup, "fcntl set O_ASYNC failed");
pid = getpid();
ret = setpgrp();
if (ret < 0)
tst_brkm(TBROK | TERRNO, cleanup, "setpgrp() failed");
pgrp_pid = getpgid(0);
if (pgrp_pid < 0)
tst_brkm(TBROK | TERRNO, cleanup, "getpgid() failed");
#if defined(HAVE_STRUCT_F_OWNER_EX)
if ((tst_kvercmp(2, 6, 32)) >= 0) {
ownex_enabled = 1;
/* get original f_owner_ex info */
TEST(fcntl(test_fd, F_GETOWN_EX, &orig_own_ex));
if (TEST_RETURN < 0) {
tst_brkm(TFAIL | TTERRNO, cleanup,
"fcntl get original f_owner_ex info failed");
}
}
#endif
/* get original pid info */
TEST(fcntl(test_fd, F_GETOWN));
if (TEST_RETURN < 0) {
tst_brkm(TFAIL | TTERRNO, cleanup,
"fcntl get original pid info failed");
}
orig_pid = TEST_RETURN;
sigemptyset(&newset);
sigaddset(&newset, SIGUSR1);
sigaddset(&newset, SIGIO);
if (sigprocmask(SIG_SETMASK, &newset, &oldset) < 0)
tst_brkm(TBROK | TERRNO, cleanup, "sigprocmask failed");
timeout.tv_sec = 5;
timeout.tv_nsec = 0;
}
static void setown_pid_test(void)
{
TEST(fcntl(test_fd, F_SETOWN, pid));
if (TEST_RETURN < 0) {
tst_brkm(TFAIL | TTERRNO, cleanup,
"fcntl(F_SETOWN) set process id failed");
}
test_set_and_get_sig(SIGUSR1, "F_GETOWN, F_SETOWN for process ID");
TEST(fcntl(test_fd, F_SETOWN, orig_pid));
if (TEST_RETURN < 0) {
tst_brkm(TFAIL | TTERRNO, cleanup,
"fcntl(F_SETOWN) restore orig_pid failed");
}
}
static void setown_pgrp_test(void)
{
TEST(fcntl(test_fd, F_SETOWN, -pgrp_pid));
if (TEST_RETURN < 0) {
tst_brkm(TFAIL | TTERRNO, cleanup,
"fcntl(F_SETOWN) set process group id failed");
}
test_set_and_get_sig(SIGUSR1,
"F_GETOWN, F_SETOWN for process group ID");
TEST(fcntl(test_fd, F_SETOWN, orig_pid));
if (TEST_RETURN < 0) {
tst_brkm(TFAIL | TTERRNO, cleanup,
"fcntl(F_SETOWN) restore orig_pid failed");
}
}
#if defined(HAVE_STRUCT_F_OWNER_EX)
static void setownex_cleanup(void)
{
TEST(fcntl(test_fd, F_SETOWN_EX, &orig_own_ex));
if (TEST_RETURN < 0) {
tst_brkm(TFAIL | TTERRNO, cleanup,
"fcntl F_SETOWN_EX restore orig_own_ex failed");
}
}
static void setownex_tid_test(void)
{
static struct f_owner_ex tst_own_ex;
if (ownex_enabled == 0) {
tst_resm(TCONF, "%s", ownex_tconf_msg);
return;
}
tst_own_ex.type = F_OWNER_TID;
tst_own_ex.pid = ltp_syscall(__NR_gettid);
TEST(fcntl(test_fd, F_SETOWN_EX, &tst_own_ex));
if (TEST_RETURN < 0) {
tst_brkm(TFAIL | TTERRNO, cleanup,
"fcntl F_SETOWN_EX failed");
}
test_set_and_get_sig(SIGUSR1, "F_GETOWN_EX, F_SETOWN_EX for thread ID");
setownex_cleanup();
}
static void setownex_pid_test(void)
{
static struct f_owner_ex tst_own_ex;
if (ownex_enabled == 0) {
tst_resm(TCONF, "%s", ownex_tconf_msg);
return;
}
tst_own_ex.type = F_OWNER_PID;
tst_own_ex.pid = pid;
TEST(fcntl(test_fd, F_SETOWN_EX, &tst_own_ex));
if (TEST_RETURN < 0) {
tst_brkm(TFAIL | TTERRNO, cleanup,
"fcntl F_SETOWN_EX failed");
}
test_set_and_get_sig(SIGUSR1,
"F_GETOWN_EX, F_SETOWN_EX for process ID");
setownex_cleanup();
}
static void setownex_pgrp_test(void)
{
static struct f_owner_ex tst_own_ex;
if (ownex_enabled == 0) {
tst_resm(TCONF, "%s", ownex_tconf_msg);
return;
}
tst_own_ex.type = F_OWNER_PGRP;
tst_own_ex.pid = pgrp_pid;
TEST(fcntl(test_fd, F_SETOWN_EX, &tst_own_ex));
if (TEST_RETURN < 0) {
tst_brkm(TFAIL | TTERRNO, cleanup,
"fcntl F_SETOWN_EX failed");
}
test_set_and_get_sig(SIGUSR1,
"F_GETOWN_EX, F_SETOWN_EX for process group ID");
setownex_cleanup();
}
#endif
static void test_set_and_get_sig(int sig, char *des)
{
int orig_sig;
TEST(fcntl(test_fd, F_GETSIG));
if (TEST_RETURN < 0) {
tst_brkm(TFAIL | TTERRNO, cleanup,
"fcntl(fd, F_GETSIG) get orig_sig failed");
}
orig_sig = TEST_RETURN;
if (orig_sig == 0 || orig_sig == SIGIO)
tst_resm(TINFO, "default io events signal is SIGIO");
TEST(fcntl(test_fd, F_SETSIG, sig));
if (TEST_RETURN < 0) {
tst_brkm(TFAIL | TTERRNO, cleanup,
"fcntl(fd, F_SETSIG, SIG: %d) failed", sig);
}
TEST(fcntl(test_fd, F_GETSIG));
if (TEST_RETURN < 0) {
tst_brkm(TFAIL | TTERRNO, cleanup,
"fcntl(fd, F_GETSIG) get the set signal failed");
}
if (TEST_RETURN != sig) {
tst_brkm(TFAIL | TTERRNO, cleanup,
"fcntl F_SETSIG set SIG: %d failed", sig);
}
check_io_signal(des);
/* restore the default signal*/
TEST(fcntl(test_fd, F_SETSIG, orig_sig));
if (TEST_RETURN < 0) {
tst_brkm(TFAIL | TTERRNO, cleanup,
"fcntl restore default signal failed");
}
}
static void signal_parent(void)
{
int ret, fd;
fd = pipe_fds[1];
close(pipe_fds[0]);
ret = setpgrp();
if (ret < 0) {
fprintf(stderr, "child process(%d) setpgrp() failed: %s \n",
getpid(), strerror(errno));
}
/* Wait for parent process to enter sigtimedwait(). */
tst_process_state_wait2(getppid(), 'S');
ret = write(fd, "c", 1);
switch (ret) {
case 0:
fprintf(stderr, "No data written, something is wrong\n");
break;
case -1:
fprintf(stderr, "Failed to write to pipe: %s\n",
strerror(errno));
break;
}
close(fd);
return;
}
static void check_io_signal(char *des)
{
int ret;
char c;
pid_t child;
child = tst_fork();
if (child < 0)
tst_brkm(TBROK | TERRNO, cleanup, "fork failed");
if (child == 0) {
signal_parent();
exit(0);
} else {
ret = sigtimedwait(&newset, NULL, &timeout);
if (ret == -1) {
tst_brkm(TBROK | TERRNO, NULL,
"sigtimedwait() failed.");
}
switch (ret) {
case SIGUSR1:
tst_resm(TPASS, "fcntl test %s success", des);
break;
case SIGIO:
tst_resm(TFAIL, "received default SIGIO, fcntl test "
"%s failed", des);
break;
default:
tst_brkm(TBROK, cleanup, "fcntl io events "
"signal mechanism work abnormally");
}
SAFE_READ(cleanup, 1, test_fd, &c, 1);
wait(NULL);
}
}
static void cleanup(void)
{
if (sigprocmask(SIG_SETMASK, &oldset, NULL) < 0)
tst_resm(TWARN | TERRNO, "sigprocmask restore oldset failed");
if (pipe_fds[0] > 0 && close(pipe_fds[0]) == -1)
tst_resm(TWARN | TERRNO, "close(%d) failed", pipe_fds[0]);
if (pipe_fds[1] > 0 && close(pipe_fds[1]) == -1)
tst_resm(TWARN | TERRNO, "close(%d) failed", pipe_fds[1]);
}