1*0d02a753SJohn Stultz /* Time bounds setting test
2*0d02a753SJohn Stultz * by: john stultz (johnstul@us.ibm.com)
3*0d02a753SJohn Stultz * (C) Copyright IBM 2012
4*0d02a753SJohn Stultz * Licensed under the GPLv2
5*0d02a753SJohn Stultz *
6*0d02a753SJohn Stultz * NOTE: This is a meta-test which sets the time to edge cases then
7*0d02a753SJohn Stultz * uses other tests to detect problems. Thus this test requires that
8*0d02a753SJohn Stultz * the inconsistency-check and nanosleep tests be present in the same
9*0d02a753SJohn Stultz * directory it is run from.
10*0d02a753SJohn Stultz *
11*0d02a753SJohn Stultz * To build:
12*0d02a753SJohn Stultz * $ gcc set-2038.c -o set-2038 -lrt
13*0d02a753SJohn Stultz *
14*0d02a753SJohn Stultz * This program is free software: you can redistribute it and/or modify
15*0d02a753SJohn Stultz * it under the terms of the GNU General Public License as published by
16*0d02a753SJohn Stultz * the Free Software Foundation, either version 2 of the License, or
17*0d02a753SJohn Stultz * (at your option) any later version.
18*0d02a753SJohn Stultz *
19*0d02a753SJohn Stultz * This program is distributed in the hope that it will be useful,
20*0d02a753SJohn Stultz * but WITHOUT ANY WARRANTY; without even the implied warranty of
21*0d02a753SJohn Stultz * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
22*0d02a753SJohn Stultz * GNU General Public License for more details.
23*0d02a753SJohn Stultz */
24*0d02a753SJohn Stultz
25*0d02a753SJohn Stultz #include <stdio.h>
26*0d02a753SJohn Stultz #include <stdlib.h>
27*0d02a753SJohn Stultz #include <unistd.h>
28*0d02a753SJohn Stultz #include <time.h>
29*0d02a753SJohn Stultz #include <sys/time.h>
30*0d02a753SJohn Stultz #include "../kselftest.h"
31*0d02a753SJohn Stultz
32*0d02a753SJohn Stultz #define NSEC_PER_SEC 1000000000LL
33*0d02a753SJohn Stultz
34*0d02a753SJohn Stultz #define KTIME_MAX ((long long)~((unsigned long long)1 << 63))
35*0d02a753SJohn Stultz #define KTIME_SEC_MAX (KTIME_MAX / NSEC_PER_SEC)
36*0d02a753SJohn Stultz
37*0d02a753SJohn Stultz #define YEAR_1901 (-0x7fffffffL)
38*0d02a753SJohn Stultz #define YEAR_1970 1
39*0d02a753SJohn Stultz #define YEAR_2038 0x7fffffffL /*overflows 32bit time_t */
40*0d02a753SJohn Stultz #define YEAR_2262 KTIME_SEC_MAX /*overflows 64bit ktime_t */
41*0d02a753SJohn Stultz #define YEAR_MAX ((long long)((1ULL<<63)-1)) /*overflows 64bit time_t */
42*0d02a753SJohn Stultz
is32bits(void)43*0d02a753SJohn Stultz int is32bits(void)
44*0d02a753SJohn Stultz {
45*0d02a753SJohn Stultz return (sizeof(long) == 4);
46*0d02a753SJohn Stultz }
47*0d02a753SJohn Stultz
settime(long long time)48*0d02a753SJohn Stultz int settime(long long time)
49*0d02a753SJohn Stultz {
50*0d02a753SJohn Stultz struct timeval now;
51*0d02a753SJohn Stultz int ret;
52*0d02a753SJohn Stultz
53*0d02a753SJohn Stultz now.tv_sec = (time_t)time;
54*0d02a753SJohn Stultz now.tv_usec = 0;
55*0d02a753SJohn Stultz
56*0d02a753SJohn Stultz ret = settimeofday(&now, NULL);
57*0d02a753SJohn Stultz
58*0d02a753SJohn Stultz printf("Setting time to 0x%lx: %d\n", (long)time, ret);
59*0d02a753SJohn Stultz return ret;
60*0d02a753SJohn Stultz }
61*0d02a753SJohn Stultz
do_tests(void)62*0d02a753SJohn Stultz int do_tests(void)
63*0d02a753SJohn Stultz {
64*0d02a753SJohn Stultz int ret;
65*0d02a753SJohn Stultz
66*0d02a753SJohn Stultz ret = system("date");
67*0d02a753SJohn Stultz ret = system("./inconsistency-check -c 0 -t 20");
68*0d02a753SJohn Stultz ret |= system("./nanosleep");
69*0d02a753SJohn Stultz ret |= system("./nsleep-lat");
70*0d02a753SJohn Stultz return ret;
71*0d02a753SJohn Stultz
72*0d02a753SJohn Stultz }
73*0d02a753SJohn Stultz
main(int argc,char * argv[])74*0d02a753SJohn Stultz int main(int argc, char *argv[])
75*0d02a753SJohn Stultz {
76*0d02a753SJohn Stultz int ret = 0;
77*0d02a753SJohn Stultz int opt, dangerous = 0;
78*0d02a753SJohn Stultz time_t start;
79*0d02a753SJohn Stultz
80*0d02a753SJohn Stultz /* Process arguments */
81*0d02a753SJohn Stultz while ((opt = getopt(argc, argv, "d")) != -1) {
82*0d02a753SJohn Stultz switch (opt) {
83*0d02a753SJohn Stultz case 'd':
84*0d02a753SJohn Stultz dangerous = 1;
85*0d02a753SJohn Stultz }
86*0d02a753SJohn Stultz }
87*0d02a753SJohn Stultz
88*0d02a753SJohn Stultz start = time(0);
89*0d02a753SJohn Stultz
90*0d02a753SJohn Stultz /* First test that crazy values don't work */
91*0d02a753SJohn Stultz if (!settime(YEAR_1901)) {
92*0d02a753SJohn Stultz ret = -1;
93*0d02a753SJohn Stultz goto out;
94*0d02a753SJohn Stultz }
95*0d02a753SJohn Stultz if (!settime(YEAR_MAX)) {
96*0d02a753SJohn Stultz ret = -1;
97*0d02a753SJohn Stultz goto out;
98*0d02a753SJohn Stultz }
99*0d02a753SJohn Stultz if (!is32bits() && !settime(YEAR_2262)) {
100*0d02a753SJohn Stultz ret = -1;
101*0d02a753SJohn Stultz goto out;
102*0d02a753SJohn Stultz }
103*0d02a753SJohn Stultz
104*0d02a753SJohn Stultz /* Now test behavior near edges */
105*0d02a753SJohn Stultz settime(YEAR_1970);
106*0d02a753SJohn Stultz ret = do_tests();
107*0d02a753SJohn Stultz if (ret)
108*0d02a753SJohn Stultz goto out;
109*0d02a753SJohn Stultz
110*0d02a753SJohn Stultz settime(YEAR_2038 - 600);
111*0d02a753SJohn Stultz ret = do_tests();
112*0d02a753SJohn Stultz if (ret)
113*0d02a753SJohn Stultz goto out;
114*0d02a753SJohn Stultz
115*0d02a753SJohn Stultz /* The rest of the tests can blowup on 32bit systems */
116*0d02a753SJohn Stultz if (is32bits() && !dangerous)
117*0d02a753SJohn Stultz goto out;
118*0d02a753SJohn Stultz /* Test rollover behavior 32bit edge */
119*0d02a753SJohn Stultz settime(YEAR_2038 - 10);
120*0d02a753SJohn Stultz ret = do_tests();
121*0d02a753SJohn Stultz if (ret)
122*0d02a753SJohn Stultz goto out;
123*0d02a753SJohn Stultz
124*0d02a753SJohn Stultz settime(YEAR_2262 - 600);
125*0d02a753SJohn Stultz ret = do_tests();
126*0d02a753SJohn Stultz
127*0d02a753SJohn Stultz out:
128*0d02a753SJohn Stultz /* restore clock */
129*0d02a753SJohn Stultz settime(start);
130*0d02a753SJohn Stultz if (ret)
131*0d02a753SJohn Stultz return ksft_exit_fail();
132*0d02a753SJohn Stultz return ksft_exit_pass();
133*0d02a753SJohn Stultz }
134