Loading...
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125 126 127 128 129 130 131 132 133 134 135 136 137 138 139 140 141 142 143 144 145 146 147 148 149 150 151 152 153 154 155 156 157 158 159 160 161 162 163 164 165 166 167 168 169 170 171 172 173 174 175 176 177 178 179 180 181 182 183 184 185 186 187 188 189 190 191 192 193 194 195 196 197 198 199 200 201 202 203 204 205 206 207 208 209 210 211 212 213 214 215 216 217 218 219 220 221 222 223 224 225 226 227 228 229 230 231 232 233 234 235 236 237 238 239 240 241 242 243 244 245 246 247 248 249 250 251 252 253 254 255 256 257 258 259 260 261 262 263 264 265 266 267 268 269 270 271 272 273 274 275 276 277 278 279 280 281 282 283 284 285 286 287 288 289 290 291 292 293 294 295 296 297 298 299 300 301 302 303 304 305 306 307 308 309 310 311 312 313 314 315 316 317 318 319 320 321 322 323 324 325 326 327 328 329 330 331 332 333 334 335 336 337 338 339 340 341 342 343 344 345 346 347 348 349 350 351 352 | /*
* Copyright (c) 2018 Intel Corporation
*
* SPDX-License-Identifier: Apache-2.0
*/
#include <zephyr.h>
#include <ztest.h>
#include <irq_offload.h>
#include <kernel_structs.h> /* for _THREAD_PENDING */
/* Explicit preemption test. Works by creating a set of threads in
* each priority class (cooperative, preemptive, metairq) which all go
* to sleep. Then one is woken up (from a low priority manager
* thread) and arranges to wake up one other thread and validate that
* the next thread to be run is correct according to the documented
* rules.
*
* The wakeup test is repeated for all four combinations of threads
* either holding or not holding the scheduler lock, and by a
* synchronous wake vs. a wake in a (offloaded) interrupt.
*/
#if defined(CONFIG_SMP) && CONFIG_MP_NUM_CPUS > 1
#error Preemption test requires single-CPU operation
#endif
#if CONFIG_NUM_METAIRQ_PRIORITIES < 1
#error Need one metairq priority
#endif
#if CONFIG_NUM_COOP_PRIORITIES < 2
#error Need two cooperative priorities
#endif
#if CONFIG_NUM_PREEMPT_PRIORITIES < 2
#error Need two preemptible priorities
#endif
/* Two threads at each priority (to test the case of waking up a
* thread of equal priority). But only one metairq, as it isn't
* technically legal to have more than one at the same priority.
*/
const enum { METAIRQ, COOP, PREEMPTIBLE } worker_priorities[] = {
METAIRQ,
COOP, COOP,
PREEMPTIBLE, PREEMPTIBLE,
};
#define NUM_THREADS ARRAY_SIZE(worker_priorities)
#define STACK_SIZE (640 + CONFIG_TEST_EXTRA_STACKSIZE)
k_tid_t last_thread;
struct k_thread manager_thread;
K_THREAD_STACK_DEFINE(manager_stack, STACK_SIZE);
struct k_thread worker_threads[NUM_THREADS];
K_THREAD_STACK_ARRAY_DEFINE(worker_stacks, NUM_THREADS, STACK_SIZE);
struct k_thread manager_thread;
struct k_sem worker_sems[NUM_THREADS];
/* Command to worker: who to wake up */
int target;
/* Command to worker: use a sched_lock()? */
int do_lock;
/* Command to worker: use irq_offload() to indirect the wakeup? */
int do_irq;
/* Command to worker: sleep after wakeup? */
int do_sleep;
/* Command to worker: yield after wakeup? */
int do_yield;
K_SEM_DEFINE(main_sem, 0, 1);
void wakeup_src_thread(int id)
{
zassert_true(k_current_get() == &manager_thread, "");
/* irq_offload() on ARM appears not to do what we want. It
* doesn't appear to go through the normal exception return
* path and always returns back into the calling context, so
* it can't be used to fake preemption.
*/
if (do_irq && IS_ENABLED(CONFIG_ARM)) {
return;
}
last_thread = NULL;
/* A little bit of white-box inspection: check that all the
* worker threads are pending.
*/
for (int i = 0; i < NUM_THREADS; i++) {
k_tid_t th = &worker_threads[i];
zassert_equal(strcmp(k_thread_state_str(th), "pending"),
0, "worker thread %d not pending?", i);
}
/* Wake the src worker up */
last_thread = NULL;
k_sem_give(&worker_sems[id]);
while (do_sleep
&& !(worker_threads[id].base.thread_state & _THREAD_PENDING)) {
/* spin, waiting on the sleep timeout */
#if defined(CONFIG_ARCH_POSIX)
/**
* In the posix arch busy wait loops waiting for something to
* happen need to halt the CPU due to the infinitely fast clock
* assumption. (Or in plain English: otherwise you hang in this
* loop. Because the posix arch emulates having 1 CPU by only
* enabling 1 thread at a time. And because it assumes code
* executes in 0 time: it always waits for the code to finish
* and it letting the cpu sleep before letting time pass)
*/
k_busy_wait(50);
#endif
}
/* We are lowest priority, SOMEONE must have run */
zassert_true(!!last_thread, "");
}
void manager(void *p1, void *p2, void *p3)
{
for (int src = 0; src < NUM_THREADS; src++) {
for (target = 0; target < NUM_THREADS; target++) {
if (src == target) {
continue;
}
for (do_lock = 0; do_lock < 2; do_lock++) {
for (do_irq = 0; do_irq < 2; do_irq++) {
do_yield = 0;
do_sleep = 0;
wakeup_src_thread(src);
do_yield = 1;
do_sleep = 0;
wakeup_src_thread(src);
do_yield = 0;
do_sleep = 1;
wakeup_src_thread(src);
}
}
}
}
k_sem_give(&main_sem);
}
void irq_waker(void *p)
{
ARG_UNUSED(p);
k_sem_give(&worker_sems[target]);
}
#define PRI(n) (worker_priorities[n])
void validate_wakeup(int src, int target, k_tid_t last_thread)
{
int preempted = &worker_threads[target] == last_thread;
int src_wins = PRI(src) < PRI(target);
int target_wins = PRI(target) < PRI(src);
int tie = PRI(src) == PRI(target);
if (do_sleep) {
zassert_true(preempted, "sleeping must let any worker run");
return;
}
if (do_yield) {
if (preempted) {
zassert_false(src_wins,
"src (pri %d) should not have yielded to tgt (%d)",
PRI(src), PRI(target));
} else {
zassert_true(src_wins,
"src (pri %d) should have yielded to tgt (%d)",
PRI(src), PRI(target));
}
return;
}
if (preempted) {
zassert_true(target_wins, "preemption must raise priority");
}
if (PRI(target) == METAIRQ) {
zassert_true(preempted,
"metairq threads must always preempt");
} else {
zassert_false(do_lock && preempted,
"threads holding scheduler lock must not be preempted");
zassert_false(preempted && src_wins,
"lower priority threads must never preempt");
if (!do_lock) {
zassert_false(!preempted && target_wins,
"higher priority thread should have preempted");
/* The scheudler implements a 'first added to
* queue' policy for threads within a single
* priority, so the last thread woken up (the
* target) must never run before the source
* thread.
*
* NOTE: I checked, and Zephyr doesn't
* actually document this behavior, though a
* few other tests rely on it IIRC. IMHO
* there are good arguments for either this
* policy OR the opposite ("run newly woken
* threads first"), and long term we may want
* to revisit this particular check and maybe
* make the poilicy configurable.
*/
zassert_false(preempted && tie,
"tied priority should not preempt");
}
}
}
void worker(void *p1, void *p2, void *p3)
{
int id = POINTER_TO_INT(p1);
k_tid_t curr = &worker_threads[id], prev;
ARG_UNUSED(p2);
ARG_UNUSED(p3);
zassert_true(id >= 0 && id < NUM_THREADS, "");
zassert_true(curr == k_current_get(), "");
while (1) {
/* Wait for the manager or another test thread to wake
* us up
*/
k_sem_take(&worker_sems[id], K_FOREVER);
last_thread = curr;
/* If we're the wakeup target, setting last_thread is
* all we do
*/
if (id == target) {
continue;
}
if (do_lock) {
k_sched_lock();
}
if (do_irq) {
/* Do the sem_give() in a IRQ to validate that
* ISR return does the right thing
*/
irq_offload(irq_waker, NULL);
prev = last_thread;
} else {
/* Do the sem_give() directly to validate that
* the synchronous scheduling does the right
* thing
*/
k_sem_give(&worker_sems[target]);
prev = last_thread;
}
if (do_lock) {
k_sched_unlock();
}
if (do_yield) {
k_yield();
prev = last_thread;
}
if (do_sleep) {
u64_t start = k_uptime_get();
k_sleep(K_MSEC(1));
zassert_true(k_uptime_get() - start > 0,
"didn't sleep");
prev = last_thread;
}
validate_wakeup(id, target, prev);
}
}
/**
* @brief Test preemption
*
* @ingroup kernel_sched_tests
*/
void test_preempt(void)
{
int priority;
for (int i = 0; i < NUM_THREADS; i++) {
k_sem_init(&worker_sems[i], 0, 1);
if (worker_priorities[i] == METAIRQ) {
priority = K_HIGHEST_THREAD_PRIO;
} else if (worker_priorities[i] == COOP) {
priority = K_HIGHEST_THREAD_PRIO
+ CONFIG_NUM_METAIRQ_PRIORITIES;
zassert_true(priority < K_PRIO_PREEMPT(0), "");
} else {
priority = K_LOWEST_APPLICATION_THREAD_PRIO - 1;
zassert_true(priority >= K_PRIO_PREEMPT(0), "");
}
k_thread_create(&worker_threads[i],
worker_stacks[i], STACK_SIZE,
worker, INT_TO_POINTER(i), NULL, NULL,
priority, 0, K_NO_WAIT);
}
k_thread_create(&manager_thread, manager_stack, STACK_SIZE,
manager, NULL, NULL, NULL,
K_LOWEST_APPLICATION_THREAD_PRIO, 0, K_NO_WAIT);
/* We don't control the priority of this thread so can't make
* it part of the test. Just get out of the way until the
* test is done
*/
k_sem_take(&main_sem, K_FOREVER);
}
void test_main(void)
{
ztest_test_suite(suite_preempt,
ztest_unit_test(test_preempt));
ztest_run_test_suite(suite_preempt);
}
|