2013-06-18 17:21:38 +02:00
|
|
|
/**
|
2013-06-22 05:11:53 +02:00
|
|
|
* Character device messaging loop.
|
2013-06-18 17:21:38 +02:00
|
|
|
*
|
|
|
|
* Copyright (C) 2013, INRIA.
|
|
|
|
*
|
2014-07-31 19:45:27 +02:00
|
|
|
* 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.
|
2013-06-18 17:21:38 +02:00
|
|
|
*
|
|
|
|
* @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>
|
2013-11-14 12:46:53 +01:00
|
|
|
|
|
|
|
#include "irq.h"
|
|
|
|
#include "msg.h"
|
|
|
|
|
|
|
|
#include "ringbuffer.h"
|
|
|
|
#include "posix_io.h"
|
2010-09-30 15:10:39 +02:00
|
|
|
|
2014-02-24 22:40:02 +01:00
|
|
|
/* increase stack size in uart0 when setting this to 1 */
|
2013-07-24 00:36:06 +02:00
|
|
|
#define ENABLE_DEBUG (0)
|
2013-11-14 12:46:53 +01:00
|
|
|
#include "debug.h"
|
2010-09-30 15:10:39 +02:00
|
|
|
|
2013-06-22 05:11:53 +02:00
|
|
|
static int min(int a, int b)
|
|
|
|
{
|
2013-06-24 22:37:35 +02:00
|
|
|
if (b > a) {
|
2013-06-22 05:11:53 +02:00
|
|
|
return a;
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
return b;
|
|
|
|
}
|
2010-09-30 15:10:39 +02:00
|
|
|
}
|
|
|
|
|
2013-06-22 05:11:53 +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
|
|
|
|
2014-08-06 09:27:23 +02:00
|
|
|
kernel_pid_t reader_pid = KERNEL_PID_UNDEF;
|
2011-03-08 11:43:21 +01:00
|
|
|
struct posix_iop_t *r = NULL;
|
2010-09-30 15:10:39 +02:00
|
|
|
|
|
|
|
puts("UART0 thread started.");
|
|
|
|
|
2013-06-24 22:37:35 +02:00
|
|
|
while (1) {
|
2010-09-30 15:10:39 +02:00
|
|
|
msg_receive(&m);
|
|
|
|
|
2014-10-10 11:28:38 +02:00
|
|
|
if (!msg_sent_by_int(&m)) {
|
2013-12-23 18:18:33 +01:00
|
|
|
DEBUG("Receiving message from another thread: ");
|
2013-06-22 05:11:53 +02:00
|
|
|
|
|
|
|
switch(m.type) {
|
2010-09-30 15:10:39 +02:00
|
|
|
case OPEN:
|
2013-12-23 18:18:33 +01:00
|
|
|
DEBUG("OPEN\n");
|
2014-08-06 09:27:23 +02:00
|
|
|
if (reader_pid == KERNEL_PID_UNDEF) {
|
2010-09-30 15:10:39 +02:00
|
|
|
reader_pid = m.sender_pid;
|
2012-02-16 21:33:41 +01:00
|
|
|
/* no error */
|
|
|
|
m.content.value = 0;
|
|
|
|
}
|
|
|
|
else {
|
2010-09-30 15:10:39 +02:00
|
|
|
m.content.value = -EBUSY;
|
|
|
|
}
|
2013-06-22 05:11:53 +02:00
|
|
|
|
|
|
|
msg_reply(&m, &m);
|
2010-09-30 15:10:39 +02:00
|
|
|
break;
|
2013-06-22 05:11:53 +02:00
|
|
|
|
2010-09-30 15:10:39 +02:00
|
|
|
case READ:
|
2013-12-23 18:18:33 +01:00
|
|
|
DEBUG("READ\n");
|
2013-06-24 22:37:35 +02:00
|
|
|
if (m.sender_pid != reader_pid) {
|
2010-09-30 15:10:39 +02:00
|
|
|
m.content.value = -EINVAL;
|
2012-02-16 21:33:41 +01:00
|
|
|
r = NULL;
|
2010-09-30 15:10:39 +02:00
|
|
|
msg_reply(&m, &m);
|
2012-02-16 21:33:41 +01:00
|
|
|
}
|
|
|
|
else {
|
2013-04-02 02:02:58 +02:00
|
|
|
r = (struct posix_iop_t *)(void*)m.content.ptr;
|
2010-09-30 15:10:39 +02:00
|
|
|
}
|
2013-06-22 05:11:53 +02:00
|
|
|
|
2010-09-30 15:10:39 +02:00
|
|
|
break;
|
2013-06-22 05:11:53 +02:00
|
|
|
|
2010-09-30 15:10:39 +02:00
|
|
|
case CLOSE:
|
2013-12-23 18:18:33 +01:00
|
|
|
DEBUG("CLOSE\n");
|
2013-06-24 22:37:35 +02:00
|
|
|
if (m.sender_pid == reader_pid) {
|
2014-07-06 22:57:56 +02:00
|
|
|
DEBUG("uart0_thread: closing file from %" PRIkernel_pid "\n", reader_pid);
|
2014-08-06 09:27:23 +02:00
|
|
|
reader_pid = KERNEL_PID_UNDEF;
|
2010-09-30 15:10:39 +02:00
|
|
|
r = NULL;
|
|
|
|
m.content.value = 0;
|
2012-02-16 21:33:41 +01:00
|
|
|
}
|
|
|
|
else {
|
2010-09-30 15:10:39 +02:00
|
|
|
m.content.value = -EINVAL;
|
|
|
|
}
|
2013-06-22 05:11:53 +02:00
|
|
|
|
|
|
|
msg_reply(&m, &m);
|
2010-09-30 15:10:39 +02:00
|
|
|
break;
|
2013-06-22 05:11:53 +02:00
|
|
|
|
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);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-06-24 22:37:35 +02:00
|
|
|
if (rb->avail && (r != NULL)) {
|
2013-12-23 18:18:33 +01:00
|
|
|
DEBUG("Data is available\n");
|
2014-10-27 16:18:40 +01:00
|
|
|
unsigned state = disableIRQ();
|
2010-09-30 15:10:39 +02:00
|
|
|
int nbytes = min(r->nbytes, rb->avail);
|
2015-01-28 01:49:23 +01:00
|
|
|
DEBUG("uart0_thread [%i]: sending %i bytes received from %" PRIkernel_pid " to pid %" PRIkernel_pid "\n", thread_getpid(), nbytes, m.sender_pid, reader_pid);
|
2014-05-03 16:39:39 +02:00
|
|
|
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;
|
2013-06-22 05:11:53 +02:00
|
|
|
m.content.ptr = (char *)r;
|
2010-09-30 15:10:39 +02:00
|
|
|
|
|
|
|
msg_reply(&m, &m);
|
|
|
|
|
|
|
|
r = NULL;
|
|
|
|
restoreIRQ(state);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2014-03-04 20:20:01 +01:00
|
|
|
|
|
|
|
void *chardev_thread_entry(void *rb_)
|
|
|
|
{
|
|
|
|
ringbuffer_t *rb = rb_;
|
|
|
|
chardev_loop(rb);
|
|
|
|
return NULL;
|
|
|
|
}
|