/* Time inconsistency check test * by: john stultz (johnstul@us.ibm.com) * (C) Copyright IBM 2003, 2004, 2005 */ /* * Copyright (C) 2003-2006 IBM * * This program is free software; you can redistribute it and/or * modify it under the terms of the GNU General Public License as * published by the Free Software Foundation; either version 2 of the * License, or (at your option) any later version. * * This program is distributed in the hope that it will be useful, but * WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU * General Public License for more details. * * You should have received a copy of the GNU General Public License * along with this program; if not, write to the Free Software * Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA * 02111-1307, USA. */ #include <stdio.h> #include <time.h> #include <sys/time.h> #define CALLS_PER_LOOP 64 #define NSEC_PER_SEC 1000000000ULL /* returns 1 if a <= b, 0 otherwise */ static inline int in_order(struct timespec a, struct timespec b) { if (a.tv_sec < b.tv_sec) return 1; if (a.tv_sec > b.tv_sec) return 0; if (a.tv_nsec > b.tv_nsec) return 0; return 1; } int main(int argc, char *argv[]) { struct timespec list[CALLS_PER_LOOP]; int i, inconsistent; unsigned long seconds = -1; long now, then; int clock_type = CLOCK_MONOTONIC; if (argc > 1) seconds = atol(argv[1]); /* make sure CLOCK_MONOTONIC is supported */ if (clock_gettime(clock_type, &list[0])) { printf("Using CLOCK_REALTIME\n"); clock_type = CLOCK_REALTIME; } clock_gettime(clock_type, &list[0]); now = then = list[0].tv_sec; /* timestamp start of test */ system("date"); while (seconds == -1 || now - then < seconds) { inconsistent = 0; /* Fill list */ for (i = 0; i < CALLS_PER_LOOP; i++) clock_gettime(clock_type, &list[i]); /* Check for inconsistencies */ for (i = 0; i < CALLS_PER_LOOP - 1; i++) if (!in_order(list[i], list[i + 1])) inconsistent = i; /* display inconsistency */ if (inconsistent) { unsigned long long delta; for (i = 0; i < CALLS_PER_LOOP; i++) { if (i == inconsistent) printf("--------------------\n"); printf("%lu:%lu\n", list[i].tv_sec, list[i].tv_nsec); if (i == inconsistent + 1) printf("--------------------\n"); } delta = list[inconsistent].tv_sec * NSEC_PER_SEC; delta += list[inconsistent].tv_nsec; delta -= list[inconsistent + 1].tv_sec * NSEC_PER_SEC; delta -= list[inconsistent + 1].tv_nsec; printf("Delta: %llu ns\n", delta); fflush(0); /* timestamp inconsistency */ system("date"); return -1; } now = list[0].tv_sec; } return 0; }