1
0
mirror of https://github.com/RIOT-OS/RIOT.git synced 2025-01-18 12:52:44 +01:00
RIOT/sys/chardev_thread.c

132 lines
3.4 KiB
C
Raw Normal View History

/**
* Character device messaging loop.
*
* Copyright (C) 2013, INRIA.
*
* 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 chardev
* @{
* @file chardev_thread.c
* @brief Runs an infinite loop in a separate thread to handle access to character devices such as uart.
* @author Kaspar Schleiser <kaspar@schleiser.de>
* @}
*/
2010-09-30 15:10:39 +02:00
#include <stdio.h>
#include <errno.h>
#include "irq.h"
#include "msg.h"
#include "ringbuffer.h"
#include "posix_io.h"
2010-09-30 15:10:39 +02:00
/* increase stack size in uart0 when setting this to 1 */
#define ENABLE_DEBUG (0)
#include "debug.h"
2010-09-30 15:10:39 +02:00
static int min(int a, int b)
{
if (b > a) {
return a;
}
else {
return b;
}
2010-09-30 15:10:39 +02:00
}
void chardev_loop(ringbuffer_t *rb)
{
2011-03-08 10:54:40 +01:00
msg_t m;
2010-09-30 15:10:39 +02:00
kernel_pid_t reader_pid = KERNEL_PID_UNDEF;
struct posix_iop_t *r = NULL;
2010-09-30 15:10:39 +02:00
puts("UART0 thread started.");
while (1) {
2010-09-30 15:10:39 +02:00
msg_receive(&m);
if (!msg_sent_by_int(&m)) {
2013-12-23 18:18:33 +01:00
DEBUG("Receiving message from another thread: ");
switch(m.type) {
2010-09-30 15:10:39 +02:00
case OPEN:
2013-12-23 18:18:33 +01:00
DEBUG("OPEN\n");
if (reader_pid == KERNEL_PID_UNDEF) {
2010-09-30 15:10:39 +02:00
reader_pid = m.sender_pid;
/* no error */
m.content.value = 0;
}
else {
2010-09-30 15:10:39 +02:00
m.content.value = -EBUSY;
}
msg_reply(&m, &m);
2010-09-30 15:10:39 +02:00
break;
2010-09-30 15:10:39 +02:00
case READ:
2013-12-23 18:18:33 +01:00
DEBUG("READ\n");
if (m.sender_pid != reader_pid) {
2010-09-30 15:10:39 +02:00
m.content.value = -EINVAL;
r = NULL;
2010-09-30 15:10:39 +02:00
msg_reply(&m, &m);
}
else {
r = (struct posix_iop_t *)(void*)m.content.ptr;
2010-09-30 15:10:39 +02:00
}
2010-09-30 15:10:39 +02:00
break;
2010-09-30 15:10:39 +02:00
case CLOSE:
2013-12-23 18:18:33 +01:00
DEBUG("CLOSE\n");
if (m.sender_pid == reader_pid) {
DEBUG("uart0_thread: closing file from %" PRIkernel_pid "\n", reader_pid);
reader_pid = KERNEL_PID_UNDEF;
2010-09-30 15:10:39 +02:00
r = NULL;
m.content.value = 0;
}
else {
2010-09-30 15:10:39 +02:00
m.content.value = -EINVAL;
}
msg_reply(&m, &m);
2010-09-30 15:10:39 +02:00
break;
2010-09-30 15:10:39 +02:00
default:
2013-12-23 18:18:33 +01:00
DEBUG("UNKNOWN\n");
2010-09-30 15:10:39 +02:00
m.content.value = -EINVAL;
msg_reply(&m, &m);
}
}
if (rb->avail && (r != NULL)) {
2013-12-23 18:18:33 +01:00
DEBUG("Data is available\n");
unsigned state = disableIRQ();
2010-09-30 15:10:39 +02:00
int nbytes = min(r->nbytes, rb->avail);
DEBUG("uart0_thread [%i]: sending %i bytes received from %" PRIkernel_pid " to pid %" PRIkernel_pid "\n", pid, nbytes, m.sender_pid, reader_pid);
ringbuffer_get(rb, r->buffer, nbytes);
2010-09-30 15:10:39 +02:00
r->nbytes = nbytes;
m.sender_pid = reader_pid;
m.type = OPEN;
m.content.ptr = (char *)r;
2010-09-30 15:10:39 +02:00
msg_reply(&m, &m);
r = NULL;
restoreIRQ(state);
}
}
}
void *chardev_thread_entry(void *rb_)
{
ringbuffer_t *rb = rb_;
chardev_loop(rb);
return NULL;
}