]> git.saurik.com Git - apple/xnu.git/blobdiff - tools/tests/zero-to-n/zero-to-n.c
xnu-6153.141.1.tar.gz
[apple/xnu.git] / tools / tests / zero-to-n / zero-to-n.c
index 87ce83bb7b7e2b3541243a94693f9e2eb88923e4..db9ed81c5d6b3f6a002348b301029d8b67cb6aa7 100644 (file)
@@ -2,7 +2,7 @@
  * Copyright (c) 2009 Apple Inc. All rights reserved.
  *
  * @APPLE_OSREFERENCE_LICENSE_HEADER_START@
- * 
+ *
  * This file contains Original Code and/or Modifications of Original Code
  * as defined in and that are subject to the Apple Public Source License
  * Version 2.0 (the 'License'). You may not use this file except in
  * unlawful or unlicensed copies of an Apple operating system, or to
  * circumvent, violate, or enable the circumvention or violation of, any
  * terms of an Apple operating system software license agreement.
- * 
+ *
  * Please obtain a copy of the License at
  * http://www.opensource.apple.com/apsl/ and read it before using this file.
- * 
+ *
  * The Original Code and all software distributed under the License are
  * distributed on an 'AS IS' basis, WITHOUT WARRANTY OF ANY KIND, EITHER
  * EXPRESS OR IMPLIED, AND APPLE HEREBY DISCLAIMS ALL SUCH WARRANTIES,
@@ -22,7 +22,7 @@
  * FITNESS FOR A PARTICULAR PURPOSE, QUIET ENJOYMENT OR NON-INFRINGEMENT.
  * Please see the License for the specific language governing rights and
  * limitations under the License.
- * 
+ *
  * @APPLE_OSREFERENCE_LICENSE_HEADER_END@
  */
 #include <unistd.h>
@@ -44,8 +44,6 @@
 #include <sys/spawn_internal.h>
 #include <mach-o/dyld.h>
 
-#include <libkern/OSAtomic.h>
-
 #include <mach/mach_time.h>
 #include <mach/mach.h>
 #include <mach/task.h>
 
 #include <sys/resource.h>
 
+#include <stdatomic.h>
+
+#include <os/tsd.h>
+
 typedef enum wake_type { WAKE_BROADCAST_ONESEM, WAKE_BROADCAST_PERTHREAD, WAKE_CHAIN, WAKE_HOP } wake_type_t;
 typedef enum my_policy_type { MY_POLICY_REALTIME, MY_POLICY_TIMESHARE, MY_POLICY_FIXEDPRI } my_policy_type_t;
 
@@ -62,14 +64,16 @@ typedef enum my_policy_type { MY_POLICY_REALTIME, MY_POLICY_TIMESHARE, MY_POLICY
 #define mach_assert_zero_t(tid, error) do { if ((error) != 0) { fprintf(stderr, "[FAIL] Thread %d error %d (%s) ", (tid), (error), mach_error_string(error)); assert(error == 0); } } while (0)
 #define assert_zero_t(tid, error)      do { if ((error) != 0) { fprintf(stderr, "[FAIL] Thread %d error %d ", (tid), (error)); assert(error == 0); } } while (0)
 
-#define CONSTRAINT_NANOS       (20000000ll)    /* 20 ms */
-#define COMPUTATION_NANOS      (10000000ll)    /* 10 ms */
-#define TRACEWORTHY_NANOS      (10000000ll)    /* 10 ms */
+#define CONSTRAINT_NANOS        (20000000ll)    /* 20 ms */
+#define COMPUTATION_NANOS       (10000000ll)    /* 10 ms */
+#define RT_CHURN_COMP_NANOS     ( 1000000ll)    /*  1 ms */
+#define TRACEWORTHY_NANOS       (10000000ll)    /* 10 ms */
+#define TRACEWORTHY_NANOS_TEST  ( 2000000ll)    /*  2 ms */
 
 #if DEBUG
-#define debug_log(args...) printf(args)
+#define debug_log(args ...) printf(args)
 #else
-#define debug_log(args...) do { } while(0)
+#define debug_log(args ...) do { } while(0)
 #endif
 
 /* Declarations */
@@ -80,8 +84,14 @@ static my_policy_type_t         parse_thread_policy(const char *str);
 static void                     selfexec_with_apptype(int argc, char *argv[]);
 static void                     parse_args(int argc, char *argv[]);
 
+static __attribute__((aligned(128))) _Atomic uint32_t   g_done_threads;
+static __attribute__((aligned(128))) _Atomic boolean_t  g_churn_stop = FALSE;
+static __attribute__((aligned(128))) _Atomic uint64_t   g_churn_stopped_at = 0;
+
 /* Global variables (general) */
 static uint32_t                 g_numcpus;
+static uint32_t                 g_nphysicalcpu;
+static uint32_t                 g_nlogicalcpu;
 static uint32_t                 g_numthreads;
 static wake_type_t              g_waketype;
 static policy_t                 g_policy;
@@ -89,7 +99,6 @@ static uint32_t                 g_iterations;
 static struct mach_timebase_info g_mti;
 static semaphore_t              g_main_sem;
 static uint64_t                *g_thread_endtimes_abs;
-static volatile uint32_t        g_done_threads;
 static boolean_t                g_verbose       = FALSE;
 static boolean_t                g_do_affinity   = FALSE;
 static uint64_t                 g_starttime_abs;
@@ -97,10 +106,10 @@ static uint32_t                 g_iteration_sleeptime_us = 0;
 static uint32_t                 g_priority = 0;
 static uint32_t                 g_churn_pri = 0;
 static uint32_t                 g_churn_count = 0;
-static uint64_t                 g_churn_stopped_at = 0;
-static boolean_t                g_churn_stop = FALSE;
+static uint32_t                 g_rt_churn_count = 0;
 
 static pthread_t*               g_churn_threads = NULL;
+static pthread_t*               g_rt_churn_threads = NULL;
 
 /* Threshold for dropping a 'bad run' tracepoint */
 static uint64_t                 g_traceworthy_latency_ns = TRACEWORTHY_NANOS;
@@ -117,6 +126,20 @@ static boolean_t                g_do_all_spin = FALSE;
 /* Every thread backgrounds temporarily before parking */
 static boolean_t                g_drop_priority = FALSE;
 
+/* Test whether realtime threads are scheduled on the separate CPUs */
+static boolean_t                g_test_rt = FALSE;
+
+static boolean_t                g_rt_churn = FALSE;
+
+/* On SMT machines, test whether realtime threads are scheduled on the correct CPUs */
+static boolean_t                g_test_rt_smt = FALSE;
+
+/* Test whether realtime threads are successfully avoiding CPU 0 on Intel */
+static boolean_t                g_test_rt_avoid0 = FALSE;
+
+/* Print a histgram showing how many threads ran on each CPU */
+static boolean_t                g_histogram = FALSE;
+
 /* One randomly chosen thread holds up the train for a certain duration. */
 static boolean_t                g_do_one_long_spin = FALSE;
 static uint32_t                 g_one_long_spin_id = 0;
@@ -133,10 +156,20 @@ static semaphore_t              g_broadcastsem;
 static semaphore_t              g_leadersem;
 static semaphore_t              g_readysem;
 static semaphore_t              g_donesem;
+static semaphore_t              g_rt_churn_sem;
+static semaphore_t              g_rt_churn_start_sem;
 
 /* Global variables (chain) */
 static semaphore_t             *g_semarr;
 
+typedef struct {
+       __attribute__((aligned(128))) uint32_t current;
+       uint32_t accum;
+} histogram_t;
+
+static histogram_t             *g_cpu_histogram;
+static _Atomic uint64_t        *g_cpu_map;
+
 static uint64_t
 abs_to_nanos(uint64_t abstime)
 {
@@ -152,8 +185,10 @@ nanos_to_abs(uint64_t ns)
 inline static void
 yield(void)
 {
-#if   defined(__x86_64__) || defined(__i386__)
-       asm volatile("pause");
+#if defined(__arm__) || defined(__arm64__)
+       asm volatile ("yield");
+#elif defined(__x86_64__) || defined(__i386__)
+       asm volatile ("pause");
 #else
 #error Unrecognized architecture
 #endif
@@ -170,13 +205,13 @@ churn_thread(__unused void *arg)
         */
 
        while (g_churn_stop == FALSE &&
-              mach_absolute_time() < (g_starttime_abs + NSEC_PER_SEC)) {
+           mach_absolute_time() < (g_starttime_abs + NSEC_PER_SEC)) {
                spin_count++;
                yield();
        }
 
        /* This is totally racy, but only here to detect if anyone stops early */
-       g_churn_stopped_at += spin_count;
+       atomic_fetch_add_explicit(&g_churn_stopped_at, spin_count, memory_order_relaxed);
 
        return NULL;
 }
@@ -184,8 +219,9 @@ churn_thread(__unused void *arg)
 static void
 create_churn_threads()
 {
-       if (g_churn_count == 0)
+       if (g_churn_count == 0) {
                g_churn_count = g_numcpus - 1;
+       }
 
        errno_t err;
 
@@ -196,47 +232,176 @@ create_churn_threads()
        g_churn_threads = (pthread_t*) valloc(sizeof(pthread_t) * g_churn_count);
        assert(g_churn_threads);
 
-       if ((err = pthread_attr_init(&attr)))
+       if ((err = pthread_attr_init(&attr))) {
                errc(EX_OSERR, err, "pthread_attr_init");
+       }
 
-       if ((err = pthread_attr_setschedparam(&attr, &param)))
+       if ((err = pthread_attr_setschedparam(&attr, &param))) {
                errc(EX_OSERR, err, "pthread_attr_setschedparam");
+       }
 
-       if ((err = pthread_attr_setschedpolicy(&attr, SCHED_RR)))
+       if ((err = pthread_attr_setschedpolicy(&attr, SCHED_RR))) {
                errc(EX_OSERR, err, "pthread_attr_setschedpolicy");
+       }
 
-       for (uint32_t i = 0 ; i < g_churn_count ; i++) {
+       for (uint32_t i = 0; i < g_churn_count; i++) {
                pthread_t new_thread;
 
-               if ((err = pthread_create(&new_thread, &attr, churn_thread, NULL)))
+               if ((err = pthread_create(&new_thread, &attr, churn_thread, NULL))) {
                        errc(EX_OSERR, err, "pthread_create");
+               }
                g_churn_threads[i] = new_thread;
        }
 
-       if ((err = pthread_attr_destroy(&attr)))
+       if ((err = pthread_attr_destroy(&attr))) {
                errc(EX_OSERR, err, "pthread_attr_destroy");
+       }
 }
 
 static void
 join_churn_threads(void)
 {
-       if (g_churn_stopped_at != 0)
+       if (atomic_load_explicit(&g_churn_stopped_at, memory_order_seq_cst) != 0) {
                printf("Warning: Some of the churn threads may have stopped early: %lld\n",
-                      g_churn_stopped_at);
-
-       OSMemoryBarrier();
+                   g_churn_stopped_at);
+       }
 
-       g_churn_stop = TRUE;
+       atomic_store_explicit(&g_churn_stop, TRUE, memory_order_seq_cst);
 
        /* Rejoin churn threads */
        for (uint32_t i = 0; i < g_churn_count; i++) {
                errno_t err = pthread_join(g_churn_threads[i], NULL);
-               if (err) errc(EX_OSERR, err, "pthread_join %d", i);
+               if (err) {
+                       errc(EX_OSERR, err, "pthread_join %d", i);
+               }
        }
 }
 
 /*
- * Figure out what thread policy to use 
+ * Set policy
+ */
+static int
+rt_churn_thread_setup(void)
+{
+       kern_return_t kr;
+       thread_time_constraint_policy_data_t pol;
+
+       /* Hard-coded realtime parameters (similar to what Digi uses) */
+       pol.period      = 100000;
+       pol.constraint  = (uint32_t) nanos_to_abs(CONSTRAINT_NANOS * 2);
+       pol.computation = (uint32_t) nanos_to_abs(RT_CHURN_COMP_NANOS * 2);
+       pol.preemptible = 0;         /* Ignored by OS */
+
+       kr = thread_policy_set(mach_thread_self(), THREAD_TIME_CONSTRAINT_POLICY,
+           (thread_policy_t) &pol, THREAD_TIME_CONSTRAINT_POLICY_COUNT);
+       mach_assert_zero_t(0, kr);
+
+       return 0;
+}
+
+static void *
+rt_churn_thread(__unused void *arg)
+{
+       rt_churn_thread_setup();
+
+       for (uint32_t i = 0; i < g_iterations; i++) {
+               kern_return_t kr = semaphore_wait_signal(g_rt_churn_start_sem, g_rt_churn_sem);
+               mach_assert_zero_t(0, kr);
+
+               volatile double x = 0.0;
+               volatile double y = 0.0;
+
+               uint64_t endspin = mach_absolute_time() + nanos_to_abs(RT_CHURN_COMP_NANOS);
+               while (mach_absolute_time() < endspin) {
+                       y = y + 1.5 + x;
+                       x = sqrt(y);
+               }
+       }
+
+       kern_return_t kr = semaphore_signal(g_rt_churn_sem);
+       mach_assert_zero_t(0, kr);
+
+       return NULL;
+}
+
+static void
+wait_for_rt_churn_threads(void)
+{
+       for (uint32_t i = 0; i < g_rt_churn_count; i++) {
+               kern_return_t kr = semaphore_wait(g_rt_churn_sem);
+               mach_assert_zero_t(0, kr);
+       }
+}
+
+static void
+start_rt_churn_threads(void)
+{
+       for (uint32_t i = 0; i < g_rt_churn_count; i++) {
+               kern_return_t kr = semaphore_signal(g_rt_churn_start_sem);
+               mach_assert_zero_t(0, kr);
+       }
+}
+
+static void
+create_rt_churn_threads(void)
+{
+       if (g_rt_churn_count == 0) {
+               /* Leave 1 CPU to ensure that the main thread can make progress */
+               g_rt_churn_count = g_numcpus - 1;
+       }
+
+       errno_t err;
+
+       struct sched_param param = { .sched_priority = (int)g_churn_pri };
+       pthread_attr_t attr;
+
+       /* Array for churn threads */
+       g_rt_churn_threads = (pthread_t*) valloc(sizeof(pthread_t) * g_rt_churn_count);
+       assert(g_rt_churn_threads);
+
+       if ((err = pthread_attr_init(&attr))) {
+               errc(EX_OSERR, err, "pthread_attr_init");
+       }
+
+       if ((err = pthread_attr_setschedparam(&attr, &param))) {
+               errc(EX_OSERR, err, "pthread_attr_setschedparam");
+       }
+
+       if ((err = pthread_attr_setschedpolicy(&attr, SCHED_RR))) {
+               errc(EX_OSERR, err, "pthread_attr_setschedpolicy");
+       }
+
+       for (uint32_t i = 0; i < g_rt_churn_count; i++) {
+               pthread_t new_thread;
+
+               if ((err = pthread_create(&new_thread, &attr, rt_churn_thread, NULL))) {
+                       errc(EX_OSERR, err, "pthread_create");
+               }
+               g_rt_churn_threads[i] = new_thread;
+       }
+
+       if ((err = pthread_attr_destroy(&attr))) {
+               errc(EX_OSERR, err, "pthread_attr_destroy");
+       }
+
+       /* Wait until all threads have checked in */
+       wait_for_rt_churn_threads();
+}
+
+static void
+join_rt_churn_threads(void)
+{
+       /* Rejoin rt churn threads */
+       for (uint32_t i = 0; i < g_rt_churn_count; i++) {
+               errno_t err = pthread_join(g_rt_churn_threads[i], NULL);
+               if (err) {
+                       errc(EX_OSERR, err, "pthread_join %d", i);
+               }
+       }
+}
+
+/*
+ * Figure out what thread policy to use
  */
 static my_policy_type_t
 parse_thread_policy(const char *str)
@@ -256,7 +421,7 @@ parse_thread_policy(const char *str)
  * Figure out what wakeup pattern to use
  */
 static wake_type_t
-parse_wakeup_pattern(const char *str) 
+parse_wakeup_pattern(const char *str)
 {
        if (strcmp(str, "chain") == 0) {
                return WAKE_CHAIN;
@@ -283,34 +448,38 @@ thread_setup(uint32_t my_id)
 
        if (g_priority) {
                int policy = SCHED_OTHER;
-               if (g_policy == MY_POLICY_FIXEDPRI)
+               if (g_policy == MY_POLICY_FIXEDPRI) {
                        policy = SCHED_RR;
+               }
 
                struct sched_param param = {.sched_priority = (int)g_priority};
-               if ((ret = pthread_setschedparam(pthread_self(), policy, &param)))
+               if ((ret = pthread_setschedparam(pthread_self(), policy, &param))) {
                        errc(EX_OSERR, ret, "pthread_setschedparam: %d", my_id);
+               }
        }
 
        switch (g_policy) {
-               case MY_POLICY_TIMESHARE:
-                       break;
-               case MY_POLICY_REALTIME:
-                       /* Hard-coded realtime parameters (similar to what Digi uses) */
-                       pol.period      = 100000;
-                       pol.constraint  = (uint32_t) nanos_to_abs(CONSTRAINT_NANOS);
-                       pol.computation = (uint32_t) nanos_to_abs(COMPUTATION_NANOS);
-                       pol.preemptible = 0; /* Ignored by OS */
-
-                       kr = thread_policy_set(mach_thread_self(), THREAD_TIME_CONSTRAINT_POLICY,
-                                              (thread_policy_t) &pol, THREAD_TIME_CONSTRAINT_POLICY_COUNT);
-                       mach_assert_zero_t(my_id, kr);
-                       break;
-               case MY_POLICY_FIXEDPRI:
-                       ret = pthread_set_fixedpriority_self();
-                       if (ret) errc(EX_OSERR, ret, "pthread_set_fixedpriority_self");
-                       break;
-               default:
-                       errx(EX_USAGE, "invalid policy type %d", g_policy);
+       case MY_POLICY_TIMESHARE:
+               break;
+       case MY_POLICY_REALTIME:
+               /* Hard-coded realtime parameters (similar to what Digi uses) */
+               pol.period      = 100000;
+               pol.constraint  = (uint32_t) nanos_to_abs(CONSTRAINT_NANOS);
+               pol.computation = (uint32_t) nanos_to_abs(COMPUTATION_NANOS);
+               pol.preemptible = 0;         /* Ignored by OS */
+
+               kr = thread_policy_set(mach_thread_self(), THREAD_TIME_CONSTRAINT_POLICY,
+                   (thread_policy_t) &pol, THREAD_TIME_CONSTRAINT_POLICY_COUNT);
+               mach_assert_zero_t(my_id, kr);
+               break;
+       case MY_POLICY_FIXEDPRI:
+               ret = pthread_set_fixedpriority_self();
+               if (ret) {
+                       errc(EX_OSERR, ret, "pthread_set_fixedpriority_self");
+               }
+               break;
+       default:
+               errx(EX_USAGE, "invalid policy type %d", g_policy);
        }
 
        if (g_do_affinity) {
@@ -319,7 +488,7 @@ thread_setup(uint32_t my_id)
                affinity.affinity_tag = my_id % 2;
 
                kr = thread_policy_set(mach_thread_self(), THREAD_AFFINITY_POLICY,
-                                      (thread_policy_t)&affinity, THREAD_AFFINITY_POLICY_COUNT);
+                   (thread_policy_t)&affinity, THREAD_AFFINITY_POLICY_COUNT);
                mach_assert_zero_t(my_id, kr);
        }
 
@@ -349,8 +518,9 @@ worker_thread(void *arg)
                         */
 
                        /* Give the worker threads undisturbed time to finish before waiting on them */
-                       if (g_do_sleep)
+                       if (g_do_sleep) {
                                usleep(g_iteration_sleeptime_us);
+                       }
 
                        debug_log("%d Leader thread wait for ready\n", i);
 
@@ -360,13 +530,22 @@ worker_thread(void *arg)
                         * TODO: Invent 'semaphore wait for N signals'
                         */
 
-                       for (uint32_t j = 0 ; j < g_numthreads - 1; j++) {
+                       for (uint32_t j = 0; j < g_numthreads - 1; j++) {
                                kr = semaphore_wait(g_readysem);
                                mach_assert_zero_t(my_id, kr);
                        }
 
                        debug_log("%d Leader thread wait\n", i);
 
+                       if (i > 0) {
+                               for (int cpuid = 0; cpuid < g_numcpus; cpuid++) {
+                                       if (g_cpu_histogram[cpuid].current == 1) {
+                                               atomic_fetch_or_explicit(&g_cpu_map[i - 1], (1UL << cpuid), memory_order_relaxed);
+                                               g_cpu_histogram[cpuid].current = 0;
+                                       }
+                               }
+                       }
+
                        /* Signal main thread and wait for start of iteration */
 
                        kr = semaphore_wait_signal(g_leadersem, g_main_sem);
@@ -376,7 +555,7 @@ worker_thread(void *arg)
 
                        debug_log("%d Leader thread go\n", i);
 
-                       assert_zero_t(my_id, g_done_threads);
+                       assert_zero_t(my_id, atomic_load_explicit(&g_done_threads, memory_order_relaxed));
 
                        switch (g_waketype) {
                        case WAKE_BROADCAST_ONESEM:
@@ -404,7 +583,7 @@ worker_thread(void *arg)
                         * records when she wakes up, and possibly
                         * wakes up a friend.
                         */
-                       switch(g_waketype)  {
+                       switch (g_waketype) {
                        case WAKE_BROADCAST_ONESEM:
                                kr = semaphore_wait_signal(g_broadcastsem, g_readysem);
                                mach_assert_zero_t(my_id, kr);
@@ -454,7 +633,11 @@ worker_thread(void *arg)
                        }
                }
 
-               debug_log("Thread %p woke up for iteration %d.\n", pthread_self(), i);
+               unsigned int cpuid =  _os_cpu_number();
+               assert(cpuid < g_numcpus);
+               debug_log("Thread %p woke up on CPU %d for iteration %d.\n", pthread_self(), cpuid, i);
+               g_cpu_histogram[cpuid].current = 1;
+               g_cpu_histogram[cpuid].accum++;
 
                if (g_do_one_long_spin && g_one_long_spin_id == my_id) {
                        /* One randomly chosen thread holds up the train for a while. */
@@ -476,21 +659,23 @@ worker_thread(void *arg)
                        }
                }
 
-               int32_t new = OSAtomicIncrement32((volatile int32_t *)&g_done_threads);
-               (void)new;
+               uint32_t done_threads;
+               done_threads = atomic_fetch_add_explicit(&g_done_threads, 1, memory_order_relaxed) + 1;
 
-               debug_log("Thread %p new value is %d, iteration %d\n", pthread_self(), new, i);
+               debug_log("Thread %p new value is %d, iteration %d\n", pthread_self(), done_threads, i);
 
                if (g_drop_priority) {
                        /* Drop priority to BG momentarily */
                        errno_t ret = setpriority(PRIO_DARWIN_THREAD, 0, PRIO_DARWIN_BG);
-                       if (ret) errc(EX_OSERR, ret, "setpriority PRIO_DARWIN_BG");
+                       if (ret) {
+                               errc(EX_OSERR, ret, "setpriority PRIO_DARWIN_BG");
+                       }
                }
 
                if (g_do_all_spin) {
                        /* Everyone spins until the last thread checks in. */
 
-                       while (g_done_threads < g_numthreads) {
+                       while (atomic_load_explicit(&g_done_threads, memory_order_relaxed) < g_numthreads) {
                                y = y + 1.5 + x;
                                x = sqrt(y);
                        }
@@ -499,7 +684,9 @@ worker_thread(void *arg)
                if (g_drop_priority) {
                        /* Restore normal priority */
                        errno_t ret = setpriority(PRIO_DARWIN_THREAD, 0, 0);
-                       if (ret) errc(EX_OSERR, ret, "setpriority 0");
+                       if (ret) {
+                               errc(EX_OSERR, ret, "setpriority 0");
+                       }
                }
 
                debug_log("Thread %p done spinning, iteration %d\n", pthread_self(), i);
@@ -507,17 +694,25 @@ worker_thread(void *arg)
 
        if (my_id == 0) {
                /* Give the worker threads undisturbed time to finish before waiting on them */
-               if (g_do_sleep)
+               if (g_do_sleep) {
                        usleep(g_iteration_sleeptime_us);
+               }
 
                /* Wait for the worker threads to finish */
-               for (uint32_t i = 0 ; i < g_numthreads - 1; i++) {
+               for (uint32_t i = 0; i < g_numthreads - 1; i++) {
                        kr = semaphore_wait(g_readysem);
                        mach_assert_zero_t(my_id, kr);
                }
 
                /* Tell everyone and the main thread that the last iteration is done */
-               debug_log("%d Leader thread done\n", i);
+               debug_log("%d Leader thread done\n", g_iterations - 1);
+
+               for (int cpuid = 0; cpuid < g_numcpus; cpuid++) {
+                       if (g_cpu_histogram[cpuid].current == 1) {
+                               atomic_fetch_or_explicit(&g_cpu_map[g_iterations - 1], (1UL << cpuid), memory_order_relaxed);
+                               g_cpu_histogram[cpuid].current = 0;
+                       }
+               }
 
                kr = semaphore_signal_all(g_main_sem);
                mach_assert_zero_t(my_id, kr);
@@ -540,8 +735,8 @@ compute_stats(uint64_t *values, uint64_t count, float *averagep, uint64_t *maxp,
        uint64_t _sum = 0;
        uint64_t _max = 0;
        uint64_t _min = UINT64_MAX;
-       float    _avg = 0;
-       float    _dev = 0;
+       float    _avg = 0;
+       float    _dev = 0;
 
        for (i = 0; i < count; i++) {
                _sum += values[i];
@@ -550,12 +745,12 @@ compute_stats(uint64_t *values, uint64_t count, float *averagep, uint64_t *maxp,
        }
 
        _avg = ((float)_sum) / ((float)count);
-       
+
        _dev = 0;
        for (i = 0; i < count; i++) {
                _dev += powf((((float)values[i]) - _avg), 2);
        }
-       
+
        _dev /= count;
        _dev = sqrtf(_dev);
 
@@ -571,18 +766,23 @@ main(int argc, char **argv)
        errno_t ret;
        kern_return_t kr;
 
-       pthread_t       *threads;
-       uint64_t        *worst_latencies_ns;
-       uint64_t        *worst_latencies_from_first_ns;
-       uint64_t        max, min;
-       float           avg, stddev;
+       pthread_t       *threads;
+       uint64_t        *worst_latencies_ns;
+       uint64_t        *worst_latencies_from_first_ns;
+       uint64_t        max, min;
+       float           avg, stddev;
 
-       for (int i = 0; i < argc; i++)
-               if (strcmp(argv[i], "--switched_apptype") == 0)
+       bool test_fail = false;
+
+       for (int i = 0; i < argc; i++) {
+               if (strcmp(argv[i], "--switched_apptype") == 0) {
                        g_seen_apptype = TRUE;
+               }
+       }
 
-       if (!g_seen_apptype)
+       if (!g_seen_apptype) {
                selfexec_with_apptype(argc, argv);
+       }
 
        parse_args(argc, argv);
 
@@ -592,10 +792,70 @@ main(int argc, char **argv)
 
        size_t ncpu_size = sizeof(g_numcpus);
        ret = sysctlbyname("hw.ncpu", &g_numcpus, &ncpu_size, NULL, 0);
-       if (ret) err(EX_OSERR, "Failed sysctlbyname(hw.ncpu)");
+       if (ret) {
+               err(EX_OSERR, "Failed sysctlbyname(hw.ncpu)");
+       }
+       assert(g_numcpus <= 64); /* g_cpu_map needs to be extended for > 64 cpus */
+
+       size_t physicalcpu_size = sizeof(g_nphysicalcpu);
+       ret = sysctlbyname("hw.physicalcpu", &g_nphysicalcpu, &physicalcpu_size, NULL, 0);
+       if (ret) {
+               err(EX_OSERR, "Failed sysctlbyname(hw.physicalcpu)");
+       }
+
+       size_t logicalcpu_size = sizeof(g_nlogicalcpu);
+       ret = sysctlbyname("hw.logicalcpu", &g_nlogicalcpu, &logicalcpu_size, NULL, 0);
+       if (ret) {
+               err(EX_OSERR, "Failed sysctlbyname(hw.logicalcpu)");
+       }
 
-       if (g_do_each_spin)
+       if (g_test_rt) {
+               if (g_numthreads == 0) {
+                       g_numthreads = g_numcpus;
+               }
+               g_policy = MY_POLICY_REALTIME;
+               g_do_all_spin = TRUE;
+               g_histogram = true;
+               /* Don't change g_traceworthy_latency_ns if it's explicity been set to something other than the default */
+               if (g_traceworthy_latency_ns == TRACEWORTHY_NANOS) {
+                       g_traceworthy_latency_ns = TRACEWORTHY_NANOS_TEST;
+               }
+       } else if (g_test_rt_smt) {
+               if (g_nlogicalcpu != 2 * g_nphysicalcpu) {
+                       /* Not SMT */
+                       printf("Attempt to run --test-rt-smt on a non-SMT device\n");
+                       exit(0);
+               }
+
+               if (g_numthreads == 0) {
+                       g_numthreads = g_nphysicalcpu;
+               }
+               g_policy = MY_POLICY_REALTIME;
+               g_do_all_spin = TRUE;
+               g_histogram = true;
+       } else if (g_test_rt_avoid0) {
+#if defined(__x86_64__) || defined(__i386__)
+               if (g_numthreads == 0) {
+                       g_numthreads = g_nphysicalcpu - 1;
+               }
+               if (g_numthreads == 0) {
+                       printf("Attempt to run --test-rt-avoid0 on a uniprocessor\n");
+                       exit(0);
+               }
+               g_policy = MY_POLICY_REALTIME;
+               g_do_all_spin = TRUE;
+               g_histogram = true;
+#else
+               printf("Attempt to run --test-rt-avoid0 on a non-Intel device\n");
+               exit(0);
+#endif
+       } else if (g_numthreads == 0) {
+               g_numthreads = g_numcpus;
+       }
+
+       if (g_do_each_spin) {
                g_each_spin_duration_abs = nanos_to_abs(g_each_spin_duration_ns);
+       }
 
        /* Configure the long-spin thread to take up half of its computation */
        if (g_do_one_long_spin) {
@@ -607,10 +867,12 @@ main(int argc, char **argv)
        g_iteration_sleeptime_us = g_numthreads * 20;
 
        uint32_t threads_per_core = (g_numthreads / g_numcpus) + 1;
-       if (g_do_each_spin)
+       if (g_do_each_spin) {
                g_iteration_sleeptime_us += threads_per_core * (g_each_spin_duration_ns / NSEC_PER_USEC);
-       if (g_do_one_long_spin)
+       }
+       if (g_do_one_long_spin) {
                g_iteration_sleeptime_us += g_one_long_spin_length_ns / NSEC_PER_USEC;
+       }
 
        /* Arrays for threads and their wakeup times */
        threads = (pthread_t*) valloc(sizeof(pthread_t) * g_numthreads);
@@ -623,7 +885,9 @@ main(int argc, char **argv)
 
        /* Ensure the allocation is pre-faulted */
        ret = memset_s(g_thread_endtimes_abs, endtimes_size, 0, endtimes_size);
-       if (ret) errc(EX_OSERR, ret, "memset_s endtimes");
+       if (ret) {
+               errc(EX_OSERR, ret, "memset_s endtimes");
+       }
 
        size_t latencies_size = sizeof(uint64_t) * g_iterations;
 
@@ -632,14 +896,36 @@ main(int argc, char **argv)
 
        /* Ensure the allocation is pre-faulted */
        ret = memset_s(worst_latencies_ns, latencies_size, 0, latencies_size);
-       if (ret) errc(EX_OSERR, ret, "memset_s latencies");
+       if (ret) {
+               errc(EX_OSERR, ret, "memset_s latencies");
+       }
 
        worst_latencies_from_first_ns = (uint64_t*) valloc(latencies_size);
        assert(worst_latencies_from_first_ns);
 
        /* Ensure the allocation is pre-faulted */
        ret = memset_s(worst_latencies_from_first_ns, latencies_size, 0, latencies_size);
-       if (ret) errc(EX_OSERR, ret, "memset_s latencies_from_first");
+       if (ret) {
+               errc(EX_OSERR, ret, "memset_s latencies_from_first");
+       }
+
+       size_t histogram_size = sizeof(histogram_t) * g_numcpus;
+       g_cpu_histogram = (histogram_t *)valloc(histogram_size);
+       assert(g_cpu_histogram);
+       /* Ensure the allocation is pre-faulted */
+       ret = memset_s(g_cpu_histogram, histogram_size, 0, histogram_size);
+       if (ret) {
+               errc(EX_OSERR, ret, "memset_s g_cpu_histogram");
+       }
+
+       size_t map_size = sizeof(uint64_t) * g_iterations;
+       g_cpu_map = (_Atomic uint64_t *)valloc(map_size);
+       assert(g_cpu_map);
+       /* Ensure the allocation is pre-faulted */
+       ret = memset_s(g_cpu_map, map_size, 0, map_size);
+       if (ret) {
+               errc(EX_OSERR, ret, "memset_s g_cpu_map");
+       }
 
        kr = semaphore_create(mach_task_self(), &g_main_sem, SYNC_POLICY_FIFO, 0);
        mach_assert_zero(kr);
@@ -648,7 +934,6 @@ main(int argc, char **argv)
        if (g_waketype == WAKE_CHAIN ||
            g_waketype == WAKE_BROADCAST_PERTHREAD ||
            g_waketype == WAKE_HOP) {
-
                g_semarr = valloc(sizeof(semaphore_t) * g_numthreads);
                assert(g_semarr);
 
@@ -673,43 +958,64 @@ main(int argc, char **argv)
        kr = semaphore_create(mach_task_self(), &g_readysem, SYNC_POLICY_FIFO, 0);
        mach_assert_zero(kr);
 
+       kr = semaphore_create(mach_task_self(), &g_rt_churn_sem, SYNC_POLICY_FIFO, 0);
+       mach_assert_zero(kr);
+
+       kr = semaphore_create(mach_task_self(), &g_rt_churn_start_sem, SYNC_POLICY_FIFO, 0);
+       mach_assert_zero(kr);
+
+       atomic_store_explicit(&g_done_threads, 0, memory_order_relaxed);
+
        /* Create the threads */
-       g_done_threads = 0;
        for (uint32_t i = 0; i < g_numthreads; i++) {
                ret = pthread_create(&threads[i], NULL, worker_thread, (void*)(uintptr_t)i);
-               if (ret) errc(EX_OSERR, ret, "pthread_create %d", i);
+               if (ret) {
+                       errc(EX_OSERR, ret, "pthread_create %d", i);
+               }
        }
 
        ret = setpriority(PRIO_DARWIN_ROLE, 0, PRIO_DARWIN_ROLE_UI_FOCAL);
-       if (ret) errc(EX_OSERR, ret, "setpriority");
+       if (ret) {
+               errc(EX_OSERR, ret, "setpriority");
+       }
 
        thread_setup(0);
 
        g_starttime_abs = mach_absolute_time();
 
-       if (g_churn_pri)
+       if (g_churn_pri) {
                create_churn_threads();
+       }
+       if (g_rt_churn) {
+               create_rt_churn_threads();
+       }
 
        /* Let everyone get settled */
        kr = semaphore_wait(g_main_sem);
        mach_assert_zero(kr);
 
        /* Give the system a bit more time to settle */
-       if (g_do_sleep)
+       if (g_do_sleep) {
                usleep(g_iteration_sleeptime_us);
+       }
 
        /* Go! */
        for (uint32_t i = 0; i < g_iterations; i++) {
                uint32_t j;
                uint64_t worst_abs = 0, best_abs = UINT64_MAX;
 
-               if (g_do_one_long_spin)
+               if (g_do_one_long_spin) {
                        g_one_long_spin_id = (uint32_t)rand() % g_numthreads;
+               }
+
+               if (g_rt_churn) {
+                       start_rt_churn_threads();
+                       usleep(100);
+               }
 
                debug_log("%d Main thread reset\n", i);
 
-               g_done_threads = 0;
-               OSMemoryBarrier();
+               atomic_store_explicit(&g_done_threads, 0, memory_order_seq_cst);
 
                g_starttime_abs = mach_absolute_time();
 
@@ -719,6 +1025,12 @@ main(int argc, char **argv)
 
                debug_log("%d Main thread return\n", i);
 
+               assert(atomic_load_explicit(&g_done_threads, memory_order_relaxed) == g_numthreads);
+
+               if (g_rt_churn) {
+                       wait_for_rt_churn_threads();
+               }
+
                /*
                 * We report the worst latencies relative to start time
                 * and relative to the lead worker thread.
@@ -729,13 +1041,13 @@ main(int argc, char **argv)
                        latency_abs = g_thread_endtimes_abs[j] - g_starttime_abs;
                        worst_abs = worst_abs < latency_abs ? latency_abs : worst_abs;
                }
-       
+
                worst_latencies_ns[i] = abs_to_nanos(worst_abs);
 
                worst_abs = 0;
                for (j = 1; j < g_numthreads; j++) {
                        uint64_t latency_abs;
-               
+
                        latency_abs = g_thread_endtimes_abs[j] - g_thread_endtimes_abs[0];
                        worst_abs = worst_abs < latency_abs ? latency_abs : worst_abs;
                        best_abs = best_abs > latency_abs ? latency_abs : best_abs;
@@ -750,23 +1062,32 @@ main(int argc, char **argv)
                        /* Ariadne's ad-hoc test signpost */
                        kdebug_trace(ARIADNEDBG_CODE(0, 0), worst_latencies_from_first_ns[i], g_traceworthy_latency_ns, 0, 0);
 
-                       if (g_verbose)
+                       if (g_verbose) {
                                printf("Worst on this round was %.2f us.\n", ((float)worst_latencies_from_first_ns[i]) / 1000.0);
+                       }
                }
 
                /* Give the system a bit more time to settle */
-               if (g_do_sleep)
+               if (g_do_sleep) {
                        usleep(g_iteration_sleeptime_us);
+               }
        }
 
        /* Rejoin threads */
        for (uint32_t i = 0; i < g_numthreads; i++) {
                ret = pthread_join(threads[i], NULL);
-               if (ret) errc(EX_OSERR, ret, "pthread_join %d", i);
+               if (ret) {
+                       errc(EX_OSERR, ret, "pthread_join %d", i);
+               }
+       }
+
+       if (g_rt_churn) {
+               join_rt_churn_threads();
        }
 
-       if (g_churn_pri)
+       if (g_churn_pri) {
                join_churn_threads();
+       }
 
        compute_stats(worst_latencies_ns, g_iterations, &avg, &max, &min, &stddev);
        printf("Results (from a stop):\n");
@@ -790,12 +1111,57 @@ main(int argc, char **argv)
        }
 #endif
 
+       if (g_histogram) {
+               putchar('\n');
+
+               for (uint32_t i = 0; i < g_numcpus; i++) {
+                       printf("%d\t%d\n", i, g_cpu_histogram[i].accum);
+               }
+       }
+
+       if (g_test_rt || g_test_rt_smt || g_test_rt_avoid0) {
+#define PRIMARY   0x5555555555555555ULL
+#define SECONDARY 0xaaaaaaaaaaaaaaaaULL
+
+               int fail_count = 0;
+
+               for (uint32_t i = 0; i < g_iterations; i++) {
+                       bool secondary = false;
+                       bool fail = false;
+                       uint64_t map = g_cpu_map[i];
+                       if (g_test_rt_smt) {
+                               /* Test for one or more threads running on secondary cores unexpectedly (WARNING) */
+                               secondary = (map & SECONDARY);
+                               /* Test for threads running on both primary and secondary cpus of the same core (FAIL) */
+                               fail = ((map & PRIMARY) & ((map & SECONDARY) >> 1));
+                       } else if (g_test_rt) {
+                               fail = (__builtin_popcountll(map) != g_numthreads) && (worst_latencies_ns[i] > g_traceworthy_latency_ns);
+                       } else if (g_test_rt_avoid0) {
+                               fail = ((map & 0x1) == 0x1);
+                       }
+                       if (secondary || fail) {
+                               printf("Iteration %d: 0x%llx%s%s\n", i, map,
+                                   secondary ? " SECONDARY" : "",
+                                   fail ? " FAIL" : "");
+                       }
+                       test_fail |= fail;
+                       fail_count += fail;
+               }
+
+               if (test_fail && (g_iterations >= 100) && (fail_count <= g_iterations / 100)) {
+                       printf("99%% or better success rate\n");
+                       test_fail = 0;
+               }
+       }
+
        free(threads);
        free(g_thread_endtimes_abs);
        free(worst_latencies_ns);
        free(worst_latencies_from_first_ns);
+       free(g_cpu_histogram);
+       free(g_cpu_map);
 
-       return 0;
+       return test_fail;
 }
 
 /*
@@ -815,26 +1181,36 @@ selfexec_with_apptype(int argc, char *argv[])
        uint32_t prog_size = PATH_MAX;
 
        ret = _NSGetExecutablePath(prog, &prog_size);
-       if (ret) err(EX_OSERR, "_NSGetExecutablePath");
+       if (ret) {
+               err(EX_OSERR, "_NSGetExecutablePath");
+       }
 
-       for (i=0; i < argc; i++) {
+       for (i = 0; i < argc; i++) {
                new_argv[i] = argv[i];
        }
 
        new_argv[i]   = "--switched_apptype";
-       new_argv[i+1] = NULL;
+       new_argv[i + 1] = NULL;
 
        ret = posix_spawnattr_init(&attr);
-       if (ret) errc(EX_OSERR, ret, "posix_spawnattr_init");
+       if (ret) {
+               errc(EX_OSERR, ret, "posix_spawnattr_init");
+       }
 
        ret = posix_spawnattr_setflags(&attr, POSIX_SPAWN_SETEXEC);
-       if (ret) errc(EX_OSERR, ret, "posix_spawnattr_setflags");
+       if (ret) {
+               errc(EX_OSERR, ret, "posix_spawnattr_setflags");
+       }
 
        ret = posix_spawnattr_setprocesstype_np(&attr, POSIX_SPAWN_PROC_TYPE_APP_DEFAULT);
-       if (ret) errc(EX_OSERR, ret, "posix_spawnattr_setprocesstype_np");
+       if (ret) {
+               errc(EX_OSERR, ret, "posix_spawnattr_setprocesstype_np");
+       }
 
        ret = posix_spawn(NULL, prog, NULL, &attr, new_argv, environ);
-       if (ret) errc(EX_OSERR, ret, "posix_spawn");
+       if (ret) {
+               errc(EX_OSERR, ret, "posix_spawn");
+       }
 }
 
 /*
@@ -844,11 +1220,11 @@ static void __attribute__((noreturn))
 usage()
 {
        errx(EX_USAGE, "Usage: %s <threads> <chain | hop | broadcast-single-sem | broadcast-per-thread> "
-            "<realtime | timeshare | fixed> <iterations>\n\t\t"
-            "[--trace <traceworthy latency in ns>] "
-            "[--verbose] [--spin-one] [--spin-all] [--spin-time <nanos>] [--affinity]\n\t\t"
-            "[--no-sleep] [--drop-priority] [--churn-pri <pri>] [--churn-count <n>]",
-            getprogname());
+           "<realtime | timeshare | fixed> <iterations>\n\t\t"
+           "[--trace <traceworthy latency in ns>] "
+           "[--verbose] [--spin-one] [--spin-all] [--spin-time <nanos>] [--affinity]\n\t\t"
+           "[--no-sleep] [--drop-priority] [--churn-pri <pri>] [--churn-count <n>]",
+           getprogname());
 }
 
 static struct option* g_longopts;
@@ -862,9 +1238,10 @@ read_dec_arg()
 
        uint32_t arg_val = (uint32_t)strtoull(optarg, &cp, 10);
 
-       if (cp == optarg || *cp)
+       if (cp == optarg || *cp) {
                errx(EX_USAGE, "arg --%s requires a decimal number, found \"%s\"",
-                    g_longopts[option_index].name, optarg);
+                   g_longopts[option_index].name, optarg);
+       }
 
        return arg_val;
 }
@@ -879,23 +1256,32 @@ parse_args(int argc, char *argv[])
                OPT_PRIORITY,
                OPT_CHURN_PRI,
                OPT_CHURN_COUNT,
+               OPT_RT_CHURN_COUNT,
        };
 
        static struct option longopts[] = {
+               /* BEGIN IGNORE CODESTYLE */
                { "spin-time",          required_argument,      NULL,                           OPT_SPIN_TIME },
                { "trace",              required_argument,      NULL,                           OPT_TRACE     },
                { "priority",           required_argument,      NULL,                           OPT_PRIORITY  },
                { "churn-pri",          required_argument,      NULL,                           OPT_CHURN_PRI },
                { "churn-count",        required_argument,      NULL,                           OPT_CHURN_COUNT },
+               { "rt-churn-count",     required_argument,      NULL,                           OPT_RT_CHURN_COUNT },
                { "switched_apptype",   no_argument,            (int*)&g_seen_apptype,          TRUE },
                { "spin-one",           no_argument,            (int*)&g_do_one_long_spin,      TRUE },
                { "spin-all",           no_argument,            (int*)&g_do_all_spin,           TRUE },
                { "affinity",           no_argument,            (int*)&g_do_affinity,           TRUE },
                { "no-sleep",           no_argument,            (int*)&g_do_sleep,              FALSE },
                { "drop-priority",      no_argument,            (int*)&g_drop_priority,         TRUE },
+               { "test-rt",            no_argument,            (int*)&g_test_rt,               TRUE },
+               { "test-rt-smt",        no_argument,            (int*)&g_test_rt_smt,           TRUE },
+               { "test-rt-avoid0",     no_argument,            (int*)&g_test_rt_avoid0,        TRUE },
+               { "rt-churn",           no_argument,            (int*)&g_rt_churn,              TRUE },
+               { "histogram",          no_argument,            (int*)&g_histogram,             TRUE },
                { "verbose",            no_argument,            (int*)&g_verbose,               TRUE },
                { "help",               no_argument,            NULL,                           'h' },
                { NULL,                 0,                      NULL,                           0 }
+               /* END IGNORE CODESTYLE */
        };
 
        g_longopts = longopts;
@@ -922,6 +1308,9 @@ parse_args(int argc, char *argv[])
                case OPT_CHURN_COUNT:
                        g_churn_count = read_dec_arg();
                        break;
+               case OPT_RT_CHURN_COUNT:
+                       g_rt_churn_count = read_dec_arg();
+                       break;
                case '?':
                case 'h':
                default:
@@ -953,11 +1342,9 @@ parse_args(int argc, char *argv[])
        /* How many threads? */
        g_numthreads = (uint32_t)strtoull(argv[0], &cp, 10);
 
-       if (cp == argv[0] || *cp)
+       if (cp == argv[0] || *cp) {
                errx(EX_USAGE, "numthreads requires a decimal number, found \"%s\"", argv[0]);
-
-       if (g_numthreads < 1)
-               errx(EX_USAGE, "Must use at least one thread");
+       }
 
        /* What wakeup pattern? */
        g_waketype = parse_wakeup_pattern(argv[1]);
@@ -968,17 +1355,19 @@ parse_args(int argc, char *argv[])
        /* Iterations */
        g_iterations = (uint32_t)strtoull(argv[3], &cp, 10);
 
-       if (cp == argv[3] || *cp)
+       if (cp == argv[3] || *cp) {
                errx(EX_USAGE, "numthreads requires a decimal number, found \"%s\"", argv[3]);
+       }
 
-       if (g_iterations < 1)
+       if (g_iterations < 1) {
                errx(EX_USAGE, "Must have at least one iteration");
+       }
 
-       if (g_numthreads == 1 && g_waketype == WAKE_CHAIN)
+       if (g_numthreads == 1 && g_waketype == WAKE_CHAIN) {
                errx(EX_USAGE, "chain mode requires more than one thread");
+       }
 
-       if (g_numthreads == 1 && g_waketype == WAKE_HOP)
+       if (g_numthreads == 1 && g_waketype == WAKE_HOP) {
                errx(EX_USAGE, "hop mode requires more than one thread");
+       }
 }
-
-