/* * Copyright (C) 2015 Kaspar Schleiser * 2013 Freie UniversitÀt Berlin * * This file is subject to the terms and conditions of the GNU Lesser * General Public License v2.1. See the file LICENSE in the top level * directory for more details. */ /** * @ingroup core_sync * @{ * * @file * @brief Kernel mutex implementation * * @author Kaspar Schleiser * @author Joakim NohlgÄrd * * @} */ #include #include #include #include "mutex.h" #include "thread.h" #include "sched.h" #include "irq.h" #include "list.h" #define ENABLE_DEBUG 0 #include "debug.h" int mutex_lock(mutex_t *mutex) { unsigned irq_state = irq_disable(); DEBUG("PID[%" PRIkernel_pid "]: Mutex in use.\n", thread_getpid()); if (mutex->queue.next == NULL) { /* mutex is unlocked. */ mutex->queue.next = MUTEX_LOCKED; DEBUG("PID[%" PRIkernel_pid "]: mutex_wait_and_lock early out.\n", thread_getpid()); irq_restore(irq_state); return 0; } thread_t *me = thread_get_active(); DEBUG("PID[%" PRIkernel_pid "]: Adding node to mutex queue: prio: %" PRIu32 "\n", thread_getpid(), (uint32_t)me->priority); sched_set_status(me, STATUS_MUTEX_BLOCKED); if (mutex->queue.next == MUTEX_LOCKED) { mutex->queue.next = (list_node_t *)&me->rq_entry; mutex->queue.next->next = NULL; } else { thread_add_to_list(&mutex->queue, me); } irq_restore(irq_state); thread_yield_higher(); /* We were woken up by scheduler. Waker removed us from queue. */ return 0; } void mutex_unlock(mutex_t *mutex) { unsigned irqstate = irq_disable(); DEBUG("mutex_unlock(): queue.next: %p pid: %" PRIkernel_pid "\n", (void *)mutex->queue.next, thread_getpid()); if (mutex->queue.next == NULL) { /* the mutex was not locked */ irq_restore(irqstate); return; } if (mutex->queue.next == MUTEX_LOCKED) { mutex->queue.next = NULL; /* the mutex was locked and no thread was waiting for it */ irq_restore(irqstate); return; } list_node_t *next = list_remove_head(&mutex->queue); thread_t *process = container_of((clist_node_t *)next, thread_t, rq_entry); DEBUG("mutex_unlock: waking up waiting thread %" PRIkernel_pid "\n", process->pid); sched_set_status(process, STATUS_PENDING); if (!mutex->queue.next) { mutex->queue.next = MUTEX_LOCKED; } uint16_t process_priority = process->priority; irq_restore(irqstate); sched_switch(process_priority); } void mutex_unlock_and_sleep(mutex_t *mutex) { DEBUG("PID[%" PRIkernel_pid "]: unlocking mutex. queue.next: %p, and " "taking a nap\n", thread_getpid(), (void *)mutex->queue.next); unsigned irqstate = irq_disable(); if (mutex->queue.next) { if (mutex->queue.next == MUTEX_LOCKED) { mutex->queue.next = NULL; } else { list_node_t *next = list_remove_head(&mutex->queue); thread_t *process = container_of((clist_node_t *)next, thread_t, rq_entry); DEBUG("PID[%" PRIkernel_pid "]: waking up waiter.\n", process->pid); sched_set_status(process, STATUS_PENDING); if (!mutex->queue.next) { mutex->queue.next = MUTEX_LOCKED; } } } DEBUG("PID[%" PRIkernel_pid "]: going to sleep.\n", thread_getpid()); sched_set_status(thread_get_active(), STATUS_SLEEPING); irq_restore(irqstate); thread_yield_higher(); }