2015-12-30 15:39:43 +01:00
|
|
|
/*
|
|
|
|
* Copyright (C) 2016 Kaspar Schleiser <kaspar@schleiser.de>
|
|
|
|
*
|
|
|
|
* 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_mbox
|
|
|
|
* @{
|
|
|
|
*
|
|
|
|
* @file
|
|
|
|
* @brief mailbox implementation
|
|
|
|
*
|
|
|
|
* @author Kaspar Schleiser <kaspar@schleiser.de>
|
|
|
|
*
|
|
|
|
* @}
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include <string.h>
|
|
|
|
|
|
|
|
#include "mbox.h"
|
|
|
|
#include "irq.h"
|
|
|
|
#include "sched.h"
|
|
|
|
#include "thread.h"
|
|
|
|
|
|
|
|
#define ENABLE_DEBUG (0)
|
|
|
|
#include "debug.h"
|
|
|
|
|
|
|
|
static void _wake_waiter(thread_t *thread, unsigned irqstate)
|
|
|
|
{
|
|
|
|
sched_set_status(thread, STATUS_PENDING);
|
|
|
|
|
|
|
|
DEBUG("mbox: Thread %"PRIkernel_pid": _wake_waiter(): waking up "
|
|
|
|
"%"PRIkernel_pid".\n", sched_active_pid, thread->pid);
|
|
|
|
|
|
|
|
uint16_t process_priority = thread->priority;
|
|
|
|
irq_restore(irqstate);
|
|
|
|
sched_switch(process_priority);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void _wait(list_node_t *wait_list, unsigned irqstate)
|
|
|
|
{
|
|
|
|
DEBUG("mbox: Thread %"PRIkernel_pid" _wait(): going blocked.\n",
|
|
|
|
sched_active_pid);
|
|
|
|
|
|
|
|
thread_t *me = (thread_t*) sched_active_thread;
|
|
|
|
sched_set_status(me, STATUS_MBOX_BLOCKED);
|
|
|
|
thread_add_to_list(wait_list, me);
|
|
|
|
irq_restore(irqstate);
|
|
|
|
thread_yield();
|
|
|
|
|
|
|
|
DEBUG("mbox: Thread %"PRIkernel_pid" _wait(): woke up.\n",
|
|
|
|
sched_active_pid);
|
|
|
|
}
|
|
|
|
|
|
|
|
int _mbox_put(mbox_t *mbox, msg_t *msg, int blocking)
|
|
|
|
{
|
|
|
|
unsigned irqstate = irq_disable();
|
|
|
|
|
2019-01-07 20:58:35 +01:00
|
|
|
list_node_t *next = list_remove_head(&mbox->readers);
|
2015-12-30 15:39:43 +01:00
|
|
|
if (next) {
|
|
|
|
DEBUG("mbox: Thread %"PRIkernel_pid" mbox 0x%08x: _tryput(): "
|
|
|
|
"there's a waiter.\n", sched_active_pid, (unsigned)mbox);
|
|
|
|
thread_t *thread = container_of((clist_node_t*)next, thread_t, rq_entry);
|
|
|
|
*(msg_t *)thread->wait_data = *msg;
|
|
|
|
_wake_waiter(thread, irqstate);
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
if (cib_full(&mbox->cib)) {
|
|
|
|
if (blocking) {
|
|
|
|
_wait(&mbox->writers, irqstate);
|
|
|
|
irqstate = irq_disable();
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
irq_restore(irqstate);
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
DEBUG("mbox: Thread %"PRIkernel_pid" mbox 0x%08x: _tryput(): "
|
|
|
|
"queued message.\n", sched_active_pid, (unsigned)mbox);
|
|
|
|
msg->sender_pid = sched_active_pid;
|
|
|
|
/* copy msg into queue */
|
|
|
|
mbox->msg_array[cib_put_unsafe(&mbox->cib)] = *msg;
|
|
|
|
irq_restore(irqstate);
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
int _mbox_get(mbox_t *mbox, msg_t *msg, int blocking)
|
|
|
|
{
|
|
|
|
unsigned irqstate = irq_disable();
|
|
|
|
|
|
|
|
if (cib_avail(&mbox->cib)) {
|
|
|
|
DEBUG("mbox: Thread %"PRIkernel_pid" mbox 0x%08x: _tryget(): "
|
|
|
|
"got queued message.\n", sched_active_pid, (unsigned)mbox);
|
|
|
|
/* copy msg from queue */
|
|
|
|
*msg = mbox->msg_array[cib_get_unsafe(&mbox->cib)];
|
2019-01-07 20:58:35 +01:00
|
|
|
list_node_t *next = list_remove_head(&mbox->writers);
|
2015-12-30 15:39:43 +01:00
|
|
|
if (next) {
|
|
|
|
thread_t *thread = container_of((clist_node_t*)next, thread_t, rq_entry);
|
|
|
|
_wake_waiter(thread, irqstate);
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
irq_restore(irqstate);
|
|
|
|
}
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
else if (blocking) {
|
|
|
|
sched_active_thread->wait_data = (void*)msg;
|
|
|
|
_wait(&mbox->readers, irqstate);
|
|
|
|
/* sender has copied message */
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
irq_restore(irqstate);
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
}
|