| 12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788899091929394959697989910010110210310410510610710810911011111211311411511611711811912012112212312412512612712812913013113213313413513613713813914014114214314414514614714814915015115215315415515615715815916016116216316416516616716816917017117217317417517617717817918018118218318418518618718818919019119219319419519619719819920020120220320420520620720820921021121221321421521621721821922022122222322422522622722822923023123223323423523623723823924024124224324424524624724824925025125225325425525625725825926026126226326426526626726826927027127227327427527627727827928028128228328428528628728828929029129229329429529629729829930030130230330430530630730830931031131231331431531631731831932032132232332432532632732832933033133233333433533633733833934034134234334434534634734834935035135235335435535635735835936036136236336436536636736836937037137237337437537637737837938038138238338438538638738838939039139239339439539639739839940040140240340440540640740840941041141241341441541641741841942042142242342442542642742842943043143243343443543643743843944044144244344444544644744844945045145245345445545645745845946046146246346446546646746846947047147247347447547647747847948048148248348448548648748848949049149249349449549649749849950050150250350450550650750850951051151251351451551651751851952052152252352452552652752852953053153253353453553653753853954054154254354454554654754854955055155255355455555655755855956056156256356456556656756856957057157257357457557657757857958058158258358458558658758858959059159259359459559659759859960060160260360460560660760860961061161261361461561661761861962062162262362462562662762862963063163263363463563663763863964064164264364464564664764864965065165265365465565665765865966066166266366466566666766866967067167267367467567667767867968068168268368468568668768868969069169269369469569669769869970070170270370470570670770870971071171271371471571671771871972072172272372472572672772872973073173273373473573673773873974074174274374474574674774874975075175275375475575675775875976076176276376476576676776876977077177277377477577677777877978078178278378478578678778878979079179279379479579679779879980080180280380480580680780880981081181281381481581681781881982082182282382482582682782882983083183283383483583683783883984084184284384484584684784884985085185285385485585685785885986086186286386486586686786886987087187287387487587687787887988088188288388488588688788888989089189289389489589689789889990090190290390490590690790890991091191291391491591691791891992092192292392492592692792892993093193293393493593693793893994094194294394494594694794894995095195295395495595695795895996096196296396496596696796896997097197297397497597697797897998098198298398498598698798898999099199299399499599699799899910001001100210031004100510061007100810091010101110121013101410151016101710181019102010211022102310241025102610271028102910301031103210331034103510361037103810391040104110421043104410451046104710481049105010511052105310541055 |
- // Copyright 2015-2017 Espressif Systems (Shanghai) PTE LTD
- //
- // Licensed under the Apache License, Version 2.0 (the "License");
- // you may not use this file except in compliance with the License.
- // You may obtain a copy of the License at
- //
- // http://www.apache.org/licenses/LICENSE-2.0
- //
- // Unless required by applicable law or agreed to in writing, software
- // distributed under the License is distributed on an "AS IS" BASIS,
- // WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
- // See the License for the specific language governing permissions and
- // limitations under the License.
- #include <string.h>
- #include <stdbool.h>
- #include <stdarg.h>
- #include <sys/errno.h>
- #include <sys/lock.h>
- #include <sys/fcntl.h>
- #include <sys/param.h>
- #include "esp_vfs.h"
- #include "esp_vfs_dev.h"
- #include "esp_attr.h"
- #include "soc/uart_periph.h"
- #include "driver/uart.h"
- #include "sdkconfig.h"
- #include "driver/uart_select.h"
- #if CONFIG_IDF_TARGET_ESP32
- #include "esp32/rom/uart.h"
- #elif CONFIG_IDF_TARGET_ESP32S2
- #include "esp32s2/rom/uart.h"
- #endif
- // TODO: make the number of UARTs chip dependent
- #define UART_NUM SOC_UART_NUM
- // Token signifying that no character is available
- #define NONE -1
- #if CONFIG_NEWLIB_STDOUT_LINE_ENDING_CRLF
- # define DEFAULT_TX_MODE ESP_LINE_ENDINGS_CRLF
- #elif CONFIG_NEWLIB_STDOUT_LINE_ENDING_CR
- # define DEFAULT_TX_MODE ESP_LINE_ENDINGS_CR
- #else
- # define DEFAULT_TX_MODE ESP_LINE_ENDINGS_LF
- #endif
- #if CONFIG_NEWLIB_STDIN_LINE_ENDING_CRLF
- # define DEFAULT_RX_MODE ESP_LINE_ENDINGS_CRLF
- #elif CONFIG_NEWLIB_STDIN_LINE_ENDING_CR
- # define DEFAULT_RX_MODE ESP_LINE_ENDINGS_CR
- #else
- # define DEFAULT_RX_MODE ESP_LINE_ENDINGS_LF
- #endif
- // UART write bytes function type
- typedef void (*tx_func_t)(int, int);
- // UART read bytes function type
- typedef int (*rx_func_t)(int);
- // Basic functions for sending and receiving bytes over UART
- static void uart_tx_char(int fd, int c);
- static int uart_rx_char(int fd);
- // Functions for sending and receiving bytes which use UART driver
- static void uart_tx_char_via_driver(int fd, int c);
- static int uart_rx_char_via_driver(int fd);
- typedef struct {
- // Pointers to UART peripherals
- uart_dev_t* uart;
- // One-character buffer used for newline conversion code, per UART
- int peek_char;
- // per-UART locks, lazily initialized
- _lock_t read_lock;
- _lock_t write_lock;
- // Per-UART non-blocking flag. Note: default implementation does not honor this
- // flag, all reads are non-blocking. This option becomes effective if UART
- // driver is used.
- bool non_blocking;
- // Newline conversion mode when transmitting
- esp_line_endings_t tx_mode;
- // Newline conversion mode when receiving
- esp_line_endings_t rx_mode;
- // Functions used to write bytes to UART. Default to "basic" functions.
- tx_func_t tx_func;
- // Functions used to read bytes from UART. Default to "basic" functions.
- rx_func_t rx_func;
- } vfs_uart_context_t;
- #define VFS_CTX_DEFAULT_VAL(uart_dev) (vfs_uart_context_t) {\
- .uart = (uart_dev),\
- .peek_char = NONE,\
- .tx_mode = DEFAULT_TX_MODE,\
- .rx_mode = DEFAULT_RX_MODE,\
- .tx_func = uart_tx_char,\
- .rx_func = uart_rx_char,\
- }
- //If the context should be dynamically initialized, remove this structure
- //and point s_ctx to allocated data.
- static vfs_uart_context_t s_context[UART_NUM] = {
- VFS_CTX_DEFAULT_VAL(&UART0),
- VFS_CTX_DEFAULT_VAL(&UART1),
- #if UART_NUM > 2
- VFS_CTX_DEFAULT_VAL(&UART2),
- #endif
- };
- static vfs_uart_context_t* s_ctx[UART_NUM] = {
- &s_context[0],
- &s_context[1],
- #if UART_NUM > 2
- &s_context[2],
- #endif
- };
- #ifdef CONFIG_VFS_SUPPORT_SELECT
- typedef struct {
- esp_vfs_select_sem_t select_sem;
- fd_set *readfds;
- fd_set *writefds;
- fd_set *errorfds;
- fd_set readfds_orig;
- fd_set writefds_orig;
- fd_set errorfds_orig;
- } uart_select_args_t;
- static uart_select_args_t **s_registered_selects = NULL;
- static int s_registered_select_num = 0;
- static portMUX_TYPE s_registered_select_lock = portMUX_INITIALIZER_UNLOCKED;
- static esp_err_t uart_end_select(void *end_select_args);
- #endif // CONFIG_VFS_SUPPORT_SELECT
- static int uart_open(const char * path, int flags, int mode)
- {
- // this is fairly primitive, we should check if file is opened read only,
- // and error out if write is requested
- int fd = -1;
- if (strcmp(path, "/0") == 0) {
- fd = 0;
- } else if (strcmp(path, "/1") == 0) {
- fd = 1;
- } else if (strcmp(path, "/2") == 0) {
- fd = 2;
- } else {
- errno = ENOENT;
- return fd;
- }
- s_ctx[fd]->non_blocking = ((flags & O_NONBLOCK) == O_NONBLOCK);
- return fd;
- }
- static void uart_tx_char(int fd, int c)
- {
- uart_dev_t* uart = s_ctx[fd]->uart;
- while (uart->status.txfifo_cnt >= 127) {
- ;
- }
- #if CONFIG_IDF_TARGET_ESP32
- uart->fifo.rw_byte = c;
- #elif CONFIG_IDF_TARGET_ESP32S2
- uart->ahb_fifo.rw_byte = c;
- #endif
- }
- static void uart_tx_char_via_driver(int fd, int c)
- {
- char ch = (char) c;
- uart_write_bytes(fd, &ch, 1);
- }
- static int uart_rx_char(int fd)
- {
- uart_dev_t* uart = s_ctx[fd]->uart;
- if (uart->status.rxfifo_cnt == 0) {
- return NONE;
- }
- #if CONFIG_IDF_TARGET_ESP32
- return uart->fifo.rw_byte;
- #elif CONFIG_IDF_TARGET_ESP32S2
- return READ_PERI_REG(UART_FIFO_AHB_REG(fd));
- #endif
- }
- static int uart_rx_char_via_driver(int fd)
- {
- uint8_t c;
- int timeout = s_ctx[fd]->non_blocking ? 0 : portMAX_DELAY;
- int n = uart_read_bytes(fd, &c, 1, timeout);
- if (n <= 0) {
- return NONE;
- }
- return c;
- }
- static ssize_t uart_write(int fd, const void * data, size_t size)
- {
- assert(fd >=0 && fd < 3);
- const char *data_c = (const char *)data;
- /* Even though newlib does stream locking on each individual stream, we need
- * a dedicated UART lock if two streams (stdout and stderr) point to the
- * same UART.
- */
- _lock_acquire_recursive(&s_ctx[fd]->write_lock);
- for (size_t i = 0; i < size; i++) {
- int c = data_c[i];
- if (c == '\n' && s_ctx[fd]->tx_mode != ESP_LINE_ENDINGS_LF) {
- s_ctx[fd]->tx_func(fd, '\r');
- if (s_ctx[fd]->tx_mode == ESP_LINE_ENDINGS_CR) {
- continue;
- }
- }
- s_ctx[fd]->tx_func(fd, c);
- }
- _lock_release_recursive(&s_ctx[fd]->write_lock);
- return size;
- }
- /* Helper function which returns a previous character or reads a new one from
- * UART. Previous character can be returned ("pushed back") using
- * uart_return_char function.
- */
- static int uart_read_char(int fd)
- {
- /* return character from peek buffer, if it is there */
- if (s_ctx[fd]->peek_char != NONE) {
- int c = s_ctx[fd]->peek_char;
- s_ctx[fd]->peek_char = NONE;
- return c;
- }
- return s_ctx[fd]->rx_func(fd);
- }
- /* Push back a character; it will be returned by next call to uart_read_char */
- static void uart_return_char(int fd, int c)
- {
- assert(s_ctx[fd]->peek_char == NONE);
- s_ctx[fd]->peek_char = c;
- }
- static ssize_t uart_read(int fd, void* data, size_t size)
- {
- assert(fd >=0 && fd < 3);
- char *data_c = (char *) data;
- size_t received = 0;
- _lock_acquire_recursive(&s_ctx[fd]->read_lock);
- while (received < size) {
- int c = uart_read_char(fd);
- if (c == '\r') {
- if (s_ctx[fd]->rx_mode == ESP_LINE_ENDINGS_CR) {
- c = '\n';
- } else if (s_ctx[fd]->rx_mode == ESP_LINE_ENDINGS_CRLF) {
- /* look ahead */
- int c2 = uart_read_char(fd);
- if (c2 == NONE) {
- /* could not look ahead, put the current character back */
- uart_return_char(fd, c);
- break;
- }
- if (c2 == '\n') {
- /* this was \r\n sequence. discard \r, return \n */
- c = '\n';
- } else {
- /* \r followed by something else. put the second char back,
- * it will be processed on next iteration. return \r now.
- */
- uart_return_char(fd, c2);
- }
- }
- } else if (c == NONE) {
- break;
- }
- data_c[received] = (char) c;
- ++received;
- if (c == '\n') {
- break;
- }
- }
- _lock_release_recursive(&s_ctx[fd]->read_lock);
- if (received > 0) {
- return received;
- }
- errno = EWOULDBLOCK;
- return -1;
- }
- static int uart_fstat(int fd, struct stat * st)
- {
- assert(fd >=0 && fd < 3);
- st->st_mode = S_IFCHR;
- return 0;
- }
- static int uart_close(int fd)
- {
- assert(fd >=0 && fd < 3);
- return 0;
- }
- static int uart_fcntl(int fd, int cmd, int arg)
- {
- assert(fd >=0 && fd < 3);
- int result = 0;
- if (cmd == F_GETFL) {
- if (s_ctx[fd]->non_blocking) {
- result |= O_NONBLOCK;
- }
- } else if (cmd == F_SETFL) {
- s_ctx[fd]->non_blocking = (arg & O_NONBLOCK) != 0;
- } else {
- // unsupported operation
- result = -1;
- errno = ENOSYS;
- }
- return result;
- }
- #ifdef CONFIG_VFS_SUPPORT_DIR
- static int uart_access(const char *path, int amode)
- {
- int ret = -1;
- if (strcmp(path, "/0") == 0 || strcmp(path, "/1") == 0 || strcmp(path, "/2") == 0) {
- if (F_OK == amode) {
- ret = 0; //path exists
- } else {
- if ((((amode & R_OK) == R_OK) || ((amode & W_OK) == W_OK)) && ((amode & X_OK) != X_OK)) {
- ret = 0; //path is readable and/or writable but not executable
- } else {
- errno = EACCES;
- }
- }
- } else {
- errno = ENOENT;
- }
- return ret;
- }
- #endif // CONFIG_VFS_SUPPORT_DIR
- static int uart_fsync(int fd)
- {
- assert(fd >= 0 && fd < 3);
- _lock_acquire_recursive(&s_ctx[fd]->write_lock);
- uart_tx_wait_idle((uint8_t) fd);
- _lock_release_recursive(&s_ctx[fd]->write_lock);
- return 0;
- }
- #ifdef CONFIG_VFS_SUPPORT_SELECT
- static esp_err_t register_select(uart_select_args_t *args)
- {
- esp_err_t ret = ESP_ERR_INVALID_ARG;
- if (args) {
- portENTER_CRITICAL(&s_registered_select_lock);
- const int new_size = s_registered_select_num + 1;
- uart_select_args_t **new_selects;
- if ((new_selects = realloc(s_registered_selects, new_size * sizeof(uart_select_args_t *))) == NULL) {
- ret = ESP_ERR_NO_MEM;
- } else {
- s_registered_selects = new_selects;
- s_registered_selects[s_registered_select_num] = args;
- s_registered_select_num = new_size;
- ret = ESP_OK;
- }
- portEXIT_CRITICAL(&s_registered_select_lock);
- }
- return ret;
- }
- static esp_err_t unregister_select(uart_select_args_t *args)
- {
- esp_err_t ret = ESP_OK;
- if (args) {
- ret = ESP_ERR_INVALID_STATE;
- portENTER_CRITICAL(&s_registered_select_lock);
- for (int i = 0; i < s_registered_select_num; ++i) {
- if (s_registered_selects[i] == args) {
- const int new_size = s_registered_select_num - 1;
- // The item is removed by overwriting it with the last item. The subsequent rellocation will drop the
- // last item.
- s_registered_selects[i] = s_registered_selects[new_size];
- s_registered_selects = realloc(s_registered_selects, new_size * sizeof(uart_select_args_t *));
- // Shrinking a buffer with realloc is guaranteed to succeed.
- s_registered_select_num = new_size;
- ret = ESP_OK;
- break;
- }
- }
- portEXIT_CRITICAL(&s_registered_select_lock);
- }
- return ret;
- }
- static void select_notif_callback_isr(uart_port_t uart_num, uart_select_notif_t uart_select_notif, BaseType_t *task_woken)
- {
- portENTER_CRITICAL_ISR(&s_registered_select_lock);
- for (int i = 0; i < s_registered_select_num; ++i) {
- uart_select_args_t *args = s_registered_selects[i];
- if (args) {
- switch (uart_select_notif) {
- case UART_SELECT_READ_NOTIF:
- if (FD_ISSET(uart_num, &args->readfds_orig)) {
- FD_SET(uart_num, args->readfds);
- esp_vfs_select_triggered_isr(args->select_sem, task_woken);
- }
- break;
- case UART_SELECT_WRITE_NOTIF:
- if (FD_ISSET(uart_num, &args->writefds_orig)) {
- FD_SET(uart_num, args->writefds);
- esp_vfs_select_triggered_isr(args->select_sem, task_woken);
- }
- break;
- case UART_SELECT_ERROR_NOTIF:
- if (FD_ISSET(uart_num, &args->errorfds_orig)) {
- FD_SET(uart_num, args->errorfds);
- esp_vfs_select_triggered_isr(args->select_sem, task_woken);
- }
- break;
- }
- }
- }
- portEXIT_CRITICAL_ISR(&s_registered_select_lock);
- }
- static esp_err_t uart_start_select(int nfds, fd_set *readfds, fd_set *writefds, fd_set *exceptfds,
- esp_vfs_select_sem_t select_sem, void **end_select_args)
- {
- const int max_fds = MIN(nfds, UART_NUM);
- *end_select_args = NULL;
- for (int i = 0; i < max_fds; ++i) {
- if (FD_ISSET(i, readfds) || FD_ISSET(i, writefds) || FD_ISSET(i, exceptfds)) {
- if (!uart_is_driver_installed(i)) {
- return ESP_ERR_INVALID_STATE;
- }
- }
- }
- uart_select_args_t *args = malloc(sizeof(uart_select_args_t));
- if (args == NULL) {
- return ESP_ERR_NO_MEM;
- }
- args->select_sem = select_sem;
- args->readfds = readfds;
- args->writefds = writefds;
- args->errorfds = exceptfds;
- args->readfds_orig = *readfds; // store the original values because they will be set to zero
- args->writefds_orig = *writefds;
- args->errorfds_orig = *exceptfds;
- FD_ZERO(readfds);
- FD_ZERO(writefds);
- FD_ZERO(exceptfds);
- portENTER_CRITICAL(uart_get_selectlock());
- //uart_set_select_notif_callback sets the callbacks in UART ISR
- for (int i = 0; i < max_fds; ++i) {
- if (FD_ISSET(i, &args->readfds_orig) || FD_ISSET(i, &args->writefds_orig) || FD_ISSET(i, &args->errorfds_orig)) {
- uart_set_select_notif_callback(i, select_notif_callback_isr);
- }
- }
- for (int i = 0; i < max_fds; ++i) {
- if (FD_ISSET(i, &args->readfds_orig)) {
- size_t buffered_size;
- if (uart_get_buffered_data_len(i, &buffered_size) == ESP_OK && buffered_size > 0) {
- // signalize immediately when data is buffered
- FD_SET(i, readfds);
- esp_vfs_select_triggered(args->select_sem);
- }
- }
- }
- esp_err_t ret = register_select(args);
- if (ret != ESP_OK) {
- portEXIT_CRITICAL(uart_get_selectlock());
- free(args);
- return ret;
- }
- portEXIT_CRITICAL(uart_get_selectlock());
- *end_select_args = args;
- return ESP_OK;
- }
- static esp_err_t uart_end_select(void *end_select_args)
- {
- uart_select_args_t *args = end_select_args;
- portENTER_CRITICAL(uart_get_selectlock());
- esp_err_t ret = unregister_select(args);
- for (int i = 0; i < UART_NUM; ++i) {
- uart_set_select_notif_callback(i, NULL);
- }
- portEXIT_CRITICAL(uart_get_selectlock());
- if (args) {
- free(args);
- }
- return ret;
- }
- #endif // CONFIG_VFS_SUPPORT_SELECT
- #ifdef CONFIG_VFS_SUPPORT_TERMIOS
- static int uart_tcsetattr(int fd, int optional_actions, const struct termios *p)
- {
- if (fd < 0 || fd >= UART_NUM) {
- errno = EBADF;
- return -1;
- }
- if (p == NULL) {
- errno = EINVAL;
- return -1;
- }
- switch (optional_actions) {
- case TCSANOW:
- // nothing to do
- break;
- case TCSADRAIN:
- if (uart_wait_tx_done(fd, portMAX_DELAY) != ESP_OK) {
- errno = EINVAL;
- return -1;
- }
- /* FALLTHRU */
- case TCSAFLUSH:
- if (uart_flush_input(fd) != ESP_OK) {
- errno = EINVAL;
- return -1;
- }
- break;
- default:
- errno = EINVAL;
- return -1;
- }
- if (p->c_iflag & IGNCR) {
- s_ctx[fd]->rx_mode = ESP_LINE_ENDINGS_CRLF;
- } else if (p->c_iflag & ICRNL) {
- s_ctx[fd]->rx_mode = ESP_LINE_ENDINGS_CR;
- } else {
- s_ctx[fd]->rx_mode = ESP_LINE_ENDINGS_LF;
- }
- // output line endings are not supported because there is no alternative in termios for converting LF to CR
- {
- uart_word_length_t data_bits;
- const tcflag_t csize_bits = p->c_cflag & CSIZE;
- switch (csize_bits) {
- case CS5:
- data_bits = UART_DATA_5_BITS;
- break;
- case CS6:
- data_bits = UART_DATA_6_BITS;
- break;
- case CS7:
- data_bits = UART_DATA_7_BITS;
- break;
- case CS8:
- data_bits = UART_DATA_8_BITS;
- break;
- default:
- errno = EINVAL;
- return -1;
- }
- if (uart_set_word_length(fd, data_bits) != ESP_OK) {
- errno = EINVAL;
- return -1;
- }
- }
- if (uart_set_stop_bits(fd, (p->c_cflag & CSTOPB) ? UART_STOP_BITS_2 : UART_STOP_BITS_1) != ESP_OK) {
- errno = EINVAL;
- return -1;
- }
- if (uart_set_parity(fd, (p->c_cflag & PARENB) ?
- ((p->c_cflag & PARODD) ? UART_PARITY_ODD : UART_PARITY_EVEN)
- :
- UART_PARITY_DISABLE) != ESP_OK) {
- errno = EINVAL;
- return -1;
- }
- if (p->c_cflag & (CBAUD | CBAUDEX)) {
- if (p->c_ispeed != p->c_ospeed) {
- errno = EINVAL;
- return -1;
- } else {
- uint32_t b;
- if (p->c_cflag & BOTHER) {
- b = p->c_ispeed;
- } else {
- switch (p->c_ispeed) {
- case B0:
- b = 0;
- break;
- case B50:
- b = 50;
- break;
- case B75:
- b = 75;
- break;
- case B110:
- b = 110;
- break;
- case B134:
- b = 134;
- break;
- case B150:
- b = 150;
- break;
- case B200:
- b = 200;
- break;
- case B300:
- b = 300;
- break;
- case B600:
- b = 600;
- break;
- case B1200:
- b = 1200;
- break;
- case B1800:
- b = 1800;
- break;
- case B2400:
- b = 2400;
- break;
- case B4800:
- b = 4800;
- break;
- case B9600:
- b = 9600;
- break;
- case B19200:
- b = 19200;
- break;
- case B38400:
- b = 38400;
- break;
- case B57600:
- b = 57600;
- break;
- case B115200:
- b = 115200;
- break;
- case B230400:
- b = 230400;
- break;
- case B460800:
- b = 460800;
- break;
- case B500000:
- b = 500000;
- break;
- case B576000:
- b = 576000;
- break;
- case B921600:
- b = 921600;
- break;
- case B1000000:
- b = 1000000;
- break;
- case B1152000:
- b = 1152000;
- break;
- case B1500000:
- b = 1500000;
- break;
- case B2000000:
- b = 2000000;
- break;
- case B2500000:
- b = 2500000;
- break;
- case B3000000:
- b = 3000000;
- break;
- case B3500000:
- b = 3500000;
- break;
- case B4000000:
- b = 4000000;
- break;
- default:
- errno = EINVAL;
- return -1;
- }
- }
- if (uart_set_baudrate(fd, b) != ESP_OK) {
- errno = EINVAL;
- return -1;
- }
- }
- }
- return 0;
- }
- static int uart_tcgetattr(int fd, struct termios *p)
- {
- if (fd < 0 || fd >= UART_NUM) {
- errno = EBADF;
- return -1;
- }
- if (p == NULL) {
- errno = EINVAL;
- return -1;
- }
- memset(p, 0, sizeof(struct termios));
- if (s_ctx[fd]->rx_mode == ESP_LINE_ENDINGS_CRLF) {
- p->c_iflag |= IGNCR;
- } else if (s_ctx[fd]->rx_mode == ESP_LINE_ENDINGS_CR) {
- p->c_iflag |= ICRNL;
- }
- {
- uart_word_length_t data_bits;
- if (uart_get_word_length(fd, &data_bits) != ESP_OK) {
- errno = EINVAL;
- return -1;
- }
- p->c_cflag &= (~CSIZE);
- switch (data_bits) {
- case UART_DATA_5_BITS:
- p->c_cflag |= CS5;
- break;
- case UART_DATA_6_BITS:
- p->c_cflag |= CS6;
- break;
- case UART_DATA_7_BITS:
- p->c_cflag |= CS7;
- break;
- case UART_DATA_8_BITS:
- p->c_cflag |= CS8;
- break;
- default:
- errno = ENOSYS;
- return -1;
- }
- }
- {
- uart_stop_bits_t stop_bits;
- if (uart_get_stop_bits(fd, &stop_bits) != ESP_OK) {
- errno = EINVAL;
- return -1;
- }
- switch (stop_bits) {
- case UART_STOP_BITS_1:
- // nothing to do
- break;
- case UART_STOP_BITS_2:
- p->c_cflag |= CSTOPB;
- break;
- default:
- // UART_STOP_BITS_1_5 is unsupported by termios
- errno = ENOSYS;
- return -1;
- }
- }
- {
- uart_parity_t parity_mode;
- if (uart_get_parity(fd, &parity_mode) != ESP_OK) {
- errno = EINVAL;
- return -1;
- }
- switch (parity_mode) {
- case UART_PARITY_EVEN:
- p->c_cflag |= PARENB;
- break;
- case UART_PARITY_ODD:
- p->c_cflag |= (PARENB | PARODD);
- break;
- case UART_PARITY_DISABLE:
- // nothing to do
- break;
- default:
- errno = ENOSYS;
- return -1;
- }
- }
- {
- uint32_t baudrate;
- if (uart_get_baudrate(fd, &baudrate) != ESP_OK) {
- errno = EINVAL;
- return -1;
- }
- p->c_cflag |= (CBAUD | CBAUDEX);
- speed_t sp;
- switch (baudrate) {
- case 0:
- sp = B0;
- break;
- case 50:
- sp = B50;
- break;
- case 75:
- sp = B75;
- break;
- case 110:
- sp = B110;
- break;
- case 134:
- sp = B134;
- break;
- case 150:
- sp = B150;
- break;
- case 200:
- sp = B200;
- break;
- case 300:
- sp = B300;
- break;
- case 600:
- sp = B600;
- break;
- case 1200:
- sp = B1200;
- break;
- case 1800:
- sp = B1800;
- break;
- case 2400:
- sp = B2400;
- break;
- case 4800:
- sp = B4800;
- break;
- case 9600:
- sp = B9600;
- break;
- case 19200:
- sp = B19200;
- break;
- case 38400:
- sp = B38400;
- break;
- case 57600:
- sp = B57600;
- break;
- case 115200:
- sp = B115200;
- break;
- case 230400:
- sp = B230400;
- break;
- case 460800:
- sp = B460800;
- break;
- case 500000:
- sp = B500000;
- break;
- case 576000:
- sp = B576000;
- break;
- case 921600:
- sp = B921600;
- break;
- case 1000000:
- sp = B1000000;
- break;
- case 1152000:
- sp = B1152000;
- break;
- case 1500000:
- sp = B1500000;
- break;
- case 2000000:
- sp = B2000000;
- break;
- case 2500000:
- sp = B2500000;
- break;
- case 3000000:
- sp = B3000000;
- break;
- case 3500000:
- sp = B3500000;
- break;
- case 4000000:
- sp = B4000000;
- break;
- default:
- p->c_cflag |= BOTHER;
- sp = baudrate;
- break;
- }
- p->c_ispeed = p->c_ospeed = sp;
- }
- return 0;
- }
- static int uart_tcdrain(int fd)
- {
- if (fd < 0 || fd >= UART_NUM) {
- errno = EBADF;
- return -1;
- }
- if (uart_wait_tx_done(fd, portMAX_DELAY) != ESP_OK) {
- errno = EINVAL;
- return -1;
- }
- return 0;
- }
- static int uart_tcflush(int fd, int select)
- {
- if (fd < 0 || fd >= UART_NUM) {
- errno = EBADF;
- return -1;
- }
- if (select == TCIFLUSH) {
- if (uart_flush_input(fd) != ESP_OK) {
- errno = EINVAL;
- return -1;
- }
- } else {
- // output flushing is not supported
- errno = EINVAL;
- return -1;
- }
- return 0;
- }
- #endif // CONFIG_VFS_SUPPORT_TERMIOS
- void esp_vfs_dev_uart_register(void)
- {
- esp_vfs_t vfs = {
- .flags = ESP_VFS_FLAG_DEFAULT,
- .write = &uart_write,
- .open = &uart_open,
- .fstat = &uart_fstat,
- .close = &uart_close,
- .read = &uart_read,
- .fcntl = &uart_fcntl,
- .fsync = &uart_fsync,
- #ifdef CONFIG_VFS_SUPPORT_DIR
- .access = &uart_access,
- #endif // CONFIG_VFS_SUPPORT_DIR
- #ifdef CONFIG_VFS_SUPPORT_SELECT
- .start_select = &uart_start_select,
- .end_select = &uart_end_select,
- #endif // CONFIG_VFS_SUPPORT_SELECT
- #ifdef CONFIG_VFS_SUPPORT_TERMIOS
- .tcsetattr = &uart_tcsetattr,
- .tcgetattr = &uart_tcgetattr,
- .tcdrain = &uart_tcdrain,
- .tcflush = &uart_tcflush,
- #endif // CONFIG_VFS_SUPPORT_TERMIOS
- };
- ESP_ERROR_CHECK(esp_vfs_register("/dev/uart", &vfs, NULL));
- }
- int esp_vfs_dev_uart_port_set_rx_line_endings(int uart_num, esp_line_endings_t mode)
- {
- if (uart_num < 0 || uart_num >= UART_NUM) {
- errno = EBADF;
- return -1;
- }
- s_ctx[uart_num]->rx_mode = mode;
- return 0;
- }
- int esp_vfs_dev_uart_port_set_tx_line_endings(int uart_num, esp_line_endings_t mode)
- {
- if (uart_num < 0 || uart_num >= UART_NUM) {
- errno = EBADF;
- return -1;
- }
- s_ctx[uart_num]->tx_mode = mode;
- return 0;
- }
- void esp_vfs_dev_uart_set_rx_line_endings(esp_line_endings_t mode)
- {
- for (int i = 0; i < UART_NUM; ++i) {
- s_ctx[i]->rx_mode = mode;
- }
- }
- void esp_vfs_dev_uart_set_tx_line_endings(esp_line_endings_t mode)
- {
- for (int i = 0; i < UART_NUM; ++i) {
- s_ctx[i]->tx_mode = mode;
- }
- }
- void esp_vfs_dev_uart_use_nonblocking(int uart_num)
- {
- _lock_acquire_recursive(&s_ctx[uart_num]->read_lock);
- _lock_acquire_recursive(&s_ctx[uart_num]->write_lock);
- s_ctx[uart_num]->tx_func = uart_tx_char;
- s_ctx[uart_num]->rx_func = uart_rx_char;
- _lock_release_recursive(&s_ctx[uart_num]->write_lock);
- _lock_release_recursive(&s_ctx[uart_num]->read_lock);
- }
- void esp_vfs_dev_uart_use_driver(int uart_num)
- {
- _lock_acquire_recursive(&s_ctx[uart_num]->read_lock);
- _lock_acquire_recursive(&s_ctx[uart_num]->write_lock);
- s_ctx[uart_num]->tx_func = uart_tx_char_via_driver;
- s_ctx[uart_num]->rx_func = uart_rx_char_via_driver;
- _lock_release_recursive(&s_ctx[uart_num]->write_lock);
- _lock_release_recursive(&s_ctx[uart_num]->read_lock);
- }
|