1*053f45beSAndroid Build Coastguard Worker /* Make sure timers don't return early
2*053f45beSAndroid Build Coastguard Worker * by: john stultz ([email protected])
3*053f45beSAndroid Build Coastguard Worker * John Stultz ([email protected])
4*053f45beSAndroid Build Coastguard Worker * (C) Copyright IBM 2012
5*053f45beSAndroid Build Coastguard Worker * (C) Copyright Linaro 2013 2015
6*053f45beSAndroid Build Coastguard Worker * Licensed under the GPLv2
7*053f45beSAndroid Build Coastguard Worker *
8*053f45beSAndroid Build Coastguard Worker * To build:
9*053f45beSAndroid Build Coastguard Worker * $ gcc nanosleep.c -o nanosleep -lrt
10*053f45beSAndroid Build Coastguard Worker *
11*053f45beSAndroid Build Coastguard Worker * This program is free software: you can redistribute it and/or modify
12*053f45beSAndroid Build Coastguard Worker * it under the terms of the GNU General Public License as published by
13*053f45beSAndroid Build Coastguard Worker * the Free Software Foundation, either version 2 of the License, or
14*053f45beSAndroid Build Coastguard Worker * (at your option) any later version.
15*053f45beSAndroid Build Coastguard Worker *
16*053f45beSAndroid Build Coastguard Worker * This program is distributed in the hope that it will be useful,
17*053f45beSAndroid Build Coastguard Worker * but WITHOUT ANY WARRANTY; without even the implied warranty of
18*053f45beSAndroid Build Coastguard Worker * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
19*053f45beSAndroid Build Coastguard Worker * GNU General Public License for more details.
20*053f45beSAndroid Build Coastguard Worker */
21*053f45beSAndroid Build Coastguard Worker
22*053f45beSAndroid Build Coastguard Worker #include <errno.h>
23*053f45beSAndroid Build Coastguard Worker #include <stdio.h>
24*053f45beSAndroid Build Coastguard Worker #include <stdlib.h>
25*053f45beSAndroid Build Coastguard Worker #include <time.h>
26*053f45beSAndroid Build Coastguard Worker #include <sys/time.h>
27*053f45beSAndroid Build Coastguard Worker #include <sys/timex.h>
28*053f45beSAndroid Build Coastguard Worker #include <string.h>
29*053f45beSAndroid Build Coastguard Worker #include <signal.h>
30*053f45beSAndroid Build Coastguard Worker #include "../kselftest.h"
31*053f45beSAndroid Build Coastguard Worker
32*053f45beSAndroid Build Coastguard Worker #define NSEC_PER_SEC 1000000000ULL
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
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 if (a.tv_sec < b.tv_sec)
83*053f45beSAndroid Build Coastguard Worker return 1;
84*053f45beSAndroid Build Coastguard Worker if (a.tv_sec > b.tv_sec)
85*053f45beSAndroid Build Coastguard Worker return 0;
86*053f45beSAndroid Build Coastguard Worker if (a.tv_nsec > b.tv_nsec)
87*053f45beSAndroid Build Coastguard Worker return 0;
88*053f45beSAndroid Build Coastguard Worker return 1;
89*053f45beSAndroid Build Coastguard Worker }
90*053f45beSAndroid Build Coastguard Worker
timespec_add(struct timespec ts,unsigned long long ns)91*053f45beSAndroid Build Coastguard Worker struct timespec timespec_add(struct timespec ts, unsigned long long ns)
92*053f45beSAndroid Build Coastguard Worker {
93*053f45beSAndroid Build Coastguard Worker ts.tv_nsec += ns;
94*053f45beSAndroid Build Coastguard Worker while (ts.tv_nsec >= NSEC_PER_SEC) {
95*053f45beSAndroid Build Coastguard Worker ts.tv_nsec -= NSEC_PER_SEC;
96*053f45beSAndroid Build Coastguard Worker ts.tv_sec++;
97*053f45beSAndroid Build Coastguard Worker }
98*053f45beSAndroid Build Coastguard Worker return ts;
99*053f45beSAndroid Build Coastguard Worker }
100*053f45beSAndroid Build Coastguard Worker
nanosleep_test(int clockid,long long ns)101*053f45beSAndroid Build Coastguard Worker int nanosleep_test(int clockid, long long ns)
102*053f45beSAndroid Build Coastguard Worker {
103*053f45beSAndroid Build Coastguard Worker struct timespec now, target, rel;
104*053f45beSAndroid Build Coastguard Worker
105*053f45beSAndroid Build Coastguard Worker /* First check abs time */
106*053f45beSAndroid Build Coastguard Worker if (clock_gettime(clockid, &now))
107*053f45beSAndroid Build Coastguard Worker return UNSUPPORTED;
108*053f45beSAndroid Build Coastguard Worker target = timespec_add(now, ns);
109*053f45beSAndroid Build Coastguard Worker
110*053f45beSAndroid Build Coastguard Worker if (clock_nanosleep(clockid, TIMER_ABSTIME, &target, NULL))
111*053f45beSAndroid Build Coastguard Worker return UNSUPPORTED;
112*053f45beSAndroid Build Coastguard Worker clock_gettime(clockid, &now);
113*053f45beSAndroid Build Coastguard Worker
114*053f45beSAndroid Build Coastguard Worker if (!in_order(target, now))
115*053f45beSAndroid Build Coastguard Worker return -1;
116*053f45beSAndroid Build Coastguard Worker
117*053f45beSAndroid Build Coastguard Worker /* Second check reltime */
118*053f45beSAndroid Build Coastguard Worker clock_gettime(clockid, &now);
119*053f45beSAndroid Build Coastguard Worker rel.tv_sec = 0;
120*053f45beSAndroid Build Coastguard Worker rel.tv_nsec = 0;
121*053f45beSAndroid Build Coastguard Worker rel = timespec_add(rel, ns);
122*053f45beSAndroid Build Coastguard Worker target = timespec_add(now, ns);
123*053f45beSAndroid Build Coastguard Worker clock_nanosleep(clockid, 0, &rel, NULL);
124*053f45beSAndroid Build Coastguard Worker clock_gettime(clockid, &now);
125*053f45beSAndroid Build Coastguard Worker
126*053f45beSAndroid Build Coastguard Worker if (!in_order(target, now))
127*053f45beSAndroid Build Coastguard Worker return -1;
128*053f45beSAndroid Build Coastguard Worker return 0;
129*053f45beSAndroid Build Coastguard Worker }
130*053f45beSAndroid Build Coastguard Worker
main(int argc,char ** argv)131*053f45beSAndroid Build Coastguard Worker int main(int argc, char **argv)
132*053f45beSAndroid Build Coastguard Worker {
133*053f45beSAndroid Build Coastguard Worker long long length;
134*053f45beSAndroid Build Coastguard Worker int clockid, ret;
135*053f45beSAndroid Build Coastguard Worker
136*053f45beSAndroid Build Coastguard Worker ksft_print_header();
137*053f45beSAndroid Build Coastguard Worker ksft_set_plan(NR_CLOCKIDS);
138*053f45beSAndroid Build Coastguard Worker
139*053f45beSAndroid Build Coastguard Worker for (clockid = CLOCK_REALTIME; clockid < NR_CLOCKIDS; clockid++) {
140*053f45beSAndroid Build Coastguard Worker
141*053f45beSAndroid Build Coastguard Worker /* Skip cputime clockids since nanosleep won't increment cputime */
142*053f45beSAndroid Build Coastguard Worker if (clockid == CLOCK_PROCESS_CPUTIME_ID ||
143*053f45beSAndroid Build Coastguard Worker clockid == CLOCK_THREAD_CPUTIME_ID ||
144*053f45beSAndroid Build Coastguard Worker clockid == CLOCK_HWSPECIFIC) {
145*053f45beSAndroid Build Coastguard Worker ksft_test_result_skip("%-31s\n", clockstring(clockid));
146*053f45beSAndroid Build Coastguard Worker continue;
147*053f45beSAndroid Build Coastguard Worker }
148*053f45beSAndroid Build Coastguard Worker
149*053f45beSAndroid Build Coastguard Worker fflush(stdout);
150*053f45beSAndroid Build Coastguard Worker
151*053f45beSAndroid Build Coastguard Worker length = 10;
152*053f45beSAndroid Build Coastguard Worker while (length <= (NSEC_PER_SEC * 10)) {
153*053f45beSAndroid Build Coastguard Worker ret = nanosleep_test(clockid, length);
154*053f45beSAndroid Build Coastguard Worker if (ret == UNSUPPORTED) {
155*053f45beSAndroid Build Coastguard Worker ksft_test_result_skip("%-31s\n", clockstring(clockid));
156*053f45beSAndroid Build Coastguard Worker goto next;
157*053f45beSAndroid Build Coastguard Worker }
158*053f45beSAndroid Build Coastguard Worker if (ret < 0) {
159*053f45beSAndroid Build Coastguard Worker ksft_test_result_fail("%-31s\n", clockstring(clockid));
160*053f45beSAndroid Build Coastguard Worker ksft_exit_fail();
161*053f45beSAndroid Build Coastguard Worker }
162*053f45beSAndroid Build Coastguard Worker length *= 100;
163*053f45beSAndroid Build Coastguard Worker }
164*053f45beSAndroid Build Coastguard Worker ksft_test_result_pass("%-31s\n", clockstring(clockid));
165*053f45beSAndroid Build Coastguard Worker next:
166*053f45beSAndroid Build Coastguard Worker ret = 0;
167*053f45beSAndroid Build Coastguard Worker }
168*053f45beSAndroid Build Coastguard Worker ksft_exit_pass();
169*053f45beSAndroid Build Coastguard Worker }
170