| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301 | /* * newlib_stubs.c * *  Created on: 2 Nov 2010 *      Author: nanoage.co.uk */#include <errno.h>#include <sys/stat.h>#include <sys/times.h>#include <sys/unistd.h>#include "stm32f4xx_usart.h"#include "stm32f4xx.h"#ifdef SWOTRACE/* Set STDIO_USART to 255 as flag that trace output on SWO pin used instead */#define STDIO_USART 255#else/* Set USART number */#define STDIO_USART 1 /* Disable UART stdio */#endif /* SWOTRACE */#ifndef STDOUT_USART#define STDOUT_USART STDIO_USART#endif#ifndef STDERR_USART#define STDERR_USART STDIO_USART#endif#ifndef STDIN_USART#define STDIN_USART STDIO_USART#endif#undef errnoextern int errno;/* Variable to receive characters on SWO */volatile int32_t ITM_RxBuffer;/* environ A pointer to a list of environment variables and their values. For a minimal environment, this empty list is adequate: */char *__env[1] = { 0 };char **environ = __env;int _write(int file, char *ptr, int len);void _exit(int status) {    _write(1, "exit", 4);    while (1) {        ;    }}int _close(int file) {    return -1;}/* execve Transfer control to a new process. Minimal implementation (for a system without processes): */int _execve(char *name, char **argv, char **env) {    errno = ENOMEM;    return -1;}/* fork Create a new process. Minimal implementation (for a system without processes): */int _fork() {    errno = EAGAIN;    return -1;}/* fstat Status of an open file. For consistency with other minimal implementations in these examples, all files are regarded as character special devices. The `sys/stat.h' header file required is distributed in the `include' subdirectory for this C library. */int _fstat(int file, struct stat *st) {    st->st_mode = S_IFCHR;    return 0;}/* getpid Process-ID; this is sometimes used to generate strings unlikely to conflict with other processes. Minimal implementation, for a system without processes: */int _getpid() {    return 1;}/* isatty Query whether output stream is a terminal. For consistency with the other minimal implementations, */int _isatty(int file) {    switch (file){    case STDOUT_FILENO:    case STDERR_FILENO:    case STDIN_FILENO:        return 1;    default:        //errno = ENOTTY;        errno = EBADF;        return 0;    }}/* kill Send a signal. Minimal implementation: */int _kill(int pid, int sig) {    errno = EINVAL;    return (-1);}/* link Establish a new name for an existing file. Minimal implementation: */int _link(char *old, char *new) {    errno = EMLINK;    return -1;}/* lseek Set position in a file. Minimal implementation: */int _lseek(int file, int ptr, int dir) {    return 0;}/* sbrk Increase program data space. Malloc and related functions depend on this */caddr_t _sbrk(int incr) {    extern char _ebss; // Defined by the linker    extern char __bss_end__;    static char *heap_end;    char *prev_heap_end;    if (heap_end == 0) {    	//heap_end = &_ebss;    	heap_end = &__bss_end__;    }    prev_heap_end = heap_end;char * stack = (char*) __get_MSP();     if (heap_end + incr >  stack)     {         _write (STDERR_FILENO, "Heap and stack collision\n", 25);         errno = ENOMEM;         return  (caddr_t) -1;         //abort ();     }    heap_end += incr;    return (caddr_t) prev_heap_end;}/* read Read a character to a file. `libc' subroutines will use this system routine for input from all files, including stdin Returns -1 on error or blocks until the number of characters have been read. */int _read(int file, char *ptr, int len) {    int n;    int num = 0;    switch (file) {    case STDIN_FILENO:        for (n = 0; n < len; n++) {#if   STDIN_USART == 1            while ((USART1->SR & USART_FLAG_RXNE) == (uint16_t)RESET) {}            char c = (char)(USART1->DR & (uint16_t)0x01FF);#elif STDIN_USART == 2            while ((USART2->SR & USART_FLAG_RXNE) == (uint16_t) RESET) {}            char c = (char) (USART2->DR & (uint16_t) 0x01FF);#elif STDIN_USART == 3            while ((USART3->SR & USART_FLAG_RXNE) == (uint16_t)RESET) {}            char c = (char)(USART3->DR & (uint16_t)0x01FF);#elif STDIN_USART == 255            /* TODO Test this feature */            char c = (char)ITM_ReceiveChar();#elif STDIN_USART == 0            /* Return NULL */            char c = '\0';#endif            *ptr++ = c;            num++;        }        break;    default:        errno = EBADF;        return -1;    }    return num;}/* stat Status of a file (by name). Minimal implementation: int    _EXFUN(stat,( const char *__path, struct stat *__sbuf )); */int _stat(const char *filepath, struct stat *st) {    st->st_mode = S_IFCHR;    return 0;}/* times Timing information for current process. Minimal implementation: */clock_t _times(struct tms *buf) {    return -1;}/* unlink Remove a file's directory entry. Minimal implementation: */int _unlink(char *name) {    errno = ENOENT;    return -1;}/* wait Wait for a child process. Minimal implementation: */int _wait(int *status) {    errno = ECHILD;    return -1;}/* write Write a character to a file. `libc' subroutines will use this system routine for output to all files, including stdout Returns -1 on error or number of bytes sent */int _write(int file, char *ptr, int len) {    int n;    switch (file) {    case STDOUT_FILENO: /*stdout*/        for (n = 0; n < len; n++) {#if STDOUT_USART == 1            while ((USART1->SR & USART_FLAG_TC) == (uint16_t)RESET) {}            USART1->DR = (*ptr++ & (uint16_t)0x01FF);#elif  STDOUT_USART == 2            while ((USART2->SR & USART_FLAG_TC) == (uint16_t) RESET) {            }            USART2->DR = (*ptr++ & (uint16_t) 0x01FF);#elif  STDOUT_USART == 3            while ((USART3->SR & USART_FLAG_TC) == (uint16_t)RESET) {}            USART3->DR = (*ptr++ & (uint16_t)0x01FF);#elif STDOUT_USART == 255            ITM_SendChar(*ptr++ & (uint16_t)0x00FF);#endif        }        break;    case STDERR_FILENO: /* stderr */        for (n = 0; n < len; n++) {#if STDERR_USART == 1            while ((USART1->SR & USART_FLAG_TC) == (uint16_t)RESET) {}            USART1->DR = (*ptr++ & (uint16_t)0x01FF);#elif  STDERR_USART == 2            while ((USART2->SR & USART_FLAG_TC) == (uint16_t) RESET) {            }            USART2->DR = (*ptr++ & (uint16_t) 0x01FF);#elif  STDERR_USART == 3            while ((USART3->SR & USART_FLAG_TC) == (uint16_t)RESET) {}            USART3->DR = (*ptr++ & (uint16_t)0x01FF);#elif STDERR_USART == 255            ITM_SendChar(*ptr++ & (uint16_t)0x00FF);#endif        }        break;    default:        errno = EBADF;        return -1;    }    return len;}
 |