blob: 10743297a001ed524046ac79c0e6e238f61ece45 [file] [log] [blame]
Andrew Victor1e6c9c22006-01-10 16:59:27 +00001/*
Haavard Skinnemoen7192f922006-10-04 16:02:05 +02002 * Driver for Atmel AT91 / AT32 Serial ports
Andrew Victor1e6c9c22006-01-10 16:59:27 +00003 * Copyright (C) 2003 Rick Bronson
4 *
5 * Based on drivers/char/serial_sa1100.c, by Deep Blue Solutions Ltd.
6 * Based on drivers/char/serial.c, by Linus Torvalds, Theodore Ts'o.
7 *
Chip Coldwella6670612008-02-08 04:21:06 -08008 * DMA support added by Chip Coldwell.
9 *
Andrew Victor1e6c9c22006-01-10 16:59:27 +000010 * This program is free software; you can redistribute it and/or modify
11 * it under the terms of the GNU General Public License as published by
12 * the Free Software Foundation; either version 2 of the License, or
13 * (at your option) any later version.
14 *
15 * This program is distributed in the hope that it will be useful,
16 * but WITHOUT ANY WARRANTY; without even the implied warranty of
17 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
18 * GNU General Public License for more details.
19 *
20 * You should have received a copy of the GNU General Public License
21 * along with this program; if not, write to the Free Software
22 * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
23 *
24 */
Andrew Victor1e6c9c22006-01-10 16:59:27 +000025#include <linux/module.h>
26#include <linux/tty.h>
27#include <linux/ioport.h>
28#include <linux/slab.h>
29#include <linux/init.h>
30#include <linux/serial.h>
Andrew Victorafefc412006-06-19 19:53:19 +010031#include <linux/clk.h>
Andrew Victor1e6c9c22006-01-10 16:59:27 +000032#include <linux/console.h>
33#include <linux/sysrq.h>
34#include <linux/tty_flip.h>
Andrew Victorafefc412006-06-19 19:53:19 +010035#include <linux/platform_device.h>
Chip Coldwella6670612008-02-08 04:21:06 -080036#include <linux/dma-mapping.h>
Andrew Victor93a3ddc2007-02-08 11:31:22 +010037#include <linux/atmel_pdc.h>
Guennadi Liakhovetskifa3218d2008-01-29 15:43:13 +010038#include <linux/atmel_serial.h>
Claudio Scordinoe8faff72010-05-03 13:31:28 +010039#include <linux/uaccess.h>
Andrew Victor1e6c9c22006-01-10 16:59:27 +000040
41#include <asm/io.h>
Peter Huewef7512e72010-06-29 19:35:39 +020042#include <asm/ioctls.h>
Andrew Victor1e6c9c22006-01-10 16:59:27 +000043
Andrew Victorafefc412006-06-19 19:53:19 +010044#include <asm/mach/serial_at91.h>
Russell Kinga09e64f2008-08-05 16:14:15 +010045#include <mach/board.h>
Andrew Victor93a3ddc2007-02-08 11:31:22 +010046
Haavard Skinnemoenacca9b82006-10-04 16:02:09 +020047#ifdef CONFIG_ARM
Russell Kinga09e64f2008-08-05 16:14:15 +010048#include <mach/cpu.h>
49#include <mach/gpio.h>
Haavard Skinnemoenacca9b82006-10-04 16:02:09 +020050#endif
Andrew Victor1e6c9c22006-01-10 16:59:27 +000051
Chip Coldwella6670612008-02-08 04:21:06 -080052#define PDC_BUFFER_SIZE 512
53/* Revisit: We should calculate this based on the actual port settings */
54#define PDC_RX_TIMEOUT (3 * 10) /* 3 bytes */
55
Haavard Skinnemoen749c4e62006-10-04 16:02:02 +020056#if defined(CONFIG_SERIAL_ATMEL_CONSOLE) && defined(CONFIG_MAGIC_SYSRQ)
Andrew Victor1e6c9c22006-01-10 16:59:27 +000057#define SUPPORT_SYSRQ
58#endif
59
60#include <linux/serial_core.h>
61
Claudio Scordinoe8faff72010-05-03 13:31:28 +010062static void atmel_start_rx(struct uart_port *port);
63static void atmel_stop_rx(struct uart_port *port);
64
Haavard Skinnemoen749c4e62006-10-04 16:02:02 +020065#ifdef CONFIG_SERIAL_ATMEL_TTYAT
Andrew Victor1e6c9c22006-01-10 16:59:27 +000066
67/* Use device name ttyAT, major 204 and minor 154-169. This is necessary if we
68 * should coexist with the 8250 driver, such as if we have an external 16C550
69 * UART. */
Haavard Skinnemoen7192f922006-10-04 16:02:05 +020070#define SERIAL_ATMEL_MAJOR 204
Andrew Victor1e6c9c22006-01-10 16:59:27 +000071#define MINOR_START 154
Haavard Skinnemoen7192f922006-10-04 16:02:05 +020072#define ATMEL_DEVICENAME "ttyAT"
Andrew Victor1e6c9c22006-01-10 16:59:27 +000073
74#else
75
76/* Use device name ttyS, major 4, minor 64-68. This is the usual serial port
77 * name, but it is legally reserved for the 8250 driver. */
Haavard Skinnemoen7192f922006-10-04 16:02:05 +020078#define SERIAL_ATMEL_MAJOR TTY_MAJOR
Andrew Victor1e6c9c22006-01-10 16:59:27 +000079#define MINOR_START 64
Haavard Skinnemoen7192f922006-10-04 16:02:05 +020080#define ATMEL_DEVICENAME "ttyS"
Andrew Victor1e6c9c22006-01-10 16:59:27 +000081
82#endif
83
Haavard Skinnemoen7192f922006-10-04 16:02:05 +020084#define ATMEL_ISR_PASS_LIMIT 256
Andrew Victor1e6c9c22006-01-10 16:59:27 +000085
Remy Bohmerb843aa22008-02-08 04:21:01 -080086/* UART registers. CR is write-only, hence no GET macro */
Haavard Skinnemoen544fc722007-02-12 00:52:15 -080087#define UART_PUT_CR(port,v) __raw_writel(v, (port)->membase + ATMEL_US_CR)
88#define UART_GET_MR(port) __raw_readl((port)->membase + ATMEL_US_MR)
89#define UART_PUT_MR(port,v) __raw_writel(v, (port)->membase + ATMEL_US_MR)
90#define UART_PUT_IER(port,v) __raw_writel(v, (port)->membase + ATMEL_US_IER)
91#define UART_PUT_IDR(port,v) __raw_writel(v, (port)->membase + ATMEL_US_IDR)
92#define UART_GET_IMR(port) __raw_readl((port)->membase + ATMEL_US_IMR)
93#define UART_GET_CSR(port) __raw_readl((port)->membase + ATMEL_US_CSR)
94#define UART_GET_CHAR(port) __raw_readl((port)->membase + ATMEL_US_RHR)
95#define UART_PUT_CHAR(port,v) __raw_writel(v, (port)->membase + ATMEL_US_THR)
96#define UART_GET_BRGR(port) __raw_readl((port)->membase + ATMEL_US_BRGR)
97#define UART_PUT_BRGR(port,v) __raw_writel(v, (port)->membase + ATMEL_US_BRGR)
98#define UART_PUT_RTOR(port,v) __raw_writel(v, (port)->membase + ATMEL_US_RTOR)
Claudio Scordinoe8faff72010-05-03 13:31:28 +010099#define UART_PUT_TTGR(port, v) __raw_writel(v, (port)->membase + ATMEL_US_TTGR)
Andrew Victor1e6c9c22006-01-10 16:59:27 +0000100
Andrew Victor1e6c9c22006-01-10 16:59:27 +0000101 /* PDC registers */
Haavard Skinnemoen544fc722007-02-12 00:52:15 -0800102#define UART_PUT_PTCR(port,v) __raw_writel(v, (port)->membase + ATMEL_PDC_PTCR)
103#define UART_GET_PTSR(port) __raw_readl((port)->membase + ATMEL_PDC_PTSR)
Andrew Victorafefc412006-06-19 19:53:19 +0100104
Haavard Skinnemoen544fc722007-02-12 00:52:15 -0800105#define UART_PUT_RPR(port,v) __raw_writel(v, (port)->membase + ATMEL_PDC_RPR)
106#define UART_GET_RPR(port) __raw_readl((port)->membase + ATMEL_PDC_RPR)
107#define UART_PUT_RCR(port,v) __raw_writel(v, (port)->membase + ATMEL_PDC_RCR)
108#define UART_PUT_RNPR(port,v) __raw_writel(v, (port)->membase + ATMEL_PDC_RNPR)
109#define UART_PUT_RNCR(port,v) __raw_writel(v, (port)->membase + ATMEL_PDC_RNCR)
Andrew Victor1e6c9c22006-01-10 16:59:27 +0000110
Haavard Skinnemoen544fc722007-02-12 00:52:15 -0800111#define UART_PUT_TPR(port,v) __raw_writel(v, (port)->membase + ATMEL_PDC_TPR)
112#define UART_PUT_TCR(port,v) __raw_writel(v, (port)->membase + ATMEL_PDC_TCR)
Marc Pignat39d4c922008-04-02 13:04:42 -0700113#define UART_GET_TCR(port) __raw_readl((port)->membase + ATMEL_PDC_TCR)
Andrew Victor1e6c9c22006-01-10 16:59:27 +0000114
Haavard Skinnemoen71f2e2b2006-10-04 16:02:07 +0200115static int (*atmel_open_hook)(struct uart_port *);
116static void (*atmel_close_hook)(struct uart_port *);
Andrew Victor1e6c9c22006-01-10 16:59:27 +0000117
Chip Coldwella6670612008-02-08 04:21:06 -0800118struct atmel_dma_buffer {
119 unsigned char *buf;
120 dma_addr_t dma_addr;
121 unsigned int dma_size;
122 unsigned int ofs;
123};
124
Remy Bohmer1ecc26b2008-02-08 04:21:05 -0800125struct atmel_uart_char {
126 u16 status;
127 u16 ch;
128};
129
130#define ATMEL_SERIAL_RINGSIZE 1024
131
Andrew Victorafefc412006-06-19 19:53:19 +0100132/*
133 * We wrap our port structure around the generic uart_port.
134 */
Haavard Skinnemoen7192f922006-10-04 16:02:05 +0200135struct atmel_uart_port {
Andrew Victorafefc412006-06-19 19:53:19 +0100136 struct uart_port uart; /* uart */
137 struct clk *clk; /* uart clock */
Anti Sullinf05596d2008-09-22 13:57:54 -0700138 int may_wakeup; /* cached value of device_may_wakeup for times we need to disable it */
139 u32 backup_imr; /* IMR saved during suspend */
Haavard Skinnemoen9e6077b2007-07-15 23:40:36 -0700140 int break_active; /* break being received */
Remy Bohmer1ecc26b2008-02-08 04:21:05 -0800141
Chip Coldwella6670612008-02-08 04:21:06 -0800142 short use_dma_rx; /* enable PDC receiver */
143 short pdc_rx_idx; /* current PDC RX buffer */
144 struct atmel_dma_buffer pdc_rx[2]; /* PDC receier */
145
146 short use_dma_tx; /* enable PDC transmitter */
147 struct atmel_dma_buffer pdc_tx; /* PDC transmitter */
148
Remy Bohmer1ecc26b2008-02-08 04:21:05 -0800149 struct tasklet_struct tasklet;
150 unsigned int irq_status;
151 unsigned int irq_status_prev;
152
153 struct circ_buf rx_ring;
Claudio Scordinoe8faff72010-05-03 13:31:28 +0100154
155 struct serial_rs485 rs485; /* rs485 settings */
156 unsigned int tx_done_mask;
Andrew Victorafefc412006-06-19 19:53:19 +0100157};
158
Haavard Skinnemoen7192f922006-10-04 16:02:05 +0200159static struct atmel_uart_port atmel_ports[ATMEL_MAX_UART];
Nicolas Ferre4cbf9f42011-10-12 18:06:59 +0200160static unsigned long atmel_ports_in_use;
Andrew Victorafefc412006-06-19 19:53:19 +0100161
Andrew Victor1e6c9c22006-01-10 16:59:27 +0000162#ifdef SUPPORT_SYSRQ
Haavard Skinnemoen7192f922006-10-04 16:02:05 +0200163static struct console atmel_console;
Andrew Victor1e6c9c22006-01-10 16:59:27 +0000164#endif
165
Haavard Skinnemoenc811ab82008-02-08 04:21:08 -0800166static inline struct atmel_uart_port *
167to_atmel_uart_port(struct uart_port *uart)
168{
169 return container_of(uart, struct atmel_uart_port, uart);
170}
171
Chip Coldwella6670612008-02-08 04:21:06 -0800172#ifdef CONFIG_SERIAL_ATMEL_PDC
173static bool atmel_use_dma_rx(struct uart_port *port)
174{
Haavard Skinnemoenc811ab82008-02-08 04:21:08 -0800175 struct atmel_uart_port *atmel_port = to_atmel_uart_port(port);
Chip Coldwella6670612008-02-08 04:21:06 -0800176
177 return atmel_port->use_dma_rx;
178}
179
180static bool atmel_use_dma_tx(struct uart_port *port)
181{
Haavard Skinnemoenc811ab82008-02-08 04:21:08 -0800182 struct atmel_uart_port *atmel_port = to_atmel_uart_port(port);
Chip Coldwella6670612008-02-08 04:21:06 -0800183
184 return atmel_port->use_dma_tx;
185}
186#else
187static bool atmel_use_dma_rx(struct uart_port *port)
188{
189 return false;
190}
191
192static bool atmel_use_dma_tx(struct uart_port *port)
193{
194 return false;
195}
196#endif
197
Claudio Scordinoe8faff72010-05-03 13:31:28 +0100198/* Enable or disable the rs485 support */
199void atmel_config_rs485(struct uart_port *port, struct serial_rs485 *rs485conf)
200{
201 struct atmel_uart_port *atmel_port = to_atmel_uart_port(port);
202 unsigned int mode;
203
204 spin_lock(&port->lock);
205
206 /* Disable interrupts */
207 UART_PUT_IDR(port, atmel_port->tx_done_mask);
208
209 mode = UART_GET_MR(port);
210
211 /* Resetting serial mode to RS232 (0x0) */
212 mode &= ~ATMEL_US_USMODE;
213
214 atmel_port->rs485 = *rs485conf;
215
216 if (rs485conf->flags & SER_RS485_ENABLED) {
217 dev_dbg(port->dev, "Setting UART to RS485\n");
218 atmel_port->tx_done_mask = ATMEL_US_TXEMPTY;
Claudio Scordino1b633182010-07-20 15:26:47 -0700219 if (rs485conf->flags & SER_RS485_RTS_AFTER_SEND)
220 UART_PUT_TTGR(port, rs485conf->delay_rts_after_send);
Claudio Scordinoe8faff72010-05-03 13:31:28 +0100221 mode |= ATMEL_US_USMODE_RS485;
222 } else {
223 dev_dbg(port->dev, "Setting UART to RS232\n");
224 if (atmel_use_dma_tx(port))
225 atmel_port->tx_done_mask = ATMEL_US_ENDTX |
226 ATMEL_US_TXBUFE;
227 else
228 atmel_port->tx_done_mask = ATMEL_US_TXRDY;
229 }
230 UART_PUT_MR(port, mode);
231
232 /* Enable interrupts */
233 UART_PUT_IER(port, atmel_port->tx_done_mask);
234
235 spin_unlock(&port->lock);
236
237}
238
Andrew Victor1e6c9c22006-01-10 16:59:27 +0000239/*
240 * Return TIOCSER_TEMT when transmitter FIFO and Shift register is empty.
241 */
Haavard Skinnemoen7192f922006-10-04 16:02:05 +0200242static u_int atmel_tx_empty(struct uart_port *port)
Andrew Victor1e6c9c22006-01-10 16:59:27 +0000243{
Haavard Skinnemoen7192f922006-10-04 16:02:05 +0200244 return (UART_GET_CSR(port) & ATMEL_US_TXEMPTY) ? TIOCSER_TEMT : 0;
Andrew Victor1e6c9c22006-01-10 16:59:27 +0000245}
246
247/*
248 * Set state of the modem control output lines
249 */
Haavard Skinnemoen7192f922006-10-04 16:02:05 +0200250static void atmel_set_mctrl(struct uart_port *port, u_int mctrl)
Andrew Victor1e6c9c22006-01-10 16:59:27 +0000251{
252 unsigned int control = 0;
Andrew Victorafefc412006-06-19 19:53:19 +0100253 unsigned int mode;
Claudio Scordinoe8faff72010-05-03 13:31:28 +0100254 struct atmel_uart_port *atmel_port = to_atmel_uart_port(port);
Andrew Victor1e6c9c22006-01-10 16:59:27 +0000255
Andrew Victorc2f5ccf2006-12-04 09:37:20 +0100256#ifdef CONFIG_ARCH_AT91RM9200
Andrew Victor79da7a62006-11-30 11:58:14 +0100257 if (cpu_is_at91rm9200()) {
Andrew Victorafefc412006-06-19 19:53:19 +0100258 /*
Remy Bohmerb843aa22008-02-08 04:21:01 -0800259 * AT91RM9200 Errata #39: RTS0 is not internally connected
260 * to PA21. We need to drive the pin manually.
Andrew Victorafefc412006-06-19 19:53:19 +0100261 */
Andrew Victor72729912006-09-27 09:44:11 +0100262 if (port->mapbase == AT91RM9200_BASE_US0) {
Andrew Victorafefc412006-06-19 19:53:19 +0100263 if (mctrl & TIOCM_RTS)
Andrew Victor20e65272006-07-05 14:31:13 +0100264 at91_set_gpio_value(AT91_PIN_PA21, 0);
Andrew Victorafefc412006-06-19 19:53:19 +0100265 else
Andrew Victor20e65272006-07-05 14:31:13 +0100266 at91_set_gpio_value(AT91_PIN_PA21, 1);
Andrew Victorafefc412006-06-19 19:53:19 +0100267 }
Andrew Victor1e6c9c22006-01-10 16:59:27 +0000268 }
Haavard Skinnemoenacca9b82006-10-04 16:02:09 +0200269#endif
Andrew Victor1e6c9c22006-01-10 16:59:27 +0000270
271 if (mctrl & TIOCM_RTS)
Haavard Skinnemoen7192f922006-10-04 16:02:05 +0200272 control |= ATMEL_US_RTSEN;
Andrew Victor1e6c9c22006-01-10 16:59:27 +0000273 else
Haavard Skinnemoen7192f922006-10-04 16:02:05 +0200274 control |= ATMEL_US_RTSDIS;
Andrew Victor1e6c9c22006-01-10 16:59:27 +0000275
276 if (mctrl & TIOCM_DTR)
Haavard Skinnemoen7192f922006-10-04 16:02:05 +0200277 control |= ATMEL_US_DTREN;
Andrew Victor1e6c9c22006-01-10 16:59:27 +0000278 else
Haavard Skinnemoen7192f922006-10-04 16:02:05 +0200279 control |= ATMEL_US_DTRDIS;
Andrew Victor1e6c9c22006-01-10 16:59:27 +0000280
Andrew Victorafefc412006-06-19 19:53:19 +0100281 UART_PUT_CR(port, control);
282
283 /* Local loopback mode? */
Haavard Skinnemoen7192f922006-10-04 16:02:05 +0200284 mode = UART_GET_MR(port) & ~ATMEL_US_CHMODE;
Andrew Victorafefc412006-06-19 19:53:19 +0100285 if (mctrl & TIOCM_LOOP)
Haavard Skinnemoen7192f922006-10-04 16:02:05 +0200286 mode |= ATMEL_US_CHMODE_LOC_LOOP;
Andrew Victorafefc412006-06-19 19:53:19 +0100287 else
Haavard Skinnemoen7192f922006-10-04 16:02:05 +0200288 mode |= ATMEL_US_CHMODE_NORMAL;
Claudio Scordinoe8faff72010-05-03 13:31:28 +0100289
290 /* Resetting serial mode to RS232 (0x0) */
291 mode &= ~ATMEL_US_USMODE;
292
293 if (atmel_port->rs485.flags & SER_RS485_ENABLED) {
294 dev_dbg(port->dev, "Setting UART to RS485\n");
Claudio Scordino1b633182010-07-20 15:26:47 -0700295 if (atmel_port->rs485.flags & SER_RS485_RTS_AFTER_SEND)
296 UART_PUT_TTGR(port,
297 atmel_port->rs485.delay_rts_after_send);
Claudio Scordinoe8faff72010-05-03 13:31:28 +0100298 mode |= ATMEL_US_USMODE_RS485;
299 } else {
300 dev_dbg(port->dev, "Setting UART to RS232\n");
301 }
Andrew Victorafefc412006-06-19 19:53:19 +0100302 UART_PUT_MR(port, mode);
Andrew Victor1e6c9c22006-01-10 16:59:27 +0000303}
304
305/*
306 * Get state of the modem control input lines
307 */
Haavard Skinnemoen7192f922006-10-04 16:02:05 +0200308static u_int atmel_get_mctrl(struct uart_port *port)
Andrew Victor1e6c9c22006-01-10 16:59:27 +0000309{
310 unsigned int status, ret = 0;
311
312 status = UART_GET_CSR(port);
313
314 /*
315 * The control signals are active low.
316 */
Haavard Skinnemoen7192f922006-10-04 16:02:05 +0200317 if (!(status & ATMEL_US_DCD))
Andrew Victor1e6c9c22006-01-10 16:59:27 +0000318 ret |= TIOCM_CD;
Haavard Skinnemoen7192f922006-10-04 16:02:05 +0200319 if (!(status & ATMEL_US_CTS))
Andrew Victor1e6c9c22006-01-10 16:59:27 +0000320 ret |= TIOCM_CTS;
Haavard Skinnemoen7192f922006-10-04 16:02:05 +0200321 if (!(status & ATMEL_US_DSR))
Andrew Victor1e6c9c22006-01-10 16:59:27 +0000322 ret |= TIOCM_DSR;
Haavard Skinnemoen7192f922006-10-04 16:02:05 +0200323 if (!(status & ATMEL_US_RI))
Andrew Victor1e6c9c22006-01-10 16:59:27 +0000324 ret |= TIOCM_RI;
325
326 return ret;
327}
328
329/*
330 * Stop transmitting.
331 */
Haavard Skinnemoen7192f922006-10-04 16:02:05 +0200332static void atmel_stop_tx(struct uart_port *port)
Andrew Victor1e6c9c22006-01-10 16:59:27 +0000333{
Claudio Scordinoe8faff72010-05-03 13:31:28 +0100334 struct atmel_uart_port *atmel_port = to_atmel_uart_port(port);
335
Chip Coldwella6670612008-02-08 04:21:06 -0800336 if (atmel_use_dma_tx(port)) {
337 /* disable PDC transmit */
338 UART_PUT_PTCR(port, ATMEL_PDC_TXTDIS);
Claudio Scordinoe8faff72010-05-03 13:31:28 +0100339 }
340 /* Disable interrupts */
341 UART_PUT_IDR(port, atmel_port->tx_done_mask);
342
Bernhard Roth83cac9f2011-08-24 09:48:23 +0200343 if ((atmel_port->rs485.flags & SER_RS485_ENABLED) &&
344 !(atmel_port->rs485.flags & SER_RS485_RX_DURING_TX))
Claudio Scordinoe8faff72010-05-03 13:31:28 +0100345 atmel_start_rx(port);
Andrew Victor1e6c9c22006-01-10 16:59:27 +0000346}
347
348/*
349 * Start transmitting.
350 */
Haavard Skinnemoen7192f922006-10-04 16:02:05 +0200351static void atmel_start_tx(struct uart_port *port)
Andrew Victor1e6c9c22006-01-10 16:59:27 +0000352{
Claudio Scordinoe8faff72010-05-03 13:31:28 +0100353 struct atmel_uart_port *atmel_port = to_atmel_uart_port(port);
354
Chip Coldwella6670612008-02-08 04:21:06 -0800355 if (atmel_use_dma_tx(port)) {
356 if (UART_GET_PTSR(port) & ATMEL_PDC_TXTEN)
357 /* The transmitter is already running. Yes, we
358 really need this.*/
359 return;
360
Bernhard Roth83cac9f2011-08-24 09:48:23 +0200361 if ((atmel_port->rs485.flags & SER_RS485_ENABLED) &&
362 !(atmel_port->rs485.flags & SER_RS485_RX_DURING_TX))
Claudio Scordinoe8faff72010-05-03 13:31:28 +0100363 atmel_stop_rx(port);
364
Chip Coldwella6670612008-02-08 04:21:06 -0800365 /* re-enable PDC transmit */
366 UART_PUT_PTCR(port, ATMEL_PDC_TXTEN);
Claudio Scordinoe8faff72010-05-03 13:31:28 +0100367 }
368 /* Enable interrupts */
369 UART_PUT_IER(port, atmel_port->tx_done_mask);
370}
371
372/*
373 * start receiving - port is in process of being opened.
374 */
375static void atmel_start_rx(struct uart_port *port)
376{
377 UART_PUT_CR(port, ATMEL_US_RSTSTA); /* reset status and receiver */
378
379 if (atmel_use_dma_rx(port)) {
380 /* enable PDC controller */
381 UART_PUT_IER(port, ATMEL_US_ENDRX | ATMEL_US_TIMEOUT |
382 port->read_status_mask);
383 UART_PUT_PTCR(port, ATMEL_PDC_RXTEN);
384 } else {
385 UART_PUT_IER(port, ATMEL_US_RXRDY);
386 }
Andrew Victor1e6c9c22006-01-10 16:59:27 +0000387}
388
389/*
390 * Stop receiving - port is in process of being closed.
391 */
Haavard Skinnemoen7192f922006-10-04 16:02:05 +0200392static void atmel_stop_rx(struct uart_port *port)
Andrew Victor1e6c9c22006-01-10 16:59:27 +0000393{
Chip Coldwella6670612008-02-08 04:21:06 -0800394 if (atmel_use_dma_rx(port)) {
395 /* disable PDC receive */
396 UART_PUT_PTCR(port, ATMEL_PDC_RXTDIS);
Claudio Scordinoe8faff72010-05-03 13:31:28 +0100397 UART_PUT_IDR(port, ATMEL_US_ENDRX | ATMEL_US_TIMEOUT |
398 port->read_status_mask);
399 } else {
Chip Coldwella6670612008-02-08 04:21:06 -0800400 UART_PUT_IDR(port, ATMEL_US_RXRDY);
Claudio Scordinoe8faff72010-05-03 13:31:28 +0100401 }
Andrew Victor1e6c9c22006-01-10 16:59:27 +0000402}
403
404/*
405 * Enable modem status interrupts
406 */
Haavard Skinnemoen7192f922006-10-04 16:02:05 +0200407static void atmel_enable_ms(struct uart_port *port)
Andrew Victor1e6c9c22006-01-10 16:59:27 +0000408{
Remy Bohmerb843aa22008-02-08 04:21:01 -0800409 UART_PUT_IER(port, ATMEL_US_RIIC | ATMEL_US_DSRIC
410 | ATMEL_US_DCDIC | ATMEL_US_CTSIC);
Andrew Victor1e6c9c22006-01-10 16:59:27 +0000411}
412
413/*
414 * Control the transmission of a break signal
415 */
Haavard Skinnemoen7192f922006-10-04 16:02:05 +0200416static void atmel_break_ctl(struct uart_port *port, int break_state)
Andrew Victor1e6c9c22006-01-10 16:59:27 +0000417{
418 if (break_state != 0)
Haavard Skinnemoen7192f922006-10-04 16:02:05 +0200419 UART_PUT_CR(port, ATMEL_US_STTBRK); /* start break */
Andrew Victor1e6c9c22006-01-10 16:59:27 +0000420 else
Haavard Skinnemoen7192f922006-10-04 16:02:05 +0200421 UART_PUT_CR(port, ATMEL_US_STPBRK); /* stop break */
Andrew Victor1e6c9c22006-01-10 16:59:27 +0000422}
423
424/*
Remy Bohmer1ecc26b2008-02-08 04:21:05 -0800425 * Stores the incoming character in the ring buffer
426 */
427static void
428atmel_buffer_rx_char(struct uart_port *port, unsigned int status,
429 unsigned int ch)
430{
Haavard Skinnemoenc811ab82008-02-08 04:21:08 -0800431 struct atmel_uart_port *atmel_port = to_atmel_uart_port(port);
Remy Bohmer1ecc26b2008-02-08 04:21:05 -0800432 struct circ_buf *ring = &atmel_port->rx_ring;
433 struct atmel_uart_char *c;
434
435 if (!CIRC_SPACE(ring->head, ring->tail, ATMEL_SERIAL_RINGSIZE))
436 /* Buffer overflow, ignore char */
437 return;
438
439 c = &((struct atmel_uart_char *)ring->buf)[ring->head];
440 c->status = status;
441 c->ch = ch;
442
443 /* Make sure the character is stored before we update head. */
444 smp_wmb();
445
446 ring->head = (ring->head + 1) & (ATMEL_SERIAL_RINGSIZE - 1);
447}
448
449/*
Chip Coldwella6670612008-02-08 04:21:06 -0800450 * Deal with parity, framing and overrun errors.
451 */
452static void atmel_pdc_rxerr(struct uart_port *port, unsigned int status)
453{
454 /* clear error */
455 UART_PUT_CR(port, ATMEL_US_RSTSTA);
456
457 if (status & ATMEL_US_RXBRK) {
458 /* ignore side-effect */
459 status &= ~(ATMEL_US_PARE | ATMEL_US_FRAME);
460 port->icount.brk++;
461 }
462 if (status & ATMEL_US_PARE)
463 port->icount.parity++;
464 if (status & ATMEL_US_FRAME)
465 port->icount.frame++;
466 if (status & ATMEL_US_OVRE)
467 port->icount.overrun++;
468}
469
470/*
Andrew Victor1e6c9c22006-01-10 16:59:27 +0000471 * Characters received (called from interrupt handler)
472 */
David Howells7d12e782006-10-05 14:55:46 +0100473static void atmel_rx_chars(struct uart_port *port)
Andrew Victor1e6c9c22006-01-10 16:59:27 +0000474{
Haavard Skinnemoenc811ab82008-02-08 04:21:08 -0800475 struct atmel_uart_port *atmel_port = to_atmel_uart_port(port);
Remy Bohmer1ecc26b2008-02-08 04:21:05 -0800476 unsigned int status, ch;
Andrew Victor1e6c9c22006-01-10 16:59:27 +0000477
Andrew Victorafefc412006-06-19 19:53:19 +0100478 status = UART_GET_CSR(port);
Haavard Skinnemoen7192f922006-10-04 16:02:05 +0200479 while (status & ATMEL_US_RXRDY) {
Andrew Victor1e6c9c22006-01-10 16:59:27 +0000480 ch = UART_GET_CHAR(port);
481
Andrew Victor1e6c9c22006-01-10 16:59:27 +0000482 /*
483 * note that the error handling code is
484 * out of the main execution path
485 */
Haavard Skinnemoen9e6077b2007-07-15 23:40:36 -0700486 if (unlikely(status & (ATMEL_US_PARE | ATMEL_US_FRAME
487 | ATMEL_US_OVRE | ATMEL_US_RXBRK)
488 || atmel_port->break_active)) {
Remy Bohmer1ecc26b2008-02-08 04:21:05 -0800489
Remy Bohmerb843aa22008-02-08 04:21:01 -0800490 /* clear error */
491 UART_PUT_CR(port, ATMEL_US_RSTSTA);
Remy Bohmer1ecc26b2008-02-08 04:21:05 -0800492
Haavard Skinnemoen9e6077b2007-07-15 23:40:36 -0700493 if (status & ATMEL_US_RXBRK
494 && !atmel_port->break_active) {
Haavard Skinnemoen9e6077b2007-07-15 23:40:36 -0700495 atmel_port->break_active = 1;
496 UART_PUT_IER(port, ATMEL_US_RXBRK);
Haavard Skinnemoen9e6077b2007-07-15 23:40:36 -0700497 } else {
498 /*
499 * This is either the end-of-break
500 * condition or we've received at
501 * least one character without RXBRK
502 * being set. In both cases, the next
503 * RXBRK will indicate start-of-break.
504 */
505 UART_PUT_IDR(port, ATMEL_US_RXBRK);
506 status &= ~ATMEL_US_RXBRK;
507 atmel_port->break_active = 0;
Andrew Victorafefc412006-06-19 19:53:19 +0100508 }
Andrew Victor1e6c9c22006-01-10 16:59:27 +0000509 }
510
Remy Bohmer1ecc26b2008-02-08 04:21:05 -0800511 atmel_buffer_rx_char(port, status, ch);
Andrew Victorafefc412006-06-19 19:53:19 +0100512 status = UART_GET_CSR(port);
Andrew Victor1e6c9c22006-01-10 16:59:27 +0000513 }
514
Remy Bohmer1ecc26b2008-02-08 04:21:05 -0800515 tasklet_schedule(&atmel_port->tasklet);
Andrew Victor1e6c9c22006-01-10 16:59:27 +0000516}
517
518/*
Remy Bohmer1ecc26b2008-02-08 04:21:05 -0800519 * Transmit characters (called from tasklet with TXRDY interrupt
520 * disabled)
Andrew Victor1e6c9c22006-01-10 16:59:27 +0000521 */
Haavard Skinnemoen7192f922006-10-04 16:02:05 +0200522static void atmel_tx_chars(struct uart_port *port)
Andrew Victor1e6c9c22006-01-10 16:59:27 +0000523{
Alan Coxebd2c8f2009-09-19 13:13:28 -0700524 struct circ_buf *xmit = &port->state->xmit;
Claudio Scordinoe8faff72010-05-03 13:31:28 +0100525 struct atmel_uart_port *atmel_port = to_atmel_uart_port(port);
Andrew Victor1e6c9c22006-01-10 16:59:27 +0000526
Claudio Scordinoe8faff72010-05-03 13:31:28 +0100527 if (port->x_char && UART_GET_CSR(port) & atmel_port->tx_done_mask) {
Andrew Victor1e6c9c22006-01-10 16:59:27 +0000528 UART_PUT_CHAR(port, port->x_char);
529 port->icount.tx++;
530 port->x_char = 0;
Andrew Victor1e6c9c22006-01-10 16:59:27 +0000531 }
Remy Bohmer1ecc26b2008-02-08 04:21:05 -0800532 if (uart_circ_empty(xmit) || uart_tx_stopped(port))
Andrew Victor1e6c9c22006-01-10 16:59:27 +0000533 return;
Andrew Victor1e6c9c22006-01-10 16:59:27 +0000534
Claudio Scordinoe8faff72010-05-03 13:31:28 +0100535 while (UART_GET_CSR(port) & atmel_port->tx_done_mask) {
Andrew Victor1e6c9c22006-01-10 16:59:27 +0000536 UART_PUT_CHAR(port, xmit->buf[xmit->tail]);
537 xmit->tail = (xmit->tail + 1) & (UART_XMIT_SIZE - 1);
538 port->icount.tx++;
539 if (uart_circ_empty(xmit))
540 break;
541 }
542
543 if (uart_circ_chars_pending(xmit) < WAKEUP_CHARS)
544 uart_write_wakeup(port);
545
Remy Bohmer1ecc26b2008-02-08 04:21:05 -0800546 if (!uart_circ_empty(xmit))
Claudio Scordinoe8faff72010-05-03 13:31:28 +0100547 /* Enable interrupts */
548 UART_PUT_IER(port, atmel_port->tx_done_mask);
Andrew Victor1e6c9c22006-01-10 16:59:27 +0000549}
550
551/*
Remy Bohmerb843aa22008-02-08 04:21:01 -0800552 * receive interrupt handler.
553 */
554static void
555atmel_handle_receive(struct uart_port *port, unsigned int pending)
556{
Haavard Skinnemoenc811ab82008-02-08 04:21:08 -0800557 struct atmel_uart_port *atmel_port = to_atmel_uart_port(port);
Remy Bohmerb843aa22008-02-08 04:21:01 -0800558
Chip Coldwella6670612008-02-08 04:21:06 -0800559 if (atmel_use_dma_rx(port)) {
560 /*
561 * PDC receive. Just schedule the tasklet and let it
562 * figure out the details.
563 *
564 * TODO: We're not handling error flags correctly at
565 * the moment.
566 */
567 if (pending & (ATMEL_US_ENDRX | ATMEL_US_TIMEOUT)) {
568 UART_PUT_IDR(port, (ATMEL_US_ENDRX
569 | ATMEL_US_TIMEOUT));
570 tasklet_schedule(&atmel_port->tasklet);
571 }
572
573 if (pending & (ATMEL_US_RXBRK | ATMEL_US_OVRE |
574 ATMEL_US_FRAME | ATMEL_US_PARE))
575 atmel_pdc_rxerr(port, pending);
576 }
577
Remy Bohmerb843aa22008-02-08 04:21:01 -0800578 /* Interrupt receive */
579 if (pending & ATMEL_US_RXRDY)
580 atmel_rx_chars(port);
581 else if (pending & ATMEL_US_RXBRK) {
582 /*
583 * End of break detected. If it came along with a
584 * character, atmel_rx_chars will handle it.
585 */
586 UART_PUT_CR(port, ATMEL_US_RSTSTA);
587 UART_PUT_IDR(port, ATMEL_US_RXBRK);
588 atmel_port->break_active = 0;
589 }
590}
591
592/*
Remy Bohmer1ecc26b2008-02-08 04:21:05 -0800593 * transmit interrupt handler. (Transmit is IRQF_NODELAY safe)
Remy Bohmerb843aa22008-02-08 04:21:01 -0800594 */
595static void
596atmel_handle_transmit(struct uart_port *port, unsigned int pending)
597{
Haavard Skinnemoenc811ab82008-02-08 04:21:08 -0800598 struct atmel_uart_port *atmel_port = to_atmel_uart_port(port);
Remy Bohmer1ecc26b2008-02-08 04:21:05 -0800599
Claudio Scordinoe8faff72010-05-03 13:31:28 +0100600 if (pending & atmel_port->tx_done_mask) {
601 /* Either PDC or interrupt transmission */
602 UART_PUT_IDR(port, atmel_port->tx_done_mask);
603 tasklet_schedule(&atmel_port->tasklet);
Remy Bohmer1ecc26b2008-02-08 04:21:05 -0800604 }
Remy Bohmerb843aa22008-02-08 04:21:01 -0800605}
606
607/*
608 * status flags interrupt handler.
609 */
610static void
611atmel_handle_status(struct uart_port *port, unsigned int pending,
612 unsigned int status)
613{
Haavard Skinnemoenc811ab82008-02-08 04:21:08 -0800614 struct atmel_uart_port *atmel_port = to_atmel_uart_port(port);
Remy Bohmer1ecc26b2008-02-08 04:21:05 -0800615
Remy Bohmerb843aa22008-02-08 04:21:01 -0800616 if (pending & (ATMEL_US_RIIC | ATMEL_US_DSRIC | ATMEL_US_DCDIC
Remy Bohmer1ecc26b2008-02-08 04:21:05 -0800617 | ATMEL_US_CTSIC)) {
618 atmel_port->irq_status = status;
619 tasklet_schedule(&atmel_port->tasklet);
620 }
Remy Bohmerb843aa22008-02-08 04:21:01 -0800621}
622
623/*
Andrew Victor1e6c9c22006-01-10 16:59:27 +0000624 * Interrupt handler
625 */
David Howells7d12e782006-10-05 14:55:46 +0100626static irqreturn_t atmel_interrupt(int irq, void *dev_id)
Andrew Victor1e6c9c22006-01-10 16:59:27 +0000627{
628 struct uart_port *port = dev_id;
629 unsigned int status, pending, pass_counter = 0;
630
Chip Coldwella6670612008-02-08 04:21:06 -0800631 do {
632 status = UART_GET_CSR(port);
633 pending = status & UART_GET_IMR(port);
634 if (!pending)
635 break;
636
Remy Bohmerb843aa22008-02-08 04:21:01 -0800637 atmel_handle_receive(port, pending);
638 atmel_handle_status(port, pending, status);
639 atmel_handle_transmit(port, pending);
Chip Coldwella6670612008-02-08 04:21:06 -0800640 } while (pass_counter++ < ATMEL_ISR_PASS_LIMIT);
Andrew Victor1e6c9c22006-01-10 16:59:27 +0000641
Haavard Skinnemoen0400b692008-02-23 15:23:36 -0800642 return pass_counter ? IRQ_HANDLED : IRQ_NONE;
Andrew Victor1e6c9c22006-01-10 16:59:27 +0000643}
644
Chip Coldwella6670612008-02-08 04:21:06 -0800645/*
646 * Called from tasklet with ENDTX and TXBUFE interrupts disabled.
647 */
648static void atmel_tx_dma(struct uart_port *port)
649{
Haavard Skinnemoenc811ab82008-02-08 04:21:08 -0800650 struct atmel_uart_port *atmel_port = to_atmel_uart_port(port);
Alan Coxebd2c8f2009-09-19 13:13:28 -0700651 struct circ_buf *xmit = &port->state->xmit;
Chip Coldwella6670612008-02-08 04:21:06 -0800652 struct atmel_dma_buffer *pdc = &atmel_port->pdc_tx;
653 int count;
654
Michael Trimarchiba0657f2008-04-02 13:04:41 -0700655 /* nothing left to transmit? */
656 if (UART_GET_TCR(port))
657 return;
658
Chip Coldwella6670612008-02-08 04:21:06 -0800659 xmit->tail += pdc->ofs;
660 xmit->tail &= UART_XMIT_SIZE - 1;
661
662 port->icount.tx += pdc->ofs;
663 pdc->ofs = 0;
664
Michael Trimarchiba0657f2008-04-02 13:04:41 -0700665 /* more to transmit - setup next transfer */
Chip Coldwella6670612008-02-08 04:21:06 -0800666
Michael Trimarchiba0657f2008-04-02 13:04:41 -0700667 /* disable PDC transmit */
668 UART_PUT_PTCR(port, ATMEL_PDC_TXTDIS);
669
Itai Levi1f140812009-01-15 13:50:43 -0800670 if (!uart_circ_empty(xmit) && !uart_tx_stopped(port)) {
Chip Coldwella6670612008-02-08 04:21:06 -0800671 dma_sync_single_for_device(port->dev,
672 pdc->dma_addr,
673 pdc->dma_size,
674 DMA_TO_DEVICE);
675
676 count = CIRC_CNT_TO_END(xmit->head, xmit->tail, UART_XMIT_SIZE);
677 pdc->ofs = count;
678
679 UART_PUT_TPR(port, pdc->dma_addr + xmit->tail);
680 UART_PUT_TCR(port, count);
Claudio Scordinoe8faff72010-05-03 13:31:28 +0100681 /* re-enable PDC transmit */
Chip Coldwella6670612008-02-08 04:21:06 -0800682 UART_PUT_PTCR(port, ATMEL_PDC_TXTEN);
Claudio Scordinoe8faff72010-05-03 13:31:28 +0100683 /* Enable interrupts */
684 UART_PUT_IER(port, atmel_port->tx_done_mask);
685 } else {
Bernhard Roth83cac9f2011-08-24 09:48:23 +0200686 if ((atmel_port->rs485.flags & SER_RS485_ENABLED) &&
687 !(atmel_port->rs485.flags & SER_RS485_RX_DURING_TX)) {
Claudio Scordinoe8faff72010-05-03 13:31:28 +0100688 /* DMA done, stop TX, start RX for RS485 */
689 atmel_start_rx(port);
690 }
Chip Coldwella6670612008-02-08 04:21:06 -0800691 }
692
693 if (uart_circ_chars_pending(xmit) < WAKEUP_CHARS)
694 uart_write_wakeup(port);
695}
696
Remy Bohmer1ecc26b2008-02-08 04:21:05 -0800697static void atmel_rx_from_ring(struct uart_port *port)
698{
Haavard Skinnemoenc811ab82008-02-08 04:21:08 -0800699 struct atmel_uart_port *atmel_port = to_atmel_uart_port(port);
Remy Bohmer1ecc26b2008-02-08 04:21:05 -0800700 struct circ_buf *ring = &atmel_port->rx_ring;
701 unsigned int flg;
702 unsigned int status;
703
704 while (ring->head != ring->tail) {
705 struct atmel_uart_char c;
706
707 /* Make sure c is loaded after head. */
708 smp_rmb();
709
710 c = ((struct atmel_uart_char *)ring->buf)[ring->tail];
711
712 ring->tail = (ring->tail + 1) & (ATMEL_SERIAL_RINGSIZE - 1);
713
714 port->icount.rx++;
715 status = c.status;
716 flg = TTY_NORMAL;
717
718 /*
719 * note that the error handling code is
720 * out of the main execution path
721 */
722 if (unlikely(status & (ATMEL_US_PARE | ATMEL_US_FRAME
723 | ATMEL_US_OVRE | ATMEL_US_RXBRK))) {
724 if (status & ATMEL_US_RXBRK) {
725 /* ignore side-effect */
726 status &= ~(ATMEL_US_PARE | ATMEL_US_FRAME);
727
728 port->icount.brk++;
729 if (uart_handle_break(port))
730 continue;
731 }
732 if (status & ATMEL_US_PARE)
733 port->icount.parity++;
734 if (status & ATMEL_US_FRAME)
735 port->icount.frame++;
736 if (status & ATMEL_US_OVRE)
737 port->icount.overrun++;
738
739 status &= port->read_status_mask;
740
741 if (status & ATMEL_US_RXBRK)
742 flg = TTY_BREAK;
743 else if (status & ATMEL_US_PARE)
744 flg = TTY_PARITY;
745 else if (status & ATMEL_US_FRAME)
746 flg = TTY_FRAME;
747 }
748
749
750 if (uart_handle_sysrq_char(port, c.ch))
751 continue;
752
753 uart_insert_char(port, status, ATMEL_US_OVRE, c.ch, flg);
754 }
755
756 /*
757 * Drop the lock here since it might end up calling
758 * uart_start(), which takes the lock.
759 */
760 spin_unlock(&port->lock);
Alan Coxebd2c8f2009-09-19 13:13:28 -0700761 tty_flip_buffer_push(port->state->port.tty);
Remy Bohmer1ecc26b2008-02-08 04:21:05 -0800762 spin_lock(&port->lock);
763}
764
Chip Coldwella6670612008-02-08 04:21:06 -0800765static void atmel_rx_from_dma(struct uart_port *port)
766{
Haavard Skinnemoenc811ab82008-02-08 04:21:08 -0800767 struct atmel_uart_port *atmel_port = to_atmel_uart_port(port);
Alan Coxebd2c8f2009-09-19 13:13:28 -0700768 struct tty_struct *tty = port->state->port.tty;
Chip Coldwella6670612008-02-08 04:21:06 -0800769 struct atmel_dma_buffer *pdc;
770 int rx_idx = atmel_port->pdc_rx_idx;
771 unsigned int head;
772 unsigned int tail;
773 unsigned int count;
774
775 do {
776 /* Reset the UART timeout early so that we don't miss one */
777 UART_PUT_CR(port, ATMEL_US_STTTO);
778
779 pdc = &atmel_port->pdc_rx[rx_idx];
780 head = UART_GET_RPR(port) - pdc->dma_addr;
781 tail = pdc->ofs;
782
783 /* If the PDC has switched buffers, RPR won't contain
784 * any address within the current buffer. Since head
785 * is unsigned, we just need a one-way comparison to
786 * find out.
787 *
788 * In this case, we just need to consume the entire
789 * buffer and resubmit it for DMA. This will clear the
790 * ENDRX bit as well, so that we can safely re-enable
791 * all interrupts below.
792 */
793 head = min(head, pdc->dma_size);
794
795 if (likely(head != tail)) {
796 dma_sync_single_for_cpu(port->dev, pdc->dma_addr,
797 pdc->dma_size, DMA_FROM_DEVICE);
798
799 /*
800 * head will only wrap around when we recycle
801 * the DMA buffer, and when that happens, we
802 * explicitly set tail to 0. So head will
803 * always be greater than tail.
804 */
805 count = head - tail;
806
807 tty_insert_flip_string(tty, pdc->buf + pdc->ofs, count);
808
809 dma_sync_single_for_device(port->dev, pdc->dma_addr,
810 pdc->dma_size, DMA_FROM_DEVICE);
811
812 port->icount.rx += count;
813 pdc->ofs = head;
814 }
815
816 /*
817 * If the current buffer is full, we need to check if
818 * the next one contains any additional data.
819 */
820 if (head >= pdc->dma_size) {
821 pdc->ofs = 0;
822 UART_PUT_RNPR(port, pdc->dma_addr);
823 UART_PUT_RNCR(port, pdc->dma_size);
824
825 rx_idx = !rx_idx;
826 atmel_port->pdc_rx_idx = rx_idx;
827 }
828 } while (head >= pdc->dma_size);
829
830 /*
831 * Drop the lock here since it might end up calling
832 * uart_start(), which takes the lock.
833 */
834 spin_unlock(&port->lock);
835 tty_flip_buffer_push(tty);
836 spin_lock(&port->lock);
837
838 UART_PUT_IER(port, ATMEL_US_ENDRX | ATMEL_US_TIMEOUT);
839}
840
Remy Bohmer1ecc26b2008-02-08 04:21:05 -0800841/*
842 * tasklet handling tty stuff outside the interrupt handler.
843 */
844static void atmel_tasklet_func(unsigned long data)
845{
846 struct uart_port *port = (struct uart_port *)data;
Haavard Skinnemoenc811ab82008-02-08 04:21:08 -0800847 struct atmel_uart_port *atmel_port = to_atmel_uart_port(port);
Remy Bohmer1ecc26b2008-02-08 04:21:05 -0800848 unsigned int status;
849 unsigned int status_change;
850
851 /* The interrupt handler does not take the lock */
852 spin_lock(&port->lock);
853
Chip Coldwella6670612008-02-08 04:21:06 -0800854 if (atmel_use_dma_tx(port))
855 atmel_tx_dma(port);
856 else
857 atmel_tx_chars(port);
Remy Bohmer1ecc26b2008-02-08 04:21:05 -0800858
859 status = atmel_port->irq_status;
860 status_change = status ^ atmel_port->irq_status_prev;
861
862 if (status_change & (ATMEL_US_RI | ATMEL_US_DSR
863 | ATMEL_US_DCD | ATMEL_US_CTS)) {
864 /* TODO: All reads to CSR will clear these interrupts! */
865 if (status_change & ATMEL_US_RI)
866 port->icount.rng++;
867 if (status_change & ATMEL_US_DSR)
868 port->icount.dsr++;
869 if (status_change & ATMEL_US_DCD)
870 uart_handle_dcd_change(port, !(status & ATMEL_US_DCD));
871 if (status_change & ATMEL_US_CTS)
872 uart_handle_cts_change(port, !(status & ATMEL_US_CTS));
873
Alan Coxbdc04e32009-09-19 13:13:31 -0700874 wake_up_interruptible(&port->state->port.delta_msr_wait);
Remy Bohmer1ecc26b2008-02-08 04:21:05 -0800875
876 atmel_port->irq_status_prev = status;
877 }
878
Chip Coldwella6670612008-02-08 04:21:06 -0800879 if (atmel_use_dma_rx(port))
880 atmel_rx_from_dma(port);
881 else
882 atmel_rx_from_ring(port);
Remy Bohmer1ecc26b2008-02-08 04:21:05 -0800883
884 spin_unlock(&port->lock);
885}
886
Andrew Victor1e6c9c22006-01-10 16:59:27 +0000887/*
888 * Perform initialization and enable port for reception
889 */
Haavard Skinnemoen7192f922006-10-04 16:02:05 +0200890static int atmel_startup(struct uart_port *port)
Andrew Victor1e6c9c22006-01-10 16:59:27 +0000891{
Haavard Skinnemoenc811ab82008-02-08 04:21:08 -0800892 struct atmel_uart_port *atmel_port = to_atmel_uart_port(port);
Alan Coxebd2c8f2009-09-19 13:13:28 -0700893 struct tty_struct *tty = port->state->port.tty;
Andrew Victor1e6c9c22006-01-10 16:59:27 +0000894 int retval;
895
896 /*
897 * Ensure that no interrupts are enabled otherwise when
898 * request_irq() is called we could get stuck trying to
899 * handle an unexpected interrupt
900 */
901 UART_PUT_IDR(port, -1);
902
903 /*
904 * Allocate the IRQ
905 */
Remy Bohmerb843aa22008-02-08 04:21:01 -0800906 retval = request_irq(port->irq, atmel_interrupt, IRQF_SHARED,
Haavard Skinnemoenae161062008-02-08 04:21:08 -0800907 tty ? tty->name : "atmel_serial", port);
Andrew Victor1e6c9c22006-01-10 16:59:27 +0000908 if (retval) {
Haavard Skinnemoen7192f922006-10-04 16:02:05 +0200909 printk("atmel_serial: atmel_startup - Can't get irq\n");
Andrew Victor1e6c9c22006-01-10 16:59:27 +0000910 return retval;
911 }
912
913 /*
Chip Coldwella6670612008-02-08 04:21:06 -0800914 * Initialize DMA (if necessary)
915 */
916 if (atmel_use_dma_rx(port)) {
917 int i;
918
919 for (i = 0; i < 2; i++) {
920 struct atmel_dma_buffer *pdc = &atmel_port->pdc_rx[i];
921
922 pdc->buf = kmalloc(PDC_BUFFER_SIZE, GFP_KERNEL);
923 if (pdc->buf == NULL) {
924 if (i != 0) {
925 dma_unmap_single(port->dev,
926 atmel_port->pdc_rx[0].dma_addr,
927 PDC_BUFFER_SIZE,
928 DMA_FROM_DEVICE);
929 kfree(atmel_port->pdc_rx[0].buf);
930 }
931 free_irq(port->irq, port);
932 return -ENOMEM;
933 }
934 pdc->dma_addr = dma_map_single(port->dev,
935 pdc->buf,
936 PDC_BUFFER_SIZE,
937 DMA_FROM_DEVICE);
938 pdc->dma_size = PDC_BUFFER_SIZE;
939 pdc->ofs = 0;
940 }
941
942 atmel_port->pdc_rx_idx = 0;
943
944 UART_PUT_RPR(port, atmel_port->pdc_rx[0].dma_addr);
945 UART_PUT_RCR(port, PDC_BUFFER_SIZE);
946
947 UART_PUT_RNPR(port, atmel_port->pdc_rx[1].dma_addr);
948 UART_PUT_RNCR(port, PDC_BUFFER_SIZE);
949 }
950 if (atmel_use_dma_tx(port)) {
951 struct atmel_dma_buffer *pdc = &atmel_port->pdc_tx;
Alan Coxebd2c8f2009-09-19 13:13:28 -0700952 struct circ_buf *xmit = &port->state->xmit;
Chip Coldwella6670612008-02-08 04:21:06 -0800953
954 pdc->buf = xmit->buf;
955 pdc->dma_addr = dma_map_single(port->dev,
956 pdc->buf,
957 UART_XMIT_SIZE,
958 DMA_TO_DEVICE);
959 pdc->dma_size = UART_XMIT_SIZE;
960 pdc->ofs = 0;
961 }
962
963 /*
Andrew Victor1e6c9c22006-01-10 16:59:27 +0000964 * If there is a specific "open" function (to register
965 * control line interrupts)
966 */
Haavard Skinnemoen71f2e2b2006-10-04 16:02:07 +0200967 if (atmel_open_hook) {
968 retval = atmel_open_hook(port);
Andrew Victor1e6c9c22006-01-10 16:59:27 +0000969 if (retval) {
970 free_irq(port->irq, port);
971 return retval;
972 }
973 }
974
Atsushi Nemoto27c0c8e2009-02-18 14:48:28 -0800975 /* Save current CSR for comparison in atmel_tasklet_func() */
976 atmel_port->irq_status_prev = UART_GET_CSR(port);
977 atmel_port->irq_status = atmel_port->irq_status_prev;
978
Andrew Victor1e6c9c22006-01-10 16:59:27 +0000979 /*
980 * Finally, enable the serial port
981 */
Haavard Skinnemoen7192f922006-10-04 16:02:05 +0200982 UART_PUT_CR(port, ATMEL_US_RSTSTA | ATMEL_US_RSTRX);
Remy Bohmerb843aa22008-02-08 04:21:01 -0800983 /* enable xmit & rcvr */
984 UART_PUT_CR(port, ATMEL_US_TXEN | ATMEL_US_RXEN);
Andrew Victorafefc412006-06-19 19:53:19 +0100985
Chip Coldwella6670612008-02-08 04:21:06 -0800986 if (atmel_use_dma_rx(port)) {
987 /* set UART timeout */
988 UART_PUT_RTOR(port, PDC_RX_TIMEOUT);
989 UART_PUT_CR(port, ATMEL_US_STTTO);
990
991 UART_PUT_IER(port, ATMEL_US_ENDRX | ATMEL_US_TIMEOUT);
992 /* enable PDC controller */
993 UART_PUT_PTCR(port, ATMEL_PDC_RXTEN);
994 } else {
995 /* enable receive only */
996 UART_PUT_IER(port, ATMEL_US_RXRDY);
997 }
Andrew Victorafefc412006-06-19 19:53:19 +0100998
Andrew Victor1e6c9c22006-01-10 16:59:27 +0000999 return 0;
1000}
1001
1002/*
1003 * Disable the port
1004 */
Haavard Skinnemoen7192f922006-10-04 16:02:05 +02001005static void atmel_shutdown(struct uart_port *port)
Andrew Victor1e6c9c22006-01-10 16:59:27 +00001006{
Haavard Skinnemoenc811ab82008-02-08 04:21:08 -08001007 struct atmel_uart_port *atmel_port = to_atmel_uart_port(port);
Chip Coldwella6670612008-02-08 04:21:06 -08001008 /*
1009 * Ensure everything is stopped.
1010 */
1011 atmel_stop_rx(port);
1012 atmel_stop_tx(port);
1013
1014 /*
1015 * Shut-down the DMA.
1016 */
1017 if (atmel_use_dma_rx(port)) {
1018 int i;
1019
1020 for (i = 0; i < 2; i++) {
1021 struct atmel_dma_buffer *pdc = &atmel_port->pdc_rx[i];
1022
1023 dma_unmap_single(port->dev,
1024 pdc->dma_addr,
1025 pdc->dma_size,
1026 DMA_FROM_DEVICE);
1027 kfree(pdc->buf);
1028 }
1029 }
1030 if (atmel_use_dma_tx(port)) {
1031 struct atmel_dma_buffer *pdc = &atmel_port->pdc_tx;
1032
1033 dma_unmap_single(port->dev,
1034 pdc->dma_addr,
1035 pdc->dma_size,
1036 DMA_TO_DEVICE);
1037 }
1038
Andrew Victor1e6c9c22006-01-10 16:59:27 +00001039 /*
1040 * Disable all interrupts, port and break condition.
1041 */
Haavard Skinnemoen7192f922006-10-04 16:02:05 +02001042 UART_PUT_CR(port, ATMEL_US_RSTSTA);
Andrew Victor1e6c9c22006-01-10 16:59:27 +00001043 UART_PUT_IDR(port, -1);
1044
1045 /*
1046 * Free the interrupt
1047 */
1048 free_irq(port->irq, port);
1049
1050 /*
1051 * If there is a specific "close" function (to unregister
1052 * control line interrupts)
1053 */
Haavard Skinnemoen71f2e2b2006-10-04 16:02:07 +02001054 if (atmel_close_hook)
1055 atmel_close_hook(port);
Andrew Victor1e6c9c22006-01-10 16:59:27 +00001056}
1057
1058/*
Haavard Skinnemoen9afd5612008-07-16 21:52:46 +01001059 * Flush any TX data submitted for DMA. Called when the TX circular
1060 * buffer is reset.
1061 */
1062static void atmel_flush_buffer(struct uart_port *port)
1063{
1064 struct atmel_uart_port *atmel_port = to_atmel_uart_port(port);
1065
1066 if (atmel_use_dma_tx(port)) {
1067 UART_PUT_TCR(port, 0);
1068 atmel_port->pdc_tx.ofs = 0;
1069 }
1070}
1071
1072/*
Andrew Victor1e6c9c22006-01-10 16:59:27 +00001073 * Power / Clock management.
1074 */
Remy Bohmerb843aa22008-02-08 04:21:01 -08001075static void atmel_serial_pm(struct uart_port *port, unsigned int state,
1076 unsigned int oldstate)
Andrew Victor1e6c9c22006-01-10 16:59:27 +00001077{
Haavard Skinnemoenc811ab82008-02-08 04:21:08 -08001078 struct atmel_uart_port *atmel_port = to_atmel_uart_port(port);
Andrew Victorafefc412006-06-19 19:53:19 +01001079
Andrew Victor1e6c9c22006-01-10 16:59:27 +00001080 switch (state) {
Remy Bohmerb843aa22008-02-08 04:21:01 -08001081 case 0:
1082 /*
1083 * Enable the peripheral clock for this serial port.
1084 * This is called on uart_open() or a resume event.
1085 */
1086 clk_enable(atmel_port->clk);
Anti Sullinf05596d2008-09-22 13:57:54 -07001087
1088 /* re-enable interrupts if we disabled some on suspend */
1089 UART_PUT_IER(port, atmel_port->backup_imr);
Remy Bohmerb843aa22008-02-08 04:21:01 -08001090 break;
1091 case 3:
Anti Sullinf05596d2008-09-22 13:57:54 -07001092 /* Back up the interrupt mask and disable all interrupts */
1093 atmel_port->backup_imr = UART_GET_IMR(port);
1094 UART_PUT_IDR(port, -1);
1095
Remy Bohmerb843aa22008-02-08 04:21:01 -08001096 /*
1097 * Disable the peripheral clock for this serial port.
1098 * This is called on uart_close() or a suspend event.
1099 */
1100 clk_disable(atmel_port->clk);
1101 break;
1102 default:
1103 printk(KERN_ERR "atmel_serial: unknown pm %d\n", state);
Andrew Victor1e6c9c22006-01-10 16:59:27 +00001104 }
1105}
1106
1107/*
1108 * Change the port parameters
1109 */
Remy Bohmerb843aa22008-02-08 04:21:01 -08001110static void atmel_set_termios(struct uart_port *port, struct ktermios *termios,
1111 struct ktermios *old)
Andrew Victor1e6c9c22006-01-10 16:59:27 +00001112{
1113 unsigned long flags;
1114 unsigned int mode, imr, quot, baud;
Claudio Scordinoe8faff72010-05-03 13:31:28 +01001115 struct atmel_uart_port *atmel_port = to_atmel_uart_port(port);
Andrew Victor1e6c9c22006-01-10 16:59:27 +00001116
Andrew Victor03abeac2007-05-03 12:26:24 +01001117 /* Get current mode register */
Remy Bohmerb843aa22008-02-08 04:21:01 -08001118 mode = UART_GET_MR(port) & ~(ATMEL_US_USCLKS | ATMEL_US_CHRL
Peter Ma8e706c42009-03-31 10:31:02 -07001119 | ATMEL_US_NBSTOP | ATMEL_US_PAR
1120 | ATMEL_US_USMODE);
Andrew Victor03abeac2007-05-03 12:26:24 +01001121
Remy Bohmerb843aa22008-02-08 04:21:01 -08001122 baud = uart_get_baud_rate(port, termios, old, 0, port->uartclk / 16);
Andrew Victor1e6c9c22006-01-10 16:59:27 +00001123 quot = uart_get_divisor(port, baud);
1124
Remy Bohmerb843aa22008-02-08 04:21:01 -08001125 if (quot > 65535) { /* BRGR is 16-bit, so switch to slower clock */
Andrew Victor03abeac2007-05-03 12:26:24 +01001126 quot /= 8;
1127 mode |= ATMEL_US_USCLKS_MCK_DIV8;
1128 }
Andrew Victor1e6c9c22006-01-10 16:59:27 +00001129
1130 /* byte size */
1131 switch (termios->c_cflag & CSIZE) {
1132 case CS5:
Haavard Skinnemoen7192f922006-10-04 16:02:05 +02001133 mode |= ATMEL_US_CHRL_5;
Andrew Victor1e6c9c22006-01-10 16:59:27 +00001134 break;
1135 case CS6:
Haavard Skinnemoen7192f922006-10-04 16:02:05 +02001136 mode |= ATMEL_US_CHRL_6;
Andrew Victor1e6c9c22006-01-10 16:59:27 +00001137 break;
1138 case CS7:
Haavard Skinnemoen7192f922006-10-04 16:02:05 +02001139 mode |= ATMEL_US_CHRL_7;
Andrew Victor1e6c9c22006-01-10 16:59:27 +00001140 break;
1141 default:
Haavard Skinnemoen7192f922006-10-04 16:02:05 +02001142 mode |= ATMEL_US_CHRL_8;
Andrew Victor1e6c9c22006-01-10 16:59:27 +00001143 break;
1144 }
1145
1146 /* stop bits */
1147 if (termios->c_cflag & CSTOPB)
Haavard Skinnemoen7192f922006-10-04 16:02:05 +02001148 mode |= ATMEL_US_NBSTOP_2;
Andrew Victor1e6c9c22006-01-10 16:59:27 +00001149
1150 /* parity */
1151 if (termios->c_cflag & PARENB) {
Remy Bohmerb843aa22008-02-08 04:21:01 -08001152 /* Mark or Space parity */
1153 if (termios->c_cflag & CMSPAR) {
Andrew Victor1e6c9c22006-01-10 16:59:27 +00001154 if (termios->c_cflag & PARODD)
Haavard Skinnemoen7192f922006-10-04 16:02:05 +02001155 mode |= ATMEL_US_PAR_MARK;
Andrew Victor1e6c9c22006-01-10 16:59:27 +00001156 else
Haavard Skinnemoen7192f922006-10-04 16:02:05 +02001157 mode |= ATMEL_US_PAR_SPACE;
Remy Bohmerb843aa22008-02-08 04:21:01 -08001158 } else if (termios->c_cflag & PARODD)
Haavard Skinnemoen7192f922006-10-04 16:02:05 +02001159 mode |= ATMEL_US_PAR_ODD;
Andrew Victor1e6c9c22006-01-10 16:59:27 +00001160 else
Haavard Skinnemoen7192f922006-10-04 16:02:05 +02001161 mode |= ATMEL_US_PAR_EVEN;
Remy Bohmerb843aa22008-02-08 04:21:01 -08001162 } else
Haavard Skinnemoen7192f922006-10-04 16:02:05 +02001163 mode |= ATMEL_US_PAR_NONE;
Andrew Victor1e6c9c22006-01-10 16:59:27 +00001164
Peter Ma8e706c42009-03-31 10:31:02 -07001165 /* hardware handshake (RTS/CTS) */
1166 if (termios->c_cflag & CRTSCTS)
1167 mode |= ATMEL_US_USMODE_HWHS;
1168 else
1169 mode |= ATMEL_US_USMODE_NORMAL;
1170
Andrew Victor1e6c9c22006-01-10 16:59:27 +00001171 spin_lock_irqsave(&port->lock, flags);
1172
Haavard Skinnemoen7192f922006-10-04 16:02:05 +02001173 port->read_status_mask = ATMEL_US_OVRE;
Andrew Victor1e6c9c22006-01-10 16:59:27 +00001174 if (termios->c_iflag & INPCK)
Haavard Skinnemoen7192f922006-10-04 16:02:05 +02001175 port->read_status_mask |= (ATMEL_US_FRAME | ATMEL_US_PARE);
Andrew Victor1e6c9c22006-01-10 16:59:27 +00001176 if (termios->c_iflag & (BRKINT | PARMRK))
Haavard Skinnemoen7192f922006-10-04 16:02:05 +02001177 port->read_status_mask |= ATMEL_US_RXBRK;
Andrew Victor1e6c9c22006-01-10 16:59:27 +00001178
Chip Coldwella6670612008-02-08 04:21:06 -08001179 if (atmel_use_dma_rx(port))
1180 /* need to enable error interrupts */
1181 UART_PUT_IER(port, port->read_status_mask);
1182
Andrew Victor1e6c9c22006-01-10 16:59:27 +00001183 /*
1184 * Characters to ignore
1185 */
1186 port->ignore_status_mask = 0;
1187 if (termios->c_iflag & IGNPAR)
Haavard Skinnemoen7192f922006-10-04 16:02:05 +02001188 port->ignore_status_mask |= (ATMEL_US_FRAME | ATMEL_US_PARE);
Andrew Victor1e6c9c22006-01-10 16:59:27 +00001189 if (termios->c_iflag & IGNBRK) {
Haavard Skinnemoen7192f922006-10-04 16:02:05 +02001190 port->ignore_status_mask |= ATMEL_US_RXBRK;
Andrew Victor1e6c9c22006-01-10 16:59:27 +00001191 /*
1192 * If we're ignoring parity and break indicators,
1193 * ignore overruns too (for real raw support).
1194 */
1195 if (termios->c_iflag & IGNPAR)
Haavard Skinnemoen7192f922006-10-04 16:02:05 +02001196 port->ignore_status_mask |= ATMEL_US_OVRE;
Andrew Victor1e6c9c22006-01-10 16:59:27 +00001197 }
Remy Bohmerb843aa22008-02-08 04:21:01 -08001198 /* TODO: Ignore all characters if CREAD is set.*/
Andrew Victor1e6c9c22006-01-10 16:59:27 +00001199
1200 /* update the per-port timeout */
1201 uart_update_timeout(port, termios->c_cflag, baud);
1202
Haavard Skinnemoen0ccad872009-06-16 17:02:03 +01001203 /*
1204 * save/disable interrupts. The tty layer will ensure that the
1205 * transmitter is empty if requested by the caller, so there's
1206 * no need to wait for it here.
1207 */
Remy Bohmerb843aa22008-02-08 04:21:01 -08001208 imr = UART_GET_IMR(port);
1209 UART_PUT_IDR(port, -1);
Andrew Victor1e6c9c22006-01-10 16:59:27 +00001210
1211 /* disable receiver and transmitter */
Haavard Skinnemoen7192f922006-10-04 16:02:05 +02001212 UART_PUT_CR(port, ATMEL_US_TXDIS | ATMEL_US_RXDIS);
Andrew Victor1e6c9c22006-01-10 16:59:27 +00001213
Claudio Scordinoe8faff72010-05-03 13:31:28 +01001214 /* Resetting serial mode to RS232 (0x0) */
1215 mode &= ~ATMEL_US_USMODE;
1216
1217 if (atmel_port->rs485.flags & SER_RS485_ENABLED) {
1218 dev_dbg(port->dev, "Setting UART to RS485\n");
Claudio Scordino1b633182010-07-20 15:26:47 -07001219 if (atmel_port->rs485.flags & SER_RS485_RTS_AFTER_SEND)
1220 UART_PUT_TTGR(port,
1221 atmel_port->rs485.delay_rts_after_send);
Claudio Scordinoe8faff72010-05-03 13:31:28 +01001222 mode |= ATMEL_US_USMODE_RS485;
1223 } else {
1224 dev_dbg(port->dev, "Setting UART to RS232\n");
1225 }
1226
Andrew Victor1e6c9c22006-01-10 16:59:27 +00001227 /* set the parity, stop bits and data size */
1228 UART_PUT_MR(port, mode);
1229
1230 /* set the baud rate */
1231 UART_PUT_BRGR(port, quot);
Haavard Skinnemoen7192f922006-10-04 16:02:05 +02001232 UART_PUT_CR(port, ATMEL_US_RSTSTA | ATMEL_US_RSTRX);
1233 UART_PUT_CR(port, ATMEL_US_TXEN | ATMEL_US_RXEN);
Andrew Victor1e6c9c22006-01-10 16:59:27 +00001234
1235 /* restore interrupts */
1236 UART_PUT_IER(port, imr);
1237
1238 /* CTS flow-control and modem-status interrupts */
1239 if (UART_ENABLE_MS(port, termios->c_cflag))
1240 port->ops->enable_ms(port);
1241
1242 spin_unlock_irqrestore(&port->lock, flags);
1243}
1244
Viktar Palstsiuk42bd7a42011-02-09 15:26:13 +01001245static void atmel_set_ldisc(struct uart_port *port, int new)
1246{
1247 int line = port->line;
1248
1249 if (line >= port->state->port.tty->driver->num)
1250 return;
1251
1252 if (port->state->port.tty->ldisc->ops->num == N_PPS) {
1253 port->flags |= UPF_HARDPPS_CD;
1254 atmel_enable_ms(port);
1255 } else {
1256 port->flags &= ~UPF_HARDPPS_CD;
1257 }
1258}
1259
Andrew Victor1e6c9c22006-01-10 16:59:27 +00001260/*
1261 * Return string describing the specified port
1262 */
Haavard Skinnemoen7192f922006-10-04 16:02:05 +02001263static const char *atmel_type(struct uart_port *port)
Andrew Victor1e6c9c22006-01-10 16:59:27 +00001264{
Haavard Skinnemoen9ab4f882006-10-04 16:02:06 +02001265 return (port->type == PORT_ATMEL) ? "ATMEL_SERIAL" : NULL;
Andrew Victor1e6c9c22006-01-10 16:59:27 +00001266}
1267
1268/*
1269 * Release the memory region(s) being used by 'port'.
1270 */
Haavard Skinnemoen7192f922006-10-04 16:02:05 +02001271static void atmel_release_port(struct uart_port *port)
Andrew Victor1e6c9c22006-01-10 16:59:27 +00001272{
Andrew Victorafefc412006-06-19 19:53:19 +01001273 struct platform_device *pdev = to_platform_device(port->dev);
1274 int size = pdev->resource[0].end - pdev->resource[0].start + 1;
1275
1276 release_mem_region(port->mapbase, size);
1277
1278 if (port->flags & UPF_IOREMAP) {
1279 iounmap(port->membase);
1280 port->membase = NULL;
1281 }
Andrew Victor1e6c9c22006-01-10 16:59:27 +00001282}
1283
1284/*
1285 * Request the memory region(s) being used by 'port'.
1286 */
Haavard Skinnemoen7192f922006-10-04 16:02:05 +02001287static int atmel_request_port(struct uart_port *port)
Andrew Victor1e6c9c22006-01-10 16:59:27 +00001288{
Andrew Victorafefc412006-06-19 19:53:19 +01001289 struct platform_device *pdev = to_platform_device(port->dev);
1290 int size = pdev->resource[0].end - pdev->resource[0].start + 1;
Andrew Victor1e6c9c22006-01-10 16:59:27 +00001291
Haavard Skinnemoen7192f922006-10-04 16:02:05 +02001292 if (!request_mem_region(port->mapbase, size, "atmel_serial"))
Andrew Victorafefc412006-06-19 19:53:19 +01001293 return -EBUSY;
1294
1295 if (port->flags & UPF_IOREMAP) {
1296 port->membase = ioremap(port->mapbase, size);
1297 if (port->membase == NULL) {
1298 release_mem_region(port->mapbase, size);
1299 return -ENOMEM;
1300 }
1301 }
1302
1303 return 0;
Andrew Victor1e6c9c22006-01-10 16:59:27 +00001304}
1305
1306/*
1307 * Configure/autoconfigure the port.
1308 */
Haavard Skinnemoen7192f922006-10-04 16:02:05 +02001309static void atmel_config_port(struct uart_port *port, int flags)
Andrew Victor1e6c9c22006-01-10 16:59:27 +00001310{
1311 if (flags & UART_CONFIG_TYPE) {
Haavard Skinnemoen9ab4f882006-10-04 16:02:06 +02001312 port->type = PORT_ATMEL;
Haavard Skinnemoen7192f922006-10-04 16:02:05 +02001313 atmel_request_port(port);
Andrew Victor1e6c9c22006-01-10 16:59:27 +00001314 }
1315}
1316
1317/*
1318 * Verify the new serial_struct (for TIOCSSERIAL).
1319 */
Haavard Skinnemoen7192f922006-10-04 16:02:05 +02001320static int atmel_verify_port(struct uart_port *port, struct serial_struct *ser)
Andrew Victor1e6c9c22006-01-10 16:59:27 +00001321{
1322 int ret = 0;
Haavard Skinnemoen9ab4f882006-10-04 16:02:06 +02001323 if (ser->type != PORT_UNKNOWN && ser->type != PORT_ATMEL)
Andrew Victor1e6c9c22006-01-10 16:59:27 +00001324 ret = -EINVAL;
1325 if (port->irq != ser->irq)
1326 ret = -EINVAL;
1327 if (ser->io_type != SERIAL_IO_MEM)
1328 ret = -EINVAL;
1329 if (port->uartclk / 16 != ser->baud_base)
1330 ret = -EINVAL;
1331 if ((void *)port->mapbase != ser->iomem_base)
1332 ret = -EINVAL;
1333 if (port->iobase != ser->port)
1334 ret = -EINVAL;
1335 if (ser->hub6 != 0)
1336 ret = -EINVAL;
1337 return ret;
1338}
1339
Albin Tonnerre8fe2d542009-12-09 12:31:32 -08001340#ifdef CONFIG_CONSOLE_POLL
1341static int atmel_poll_get_char(struct uart_port *port)
1342{
1343 while (!(UART_GET_CSR(port) & ATMEL_US_RXRDY))
1344 cpu_relax();
1345
1346 return UART_GET_CHAR(port);
1347}
1348
1349static void atmel_poll_put_char(struct uart_port *port, unsigned char ch)
1350{
1351 while (!(UART_GET_CSR(port) & ATMEL_US_TXRDY))
1352 cpu_relax();
1353
1354 UART_PUT_CHAR(port, ch);
1355}
1356#endif
1357
Claudio Scordinoe8faff72010-05-03 13:31:28 +01001358static int
1359atmel_ioctl(struct uart_port *port, unsigned int cmd, unsigned long arg)
1360{
1361 struct serial_rs485 rs485conf;
1362
1363 switch (cmd) {
1364 case TIOCSRS485:
1365 if (copy_from_user(&rs485conf, (struct serial_rs485 *) arg,
1366 sizeof(rs485conf)))
1367 return -EFAULT;
1368
1369 atmel_config_rs485(port, &rs485conf);
1370 break;
1371
1372 case TIOCGRS485:
1373 if (copy_to_user((struct serial_rs485 *) arg,
1374 &(to_atmel_uart_port(port)->rs485),
1375 sizeof(rs485conf)))
1376 return -EFAULT;
1377 break;
1378
1379 default:
1380 return -ENOIOCTLCMD;
1381 }
1382 return 0;
1383}
1384
1385
1386
Haavard Skinnemoen7192f922006-10-04 16:02:05 +02001387static struct uart_ops atmel_pops = {
1388 .tx_empty = atmel_tx_empty,
1389 .set_mctrl = atmel_set_mctrl,
1390 .get_mctrl = atmel_get_mctrl,
1391 .stop_tx = atmel_stop_tx,
1392 .start_tx = atmel_start_tx,
1393 .stop_rx = atmel_stop_rx,
1394 .enable_ms = atmel_enable_ms,
1395 .break_ctl = atmel_break_ctl,
1396 .startup = atmel_startup,
1397 .shutdown = atmel_shutdown,
Haavard Skinnemoen9afd5612008-07-16 21:52:46 +01001398 .flush_buffer = atmel_flush_buffer,
Haavard Skinnemoen7192f922006-10-04 16:02:05 +02001399 .set_termios = atmel_set_termios,
Viktar Palstsiuk42bd7a42011-02-09 15:26:13 +01001400 .set_ldisc = atmel_set_ldisc,
Haavard Skinnemoen7192f922006-10-04 16:02:05 +02001401 .type = atmel_type,
1402 .release_port = atmel_release_port,
1403 .request_port = atmel_request_port,
1404 .config_port = atmel_config_port,
1405 .verify_port = atmel_verify_port,
1406 .pm = atmel_serial_pm,
Claudio Scordinoe8faff72010-05-03 13:31:28 +01001407 .ioctl = atmel_ioctl,
Albin Tonnerre8fe2d542009-12-09 12:31:32 -08001408#ifdef CONFIG_CONSOLE_POLL
1409 .poll_get_char = atmel_poll_get_char,
1410 .poll_put_char = atmel_poll_put_char,
1411#endif
Andrew Victor1e6c9c22006-01-10 16:59:27 +00001412};
1413
Andrew Victorafefc412006-06-19 19:53:19 +01001414/*
1415 * Configure the port from the platform device resource info.
1416 */
Remy Bohmerb843aa22008-02-08 04:21:01 -08001417static void __devinit atmel_init_port(struct atmel_uart_port *atmel_port,
1418 struct platform_device *pdev)
Andrew Victor1e6c9c22006-01-10 16:59:27 +00001419{
Haavard Skinnemoen7192f922006-10-04 16:02:05 +02001420 struct uart_port *port = &atmel_port->uart;
Nicolas Ferre1acfc7e2011-10-12 18:06:57 +02001421 struct atmel_uart_data *pdata = pdev->dev.platform_data;
Andrew Victor1e6c9c22006-01-10 16:59:27 +00001422
Claudio Scordinoe8faff72010-05-03 13:31:28 +01001423 port->iotype = UPIO_MEM;
1424 port->flags = UPF_BOOT_AUTOCONF;
1425 port->ops = &atmel_pops;
1426 port->fifosize = 1;
Claudio Scordinoe8faff72010-05-03 13:31:28 +01001427 port->dev = &pdev->dev;
Andrew Victorafefc412006-06-19 19:53:19 +01001428 port->mapbase = pdev->resource[0].start;
1429 port->irq = pdev->resource[1].start;
1430
Remy Bohmer1ecc26b2008-02-08 04:21:05 -08001431 tasklet_init(&atmel_port->tasklet, atmel_tasklet_func,
1432 (unsigned long)port);
1433
1434 memset(&atmel_port->rx_ring, 0, sizeof(atmel_port->rx_ring));
1435
Nicolas Ferre588edbf2011-10-12 18:06:58 +02001436 if (pdata->regs) {
Haavard Skinnemoen75d35212006-10-04 16:02:08 +02001437 /* Already mapped by setup code */
Nicolas Ferre1acfc7e2011-10-12 18:06:57 +02001438 port->membase = pdata->regs;
Nicolas Ferre588edbf2011-10-12 18:06:58 +02001439 } else {
Andrew Victorafefc412006-06-19 19:53:19 +01001440 port->flags |= UPF_IOREMAP;
1441 port->membase = NULL;
1442 }
1443
Remy Bohmerb843aa22008-02-08 04:21:01 -08001444 /* for console, the clock could already be configured */
1445 if (!atmel_port->clk) {
Haavard Skinnemoen7192f922006-10-04 16:02:05 +02001446 atmel_port->clk = clk_get(&pdev->dev, "usart");
1447 clk_enable(atmel_port->clk);
1448 port->uartclk = clk_get_rate(atmel_port->clk);
David Brownell06a7f052008-11-06 12:53:40 -08001449 clk_disable(atmel_port->clk);
1450 /* only enable clock when USART is in use */
Andrew Victorafefc412006-06-19 19:53:19 +01001451 }
Chip Coldwella6670612008-02-08 04:21:06 -08001452
Nicolas Ferre588edbf2011-10-12 18:06:58 +02001453 atmel_port->use_dma_rx = pdata->use_dma_rx;
1454 atmel_port->use_dma_tx = pdata->use_dma_tx;
Nicolas Ferre1acfc7e2011-10-12 18:06:57 +02001455 atmel_port->rs485 = pdata->rs485;
Nicolas Ferre588edbf2011-10-12 18:06:58 +02001456
Claudio Scordinoe8faff72010-05-03 13:31:28 +01001457 /* Use TXEMPTY for interrupt when rs485 else TXRDY or ENDTX|TXBUFE */
1458 if (atmel_port->rs485.flags & SER_RS485_ENABLED)
1459 atmel_port->tx_done_mask = ATMEL_US_TXEMPTY;
1460 else if (atmel_use_dma_tx(port)) {
Chip Coldwella6670612008-02-08 04:21:06 -08001461 port->fifosize = PDC_BUFFER_SIZE;
Claudio Scordinoe8faff72010-05-03 13:31:28 +01001462 atmel_port->tx_done_mask = ATMEL_US_ENDTX | ATMEL_US_TXBUFE;
1463 } else {
1464 atmel_port->tx_done_mask = ATMEL_US_TXRDY;
1465 }
Andrew Victor1e6c9c22006-01-10 16:59:27 +00001466}
1467
Andrew Victorafefc412006-06-19 19:53:19 +01001468/*
1469 * Register board-specific modem-control line handlers.
1470 */
Haavard Skinnemoen71f2e2b2006-10-04 16:02:07 +02001471void __init atmel_register_uart_fns(struct atmel_port_fns *fns)
Andrew Victor1e6c9c22006-01-10 16:59:27 +00001472{
1473 if (fns->enable_ms)
Haavard Skinnemoen7192f922006-10-04 16:02:05 +02001474 atmel_pops.enable_ms = fns->enable_ms;
Andrew Victor1e6c9c22006-01-10 16:59:27 +00001475 if (fns->get_mctrl)
Haavard Skinnemoen7192f922006-10-04 16:02:05 +02001476 atmel_pops.get_mctrl = fns->get_mctrl;
Andrew Victor1e6c9c22006-01-10 16:59:27 +00001477 if (fns->set_mctrl)
Haavard Skinnemoen7192f922006-10-04 16:02:05 +02001478 atmel_pops.set_mctrl = fns->set_mctrl;
Haavard Skinnemoen71f2e2b2006-10-04 16:02:07 +02001479 atmel_open_hook = fns->open;
1480 atmel_close_hook = fns->close;
Haavard Skinnemoen7192f922006-10-04 16:02:05 +02001481 atmel_pops.pm = fns->pm;
1482 atmel_pops.set_wake = fns->set_wake;
Andrew Victor1e6c9c22006-01-10 16:59:27 +00001483}
1484
Haavard Skinnemoen749c4e62006-10-04 16:02:02 +02001485#ifdef CONFIG_SERIAL_ATMEL_CONSOLE
Haavard Skinnemoen7192f922006-10-04 16:02:05 +02001486static void atmel_console_putchar(struct uart_port *port, int ch)
Russell Kingd3587882006-03-20 20:00:09 +00001487{
Haavard Skinnemoen7192f922006-10-04 16:02:05 +02001488 while (!(UART_GET_CSR(port) & ATMEL_US_TXRDY))
Haavard Skinnemoen829dd812008-02-08 04:21:02 -08001489 cpu_relax();
Russell Kingd3587882006-03-20 20:00:09 +00001490 UART_PUT_CHAR(port, ch);
1491}
Andrew Victor1e6c9c22006-01-10 16:59:27 +00001492
1493/*
1494 * Interrupts are disabled on entering
1495 */
Haavard Skinnemoen7192f922006-10-04 16:02:05 +02001496static void atmel_console_write(struct console *co, const char *s, u_int count)
Andrew Victor1e6c9c22006-01-10 16:59:27 +00001497{
Haavard Skinnemoen7192f922006-10-04 16:02:05 +02001498 struct uart_port *port = &atmel_ports[co->index].uart;
Claudio Scordinoe8faff72010-05-03 13:31:28 +01001499 struct atmel_uart_port *atmel_port = to_atmel_uart_port(port);
Russell Kingd3587882006-03-20 20:00:09 +00001500 unsigned int status, imr;
Marc Pignat39d4c922008-04-02 13:04:42 -07001501 unsigned int pdc_tx;
Andrew Victor1e6c9c22006-01-10 16:59:27 +00001502
1503 /*
Remy Bohmerb843aa22008-02-08 04:21:01 -08001504 * First, save IMR and then disable interrupts
Andrew Victor1e6c9c22006-01-10 16:59:27 +00001505 */
Remy Bohmerb843aa22008-02-08 04:21:01 -08001506 imr = UART_GET_IMR(port);
Claudio Scordinoe8faff72010-05-03 13:31:28 +01001507 UART_PUT_IDR(port, ATMEL_US_RXRDY | atmel_port->tx_done_mask);
Andrew Victor1e6c9c22006-01-10 16:59:27 +00001508
Marc Pignat39d4c922008-04-02 13:04:42 -07001509 /* Store PDC transmit status and disable it */
1510 pdc_tx = UART_GET_PTSR(port) & ATMEL_PDC_TXTEN;
1511 UART_PUT_PTCR(port, ATMEL_PDC_TXTDIS);
1512
Haavard Skinnemoen7192f922006-10-04 16:02:05 +02001513 uart_console_write(port, s, count, atmel_console_putchar);
Andrew Victor1e6c9c22006-01-10 16:59:27 +00001514
1515 /*
Remy Bohmerb843aa22008-02-08 04:21:01 -08001516 * Finally, wait for transmitter to become empty
1517 * and restore IMR
Andrew Victor1e6c9c22006-01-10 16:59:27 +00001518 */
1519 do {
1520 status = UART_GET_CSR(port);
Haavard Skinnemoen7192f922006-10-04 16:02:05 +02001521 } while (!(status & ATMEL_US_TXRDY));
Marc Pignat39d4c922008-04-02 13:04:42 -07001522
1523 /* Restore PDC transmit status */
1524 if (pdc_tx)
1525 UART_PUT_PTCR(port, ATMEL_PDC_TXTEN);
1526
Remy Bohmerb843aa22008-02-08 04:21:01 -08001527 /* set interrupts back the way they were */
1528 UART_PUT_IER(port, imr);
Andrew Victor1e6c9c22006-01-10 16:59:27 +00001529}
1530
1531/*
Remy Bohmerb843aa22008-02-08 04:21:01 -08001532 * If the port was already initialised (eg, by a boot loader),
1533 * try to determine the current setup.
Andrew Victor1e6c9c22006-01-10 16:59:27 +00001534 */
Remy Bohmerb843aa22008-02-08 04:21:01 -08001535static void __init atmel_console_get_options(struct uart_port *port, int *baud,
1536 int *parity, int *bits)
Andrew Victor1e6c9c22006-01-10 16:59:27 +00001537{
1538 unsigned int mr, quot;
1539
Haavard Skinnemoen1c0fd822008-02-08 04:21:03 -08001540 /*
1541 * If the baud rate generator isn't running, the port wasn't
1542 * initialized by the boot loader.
1543 */
Haavard Skinnemoen9c81c5c2008-06-05 22:46:39 -07001544 quot = UART_GET_BRGR(port) & ATMEL_US_CD;
Haavard Skinnemoen1c0fd822008-02-08 04:21:03 -08001545 if (!quot)
1546 return;
Andrew Victor1e6c9c22006-01-10 16:59:27 +00001547
Haavard Skinnemoen7192f922006-10-04 16:02:05 +02001548 mr = UART_GET_MR(port) & ATMEL_US_CHRL;
1549 if (mr == ATMEL_US_CHRL_8)
Andrew Victor1e6c9c22006-01-10 16:59:27 +00001550 *bits = 8;
1551 else
1552 *bits = 7;
1553
Haavard Skinnemoen7192f922006-10-04 16:02:05 +02001554 mr = UART_GET_MR(port) & ATMEL_US_PAR;
1555 if (mr == ATMEL_US_PAR_EVEN)
Andrew Victor1e6c9c22006-01-10 16:59:27 +00001556 *parity = 'e';
Haavard Skinnemoen7192f922006-10-04 16:02:05 +02001557 else if (mr == ATMEL_US_PAR_ODD)
Andrew Victor1e6c9c22006-01-10 16:59:27 +00001558 *parity = 'o';
1559
Haavard Skinnemoen4d5e3922006-10-04 16:02:11 +02001560 /*
1561 * The serial core only rounds down when matching this to a
1562 * supported baud rate. Make sure we don't end up slightly
1563 * lower than one of those, as it would make us fall through
1564 * to a much lower baud rate than we really want.
1565 */
Haavard Skinnemoen4d5e3922006-10-04 16:02:11 +02001566 *baud = port->uartclk / (16 * (quot - 1));
Andrew Victor1e6c9c22006-01-10 16:59:27 +00001567}
1568
Haavard Skinnemoen7192f922006-10-04 16:02:05 +02001569static int __init atmel_console_setup(struct console *co, char *options)
Andrew Victor1e6c9c22006-01-10 16:59:27 +00001570{
Haavard Skinnemoen7192f922006-10-04 16:02:05 +02001571 struct uart_port *port = &atmel_ports[co->index].uart;
Andrew Victor1e6c9c22006-01-10 16:59:27 +00001572 int baud = 115200;
1573 int bits = 8;
1574 int parity = 'n';
1575 int flow = 'n';
1576
Remy Bohmerb843aa22008-02-08 04:21:01 -08001577 if (port->membase == NULL) {
1578 /* Port not initialized yet - delay setup */
Andrew Victorafefc412006-06-19 19:53:19 +01001579 return -ENODEV;
Remy Bohmerb843aa22008-02-08 04:21:01 -08001580 }
Andrew Victor1e6c9c22006-01-10 16:59:27 +00001581
David Brownell06a7f052008-11-06 12:53:40 -08001582 clk_enable(atmel_ports[co->index].clk);
1583
Remy Bohmerb843aa22008-02-08 04:21:01 -08001584 UART_PUT_IDR(port, -1);
Haavard Skinnemoen7192f922006-10-04 16:02:05 +02001585 UART_PUT_CR(port, ATMEL_US_RSTSTA | ATMEL_US_RSTRX);
1586 UART_PUT_CR(port, ATMEL_US_TXEN | ATMEL_US_RXEN);
Andrew Victor1e6c9c22006-01-10 16:59:27 +00001587
1588 if (options)
1589 uart_parse_options(options, &baud, &parity, &bits, &flow);
1590 else
Haavard Skinnemoen7192f922006-10-04 16:02:05 +02001591 atmel_console_get_options(port, &baud, &parity, &bits);
Andrew Victor1e6c9c22006-01-10 16:59:27 +00001592
1593 return uart_set_options(port, co, baud, parity, bits, flow);
1594}
1595
Haavard Skinnemoen7192f922006-10-04 16:02:05 +02001596static struct uart_driver atmel_uart;
Andrew Victor1e6c9c22006-01-10 16:59:27 +00001597
Haavard Skinnemoen7192f922006-10-04 16:02:05 +02001598static struct console atmel_console = {
1599 .name = ATMEL_DEVICENAME,
1600 .write = atmel_console_write,
Andrew Victor1e6c9c22006-01-10 16:59:27 +00001601 .device = uart_console_device,
Haavard Skinnemoen7192f922006-10-04 16:02:05 +02001602 .setup = atmel_console_setup,
Andrew Victor1e6c9c22006-01-10 16:59:27 +00001603 .flags = CON_PRINTBUFFER,
1604 .index = -1,
Haavard Skinnemoen7192f922006-10-04 16:02:05 +02001605 .data = &atmel_uart,
Andrew Victor1e6c9c22006-01-10 16:59:27 +00001606};
1607
David Brownell06a7f052008-11-06 12:53:40 -08001608#define ATMEL_CONSOLE_DEVICE (&atmel_console)
Andrew Victor1e6c9c22006-01-10 16:59:27 +00001609
Andrew Victorafefc412006-06-19 19:53:19 +01001610/*
1611 * Early console initialization (before VM subsystem initialized).
1612 */
Haavard Skinnemoen7192f922006-10-04 16:02:05 +02001613static int __init atmel_console_init(void)
Andrew Victor1e6c9c22006-01-10 16:59:27 +00001614{
Haavard Skinnemoen73e27982006-10-04 16:02:04 +02001615 if (atmel_default_console_device) {
Nicolas Ferre4cbf9f42011-10-12 18:06:59 +02001616 int id = atmel_default_console_device->id;
1617 struct atmel_uart_port *port = &atmel_ports[id];
1618
1619 set_bit(id, &atmel_ports_in_use);
1620 port->backup_imr = 0;
1621 port->uart.line = id;
1622
1623 add_preferred_console(ATMEL_DEVICENAME, id, NULL);
1624 atmel_init_port(port, atmel_default_console_device);
Haavard Skinnemoen7192f922006-10-04 16:02:05 +02001625 register_console(&atmel_console);
Andrew Victorafefc412006-06-19 19:53:19 +01001626 }
Andrew Victor1e6c9c22006-01-10 16:59:27 +00001627
Andrew Victor1e6c9c22006-01-10 16:59:27 +00001628 return 0;
1629}
Remy Bohmerb843aa22008-02-08 04:21:01 -08001630
Haavard Skinnemoen7192f922006-10-04 16:02:05 +02001631console_initcall(atmel_console_init);
Andrew Victor1e6c9c22006-01-10 16:59:27 +00001632
Andrew Victorafefc412006-06-19 19:53:19 +01001633/*
1634 * Late console initialization.
1635 */
Haavard Skinnemoen7192f922006-10-04 16:02:05 +02001636static int __init atmel_late_console_init(void)
Andrew Victorafefc412006-06-19 19:53:19 +01001637{
Remy Bohmerb843aa22008-02-08 04:21:01 -08001638 if (atmel_default_console_device
1639 && !(atmel_console.flags & CON_ENABLED))
Haavard Skinnemoen7192f922006-10-04 16:02:05 +02001640 register_console(&atmel_console);
Andrew Victorafefc412006-06-19 19:53:19 +01001641
1642 return 0;
1643}
Remy Bohmerb843aa22008-02-08 04:21:01 -08001644
Haavard Skinnemoen7192f922006-10-04 16:02:05 +02001645core_initcall(atmel_late_console_init);
Andrew Victorafefc412006-06-19 19:53:19 +01001646
Haavard Skinnemoendfa7f342008-02-08 04:21:04 -08001647static inline bool atmel_is_console_port(struct uart_port *port)
1648{
1649 return port->cons && port->cons->index == port->line;
1650}
1651
Andrew Victor1e6c9c22006-01-10 16:59:27 +00001652#else
Haavard Skinnemoen7192f922006-10-04 16:02:05 +02001653#define ATMEL_CONSOLE_DEVICE NULL
Haavard Skinnemoendfa7f342008-02-08 04:21:04 -08001654
1655static inline bool atmel_is_console_port(struct uart_port *port)
1656{
1657 return false;
1658}
Andrew Victor1e6c9c22006-01-10 16:59:27 +00001659#endif
1660
Haavard Skinnemoen7192f922006-10-04 16:02:05 +02001661static struct uart_driver atmel_uart = {
Remy Bohmerb843aa22008-02-08 04:21:01 -08001662 .owner = THIS_MODULE,
1663 .driver_name = "atmel_serial",
1664 .dev_name = ATMEL_DEVICENAME,
1665 .major = SERIAL_ATMEL_MAJOR,
1666 .minor = MINOR_START,
1667 .nr = ATMEL_MAX_UART,
1668 .cons = ATMEL_CONSOLE_DEVICE,
Andrew Victor1e6c9c22006-01-10 16:59:27 +00001669};
1670
Andrew Victorafefc412006-06-19 19:53:19 +01001671#ifdef CONFIG_PM
Haavard Skinnemoenf826caa2008-02-24 14:34:45 +01001672static bool atmel_serial_clk_will_stop(void)
1673{
1674#ifdef CONFIG_ARCH_AT91
1675 return at91_suspend_entering_slow_clock();
1676#else
1677 return false;
1678#endif
1679}
1680
Remy Bohmerb843aa22008-02-08 04:21:01 -08001681static int atmel_serial_suspend(struct platform_device *pdev,
1682 pm_message_t state)
Andrew Victor1e6c9c22006-01-10 16:59:27 +00001683{
Andrew Victorafefc412006-06-19 19:53:19 +01001684 struct uart_port *port = platform_get_drvdata(pdev);
Haavard Skinnemoenc811ab82008-02-08 04:21:08 -08001685 struct atmel_uart_port *atmel_port = to_atmel_uart_port(port);
Andrew Victor1e6c9c22006-01-10 16:59:27 +00001686
Haavard Skinnemoene1c609e2008-03-14 14:54:13 +01001687 if (atmel_is_console_port(port) && console_suspend_enabled) {
1688 /* Drain the TX shifter */
1689 while (!(UART_GET_CSR(port) & ATMEL_US_TXEMPTY))
1690 cpu_relax();
1691 }
1692
Anti Sullinf05596d2008-09-22 13:57:54 -07001693 /* we can not wake up if we're running on slow clock */
1694 atmel_port->may_wakeup = device_may_wakeup(&pdev->dev);
1695 if (atmel_serial_clk_will_stop())
1696 device_set_wakeup_enable(&pdev->dev, 0);
1697
1698 uart_suspend_port(&atmel_uart, port);
Andrew Victor1e6c9c22006-01-10 16:59:27 +00001699
1700 return 0;
1701}
1702
Haavard Skinnemoen7192f922006-10-04 16:02:05 +02001703static int atmel_serial_resume(struct platform_device *pdev)
Andrew Victorafefc412006-06-19 19:53:19 +01001704{
1705 struct uart_port *port = platform_get_drvdata(pdev);
Haavard Skinnemoenc811ab82008-02-08 04:21:08 -08001706 struct atmel_uart_port *atmel_port = to_atmel_uart_port(port);
Andrew Victorafefc412006-06-19 19:53:19 +01001707
Anti Sullinf05596d2008-09-22 13:57:54 -07001708 uart_resume_port(&atmel_uart, port);
1709 device_set_wakeup_enable(&pdev->dev, atmel_port->may_wakeup);
Andrew Victorafefc412006-06-19 19:53:19 +01001710
1711 return 0;
1712}
1713#else
Haavard Skinnemoen7192f922006-10-04 16:02:05 +02001714#define atmel_serial_suspend NULL
1715#define atmel_serial_resume NULL
Andrew Victorafefc412006-06-19 19:53:19 +01001716#endif
1717
Haavard Skinnemoen7192f922006-10-04 16:02:05 +02001718static int __devinit atmel_serial_probe(struct platform_device *pdev)
Andrew Victorafefc412006-06-19 19:53:19 +01001719{
Haavard Skinnemoen7192f922006-10-04 16:02:05 +02001720 struct atmel_uart_port *port;
Jean-Christophe PLAGNIOL-VILLARDdeba1a02011-06-21 12:17:00 +08001721 struct atmel_uart_data *pdata = pdev->dev.platform_data;
Remy Bohmer1ecc26b2008-02-08 04:21:05 -08001722 void *data;
Nicolas Ferre4cbf9f42011-10-12 18:06:59 +02001723 int ret = -ENODEV;
Andrew Victorafefc412006-06-19 19:53:19 +01001724
Haavard Skinnemoen9d09daf2009-10-26 16:50:02 -07001725 BUILD_BUG_ON(ATMEL_SERIAL_RINGSIZE & (ATMEL_SERIAL_RINGSIZE - 1));
Remy Bohmer1ecc26b2008-02-08 04:21:05 -08001726
Nicolas Ferre4cbf9f42011-10-12 18:06:59 +02001727 if (pdata)
1728 ret = pdata->num;
1729
1730 if (ret < 0)
1731 /* port id not found in platform data:
1732 * auto-enumerate it */
1733 ret = find_first_zero_bit(&atmel_ports_in_use,
1734 sizeof(atmel_ports_in_use));
1735
1736 if (ret > ATMEL_MAX_UART) {
1737 ret = -ENODEV;
1738 goto err;
1739 }
1740
1741 if (test_and_set_bit(ret, &atmel_ports_in_use)) {
1742 /* port already in use */
1743 ret = -EBUSY;
1744 goto err;
1745 }
1746
1747 port = &atmel_ports[ret];
Anti Sullinf05596d2008-09-22 13:57:54 -07001748 port->backup_imr = 0;
Nicolas Ferre4cbf9f42011-10-12 18:06:59 +02001749 port->uart.line = ret;
Anti Sullinf05596d2008-09-22 13:57:54 -07001750
Haavard Skinnemoen7192f922006-10-04 16:02:05 +02001751 atmel_init_port(port, pdev);
Andrew Victorafefc412006-06-19 19:53:19 +01001752
Chip Coldwella6670612008-02-08 04:21:06 -08001753 if (!atmel_use_dma_rx(&port->uart)) {
1754 ret = -ENOMEM;
Haavard Skinnemoen64334712008-02-08 04:21:07 -08001755 data = kmalloc(sizeof(struct atmel_uart_char)
1756 * ATMEL_SERIAL_RINGSIZE, GFP_KERNEL);
Chip Coldwella6670612008-02-08 04:21:06 -08001757 if (!data)
1758 goto err_alloc_ring;
1759 port->rx_ring.buf = data;
1760 }
Remy Bohmer1ecc26b2008-02-08 04:21:05 -08001761
Haavard Skinnemoen7192f922006-10-04 16:02:05 +02001762 ret = uart_add_one_port(&atmel_uart, &port->uart);
Haavard Skinnemoendfa7f342008-02-08 04:21:04 -08001763 if (ret)
1764 goto err_add_port;
1765
Albin Tonnerre8da14b52009-07-29 15:04:18 -07001766#ifdef CONFIG_SERIAL_ATMEL_CONSOLE
David Brownell06a7f052008-11-06 12:53:40 -08001767 if (atmel_is_console_port(&port->uart)
1768 && ATMEL_CONSOLE_DEVICE->flags & CON_ENABLED) {
1769 /*
1770 * The serial core enabled the clock for us, so undo
1771 * the clk_enable() in atmel_console_setup()
1772 */
1773 clk_disable(port->clk);
1774 }
Albin Tonnerre8da14b52009-07-29 15:04:18 -07001775#endif
David Brownell06a7f052008-11-06 12:53:40 -08001776
Haavard Skinnemoendfa7f342008-02-08 04:21:04 -08001777 device_init_wakeup(&pdev->dev, 1);
1778 platform_set_drvdata(pdev, port);
1779
Claudio Scordino5dfbd1d72011-01-13 15:45:39 -08001780 if (port->rs485.flags & SER_RS485_ENABLED) {
1781 UART_PUT_MR(&port->uart, ATMEL_US_USMODE_NORMAL);
1782 UART_PUT_CR(&port->uart, ATMEL_US_RTSEN);
1783 }
1784
Haavard Skinnemoendfa7f342008-02-08 04:21:04 -08001785 return 0;
1786
1787err_add_port:
Remy Bohmer1ecc26b2008-02-08 04:21:05 -08001788 kfree(port->rx_ring.buf);
1789 port->rx_ring.buf = NULL;
1790err_alloc_ring:
Haavard Skinnemoendfa7f342008-02-08 04:21:04 -08001791 if (!atmel_is_console_port(&port->uart)) {
Haavard Skinnemoendfa7f342008-02-08 04:21:04 -08001792 clk_put(port->clk);
1793 port->clk = NULL;
Andrew Victorafefc412006-06-19 19:53:19 +01001794 }
Nicolas Ferre4cbf9f42011-10-12 18:06:59 +02001795err:
Andrew Victorafefc412006-06-19 19:53:19 +01001796 return ret;
1797}
1798
Haavard Skinnemoen7192f922006-10-04 16:02:05 +02001799static int __devexit atmel_serial_remove(struct platform_device *pdev)
Andrew Victorafefc412006-06-19 19:53:19 +01001800{
1801 struct uart_port *port = platform_get_drvdata(pdev);
Haavard Skinnemoenc811ab82008-02-08 04:21:08 -08001802 struct atmel_uart_port *atmel_port = to_atmel_uart_port(port);
Andrew Victorafefc412006-06-19 19:53:19 +01001803 int ret = 0;
1804
Andrew Victorafefc412006-06-19 19:53:19 +01001805 device_init_wakeup(&pdev->dev, 0);
1806 platform_set_drvdata(pdev, NULL);
1807
Haavard Skinnemoendfa7f342008-02-08 04:21:04 -08001808 ret = uart_remove_one_port(&atmel_uart, port);
1809
Remy Bohmer1ecc26b2008-02-08 04:21:05 -08001810 tasklet_kill(&atmel_port->tasklet);
1811 kfree(atmel_port->rx_ring.buf);
1812
Haavard Skinnemoendfa7f342008-02-08 04:21:04 -08001813 /* "port" is allocated statically, so we shouldn't free it */
1814
Nicolas Ferre4cbf9f42011-10-12 18:06:59 +02001815 clear_bit(port->line, &atmel_ports_in_use);
1816
Haavard Skinnemoendfa7f342008-02-08 04:21:04 -08001817 clk_put(atmel_port->clk);
Andrew Victorafefc412006-06-19 19:53:19 +01001818
1819 return ret;
1820}
1821
Haavard Skinnemoen7192f922006-10-04 16:02:05 +02001822static struct platform_driver atmel_serial_driver = {
1823 .probe = atmel_serial_probe,
1824 .remove = __devexit_p(atmel_serial_remove),
1825 .suspend = atmel_serial_suspend,
1826 .resume = atmel_serial_resume,
Andrew Victorafefc412006-06-19 19:53:19 +01001827 .driver = {
Haavard Skinnemoen1e8ea802006-10-04 16:02:03 +02001828 .name = "atmel_usart",
Andrew Victorafefc412006-06-19 19:53:19 +01001829 .owner = THIS_MODULE,
1830 },
1831};
1832
Haavard Skinnemoen7192f922006-10-04 16:02:05 +02001833static int __init atmel_serial_init(void)
Andrew Victorafefc412006-06-19 19:53:19 +01001834{
1835 int ret;
1836
Haavard Skinnemoen7192f922006-10-04 16:02:05 +02001837 ret = uart_register_driver(&atmel_uart);
Andrew Victorafefc412006-06-19 19:53:19 +01001838 if (ret)
1839 return ret;
1840
Haavard Skinnemoen7192f922006-10-04 16:02:05 +02001841 ret = platform_driver_register(&atmel_serial_driver);
Andrew Victorafefc412006-06-19 19:53:19 +01001842 if (ret)
Haavard Skinnemoen7192f922006-10-04 16:02:05 +02001843 uart_unregister_driver(&atmel_uart);
Andrew Victorafefc412006-06-19 19:53:19 +01001844
1845 return ret;
1846}
1847
Haavard Skinnemoen7192f922006-10-04 16:02:05 +02001848static void __exit atmel_serial_exit(void)
Andrew Victor1e6c9c22006-01-10 16:59:27 +00001849{
Haavard Skinnemoen7192f922006-10-04 16:02:05 +02001850 platform_driver_unregister(&atmel_serial_driver);
1851 uart_unregister_driver(&atmel_uart);
Andrew Victor1e6c9c22006-01-10 16:59:27 +00001852}
1853
Haavard Skinnemoen7192f922006-10-04 16:02:05 +02001854module_init(atmel_serial_init);
1855module_exit(atmel_serial_exit);
Andrew Victor1e6c9c22006-01-10 16:59:27 +00001856
1857MODULE_AUTHOR("Rick Bronson");
Haavard Skinnemoen7192f922006-10-04 16:02:05 +02001858MODULE_DESCRIPTION("Atmel AT91 / AT32 serial port driver");
Andrew Victor1e6c9c22006-01-10 16:59:27 +00001859MODULE_LICENSE("GPL");
Kay Sieverse169c132008-04-15 14:34:35 -07001860MODULE_ALIAS("platform:atmel_usart");