/* * Copyright (C) 2017 Freie Universität Berlin * * 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. */ /** * @{ * * @file * @brief Implements various POSIX syscalls * @author Martine Lenders */ #include #include #include #include #include "irq.h" #ifdef MODULE_VFS #include #include "vfs.h" #elif defined(MODULE_STDIO_UART) #include "stdio_uart.h" #endif int open(const char *name, int flags, ...) { #ifdef MODULE_VFS unsigned mode = 0; if ((flags & O_CREAT)) { va_list ap; va_start(ap, flags); mode = va_arg(ap, unsigned); va_end(ap); } int fd = vfs_open(name, flags, mode); if (fd < 0) { /* vfs returns negative error codes */ errno = -fd; return -1; } return fd; #else (void)name; (void)flags; errno = ENODEV; return -1; #endif } int close(int fd) { #ifdef MODULE_VFS int res = vfs_close(fd); if (res < 0) { errno = -res; return -1; } return res; #else (void)fd; errno = ENODEV; return -1; #endif } off_t lseek(int fd, off_t off, int whence) { #ifdef MODULE_VFS off_t res = vfs_lseek(fd, off, whence); if (res < 0) { errno = -res; return -1; } return res; #else (void)fd; (void)off; (void)whence; errno = ENODEV; return -1; #endif } int fcntl(int fd, int cmd, ...) { #ifdef MODULE_VFS unsigned long arg; va_list ap; va_start(ap, cmd); arg = va_arg(ap, unsigned long); va_end(ap); int res = vfs_fcntl(fd, cmd, arg); if (res < 0) { errno = -res; return -1; } return res; #else (void)fd; (void)cmd; errno = ENODEV; return -1; #endif } int fstat(int fd, struct stat *buf) { #ifdef MODULE_VFS int res = vfs_fstat(fd, buf); if (res < 0) { errno = -res; return -1; } return res; #else (void)fd; (void)buf; errno = ENODEV; return -1; #endif } ssize_t read(int fd, void *dest, size_t count) { #ifdef MODULE_VFS ssize_t res = vfs_read(fd, dest, count); if (res < 0) { errno = -res; return -1; } return res; #elif defined(MODULE_STDIO_UART) if (fd == 0) { return stdio_read(dest, count); } errno = EBADF; return -1; #else (void)fd; (void)dest; (void)count; errno = ENODEV; return -1; #endif } ssize_t write(int fd, const void *src, size_t count) { #ifdef MODULE_VFS ssize_t res = vfs_write(fd, src, count); if (res < 0) { errno = -res; return -1; } return res; #elif defined(MODULE_STDIO_UART) if (fd == 0) { return stdio_write(src, count); } errno = EBADF; return -1; #else (void)fd; (void)src; (void)count; errno = ENODEV; return -1; #endif } /* * Following functions are wrappers around the according avr-libc system * functions to avoid their preemption by disabling the interrupts for the * time of their execution. */ extern void *__real_malloc(size_t size); extern void __real_free(void *ptr); extern void *__real_calloc(size_t nmemb, size_t size); extern void *__real_realloc(void *ptr, size_t size); void *__wrap_malloc(size_t size) { unsigned state = irq_disable(); void *ptr = __real_malloc(size); irq_restore(state); return ptr; } void __wrap_free(void *ptr) { unsigned state = irq_disable(); __real_free(ptr); irq_restore(state); } void *__wrap_calloc(size_t nmemb, size_t size) { unsigned state = irq_disable(); void *ptr = __real_calloc(nmemb, size); irq_restore(state); return ptr; } void *__wrap_realloc(void *ptr, size_t size) { unsigned state = irq_disable(); void *new = __real_realloc(ptr, size); irq_restore(state); return new; } /** @} */