* arch/arm/src/imx6/imx_serial.c
*
* Licensed to the Apache Software Foundation (ASF) under one or more
* contributor license agreements. See the NOTICE file distributed with
* this work for additional information regarding copyright ownership. The
* ASF licenses this file to you 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.
*
****************************************************************************/
* Included Files
****************************************************************************/
#include <nuttx/config.h>
#include <sys/types.h>
#include <stdint.h>
#include <stdbool.h>
#include <unistd.h>
#include <string.h>
#include <assert.h>
#include <errno.h>
#include <debug.h>
#ifdef CONFIG_SERIAL_TERMIOS
# include <termios.h>
#endif
#include <nuttx/irq.h>
#include <nuttx/arch.h>
#include <nuttx/spinlock.h>
#include <nuttx/init.h>
#include <nuttx/fs/ioctl.h>
#include <nuttx/serial/serial.h>
#include "chip.h"
#include "arm_internal.h"
#include "gic.h"
#include "hardware/imx_uart.h"
#include "imx_config.h"
#include "imx_lowputc.h"
#ifdef USE_SERIALDRIVER
* Pre-processor Definitions
****************************************************************************/
* always be ttyS0. If there is no console then will use the lowest
* numbered UART.
*/
#if defined(CONFIG_UART1_SERIAL_CONSOLE)
# define CONSOLE_DEV g_uart1port
# define TTYS0_DEV g_uart1port
# define UART1_ASSIGNED 1
#elif defined(CONFIG_UART2_SERIAL_CONSOLE)
# define CONSOLE_DEV g_uart2port
# define TTYS0_DEV g_uart2port
# define UART2_ASSIGNED 1
#elif defined(CONFIG_UART3_SERIAL_CONSOLE)
# define CONSOLE_DEV g_uart3port
# define TTYS0_DEV g_uart3port
# define UART3_ASSIGNED 1
#elif defined(CONFIG_UART4_SERIAL_CONSOLE)
# define CONSOLE_DEV g_uart4port
# define TTYS0_DEV g_uart4port
# define UART4_ASSIGNED 1
#elif defined(CONFIG_UART5_SERIAL_CONSOLE)
# define CONSOLE_DEV g_uart5port
# define TTYS5_DEV g_uart5port
# define UART5_ASSIGNED 1
#else
# undef CONSOLE_DEV
# if defined(CONFIG_IMX6_UART1)
# define TTYS0_DEV g_uart1port
# define UART1_ASSIGNED 1
# elif defined(CONFIG_IMX6_UART2)
# define TTYS0_DEV g_uart2port
# define UART2_ASSIGNED 1
# elif defined(CONFIG_IMX6_UART3)
# define TTYS0_DEV g_uart3port
# define UART3_ASSIGNED 1
# elif defined(CONFIG_IMX6_UART4)
# define TTYS0_DEV g_uart4port
# define UART4_ASSIGNED 1
# elif defined(CONFIG_IMX6_UART5)
# define TTYS0_DEV g_uart5port
# define UART5_ASSIGNED 1
# endif
#endif
* One of UART1-5 could be the console; one of UART1-5 has already been
* assigned to ttys0.
*/
#if defined(CONFIG_IMX6_UART1) && !defined(UART1_ASSIGNED)
# define TTYS1_DEV g_uart1port
# define UART1_ASSIGNED 1
#elif defined(CONFIG_IMX6_UART2) && !defined(UART2_ASSIGNED)
# define TTYS1_DEV g_uart2port
# define UART2_ASSIGNED 1
#elif defined(CONFIG_IMX6_UART3) && !defined(UART3_ASSIGNED)
# define TTYS1_DEV g_uart3port
# define UART3_ASSIGNED 1
#elif defined(CONFIG_IMX6_UART4) && !defined(UART4_ASSIGNED)
# define TTYS1_DEV g_uart4port
# define UART4_ASSIGNED 1
#elif defined(CONFIG_IMX6_UART5) && !defined(UART5_ASSIGNED)
# define TTYS1_DEV g_uart5port
# define UART5_ASSIGNED 1
#endif
* was either assigned as ttyS0 or ttys1. One of UART 1-5 could be the
* console. One of UART2-5 has already been assigned to ttys0 or ttyS1.
*/
#if defined(CONFIG_IMX6_UART2) && !defined(UART2_ASSIGNED)
# define TTYS2_DEV g_uart2port
# define UART2_ASSIGNED 1
#elif defined(CONFIG_IMX6_UART3) && !defined(UART3_ASSIGNED)
# define TTYS2_DEV g_uart3port
# define UART3_ASSIGNED 1
#elif defined(CONFIG_IMX6_UART4) && !defined(UART4_ASSIGNED)
# define TTYS2_DEV g_uart4port
# define UART4_ASSIGNED 1
#elif defined(CONFIG_IMX6_UART5) && !defined(UART5_ASSIGNED)
# define TTYS2_DEV g_uart5port
# define UART5_ASSIGNED 1
#endif
* those have already been assigned to ttsyS0, 1, or 2. One of
* UART32-5 could also be the console. One of UART3-5 has already
* been assigned to ttys0, 1, or 3.
*/
#if defined(CONFIG_IMX6_UART3) && !defined(UART3_ASSIGNED)
# define TTYS3_DEV g_uart3port
# define UART3_ASSIGNED 1
#elif defined(CONFIG_IMX6_UART4) && !defined(UART4_ASSIGNED)
# define TTYS3_DEV g_uart4port
# define UART4_ASSIGNED 1
#elif defined(CONFIG_IMX6_UART5) && !defined(UART5_ASSIGNED)
# define TTYS3_DEV g_uart5port
# define UART5_ASSIGNED 1
#endif
* those have already been assigned to ttsyS0, 1, 2 or 3. One of
* UART 4-5 could be the console. One of UART4-5 has already been
* assigned to ttys0, 1, 3, or 4.
*/
#if defined(CONFIG_IMX6_UART4) && !defined(UART4_ASSIGNED)
# define TTYS4_DEV g_uart4port
# define UART4_ASSIGNED 1
#elif defined(CONFIG_IMX6_UART5) && !defined(UART5_ASSIGNED)
# define TTYS4_DEV g_uart5port
# define UART5_ASSIGNED 1
#endif
#if defined(CONFIG_IMX6_UART5) && !defined(UART5_ASSIGNED)
# errnor UART5 was not assigned to a TTY.
#endif
* Private Types
****************************************************************************/
struct imx_uart_s
{
uint32_t uartbase;
uint32_t baud;
uint32_t ie;
uint32_t ucr1;
uint8_t irq;
uint8_t parity;
spinlock_t lock;
uint8_t bits;
uint8_t stopbits2:1;
#ifdef CONFIG_SERIAL_IFLOWCONTROL
uint8_t iflow:1;
#endif
#ifdef CONFIG_SERIAL_OFLOWCONTROL
uint8_t oflow:1;
#endif
};
* Private Function Prototypes
****************************************************************************/
static inline uint32_t imx_serialin(struct imx_uart_s *priv,
uint32_t offset);
static inline void imx_serialout(struct imx_uart_s *priv, uint32_t offset,
uint32_t value);
static inline void imx_disableuartint(struct imx_uart_s *priv,
uint32_t *ucr1);
static inline void imx_restoreuartint(struct imx_uart_s *priv,
uint32_t ucr1);
static inline void imx_waittxready(struct imx_uart_s *priv);
static int imx_setup(struct uart_dev_s *dev);
static void imx_shutdown(struct uart_dev_s *dev);
static int imx_attach(struct uart_dev_s *dev);
static void imx_detach(struct uart_dev_s *dev);
static int imx_interrupt(int irq, void *context, void *arg);
static int imx_ioctl(struct file *filep, int cmd, unsigned long arg);
static int imx_receive(struct uart_dev_s *dev, unsigned int *status);
static void imx_rxint(struct uart_dev_s *dev, bool enable);
static bool imx_rxavailable(struct uart_dev_s *dev);
static void imx_send(struct uart_dev_s *dev, int ch);
static void imx_txint(struct uart_dev_s *dev, bool enable);
static bool imx_txready(struct uart_dev_s *dev);
static bool imx_txempty(struct uart_dev_s *dev);
* Private Data
****************************************************************************/
static const struct uart_ops_s g_uart_ops =
{
.setup = imx_setup,
.shutdown = imx_shutdown,
.attach = imx_attach,
.detach = imx_detach,
.ioctl = imx_ioctl,
.receive = imx_receive,
.rxint = imx_rxint,
.rxavailable = imx_rxavailable,
#ifdef CONFIG_SERIAL_IFLOWCONTROL
.rxflowcontrol = NULL,
#endif
.send = imx_send,
.txint = imx_txint,
.txready = imx_txready,
.txempty = imx_txempty,
};
#ifdef CONFIG_IMX6_UART1
static char g_uart1rxbuffer[CONFIG_UART1_RXBUFSIZE];
static char g_uart1txbuffer[CONFIG_UART1_TXBUFSIZE];
#endif
#ifdef CONFIG_IMX6_UART2
static char g_uart2rxbuffer[CONFIG_UART2_RXBUFSIZE];
static char g_uart2txbuffer[CONFIG_UART2_TXBUFSIZE];
#endif
#ifdef CONFIG_IMX6_UART3
static char g_uart3rxbuffer[CONFIG_UART3_RXBUFSIZE];
static char g_uart3txbuffer[CONFIG_UART3_TXBUFSIZE];
#endif
#ifdef CONFIG_IMX6_UART4
static char g_uart4rxbuffer[CONFIG_UART4_RXBUFSIZE];
static char g_uart4txbuffer[CONFIG_UART4_TXBUFSIZE];
#endif
#ifdef CONFIG_IMX6_UART5
static char g_uart5rxbuffer[CONFIG_UART5_RXBUFSIZE];
static char g_uart5txbuffer[CONFIG_UART5_TXBUFSIZE];
#endif
#ifdef CONFIG_IMX6_UART1
static struct imx_uart_s g_uart1priv =
{
.uartbase = IMX_UART1_VBASE,
.baud = CONFIG_UART1_BAUD,
.irq = IMX_IRQ_UART1,
.parity = CONFIG_UART1_PARITY,
.lock = SP_UNLOCKED,
.bits = CONFIG_UART1_BITS,
.stopbits2 = CONFIG_UART1_2STOP,
};
static struct uart_dev_s g_uart1port =
{
.recv =
{
.size = CONFIG_UART1_RXBUFSIZE,
.buffer = g_uart1rxbuffer,
},
.xmit =
{
.size = CONFIG_UART1_TXBUFSIZE,
.buffer = g_uart1txbuffer,
},
.ops = &g_uart_ops,
.priv = &g_uart1priv,
};
#endif
#ifdef CONFIG_IMX6_UART2
static struct imx_uart_s g_uart2priv =
{
.uartbase = IMX_UART2_VBASE,
.baud = CONFIG_UART2_BAUD,
.irq = IMX_IRQ_UART2,
.parity = CONFIG_UART2_PARITY,
.lock = SP_UNLOCKED,
.bits = CONFIG_UART2_BITS,
.stopbits2 = CONFIG_UART2_2STOP,
};
static struct uart_dev_s g_uart2port =
{
.recv =
{
.size = CONFIG_UART2_RXBUFSIZE,
.buffer = g_uart2rxbuffer,
},
.xmit =
{
.size = CONFIG_UART2_TXBUFSIZE,
.buffer = g_uart2txbuffer,
},
.ops = &g_uart_ops,
.priv = &g_uart2priv,
};
#endif
#ifdef CONFIG_IMX6_UART3
static struct imx_uart_s g_uart3priv =
{
.uartbase = IMX_UART3_REGISTER_BASE,
.baud = IMX_UART3_VBASE,
.irq = IMX_IRQ_UART3,
.parity = CONFIG_UART3_PARITY,
.lock = SP_UNLOCKED,
.bits = CONFIG_UART3_BITS,
.stopbits2 = CONFIG_UART3_2STOP,
};
static struct uart_dev_s g_uart3port =
{
.recv =
{
.size = CONFIG_UART3_RXBUFSIZE,
.buffer = g_uart3rxbuffer,
},
.xmit =
{
.size = CONFIG_UART3_TXBUFSIZE,
.buffer = g_uart3txbuffer,
},
.ops = &g_uart_ops,
.priv = &g_uart3priv,
};
#endif
#ifdef CONFIG_IMX6_UART4
static struct imx_uart_s g_uart4priv =
{
.uartbase = IMX_UART4_REGISTER_BASE,
.baud = IMX_UART4_VBASE,
.irq = IMX_IRQ_UART4,
.parity = CONFIG_UART4_PARITY,
.lock = SP_UNLOCKED,
.bits = CONFIG_UART4_BITS,
.stopbits2 = CONFIG_UART4_2STOP,
};
static struct uart_dev_s g_uart4port =
{
.recv =
{
.size = CONFIG_UART4_RXBUFSIZE,
.buffer = g_uart4rxbuffer,
},
.xmit =
{
.size = CONFIG_UART4_TXBUFSIZE,
.buffer = g_uart4txbuffer,
},
.ops = &g_uart_ops,
.priv = &g_uart4priv,
};
#endif
#ifdef CONFIG_IMX6_UART5
static struct imx_uart_s g_uart5priv =
{
.uartbase = IMX_UART5_REGISTER_BASE,
.baud = IMX_UART5_VBASE,
.irq = IMX_IRQ_UART5,
.parity = CONFIG_UART5_PARITY,
.lock = SP_UNLOCKED,
.bits = CONFIG_UART5_BITS,
.stopbits2 = CONFIG_UART5_2STOP,
};
static struct uart_dev_s g_uart5port =
{
.recv =
{
.size = CONFIG_UART5_RXBUFSIZE,
.buffer = g_uart5rxbuffer,
},
.xmit =
{
.size = CONFIG_UART5_TXBUFSIZE,
.buffer = g_uart5txbuffer,
},
.ops = &g_uart_ops,
.priv = &g_uart5priv,
};
#endif
* Private Functions
****************************************************************************/
* Name: imx_serialin
****************************************************************************/
static inline uint32_t imx_serialin(struct imx_uart_s *priv, uint32_t offset)
{
return getreg32(priv->uartbase + offset);
}
* Name: imx_serialout
****************************************************************************/
static inline void imx_serialout(struct imx_uart_s *priv, uint32_t offset,
uint32_t value)
{
putreg32(value, priv->uartbase + offset);
}
* Name: imx_disableuartint
****************************************************************************/
static inline void imx_disableuartint(struct imx_uart_s *priv,
uint32_t *ucr1)
{
if (ucr1 != NULL)
{
*ucr1 = priv->ucr1 & (UART_UCR1_RRDYEN | UART_UCR1_TXEMPTYEN);
}
priv->ucr1 &= ~(UART_UCR1_RRDYEN | UART_UCR1_TXEMPTYEN);
imx_serialout(priv, UART_UCR1_OFFSET, priv->ucr1);
}
* Name: imx_restoreuartint
****************************************************************************/
static inline void imx_restoreuartint(struct imx_uart_s *priv, uint32_t ucr1)
{
* enabled/disabled.
*/
priv->ucr1 &= ~(UART_UCR1_RRDYEN | UART_UCR1_TXEMPTYEN);
priv->ucr1 |= ucr1 & (UART_UCR1_RRDYEN | UART_UCR1_TXEMPTYEN);
imx_serialout(priv, UART_UCR1_OFFSET, priv->ucr1);
}
* Name: imx_waittxready
****************************************************************************/
static inline void imx_waittxready(struct imx_uart_s *priv)
{
int tmp;
for (tmp = 1000 ; tmp > 0 ; tmp--)
{
if ((imx_serialin(priv, UART_UTS_OFFSET) & UART_UTS_TXFULL) == 0)
{
break;
}
}
}
* Name: imx_setup
*
* Description:
* Configure the UART baud, bits, parity, fifos, etc. This
* method is called the first time that the serial port is
* opened.
*
****************************************************************************/
static int imx_setup(struct uart_dev_s *dev)
{
struct imx_uart_s *priv = (struct imx_uart_s *)dev->priv;
#ifndef CONFIG_SUPPRESS_UART_CONFIG
struct uart_config_s config;
int ret;
config.baud = priv->baud;
config.parity = priv->parity;
config.bits = priv->bits;
config.stopbits2 = priv->stopbits2;
ret = imx_uart_configure(priv->uartbase, &config);
priv->ucr1 = imx_serialin(priv, UART_UCR1_OFFSET);
return ret;
#else
priv->ucr1 = imx_serialin(priv, UART_UCR1_OFFSET);
return OK;
#endif
}
* Name: imx_shutdown
*
* Description:
* Disable the UART. This method is called when the serial
* port is closed
*
****************************************************************************/
static void imx_shutdown(struct uart_dev_s *dev)
{
struct imx_uart_s *priv = (struct imx_uart_s *)dev->priv;
imx_serialout(priv, UART_UCR1_OFFSET, 0);
imx_serialout(priv, UART_UCR2_OFFSET, 0);
imx_serialout(priv, UART_UCR3_OFFSET, 0);
imx_serialout(priv, UART_UCR4_OFFSET, 0);
}
* Name: imx_attach
*
* Description:
* Configure the UART to operation in interrupt driven mode. This method
* is called when the serial port is opened. Normally, this is just after
* the setup() method is called, however, the serial console may operate in
* a non-interrupt driven mode during the boot phase.
*
* RX and TX interrupts are not enabled when by the attach method (unless
* the hardware supports multiple levels of interrupt enabling). The RX
* and TX interrupts are not enabled until the txint() and rxint() methods
* are called.
*
****************************************************************************/
static int imx_attach(struct uart_dev_s *dev)
{
struct imx_uart_s *priv = (struct imx_uart_s *)dev->priv;
int ret;
ret = irq_attach(priv->irq, imx_interrupt, dev);
if (ret == OK)
{
up_set_irq_type(priv->irq, IRQ_HIGH_LEVEL);
* in the UART
*/
up_enable_irq(priv->irq);
}
return ret;
}
* Name: imx_detach
*
* Description:
* Detach UART interrupts. This method is called when the serial port is
* closed normally just before the shutdown method is called. The
* exception is the serial console which is never shutdown.
*
****************************************************************************/
static void imx_detach(struct uart_dev_s *dev)
{
struct imx_uart_s *priv = (struct imx_uart_s *)dev->priv;
up_disable_irq(priv->irq);
irq_detach(priv->irq);
}
* Name: imx_interrupt (and front-ends)
*
* Description:
* This is the UART interrupt handler. It will be invoked when an
* interrupt is received on the 'irq'. It should call uart_xmitchars or
* uart_recvchars to perform the appropriate data transfers. The
* interrupt handling logic must be able to map the 'arg' to the
* appropriate uart_dev_s structure in order to call these functions.
*
****************************************************************************/
static int imx_interrupt(int irq, void *context, void *arg)
{
struct uart_dev_s *dev = (struct uart_dev_s *)arg;
struct imx_uart_s *priv;
uint32_t usr1;
int passes = 0;
DEBUGASSERT(dev != NULL && dev->priv != NULL);
priv = (struct imx_uart_s *)dev->priv;
* until we have been looping for a long time.
*/
for (; ; )
{
* termination conditions
*/
usr1 = imx_serialin(priv, UART_USR1_OFFSET);
usr1 &= (UART_USR1_RRDY | UART_USR1_TRDY);
if (usr1 == 0 || passes > 256)
{
return OK;
}
if (usr1 & UART_USR1_RRDY)
{
uart_recvchars(dev);
}
if (usr1 & UART_USR1_TRDY &&
(imx_serialin(priv, UART_UCR1_OFFSET) & UART_UCR1_TXEMPTYEN) != 0)
{
uart_xmitchars(dev);
}
* is some hardware failure condition.
*/
passes++;
}
}
* Name: imx_ioctl
*
* Description:
* All ioctl calls will be routed through this method
*
****************************************************************************/
static int imx_ioctl(struct file *filep, int cmd, unsigned long arg)
{
#if defined(CONFIG_SERIAL_TIOCSERGSTRUCT) || defined(CONFIG_SERIAL_TERMIOS)
struct inode *inode = filep->f_inode;
struct uart_dev_s *dev = inode->i_private;
irqstate_t flags;
#endif
int ret = OK;
switch (cmd)
{
#ifdef CONFIG_SERIAL_TIOCSERGSTRUCT
case TIOCSERGSTRUCT:
{
struct imx_uart_s *user = (struct imx_uart_s *)arg;
if (!user)
{
ret = -EINVAL;
}
else
{
memcpy(user, dev, sizeof(struct imx_uart_s));
}
}
break;
#endif
#ifdef CONFIG_SERIAL_TERMIOS
case TCGETS:
{
struct termios *termiosp = (struct termios *)arg;
struct imx_uart_s *priv = (struct imx_uart_s *)dev->priv;
if (!termiosp)
{
ret = -EINVAL;
break;
}
termiosp->c_cflag = ((priv->parity != 0) ? PARENB : 0) |
((priv->parity == 1) ? PARODD : 0);
termiosp->c_cflag |= (priv->stopbits2) ? CSTOPB : 0;
#ifdef CONFIG_SERIAL_OFLOWCONTROL
termiosp->c_cflag |= ((priv->oflow) ? CCTS_OFLOW : 0);
#endif
#ifdef CONFIG_SERIAL_IFLOWCONTROL
termiosp->c_cflag |= ((priv->iflow) ? CRTS_IFLOW : 0);
#endif
cfsetispeed(termiosp, priv->baud);
switch (priv->bits)
{
case 5:
termiosp->c_cflag |= CS5;
break;
case 6:
termiosp->c_cflag |= CS6;
break;
case 7:
termiosp->c_cflag |= CS7;
break;
default:
case 8:
termiosp->c_cflag |= CS8;
break;
#if defined(CS9)
case 9:
termiosp->c_cflag |= CS9;
break;
#endif
}
}
break;
case TCSETS:
{
struct termios *termiosp = (struct termios *)arg;
struct imx_uart_s *priv = (struct imx_uart_s *)dev->priv;
uint32_t baud;
uint32_t ie;
uint8_t parity;
uint8_t nbits;
bool stop2;
if ((!termiosp)
#ifdef CONFIG_SERIAL_OFLOWCONTROL
|| ((termiosp->c_cflag & CCTS_OFLOW) && (priv->cts_gpio == 0))
#endif
#ifdef CONFIG_SERIAL_IFLOWCONTROL
|| ((termiosp->c_cflag & CRTS_IFLOW) && (priv->rts_gpio == 0))
#endif
)
{
ret = -EINVAL;
break;
}
ret = OK;
baud = cfgetispeed(termiosp);
switch (termiosp->c_cflag & CSIZE)
{
case CS5:
nbits = 5;
break;
case CS6:
nbits = 6;
break;
case CS7:
nbits = 7;
break;
case CS8:
nbits = 8;
break;
#if defined(CS9)
case CS9:
nbits = 9;
break;
#endif
default:
ret = -EINVAL;
break;
}
if ((termiosp->c_cflag & PARENB) != 0)
{
parity = (termiosp->c_cflag & PARODD) ? 1 : 2;
}
else
{
parity = 0;
}
stop2 = (termiosp->c_cflag & CSTOPB) != 0;
if (ret == OK)
{
priv->baud = baud;
priv->parity = parity;
priv->bits = nbits;
priv->stopbits2 = stop2;
#ifdef CONFIG_SERIAL_OFLOWCONTROL
priv->oflow = (termiosp->c_cflag & CCTS_OFLOW) != 0;
#endif
#ifdef CONFIG_SERIAL_IFLOWCONTROL
priv->iflow = (termiosp->c_cflag & CRTS_IFLOW) != 0;
#endif
* implement TCSADRAIN / TCSAFLUSH
*/
flags = spin_lock_irqsave(&priv->lock);
imx_disableuartint(priv, &ie);
ret = imx_setup(dev);
imx_restoreuartint(priv, ie);
priv->ie = ie;
spin_unlock_irqrestore(&priv->lock, flags);
}
}
break;
#endif
case TIOCSBRK:
case TIOCCBRK:
default:
ret = -ENOTTY;
break;
}
return ret;
}
* Name: imx_receive
*
* Description:
* Called (usually) from the interrupt level to receive one
* character from the UART. Error bits associated with the
* receipt are provided in the return 'status'.
*
****************************************************************************/
static int imx_receive(struct uart_dev_s *dev, unsigned int *status)
{
struct imx_uart_s *priv = (struct imx_uart_s *)dev->priv;
uint32_t rxd0;
rxd0 = imx_serialin(priv, UART_RXD_OFFSET);
*status = rxd0;
return (rxd0 & UART_RXD_DATA_MASK) >> UART_RXD_DATA_SHIFT;
}
* Name: imx_rxint
*
* Description:
* Call to enable or disable RX interrupts
*
****************************************************************************/
static void imx_rxint(struct uart_dev_s *dev, bool enable)
{
struct imx_uart_s *priv = (struct imx_uart_s *)dev->priv;
if (enable)
{
#ifndef CONFIG_SUPPRESS_SERIAL_INTS
priv->ucr1 |= UART_UCR1_RRDYEN;
#endif
}
else
{
priv->ucr1 &= ~UART_UCR1_RRDYEN;
}
imx_serialout(priv, UART_UCR1_OFFSET, priv->ucr1);
}
* Name: imx_rxavailable
*
* Description:
* Return true if the receive fifo is not empty
*
****************************************************************************/
static bool imx_rxavailable(struct uart_dev_s *dev)
{
struct imx_uart_s *priv = (struct imx_uart_s *)dev->priv;
return ((imx_serialin(priv, UART_USR2_OFFSET) & UART_USR2_RDR) != 0);
}
* Name: imx_send
*
* Description:
* This method will send one byte on the UART
*
****************************************************************************/
static void imx_send(struct uart_dev_s *dev, int ch)
{
struct imx_uart_s *priv = (struct imx_uart_s *)dev->priv;
imx_serialout(priv, UART_TXD_OFFSET, (uint32_t)ch);
}
* Name: imx_txint
*
* Description:
* Call to enable or disable TX interrupts
*
****************************************************************************/
static void imx_txint(struct uart_dev_s *dev, bool enable)
{
struct imx_uart_s *priv = (struct imx_uart_s *)dev->priv;
* there may still be a transmission in progress).
*/
if (enable)
{
#ifndef CONFIG_SUPPRESS_SERIAL_INTS
priv->ucr1 |= UART_UCR1_TXEMPTYEN;
#endif
}
else
{
priv->ucr1 &= ~UART_UCR1_TXEMPTYEN;
}
imx_serialout(priv, UART_UCR1_OFFSET, priv->ucr1);
}
* Name: imx_txready
*
* Description:
* Return true if the tranmsit fifo is not full
*
****************************************************************************/
static bool imx_txready(struct uart_dev_s *dev)
{
struct imx_uart_s *priv = (struct imx_uart_s *)dev->priv;
return ((imx_serialin(priv, UART_UTS_OFFSET) & UART_UTS_TXFULL) == 0);
}
* Name: imx_txempty
*
* Description:
* Return true if the transmit fifo is empty
*
****************************************************************************/
static bool imx_txempty(struct uart_dev_s *dev)
{
struct imx_uart_s *priv = (struct imx_uart_s *)dev->priv;
return ((imx_serialin(priv, UART_USR2_OFFSET) & UART_USR2_TXDC) != 0);
}
* Public Functions
****************************************************************************/
* Name: imx_earlyserialinit
*
* Description:
* Performs the low level UART initialization early in
* debug so that the serial console will be available
* during bootup. This must be called before arm_serialinit.
*
****************************************************************************/
void imx_earlyserialinit(void)
{
* -- including all clocking and pin configuration -- was performed by the
* function imx_lowsetup() earlier in the boot sequence.
*/
* when they are first opened.
*/
#ifdef CONSOLE_DEV
CONSOLE_DEV.isconsole = true;
imx_setup(&CONSOLE_DEV);
#endif
}
* Name: arm_serialinit
*
* Description:
* Register serial console and serial ports. This assumes
* that imx_earlyserialinit was called previously.
*
****************************************************************************/
void arm_serialinit(void)
{
#ifdef CONSOLE_DEV
uart_register("/dev/console", &CONSOLE_DEV);
#endif
#ifdef TTYS0_DEV
uart_register("/dev/ttyS0", &TTYS0_DEV);
# ifdef TTYS1_DEV
uart_register("/dev/ttyS1", &TTYS1_DEV);
# ifdef TTYS2_DEV
uart_register("/dev/ttyS2", &TTYS2_DEV);
# ifdef TTYS3_DEV
uart_register("/dev/ttyS3", &TTYS2_DEV);
# ifdef TTYS4_DEV
uart_register("/dev/ttyS4", &TTYS2_DEV);
# endif
# endif
# endif
# endif
#endif
}
* Name: up_putc
*
* Description:
* Provide priority, low-level access to support OS debug
* writes
*
****************************************************************************/
void up_putc(int ch)
{
struct imx_uart_s *priv = (struct imx_uart_s *)CONSOLE_DEV.priv;
uint32_t ier;
* a byte.
*/
imx_disableuartint(priv, &ier);
imx_lowputc(ch);
imx_restoreuartint(priv, ier);
}
#else
* Pre-processor Definitions
****************************************************************************/
# undef IMX_CONSOLE_VBASE
# if defined(CONFIG_UART1_SERIAL_CONSOLE)
# define IMX_CONSOLE_VBASE IMX_UART1_VBASE
# elif defined(CONFIG_UART2_SERIAL_CONSOLE)
# define IMX_CONSOLE_VBASE IMX_UART2_VBASE
# elif defined(CONFIG_UART3_SERIAL_CONSOLE)
# define IMX_CONSOLE_VBASE IMX_UART3_VBASE
# elif defined(CONFIG_UART4_SERIAL_CONSOLE)
# define IMX_CONSOLE_VBASE IMX_UART4_VBASE
# elif defined(CONFIG_UART5_SERIAL_CONSOLE)
# define IMX_CONSOLE_VBASE IMX_UART5_VBASE
# endif
* Private Functions
****************************************************************************/
#ifdef IMX_CONSOLE_VBASE
static inline void imx_waittxready(void)
{
int tmp;
for (tmp = 1000 ; tmp > 0 ; tmp--)
{
* in the TX FIFO.
*/
if ((getreg32(IMX_CONSOLE_VBASE + UART_UTS) & UART_UTS_TXFULL) == 0)
{
break;
}
}
}
#endif
* Public Functions
****************************************************************************/
void up_putc(int ch)
{
#ifdef IMX_CONSOLE_VBASE
imx_waittxready();
putreg32((uint16_t)ch, IMX_CONSOLE_VBASE + UART_TXD_OFFSET);
#endif
}
#endif