/* * Copyright (C) 2013 INRIA * 2015 Kaspar Schleiser * 2016 Eistec AB * 2018 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 * @{ * * @file * @brief STDIO over UART implementation * * This file implements a UART callback and the STDIO read/write functions * * @author Oliver Hahm * @author Ludwig Knüpfer * @author Kaspar Schleiser * @author Joakim Nohlgård * @author Hauke Petersen * * @} */ #include #include #include "board.h" #include "isrpipe.h" #include "kernel_defines.h" #include "periph/uart.h" #include "stdio_uart.h" #if MODULE_VFS #include "vfs.h" #endif #define ENABLE_DEBUG 0 #include "debug.h" static uint8_t _rx_buf_mem[STDIO_UART_RX_BUFSIZE]; isrpipe_t stdio_uart_isrpipe = ISRPIPE_INIT(_rx_buf_mem); static void _isrpipe_write_one_wrapper(void *arg, uint8_t value) { isrpipe_write_one(arg, value); } void stdio_init(void) { uart_rx_cb_t cb = NULL; void *arg = NULL; if (IS_USED(MODULE_STDIO_UART_RX)) { cb = _isrpipe_write_one_wrapper; arg = &stdio_uart_isrpipe; } uart_init(STDIO_UART_DEV, STDIO_UART_BAUDRATE, cb, arg); #if MODULE_VFS vfs_bind_stdio(); #endif } #if IS_USED(MODULE_STDIO_AVAILABLE) int stdio_available(void) { return tsrb_avail(&stdio_uart_isrpipe.tsrb); } #endif ssize_t stdio_read(void* buffer, size_t count) { if (IS_USED(MODULE_STDIO_UART_RX)) { return (ssize_t)isrpipe_read(&stdio_uart_isrpipe, buffer, count); } return -ENOTSUP; } ssize_t stdio_write(const void *buffer, size_t len) { ssize_t result = len; if (IS_USED(MODULE_STDIO_UART_ONLCR)) { static const uint8_t crlf[2] = { (uint8_t)'\r', (uint8_t)'\n' }; const uint8_t *buf = buffer; while (len) { const uint8_t *pos = memchr(buf, '\n', len); size_t chunk_len = (pos != NULL) ? (uintptr_t)pos - (uintptr_t)buf : len; uart_write(STDIO_UART_DEV, buf, chunk_len); buf += chunk_len; len -= chunk_len; if (len) { uart_write(STDIO_UART_DEV, crlf, sizeof(crlf)); buf++; len--; } } } else { uart_write(STDIO_UART_DEV, (const uint8_t *)buffer, len); } return result; }