This is an automated email from the git hooks/post-receive script.

jforbes pushed a commit to branch master
in repository kernel-tests.

commit 2947f58f098a4fdbe36bbaee850043f09bb384f0
Author: Justin M. Forbes <[email protected]>
Date:   Mon Feb 23 11:04:51 2015 -0600

    Add posix timer tests
---
 default/posix_timers/posix_timers.c | 223 ++++++++++++++++++++++++++++++++++++
 default/posix_timers/runtest.sh     |  20 ++++
 2 files changed, 243 insertions(+)

diff --git a/default/posix_timers/posix_timers.c 
b/default/posix_timers/posix_timers.c
new file mode 100644
index 0000000..d2227d4
--- /dev/null
+++ b/default/posix_timers/posix_timers.c
@@ -0,0 +1,223 @@
+/*
+ * Copyright (C) 2013 Red Hat, Inc., Frederic Weisbecker <[email protected]>
+ *
+ * Licensed under the terms of the GNU GPL License version 2
+ *
+ * Selftests for a few posix timers interface.
+ *
+ * Kernel loop code stolen from Steven Rostedt <[email protected]>
+ */
+
+#include <sys/time.h>
+#include <stdio.h>
+#include <signal.h>
+#include <unistd.h>
+#include <time.h>
+#include <pthread.h>
+
+#include "../../include/kselftest.h"
+
+#define DELAY 2
+#define USECS_PER_SEC 1000000
+
+static volatile int done;
+
+/* Busy loop in userspace to elapse ITIMER_VIRTUAL */
+static void user_loop(void)
+{
+       while (!done);
+}
+
+/*
+ * Try to spend as much time as possible in kernelspace
+ * to elapse ITIMER_PROF.
+ */
+static void kernel_loop(void)
+{
+       void *addr = sbrk(0);
+
+       while (!done) {
+               brk(addr + 4096);
+               brk(addr);
+       }
+}
+
+/*
+ * Sleep until ITIMER_REAL expiration.
+ */
+static void idle_loop(void)
+{
+       pause();
+}
+
+static void sig_handler(int nr)
+{
+       done = 1;
+}
+
+/*
+ * Check the expected timer expiration matches the GTOD elapsed delta since
+ * we armed the timer. Keep a 0.5 sec error margin due to various jitter.
+ */
+static int check_diff(struct timeval start, struct timeval end)
+{
+       long long diff;
+
+       diff = end.tv_usec - start.tv_usec;
+       diff += (end.tv_sec - start.tv_sec) * USECS_PER_SEC;
+
+       if (abs(diff - DELAY * USECS_PER_SEC) > USECS_PER_SEC / 2) {
+               printf("Diff too high: %lld..", diff);
+               return -1;
+       }
+
+       return 0;
+}
+
+static int check_itimer(int which)
+{
+       int err;
+       struct timeval start, end;
+       struct itimerval val = {
+               .it_value.tv_sec = DELAY,
+       };
+
+       printf("Check itimer ");
+
+       if (which == ITIMER_VIRTUAL)
+               printf("virtual... ");
+       else if (which == ITIMER_PROF)
+               printf("prof... ");
+       else if (which == ITIMER_REAL)
+               printf("real... ");
+
+       fflush(stdout);
+
+       done = 0;
+
+       if (which == ITIMER_VIRTUAL)
+               signal(SIGVTALRM, sig_handler);
+       else if (which == ITIMER_PROF)
+               signal(SIGPROF, sig_handler);
+       else if (which == ITIMER_REAL)
+               signal(SIGALRM, sig_handler);
+
+       err = gettimeofday(&start, NULL);
+       if (err < 0) {
+               perror("Can't call gettimeofday()\n");
+               return -1;
+       }
+
+       err = setitimer(which, &val, NULL);
+       if (err < 0) {
+               perror("Can't set timer\n");
+               return -1;
+       }
+
+       if (which == ITIMER_VIRTUAL)
+               user_loop();
+       else if (which == ITIMER_PROF)
+               kernel_loop();
+       else if (which == ITIMER_REAL)
+               idle_loop();
+
+       gettimeofday(&end, NULL);
+       if (err < 0) {
+               perror("Can't call gettimeofday()\n");
+               return -1;
+       }
+
+       if (!check_diff(start, end))
+               printf("[OK]\n");
+       else
+               printf("[FAIL]\n");
+
+       return 0;
+}
+
+static int check_timer_create(int which)
+{
+       int err;
+       timer_t id;
+       struct timeval start, end;
+       struct itimerspec val = {
+               .it_value.tv_sec = DELAY,
+       };
+
+       printf("Check timer_create() ");
+       if (which == CLOCK_THREAD_CPUTIME_ID) {
+               printf("per thread... ");
+       } else if (which == CLOCK_PROCESS_CPUTIME_ID) {
+               printf("per process... ");
+       }
+       fflush(stdout);
+
+       done = 0;
+       err = timer_create(which, NULL, &id);
+       if (err < 0) {
+               perror("Can't create timer\n");
+               return -1;
+       }
+       signal(SIGALRM, sig_handler);
+
+       err = gettimeofday(&start, NULL);
+       if (err < 0) {
+               perror("Can't call gettimeofday()\n");
+               return -1;
+       }
+
+       err = timer_settime(id, 0, &val, NULL);
+       if (err < 0) {
+               perror("Can't set timer\n");
+               return -1;
+       }
+
+       user_loop();
+
+       gettimeofday(&end, NULL);
+       if (err < 0) {
+               perror("Can't call gettimeofday()\n");
+               return -1;
+       }
+
+       if (!check_diff(start, end))
+               printf("[OK]\n");
+       else
+               printf("[FAIL]\n");
+
+       return 0;
+}
+
+int main(int argc, char **argv)
+{
+       int err;
+
+       printf("Testing posix timers. False negative may happen on CPU 
execution \n");
+       printf("based timers if other threads run on the CPU...\n");
+
+       if (check_itimer(ITIMER_VIRTUAL) < 0)
+               return ksft_exit_fail();
+
+       if (check_itimer(ITIMER_PROF) < 0)
+               return ksft_exit_fail();
+
+       if (check_itimer(ITIMER_REAL) < 0)
+               return ksft_exit_fail();
+
+       if (check_timer_create(CLOCK_THREAD_CPUTIME_ID) < 0)
+               return ksft_exit_fail();
+
+       /*
+        * It's unfortunately hard to reliably test a timer expiration
+        * on parallel multithread cputime. We could arm it to expire
+        * on DELAY * nr_threads, with nr_threads busy looping, then wait
+        * the normal DELAY since the time is elapsing nr_threads faster.
+        * But for that we need to ensure we have real physical free CPUs
+        * to ensure true parallelism. So test only one thread until we
+        * find a better solution.
+        */
+       if (check_timer_create(CLOCK_PROCESS_CPUTIME_ID) < 0)
+               return ksft_exit_fail();
+
+       return ksft_exit_pass();
+}
diff --git a/default/posix_timers/runtest.sh b/default/posix_timers/runtest.sh
new file mode 100755
index 0000000..95de0d0
--- /dev/null
+++ b/default/posix_timers/runtest.sh
@@ -0,0 +1,20 @@
+#!/bin/sh
+#
+# Licensed under the terms of the GNU GPL License version 2
+
+# Test posix timers
+
+# Build source if it is not already built
+if [ ! -f ./posix_timers ]; then
+       gcc -o posix_timers posix_timers.c -lrt
+       if [ "$?" -ne "0" ]; then
+               echo "posix_timers build failed."
+               exit 3
+       fi
+fi
+
+# Run the test
+./posix_timers
+if [ "$?" -ne "0" ]; then
+       exit -1
+fi

-- 
To stop receiving notification emails like this one, please contact
the administrator of this repository.
_______________________________________________
kernel mailing list
[email protected]
https://lists.fedoraproject.org/admin/lists/[email protected]

Reply via email to