xref: /aosp_15_r20/external/linux-kselftest/tools/testing/selftests/timers/inconsistency-check.c (revision 053f45be4e351dfd5e965df293cd45b779f579ee)
1*053f45beSAndroid Build Coastguard Worker /* Time inconsistency check test
2*053f45beSAndroid Build Coastguard Worker  *		by: john stultz ([email protected])
3*053f45beSAndroid Build Coastguard Worker  *		(C) Copyright IBM 2003, 2004, 2005, 2012
4*053f45beSAndroid Build Coastguard Worker  *		(C) Copyright Linaro Limited 2015
5*053f45beSAndroid Build Coastguard Worker  *		Licensed under the GPLv2
6*053f45beSAndroid Build Coastguard Worker  *
7*053f45beSAndroid Build Coastguard Worker  *  To build:
8*053f45beSAndroid Build Coastguard Worker  *	$ gcc inconsistency-check.c -o inconsistency-check -lrt
9*053f45beSAndroid Build Coastguard Worker  *
10*053f45beSAndroid Build Coastguard Worker  *   This program is free software: you can redistribute it and/or modify
11*053f45beSAndroid Build Coastguard Worker  *   it under the terms of the GNU General Public License as published by
12*053f45beSAndroid Build Coastguard Worker  *   the Free Software Foundation, either version 2 of the License, or
13*053f45beSAndroid Build Coastguard Worker  *   (at your option) any later version.
14*053f45beSAndroid Build Coastguard Worker  *
15*053f45beSAndroid Build Coastguard Worker  *   This program is distributed in the hope that it will be useful,
16*053f45beSAndroid Build Coastguard Worker  *   but WITHOUT ANY WARRANTY; without even the implied warranty of
17*053f45beSAndroid Build Coastguard Worker  *   MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
18*053f45beSAndroid Build Coastguard Worker  *   GNU General Public License for more details.
19*053f45beSAndroid Build Coastguard Worker  */
20*053f45beSAndroid Build Coastguard Worker 
21*053f45beSAndroid Build Coastguard Worker 
22*053f45beSAndroid Build Coastguard Worker 
23*053f45beSAndroid Build Coastguard Worker #include <stdio.h>
24*053f45beSAndroid Build Coastguard Worker #include <unistd.h>
25*053f45beSAndroid Build Coastguard Worker #include <stdlib.h>
26*053f45beSAndroid Build Coastguard Worker #include <time.h>
27*053f45beSAndroid Build Coastguard Worker #include <sys/time.h>
28*053f45beSAndroid Build Coastguard Worker #include <sys/timex.h>
29*053f45beSAndroid Build Coastguard Worker #include <string.h>
30*053f45beSAndroid Build Coastguard Worker #include <signal.h>
31*053f45beSAndroid Build Coastguard Worker #include "../kselftest.h"
32*053f45beSAndroid Build Coastguard Worker 
33*053f45beSAndroid Build Coastguard Worker #define CALLS_PER_LOOP 64
34*053f45beSAndroid Build Coastguard Worker #define NSEC_PER_SEC 1000000000ULL
35*053f45beSAndroid Build Coastguard Worker 
36*053f45beSAndroid Build Coastguard Worker #define CLOCK_REALTIME			0
37*053f45beSAndroid Build Coastguard Worker #define CLOCK_MONOTONIC			1
38*053f45beSAndroid Build Coastguard Worker #define CLOCK_PROCESS_CPUTIME_ID	2
39*053f45beSAndroid Build Coastguard Worker #define CLOCK_THREAD_CPUTIME_ID		3
40*053f45beSAndroid Build Coastguard Worker #define CLOCK_MONOTONIC_RAW		4
41*053f45beSAndroid Build Coastguard Worker #define CLOCK_REALTIME_COARSE		5
42*053f45beSAndroid Build Coastguard Worker #define CLOCK_MONOTONIC_COARSE		6
43*053f45beSAndroid Build Coastguard Worker #define CLOCK_BOOTTIME			7
44*053f45beSAndroid Build Coastguard Worker #define CLOCK_REALTIME_ALARM		8
45*053f45beSAndroid Build Coastguard Worker #define CLOCK_BOOTTIME_ALARM		9
46*053f45beSAndroid Build Coastguard Worker #define CLOCK_HWSPECIFIC		10
47*053f45beSAndroid Build Coastguard Worker #define CLOCK_TAI			11
48*053f45beSAndroid Build Coastguard Worker #define NR_CLOCKIDS			12
49*053f45beSAndroid Build Coastguard Worker 
clockstring(int clockid)50*053f45beSAndroid Build Coastguard Worker char *clockstring(int clockid)
51*053f45beSAndroid Build Coastguard Worker {
52*053f45beSAndroid Build Coastguard Worker 	switch (clockid) {
53*053f45beSAndroid Build Coastguard Worker 	case CLOCK_REALTIME:
54*053f45beSAndroid Build Coastguard Worker 		return "CLOCK_REALTIME";
55*053f45beSAndroid Build Coastguard Worker 	case CLOCK_MONOTONIC:
56*053f45beSAndroid Build Coastguard Worker 		return "CLOCK_MONOTONIC";
57*053f45beSAndroid Build Coastguard Worker 	case CLOCK_PROCESS_CPUTIME_ID:
58*053f45beSAndroid Build Coastguard Worker 		return "CLOCK_PROCESS_CPUTIME_ID";
59*053f45beSAndroid Build Coastguard Worker 	case CLOCK_THREAD_CPUTIME_ID:
60*053f45beSAndroid Build Coastguard Worker 		return "CLOCK_THREAD_CPUTIME_ID";
61*053f45beSAndroid Build Coastguard Worker 	case CLOCK_MONOTONIC_RAW:
62*053f45beSAndroid Build Coastguard Worker 		return "CLOCK_MONOTONIC_RAW";
63*053f45beSAndroid Build Coastguard Worker 	case CLOCK_REALTIME_COARSE:
64*053f45beSAndroid Build Coastguard Worker 		return "CLOCK_REALTIME_COARSE";
65*053f45beSAndroid Build Coastguard Worker 	case CLOCK_MONOTONIC_COARSE:
66*053f45beSAndroid Build Coastguard Worker 		return "CLOCK_MONOTONIC_COARSE";
67*053f45beSAndroid Build Coastguard Worker 	case CLOCK_BOOTTIME:
68*053f45beSAndroid Build Coastguard Worker 		return "CLOCK_BOOTTIME";
69*053f45beSAndroid Build Coastguard Worker 	case CLOCK_REALTIME_ALARM:
70*053f45beSAndroid Build Coastguard Worker 		return "CLOCK_REALTIME_ALARM";
71*053f45beSAndroid Build Coastguard Worker 	case CLOCK_BOOTTIME_ALARM:
72*053f45beSAndroid Build Coastguard Worker 		return "CLOCK_BOOTTIME_ALARM";
73*053f45beSAndroid Build Coastguard Worker 	case CLOCK_TAI:
74*053f45beSAndroid Build Coastguard Worker 		return "CLOCK_TAI";
75*053f45beSAndroid Build Coastguard Worker 	}
76*053f45beSAndroid Build Coastguard Worker 	return "UNKNOWN_CLOCKID";
77*053f45beSAndroid Build Coastguard Worker }
78*053f45beSAndroid Build Coastguard Worker 
79*053f45beSAndroid Build Coastguard Worker /* returns 1 if a <= b, 0 otherwise */
in_order(struct timespec a,struct timespec b)80*053f45beSAndroid Build Coastguard Worker static inline int in_order(struct timespec a, struct timespec b)
81*053f45beSAndroid Build Coastguard Worker {
82*053f45beSAndroid Build Coastguard Worker 	/* use unsigned to avoid false positives on 2038 rollover */
83*053f45beSAndroid Build Coastguard Worker 	if ((unsigned long)a.tv_sec < (unsigned long)b.tv_sec)
84*053f45beSAndroid Build Coastguard Worker 		return 1;
85*053f45beSAndroid Build Coastguard Worker 	if ((unsigned long)a.tv_sec > (unsigned long)b.tv_sec)
86*053f45beSAndroid Build Coastguard Worker 		return 0;
87*053f45beSAndroid Build Coastguard Worker 	if (a.tv_nsec > b.tv_nsec)
88*053f45beSAndroid Build Coastguard Worker 		return 0;
89*053f45beSAndroid Build Coastguard Worker 	return 1;
90*053f45beSAndroid Build Coastguard Worker }
91*053f45beSAndroid Build Coastguard Worker 
92*053f45beSAndroid Build Coastguard Worker 
93*053f45beSAndroid Build Coastguard Worker 
consistency_test(int clock_type,unsigned long seconds)94*053f45beSAndroid Build Coastguard Worker int consistency_test(int clock_type, unsigned long seconds)
95*053f45beSAndroid Build Coastguard Worker {
96*053f45beSAndroid Build Coastguard Worker 	struct timespec list[CALLS_PER_LOOP];
97*053f45beSAndroid Build Coastguard Worker 	int i, inconsistent;
98*053f45beSAndroid Build Coastguard Worker 	long now, then;
99*053f45beSAndroid Build Coastguard Worker 	time_t t;
100*053f45beSAndroid Build Coastguard Worker 	char *start_str;
101*053f45beSAndroid Build Coastguard Worker 
102*053f45beSAndroid Build Coastguard Worker 	clock_gettime(clock_type, &list[0]);
103*053f45beSAndroid Build Coastguard Worker 	now = then = list[0].tv_sec;
104*053f45beSAndroid Build Coastguard Worker 
105*053f45beSAndroid Build Coastguard Worker 	/* timestamp start of test */
106*053f45beSAndroid Build Coastguard Worker 	t = time(0);
107*053f45beSAndroid Build Coastguard Worker 	start_str = ctime(&t);
108*053f45beSAndroid Build Coastguard Worker 
109*053f45beSAndroid Build Coastguard Worker 	while (seconds == -1 || now - then < seconds) {
110*053f45beSAndroid Build Coastguard Worker 		inconsistent = -1;
111*053f45beSAndroid Build Coastguard Worker 
112*053f45beSAndroid Build Coastguard Worker 		/* Fill list */
113*053f45beSAndroid Build Coastguard Worker 		for (i = 0; i < CALLS_PER_LOOP; i++)
114*053f45beSAndroid Build Coastguard Worker 			clock_gettime(clock_type, &list[i]);
115*053f45beSAndroid Build Coastguard Worker 
116*053f45beSAndroid Build Coastguard Worker 		/* Check for inconsistencies */
117*053f45beSAndroid Build Coastguard Worker 		for (i = 0; i < CALLS_PER_LOOP - 1; i++)
118*053f45beSAndroid Build Coastguard Worker 			if (!in_order(list[i], list[i+1]))
119*053f45beSAndroid Build Coastguard Worker 				inconsistent = i;
120*053f45beSAndroid Build Coastguard Worker 
121*053f45beSAndroid Build Coastguard Worker 		/* display inconsistency */
122*053f45beSAndroid Build Coastguard Worker 		if (inconsistent >= 0) {
123*053f45beSAndroid Build Coastguard Worker 			unsigned long long delta;
124*053f45beSAndroid Build Coastguard Worker 
125*053f45beSAndroid Build Coastguard Worker 			ksft_print_msg("\%s\n", start_str);
126*053f45beSAndroid Build Coastguard Worker 			for (i = 0; i < CALLS_PER_LOOP; i++) {
127*053f45beSAndroid Build Coastguard Worker 				if (i == inconsistent)
128*053f45beSAndroid Build Coastguard Worker 					ksft_print_msg("--------------------\n");
129*053f45beSAndroid Build Coastguard Worker 				ksft_print_msg("%lu:%lu\n", list[i].tv_sec,
130*053f45beSAndroid Build Coastguard Worker 							list[i].tv_nsec);
131*053f45beSAndroid Build Coastguard Worker 				if (i == inconsistent + 1)
132*053f45beSAndroid Build Coastguard Worker 					ksft_print_msg("--------------------\n");
133*053f45beSAndroid Build Coastguard Worker 			}
134*053f45beSAndroid Build Coastguard Worker 			delta = list[inconsistent].tv_sec * NSEC_PER_SEC;
135*053f45beSAndroid Build Coastguard Worker 			delta += list[inconsistent].tv_nsec;
136*053f45beSAndroid Build Coastguard Worker 			delta -= list[inconsistent+1].tv_sec * NSEC_PER_SEC;
137*053f45beSAndroid Build Coastguard Worker 			delta -= list[inconsistent+1].tv_nsec;
138*053f45beSAndroid Build Coastguard Worker 			ksft_print_msg("Delta: %llu ns\n", delta);
139*053f45beSAndroid Build Coastguard Worker 			fflush(0);
140*053f45beSAndroid Build Coastguard Worker 			/* timestamp inconsistency*/
141*053f45beSAndroid Build Coastguard Worker 			t = time(0);
142*053f45beSAndroid Build Coastguard Worker 			ksft_print_msg("%s\n", ctime(&t));
143*053f45beSAndroid Build Coastguard Worker 			return -1;
144*053f45beSAndroid Build Coastguard Worker 		}
145*053f45beSAndroid Build Coastguard Worker 		now = list[0].tv_sec;
146*053f45beSAndroid Build Coastguard Worker 	}
147*053f45beSAndroid Build Coastguard Worker 	return 0;
148*053f45beSAndroid Build Coastguard Worker }
149*053f45beSAndroid Build Coastguard Worker 
150*053f45beSAndroid Build Coastguard Worker 
main(int argc,char * argv[])151*053f45beSAndroid Build Coastguard Worker int main(int argc, char *argv[])
152*053f45beSAndroid Build Coastguard Worker {
153*053f45beSAndroid Build Coastguard Worker 	int clockid, opt;
154*053f45beSAndroid Build Coastguard Worker 	int userclock = CLOCK_REALTIME;
155*053f45beSAndroid Build Coastguard Worker 	int maxclocks = NR_CLOCKIDS;
156*053f45beSAndroid Build Coastguard Worker 	int runtime = 10;
157*053f45beSAndroid Build Coastguard Worker 	struct timespec ts;
158*053f45beSAndroid Build Coastguard Worker 
159*053f45beSAndroid Build Coastguard Worker 	/* Process arguments */
160*053f45beSAndroid Build Coastguard Worker 	while ((opt = getopt(argc, argv, "t:c:")) != -1) {
161*053f45beSAndroid Build Coastguard Worker 		switch (opt) {
162*053f45beSAndroid Build Coastguard Worker 		case 't':
163*053f45beSAndroid Build Coastguard Worker 			runtime = atoi(optarg);
164*053f45beSAndroid Build Coastguard Worker 			break;
165*053f45beSAndroid Build Coastguard Worker 		case 'c':
166*053f45beSAndroid Build Coastguard Worker 			userclock = atoi(optarg);
167*053f45beSAndroid Build Coastguard Worker 			maxclocks = userclock + 1;
168*053f45beSAndroid Build Coastguard Worker 			break;
169*053f45beSAndroid Build Coastguard Worker 		default:
170*053f45beSAndroid Build Coastguard Worker 			printf("Usage: %s [-t <secs>] [-c <clockid>]\n", argv[0]);
171*053f45beSAndroid Build Coastguard Worker 			printf("	-t: Number of seconds to run\n");
172*053f45beSAndroid Build Coastguard Worker 			printf("	-c: clockid to use (default, all clockids)\n");
173*053f45beSAndroid Build Coastguard Worker 			exit(-1);
174*053f45beSAndroid Build Coastguard Worker 		}
175*053f45beSAndroid Build Coastguard Worker 	}
176*053f45beSAndroid Build Coastguard Worker 
177*053f45beSAndroid Build Coastguard Worker 	setbuf(stdout, NULL);
178*053f45beSAndroid Build Coastguard Worker 
179*053f45beSAndroid Build Coastguard Worker 	ksft_print_header();
180*053f45beSAndroid Build Coastguard Worker 	ksft_set_plan(maxclocks - userclock);
181*053f45beSAndroid Build Coastguard Worker 
182*053f45beSAndroid Build Coastguard Worker 	for (clockid = userclock; clockid < maxclocks; clockid++) {
183*053f45beSAndroid Build Coastguard Worker 
184*053f45beSAndroid Build Coastguard Worker 		if (clockid == CLOCK_HWSPECIFIC || clock_gettime(clockid, &ts)) {
185*053f45beSAndroid Build Coastguard Worker 			ksft_test_result_skip("%-31s\n", clockstring(clockid));
186*053f45beSAndroid Build Coastguard Worker 			continue;
187*053f45beSAndroid Build Coastguard Worker 		}
188*053f45beSAndroid Build Coastguard Worker 
189*053f45beSAndroid Build Coastguard Worker 		if (consistency_test(clockid, runtime)) {
190*053f45beSAndroid Build Coastguard Worker 			ksft_test_result_fail("%-31s\n", clockstring(clockid));
191*053f45beSAndroid Build Coastguard Worker 			ksft_exit_fail();
192*053f45beSAndroid Build Coastguard Worker 		} else {
193*053f45beSAndroid Build Coastguard Worker 			ksft_test_result_pass("%-31s\n", clockstring(clockid));
194*053f45beSAndroid Build Coastguard Worker 		}
195*053f45beSAndroid Build Coastguard Worker 	}
196*053f45beSAndroid Build Coastguard Worker 	ksft_exit_pass();
197*053f45beSAndroid Build Coastguard Worker }
198