aboutsummaryrefslogtreecommitdiffstats
path: root/drivers/serial/sh-sci.c
diff options
context:
space:
mode:
Diffstat (limited to 'drivers/serial/sh-sci.c')
-rw-r--r--drivers/serial/sh-sci.c45
1 files changed, 33 insertions, 12 deletions
diff --git a/drivers/serial/sh-sci.c b/drivers/serial/sh-sci.c
index f5aebc9f27e7..ac658a7a27be 100644
--- a/drivers/serial/sh-sci.c
+++ b/drivers/serial/sh-sci.c
@@ -3,7 +3,7 @@
3 * 3 *
4 * SuperH on-chip serial module support. (SCI with no FIFO / with FIFO) 4 * SuperH on-chip serial module support. (SCI with no FIFO / with FIFO)
5 * 5 *
6 * Copyright (C) 2002 - 2006 Paul Mundt 6 * Copyright (C) 2002 - 2008 Paul Mundt
7 * Modified to support SH7720 SCIF. Markus Brunner, Mark Jonas (Jul 2007). 7 * Modified to support SH7720 SCIF. Markus Brunner, Mark Jonas (Jul 2007).
8 * 8 *
9 * based off of the old drivers/char/sh-sci.c by: 9 * based off of the old drivers/char/sh-sci.c by:
@@ -46,6 +46,7 @@
46#include <linux/cpufreq.h> 46#include <linux/cpufreq.h>
47#include <linux/clk.h> 47#include <linux/clk.h>
48#include <linux/ctype.h> 48#include <linux/ctype.h>
49#include <linux/err.h>
49 50
50#ifdef CONFIG_SUPERH 51#ifdef CONFIG_SUPERH
51#include <asm/clock.h> 52#include <asm/clock.h>
@@ -1145,12 +1146,16 @@ static void sci_config_port(struct uart_port *port, int flags)
1145 break; 1146 break;
1146 } 1147 }
1147 1148
1148#if defined(CONFIG_CPU_SUBTYPE_SH5_101) || defined(CONFIG_CPU_SUBTYPE_SH5_103) 1149 if (port->flags & UPF_IOREMAP && !port->membase) {
1149 if (port->mapbase == 0) 1150#if defined(CONFIG_SUPERH64)
1150 port->mapbase = onchip_remap(SCIF_ADDR_SH5, 1024, "SCIF"); 1151 port->mapbase = onchip_remap(SCIF_ADDR_SH5, 1024, "SCIF");
1151 1152 port->membase = (void __iomem *)port->mapbase;
1152 port->membase = (void __iomem *)port->mapbase; 1153#else
1154 port->membase = ioremap_nocache(port->mapbase, 0x40);
1153#endif 1155#endif
1156
1157 printk(KERN_ERR "sci: can't remap port#%d\n", port->line);
1158 }
1154} 1159}
1155 1160
1156static int sci_verify_port(struct uart_port *port, struct serial_struct *ser) 1161static int sci_verify_port(struct uart_port *port, struct serial_struct *ser)
@@ -1436,7 +1441,7 @@ static struct uart_driver sci_uart_driver = {
1436static int __devinit sci_probe(struct platform_device *dev) 1441static int __devinit sci_probe(struct platform_device *dev)
1437{ 1442{
1438 struct plat_sci_port *p = dev->dev.platform_data; 1443 struct plat_sci_port *p = dev->dev.platform_data;
1439 int i; 1444 int i, ret = -EINVAL;
1440 1445
1441 for (i = 0; p && p->flags != 0; p++, i++) { 1446 for (i = 0; p && p->flags != 0; p++, i++) {
1442 struct sci_port *sciport = &sci_ports[i]; 1447 struct sci_port *sciport = &sci_ports[i];
@@ -1453,12 +1458,22 @@ static int __devinit sci_probe(struct platform_device *dev)
1453 1458
1454 sciport->port.mapbase = p->mapbase; 1459 sciport->port.mapbase = p->mapbase;
1455 1460
1456 /* 1461 if (p->mapbase && !p->membase) {
1457 * For the simple (and majority of) cases where we don't need 1462 if (p->flags & UPF_IOREMAP) {
1458 * to do any remapping, just cast the cookie directly. 1463 p->membase = ioremap_nocache(p->mapbase, 0x40);
1459 */ 1464 if (IS_ERR(p->membase)) {
1460 if (p->mapbase && !p->membase && !(p->flags & UPF_IOREMAP)) 1465 ret = PTR_ERR(p->membase);
1461 p->membase = (void __iomem *)p->mapbase; 1466 goto err_unreg;
1467 }
1468 } else {
1469 /*
1470 * For the simple (and majority of) cases
1471 * where we don't need to do any remapping,
1472 * just cast the cookie directly.
1473 */
1474 p->membase = (void __iomem *)p->mapbase;
1475 }
1476 }
1462 1477
1463 sciport->port.membase = p->membase; 1478 sciport->port.membase = p->membase;
1464 1479
@@ -1489,6 +1504,12 @@ static int __devinit sci_probe(struct platform_device *dev)
1489#endif 1504#endif
1490 1505
1491 return 0; 1506 return 0;
1507
1508err_unreg:
1509 for (i = i - 1; i >= 0; i--)
1510 uart_remove_one_port(&sci_uart_driver, &sci_ports[i].port);
1511
1512 return ret;
1492} 1513}
1493 1514
1494static int __devexit sci_remove(struct platform_device *dev) 1515static int __devexit sci_remove(struct platform_device *dev)