/* * Copyright (c) 2004, QUALCOMM Inc. All rights reserved. * Created by: abisain REMOVE-THIS AT qualcomm DOT com * This file is licensed under the GPL license. For the full content * of this license, see the COPYING file at the top level of this * source tree. * * Test that pthread_barrier_wait() * shall wakeup a high priority thread even when a low priority thread * is running * * Steps: * 1. Create a barrier object * 2. Create a high priority thread and make it wait on the barrier * 3. Create a low priority thread and let it busy-loop * 4. Both low and high prio threads run on same CPU * 5. Call the final barrier_wait from main * 6. Check that the higher priority thread got woken up * and preempted low priority thread */ #define _XOPEN_SOURCE 600 #include "affinity.h" #include <pthread.h> #include <stdlib.h> #include <stdio.h> #include <unistd.h> #include <sys/time.h> #include "posixtest.h" #include "safe_helpers.h" #define TEST "5-4" #define AREA "scheduler" #define ERROR_PREFIX "unexpected error: " AREA " " TEST ": " #define HIGH_PRIORITY 10 #define LOW_PRIORITY 5 #define RUNTIME 5 pthread_barrier_t barrier; static volatile int woken_up; static volatile int low_done; float timediff(struct timespec t2, struct timespec t1) { float diff = t2.tv_sec - t1.tv_sec; diff += (t2.tv_nsec - t1.tv_nsec) / 1000000000.0; return diff; } int my_pthread_barrier_wait(pthread_barrier_t *p) { int rc; rc = pthread_barrier_wait(p); if (rc == PTHREAD_BARRIER_SERIAL_THREAD) rc = 0; return rc; } void *hi_prio_thread(void *tmp) { struct sched_param param; int policy; (void) tmp; set_affinity_single(); SAFE_PFUNC(pthread_getschedparam(pthread_self(), &policy, ¶m)); if (policy != SCHED_RR || param.sched_priority != HIGH_PRIORITY) { printf("Error: the policy or priority not correct\n"); exit(PTS_UNRESOLVED); } SAFE_PFUNC(my_pthread_barrier_wait(&barrier)); /* This variable is unprotected because the scheduling removes * the contention */ if (!low_done) woken_up = 1; pthread_exit(NULL); } void *low_prio_thread(void *tmp) { struct timespec start_timespec, current_timespec; struct sched_param param; int policy; (void) tmp; set_affinity_single(); SAFE_PFUNC(pthread_getschedparam(pthread_self(), &policy, ¶m)); if (policy != SCHED_RR || param.sched_priority != LOW_PRIORITY) { printf("Error: the policy or priority not correct\n"); exit(PTS_UNRESOLVED); } clock_gettime(CLOCK_REALTIME, &start_timespec); while (!woken_up) { clock_gettime(CLOCK_REALTIME, ¤t_timespec); if (timediff(current_timespec, start_timespec) > RUNTIME) break; } low_done = 1; pthread_exit(NULL); } int main() { pthread_t high_id, low_id; pthread_attr_t high_attr; struct sched_param param; SAFE_PFUNC(pthread_barrier_init(&barrier, NULL, 2)); /* Create the higher priority */ SAFE_PFUNC(pthread_attr_init(&high_attr)); SAFE_PFUNC(pthread_attr_setinheritsched(&high_attr, PTHREAD_EXPLICIT_SCHED)); SAFE_PFUNC(pthread_attr_setschedpolicy(&high_attr, SCHED_RR)); param.sched_priority = HIGH_PRIORITY; SAFE_PFUNC(pthread_attr_setschedparam(&high_attr, ¶m)); SAFE_PFUNC(pthread_create(&high_id, &high_attr, hi_prio_thread, NULL)); /* run main with same priority as low prio thread */ param.sched_priority = LOW_PRIORITY; SAFE_PFUNC(pthread_setschedparam(pthread_self(), SCHED_RR, ¶m)); /* Create the low priority thread (inherits sched policy from main) */ SAFE_PFUNC(pthread_create(&low_id, NULL, low_prio_thread, NULL)); sleep(1); SAFE_PFUNC(my_pthread_barrier_wait(&barrier)); /* Wait for the threads to exit */ SAFE_PFUNC(pthread_join(low_id, NULL)); if (!woken_up) { printf("High priority was not woken up. Test FAILED\n"); exit(PTS_FAIL); } SAFE_PFUNC(pthread_join(high_id, NULL)); printf("Test PASSED\n"); exit(PTS_PASS); }