xref: /freebsd-src/contrib/netbsd-tests/lib/libc/sys/t_clock_gettime.c (revision 1a36faad54665288ed4eb839d2a4699ae2ead45e)
163d1fd59SEnji Cooper /* $NetBSD: t_clock_gettime.c,v 1.3 2017/01/13 21:30:41 christos Exp $ */
257718be8SEnji Cooper 
357718be8SEnji Cooper /*-
457718be8SEnji Cooper  * Copyright (c) 2008 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 Frank Kardel.
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 
3257718be8SEnji Cooper /*-
3357718be8SEnji Cooper  * Copyright (c) 2006 Frank Kardel
3457718be8SEnji Cooper  * All rights reserved.
3557718be8SEnji Cooper  *
3657718be8SEnji Cooper  * Redistribution and use in source and binary forms, with or without
3757718be8SEnji Cooper  * modification, are permitted provided that the following conditions
3857718be8SEnji Cooper  * are met:
3957718be8SEnji Cooper  * 1. Redistributions of source code must retain the above copyright
4057718be8SEnji Cooper  *    notice, this list of conditions and the following disclaimer.
4157718be8SEnji Cooper  * 2. Redistributions in binary form must reproduce the above copyright
4257718be8SEnji Cooper  *    notice, this list of conditions and the following disclaimer in the
4357718be8SEnji Cooper  *    documentation and/or other materials provided with the distribution.
4457718be8SEnji Cooper  *
4557718be8SEnji Cooper  * THIS SOFTWARE IS PROVIDED BY THE NETBSD FOUNDATION, INC. AND CONTRIBUTORS
4657718be8SEnji Cooper  * ``AS IS'' AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED
4757718be8SEnji Cooper  * TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR
4857718be8SEnji Cooper  * PURPOSE ARE DISCLAIMED.  IN NO EVENT SHALL THE FOUNDATION OR CONTRIBUTORS
4957718be8SEnji Cooper  * BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
5057718be8SEnji Cooper  * CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
5157718be8SEnji Cooper  * SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
5257718be8SEnji Cooper  * INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
5357718be8SEnji Cooper  * CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
5457718be8SEnji Cooper  * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
5557718be8SEnji Cooper  * POSSIBILITY OF SUCH DAMAGE.
5657718be8SEnji Cooper  */
5757718be8SEnji Cooper 
5857718be8SEnji Cooper #include <sys/cdefs.h>
5957718be8SEnji Cooper __COPYRIGHT("@(#) Copyright (c) 2008\
6057718be8SEnji Cooper  The NetBSD Foundation, inc. All rights reserved.");
6163d1fd59SEnji Cooper __RCSID("$NetBSD: t_clock_gettime.c,v 1.3 2017/01/13 21:30:41 christos Exp $");
6257718be8SEnji Cooper 
6357718be8SEnji Cooper #include <sys/param.h>
6457718be8SEnji Cooper #include <sys/sysctl.h>
6557718be8SEnji Cooper 
66*04f40d86SEnji Cooper 
6757718be8SEnji Cooper #include <atf-c.h>
6857718be8SEnji Cooper #include <errno.h>
6963d1fd59SEnji Cooper #include <limits.h>
7057718be8SEnji Cooper #include <stdio.h>
7163d1fd59SEnji Cooper #include <stdint.h>
7257718be8SEnji Cooper #include <stdlib.h>
7357718be8SEnji Cooper #include <string.h>
7457718be8SEnji Cooper #include <time.h>
7557718be8SEnji Cooper #include <unistd.h>
7657718be8SEnji Cooper 
77bc3c590bSEnji Cooper #include "h_macros.h"
7857718be8SEnji Cooper 
7957718be8SEnji Cooper #define MINPOSDIFF	15000000	/* 15 ms for now */
8057718be8SEnji Cooper #define TIMEOUT		5
8157718be8SEnji Cooper 
8257718be8SEnji Cooper #define TC_HARDWARE	"kern.timecounter.hardware"
8357718be8SEnji Cooper #define TC_CHOICE	"kern.timecounter.choice"
8457718be8SEnji Cooper 
8557718be8SEnji Cooper static void
check_timecounter(void)8657718be8SEnji Cooper check_timecounter(void)
8757718be8SEnji Cooper {
8857718be8SEnji Cooper 	struct timespec tsa, tsb, tsl, res;
8957718be8SEnji Cooper 	long long mindiff = INTMAX_MAX;
9057718be8SEnji Cooper 	time_t endlimit;
9157718be8SEnji Cooper 
9257718be8SEnji Cooper #define CL(x) \
9357718be8SEnji Cooper 	do { \
9457718be8SEnji Cooper 		if ((x) != -1) \
9557718be8SEnji Cooper 			break; \
9657718be8SEnji Cooper 		atf_tc_fail_nonfatal("%s: %s", #x, strerror(errno)); \
9757718be8SEnji Cooper 		return; \
9857718be8SEnji Cooper 	} while (0)
9957718be8SEnji Cooper 
10057718be8SEnji Cooper 	CL(clock_gettime(CLOCK_REALTIME, &tsa));
10157718be8SEnji Cooper 	tsl = tsa;
10257718be8SEnji Cooper 
10357718be8SEnji Cooper 	CL(time(&endlimit));
10457718be8SEnji Cooper 	endlimit += TIMEOUT + 1;
10557718be8SEnji Cooper 
10657718be8SEnji Cooper 	while ((time_t)tsa.tv_sec < endlimit) {
10757718be8SEnji Cooper 		long long diff;
10857718be8SEnji Cooper 
10957718be8SEnji Cooper 		CL(clock_gettime(CLOCK_REALTIME, &tsb));
11057718be8SEnji Cooper 		diff = 1000000000LL * (tsb.tv_sec - tsa.tv_sec)
11157718be8SEnji Cooper 		    + tsb.tv_nsec - tsa.tv_nsec;
11257718be8SEnji Cooper 
11357718be8SEnji Cooper 		if (diff > 0 && mindiff > diff)
11457718be8SEnji Cooper 			mindiff = diff;
11557718be8SEnji Cooper 
11657718be8SEnji Cooper 		if (diff < 0 || diff > MINPOSDIFF) {
11757718be8SEnji Cooper 			long long elapsed;
11857718be8SEnji Cooper 			(void)printf("%stime TSA: 0x%jx.%08jx, TSB: 0x%jx.%08jx, "
11957718be8SEnji Cooper 			    "diff = %lld nsec, ", (diff < 0) ? "BAD " : "",
12057718be8SEnji Cooper 			    (uintmax_t)tsa.tv_sec, (uintmax_t)tsa.tv_nsec,
12157718be8SEnji Cooper 			    (uintmax_t)tsb.tv_sec, (uintmax_t)tsb.tv_nsec, diff);
12257718be8SEnji Cooper 
12357718be8SEnji Cooper 			elapsed = 1000000000LL * (tsb.tv_sec - tsl.tv_sec)
12457718be8SEnji Cooper 			    + tsb.tv_nsec - tsl.tv_nsec;
12557718be8SEnji Cooper 
12657718be8SEnji Cooper 
12757718be8SEnji Cooper 			(void)printf("%lld nsec\n", elapsed);
12857718be8SEnji Cooper 			tsl = tsb;
12957718be8SEnji Cooper 
13057718be8SEnji Cooper 			ATF_CHECK(diff >= 0);
13157718be8SEnji Cooper 			if (diff < 0)
13257718be8SEnji Cooper 				return;
13357718be8SEnji Cooper 		}
13457718be8SEnji Cooper 
13557718be8SEnji Cooper 		tsa.tv_sec = tsb.tv_sec;
13657718be8SEnji Cooper 		tsa.tv_nsec = tsb.tv_nsec;
13757718be8SEnji Cooper 	}
13857718be8SEnji Cooper 
13957718be8SEnji Cooper 	if (clock_getres(CLOCK_REALTIME, &res) == 0) {
14057718be8SEnji Cooper 		long long r = res.tv_sec * 1000000000 + res.tv_nsec;
14157718be8SEnji Cooper 
14257718be8SEnji Cooper 		(void)printf("Claimed resolution: %lld nsec (%f Hz) or "
14357718be8SEnji Cooper 		    "better\n", r, 1.0 / r * 1e9);
14457718be8SEnji Cooper 		(void)printf("Observed minimum non zero delta: %lld "
14557718be8SEnji Cooper 		    "nsec\n", mindiff);
14657718be8SEnji Cooper 	}
14757718be8SEnji Cooper 
14857718be8SEnji Cooper #undef CL
14957718be8SEnji Cooper }
15057718be8SEnji Cooper 
15157718be8SEnji Cooper ATF_TC(clock_gettime_real);
ATF_TC_HEAD(clock_gettime_real,tc)15257718be8SEnji Cooper ATF_TC_HEAD(clock_gettime_real, tc)
15357718be8SEnji Cooper {
15457718be8SEnji Cooper 	atf_tc_set_md_var(tc, "require.user", "root");
15557718be8SEnji Cooper 	atf_tc_set_md_var(tc, "descr",
15657718be8SEnji Cooper 	    "Checks the monotonicity of the CLOCK_REALTIME implementation");
15757718be8SEnji Cooper 	atf_tc_set_md_var(tc, "timeout", "300");
15857718be8SEnji Cooper }
15957718be8SEnji Cooper 
ATF_TC_BODY(clock_gettime_real,tc)16057718be8SEnji Cooper ATF_TC_BODY(clock_gettime_real, tc)
16157718be8SEnji Cooper {
16257718be8SEnji Cooper 	char name[128], cbuf[512], ctrbuf[10240];
16357718be8SEnji Cooper 	size_t cbufsiz = sizeof(cbuf);
16457718be8SEnji Cooper 	size_t ctrbufsiz = sizeof(ctrbuf);
16557718be8SEnji Cooper 	const char *p;
16657718be8SEnji Cooper 	char *save;
16757718be8SEnji Cooper 	int quality, n;
16857718be8SEnji Cooper 
16957718be8SEnji Cooper 	if (sysctlbyname(TC_HARDWARE, cbuf, &cbufsiz, NULL, 0) != 0) {
17057718be8SEnji Cooper 		(void)printf("\nChecking legacy time implementation "
17157718be8SEnji Cooper 		    "for %d seconds\n", TIMEOUT);
17257718be8SEnji Cooper 		check_timecounter();
17357718be8SEnji Cooper 		return;
17457718be8SEnji Cooper 		/* NOTREACHED */
17557718be8SEnji Cooper 	}
17657718be8SEnji Cooper 	(void)printf("%s = %s\n", TC_HARDWARE, cbuf);
17757718be8SEnji Cooper 	REQUIRE_LIBC(save = strdup(cbuf), NULL);
17857718be8SEnji Cooper 
17957718be8SEnji Cooper 	RL(sysctlbyname(TC_CHOICE, ctrbuf, &ctrbufsiz, NULL, 0));
18057718be8SEnji Cooper 	(void)printf("%s = %s\n", TC_CHOICE, ctrbuf);
18157718be8SEnji Cooper 
18257718be8SEnji Cooper 	for (p = ctrbuf, n = 0; sscanf(p, "%127[^(](q=%d, f=%*u Hz)%*[ ]%n",
18357718be8SEnji Cooper 	    name, &quality, &n) == 2; p += n) {
18457718be8SEnji Cooper 		struct timespec ts;
18557718be8SEnji Cooper 		int ret;
18657718be8SEnji Cooper 
18757718be8SEnji Cooper 		if (quality < 0)
18857718be8SEnji Cooper 			continue;
18957718be8SEnji Cooper 
19057718be8SEnji Cooper 		(void)printf("\nChecking %s for %d seconds\n", name, TIMEOUT);
19157718be8SEnji Cooper 		CHECK_LIBC(ret = sysctlbyname(TC_HARDWARE, NULL, 0,
19257718be8SEnji Cooper 		    name, strlen(name)), -1);
19357718be8SEnji Cooper 		if (ret == -1)
19457718be8SEnji Cooper 			continue;
19557718be8SEnji Cooper 
19657718be8SEnji Cooper 		/* wait a bit to select new counter in clockinterrupt */
19757718be8SEnji Cooper 		ts.tv_sec = 0;
19857718be8SEnji Cooper 		ts.tv_nsec = 100000000;
19957718be8SEnji Cooper 		(void)nanosleep(&ts, NULL);
20057718be8SEnji Cooper 
20157718be8SEnji Cooper 		check_timecounter();
20257718be8SEnji Cooper 	}
20357718be8SEnji Cooper 
20457718be8SEnji Cooper 	RL(sysctlbyname(TC_HARDWARE, NULL, 0, save, strlen(save)));
20557718be8SEnji Cooper }
20657718be8SEnji Cooper 
ATF_TP_ADD_TCS(tp)20757718be8SEnji Cooper ATF_TP_ADD_TCS(tp)
20857718be8SEnji Cooper {
20957718be8SEnji Cooper 
21057718be8SEnji Cooper 	ATF_TP_ADD_TC(tp, clock_gettime_real);
21157718be8SEnji Cooper 
21257718be8SEnji Cooper 	return atf_no_error();
21357718be8SEnji Cooper }
214