diff options
author | David Daney <david.daney@cavium.com> | 2012-07-05 12:12:40 -0400 |
---|---|---|
committer | Ralf Baechle <ralf@linux-mips.org> | 2012-07-23 08:54:53 -0400 |
commit | b59b284101454823c77c9635f3ee8cd4f979fe6e (patch) | |
tree | d1b70952c0000195151c17a121edf6e326e06d29 /arch/mips | |
parent | df9244c5365c12d0cd1452323d3dc3ef91d80173 (diff) |
MIPS: Octeon: Use device tree to register serial ports.
Switch to using the device tree to register serial ports.
Add all the ports with compatible = "cavium,octeon-3860-uart". Octeon serial
ports have their own device type, required port flags, and I/O
functions, so using of_serial.c is not indicated.
We need to do this as late_initcall, as the 8250 driver must be
initialized before we add any ports. 8250 initialization is done at
device_initcall time.
The OCTEON_IRQ_UART{0,1,2} symbols are removed as they are now unused
and interfere with irq_domain used by the device tree code.
Signed-off-by: David Daney <david.daney@cavium.com>
Acked-by: Grant Likely <grant.likely@secretlab.ca>
Cc: linux-mips@linux-mips.org
Cc: devicetree-discuss@lists.ozlabs.org
Cc: Rob Herring <rob.herring@calxeda.com>
Cc: linux-kernel@vger.kernel.org
Cc: David Daney <david.daney@cavium.com>
Patchwork: https://patchwork.linux-mips.org/patch/3942/
Signed-off-by: Ralf Baechle <ralf@linux-mips.org>
Diffstat (limited to 'arch/mips')
-rw-r--r-- | arch/mips/cavium-octeon/octeon-irq.c | 4 | ||||
-rw-r--r-- | arch/mips/cavium-octeon/serial.c | 134 |
2 files changed, 53 insertions, 85 deletions
diff --git a/arch/mips/cavium-octeon/octeon-irq.c b/arch/mips/cavium-octeon/octeon-irq.c index 5fb76aa346be..7fb1f222b8a5 100644 --- a/arch/mips/cavium-octeon/octeon-irq.c +++ b/arch/mips/cavium-octeon/octeon-irq.c | |||
@@ -1184,9 +1184,6 @@ static void __init octeon_irq_init_ciu(void) | |||
1184 | octeon_irq_set_ciu_mapping(OCTEON_IRQ_MBOX0, 0, 32, chip_mbox, handle_percpu_irq); | 1184 | octeon_irq_set_ciu_mapping(OCTEON_IRQ_MBOX0, 0, 32, chip_mbox, handle_percpu_irq); |
1185 | octeon_irq_set_ciu_mapping(OCTEON_IRQ_MBOX1, 0, 33, chip_mbox, handle_percpu_irq); | 1185 | octeon_irq_set_ciu_mapping(OCTEON_IRQ_MBOX1, 0, 33, chip_mbox, handle_percpu_irq); |
1186 | 1186 | ||
1187 | octeon_irq_set_ciu_mapping(OCTEON_IRQ_UART0, 0, 34, chip, handle_level_irq); | ||
1188 | octeon_irq_set_ciu_mapping(OCTEON_IRQ_UART1, 0, 35, chip, handle_level_irq); | ||
1189 | |||
1190 | for (i = 0; i < 4; i++) | 1187 | for (i = 0; i < 4; i++) |
1191 | octeon_irq_set_ciu_mapping(i + OCTEON_IRQ_PCI_INT0, 0, i + 36, chip, handle_level_irq); | 1188 | octeon_irq_set_ciu_mapping(i + OCTEON_IRQ_PCI_INT0, 0, i + 36, chip, handle_level_irq); |
1192 | for (i = 0; i < 4; i++) | 1189 | for (i = 0; i < 4; i++) |
@@ -1203,7 +1200,6 @@ static void __init octeon_irq_init_ciu(void) | |||
1203 | for (i = 0; i < 16; i++) | 1200 | for (i = 0; i < 16; i++) |
1204 | octeon_irq_set_ciu_mapping(i + OCTEON_IRQ_WDOG0, 1, i + 0, chip_wd, handle_level_irq); | 1201 | octeon_irq_set_ciu_mapping(i + OCTEON_IRQ_WDOG0, 1, i + 0, chip_wd, handle_level_irq); |
1205 | 1202 | ||
1206 | octeon_irq_set_ciu_mapping(OCTEON_IRQ_UART2, 1, 16, chip, handle_level_irq); | ||
1207 | octeon_irq_set_ciu_mapping(OCTEON_IRQ_USB1, 1, 17, chip, handle_level_irq); | 1203 | octeon_irq_set_ciu_mapping(OCTEON_IRQ_USB1, 1, 17, chip, handle_level_irq); |
1208 | 1204 | ||
1209 | gpio_node = of_find_compatible_node(NULL, NULL, "cavium,octeon-3860-gpio"); | 1205 | gpio_node = of_find_compatible_node(NULL, NULL, "cavium,octeon-3860-gpio"); |
diff --git a/arch/mips/cavium-octeon/serial.c b/arch/mips/cavium-octeon/serial.c index 057f0ae88c99..138b2216b4f8 100644 --- a/arch/mips/cavium-octeon/serial.c +++ b/arch/mips/cavium-octeon/serial.c | |||
@@ -43,95 +43,67 @@ void octeon_serial_out(struct uart_port *up, int offset, int value) | |||
43 | cvmx_write_csr((uint64_t)(up->membase + (offset << 3)), (u8)value); | 43 | cvmx_write_csr((uint64_t)(up->membase + (offset << 3)), (u8)value); |
44 | } | 44 | } |
45 | 45 | ||
46 | /* | 46 | static int __devinit octeon_serial_probe(struct platform_device *pdev) |
47 | * Allocated in .bss, so it is all zeroed. | ||
48 | */ | ||
49 | #define OCTEON_MAX_UARTS 3 | ||
50 | static struct plat_serial8250_port octeon_uart8250_data[OCTEON_MAX_UARTS + 1]; | ||
51 | static struct platform_device octeon_uart8250_device = { | ||
52 | .name = "serial8250", | ||
53 | .id = PLAT8250_DEV_PLATFORM, | ||
54 | .dev = { | ||
55 | .platform_data = octeon_uart8250_data, | ||
56 | }, | ||
57 | }; | ||
58 | |||
59 | static void __init octeon_uart_set_common(struct plat_serial8250_port *p) | ||
60 | { | 47 | { |
61 | p->flags = ASYNC_SKIP_TEST | UPF_SHARE_IRQ | UPF_FIXED_TYPE; | 48 | int irq, res; |
62 | p->type = PORT_OCTEON; | 49 | struct resource *res_mem; |
63 | p->iotype = UPIO_MEM; | 50 | struct uart_port port; |
64 | p->regshift = 3; /* I/O addresses are every 8 bytes */ | 51 | |
52 | /* All adaptors have an irq. */ | ||
53 | irq = platform_get_irq(pdev, 0); | ||
54 | if (irq < 0) | ||
55 | return irq; | ||
56 | |||
57 | memset(&port, 0, sizeof(port)); | ||
58 | |||
59 | port.flags = ASYNC_SKIP_TEST | UPF_SHARE_IRQ | UPF_FIXED_TYPE; | ||
60 | port.type = PORT_OCTEON; | ||
61 | port.iotype = UPIO_MEM; | ||
62 | port.regshift = 3; | ||
63 | port.dev = &pdev->dev; | ||
64 | |||
65 | if (octeon_is_simulation()) | 65 | if (octeon_is_simulation()) |
66 | /* Make simulator output fast*/ | 66 | /* Make simulator output fast*/ |
67 | p->uartclk = 115200 * 16; | 67 | port.uartclk = 115200 * 16; |
68 | else | 68 | else |
69 | p->uartclk = octeon_get_io_clock_rate(); | 69 | port.uartclk = octeon_get_io_clock_rate(); |
70 | p->serial_in = octeon_serial_in; | ||
71 | p->serial_out = octeon_serial_out; | ||
72 | } | ||
73 | 70 | ||
74 | static int __init octeon_serial_init(void) | 71 | port.serial_in = octeon_serial_in; |
75 | { | 72 | port.serial_out = octeon_serial_out; |
76 | int enable_uart0; | 73 | port.irq = irq; |
77 | int enable_uart1; | ||
78 | int enable_uart2; | ||
79 | struct plat_serial8250_port *p; | ||
80 | |||
81 | #ifdef CONFIG_CAVIUM_OCTEON_2ND_KERNEL | ||
82 | /* | ||
83 | * If we are configured to run as the second of two kernels, | ||
84 | * disable uart0 and enable uart1. Uart0 is owned by the first | ||
85 | * kernel | ||
86 | */ | ||
87 | enable_uart0 = 0; | ||
88 | enable_uart1 = 1; | ||
89 | #else | ||
90 | /* | ||
91 | * We are configured for the first kernel. We'll enable uart0 | ||
92 | * if the bootloader told us to use 0, otherwise will enable | ||
93 | * uart 1. | ||
94 | */ | ||
95 | enable_uart0 = (octeon_get_boot_uart() == 0); | ||
96 | enable_uart1 = (octeon_get_boot_uart() == 1); | ||
97 | #ifdef CONFIG_KGDB | ||
98 | enable_uart1 = 1; | ||
99 | #endif | ||
100 | #endif | ||
101 | |||
102 | /* Right now CN52XX is the only chip with a third uart */ | ||
103 | enable_uart2 = OCTEON_IS_MODEL(OCTEON_CN52XX); | ||
104 | |||
105 | p = octeon_uart8250_data; | ||
106 | if (enable_uart0) { | ||
107 | /* Add a ttyS device for hardware uart 0 */ | ||
108 | octeon_uart_set_common(p); | ||
109 | p->membase = (void *) CVMX_MIO_UARTX_RBR(0); | ||
110 | p->mapbase = CVMX_MIO_UARTX_RBR(0) & ((1ull << 49) - 1); | ||
111 | p->irq = OCTEON_IRQ_UART0; | ||
112 | p++; | ||
113 | } | ||
114 | 74 | ||
115 | if (enable_uart1) { | 75 | res_mem = platform_get_resource(pdev, IORESOURCE_MEM, 0); |
116 | /* Add a ttyS device for hardware uart 1 */ | 76 | if (res_mem == NULL) { |
117 | octeon_uart_set_common(p); | 77 | dev_err(&pdev->dev, "found no memory resource\n"); |
118 | p->membase = (void *) CVMX_MIO_UARTX_RBR(1); | 78 | return -ENXIO; |
119 | p->mapbase = CVMX_MIO_UARTX_RBR(1) & ((1ull << 49) - 1); | ||
120 | p->irq = OCTEON_IRQ_UART1; | ||
121 | p++; | ||
122 | } | ||
123 | if (enable_uart2) { | ||
124 | /* Add a ttyS device for hardware uart 2 */ | ||
125 | octeon_uart_set_common(p); | ||
126 | p->membase = (void *) CVMX_MIO_UART2_RBR; | ||
127 | p->mapbase = CVMX_MIO_UART2_RBR & ((1ull << 49) - 1); | ||
128 | p->irq = OCTEON_IRQ_UART2; | ||
129 | p++; | ||
130 | } | 79 | } |
80 | port.mapbase = res_mem->start; | ||
81 | port.membase = ioremap(res_mem->start, resource_size(res_mem)); | ||
131 | 82 | ||
132 | BUG_ON(p > &octeon_uart8250_data[OCTEON_MAX_UARTS]); | 83 | res = serial8250_register_port(&port); |
133 | 84 | ||
134 | return platform_device_register(&octeon_uart8250_device); | 85 | return res >= 0 ? 0 : res; |
135 | } | 86 | } |
136 | 87 | ||
137 | device_initcall(octeon_serial_init); | 88 | static struct of_device_id octeon_serial_match[] = { |
89 | { | ||
90 | .compatible = "cavium,octeon-3860-uart", | ||
91 | }, | ||
92 | {}, | ||
93 | }; | ||
94 | MODULE_DEVICE_TABLE(of, octeon_serial_match); | ||
95 | |||
96 | static struct platform_driver octeon_serial_driver = { | ||
97 | .probe = octeon_serial_probe, | ||
98 | .driver = { | ||
99 | .owner = THIS_MODULE, | ||
100 | .name = "octeon_serial", | ||
101 | .of_match_table = octeon_serial_match, | ||
102 | }, | ||
103 | }; | ||
104 | |||
105 | static int __init octeon_serial_init(void) | ||
106 | { | ||
107 | return platform_driver_register(&octeon_serial_driver); | ||
108 | } | ||
109 | late_initcall(octeon_serial_init); | ||