* arch/arm/src/nrf91/nrf91_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>
#include <nuttx/arch.h>
#include <nuttx/spinlock.h>
#include <nuttx/fs/ioctl.h>
#include <nuttx/serial/serial.h>
#ifdef CONFIG_SERIAL_TERMIOS
# include <termios.h>
#endif
#include <arch/board/board.h>
#include "arm_internal.h"
#include "chip.h"
#include "nrf91_config.h"
#include "hardware/nrf91_uarte.h"
#include "nrf91_clockconfig.h"
#include "nrf91_lowputc.h"
#include "nrf91_serial.h"
* Pre-processor Definitions
****************************************************************************/
#ifdef HAVE_UART1
# define NRF91_NUART 2
#else
# define NRF91_NUART 1
#endif
* provide some minimal implementation of up_putc.
*/
#if defined(HAVE_UART_DEVICE) && defined(USE_SERIALDRIVER)
* always be ttyS0. If there is no console then will use the lowest
* numbered UART.
*/
#ifdef CONFIG_UART0_SERIAL_CONSOLE
# define CONSOLE_DEV g_uart0port
# define TTYS0_DEV g_uart0port
#elif defined(CONFIG_UART1_SERIAL_CONSOLE)
# define CONSOLE_DEV g_uart1port
# define TTYS0_DEV g_uart1port
#endif
* Private Types
****************************************************************************/
struct nrf91_dev_s
{
uintptr_t uartbase;
uint8_t irq;
bool rx_available;
struct uart_config_s config;
};
* Private Function Prototypes
****************************************************************************/
static int nrf91_setup(struct uart_dev_s *dev);
static void nrf91_shutdown(struct uart_dev_s *dev);
static int nrf91_attach(struct uart_dev_s *dev);
static void nrf91_detach(struct uart_dev_s *dev);
static int nrf91_interrupt(int irq, void *context, void *arg);
static int nrf91_ioctl(struct file *filep, int cmd, unsigned long arg);
static int nrf91_receive(struct uart_dev_s *dev, unsigned int *status);
static void nrf91_rxint(struct uart_dev_s *dev, bool enable);
static bool nrf91_rxavailable(struct uart_dev_s *dev);
static void nrf91_send(struct uart_dev_s *dev, int ch);
static void nrf91_txint(struct uart_dev_s *dev, bool enable);
static bool nrf91_txready(struct uart_dev_s *dev);
static bool nrf91_txempty(struct uart_dev_s *dev);
* Private Data
****************************************************************************/
static const struct uart_ops_s g_uart_ops =
{
.setup = nrf91_setup,
.shutdown = nrf91_shutdown,
.attach = nrf91_attach,
.detach = nrf91_detach,
.ioctl = nrf91_ioctl,
.receive = nrf91_receive,
.rxint = nrf91_rxint,
.rxavailable = nrf91_rxavailable,
#ifdef CONFIG_SERIAL_IFLOWCONTROL
.rxflowcontrol = NULL,
#endif
.send = nrf91_send,
.txint = nrf91_txint,
.txready = nrf91_txready,
.txempty = nrf91_txempty,
};
#ifdef HAVE_UART0
static char g_uart0rxbuffer[CONFIG_UART0_RXBUFSIZE];
static char g_uart0txbuffer[CONFIG_UART0_TXBUFSIZE];
#endif
#ifdef HAVE_UART1
static char g_uart1rxbuffer[CONFIG_UART1_RXBUFSIZE];
static char g_uart1txbuffer[CONFIG_UART1_TXBUFSIZE];
#endif
#ifdef HAVE_UART0
static struct nrf91_dev_s g_uart0priv =
{
.uartbase = NRF91_UART0_BASE,
.irq = NRF91_IRQ_SERIAL0,
.rx_available = false,
.config =
{
.baud = CONFIG_UART0_BAUD,
.parity = CONFIG_UART0_PARITY,
.bits = CONFIG_UART0_BITS,
.stopbits2 = CONFIG_UART0_2STOP,
#ifdef CONFIG_UART0_IFLOWCONTROL
.iflow = true,
#endif
#ifdef CONFIG_UART0_OFLOWCONTROL
.oflow = true,
#endif
.txpin = BOARD_UART0_TX_PIN,
.rxpin = BOARD_UART0_RX_PIN,
}
};
static uart_dev_t g_uart0port =
{
.recv =
{
.size = CONFIG_UART0_RXBUFSIZE,
.buffer = g_uart0rxbuffer,
},
.xmit =
{
.size = CONFIG_UART0_TXBUFSIZE,
.buffer = g_uart0txbuffer,
},
.ops = &g_uart_ops,
.priv = &g_uart0priv,
};
#endif
#ifdef HAVE_UART1
static struct nrf91_dev_s g_uart1priv =
{
.uartbase = NRF91_UART1_BASE,
.irq = NRF91_IRQ_SERIAL1,
.rx_available = false,
.config =
{
.baud = CONFIG_UART1_BAUD,
.parity = CONFIG_UART1_PARITY,
.bits = CONFIG_UART1_BITS,
.stopbits2 = CONFIG_UART1_2STOP,
#ifdef CONFIG_UART1_IFLOWCONTROL
.iflow = true,
#endif
#ifdef CONFIG_UART1_OFLOWCONTROL
.oflow = true,
#endif
.txpin = BOARD_UART1_TX_PIN,
.rxpin = BOARD_UART1_RX_PIN,
}
};
static uart_dev_t 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
static struct uart_dev_s * const g_uart_devs[NRF91_NUART] =
{
#ifdef HAVE_UART0
[0] = &g_uart0port,
#endif
#ifdef HAVE_UART1
[1] = &g_uart1port
#endif
};
* Private Functions
****************************************************************************/
* Name: nrf91_setup
*
* Description:
* Configure the UART baud, bits, parity, etc. This method is called the
* first time that the serial port is opened.
*
****************************************************************************/
static int nrf91_setup(struct uart_dev_s *dev)
{
#ifndef CONFIG_SUPPRESS_UART_CONFIG
struct nrf91_dev_s *priv = (struct nrf91_dev_s *)dev->priv;
* Rx interrupt keeps firing.
* configuring is done on __start
*
* UPDATE 19.12.2019: No problems described above were observed,
* but just in case we leave the above note for some time.
*/
nrf91_usart_configure(priv->uartbase, &priv->config);
#endif
return OK;
}
* Name: nrf91_shutdown
*
* Description:
* Disable the UART. This method is called when the serial
* port is closed
*
****************************************************************************/
static void nrf91_shutdown(struct uart_dev_s *dev)
{
struct nrf91_dev_s *priv = (struct nrf91_dev_s *)dev->priv;
nrf91_usart_disable(priv->uartbase, &priv->config);
}
* Name: nrf91_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 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 nrf91_attach(struct uart_dev_s *dev)
{
struct nrf91_dev_s *priv = (struct nrf91_dev_s *)dev->priv;
int ret;
* disabled in the C2 register.
*/
ret = irq_attach(priv->irq, nrf91_interrupt, dev);
if (ret == OK)
{
up_enable_irq(priv->irq);
}
return ret;
}
* Name: nrf91_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 nrf91_detach(struct uart_dev_s *dev)
{
struct nrf91_dev_s *priv = (struct nrf91_dev_s *)dev->priv;
putreg32(UART_INT_RXDRDY, priv->uartbase + NRF91_UART_INTENCLR_OFFSET);
up_disable_irq(priv->irq);
irq_detach(priv->irq);
}
* Name: nrf91_interrupt
*
* 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 nrf91_interrupt(int irq, void *context, void *arg)
{
struct uart_dev_s *dev = (struct uart_dev_s *)arg;
struct nrf91_dev_s *priv;
uint32_t regval;
DEBUGASSERT(dev != NULL && dev->priv != NULL);
priv = (struct nrf91_dev_s *)dev->priv;
regval = getreg32(priv->uartbase + NRF91_UART_EVENTS_RXDRDY_OFFSET);
if (regval != 0)
{
putreg32(0, priv->uartbase + NRF91_UART_EVENTS_RXDRDY_OFFSET);
priv->rx_available = true;
uart_recvchars(dev);
}
putreg32(0, priv->uartbase + NRF91_UART_ERRORSRC_OFFSET);
return OK;
}
* Name: nrf91_set_format
*
* Description:
* Set the serial line format and speed.
*
****************************************************************************/
#ifdef CONFIG_SERIAL_TERMIOS
void nrf91_set_format(struct uart_dev_s *dev)
{
struct nrf91_dev_s *priv = (struct nrf91_dev_s *)dev->priv;
nrf91_usart_setformat(priv->uartbase, &priv->config);
}
#endif
* Name: nrf91_ioctl
*
* Description:
* All ioctl calls will be routed through this method
*
****************************************************************************/
static int nrf91_ioctl(struct file *filep, int cmd, unsigned long arg)
{
#ifdef CONFIG_SERIAL_TERMIOS
struct inode *inode = filep->f_inode;
struct uart_dev_s *dev = inode->i_private;
struct nrf91_dev_s *priv = (struct nrf91_dev_s *)dev->priv;
struct uart_config_s *config = &priv->config;
#endif
int ret = OK;
switch (cmd)
{
#ifdef CONFIG_SERIAL_TERMIOS
case TCGETS:
{
struct termios *termiosp = (struct termios *)arg;
if (!termiosp)
{
ret = -EINVAL;
break;
}
termiosp->c_cflag = ((config->parity != 0) ? PARENB : 0)
| ((config->parity == 1) ? PARODD : 0)
| ((config->stopbits2) ? CSTOPB : 0) |
#ifdef CONFIG_SERIAL_OFLOWCONTROL
((config->oflow) ? CCTS_OFLOW : 0) |
#endif
#ifdef CONFIG_SERIAL_IFLOWCONTROL
((config->iflow) ? CRTS_IFLOW : 0) |
#endif
CS8;
cfsetispeed(termiosp, config->baud);
break;
}
case TCSETS:
{
struct termios *termiosp = (struct termios *)arg;
if (!termiosp)
{
ret = -EINVAL;
break;
}
if ((termiosp->c_cflag & CSIZE) != CS8)
{
ret = -EINVAL;
break;
}
#ifndef HAVE_UART_STOPBITS
if ((termiosp->c_cflag & CSTOPB) != 0)
{
ret = -EINVAL;
break;
}
#endif
if (termiosp->c_cflag & PARODD)
{
ret = -EINVAL;
break;
}
if (termiosp->c_cflag & PARENB)
{
config->parity = (termiosp->c_cflag & PARODD) ? 1 : 2;
}
else
{
config->parity = 0;
}
#ifdef HAVE_UART_STOPBITS
config->stopbits2 = (termiosp->c_cflag & CSTOPB) != 0;
#endif
* that only one speed is supported.
*/
config->baud = cfgetispeed(termiosp);
nrf91_set_format(dev);
break;
}
#endif
default:
{
ret = -ENOTTY;
break;
}
}
return ret;
}
* Name: nrf91_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 nrf91_receive(struct uart_dev_s *dev, unsigned int *status)
{
struct nrf91_dev_s *priv = (struct nrf91_dev_s *)dev->priv;
uint32_t data;
data = getreg32(priv->uartbase + NRF91_UART_RXD_OFFSET);
priv->rx_available = false;
if (status)
{
*status = 0x00;
}
return data;
}
* Name: nrf91_rxint
*
* Description:
* Call to enable or disable RX interrupts
*
****************************************************************************/
static void nrf91_rxint(struct uart_dev_s *dev, bool enable)
{
struct nrf91_dev_s *priv = (struct nrf91_dev_s *)dev->priv;
if (enable)
{
#ifndef CONFIG_SUPPRESS_SERIAL_INTS
* (or an Rx timeout occurs).
*/
putreg32(UART_INT_RXDRDY, priv->uartbase + NRF91_UART_INTENSET_OFFSET);
putreg32(1, priv->uartbase + NRF91_UART_TASKS_STARTRX_OFFSET);
#endif
}
else
{
putreg32(UART_INT_RXDRDY, priv->uartbase + NRF91_UART_INTENCLR_OFFSET);
putreg32(1, priv->uartbase + NRF91_UART_TASKS_STOPRX_OFFSET);
}
}
* Name: nrf91_rxavailable
*
* Description:
* Return true if the receive register is not empty
*
****************************************************************************/
static bool nrf91_rxavailable(struct uart_dev_s *dev)
{
struct nrf91_dev_s *priv = (struct nrf91_dev_s *)dev->priv;
return priv->rx_available;
}
* Name: nrf91_send
*
* Description:
* This method will send one byte on the UART.
*
****************************************************************************/
static void nrf91_send(struct uart_dev_s *dev, int ch)
{
struct nrf91_dev_s *priv = (struct nrf91_dev_s *)dev->priv;
putreg32(0, priv->uartbase + NRF91_UART_EVENTS_TXDRDY_OFFSET);
putreg32(1, priv->uartbase + NRF91_UART_TASKS_STARTTX_OFFSET);
putreg32(ch, priv->uartbase + NRF91_UART_TXD_OFFSET);
while (getreg32(priv->uartbase + NRF91_UART_EVENTS_TXDRDY_OFFSET) == 0)
{
}
putreg32(1, priv->uartbase + NRF91_UART_TASKS_STOPTX_OFFSET);
}
* Name: nrf91_txint
*
* Description:
* Call to enable or disable TX interrupts
*
****************************************************************************/
static void nrf91_txint(struct uart_dev_s *dev, bool enable)
{
if (enable)
{
#ifndef CONFIG_SUPPRESS_SERIAL_INTS
irqstate_t flags;
flags = enter_critical_section();
* interrupts disabled (note this may recurse).
*/
uart_xmitchars(dev);
leave_critical_section(flags);
#endif
}
else
{
}
}
* Name: nrf91_txready
*
* Description:
* Return true if the tranmsit data register is empty
*
****************************************************************************/
static bool nrf91_txready(struct uart_dev_s *dev)
{
return true;
}
* Name: nrf91_txempty
*
* Description:
* Return true if the transmit data register is empty
*
****************************************************************************/
static bool nrf91_txempty(struct uart_dev_s *dev)
{
return true;
}
* Public Functions
****************************************************************************/
* Name: nrf91_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 nrf91_serialinit. NOTE: This function depends on GPIO pin
* configuration performed in nrf91_lowsetup() and main clock
* initialization performed in nrf_clock_configure().
*
****************************************************************************/
#ifdef USE_EARLYSERIALINIT
void nrf91_earlyserialinit(void)
{
#ifdef HAVE_UART_CONSOLE
CONSOLE_DEV.isconsole = true;
nrf91_setup(&CONSOLE_DEV);
#endif
}
#endif
* Name: arm_serialinit
*
* Description:
* Register serial console and serial ports. This assumes
* that nrf91_earlyserialinit was called previously.
*
* Input Parameters:
* None
*
* Returns Value:
* None
*
****************************************************************************/
void arm_serialinit(void)
{
unsigned minor = 0;
unsigned i = 0;
char devname[16];
#ifdef HAVE_UART_CONSOLE
uart_register("/dev/console", &CONSOLE_DEV);
uart_register("/dev/ttyS0", &TTYS0_DEV);
minor = 1;
#endif
strlcpy(devname, "/dev/ttySx", sizeof(devname));
for (i = 0; i < NRF91_NUART; i++)
{
if (g_uart_devs[i] == NULL)
{
continue;
}
if (g_uart_devs[i]->isconsole)
{
continue;
}
devname[9] = '0' + minor++;
uart_register(devname, g_uart_devs[i]);
}
}
* Name: up_putc
*
* Description:
* Provide priority, low-level access to support OS debug writes
*
****************************************************************************/
void up_putc(int ch)
{
#ifdef HAVE_UART_CONSOLE
arm_lowputc(ch);
#endif
}
#else
* Name: up_putc
*
* Description:
* Provide priority, low-level access to support OS debug writes
*
****************************************************************************/
void up_putc(int ch)
{
#ifdef HAVE_UART_CONSOLE
arm_lowputc(ch);
#endif
}
#endif