xref: /aosp_15_r20/external/linux-kselftest/tools/testing/selftests/timers/nsleep-lat.c (revision 053f45be4e351dfd5e965df293cd45b779f579ee)
1*053f45beSAndroid Build Coastguard Worker /* Measure nanosleep timer latency
2*053f45beSAndroid Build Coastguard Worker  *              by: john stultz ([email protected])
3*053f45beSAndroid Build Coastguard Worker  *		(C) Copyright Linaro 2013
4*053f45beSAndroid Build Coastguard Worker  *              Licensed under the GPLv2
5*053f45beSAndroid Build Coastguard Worker  *
6*053f45beSAndroid Build Coastguard Worker  *  To build:
7*053f45beSAndroid Build Coastguard Worker  *	$ gcc nsleep-lat.c -o nsleep-lat -lrt
8*053f45beSAndroid Build Coastguard Worker  *
9*053f45beSAndroid Build Coastguard Worker  *   This program is free software: you can redistribute it and/or modify
10*053f45beSAndroid Build Coastguard Worker  *   it under the terms of the GNU General Public License as published by
11*053f45beSAndroid Build Coastguard Worker  *   the Free Software Foundation, either version 2 of the License, or
12*053f45beSAndroid Build Coastguard Worker  *   (at your option) any later version.
13*053f45beSAndroid Build Coastguard Worker  *
14*053f45beSAndroid Build Coastguard Worker  *   This program is distributed in the hope that it will be useful,
15*053f45beSAndroid Build Coastguard Worker  *   but WITHOUT ANY WARRANTY; without even the implied warranty of
16*053f45beSAndroid Build Coastguard Worker  *   MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
17*053f45beSAndroid Build Coastguard Worker  *   GNU General Public License for more details.
18*053f45beSAndroid Build Coastguard Worker  */
19*053f45beSAndroid Build Coastguard Worker 
20*053f45beSAndroid Build Coastguard Worker #include <stdio.h>
21*053f45beSAndroid Build Coastguard Worker #include <stdlib.h>
22*053f45beSAndroid Build Coastguard Worker #include <time.h>
23*053f45beSAndroid Build Coastguard Worker #include <sys/time.h>
24*053f45beSAndroid Build Coastguard Worker #include <sys/timex.h>
25*053f45beSAndroid Build Coastguard Worker #include <string.h>
26*053f45beSAndroid Build Coastguard Worker #include <signal.h>
27*053f45beSAndroid Build Coastguard Worker #include "../kselftest.h"
28*053f45beSAndroid Build Coastguard Worker 
29*053f45beSAndroid Build Coastguard Worker #define NSEC_PER_SEC 1000000000ULL
30*053f45beSAndroid Build Coastguard Worker 
31*053f45beSAndroid Build Coastguard Worker #define UNRESONABLE_LATENCY 40000000 /* 40ms in nanosecs */
32*053f45beSAndroid Build Coastguard Worker 
33*053f45beSAndroid Build Coastguard Worker 
34*053f45beSAndroid Build Coastguard Worker #define CLOCK_REALTIME			0
35*053f45beSAndroid Build Coastguard Worker #define CLOCK_MONOTONIC			1
36*053f45beSAndroid Build Coastguard Worker #define CLOCK_PROCESS_CPUTIME_ID	2
37*053f45beSAndroid Build Coastguard Worker #define CLOCK_THREAD_CPUTIME_ID		3
38*053f45beSAndroid Build Coastguard Worker #define CLOCK_MONOTONIC_RAW		4
39*053f45beSAndroid Build Coastguard Worker #define CLOCK_REALTIME_COARSE		5
40*053f45beSAndroid Build Coastguard Worker #define CLOCK_MONOTONIC_COARSE		6
41*053f45beSAndroid Build Coastguard Worker #define CLOCK_BOOTTIME			7
42*053f45beSAndroid Build Coastguard Worker #define CLOCK_REALTIME_ALARM		8
43*053f45beSAndroid Build Coastguard Worker #define CLOCK_BOOTTIME_ALARM		9
44*053f45beSAndroid Build Coastguard Worker #define CLOCK_HWSPECIFIC		10
45*053f45beSAndroid Build Coastguard Worker #define CLOCK_TAI			11
46*053f45beSAndroid Build Coastguard Worker #define NR_CLOCKIDS			12
47*053f45beSAndroid Build Coastguard Worker 
48*053f45beSAndroid Build Coastguard Worker #define UNSUPPORTED 0xf00f
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 
timespec_add(struct timespec ts,unsigned long long ns)79*053f45beSAndroid Build Coastguard Worker struct timespec timespec_add(struct timespec ts, unsigned long long ns)
80*053f45beSAndroid Build Coastguard Worker {
81*053f45beSAndroid Build Coastguard Worker 	ts.tv_nsec += ns;
82*053f45beSAndroid Build Coastguard Worker 	while (ts.tv_nsec >= NSEC_PER_SEC) {
83*053f45beSAndroid Build Coastguard Worker 		ts.tv_nsec -= NSEC_PER_SEC;
84*053f45beSAndroid Build Coastguard Worker 		ts.tv_sec++;
85*053f45beSAndroid Build Coastguard Worker 	}
86*053f45beSAndroid Build Coastguard Worker 	return ts;
87*053f45beSAndroid Build Coastguard Worker }
88*053f45beSAndroid Build Coastguard Worker 
89*053f45beSAndroid Build Coastguard Worker 
timespec_sub(struct timespec a,struct timespec b)90*053f45beSAndroid Build Coastguard Worker long long timespec_sub(struct timespec a, struct timespec b)
91*053f45beSAndroid Build Coastguard Worker {
92*053f45beSAndroid Build Coastguard Worker 	long long ret = NSEC_PER_SEC * b.tv_sec + b.tv_nsec;
93*053f45beSAndroid Build Coastguard Worker 
94*053f45beSAndroid Build Coastguard Worker 	ret -= NSEC_PER_SEC * a.tv_sec + a.tv_nsec;
95*053f45beSAndroid Build Coastguard Worker 	return ret;
96*053f45beSAndroid Build Coastguard Worker }
97*053f45beSAndroid Build Coastguard Worker 
nanosleep_lat_test(int clockid,long long ns)98*053f45beSAndroid Build Coastguard Worker int nanosleep_lat_test(int clockid, long long ns)
99*053f45beSAndroid Build Coastguard Worker {
100*053f45beSAndroid Build Coastguard Worker 	struct timespec start, end, target;
101*053f45beSAndroid Build Coastguard Worker 	long long latency = 0;
102*053f45beSAndroid Build Coastguard Worker 	int i, count;
103*053f45beSAndroid Build Coastguard Worker 
104*053f45beSAndroid Build Coastguard Worker 	target.tv_sec = ns/NSEC_PER_SEC;
105*053f45beSAndroid Build Coastguard Worker 	target.tv_nsec = ns%NSEC_PER_SEC;
106*053f45beSAndroid Build Coastguard Worker 
107*053f45beSAndroid Build Coastguard Worker 	if (clock_gettime(clockid, &start))
108*053f45beSAndroid Build Coastguard Worker 		return UNSUPPORTED;
109*053f45beSAndroid Build Coastguard Worker 	if (clock_nanosleep(clockid, 0, &target, NULL))
110*053f45beSAndroid Build Coastguard Worker 		return UNSUPPORTED;
111*053f45beSAndroid Build Coastguard Worker 
112*053f45beSAndroid Build Coastguard Worker 	count = 10;
113*053f45beSAndroid Build Coastguard Worker 
114*053f45beSAndroid Build Coastguard Worker 	/* First check relative latency */
115*053f45beSAndroid Build Coastguard Worker 	clock_gettime(clockid, &start);
116*053f45beSAndroid Build Coastguard Worker 	for (i = 0; i < count; i++)
117*053f45beSAndroid Build Coastguard Worker 		clock_nanosleep(clockid, 0, &target, NULL);
118*053f45beSAndroid Build Coastguard Worker 	clock_gettime(clockid, &end);
119*053f45beSAndroid Build Coastguard Worker 
120*053f45beSAndroid Build Coastguard Worker 	if (((timespec_sub(start, end)/count)-ns) > UNRESONABLE_LATENCY) {
121*053f45beSAndroid Build Coastguard Worker 		printf("Large rel latency: %lld ns :", (timespec_sub(start, end)/count)-ns);
122*053f45beSAndroid Build Coastguard Worker 		return -1;
123*053f45beSAndroid Build Coastguard Worker 	}
124*053f45beSAndroid Build Coastguard Worker 
125*053f45beSAndroid Build Coastguard Worker 	/* Next check absolute latency */
126*053f45beSAndroid Build Coastguard Worker 	for (i = 0; i < count; i++) {
127*053f45beSAndroid Build Coastguard Worker 		clock_gettime(clockid, &start);
128*053f45beSAndroid Build Coastguard Worker 		target = timespec_add(start, ns);
129*053f45beSAndroid Build Coastguard Worker 		clock_nanosleep(clockid, TIMER_ABSTIME, &target, NULL);
130*053f45beSAndroid Build Coastguard Worker 		clock_gettime(clockid, &end);
131*053f45beSAndroid Build Coastguard Worker 		latency += timespec_sub(target, end);
132*053f45beSAndroid Build Coastguard Worker 	}
133*053f45beSAndroid Build Coastguard Worker 
134*053f45beSAndroid Build Coastguard Worker 	if (latency/count > UNRESONABLE_LATENCY) {
135*053f45beSAndroid Build Coastguard Worker 		printf("Large abs latency: %lld ns :", latency/count);
136*053f45beSAndroid Build Coastguard Worker 		return -1;
137*053f45beSAndroid Build Coastguard Worker 	}
138*053f45beSAndroid Build Coastguard Worker 
139*053f45beSAndroid Build Coastguard Worker 	return 0;
140*053f45beSAndroid Build Coastguard Worker }
141*053f45beSAndroid Build Coastguard Worker 
142*053f45beSAndroid Build Coastguard Worker 
143*053f45beSAndroid Build Coastguard Worker 
main(int argc,char ** argv)144*053f45beSAndroid Build Coastguard Worker int main(int argc, char **argv)
145*053f45beSAndroid Build Coastguard Worker {
146*053f45beSAndroid Build Coastguard Worker 	long long length;
147*053f45beSAndroid Build Coastguard Worker 	int clockid, ret;
148*053f45beSAndroid Build Coastguard Worker 
149*053f45beSAndroid Build Coastguard Worker 	for (clockid = CLOCK_REALTIME; clockid < NR_CLOCKIDS; clockid++) {
150*053f45beSAndroid Build Coastguard Worker 
151*053f45beSAndroid Build Coastguard Worker 		/* Skip cputime clockids since nanosleep won't increment cputime */
152*053f45beSAndroid Build Coastguard Worker 		if (clockid == CLOCK_PROCESS_CPUTIME_ID ||
153*053f45beSAndroid Build Coastguard Worker 				clockid == CLOCK_THREAD_CPUTIME_ID ||
154*053f45beSAndroid Build Coastguard Worker 				clockid == CLOCK_HWSPECIFIC)
155*053f45beSAndroid Build Coastguard Worker 			continue;
156*053f45beSAndroid Build Coastguard Worker 
157*053f45beSAndroid Build Coastguard Worker 		printf("nsleep latency %-26s ", clockstring(clockid));
158*053f45beSAndroid Build Coastguard Worker 		fflush(stdout);
159*053f45beSAndroid Build Coastguard Worker 
160*053f45beSAndroid Build Coastguard Worker 		length = 10;
161*053f45beSAndroid Build Coastguard Worker 		while (length <= (NSEC_PER_SEC * 10)) {
162*053f45beSAndroid Build Coastguard Worker 			ret = nanosleep_lat_test(clockid, length);
163*053f45beSAndroid Build Coastguard Worker 			if (ret)
164*053f45beSAndroid Build Coastguard Worker 				break;
165*053f45beSAndroid Build Coastguard Worker 			length *= 100;
166*053f45beSAndroid Build Coastguard Worker 
167*053f45beSAndroid Build Coastguard Worker 		}
168*053f45beSAndroid Build Coastguard Worker 
169*053f45beSAndroid Build Coastguard Worker 		if (ret == UNSUPPORTED) {
170*053f45beSAndroid Build Coastguard Worker 			printf("[UNSUPPORTED]\n");
171*053f45beSAndroid Build Coastguard Worker 			continue;
172*053f45beSAndroid Build Coastguard Worker 		}
173*053f45beSAndroid Build Coastguard Worker 		if (ret < 0) {
174*053f45beSAndroid Build Coastguard Worker 			printf("[FAILED]\n");
175*053f45beSAndroid Build Coastguard Worker 			return ksft_exit_fail();
176*053f45beSAndroid Build Coastguard Worker 		}
177*053f45beSAndroid Build Coastguard Worker 		printf("[OK]\n");
178*053f45beSAndroid Build Coastguard Worker 	}
179*053f45beSAndroid Build Coastguard Worker 	return ksft_exit_pass();
180*053f45beSAndroid Build Coastguard Worker }
181