2006-12-07 07:35:17 +03:00
/*
* uartlite . c : Serial driver for Xilinx uartlite serial controller
*
2007-10-02 06:16:04 +04:00
* Copyright ( C ) 2006 Peter Korsgaard < jacmet @ sunsite . dk >
* Copyright ( C ) 2007 Secret Lab Technologies Ltd .
2006-12-07 07:35:17 +03:00
*
* This file is licensed under the terms of the GNU General Public License
* version 2. This program is licensed " as is " without any warranty of any
* kind , whether express or implied .
*/
# include <linux/platform_device.h>
# include <linux/module.h>
# include <linux/console.h>
# include <linux/serial.h>
# include <linux/serial_core.h>
# include <linux/tty.h>
# include <linux/delay.h>
# include <linux/interrupt.h>
2008-01-08 22:35:05 +03:00
# include <linux/init.h>
2006-12-07 07:35:17 +03:00
# include <asm/io.h>
2007-10-02 06:16:04 +04:00
# if defined(CONFIG_OF)
2008-01-08 22:35:05 +03:00
# include <linux/of.h>
2007-10-02 06:16:04 +04:00
# include <linux/of_device.h>
# include <linux/of_platform.h>
2008-01-08 22:35:05 +03:00
/* Match table for of_platform binding */
static struct of_device_id ulite_of_match [ ] __devinitdata = {
{ . compatible = " xlnx,opb-uartlite-1.00.b " , } ,
{ . compatible = " xlnx,xps-uartlite-1.00.a " , } ,
{ }
} ;
MODULE_DEVICE_TABLE ( of , ulite_of_match ) ;
2007-10-02 06:16:04 +04:00
# endif
2006-12-07 07:35:17 +03:00
2007-10-02 06:15:49 +04:00
# define ULITE_NAME "ttyUL"
2006-12-07 07:35:17 +03:00
# define ULITE_MAJOR 204
# define ULITE_MINOR 187
# define ULITE_NR_UARTS 4
2007-10-02 06:15:59 +04:00
/* ---------------------------------------------------------------------
* Register definitions
*
* For register details see datasheet :
* http : //www.xilinx.com/bvdocs/ipcenter/data_sheet/opb_uartlite.pdf
*/
2006-12-07 07:35:17 +03:00
# define ULITE_RX 0x00
# define ULITE_TX 0x04
# define ULITE_STATUS 0x08
# define ULITE_CONTROL 0x0c
# define ULITE_REGION 16
# define ULITE_STATUS_RXVALID 0x01
# define ULITE_STATUS_RXFULL 0x02
# define ULITE_STATUS_TXEMPTY 0x04
# define ULITE_STATUS_TXFULL 0x08
# define ULITE_STATUS_IE 0x10
# define ULITE_STATUS_OVERRUN 0x20
# define ULITE_STATUS_FRAME 0x40
# define ULITE_STATUS_PARITY 0x80
# define ULITE_CONTROL_RST_TX 0x01
# define ULITE_CONTROL_RST_RX 0x02
# define ULITE_CONTROL_IE 0x10
2007-10-02 06:15:44 +04:00
static struct uart_port ulite_ports [ ULITE_NR_UARTS ] ;
2006-12-07 07:35:17 +03:00
2007-10-02 06:15:59 +04:00
/* ---------------------------------------------------------------------
* Core UART driver operations
*/
2006-12-07 07:35:17 +03:00
static int ulite_receive ( struct uart_port * port , int stat )
{
2008-07-17 00:54:42 +04:00
struct tty_struct * tty = port - > info - > port . tty ;
2006-12-07 07:35:17 +03:00
unsigned char ch = 0 ;
char flag = TTY_NORMAL ;
if ( ( stat & ( ULITE_STATUS_RXVALID | ULITE_STATUS_OVERRUN
| ULITE_STATUS_FRAME ) ) = = 0 )
return 0 ;
/* stats */
if ( stat & ULITE_STATUS_RXVALID ) {
port - > icount . rx + + ;
2007-10-02 20:47:02 +04:00
ch = readb ( port - > membase + ULITE_RX ) ;
2006-12-07 07:35:17 +03:00
if ( stat & ULITE_STATUS_PARITY )
port - > icount . parity + + ;
}
if ( stat & ULITE_STATUS_OVERRUN )
port - > icount . overrun + + ;
if ( stat & ULITE_STATUS_FRAME )
port - > icount . frame + + ;
/* drop byte with parity error if IGNPAR specificed */
if ( stat & port - > ignore_status_mask & ULITE_STATUS_PARITY )
stat & = ~ ULITE_STATUS_RXVALID ;
stat & = port - > read_status_mask ;
if ( stat & ULITE_STATUS_PARITY )
flag = TTY_PARITY ;
stat & = ~ port - > ignore_status_mask ;
if ( stat & ULITE_STATUS_RXVALID )
tty_insert_flip_char ( tty , ch , flag ) ;
if ( stat & ULITE_STATUS_FRAME )
tty_insert_flip_char ( tty , 0 , TTY_FRAME ) ;
if ( stat & ULITE_STATUS_OVERRUN )
tty_insert_flip_char ( tty , 0 , TTY_OVERRUN ) ;
return 1 ;
}
static int ulite_transmit ( struct uart_port * port , int stat )
{
struct circ_buf * xmit = & port - > info - > xmit ;
if ( stat & ULITE_STATUS_TXFULL )
return 0 ;
if ( port - > x_char ) {
2007-10-02 20:47:02 +04:00
writeb ( port - > x_char , port - > membase + ULITE_TX ) ;
2006-12-07 07:35:17 +03:00
port - > x_char = 0 ;
port - > icount . tx + + ;
return 1 ;
}
if ( uart_circ_empty ( xmit ) | | uart_tx_stopped ( port ) )
return 0 ;
2007-10-02 20:47:02 +04:00
writeb ( xmit - > buf [ xmit - > tail ] , port - > membase + ULITE_TX ) ;
2006-12-07 07:35:17 +03:00
xmit - > tail = ( xmit - > tail + 1 ) & ( UART_XMIT_SIZE - 1 ) ;
port - > icount . tx + + ;
/* wake up */
if ( uart_circ_chars_pending ( xmit ) < WAKEUP_CHARS )
uart_write_wakeup ( port ) ;
return 1 ;
}
static irqreturn_t ulite_isr ( int irq , void * dev_id )
{
2008-02-06 12:36:20 +03:00
struct uart_port * port = dev_id ;
2006-12-07 07:35:17 +03:00
int busy ;
do {
2007-10-02 20:47:02 +04:00
int stat = readb ( port - > membase + ULITE_STATUS ) ;
2006-12-07 07:35:17 +03:00
busy = ulite_receive ( port , stat ) ;
busy | = ulite_transmit ( port , stat ) ;
} while ( busy ) ;
2008-07-17 00:54:42 +04:00
tty_flip_buffer_push ( port - > info - > port . tty ) ;
2006-12-07 07:35:17 +03:00
return IRQ_HANDLED ;
}
static unsigned int ulite_tx_empty ( struct uart_port * port )
{
unsigned long flags ;
unsigned int ret ;
spin_lock_irqsave ( & port - > lock , flags ) ;
2007-10-02 20:47:02 +04:00
ret = readb ( port - > membase + ULITE_STATUS ) ;
2006-12-07 07:35:17 +03:00
spin_unlock_irqrestore ( & port - > lock , flags ) ;
return ret & ULITE_STATUS_TXEMPTY ? TIOCSER_TEMT : 0 ;
}
static unsigned int ulite_get_mctrl ( struct uart_port * port )
{
return TIOCM_CTS | TIOCM_DSR | TIOCM_CAR ;
}
static void ulite_set_mctrl ( struct uart_port * port , unsigned int mctrl )
{
/* N/A */
}
static void ulite_stop_tx ( struct uart_port * port )
{
/* N/A */
}
static void ulite_start_tx ( struct uart_port * port )
{
2007-10-02 20:47:02 +04:00
ulite_transmit ( port , readb ( port - > membase + ULITE_STATUS ) ) ;
2006-12-07 07:35:17 +03:00
}
static void ulite_stop_rx ( struct uart_port * port )
{
/* don't forward any more data (like !CREAD) */
port - > ignore_status_mask = ULITE_STATUS_RXVALID | ULITE_STATUS_PARITY
| ULITE_STATUS_FRAME | ULITE_STATUS_OVERRUN ;
}
static void ulite_enable_ms ( struct uart_port * port )
{
/* N/A */
}
static void ulite_break_ctl ( struct uart_port * port , int ctl )
{
/* N/A */
}
static int ulite_startup ( struct uart_port * port )
{
int ret ;
ret = request_irq ( port - > irq , ulite_isr ,
IRQF_DISABLED | IRQF_SAMPLE_RANDOM , " uartlite " , port ) ;
if ( ret )
return ret ;
2007-10-02 20:47:02 +04:00
writeb ( ULITE_CONTROL_RST_RX | ULITE_CONTROL_RST_TX ,
port - > membase + ULITE_CONTROL ) ;
writeb ( ULITE_CONTROL_IE , port - > membase + ULITE_CONTROL ) ;
2006-12-07 07:35:17 +03:00
return 0 ;
}
static void ulite_shutdown ( struct uart_port * port )
{
2007-10-02 20:47:02 +04:00
writeb ( 0 , port - > membase + ULITE_CONTROL ) ;
readb ( port - > membase + ULITE_CONTROL ) ; /* dummy */
2006-12-07 07:35:17 +03:00
free_irq ( port - > irq , port ) ;
}
2006-12-08 13:38:45 +03:00
static void ulite_set_termios ( struct uart_port * port , struct ktermios * termios ,
struct ktermios * old )
2006-12-07 07:35:17 +03:00
{
unsigned long flags ;
unsigned int baud ;
spin_lock_irqsave ( & port - > lock , flags ) ;
port - > read_status_mask = ULITE_STATUS_RXVALID | ULITE_STATUS_OVERRUN
| ULITE_STATUS_TXFULL ;
if ( termios - > c_iflag & INPCK )
port - > read_status_mask | =
ULITE_STATUS_PARITY | ULITE_STATUS_FRAME ;
port - > ignore_status_mask = 0 ;
if ( termios - > c_iflag & IGNPAR )
port - > ignore_status_mask | = ULITE_STATUS_PARITY
| ULITE_STATUS_FRAME | ULITE_STATUS_OVERRUN ;
/* ignore all characters if CREAD is not set */
if ( ( termios - > c_cflag & CREAD ) = = 0 )
port - > ignore_status_mask | =
ULITE_STATUS_RXVALID | ULITE_STATUS_PARITY
| ULITE_STATUS_FRAME | ULITE_STATUS_OVERRUN ;
/* update timeout */
baud = uart_get_baud_rate ( port , termios , old , 0 , 460800 ) ;
uart_update_timeout ( port , termios - > c_cflag , baud ) ;
spin_unlock_irqrestore ( & port - > lock , flags ) ;
}
static const char * ulite_type ( struct uart_port * port )
{
return port - > type = = PORT_UARTLITE ? " uartlite " : NULL ;
}
static void ulite_release_port ( struct uart_port * port )
{
release_mem_region ( port - > mapbase , ULITE_REGION ) ;
iounmap ( port - > membase ) ;
2007-02-09 19:38:25 +03:00
port - > membase = NULL ;
2006-12-07 07:35:17 +03:00
}
static int ulite_request_port ( struct uart_port * port )
{
2008-01-08 22:35:05 +03:00
pr_debug ( " ulite console: port=%p; port->mapbase=%x \n " ,
port , port - > mapbase ) ;
2006-12-07 07:35:17 +03:00
if ( ! request_mem_region ( port - > mapbase , ULITE_REGION , " uartlite " ) ) {
dev_err ( port - > dev , " Memory region busy \n " ) ;
return - EBUSY ;
}
port - > membase = ioremap ( port - > mapbase , ULITE_REGION ) ;
if ( ! port - > membase ) {
dev_err ( port - > dev , " Unable to map registers \n " ) ;
release_mem_region ( port - > mapbase , ULITE_REGION ) ;
return - EBUSY ;
}
return 0 ;
}
static void ulite_config_port ( struct uart_port * port , int flags )
{
2006-12-22 18:38:40 +03:00
if ( ! ulite_request_port ( port ) )
port - > type = PORT_UARTLITE ;
2006-12-07 07:35:17 +03:00
}
static int ulite_verify_port ( struct uart_port * port , struct serial_struct * ser )
{
/* we don't want the core code to modify any port params */
return - EINVAL ;
}
static struct uart_ops ulite_ops = {
. tx_empty = ulite_tx_empty ,
. set_mctrl = ulite_set_mctrl ,
. get_mctrl = ulite_get_mctrl ,
. stop_tx = ulite_stop_tx ,
. start_tx = ulite_start_tx ,
. stop_rx = ulite_stop_rx ,
. enable_ms = ulite_enable_ms ,
. break_ctl = ulite_break_ctl ,
. startup = ulite_startup ,
. shutdown = ulite_shutdown ,
. set_termios = ulite_set_termios ,
. type = ulite_type ,
. release_port = ulite_release_port ,
. request_port = ulite_request_port ,
. config_port = ulite_config_port ,
. verify_port = ulite_verify_port
} ;
2007-10-02 06:15:59 +04:00
/* ---------------------------------------------------------------------
* Console driver operations
*/
2006-12-07 07:35:17 +03:00
# ifdef CONFIG_SERIAL_UARTLITE_CONSOLE
static void ulite_console_wait_tx ( struct uart_port * port )
{
int i ;
2007-10-23 08:27:46 +04:00
u8 val ;
2006-12-07 07:35:17 +03:00
2007-10-23 08:27:46 +04:00
/* Spin waiting for TX fifo to have space available */
for ( i = 0 ; i < 100000 ; i + + ) {
val = readb ( port - > membase + ULITE_STATUS ) ;
if ( ( val & ULITE_STATUS_TXFULL ) = = 0 )
2006-12-07 07:35:17 +03:00
break ;
2007-10-23 08:27:46 +04:00
cpu_relax ( ) ;
2006-12-07 07:35:17 +03:00
}
}
static void ulite_console_putchar ( struct uart_port * port , int ch )
{
ulite_console_wait_tx ( port ) ;
2007-10-02 20:47:02 +04:00
writeb ( ch , port - > membase + ULITE_TX ) ;
2006-12-07 07:35:17 +03:00
}
static void ulite_console_write ( struct console * co , const char * s ,
unsigned int count )
{
2007-10-02 06:15:44 +04:00
struct uart_port * port = & ulite_ports [ co - > index ] ;
2006-12-07 07:35:17 +03:00
unsigned long flags ;
unsigned int ier ;
int locked = 1 ;
if ( oops_in_progress ) {
locked = spin_trylock_irqsave ( & port - > lock , flags ) ;
} else
spin_lock_irqsave ( & port - > lock , flags ) ;
/* save and disable interrupt */
2007-10-02 20:47:02 +04:00
ier = readb ( port - > membase + ULITE_STATUS ) & ULITE_STATUS_IE ;
writeb ( 0 , port - > membase + ULITE_CONTROL ) ;
2006-12-07 07:35:17 +03:00
uart_console_write ( port , s , count , ulite_console_putchar ) ;
ulite_console_wait_tx ( port ) ;
/* restore interrupt state */
if ( ier )
2007-10-02 20:47:02 +04:00
writeb ( ULITE_CONTROL_IE , port - > membase + ULITE_CONTROL ) ;
2006-12-07 07:35:17 +03:00
if ( locked )
spin_unlock_irqrestore ( & port - > lock , flags ) ;
}
static int __init ulite_console_setup ( struct console * co , char * options )
{
struct uart_port * port ;
int baud = 9600 ;
int bits = 8 ;
int parity = ' n ' ;
int flow = ' n ' ;
if ( co - > index < 0 | | co - > index > = ULITE_NR_UARTS )
return - EINVAL ;
2007-10-02 06:15:44 +04:00
port = & ulite_ports [ co - > index ] ;
2006-12-07 07:35:17 +03:00
2008-02-06 20:23:41 +03:00
/* Has the device been initialized yet? */
2007-10-02 06:16:09 +04:00
if ( ! port - > mapbase ) {
pr_debug ( " console on ttyUL%i not present \n " , co - > index ) ;
return - ENODEV ;
}
2006-12-07 07:35:17 +03:00
/* not initialized yet? */
2007-10-02 06:16:04 +04:00
if ( ! port - > membase ) {
2007-10-02 06:16:09 +04:00
if ( ulite_request_port ( port ) )
return - ENODEV ;
2007-10-02 06:16:04 +04:00
}
2006-12-07 07:35:17 +03:00
if ( options )
uart_parse_options ( options , & baud , & parity , & bits , & flow ) ;
return uart_set_options ( port , co , baud , parity , bits , flow ) ;
}
static struct uart_driver ulite_uart_driver ;
static struct console ulite_console = {
2007-10-02 06:15:49 +04:00
. name = ULITE_NAME ,
2006-12-07 07:35:17 +03:00
. write = ulite_console_write ,
. device = uart_console_device ,
. setup = ulite_console_setup ,
. flags = CON_PRINTBUFFER ,
. index = - 1 , /* Specified on the cmdline (e.g. console=ttyUL0 ) */
. data = & ulite_uart_driver ,
} ;
static int __init ulite_console_init ( void )
{
register_console ( & ulite_console ) ;
return 0 ;
}
console_initcall ( ulite_console_init ) ;
# endif /* CONFIG_SERIAL_UARTLITE_CONSOLE */
static struct uart_driver ulite_uart_driver = {
. owner = THIS_MODULE ,
. driver_name = " uartlite " ,
2007-10-02 06:15:49 +04:00
. dev_name = ULITE_NAME ,
2006-12-07 07:35:17 +03:00
. major = ULITE_MAJOR ,
. minor = ULITE_MINOR ,
. nr = ULITE_NR_UARTS ,
# ifdef CONFIG_SERIAL_UARTLITE_CONSOLE
. cons = & ulite_console ,
# endif
} ;
2007-10-02 06:15:59 +04:00
/* ---------------------------------------------------------------------
* Port assignment functions ( mapping devices to uart_port structures )
*/
/** ulite_assign: register a uartlite device with the driver
*
* @ dev : pointer to device structure
* @ id : requested id number . Pass - 1 for automatic port assignment
* @ base : base address of uartlite registers
* @ irq : irq number for uartlite
*
* Returns : 0 on success , < 0 otherwise
*/
2007-10-02 06:15:54 +04:00
static int __devinit ulite_assign ( struct device * dev , int id , u32 base , int irq )
2006-12-07 07:35:17 +03:00
{
struct uart_port * port ;
2007-10-02 06:15:54 +04:00
int rc ;
2006-12-07 07:35:17 +03:00
2007-10-02 06:15:54 +04:00
/* if id = -1; then scan for a free id and use that */
if ( id < 0 ) {
for ( id = 0 ; id < ULITE_NR_UARTS ; id + + )
if ( ulite_ports [ id ] . mapbase = = 0 )
break ;
}
if ( id < 0 | | id > = ULITE_NR_UARTS ) {
dev_err ( dev , " %s%i too large \n " , ULITE_NAME , id ) ;
2006-12-07 07:35:17 +03:00
return - EINVAL ;
2007-10-02 06:15:54 +04:00
}
2006-12-07 07:35:17 +03:00
2007-10-02 06:16:09 +04:00
if ( ( ulite_ports [ id ] . mapbase ) & & ( ulite_ports [ id ] . mapbase ! = base ) ) {
2007-10-02 06:15:54 +04:00
dev_err ( dev , " cannot assign to %s%i; it is already in use \n " ,
ULITE_NAME , id ) ;
2006-12-07 07:35:17 +03:00
return - EBUSY ;
2007-10-02 06:15:54 +04:00
}
2006-12-07 07:35:17 +03:00
2007-10-02 06:15:54 +04:00
port = & ulite_ports [ id ] ;
2006-12-07 07:35:17 +03:00
2007-10-02 06:15:54 +04:00
spin_lock_init ( & port - > lock ) ;
port - > fifosize = 16 ;
port - > regshift = 2 ;
port - > iotype = UPIO_MEM ;
port - > iobase = 1 ; /* mark port in use */
port - > mapbase = base ;
port - > membase = NULL ;
port - > ops = & ulite_ops ;
port - > irq = irq ;
port - > flags = UPF_BOOT_AUTOCONF ;
port - > dev = dev ;
port - > type = PORT_UNKNOWN ;
port - > line = id ;
dev_set_drvdata ( dev , port ) ;
/* Register the port */
rc = uart_add_one_port ( & ulite_uart_driver , port ) ;
if ( rc ) {
dev_err ( dev , " uart_add_one_port() failed; err=%i \n " , rc ) ;
port - > mapbase = 0 ;
dev_set_drvdata ( dev , NULL ) ;
return rc ;
}
2006-12-07 07:35:17 +03:00
2007-10-02 06:15:54 +04:00
return 0 ;
}
2006-12-07 07:35:17 +03:00
2007-10-02 06:15:59 +04:00
/** ulite_release: register a uartlite device with the driver
*
* @ dev : pointer to device structure
*/
2008-01-08 22:35:09 +03:00
static int __devexit ulite_release ( struct device * dev )
2007-10-02 06:15:54 +04:00
{
struct uart_port * port = dev_get_drvdata ( dev ) ;
int rc = 0 ;
2006-12-07 07:35:17 +03:00
2007-10-02 06:15:54 +04:00
if ( port ) {
rc = uart_remove_one_port ( & ulite_uart_driver , port ) ;
dev_set_drvdata ( dev , NULL ) ;
port - > mapbase = 0 ;
}
2006-12-07 07:35:17 +03:00
2007-10-02 06:15:54 +04:00
return rc ;
2006-12-07 07:35:17 +03:00
}
2007-10-02 06:15:59 +04:00
/* ---------------------------------------------------------------------
* Platform bus binding
*/
2007-10-02 06:15:54 +04:00
static int __devinit ulite_probe ( struct platform_device * pdev )
2006-12-07 07:35:17 +03:00
{
2007-10-02 06:15:54 +04:00
struct resource * res , * res2 ;
2006-12-07 07:35:17 +03:00
2007-10-02 06:15:54 +04:00
res = platform_get_resource ( pdev , IORESOURCE_MEM , 0 ) ;
if ( ! res )
return - ENODEV ;
2006-12-07 07:35:17 +03:00
2007-10-02 06:15:54 +04:00
res2 = platform_get_resource ( pdev , IORESOURCE_IRQ , 0 ) ;
if ( ! res2 )
return - ENODEV ;
2006-12-07 07:35:17 +03:00
2007-10-02 06:15:54 +04:00
return ulite_assign ( & pdev - > dev , pdev - > id , res - > start , res2 - > start ) ;
}
2006-12-07 07:35:17 +03:00
2008-01-08 22:35:09 +03:00
static int __devexit ulite_remove ( struct platform_device * pdev )
2007-10-02 06:15:54 +04:00
{
return ulite_release ( & pdev - > dev ) ;
2006-12-07 07:35:17 +03:00
}
2008-04-16 01:34:35 +04:00
/* work with hotplug and coldplug */
MODULE_ALIAS ( " platform:uartlite " ) ;
2006-12-07 07:35:17 +03:00
static struct platform_driver ulite_platform_driver = {
. probe = ulite_probe ,
2008-01-08 22:35:09 +03:00
. remove = __devexit_p ( ulite_remove ) ,
2006-12-07 07:35:17 +03:00
. driver = {
. owner = THIS_MODULE ,
. name = " uartlite " ,
} ,
} ;
2007-10-02 06:16:04 +04:00
/* ---------------------------------------------------------------------
* OF bus bindings
*/
# if defined(CONFIG_OF)
static int __devinit
ulite_of_probe ( struct of_device * op , const struct of_device_id * match )
{
struct resource res ;
const unsigned int * id ;
int irq , rc ;
2008-04-30 11:55:10 +04:00
dev_dbg ( & op - > dev , " %s(%p, %p) \n " , __func__ , op , match ) ;
2007-10-02 06:16:04 +04:00
rc = of_address_to_resource ( op - > node , 0 , & res ) ;
if ( rc ) {
2007-10-02 20:47:02 +04:00
dev_err ( & op - > dev , " invalid address \n " ) ;
2007-10-02 06:16:04 +04:00
return rc ;
}
irq = irq_of_parse_and_map ( op - > node , 0 ) ;
id = of_get_property ( op - > node , " port-number " , NULL ) ;
2007-10-02 20:47:02 +04:00
return ulite_assign ( & op - > dev , id ? * id : - 1 , res . start + 3 , irq ) ;
2007-10-02 06:16:04 +04:00
}
static int __devexit ulite_of_remove ( struct of_device * op )
{
return ulite_release ( & op - > dev ) ;
}
static struct of_platform_driver ulite_of_driver = {
. owner = THIS_MODULE ,
. name = " uartlite " ,
. match_table = ulite_of_match ,
. probe = ulite_of_probe ,
. remove = __devexit_p ( ulite_of_remove ) ,
. driver = {
. name = " uartlite " ,
} ,
} ;
/* Registration helpers to keep the number of #ifdefs to a minimum */
static inline int __init ulite_of_register ( void )
{
pr_debug ( " uartlite: calling of_register_platform_driver() \n " ) ;
return of_register_platform_driver ( & ulite_of_driver ) ;
}
static inline void __exit ulite_of_unregister ( void )
{
of_unregister_platform_driver ( & ulite_of_driver ) ;
}
# else /* CONFIG_OF */
/* CONFIG_OF not enabled; do nothing helpers */
static inline int __init ulite_of_register ( void ) { return 0 ; }
static inline void __exit ulite_of_unregister ( void ) { }
# endif /* CONFIG_OF */
2007-10-02 06:15:59 +04:00
/* ---------------------------------------------------------------------
* Module setup / teardown
*/
2006-12-07 07:35:17 +03:00
int __init ulite_init ( void )
{
int ret ;
2007-10-02 06:16:04 +04:00
pr_debug ( " uartlite: calling uart_register_driver() \n " ) ;
2006-12-07 07:35:17 +03:00
ret = uart_register_driver ( & ulite_uart_driver ) ;
if ( ret )
2007-10-02 06:16:04 +04:00
goto err_uart ;
ret = ulite_of_register ( ) ;
if ( ret )
goto err_of ;
2006-12-07 07:35:17 +03:00
2007-10-02 06:16:04 +04:00
pr_debug ( " uartlite: calling platform_driver_register() \n " ) ;
2006-12-07 07:35:17 +03:00
ret = platform_driver_register ( & ulite_platform_driver ) ;
if ( ret )
2007-10-02 06:16:04 +04:00
goto err_plat ;
return 0 ;
2006-12-07 07:35:17 +03:00
2007-10-02 06:16:04 +04:00
err_plat :
ulite_of_unregister ( ) ;
err_of :
uart_unregister_driver ( & ulite_uart_driver ) ;
err_uart :
printk ( KERN_ERR " registering uartlite driver failed: err=%i " , ret ) ;
2006-12-07 07:35:17 +03:00
return ret ;
}
void __exit ulite_exit ( void )
{
platform_driver_unregister ( & ulite_platform_driver ) ;
2007-10-02 06:16:04 +04:00
ulite_of_unregister ( ) ;
2006-12-07 07:35:17 +03:00
uart_unregister_driver ( & ulite_uart_driver ) ;
}
module_init ( ulite_init ) ;
module_exit ( ulite_exit ) ;
MODULE_AUTHOR ( " Peter Korsgaard <jacmet@sunsite.dk> " ) ;
MODULE_DESCRIPTION ( " Xilinx uartlite serial driver " ) ;
MODULE_LICENSE ( " GPL " ) ;