1 /*
2  * Copyright (c) 1991, 1992 Paul Kranenburg <pk@cs.few.eur.nl>
3  * Copyright (c) 1993 Branko Lankester <branko@hacktic.nl>
4  * Copyright (c) 1993, 1994, 1995, 1996 Rick Sladkey <jrs@world.std.com>
5  * All rights reserved.
6  *
7  * Redistribution and use in source and binary forms, with or without
8  * modification, are permitted provided that the following conditions
9  * are met:
10  * 1. Redistributions of source code must retain the above copyright
11  *    notice, this list of conditions and the following disclaimer.
12  * 2. Redistributions in binary form must reproduce the above copyright
13  *    notice, this list of conditions and the following disclaimer in the
14  *    documentation and/or other materials provided with the distribution.
15  * 3. The name of the author may not be used to endorse or promote products
16  *    derived from this software without specific prior written permission.
17  *
18  * THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``AS IS'' AND ANY EXPRESS OR
19  * IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES
20  * OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED.
21  * IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY DIRECT, INDIRECT,
22  * INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT
23  * NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
24  * DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
25  * THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
26  * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF
27  * THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
28  */
29 
30 #include "defs.h"
31 #include <fcntl.h>
32 #include <signal.h>
33 #include <sys/timex.h>
34 
35 static void
print_timezone(struct tcb * const tcp,const kernel_ulong_t addr)36 print_timezone(struct tcb *const tcp, const kernel_ulong_t addr)
37 {
38 	struct timezone tz;
39 
40 	if (umove_or_printaddr(tcp, addr, &tz))
41 		return;
42 
43 	tprintf("{tz_minuteswest=%d, tz_dsttime=%d}",
44 		tz.tz_minuteswest, tz.tz_dsttime);
45 }
46 
SYS_FUNC(gettimeofday)47 SYS_FUNC(gettimeofday)
48 {
49 	if (exiting(tcp)) {
50 		print_timeval(tcp, tcp->u_arg[0]);
51 		tprints(", ");
52 		print_timezone(tcp, tcp->u_arg[1]);
53 	}
54 	return 0;
55 }
56 
57 #ifdef ALPHA
SYS_FUNC(osf_gettimeofday)58 SYS_FUNC(osf_gettimeofday)
59 {
60 	if (exiting(tcp)) {
61 		print_timeval32(tcp, tcp->u_arg[0]);
62 		tprints(", ");
63 		print_timezone(tcp, tcp->u_arg[1]);
64 	}
65 	return 0;
66 }
67 #endif
68 
SYS_FUNC(settimeofday)69 SYS_FUNC(settimeofday)
70 {
71 	print_timeval(tcp, tcp->u_arg[0]);
72 	tprints(", ");
73 	print_timezone(tcp, tcp->u_arg[1]);
74 
75 	return RVAL_DECODED;
76 }
77 
78 #ifdef ALPHA
SYS_FUNC(osf_settimeofday)79 SYS_FUNC(osf_settimeofday)
80 {
81 	print_timeval32(tcp, tcp->u_arg[0]);
82 	tprints(", ");
83 	print_timezone(tcp, tcp->u_arg[1]);
84 
85 	return RVAL_DECODED;
86 }
87 #endif
88 
SYS_FUNC(nanosleep)89 SYS_FUNC(nanosleep)
90 {
91 	if (entering(tcp)) {
92 		print_timespec(tcp, tcp->u_arg[0]);
93 		tprints(", ");
94 	} else {
95 
96 		/*
97 		 * Second (returned) timespec is only significant if syscall
98 		 * was interrupted.  On success and in case of other errors we
99 		 * print only its address, since kernel doesn't modify it,
100 		 * and printing the value may show uninitialized data.
101 		 */
102 		if (is_erestart(tcp)) {
103 			temporarily_clear_syserror(tcp);
104 			print_timespec(tcp, tcp->u_arg[1]);
105 			restore_cleared_syserror(tcp);
106 		} else {
107 			printaddr(tcp->u_arg[1]);
108 		}
109 	}
110 	return 0;
111 }
112 
113 #include "xlat/itimer_which.h"
114 
SYS_FUNC(getitimer)115 SYS_FUNC(getitimer)
116 {
117 	if (entering(tcp)) {
118 		printxval(itimer_which, tcp->u_arg[0], "ITIMER_???");
119 		tprints(", ");
120 	} else {
121 		print_itimerval(tcp, tcp->u_arg[1]);
122 	}
123 	return 0;
124 }
125 
126 #ifdef ALPHA
SYS_FUNC(osf_getitimer)127 SYS_FUNC(osf_getitimer)
128 {
129 	if (entering(tcp)) {
130 		printxval(itimer_which, tcp->u_arg[0], "ITIMER_???");
131 		tprints(", ");
132 	} else {
133 		print_itimerval32(tcp, tcp->u_arg[1]);
134 	}
135 	return 0;
136 }
137 #endif
138 
SYS_FUNC(setitimer)139 SYS_FUNC(setitimer)
140 {
141 	if (entering(tcp)) {
142 		printxval(itimer_which, tcp->u_arg[0], "ITIMER_???");
143 		tprints(", ");
144 		print_itimerval(tcp, tcp->u_arg[1]);
145 		tprints(", ");
146 	} else {
147 		print_itimerval(tcp, tcp->u_arg[2]);
148 	}
149 	return 0;
150 }
151 
152 #ifdef ALPHA
SYS_FUNC(osf_setitimer)153 SYS_FUNC(osf_setitimer)
154 {
155 	if (entering(tcp)) {
156 		printxval(itimer_which, tcp->u_arg[0], "ITIMER_???");
157 		tprints(", ");
158 		print_itimerval32(tcp, tcp->u_arg[1]);
159 		tprints(", ");
160 	} else {
161 		print_itimerval32(tcp, tcp->u_arg[2]);
162 	}
163 	return 0;
164 }
165 #endif
166 
167 #include "xlat/adjtimex_state.h"
168 
169 static int
do_adjtimex(struct tcb * const tcp,const kernel_ulong_t addr)170 do_adjtimex(struct tcb *const tcp, const kernel_ulong_t addr)
171 {
172 	if (print_timex(tcp, addr))
173 		return 0;
174 	tcp->auxstr = xlookup(adjtimex_state, (kernel_ulong_t) tcp->u_rval);
175 	if (tcp->auxstr)
176 		return RVAL_STR;
177 	return 0;
178 }
179 
SYS_FUNC(adjtimex)180 SYS_FUNC(adjtimex)
181 {
182 	if (exiting(tcp))
183 		return do_adjtimex(tcp, tcp->u_arg[0]);
184 	return 0;
185 }
186 
187 #include "xlat/clockflags.h"
188 #include "xlat/clocknames.h"
189 
190 static void
printclockname(int clockid)191 printclockname(int clockid)
192 {
193 #ifdef CLOCKID_TO_FD
194 # include "xlat/cpuclocknames.h"
195 
196 	if (clockid < 0) {
197 		if ((clockid & CLOCKFD_MASK) == CLOCKFD)
198 			tprintf("FD_TO_CLOCKID(%d)", CLOCKID_TO_FD(clockid));
199 		else {
200 			if(CPUCLOCK_PERTHREAD(clockid))
201 				tprintf("MAKE_THREAD_CPUCLOCK(%d,", CPUCLOCK_PID(clockid));
202 			else
203 				tprintf("MAKE_PROCESS_CPUCLOCK(%d,", CPUCLOCK_PID(clockid));
204 			printxval(cpuclocknames, clockid & CLOCKFD_MASK, "CPUCLOCK_???");
205 			tprints(")");
206 		}
207 	}
208 	else
209 #endif
210 		printxval(clocknames, clockid, "CLOCK_???");
211 }
212 
SYS_FUNC(clock_settime)213 SYS_FUNC(clock_settime)
214 {
215 	printclockname(tcp->u_arg[0]);
216 	tprints(", ");
217 	print_timespec(tcp, tcp->u_arg[1]);
218 
219 	return RVAL_DECODED;
220 }
221 
SYS_FUNC(clock_gettime)222 SYS_FUNC(clock_gettime)
223 {
224 	if (entering(tcp)) {
225 		printclockname(tcp->u_arg[0]);
226 		tprints(", ");
227 	} else {
228 		print_timespec(tcp, tcp->u_arg[1]);
229 	}
230 	return 0;
231 }
232 
SYS_FUNC(clock_nanosleep)233 SYS_FUNC(clock_nanosleep)
234 {
235 	if (entering(tcp)) {
236 		printclockname(tcp->u_arg[0]);
237 		tprints(", ");
238 		printflags(clockflags, tcp->u_arg[1], "TIMER_???");
239 		tprints(", ");
240 		print_timespec(tcp, tcp->u_arg[2]);
241 		tprints(", ");
242 	} else {
243 		/*
244 		 * Second (returned) timespec is only significant
245 		 * if syscall was interrupted and flags is not TIMER_ABSTIME.
246 		 */
247 		if (!tcp->u_arg[1] && is_erestart(tcp)) {
248 			temporarily_clear_syserror(tcp);
249 			print_timespec(tcp, tcp->u_arg[3]);
250 			restore_cleared_syserror(tcp);
251 		} else {
252 			printaddr(tcp->u_arg[3]);
253 		}
254 	}
255 	return 0;
256 }
257 
SYS_FUNC(clock_adjtime)258 SYS_FUNC(clock_adjtime)
259 {
260 	if (exiting(tcp))
261 		return do_adjtimex(tcp, tcp->u_arg[1]);
262 	printclockname(tcp->u_arg[0]);
263 	tprints(", ");
264 	return 0;
265 }
266 
SYS_FUNC(timer_create)267 SYS_FUNC(timer_create)
268 {
269 	if (entering(tcp)) {
270 		printclockname(tcp->u_arg[0]);
271 		tprints(", ");
272 		print_sigevent(tcp, tcp->u_arg[1]);
273 		tprints(", ");
274 	} else {
275 		printnum_int(tcp, tcp->u_arg[2], "%d");
276 	}
277 	return 0;
278 }
279 
SYS_FUNC(timer_settime)280 SYS_FUNC(timer_settime)
281 {
282 	if (entering(tcp)) {
283 		tprintf("%d, ", (int) tcp->u_arg[0]);
284 		printflags(clockflags, tcp->u_arg[1], "TIMER_???");
285 		tprints(", ");
286 		print_itimerspec(tcp, tcp->u_arg[2]);
287 		tprints(", ");
288 	} else {
289 		print_itimerspec(tcp, tcp->u_arg[3]);
290 	}
291 	return 0;
292 }
293 
SYS_FUNC(timer_gettime)294 SYS_FUNC(timer_gettime)
295 {
296 	if (entering(tcp)) {
297 		tprintf("%d, ", (int) tcp->u_arg[0]);
298 	} else {
299 		print_itimerspec(tcp, tcp->u_arg[1]);
300 	}
301 	return 0;
302 }
303 
304 #include "xlat/timerfdflags.h"
305 
SYS_FUNC(timerfd_create)306 SYS_FUNC(timerfd_create)
307 {
308 	printclockname(tcp->u_arg[0]);
309 	tprints(", ");
310 	printflags(timerfdflags, tcp->u_arg[1], "TFD_???");
311 
312 	return RVAL_DECODED | RVAL_FD;
313 }
314 
SYS_FUNC(timerfd_settime)315 SYS_FUNC(timerfd_settime)
316 {
317 	if (entering(tcp)) {
318 		printfd(tcp, tcp->u_arg[0]);
319 		tprints(", ");
320 		printflags(timerfdflags, tcp->u_arg[1], "TFD_???");
321 		tprints(", ");
322 		print_itimerspec(tcp, tcp->u_arg[2]);
323 		tprints(", ");
324 	} else {
325 		print_itimerspec(tcp, tcp->u_arg[3]);
326 	}
327 	return 0;
328 }
329 
SYS_FUNC(timerfd_gettime)330 SYS_FUNC(timerfd_gettime)
331 {
332 	if (entering(tcp)) {
333 		printfd(tcp, tcp->u_arg[0]);
334 		tprints(", ");
335 	} else {
336 		print_itimerspec(tcp, tcp->u_arg[1]);
337 	}
338 	return 0;
339 }
340