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>
7 * Redistribution and use in source and binary forms, with or without
8 * modification, are permitted provided that the following conditions
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.
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.
32 #include <linux/version.h>
33 #include <sys/timex.h>
36 #define UTIME_NOW ((1l << 30) - 1l)
39 #define UTIME_OMIT ((1l << 30) - 2l)
42 #if SUPPORTED_PERSONALITIES > 1
43 # if defined X86_64 || defined X32
44 # define current_time_t_is_compat (current_personality == 1)
46 # define current_time_t_is_compat (current_wordsize == 4)
49 # define current_time_t_is_compat 0
54 u_int32_t tv_sec, tv_usec;
58 tprint_timeval32(struct tcb *tcp, const struct timeval32 *tv)
60 tprintf("{%u, %u}", tv->tv_sec, tv->tv_usec);
64 tprint_timeval(struct tcb *tcp, const struct timeval *tv)
66 tprintf("{%ju, %ju}", (uintmax_t) tv->tv_sec, (uintmax_t) tv->tv_usec);
70 printtv_bitness(struct tcb *tcp, long addr, enum bitness_t bitness, int special)
72 char buf[TIMEVAL_TEXT_BUFSIZE];
73 sprinttv(buf, tcp, addr, bitness, special);
78 do_sprinttv(char *buf, const uintmax_t sec, const uintmax_t usec,
84 return stpcpy(buf, "UTIME_NOW");
86 return stpcpy(buf, "UTIME_OMIT");
89 return buf + sprintf(buf, "{%ju, %ju}", sec, usec);
93 sprinttv(char *buf, struct tcb *tcp, long addr, enum bitness_t bitness, int special)
96 return stpcpy(buf, "NULL");
98 if (!verbose(tcp) || (exiting(tcp) && syserror(tcp)))
99 return buf + sprintf(buf, "%#lx", addr);
101 if (bitness == BITNESS_32 || current_time_t_is_compat)
105 if (umove(tcp, addr, &tv) >= 0)
106 return do_sprinttv(buf, tv.tv_sec, tv.tv_usec, special);
110 if (umove(tcp, addr, &tv) >= 0)
111 return do_sprinttv(buf, tv.tv_sec, tv.tv_usec, special);
114 return buf + sprintf(buf, "%#lx", addr);
118 print_timespec(struct tcb *tcp, long addr)
120 char buf[TIMESPEC_TEXT_BUFSIZE];
121 sprint_timespec(buf, tcp, addr);
126 sprint_timespec(char *buf, struct tcb *tcp, long addr)
130 else if (!verbose(tcp))
131 sprintf(buf, "%#lx", addr);
135 #if SUPPORTED_PERSONALITIES > 1
136 if (current_time_t_is_compat) {
139 rc = umove(tcp, addr, &tv);
141 sprintf(buf, "{%u, %u}",
142 tv.tv_sec, tv.tv_usec);
148 rc = umove(tcp, addr, &ts);
150 sprintf(buf, "{%ju, %ju}",
151 (uintmax_t) ts.tv_sec,
152 (uintmax_t) ts.tv_nsec);
155 strcpy(buf, "{...}");
162 printnum_long(tcp, tcp->u_arg[0], "%ld");
167 SYS_FUNC(gettimeofday)
170 printtv(tcp, tcp->u_arg[0]);
172 printtv(tcp, tcp->u_arg[1]);
178 SYS_FUNC(osf_gettimeofday)
181 printtv_bitness(tcp, tcp->u_arg[0], BITNESS_32, 0);
183 printtv_bitness(tcp, tcp->u_arg[1], BITNESS_32, 0);
189 SYS_FUNC(settimeofday)
191 printtv(tcp, tcp->u_arg[0]);
193 printtv(tcp, tcp->u_arg[1]);
199 SYS_FUNC(osf_settimeofday)
201 printtv_bitness(tcp, tcp->u_arg[0], BITNESS_32, 0);
203 printtv_bitness(tcp, tcp->u_arg[1], BITNESS_32, 0);
212 printtv(tcp, tcp->u_arg[0]);
215 printtv(tcp, tcp->u_arg[1]);
223 print_timespec(tcp, tcp->u_arg[0]);
226 /* Second (returned) timespec is only significant
227 * if syscall was interrupted. On success, we print
228 * only its address, since kernel doesn't modify it,
229 * and printing the value may show uninitialized data.
231 switch (tcp->u_error) {
233 /* Not interrupted (slept entire interval) */
234 printaddr(tcp->u_arg[1]);
239 case ERESTART_RESTARTBLOCK:
241 print_timespec(tcp, tcp->u_arg[1]);
247 #include "xlat/itimer_which.h"
250 printitv_bitness(struct tcb *tcp, long addr, enum bitness_t bitness)
252 if (bitness == BITNESS_32 || current_time_t_is_compat) {
254 struct timeval32 it_interval, it_value;
257 if (!umove_or_printaddr(tcp, addr, &itv)) {
258 tprints("{it_interval=");
259 tprint_timeval32(tcp, &itv.it_interval);
260 tprints(", it_value=");
261 tprint_timeval32(tcp, &itv.it_value);
265 struct itimerval itv;
267 if (!umove_or_printaddr(tcp, addr, &itv)) {
268 tprints("{it_interval=");
269 tprint_timeval(tcp, &itv.it_interval);
270 tprints(", it_value=");
271 tprint_timeval(tcp, &itv.it_value);
277 #define printitv(tcp, addr) \
278 printitv_bitness((tcp), (addr), BITNESS_CURRENT)
283 printxval(itimer_which, tcp->u_arg[0], "ITIMER_???");
286 printitv(tcp, tcp->u_arg[1]);
292 SYS_FUNC(osf_getitimer)
295 printxval(itimer_which, tcp->u_arg[0], "ITIMER_???");
298 printitv_bitness(tcp, tcp->u_arg[1], BITNESS_32);
307 printxval(itimer_which, tcp->u_arg[0], "ITIMER_???");
309 printitv(tcp, tcp->u_arg[1]);
312 printitv(tcp, tcp->u_arg[2]);
318 SYS_FUNC(osf_setitimer)
321 printxval(itimer_which, tcp->u_arg[0], "ITIMER_???");
323 printitv_bitness(tcp, tcp->u_arg[1], BITNESS_32);
326 printitv_bitness(tcp, tcp->u_arg[2], BITNESS_32);
332 #include "xlat/adjtimex_modes.h"
333 #include "xlat/adjtimex_status.h"
334 #include "xlat/adjtimex_state.h"
336 #if SUPPORTED_PERSONALITIES > 1
338 tprint_timex32(struct tcb *tcp, long addr)
350 struct timeval32 time;
362 if (umove_or_printaddr(tcp, addr, &tx))
366 printflags(adjtimex_modes, tx.modes, "ADJ_???");
367 tprintf(", offset=%d, freq=%d, maxerror=%d, ",
368 tx.offset, tx.freq, tx.maxerror);
369 tprintf("esterror=%u, status=", tx.esterror);
370 printflags(adjtimex_status, tx.status, "STA_???");
371 tprintf(", constant=%d, precision=%u, ",
372 tx.constant, tx.precision);
373 tprintf("tolerance=%d, time=", tx.tolerance);
374 tprint_timeval32(tcp, &tx.time);
375 tprintf(", tick=%d, ppsfreq=%d, jitter=%d",
376 tx.tick, tx.ppsfreq, tx.jitter);
377 tprintf(", shift=%d, stabil=%d, jitcnt=%d",
378 tx.shift, tx.stabil, tx.jitcnt);
379 tprintf(", calcnt=%d, errcnt=%d, stbcnt=%d",
380 tx.calcnt, tx.errcnt, tx.stbcnt);
384 #endif /* SUPPORTED_PERSONALITIES > 1 */
387 tprint_timex(struct tcb *tcp, long addr)
391 #if SUPPORTED_PERSONALITIES > 1
392 if (current_time_t_is_compat)
393 return tprint_timex32(tcp, addr);
395 if (umove_or_printaddr(tcp, addr, &tx))
398 #if LINUX_VERSION_CODE < 66332
399 tprintf("{mode=%d, offset=%ld, frequency=%ld, ",
400 tx.mode, tx.offset, tx.frequency);
401 tprintf("maxerror=%ld, esterror=%lu, status=%u, ",
402 tx.maxerror, tx.esterror, tx.status);
403 tprintf("time_constant=%ld, precision=%lu, ",
404 tx.time_constant, tx.precision);
405 tprintf("tolerance=%ld, time=", tx.tolerance);
406 tprint_timeval(tcp, &tx.time);
409 printflags(adjtimex_modes, tx.modes, "ADJ_???");
410 tprintf(", offset=%jd, freq=%jd, maxerror=%ju, esterror=%ju, status=",
411 (intmax_t) tx.offset, (intmax_t) tx.freq,
412 (uintmax_t) tx.maxerror, (uintmax_t) tx.esterror);
413 printflags(adjtimex_status, tx.status, "STA_???");
414 tprintf(", constant=%jd, precision=%ju, tolerance=%jd, time=",
415 (intmax_t) tx.constant, (uintmax_t) tx.precision,
416 (intmax_t) tx.tolerance);
417 tprint_timeval(tcp, &tx.time);
418 tprintf(", tick=%jd, ppsfreq=%jd, jitter=%jd",
419 (intmax_t) tx.tick, (intmax_t) tx.ppsfreq, (intmax_t) tx.jitter);
420 tprintf(", shift=%d, stabil=%jd, jitcnt=%jd",
421 tx.shift, (intmax_t) tx.stabil, (intmax_t) tx.jitcnt);
422 tprintf(", calcnt=%jd, errcnt=%jd, stbcnt=%jd",
423 (intmax_t) tx.calcnt, (intmax_t) tx.errcnt, (intmax_t) tx.stbcnt);
430 do_adjtimex(struct tcb *tcp, long addr)
432 if (tprint_timex(tcp, addr))
434 tcp->auxstr = xlookup(adjtimex_state, tcp->u_rval);
443 return do_adjtimex(tcp, tcp->u_arg[0]);
447 #include "xlat/clockflags.h"
448 #include "xlat/clocknames.h"
451 printclockname(int clockid)
454 # include "xlat/cpuclocknames.h"
457 if ((clockid & CLOCKFD_MASK) == CLOCKFD)
458 tprintf("FD_TO_CLOCKID(%d)", CLOCKID_TO_FD(clockid));
460 if(CPUCLOCK_PERTHREAD(clockid))
461 tprintf("MAKE_THREAD_CPUCLOCK(%d,", CPUCLOCK_PID(clockid));
463 tprintf("MAKE_PROCESS_CPUCLOCK(%d,", CPUCLOCK_PID(clockid));
464 printxval(cpuclocknames, clockid & CLOCKFD_MASK, "CPUCLOCK_???");
470 printxval(clocknames, clockid, "CLOCK_???");
473 SYS_FUNC(clock_settime)
475 printclockname(tcp->u_arg[0]);
477 printtv(tcp, tcp->u_arg[1]);
482 SYS_FUNC(clock_gettime)
485 printclockname(tcp->u_arg[0]);
488 printtv(tcp, tcp->u_arg[1]);
493 SYS_FUNC(clock_nanosleep)
496 printclockname(tcp->u_arg[0]);
498 printflags(clockflags, tcp->u_arg[1], "TIMER_???");
500 printtv(tcp, tcp->u_arg[2]);
503 printtv(tcp, tcp->u_arg[3]);
508 SYS_FUNC(clock_adjtime)
511 return do_adjtimex(tcp, tcp->u_arg[1]);
512 printclockname(tcp->u_arg[0]);
517 #ifndef SIGEV_THREAD_ID
518 # define SIGEV_THREAD_ID 4
520 #include "xlat/sigev_value.h"
522 #if SUPPORTED_PERSONALITIES > 1
524 printsigevent32(struct tcb *tcp, long arg)
534 int function, attribute;
539 if (!umove_or_printaddr(tcp, arg, &sev)) {
540 tprintf("{%#x, ", sev.sigev_value);
541 if (sev.sigev_notify == SIGEV_SIGNAL)
542 tprintf("%s, ", signame(sev.sigev_signo));
544 tprintf("%u, ", sev.sigev_signo);
545 printxval(sigev_value, sev.sigev_notify, "SIGEV_???");
547 if (sev.sigev_notify == SIGEV_THREAD_ID)
548 tprintf("{%d}", sev.un.tid);
549 else if (sev.sigev_notify == SIGEV_THREAD)
550 tprintf("{%#x, %#x}",
551 sev.un.thread.function,
552 sev.un.thread.attribute);
561 printsigevent(struct tcb *tcp, long arg)
565 #if SUPPORTED_PERSONALITIES > 1
566 if (current_wordsize == 4) {
567 printsigevent32(tcp, arg);
571 if (!umove_or_printaddr(tcp, arg, &sev)) {
572 tprintf("{%p, ", sev.sigev_value.sival_ptr);
573 if (sev.sigev_notify == SIGEV_SIGNAL)
574 tprintf("%s, ", signame(sev.sigev_signo));
576 tprintf("%u, ", sev.sigev_signo);
577 printxval(sigev_value, sev.sigev_notify, "SIGEV_???");
579 if (sev.sigev_notify == SIGEV_THREAD_ID)
580 #if defined(HAVE_STRUCT_SIGEVENT__SIGEV_UN__PAD)
581 /* _pad[0] is the _tid field which might not be
582 present in the userlevel definition of the
584 tprintf("{%d}", sev._sigev_un._pad[0]);
585 #elif defined(HAVE_STRUCT_SIGEVENT___PAD)
586 tprintf("{%d}", sev.__pad[0]);
588 # warning unfamiliar struct sigevent => incomplete SIGEV_THREAD_ID decoding
591 else if (sev.sigev_notify == SIGEV_THREAD)
592 tprintf("{%p, %p}", sev.sigev_notify_function,
593 sev.sigev_notify_attributes);
600 SYS_FUNC(timer_create)
603 printclockname(tcp->u_arg[0]);
605 printsigevent(tcp, tcp->u_arg[1]);
608 printnum_int(tcp, tcp->u_arg[2], "%d");
613 SYS_FUNC(timer_settime)
616 tprintf("%d, ", (int) tcp->u_arg[0]);
617 printflags(clockflags, tcp->u_arg[1], "TIMER_???");
619 printitv(tcp, tcp->u_arg[2]);
622 printitv(tcp, tcp->u_arg[3]);
627 SYS_FUNC(timer_gettime)
630 tprintf("%d, ", (int) tcp->u_arg[0]);
632 printitv(tcp, tcp->u_arg[1]);
637 #include "xlat/timerfdflags.h"
641 /* It does not matter that the kernel uses itimerspec. */
642 tprintf("%ld, ", tcp->u_arg[0]);
643 printclockname(tcp->u_arg[0]);
645 printflags(timerfdflags, tcp->u_arg[2], "TFD_???");
647 printitv(tcp, tcp->u_arg[3]);
652 SYS_FUNC(timerfd_create)
654 printclockname(tcp->u_arg[0]);
656 printflags(timerfdflags, tcp->u_arg[1], "TFD_???");
661 SYS_FUNC(timerfd_settime)
663 printfd(tcp, tcp->u_arg[0]);
665 printflags(timerfdflags, tcp->u_arg[1], "TFD_???");
667 printitv(tcp, tcp->u_arg[2]);
669 printitv(tcp, tcp->u_arg[3]);
674 SYS_FUNC(timerfd_gettime)
677 printfd(tcp, tcp->u_arg[0]);
680 printitv(tcp, tcp->u_arg[1]);