diff options
Diffstat (limited to 'arch/arm/mach-omap2/serial.c')
-rw-r--r-- | arch/arm/mach-omap2/serial.c | 80 |
1 files changed, 63 insertions, 17 deletions
diff --git a/arch/arm/mach-omap2/serial.c b/arch/arm/mach-omap2/serial.c index e10a02df6e1d..b79bc8926cc9 100644 --- a/arch/arm/mach-omap2/serial.c +++ b/arch/arm/mach-omap2/serial.c | |||
@@ -23,6 +23,7 @@ | |||
23 | #include <linux/serial_reg.h> | 23 | #include <linux/serial_reg.h> |
24 | #include <linux/clk.h> | 24 | #include <linux/clk.h> |
25 | #include <linux/io.h> | 25 | #include <linux/io.h> |
26 | #include <linux/delay.h> | ||
26 | 27 | ||
27 | #include <plat/common.h> | 28 | #include <plat/common.h> |
28 | #include <plat/board.h> | 29 | #include <plat/board.h> |
@@ -80,7 +81,6 @@ static LIST_HEAD(uart_list); | |||
80 | 81 | ||
81 | static struct plat_serial8250_port serial_platform_data0[] = { | 82 | static struct plat_serial8250_port serial_platform_data0[] = { |
82 | { | 83 | { |
83 | .mapbase = OMAP_UART1_BASE, | ||
84 | .irq = 72, | 84 | .irq = 72, |
85 | .flags = UPF_BOOT_AUTOCONF, | 85 | .flags = UPF_BOOT_AUTOCONF, |
86 | .iotype = UPIO_MEM, | 86 | .iotype = UPIO_MEM, |
@@ -93,7 +93,6 @@ static struct plat_serial8250_port serial_platform_data0[] = { | |||
93 | 93 | ||
94 | static struct plat_serial8250_port serial_platform_data1[] = { | 94 | static struct plat_serial8250_port serial_platform_data1[] = { |
95 | { | 95 | { |
96 | .mapbase = OMAP_UART2_BASE, | ||
97 | .irq = 73, | 96 | .irq = 73, |
98 | .flags = UPF_BOOT_AUTOCONF, | 97 | .flags = UPF_BOOT_AUTOCONF, |
99 | .iotype = UPIO_MEM, | 98 | .iotype = UPIO_MEM, |
@@ -106,7 +105,6 @@ static struct plat_serial8250_port serial_platform_data1[] = { | |||
106 | 105 | ||
107 | static struct plat_serial8250_port serial_platform_data2[] = { | 106 | static struct plat_serial8250_port serial_platform_data2[] = { |
108 | { | 107 | { |
109 | .mapbase = OMAP_UART3_BASE, | ||
110 | .irq = 74, | 108 | .irq = 74, |
111 | .flags = UPF_BOOT_AUTOCONF, | 109 | .flags = UPF_BOOT_AUTOCONF, |
112 | .iotype = UPIO_MEM, | 110 | .iotype = UPIO_MEM, |
@@ -117,10 +115,9 @@ static struct plat_serial8250_port serial_platform_data2[] = { | |||
117 | } | 115 | } |
118 | }; | 116 | }; |
119 | 117 | ||
120 | #ifdef CONFIG_ARCH_OMAP4 | 118 | #if defined(CONFIG_ARCH_OMAP3) || defined(CONFIG_ARCH_OMAP4) |
121 | static struct plat_serial8250_port serial_platform_data3[] = { | 119 | static struct plat_serial8250_port serial_platform_data3[] = { |
122 | { | 120 | { |
123 | .mapbase = OMAP_UART4_BASE, | ||
124 | .irq = 70, | 121 | .irq = 70, |
125 | .flags = UPF_BOOT_AUTOCONF, | 122 | .flags = UPF_BOOT_AUTOCONF, |
126 | .iotype = UPIO_MEM, | 123 | .iotype = UPIO_MEM, |
@@ -130,7 +127,26 @@ static struct plat_serial8250_port serial_platform_data3[] = { | |||
130 | .flags = 0 | 127 | .flags = 0 |
131 | } | 128 | } |
132 | }; | 129 | }; |
130 | |||
131 | static inline void omap2_set_globals_uart4(struct omap_globals *omap2_globals) | ||
132 | { | ||
133 | serial_platform_data3[0].mapbase = omap2_globals->uart4_phys; | ||
134 | } | ||
135 | #else | ||
136 | static inline void omap2_set_globals_uart4(struct omap_globals *omap2_globals) | ||
137 | { | ||
138 | } | ||
133 | #endif | 139 | #endif |
140 | |||
141 | void __init omap2_set_globals_uart(struct omap_globals *omap2_globals) | ||
142 | { | ||
143 | serial_platform_data0[0].mapbase = omap2_globals->uart1_phys; | ||
144 | serial_platform_data1[0].mapbase = omap2_globals->uart2_phys; | ||
145 | serial_platform_data2[0].mapbase = omap2_globals->uart3_phys; | ||
146 | if (cpu_is_omap3630() || cpu_is_omap44xx()) | ||
147 | omap2_set_globals_uart4(omap2_globals); | ||
148 | } | ||
149 | |||
134 | static inline unsigned int __serial_read_reg(struct uart_port *up, | 150 | static inline unsigned int __serial_read_reg(struct uart_port *up, |
135 | int offset) | 151 | int offset) |
136 | { | 152 | { |
@@ -145,6 +161,13 @@ static inline unsigned int serial_read_reg(struct plat_serial8250_port *up, | |||
145 | return (unsigned int)__raw_readb(up->membase + offset); | 161 | return (unsigned int)__raw_readb(up->membase + offset); |
146 | } | 162 | } |
147 | 163 | ||
164 | static inline void __serial_write_reg(struct uart_port *up, int offset, | ||
165 | int value) | ||
166 | { | ||
167 | offset <<= up->regshift; | ||
168 | __raw_writeb(value, up->membase + offset); | ||
169 | } | ||
170 | |||
148 | static inline void serial_write_reg(struct plat_serial8250_port *p, int offset, | 171 | static inline void serial_write_reg(struct plat_serial8250_port *p, int offset, |
149 | int value) | 172 | int value) |
150 | { | 173 | { |
@@ -574,7 +597,7 @@ static struct omap_uart_state omap_uart[] = { | |||
574 | }, | 597 | }, |
575 | }, | 598 | }, |
576 | }, | 599 | }, |
577 | #ifdef CONFIG_ARCH_OMAP4 | 600 | #if defined(CONFIG_ARCH_OMAP3) || defined(CONFIG_ARCH_OMAP4) |
578 | { | 601 | { |
579 | .pdev = { | 602 | .pdev = { |
580 | .name = "serial8250", | 603 | .name = "serial8250", |
@@ -605,6 +628,20 @@ static unsigned int serial_in_override(struct uart_port *up, int offset) | |||
605 | return __serial_read_reg(up, offset); | 628 | return __serial_read_reg(up, offset); |
606 | } | 629 | } |
607 | 630 | ||
631 | static void serial_out_override(struct uart_port *up, int offset, int value) | ||
632 | { | ||
633 | unsigned int status, tmout = 10000; | ||
634 | |||
635 | status = __serial_read_reg(up, UART_LSR); | ||
636 | while (!(status & UART_LSR_THRE)) { | ||
637 | /* Wait up to 10ms for the character(s) to be sent. */ | ||
638 | if (--tmout == 0) | ||
639 | break; | ||
640 | udelay(1); | ||
641 | status = __serial_read_reg(up, UART_LSR); | ||
642 | } | ||
643 | __serial_write_reg(up, offset, value); | ||
644 | } | ||
608 | void __init omap_serial_early_init(void) | 645 | void __init omap_serial_early_init(void) |
609 | { | 646 | { |
610 | int i; | 647 | int i; |
@@ -701,15 +738,19 @@ void __init omap_serial_init_port(int port) | |||
701 | DEV_CREATE_FILE(dev, &dev_attr_sleep_timeout); | 738 | DEV_CREATE_FILE(dev, &dev_attr_sleep_timeout); |
702 | } | 739 | } |
703 | 740 | ||
704 | /* omap44xx: Never read empty UART fifo | 741 | /* |
705 | * omap3xxx: Never read empty UART fifo on UARTs | 742 | * omap44xx: Never read empty UART fifo |
706 | * with IP rev >=0x52 | 743 | * omap3xxx: Never read empty UART fifo on UARTs |
707 | */ | 744 | * with IP rev >=0x52 |
708 | if (cpu_is_omap44xx()) | 745 | */ |
709 | uart->p->serial_in = serial_in_override; | 746 | if (cpu_is_omap44xx()) { |
710 | else if ((serial_read_reg(uart->p, UART_OMAP_MVER) & 0xFF) | 747 | uart->p->serial_in = serial_in_override; |
711 | >= UART_OMAP_NO_EMPTY_FIFO_READ_IP_REV) | 748 | uart->p->serial_out = serial_out_override; |
712 | uart->p->serial_in = serial_in_override; | 749 | } else if ((serial_read_reg(uart->p, UART_OMAP_MVER) & 0xFF) |
750 | >= UART_OMAP_NO_EMPTY_FIFO_READ_IP_REV) { | ||
751 | uart->p->serial_in = serial_in_override; | ||
752 | uart->p->serial_out = serial_out_override; | ||
753 | } | ||
713 | } | 754 | } |
714 | 755 | ||
715 | /** | 756 | /** |
@@ -721,8 +762,13 @@ void __init omap_serial_init_port(int port) | |||
721 | */ | 762 | */ |
722 | void __init omap_serial_init(void) | 763 | void __init omap_serial_init(void) |
723 | { | 764 | { |
724 | int i; | 765 | int i, nr_ports; |
766 | |||
767 | if (!(cpu_is_omap3630() || cpu_is_omap4430())) | ||
768 | nr_ports = 3; | ||
769 | else | ||
770 | nr_ports = ARRAY_SIZE(omap_uart); | ||
725 | 771 | ||
726 | for (i = 0; i < ARRAY_SIZE(omap_uart); i++) | 772 | for (i = 0; i < nr_ports; i++) |
727 | omap_serial_init_port(i); | 773 | omap_serial_init_port(i); |
728 | } | 774 | } |