2013-09-21 15:23:40 +02:00
|
|
|
/*
|
|
|
|
* Copyright (C) 2013 Freie Universität Berlin
|
|
|
|
*
|
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-09-21 15:23:40 +02:00
|
|
|
*/
|
|
|
|
|
2014-01-31 15:00:21 +01:00
|
|
|
/**
|
|
|
|
* @ingroup posix
|
|
|
|
* @{
|
|
|
|
* @file fd.c
|
|
|
|
* @brief Providing unifying file descriptor wrapper for POSIX-compliant
|
|
|
|
* operations.
|
|
|
|
* @author Christian Mehlis <mehlis@inf.fu-berlin.de>
|
2015-02-08 18:51:25 +01:00
|
|
|
* @author Martine Lenders <mlenders@inf.fu-berlin.de>
|
2014-01-31 15:00:21 +01:00
|
|
|
*/
|
2013-09-21 15:23:40 +02:00
|
|
|
#include <errno.h>
|
|
|
|
#include <stdio.h>
|
|
|
|
#include <stdlib.h>
|
|
|
|
#include <string.h>
|
|
|
|
|
|
|
|
#include "posix_io.h"
|
2014-10-02 10:12:28 +02:00
|
|
|
#ifdef MODULE_UART0
|
2013-09-21 15:23:40 +02:00
|
|
|
#include "board_uart0.h"
|
2014-10-02 10:12:28 +02:00
|
|
|
#endif
|
2013-09-21 15:23:40 +02:00
|
|
|
#include "unistd.h"
|
|
|
|
|
|
|
|
#include "fd.h"
|
|
|
|
|
2014-01-20 14:05:21 +01:00
|
|
|
#ifdef CPU_MSP430
|
2013-09-21 15:23:40 +02:00
|
|
|
#define FD_MAX 5
|
|
|
|
#else
|
|
|
|
#define FD_MAX 15
|
|
|
|
#endif
|
|
|
|
|
|
|
|
static fd_t fd_table[FD_MAX];
|
|
|
|
|
|
|
|
int fd_init(void)
|
|
|
|
{
|
|
|
|
memset(fd_table, 0, sizeof(fd_t) * FD_MAX);
|
|
|
|
|
2014-10-02 10:12:28 +02:00
|
|
|
#ifdef MODULE_UART0
|
2013-09-21 15:23:40 +02:00
|
|
|
posix_open(uart0_handler_pid, 0);
|
|
|
|
fd_t fd = {
|
2015-03-11 17:13:26 +01:00
|
|
|
.internal_active = 1,
|
|
|
|
.internal_fd = (int)uart0_handler_pid,
|
2013-09-21 15:23:40 +02:00
|
|
|
.read = (ssize_t ( *)(int, void *, size_t))posix_read,
|
|
|
|
.write = (ssize_t ( *)(int, const void *, size_t))posix_write,
|
|
|
|
.close = posix_close
|
|
|
|
};
|
|
|
|
memcpy(&fd_table[STDIN_FILENO], &fd, sizeof(fd_t));
|
|
|
|
memcpy(&fd_table[STDOUT_FILENO], &fd, sizeof(fd_t));
|
|
|
|
memcpy(&fd_table[STDERR_FILENO], &fd, sizeof(fd_t));
|
2014-10-02 10:12:28 +02:00
|
|
|
#endif
|
2013-09-21 15:23:40 +02:00
|
|
|
return FD_MAX;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int fd_get_next_free(void)
|
|
|
|
{
|
|
|
|
for (int i = 0; i < FD_MAX; i++) {
|
|
|
|
fd_t *cur = &fd_table[i];
|
|
|
|
|
2015-03-11 17:13:26 +01:00
|
|
|
if (!cur->internal_active) {
|
2013-09-21 15:23:40 +02:00
|
|
|
return i;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
int fd_new(int internal_fd, ssize_t (*internal_read)(int, void *, size_t),
|
|
|
|
ssize_t (*internal_write)(int, const void *, size_t),
|
2014-08-11 13:10:12 +02:00
|
|
|
int (*internal_close)(int))
|
2013-09-21 15:23:40 +02:00
|
|
|
{
|
|
|
|
int fd = fd_get_next_free();
|
|
|
|
|
|
|
|
if (fd >= 0) {
|
|
|
|
fd_t *fd_s = fd_get(fd);
|
2015-03-11 17:13:26 +01:00
|
|
|
fd_s->internal_active = 1;
|
|
|
|
fd_s->internal_fd = internal_fd;
|
2013-09-21 15:23:40 +02:00
|
|
|
fd_s->read = internal_read;
|
|
|
|
fd_s->write = internal_write;
|
|
|
|
fd_s->close = internal_close;
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
errno = ENFILE;
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
return fd;
|
|
|
|
}
|
|
|
|
|
|
|
|
fd_t *fd_get(int fd)
|
|
|
|
{
|
|
|
|
if (fd >= 0 && fd < FD_MAX) {
|
|
|
|
return &fd_table[fd];
|
|
|
|
}
|
|
|
|
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
void fd_destroy(int fd)
|
|
|
|
{
|
|
|
|
fd_t *cur = fd_get(fd);
|
|
|
|
|
|
|
|
if (!cur) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
memset(cur, 0, sizeof(fd_t));
|
|
|
|
}
|
2014-01-31 15:00:21 +01:00
|
|
|
|
|
|
|
/**
|
|
|
|
* @}
|
|
|
|
*/
|