mpc52xx: switch to 3.2.9
[openwrt.git] / target / linux / ar71xx / patches-3.2 / 014-SERIAL-AR933X-Add-driver-for-the-built-in-UART-of-th.patch
1 From 1de387abd06fb67aaa8e27a48e378ffd9aaddd74 Mon Sep 17 00:00:00 2001
2 From: Gabor Juhos <juhosg@openwrt.org>
3 Date: Mon, 20 Jun 2011 19:26:11 +0200
4 Subject: [PATCH 14/27] SERIAL: AR933X: Add driver for the built-in UART of the SoC
5
6 This patch adds the driver for the built-in UART of the
7 Atheros AR933X SoCs.
8
9 Signed-off-by: Gabor Juhos <juhosg@openwrt.org>
10 Cc: linux-mips@linux-mips.org
11 Cc: Kathy Giori <kgiori@qca.qualcomm.com>
12 Cc: "Luis R. Rodriguez" <rodrigue@qca.qualcomm.com>
13 Cc: Alan Cox <alan@lxorguk.ukuu.org.uk>
14 Cc: linux-serial@vger.kernel.org
15 Patchwork: https://patchwork.linux-mips.org/patch/2526/
16 Signed-off-by: Alan Cox <alan@linux.intel.com>
17 Signed-off-by: Ralf Baechle <ralf@linux-mips.org>
18 ---
19 .../include/asm/mach-ath79/ar933x_uart_platform.h | 18 +
20 drivers/tty/serial/Kconfig | 23 +
21 drivers/tty/serial/Makefile | 1 +
22 drivers/tty/serial/ar933x_uart.c | 688 ++++++++++++++++++++
23 include/linux/serial_core.h | 4 +
24 5 files changed, 734 insertions(+), 0 deletions(-)
25 create mode 100644 arch/mips/include/asm/mach-ath79/ar933x_uart_platform.h
26 create mode 100644 drivers/tty/serial/ar933x_uart.c
27
28 --- /dev/null
29 +++ b/arch/mips/include/asm/mach-ath79/ar933x_uart_platform.h
30 @@ -0,0 +1,18 @@
31 +/*
32 + * Platform data definition for Atheros AR933X UART
33 + *
34 + * Copyright (C) 2011 Gabor Juhos <juhosg@openwrt.org>
35 + *
36 + * This program is free software; you can redistribute it and/or modify it
37 + * under the terms of the GNU General Public License version 2 as published
38 + * by the Free Software Foundation.
39 + */
40 +
41 +#ifndef _AR933X_UART_PLATFORM_H
42 +#define _AR933X_UART_PLATFORM_H
43 +
44 +struct ar933x_uart_platform_data {
45 + unsigned uartclk;
46 +};
47 +
48 +#endif /* _AR933X_UART_PLATFORM_H */
49 --- a/drivers/tty/serial/Kconfig
50 +++ b/drivers/tty/serial/Kconfig
51 @@ -1610,4 +1610,27 @@ config SERIAL_XILINX_PS_UART_CONSOLE
52 help
53 Enable a Xilinx PS UART port to be the system console.
54
55 +config SERIAL_AR933X
56 + bool "AR933X serial port support"
57 + depends on SOC_AR933X
58 + select SERIAL_CORE
59 + help
60 + If you have an Atheros AR933X SOC based board and want to use the
61 + built-in UART of the SoC, say Y to this option.
62 +
63 +config SERIAL_AR933X_CONSOLE
64 + bool "Console on AR933X serial port"
65 + depends on SERIAL_AR933X=y
66 + select SERIAL_CORE_CONSOLE
67 + help
68 + Enable a built-in UART port of the AR933X to be the system console.
69 +
70 +config SERIAL_AR933X_NR_UARTS
71 + int "Maximum number of AR933X serial ports"
72 + depends on SERIAL_AR933X
73 + default "2"
74 + help
75 + Set this to the number of serial ports you want the driver
76 + to support.
77 +
78 endmenu
79 --- a/drivers/tty/serial/Makefile
80 +++ b/drivers/tty/serial/Makefile
81 @@ -94,3 +94,4 @@ obj-$(CONFIG_SERIAL_MSM_SMD) += msm_smd_
82 obj-$(CONFIG_SERIAL_MXS_AUART) += mxs-auart.o
83 obj-$(CONFIG_SERIAL_LANTIQ) += lantiq.o
84 obj-$(CONFIG_SERIAL_XILINX_PS_UART) += xilinx_uartps.o
85 +obj-$(CONFIG_SERIAL_AR933X) += ar933x_uart.o
86 --- /dev/null
87 +++ b/drivers/tty/serial/ar933x_uart.c
88 @@ -0,0 +1,688 @@
89 +/*
90 + * Atheros AR933X SoC built-in UART driver
91 + *
92 + * Copyright (C) 2011 Gabor Juhos <juhosg@openwrt.org>
93 + *
94 + * Based on drivers/char/serial.c, by Linus Torvalds, Theodore Ts'o.
95 + *
96 + * This program is free software; you can redistribute it and/or modify it
97 + * under the terms of the GNU General Public License version 2 as published
98 + * by the Free Software Foundation.
99 + */
100 +
101 +#include <linux/module.h>
102 +#include <linux/ioport.h>
103 +#include <linux/init.h>
104 +#include <linux/console.h>
105 +#include <linux/sysrq.h>
106 +#include <linux/delay.h>
107 +#include <linux/platform_device.h>
108 +#include <linux/tty.h>
109 +#include <linux/tty_flip.h>
110 +#include <linux/serial_core.h>
111 +#include <linux/serial.h>
112 +#include <linux/slab.h>
113 +#include <linux/io.h>
114 +#include <linux/irq.h>
115 +
116 +#include <asm/mach-ath79/ar933x_uart.h>
117 +#include <asm/mach-ath79/ar933x_uart_platform.h>
118 +
119 +#define DRIVER_NAME "ar933x-uart"
120 +
121 +#define AR933X_DUMMY_STATUS_RD 0x01
122 +
123 +static struct uart_driver ar933x_uart_driver;
124 +
125 +struct ar933x_uart_port {
126 + struct uart_port port;
127 + unsigned int ier; /* shadow Interrupt Enable Register */
128 +};
129 +
130 +static inline unsigned int ar933x_uart_read(struct ar933x_uart_port *up,
131 + int offset)
132 +{
133 + return readl(up->port.membase + offset);
134 +}
135 +
136 +static inline void ar933x_uart_write(struct ar933x_uart_port *up,
137 + int offset, unsigned int value)
138 +{
139 + writel(value, up->port.membase + offset);
140 +}
141 +
142 +static inline void ar933x_uart_rmw(struct ar933x_uart_port *up,
143 + unsigned int offset,
144 + unsigned int mask,
145 + unsigned int val)
146 +{
147 + unsigned int t;
148 +
149 + t = ar933x_uart_read(up, offset);
150 + t &= ~mask;
151 + t |= val;
152 + ar933x_uart_write(up, offset, t);
153 +}
154 +
155 +static inline void ar933x_uart_rmw_set(struct ar933x_uart_port *up,
156 + unsigned int offset,
157 + unsigned int val)
158 +{
159 + ar933x_uart_rmw(up, offset, 0, val);
160 +}
161 +
162 +static inline void ar933x_uart_rmw_clear(struct ar933x_uart_port *up,
163 + unsigned int offset,
164 + unsigned int val)
165 +{
166 + ar933x_uart_rmw(up, offset, val, 0);
167 +}
168 +
169 +static inline void ar933x_uart_start_tx_interrupt(struct ar933x_uart_port *up)
170 +{
171 + up->ier |= AR933X_UART_INT_TX_EMPTY;
172 + ar933x_uart_write(up, AR933X_UART_INT_EN_REG, up->ier);
173 +}
174 +
175 +static inline void ar933x_uart_stop_tx_interrupt(struct ar933x_uart_port *up)
176 +{
177 + up->ier &= ~AR933X_UART_INT_TX_EMPTY;
178 + ar933x_uart_write(up, AR933X_UART_INT_EN_REG, up->ier);
179 +}
180 +
181 +static inline void ar933x_uart_putc(struct ar933x_uart_port *up, int ch)
182 +{
183 + unsigned int rdata;
184 +
185 + rdata = ch & AR933X_UART_DATA_TX_RX_MASK;
186 + rdata |= AR933X_UART_DATA_TX_CSR;
187 + ar933x_uart_write(up, AR933X_UART_DATA_REG, rdata);
188 +}
189 +
190 +static unsigned int ar933x_uart_tx_empty(struct uart_port *port)
191 +{
192 + struct ar933x_uart_port *up = (struct ar933x_uart_port *) port;
193 + unsigned long flags;
194 + unsigned int rdata;
195 +
196 + spin_lock_irqsave(&up->port.lock, flags);
197 + rdata = ar933x_uart_read(up, AR933X_UART_DATA_REG);
198 + spin_unlock_irqrestore(&up->port.lock, flags);
199 +
200 + return (rdata & AR933X_UART_DATA_TX_CSR) ? 0 : TIOCSER_TEMT;
201 +}
202 +
203 +static unsigned int ar933x_uart_get_mctrl(struct uart_port *port)
204 +{
205 + return TIOCM_CAR;
206 +}
207 +
208 +static void ar933x_uart_set_mctrl(struct uart_port *port, unsigned int mctrl)
209 +{
210 +}
211 +
212 +static void ar933x_uart_start_tx(struct uart_port *port)
213 +{
214 + struct ar933x_uart_port *up = (struct ar933x_uart_port *) port;
215 +
216 + ar933x_uart_start_tx_interrupt(up);
217 +}
218 +
219 +static void ar933x_uart_stop_tx(struct uart_port *port)
220 +{
221 + struct ar933x_uart_port *up = (struct ar933x_uart_port *) port;
222 +
223 + ar933x_uart_stop_tx_interrupt(up);
224 +}
225 +
226 +static void ar933x_uart_stop_rx(struct uart_port *port)
227 +{
228 + struct ar933x_uart_port *up = (struct ar933x_uart_port *) port;
229 +
230 + up->ier &= ~AR933X_UART_INT_RX_VALID;
231 + ar933x_uart_write(up, AR933X_UART_INT_EN_REG, up->ier);
232 +}
233 +
234 +static void ar933x_uart_break_ctl(struct uart_port *port, int break_state)
235 +{
236 + struct ar933x_uart_port *up = (struct ar933x_uart_port *) port;
237 + unsigned long flags;
238 +
239 + spin_lock_irqsave(&up->port.lock, flags);
240 + if (break_state == -1)
241 + ar933x_uart_rmw_set(up, AR933X_UART_CS_REG,
242 + AR933X_UART_CS_TX_BREAK);
243 + else
244 + ar933x_uart_rmw_clear(up, AR933X_UART_CS_REG,
245 + AR933X_UART_CS_TX_BREAK);
246 + spin_unlock_irqrestore(&up->port.lock, flags);
247 +}
248 +
249 +static void ar933x_uart_enable_ms(struct uart_port *port)
250 +{
251 +}
252 +
253 +static void ar933x_uart_set_termios(struct uart_port *port,
254 + struct ktermios *new,
255 + struct ktermios *old)
256 +{
257 + struct ar933x_uart_port *up = (struct ar933x_uart_port *) port;
258 + unsigned int cs;
259 + unsigned long flags;
260 + unsigned int baud, scale;
261 +
262 + /* Only CS8 is supported */
263 + new->c_cflag &= ~CSIZE;
264 + new->c_cflag |= CS8;
265 +
266 + /* Only one stop bit is supported */
267 + new->c_cflag &= ~CSTOPB;
268 +
269 + cs = 0;
270 + if (new->c_cflag & PARENB) {
271 + if (!(new->c_cflag & PARODD))
272 + cs |= AR933X_UART_CS_PARITY_EVEN;
273 + else
274 + cs |= AR933X_UART_CS_PARITY_ODD;
275 + } else {
276 + cs |= AR933X_UART_CS_PARITY_NONE;
277 + }
278 +
279 + /* Mark/space parity is not supported */
280 + new->c_cflag &= ~CMSPAR;
281 +
282 + baud = uart_get_baud_rate(port, new, old, 0, port->uartclk / 16);
283 + scale = (port->uartclk / (16 * baud)) - 1;
284 +
285 + /*
286 + * Ok, we're now changing the port state. Do it with
287 + * interrupts disabled.
288 + */
289 + spin_lock_irqsave(&up->port.lock, flags);
290 +
291 + /* Update the per-port timeout. */
292 + uart_update_timeout(port, new->c_cflag, baud);
293 +
294 + up->port.ignore_status_mask = 0;
295 +
296 + /* ignore all characters if CREAD is not set */
297 + if ((new->c_cflag & CREAD) == 0)
298 + up->port.ignore_status_mask |= AR933X_DUMMY_STATUS_RD;
299 +
300 + ar933x_uart_write(up, AR933X_UART_CLOCK_REG,
301 + scale << AR933X_UART_CLOCK_SCALE_S | 8192);
302 +
303 + /* setup configuration register */
304 + ar933x_uart_rmw(up, AR933X_UART_CS_REG, AR933X_UART_CS_PARITY_M, cs);
305 +
306 + /* enable host interrupt */
307 + ar933x_uart_rmw_set(up, AR933X_UART_CS_REG,
308 + AR933X_UART_CS_HOST_INT_EN);
309 +
310 + spin_unlock_irqrestore(&up->port.lock, flags);
311 +
312 + if (tty_termios_baud_rate(new))
313 + tty_termios_encode_baud_rate(new, baud, baud);
314 +}
315 +
316 +static void ar933x_uart_rx_chars(struct ar933x_uart_port *up)
317 +{
318 + struct tty_struct *tty;
319 + int max_count = 256;
320 +
321 + tty = tty_port_tty_get(&up->port.state->port);
322 + do {
323 + unsigned int rdata;
324 + unsigned char ch;
325 +
326 + rdata = ar933x_uart_read(up, AR933X_UART_DATA_REG);
327 + if ((rdata & AR933X_UART_DATA_RX_CSR) == 0)
328 + break;
329 +
330 + /* remove the character from the FIFO */
331 + ar933x_uart_write(up, AR933X_UART_DATA_REG,
332 + AR933X_UART_DATA_RX_CSR);
333 +
334 + if (!tty) {
335 + /* discard the data if no tty available */
336 + continue;
337 + }
338 +
339 + up->port.icount.rx++;
340 + ch = rdata & AR933X_UART_DATA_TX_RX_MASK;
341 +
342 + if (uart_handle_sysrq_char(&up->port, ch))
343 + continue;
344 +
345 + if ((up->port.ignore_status_mask & AR933X_DUMMY_STATUS_RD) == 0)
346 + tty_insert_flip_char(tty, ch, TTY_NORMAL);
347 + } while (max_count-- > 0);
348 +
349 + if (tty) {
350 + tty_flip_buffer_push(tty);
351 + tty_kref_put(tty);
352 + }
353 +}
354 +
355 +static void ar933x_uart_tx_chars(struct ar933x_uart_port *up)
356 +{
357 + struct circ_buf *xmit = &up->port.state->xmit;
358 + int count;
359 +
360 + if (uart_tx_stopped(&up->port))
361 + return;
362 +
363 + count = up->port.fifosize;
364 + do {
365 + unsigned int rdata;
366 +
367 + rdata = ar933x_uart_read(up, AR933X_UART_DATA_REG);
368 + if ((rdata & AR933X_UART_DATA_TX_CSR) == 0)
369 + break;
370 +
371 + if (up->port.x_char) {
372 + ar933x_uart_putc(up, up->port.x_char);
373 + up->port.icount.tx++;
374 + up->port.x_char = 0;
375 + continue;
376 + }
377 +
378 + if (uart_circ_empty(xmit))
379 + break;
380 +
381 + ar933x_uart_putc(up, xmit->buf[xmit->tail]);
382 +
383 + xmit->tail = (xmit->tail + 1) & (UART_XMIT_SIZE - 1);
384 + up->port.icount.tx++;
385 + } while (--count > 0);
386 +
387 + if (uart_circ_chars_pending(xmit) < WAKEUP_CHARS)
388 + uart_write_wakeup(&up->port);
389 +
390 + if (!uart_circ_empty(xmit))
391 + ar933x_uart_start_tx_interrupt(up);
392 +}
393 +
394 +static irqreturn_t ar933x_uart_interrupt(int irq, void *dev_id)
395 +{
396 + struct ar933x_uart_port *up = dev_id;
397 + unsigned int status;
398 +
399 + status = ar933x_uart_read(up, AR933X_UART_CS_REG);
400 + if ((status & AR933X_UART_CS_HOST_INT) == 0)
401 + return IRQ_NONE;
402 +
403 + spin_lock(&up->port.lock);
404 +
405 + status = ar933x_uart_read(up, AR933X_UART_INT_REG);
406 + status &= ar933x_uart_read(up, AR933X_UART_INT_EN_REG);
407 +
408 + if (status & AR933X_UART_INT_RX_VALID) {
409 + ar933x_uart_write(up, AR933X_UART_INT_REG,
410 + AR933X_UART_INT_RX_VALID);
411 + ar933x_uart_rx_chars(up);
412 + }
413 +
414 + if (status & AR933X_UART_INT_TX_EMPTY) {
415 + ar933x_uart_write(up, AR933X_UART_INT_REG,
416 + AR933X_UART_INT_TX_EMPTY);
417 + ar933x_uart_stop_tx_interrupt(up);
418 + ar933x_uart_tx_chars(up);
419 + }
420 +
421 + spin_unlock(&up->port.lock);
422 +
423 + return IRQ_HANDLED;
424 +}
425 +
426 +static int ar933x_uart_startup(struct uart_port *port)
427 +{
428 + struct ar933x_uart_port *up = (struct ar933x_uart_port *) port;
429 + unsigned long flags;
430 + int ret;
431 +
432 + ret = request_irq(up->port.irq, ar933x_uart_interrupt,
433 + up->port.irqflags, dev_name(up->port.dev), up);
434 + if (ret)
435 + return ret;
436 +
437 + spin_lock_irqsave(&up->port.lock, flags);
438 +
439 + /* Enable HOST interrupts */
440 + ar933x_uart_rmw_set(up, AR933X_UART_CS_REG,
441 + AR933X_UART_CS_HOST_INT_EN);
442 +
443 + /* Enable RX interrupts */
444 + up->ier = AR933X_UART_INT_RX_VALID;
445 + ar933x_uart_write(up, AR933X_UART_INT_EN_REG, up->ier);
446 +
447 + spin_unlock_irqrestore(&up->port.lock, flags);
448 +
449 + return 0;
450 +}
451 +
452 +static void ar933x_uart_shutdown(struct uart_port *port)
453 +{
454 + struct ar933x_uart_port *up = (struct ar933x_uart_port *) port;
455 +
456 + /* Disable all interrupts */
457 + up->ier = 0;
458 + ar933x_uart_write(up, AR933X_UART_INT_EN_REG, up->ier);
459 +
460 + /* Disable break condition */
461 + ar933x_uart_rmw_clear(up, AR933X_UART_CS_REG,
462 + AR933X_UART_CS_TX_BREAK);
463 +
464 + free_irq(up->port.irq, up);
465 +}
466 +
467 +static const char *ar933x_uart_type(struct uart_port *port)
468 +{
469 + return (port->type == PORT_AR933X) ? "AR933X UART" : NULL;
470 +}
471 +
472 +static void ar933x_uart_release_port(struct uart_port *port)
473 +{
474 + /* Nothing to release ... */
475 +}
476 +
477 +static int ar933x_uart_request_port(struct uart_port *port)
478 +{
479 + /* UARTs always present */
480 + return 0;
481 +}
482 +
483 +static void ar933x_uart_config_port(struct uart_port *port, int flags)
484 +{
485 + if (flags & UART_CONFIG_TYPE)
486 + port->type = PORT_AR933X;
487 +}
488 +
489 +static int ar933x_uart_verify_port(struct uart_port *port,
490 + struct serial_struct *ser)
491 +{
492 + if (ser->type != PORT_UNKNOWN &&
493 + ser->type != PORT_AR933X)
494 + return -EINVAL;
495 +
496 + if (ser->irq < 0 || ser->irq >= NR_IRQS)
497 + return -EINVAL;
498 +
499 + if (ser->baud_base < 28800)
500 + return -EINVAL;
501 +
502 + return 0;
503 +}
504 +
505 +static struct uart_ops ar933x_uart_ops = {
506 + .tx_empty = ar933x_uart_tx_empty,
507 + .set_mctrl = ar933x_uart_set_mctrl,
508 + .get_mctrl = ar933x_uart_get_mctrl,
509 + .stop_tx = ar933x_uart_stop_tx,
510 + .start_tx = ar933x_uart_start_tx,
511 + .stop_rx = ar933x_uart_stop_rx,
512 + .enable_ms = ar933x_uart_enable_ms,
513 + .break_ctl = ar933x_uart_break_ctl,
514 + .startup = ar933x_uart_startup,
515 + .shutdown = ar933x_uart_shutdown,
516 + .set_termios = ar933x_uart_set_termios,
517 + .type = ar933x_uart_type,
518 + .release_port = ar933x_uart_release_port,
519 + .request_port = ar933x_uart_request_port,
520 + .config_port = ar933x_uart_config_port,
521 + .verify_port = ar933x_uart_verify_port,
522 +};
523 +
524 +#ifdef CONFIG_SERIAL_AR933X_CONSOLE
525 +
526 +static struct ar933x_uart_port *
527 +ar933x_console_ports[CONFIG_SERIAL_AR933X_NR_UARTS];
528 +
529 +static void ar933x_uart_wait_xmitr(struct ar933x_uart_port *up)
530 +{
531 + unsigned int status;
532 + unsigned int timeout = 60000;
533 +
534 + /* Wait up to 60ms for the character(s) to be sent. */
535 + do {
536 + status = ar933x_uart_read(up, AR933X_UART_DATA_REG);
537 + if (--timeout == 0)
538 + break;
539 + udelay(1);
540 + } while ((status & AR933X_UART_DATA_TX_CSR) == 0);
541 +}
542 +
543 +static void ar933x_uart_console_putchar(struct uart_port *port, int ch)
544 +{
545 + struct ar933x_uart_port *up = (struct ar933x_uart_port *) port;
546 +
547 + ar933x_uart_wait_xmitr(up);
548 + ar933x_uart_putc(up, ch);
549 +}
550 +
551 +static void ar933x_uart_console_write(struct console *co, const char *s,
552 + unsigned int count)
553 +{
554 + struct ar933x_uart_port *up = ar933x_console_ports[co->index];
555 + unsigned long flags;
556 + unsigned int int_en;
557 + int locked = 1;
558 +
559 + local_irq_save(flags);
560 +
561 + if (up->port.sysrq)
562 + locked = 0;
563 + else if (oops_in_progress)
564 + locked = spin_trylock(&up->port.lock);
565 + else
566 + spin_lock(&up->port.lock);
567 +
568 + /*
569 + * First save the IER then disable the interrupts
570 + */
571 + int_en = ar933x_uart_read(up, AR933X_UART_INT_EN_REG);
572 + ar933x_uart_write(up, AR933X_UART_INT_EN_REG, 0);
573 +
574 + uart_console_write(&up->port, s, count, ar933x_uart_console_putchar);
575 +
576 + /*
577 + * Finally, wait for transmitter to become empty
578 + * and restore the IER
579 + */
580 + ar933x_uart_wait_xmitr(up);
581 + ar933x_uart_write(up, AR933X_UART_INT_EN_REG, int_en);
582 +
583 + ar933x_uart_write(up, AR933X_UART_INT_REG, AR933X_UART_INT_ALLINTS);
584 +
585 + if (locked)
586 + spin_unlock(&up->port.lock);
587 +
588 + local_irq_restore(flags);
589 +}
590 +
591 +static int ar933x_uart_console_setup(struct console *co, char *options)
592 +{
593 + struct ar933x_uart_port *up;
594 + int baud = 115200;
595 + int bits = 8;
596 + int parity = 'n';
597 + int flow = 'n';
598 +
599 + if (co->index < 0 || co->index >= CONFIG_SERIAL_AR933X_NR_UARTS)
600 + return -EINVAL;
601 +
602 + up = ar933x_console_ports[co->index];
603 + if (!up)
604 + return -ENODEV;
605 +
606 + if (options)
607 + uart_parse_options(options, &baud, &parity, &bits, &flow);
608 +
609 + return uart_set_options(&up->port, co, baud, parity, bits, flow);
610 +}
611 +
612 +static struct console ar933x_uart_console = {
613 + .name = "ttyATH",
614 + .write = ar933x_uart_console_write,
615 + .device = uart_console_device,
616 + .setup = ar933x_uart_console_setup,
617 + .flags = CON_PRINTBUFFER,
618 + .index = -1,
619 + .data = &ar933x_uart_driver,
620 +};
621 +
622 +static void ar933x_uart_add_console_port(struct ar933x_uart_port *up)
623 +{
624 + ar933x_console_ports[up->port.line] = up;
625 +}
626 +
627 +#define AR933X_SERIAL_CONSOLE (&ar933x_uart_console)
628 +
629 +#else
630 +
631 +static inline void ar933x_uart_add_console_port(struct ar933x_uart_port *up) {}
632 +
633 +#define AR933X_SERIAL_CONSOLE NULL
634 +
635 +#endif /* CONFIG_SERIAL_AR933X_CONSOLE */
636 +
637 +static struct uart_driver ar933x_uart_driver = {
638 + .owner = THIS_MODULE,
639 + .driver_name = DRIVER_NAME,
640 + .dev_name = "ttyATH",
641 + .nr = CONFIG_SERIAL_AR933X_NR_UARTS,
642 + .cons = AR933X_SERIAL_CONSOLE,
643 +};
644 +
645 +static int __devinit ar933x_uart_probe(struct platform_device *pdev)
646 +{
647 + struct ar933x_uart_platform_data *pdata;
648 + struct ar933x_uart_port *up;
649 + struct uart_port *port;
650 + struct resource *mem_res;
651 + struct resource *irq_res;
652 + int id;
653 + int ret;
654 +
655 + pdata = pdev->dev.platform_data;
656 + if (!pdata)
657 + return -EINVAL;
658 +
659 + id = pdev->id;
660 + if (id == -1)
661 + id = 0;
662 +
663 + if (id > CONFIG_SERIAL_AR933X_NR_UARTS)
664 + return -EINVAL;
665 +
666 + mem_res = platform_get_resource(pdev, IORESOURCE_MEM, 0);
667 + if (!mem_res) {
668 + dev_err(&pdev->dev, "no MEM resource\n");
669 + return -EINVAL;
670 + }
671 +
672 + irq_res = platform_get_resource(pdev, IORESOURCE_IRQ, 0);
673 + if (!irq_res) {
674 + dev_err(&pdev->dev, "no IRQ resource\n");
675 + return -EINVAL;
676 + }
677 +
678 + up = kzalloc(sizeof(struct ar933x_uart_port), GFP_KERNEL);
679 + if (!up)
680 + return -ENOMEM;
681 +
682 + port = &up->port;
683 + port->mapbase = mem_res->start;
684 +
685 + port->membase = ioremap(mem_res->start, AR933X_UART_REGS_SIZE);
686 + if (!port->membase) {
687 + ret = -ENOMEM;
688 + goto err_free_up;
689 + }
690 +
691 + port->line = id;
692 + port->irq = irq_res->start;
693 + port->dev = &pdev->dev;
694 + port->type = PORT_AR933X;
695 + port->iotype = UPIO_MEM32;
696 + port->uartclk = pdata->uartclk;
697 +
698 + port->regshift = 2;
699 + port->fifosize = AR933X_UART_FIFO_SIZE;
700 + port->ops = &ar933x_uart_ops;
701 +
702 + ar933x_uart_add_console_port(up);
703 +
704 + ret = uart_add_one_port(&ar933x_uart_driver, &up->port);
705 + if (ret)
706 + goto err_unmap;
707 +
708 + platform_set_drvdata(pdev, up);
709 + return 0;
710 +
711 +err_unmap:
712 + iounmap(up->port.membase);
713 +err_free_up:
714 + kfree(up);
715 + return ret;
716 +}
717 +
718 +static int __devexit ar933x_uart_remove(struct platform_device *pdev)
719 +{
720 + struct ar933x_uart_port *up;
721 +
722 + up = platform_get_drvdata(pdev);
723 + platform_set_drvdata(pdev, NULL);
724 +
725 + if (up) {
726 + uart_remove_one_port(&ar933x_uart_driver, &up->port);
727 + iounmap(up->port.membase);
728 + kfree(up);
729 + }
730 +
731 + return 0;
732 +}
733 +
734 +static struct platform_driver ar933x_uart_platform_driver = {
735 + .probe = ar933x_uart_probe,
736 + .remove = __devexit_p(ar933x_uart_remove),
737 + .driver = {
738 + .name = DRIVER_NAME,
739 + .owner = THIS_MODULE,
740 + },
741 +};
742 +
743 +static int __init ar933x_uart_init(void)
744 +{
745 + int ret;
746 +
747 + ar933x_uart_driver.nr = CONFIG_SERIAL_AR933X_NR_UARTS;
748 + ret = uart_register_driver(&ar933x_uart_driver);
749 + if (ret)
750 + goto err_out;
751 +
752 + ret = platform_driver_register(&ar933x_uart_platform_driver);
753 + if (ret)
754 + goto err_unregister_uart_driver;
755 +
756 + return 0;
757 +
758 +err_unregister_uart_driver:
759 + uart_unregister_driver(&ar933x_uart_driver);
760 +err_out:
761 + return ret;
762 +}
763 +
764 +static void __exit ar933x_uart_exit(void)
765 +{
766 + platform_driver_unregister(&ar933x_uart_platform_driver);
767 + uart_unregister_driver(&ar933x_uart_driver);
768 +}
769 +
770 +module_init(ar933x_uart_init);
771 +module_exit(ar933x_uart_exit);
772 +
773 +MODULE_DESCRIPTION("Atheros AR933X UART driver");
774 +MODULE_AUTHOR("Gabor Juhos <juhosg@openwrt.org>");
775 +MODULE_LICENSE("GPL v2");
776 +MODULE_ALIAS("platform:" DRIVER_NAME);
777 --- a/include/linux/serial_core.h
778 +++ b/include/linux/serial_core.h
779 @@ -207,6 +207,10 @@
780 /* Xilinx PSS UART */
781 #define PORT_XUARTPS 98
782
783 +/* Atheros AR933X SoC */
784 +#define PORT_AR933X 99
785 +
786 +
787 #ifdef __KERNEL__
788
789 #include <linux/compiler.h>
This page took 0.076742 seconds and 5 git commands to generate.