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