/*
* Copyright (c) Crackerjack Project., 2007-2008 ,Hitachi, Ltd
* Author(s): Takahiro Yasui <takahiro.yasui.mp@hitachi.com>,
* Yumiko Sugita <yumiko.sugita.yf@hitachi.com>,
* Satoshi Fujiwara <sa-fuji@sdl.hitachi.co.jp>
* Copyright (c) 2016 Linux Test Project
*
* 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 would 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 the Free Software Foundation,
* Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA
*/
#include <limits.h>
#include "lapi/syscalls.h"
#include "tst_sig_proc.h"
#include "tst_timer.h"
#include "tst_test.h"
static void sighandler(int sig LTP_ATTRIBUTE_UNUSED)
{
}
enum test_type {
NORMAL,
SEND_SIGINT,
};
#define TYPE_NAME(x) .ttype = x, .desc = #x
struct test_case {
clockid_t clk_id; /* clock_* clock type parameter */
int ttype; /* test type (enum) */
const char *desc; /* test description (name) */
int flags; /* clock_nanosleep flags parameter */
struct timespec rq;
int exp_ret;
int exp_err;
};
/*
* test status of errors on man page
* EINTR v (function was interrupted by a signal)
* EINVAL v (invalid tv_nsec, etc.)
*/
static struct test_case tcase[] = {
{
TYPE_NAME(NORMAL),
.clk_id = CLOCK_REALTIME,
.flags = 0,
.rq = (struct timespec) {.tv_sec = 0, .tv_nsec = -1},
.exp_ret = EINVAL,
.exp_err = 0,
},
{
TYPE_NAME(NORMAL),
.clk_id = CLOCK_REALTIME,
.flags = 0,
.rq = (struct timespec) {.tv_sec = 0, .tv_nsec = 1000000000},
.exp_ret = EINVAL,
.exp_err = 0,
},
{
TYPE_NAME(NORMAL),
.clk_id = CLOCK_THREAD_CPUTIME_ID,
.flags = 0,
.rq = (struct timespec) {.tv_sec = 0, .tv_nsec = 500000000},
.exp_ret = EINVAL,
.exp_err = 0,
},
{
TYPE_NAME(SEND_SIGINT),
.clk_id = CLOCK_REALTIME,
.flags = 0,
.rq = (struct timespec) {.tv_sec = 10, .tv_nsec = 0},
.exp_ret = EINTR,
.exp_err = 0,
},
};
void setup(void)
{
SAFE_SIGNAL(SIGINT, sighandler);
}
static void do_test(unsigned int i)
{
struct test_case *tc = &tcase[i];
struct timespec rm = {0};
pid_t pid = 0;
tst_res(TINFO, "case %s", tc->desc);
if (tc->ttype == SEND_SIGINT)
pid = create_sig_proc(SIGINT, 40, 500000);
TEST(clock_nanosleep(tc->clk_id, tc->flags, &tc->rq, &rm));
if (pid) {
SAFE_KILL(pid, SIGTERM);
SAFE_WAIT(NULL);
}
if (tc->ttype == SEND_SIGINT) {
long long expect_ms = tst_timespec_to_ms(tc->rq);
long long remain_ms = tst_timespec_to_ms(rm);
tst_res(TINFO, "remain time: %lds %ldns", rm.tv_sec, rm.tv_nsec);
if (!rm.tv_sec && !rm.tv_nsec) {
tst_res(TFAIL | TTERRNO,
"The clock_nanosleep() haven't updated"
" timestamp with remaining time");
return;
}
if (remain_ms > expect_ms) {
tst_res(TFAIL| TTERRNO,
"remaining time > requested time (%lld > %lld)",
remain_ms, expect_ms);
return;
}
}
if (TEST_RETURN != tc->exp_ret) {
tst_res(TFAIL | TTERRNO, "returned %ld, expected %d,"
" expected errno: %s (%d)", TEST_RETURN,
tc->exp_ret, tst_strerrno(tc->exp_err), tc->exp_err);
return;
}
tst_res(TPASS, "returned %s (%ld)",
tst_strerrno(TEST_RETURN), TEST_RETURN);
}
static struct tst_test test = {
.tcnt = ARRAY_SIZE(tcase),
.test = do_test,
.setup = setup,
.forks_child = 1,
};