xref: /freebsd-src/contrib/netbsd-tests/lib/libc/sys/t_nanosleep.c (revision 70c144dc782e3148c42698508974edba686d4bde)
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