|  | /* | 
|  | * Copyright (c) 2023 ENE Technology Inc. | 
|  | * | 
|  | * SPDX-License-Identifier: Apache-2.0 | 
|  | */ | 
|  |  | 
|  | #define DT_DRV_COMPAT ene_kb1200_uart | 
|  |  | 
|  | #include <zephyr/kernel.h> | 
|  | #include <zephyr/drivers/uart.h> | 
|  | #include <zephyr/drivers/pinctrl.h> | 
|  | #include <reg/ser.h> | 
|  |  | 
|  | struct kb1200_uart_config { | 
|  | #ifdef CONFIG_UART_INTERRUPT_DRIVEN | 
|  | void (*irq_cfg_func)(void); | 
|  | #endif | 
|  | struct serial_regs *ser; | 
|  | const struct pinctrl_dev_config *pcfg; | 
|  | }; | 
|  |  | 
|  | struct kb1200_uart_data { | 
|  | uart_irq_callback_user_data_t callback; | 
|  | struct uart_config current_config; | 
|  | void *callback_data; | 
|  | uint8_t pending_flag_data; | 
|  | }; | 
|  |  | 
|  | static int kb1200_uart_err_check(const struct device *dev) | 
|  | { | 
|  | const struct kb1200_uart_config *config = dev->config; | 
|  | int err = 0; | 
|  |  | 
|  | if (config->ser->SERSTS & SERSTS_RX_OVERRUN) { | 
|  | err |= UART_ERROR_OVERRUN; | 
|  | } | 
|  | if (config->ser->SERSTS & SERSTS_PARITY_ERROR) { | 
|  | err |= UART_ERROR_PARITY; | 
|  | } | 
|  | if (config->ser->SERSTS & SERSTS_FRAME_ERROR) { | 
|  | err |= UART_ERROR_FRAMING; | 
|  | } | 
|  | return err; | 
|  | } | 
|  |  | 
|  | #ifdef CONFIG_UART_USE_RUNTIME_CONFIGURE | 
|  | static int kb1200_uart_configure(const struct device *dev, const struct uart_config *cfg) | 
|  | { | 
|  | uint16_t reg_baudrate = 0; | 
|  | uint8_t reg_parity = 0; | 
|  | int ret = 0; | 
|  | const struct kb1200_uart_config *config = dev->config; | 
|  | struct kb1200_uart_data *data = dev->data; | 
|  |  | 
|  | reg_baudrate = (DIVIDER_BASE_CLK / cfg->baudrate) - 1; | 
|  |  | 
|  | switch (cfg->parity) { | 
|  | case UART_CFG_PARITY_NONE: | 
|  | reg_parity = SERCFG_PARITY_NONE; | 
|  | break; | 
|  | case UART_CFG_PARITY_ODD: | 
|  | reg_parity = SERCFG_PARITY_ODD; | 
|  | break; | 
|  | case UART_CFG_PARITY_EVEN: | 
|  | reg_parity = SERCFG_PARITY_EVEN; | 
|  | break; | 
|  | case UART_CFG_PARITY_MARK: | 
|  | case UART_CFG_PARITY_SPACE: | 
|  | default: | 
|  | ret = -ENOTSUP; | 
|  | break; | 
|  | } | 
|  |  | 
|  | switch (cfg->stop_bits) { | 
|  | case UART_CFG_STOP_BITS_1: | 
|  | break; | 
|  | case UART_CFG_STOP_BITS_0_5: | 
|  | case UART_CFG_STOP_BITS_1_5: | 
|  | case UART_CFG_STOP_BITS_2: | 
|  | default: | 
|  | ret = -ENOTSUP; | 
|  | break; | 
|  | } | 
|  |  | 
|  | switch (cfg->data_bits) { | 
|  | case UART_CFG_DATA_BITS_8: | 
|  | break; | 
|  | case UART_CFG_DATA_BITS_5: | 
|  | case UART_CFG_DATA_BITS_6: | 
|  | case UART_CFG_DATA_BITS_7: | 
|  | case UART_CFG_DATA_BITS_9: | 
|  | default: | 
|  | ret = -ENOTSUP; | 
|  | break; | 
|  | } | 
|  |  | 
|  | switch (cfg->flow_ctrl) { | 
|  | case UART_CFG_FLOW_CTRL_NONE: | 
|  | break; | 
|  | case UART_CFG_FLOW_CTRL_RTS_CTS: | 
|  | case UART_CFG_FLOW_CTRL_DTR_DSR: | 
|  | case UART_CFG_FLOW_CTRL_RS485: | 
|  | default: | 
|  | ret = -ENOTSUP; | 
|  | break; | 
|  | } | 
|  | config->ser->SERCFG = | 
|  | (reg_baudrate << 16) | (reg_parity << 2) | (SERIE_RX_ENABLE | SERIE_TX_ENABLE); | 
|  | config->ser->SERCTRL = SERCTRL_MODE1; | 
|  | data->current_config = *cfg; | 
|  | return ret; | 
|  | } | 
|  |  | 
|  | static int kb1200_uart_config_get(const struct device *dev, struct uart_config *cfg) | 
|  | { | 
|  | struct kb1200_uart_data *data = dev->data; | 
|  |  | 
|  | *cfg = data->current_config; | 
|  | return 0; | 
|  | } | 
|  | #endif /* CONFIG_UART_USE_RUNTIME_CONFIGURE */ | 
|  |  | 
|  | #ifdef CONFIG_UART_INTERRUPT_DRIVEN | 
|  | static int kb1200_uart_fifo_fill(const struct device *dev, const uint8_t *tx_data, int size) | 
|  | { | 
|  | const struct kb1200_uart_config *config = dev->config; | 
|  | uint16_t tx_bytes = 0U; | 
|  |  | 
|  | while ((size - tx_bytes) > 0) { | 
|  | /* Check Tx FIFO not Full*/ | 
|  | while (config->ser->SERSTS & SERSTS_TX_FULL) | 
|  | ; | 
|  | /* Put a character into	Tx FIFO	*/ | 
|  | config->ser->SERTBUF = tx_data[tx_bytes]; | 
|  | tx_bytes++; | 
|  | } | 
|  | return tx_bytes; | 
|  | } | 
|  |  | 
|  | static int kb1200_uart_fifo_read(const struct device *dev, uint8_t *rx_data, const int size) | 
|  | { | 
|  | const struct kb1200_uart_config *config = dev->config; | 
|  | uint16_t rx_bytes = 0U; | 
|  |  | 
|  | /* Check Rx FIFO not Empty*/ | 
|  | while ((size - rx_bytes > 0) && (!(config->ser->SERSTS & SERSTS_RX_EMPTY))) { | 
|  | /* Put a character into	Tx FIFO	*/ | 
|  | rx_data[rx_bytes] = config->ser->SERRBUF; | 
|  | rx_bytes++; | 
|  | } | 
|  | return rx_bytes; | 
|  | } | 
|  |  | 
|  | static void kb1200_uart_irq_tx_enable(const struct device *dev) | 
|  | { | 
|  | const struct kb1200_uart_config *config = dev->config; | 
|  |  | 
|  | config->ser->SERPF = SERPF_TX_EMPTY; | 
|  | config->ser->SERIE |= SERIE_TX_ENABLE; | 
|  | } | 
|  |  | 
|  | static void kb1200_uart_irq_tx_disable(const struct device *dev) | 
|  | { | 
|  | const struct kb1200_uart_config *config = dev->config; | 
|  |  | 
|  | config->ser->SERIE &= ~SERIE_TX_ENABLE; | 
|  | config->ser->SERPF = SERPF_TX_EMPTY; | 
|  | } | 
|  |  | 
|  | static int kb1200_uart_irq_tx_ready(const struct device *dev) | 
|  | { | 
|  | struct kb1200_uart_data *data = dev->data; | 
|  |  | 
|  | return (data->pending_flag_data & SERPF_TX_EMPTY) ? 1 : 0; | 
|  | } | 
|  |  | 
|  | static void kb1200_uart_irq_rx_enable(const struct device *dev) | 
|  | { | 
|  | const struct kb1200_uart_config *config = dev->config; | 
|  |  | 
|  | config->ser->SERPF = SERPF_RX_CNT_FULL; | 
|  | config->ser->SERIE |= SERIE_RX_ENABLE; | 
|  | } | 
|  |  | 
|  | static void kb1200_uart_irq_rx_disable(const struct device *dev) | 
|  | { | 
|  | const struct kb1200_uart_config *config = dev->config; | 
|  |  | 
|  | config->ser->SERIE &= (~SERIE_RX_ENABLE); | 
|  | config->ser->SERPF = SERPF_RX_CNT_FULL; | 
|  | } | 
|  |  | 
|  | static int kb1200_uart_irq_rx_ready(const struct device *dev) | 
|  | { | 
|  | struct kb1200_uart_data *data = dev->data; | 
|  |  | 
|  | return (data->pending_flag_data & SERPF_RX_CNT_FULL) ? 1 : 0; | 
|  | } | 
|  |  | 
|  | static void kb1200_uart_irq_err_enable(const struct device *dev) | 
|  | { | 
|  | const struct kb1200_uart_config *config = dev->config; | 
|  |  | 
|  | config->ser->SERPF = SERPF_RX_ERROR; | 
|  | config->ser->SERIE |= SERIE_RX_ERROR; | 
|  | } | 
|  |  | 
|  | static void kb1200_uart_irq_err_disable(const struct device *dev) | 
|  | { | 
|  | const struct kb1200_uart_config *config = dev->config; | 
|  |  | 
|  | config->ser->SERIE &= (~SERIE_RX_ERROR); | 
|  | config->ser->SERPF = SERPF_RX_ERROR; | 
|  | } | 
|  |  | 
|  | static int kb1200_uart_irq_is_pending(const struct device *dev) | 
|  | { | 
|  | struct kb1200_uart_data *data = dev->data; | 
|  |  | 
|  | return (data->pending_flag_data) ? 1 : 0; | 
|  | } | 
|  |  | 
|  | static int kb1200_uart_irq_update(const struct device *dev) | 
|  | { | 
|  | struct kb1200_uart_data *data = dev->data; | 
|  | const struct kb1200_uart_config *config = dev->config; | 
|  |  | 
|  | data->pending_flag_data = (config->ser->SERPF) & (config->ser->SERIE); | 
|  | /*clear	pending	flag*/ | 
|  | config->ser->SERPF = data->pending_flag_data; | 
|  | return 1; | 
|  | } | 
|  |  | 
|  | static void kb1200_uart_irq_callback_set(const struct device *dev, uart_irq_callback_user_data_t cb, | 
|  | void *cb_data) | 
|  | { | 
|  | struct kb1200_uart_data *data = dev->data; | 
|  |  | 
|  | data->callback = cb; | 
|  | data->callback_data = cb_data; | 
|  | } | 
|  |  | 
|  | static void kb1200_uart_irq_handler(const struct device *dev) | 
|  | { | 
|  | struct kb1200_uart_data *data = dev->data; | 
|  |  | 
|  | if (data->callback) { | 
|  | data->callback(dev, data->callback_data); | 
|  | } | 
|  | } | 
|  | #endif /* CONFIG_UART_INTERRUPT_DRIVEN */ | 
|  |  | 
|  | static int kb1200_uart_poll_in(const struct device *dev, unsigned char *c) | 
|  | { | 
|  | #ifdef CONFIG_UART_INTERRUPT_DRIVEN | 
|  | return kb1200_uart_fifo_read(dev, c, 1) ? 0 : -1; | 
|  | #else | 
|  | const struct kb1200_uart_config *config = dev->config; | 
|  |  | 
|  | /* Check Rx FIFO not Empty*/ | 
|  | if (config->ser->SERSTS & SERSTS_RX_EMPTY) { | 
|  | return -1; | 
|  | } | 
|  | /* Put a character into Tx FIFO */ | 
|  | *c = config->ser->SERRBUF; | 
|  | return 0; | 
|  | #endif /* CONFIG_UART_INTERRUPT_DRIVEN */ | 
|  | } | 
|  |  | 
|  | static void kb1200_uart_poll_out(const struct device *dev, unsigned char c) | 
|  | { | 
|  | #ifdef CONFIG_UART_INTERRUPT_DRIVEN | 
|  | kb1200_uart_fifo_fill(dev, &c, 1); | 
|  | #else | 
|  | const struct kb1200_uart_config *config = dev->config; | 
|  |  | 
|  | /* Wait	Tx FIFO	not Full*/ | 
|  | while (config->ser->SERSTS & SER_TxFull) { | 
|  | ; | 
|  | } | 
|  | /* Put a character into	Tx FIFO */ | 
|  | config->ser->SERTBUF = c; | 
|  | #endif /* CONFIG_UART_INTERRUPT_DRIVEN */ | 
|  | } | 
|  |  | 
|  | static DEVICE_API(uart, kb1200_uart_api) = { | 
|  | .poll_in = kb1200_uart_poll_in, | 
|  | .poll_out = kb1200_uart_poll_out, | 
|  | .err_check = kb1200_uart_err_check, | 
|  | #ifdef CONFIG_UART_USE_RUNTIME_CONFIGURE | 
|  | .configure = kb1200_uart_configure, | 
|  | .config_get = kb1200_uart_config_get, | 
|  | #endif /* CONFIG_UART_USE_RUNTIME_CONFIGURE */ | 
|  | #ifdef CONFIG_UART_INTERRUPT_DRIVEN | 
|  | .fifo_fill = kb1200_uart_fifo_fill, | 
|  | .fifo_read = kb1200_uart_fifo_read, | 
|  | .irq_tx_enable = kb1200_uart_irq_tx_enable, | 
|  | .irq_tx_disable = kb1200_uart_irq_tx_disable, | 
|  | .irq_tx_ready = kb1200_uart_irq_tx_ready, | 
|  | .irq_rx_enable = kb1200_uart_irq_rx_enable, | 
|  | .irq_rx_disable = kb1200_uart_irq_rx_disable, | 
|  | .irq_rx_ready = kb1200_uart_irq_rx_ready, | 
|  | .irq_err_enable = kb1200_uart_irq_err_enable, | 
|  | .irq_err_disable = kb1200_uart_irq_err_disable, | 
|  | .irq_is_pending = kb1200_uart_irq_is_pending, | 
|  | .irq_update = kb1200_uart_irq_update, | 
|  | .irq_callback_set = kb1200_uart_irq_callback_set, | 
|  | #endif /* CONFIG_UART_INTERRUPT_DRIVEN */ | 
|  | }; | 
|  |  | 
|  | #ifdef CONFIG_UART_INTERRUPT_DRIVEN | 
|  |  | 
|  | /* GPIO module instances */ | 
|  | #define KB1200_UART_DEV(inst) DEVICE_DT_INST_GET(inst), | 
|  | static const struct device *const uart_devices[] = {DT_INST_FOREACH_STATUS_OKAY(KB1200_UART_DEV)}; | 
|  | static void kb1200_uart_isr_wrap(const struct device *dev) | 
|  | { | 
|  | for (size_t i = 0; i < ARRAY_SIZE(uart_devices); i++) { | 
|  | const struct device *dev_ = uart_devices[i]; | 
|  | const struct kb1200_uart_config *config = dev_->config; | 
|  |  | 
|  | if (config->ser->SERIE & config->ser->SERPF) { | 
|  | kb1200_uart_irq_handler(dev_); | 
|  | } | 
|  | } | 
|  | } | 
|  | #endif /* CONFIG_UART_INTERRUPT_DRIVEN */ | 
|  |  | 
|  | static int kb1200_uart_init(const struct device *dev) | 
|  | { | 
|  | int ret; | 
|  | const struct kb1200_uart_config *config = dev->config; | 
|  | struct kb1200_uart_data *data = dev->data; | 
|  |  | 
|  | ret = pinctrl_apply_state(config->pcfg, PINCTRL_STATE_DEFAULT); | 
|  | if (ret != 0) { | 
|  | return ret; | 
|  | } | 
|  |  | 
|  | kb1200_uart_configure(dev, &data->current_config); | 
|  | #ifdef CONFIG_UART_INTERRUPT_DRIVEN | 
|  | config->irq_cfg_func(); | 
|  | #endif /* CONFIG_UART_INTERRUPT_DRIVEN */ | 
|  |  | 
|  | return 0; | 
|  | } | 
|  |  | 
|  | #ifdef CONFIG_UART_INTERRUPT_DRIVEN | 
|  | static bool init_irq = true; | 
|  | static void kb1200_uart_irq_init(void) | 
|  | { | 
|  | if (init_irq) { | 
|  | init_irq = false; | 
|  | IRQ_CONNECT(DT_INST_IRQN(0), DT_INST_IRQ(0, priority), kb1200_uart_isr_wrap, NULL, | 
|  | 0); | 
|  | irq_enable(DT_INST_IRQN(0)); | 
|  | } | 
|  | } | 
|  | #endif /* CONFIG_UART_INTERRUPT_DRIVEN */ | 
|  |  | 
|  | #define KB1200_UART_INIT(n)                                                                        \ | 
|  | PINCTRL_DT_INST_DEFINE(n);                                                                 \ | 
|  | static struct kb1200_uart_data kb1200_uart_data_##n = {                                    \ | 
|  | .current_config = {                                                                \ | 
|  | .baudrate = DT_INST_PROP(n, current_speed),                        \ | 
|  | .parity = UART_CFG_PARITY_NONE,                                    \ | 
|  | .stop_bits = UART_CFG_STOP_BITS_1,                                 \ | 
|  | .data_bits = UART_CFG_DATA_BITS_8,                                 \ | 
|  | .flow_ctrl = UART_CFG_FLOW_CTRL_NONE,                              \ | 
|  | },                                                                         \ | 
|  | };                                                                                         \ | 
|  | static const struct kb1200_uart_config kb1200_uart_config_##n = {                          \ | 
|  | IF_ENABLED(CONFIG_UART_INTERRUPT_DRIVEN, (.irq_cfg_func = kb1200_uart_irq_init,))  \ | 
|  | .ser = (struct serial_regs *)DT_INST_REG_ADDR(n),                                  \ | 
|  | .pcfg = PINCTRL_DT_INST_DEV_CONFIG_GET(n)};                                        \ | 
|  | DEVICE_DT_INST_DEFINE(n, kb1200_uart_init, NULL, &kb1200_uart_data_##n,                    \ | 
|  | &kb1200_uart_config_##n, PRE_KERNEL_1, CONFIG_SERIAL_INIT_PRIORITY,  \ | 
|  | &kb1200_uart_api); | 
|  |  | 
|  | DT_INST_FOREACH_STATUS_OKAY(KB1200_UART_INIT) |