2015-04-22 11:54:24 +02:00
|
|
|
/*
|
|
|
|
* Copyright (C) 2015 Kaspar Schleiser <kaspar@schleiser.de>
|
|
|
|
* 2014 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.
|
|
|
|
*/
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @ingroup sys_newlib
|
|
|
|
* @{
|
|
|
|
*
|
|
|
|
* @file
|
|
|
|
* @brief Newlib system call implementations
|
|
|
|
*
|
|
|
|
* @author Michael Baar <michael.baar@fu-berlin.de>
|
|
|
|
* @author Stefan Pfeiffer <pfeiffer@inf.fu-berlin.de>
|
|
|
|
* @author Hauke Petersen <hauke.petersen@fu-berlin.de>
|
|
|
|
* @author Kaspar Schleiser <kaspar@schleiser.de>
|
|
|
|
*
|
|
|
|
* @}
|
|
|
|
*/
|
|
|
|
|
2016-03-31 10:32:51 +02:00
|
|
|
#include <unistd.h>
|
|
|
|
#include <reent.h>
|
2015-04-22 11:54:24 +02:00
|
|
|
#include <errno.h>
|
|
|
|
#include <stdio.h>
|
|
|
|
#include <stdlib.h>
|
|
|
|
#include <string.h>
|
|
|
|
#include <sys/stat.h>
|
|
|
|
#include <sys/unistd.h>
|
|
|
|
#include <stdint.h>
|
|
|
|
|
|
|
|
#include "cpu.h"
|
|
|
|
#include "board.h"
|
2016-02-29 21:55:56 +01:00
|
|
|
#include "sched.h"
|
2015-04-22 11:54:24 +02:00
|
|
|
#include "thread.h"
|
2015-09-29 13:41:33 +02:00
|
|
|
#include "reboot.h"
|
2015-04-22 11:54:24 +02:00
|
|
|
#include "irq.h"
|
2015-09-29 13:41:33 +02:00
|
|
|
#include "log.h"
|
2015-04-22 11:54:24 +02:00
|
|
|
|
2015-06-04 11:50:03 +02:00
|
|
|
#include "uart_stdio.h"
|
2015-04-22 11:54:24 +02:00
|
|
|
|
2015-11-17 12:50:48 +01:00
|
|
|
#ifdef MODULE_XTIMER
|
|
|
|
#include <sys/time.h>
|
|
|
|
#include "div.h"
|
|
|
|
#include "xtimer.h"
|
|
|
|
#endif
|
|
|
|
|
2015-04-22 11:54:24 +02:00
|
|
|
/**
|
|
|
|
* @brief manage the heap
|
|
|
|
*/
|
|
|
|
extern char _sheap; /* start of the heap */
|
|
|
|
extern char _eheap; /* end of the heap */
|
2015-08-04 21:05:08 +02:00
|
|
|
char *heap_top = &_sheap + 4;
|
2015-04-22 11:54:24 +02:00
|
|
|
|
|
|
|
/**
|
|
|
|
* @brief Initialize NewLib, called by __libc_init_array() from the startup script
|
|
|
|
*/
|
|
|
|
void _init(void)
|
|
|
|
{
|
2015-06-04 11:50:03 +02:00
|
|
|
uart_stdio_init();
|
2015-04-22 11:54:24 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @brief Free resources on NewLib de-initialization, not used for RIOT
|
|
|
|
*/
|
|
|
|
void _fini(void)
|
|
|
|
{
|
|
|
|
/* nothing to do here */
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @brief Exit a program without cleaning up files
|
|
|
|
*
|
|
|
|
* If your system doesn't provide this, it is best to avoid linking with subroutines that
|
|
|
|
* require it (exit, system).
|
|
|
|
*
|
|
|
|
* @param n the exit code, 0 for all OK, >0 for not OK
|
|
|
|
*/
|
|
|
|
void _exit(int n)
|
|
|
|
{
|
2015-09-29 13:41:33 +02:00
|
|
|
LOG_INFO("#! exit %i: resetting\n", n);
|
|
|
|
reboot();
|
2015-04-22 11:54:24 +02:00
|
|
|
while(1);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @brief Allocate memory from the heap.
|
|
|
|
*
|
|
|
|
* The current heap implementation is very rudimentary, it is only able to allocate
|
2015-11-23 19:20:47 +01:00
|
|
|
* memory. But it does not have any means to free memory again
|
2015-04-22 11:54:24 +02:00
|
|
|
*
|
2015-11-23 19:20:47 +01:00
|
|
|
* @return pointer to the newly allocated memory on success
|
|
|
|
* @return pointer set to address `-1` on failure
|
2015-04-22 11:54:24 +02:00
|
|
|
*/
|
2015-08-04 21:05:08 +02:00
|
|
|
void *_sbrk_r(struct _reent *r, ptrdiff_t incr)
|
2015-04-22 11:54:24 +02:00
|
|
|
{
|
2016-03-19 09:25:47 +01:00
|
|
|
unsigned int state = irq_disable();
|
2015-08-04 21:05:08 +02:00
|
|
|
void *res = heap_top;
|
2015-04-22 11:54:24 +02:00
|
|
|
|
2015-08-04 21:05:08 +02:00
|
|
|
if ((heap_top + incr > &_eheap) || (heap_top + incr < &_sheap)) {
|
2015-04-22 11:54:24 +02:00
|
|
|
r->_errno = ENOMEM;
|
2015-08-04 21:05:08 +02:00
|
|
|
res = (void *)-1;
|
2015-04-22 11:54:24 +02:00
|
|
|
}
|
|
|
|
else {
|
|
|
|
heap_top += incr;
|
|
|
|
}
|
|
|
|
|
2016-03-19 09:25:47 +01:00
|
|
|
irq_restore(state);
|
2015-04-22 11:54:24 +02:00
|
|
|
return res;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @brief Get the process-ID of the current thread
|
|
|
|
*
|
|
|
|
* @return the process ID of the current thread
|
|
|
|
*/
|
2015-08-04 21:04:34 +02:00
|
|
|
pid_t _getpid(void)
|
2015-04-22 11:54:24 +02:00
|
|
|
{
|
|
|
|
return sched_active_pid;
|
|
|
|
}
|
|
|
|
|
2015-08-04 19:55:49 +02:00
|
|
|
/**
|
|
|
|
* @brief Get the process-ID of the current thread
|
|
|
|
*
|
|
|
|
* @return the process ID of the current thread
|
|
|
|
*/
|
|
|
|
pid_t _getpid_r(struct _reent *ptr)
|
|
|
|
{
|
|
|
|
(void) ptr;
|
|
|
|
return sched_active_pid;
|
|
|
|
}
|
|
|
|
|
2015-04-22 11:54:24 +02:00
|
|
|
/**
|
|
|
|
* @brief Send a signal to a given thread
|
|
|
|
*
|
|
|
|
* @param r TODO
|
|
|
|
* @param pid TODO
|
|
|
|
* @param sig TODO
|
|
|
|
*
|
|
|
|
* @return TODO
|
|
|
|
*/
|
|
|
|
__attribute__ ((weak))
|
2015-08-04 21:04:34 +02:00
|
|
|
int _kill_r(struct _reent *r, pid_t pid, int sig)
|
2015-04-22 11:54:24 +02:00
|
|
|
{
|
2015-05-27 17:35:57 +02:00
|
|
|
(void) pid;
|
|
|
|
(void) sig;
|
2015-04-22 11:54:24 +02:00
|
|
|
r->_errno = ESRCH; /* not implemented yet */
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @brief Open a file
|
|
|
|
*
|
|
|
|
* @param r TODO
|
|
|
|
* @param name TODO
|
|
|
|
* @param mode TODO
|
|
|
|
*
|
|
|
|
* @return TODO
|
|
|
|
*/
|
2016-03-31 10:32:51 +02:00
|
|
|
int _open_r(struct _reent *r, const char *name, int flags, int mode)
|
2015-04-22 11:54:24 +02:00
|
|
|
{
|
2015-05-27 17:35:57 +02:00
|
|
|
(void) name;
|
2016-03-31 10:32:51 +02:00
|
|
|
(void) flags;
|
2015-05-27 17:35:57 +02:00
|
|
|
(void) mode;
|
2015-04-22 11:54:24 +02:00
|
|
|
r->_errno = ENODEV; /* not implemented yet */
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @brief Read from a file
|
|
|
|
*
|
|
|
|
* All input is read from UART_0. The function will block until a byte is actually read.
|
|
|
|
*
|
|
|
|
* Note: the read function does not buffer - data will be lost if the function is not
|
|
|
|
* called fast enough.
|
|
|
|
*
|
|
|
|
* TODO: implement more sophisticated read call.
|
|
|
|
*
|
|
|
|
* @param r TODO
|
|
|
|
* @param fd TODO
|
|
|
|
* @param buffer TODO
|
|
|
|
* @param int TODO
|
|
|
|
*
|
|
|
|
* @return TODO
|
|
|
|
*/
|
|
|
|
int _read_r(struct _reent *r, int fd, void *buffer, unsigned int count)
|
|
|
|
{
|
2015-07-04 11:16:54 +02:00
|
|
|
(void)r;
|
|
|
|
(void)fd;
|
2015-06-04 11:50:03 +02:00
|
|
|
return uart_stdio_read(buffer, count);
|
2015-04-22 11:54:24 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @brief Write characters to a file
|
|
|
|
*
|
|
|
|
* All output is currently directed to UART_0, independent of the given file descriptor.
|
|
|
|
* The write call will further block until the byte is actually written to the UART.
|
|
|
|
*
|
|
|
|
* TODO: implement more sophisticated write call.
|
|
|
|
*
|
|
|
|
* @param r TODO
|
|
|
|
* @param fd TODO
|
|
|
|
* @param data TODO
|
|
|
|
* @param int TODO
|
|
|
|
*
|
|
|
|
* @return TODO
|
|
|
|
*/
|
|
|
|
int _write_r(struct _reent *r, int fd, const void *data, unsigned int count)
|
|
|
|
{
|
2015-05-27 17:35:57 +02:00
|
|
|
(void) r;
|
|
|
|
(void) fd;
|
2015-06-04 11:50:03 +02:00
|
|
|
return uart_stdio_write(data, count);
|
2015-04-22 11:54:24 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @brief Close a file
|
|
|
|
*
|
|
|
|
* @param r TODO
|
|
|
|
* @param fd TODO
|
|
|
|
*
|
|
|
|
* @return TODO
|
|
|
|
*/
|
|
|
|
int _close_r(struct _reent *r, int fd)
|
|
|
|
{
|
2015-05-27 17:35:57 +02:00
|
|
|
(void) fd;
|
2015-04-22 11:54:24 +02:00
|
|
|
r->_errno = ENODEV; /* not implemented yet */
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @brief Set position in a file
|
|
|
|
*
|
|
|
|
* @param r TODO
|
|
|
|
* @param fd TODO
|
|
|
|
* @param pos TODO
|
|
|
|
* @param dir TODO
|
|
|
|
*
|
|
|
|
* @return TODO
|
|
|
|
*/
|
|
|
|
_off_t _lseek_r(struct _reent *r, int fd, _off_t pos, int dir)
|
|
|
|
{
|
2015-05-27 17:35:57 +02:00
|
|
|
(void) fd;
|
|
|
|
(void) pos;
|
|
|
|
(void) dir;
|
2015-04-22 11:54:24 +02:00
|
|
|
r->_errno = ENODEV; /* not implemented yet */
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @brief Status of an open file
|
|
|
|
*
|
|
|
|
* @param r TODO
|
|
|
|
* @param fd TODO
|
|
|
|
* @param stat TODO
|
|
|
|
*
|
|
|
|
* @return TODO
|
|
|
|
*/
|
2015-05-27 17:35:57 +02:00
|
|
|
int _fstat_r(struct _reent *r, int fd, struct stat *st)
|
2015-04-22 11:54:24 +02:00
|
|
|
{
|
2015-05-27 17:35:57 +02:00
|
|
|
(void) fd;
|
|
|
|
(void) st;
|
2015-04-22 11:54:24 +02:00
|
|
|
r->_errno = ENODEV; /* not implemented yet */
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @brief Status of a file (by name)
|
|
|
|
*
|
|
|
|
* @param r TODO
|
|
|
|
* @param name TODO
|
|
|
|
* @param stat TODO
|
|
|
|
*
|
|
|
|
* @return TODO
|
|
|
|
*/
|
2016-03-31 10:32:51 +02:00
|
|
|
int _stat_r(struct _reent *r, const char *name, struct stat *st)
|
2015-04-22 11:54:24 +02:00
|
|
|
{
|
2015-05-27 17:35:57 +02:00
|
|
|
(void) name;
|
|
|
|
(void) st;
|
2015-04-22 11:54:24 +02:00
|
|
|
r->_errno = ENODEV; /* not implemented yet */
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @brief Query whether output stream is a terminal
|
|
|
|
*
|
|
|
|
* @param r TODO
|
|
|
|
* @param fd TODO
|
|
|
|
*
|
|
|
|
* @return TODO
|
|
|
|
*/
|
|
|
|
int _isatty_r(struct _reent *r, int fd)
|
|
|
|
{
|
|
|
|
r->_errno = 0;
|
|
|
|
|
|
|
|
if(fd == STDIN_FILENO || fd == STDOUT_FILENO || fd == STDERR_FILENO) {
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @brief Remove a file's directory entry
|
|
|
|
*
|
|
|
|
* @param r TODO
|
|
|
|
* @param path TODO
|
|
|
|
*
|
|
|
|
* @return TODO
|
|
|
|
*/
|
2016-03-31 10:32:51 +02:00
|
|
|
int _unlink_r(struct _reent *r, const char *path)
|
2015-04-22 11:54:24 +02:00
|
|
|
{
|
2015-05-27 17:35:57 +02:00
|
|
|
(void) path;
|
2015-04-22 11:54:24 +02:00
|
|
|
r->_errno = ENODEV; /* not implemented yet */
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @brief Send a signal to a thread
|
|
|
|
*
|
|
|
|
* @param[in] pid the pid to send to
|
|
|
|
* @param[in] sig the signal to send
|
|
|
|
*
|
|
|
|
* @return TODO
|
|
|
|
*/
|
|
|
|
__attribute__ ((weak))
|
2015-08-04 21:04:34 +02:00
|
|
|
int _kill(pid_t pid, int sig)
|
2015-04-22 11:54:24 +02:00
|
|
|
{
|
2015-05-27 17:35:57 +02:00
|
|
|
(void) pid;
|
|
|
|
(void) sig;
|
2015-04-22 11:54:24 +02:00
|
|
|
errno = ESRCH; /* not implemented yet */
|
|
|
|
return -1;
|
|
|
|
}
|
2015-11-17 12:50:48 +01:00
|
|
|
|
|
|
|
#ifdef MODULE_XTIMER
|
|
|
|
int _gettimeofday_r(struct _reent *r, struct timeval *restrict tp, void *restrict tzp)
|
|
|
|
{
|
|
|
|
(void)tzp;
|
2015-11-28 12:46:30 +01:00
|
|
|
(void) r;
|
2015-11-17 12:50:48 +01:00
|
|
|
uint64_t now = xtimer_now64();
|
|
|
|
tp->tv_sec = div_u64_by_1000000(now);
|
|
|
|
tp->tv_usec = now - (tp->tv_sec * SEC_IN_USEC);
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
#endif
|