diff options
Diffstat (limited to 'drivers/tty/serial/s3c2410.c')
| -rw-r--r-- | drivers/tty/serial/s3c2410.c | 118 | 
1 files changed, 118 insertions, 0 deletions
diff --git a/drivers/tty/serial/s3c2410.c b/drivers/tty/serial/s3c2410.c new file mode 100644 index 00000000000..73f089d3efd --- /dev/null +++ b/drivers/tty/serial/s3c2410.c @@ -0,0 +1,118 @@ +/* linux/drivers/serial/s3c2410.c + * + * Driver for Samsung S3C2410 SoC onboard UARTs. + * + * Ben Dooks, Copyright (c) 2003-2008 Simtec Electronics + *	http://armlinux.simtec.co.uk/ + * + * This program is free software; you can redistribute it and/or modify + * it under the terms of the GNU General Public License version 2 as + * published by the Free Software Foundation. +*/ + +#include <linux/module.h> +#include <linux/ioport.h> +#include <linux/io.h> +#include <linux/platform_device.h> +#include <linux/init.h> +#include <linux/serial_core.h> +#include <linux/serial.h> + +#include <asm/irq.h> +#include <mach/hardware.h> + +#include <plat/regs-serial.h> +#include <mach/regs-gpio.h> + +#include "samsung.h" + +static int s3c2410_serial_setsource(struct uart_port *port, +				    struct s3c24xx_uart_clksrc *clk) +{ +	unsigned long ucon = rd_regl(port, S3C2410_UCON); + +	if (strcmp(clk->name, "uclk") == 0) +		ucon |= S3C2410_UCON_UCLK; +	else +		ucon &= ~S3C2410_UCON_UCLK; + +	wr_regl(port, S3C2410_UCON, ucon); +	return 0; +} + +static int s3c2410_serial_getsource(struct uart_port *port, +				    struct s3c24xx_uart_clksrc *clk) +{ +	unsigned long ucon = rd_regl(port, S3C2410_UCON); + +	clk->divisor = 1; +	clk->name = (ucon & S3C2410_UCON_UCLK) ? "uclk" : "pclk"; + +	return 0; +} + +static int s3c2410_serial_resetport(struct uart_port *port, +				    struct s3c2410_uartcfg *cfg) +{ +	dbg("s3c2410_serial_resetport: port=%p (%08lx), cfg=%p\n", +	    port, port->mapbase, cfg); + +	wr_regl(port, S3C2410_UCON,  cfg->ucon); +	wr_regl(port, S3C2410_ULCON, cfg->ulcon); + +	/* reset both fifos */ + +	wr_regl(port, S3C2410_UFCON, cfg->ufcon | S3C2410_UFCON_RESETBOTH); +	wr_regl(port, S3C2410_UFCON, cfg->ufcon); + +	return 0; +} + +static struct s3c24xx_uart_info s3c2410_uart_inf = { +	.name		= "Samsung S3C2410 UART", +	.type		= PORT_S3C2410, +	.fifosize	= 16, +	.rx_fifomask	= S3C2410_UFSTAT_RXMASK, +	.rx_fifoshift	= S3C2410_UFSTAT_RXSHIFT, +	.rx_fifofull	= S3C2410_UFSTAT_RXFULL, +	.tx_fifofull	= S3C2410_UFSTAT_TXFULL, +	.tx_fifomask	= S3C2410_UFSTAT_TXMASK, +	.tx_fifoshift	= S3C2410_UFSTAT_TXSHIFT, +	.get_clksrc	= s3c2410_serial_getsource, +	.set_clksrc	= s3c2410_serial_setsource, +	.reset_port	= s3c2410_serial_resetport, +}; + +static int s3c2410_serial_probe(struct platform_device *dev) +{ +	return s3c24xx_serial_probe(dev, &s3c2410_uart_inf); +} + +static struct platform_driver s3c2410_serial_driver = { +	.probe		= s3c2410_serial_probe, +	.remove		= __devexit_p(s3c24xx_serial_remove), +	.driver		= { +		.name	= "s3c2410-uart", +		.owner	= THIS_MODULE, +	}, +}; + +s3c24xx_console_init(&s3c2410_serial_driver, &s3c2410_uart_inf); + +static int __init s3c2410_serial_init(void) +{ +	return s3c24xx_serial_init(&s3c2410_serial_driver, &s3c2410_uart_inf); +} + +static void __exit s3c2410_serial_exit(void) +{ +	platform_driver_unregister(&s3c2410_serial_driver); +} + +module_init(s3c2410_serial_init); +module_exit(s3c2410_serial_exit); + +MODULE_LICENSE("GPL v2"); +MODULE_AUTHOR("Ben Dooks <ben@simtec.co.uk>"); +MODULE_DESCRIPTION("Samsung S3C2410 SoC Serial port driver"); +MODULE_ALIAS("platform:s3c2410-uart");  |