/* See the end of this file for copyright, license, and warranty information. */ #include #include #include #include #include #include #include #include void mutex_init(struct mutex *mutex) { mutex->lock = 0; spin_init(&mutex->wait_queue_lock); list_init(&mutex->wait_queue); } void mutex_lock(struct mutex *mutex) { if (mutex_trylock(mutex) != 0) { struct mutex_wait *entry = malloc(sizeof(*entry)); if (entry == NULL) { _spin_lock(&mutex->lock); /* fall back to spinning */ return; } spin_lock(&mutex->wait_queue_lock); entry->task = current; list_insert(&mutex->wait_queue, &entry->link); spin_unlock(&mutex->wait_queue_lock); yield(TASK_LOCKWAIT); } } void mutex_unlock(struct mutex *mutex) { struct mutex_wait *waiter = NULL; spin_lock(&mutex->wait_queue_lock); if (!list_is_empty(&mutex->wait_queue)) { waiter = list_first_entry(&mutex->wait_queue, struct mutex_wait, link); list_delete(&waiter->link); } spin_unlock(&mutex->wait_queue_lock); if (waiter != NULL) { struct task *task = waiter->task; free(waiter); current->state = TASK_QUEUE; do_switch(current, task); } else { _mutex_unlock(&mutex->lock); } } /* * This file is part of Ardix. * Copyright (c) 2021 Felix Kopp . * * Ardix is non-violent software: you may only use, redistribute, * and/or modify it under the terms of the CNPLv6+ as found in * the LICENSE file in the source code root directory or at * . * * Ardix comes with ABSOLUTELY NO WARRANTY, to the extent * permitted by applicable law. See the CNPLv6+ for details. */