9f0f9b8875
The var "uart_xlnx_ps_driver_api" is defined twice in the source, so remove one definition of it. Signed-off-by: Florian La Roche <Florian.LaRoche@gmail.com>
1183 lines
36 KiB
C
1183 lines
36 KiB
C
/* uart_xlnx_ps.c - Xilinx Zynq family serial driver */
|
|
|
|
/*
|
|
* Copyright (c) 2018 Xilinx, Inc.
|
|
*
|
|
* SPDX-License-Identifier: Apache-2.0
|
|
*/
|
|
|
|
#define DT_DRV_COMPAT xlnx_xuartps
|
|
|
|
/**
|
|
* @brief Xilinx Zynq Family Serial Driver
|
|
*
|
|
* This is the driver for the Xilinx Zynq family cadence serial device.
|
|
*
|
|
* Before individual UART port can be used, uart_xlnx_ps_init() has to be
|
|
* called to setup the port.
|
|
*
|
|
* - the following macro for the number of bytes between register addresses:
|
|
*
|
|
* UART_REG_ADDR_INTERVAL
|
|
*/
|
|
|
|
#include <errno.h>
|
|
#include <zephyr/kernel.h>
|
|
#include <zephyr/arch/cpu.h>
|
|
#include <zephyr/types.h>
|
|
#include <soc.h>
|
|
|
|
#include <zephyr/init.h>
|
|
#include <zephyr/toolchain.h>
|
|
#include <zephyr/linker/sections.h>
|
|
#include <zephyr/drivers/uart.h>
|
|
#include <zephyr/sys/sys_io.h>
|
|
#include <zephyr/irq.h>
|
|
|
|
#ifdef CONFIG_PINCTRL
|
|
#include <zephyr/drivers/pinctrl.h>
|
|
#endif
|
|
|
|
/* For all register offsets and bits / bit masks:
|
|
* Comp. Xilinx Zynq-7000 Technical Reference Manual (ug585), chap. B.33
|
|
*/
|
|
|
|
/* Register offsets within the UART device's register space */
|
|
#define XUARTPS_CR_OFFSET 0x0000U /**< Control Register [8:0] */
|
|
#define XUARTPS_MR_OFFSET 0x0004U /**< Mode Register [9:0] */
|
|
#define XUARTPS_IER_OFFSET 0x0008U /**< Interrupt Enable [12:0] */
|
|
#define XUARTPS_IDR_OFFSET 0x000CU /**< Interrupt Disable [12:0] */
|
|
#define XUARTPS_IMR_OFFSET 0x0010U /**< Interrupt Mask [12:0] */
|
|
#define XUARTPS_ISR_OFFSET 0x0014U /**< Interrupt Status [12:0]*/
|
|
#define XUARTPS_BAUDGEN_OFFSET 0x0018U /**< Baud Rate Generator [15:0] */
|
|
#define XUARTPS_RXTOUT_OFFSET 0x001CU /**< RX Timeout [7:0] */
|
|
#define XUARTPS_RXWM_OFFSET 0x0020U /**< RX FIFO Trigger Level [5:0] */
|
|
#define XUARTPS_MODEMCR_OFFSET 0x0024U /**< Modem Control [5:0] */
|
|
#define XUARTPS_MODEMSR_OFFSET 0x0028U /**< Modem Status [8:0] */
|
|
#define XUARTPS_SR_OFFSET 0x002CU /**< Channel Status [14:0] */
|
|
#define XUARTPS_FIFO_OFFSET 0x0030U /**< FIFO [7:0] */
|
|
#define XUARTPS_BAUDDIV_OFFSET 0x0034U /**< Baud Rate Divider [7:0] */
|
|
#define XUARTPS_FLOWDEL_OFFSET 0x0038U /**< Flow Delay [5:0] */
|
|
#define XUARTPS_TXWM_OFFSET 0x0044U /**< TX FIFO Trigger Level [5:0] */
|
|
#define XUARTPS_RXBS_OFFSET 0x0048U /**< RX FIFO Byte Status [11:0] */
|
|
|
|
/* Control Register Bits Definition */
|
|
#define XUARTPS_CR_STOPBRK 0x00000100U /**< Stop transmission of break */
|
|
#define XUARTPS_CR_STARTBRK 0x00000080U /**< Set break */
|
|
#define XUARTPS_CR_TORST 0x00000040U /**< RX timeout counter restart */
|
|
#define XUARTPS_CR_TX_DIS 0x00000020U /**< TX disabled. */
|
|
#define XUARTPS_CR_TX_EN 0x00000010U /**< TX enabled */
|
|
#define XUARTPS_CR_RX_DIS 0x00000008U /**< RX disabled. */
|
|
#define XUARTPS_CR_RX_EN 0x00000004U /**< RX enabled */
|
|
#define XUARTPS_CR_EN_DIS_MASK 0x0000003CU /**< Enable/disable Mask */
|
|
#define XUARTPS_CR_TXRST 0x00000002U /**< TX logic reset */
|
|
#define XUARTPS_CR_RXRST 0x00000001U /**< RX logic reset */
|
|
|
|
/* Mode Register Bits Definition */
|
|
#define XUARTPS_MR_CCLK 0x00000400U /**< Input clock select */
|
|
#define XUARTPS_MR_CHMODE_R_LOOP 0x00000300U /**< Remote loopback mode */
|
|
#define XUARTPS_MR_CHMODE_L_LOOP 0x00000200U /**< Local loopback mode */
|
|
#define XUARTPS_MR_CHMODE_ECHO 0x00000100U /**< Auto echo mode */
|
|
#define XUARTPS_MR_CHMODE_NORM 0x00000000U /**< Normal mode */
|
|
#define XUARTPS_MR_CHMODE_SHIFT 8U /**< Mode shift */
|
|
#define XUARTPS_MR_CHMODE_MASK 0x00000300U /**< Mode mask */
|
|
#define XUARTPS_MR_STOPMODE_2_BIT 0x00000080U /**< 2 stop bits */
|
|
#define XUARTPS_MR_STOPMODE_1_5_BIT 0x00000040U /**< 1.5 stop bits */
|
|
#define XUARTPS_MR_STOPMODE_1_BIT 0x00000000U /**< 1 stop bit */
|
|
#define XUARTPS_MR_STOPMODE_SHIFT 6U /**< Stop bits shift */
|
|
#define XUARTPS_MR_STOPMODE_MASK 0x000000A0U /**< Stop bits mask */
|
|
#define XUARTPS_MR_PARITY_NONE 0x00000020U /**< No parity mode */
|
|
#define XUARTPS_MR_PARITY_MARK 0x00000018U /**< Mark parity mode */
|
|
#define XUARTPS_MR_PARITY_SPACE 0x00000010U /**< Space parity mode */
|
|
#define XUARTPS_MR_PARITY_ODD 0x00000008U /**< Odd parity mode */
|
|
#define XUARTPS_MR_PARITY_EVEN 0x00000000U /**< Even parity mode */
|
|
#define XUARTPS_MR_PARITY_SHIFT 3U /**< Parity setting shift */
|
|
#define XUARTPS_MR_PARITY_MASK 0x00000038U /**< Parity mask */
|
|
#define XUARTPS_MR_CHARLEN_6_BIT 0x00000006U /**< 6 bits data */
|
|
#define XUARTPS_MR_CHARLEN_7_BIT 0x00000004U /**< 7 bits data */
|
|
#define XUARTPS_MR_CHARLEN_8_BIT 0x00000000U /**< 8 bits data */
|
|
#define XUARTPS_MR_CHARLEN_SHIFT 1U /**< Data Length shift */
|
|
#define XUARTPS_MR_CHARLEN_MASK 0x00000006U /**< Data length mask */
|
|
#define XUARTPS_MR_CLKSEL 0x00000001U /**< Input clock select */
|
|
|
|
/* Interrupt Register Bits Definition */
|
|
#define XUARTPS_IXR_RBRK 0x00002000U /**< Rx FIFO break detect interrupt */
|
|
#define XUARTPS_IXR_TOVR 0x00001000U /**< Tx FIFO Overflow interrupt */
|
|
#define XUARTPS_IXR_TNFUL 0x00000800U /**< Tx FIFO Nearly Full interrupt */
|
|
#define XUARTPS_IXR_TTRIG 0x00000400U /**< Tx Trig interrupt */
|
|
#define XUARTPS_IXR_DMS 0x00000200U /**< Modem status change interrupt */
|
|
#define XUARTPS_IXR_TOUT 0x00000100U /**< Timeout error interrupt */
|
|
#define XUARTPS_IXR_PARITY 0x00000080U /**< Parity error interrupt */
|
|
#define XUARTPS_IXR_FRAMING 0x00000040U /**< Framing error interrupt */
|
|
#define XUARTPS_IXR_RXOVR 0x00000020U /**< Overrun error interrupt */
|
|
#define XUARTPS_IXR_TXFULL 0x00000010U /**< TX FIFO full interrupt. */
|
|
#define XUARTPS_IXR_TXEMPTY 0x00000008U /**< TX FIFO empty interrupt. */
|
|
#define XUARTPS_IXR_RXFULL 0x00000004U /**< RX FIFO full interrupt. */
|
|
#define XUARTPS_IXR_RXEMPTY 0x00000002U /**< RX FIFO empty interrupt. */
|
|
#define XUARTPS_IXR_RTRIG 0x00000001U /**< RX FIFO trigger interrupt. */
|
|
#define XUARTPS_IXR_MASK 0x00003FFFU /**< Valid bit mask */
|
|
|
|
/* Modem Control Register Bits Definition */
|
|
#define XUARTPS_MODEMCR_FCM_RTS_CTS 0x00000020 /**< RTS/CTS hardware flow control. */
|
|
#define XUARTPS_MODEMCR_FCM_NONE 0x00000000 /**< No hardware flow control. */
|
|
#define XUARTPS_MODEMCR_FCM_MASK 0x00000020 /**< Hardware flow control mask. */
|
|
#define XUARTPS_MODEMCR_RTS_SHIFT 1U /**< RTS bit shift */
|
|
#define XUARTPS_MODEMCR_DTR_SHIFT 0U /**< DTR bit shift */
|
|
|
|
/* Channel Status Register */
|
|
#define XUARTPS_SR_TNFUL 0x00004000U /**< TX FIFO Nearly Full Status */
|
|
#define XUARTPS_SR_TTRIG 0x00002000U /**< TX FIFO Trigger Status */
|
|
#define XUARTPS_SR_FLOWDEL 0x00001000U /**< RX FIFO fill over flow delay */
|
|
#define XUARTPS_SR_TACTIVE 0x00000800U /**< TX active */
|
|
#define XUARTPS_SR_RACTIVE 0x00000400U /**< RX active */
|
|
#define XUARTPS_SR_TXFULL 0x00000010U /**< TX FIFO full */
|
|
#define XUARTPS_SR_TXEMPTY 0x00000008U /**< TX FIFO empty */
|
|
#define XUARTPS_SR_RXFULL 0x00000004U /**< RX FIFO full */
|
|
#define XUARTPS_SR_RXEMPTY 0x00000002U /**< RX FIFO empty */
|
|
#define XUARTPS_SR_RTRIG 0x00000001U /**< RX FIFO fill over trigger */
|
|
|
|
/** Device configuration structure */
|
|
struct uart_xlnx_ps_dev_config {
|
|
DEVICE_MMIO_ROM;
|
|
uint32_t sys_clk_freq;
|
|
#ifdef CONFIG_UART_INTERRUPT_DRIVEN
|
|
uart_irq_config_func_t irq_config_func;
|
|
#endif
|
|
#ifdef CONFIG_PINCTRL
|
|
const struct pinctrl_dev_config *pincfg;
|
|
#endif
|
|
uint32_t baud_rate;
|
|
};
|
|
|
|
/** Device data structure */
|
|
struct uart_xlnx_ps_dev_data_t {
|
|
DEVICE_MMIO_RAM;
|
|
uint32_t parity;
|
|
uint32_t stopbits;
|
|
uint32_t databits;
|
|
uint32_t flowctrl;
|
|
|
|
#ifdef CONFIG_UART_INTERRUPT_DRIVEN
|
|
uart_irq_callback_user_data_t user_cb;
|
|
void *user_data;
|
|
#endif
|
|
};
|
|
|
|
/**
|
|
* @brief Disables the UART's RX and TX function.
|
|
*
|
|
* Writes 'Disable RX' and 'Disable TX' command bits into the respective
|
|
* UART's Command Register, thus disabling the operation of the UART.
|
|
*
|
|
* While writing the disable command bits, the opposing enable command
|
|
* bits, which are set when enabling the UART, are cleared.
|
|
*
|
|
* This function must be called before any configuration parameters
|
|
* of the UART are modified at run-time.
|
|
*
|
|
* @param reg_base Base address of the respective UART's register space.
|
|
*/
|
|
static void xlnx_ps_disable_uart(uintptr_t reg_base)
|
|
{
|
|
uint32_t reg_val = sys_read32(reg_base + XUARTPS_CR_OFFSET);
|
|
|
|
reg_val &= (~XUARTPS_CR_EN_DIS_MASK);
|
|
/* Set control register bits [5]: TX_DIS and [3]: RX_DIS */
|
|
reg_val |= XUARTPS_CR_TX_DIS | XUARTPS_CR_RX_DIS;
|
|
sys_write32(reg_val, reg_base + XUARTPS_CR_OFFSET);
|
|
}
|
|
|
|
/**
|
|
* @brief Enables the UART's RX and TX function.
|
|
*
|
|
* Writes 'Enable RX' and 'Enable TX' command bits into the respective
|
|
* UART's Command Register, thus enabling the operation of the UART.
|
|
*
|
|
* While writing the enable command bits, the opposing disable command
|
|
* bits, which are set when disabling the UART, are cleared.
|
|
*
|
|
* This function must not be called while any configuration parameters
|
|
* of the UART are being modified at run-time.
|
|
*
|
|
* @param reg_base Base address of the respective UART's register space.
|
|
*/
|
|
static void xlnx_ps_enable_uart(uintptr_t reg_base)
|
|
{
|
|
uint32_t reg_val = sys_read32(reg_base + XUARTPS_CR_OFFSET);
|
|
|
|
reg_val &= (~XUARTPS_CR_EN_DIS_MASK);
|
|
/* Set control register bits [4]: TX_EN and [2]: RX_EN */
|
|
reg_val |= XUARTPS_CR_TX_EN | XUARTPS_CR_RX_EN;
|
|
sys_write32(reg_val, reg_base + XUARTPS_CR_OFFSET);
|
|
}
|
|
|
|
/**
|
|
* @brief Calculates and sets the values of the BAUDDIV and BAUDGEN registers.
|
|
*
|
|
* Calculates and sets the values of the BAUDDIV and BAUDGEN registers, which
|
|
* determine the prescaler applied to the clock driving the UART, based on
|
|
* the target baud rate, which is provided as a decimal value.
|
|
*
|
|
* The calculation of the values to be written to the BAUDDIV and BAUDGEN
|
|
* registers is described in the Zynq-7000 TRM, chapter 19.2.3 'Baud Rate
|
|
* Generator'.
|
|
*
|
|
* @param dev UART device struct
|
|
* @param baud_rate The desired baud rate as a decimal value
|
|
*/
|
|
static void set_baudrate(const struct device *dev, uint32_t baud_rate)
|
|
{
|
|
const struct uart_xlnx_ps_dev_config *dev_cfg = dev->config;
|
|
uint32_t baud = dev_cfg->baud_rate;
|
|
uint32_t clk_freq = dev_cfg->sys_clk_freq;
|
|
uintptr_t reg_base = DEVICE_MMIO_GET(dev);
|
|
uint32_t divisor, generator;
|
|
|
|
/* Calculate divisor and baud rate generator value */
|
|
if ((baud != 0) && (clk_freq != 0)) {
|
|
/* Covering case where input clock is so slow */
|
|
if (clk_freq < 1000000U && baud > 4800U) {
|
|
baud = 4800;
|
|
}
|
|
|
|
for (divisor = 4; divisor < 255; divisor++) {
|
|
uint32_t tmpbaud, bauderr;
|
|
|
|
generator = clk_freq / (baud * (divisor + 1));
|
|
if (generator < 2 || generator > 65535) {
|
|
continue;
|
|
}
|
|
tmpbaud = clk_freq / (generator * (divisor + 1));
|
|
|
|
if (baud > tmpbaud) {
|
|
bauderr = baud - tmpbaud;
|
|
} else {
|
|
bauderr = tmpbaud - baud;
|
|
}
|
|
if (((bauderr * 100) / baud) < 3) {
|
|
break;
|
|
}
|
|
}
|
|
|
|
/*
|
|
* Set baud rate divisor and generator.
|
|
* -> This function is always called from a context in which
|
|
* the receiver/transmitter is disabled, the baud rate can
|
|
* be changed safely at this time.
|
|
*/
|
|
sys_write32(divisor, reg_base + XUARTPS_BAUDDIV_OFFSET);
|
|
sys_write32(generator, reg_base + XUARTPS_BAUDGEN_OFFSET);
|
|
}
|
|
}
|
|
|
|
/**
|
|
* @brief Initialize individual UART port
|
|
*
|
|
* This routine is called to reset the chip in a quiescent state.
|
|
*
|
|
* @param dev UART device struct
|
|
*
|
|
* @return 0 if successful, failed otherwise
|
|
*/
|
|
static int uart_xlnx_ps_init(const struct device *dev)
|
|
{
|
|
const struct uart_xlnx_ps_dev_config *dev_cfg = dev->config;
|
|
uint32_t reg_val;
|
|
#ifdef CONFIG_PINCTRL
|
|
int err;
|
|
#endif
|
|
|
|
DEVICE_MMIO_MAP(dev, K_MEM_CACHE_NONE);
|
|
uintptr_t reg_base = DEVICE_MMIO_GET(dev);
|
|
|
|
/* Disable RX/TX before changing any configuration data */
|
|
xlnx_ps_disable_uart(reg_base);
|
|
|
|
#ifdef CONFIG_PINCTRL
|
|
err = pinctrl_apply_state(dev_cfg->pincfg, PINCTRL_STATE_DEFAULT);
|
|
if (err < 0) {
|
|
return err;
|
|
}
|
|
#endif
|
|
|
|
/* Set initial character length / start/stop bit / parity configuration */
|
|
reg_val = sys_read32(reg_base + XUARTPS_MR_OFFSET);
|
|
reg_val &= (~(XUARTPS_MR_CHARLEN_MASK | XUARTPS_MR_STOPMODE_MASK |
|
|
XUARTPS_MR_PARITY_MASK));
|
|
reg_val |= XUARTPS_MR_CHARLEN_8_BIT | XUARTPS_MR_STOPMODE_1_BIT |
|
|
XUARTPS_MR_PARITY_NONE;
|
|
sys_write32(reg_val, reg_base + XUARTPS_MR_OFFSET);
|
|
|
|
/* Set RX FIFO trigger at 1 data bytes. */
|
|
sys_write32(0x01U, reg_base + XUARTPS_RXWM_OFFSET);
|
|
|
|
/* Disable all interrupts, polling mode is default */
|
|
sys_write32(XUARTPS_IXR_MASK, reg_base + XUARTPS_IDR_OFFSET);
|
|
|
|
/* Set the baud rate */
|
|
set_baudrate(dev, dev_cfg->baud_rate);
|
|
|
|
#ifdef CONFIG_UART_INTERRUPT_DRIVEN
|
|
|
|
/* Clear any pending interrupt flags */
|
|
sys_write32(XUARTPS_IXR_MASK, reg_base + XUARTPS_ISR_OFFSET);
|
|
|
|
/* Attach to & unmask the corresponding interrupt vector */
|
|
dev_cfg->irq_config_func(dev);
|
|
|
|
#endif
|
|
|
|
xlnx_ps_enable_uart(reg_base);
|
|
|
|
return 0;
|
|
}
|
|
|
|
/**
|
|
* @brief Poll the device for input.
|
|
*
|
|
* @param dev UART device struct
|
|
* @param c Pointer to character
|
|
*
|
|
* @return 0 if a character arrived, -1 if the input buffer if empty.
|
|
*/
|
|
static int uart_xlnx_ps_poll_in(const struct device *dev, unsigned char *c)
|
|
{
|
|
uintptr_t reg_base = DEVICE_MMIO_GET(dev);
|
|
uint32_t reg_val = sys_read32(reg_base + XUARTPS_SR_OFFSET);
|
|
|
|
if ((reg_val & XUARTPS_SR_RXEMPTY) == 0) {
|
|
*c = (unsigned char)sys_read32(reg_base + XUARTPS_FIFO_OFFSET);
|
|
return 0;
|
|
} else {
|
|
return -1;
|
|
}
|
|
}
|
|
|
|
/**
|
|
* @brief Output a character in polled mode.
|
|
*
|
|
* Checks if the transmitter is empty. If empty, a character is written to
|
|
* the data register.
|
|
*
|
|
* If the hardware flow control is enabled then the handshake signal CTS has to
|
|
* be asserted in order to send a character.
|
|
*
|
|
* @param dev UART device struct
|
|
* @param c Character to send
|
|
*
|
|
* @return Sent character
|
|
*/
|
|
static void uart_xlnx_ps_poll_out(const struct device *dev, unsigned char c)
|
|
{
|
|
uintptr_t reg_base = DEVICE_MMIO_GET(dev);
|
|
uint32_t reg_val;
|
|
|
|
/* wait for transmitter to ready to accept a character */
|
|
do {
|
|
reg_val = sys_read32(reg_base + XUARTPS_SR_OFFSET);
|
|
} while ((reg_val & XUARTPS_SR_TXEMPTY) == 0);
|
|
|
|
sys_write32((uint32_t)(c & 0xFF), reg_base + XUARTPS_FIFO_OFFSET);
|
|
|
|
do {
|
|
reg_val = sys_read32(reg_base + XUARTPS_SR_OFFSET);
|
|
} while ((reg_val & XUARTPS_SR_TXEMPTY) == 0);
|
|
}
|
|
|
|
/**
|
|
* @brief Converts a parity enum value to a Mode Register bit mask.
|
|
*
|
|
* Converts a value of an enumeration type provided by the driver
|
|
* framework for the configuration of the UART's parity setting
|
|
* into a bit mask within the Mode Register.
|
|
*
|
|
* It is assumed that the Mode Register contents that are being
|
|
* modified within this function come with the bits modified by
|
|
* this function already masked out by the caller.
|
|
*
|
|
* @param mode_reg Pointer to the Mode Register contents to which
|
|
* the parity configuration shall be added.
|
|
* @param parity Enumeration value to be converted to a bit mask.
|
|
*
|
|
* @return Indication of success, always true for this function
|
|
* as all parity modes supported by the API are also supported
|
|
* by the hardware.
|
|
*/
|
|
static inline bool uart_xlnx_ps_cfg2ll_parity(
|
|
uint32_t *mode_reg,
|
|
enum uart_config_parity parity)
|
|
{
|
|
/*
|
|
* Translate the new parity configuration to the mode register's
|
|
* bits [5..3] (PAR):
|
|
* 000b : even
|
|
* 001b : odd
|
|
* 010b : space
|
|
* 011b : mark
|
|
* 1xxb : none
|
|
*/
|
|
|
|
switch (parity) {
|
|
default:
|
|
case UART_CFG_PARITY_EVEN:
|
|
*mode_reg |= XUARTPS_MR_PARITY_EVEN;
|
|
break;
|
|
case UART_CFG_PARITY_ODD:
|
|
*mode_reg |= XUARTPS_MR_PARITY_ODD;
|
|
break;
|
|
case UART_CFG_PARITY_SPACE:
|
|
*mode_reg |= XUARTPS_MR_PARITY_SPACE;
|
|
break;
|
|
case UART_CFG_PARITY_MARK:
|
|
*mode_reg |= XUARTPS_MR_PARITY_MARK;
|
|
break;
|
|
case UART_CFG_PARITY_NONE:
|
|
*mode_reg |= XUARTPS_MR_PARITY_NONE;
|
|
break;
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
/**
|
|
* @brief Converts a stop bit enum value to a Mode Register bit mask.
|
|
*
|
|
* Converts a value of an enumeration type provided by the driver
|
|
* framework for the configuration of the UART's stop bit setting
|
|
* into a bit mask within the Mode Register.
|
|
*
|
|
* It is assumed that the Mode Register contents that are being
|
|
* modified within this function come with the bits modified by
|
|
* this function already masked out by the caller.
|
|
*
|
|
* @param mode_reg Pointer to the Mode Register contents to which
|
|
* the stop bit configuration shall be added.
|
|
* @param stopbits Enumeration value to be converted to a bit mask.
|
|
*
|
|
* @return Indication of success or failure in case of an unsupported
|
|
* stop bit configuration being provided by the caller.
|
|
*/
|
|
static inline bool uart_xlnx_ps_cfg2ll_stopbits(
|
|
uint32_t *mode_reg,
|
|
enum uart_config_stop_bits stopbits)
|
|
{
|
|
/*
|
|
* Translate the new stop bit configuration to the mode register's
|
|
* bits [7..6] (NBSTOP):
|
|
* 00b : 1 stop bit
|
|
* 01b : 1.5 stop bits
|
|
* 10b : 2 stop bits
|
|
* 11b : reserved
|
|
*/
|
|
|
|
switch (stopbits) {
|
|
case UART_CFG_STOP_BITS_0_5:
|
|
/* Controller doesn't support 0.5 stop bits */
|
|
return false;
|
|
default:
|
|
case UART_CFG_STOP_BITS_1:
|
|
*mode_reg |= XUARTPS_MR_STOPMODE_1_BIT;
|
|
break;
|
|
case UART_CFG_STOP_BITS_1_5:
|
|
*mode_reg |= XUARTPS_MR_STOPMODE_1_5_BIT;
|
|
break;
|
|
case UART_CFG_STOP_BITS_2:
|
|
*mode_reg |= XUARTPS_MR_STOPMODE_2_BIT;
|
|
break;
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
/**
|
|
* @brief Converts a data bit enum value to a Mode Register bit mask.
|
|
*
|
|
* Converts a value of an enumeration type provided by the driver
|
|
* framework for the configuration of the UART's data bit setting
|
|
* into a bit mask within the Mode Register.
|
|
*
|
|
* It is assumed that the Mode Register contents that are being
|
|
* modified within this function come with the bits modified by
|
|
* this function already masked out by the caller.
|
|
*
|
|
* @param mode_reg Pointer to the Mode Register contents to which
|
|
* the data bit configuration shall be added.
|
|
* @param databits Enumeration value to be converted to a bit mask.
|
|
*
|
|
* @return Indication of success or failure in case of an unsupported
|
|
* data bit configuration being provided by the caller.
|
|
*/
|
|
static inline bool uart_xlnx_ps_cfg2ll_databits(
|
|
uint32_t *mode_reg,
|
|
enum uart_config_data_bits databits)
|
|
{
|
|
/*
|
|
* Translate the new data bit configuration to the mode register's
|
|
* bits [2..1] (CHRL):
|
|
* 0xb : 8 data bits
|
|
* 10b : 7 data bits
|
|
* 11b : 6 data bits
|
|
*/
|
|
|
|
switch (databits) {
|
|
case UART_CFG_DATA_BITS_5:
|
|
case UART_CFG_DATA_BITS_9:
|
|
/* Controller doesn't support 5 or 9 data bits */
|
|
return false;
|
|
default:
|
|
case UART_CFG_DATA_BITS_8:
|
|
*mode_reg |= XUARTPS_MR_CHARLEN_8_BIT;
|
|
break;
|
|
case UART_CFG_DATA_BITS_7:
|
|
*mode_reg |= XUARTPS_MR_CHARLEN_7_BIT;
|
|
break;
|
|
case UART_CFG_DATA_BITS_6:
|
|
*mode_reg |= XUARTPS_MR_CHARLEN_6_BIT;
|
|
break;
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
/**
|
|
* @brief Converts a flow control enum value to a Modem Control
|
|
* Register bit mask.
|
|
*
|
|
* Converts a value of an enumeration type provided by the driver
|
|
* framework for the configuration of the UART's flow control
|
|
* setting into a bit mask within the Modem Control Register.
|
|
*
|
|
* It is assumed that the Modem Control Register contents that are
|
|
* being modified within this function come with the bits modified
|
|
* by this function already masked out by the caller.
|
|
*
|
|
* @param modemcr_reg Pointer to the Modem Control Register contents
|
|
* to which the flow control configuration shall
|
|
* be added.
|
|
* @param hwctrl Enumeration value to be converted to a bit mask.
|
|
*
|
|
* @return Indication of success or failure in case of an unsupported
|
|
* flow control configuration being provided by the caller.
|
|
*/
|
|
static inline bool uart_xlnx_ps_cfg2ll_hwctrl(
|
|
uint32_t *modemcr_reg,
|
|
enum uart_config_flow_control hwctrl)
|
|
{
|
|
/*
|
|
* Translate the new flow control configuration to the modem
|
|
* control register's bit [5] (FCM):
|
|
* 0b : no flow control
|
|
* 1b : RTS/CTS
|
|
*/
|
|
|
|
if (hwctrl == UART_CFG_FLOW_CTRL_RTS_CTS) {
|
|
*modemcr_reg |= XUARTPS_MODEMCR_FCM_RTS_CTS;
|
|
} else if (hwctrl == UART_CFG_FLOW_CTRL_NONE) {
|
|
*modemcr_reg |= XUARTPS_MODEMCR_FCM_NONE;
|
|
} else {
|
|
/* Only no flow control or RTS/CTS is supported. */
|
|
return false;
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
#ifdef CONFIG_UART_USE_RUNTIME_CONFIGURE
|
|
/**
|
|
* @brief Configures the UART device at run-time.
|
|
*
|
|
* Configures the UART device at run-time according to the
|
|
* configuration data provided by the caller.
|
|
*
|
|
* @param dev UART device struct
|
|
* @param cfg The configuration parameters to be applied.
|
|
*
|
|
* @return 0 if the configuration completed successfully, ENOTSUP
|
|
* error if an unsupported configuration parameter is detected.
|
|
*/
|
|
static int uart_xlnx_ps_configure(const struct device *dev,
|
|
const struct uart_config *cfg)
|
|
{
|
|
struct uart_xlnx_ps_dev_config *dev_cfg =
|
|
(struct uart_xlnx_ps_dev_config *)dev->config;
|
|
|
|
uintptr_t reg_base = DEVICE_MMIO_GET(dev);
|
|
uint32_t mode_reg = 0;
|
|
uint32_t modemcr_reg = 0;
|
|
|
|
/* Read the current mode register & modem control register values */
|
|
mode_reg = sys_read32(reg_base + XUARTPS_MR_OFFSET);
|
|
modemcr_reg = sys_read32(reg_base + XUARTPS_MODEMCR_OFFSET);
|
|
|
|
/* Mask out all items that might be re-configured */
|
|
mode_reg &= (~XUARTPS_MR_PARITY_MASK);
|
|
mode_reg &= (~XUARTPS_MR_STOPMODE_MASK);
|
|
mode_reg &= (~XUARTPS_MR_CHARLEN_MASK);
|
|
modemcr_reg &= (~XUARTPS_MODEMCR_FCM_MASK);
|
|
|
|
/* Assemble the updated registers, validity checks contained within */
|
|
if ((!uart_xlnx_ps_cfg2ll_parity(&mode_reg, cfg->parity)) ||
|
|
(!uart_xlnx_ps_cfg2ll_stopbits(&mode_reg, cfg->stop_bits)) ||
|
|
(!uart_xlnx_ps_cfg2ll_databits(&mode_reg, cfg->data_bits)) ||
|
|
(!uart_xlnx_ps_cfg2ll_hwctrl(&modemcr_reg, cfg->flow_ctrl))) {
|
|
return -ENOTSUP;
|
|
}
|
|
|
|
/* Disable the controller before modifying any config registers */
|
|
xlnx_ps_disable_uart(reg_base);
|
|
|
|
/* Set the baud rate */
|
|
set_baudrate(dev, cfg->baudrate);
|
|
dev_cfg->baud_rate = cfg->baudrate;
|
|
|
|
/* Write the two control registers */
|
|
sys_write32(mode_reg, reg_base + XUARTPS_MR_OFFSET);
|
|
sys_write32(modemcr_reg, reg_base + XUARTPS_MODEMCR_OFFSET);
|
|
|
|
/* Re-enable the controller */
|
|
xlnx_ps_enable_uart(reg_base);
|
|
|
|
return 0;
|
|
};
|
|
#endif /* CONFIG_UART_USE_RUNTIME_CONFIGURE */
|
|
|
|
/**
|
|
* @brief Converts a Mode Register bit mask to a parity configuration
|
|
* enum value.
|
|
*
|
|
* Converts a bit mask representing the UART's parity setting within
|
|
* the UART's Mode Register into a value of an enumeration type provided
|
|
* by the UART driver API.
|
|
*
|
|
* @param mode_reg The current Mode Register contents from which the
|
|
* parity setting shall be extracted.
|
|
*
|
|
* @return The current parity setting mapped to the UART driver API's
|
|
* enum type.
|
|
*/
|
|
static inline enum uart_config_parity uart_xlnx_ps_ll2cfg_parity(
|
|
uint32_t mode_reg)
|
|
{
|
|
/*
|
|
* Obtain the current parity configuration from the mode register's
|
|
* bits [5..3] (PAR):
|
|
* 000b : even -> reset value
|
|
* 001b : odd
|
|
* 010b : space
|
|
* 011b : mark
|
|
* 1xxb : none
|
|
*/
|
|
|
|
switch ((mode_reg & XUARTPS_MR_PARITY_MASK)) {
|
|
case XUARTPS_MR_PARITY_EVEN:
|
|
default:
|
|
return UART_CFG_PARITY_EVEN;
|
|
case XUARTPS_MR_PARITY_ODD:
|
|
return UART_CFG_PARITY_ODD;
|
|
case XUARTPS_MR_PARITY_SPACE:
|
|
return UART_CFG_PARITY_SPACE;
|
|
case XUARTPS_MR_PARITY_MARK:
|
|
return UART_CFG_PARITY_MARK;
|
|
case XUARTPS_MR_PARITY_NONE:
|
|
return UART_CFG_PARITY_NONE;
|
|
}
|
|
}
|
|
|
|
/**
|
|
* @brief Converts a Mode Register bit mask to a stop bit configuration
|
|
* enum value.
|
|
*
|
|
* Converts a bit mask representing the UART's stop bit setting within
|
|
* the UART's Mode Register into a value of an enumeration type provided
|
|
* by the UART driver API.
|
|
*
|
|
* @param mode_reg The current Mode Register contents from which the
|
|
* stop bit setting shall be extracted.
|
|
*
|
|
* @return The current stop bit setting mapped to the UART driver API's
|
|
* enum type.
|
|
*/
|
|
static inline enum uart_config_stop_bits uart_xlnx_ps_ll2cfg_stopbits(
|
|
uint32_t mode_reg)
|
|
{
|
|
/*
|
|
* Obtain the current stop bit configuration from the mode register's
|
|
* bits [7..6] (NBSTOP):
|
|
* 00b : 1 stop bit -> reset value
|
|
* 01b : 1.5 stop bits
|
|
* 10b : 2 stop bits
|
|
* 11b : reserved
|
|
*/
|
|
|
|
switch ((mode_reg & XUARTPS_MR_STOPMODE_MASK)) {
|
|
case XUARTPS_MR_STOPMODE_1_BIT:
|
|
default:
|
|
return UART_CFG_STOP_BITS_1;
|
|
case XUARTPS_MR_STOPMODE_1_5_BIT:
|
|
return UART_CFG_STOP_BITS_1_5;
|
|
case XUARTPS_MR_STOPMODE_2_BIT:
|
|
return UART_CFG_STOP_BITS_2;
|
|
}
|
|
}
|
|
|
|
/**
|
|
* @brief Converts a Mode Register bit mask to a data bit configuration
|
|
* enum value.
|
|
*
|
|
* Converts a bit mask representing the UART's data bit setting within
|
|
* the UART's Mode Register into a value of an enumeration type provided
|
|
* by the UART driver API.
|
|
*
|
|
* @param mode_reg The current Mode Register contents from which the
|
|
* data bit setting shall be extracted.
|
|
*
|
|
* @return The current data bit setting mapped to the UART driver API's
|
|
* enum type.
|
|
*/
|
|
static inline enum uart_config_data_bits uart_xlnx_ps_ll2cfg_databits(
|
|
uint32_t mode_reg)
|
|
{
|
|
/*
|
|
* Obtain the current data bit configuration from the mode register's
|
|
* bits [2..1] (CHRL):
|
|
* 0xb : 8 data bits -> reset value
|
|
* 10b : 7 data bits
|
|
* 11b : 6 data bits
|
|
*/
|
|
|
|
switch ((mode_reg & XUARTPS_MR_CHARLEN_MASK)) {
|
|
case XUARTPS_MR_CHARLEN_8_BIT:
|
|
default:
|
|
return UART_CFG_DATA_BITS_8;
|
|
case XUARTPS_MR_CHARLEN_7_BIT:
|
|
return UART_CFG_DATA_BITS_7;
|
|
case XUARTPS_MR_CHARLEN_6_BIT:
|
|
return UART_CFG_DATA_BITS_6;
|
|
}
|
|
}
|
|
|
|
/**
|
|
* @brief Converts a Modem Control Register bit mask to a flow control
|
|
* configuration enum value.
|
|
*
|
|
* Converts a bit mask representing the UART's flow control setting within
|
|
* the UART's Modem Control Register into a value of an enumeration type
|
|
* provided by the UART driver API.
|
|
*
|
|
* @param modemcr_reg The current Modem Control Register contents from
|
|
* which the parity setting shall be extracted.
|
|
*
|
|
* @return The current flow control setting mapped to the UART driver API's
|
|
* enum type.
|
|
*/
|
|
static inline enum uart_config_flow_control uart_xlnx_ps_ll2cfg_hwctrl(
|
|
uint32_t modemcr_reg)
|
|
{
|
|
/*
|
|
* Obtain the current flow control configuration from the modem
|
|
* control register's bit [5] (FCM):
|
|
* 0b : no flow control -> reset value
|
|
* 1b : RTS/CTS
|
|
*/
|
|
|
|
if ((modemcr_reg & XUARTPS_MODEMCR_FCM_MASK)
|
|
== XUARTPS_MODEMCR_FCM_RTS_CTS) {
|
|
return UART_CFG_FLOW_CTRL_RTS_CTS;
|
|
}
|
|
|
|
return UART_CFG_FLOW_CTRL_NONE;
|
|
}
|
|
|
|
#ifdef CONFIG_UART_USE_RUNTIME_CONFIGURE
|
|
/**
|
|
* @brief Returns the current configuration of the UART at run-time.
|
|
*
|
|
* Returns the current configuration of the UART at run-time by obtaining
|
|
* the current configuration from the UART's Mode and Modem Control Registers
|
|
* (exception: baud rate).
|
|
*
|
|
* @param dev UART device struct
|
|
* @param cfg Pointer to the data structure to which the current configuration
|
|
* shall be written.
|
|
*
|
|
* @return always 0.
|
|
*/
|
|
static int uart_xlnx_ps_config_get(const struct device *dev,
|
|
struct uart_config *cfg)
|
|
{
|
|
const struct uart_xlnx_ps_dev_config *dev_cfg = dev->config;
|
|
uintptr_t reg_base = DEVICE_MMIO_GET(dev);
|
|
|
|
/*
|
|
* Read the Mode & Modem control registers - they contain
|
|
* the current data / stop bit and parity settings (Mode
|
|
* Register) and the current flow control setting (Modem
|
|
* Control register).
|
|
*/
|
|
uint32_t mode_reg = sys_read32(reg_base + XUARTPS_MR_OFFSET);
|
|
uint32_t modemcr_reg = sys_read32(reg_base + XUARTPS_MODEMCR_OFFSET);
|
|
|
|
cfg->baudrate = dev_cfg->baud_rate;
|
|
cfg->parity = uart_xlnx_ps_ll2cfg_parity(mode_reg);
|
|
cfg->stop_bits = uart_xlnx_ps_ll2cfg_stopbits(mode_reg);
|
|
cfg->data_bits = uart_xlnx_ps_ll2cfg_databits(mode_reg);
|
|
cfg->flow_ctrl = uart_xlnx_ps_ll2cfg_hwctrl(modemcr_reg);
|
|
|
|
return 0;
|
|
}
|
|
#endif /* CONFIG_UART_USE_RUNTIME_CONFIGURE */
|
|
|
|
#if CONFIG_UART_INTERRUPT_DRIVEN
|
|
|
|
/**
|
|
* @brief Fill FIFO with data
|
|
*
|
|
* @param dev UART device struct
|
|
* @param tx_data Data to transmit
|
|
* @param size Number of bytes to send
|
|
*
|
|
* @return Number of bytes sent
|
|
*/
|
|
static int uart_xlnx_ps_fifo_fill(const struct device *dev,
|
|
const uint8_t *tx_data,
|
|
int size)
|
|
{
|
|
uintptr_t reg_base = DEVICE_MMIO_GET(dev);
|
|
uint32_t data_iter = 0;
|
|
|
|
sys_write32(XUARTPS_IXR_TXEMPTY, reg_base + XUARTPS_IDR_OFFSET);
|
|
while (size--) {
|
|
while ((sys_read32(reg_base + XUARTPS_SR_OFFSET) & XUARTPS_SR_TXFULL) != 0) {
|
|
}
|
|
sys_write32((uint32_t)tx_data[data_iter++], reg_base + XUARTPS_FIFO_OFFSET);
|
|
}
|
|
sys_write32(XUARTPS_IXR_TXEMPTY, reg_base + XUARTPS_IER_OFFSET);
|
|
|
|
return data_iter;
|
|
}
|
|
|
|
/**
|
|
* @brief Read data from FIFO
|
|
*
|
|
* @param dev UART device struct
|
|
* @param rxData Data container
|
|
* @param size Container size
|
|
*
|
|
* @return Number of bytes read
|
|
*/
|
|
static int uart_xlnx_ps_fifo_read(const struct device *dev, uint8_t *rx_data,
|
|
const int size)
|
|
{
|
|
uintptr_t reg_base = DEVICE_MMIO_GET(dev);
|
|
uint32_t reg_val = sys_read32(reg_base + XUARTPS_SR_OFFSET);
|
|
int inum = 0;
|
|
|
|
while (inum < size && (reg_val & XUARTPS_SR_RXEMPTY) == 0) {
|
|
rx_data[inum] = (uint8_t)sys_read32(reg_base
|
|
+ XUARTPS_FIFO_OFFSET);
|
|
inum++;
|
|
reg_val = sys_read32(reg_base + XUARTPS_SR_OFFSET);
|
|
}
|
|
|
|
return inum;
|
|
}
|
|
|
|
/**
|
|
* @brief Enable TX interrupt in IER
|
|
*
|
|
* @param dev UART device struct
|
|
*/
|
|
static void uart_xlnx_ps_irq_tx_enable(const struct device *dev)
|
|
{
|
|
uintptr_t reg_base = DEVICE_MMIO_GET(dev);
|
|
|
|
sys_write32(
|
|
(XUARTPS_IXR_TTRIG | XUARTPS_IXR_TXEMPTY),
|
|
reg_base + XUARTPS_IER_OFFSET);
|
|
}
|
|
|
|
/**
|
|
* @brief Disable TX interrupt in IER
|
|
*
|
|
* @param dev UART device struct
|
|
*/
|
|
static void uart_xlnx_ps_irq_tx_disable(const struct device *dev)
|
|
{
|
|
uintptr_t reg_base = DEVICE_MMIO_GET(dev);
|
|
|
|
sys_write32(
|
|
(XUARTPS_IXR_TTRIG | XUARTPS_IXR_TXEMPTY),
|
|
reg_base + XUARTPS_IDR_OFFSET);
|
|
}
|
|
|
|
/**
|
|
* @brief Check if Tx IRQ has been raised
|
|
*
|
|
* @param dev UART device struct
|
|
*
|
|
* @return 1 if an IRQ is ready, 0 otherwise
|
|
*/
|
|
static int uart_xlnx_ps_irq_tx_ready(const struct device *dev)
|
|
{
|
|
uintptr_t reg_base = DEVICE_MMIO_GET(dev);
|
|
uint32_t reg_val = sys_read32(reg_base + XUARTPS_SR_OFFSET);
|
|
|
|
if ((reg_val & (XUARTPS_SR_TTRIG | XUARTPS_SR_TXEMPTY)) == 0) {
|
|
return 0;
|
|
} else {
|
|
return 1;
|
|
}
|
|
}
|
|
|
|
/**
|
|
* @brief Check if nothing remains to be transmitted
|
|
*
|
|
* @param dev UART device struct
|
|
*
|
|
* @return 1 if nothing remains to be transmitted, 0 otherwise
|
|
*/
|
|
static int uart_xlnx_ps_irq_tx_complete(const struct device *dev)
|
|
{
|
|
uintptr_t reg_base = DEVICE_MMIO_GET(dev);
|
|
uint32_t reg_val = sys_read32(reg_base + XUARTPS_SR_OFFSET);
|
|
|
|
if ((reg_val & XUARTPS_SR_TXEMPTY) == 0) {
|
|
return 0;
|
|
} else {
|
|
return 1;
|
|
}
|
|
}
|
|
|
|
/**
|
|
* @brief Enable RX interrupt in IER
|
|
*
|
|
* @param dev UART device struct
|
|
*/
|
|
static void uart_xlnx_ps_irq_rx_enable(const struct device *dev)
|
|
{
|
|
uintptr_t reg_base = DEVICE_MMIO_GET(dev);
|
|
|
|
sys_write32(XUARTPS_IXR_RTRIG, reg_base + XUARTPS_IER_OFFSET);
|
|
}
|
|
|
|
/**
|
|
* @brief Disable RX interrupt in IER
|
|
*
|
|
* @param dev UART device struct
|
|
*/
|
|
static void uart_xlnx_ps_irq_rx_disable(const struct device *dev)
|
|
{
|
|
uintptr_t reg_base = DEVICE_MMIO_GET(dev);
|
|
|
|
sys_write32(XUARTPS_IXR_RTRIG, reg_base + XUARTPS_IDR_OFFSET);
|
|
}
|
|
|
|
/**
|
|
* @brief Check if Rx IRQ has been raised
|
|
*
|
|
* @param dev UART device struct
|
|
*
|
|
* @return 1 if an IRQ is ready, 0 otherwise
|
|
*/
|
|
static int uart_xlnx_ps_irq_rx_ready(const struct device *dev)
|
|
{
|
|
uintptr_t reg_base = DEVICE_MMIO_GET(dev);
|
|
uint32_t reg_val = sys_read32(reg_base + XUARTPS_ISR_OFFSET);
|
|
|
|
if ((reg_val & XUARTPS_IXR_RTRIG) == 0) {
|
|
return 0;
|
|
} else {
|
|
sys_write32(XUARTPS_IXR_RTRIG, reg_base + XUARTPS_ISR_OFFSET);
|
|
return 1;
|
|
}
|
|
}
|
|
|
|
/**
|
|
* @brief Enable error interrupt in IER
|
|
*
|
|
* @param dev UART device struct
|
|
*/
|
|
static void uart_xlnx_ps_irq_err_enable(const struct device *dev)
|
|
{
|
|
uintptr_t reg_base = DEVICE_MMIO_GET(dev);
|
|
|
|
sys_write32(
|
|
XUARTPS_IXR_TOVR /* [12] Transmitter FIFO Overflow */
|
|
| XUARTPS_IXR_TOUT /* [8] Receiver Timerout */
|
|
| XUARTPS_IXR_PARITY /* [7] Parity Error */
|
|
| XUARTPS_IXR_FRAMING /* [6] Receiver Framing Error */
|
|
| XUARTPS_IXR_RXOVR, /* [5] Receiver Overflow Error */
|
|
reg_base + XUARTPS_IER_OFFSET);
|
|
}
|
|
|
|
/**
|
|
* @brief Disable error interrupt in IER
|
|
*
|
|
* @param dev UART device struct
|
|
*
|
|
* @return 1 if an IRQ is ready, 0 otherwise
|
|
*/
|
|
static void uart_xlnx_ps_irq_err_disable(const struct device *dev)
|
|
{
|
|
uintptr_t reg_base = DEVICE_MMIO_GET(dev);
|
|
|
|
sys_write32(
|
|
XUARTPS_IXR_TOVR /* [12] Transmitter FIFO Overflow */
|
|
| XUARTPS_IXR_TOUT /* [8] Receiver Timerout */
|
|
| XUARTPS_IXR_PARITY /* [7] Parity Error */
|
|
| XUARTPS_IXR_FRAMING /* [6] Receiver Framing Error */
|
|
| XUARTPS_IXR_RXOVR, /* [5] Receiver Overflow Error */
|
|
reg_base + XUARTPS_IDR_OFFSET);
|
|
}
|
|
|
|
/**
|
|
* @brief Check if any IRQ is pending
|
|
*
|
|
* @param dev UART device struct
|
|
*
|
|
* @return 1 if an IRQ is pending, 0 otherwise
|
|
*/
|
|
static int uart_xlnx_ps_irq_is_pending(const struct device *dev)
|
|
{
|
|
uintptr_t reg_base = DEVICE_MMIO_GET(dev);
|
|
uint32_t reg_imr = sys_read32(reg_base + XUARTPS_IMR_OFFSET);
|
|
uint32_t reg_isr = sys_read32(reg_base + XUARTPS_ISR_OFFSET);
|
|
|
|
if ((reg_imr & reg_isr) != 0) {
|
|
return 1;
|
|
} else {
|
|
return 0;
|
|
}
|
|
}
|
|
|
|
/**
|
|
* @brief Update cached contents of IIR
|
|
*
|
|
* @param dev UART device struct
|
|
*
|
|
* @return Always 1
|
|
*/
|
|
static int uart_xlnx_ps_irq_update(const struct device *dev)
|
|
{
|
|
ARG_UNUSED(dev);
|
|
return 1;
|
|
}
|
|
|
|
/**
|
|
* @brief Set the callback function pointer for IRQ.
|
|
*
|
|
* @param dev UART device struct
|
|
* @param cb Callback function pointer.
|
|
*/
|
|
static void uart_xlnx_ps_irq_callback_set(const struct device *dev,
|
|
uart_irq_callback_user_data_t cb,
|
|
void *cb_data)
|
|
{
|
|
struct uart_xlnx_ps_dev_data_t *dev_data = dev->data;
|
|
|
|
dev_data->user_cb = cb;
|
|
dev_data->user_data = cb_data;
|
|
}
|
|
|
|
/**
|
|
* @brief Interrupt ce routine.
|
|
*
|
|
* This simply calls the callback function, if one exists.
|
|
*
|
|
* @param arg Argument to ISR.
|
|
*/
|
|
static void uart_xlnx_ps_isr(const struct device *dev)
|
|
{
|
|
const struct uart_xlnx_ps_dev_data_t *data = dev->data;
|
|
|
|
if (data->user_cb) {
|
|
data->user_cb(dev, data->user_data);
|
|
}
|
|
}
|
|
#endif /* CONFIG_UART_INTERRUPT_DRIVEN */
|
|
|
|
static const struct uart_driver_api uart_xlnx_ps_driver_api = {
|
|
.poll_in = uart_xlnx_ps_poll_in,
|
|
.poll_out = uart_xlnx_ps_poll_out,
|
|
#ifdef CONFIG_UART_USE_RUNTIME_CONFIGURE
|
|
.configure = uart_xlnx_ps_configure,
|
|
.config_get = uart_xlnx_ps_config_get,
|
|
#endif
|
|
#ifdef CONFIG_UART_INTERRUPT_DRIVEN
|
|
.fifo_fill = uart_xlnx_ps_fifo_fill,
|
|
.fifo_read = uart_xlnx_ps_fifo_read,
|
|
.irq_tx_enable = uart_xlnx_ps_irq_tx_enable,
|
|
.irq_tx_disable = uart_xlnx_ps_irq_tx_disable,
|
|
.irq_tx_ready = uart_xlnx_ps_irq_tx_ready,
|
|
.irq_tx_complete = uart_xlnx_ps_irq_tx_complete,
|
|
.irq_rx_enable = uart_xlnx_ps_irq_rx_enable,
|
|
.irq_rx_disable = uart_xlnx_ps_irq_rx_disable,
|
|
.irq_rx_ready = uart_xlnx_ps_irq_rx_ready,
|
|
.irq_err_enable = uart_xlnx_ps_irq_err_enable,
|
|
.irq_err_disable = uart_xlnx_ps_irq_err_disable,
|
|
.irq_is_pending = uart_xlnx_ps_irq_is_pending,
|
|
.irq_update = uart_xlnx_ps_irq_update,
|
|
.irq_callback_set = uart_xlnx_ps_irq_callback_set,
|
|
#endif
|
|
};
|
|
|
|
#ifdef CONFIG_UART_INTERRUPT_DRIVEN
|
|
|
|
#define UART_XLNX_PS_IRQ_CONF_FUNC_SET(port) \
|
|
.irq_config_func = uart_xlnx_ps_irq_config_##port,
|
|
|
|
#define UART_XLNX_PS_IRQ_CONF_FUNC(port) \
|
|
static void uart_xlnx_ps_irq_config_##port(const struct device *dev) \
|
|
{ \
|
|
IRQ_CONNECT(DT_INST_IRQN(port), \
|
|
DT_INST_IRQ(port, priority), \
|
|
uart_xlnx_ps_isr, DEVICE_DT_INST_GET(port), \
|
|
0); \
|
|
irq_enable(DT_INST_IRQN(port)); \
|
|
}
|
|
|
|
#else
|
|
|
|
#define UART_XLNX_PS_IRQ_CONF_FUNC_SET(port)
|
|
#define UART_XLNX_PS_IRQ_CONF_FUNC(port)
|
|
|
|
#endif /*CONFIG_UART_INTERRUPT_DRIVEN */
|
|
|
|
#define UART_XLNX_PS_DEV_DATA(port) \
|
|
static struct uart_xlnx_ps_dev_data_t uart_xlnx_ps_dev_data_##port
|
|
|
|
#if CONFIG_PINCTRL
|
|
#define UART_XLNX_PS_PINCTRL_DEFINE(port) PINCTRL_DT_INST_DEFINE(port);
|
|
#define UART_XLNX_PS_PINCTRL_INIT(port) .pincfg = PINCTRL_DT_INST_DEV_CONFIG_GET(port),
|
|
#else
|
|
#define UART_XLNX_PS_PINCTRL_DEFINE(port)
|
|
#define UART_XLNX_PS_PINCTRL_INIT(port)
|
|
#endif /* CONFIG_PINCTRL */
|
|
|
|
#define UART_XLNX_PS_DEV_CFG(port) \
|
|
static struct uart_xlnx_ps_dev_config uart_xlnx_ps_dev_cfg_##port = { \
|
|
DEVICE_MMIO_ROM_INIT(DT_DRV_INST(port)), \
|
|
.sys_clk_freq = DT_INST_PROP(port, clock_frequency), \
|
|
.baud_rate = DT_INST_PROP(port, current_speed), \
|
|
UART_XLNX_PS_IRQ_CONF_FUNC_SET(port) \
|
|
UART_XLNX_PS_PINCTRL_INIT(port) \
|
|
}
|
|
|
|
#define UART_XLNX_PS_INIT(port) \
|
|
DEVICE_DT_INST_DEFINE(port, \
|
|
uart_xlnx_ps_init, \
|
|
NULL, \
|
|
&uart_xlnx_ps_dev_data_##port, \
|
|
&uart_xlnx_ps_dev_cfg_##port, \
|
|
PRE_KERNEL_1, CONFIG_SERIAL_INIT_PRIORITY, \
|
|
&uart_xlnx_ps_driver_api)
|
|
|
|
#define UART_XLNX_INSTANTIATE(inst) \
|
|
UART_XLNX_PS_PINCTRL_DEFINE(inst) \
|
|
UART_XLNX_PS_IRQ_CONF_FUNC(inst); \
|
|
UART_XLNX_PS_DEV_DATA(inst); \
|
|
UART_XLNX_PS_DEV_CFG(inst); \
|
|
UART_XLNX_PS_INIT(inst);
|
|
|
|
DT_INST_FOREACH_STATUS_OKAY(UART_XLNX_INSTANTIATE)
|