/* * Copyright (c) 2006-2021, RT-Thread Development Team * * SPDX-License-Identifier: Apache-2.0 * * Change Logs: * Date Author Notes * 2021-02-11 Meco Man remove _gettimeofday_r() and _times_r() * 2020-02-13 Meco Man re-implement exit() and abort() * 2020-02-21 Meco Man improve and beautify syscalls * 2020-02-24 Meco Man fix bug of _isatty_r() */ #include #include #include #include #include #include #include #include #include #ifdef RT_USING_POSIX_DEVIO #include "libc.h" #endif /* RT_USING_POSIX_DEVIO */ #ifdef RT_USING_MODULE #include #endif /* RT_USING_MODULE */ #define DBG_TAG "newlib.syscalls" #define DBG_LVL DBG_INFO #include #ifdef RT_USING_HEAP /* Memory routine */ void *_malloc_r (struct _reent *ptr, size_t size) { void* result; result = (void*)rt_malloc (size); if (result == RT_NULL) { ptr->_errno = ENOMEM; } return result; } void *_realloc_r (struct _reent *ptr, void *old, size_t newlen) { void* result; result = (void*)rt_realloc (old, newlen); if (result == RT_NULL) { ptr->_errno = ENOMEM; } return result; } void *_calloc_r (struct _reent *ptr, size_t size, size_t len) { void* result; result = (void*)rt_calloc (size, len); if (result == RT_NULL) { ptr->_errno = ENOMEM; } return result; } void _free_r (struct _reent *ptr, void *addr) { rt_free (addr); } #else void * _sbrk_r(struct _reent *ptr, ptrdiff_t incr) { LOG_E("Please enable RT_USING_HEAP"); RT_ASSERT(0); return RT_NULL; } #endif /*RT_USING_HEAP*/ void __libc_init_array(void) { /* we not use __libc init_aray to initialize C++ objects */ /* __libc_init_array is ARM code, not Thumb; it will cause a hardfault. */ } #ifdef RT_USING_LIBC /* Reentrant versions of system calls. */ #ifndef _REENT_ONLY int *__errno () { return _rt_errno(); } #endif int _getpid_r(struct _reent *ptr) { return 0; } int _close_r(struct _reent *ptr, int fd) { #ifdef DFS_USING_POSIX return close(fd); #else ptr->_errno = ENOTSUP; return -1; #endif /* DFS_USING_POSIX */ } int _execve_r(struct _reent *ptr, const char * name, char *const *argv, char *const *env) { ptr->_errno = ENOTSUP; return -1; } int _fcntl_r(struct _reent *ptr, int fd, int cmd, int arg) { ptr->_errno = ENOTSUP; return -1; } int _fork_r(struct _reent *ptr) { ptr->_errno = ENOTSUP; return -1; } int _fstat_r(struct _reent *ptr, int fd, struct stat *pstat) { ptr->_errno = ENOTSUP; return -1; } int _isatty_r(struct _reent *ptr, int fd) { if (fd >=0 && fd < 3) { return 1; } else { return 0; } } int _kill_r(struct _reent *ptr, int pid, int sig) { ptr->_errno = ENOTSUP; return -1; } int _link_r(struct _reent *ptr, const char *old, const char *new) { ptr->_errno = ENOTSUP; return -1; } int _wait_r(struct _reent *ptr, int *status) { ptr->_errno = ENOTSUP; return -1; } mode_t umask(mode_t mask) { return 022; } int flock(int fd, int operation) { return 0; } _off_t _lseek_r(struct _reent *ptr, int fd, _off_t pos, int whence) { #ifdef DFS_USING_POSIX _off_t rc; rc = lseek(fd, pos, whence); return rc; #else ptr->_errno = ENOTSUP; return -1; #endif /* DFS_USING_POSIX */ } int _mkdir_r(struct _reent *ptr, const char *name, int mode) { #ifdef DFS_USING_POSIX int rc; rc = mkdir(name, mode); return rc; #else ptr->_errno = ENOTSUP; return -1; #endif /* DFS_USING_POSIX */ } int _open_r(struct _reent *ptr, const char *file, int flags, int mode) { #ifdef DFS_USING_POSIX int rc; rc = open(file, flags, mode); return rc; #else ptr->_errno = ENOTSUP; return -1; #endif /* DFS_USING_POSIX */ } _ssize_t _read_r(struct _reent *ptr, int fd, void *buf, size_t nbytes) { #ifdef DFS_USING_POSIX _ssize_t rc; if (fd == STDIN_FILENO) { #ifdef RT_USING_POSIX_DEVIO if (libc_stdio_get_console() < 0) { LOG_W("Do not invoke standard input before initializing libc"); return 0; } #else ptr->_errno = ENOTSUP; return -1; #endif /* RT_USING_POSIX_DEVIO */ } else if (fd == STDOUT_FILENO || fd == STDERR_FILENO) { ptr->_errno = ENOTSUP; return -1; } rc = read(fd, buf, nbytes); return rc; #else ptr->_errno = ENOTSUP; return -1; #endif /* DFS_USING_POSIX */ } int _rename_r(struct _reent *ptr, const char *old, const char *new) { #ifdef DFS_USING_POSIX int rc; rc = rename(old, new); return rc; #else ptr->_errno = ENOTSUP; return -1; #endif /* DFS_USING_POSIX */ } int _stat_r(struct _reent *ptr, const char *file, struct stat *pstat) { #ifdef DFS_USING_POSIX int rc; rc = stat(file, pstat); return rc; #else ptr->_errno = ENOTSUP; return -1; #endif /* DFS_USING_POSIX */ } int _unlink_r(struct _reent *ptr, const char *file) { #ifdef DFS_USING_POSIX return unlink(file); #else ptr->_errno = ENOTSUP; return -1; #endif /* DFS_USING_POSIX */ } _ssize_t _write_r(struct _reent *ptr, int fd, const void *buf, size_t nbytes) { #ifdef DFS_USING_POSIX _ssize_t rc; #endif /* DFS_USING_POSIX */ if (fd == STDOUT_FILENO || fd == STDERR_FILENO) { #ifdef RT_USING_CONSOLE rt_device_t console; console = rt_console_get_device(); if (console) return rt_device_write(console, -1, buf, nbytes); #else ptr->_errno = ENOTSUP; return -1; #endif /* RT_USING_CONSOLE */ } else if (fd == STDIN_FILENO) { ptr->_errno = ENOTSUP; return -1; } #ifdef DFS_USING_POSIX rc = write(fd, buf, nbytes); return rc; #else ptr->_errno = ENOTSUP; return -1; #endif /* DFS_USING_POSIX */ } /* for exit() and abort() */ __attribute__ ((noreturn)) void _exit (int status) { extern void __rt_libc_exit(int status); __rt_libc_exit(status); while(1); } /* These functions are implemented and replaced by the 'common/time.c' file int _gettimeofday_r(struct _reent *ptr, struct timeval *__tp, void *__tzp); _CLOCK_T_ _times_r(struct _reent *ptr, struct tms *ptms); */ #endif /* RT_USING_LIBC */