2011-01-11 15:54:54 +01:00
/*
* Freescale STMP37XX / STMP378X Application UART driver
*
* Author : dmitry pervushin < dimka @ embeddedalley . com >
*
* Copyright 2008 - 2010 Freescale Semiconductor , Inc .
* Copyright 2008 Embedded Alley Solutions , Inc All Rights Reserved .
*
* The code contained herein is licensed under the GNU General Public
* License . You may obtain a copy of the GNU General Public License
* Version 2 or later at the following locations :
*
* http : //www.opensource.org/licenses/gpl-license.html
* http : //www.gnu.org/copyleft/gpl.html
*/
2014-10-10 13:13:28 +02:00
# if defined(CONFIG_SERIAL_MXS_AUART_CONSOLE) && defined(CONFIG_MAGIC_SYSRQ)
# define SUPPORT_SYSRQ
# endif
2011-01-11 15:54:54 +01:00
# include <linux/kernel.h>
# include <linux/errno.h>
# include <linux/init.h>
# include <linux/console.h>
# include <linux/interrupt.h>
# include <linux/module.h>
# include <linux/slab.h>
# include <linux/wait.h>
# include <linux/tty.h>
# include <linux/tty_driver.h>
# include <linux/tty_flip.h>
# include <linux/serial.h>
# include <linux/serial_core.h>
# include <linux/platform_device.h>
# include <linux/device.h>
# include <linux/clk.h>
# include <linux/delay.h>
# include <linux/io.h>
2012-06-18 10:06:09 -03:00
# include <linux/of_device.h>
2012-11-16 16:03:53 +08:00
# include <linux/dma-mapping.h>
2013-02-26 13:47:41 +08:00
# include <linux/dmaengine.h>
2011-01-11 15:54:54 +01:00
# include <asm/cacheflush.h>
2014-10-10 18:53:26 +02:00
# include <linux/gpio.h>
# include <linux/gpio/consumer.h>
2014-10-10 18:53:25 +02:00
# include <linux/err.h>
2014-10-10 18:53:26 +02:00
# include <linux/irq.h>
2014-10-10 18:53:25 +02:00
# include "serial_mctrl_gpio.h"
2011-01-11 15:54:54 +01:00
# define MXS_AUART_PORTS 5
2013-10-03 09:32:03 +02:00
# define MXS_AUART_FIFO_SIZE 16
2011-01-11 15:54:54 +01:00
# define AUART_CTRL0 0x00000000
# define AUART_CTRL0_SET 0x00000004
# define AUART_CTRL0_CLR 0x00000008
# define AUART_CTRL0_TOG 0x0000000c
# define AUART_CTRL1 0x00000010
# define AUART_CTRL1_SET 0x00000014
# define AUART_CTRL1_CLR 0x00000018
# define AUART_CTRL1_TOG 0x0000001c
# define AUART_CTRL2 0x00000020
# define AUART_CTRL2_SET 0x00000024
# define AUART_CTRL2_CLR 0x00000028
# define AUART_CTRL2_TOG 0x0000002c
# define AUART_LINECTRL 0x00000030
# define AUART_LINECTRL_SET 0x00000034
# define AUART_LINECTRL_CLR 0x00000038
# define AUART_LINECTRL_TOG 0x0000003c
# define AUART_LINECTRL2 0x00000040
# define AUART_LINECTRL2_SET 0x00000044
# define AUART_LINECTRL2_CLR 0x00000048
# define AUART_LINECTRL2_TOG 0x0000004c
# define AUART_INTR 0x00000050
# define AUART_INTR_SET 0x00000054
# define AUART_INTR_CLR 0x00000058
# define AUART_INTR_TOG 0x0000005c
# define AUART_DATA 0x00000060
# define AUART_STAT 0x00000070
# define AUART_DEBUG 0x00000080
# define AUART_VERSION 0x00000090
# define AUART_AUTOBAUD 0x000000a0
# define AUART_CTRL0_SFTRST (1 << 31)
# define AUART_CTRL0_CLKGATE (1 << 30)
2012-11-16 16:03:53 +08:00
# define AUART_CTRL0_RXTO_ENABLE (1 << 27)
# define AUART_CTRL0_RXTIMEOUT(v) (((v) & 0x7ff) << 16)
# define AUART_CTRL0_XFER_COUNT(v) ((v) & 0xffff)
# define AUART_CTRL1_XFER_COUNT(v) ((v) & 0xffff)
# define AUART_CTRL2_DMAONERR (1 << 26)
# define AUART_CTRL2_TXDMAE (1 << 25)
# define AUART_CTRL2_RXDMAE (1 << 24)
2011-01-11 15:54:54 +01:00
# define AUART_CTRL2_CTSEN (1 << 15)
2012-08-08 10:37:59 +08:00
# define AUART_CTRL2_RTSEN (1 << 14)
2011-01-11 15:54:54 +01:00
# define AUART_CTRL2_RTS (1 << 11)
# define AUART_CTRL2_RXE (1 << 9)
# define AUART_CTRL2_TXE (1 << 8)
# define AUART_CTRL2_UARTEN (1 << 0)
2015-08-11 11:46:01 +00:00
# define AUART_LINECTRL_BAUD_DIV_MAX 0x003fffc0
# define AUART_LINECTRL_BAUD_DIV_MIN 0x000000ec
2011-01-11 15:54:54 +01:00
# define AUART_LINECTRL_BAUD_DIVINT_SHIFT 16
# define AUART_LINECTRL_BAUD_DIVINT_MASK 0xffff0000
# define AUART_LINECTRL_BAUD_DIVINT(v) (((v) & 0xffff) << 16)
# define AUART_LINECTRL_BAUD_DIVFRAC_SHIFT 8
# define AUART_LINECTRL_BAUD_DIVFRAC_MASK 0x00003f00
# define AUART_LINECTRL_BAUD_DIVFRAC(v) (((v) & 0x3f) << 8)
# define AUART_LINECTRL_WLEN_MASK 0x00000060
# define AUART_LINECTRL_WLEN(v) (((v) & 0x3) << 5)
# define AUART_LINECTRL_FEN (1 << 4)
# define AUART_LINECTRL_STP2 (1 << 3)
# define AUART_LINECTRL_EPS (1 << 2)
# define AUART_LINECTRL_PEN (1 << 1)
# define AUART_LINECTRL_BRK (1 << 0)
# define AUART_INTR_RTIEN (1 << 22)
# define AUART_INTR_TXIEN (1 << 21)
# define AUART_INTR_RXIEN (1 << 20)
# define AUART_INTR_CTSMIEN (1 << 17)
# define AUART_INTR_RTIS (1 << 6)
# define AUART_INTR_TXIS (1 << 5)
# define AUART_INTR_RXIS (1 << 4)
# define AUART_INTR_CTSMIS (1 << 1)
# define AUART_STAT_BUSY (1 << 29)
# define AUART_STAT_CTS (1 << 28)
# define AUART_STAT_TXFE (1 << 27)
# define AUART_STAT_TXFF (1 << 25)
# define AUART_STAT_RXFE (1 << 24)
# define AUART_STAT_OERR (1 << 19)
# define AUART_STAT_BERR (1 << 18)
# define AUART_STAT_PERR (1 << 17)
# define AUART_STAT_FERR (1 << 16)
2012-11-16 16:03:53 +08:00
# define AUART_STAT_RXCOUNT_MASK 0xffff
2011-01-11 15:54:54 +01:00
static struct uart_driver auart_driver ;
2012-11-16 16:03:52 +08:00
enum mxs_auart_type {
IMX23_AUART ,
IMX28_AUART ,
} ;
2011-01-11 15:54:54 +01:00
struct mxs_auart_port {
struct uart_port port ;
2012-11-16 16:03:53 +08:00
# define MXS_AUART_DMA_ENABLED 0x2
# define MXS_AUART_DMA_TX_SYNC 2 /* bit 2 */
# define MXS_AUART_DMA_RX_READY 3 /* bit 3 */
2013-08-03 10:09:14 -04:00
# define MXS_AUART_RTSCTS 4 /* bit 4 */
2012-11-16 16:03:53 +08:00
unsigned long flags ;
2014-10-10 18:53:26 +02:00
unsigned int mctrl_prev ;
2012-11-16 16:03:52 +08:00
enum mxs_auart_type devtype ;
2011-01-11 15:54:54 +01:00
struct clk * clk ;
struct device * dev ;
2012-11-16 16:03:53 +08:00
/* for DMA */
struct scatterlist tx_sgl ;
struct dma_chan * tx_dma_chan ;
void * tx_dma_buf ;
struct scatterlist rx_sgl ;
struct dma_chan * rx_dma_chan ;
void * rx_dma_buf ;
2014-10-10 18:53:25 +02:00
struct mctrl_gpios * gpios ;
2014-10-10 18:53:26 +02:00
int gpio_irq [ UART_GPIO_MAX ] ;
bool ms_irq_enabled ;
2011-01-11 15:54:54 +01:00
} ;
2015-05-02 00:40:03 +09:00
static const struct platform_device_id mxs_auart_devtype [ ] = {
2012-11-16 16:03:52 +08:00
{ . name = " mxs-auart-imx23 " , . driver_data = IMX23_AUART } ,
{ . name = " mxs-auart-imx28 " , . driver_data = IMX28_AUART } ,
{ /* sentinel */ }
} ;
MODULE_DEVICE_TABLE ( platform , mxs_auart_devtype ) ;
2015-03-16 20:17:11 +01:00
static const struct of_device_id mxs_auart_dt_ids [ ] = {
2012-11-16 16:03:52 +08:00
{
. compatible = " fsl,imx28-auart " ,
. data = & mxs_auart_devtype [ IMX28_AUART ]
} , {
. compatible = " fsl,imx23-auart " ,
. data = & mxs_auart_devtype [ IMX23_AUART ]
} , { /* sentinel */ }
} ;
MODULE_DEVICE_TABLE ( of , mxs_auart_dt_ids ) ;
static inline int is_imx28_auart ( struct mxs_auart_port * s )
{
return s - > devtype = = IMX28_AUART ;
}
2012-11-16 16:03:53 +08:00
static inline bool auart_dma_enabled ( struct mxs_auart_port * s )
{
return s - > flags & MXS_AUART_DMA_ENABLED ;
}
2011-01-11 15:54:54 +01:00
static void mxs_auart_stop_tx ( struct uart_port * u ) ;
# define to_auart_port(u) container_of(u, struct mxs_auart_port, port)
2012-11-16 16:03:53 +08:00
static void mxs_auart_tx_chars ( struct mxs_auart_port * s ) ;
static void dma_tx_callback ( void * param )
2011-01-11 15:54:54 +01:00
{
2012-11-16 16:03:53 +08:00
struct mxs_auart_port * s = param ;
2011-01-11 15:54:54 +01:00
struct circ_buf * xmit = & s - > port . state - > xmit ;
2012-11-16 16:03:53 +08:00
dma_unmap_sg ( s - > dev , & s - > tx_sgl , 1 , DMA_TO_DEVICE ) ;
/* clear the bit used to serialize the DMA tx. */
clear_bit ( MXS_AUART_DMA_TX_SYNC , & s - > flags ) ;
2014-03-17 18:06:10 +01:00
smp_mb__after_atomic ( ) ;
2012-11-16 16:03:53 +08:00
/* wake up the possible processes. */
if ( uart_circ_chars_pending ( xmit ) < WAKEUP_CHARS )
uart_write_wakeup ( & s - > port ) ;
mxs_auart_tx_chars ( s ) ;
}
static int mxs_auart_dma_tx ( struct mxs_auart_port * s , int size )
{
struct dma_async_tx_descriptor * desc ;
struct scatterlist * sgl = & s - > tx_sgl ;
struct dma_chan * channel = s - > tx_dma_chan ;
u32 pio ;
/* [1] : send PIO. Note, the first pio word is CTRL1. */
pio = AUART_CTRL1_XFER_COUNT ( size ) ;
desc = dmaengine_prep_slave_sg ( channel , ( struct scatterlist * ) & pio ,
1 , DMA_TRANS_NONE , 0 ) ;
if ( ! desc ) {
dev_err ( s - > dev , " step 1 error \n " ) ;
return - EINVAL ;
}
/* [2] : set DMA buffer. */
sg_init_one ( sgl , s - > tx_dma_buf , size ) ;
dma_map_sg ( s - > dev , sgl , 1 , DMA_TO_DEVICE ) ;
desc = dmaengine_prep_slave_sg ( channel , sgl ,
1 , DMA_MEM_TO_DEV , DMA_PREP_INTERRUPT | DMA_CTRL_ACK ) ;
if ( ! desc ) {
dev_err ( s - > dev , " step 2 error \n " ) ;
return - EINVAL ;
}
/* [3] : submit the DMA */
desc - > callback = dma_tx_callback ;
desc - > callback_param = s ;
dmaengine_submit ( desc ) ;
dma_async_issue_pending ( channel ) ;
return 0 ;
}
static void mxs_auart_tx_chars ( struct mxs_auart_port * s )
{
struct circ_buf * xmit = & s - > port . state - > xmit ;
if ( auart_dma_enabled ( s ) ) {
2013-01-07 23:11:06 -02:00
u32 i = 0 ;
2012-11-16 16:03:53 +08:00
int size ;
void * buffer = s - > tx_dma_buf ;
if ( test_and_set_bit ( MXS_AUART_DMA_TX_SYNC , & s - > flags ) )
return ;
while ( ! uart_circ_empty ( xmit ) & & ! uart_tx_stopped ( & s - > port ) ) {
size = min_t ( u32 , UART_XMIT_SIZE - i ,
CIRC_CNT_TO_END ( xmit - > head ,
xmit - > tail ,
UART_XMIT_SIZE ) ) ;
memcpy ( buffer + i , xmit - > buf + xmit - > tail , size ) ;
xmit - > tail = ( xmit - > tail + size ) & ( UART_XMIT_SIZE - 1 ) ;
i + = size ;
if ( i > = UART_XMIT_SIZE )
break ;
}
if ( uart_tx_stopped ( & s - > port ) )
mxs_auart_stop_tx ( & s - > port ) ;
if ( i ) {
mxs_auart_dma_tx ( s , i ) ;
} else {
clear_bit ( MXS_AUART_DMA_TX_SYNC , & s - > flags ) ;
2014-03-17 18:06:10 +01:00
smp_mb__after_atomic ( ) ;
2012-11-16 16:03:53 +08:00
}
return ;
}
2011-01-11 15:54:54 +01:00
while ( ! ( readl ( s - > port . membase + AUART_STAT ) &
AUART_STAT_TXFF ) ) {
if ( s - > port . x_char ) {
s - > port . icount . tx + + ;
writel ( s - > port . x_char ,
s - > port . membase + AUART_DATA ) ;
s - > port . x_char = 0 ;
continue ;
}
if ( ! uart_circ_empty ( xmit ) & & ! uart_tx_stopped ( & s - > port ) ) {
s - > port . icount . tx + + ;
writel ( xmit - > buf [ xmit - > tail ] ,
s - > port . membase + AUART_DATA ) ;
xmit - > tail = ( xmit - > tail + 1 ) & ( UART_XMIT_SIZE - 1 ) ;
} else
break ;
}
2011-11-22 14:22:56 +01:00
if ( uart_circ_chars_pending ( xmit ) < WAKEUP_CHARS )
uart_write_wakeup ( & s - > port ) ;
2011-01-11 15:54:54 +01:00
if ( uart_circ_empty ( & ( s - > port . state - > xmit ) ) )
writel ( AUART_INTR_TXIEN ,
s - > port . membase + AUART_INTR_CLR ) ;
else
writel ( AUART_INTR_TXIEN ,
s - > port . membase + AUART_INTR_SET ) ;
if ( uart_tx_stopped ( & s - > port ) )
mxs_auart_stop_tx ( & s - > port ) ;
}
static void mxs_auart_rx_char ( struct mxs_auart_port * s )
{
int flag ;
u32 stat ;
u8 c ;
c = readl ( s - > port . membase + AUART_DATA ) ;
stat = readl ( s - > port . membase + AUART_STAT ) ;
flag = TTY_NORMAL ;
s - > port . icount . rx + + ;
if ( stat & AUART_STAT_BERR ) {
s - > port . icount . brk + + ;
if ( uart_handle_break ( & s - > port ) )
goto out ;
} else if ( stat & AUART_STAT_PERR ) {
s - > port . icount . parity + + ;
} else if ( stat & AUART_STAT_FERR ) {
s - > port . icount . frame + + ;
}
/*
* Mask off conditions which should be ingored .
*/
stat & = s - > port . read_status_mask ;
if ( stat & AUART_STAT_BERR ) {
flag = TTY_BREAK ;
} else if ( stat & AUART_STAT_PERR )
flag = TTY_PARITY ;
else if ( stat & AUART_STAT_FERR )
flag = TTY_FRAME ;
if ( stat & AUART_STAT_OERR )
s - > port . icount . overrun + + ;
if ( uart_handle_sysrq_char ( & s - > port , c ) )
goto out ;
uart_insert_char ( & s - > port , stat , AUART_STAT_OERR , c , flag ) ;
out :
writel ( stat , s - > port . membase + AUART_STAT ) ;
}
static void mxs_auart_rx_chars ( struct mxs_auart_port * s )
{
u32 stat = 0 ;
for ( ; ; ) {
stat = readl ( s - > port . membase + AUART_STAT ) ;
if ( stat & AUART_STAT_RXFE )
break ;
mxs_auart_rx_char ( s ) ;
}
writel ( stat , s - > port . membase + AUART_STAT ) ;
2013-01-03 15:53:06 +01:00
tty_flip_buffer_push ( & s - > port . state - > port ) ;
2011-01-11 15:54:54 +01:00
}
static int mxs_auart_request_port ( struct uart_port * u )
{
return 0 ;
}
static int mxs_auart_verify_port ( struct uart_port * u ,
struct serial_struct * ser )
{
if ( u - > type ! = PORT_UNKNOWN & & u - > type ! = PORT_IMX )
return - EINVAL ;
return 0 ;
}
static void mxs_auart_config_port ( struct uart_port * u , int flags )
{
}
static const char * mxs_auart_type ( struct uart_port * u )
{
struct mxs_auart_port * s = to_auart_port ( u ) ;
return dev_name ( s - > dev ) ;
}
static void mxs_auart_release_port ( struct uart_port * u )
{
}
static void mxs_auart_set_mctrl ( struct uart_port * u , unsigned mctrl )
{
2014-10-10 18:53:25 +02:00
struct mxs_auart_port * s = to_auart_port ( u ) ;
2011-01-11 15:54:54 +01:00
u32 ctrl = readl ( u - > membase + AUART_CTRL2 ) ;
2012-12-13 14:27:43 +01:00
ctrl & = ~ ( AUART_CTRL2_RTSEN | AUART_CTRL2_RTS ) ;
2012-08-08 10:37:59 +08:00
if ( mctrl & TIOCM_RTS ) {
2014-09-10 15:06:24 -04:00
if ( uart_cts_enabled ( u ) )
2012-08-08 10:37:59 +08:00
ctrl | = AUART_CTRL2_RTSEN ;
2012-12-13 14:27:43 +01:00
else
ctrl | = AUART_CTRL2_RTS ;
2012-08-08 10:37:59 +08:00
}
2011-01-11 15:54:54 +01:00
writel ( ctrl , u - > membase + AUART_CTRL2 ) ;
2014-10-10 18:53:25 +02:00
mctrl_gpio_set ( s - > gpios , mctrl ) ;
2011-01-11 15:54:54 +01:00
}
2014-10-10 18:53:26 +02:00
# define MCTRL_ANY_DELTA (TIOCM_RI | TIOCM_DSR | TIOCM_CD | TIOCM_CTS)
static u32 mxs_auart_modem_status ( struct mxs_auart_port * s , u32 mctrl )
{
u32 mctrl_diff ;
mctrl_diff = mctrl ^ s - > mctrl_prev ;
s - > mctrl_prev = mctrl ;
if ( mctrl_diff & MCTRL_ANY_DELTA & & s - > ms_irq_enabled & &
s - > port . state ! = NULL ) {
if ( mctrl_diff & TIOCM_RI )
s - > port . icount . rng + + ;
if ( mctrl_diff & TIOCM_DSR )
s - > port . icount . dsr + + ;
if ( mctrl_diff & TIOCM_CD )
uart_handle_dcd_change ( & s - > port , mctrl & TIOCM_CD ) ;
if ( mctrl_diff & TIOCM_CTS )
uart_handle_cts_change ( & s - > port , mctrl & TIOCM_CTS ) ;
wake_up_interruptible ( & s - > port . state - > port . delta_msr_wait ) ;
}
return mctrl ;
}
2011-01-11 15:54:54 +01:00
static u32 mxs_auart_get_mctrl ( struct uart_port * u )
{
2014-10-10 18:53:25 +02:00
struct mxs_auart_port * s = to_auart_port ( u ) ;
2011-01-11 15:54:54 +01:00
u32 stat = readl ( u - > membase + AUART_STAT ) ;
2014-10-10 18:53:24 +02:00
u32 mctrl = 0 ;
2011-01-11 15:54:54 +01:00
if ( stat & AUART_STAT_CTS )
mctrl | = TIOCM_CTS ;
2014-10-10 18:53:25 +02:00
return mctrl_gpio_get ( s - > gpios , & mctrl ) ;
2011-01-11 15:54:54 +01:00
}
2014-10-10 18:53:26 +02:00
/*
* Enable modem status interrupts
*/
static void mxs_auart_enable_ms ( struct uart_port * port )
{
struct mxs_auart_port * s = to_auart_port ( port ) ;
/*
* Interrupt should not be enabled twice
*/
if ( s - > ms_irq_enabled )
return ;
s - > ms_irq_enabled = true ;
if ( s - > gpio_irq [ UART_GPIO_CTS ] > = 0 )
enable_irq ( s - > gpio_irq [ UART_GPIO_CTS ] ) ;
/* TODO: enable AUART_INTR_CTSMIEN otherwise */
if ( s - > gpio_irq [ UART_GPIO_DSR ] > = 0 )
enable_irq ( s - > gpio_irq [ UART_GPIO_DSR ] ) ;
if ( s - > gpio_irq [ UART_GPIO_RI ] > = 0 )
enable_irq ( s - > gpio_irq [ UART_GPIO_RI ] ) ;
if ( s - > gpio_irq [ UART_GPIO_DCD ] > = 0 )
enable_irq ( s - > gpio_irq [ UART_GPIO_DCD ] ) ;
}
/*
* Disable modem status interrupts
*/
static void mxs_auart_disable_ms ( struct uart_port * port )
{
struct mxs_auart_port * s = to_auart_port ( port ) ;
/*
* Interrupt should not be disabled twice
*/
if ( ! s - > ms_irq_enabled )
return ;
s - > ms_irq_enabled = false ;
if ( s - > gpio_irq [ UART_GPIO_CTS ] > = 0 )
disable_irq ( s - > gpio_irq [ UART_GPIO_CTS ] ) ;
/* TODO: disable AUART_INTR_CTSMIEN otherwise */
if ( s - > gpio_irq [ UART_GPIO_DSR ] > = 0 )
disable_irq ( s - > gpio_irq [ UART_GPIO_DSR ] ) ;
if ( s - > gpio_irq [ UART_GPIO_RI ] > = 0 )
disable_irq ( s - > gpio_irq [ UART_GPIO_RI ] ) ;
if ( s - > gpio_irq [ UART_GPIO_DCD ] > = 0 )
disable_irq ( s - > gpio_irq [ UART_GPIO_DCD ] ) ;
}
2012-11-16 16:03:53 +08:00
static int mxs_auart_dma_prep_rx ( struct mxs_auart_port * s ) ;
static void dma_rx_callback ( void * arg )
{
struct mxs_auart_port * s = ( struct mxs_auart_port * ) arg ;
2013-01-03 15:53:04 +01:00
struct tty_port * port = & s - > port . state - > port ;
2012-11-16 16:03:53 +08:00
int count ;
u32 stat ;
2012-11-22 15:06:30 +08:00
dma_unmap_sg ( s - > dev , & s - > rx_sgl , 1 , DMA_FROM_DEVICE ) ;
2012-11-16 16:03:53 +08:00
stat = readl ( s - > port . membase + AUART_STAT ) ;
stat & = ~ ( AUART_STAT_OERR | AUART_STAT_BERR |
AUART_STAT_PERR | AUART_STAT_FERR ) ;
count = stat & AUART_STAT_RXCOUNT_MASK ;
2013-01-03 15:53:04 +01:00
tty_insert_flip_string ( port , s - > rx_dma_buf , count ) ;
2012-11-16 16:03:53 +08:00
writel ( stat , s - > port . membase + AUART_STAT ) ;
2013-01-03 15:53:06 +01:00
tty_flip_buffer_push ( port ) ;
2012-11-16 16:03:53 +08:00
/* start the next DMA for RX. */
mxs_auart_dma_prep_rx ( s ) ;
}
static int mxs_auart_dma_prep_rx ( struct mxs_auart_port * s )
{
struct dma_async_tx_descriptor * desc ;
struct scatterlist * sgl = & s - > rx_sgl ;
struct dma_chan * channel = s - > rx_dma_chan ;
u32 pio [ 1 ] ;
/* [1] : send PIO */
pio [ 0 ] = AUART_CTRL0_RXTO_ENABLE
| AUART_CTRL0_RXTIMEOUT ( 0x80 )
| AUART_CTRL0_XFER_COUNT ( UART_XMIT_SIZE ) ;
desc = dmaengine_prep_slave_sg ( channel , ( struct scatterlist * ) pio ,
1 , DMA_TRANS_NONE , 0 ) ;
if ( ! desc ) {
dev_err ( s - > dev , " step 1 error \n " ) ;
return - EINVAL ;
}
/* [2] : send DMA request */
sg_init_one ( sgl , s - > rx_dma_buf , UART_XMIT_SIZE ) ;
dma_map_sg ( s - > dev , sgl , 1 , DMA_FROM_DEVICE ) ;
desc = dmaengine_prep_slave_sg ( channel , sgl , 1 , DMA_DEV_TO_MEM ,
DMA_PREP_INTERRUPT | DMA_CTRL_ACK ) ;
if ( ! desc ) {
dev_err ( s - > dev , " step 2 error \n " ) ;
return - 1 ;
}
/* [3] : submit the DMA, but do not issue it. */
desc - > callback = dma_rx_callback ;
desc - > callback_param = s ;
dmaengine_submit ( desc ) ;
dma_async_issue_pending ( channel ) ;
return 0 ;
}
static void mxs_auart_dma_exit_channel ( struct mxs_auart_port * s )
{
if ( s - > tx_dma_chan ) {
dma_release_channel ( s - > tx_dma_chan ) ;
s - > tx_dma_chan = NULL ;
}
if ( s - > rx_dma_chan ) {
dma_release_channel ( s - > rx_dma_chan ) ;
s - > rx_dma_chan = NULL ;
}
kfree ( s - > tx_dma_buf ) ;
kfree ( s - > rx_dma_buf ) ;
s - > tx_dma_buf = NULL ;
s - > rx_dma_buf = NULL ;
}
static void mxs_auart_dma_exit ( struct mxs_auart_port * s )
{
writel ( AUART_CTRL2_TXDMAE | AUART_CTRL2_RXDMAE | AUART_CTRL2_DMAONERR ,
s - > port . membase + AUART_CTRL2_CLR ) ;
mxs_auart_dma_exit_channel ( s ) ;
s - > flags & = ~ MXS_AUART_DMA_ENABLED ;
clear_bit ( MXS_AUART_DMA_TX_SYNC , & s - > flags ) ;
clear_bit ( MXS_AUART_DMA_RX_READY , & s - > flags ) ;
}
static int mxs_auart_dma_init ( struct mxs_auart_port * s )
{
if ( auart_dma_enabled ( s ) )
return 0 ;
/* init for RX */
2013-02-26 13:47:41 +08:00
s - > rx_dma_chan = dma_request_slave_channel ( s - > dev , " rx " ) ;
2012-11-16 16:03:53 +08:00
if ( ! s - > rx_dma_chan )
goto err_out ;
s - > rx_dma_buf = kzalloc ( UART_XMIT_SIZE , GFP_KERNEL | GFP_DMA ) ;
if ( ! s - > rx_dma_buf )
goto err_out ;
/* init for TX */
2013-02-26 13:47:41 +08:00
s - > tx_dma_chan = dma_request_slave_channel ( s - > dev , " tx " ) ;
2012-11-16 16:03:53 +08:00
if ( ! s - > tx_dma_chan )
goto err_out ;
s - > tx_dma_buf = kzalloc ( UART_XMIT_SIZE , GFP_KERNEL | GFP_DMA ) ;
if ( ! s - > tx_dma_buf )
goto err_out ;
/* set the flags */
s - > flags | = MXS_AUART_DMA_ENABLED ;
dev_dbg ( s - > dev , " enabled the DMA support. " ) ;
2013-10-03 09:32:03 +02:00
/* The DMA buffer is now the FIFO the TTY subsystem can use */
s - > port . fifosize = UART_XMIT_SIZE ;
2012-11-16 16:03:53 +08:00
return 0 ;
err_out :
mxs_auart_dma_exit_channel ( s ) ;
return - EINVAL ;
}
2014-10-10 18:53:25 +02:00
# define RTS_AT_AUART() IS_ERR_OR_NULL(mctrl_gpio_to_gpiod(s->gpios, \
UART_GPIO_RTS ) )
# define CTS_AT_AUART() IS_ERR_OR_NULL(mctrl_gpio_to_gpiod(s->gpios, \
UART_GPIO_CTS ) )
2011-01-11 15:54:54 +01:00
static void mxs_auart_settermios ( struct uart_port * u ,
struct ktermios * termios ,
struct ktermios * old )
{
2012-11-16 16:03:53 +08:00
struct mxs_auart_port * s = to_auart_port ( u ) ;
2011-01-11 15:54:54 +01:00
u32 bm , ctrl , ctrl2 , div ;
2015-08-11 11:46:01 +00:00
unsigned int cflag , baud , baud_min , baud_max ;
2011-01-11 15:54:54 +01:00
cflag = termios - > c_cflag ;
ctrl = AUART_LINECTRL_FEN ;
ctrl2 = readl ( u - > membase + AUART_CTRL2 ) ;
/* byte size */
switch ( cflag & CSIZE ) {
case CS5 :
bm = 0 ;
break ;
case CS6 :
bm = 1 ;
break ;
case CS7 :
bm = 2 ;
break ;
case CS8 :
bm = 3 ;
break ;
default :
return ;
}
ctrl | = AUART_LINECTRL_WLEN ( bm ) ;
/* parity */
if ( cflag & PARENB ) {
ctrl | = AUART_LINECTRL_PEN ;
if ( ( cflag & PARODD ) = = 0 )
ctrl | = AUART_LINECTRL_EPS ;
}
u - > read_status_mask = 0 ;
if ( termios - > c_iflag & INPCK )
u - > read_status_mask | = AUART_STAT_PERR ;
2014-06-16 08:10:41 -04:00
if ( termios - > c_iflag & ( IGNBRK | BRKINT | PARMRK ) )
2011-01-11 15:54:54 +01:00
u - > read_status_mask | = AUART_STAT_BERR ;
/*
* Characters to ignore
*/
u - > ignore_status_mask = 0 ;
if ( termios - > c_iflag & IGNPAR )
u - > ignore_status_mask | = AUART_STAT_PERR ;
if ( termios - > c_iflag & IGNBRK ) {
u - > ignore_status_mask | = AUART_STAT_BERR ;
/*
* If we ' re ignoring parity and break indicators ,
* ignore overruns too ( for real raw support ) .
*/
if ( termios - > c_iflag & IGNPAR )
u - > ignore_status_mask | = AUART_STAT_OERR ;
}
/*
* ignore all characters if CREAD is not set
*/
if ( cflag & CREAD )
ctrl2 | = AUART_CTRL2_RXE ;
else
ctrl2 & = ~ AUART_CTRL2_RXE ;
/* figure out the stop bits requested */
if ( cflag & CSTOPB )
ctrl | = AUART_LINECTRL_STP2 ;
/* figure out the hardware flow control settings */
2014-10-10 18:53:25 +02:00
ctrl2 & = ~ ( AUART_CTRL2_CTSEN | AUART_CTRL2_RTSEN ) ;
2012-11-16 16:03:53 +08:00
if ( cflag & CRTSCTS ) {
/*
* The DMA has a bug ( see errata : 2836 ) in mx23 .
* So we can not implement the DMA for auart in mx23 ,
* we can only implement the DMA support for auart
* in mx28 .
*/
2013-08-03 10:09:15 -04:00
if ( is_imx28_auart ( s )
2013-08-03 10:09:14 -04:00
& & test_bit ( MXS_AUART_RTSCTS , & s - > flags ) ) {
2012-11-16 16:03:53 +08:00
if ( ! mxs_auart_dma_init ( s ) )
/* enable DMA tranfer */
ctrl2 | = AUART_CTRL2_TXDMAE | AUART_CTRL2_RXDMAE
| AUART_CTRL2_DMAONERR ;
}
2014-10-10 18:53:25 +02:00
/* Even if RTS is GPIO line RTSEN can be enabled because
* the pinctrl configuration decides about RTS pin function */
ctrl2 | = AUART_CTRL2_RTSEN ;
if ( CTS_AT_AUART ( ) )
ctrl2 | = AUART_CTRL2_CTSEN ;
2012-11-16 16:03:53 +08:00
}
2011-01-11 15:54:54 +01:00
/* set baud rate */
2015-08-11 11:46:01 +00:00
baud_min = DIV_ROUND_UP ( u - > uartclk * 32 , AUART_LINECTRL_BAUD_DIV_MAX ) ;
baud_max = u - > uartclk * 32 / AUART_LINECTRL_BAUD_DIV_MIN ;
baud = uart_get_baud_rate ( u , termios , old , baud_min , baud_max ) ;
2011-01-11 15:54:54 +01:00
div = u - > uartclk * 32 / baud ;
ctrl | = AUART_LINECTRL_BAUD_DIVFRAC ( div & 0x3F ) ;
ctrl | = AUART_LINECTRL_BAUD_DIVINT ( div > > 6 ) ;
writel ( ctrl , u - > membase + AUART_LINECTRL ) ;
writel ( ctrl2 , u - > membase + AUART_CTRL2 ) ;
2012-05-03 11:37:12 +02:00
uart_update_timeout ( u , termios - > c_cflag , baud ) ;
2012-11-16 16:03:53 +08:00
/* prepare for the DMA RX. */
if ( auart_dma_enabled ( s ) & &
! test_and_set_bit ( MXS_AUART_DMA_RX_READY , & s - > flags ) ) {
if ( ! mxs_auart_dma_prep_rx ( s ) ) {
/* Disable the normal RX interrupt. */
2012-11-22 15:06:29 +08:00
writel ( AUART_INTR_RXIEN | AUART_INTR_RTIEN ,
u - > membase + AUART_INTR_CLR ) ;
2012-11-16 16:03:53 +08:00
} else {
mxs_auart_dma_exit ( s ) ;
dev_err ( s - > dev , " We can not start up the DMA. \n " ) ;
}
}
2014-10-10 18:53:26 +02:00
/* CTS flow-control and modem-status interrupts */
if ( UART_ENABLE_MS ( u , termios - > c_cflag ) )
mxs_auart_enable_ms ( u ) ;
else
mxs_auart_disable_ms ( u ) ;
2011-01-11 15:54:54 +01:00
}
2014-11-12 20:32:49 -02:00
static void mxs_auart_set_ldisc ( struct uart_port * port ,
struct ktermios * termios )
2014-10-10 18:53:27 +02:00
{
2014-11-12 20:32:49 -02:00
if ( termios - > c_line = = N_PPS ) {
2014-10-10 18:53:27 +02:00
port - > flags | = UPF_HARDPPS_CD ;
mxs_auart_enable_ms ( port ) ;
} else {
port - > flags & = ~ UPF_HARDPPS_CD ;
}
}
2011-01-11 15:54:54 +01:00
static irqreturn_t mxs_auart_irq_handle ( int irq , void * context )
{
2013-07-04 11:28:51 +02:00
u32 istat ;
2011-01-11 15:54:54 +01:00
struct mxs_auart_port * s = context ;
2014-11-14 23:24:33 +01:00
u32 mctrl_temp = s - > mctrl_prev ;
2011-01-11 15:54:54 +01:00
u32 stat = readl ( s - > port . membase + AUART_STAT ) ;
2013-07-04 11:28:51 +02:00
istat = readl ( s - > port . membase + AUART_INTR ) ;
/* ack irq */
writel ( istat & ( AUART_INTR_RTIS
| AUART_INTR_TXIS
| AUART_INTR_RXIS
| AUART_INTR_CTSMIS ) ,
s - > port . membase + AUART_INTR_CLR ) ;
2011-01-11 15:54:54 +01:00
2014-10-10 18:53:26 +02:00
/*
* Dealing with GPIO interrupt
*/
if ( irq = = s - > gpio_irq [ UART_GPIO_CTS ] | |
irq = = s - > gpio_irq [ UART_GPIO_DCD ] | |
irq = = s - > gpio_irq [ UART_GPIO_DSR ] | |
irq = = s - > gpio_irq [ UART_GPIO_RI ] )
mxs_auart_modem_status ( s ,
2014-11-14 23:24:33 +01:00
mctrl_gpio_get ( s - > gpios , & mctrl_temp ) ) ;
2014-10-10 18:53:26 +02:00
2011-01-11 15:54:54 +01:00
if ( istat & AUART_INTR_CTSMIS ) {
2014-10-10 18:53:26 +02:00
if ( CTS_AT_AUART ( ) & & s - > ms_irq_enabled )
2014-10-10 18:53:25 +02:00
uart_handle_cts_change ( & s - > port ,
stat & AUART_STAT_CTS ) ;
2011-01-11 15:54:54 +01:00
writel ( AUART_INTR_CTSMIS ,
s - > port . membase + AUART_INTR_CLR ) ;
istat & = ~ AUART_INTR_CTSMIS ;
}
if ( istat & ( AUART_INTR_RTIS | AUART_INTR_RXIS ) ) {
2012-11-22 15:06:29 +08:00
if ( ! auart_dma_enabled ( s ) )
mxs_auart_rx_chars ( s ) ;
2011-01-11 15:54:54 +01:00
istat & = ~ ( AUART_INTR_RTIS | AUART_INTR_RXIS ) ;
}
if ( istat & AUART_INTR_TXIS ) {
mxs_auart_tx_chars ( s ) ;
istat & = ~ AUART_INTR_TXIS ;
}
return IRQ_HANDLED ;
}
2015-08-07 12:47:03 +02:00
static void mxs_auart_reset_deassert ( struct uart_port * u )
2011-01-11 15:54:54 +01:00
{
int i ;
unsigned int reg ;
writel ( AUART_CTRL0_SFTRST , u - > membase + AUART_CTRL0_CLR ) ;
for ( i = 0 ; i < 10000 ; i + + ) {
reg = readl ( u - > membase + AUART_CTRL0 ) ;
if ( ! ( reg & AUART_CTRL0_SFTRST ) )
break ;
udelay ( 3 ) ;
}
writel ( AUART_CTRL0_CLKGATE , u - > membase + AUART_CTRL0_CLR ) ;
}
2015-08-07 12:47:04 +02:00
static void mxs_auart_reset_assert ( struct uart_port * u )
{
int i ;
u32 reg ;
reg = readl ( u - > membase + AUART_CTRL0 ) ;
/* if already in reset state, keep it untouched */
if ( reg & AUART_CTRL0_SFTRST )
return ;
writel ( AUART_CTRL0_CLKGATE , u - > membase + AUART_CTRL0_CLR ) ;
writel ( AUART_CTRL0_SFTRST , u - > membase + AUART_CTRL0_SET ) ;
for ( i = 0 ; i < 1000 ; i + + ) {
reg = readl ( u - > membase + AUART_CTRL0 ) ;
/* reset is finished when the clock is gated */
if ( reg & AUART_CTRL0_CLKGATE )
return ;
udelay ( 10 ) ;
}
dev_err ( u - > dev , " Failed to reset the unit. " ) ;
}
2011-01-11 15:54:54 +01:00
static int mxs_auart_startup ( struct uart_port * u )
{
2013-12-02 01:17:58 -02:00
int ret ;
2011-01-11 15:54:54 +01:00
struct mxs_auart_port * s = to_auart_port ( u ) ;
2013-12-02 01:17:58 -02:00
ret = clk_prepare_enable ( s - > clk ) ;
if ( ret )
return ret ;
2011-01-11 15:54:54 +01:00
2015-08-07 12:47:04 +02:00
if ( uart_console ( u ) ) {
writel ( AUART_CTRL0_CLKGATE , u - > membase + AUART_CTRL0_CLR ) ;
} else {
/* reset the unit to a well known state */
mxs_auart_reset_assert ( u ) ;
mxs_auart_reset_deassert ( u ) ;
}
2011-01-11 15:54:54 +01:00
writel ( AUART_CTRL2_UARTEN , u - > membase + AUART_CTRL2_SET ) ;
writel ( AUART_INTR_RXIEN | AUART_INTR_RTIEN | AUART_INTR_CTSMIEN ,
u - > membase + AUART_INTR ) ;
2013-10-03 09:32:03 +02:00
/* Reset FIFO size (it could have changed if DMA was enabled) */
u - > fifosize = MXS_AUART_FIFO_SIZE ;
2011-01-11 15:54:54 +01:00
/*
* Enable fifo so all four bytes of a DMA word are written to
* output ( otherwise , only the LSB is written , ie . 1 in 4 bytes )
*/
writel ( AUART_LINECTRL_FEN , u - > membase + AUART_LINECTRL_SET ) ;
2014-10-10 18:53:26 +02:00
/* get initial status of modem lines */
mctrl_gpio_get ( s - > gpios , & s - > mctrl_prev ) ;
s - > ms_irq_enabled = false ;
2011-01-11 15:54:54 +01:00
return 0 ;
}
static void mxs_auart_shutdown ( struct uart_port * u )
{
struct mxs_auart_port * s = to_auart_port ( u ) ;
2014-10-10 18:53:26 +02:00
mxs_auart_disable_ms ( u ) ;
2012-11-16 16:03:53 +08:00
if ( auart_dma_enabled ( s ) )
mxs_auart_dma_exit ( s ) ;
2015-08-07 12:47:04 +02:00
if ( uart_console ( u ) ) {
writel ( AUART_CTRL2_UARTEN , u - > membase + AUART_CTRL2_CLR ) ;
writel ( AUART_INTR_RXIEN | AUART_INTR_RTIEN | AUART_INTR_CTSMIEN ,
u - > membase + AUART_INTR_CLR ) ;
writel ( AUART_CTRL0_CLKGATE , u - > membase + AUART_CTRL0_SET ) ;
} else {
mxs_auart_reset_assert ( u ) ;
}
2012-09-06 22:38:40 -04:00
2011-12-20 14:10:29 +08:00
clk_disable_unprepare ( s - > clk ) ;
2011-01-11 15:54:54 +01:00
}
static unsigned int mxs_auart_tx_empty ( struct uart_port * u )
{
2014-11-18 18:37:13 +01:00
if ( ( readl ( u - > membase + AUART_STAT ) &
( AUART_STAT_TXFE | AUART_STAT_BUSY ) ) = = AUART_STAT_TXFE )
2011-01-11 15:54:54 +01:00
return TIOCSER_TEMT ;
2014-11-18 18:37:13 +01:00
return 0 ;
2011-01-11 15:54:54 +01:00
}
static void mxs_auart_start_tx ( struct uart_port * u )
{
struct mxs_auart_port * s = to_auart_port ( u ) ;
/* enable transmitter */
writel ( AUART_CTRL2_TXE , u - > membase + AUART_CTRL2_SET ) ;
mxs_auart_tx_chars ( s ) ;
}
static void mxs_auart_stop_tx ( struct uart_port * u )
{
writel ( AUART_CTRL2_TXE , u - > membase + AUART_CTRL2_CLR ) ;
}
static void mxs_auart_stop_rx ( struct uart_port * u )
{
writel ( AUART_CTRL2_RXE , u - > membase + AUART_CTRL2_CLR ) ;
}
static void mxs_auart_break_ctl ( struct uart_port * u , int ctl )
{
if ( ctl )
writel ( AUART_LINECTRL_BRK ,
u - > membase + AUART_LINECTRL_SET ) ;
else
writel ( AUART_LINECTRL_BRK ,
u - > membase + AUART_LINECTRL_CLR ) ;
}
static struct uart_ops mxs_auart_ops = {
. tx_empty = mxs_auart_tx_empty ,
. start_tx = mxs_auart_start_tx ,
. stop_tx = mxs_auart_stop_tx ,
. stop_rx = mxs_auart_stop_rx ,
2014-10-10 18:53:26 +02:00
. enable_ms = mxs_auart_enable_ms ,
2011-01-11 15:54:54 +01:00
. break_ctl = mxs_auart_break_ctl ,
. set_mctrl = mxs_auart_set_mctrl ,
. get_mctrl = mxs_auart_get_mctrl ,
. startup = mxs_auart_startup ,
. shutdown = mxs_auart_shutdown ,
. set_termios = mxs_auart_settermios ,
2014-10-10 18:53:27 +02:00
. set_ldisc = mxs_auart_set_ldisc ,
2011-01-11 15:54:54 +01:00
. type = mxs_auart_type ,
. release_port = mxs_auart_release_port ,
. request_port = mxs_auart_request_port ,
. config_port = mxs_auart_config_port ,
. verify_port = mxs_auart_verify_port ,
} ;
static struct mxs_auart_port * auart_port [ MXS_AUART_PORTS ] ;
# ifdef CONFIG_SERIAL_MXS_AUART_CONSOLE
static void mxs_auart_console_putchar ( struct uart_port * port , int ch )
{
unsigned int to = 1000 ;
while ( readl ( port - > membase + AUART_STAT ) & AUART_STAT_TXFF ) {
if ( ! to - - )
break ;
udelay ( 1 ) ;
}
writel ( ch , port - > membase + AUART_DATA ) ;
}
static void
auart_console_write ( struct console * co , const char * str , unsigned int count )
{
struct mxs_auart_port * s ;
struct uart_port * port ;
unsigned int old_ctrl0 , old_ctrl2 ;
2013-06-28 11:49:41 +02:00
unsigned int to = 20000 ;
2011-01-11 15:54:54 +01:00
2013-04-19 21:12:17 +02:00
if ( co - > index > = MXS_AUART_PORTS | | co - > index < 0 )
2011-01-11 15:54:54 +01:00
return ;
s = auart_port [ co - > index ] ;
port = & s - > port ;
clk_enable ( s - > clk ) ;
/* First save the CR then disable the interrupts */
old_ctrl2 = readl ( port - > membase + AUART_CTRL2 ) ;
old_ctrl0 = readl ( port - > membase + AUART_CTRL0 ) ;
writel ( AUART_CTRL0_CLKGATE ,
port - > membase + AUART_CTRL0_CLR ) ;
writel ( AUART_CTRL2_UARTEN | AUART_CTRL2_TXE ,
port - > membase + AUART_CTRL2_SET ) ;
uart_console_write ( port , str , count , mxs_auart_console_putchar ) ;
2013-06-28 11:49:41 +02:00
/* Finally, wait for transmitter to become empty ... */
2011-01-11 15:54:54 +01:00
while ( readl ( port - > membase + AUART_STAT ) & AUART_STAT_BUSY ) {
2013-06-28 11:49:41 +02:00
udelay ( 1 ) ;
2011-01-11 15:54:54 +01:00
if ( ! to - - )
break ;
}
2013-06-28 11:49:41 +02:00
/*
* . . . and restore the TCR if we waited long enough for the transmitter
* to be idle . This might keep the transmitter enabled although it is
* unused , but that is better than to disable it while it is still
* transmitting .
*/
if ( ! ( readl ( port - > membase + AUART_STAT ) & AUART_STAT_BUSY ) ) {
writel ( old_ctrl0 , port - > membase + AUART_CTRL0 ) ;
writel ( old_ctrl2 , port - > membase + AUART_CTRL2 ) ;
}
2011-01-11 15:54:54 +01:00
clk_disable ( s - > clk ) ;
}
static void __init
auart_console_get_options ( struct uart_port * port , int * baud ,
int * parity , int * bits )
{
unsigned int lcr_h , quot ;
if ( ! ( readl ( port - > membase + AUART_CTRL2 ) & AUART_CTRL2_UARTEN ) )
return ;
lcr_h = readl ( port - > membase + AUART_LINECTRL ) ;
* parity = ' n ' ;
if ( lcr_h & AUART_LINECTRL_PEN ) {
if ( lcr_h & AUART_LINECTRL_EPS )
* parity = ' e ' ;
else
* parity = ' o ' ;
}
if ( ( lcr_h & AUART_LINECTRL_WLEN_MASK ) = = AUART_LINECTRL_WLEN ( 2 ) )
* bits = 7 ;
else
* bits = 8 ;
quot = ( ( readl ( port - > membase + AUART_LINECTRL )
& AUART_LINECTRL_BAUD_DIVINT_MASK ) )
> > ( AUART_LINECTRL_BAUD_DIVINT_SHIFT - 6 ) ;
quot | = ( ( readl ( port - > membase + AUART_LINECTRL )
& AUART_LINECTRL_BAUD_DIVFRAC_MASK ) )
> > AUART_LINECTRL_BAUD_DIVFRAC_SHIFT ;
if ( quot = = 0 )
quot = 1 ;
* baud = ( port - > uartclk < < 2 ) / quot ;
}
static int __init
auart_console_setup ( struct console * co , char * options )
{
struct mxs_auart_port * s ;
int baud = 9600 ;
int bits = 8 ;
int parity = ' n ' ;
int flow = ' n ' ;
int ret ;
/*
* Check whether an invalid uart number has been specified , and
* if so , search for the first available port that does have
* console support .
*/
if ( co - > index = = - 1 | | co - > index > = ARRAY_SIZE ( auart_port ) )
co - > index = 0 ;
s = auart_port [ co - > index ] ;
if ( ! s )
return - ENODEV ;
2013-12-02 01:17:58 -02:00
ret = clk_prepare_enable ( s - > clk ) ;
if ( ret )
return ret ;
2011-01-11 15:54:54 +01:00
if ( options )
uart_parse_options ( options , & baud , & parity , & bits , & flow ) ;
else
auart_console_get_options ( & s - > port , & baud , & parity , & bits ) ;
ret = uart_set_options ( & s - > port , co , baud , parity , bits , flow ) ;
2011-12-20 14:10:29 +08:00
clk_disable_unprepare ( s - > clk ) ;
2011-01-11 15:54:54 +01:00
return ret ;
}
static struct console auart_console = {
. name = " ttyAPP " ,
. write = auart_console_write ,
. device = uart_console_device ,
. setup = auart_console_setup ,
. flags = CON_PRINTBUFFER ,
. index = - 1 ,
. data = & auart_driver ,
} ;
# endif
static struct uart_driver auart_driver = {
. owner = THIS_MODULE ,
. driver_name = " ttyAPP " ,
. dev_name = " ttyAPP " ,
. major = 0 ,
. minor = 0 ,
. nr = MXS_AUART_PORTS ,
# ifdef CONFIG_SERIAL_MXS_AUART_CONSOLE
. cons = & auart_console ,
# endif
} ;
2012-06-18 10:06:09 -03:00
/*
* This function returns 1 if pdev isn ' t a device instatiated by dt , 0 if it
* could successfully get all information from dt or a negative errno .
*/
static int serial_mxs_probe_dt ( struct mxs_auart_port * s ,
struct platform_device * pdev )
{
struct device_node * np = pdev - > dev . of_node ;
int ret ;
if ( ! np )
/* no device tree device */
return 1 ;
ret = of_alias_get_id ( np , " serial " ) ;
if ( ret < 0 ) {
dev_err ( & pdev - > dev , " failed to get alias id: %d \n " , ret ) ;
return ret ;
}
s - > port . line = ret ;
2013-08-03 10:09:14 -04:00
if ( of_get_property ( np , " fsl,uart-has-rtscts " , NULL ) )
set_bit ( MXS_AUART_RTSCTS , & s - > flags ) ;
2012-06-18 10:06:09 -03:00
return 0 ;
}
2015-02-12 15:24:40 +01:00
static int mxs_auart_init_gpios ( struct mxs_auart_port * s , struct device * dev )
2014-10-10 18:53:25 +02:00
{
2014-10-10 18:53:26 +02:00
enum mctrl_gpio_idx i ;
struct gpio_desc * gpiod ;
2015-09-30 10:19:40 +02:00
s - > gpios = mctrl_gpio_init_noauto ( dev , 0 ) ;
2015-02-12 15:24:40 +01:00
if ( IS_ERR ( s - > gpios ) )
return PTR_ERR ( s - > gpios ) ;
2014-10-10 18:53:25 +02:00
/* Block (enabled before) DMA option if RTS or CTS is GPIO line */
if ( ! RTS_AT_AUART ( ) | | ! CTS_AT_AUART ( ) ) {
if ( test_bit ( MXS_AUART_RTSCTS , & s - > flags ) )
dev_warn ( dev ,
" DMA and flow control via gpio may cause some problems. DMA disabled! \n " ) ;
clear_bit ( MXS_AUART_RTSCTS , & s - > flags ) ;
}
2014-10-10 18:53:26 +02:00
for ( i = 0 ; i < UART_GPIO_MAX ; i + + ) {
gpiod = mctrl_gpio_to_gpiod ( s - > gpios , i ) ;
if ( gpiod & & ( gpiod_get_direction ( gpiod ) = = GPIOF_DIR_IN ) )
s - > gpio_irq [ i ] = gpiod_to_irq ( gpiod ) ;
else
s - > gpio_irq [ i ] = - EINVAL ;
}
2015-02-12 15:24:40 +01:00
return 0 ;
2014-10-10 18:53:25 +02:00
}
2014-10-10 18:53:26 +02:00
static void mxs_auart_free_gpio_irq ( struct mxs_auart_port * s )
{
enum mctrl_gpio_idx i ;
for ( i = 0 ; i < UART_GPIO_MAX ; i + + )
if ( s - > gpio_irq [ i ] > = 0 )
free_irq ( s - > gpio_irq [ i ] , s ) ;
}
static int mxs_auart_request_gpio_irq ( struct mxs_auart_port * s )
{
int * irq = s - > gpio_irq ;
enum mctrl_gpio_idx i ;
int err = 0 ;
for ( i = 0 ; ( i < UART_GPIO_MAX ) & & ! err ; i + + ) {
if ( irq [ i ] < 0 )
continue ;
irq_set_status_flags ( irq [ i ] , IRQ_NOAUTOEN ) ;
err = request_irq ( irq [ i ] , mxs_auart_irq_handle ,
IRQ_TYPE_EDGE_BOTH , dev_name ( s - > dev ) , s ) ;
if ( err )
dev_err ( s - > dev , " %s - Can't get %d irq \n " ,
__func__ , irq [ i ] ) ;
}
/*
* If something went wrong , rollback .
*/
while ( err & & ( - - i > = 0 ) )
if ( irq [ i ] > = 0 )
free_irq ( irq [ i ] , s ) ;
return err ;
}
2012-11-19 13:21:50 -05:00
static int mxs_auart_probe ( struct platform_device * pdev )
2011-01-11 15:54:54 +01:00
{
2012-11-16 16:03:52 +08:00
const struct of_device_id * of_id =
of_match_device ( mxs_auart_dt_ids , & pdev - > dev ) ;
2011-01-11 15:54:54 +01:00
struct mxs_auart_port * s ;
u32 version ;
2015-01-14 14:39:09 -02:00
int ret , irq ;
2011-01-11 15:54:54 +01:00
struct resource * r ;
2014-11-27 17:08:31 -02:00
s = devm_kzalloc ( & pdev - > dev , sizeof ( * s ) , GFP_KERNEL ) ;
2014-11-27 17:08:30 -02:00
if ( ! s )
return - ENOMEM ;
2011-01-11 15:54:54 +01:00
2012-06-18 10:06:09 -03:00
ret = serial_mxs_probe_dt ( s , pdev ) ;
if ( ret > 0 )
s - > port . line = pdev - > id < 0 ? 0 : pdev - > id ;
else if ( ret < 0 )
2014-11-27 17:08:31 -02:00
return ret ;
2012-06-18 10:06:09 -03:00
2012-11-16 16:03:52 +08:00
if ( of_id ) {
pdev - > id_entry = of_id - > data ;
s - > devtype = pdev - > id_entry - > driver_data ;
}
2014-11-27 17:08:32 -02:00
s - > clk = devm_clk_get ( & pdev - > dev , NULL ) ;
2014-11-27 17:08:31 -02:00
if ( IS_ERR ( s - > clk ) )
return PTR_ERR ( s - > clk ) ;
2011-01-11 15:54:54 +01:00
r = platform_get_resource ( pdev , IORESOURCE_MEM , 0 ) ;
2014-11-27 17:08:32 -02:00
if ( ! r )
return - ENXIO ;
2011-01-11 15:54:54 +01:00
s - > port . mapbase = r - > start ;
s - > port . membase = ioremap ( r - > start , resource_size ( r ) ) ;
s - > port . ops = & mxs_auart_ops ;
s - > port . iotype = UPIO_MEM ;
2013-10-03 09:32:03 +02:00
s - > port . fifosize = MXS_AUART_FIFO_SIZE ;
2011-01-11 15:54:54 +01:00
s - > port . uartclk = clk_get_rate ( s - > clk ) ;
s - > port . type = PORT_IMX ;
2013-04-19 21:06:20 +02:00
s - > port . dev = s - > dev = & pdev - > dev ;
2011-01-11 15:54:54 +01:00
2014-10-10 18:53:26 +02:00
s - > mctrl_prev = 0 ;
2015-01-14 14:39:07 -02:00
irq = platform_get_irq ( pdev , 0 ) ;
2015-01-14 14:39:08 -02:00
if ( irq < 0 )
return irq ;
2015-01-14 14:39:07 -02:00
s - > port . irq = irq ;
ret = devm_request_irq ( & pdev - > dev , irq , mxs_auart_irq_handle , 0 ,
2014-11-27 17:08:33 -02:00
dev_name ( & pdev - > dev ) , s ) ;
2011-01-11 15:54:54 +01:00
if ( ret )
2014-11-27 17:08:32 -02:00
return ret ;
2011-01-11 15:54:54 +01:00
platform_set_drvdata ( pdev , s ) ;
2015-02-12 15:24:40 +01:00
ret = mxs_auart_init_gpios ( s , & pdev - > dev ) ;
if ( ret ) {
dev_err ( & pdev - > dev , " Failed to initialize GPIOs. \n " ) ;
2015-03-10 11:24:24 -03:00
return ret ;
2015-02-12 15:24:40 +01:00
}
2014-10-10 18:53:25 +02:00
2014-10-10 18:53:26 +02:00
/*
* Get the GPIO lines IRQ
*/
ret = mxs_auart_request_gpio_irq ( s ) ;
if ( ret )
2014-11-27 17:08:33 -02:00
return ret ;
2014-10-10 18:53:26 +02:00
2012-06-18 10:06:09 -03:00
auart_port [ s - > port . line ] = s ;
2011-01-11 15:54:54 +01:00
2015-08-07 12:47:03 +02:00
mxs_auart_reset_deassert ( & s - > port ) ;
2011-01-11 15:54:54 +01:00
ret = uart_add_one_port ( & auart_driver , & s - > port ) ;
if ( ret )
2014-10-10 18:53:26 +02:00
goto out_free_gpio_irq ;
2011-01-11 15:54:54 +01:00
version = readl ( s - > port . membase + AUART_VERSION ) ;
dev_info ( & pdev - > dev , " Found APPUART %d.%d.%d \n " ,
( version > > 24 ) & 0xff ,
( version > > 16 ) & 0xff , version & 0xffff ) ;
return 0 ;
2014-10-10 18:53:26 +02:00
out_free_gpio_irq :
mxs_auart_free_gpio_irq ( s ) ;
2011-01-11 15:54:54 +01:00
auart_port [ pdev - > id ] = NULL ;
return ret ;
}
2012-11-19 13:26:18 -05:00
static int mxs_auart_remove ( struct platform_device * pdev )
2011-01-11 15:54:54 +01:00
{
struct mxs_auart_port * s = platform_get_drvdata ( pdev ) ;
uart_remove_one_port ( & auart_driver , & s - > port ) ;
auart_port [ pdev - > id ] = NULL ;
2014-10-10 18:53:26 +02:00
mxs_auart_free_gpio_irq ( s ) ;
2011-01-11 15:54:54 +01:00
return 0 ;
}
static struct platform_driver mxs_auart_driver = {
. probe = mxs_auart_probe ,
2012-11-19 13:21:34 -05:00
. remove = mxs_auart_remove ,
2011-01-11 15:54:54 +01:00
. driver = {
. name = " mxs-auart " ,
2012-06-18 10:06:09 -03:00
. of_match_table = mxs_auart_dt_ids ,
2011-01-11 15:54:54 +01:00
} ,
} ;
static int __init mxs_auart_init ( void )
{
int r ;
r = uart_register_driver ( & auart_driver ) ;
if ( r )
goto out ;
r = platform_driver_register ( & mxs_auart_driver ) ;
if ( r )
goto out_err ;
return 0 ;
out_err :
uart_unregister_driver ( & auart_driver ) ;
out :
return r ;
}
static void __exit mxs_auart_exit ( void )
{
platform_driver_unregister ( & mxs_auart_driver ) ;
uart_unregister_driver ( & auart_driver ) ;
}
module_init ( mxs_auart_init ) ;
module_exit ( mxs_auart_exit ) ;
MODULE_LICENSE ( " GPL " ) ;
MODULE_DESCRIPTION ( " Freescale MXS application uart driver " ) ;
2012-06-18 10:06:09 -03:00
MODULE_ALIAS ( " platform:mxs-auart " ) ;