2015-03-03 19:34:12 +01:00
|
|
|
/*
|
|
|
|
* Copyright (C) 2015 Hamburg University of Applied Sciences (HAW)
|
|
|
|
*
|
|
|
|
* 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 cpp11-compat
|
|
|
|
* @{
|
|
|
|
*
|
2015-05-22 07:34:41 +02:00
|
|
|
* @file
|
2015-03-03 19:34:12 +01:00
|
|
|
* @brief C++11 condition variable drop in replacement
|
|
|
|
*
|
|
|
|
* @author Raphael Hiesgen <raphael.hiesgen (at) haw-hamburg.de>
|
|
|
|
*
|
|
|
|
* @}
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include <stdexcept>
|
|
|
|
#include <system_error>
|
|
|
|
|
|
|
|
#include "irq.h"
|
|
|
|
#include "sched.h"
|
2016-03-01 16:33:57 +01:00
|
|
|
#include "thread.h"
|
2023-03-09 13:59:50 +01:00
|
|
|
#include "time_units.h"
|
|
|
|
#include "ztimer64.h"
|
2015-03-03 19:34:12 +01:00
|
|
|
#include "priority_queue.h"
|
|
|
|
|
|
|
|
#include "riot/condition_variable.hpp"
|
|
|
|
|
|
|
|
using namespace std::chrono;
|
|
|
|
|
|
|
|
namespace riot {
|
|
|
|
|
|
|
|
condition_variable::~condition_variable() { m_queue.first = NULL; }
|
|
|
|
|
|
|
|
void condition_variable::notify_one() noexcept {
|
2016-03-19 09:25:47 +01:00
|
|
|
unsigned old_state = irq_disable();
|
2015-03-03 19:34:12 +01:00
|
|
|
priority_queue_node_t* head = priority_queue_remove_head(&m_queue);
|
|
|
|
int other_prio = -1;
|
|
|
|
if (head != NULL) {
|
2016-01-04 22:29:34 +01:00
|
|
|
thread_t* other_thread = (thread_t*)sched_threads[head->data];
|
2015-03-03 19:34:12 +01:00
|
|
|
if (other_thread) {
|
|
|
|
other_prio = other_thread->priority;
|
|
|
|
sched_set_status(other_thread, STATUS_PENDING);
|
|
|
|
}
|
|
|
|
head->data = -1u;
|
|
|
|
}
|
2016-03-19 09:25:47 +01:00
|
|
|
irq_restore(old_state);
|
2015-03-03 19:34:12 +01:00
|
|
|
if (other_prio >= 0) {
|
|
|
|
sched_switch(other_prio);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void condition_variable::notify_all() noexcept {
|
2016-03-19 09:25:47 +01:00
|
|
|
unsigned old_state = irq_disable();
|
2015-03-03 19:34:12 +01:00
|
|
|
int other_prio = -1;
|
|
|
|
while (true) {
|
|
|
|
priority_queue_node_t* head = priority_queue_remove_head(&m_queue);
|
|
|
|
if (head == NULL) {
|
|
|
|
break;
|
|
|
|
}
|
2016-01-04 22:29:34 +01:00
|
|
|
thread_t* other_thread = (thread_t*)sched_threads[head->data];
|
2015-03-03 19:34:12 +01:00
|
|
|
if (other_thread) {
|
|
|
|
auto max_prio
|
|
|
|
= [](int a, int b) { return (a < 0) ? b : ((a < b) ? a : b); };
|
|
|
|
other_prio = max_prio(other_prio, other_thread->priority);
|
|
|
|
sched_set_status(other_thread, STATUS_PENDING);
|
|
|
|
}
|
|
|
|
head->data = -1u;
|
|
|
|
}
|
2016-03-19 09:25:47 +01:00
|
|
|
irq_restore(old_state);
|
2015-03-03 19:34:12 +01:00
|
|
|
if (other_prio >= 0) {
|
|
|
|
sched_switch(other_prio);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void condition_variable::wait(unique_lock<mutex>& lock) noexcept {
|
|
|
|
priority_queue_node_t n;
|
2020-08-23 21:25:54 +02:00
|
|
|
n.priority = thread_get_active()->priority;
|
|
|
|
n.data = thread_getpid();
|
2015-03-03 19:34:12 +01:00
|
|
|
n.next = NULL;
|
|
|
|
// the signaling thread may not hold the mutex, the queue is not thread safe
|
2016-03-19 09:25:47 +01:00
|
|
|
unsigned old_state = irq_disable();
|
2015-03-03 19:34:12 +01:00
|
|
|
priority_queue_add(&m_queue, &n);
|
2016-03-19 09:25:47 +01:00
|
|
|
irq_restore(old_state);
|
2015-03-03 19:34:12 +01:00
|
|
|
mutex_unlock_and_sleep(lock.mutex()->native_handle());
|
|
|
|
if (n.data != -1u) {
|
|
|
|
// on signaling n.data is set to -1u
|
|
|
|
// if it isn't set, then the wakeup is either spurious or a timer wakeup
|
2016-03-19 09:25:47 +01:00
|
|
|
old_state = irq_disable();
|
2015-03-03 19:34:12 +01:00
|
|
|
priority_queue_remove(&m_queue, &n);
|
2016-03-19 09:25:47 +01:00
|
|
|
irq_restore(old_state);
|
2015-03-03 19:34:12 +01:00
|
|
|
}
|
|
|
|
mutex_lock(lock.mutex()->native_handle());
|
|
|
|
}
|
|
|
|
|
|
|
|
cv_status condition_variable::wait_until(unique_lock<mutex>& lock,
|
|
|
|
const time_point& timeout_time) {
|
2023-03-09 13:59:50 +01:00
|
|
|
ztimer64_t timer;
|
|
|
|
uint64_t total_timeout_time_us = timeout_time.microseconds();
|
|
|
|
total_timeout_time_us += timeout_time.seconds() * US_PER_SEC;
|
|
|
|
|
|
|
|
ztimer64_set_wakeup_at(ZTIMER64_USEC, &timer, total_timeout_time_us,
|
|
|
|
thread_getpid());
|
2015-03-03 19:34:12 +01:00
|
|
|
wait(lock);
|
2023-03-09 13:59:50 +01:00
|
|
|
if (ztimer64_now(ZTIMER64_USEC) >= total_timeout_time_us) {
|
|
|
|
ztimer64_remove(ZTIMER64_USEC, &timer);
|
|
|
|
return cv_status::timeout;
|
|
|
|
}
|
|
|
|
ztimer64_remove(ZTIMER64_USEC, &timer);
|
|
|
|
return cv_status::no_timeout;
|
2015-03-03 19:34:12 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
} // namespace riot
|