2013-11-23 13:11:23 +01:00
|
|
|
/**
|
|
|
|
* Native uart0 implementation
|
|
|
|
*
|
|
|
|
* Copyright (C) 2013 Ludwig Ortmann
|
|
|
|
*
|
|
|
|
* This file is subject to the terms and conditions of the LGPLv2. See
|
|
|
|
* the file LICENSE in the top level directory for more details.
|
|
|
|
*
|
|
|
|
* @ingroup native_board
|
|
|
|
* @{
|
|
|
|
* @file
|
|
|
|
* @author Ludwig Ortmann <ludwig.ortmann@fu-berlin.de>
|
|
|
|
* @}
|
2013-05-15 16:04:48 +02:00
|
|
|
*/
|
|
|
|
|
2013-05-14 17:42:08 +02:00
|
|
|
#include <err.h>
|
|
|
|
#include <stdio.h>
|
|
|
|
#include <unistd.h>
|
2013-08-15 22:58:26 +02:00
|
|
|
#include <stdlib.h>
|
2013-11-07 17:23:08 +01:00
|
|
|
#include <string.h>
|
|
|
|
#include <stdarg.h>
|
2013-08-18 20:34:53 +02:00
|
|
|
#include <sys/types.h>
|
|
|
|
#include <sys/socket.h>
|
|
|
|
#include <netinet/in.h>
|
|
|
|
#include <netdb.h>
|
|
|
|
#include <arpa/inet.h>
|
|
|
|
#include <sys/un.h>
|
|
|
|
#include <fcntl.h>
|
|
|
|
#include <sys/types.h>
|
|
|
|
#include <sys/stat.h>
|
2013-06-03 19:37:22 +02:00
|
|
|
|
2013-05-14 17:42:08 +02:00
|
|
|
#include <sys/select.h>
|
|
|
|
|
|
|
|
#include "cpu.h"
|
|
|
|
#include "board_uart0.h"
|
2013-12-19 13:06:26 +01:00
|
|
|
#include "thread.h"
|
2013-05-14 17:42:08 +02:00
|
|
|
|
2013-11-07 17:23:08 +01:00
|
|
|
#include "native_internal.h"
|
2013-08-18 20:34:53 +02:00
|
|
|
#include "board_internal.h"
|
2013-11-07 17:23:08 +01:00
|
|
|
|
2013-12-19 13:05:49 +01:00
|
|
|
#define ENABLE_DEBUG (0)
|
|
|
|
#include "debug.h"
|
|
|
|
|
2013-08-18 20:34:53 +02:00
|
|
|
int _native_uart_sock;
|
|
|
|
int _native_uart_conn;
|
2013-06-03 19:37:22 +02:00
|
|
|
|
2013-05-14 17:42:08 +02:00
|
|
|
fd_set _native_uart_rfds;
|
|
|
|
|
2013-08-18 20:34:53 +02:00
|
|
|
/* uart API */
|
|
|
|
|
2013-11-07 17:23:08 +01:00
|
|
|
int uart0_puts(char *astring, int length)
|
2013-05-14 17:42:08 +02:00
|
|
|
{
|
2013-08-18 11:21:38 +02:00
|
|
|
int nwritten, offset;
|
|
|
|
|
|
|
|
nwritten = 0;
|
|
|
|
offset = 0;
|
|
|
|
|
2013-11-07 17:23:08 +01:00
|
|
|
while (
|
|
|
|
(length - offset > 0) && (
|
2014-03-26 13:49:01 +01:00
|
|
|
(nwritten = _native_write(
|
2013-11-07 17:23:08 +01:00
|
|
|
STDOUT_FILENO,
|
|
|
|
astring+offset,
|
|
|
|
length-offset)
|
|
|
|
) > 0)
|
|
|
|
) {
|
2013-08-18 11:21:38 +02:00
|
|
|
offset += nwritten;
|
|
|
|
}
|
|
|
|
if (nwritten == -1) {
|
2013-08-15 22:58:26 +02:00
|
|
|
err(EXIT_FAILURE, "uart0_puts: write");
|
|
|
|
}
|
2013-08-18 11:21:38 +02:00
|
|
|
else if ((length > 0) && (nwritten == 0)) {
|
|
|
|
/* XXX: handle properly */
|
|
|
|
errx(EXIT_FAILURE, "uart0_puts: Could not write to stdout. I don't know what to do now.");
|
|
|
|
}
|
|
|
|
|
2013-08-15 22:58:26 +02:00
|
|
|
return length;
|
2013-05-14 17:42:08 +02:00
|
|
|
}
|
|
|
|
|
2013-08-18 20:34:53 +02:00
|
|
|
/* internal */
|
|
|
|
|
|
|
|
void *get_in_addr(struct sockaddr *sa)
|
|
|
|
{
|
|
|
|
if (sa->sa_family == AF_INET) {
|
|
|
|
return &(((struct sockaddr_in*)sa)->sin_addr);
|
|
|
|
}
|
|
|
|
return &(((struct sockaddr_in6*)sa)->sin6_addr);
|
|
|
|
}
|
|
|
|
|
|
|
|
#ifndef UART_TCPPORT
|
|
|
|
#define UART_TCPPORT "4711"
|
|
|
|
#endif
|
|
|
|
int init_tcp_socket(char *tcpport)
|
|
|
|
{
|
|
|
|
struct addrinfo hints, *info, *p;
|
|
|
|
int i, s;
|
|
|
|
if (tcpport == NULL) {
|
|
|
|
tcpport = UART_TCPPORT;
|
|
|
|
}
|
|
|
|
|
|
|
|
memset(&hints, 0, sizeof hints);
|
|
|
|
hints.ai_family = AF_UNSPEC;
|
|
|
|
hints.ai_socktype = SOCK_STREAM;
|
|
|
|
hints.ai_flags = AI_PASSIVE;
|
|
|
|
|
|
|
|
if ((i = getaddrinfo(NULL, tcpport, &hints, &info)) != 0) {
|
|
|
|
errx(EXIT_FAILURE,
|
|
|
|
"init_uart_socket: getaddrinfo: %s", gai_strerror(i));
|
|
|
|
}
|
|
|
|
|
|
|
|
for (p = info; p != NULL; p = p->ai_next) {
|
|
|
|
if ((s = socket(p->ai_family, p->ai_socktype, p->ai_protocol)) == -1) {
|
|
|
|
warn("init_uart_socket: socket");
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
i = 1;
|
|
|
|
if (setsockopt(s, SOL_SOCKET, SO_REUSEADDR, &i, sizeof(int)) == -1) {
|
|
|
|
err(EXIT_FAILURE, "init_uart_socket: setsockopt");
|
|
|
|
}
|
|
|
|
|
|
|
|
if (bind(s, p->ai_addr, p->ai_addrlen) == -1) {
|
|
|
|
close(s);
|
|
|
|
warn("init_uart_socket: bind");
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
if (p == NULL) {
|
|
|
|
errx(EXIT_FAILURE, "init_uart_socket: failed to bind\n");
|
|
|
|
}
|
|
|
|
freeaddrinfo(info);
|
|
|
|
|
|
|
|
if (listen(s, 1) == -1) {
|
|
|
|
err(EXIT_FAILURE, "init_uart_socket: listen");
|
|
|
|
}
|
|
|
|
|
|
|
|
return s;
|
|
|
|
}
|
|
|
|
|
2014-05-07 12:36:32 +02:00
|
|
|
int init_unix_socket(void)
|
2013-08-18 20:34:53 +02:00
|
|
|
{
|
|
|
|
int s;
|
|
|
|
struct sockaddr_un sa;
|
|
|
|
|
|
|
|
if ((s = socket(AF_UNIX, SOCK_STREAM, 0)) == -1) {
|
|
|
|
err(EXIT_FAILURE, "init_unix_socket: socket");
|
|
|
|
}
|
|
|
|
|
|
|
|
sa.sun_family = AF_UNIX;
|
2014-05-13 16:53:36 +02:00
|
|
|
snprintf(sa.sun_path, sizeof(sa.sun_path), "/tmp/riot.tty.%d", _native_pid);
|
2013-08-18 20:34:53 +02:00
|
|
|
unlink(sa.sun_path); /* remove stale socket */
|
|
|
|
if (bind(s, (struct sockaddr *)&sa, SUN_LEN(&sa)) == -1) {
|
|
|
|
err(EXIT_FAILURE, "init_unix_socket: bind");
|
|
|
|
}
|
|
|
|
|
|
|
|
if (listen(s, 5) == -1) {
|
|
|
|
err(EXIT_FAILURE, "init_unix_socket: listen");
|
|
|
|
}
|
|
|
|
|
|
|
|
return s;
|
|
|
|
}
|
|
|
|
|
2014-05-07 12:36:32 +02:00
|
|
|
void handle_uart_in(void)
|
2013-05-14 17:42:08 +02:00
|
|
|
{
|
|
|
|
char buf[42];
|
|
|
|
int nread;
|
|
|
|
|
2013-08-18 20:34:53 +02:00
|
|
|
DEBUG("handle_uart_in\n");
|
2013-05-14 17:42:08 +02:00
|
|
|
|
2014-03-26 13:49:01 +01:00
|
|
|
nread = _native_read(STDIN_FILENO, buf, sizeof(buf));
|
2013-05-14 17:42:08 +02:00
|
|
|
if (nread == -1) {
|
2013-08-18 20:34:53 +02:00
|
|
|
err(1, "handle_uart_in(): read()");
|
2013-05-14 17:42:08 +02:00
|
|
|
}
|
2013-10-26 20:06:35 +02:00
|
|
|
else if (nread == 0) {
|
2013-08-18 20:34:53 +02:00
|
|
|
/* end of file / socket closed */
|
|
|
|
if (_native_uart_conn != 0) {
|
|
|
|
if (_native_null_out_file != -1) {
|
|
|
|
if (dup2(_native_null_out_file, STDOUT_FILENO) == -1) {
|
|
|
|
err(EXIT_FAILURE, "handle_uart_in: dup2(STDOUT_FILENO)");
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (dup2(_native_null_in_pipe[0], STDIN_FILENO) == -1) {
|
|
|
|
err(EXIT_FAILURE, "handle_uart_in: dup2(STDIN_FILENO)");
|
|
|
|
}
|
|
|
|
_native_uart_conn = 0;
|
|
|
|
warnx("closed stdio");
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
errx(EXIT_FAILURE, "handle_uart_in: unhandled situation!");
|
|
|
|
}
|
2013-10-26 20:06:35 +02:00
|
|
|
}
|
2013-05-14 17:42:08 +02:00
|
|
|
for(int pos = 0; pos < nread; pos++) {
|
|
|
|
uart0_handle_incoming(buf[pos]);
|
|
|
|
}
|
|
|
|
uart0_notify_thread();
|
|
|
|
|
2013-10-26 14:45:51 +02:00
|
|
|
thread_yield();
|
2013-05-14 17:42:08 +02:00
|
|
|
}
|
|
|
|
|
2014-05-07 12:36:32 +02:00
|
|
|
void handle_uart_sock(void)
|
2013-08-18 20:34:53 +02:00
|
|
|
{
|
|
|
|
int s;
|
|
|
|
socklen_t t;
|
|
|
|
struct sockaddr remote;
|
|
|
|
|
|
|
|
t = sizeof(remote);
|
|
|
|
|
|
|
|
_native_syscall_enter();
|
|
|
|
if ((s = accept(_native_uart_sock, &remote, &t)) == -1) {
|
|
|
|
err(EXIT_FAILURE, "handle_uart_sock: accept");
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
warnx("handle_uart_sock: successfully accepted socket");
|
|
|
|
}
|
|
|
|
|
|
|
|
if (dup2(s, STDOUT_FILENO) == -1) {
|
|
|
|
err(EXIT_FAILURE, "handle_uart_sock: dup2()");
|
|
|
|
}
|
|
|
|
if (dup2(s, STDIN_FILENO) == -1) {
|
|
|
|
err(EXIT_FAILURE, "handle_uart_sock: dup2()");
|
|
|
|
}
|
|
|
|
_native_syscall_leave();
|
|
|
|
|
|
|
|
_native_uart_conn = s;
|
|
|
|
}
|
|
|
|
|
2013-12-24 14:50:26 +01:00
|
|
|
#ifdef MODULE_UART0
|
2014-05-07 12:36:32 +02:00
|
|
|
void _native_handle_uart0_input(void)
|
2013-08-18 20:34:53 +02:00
|
|
|
{
|
|
|
|
if (FD_ISSET(STDIN_FILENO, &_native_rfds)) {
|
|
|
|
handle_uart_in();
|
|
|
|
}
|
|
|
|
else if ((_native_uart_sock != -1) && (FD_ISSET(_native_uart_sock, &_native_rfds))) {
|
|
|
|
handle_uart_sock();
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
DEBUG("_native_handle_uart0_input - nothing to do\n");
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-07-16 17:57:52 +02:00
|
|
|
int _native_set_uart_fds(void)
|
|
|
|
{
|
2014-01-22 18:23:10 +01:00
|
|
|
DEBUG("_native_set_uart_fds\n");
|
2013-08-18 20:34:53 +02:00
|
|
|
FD_SET(STDIN_FILENO, &_native_rfds);
|
|
|
|
if (_native_uart_sock == -1) {
|
|
|
|
return (STDIN_FILENO);
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
FD_SET(_native_uart_sock, &_native_rfds);
|
|
|
|
return ((STDIN_FILENO > _native_uart_sock) ? STDIN_FILENO : _native_uart_sock);
|
2013-11-07 17:23:08 +01:00
|
|
|
}
|
2013-07-16 17:57:52 +02:00
|
|
|
}
|
2013-12-24 14:50:26 +01:00
|
|
|
#endif
|
2013-07-16 17:57:52 +02:00
|
|
|
|
2013-08-18 20:34:53 +02:00
|
|
|
void _native_init_uart0(char *stdiotype, char *ioparam)
|
2013-05-14 17:42:08 +02:00
|
|
|
{
|
2013-08-18 20:34:53 +02:00
|
|
|
if (strcmp(stdiotype, "tcp") == 0) {
|
|
|
|
_native_uart_sock = init_tcp_socket(ioparam);
|
|
|
|
}
|
|
|
|
else if (strcmp(stdiotype, "unix") == 0) {
|
|
|
|
_native_uart_sock = init_unix_socket();
|
|
|
|
}
|
|
|
|
else if (strcmp(stdiotype, "stdio") == 0) {
|
|
|
|
_native_uart_sock = -1;
|
|
|
|
_native_uart_conn = 1;
|
|
|
|
}
|
|
|
|
else if (strcmp(stdiotype, "null") == 0) {
|
|
|
|
_native_uart_sock = -1;
|
|
|
|
_native_uart_conn = 0;
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
errx(EXIT_FAILURE, "_native_init_uart0: unknown stdio type");
|
|
|
|
}
|
2013-06-03 19:37:22 +02:00
|
|
|
|
2013-05-14 17:42:08 +02:00
|
|
|
puts("RIOT native uart0 initialized.");
|
|
|
|
}
|