mirror of
https://github.com/RIOT-OS/RIOT.git
synced 2025-01-18 12:52:44 +01:00
Koen Zandberg
47e2885f80
fcntl(fd, F_SETOWN, getpid()); doesn't seem to work on Linux to get generate a signal when an event on the GPIO fd occurs. So fall back to the same method as on OS X and call poll() in a child process.
160 lines
4.3 KiB
C
160 lines
4.3 KiB
C
/**
|
|
* Multiple asynchronus read on file descriptors
|
|
*
|
|
* Copyright (C) 2015 Ludwig Knüpfer <ludwig.knuepfer@fu-berlin.de>,
|
|
* Martine Lenders <mlenders@inf.fu-berlin.de>
|
|
* Kaspar Schleiser <kaspar@schleiser.de>
|
|
* Ell-i open source co-operative
|
|
* Takuo Yonezawa <Yonezawa-T2@mail.dnp.co.jp>
|
|
*
|
|
* 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 cpu_native
|
|
* @{
|
|
* @file
|
|
* @author Takuo Yonezawa <Yonezawa-T2@mail.dnp.co.jp>
|
|
*/
|
|
|
|
#include <err.h>
|
|
#include <signal.h>
|
|
#include <stdlib.h>
|
|
#include <unistd.h>
|
|
#include <fcntl.h>
|
|
#include <poll.h>
|
|
|
|
#include "async_read.h"
|
|
#include "native_internal.h"
|
|
|
|
static int _next_index;
|
|
static struct pollfd _fds[ASYNC_READ_NUMOF];
|
|
static async_read_t pollers[ASYNC_READ_NUMOF];
|
|
|
|
static void _sigio_child(int fd);
|
|
|
|
static void _async_io_isr(void) {
|
|
if (real_poll(_fds, _next_index, 0) > 0) {
|
|
for (int i = 0; i < _next_index; i++) {
|
|
/* handle if one of the events has happened */
|
|
if (_fds[i].revents & _fds[i].events) {
|
|
pollers[i].cb(_fds[i].fd, pollers[i].arg);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
void native_async_read_setup(void) {
|
|
register_interrupt(SIGIO, _async_io_isr);
|
|
}
|
|
|
|
void native_async_read_cleanup(void) {
|
|
unregister_interrupt(SIGIO);
|
|
|
|
for (int i = 0; i < _next_index; i++) {
|
|
if (pollers[i].child_pid) {
|
|
kill(pollers[i].child_pid, SIGKILL);
|
|
}
|
|
}
|
|
}
|
|
|
|
void native_async_read_continue(int fd) {
|
|
for (int i = 0; i < _next_index; i++) {
|
|
if (_fds[i].fd == fd && pollers[i].child_pid) {
|
|
kill(pollers[i].child_pid, SIGCONT);
|
|
}
|
|
}
|
|
}
|
|
|
|
static void _add_handler(int fd, void *arg, native_async_read_callback_t handler) {
|
|
_fds[_next_index].fd = fd;
|
|
_fds[_next_index].events = POLLIN | POLLPRI;
|
|
async_read_t *poll = &pollers[_next_index];
|
|
|
|
poll->child_pid = 0;
|
|
poll->cb = handler;
|
|
poll->arg = arg;
|
|
poll->fd = &_fds[_next_index];
|
|
}
|
|
|
|
void native_async_read_add_handler(int fd, void *arg, native_async_read_callback_t handler) {
|
|
if (_next_index >= ASYNC_READ_NUMOF) {
|
|
err(EXIT_FAILURE, "native_async_read_add_handler(): too many callbacks");
|
|
}
|
|
|
|
_add_handler(fd, arg, handler);
|
|
|
|
/* tuntap signalled IO is not working in OSX,
|
|
* * check http://sourceforge.net/p/tuntaposx/bugs/18/ */
|
|
#ifdef __MACH__
|
|
_sigio_child(_next_index);
|
|
#else
|
|
/* configure fds to send signals on io */
|
|
if (real_fcntl(fd, F_SETOWN, _native_pid) == -1) {
|
|
err(EXIT_FAILURE, "native_async_read_add_handler(): fcntl(F_SETOWN)");
|
|
}
|
|
/* set file access mode to non-blocking */
|
|
if (real_fcntl(fd, F_SETFL, O_NONBLOCK | O_ASYNC) == -1) {
|
|
err(EXIT_FAILURE, "native_async_read_add_handler(): fcntl(F_SETFL)");
|
|
}
|
|
#endif /* not OSX */
|
|
|
|
_next_index++;
|
|
}
|
|
|
|
void native_async_read_add_int_handler(int fd, void *arg, native_async_read_callback_t handler) {
|
|
if (_next_index >= ASYNC_READ_NUMOF) {
|
|
err(EXIT_FAILURE, "native_async_read_add_int_handler(): too many callbacks");
|
|
}
|
|
|
|
_add_handler(fd, arg, handler);
|
|
|
|
_sigio_child(_next_index);
|
|
_next_index++;
|
|
}
|
|
|
|
static void _sigio_child(int index)
|
|
{
|
|
struct pollfd fds = _fds[index];
|
|
async_read_t *poll = &pollers[_next_index];
|
|
pid_t parent = _native_pid;
|
|
pid_t child;
|
|
if ((child = real_fork()) == -1) {
|
|
err(EXIT_FAILURE, "sigio_child: fork");
|
|
}
|
|
if (child > 0) {
|
|
poll->child_pid = child;
|
|
|
|
/* return in parent process */
|
|
return;
|
|
}
|
|
|
|
sigset_t sigmask;
|
|
|
|
sigemptyset(&sigmask);
|
|
sigaddset(&sigmask, SIGCONT);
|
|
sigprocmask(SIG_BLOCK, &sigmask, NULL);
|
|
|
|
/* watch tap interface and signal parent process if data is
|
|
* available */
|
|
while (1) {
|
|
if (real_poll(&fds, 1, -1) == 1) {
|
|
kill(parent, SIGIO);
|
|
}
|
|
else {
|
|
kill(parent, SIGKILL);
|
|
err(EXIT_FAILURE, "osx_sigio_child: select");
|
|
}
|
|
|
|
/* If SIGCONT is sent before calling pause(), the process stops
|
|
* forever, so using sigwait instead. */
|
|
|
|
int sig;
|
|
|
|
sigemptyset(&sigmask);
|
|
sigaddset(&sigmask, SIGCONT);
|
|
sigwait(&sigmask, &sig);
|
|
}
|
|
}
|
|
/** @} */
|