163d1fd59SEnji Cooper /* $NetBSD: t_nanosleep.c,v 1.4 2017/01/13 21:15:14 christos Exp $ */
257718be8SEnji Cooper
357718be8SEnji Cooper /*-
457718be8SEnji Cooper * Copyright (c) 2011 The NetBSD Foundation, Inc.
557718be8SEnji Cooper * All rights reserved.
657718be8SEnji Cooper *
757718be8SEnji Cooper * This code is derived from software contributed to The NetBSD Foundation
857718be8SEnji Cooper * by Jukka Ruohonen.
957718be8SEnji Cooper *
1057718be8SEnji Cooper * Redistribution and use in source and binary forms, with or without
1157718be8SEnji Cooper * modification, are permitted provided that the following conditions
1257718be8SEnji Cooper * are met:
1357718be8SEnji Cooper * 1. Redistributions of source code must retain the above copyright
1457718be8SEnji Cooper * notice, this list of conditions and the following disclaimer.
1557718be8SEnji Cooper * 2. Redistributions in binary form must reproduce the above copyright
1657718be8SEnji Cooper * notice, this list of conditions and the following disclaimer in the
1757718be8SEnji Cooper * documentation and/or other materials provided with the distribution.
1857718be8SEnji Cooper *
1957718be8SEnji Cooper * THIS SOFTWARE IS PROVIDED BY THE NETBSD FOUNDATION, INC. AND CONTRIBUTORS
2057718be8SEnji Cooper * ``AS IS'' AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED
2157718be8SEnji Cooper * TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR
2257718be8SEnji Cooper * PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE FOUNDATION OR CONTRIBUTORS
2357718be8SEnji Cooper * BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
2457718be8SEnji Cooper * CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
2557718be8SEnji Cooper * SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
2657718be8SEnji Cooper * INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
2757718be8SEnji Cooper * CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
2857718be8SEnji Cooper * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
2957718be8SEnji Cooper * POSSIBILITY OF SUCH DAMAGE.
3057718be8SEnji Cooper */
3157718be8SEnji Cooper #include <sys/cdefs.h>
3263d1fd59SEnji Cooper __RCSID("$NetBSD: t_nanosleep.c,v 1.4 2017/01/13 21:15:14 christos Exp $");
3357718be8SEnji Cooper
3457718be8SEnji Cooper #include <sys/time.h>
3557718be8SEnji Cooper #include <sys/wait.h>
3657718be8SEnji Cooper
3757718be8SEnji Cooper #include <atf-c.h>
3857718be8SEnji Cooper #include <errno.h>
3957718be8SEnji Cooper #include <time.h>
4057718be8SEnji Cooper #include <signal.h>
4157718be8SEnji Cooper #include <stdio.h>
4257718be8SEnji Cooper #include <stdlib.h>
4357718be8SEnji Cooper #include <string.h>
4457718be8SEnji Cooper #include <sysexits.h>
4557718be8SEnji Cooper #include <unistd.h>
4657718be8SEnji Cooper
4757718be8SEnji Cooper static void
handler(int signo __unused)48eb780be7SEnji Cooper handler(int signo __unused)
4957718be8SEnji Cooper {
5057718be8SEnji Cooper /* Nothing. */
5157718be8SEnji Cooper }
5257718be8SEnji Cooper
53*70c144dcSBryan Drewery static int got_info;
54*70c144dcSBryan Drewery static void
info_handler(int signo __unused)55*70c144dcSBryan Drewery info_handler(int signo __unused)
56*70c144dcSBryan Drewery {
57*70c144dcSBryan Drewery
58*70c144dcSBryan Drewery got_info = 1;
59*70c144dcSBryan Drewery }
60*70c144dcSBryan Drewery
61*70c144dcSBryan Drewery
6257718be8SEnji Cooper ATF_TC(nanosleep_basic);
ATF_TC_HEAD(nanosleep_basic,tc)6357718be8SEnji Cooper ATF_TC_HEAD(nanosleep_basic, tc)
6457718be8SEnji Cooper {
6557718be8SEnji Cooper atf_tc_set_md_var(tc, "descr", "Test that nanosleep(2) works");
6657718be8SEnji Cooper }
6757718be8SEnji Cooper
ATF_TC_BODY(nanosleep_basic,tc)6857718be8SEnji Cooper ATF_TC_BODY(nanosleep_basic, tc)
6957718be8SEnji Cooper {
7057718be8SEnji Cooper static const size_t maxiter = 10;
7157718be8SEnji Cooper struct timespec ts1, ts2, tsn;
7257718be8SEnji Cooper size_t i;
7357718be8SEnji Cooper
7457718be8SEnji Cooper for (i = 1; i < maxiter; i++) {
7557718be8SEnji Cooper
7657718be8SEnji Cooper tsn.tv_sec = 0;
7757718be8SEnji Cooper tsn.tv_nsec = i;
7857718be8SEnji Cooper
7957718be8SEnji Cooper (void)memset(&ts1, 0, sizeof(struct timespec));
8057718be8SEnji Cooper (void)memset(&ts2, 0, sizeof(struct timespec));
8157718be8SEnji Cooper
8257718be8SEnji Cooper ATF_REQUIRE(clock_gettime(CLOCK_MONOTONIC, &ts1) == 0);
8357718be8SEnji Cooper ATF_REQUIRE(nanosleep(&tsn, NULL) == 0);
8457718be8SEnji Cooper ATF_REQUIRE(clock_gettime(CLOCK_MONOTONIC, &ts2) == 0);
8557718be8SEnji Cooper
8657718be8SEnji Cooper /*
8757718be8SEnji Cooper * Verify that we slept at least one nanosecond.
8857718be8SEnji Cooper */
8957718be8SEnji Cooper if (timespeccmp(&ts2, &ts1, <=) != 0) {
9057718be8SEnji Cooper
9157718be8SEnji Cooper (void)fprintf(stderr,
9257718be8SEnji Cooper "sleep time:: sec %llu, nsec %lu\n\t\t"
9357718be8SEnji Cooper "ts1: sec %llu, nsec %lu\n\t\t"
9457718be8SEnji Cooper "ts2: sec %llu, nsec %lu\n",
9557718be8SEnji Cooper (unsigned long long)tsn.tv_sec, tsn.tv_nsec,
9657718be8SEnji Cooper (unsigned long long)ts1.tv_sec, ts1.tv_nsec,
9757718be8SEnji Cooper (unsigned long long)ts2.tv_sec, ts2.tv_nsec);
9857718be8SEnji Cooper
9957718be8SEnji Cooper atf_tc_fail_nonfatal("inaccuracies in sleep time "
10057718be8SEnji Cooper "(resolution = %lu nsec)", tsn.tv_nsec);
10157718be8SEnji Cooper }
10257718be8SEnji Cooper }
10357718be8SEnji Cooper }
10457718be8SEnji Cooper
10557718be8SEnji Cooper ATF_TC(nanosleep_err);
ATF_TC_HEAD(nanosleep_err,tc)10657718be8SEnji Cooper ATF_TC_HEAD(nanosleep_err, tc)
10757718be8SEnji Cooper {
10857718be8SEnji Cooper atf_tc_set_md_var(tc, "descr",
10957718be8SEnji Cooper "Test errors from nanosleep(2) (PR bin/14558)");
11057718be8SEnji Cooper }
11157718be8SEnji Cooper
ATF_TC_BODY(nanosleep_err,tc)11257718be8SEnji Cooper ATF_TC_BODY(nanosleep_err, tc)
11357718be8SEnji Cooper {
11457718be8SEnji Cooper struct timespec ts;
11557718be8SEnji Cooper
11657718be8SEnji Cooper ts.tv_sec = 1;
11757718be8SEnji Cooper ts.tv_nsec = -1;
11857718be8SEnji Cooper errno = 0;
11957718be8SEnji Cooper ATF_REQUIRE_ERRNO(EINVAL, nanosleep(&ts, NULL) == -1);
12057718be8SEnji Cooper
12157718be8SEnji Cooper ts.tv_sec = 1;
12257718be8SEnji Cooper ts.tv_nsec = 1000000000;
12357718be8SEnji Cooper errno = 0;
12457718be8SEnji Cooper ATF_REQUIRE_ERRNO(EINVAL, nanosleep(&ts, NULL) == -1);
12557718be8SEnji Cooper
12657718be8SEnji Cooper ts.tv_sec = -1;
12757718be8SEnji Cooper ts.tv_nsec = 0;
12857718be8SEnji Cooper errno = 0;
12957718be8SEnji Cooper ATF_REQUIRE_ERRNO(0, nanosleep(&ts, NULL) == 0);
13057718be8SEnji Cooper
13157718be8SEnji Cooper errno = 0;
13257718be8SEnji Cooper ATF_REQUIRE_ERRNO(EFAULT, nanosleep((void *)-1, NULL) == -1);
13357718be8SEnji Cooper }
13457718be8SEnji Cooper
13557718be8SEnji Cooper ATF_TC(nanosleep_sig);
ATF_TC_HEAD(nanosleep_sig,tc)13657718be8SEnji Cooper ATF_TC_HEAD(nanosleep_sig, tc)
13757718be8SEnji Cooper {
13857718be8SEnji Cooper atf_tc_set_md_var(tc, "descr", "Test signal for nanosleep(2)");
13957718be8SEnji Cooper }
14057718be8SEnji Cooper
ATF_TC_BODY(nanosleep_sig,tc)14157718be8SEnji Cooper ATF_TC_BODY(nanosleep_sig, tc)
14257718be8SEnji Cooper {
14357718be8SEnji Cooper struct timespec tsn, tsr;
14457718be8SEnji Cooper pid_t pid;
14557718be8SEnji Cooper int sta;
14657718be8SEnji Cooper
14757718be8SEnji Cooper /*
14857718be8SEnji Cooper * Test that a signal interrupts nanosleep(2).
14957718be8SEnji Cooper *
15057718be8SEnji Cooper * (In which case the return value should be -1 and the
15157718be8SEnji Cooper * second parameter should contain the unslept time.)
15257718be8SEnji Cooper */
15357718be8SEnji Cooper pid = fork();
15457718be8SEnji Cooper
15557718be8SEnji Cooper ATF_REQUIRE(pid >= 0);
15657718be8SEnji Cooper ATF_REQUIRE(signal(SIGINT, handler) == 0);
15757718be8SEnji Cooper
15857718be8SEnji Cooper if (pid == 0) {
15957718be8SEnji Cooper
16057718be8SEnji Cooper tsn.tv_sec = 10;
16157718be8SEnji Cooper tsn.tv_nsec = 0;
16257718be8SEnji Cooper
16357718be8SEnji Cooper tsr.tv_sec = 0;
16457718be8SEnji Cooper tsr.tv_nsec = 0;
16557718be8SEnji Cooper
16657718be8SEnji Cooper errno = 0;
16757718be8SEnji Cooper
16857718be8SEnji Cooper if (nanosleep(&tsn, &tsr) != -1)
16957718be8SEnji Cooper _exit(EXIT_FAILURE);
17057718be8SEnji Cooper
17157718be8SEnji Cooper if (errno != EINTR)
17257718be8SEnji Cooper _exit(EXIT_FAILURE);
17357718be8SEnji Cooper
17457718be8SEnji Cooper if (tsr.tv_sec == 0 && tsr.tv_nsec == 0)
17557718be8SEnji Cooper _exit(EXIT_FAILURE);
17657718be8SEnji Cooper
17757718be8SEnji Cooper _exit(EXIT_SUCCESS);
17857718be8SEnji Cooper }
17957718be8SEnji Cooper
18057718be8SEnji Cooper (void)sleep(1);
18157718be8SEnji Cooper (void)kill(pid, SIGINT);
18257718be8SEnji Cooper (void)wait(&sta);
18357718be8SEnji Cooper
18457718be8SEnji Cooper if (WIFEXITED(sta) == 0 || WEXITSTATUS(sta) != EXIT_SUCCESS)
18557718be8SEnji Cooper atf_tc_fail("signal did not interrupt nanosleep(2)");
18657718be8SEnji Cooper }
18757718be8SEnji Cooper
188*70c144dcSBryan Drewery ATF_TC(nanosleep_eintr);
ATF_TC_HEAD(nanosleep_eintr,tc)189*70c144dcSBryan Drewery ATF_TC_HEAD(nanosleep_eintr, tc)
190*70c144dcSBryan Drewery {
191*70c144dcSBryan Drewery atf_tc_set_md_var(tc, "descr", "Test [EINTR] for nanosleep(2)");
192*70c144dcSBryan Drewery atf_tc_set_md_var(tc, "timeout", "7");
193*70c144dcSBryan Drewery }
194*70c144dcSBryan Drewery
ATF_TC_BODY(nanosleep_eintr,tc)195*70c144dcSBryan Drewery ATF_TC_BODY(nanosleep_eintr, tc)
196*70c144dcSBryan Drewery {
197*70c144dcSBryan Drewery struct sigaction act;
198*70c144dcSBryan Drewery struct timespec tso, ts;
199*70c144dcSBryan Drewery pid_t pid;
200*70c144dcSBryan Drewery int sta;
201*70c144dcSBryan Drewery
202*70c144dcSBryan Drewery /*
203*70c144dcSBryan Drewery * Test that [EINTR] properly handles rmtp for nanosleep(2).
204*70c144dcSBryan Drewery */
205*70c144dcSBryan Drewery pid = fork();
206*70c144dcSBryan Drewery
207*70c144dcSBryan Drewery ATF_REQUIRE(pid >= 0);
208*70c144dcSBryan Drewery
209*70c144dcSBryan Drewery got_info = 0;
210*70c144dcSBryan Drewery
211*70c144dcSBryan Drewery if (pid == 0) {
212*70c144dcSBryan Drewery act.sa_handler = info_handler;
213*70c144dcSBryan Drewery sigemptyset(&act.sa_mask);
214*70c144dcSBryan Drewery act.sa_flags = 0; /* Don't allow restart. */
215*70c144dcSBryan Drewery ATF_REQUIRE(sigaction(SIGINFO, &act, NULL) == 0);
216*70c144dcSBryan Drewery
217*70c144dcSBryan Drewery tso.tv_sec = 5;
218*70c144dcSBryan Drewery tso.tv_nsec = 0;
219*70c144dcSBryan Drewery
220*70c144dcSBryan Drewery ts.tv_sec = tso.tv_sec;
221*70c144dcSBryan Drewery ts.tv_nsec = tso.tv_nsec;
222*70c144dcSBryan Drewery
223*70c144dcSBryan Drewery errno = 0;
224*70c144dcSBryan Drewery while (nanosleep(&ts, &ts) != 0) {
225*70c144dcSBryan Drewery ATF_REQUIRE_MSG(timespeccmp(&ts, &tso, <=),
226*70c144dcSBryan Drewery "errno=%d ts=%0.9f should be <= last tso=%0.9f\n",
227*70c144dcSBryan Drewery errno,
228*70c144dcSBryan Drewery ts.tv_sec + ts.tv_nsec / 1e9,
229*70c144dcSBryan Drewery tso.tv_sec + tso.tv_nsec / 1e9);
230*70c144dcSBryan Drewery if (errno == EINTR && got_info == 1) {
231*70c144dcSBryan Drewery got_info = 0;
232*70c144dcSBryan Drewery errno = 0;
233*70c144dcSBryan Drewery tso.tv_sec = ts.tv_sec;
234*70c144dcSBryan Drewery tso.tv_nsec = ts.tv_nsec;
235*70c144dcSBryan Drewery continue;
236*70c144dcSBryan Drewery }
237*70c144dcSBryan Drewery _exit(EXIT_FAILURE);
238*70c144dcSBryan Drewery }
239*70c144dcSBryan Drewery
240*70c144dcSBryan Drewery if (errno != 0)
241*70c144dcSBryan Drewery _exit(EXIT_FAILURE);
242*70c144dcSBryan Drewery
243*70c144dcSBryan Drewery _exit(EXIT_SUCCESS);
244*70c144dcSBryan Drewery }
245*70c144dcSBryan Drewery
246*70c144dcSBryan Drewery /* Flood the process with SIGINFO until it exits. */
247*70c144dcSBryan Drewery do {
248*70c144dcSBryan Drewery for (int i = 0; i < 10; i++)
249*70c144dcSBryan Drewery ATF_REQUIRE(kill(pid, SIGINFO) == 0);
250*70c144dcSBryan Drewery ATF_REQUIRE(usleep(10000) == 0);
251*70c144dcSBryan Drewery } while (waitpid(pid, &sta, WNOHANG) == 0);
252*70c144dcSBryan Drewery
253*70c144dcSBryan Drewery ATF_REQUIRE(WIFEXITED(sta) == 1);
254*70c144dcSBryan Drewery
255*70c144dcSBryan Drewery if (WEXITSTATUS(sta) != EXIT_SUCCESS)
256*70c144dcSBryan Drewery atf_tc_fail("nanosleep(2) handled rtmp incorrectly");
257*70c144dcSBryan Drewery }
258*70c144dcSBryan Drewery
ATF_TP_ADD_TCS(tp)25957718be8SEnji Cooper ATF_TP_ADD_TCS(tp)
26057718be8SEnji Cooper {
26157718be8SEnji Cooper
26257718be8SEnji Cooper ATF_TP_ADD_TC(tp, nanosleep_basic);
26357718be8SEnji Cooper ATF_TP_ADD_TC(tp, nanosleep_err);
26457718be8SEnji Cooper ATF_TP_ADD_TC(tp, nanosleep_sig);
265*70c144dcSBryan Drewery ATF_TP_ADD_TC(tp, nanosleep_eintr);
26657718be8SEnji Cooper
26757718be8SEnji Cooper return atf_no_error();
26857718be8SEnji Cooper }
269