/* * High resolution timer test software * * (C) 2005-2007 Thomas Gleixner <tglx@linutronix.de> * * This program is free software; you can redistribute it and/or * modify it under the terms of the GNU General Public License Version * 2 as published by the Free Software Foundation. * */ #define VERSION_STRING "V 0.15" #include <fcntl.h> #include <getopt.h> #include <pthread.h> #include <signal.h> #include <stdlib.h> #include <stdio.h> #include <string.h> #include <time.h> #include <unistd.h> #include <linux/unistd.h> #include <sys/prctl.h> #include <sys/stat.h> #include <sys/types.h> #include <sys/time.h> #define ARRAY_SIZE(x) (sizeof(x) / sizeof((x)[0])) /* Ugly, but .... */ #define gettid() syscall(__NR_gettid) #define sigev_notify_thread_id _sigev_un._tid extern int clock_nanosleep(clockid_t __clock_id, int __flags, __const struct timespec *__req, struct timespec *__rem); #define USEC_PER_SEC 1000000 #define NSEC_PER_SEC 1000000000 #define MODE_CYCLIC 0 #define MODE_CLOCK_NANOSLEEP 1 #define MODE_SYS_ITIMER 2 #define MODE_SYS_NANOSLEEP 3 #define MODE_SYS_OFFSET 2 #define TIMER_RELTIME 0 /* Must be power of 2 ! */ #define VALBUF_SIZE 16384 #define KVARS 32 #define KVARNAMELEN 32 /* Struct to transfer parameters to the thread */ struct thread_param { int prio; int mode; int timermode; int signal; int clock; unsigned long max_cycles; struct thread_stat *stats; int bufmsk; unsigned long interval; }; /* Struct for statistics */ struct thread_stat { unsigned long cycles; unsigned long cyclesread; long min; long max; long act; double avg; long *values; pthread_t thread; int threadstarted; int tid; }; static int shutdown; static int tracelimit = 0; static int ftrace = 0; static int oldtrace = 0; /* Backup of kernel variables that we modify */ static struct kvars { char name[KVARNAMELEN]; int value; } kv[KVARS]; static char *procfileprefix = "/proc/sys/kernel/"; static int kernvar(int mode, char *name, int *value) { int retval = 1; int procfilepath; char procfilename[128]; strncpy(procfilename, procfileprefix, sizeof(procfilename)); strncat(procfilename, name, sizeof(procfilename) - sizeof(procfileprefix)); procfilepath = open(procfilename, mode); if (procfilepath >= 0) { char buffer[32]; if (mode == O_RDONLY) { if (read(procfilepath, buffer, sizeof(buffer)) > 0) { char *endptr; *value = strtol(buffer, &endptr, 0); if (endptr != buffer) retval = 0; } } else if (mode == O_WRONLY) { snprintf(buffer, sizeof(buffer), "%d\n", *value); if (write(procfilepath, buffer, strlen(buffer)) == strlen(buffer)) retval = 0; } close(procfilepath); } return retval; } static void setkernvar(char *name, int value) { int i; int oldvalue; if (kernvar(O_RDONLY, name, &oldvalue)) fprintf(stderr, "could not retrieve %s\n", name); else { for (i = 0; i < KVARS; i++) { if (!strcmp(kv[i].name, name)) break; if (kv[i].name[0] == '\0') { strncpy(kv[i].name, name, sizeof(kv[i].name)); kv[i].value = oldvalue; break; } } if (i == KVARS) fprintf(stderr, "could not backup %s (%d)\n", name, oldvalue); } if (kernvar(O_WRONLY, name, &value)) fprintf(stderr, "could not set %s to %d\n", name, value); } static void restorekernvars(void) { int i; for (i = 0; i < KVARS; i++) { if (kv[i].name[0] != '\0') { if (kernvar(O_WRONLY, kv[i].name, &kv[i].value)) fprintf(stderr, "could not restore %s to %d\n", kv[i].name, kv[i].value); } } } static inline void tsnorm(struct timespec *ts) { while (ts->tv_nsec >= NSEC_PER_SEC) { ts->tv_nsec -= NSEC_PER_SEC; ts->tv_sec++; } } static inline long calcdiff(struct timespec t1, struct timespec t2) { long diff; diff = USEC_PER_SEC * ((int) t1.tv_sec - (int) t2.tv_sec); diff += ((int) t1.tv_nsec - (int) t2.tv_nsec) / 1000; return diff; } /* * timer thread * * Modes: * - clock_nanosleep based * - cyclic timer based * * Clock: * - CLOCK_MONOTONIC * - CLOCK_REALTIME * - CLOCK_MONOTONIC_HR * - CLOCK_REALTIME_HR * */ void *timerthread(void *param) { struct thread_param *par = param; struct sched_param schedp; struct sigevent sigev; sigset_t sigset; timer_t timer; struct timespec now, next, interval; struct itimerval itimer; struct itimerspec tspec; struct thread_stat *stat = par->stats; int policy = par->prio ? SCHED_FIFO : SCHED_OTHER; int stopped = 0; interval.tv_sec = par->interval / USEC_PER_SEC; interval.tv_nsec = (par->interval % USEC_PER_SEC) * 1000; if (tracelimit) { setkernvar("trace_all_cpus", 1); setkernvar("trace_freerunning", 1); setkernvar("trace_print_on_crash", 0); setkernvar("trace_user_triggered", 1); setkernvar("trace_user_trigger_irq", -1); setkernvar("trace_verbose", 0); setkernvar("preempt_thresh", 0); setkernvar("wakeup_timing", 0); setkernvar("preempt_max_latency", 0); if (ftrace) setkernvar("mcount_enabled", 1); setkernvar("trace_enabled", 1); } stat->tid = gettid(); sigemptyset(&sigset); sigaddset(&sigset, par->signal); sigprocmask(SIG_BLOCK, &sigset, NULL); if (par->mode == MODE_CYCLIC) { sigev.sigev_notify = SIGEV_THREAD_ID | SIGEV_SIGNAL; sigev.sigev_signo = par->signal; sigev.sigev_notify_thread_id = stat->tid; timer_create(par->clock, &sigev, &timer); tspec.it_interval = interval; } memset(&schedp, 0, sizeof(schedp)); schedp.sched_priority = par->prio; sched_setscheduler(0, policy, &schedp); /* Get current time */ clock_gettime(par->clock, &now); next = now; next.tv_sec++; if (par->mode == MODE_CYCLIC) { if (par->timermode == TIMER_ABSTIME) tspec.it_value = next; else { tspec.it_value.tv_nsec = 0; tspec.it_value.tv_sec = 1; } timer_settime(timer, par->timermode, &tspec, NULL); } if (par->mode == MODE_SYS_ITIMER) { itimer.it_value.tv_sec = 1; itimer.it_value.tv_usec = 0; itimer.it_interval.tv_sec = interval.tv_sec; itimer.it_interval.tv_usec = interval.tv_nsec / 1000; setitimer (ITIMER_REAL, &itimer, NULL); } stat->threadstarted++; if (tracelimit) { if (oldtrace) gettimeofday(0,(struct timezone *)1); else prctl(0, 1); } while (!shutdown) { long diff; int sigs; /* Wait for next period */ switch (par->mode) { case MODE_CYCLIC: case MODE_SYS_ITIMER: if (sigwait(&sigset, &sigs) < 0) goto out; break; case MODE_CLOCK_NANOSLEEP: if (par->timermode == TIMER_ABSTIME) clock_nanosleep(par->clock, TIMER_ABSTIME, &next, NULL); else { clock_gettime(par->clock, &now); clock_nanosleep(par->clock, TIMER_RELTIME, &interval, NULL); next.tv_sec = now.tv_sec + interval.tv_sec; next.tv_nsec = now.tv_nsec + interval.tv_nsec; tsnorm(&next); } break; case MODE_SYS_NANOSLEEP: clock_gettime(par->clock, &now); nanosleep(&interval, NULL); next.tv_sec = now.tv_sec + interval.tv_sec; next.tv_nsec = now.tv_nsec + interval.tv_nsec; tsnorm(&next); break; } clock_gettime(par->clock, &now); diff = calcdiff(now, next); if (diff < stat->min) stat->min = diff; if (diff > stat->max) stat->max = diff; stat->avg += (double) diff; if (!stopped && tracelimit && (diff > tracelimit)) { stopped++; if (oldtrace) gettimeofday(0,0); else prctl(0, 0); shutdown++; } stat->act = diff; stat->cycles++; if (par->bufmsk) stat->values[stat->cycles & par->bufmsk] = diff; next.tv_sec += interval.tv_sec; next.tv_nsec += interval.tv_nsec; tsnorm(&next); if (par->max_cycles && par->max_cycles == stat->cycles) break; } out: if (par->mode == MODE_CYCLIC) timer_delete(timer); if (par->mode == MODE_SYS_ITIMER) { itimer.it_value.tv_sec = 0; itimer.it_value.tv_usec = 0; itimer.it_interval.tv_sec = 0; itimer.it_interval.tv_usec = 0; setitimer (ITIMER_REAL, &itimer, NULL); } /* switch to normal */ schedp.sched_priority = 0; sched_setscheduler(0, SCHED_OTHER, &schedp); stat->threadstarted = -1; return NULL; } /* Print usage information */ static void display_help(void) { printf("cyclictest %s\n", VERSION_STRING); printf("Usage:\n" "cyclictest <options>\n\n" "-b USEC --breaktrace=USEC send break trace command when latency > USEC\n" "-c CLOCK --clock=CLOCK select clock\n" " 0 = CLOCK_MONOTONIC (default)\n" " 1 = CLOCK_REALTIME\n" "-d DIST --distance=DIST distance of thread intervals in us default=500\n" "-f function trace (when -b is active)\n" "-i INTV --interval=INTV base interval of thread in us default=1000\n" "-l LOOPS --loops=LOOPS number of loops: default=0(endless)\n" "-n --nanosleep use clock_nanosleep\n" "-p PRIO --prio=PRIO priority of highest prio thread\n" "-q --quiet print only a summary on exit\n" "-r --relative use relative timer instead of absolute\n" "-s --system use sys_nanosleep and sys_setitimer\n" "-t NUM --threads=NUM number of threads: default=1\n" "-v --verbose output values on stdout for statistics\n" " format: n:c:v n=tasknum c=count v=value in us\n"); exit(0); } static int use_nanosleep; static int timermode = TIMER_ABSTIME; static int use_system; static int priority; static int num_threads = 1; static int max_cycles; static int clocksel = 0; static int verbose; static int quiet; static int interval = 1000; static int distance = 500; static int clocksources[] = { CLOCK_MONOTONIC, CLOCK_REALTIME, }; /* Process commandline options */ static void process_options (int argc, char *argv[]) { int error = 0; for (;;) { int option_index = 0; /** Options for getopt */ static struct option long_options[] = { {"breaktrace", required_argument, NULL, 'b'}, {"clock", required_argument, NULL, 'c'}, {"distance", required_argument, NULL, 'd'}, {"ftrace", no_argument, NULL, 'f'}, {"interval", required_argument, NULL, 'i'}, {"loops", required_argument, NULL, 'l'}, {"nanosleep", no_argument, NULL, 'n'}, {"priority", required_argument, NULL, 'p'}, {"quiet", no_argument, NULL, 'q'}, {"relative", no_argument, NULL, 'r'}, {"system", no_argument, NULL, 's'}, {"threads", required_argument, NULL, 't'}, {"verbose", no_argument, NULL, 'v'}, {"help", no_argument, NULL, '?'}, {NULL, 0, NULL, 0} }; int c = getopt_long (argc, argv, "b:c:d:fi:l:np:qrst:v", long_options, &option_index); if (c == -1) break; switch (c) { case 'b': tracelimit = atoi(optarg); break; case 'c': clocksel = atoi(optarg); break; case 'd': distance = atoi(optarg); break; case 'f': ftrace = 1; break; case 'i': interval = atoi(optarg); break; case 'l': max_cycles = atoi(optarg); break; case 'n': use_nanosleep = MODE_CLOCK_NANOSLEEP; break; case 'p': priority = atoi(optarg); break; case 'q': quiet = 1; break; case 'r': timermode = TIMER_RELTIME; break; case 's': use_system = MODE_SYS_OFFSET; break; case 't': num_threads = atoi(optarg); break; case 'v': verbose = 1; break; case '?': error = 1; break; } } if (clocksel < 0 || clocksel > ARRAY_SIZE(clocksources)) error = 1; if (priority < 0 || priority > 99) error = 1; if (num_threads < 1) error = 1; if (error) display_help (); } static void check_kernel(void) { size_t len; char ver[256]; int fd, maj, min, sub; fd = open("/proc/version", O_RDONLY, 0666); len = read(fd, ver, 255); close(fd); ver[len-1] = 0x0; sscanf(ver, "Linux version %d.%d.%d", &maj, &min, &sub); if (maj == 2 && min == 6 && sub < 18) oldtrace = 1; } static int check_timer(void) { struct timespec ts; if (clock_getres(CLOCK_MONOTONIC, &ts)) return 1; return (ts.tv_sec != 0 || ts.tv_nsec != 1); } static void sighand(int sig) { shutdown = 1; } static void print_stat(struct thread_param *par, int index, int verbose) { struct thread_stat *stat = par->stats; if (!verbose) { if (quiet != 1) { printf("T:%2d (%5d) P:%2d I:%ld C:%7lu " "Min:%7ld Act:%5ld Avg:%5ld Max:%8ld\n", index, stat->tid, par->prio, par->interval, stat->cycles, stat->min, stat->act, stat->cycles ? (long)(stat->avg/stat->cycles) : 0, stat->max); } } else { while (stat->cycles != stat->cyclesread) { long diff = stat->values[stat->cyclesread & par->bufmsk]; printf("%8d:%8lu:%8ld\n", index, stat->cyclesread, diff); stat->cyclesread++; } } } int main(int argc, char **argv) { sigset_t sigset; int signum = SIGALRM; int mode; struct thread_param *par; struct thread_stat *stat; int i, ret = -1; if (geteuid()) { fprintf(stderr, "cyclictest: need to run as root!\n"); exit(-1); } process_options(argc, argv); check_kernel(); if (check_timer()) fprintf(stderr, "WARNING: High resolution timers not available\n"); mode = use_nanosleep + use_system; sigemptyset(&sigset); sigaddset(&sigset, signum); sigprocmask (SIG_BLOCK, &sigset, NULL); signal(SIGINT, sighand); signal(SIGTERM, sighand); par = calloc(num_threads, sizeof(struct thread_param)); if (!par) goto out; stat = calloc(num_threads, sizeof(struct thread_stat)); if (!stat) goto outpar; for (i = 0; i < num_threads; i++) { if (verbose) { stat[i].values = calloc(VALBUF_SIZE, sizeof(long)); if (!stat[i].values) goto outall; par[i].bufmsk = VALBUF_SIZE - 1; } par[i].prio = priority; if (priority) priority--; par[i].clock = clocksources[clocksel]; par[i].mode = mode; par[i].timermode = timermode; par[i].signal = signum; par[i].interval = interval; interval += distance; par[i].max_cycles = max_cycles; par[i].stats = &stat[i]; stat[i].min = 1000000; stat[i].max = -1000000; stat[i].avg = 0.0; pthread_create(&stat[i].thread, NULL, timerthread, &par[i]); stat[i].threadstarted = 1; } while (!shutdown) { char lavg[256]; int fd, len, allstopped = 0; if (!verbose && !quiet) { fd = open("/proc/loadavg", O_RDONLY, 0666); len = read(fd, &lavg, 255); close(fd); lavg[len-1] = 0x0; printf("%s \n\n", lavg); } for (i = 0; i < num_threads; i++) { print_stat(&par[i], i, verbose); if(max_cycles && stat[i].cycles >= max_cycles) allstopped++; } usleep(10000); if (shutdown || allstopped) break; if (!verbose && !quiet) printf("\033[%dA", num_threads + 2); } ret = 0; outall: shutdown = 1; usleep(50000); if (quiet) quiet = 2; for (i = 0; i < num_threads; i++) { if (stat[i].threadstarted > 0) pthread_kill(stat[i].thread, SIGTERM); if (stat[i].threadstarted) { pthread_join(stat[i].thread, NULL); if (quiet) print_stat(&par[i], i, 0); } if (stat[i].values) free(stat[i].values); } free(stat); outpar: free(par); out: /* Be a nice program, cleanup */ restorekernvars(); exit(ret); }