blob: c549110509bbed97c12167e3b9e48e3b5e2c77dc [file] [log] [blame]
Robert Love04896a72009-06-22 18:43:11 +01001/*
Jovi Zhang99edb3d2011-03-30 05:30:41 -04002 * Driver for msm7k serial device and console
Robert Love04896a72009-06-22 18:43:11 +01003 *
4 * Copyright (C) 2007 Google, Inc.
5 * Author: Robert Love <rlove@google.com>
Stepan Moskovchenkoec8f29e2010-12-21 12:38:05 -08006 * Copyright (c) 2011, Code Aurora Forum. All rights reserved.
Robert Love04896a72009-06-22 18:43:11 +01007 *
8 * This software is licensed under the terms of the GNU General Public
9 * License version 2, as published by the Free Software Foundation, and
10 * may be copied, distributed, and modified under those terms.
11 *
12 * This program is distributed in the hope that it will be useful,
13 * but WITHOUT ANY WARRANTY; without even the implied warranty of
14 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
15 * GNU General Public License for more details.
16 */
17
18#if defined(CONFIG_SERIAL_MSM_CONSOLE) && defined(CONFIG_MAGIC_SYSRQ)
19# define SUPPORT_SYSRQ
20#endif
21
David Browncfdad2a2011-08-04 01:55:24 -070022#include <linux/atomic.h>
Robert Love04896a72009-06-22 18:43:11 +010023#include <linux/hrtimer.h>
24#include <linux/module.h>
25#include <linux/io.h>
26#include <linux/ioport.h>
27#include <linux/irq.h>
28#include <linux/init.h>
29#include <linux/console.h>
30#include <linux/tty.h>
31#include <linux/tty_flip.h>
32#include <linux/serial_core.h>
33#include <linux/serial.h>
34#include <linux/clk.h>
35#include <linux/platform_device.h>
Stepan Moskovchenkoec8f29e2010-12-21 12:38:05 -080036#include <linux/delay.h>
David Browncfdad2a2011-08-04 01:55:24 -070037#include <linux/of.h>
38#include <linux/of_device.h>
Robert Love04896a72009-06-22 18:43:11 +010039
40#include "msm_serial.h"
41
Stephen Boydf7e54d72014-01-14 12:34:55 -080042enum {
43 UARTDM_1P1 = 1,
44 UARTDM_1P2,
45 UARTDM_1P3,
46 UARTDM_1P4,
47};
48
Robert Love04896a72009-06-22 18:43:11 +010049struct msm_port {
50 struct uart_port uart;
51 char name[16];
52 struct clk *clk;
Stepan Moskovchenkoec8f29e2010-12-21 12:38:05 -080053 struct clk *pclk;
Robert Love04896a72009-06-22 18:43:11 +010054 unsigned int imr;
Stepan Moskovchenkoec8f29e2010-12-21 12:38:05 -080055 int is_uartdm;
56 unsigned int old_snap_state;
Robert Love04896a72009-06-22 18:43:11 +010057};
58
Stephen Boyd4a5662d2013-07-24 11:37:28 -070059static inline void wait_for_xmitr(struct uart_port *port)
Stepan Moskovchenkoec8f29e2010-12-21 12:38:05 -080060{
Stephen Boyd4a5662d2013-07-24 11:37:28 -070061 while (!(msm_read(port, UART_SR) & UART_SR_TX_EMPTY)) {
62 if (msm_read(port, UART_ISR) & UART_ISR_TX_READY)
63 break;
64 udelay(1);
65 }
66 msm_write(port, UART_CR_CMD_RESET_TX_READY, UART_CR);
Stepan Moskovchenkoec8f29e2010-12-21 12:38:05 -080067}
68
Robert Love04896a72009-06-22 18:43:11 +010069static void msm_stop_tx(struct uart_port *port)
70{
71 struct msm_port *msm_port = UART_TO_MSM(port);
72
73 msm_port->imr &= ~UART_IMR_TXLEV;
74 msm_write(port, msm_port->imr, UART_IMR);
75}
76
77static void msm_start_tx(struct uart_port *port)
78{
79 struct msm_port *msm_port = UART_TO_MSM(port);
80
81 msm_port->imr |= UART_IMR_TXLEV;
82 msm_write(port, msm_port->imr, UART_IMR);
83}
84
85static void msm_stop_rx(struct uart_port *port)
86{
87 struct msm_port *msm_port = UART_TO_MSM(port);
88
89 msm_port->imr &= ~(UART_IMR_RXLEV | UART_IMR_RXSTALE);
90 msm_write(port, msm_port->imr, UART_IMR);
91}
92
93static void msm_enable_ms(struct uart_port *port)
94{
95 struct msm_port *msm_port = UART_TO_MSM(port);
96
97 msm_port->imr |= UART_IMR_DELTA_CTS;
98 msm_write(port, msm_port->imr, UART_IMR);
99}
100
Stepan Moskovchenkoec8f29e2010-12-21 12:38:05 -0800101static void handle_rx_dm(struct uart_port *port, unsigned int misr)
102{
Jiri Slaby92a19f92013-01-03 15:53:03 +0100103 struct tty_port *tport = &port->state->port;
Stepan Moskovchenkoec8f29e2010-12-21 12:38:05 -0800104 unsigned int sr;
105 int count = 0;
106 struct msm_port *msm_port = UART_TO_MSM(port);
107
108 if ((msm_read(port, UART_SR) & UART_SR_OVERRUN)) {
109 port->icount.overrun++;
Jiri Slaby92a19f92013-01-03 15:53:03 +0100110 tty_insert_flip_char(tport, 0, TTY_OVERRUN);
Stepan Moskovchenkoec8f29e2010-12-21 12:38:05 -0800111 msm_write(port, UART_CR_CMD_RESET_ERR, UART_CR);
112 }
113
114 if (misr & UART_IMR_RXSTALE) {
115 count = msm_read(port, UARTDM_RX_TOTAL_SNAP) -
116 msm_port->old_snap_state;
117 msm_port->old_snap_state = 0;
118 } else {
119 count = 4 * (msm_read(port, UART_RFWR));
120 msm_port->old_snap_state += count;
121 }
122
123 /* TODO: Precise error reporting */
124
125 port->icount.rx += count;
126
127 while (count > 0) {
Stephen Boyd68252422014-06-30 14:54:01 -0700128 unsigned char buf[4];
Stepan Moskovchenkoec8f29e2010-12-21 12:38:05 -0800129
130 sr = msm_read(port, UART_SR);
131 if ((sr & UART_SR_RX_READY) == 0) {
132 msm_port->old_snap_state -= count;
133 break;
134 }
Stephen Boyd68252422014-06-30 14:54:01 -0700135 ioread32_rep(port->membase + UARTDM_RF, buf, 1);
Stepan Moskovchenkoec8f29e2010-12-21 12:38:05 -0800136 if (sr & UART_SR_RX_BREAK) {
137 port->icount.brk++;
138 if (uart_handle_break(port))
139 continue;
140 } else if (sr & UART_SR_PAR_FRAME_ERR)
141 port->icount.frame++;
142
143 /* TODO: handle sysrq */
Stephen Boyd68252422014-06-30 14:54:01 -0700144 tty_insert_flip_string(tport, buf, min(count, 4));
Stepan Moskovchenkoec8f29e2010-12-21 12:38:05 -0800145 count -= 4;
146 }
147
Viresh Kumarf77232d2013-08-19 20:14:20 +0530148 spin_unlock(&port->lock);
Jiri Slaby2e124b42013-01-03 15:53:06 +0100149 tty_flip_buffer_push(tport);
Viresh Kumarf77232d2013-08-19 20:14:20 +0530150 spin_lock(&port->lock);
151
Stepan Moskovchenkoec8f29e2010-12-21 12:38:05 -0800152 if (misr & (UART_IMR_RXSTALE))
153 msm_write(port, UART_CR_CMD_RESET_STALE_INT, UART_CR);
154 msm_write(port, 0xFFFFFF, UARTDM_DMRX);
155 msm_write(port, UART_CR_CMD_STALE_EVENT_ENABLE, UART_CR);
156}
157
Robert Love04896a72009-06-22 18:43:11 +0100158static void handle_rx(struct uart_port *port)
159{
Jiri Slaby92a19f92013-01-03 15:53:03 +0100160 struct tty_port *tport = &port->state->port;
Robert Love04896a72009-06-22 18:43:11 +0100161 unsigned int sr;
162
163 /*
164 * Handle overrun. My understanding of the hardware is that overrun
165 * is not tied to the RX buffer, so we handle the case out of band.
166 */
167 if ((msm_read(port, UART_SR) & UART_SR_OVERRUN)) {
168 port->icount.overrun++;
Jiri Slaby92a19f92013-01-03 15:53:03 +0100169 tty_insert_flip_char(tport, 0, TTY_OVERRUN);
Robert Love04896a72009-06-22 18:43:11 +0100170 msm_write(port, UART_CR_CMD_RESET_ERR, UART_CR);
171 }
172
173 /* and now the main RX loop */
174 while ((sr = msm_read(port, UART_SR)) & UART_SR_RX_READY) {
175 unsigned int c;
176 char flag = TTY_NORMAL;
177
178 c = msm_read(port, UART_RF);
179
180 if (sr & UART_SR_RX_BREAK) {
181 port->icount.brk++;
182 if (uart_handle_break(port))
183 continue;
184 } else if (sr & UART_SR_PAR_FRAME_ERR) {
185 port->icount.frame++;
186 } else {
187 port->icount.rx++;
188 }
189
190 /* Mask conditions we're ignorning. */
191 sr &= port->read_status_mask;
192
193 if (sr & UART_SR_RX_BREAK) {
194 flag = TTY_BREAK;
195 } else if (sr & UART_SR_PAR_FRAME_ERR) {
196 flag = TTY_FRAME;
197 }
198
199 if (!uart_handle_sysrq_char(port, c))
Jiri Slaby92a19f92013-01-03 15:53:03 +0100200 tty_insert_flip_char(tport, c, flag);
Robert Love04896a72009-06-22 18:43:11 +0100201 }
202
Viresh Kumarf77232d2013-08-19 20:14:20 +0530203 spin_unlock(&port->lock);
Jiri Slaby2e124b42013-01-03 15:53:06 +0100204 tty_flip_buffer_push(tport);
Viresh Kumarf77232d2013-08-19 20:14:20 +0530205 spin_lock(&port->lock);
Robert Love04896a72009-06-22 18:43:11 +0100206}
207
Stephen Boyd17fae282013-07-24 11:37:31 -0700208static void reset_dm_count(struct uart_port *port, int count)
Stepan Moskovchenkoec8f29e2010-12-21 12:38:05 -0800209{
Stephen Boyd4a5662d2013-07-24 11:37:28 -0700210 wait_for_xmitr(port);
Stephen Boyd17fae282013-07-24 11:37:31 -0700211 msm_write(port, count, UARTDM_NCF_TX);
Stephen Boyd4a5662d2013-07-24 11:37:28 -0700212 msm_read(port, UARTDM_NCF_TX);
Stepan Moskovchenkoec8f29e2010-12-21 12:38:05 -0800213}
214
Robert Love04896a72009-06-22 18:43:11 +0100215static void handle_tx(struct uart_port *port)
216{
Alan Coxebd2c8f2009-09-19 13:13:28 -0700217 struct circ_buf *xmit = &port->state->xmit;
Robert Love04896a72009-06-22 18:43:11 +0100218 struct msm_port *msm_port = UART_TO_MSM(port);
Stephen Boyd17fae282013-07-24 11:37:31 -0700219 unsigned int tx_count, num_chars;
220 unsigned int tf_pointer = 0;
Stephen Boyd68252422014-06-30 14:54:01 -0700221 void __iomem *tf;
222
223 if (msm_port->is_uartdm)
224 tf = port->membase + UARTDM_TF;
225 else
226 tf = port->membase + UART_TF;
Stephen Boyd17fae282013-07-24 11:37:31 -0700227
228 tx_count = uart_circ_chars_pending(xmit);
229 tx_count = min3(tx_count, (unsigned int)UART_XMIT_SIZE - xmit->tail,
230 port->fifosize);
Robert Love04896a72009-06-22 18:43:11 +0100231
232 if (port->x_char) {
Stepan Moskovchenkoec8f29e2010-12-21 12:38:05 -0800233 if (msm_port->is_uartdm)
Stephen Boyd17fae282013-07-24 11:37:31 -0700234 reset_dm_count(port, tx_count + 1);
Stepan Moskovchenkoec8f29e2010-12-21 12:38:05 -0800235
Stephen Boyd68252422014-06-30 14:54:01 -0700236 iowrite8_rep(tf, &port->x_char, 1);
Robert Love04896a72009-06-22 18:43:11 +0100237 port->icount.tx++;
238 port->x_char = 0;
Stephen Boyd17fae282013-07-24 11:37:31 -0700239 } else if (tx_count && msm_port->is_uartdm) {
240 reset_dm_count(port, tx_count);
Robert Love04896a72009-06-22 18:43:11 +0100241 }
242
Stephen Boyd17fae282013-07-24 11:37:31 -0700243 while (tf_pointer < tx_count) {
244 int i;
245 char buf[4] = { 0 };
Stepan Moskovchenkoec8f29e2010-12-21 12:38:05 -0800246
Stephen Boyd17fae282013-07-24 11:37:31 -0700247 if (!(msm_read(port, UART_SR) & UART_SR_TX_READY))
Robert Love04896a72009-06-22 18:43:11 +0100248 break;
Robert Love04896a72009-06-22 18:43:11 +0100249
Stepan Moskovchenkoec8f29e2010-12-21 12:38:05 -0800250 if (msm_port->is_uartdm)
Jingoo Han4f749f22013-08-08 17:38:20 +0900251 num_chars = min(tx_count - tf_pointer,
252 (unsigned int)sizeof(buf));
Stephen Boyd17fae282013-07-24 11:37:31 -0700253 else
254 num_chars = 1;
Robert Love04896a72009-06-22 18:43:11 +0100255
Stephen Boyd17fae282013-07-24 11:37:31 -0700256 for (i = 0; i < num_chars; i++) {
257 buf[i] = xmit->buf[xmit->tail + i];
258 port->icount.tx++;
259 }
260
Stephen Boyd68252422014-06-30 14:54:01 -0700261 iowrite32_rep(tf, buf, 1);
Stephen Boyd17fae282013-07-24 11:37:31 -0700262 xmit->tail = (xmit->tail + num_chars) & (UART_XMIT_SIZE - 1);
263 tf_pointer += num_chars;
Robert Love04896a72009-06-22 18:43:11 +0100264 }
265
Stephen Boyd17fae282013-07-24 11:37:31 -0700266 /* disable tx interrupts if nothing more to send */
267 if (uart_circ_empty(xmit))
268 msm_stop_tx(port);
269
Robert Love04896a72009-06-22 18:43:11 +0100270 if (uart_circ_chars_pending(xmit) < WAKEUP_CHARS)
271 uart_write_wakeup(port);
272}
273
274static void handle_delta_cts(struct uart_port *port)
275{
276 msm_write(port, UART_CR_CMD_RESET_CTS, UART_CR);
277 port->icount.cts++;
Alan Coxbdc04e32009-09-19 13:13:31 -0700278 wake_up_interruptible(&port->state->port.delta_msr_wait);
Robert Love04896a72009-06-22 18:43:11 +0100279}
280
281static irqreturn_t msm_irq(int irq, void *dev_id)
282{
283 struct uart_port *port = dev_id;
284 struct msm_port *msm_port = UART_TO_MSM(port);
285 unsigned int misr;
286
287 spin_lock(&port->lock);
288 misr = msm_read(port, UART_MISR);
289 msm_write(port, 0, UART_IMR); /* disable interrupt */
290
Stepan Moskovchenkoec8f29e2010-12-21 12:38:05 -0800291 if (misr & (UART_IMR_RXLEV | UART_IMR_RXSTALE)) {
292 if (msm_port->is_uartdm)
293 handle_rx_dm(port, misr);
294 else
295 handle_rx(port);
296 }
Robert Love04896a72009-06-22 18:43:11 +0100297 if (misr & UART_IMR_TXLEV)
298 handle_tx(port);
299 if (misr & UART_IMR_DELTA_CTS)
300 handle_delta_cts(port);
301
302 msm_write(port, msm_port->imr, UART_IMR); /* restore interrupt */
303 spin_unlock(&port->lock);
304
305 return IRQ_HANDLED;
306}
307
308static unsigned int msm_tx_empty(struct uart_port *port)
309{
310 return (msm_read(port, UART_SR) & UART_SR_TX_EMPTY) ? TIOCSER_TEMT : 0;
311}
312
313static unsigned int msm_get_mctrl(struct uart_port *port)
314{
315 return TIOCM_CAR | TIOCM_CTS | TIOCM_DSR | TIOCM_RTS;
316}
317
Stepan Moskovchenkoec8f29e2010-12-21 12:38:05 -0800318
319static void msm_reset(struct uart_port *port)
320{
Stephen Boydf7e54d72014-01-14 12:34:55 -0800321 struct msm_port *msm_port = UART_TO_MSM(port);
322
Stepan Moskovchenkoec8f29e2010-12-21 12:38:05 -0800323 /* reset everything */
324 msm_write(port, UART_CR_CMD_RESET_RX, UART_CR);
325 msm_write(port, UART_CR_CMD_RESET_TX, UART_CR);
326 msm_write(port, UART_CR_CMD_RESET_ERR, UART_CR);
327 msm_write(port, UART_CR_CMD_RESET_BREAK_INT, UART_CR);
328 msm_write(port, UART_CR_CMD_RESET_CTS, UART_CR);
329 msm_write(port, UART_CR_CMD_SET_RFR, UART_CR);
Stephen Boydf7e54d72014-01-14 12:34:55 -0800330
331 /* Disable DM modes */
332 if (msm_port->is_uartdm)
333 msm_write(port, 0, UARTDM_DMEN);
Stepan Moskovchenkoec8f29e2010-12-21 12:38:05 -0800334}
335
Stephen Boydf8fb9522013-07-24 11:37:29 -0700336static void msm_set_mctrl(struct uart_port *port, unsigned int mctrl)
Robert Love04896a72009-06-22 18:43:11 +0100337{
338 unsigned int mr;
Robert Love04896a72009-06-22 18:43:11 +0100339 mr = msm_read(port, UART_MR1);
340
341 if (!(mctrl & TIOCM_RTS)) {
342 mr &= ~UART_MR1_RX_RDY_CTL;
343 msm_write(port, mr, UART_MR1);
344 msm_write(port, UART_CR_CMD_RESET_RFR, UART_CR);
345 } else {
346 mr |= UART_MR1_RX_RDY_CTL;
347 msm_write(port, mr, UART_MR1);
348 }
349}
350
351static void msm_break_ctl(struct uart_port *port, int break_ctl)
352{
353 if (break_ctl)
354 msm_write(port, UART_CR_CMD_START_BREAK, UART_CR);
355 else
356 msm_write(port, UART_CR_CMD_STOP_BREAK, UART_CR);
357}
358
Stephen Boyd6909dad2013-07-24 11:37:30 -0700359struct msm_baud_map {
360 u16 divisor;
361 u8 code;
362 u8 rxstale;
363};
364
365static const struct msm_baud_map *
366msm_find_best_baud(struct uart_port *port, unsigned int baud)
367{
368 unsigned int i, divisor;
369 const struct msm_baud_map *entry;
370 static const struct msm_baud_map table[] = {
371 { 1536, 0x00, 1 },
372 { 768, 0x11, 1 },
373 { 384, 0x22, 1 },
374 { 192, 0x33, 1 },
375 { 96, 0x44, 1 },
376 { 48, 0x55, 1 },
377 { 32, 0x66, 1 },
378 { 24, 0x77, 1 },
379 { 16, 0x88, 1 },
380 { 12, 0x99, 6 },
381 { 8, 0xaa, 6 },
382 { 6, 0xbb, 6 },
383 { 4, 0xcc, 6 },
384 { 3, 0xdd, 8 },
385 { 2, 0xee, 16 },
386 { 1, 0xff, 31 },
387 };
388
389 divisor = uart_get_divisor(port, baud);
390
391 for (i = 0, entry = table; i < ARRAY_SIZE(table); i++, entry++)
392 if (entry->divisor <= divisor)
393 break;
394
395 return entry; /* Default to smallest divider */
396}
397
Alan Cox44da59e2009-06-22 18:43:18 +0100398static int msm_set_baud_rate(struct uart_port *port, unsigned int baud)
Robert Love04896a72009-06-22 18:43:11 +0100399{
Stephen Boyd6909dad2013-07-24 11:37:30 -0700400 unsigned int rxstale, watermark;
Stepan Moskovchenkoec8f29e2010-12-21 12:38:05 -0800401 struct msm_port *msm_port = UART_TO_MSM(port);
Stephen Boyd6909dad2013-07-24 11:37:30 -0700402 const struct msm_baud_map *entry;
Robert Love04896a72009-06-22 18:43:11 +0100403
Stephen Boyd6909dad2013-07-24 11:37:30 -0700404 entry = msm_find_best_baud(port, baud);
Robert Love04896a72009-06-22 18:43:11 +0100405
Stepan Moskovchenkoec8f29e2010-12-21 12:38:05 -0800406 if (msm_port->is_uartdm)
407 msm_write(port, UART_CR_CMD_RESET_RX, UART_CR);
408
Stephen Boyd6909dad2013-07-24 11:37:30 -0700409 msm_write(port, entry->code, UART_CSR);
Robert Love04896a72009-06-22 18:43:11 +0100410
411 /* RX stale watermark */
Stephen Boyd6909dad2013-07-24 11:37:30 -0700412 rxstale = entry->rxstale;
Robert Love04896a72009-06-22 18:43:11 +0100413 watermark = UART_IPR_STALE_LSB & rxstale;
414 watermark |= UART_IPR_RXSTALE_LAST;
415 watermark |= UART_IPR_STALE_TIMEOUT_MSB & (rxstale << 2);
416 msm_write(port, watermark, UART_IPR);
417
418 /* set RX watermark */
419 watermark = (port->fifosize * 3) / 4;
420 msm_write(port, watermark, UART_RFWR);
421
422 /* set TX watermark */
423 msm_write(port, 10, UART_TFWR);
Alan Cox44da59e2009-06-22 18:43:18 +0100424
Stepan Moskovchenkoec8f29e2010-12-21 12:38:05 -0800425 if (msm_port->is_uartdm) {
426 msm_write(port, UART_CR_CMD_RESET_STALE_INT, UART_CR);
427 msm_write(port, 0xFFFFFF, UARTDM_DMRX);
428 msm_write(port, UART_CR_CMD_STALE_EVENT_ENABLE, UART_CR);
429 }
430
Alan Cox44da59e2009-06-22 18:43:18 +0100431 return baud;
Robert Love04896a72009-06-22 18:43:11 +0100432}
433
Robert Love04896a72009-06-22 18:43:11 +0100434
435static void msm_init_clock(struct uart_port *port)
436{
437 struct msm_port *msm_port = UART_TO_MSM(port);
438
Stephen Boydf98cf832013-06-17 10:43:08 -0700439 clk_prepare_enable(msm_port->clk);
Stephen Boydbfaddb72013-08-20 23:48:02 -0700440 clk_prepare_enable(msm_port->pclk);
Abhijeet Dharmapurikar18c79d72010-05-20 15:20:23 -0700441 msm_serial_set_mnd_regs(port);
Robert Love04896a72009-06-22 18:43:11 +0100442}
443
444static int msm_startup(struct uart_port *port)
445{
446 struct msm_port *msm_port = UART_TO_MSM(port);
447 unsigned int data, rfr_level;
448 int ret;
449
450 snprintf(msm_port->name, sizeof(msm_port->name),
451 "msm_serial%d", port->line);
452
453 ret = request_irq(port->irq, msm_irq, IRQF_TRIGGER_HIGH,
454 msm_port->name, port);
455 if (unlikely(ret))
456 return ret;
457
458 msm_init_clock(port);
459
460 if (likely(port->fifosize > 12))
461 rfr_level = port->fifosize - 12;
462 else
463 rfr_level = port->fifosize;
464
465 /* set automatic RFR level */
466 data = msm_read(port, UART_MR1);
467 data &= ~UART_MR1_AUTO_RFR_LEVEL1;
468 data &= ~UART_MR1_AUTO_RFR_LEVEL0;
469 data |= UART_MR1_AUTO_RFR_LEVEL1 & (rfr_level << 2);
470 data |= UART_MR1_AUTO_RFR_LEVEL0 & rfr_level;
471 msm_write(port, data, UART_MR1);
472
473 /* make sure that RXSTALE count is non-zero */
474 data = msm_read(port, UART_IPR);
475 if (unlikely(!data)) {
476 data |= UART_IPR_RXSTALE_LAST;
477 data |= UART_IPR_STALE_LSB;
478 msm_write(port, data, UART_IPR);
479 }
480
Stepan Moskovchenkoec8f29e2010-12-21 12:38:05 -0800481 data = 0;
482 if (!port->cons || (port->cons && !(port->cons->flags & CON_ENABLED))) {
483 msm_write(port, UART_CR_CMD_PROTECTION_EN, UART_CR);
484 msm_reset(port);
485 data = UART_CR_TX_ENABLE;
486 }
Robert Love04896a72009-06-22 18:43:11 +0100487
Stepan Moskovchenkoec8f29e2010-12-21 12:38:05 -0800488 data |= UART_CR_RX_ENABLE;
489 msm_write(port, data, UART_CR); /* enable TX & RX */
490
491 /* Make sure IPR is not 0 to start with*/
492 if (msm_port->is_uartdm)
493 msm_write(port, UART_IPR_STALE_LSB, UART_IPR);
Robert Love04896a72009-06-22 18:43:11 +0100494
495 /* turn on RX and CTS interrupts */
496 msm_port->imr = UART_IMR_RXLEV | UART_IMR_RXSTALE |
497 UART_IMR_CURRENT_CTS;
Robert Love04896a72009-06-22 18:43:11 +0100498
Stepan Moskovchenkoec8f29e2010-12-21 12:38:05 -0800499 if (msm_port->is_uartdm) {
500 msm_write(port, 0xFFFFFF, UARTDM_DMRX);
501 msm_write(port, UART_CR_CMD_RESET_STALE_INT, UART_CR);
502 msm_write(port, UART_CR_CMD_STALE_EVENT_ENABLE, UART_CR);
503 }
504
505 msm_write(port, msm_port->imr, UART_IMR);
Robert Love04896a72009-06-22 18:43:11 +0100506 return 0;
507}
508
509static void msm_shutdown(struct uart_port *port)
510{
511 struct msm_port *msm_port = UART_TO_MSM(port);
512
513 msm_port->imr = 0;
514 msm_write(port, 0, UART_IMR); /* disable interrupts */
515
Stephen Boydf98cf832013-06-17 10:43:08 -0700516 clk_disable_unprepare(msm_port->clk);
Robert Love04896a72009-06-22 18:43:11 +0100517
518 free_irq(port->irq, port);
519}
520
521static void msm_set_termios(struct uart_port *port, struct ktermios *termios,
522 struct ktermios *old)
523{
524 unsigned long flags;
525 unsigned int baud, mr;
526
527 spin_lock_irqsave(&port->lock, flags);
528
529 /* calculate and set baud rate */
530 baud = uart_get_baud_rate(port, termios, old, 300, 115200);
Alan Cox44da59e2009-06-22 18:43:18 +0100531 baud = msm_set_baud_rate(port, baud);
532 if (tty_termios_baud_rate(termios))
533 tty_termios_encode_baud_rate(termios, baud, baud);
Stepan Moskovchenkoec8f29e2010-12-21 12:38:05 -0800534
Robert Love04896a72009-06-22 18:43:11 +0100535 /* calculate parity */
536 mr = msm_read(port, UART_MR2);
537 mr &= ~UART_MR2_PARITY_MODE;
538 if (termios->c_cflag & PARENB) {
539 if (termios->c_cflag & PARODD)
540 mr |= UART_MR2_PARITY_MODE_ODD;
541 else if (termios->c_cflag & CMSPAR)
542 mr |= UART_MR2_PARITY_MODE_SPACE;
543 else
544 mr |= UART_MR2_PARITY_MODE_EVEN;
545 }
546
547 /* calculate bits per char */
548 mr &= ~UART_MR2_BITS_PER_CHAR;
549 switch (termios->c_cflag & CSIZE) {
550 case CS5:
551 mr |= UART_MR2_BITS_PER_CHAR_5;
552 break;
553 case CS6:
554 mr |= UART_MR2_BITS_PER_CHAR_6;
555 break;
556 case CS7:
557 mr |= UART_MR2_BITS_PER_CHAR_7;
558 break;
559 case CS8:
560 default:
561 mr |= UART_MR2_BITS_PER_CHAR_8;
562 break;
563 }
564
565 /* calculate stop bits */
566 mr &= ~(UART_MR2_STOP_BIT_LEN_ONE | UART_MR2_STOP_BIT_LEN_TWO);
567 if (termios->c_cflag & CSTOPB)
568 mr |= UART_MR2_STOP_BIT_LEN_TWO;
569 else
570 mr |= UART_MR2_STOP_BIT_LEN_ONE;
571
572 /* set parity, bits per char, and stop bit */
573 msm_write(port, mr, UART_MR2);
574
575 /* calculate and set hardware flow control */
576 mr = msm_read(port, UART_MR1);
577 mr &= ~(UART_MR1_CTS_CTL | UART_MR1_RX_RDY_CTL);
578 if (termios->c_cflag & CRTSCTS) {
579 mr |= UART_MR1_CTS_CTL;
580 mr |= UART_MR1_RX_RDY_CTL;
581 }
582 msm_write(port, mr, UART_MR1);
583
584 /* Configure status bits to ignore based on termio flags. */
585 port->read_status_mask = 0;
586 if (termios->c_iflag & INPCK)
587 port->read_status_mask |= UART_SR_PAR_FRAME_ERR;
Peter Hurleyef8b9dd2014-06-16 08:10:41 -0400588 if (termios->c_iflag & (IGNBRK | BRKINT | PARMRK))
Robert Love04896a72009-06-22 18:43:11 +0100589 port->read_status_mask |= UART_SR_RX_BREAK;
590
591 uart_update_timeout(port, termios->c_cflag, baud);
592
593 spin_unlock_irqrestore(&port->lock, flags);
594}
595
596static const char *msm_type(struct uart_port *port)
597{
598 return "MSM";
599}
600
601static void msm_release_port(struct uart_port *port)
602{
603 struct platform_device *pdev = to_platform_device(port->dev);
Stepan Moskovchenkoec8f29e2010-12-21 12:38:05 -0800604 struct resource *uart_resource;
Robert Love04896a72009-06-22 18:43:11 +0100605 resource_size_t size;
606
Stepan Moskovchenkoec8f29e2010-12-21 12:38:05 -0800607 uart_resource = platform_get_resource(pdev, IORESOURCE_MEM, 0);
608 if (unlikely(!uart_resource))
Robert Love04896a72009-06-22 18:43:11 +0100609 return;
Stepan Moskovchenkoec8f29e2010-12-21 12:38:05 -0800610 size = resource_size(uart_resource);
Robert Love04896a72009-06-22 18:43:11 +0100611
612 release_mem_region(port->mapbase, size);
613 iounmap(port->membase);
614 port->membase = NULL;
615}
616
617static int msm_request_port(struct uart_port *port)
618{
619 struct platform_device *pdev = to_platform_device(port->dev);
Stepan Moskovchenkoec8f29e2010-12-21 12:38:05 -0800620 struct resource *uart_resource;
Robert Love04896a72009-06-22 18:43:11 +0100621 resource_size_t size;
Stepan Moskovchenkoec8f29e2010-12-21 12:38:05 -0800622 int ret;
Robert Love04896a72009-06-22 18:43:11 +0100623
David Brown886a4512011-08-02 09:02:49 -0700624 uart_resource = platform_get_resource(pdev, IORESOURCE_MEM, 0);
Stepan Moskovchenkoec8f29e2010-12-21 12:38:05 -0800625 if (unlikely(!uart_resource))
Robert Love04896a72009-06-22 18:43:11 +0100626 return -ENXIO;
Robert Love04896a72009-06-22 18:43:11 +0100627
Stepan Moskovchenkoec8f29e2010-12-21 12:38:05 -0800628 size = resource_size(uart_resource);
629
630 if (!request_mem_region(port->mapbase, size, "msm_serial"))
Robert Love04896a72009-06-22 18:43:11 +0100631 return -EBUSY;
632
633 port->membase = ioremap(port->mapbase, size);
634 if (!port->membase) {
Stepan Moskovchenkoec8f29e2010-12-21 12:38:05 -0800635 ret = -EBUSY;
636 goto fail_release_port;
637 }
638
Robert Love04896a72009-06-22 18:43:11 +0100639 return 0;
Stepan Moskovchenkoec8f29e2010-12-21 12:38:05 -0800640
Stepan Moskovchenkoec8f29e2010-12-21 12:38:05 -0800641fail_release_port:
642 release_mem_region(port->mapbase, size);
643 return ret;
Robert Love04896a72009-06-22 18:43:11 +0100644}
645
646static void msm_config_port(struct uart_port *port, int flags)
647{
Stepan Moskovchenkoec8f29e2010-12-21 12:38:05 -0800648 int ret;
Robert Love04896a72009-06-22 18:43:11 +0100649 if (flags & UART_CONFIG_TYPE) {
650 port->type = PORT_MSM;
Stepan Moskovchenkoec8f29e2010-12-21 12:38:05 -0800651 ret = msm_request_port(port);
652 if (ret)
653 return;
Robert Love04896a72009-06-22 18:43:11 +0100654 }
655}
656
657static int msm_verify_port(struct uart_port *port, struct serial_struct *ser)
658{
659 if (unlikely(ser->type != PORT_UNKNOWN && ser->type != PORT_MSM))
660 return -EINVAL;
661 if (unlikely(port->irq != ser->irq))
662 return -EINVAL;
663 return 0;
664}
665
666static void msm_power(struct uart_port *port, unsigned int state,
667 unsigned int oldstate)
668{
669 struct msm_port *msm_port = UART_TO_MSM(port);
670
671 switch (state) {
672 case 0:
Stephen Boydf98cf832013-06-17 10:43:08 -0700673 clk_prepare_enable(msm_port->clk);
Stephen Boydbfaddb72013-08-20 23:48:02 -0700674 clk_prepare_enable(msm_port->pclk);
Robert Love04896a72009-06-22 18:43:11 +0100675 break;
676 case 3:
Stephen Boydf98cf832013-06-17 10:43:08 -0700677 clk_disable_unprepare(msm_port->clk);
Stephen Boydbfaddb72013-08-20 23:48:02 -0700678 clk_disable_unprepare(msm_port->pclk);
Robert Love04896a72009-06-22 18:43:11 +0100679 break;
680 default:
681 printk(KERN_ERR "msm_serial: Unknown PM state %d\n", state);
682 }
683}
684
Stephen Boydf7e54d72014-01-14 12:34:55 -0800685#ifdef CONFIG_CONSOLE_POLL
686static int msm_poll_init(struct uart_port *port)
687{
688 struct msm_port *msm_port = UART_TO_MSM(port);
689
690 /* Enable single character mode on RX FIFO */
691 if (msm_port->is_uartdm >= UARTDM_1P4)
692 msm_write(port, UARTDM_DMEN_RX_SC_ENABLE, UARTDM_DMEN);
693
694 return 0;
695}
696
697static int msm_poll_get_char_single(struct uart_port *port)
698{
699 struct msm_port *msm_port = UART_TO_MSM(port);
700 unsigned int rf_reg = msm_port->is_uartdm ? UARTDM_RF : UART_RF;
701
702 if (!(msm_read(port, UART_SR) & UART_SR_RX_READY))
703 return NO_POLL_CHAR;
704 else
705 return msm_read(port, rf_reg) & 0xff;
706}
707
708static int msm_poll_get_char_dm_1p3(struct uart_port *port)
709{
710 int c;
711 static u32 slop;
712 static int count;
713 unsigned char *sp = (unsigned char *)&slop;
714
715 /* Check if a previous read had more than one char */
716 if (count) {
717 c = sp[sizeof(slop) - count];
718 count--;
719 /* Or if FIFO is empty */
720 } else if (!(msm_read(port, UART_SR) & UART_SR_RX_READY)) {
721 /*
722 * If RX packing buffer has less than a word, force stale to
723 * push contents into RX FIFO
724 */
725 count = msm_read(port, UARTDM_RXFS);
726 count = (count >> UARTDM_RXFS_BUF_SHIFT) & UARTDM_RXFS_BUF_MASK;
727 if (count) {
728 msm_write(port, UART_CR_CMD_FORCE_STALE, UART_CR);
729 slop = msm_read(port, UARTDM_RF);
730 c = sp[0];
731 count--;
732 } else {
733 c = NO_POLL_CHAR;
734 }
735 /* FIFO has a word */
736 } else {
737 slop = msm_read(port, UARTDM_RF);
738 c = sp[0];
739 count = sizeof(slop) - 1;
740 }
741
742 return c;
743}
744
745static int msm_poll_get_char(struct uart_port *port)
746{
747 u32 imr;
748 int c;
749 struct msm_port *msm_port = UART_TO_MSM(port);
750
751 /* Disable all interrupts */
752 imr = msm_read(port, UART_IMR);
753 msm_write(port, 0, UART_IMR);
754
755 if (msm_port->is_uartdm == UARTDM_1P3)
756 c = msm_poll_get_char_dm_1p3(port);
757 else
758 c = msm_poll_get_char_single(port);
759
760 /* Enable interrupts */
761 msm_write(port, imr, UART_IMR);
762
763 return c;
764}
765
766static void msm_poll_put_char(struct uart_port *port, unsigned char c)
767{
768 u32 imr;
769 struct msm_port *msm_port = UART_TO_MSM(port);
770
771 /* Disable all interrupts */
772 imr = msm_read(port, UART_IMR);
773 msm_write(port, 0, UART_IMR);
774
775 if (msm_port->is_uartdm)
776 reset_dm_count(port, 1);
777
778 /* Wait until FIFO is empty */
779 while (!(msm_read(port, UART_SR) & UART_SR_TX_READY))
780 cpu_relax();
781
782 /* Write a character */
783 msm_write(port, c, msm_port->is_uartdm ? UARTDM_TF : UART_TF);
784
785 /* Wait until FIFO is empty */
786 while (!(msm_read(port, UART_SR) & UART_SR_TX_READY))
787 cpu_relax();
788
789 /* Enable interrupts */
790 msm_write(port, imr, UART_IMR);
791
792 return;
793}
794#endif
795
Robert Love04896a72009-06-22 18:43:11 +0100796static struct uart_ops msm_uart_pops = {
797 .tx_empty = msm_tx_empty,
798 .set_mctrl = msm_set_mctrl,
799 .get_mctrl = msm_get_mctrl,
800 .stop_tx = msm_stop_tx,
801 .start_tx = msm_start_tx,
802 .stop_rx = msm_stop_rx,
803 .enable_ms = msm_enable_ms,
804 .break_ctl = msm_break_ctl,
805 .startup = msm_startup,
806 .shutdown = msm_shutdown,
807 .set_termios = msm_set_termios,
808 .type = msm_type,
809 .release_port = msm_release_port,
810 .request_port = msm_request_port,
811 .config_port = msm_config_port,
812 .verify_port = msm_verify_port,
813 .pm = msm_power,
Stephen Boydf7e54d72014-01-14 12:34:55 -0800814#ifdef CONFIG_CONSOLE_POLL
815 .poll_init = msm_poll_init,
816 .poll_get_char = msm_poll_get_char,
817 .poll_put_char = msm_poll_put_char,
818#endif
Robert Love04896a72009-06-22 18:43:11 +0100819};
820
821static struct msm_port msm_uart_ports[] = {
822 {
823 .uart = {
824 .iotype = UPIO_MEM,
825 .ops = &msm_uart_pops,
826 .flags = UPF_BOOT_AUTOCONF,
Stepan Moskovchenkoec8f29e2010-12-21 12:38:05 -0800827 .fifosize = 64,
Robert Love04896a72009-06-22 18:43:11 +0100828 .line = 0,
829 },
830 },
831 {
832 .uart = {
833 .iotype = UPIO_MEM,
834 .ops = &msm_uart_pops,
835 .flags = UPF_BOOT_AUTOCONF,
Stepan Moskovchenkoec8f29e2010-12-21 12:38:05 -0800836 .fifosize = 64,
Robert Love04896a72009-06-22 18:43:11 +0100837 .line = 1,
838 },
839 },
840 {
841 .uart = {
842 .iotype = UPIO_MEM,
843 .ops = &msm_uart_pops,
844 .flags = UPF_BOOT_AUTOCONF,
845 .fifosize = 64,
846 .line = 2,
847 },
848 },
849};
850
851#define UART_NR ARRAY_SIZE(msm_uart_ports)
852
853static inline struct uart_port *get_port_from_line(unsigned int line)
854{
855 return &msm_uart_ports[line].uart;
856}
857
858#ifdef CONFIG_SERIAL_MSM_CONSOLE
Robert Love04896a72009-06-22 18:43:11 +0100859static void msm_console_write(struct console *co, const char *s,
860 unsigned int count)
861{
Stephen Boyda3957e82013-08-20 23:48:06 -0700862 int i;
Robert Love04896a72009-06-22 18:43:11 +0100863 struct uart_port *port;
864 struct msm_port *msm_port;
Stephen Boyda3957e82013-08-20 23:48:06 -0700865 int num_newlines = 0;
866 bool replaced = false;
Stephen Boyd68252422014-06-30 14:54:01 -0700867 void __iomem *tf;
Robert Love04896a72009-06-22 18:43:11 +0100868
869 BUG_ON(co->index < 0 || co->index >= UART_NR);
870
871 port = get_port_from_line(co->index);
872 msm_port = UART_TO_MSM(port);
873
Stephen Boyd68252422014-06-30 14:54:01 -0700874 if (msm_port->is_uartdm)
875 tf = port->membase + UARTDM_TF;
876 else
877 tf = port->membase + UART_TF;
878
Stephen Boyda3957e82013-08-20 23:48:06 -0700879 /* Account for newlines that will get a carriage return added */
880 for (i = 0; i < count; i++)
881 if (s[i] == '\n')
882 num_newlines++;
883 count += num_newlines;
884
Robert Love04896a72009-06-22 18:43:11 +0100885 spin_lock(&port->lock);
Stephen Boyda3957e82013-08-20 23:48:06 -0700886 if (msm_port->is_uartdm)
887 reset_dm_count(port, count);
888
889 i = 0;
890 while (i < count) {
891 int j;
892 unsigned int num_chars;
893 char buf[4] = { 0 };
Stephen Boyda3957e82013-08-20 23:48:06 -0700894
895 if (msm_port->is_uartdm)
896 num_chars = min(count - i, (unsigned int)sizeof(buf));
897 else
898 num_chars = 1;
899
900 for (j = 0; j < num_chars; j++) {
901 char c = *s;
902
903 if (c == '\n' && !replaced) {
904 buf[j] = '\r';
905 j++;
906 replaced = true;
907 }
908 if (j < num_chars) {
909 buf[j] = c;
910 s++;
911 replaced = false;
912 }
913 }
914
915 while (!(msm_read(port, UART_SR) & UART_SR_TX_READY))
916 cpu_relax();
917
Stephen Boyd68252422014-06-30 14:54:01 -0700918 iowrite32_rep(tf, buf, 1);
Stephen Boyda3957e82013-08-20 23:48:06 -0700919 i += num_chars;
920 }
Robert Love04896a72009-06-22 18:43:11 +0100921 spin_unlock(&port->lock);
922}
923
924static int __init msm_console_setup(struct console *co, char *options)
925{
926 struct uart_port *port;
Stepan Moskovchenkoec8f29e2010-12-21 12:38:05 -0800927 struct msm_port *msm_port;
Rickard Strandqvist2b844ad2014-06-01 15:38:24 +0200928 int baud = 0, flow, bits, parity;
Robert Love04896a72009-06-22 18:43:11 +0100929
930 if (unlikely(co->index >= UART_NR || co->index < 0))
931 return -ENXIO;
932
933 port = get_port_from_line(co->index);
Stepan Moskovchenkoec8f29e2010-12-21 12:38:05 -0800934 msm_port = UART_TO_MSM(port);
Robert Love04896a72009-06-22 18:43:11 +0100935
936 if (unlikely(!port->membase))
937 return -ENXIO;
938
Robert Love04896a72009-06-22 18:43:11 +0100939 msm_init_clock(port);
940
941 if (options)
942 uart_parse_options(options, &baud, &parity, &bits, &flow);
943
944 bits = 8;
945 parity = 'n';
946 flow = 'n';
947 msm_write(port, UART_MR2_BITS_PER_CHAR_8 | UART_MR2_STOP_BIT_LEN_ONE,
948 UART_MR2); /* 8N1 */
949
950 if (baud < 300 || baud > 115200)
951 baud = 115200;
952 msm_set_baud_rate(port, baud);
953
954 msm_reset(port);
955
Stepan Moskovchenkoec8f29e2010-12-21 12:38:05 -0800956 if (msm_port->is_uartdm) {
957 msm_write(port, UART_CR_CMD_PROTECTION_EN, UART_CR);
958 msm_write(port, UART_CR_TX_ENABLE, UART_CR);
959 }
960
Robert Love04896a72009-06-22 18:43:11 +0100961 printk(KERN_INFO "msm_serial: console setup on port #%d\n", port->line);
962
963 return uart_set_options(port, co, baud, parity, bits, flow);
964}
965
966static struct uart_driver msm_uart_driver;
967
968static struct console msm_console = {
969 .name = "ttyMSM",
970 .write = msm_console_write,
971 .device = uart_console_device,
972 .setup = msm_console_setup,
973 .flags = CON_PRINTBUFFER,
974 .index = -1,
975 .data = &msm_uart_driver,
976};
977
978#define MSM_CONSOLE (&msm_console)
979
980#else
981#define MSM_CONSOLE NULL
982#endif
983
984static struct uart_driver msm_uart_driver = {
985 .owner = THIS_MODULE,
986 .driver_name = "msm_serial",
987 .dev_name = "ttyMSM",
988 .nr = UART_NR,
989 .cons = MSM_CONSOLE,
990};
991
David Browncfdad2a2011-08-04 01:55:24 -0700992static atomic_t msm_uart_next_id = ATOMIC_INIT(0);
993
Stephen Boydc3b5d3b2013-08-20 23:48:04 -0700994static const struct of_device_id msm_uartdm_table[] = {
Stephen Boydf7e54d72014-01-14 12:34:55 -0800995 { .compatible = "qcom,msm-uartdm-v1.1", .data = (void *)UARTDM_1P1 },
996 { .compatible = "qcom,msm-uartdm-v1.2", .data = (void *)UARTDM_1P2 },
997 { .compatible = "qcom,msm-uartdm-v1.3", .data = (void *)UARTDM_1P3 },
998 { .compatible = "qcom,msm-uartdm-v1.4", .data = (void *)UARTDM_1P4 },
Stephen Boydc3b5d3b2013-08-20 23:48:04 -0700999 { }
1000};
1001
Kumar Gala4cc29462014-06-03 15:13:22 -05001002static int msm_serial_probe(struct platform_device *pdev)
Robert Love04896a72009-06-22 18:43:11 +01001003{
1004 struct msm_port *msm_port;
1005 struct resource *resource;
1006 struct uart_port *port;
Stephen Boydf7e54d72014-01-14 12:34:55 -08001007 const struct of_device_id *id;
Roel Kluin1e091752009-12-21 16:26:49 -08001008 int irq;
Robert Love04896a72009-06-22 18:43:11 +01001009
David Browncfdad2a2011-08-04 01:55:24 -07001010 if (pdev->id == -1)
1011 pdev->id = atomic_inc_return(&msm_uart_next_id) - 1;
1012
Robert Love04896a72009-06-22 18:43:11 +01001013 if (unlikely(pdev->id < 0 || pdev->id >= UART_NR))
1014 return -ENXIO;
1015
1016 printk(KERN_INFO "msm_serial: detected port #%d\n", pdev->id);
1017
1018 port = get_port_from_line(pdev->id);
1019 port->dev = &pdev->dev;
1020 msm_port = UART_TO_MSM(port);
1021
Stephen Boydf7e54d72014-01-14 12:34:55 -08001022 id = of_match_device(msm_uartdm_table, &pdev->dev);
1023 if (id)
1024 msm_port->is_uartdm = (unsigned long)id->data;
Stepan Moskovchenkoec8f29e2010-12-21 12:38:05 -08001025 else
1026 msm_port->is_uartdm = 0;
1027
Stephen Boydbfaddb72013-08-20 23:48:02 -07001028 msm_port->clk = devm_clk_get(&pdev->dev, "core");
Stephen Boyd519b3712013-06-17 10:43:09 -07001029 if (IS_ERR(msm_port->clk))
1030 return PTR_ERR(msm_port->clk);
Stepan Moskovchenkoec8f29e2010-12-21 12:38:05 -08001031
Stephen Boyd519b3712013-06-17 10:43:09 -07001032 if (msm_port->is_uartdm) {
Stephen Boydbfaddb72013-08-20 23:48:02 -07001033 msm_port->pclk = devm_clk_get(&pdev->dev, "iface");
Stephen Boyd519b3712013-06-17 10:43:09 -07001034 if (IS_ERR(msm_port->pclk))
1035 return PTR_ERR(msm_port->pclk);
1036
David Brown7b6031a2012-09-07 14:45:03 -07001037 clk_set_rate(msm_port->clk, 1843200);
Stephen Boyd519b3712013-06-17 10:43:09 -07001038 }
Stepan Moskovchenkoec8f29e2010-12-21 12:38:05 -08001039
Robert Love04896a72009-06-22 18:43:11 +01001040 port->uartclk = clk_get_rate(msm_port->clk);
Abhijeet Dharmapurikar18c79d72010-05-20 15:20:23 -07001041 printk(KERN_INFO "uartclk = %d\n", port->uartclk);
1042
Robert Love04896a72009-06-22 18:43:11 +01001043
David Brown886a4512011-08-02 09:02:49 -07001044 resource = platform_get_resource(pdev, IORESOURCE_MEM, 0);
Robert Love04896a72009-06-22 18:43:11 +01001045 if (unlikely(!resource))
1046 return -ENXIO;
1047 port->mapbase = resource->start;
1048
Roel Kluin1e091752009-12-21 16:26:49 -08001049 irq = platform_get_irq(pdev, 0);
1050 if (unlikely(irq < 0))
Robert Love04896a72009-06-22 18:43:11 +01001051 return -ENXIO;
Roel Kluin1e091752009-12-21 16:26:49 -08001052 port->irq = irq;
Robert Love04896a72009-06-22 18:43:11 +01001053
1054 platform_set_drvdata(pdev, port);
1055
1056 return uart_add_one_port(&msm_uart_driver, port);
1057}
1058
Bill Pembertonae8d8a12012-11-19 13:26:18 -05001059static int msm_serial_remove(struct platform_device *pdev)
Robert Love04896a72009-06-22 18:43:11 +01001060{
Stephen Boyd519b3712013-06-17 10:43:09 -07001061 struct uart_port *port = platform_get_drvdata(pdev);
Robert Love04896a72009-06-22 18:43:11 +01001062
Stephen Boyd519b3712013-06-17 10:43:09 -07001063 uart_remove_one_port(&msm_uart_driver, port);
Robert Love04896a72009-06-22 18:43:11 +01001064
1065 return 0;
1066}
1067
David Browncfdad2a2011-08-04 01:55:24 -07001068static struct of_device_id msm_match_table[] = {
1069 { .compatible = "qcom,msm-uart" },
Stephen Boydc3b5d3b2013-08-20 23:48:04 -07001070 { .compatible = "qcom,msm-uartdm" },
David Browncfdad2a2011-08-04 01:55:24 -07001071 {}
1072};
1073
Robert Love04896a72009-06-22 18:43:11 +01001074static struct platform_driver msm_platform_driver = {
Robert Love04896a72009-06-22 18:43:11 +01001075 .remove = msm_serial_remove,
Andy Gross31964ff2014-04-24 11:31:22 -05001076 .probe = msm_serial_probe,
Robert Love04896a72009-06-22 18:43:11 +01001077 .driver = {
1078 .name = "msm_serial",
1079 .owner = THIS_MODULE,
David Browncfdad2a2011-08-04 01:55:24 -07001080 .of_match_table = msm_match_table,
Robert Love04896a72009-06-22 18:43:11 +01001081 },
1082};
1083
1084static int __init msm_serial_init(void)
1085{
1086 int ret;
1087
1088 ret = uart_register_driver(&msm_uart_driver);
1089 if (unlikely(ret))
1090 return ret;
1091
Andy Gross31964ff2014-04-24 11:31:22 -05001092 ret = platform_driver_register(&msm_platform_driver);
Robert Love04896a72009-06-22 18:43:11 +01001093 if (unlikely(ret))
1094 uart_unregister_driver(&msm_uart_driver);
1095
1096 printk(KERN_INFO "msm_serial: driver initialized\n");
1097
1098 return ret;
1099}
1100
1101static void __exit msm_serial_exit(void)
1102{
1103#ifdef CONFIG_SERIAL_MSM_CONSOLE
1104 unregister_console(&msm_console);
1105#endif
1106 platform_driver_unregister(&msm_platform_driver);
1107 uart_unregister_driver(&msm_uart_driver);
1108}
1109
1110module_init(msm_serial_init);
1111module_exit(msm_serial_exit);
1112
1113MODULE_AUTHOR("Robert Love <rlove@google.com>");
1114MODULE_DESCRIPTION("Driver for msm7x serial device");
1115MODULE_LICENSE("GPL");