xref: /openbmc/linux/drivers/tty/serial/ma35d1_serial.c (revision 6e4f6b5eac461867471b3f368699097b31843d23)
1930cbf92SJacky Huang // SPDX-License-Identifier: GPL-2.0+
2930cbf92SJacky Huang /*
3930cbf92SJacky Huang  *  MA35D1 serial driver
4930cbf92SJacky Huang  *  Copyright (C) 2023 Nuvoton Technology Corp.
5930cbf92SJacky Huang  */
6930cbf92SJacky Huang 
7930cbf92SJacky Huang #include <linux/bitfield.h>
8930cbf92SJacky Huang #include <linux/clk.h>
9930cbf92SJacky Huang #include <linux/delay.h>
10930cbf92SJacky Huang #include <linux/of.h>
1129e5c442SRob Herring #include <linux/platform_device.h>
12930cbf92SJacky Huang #include <linux/iopoll.h>
13930cbf92SJacky Huang #include <linux/serial_core.h>
14930cbf92SJacky Huang #include <linux/slab.h>
15930cbf92SJacky Huang #include <linux/tty_flip.h>
16930cbf92SJacky Huang #include <linux/units.h>
17930cbf92SJacky Huang 
18930cbf92SJacky Huang #define MA35_UART_NR		17
19930cbf92SJacky Huang 
20930cbf92SJacky Huang #define MA35_RBR_REG		0x00
21930cbf92SJacky Huang #define MA35_THR_REG		0x00
22930cbf92SJacky Huang #define MA35_IER_REG		0x04
23930cbf92SJacky Huang #define MA35_FCR_REG		0x08
24930cbf92SJacky Huang #define MA35_LCR_REG		0x0C
25930cbf92SJacky Huang #define MA35_MCR_REG		0x10
26930cbf92SJacky Huang #define MA35_MSR_REG		0x14
27930cbf92SJacky Huang #define MA35_FSR_REG		0x18
28930cbf92SJacky Huang #define MA35_ISR_REG		0x1C
29930cbf92SJacky Huang #define MA35_TOR_REG		0x20
30930cbf92SJacky Huang #define MA35_BAUD_REG		0x24
31930cbf92SJacky Huang #define MA35_ALTCTL_REG		0x2C
32930cbf92SJacky Huang #define MA35_FUN_SEL_REG	0x30
33930cbf92SJacky Huang #define MA35_WKCTL_REG		0x40
34930cbf92SJacky Huang #define MA35_WKSTS_REG		0x44
35930cbf92SJacky Huang 
36930cbf92SJacky Huang /* MA35_IER_REG - Interrupt Enable Register */
37930cbf92SJacky Huang #define MA35_IER_RDA_IEN	BIT(0)  /* RBR Available Interrupt Enable */
38930cbf92SJacky Huang #define MA35_IER_THRE_IEN	BIT(1)  /* THR Empty Interrupt Enable */
39930cbf92SJacky Huang #define MA35_IER_RLS_IEN	BIT(2)  /* RX Line Status Interrupt Enable */
40930cbf92SJacky Huang #define MA35_IER_RTO_IEN	BIT(4)  /* RX Time-out Interrupt Enable */
41930cbf92SJacky Huang #define MA35_IER_BUFERR_IEN	BIT(5)  /* Buffer Error Interrupt Enable */
42930cbf92SJacky Huang #define MA35_IER_TIME_OUT_EN	BIT(11) /* RX Buffer Time-out Counter Enable */
43930cbf92SJacky Huang #define MA35_IER_AUTO_RTS	BIT(12) /* nRTS Auto-flow Control Enable */
44930cbf92SJacky Huang #define MA35_IER_AUTO_CTS	BIT(13) /* nCTS Auto-flow Control Enable */
45930cbf92SJacky Huang 
46930cbf92SJacky Huang /* MA35_FCR_REG - FIFO Control Register */
47930cbf92SJacky Huang #define MA35_FCR_RFR		BIT(1)  /* RX Field Software Reset */
48930cbf92SJacky Huang #define MA35_FCR_TFR		BIT(2)  /* TX Field Software Reset */
49930cbf92SJacky Huang #define MA35_FCR_RFITL_MASK	GENMASK(7, 4) /* RX FIFO Interrupt Trigger Level */
50930cbf92SJacky Huang #define MA35_FCR_RFITL_1BYTE	FIELD_PREP(MA35_FCR_RFITL_MASK, 0)
51930cbf92SJacky Huang #define MA35_FCR_RFITL_4BYTES	FIELD_PREP(MA35_FCR_RFITL_MASK, 1)
52930cbf92SJacky Huang #define MA35_FCR_RFITL_8BYTES	FIELD_PREP(MA35_FCR_RFITL_MASK, 2)
53930cbf92SJacky Huang #define MA35_FCR_RFITL_14BYTES	FIELD_PREP(MA35_FCR_RFITL_MASK, 3)
54930cbf92SJacky Huang #define MA35_FCR_RFITL_30BYTES	FIELD_PREP(MA35_FCR_RFITL_MASK, 4)
55930cbf92SJacky Huang #define MA35_FCR_RTSTL_MASK	GENMASK(19, 16) /* nRTS Trigger Level */
56930cbf92SJacky Huang #define MA35_FCR_RTSTL_1BYTE	FIELD_PREP(MA35_FCR_RTSTL_MASK, 0)
57930cbf92SJacky Huang #define MA35_FCR_RTSTL_4BYTES	FIELD_PREP(MA35_FCR_RTSTL_MASK, 1)
58930cbf92SJacky Huang #define MA35_FCR_RTSTL_8BYTES	FIELD_PREP(MA35_FCR_RTSTL_MASK, 2)
59930cbf92SJacky Huang #define MA35_FCR_RTSTL_14BYTES	FIELD_PREP(MA35_FCR_RTSTL_MASK, 3)
60930cbf92SJacky Huang #define MA35_FCR_RTSTLL_30BYTES	FIELD_PREP(MA35_FCR_RTSTL_MASK, 4)
61930cbf92SJacky Huang 
62930cbf92SJacky Huang /* MA35_LCR_REG - Line Control Register */
63930cbf92SJacky Huang #define	MA35_LCR_NSB		BIT(2)  /* Number of “STOP Bit” */
64930cbf92SJacky Huang #define MA35_LCR_PBE		BIT(3)  /* Parity Bit Enable */
65930cbf92SJacky Huang #define MA35_LCR_EPE		BIT(4)  /* Even Parity Enable */
66930cbf92SJacky Huang #define MA35_LCR_SPE		BIT(5)  /* Stick Parity Enable */
67930cbf92SJacky Huang #define MA35_LCR_BREAK		BIT(6)  /* Break Control */
68930cbf92SJacky Huang #define MA35_LCR_WLS_MASK	GENMASK(1, 0) /* Word Length Selection */
69930cbf92SJacky Huang #define MA35_LCR_WLS_5BITS	FIELD_PREP(MA35_LCR_WLS_MASK, 0)
70930cbf92SJacky Huang #define MA35_LCR_WLS_6BITS	FIELD_PREP(MA35_LCR_WLS_MASK, 1)
71930cbf92SJacky Huang #define MA35_LCR_WLS_7BITS	FIELD_PREP(MA35_LCR_WLS_MASK, 2)
72930cbf92SJacky Huang #define MA35_LCR_WLS_8BITS	FIELD_PREP(MA35_LCR_WLS_MASK, 3)
73930cbf92SJacky Huang 
74930cbf92SJacky Huang /* MA35_MCR_REG - Modem Control Register */
75930cbf92SJacky Huang #define MA35_MCR_RTS_CTRL	BIT(1)  /* nRTS Signal Control */
76930cbf92SJacky Huang #define MA35_MCR_RTSACTLV	BIT(9)  /* nRTS Pin Active Level */
77930cbf92SJacky Huang #define MA35_MCR_RTSSTS		BIT(13) /* nRTS Pin Status (Read Only) */
78930cbf92SJacky Huang 
79930cbf92SJacky Huang /* MA35_MSR_REG - Modem Status Register */
80930cbf92SJacky Huang #define MA35_MSR_CTSDETF	BIT(0)  /* Detect nCTS State Change Flag */
81930cbf92SJacky Huang #define MA35_MSR_CTSSTS		BIT(4)  /* nCTS Pin Status (Read Only) */
82930cbf92SJacky Huang #define MA35_MSR_CTSACTLV	BIT(8)  /* nCTS Pin Active Level */
83930cbf92SJacky Huang 
84930cbf92SJacky Huang /* MA35_FSR_REG - FIFO Status Register */
85930cbf92SJacky Huang #define MA35_FSR_RX_OVER_IF	BIT(0)  /* RX Overflow Error Interrupt Flag */
86930cbf92SJacky Huang #define MA35_FSR_PEF		BIT(4)  /* Parity Error Flag*/
87930cbf92SJacky Huang #define MA35_FSR_FEF		BIT(5)  /* Framing Error Flag */
88930cbf92SJacky Huang #define MA35_FSR_BIF		BIT(6)  /* Break Interrupt Flag */
89930cbf92SJacky Huang #define MA35_FSR_RX_EMPTY	BIT(14) /* Receiver FIFO Empty (Read Only) */
90930cbf92SJacky Huang #define MA35_FSR_RX_FULL	BIT(15) /* Receiver FIFO Full (Read Only) */
91930cbf92SJacky Huang #define MA35_FSR_TX_EMPTY	BIT(22) /* Transmitter FIFO Empty (Read Only) */
92930cbf92SJacky Huang #define MA35_FSR_TX_FULL	BIT(23) /* Transmitter FIFO Full (Read Only) */
93930cbf92SJacky Huang #define MA35_FSR_TX_OVER_IF	BIT(24) /* TX Overflow Error Interrupt Flag */
94930cbf92SJacky Huang #define MA35_FSR_TE_FLAG	BIT(28) /* Transmitter Empty Flag (Read Only) */
95930cbf92SJacky Huang #define MA35_FSR_RXPTR_MSK	GENMASK(13, 8) /* TX FIFO Pointer mask */
96930cbf92SJacky Huang #define MA35_FSR_TXPTR_MSK	GENMASK(21, 16) /* RX FIFO Pointer mask */
97930cbf92SJacky Huang 
98930cbf92SJacky Huang /* MA35_ISR_REG - Interrupt Status Register */
99930cbf92SJacky Huang #define MA35_ISR_RDA_IF		BIT(0)  /* RBR Available Interrupt Flag */
100930cbf92SJacky Huang #define MA35_ISR_THRE_IF	BIT(1)  /* THR Empty Interrupt Flag */
101930cbf92SJacky Huang #define MA35_ISR_RLSIF		BIT(2)  /* Receive Line Interrupt Flag */
102930cbf92SJacky Huang #define MA35_ISR_MODEMIF	BIT(3)  /* MODEM Interrupt Flag */
103930cbf92SJacky Huang #define MA35_ISR_RXTO_IF	BIT(4)  /* RX Time-out Interrupt Flag */
104930cbf92SJacky Huang #define MA35_ISR_BUFEIF		BIT(5)  /* Buffer Error Interrupt Flag */
105930cbf92SJacky Huang #define MA35_ISR_WK_IF		BIT(6)  /* UART Wake-up Interrupt Flag */
106930cbf92SJacky Huang #define MA35_ISR_RDAINT		BIT(8)  /* RBR Available Interrupt Indicator */
107930cbf92SJacky Huang #define MA35_ISR_THRE_INT	BIT(9)  /* THR Empty Interrupt Indicator */
108930cbf92SJacky Huang #define MA35_ISR_ALL		0xFFFFFFFF
109930cbf92SJacky Huang 
110930cbf92SJacky Huang /* MA35_BAUD_REG - Baud Rate Divider Register */
111930cbf92SJacky Huang #define	MA35_BAUD_MODE_MASK	GENMASK(29, 28)
112930cbf92SJacky Huang #define MA35_BAUD_MODE0		FIELD_PREP(MA35_BAUD_MODE_MASK, 0)
113930cbf92SJacky Huang #define MA35_BAUD_MODE1		FIELD_PREP(MA35_BAUD_MODE_MASK, 2)
114930cbf92SJacky Huang #define MA35_BAUD_MODE2		FIELD_PREP(MA35_BAUD_MODE_MASK, 3)
115930cbf92SJacky Huang #define	MA35_BAUD_MASK		GENMASK(15, 0)
116930cbf92SJacky Huang 
117930cbf92SJacky Huang /* MA35_ALTCTL_REG - Alternate Control/Status Register */
118930cbf92SJacky Huang #define MA35_ALTCTL_RS485AUD	BIT(10) /* RS-485 Auto Direction Function */
119930cbf92SJacky Huang 
120930cbf92SJacky Huang /* MA35_FUN_SEL_REG - Function Select Register */
121930cbf92SJacky Huang #define MA35_FUN_SEL_MASK	GENMASK(2, 0)
122930cbf92SJacky Huang #define MA35_FUN_SEL_UART	FIELD_PREP(MA35_FUN_SEL_MASK, 0)
123930cbf92SJacky Huang #define MA35_FUN_SEL_RS485	FIELD_PREP(MA35_FUN_SEL_MASK, 3)
124930cbf92SJacky Huang 
125930cbf92SJacky Huang /* The constrain for MA35D1 UART baud rate divider */
126930cbf92SJacky Huang #define MA35_BAUD_DIV_MAX	0xFFFF
127930cbf92SJacky Huang #define MA35_BAUD_DIV_MIN	11
128930cbf92SJacky Huang 
129930cbf92SJacky Huang /* UART FIFO depth */
130930cbf92SJacky Huang #define MA35_UART_FIFO_DEPTH	32
131930cbf92SJacky Huang /* UART console clock */
132930cbf92SJacky Huang #define MA35_UART_CONSOLE_CLK	(24 * HZ_PER_MHZ)
133930cbf92SJacky Huang /* UART register ioremap size */
134930cbf92SJacky Huang #define MA35_UART_REG_SIZE	0x100
135930cbf92SJacky Huang /* Rx Timeout */
136930cbf92SJacky Huang #define MA35_UART_RX_TOUT	0x40
137930cbf92SJacky Huang 
138930cbf92SJacky Huang #define MA35_IER_CONFIG		(MA35_IER_RTO_IEN | MA35_IER_RDA_IEN | \
139930cbf92SJacky Huang 				 MA35_IER_TIME_OUT_EN | MA35_IER_BUFERR_IEN)
140930cbf92SJacky Huang 
141930cbf92SJacky Huang #define MA35_ISR_IF_CHECK	(MA35_ISR_RDA_IF | MA35_ISR_RXTO_IF | \
142930cbf92SJacky Huang 				 MA35_ISR_THRE_INT | MA35_ISR_BUFEIF)
143930cbf92SJacky Huang 
144930cbf92SJacky Huang #define MA35_FSR_TX_BOTH_EMPTY	(MA35_FSR_TE_FLAG | MA35_FSR_TX_EMPTY)
145930cbf92SJacky Huang 
146930cbf92SJacky Huang static struct uart_driver ma35d1serial_reg;
147930cbf92SJacky Huang 
148930cbf92SJacky Huang struct uart_ma35d1_port {
149930cbf92SJacky Huang 	struct uart_port port;
150930cbf92SJacky Huang 	struct clk *clk;
151930cbf92SJacky Huang 	u16 capabilities; /* port capabilities */
152930cbf92SJacky Huang 	u8 ier;
153930cbf92SJacky Huang 	u8 lcr;
154930cbf92SJacky Huang 	u8 mcr;
155930cbf92SJacky Huang 	u32 baud_rate;
156930cbf92SJacky Huang 	u32 console_baud_rate;
157930cbf92SJacky Huang 	u32 console_line;
158930cbf92SJacky Huang 	u32 console_int;
159930cbf92SJacky Huang };
160930cbf92SJacky Huang 
161930cbf92SJacky Huang static struct uart_ma35d1_port ma35d1serial_ports[MA35_UART_NR];
162930cbf92SJacky Huang 
to_ma35d1_uart_port(struct uart_port * uart)163930cbf92SJacky Huang static struct uart_ma35d1_port *to_ma35d1_uart_port(struct uart_port *uart)
164930cbf92SJacky Huang {
165930cbf92SJacky Huang 	return container_of(uart, struct uart_ma35d1_port, port);
166930cbf92SJacky Huang }
167930cbf92SJacky Huang 
serial_in(struct uart_ma35d1_port * p,u32 offset)168930cbf92SJacky Huang static u32 serial_in(struct uart_ma35d1_port *p, u32 offset)
169930cbf92SJacky Huang {
170930cbf92SJacky Huang 	return readl_relaxed(p->port.membase + offset);
171930cbf92SJacky Huang }
172930cbf92SJacky Huang 
serial_out(struct uart_ma35d1_port * p,u32 offset,u32 value)173930cbf92SJacky Huang static void serial_out(struct uart_ma35d1_port *p, u32 offset, u32 value)
174930cbf92SJacky Huang {
175930cbf92SJacky Huang 	writel_relaxed(value, p->port.membase + offset);
176930cbf92SJacky Huang }
177930cbf92SJacky Huang 
__stop_tx(struct uart_ma35d1_port * p)178930cbf92SJacky Huang static void __stop_tx(struct uart_ma35d1_port *p)
179930cbf92SJacky Huang {
180930cbf92SJacky Huang 	u32 ier;
181930cbf92SJacky Huang 
182930cbf92SJacky Huang 	ier = serial_in(p, MA35_IER_REG);
183930cbf92SJacky Huang 	if (ier & MA35_IER_THRE_IEN)
184930cbf92SJacky Huang 		serial_out(p, MA35_IER_REG, ier & ~MA35_IER_THRE_IEN);
185930cbf92SJacky Huang }
186930cbf92SJacky Huang 
ma35d1serial_stop_tx(struct uart_port * port)187930cbf92SJacky Huang static void ma35d1serial_stop_tx(struct uart_port *port)
188930cbf92SJacky Huang {
189930cbf92SJacky Huang 	struct uart_ma35d1_port *up = to_ma35d1_uart_port(port);
190930cbf92SJacky Huang 
191930cbf92SJacky Huang 	__stop_tx(up);
192930cbf92SJacky Huang }
193930cbf92SJacky Huang 
transmit_chars(struct uart_ma35d1_port * up)194930cbf92SJacky Huang static void transmit_chars(struct uart_ma35d1_port *up)
195930cbf92SJacky Huang {
196930cbf92SJacky Huang 	u32 count;
197930cbf92SJacky Huang 	u8 ch;
198930cbf92SJacky Huang 
199930cbf92SJacky Huang 	if (uart_tx_stopped(&up->port)) {
200930cbf92SJacky Huang 		ma35d1serial_stop_tx(&up->port);
201930cbf92SJacky Huang 		return;
202930cbf92SJacky Huang 	}
203930cbf92SJacky Huang 	count = MA35_UART_FIFO_DEPTH - FIELD_GET(MA35_FSR_TXPTR_MSK,
204930cbf92SJacky Huang 						 serial_in(up, MA35_FSR_REG));
205930cbf92SJacky Huang 	uart_port_tx_limited(&up->port, ch, count,
206930cbf92SJacky Huang 			     !(serial_in(up, MA35_FSR_REG) & MA35_FSR_TX_FULL),
207930cbf92SJacky Huang 			     serial_out(up, MA35_THR_REG, ch),
208930cbf92SJacky Huang 			     ({}));
209930cbf92SJacky Huang }
210930cbf92SJacky Huang 
ma35d1serial_start_tx(struct uart_port * port)211930cbf92SJacky Huang static void ma35d1serial_start_tx(struct uart_port *port)
212930cbf92SJacky Huang {
213930cbf92SJacky Huang 	struct uart_ma35d1_port *up = to_ma35d1_uart_port(port);
214930cbf92SJacky Huang 	u32 ier;
215930cbf92SJacky Huang 
216930cbf92SJacky Huang 	ier = serial_in(up, MA35_IER_REG);
217930cbf92SJacky Huang 	serial_out(up, MA35_IER_REG, ier & ~MA35_IER_THRE_IEN);
218930cbf92SJacky Huang 	transmit_chars(up);
219930cbf92SJacky Huang 	serial_out(up, MA35_IER_REG, ier | MA35_IER_THRE_IEN);
220930cbf92SJacky Huang }
221930cbf92SJacky Huang 
ma35d1serial_stop_rx(struct uart_port * port)222930cbf92SJacky Huang static void ma35d1serial_stop_rx(struct uart_port *port)
223930cbf92SJacky Huang {
224930cbf92SJacky Huang 	struct uart_ma35d1_port *up = to_ma35d1_uart_port(port);
225930cbf92SJacky Huang 	u32 ier;
226930cbf92SJacky Huang 
227930cbf92SJacky Huang 	ier = serial_in(up, MA35_IER_REG);
228930cbf92SJacky Huang 	ier &= ~MA35_IER_RDA_IEN;
229930cbf92SJacky Huang 	serial_out(up, MA35_IER_REG, ier);
230930cbf92SJacky Huang }
231930cbf92SJacky Huang 
receive_chars(struct uart_ma35d1_port * up)232930cbf92SJacky Huang static void receive_chars(struct uart_ma35d1_port *up)
233930cbf92SJacky Huang {
234930cbf92SJacky Huang 	int max_count = 256;
235930cbf92SJacky Huang 	u8 ch, flag;
236930cbf92SJacky Huang 	u32 fsr;
237930cbf92SJacky Huang 
238930cbf92SJacky Huang 	fsr = serial_in(up, MA35_FSR_REG);
239930cbf92SJacky Huang 	do {
240930cbf92SJacky Huang 		flag = TTY_NORMAL;
241930cbf92SJacky Huang 		up->port.icount.rx++;
242930cbf92SJacky Huang 
243930cbf92SJacky Huang 		if (unlikely(fsr & (MA35_FSR_BIF | MA35_FSR_FEF |
244930cbf92SJacky Huang 				    MA35_FSR_PEF | MA35_FSR_RX_OVER_IF))) {
245930cbf92SJacky Huang 			if (fsr & MA35_FSR_BIF) {
246930cbf92SJacky Huang 				up->port.icount.brk++;
247930cbf92SJacky Huang 				if (uart_handle_break(&up->port))
248930cbf92SJacky Huang 					continue;
249930cbf92SJacky Huang 			}
250930cbf92SJacky Huang 			if (fsr & MA35_FSR_FEF)
251930cbf92SJacky Huang 				up->port.icount.frame++;
252930cbf92SJacky Huang 			if (fsr & MA35_FSR_PEF)
253930cbf92SJacky Huang 				up->port.icount.parity++;
254930cbf92SJacky Huang 			if (fsr & MA35_FSR_RX_OVER_IF)
255930cbf92SJacky Huang 				up->port.icount.overrun++;
256930cbf92SJacky Huang 
257930cbf92SJacky Huang 			serial_out(up, MA35_FSR_REG,
258930cbf92SJacky Huang 				   fsr & (MA35_FSR_BIF | MA35_FSR_FEF |
259930cbf92SJacky Huang 					  MA35_FSR_PEF | MA35_FSR_RX_OVER_IF));
260930cbf92SJacky Huang 			if (fsr & MA35_FSR_BIF)
261930cbf92SJacky Huang 				flag = TTY_BREAK;
262930cbf92SJacky Huang 			else if (fsr & MA35_FSR_PEF)
263930cbf92SJacky Huang 				flag = TTY_PARITY;
264930cbf92SJacky Huang 			else if (fsr & MA35_FSR_FEF)
265930cbf92SJacky Huang 				flag = TTY_FRAME;
266930cbf92SJacky Huang 		}
267930cbf92SJacky Huang 
268930cbf92SJacky Huang 		ch = serial_in(up, MA35_RBR_REG);
269930cbf92SJacky Huang 		if (uart_handle_sysrq_char(&up->port, ch))
270930cbf92SJacky Huang 			continue;
271930cbf92SJacky Huang 
272930cbf92SJacky Huang 		spin_lock(&up->port.lock);
273930cbf92SJacky Huang 		uart_insert_char(&up->port, fsr, MA35_FSR_RX_OVER_IF, ch, flag);
274930cbf92SJacky Huang 		spin_unlock(&up->port.lock);
275930cbf92SJacky Huang 
276930cbf92SJacky Huang 		fsr = serial_in(up, MA35_FSR_REG);
277930cbf92SJacky Huang 	} while (!(fsr & MA35_FSR_RX_EMPTY) && (max_count-- > 0));
278930cbf92SJacky Huang 
279930cbf92SJacky Huang 	spin_lock(&up->port.lock);
280930cbf92SJacky Huang 	tty_flip_buffer_push(&up->port.state->port);
281930cbf92SJacky Huang 	spin_unlock(&up->port.lock);
282930cbf92SJacky Huang }
283930cbf92SJacky Huang 
ma35d1serial_interrupt(int irq,void * dev_id)284930cbf92SJacky Huang static irqreturn_t ma35d1serial_interrupt(int irq, void *dev_id)
285930cbf92SJacky Huang {
286930cbf92SJacky Huang 	struct uart_port *port = dev_id;
287930cbf92SJacky Huang 	struct uart_ma35d1_port *up = to_ma35d1_uart_port(port);
288930cbf92SJacky Huang 	u32 isr, fsr;
289930cbf92SJacky Huang 
290930cbf92SJacky Huang 	isr = serial_in(up, MA35_ISR_REG);
291930cbf92SJacky Huang 	fsr = serial_in(up, MA35_FSR_REG);
292930cbf92SJacky Huang 
293930cbf92SJacky Huang 	if (!(isr & MA35_ISR_IF_CHECK))
294930cbf92SJacky Huang 		return IRQ_NONE;
295930cbf92SJacky Huang 
296930cbf92SJacky Huang 	if (isr & (MA35_ISR_RDA_IF | MA35_ISR_RXTO_IF))
297930cbf92SJacky Huang 		receive_chars(up);
298930cbf92SJacky Huang 	if (isr & MA35_ISR_THRE_INT)
299930cbf92SJacky Huang 		transmit_chars(up);
300930cbf92SJacky Huang 	if (fsr & MA35_FSR_TX_OVER_IF)
301930cbf92SJacky Huang 		serial_out(up, MA35_FSR_REG, MA35_FSR_TX_OVER_IF);
302930cbf92SJacky Huang 
303930cbf92SJacky Huang 	return IRQ_HANDLED;
304930cbf92SJacky Huang }
305930cbf92SJacky Huang 
ma35d1serial_tx_empty(struct uart_port * port)306930cbf92SJacky Huang static u32 ma35d1serial_tx_empty(struct uart_port *port)
307930cbf92SJacky Huang {
308930cbf92SJacky Huang 	struct uart_ma35d1_port *up = to_ma35d1_uart_port(port);
309930cbf92SJacky Huang 	u32 fsr;
310930cbf92SJacky Huang 
311930cbf92SJacky Huang 	fsr = serial_in(up, MA35_FSR_REG);
312930cbf92SJacky Huang 	if ((fsr & MA35_FSR_TX_BOTH_EMPTY) == MA35_FSR_TX_BOTH_EMPTY)
313930cbf92SJacky Huang 		return TIOCSER_TEMT;
314930cbf92SJacky Huang 	else
315930cbf92SJacky Huang 		return 0;
316930cbf92SJacky Huang }
317930cbf92SJacky Huang 
ma35d1serial_get_mctrl(struct uart_port * port)318930cbf92SJacky Huang static u32 ma35d1serial_get_mctrl(struct uart_port *port)
319930cbf92SJacky Huang {
320930cbf92SJacky Huang 	struct uart_ma35d1_port *up = to_ma35d1_uart_port(port);
321930cbf92SJacky Huang 	u32 status;
322930cbf92SJacky Huang 	u32 ret = 0;
323930cbf92SJacky Huang 
324930cbf92SJacky Huang 	status = serial_in(up, MA35_MSR_REG);
325930cbf92SJacky Huang 	if (!(status & MA35_MSR_CTSSTS))
326930cbf92SJacky Huang 		ret |= TIOCM_CTS;
327930cbf92SJacky Huang 	return ret;
328930cbf92SJacky Huang }
329930cbf92SJacky Huang 
ma35d1serial_set_mctrl(struct uart_port * port,u32 mctrl)330930cbf92SJacky Huang static void ma35d1serial_set_mctrl(struct uart_port *port, u32 mctrl)
331930cbf92SJacky Huang {
332930cbf92SJacky Huang 	struct uart_ma35d1_port *up = to_ma35d1_uart_port(port);
333930cbf92SJacky Huang 	u32 mcr, msr, ier;
334930cbf92SJacky Huang 
335930cbf92SJacky Huang 	mcr = serial_in(up, MA35_MCR_REG);
336930cbf92SJacky Huang 	mcr &= ~MA35_MCR_RTS_CTRL;
337930cbf92SJacky Huang 
338930cbf92SJacky Huang 	if (mctrl & TIOCM_RTS)
339930cbf92SJacky Huang 		mcr |= MA35_MCR_RTSACTLV;
340930cbf92SJacky Huang 	else
341930cbf92SJacky Huang 		mcr &= ~MA35_MCR_RTSACTLV;
342930cbf92SJacky Huang 
343930cbf92SJacky Huang 	if (up->mcr & UART_MCR_AFE) {
344930cbf92SJacky Huang 		ier = serial_in(up, MA35_IER_REG);
345930cbf92SJacky Huang 		ier |= MA35_IER_AUTO_RTS | MA35_IER_AUTO_CTS;
346930cbf92SJacky Huang 		serial_out(up, MA35_IER_REG, ier);
347930cbf92SJacky Huang 		up->port.flags |= UPF_HARD_FLOW;
348930cbf92SJacky Huang 	} else {
349930cbf92SJacky Huang 		ier = serial_in(up, MA35_IER_REG);
350930cbf92SJacky Huang 		ier &= ~(MA35_IER_AUTO_RTS | MA35_IER_AUTO_CTS);
351930cbf92SJacky Huang 		serial_out(up, MA35_IER_REG, ier);
352930cbf92SJacky Huang 		up->port.flags &= ~UPF_HARD_FLOW;
353930cbf92SJacky Huang 	}
354930cbf92SJacky Huang 
355930cbf92SJacky Huang 	msr = serial_in(up, MA35_MSR_REG);
356930cbf92SJacky Huang 	msr |= MA35_MSR_CTSACTLV;
357930cbf92SJacky Huang 	serial_out(up, MA35_MSR_REG, msr);
358930cbf92SJacky Huang 	serial_out(up, MA35_MCR_REG, mcr);
359930cbf92SJacky Huang }
360930cbf92SJacky Huang 
ma35d1serial_break_ctl(struct uart_port * port,int break_state)361930cbf92SJacky Huang static void ma35d1serial_break_ctl(struct uart_port *port, int break_state)
362930cbf92SJacky Huang {
363930cbf92SJacky Huang 	struct uart_ma35d1_port *up = to_ma35d1_uart_port(port);
364930cbf92SJacky Huang 	unsigned long flags;
365930cbf92SJacky Huang 	u32 lcr;
366930cbf92SJacky Huang 
367930cbf92SJacky Huang 	spin_lock_irqsave(&up->port.lock, flags);
368930cbf92SJacky Huang 	lcr = serial_in(up, MA35_LCR_REG);
369930cbf92SJacky Huang 	if (break_state != 0)
370930cbf92SJacky Huang 		lcr |= MA35_LCR_BREAK;
371930cbf92SJacky Huang 	else
372930cbf92SJacky Huang 		lcr &= ~MA35_LCR_BREAK;
373930cbf92SJacky Huang 	serial_out(up, MA35_LCR_REG, lcr);
374930cbf92SJacky Huang 	spin_unlock_irqrestore(&up->port.lock, flags);
375930cbf92SJacky Huang }
376930cbf92SJacky Huang 
ma35d1serial_startup(struct uart_port * port)377930cbf92SJacky Huang static int ma35d1serial_startup(struct uart_port *port)
378930cbf92SJacky Huang {
379930cbf92SJacky Huang 	struct uart_ma35d1_port *up = to_ma35d1_uart_port(port);
380930cbf92SJacky Huang 	u32 fcr;
381930cbf92SJacky Huang 	int retval;
382930cbf92SJacky Huang 
383930cbf92SJacky Huang 	/* Reset FIFO */
384930cbf92SJacky Huang 	serial_out(up, MA35_FCR_REG, MA35_FCR_TFR | MA35_FCR_RFR);
385930cbf92SJacky Huang 
386930cbf92SJacky Huang 	/* Clear pending interrupts */
387930cbf92SJacky Huang 	serial_out(up, MA35_ISR_REG, MA35_ISR_ALL);
388930cbf92SJacky Huang 
389930cbf92SJacky Huang 	retval = request_irq(port->irq, ma35d1serial_interrupt, 0,
390930cbf92SJacky Huang 			     dev_name(port->dev), port);
391930cbf92SJacky Huang 	if (retval) {
392930cbf92SJacky Huang 		dev_err(up->port.dev, "request irq failed.\n");
393930cbf92SJacky Huang 		return retval;
394930cbf92SJacky Huang 	}
395930cbf92SJacky Huang 
396930cbf92SJacky Huang 	fcr = serial_in(up, MA35_FCR_REG);
397930cbf92SJacky Huang 	fcr |= MA35_FCR_RFITL_4BYTES | MA35_FCR_RTSTL_8BYTES;
398930cbf92SJacky Huang 	serial_out(up, MA35_FCR_REG, fcr);
399930cbf92SJacky Huang 	serial_out(up, MA35_LCR_REG, MA35_LCR_WLS_8BITS);
400930cbf92SJacky Huang 	serial_out(up, MA35_TOR_REG, MA35_UART_RX_TOUT);
401930cbf92SJacky Huang 	serial_out(up, MA35_IER_REG, MA35_IER_CONFIG);
402930cbf92SJacky Huang 	return 0;
403930cbf92SJacky Huang }
404930cbf92SJacky Huang 
ma35d1serial_shutdown(struct uart_port * port)405930cbf92SJacky Huang static void ma35d1serial_shutdown(struct uart_port *port)
406930cbf92SJacky Huang {
407930cbf92SJacky Huang 	struct uart_ma35d1_port *up = to_ma35d1_uart_port(port);
408930cbf92SJacky Huang 
409930cbf92SJacky Huang 	serial_out(up, MA35_IER_REG, 0);
410930cbf92SJacky Huang 	free_irq(port->irq, port);
411930cbf92SJacky Huang }
412930cbf92SJacky Huang 
ma35d1serial_set_termios(struct uart_port * port,struct ktermios * termios,const struct ktermios * old)413930cbf92SJacky Huang static void ma35d1serial_set_termios(struct uart_port *port,
414930cbf92SJacky Huang 				     struct ktermios *termios,
415930cbf92SJacky Huang 				     const struct ktermios *old)
416930cbf92SJacky Huang {
417930cbf92SJacky Huang 	struct uart_ma35d1_port *up = to_ma35d1_uart_port(port);
418930cbf92SJacky Huang 	unsigned long flags;
419930cbf92SJacky Huang 	u32 baud, quot;
420930cbf92SJacky Huang 	u32 lcr = 0;
421930cbf92SJacky Huang 
422930cbf92SJacky Huang 	lcr = UART_LCR_WLEN(tty_get_char_size(termios->c_cflag));
423930cbf92SJacky Huang 
424930cbf92SJacky Huang 	if (termios->c_cflag & CSTOPB)
425930cbf92SJacky Huang 		lcr |= MA35_LCR_NSB;
426930cbf92SJacky Huang 	if (termios->c_cflag & PARENB)
427930cbf92SJacky Huang 		lcr |= MA35_LCR_PBE;
428930cbf92SJacky Huang 	if (!(termios->c_cflag & PARODD))
429930cbf92SJacky Huang 		lcr |= MA35_LCR_EPE;
430930cbf92SJacky Huang 	if (termios->c_cflag & CMSPAR)
431930cbf92SJacky Huang 		lcr |= MA35_LCR_SPE;
432930cbf92SJacky Huang 
433930cbf92SJacky Huang 	baud = uart_get_baud_rate(port, termios, old,
434930cbf92SJacky Huang 				  port->uartclk / MA35_BAUD_DIV_MAX,
435930cbf92SJacky Huang 				  port->uartclk / MA35_BAUD_DIV_MIN);
436930cbf92SJacky Huang 
437930cbf92SJacky Huang 	/* MA35D1 UART baud rate equation: baudrate = UART_CLK / (quot + 2) */
438930cbf92SJacky Huang 	quot = (port->uartclk / baud) - 2;
439930cbf92SJacky Huang 
440930cbf92SJacky Huang 	/*
441930cbf92SJacky Huang 	 * Ok, we're now changing the port state.  Do it with
442930cbf92SJacky Huang 	 * interrupts disabled.
443930cbf92SJacky Huang 	 */
444930cbf92SJacky Huang 	spin_lock_irqsave(&up->port.lock, flags);
445930cbf92SJacky Huang 
446930cbf92SJacky Huang 	up->port.read_status_mask = MA35_FSR_RX_OVER_IF;
447930cbf92SJacky Huang 	if (termios->c_iflag & INPCK)
448930cbf92SJacky Huang 		up->port.read_status_mask |= MA35_FSR_FEF | MA35_FSR_PEF;
449930cbf92SJacky Huang 	if (termios->c_iflag & (BRKINT | PARMRK))
450930cbf92SJacky Huang 		up->port.read_status_mask |= MA35_FSR_BIF;
451930cbf92SJacky Huang 
452930cbf92SJacky Huang 	/* Characteres to ignore */
453930cbf92SJacky Huang 	up->port.ignore_status_mask = 0;
454930cbf92SJacky Huang 	if (termios->c_iflag & IGNPAR)
455930cbf92SJacky Huang 		up->port.ignore_status_mask |= MA35_FSR_FEF | MA35_FSR_PEF;
456930cbf92SJacky Huang 	if (termios->c_iflag & IGNBRK) {
457930cbf92SJacky Huang 		up->port.ignore_status_mask |= MA35_FSR_BIF;
458930cbf92SJacky Huang 		/*
459930cbf92SJacky Huang 		 * If we're ignoring parity and break indicators,
460930cbf92SJacky Huang 		 * ignore overruns too (for real raw support).
461930cbf92SJacky Huang 		 */
462930cbf92SJacky Huang 		if (termios->c_iflag & IGNPAR)
463930cbf92SJacky Huang 			up->port.ignore_status_mask |= MA35_FSR_RX_OVER_IF;
464930cbf92SJacky Huang 	}
465930cbf92SJacky Huang 	if (termios->c_cflag & CRTSCTS)
466930cbf92SJacky Huang 		up->mcr |= UART_MCR_AFE;
467930cbf92SJacky Huang 	else
468930cbf92SJacky Huang 		up->mcr &= ~UART_MCR_AFE;
469930cbf92SJacky Huang 
470930cbf92SJacky Huang 	uart_update_timeout(port, termios->c_cflag, baud);
471930cbf92SJacky Huang 
472930cbf92SJacky Huang 	ma35d1serial_set_mctrl(&up->port, up->port.mctrl);
473930cbf92SJacky Huang 
474930cbf92SJacky Huang 	serial_out(up, MA35_BAUD_REG, MA35_BAUD_MODE2 | FIELD_PREP(MA35_BAUD_MASK, quot));
475930cbf92SJacky Huang 
476930cbf92SJacky Huang 	serial_out(up, MA35_LCR_REG, lcr);
477930cbf92SJacky Huang 
478930cbf92SJacky Huang 	spin_unlock_irqrestore(&up->port.lock, flags);
479930cbf92SJacky Huang }
480930cbf92SJacky Huang 
ma35d1serial_type(struct uart_port * port)481930cbf92SJacky Huang static const char *ma35d1serial_type(struct uart_port *port)
482930cbf92SJacky Huang {
483930cbf92SJacky Huang 	return "ma35d1-uart";
484930cbf92SJacky Huang }
485930cbf92SJacky Huang 
ma35d1serial_config_port(struct uart_port * port,int flags)486930cbf92SJacky Huang static void ma35d1serial_config_port(struct uart_port *port, int flags)
487930cbf92SJacky Huang {
488930cbf92SJacky Huang 	/*
489930cbf92SJacky Huang 	 * Driver core for serial ports forces a non-zero value for port type.
490930cbf92SJacky Huang 	 * Write an arbitrary value here to accommodate the serial core driver,
491930cbf92SJacky Huang 	 * as ID part of UAPI is redundant.
492930cbf92SJacky Huang 	 */
493930cbf92SJacky Huang 	port->type = 1;
494930cbf92SJacky Huang }
495930cbf92SJacky Huang 
ma35d1serial_verify_port(struct uart_port * port,struct serial_struct * ser)496930cbf92SJacky Huang static int ma35d1serial_verify_port(struct uart_port *port, struct serial_struct *ser)
497930cbf92SJacky Huang {
498930cbf92SJacky Huang 	if (port->type != PORT_UNKNOWN && ser->type != 1)
499930cbf92SJacky Huang 		return -EINVAL;
500930cbf92SJacky Huang 
501930cbf92SJacky Huang 	return 0;
502930cbf92SJacky Huang }
503930cbf92SJacky Huang 
504930cbf92SJacky Huang static const struct uart_ops ma35d1serial_ops = {
505930cbf92SJacky Huang 	.tx_empty     = ma35d1serial_tx_empty,
506930cbf92SJacky Huang 	.set_mctrl    = ma35d1serial_set_mctrl,
507930cbf92SJacky Huang 	.get_mctrl    = ma35d1serial_get_mctrl,
508930cbf92SJacky Huang 	.stop_tx      = ma35d1serial_stop_tx,
509930cbf92SJacky Huang 	.start_tx     = ma35d1serial_start_tx,
510930cbf92SJacky Huang 	.stop_rx      = ma35d1serial_stop_rx,
511930cbf92SJacky Huang 	.break_ctl    = ma35d1serial_break_ctl,
512930cbf92SJacky Huang 	.startup      = ma35d1serial_startup,
513930cbf92SJacky Huang 	.shutdown     = ma35d1serial_shutdown,
514930cbf92SJacky Huang 	.set_termios  = ma35d1serial_set_termios,
515930cbf92SJacky Huang 	.type         = ma35d1serial_type,
516930cbf92SJacky Huang 	.config_port  = ma35d1serial_config_port,
517930cbf92SJacky Huang 	.verify_port  = ma35d1serial_verify_port,
518930cbf92SJacky Huang };
519930cbf92SJacky Huang 
520930cbf92SJacky Huang static const struct of_device_id ma35d1_serial_of_match[] = {
521930cbf92SJacky Huang 	{ .compatible = "nuvoton,ma35d1-uart" },
522930cbf92SJacky Huang 	{},
523930cbf92SJacky Huang };
524930cbf92SJacky Huang MODULE_DEVICE_TABLE(of, ma35d1_serial_of_match);
525930cbf92SJacky Huang 
526930cbf92SJacky Huang #ifdef CONFIG_SERIAL_NUVOTON_MA35D1_CONSOLE
527930cbf92SJacky Huang 
528930cbf92SJacky Huang static struct device_node *ma35d1serial_uart_nodes[MA35_UART_NR];
529930cbf92SJacky Huang 
wait_for_xmitr(struct uart_ma35d1_port * up)530930cbf92SJacky Huang static void wait_for_xmitr(struct uart_ma35d1_port *up)
531930cbf92SJacky Huang {
532930cbf92SJacky Huang 	unsigned int reg = 0;
533930cbf92SJacky Huang 
534930cbf92SJacky Huang 	read_poll_timeout_atomic(serial_in, reg, reg & MA35_FSR_TX_EMPTY,
535930cbf92SJacky Huang 				 1, 10000, false,
536930cbf92SJacky Huang 				 up, MA35_FSR_REG);
537930cbf92SJacky Huang }
538930cbf92SJacky Huang 
ma35d1serial_console_putchar(struct uart_port * port,unsigned char ch)539930cbf92SJacky Huang static void ma35d1serial_console_putchar(struct uart_port *port, unsigned char ch)
540930cbf92SJacky Huang {
541930cbf92SJacky Huang 	struct uart_ma35d1_port *up = to_ma35d1_uart_port(port);
542930cbf92SJacky Huang 
543930cbf92SJacky Huang 	wait_for_xmitr(up);
544930cbf92SJacky Huang 	serial_out(up, MA35_THR_REG, ch);
545930cbf92SJacky Huang }
546930cbf92SJacky Huang 
547930cbf92SJacky Huang /*
548930cbf92SJacky Huang  *  Print a string to the serial port trying not to disturb
549930cbf92SJacky Huang  *  any possible real use of the port...
550930cbf92SJacky Huang  *
551930cbf92SJacky Huang  *  The console_lock must be held when we get here.
552930cbf92SJacky Huang  */
ma35d1serial_console_write(struct console * co,const char * s,u32 count)553930cbf92SJacky Huang static void ma35d1serial_console_write(struct console *co, const char *s, u32 count)
554930cbf92SJacky Huang {
555c88a91caSAndi Shyti 	struct uart_ma35d1_port *up;
556930cbf92SJacky Huang 	unsigned long flags;
557930cbf92SJacky Huang 	int locked = 1;
558930cbf92SJacky Huang 	u32 ier;
559930cbf92SJacky Huang 
560c88a91caSAndi Shyti 	if ((co->index < 0) || (co->index >= MA35_UART_NR)) {
561c88a91caSAndi Shyti 		pr_warn("Failed to write on ononsole port %x, out of range\n",
562c88a91caSAndi Shyti 			co->index);
563c88a91caSAndi Shyti 		return;
564c88a91caSAndi Shyti 	}
565c88a91caSAndi Shyti 
566c88a91caSAndi Shyti 	up = &ma35d1serial_ports[co->index];
567c88a91caSAndi Shyti 
568930cbf92SJacky Huang 	if (up->port.sysrq)
569930cbf92SJacky Huang 		locked = 0;
570930cbf92SJacky Huang 	else if (oops_in_progress)
571930cbf92SJacky Huang 		locked = spin_trylock_irqsave(&up->port.lock, flags);
572930cbf92SJacky Huang 	else
573930cbf92SJacky Huang 		spin_lock_irqsave(&up->port.lock, flags);
574930cbf92SJacky Huang 
575930cbf92SJacky Huang 	/*
576930cbf92SJacky Huang 	 *  First save the IER then disable the interrupts
577930cbf92SJacky Huang 	 */
578930cbf92SJacky Huang 	ier = serial_in(up, MA35_IER_REG);
579930cbf92SJacky Huang 	serial_out(up, MA35_IER_REG, 0);
580930cbf92SJacky Huang 
581930cbf92SJacky Huang 	uart_console_write(&up->port, s, count, ma35d1serial_console_putchar);
582930cbf92SJacky Huang 
583930cbf92SJacky Huang 	wait_for_xmitr(up);
584930cbf92SJacky Huang 	serial_out(up, MA35_IER_REG, ier);
585930cbf92SJacky Huang 
586930cbf92SJacky Huang 	if (locked)
587930cbf92SJacky Huang 		spin_unlock_irqrestore(&up->port.lock, flags);
588930cbf92SJacky Huang }
589930cbf92SJacky Huang 
ma35d1serial_console_setup(struct console * co,char * options)590930cbf92SJacky Huang static int __init ma35d1serial_console_setup(struct console *co, char *options)
591930cbf92SJacky Huang {
592930cbf92SJacky Huang 	struct device_node *np;
593930cbf92SJacky Huang 	struct uart_ma35d1_port *p;
594930cbf92SJacky Huang 	u32 val32[4];
595930cbf92SJacky Huang 	struct uart_port *port;
596930cbf92SJacky Huang 	int baud = 115200;
597930cbf92SJacky Huang 	int bits = 8;
598930cbf92SJacky Huang 	int parity = 'n';
599930cbf92SJacky Huang 	int flow = 'n';
600930cbf92SJacky Huang 
601930cbf92SJacky Huang 	if ((co->index < 0) || (co->index >= MA35_UART_NR)) {
602930cbf92SJacky Huang 		pr_debug("Console Port%x out of range\n", co->index);
603930cbf92SJacky Huang 		return -EINVAL;
604930cbf92SJacky Huang 	}
605930cbf92SJacky Huang 
606930cbf92SJacky Huang 	np = ma35d1serial_uart_nodes[co->index];
607930cbf92SJacky Huang 	p = &ma35d1serial_ports[co->index];
608930cbf92SJacky Huang 	if (!np || !p)
609930cbf92SJacky Huang 		return -ENODEV;
610930cbf92SJacky Huang 
611930cbf92SJacky Huang 	if (of_property_read_u32_array(np, "reg", val32, ARRAY_SIZE(val32)) != 0)
612930cbf92SJacky Huang 		return -EINVAL;
613930cbf92SJacky Huang 
614930cbf92SJacky Huang 	p->port.iobase = val32[1];
615930cbf92SJacky Huang 	p->port.membase = ioremap(p->port.iobase, MA35_UART_REG_SIZE);
616930cbf92SJacky Huang 	if (!p->port.membase)
617930cbf92SJacky Huang 		return -ENOMEM;
618930cbf92SJacky Huang 
619930cbf92SJacky Huang 	p->port.ops = &ma35d1serial_ops;
620930cbf92SJacky Huang 	p->port.line = 0;
621930cbf92SJacky Huang 	p->port.uartclk = MA35_UART_CONSOLE_CLK;
622930cbf92SJacky Huang 
623930cbf92SJacky Huang 	port = &ma35d1serial_ports[co->index].port;
624930cbf92SJacky Huang 
625930cbf92SJacky Huang 	if (options)
626930cbf92SJacky Huang 		uart_parse_options(options, &baud, &parity, &bits, &flow);
627930cbf92SJacky Huang 
628930cbf92SJacky Huang 	return uart_set_options(port, co, baud, parity, bits, flow);
629930cbf92SJacky Huang }
630930cbf92SJacky Huang 
631930cbf92SJacky Huang static struct console ma35d1serial_console = {
632930cbf92SJacky Huang 	.name    = "ttyNVT",
633930cbf92SJacky Huang 	.write   = ma35d1serial_console_write,
634930cbf92SJacky Huang 	.device  = uart_console_device,
635930cbf92SJacky Huang 	.setup   = ma35d1serial_console_setup,
636930cbf92SJacky Huang 	.flags   = CON_PRINTBUFFER | CON_ENABLED,
637930cbf92SJacky Huang 	.index   = -1,
638930cbf92SJacky Huang 	.data    = &ma35d1serial_reg,
639930cbf92SJacky Huang };
640930cbf92SJacky Huang 
ma35d1serial_console_init_port(void)641930cbf92SJacky Huang static void ma35d1serial_console_init_port(void)
642930cbf92SJacky Huang {
643930cbf92SJacky Huang 	u32 i = 0;
644930cbf92SJacky Huang 	struct device_node *np;
645930cbf92SJacky Huang 
646930cbf92SJacky Huang 	for_each_matching_node(np, ma35d1_serial_of_match) {
647930cbf92SJacky Huang 		if (ma35d1serial_uart_nodes[i] == NULL) {
648930cbf92SJacky Huang 			of_node_get(np);
649930cbf92SJacky Huang 			ma35d1serial_uart_nodes[i] = np;
650930cbf92SJacky Huang 			i++;
651930cbf92SJacky Huang 			if (i == MA35_UART_NR)
652930cbf92SJacky Huang 				break;
653930cbf92SJacky Huang 		}
654930cbf92SJacky Huang 	}
655930cbf92SJacky Huang }
656930cbf92SJacky Huang 
ma35d1serial_console_init(void)657930cbf92SJacky Huang static int __init ma35d1serial_console_init(void)
658930cbf92SJacky Huang {
659930cbf92SJacky Huang 	ma35d1serial_console_init_port();
660930cbf92SJacky Huang 	register_console(&ma35d1serial_console);
661930cbf92SJacky Huang 	return 0;
662930cbf92SJacky Huang }
663930cbf92SJacky Huang console_initcall(ma35d1serial_console_init);
664930cbf92SJacky Huang 
665930cbf92SJacky Huang #define MA35D1SERIAL_CONSOLE    (&ma35d1serial_console)
666930cbf92SJacky Huang #else
667930cbf92SJacky Huang #define MA35D1SERIAL_CONSOLE    NULL
668930cbf92SJacky Huang #endif
669930cbf92SJacky Huang 
670930cbf92SJacky Huang static struct uart_driver ma35d1serial_reg = {
671930cbf92SJacky Huang 	.owner        = THIS_MODULE,
672930cbf92SJacky Huang 	.driver_name  = "serial",
673930cbf92SJacky Huang 	.dev_name     = "ttyNVT",
674930cbf92SJacky Huang 	.major        = TTY_MAJOR,
675930cbf92SJacky Huang 	.minor        = 64,
676930cbf92SJacky Huang 	.cons         = MA35D1SERIAL_CONSOLE,
677930cbf92SJacky Huang 	.nr           = MA35_UART_NR,
678930cbf92SJacky Huang };
679930cbf92SJacky Huang 
680930cbf92SJacky Huang /*
681930cbf92SJacky Huang  * Register a set of serial devices attached to a platform device.
682930cbf92SJacky Huang  * The list is terminated with a zero flags entry, which means we expect
683930cbf92SJacky Huang  * all entries to have at least UPF_BOOT_AUTOCONF set.
684930cbf92SJacky Huang  */
ma35d1serial_probe(struct platform_device * pdev)685930cbf92SJacky Huang static int ma35d1serial_probe(struct platform_device *pdev)
686930cbf92SJacky Huang {
687930cbf92SJacky Huang 	struct resource *res_mem;
688930cbf92SJacky Huang 	struct uart_ma35d1_port *up;
689930cbf92SJacky Huang 	int ret = 0;
690930cbf92SJacky Huang 
691*23efa74cSJacky Huang 	if (!pdev->dev.of_node)
692*23efa74cSJacky Huang 		return -ENODEV;
693*23efa74cSJacky Huang 
694930cbf92SJacky Huang 	ret = of_alias_get_id(pdev->dev.of_node, "serial");
695930cbf92SJacky Huang 	if (ret < 0) {
696930cbf92SJacky Huang 		dev_err(&pdev->dev, "failed to get alias/pdev id, errno %d\n", ret);
697930cbf92SJacky Huang 		return ret;
698930cbf92SJacky Huang 	}
699930cbf92SJacky Huang 	up = &ma35d1serial_ports[ret];
700930cbf92SJacky Huang 	up->port.line = ret;
701930cbf92SJacky Huang 	res_mem = platform_get_resource(pdev, IORESOURCE_MEM, 0);
702930cbf92SJacky Huang 	if (!res_mem)
703930cbf92SJacky Huang 		return -ENODEV;
704930cbf92SJacky Huang 
705930cbf92SJacky Huang 	up->port.iobase = res_mem->start;
706930cbf92SJacky Huang 	up->port.membase = ioremap(up->port.iobase, MA35_UART_REG_SIZE);
707930cbf92SJacky Huang 	up->port.ops = &ma35d1serial_ops;
708930cbf92SJacky Huang 
709930cbf92SJacky Huang 	spin_lock_init(&up->port.lock);
710930cbf92SJacky Huang 
711930cbf92SJacky Huang 	up->clk = of_clk_get(pdev->dev.of_node, 0);
712930cbf92SJacky Huang 	if (IS_ERR(up->clk)) {
713930cbf92SJacky Huang 		ret = PTR_ERR(up->clk);
714930cbf92SJacky Huang 		dev_err(&pdev->dev, "failed to get core clk: %d\n", ret);
715930cbf92SJacky Huang 		goto err_iounmap;
716930cbf92SJacky Huang 	}
717930cbf92SJacky Huang 
718930cbf92SJacky Huang 	ret = clk_prepare_enable(up->clk);
719930cbf92SJacky Huang 	if (ret)
720930cbf92SJacky Huang 		goto err_iounmap;
721930cbf92SJacky Huang 
722930cbf92SJacky Huang 	if (up->port.line != 0)
723930cbf92SJacky Huang 		up->port.uartclk = clk_get_rate(up->clk);
724930cbf92SJacky Huang 
725930cbf92SJacky Huang 	ret = platform_get_irq(pdev, 0);
726930cbf92SJacky Huang 	if (ret < 0)
727930cbf92SJacky Huang 		goto err_clk_disable;
728930cbf92SJacky Huang 
729930cbf92SJacky Huang 	up->port.irq = ret;
730930cbf92SJacky Huang 	up->port.dev = &pdev->dev;
731930cbf92SJacky Huang 	up->port.flags = UPF_BOOT_AUTOCONF;
732930cbf92SJacky Huang 
733930cbf92SJacky Huang 	platform_set_drvdata(pdev, up);
734930cbf92SJacky Huang 
735930cbf92SJacky Huang 	ret = uart_add_one_port(&ma35d1serial_reg, &up->port);
736930cbf92SJacky Huang 	if (ret < 0)
737930cbf92SJacky Huang 		goto err_free_irq;
738930cbf92SJacky Huang 
739930cbf92SJacky Huang 	return 0;
740930cbf92SJacky Huang 
741930cbf92SJacky Huang err_free_irq:
742930cbf92SJacky Huang 	free_irq(up->port.irq, &up->port);
743930cbf92SJacky Huang 
744930cbf92SJacky Huang err_clk_disable:
745930cbf92SJacky Huang 	clk_disable_unprepare(up->clk);
746930cbf92SJacky Huang 
747930cbf92SJacky Huang err_iounmap:
748930cbf92SJacky Huang 	iounmap(up->port.membase);
749930cbf92SJacky Huang 	return ret;
750930cbf92SJacky Huang }
751930cbf92SJacky Huang 
752930cbf92SJacky Huang /*
753930cbf92SJacky Huang  * Remove serial ports registered against a platform device.
754930cbf92SJacky Huang  */
ma35d1serial_remove(struct platform_device * dev)755930cbf92SJacky Huang static int ma35d1serial_remove(struct platform_device *dev)
756930cbf92SJacky Huang {
757930cbf92SJacky Huang 	struct uart_port *port = platform_get_drvdata(dev);
758930cbf92SJacky Huang 	struct uart_ma35d1_port *up = to_ma35d1_uart_port(port);
759930cbf92SJacky Huang 
760930cbf92SJacky Huang 	uart_remove_one_port(&ma35d1serial_reg, port);
761930cbf92SJacky Huang 	clk_disable_unprepare(up->clk);
762930cbf92SJacky Huang 	return 0;
763930cbf92SJacky Huang }
764930cbf92SJacky Huang 
ma35d1serial_suspend(struct platform_device * dev,pm_message_t state)765930cbf92SJacky Huang static int ma35d1serial_suspend(struct platform_device *dev, pm_message_t state)
766930cbf92SJacky Huang {
767930cbf92SJacky Huang 	struct uart_port *port = platform_get_drvdata(dev);
768930cbf92SJacky Huang 	struct uart_ma35d1_port *up = to_ma35d1_uart_port(port);
769930cbf92SJacky Huang 
770930cbf92SJacky Huang 	uart_suspend_port(&ma35d1serial_reg, &up->port);
771930cbf92SJacky Huang 	if (up->port.line == 0) {
772930cbf92SJacky Huang 		up->console_baud_rate = serial_in(up, MA35_BAUD_REG);
773930cbf92SJacky Huang 		up->console_line = serial_in(up, MA35_LCR_REG);
774930cbf92SJacky Huang 		up->console_int = serial_in(up, MA35_IER_REG);
775930cbf92SJacky Huang 	}
776930cbf92SJacky Huang 	return 0;
777930cbf92SJacky Huang }
778930cbf92SJacky Huang 
ma35d1serial_resume(struct platform_device * dev)779930cbf92SJacky Huang static int ma35d1serial_resume(struct platform_device *dev)
780930cbf92SJacky Huang {
781930cbf92SJacky Huang 	struct uart_port *port = platform_get_drvdata(dev);
782930cbf92SJacky Huang 	struct uart_ma35d1_port *up = to_ma35d1_uart_port(port);
783930cbf92SJacky Huang 
784930cbf92SJacky Huang 	if (up->port.line == 0) {
785930cbf92SJacky Huang 		serial_out(up, MA35_BAUD_REG, up->console_baud_rate);
786930cbf92SJacky Huang 		serial_out(up, MA35_LCR_REG, up->console_line);
787930cbf92SJacky Huang 		serial_out(up, MA35_IER_REG, up->console_int);
788930cbf92SJacky Huang 	}
789930cbf92SJacky Huang 	uart_resume_port(&ma35d1serial_reg, &up->port);
790930cbf92SJacky Huang 	return 0;
791930cbf92SJacky Huang }
792930cbf92SJacky Huang 
793930cbf92SJacky Huang static struct platform_driver ma35d1serial_driver = {
794930cbf92SJacky Huang 	.probe      = ma35d1serial_probe,
795930cbf92SJacky Huang 	.remove     = ma35d1serial_remove,
796930cbf92SJacky Huang 	.suspend    = ma35d1serial_suspend,
797930cbf92SJacky Huang 	.resume     = ma35d1serial_resume,
798930cbf92SJacky Huang 	.driver     = {
799930cbf92SJacky Huang 		.name   = "ma35d1-uart",
800930cbf92SJacky Huang 		.of_match_table = of_match_ptr(ma35d1_serial_of_match),
801930cbf92SJacky Huang 	},
802930cbf92SJacky Huang };
803930cbf92SJacky Huang 
ma35d1serial_init(void)804930cbf92SJacky Huang static int __init ma35d1serial_init(void)
805930cbf92SJacky Huang {
806930cbf92SJacky Huang 	int ret;
807930cbf92SJacky Huang 
808930cbf92SJacky Huang 	ret = uart_register_driver(&ma35d1serial_reg);
809930cbf92SJacky Huang 	if (ret)
810930cbf92SJacky Huang 		return ret;
811930cbf92SJacky Huang 
812930cbf92SJacky Huang 	ret = platform_driver_register(&ma35d1serial_driver);
813930cbf92SJacky Huang 	if (ret)
814930cbf92SJacky Huang 		uart_unregister_driver(&ma35d1serial_reg);
815930cbf92SJacky Huang 
816930cbf92SJacky Huang 	return ret;
817930cbf92SJacky Huang }
818930cbf92SJacky Huang 
ma35d1serial_exit(void)819930cbf92SJacky Huang static void __exit ma35d1serial_exit(void)
820930cbf92SJacky Huang {
821930cbf92SJacky Huang 	platform_driver_unregister(&ma35d1serial_driver);
822930cbf92SJacky Huang 	uart_unregister_driver(&ma35d1serial_reg);
823930cbf92SJacky Huang }
824930cbf92SJacky Huang 
825930cbf92SJacky Huang module_init(ma35d1serial_init);
826930cbf92SJacky Huang module_exit(ma35d1serial_exit);
827930cbf92SJacky Huang 
828930cbf92SJacky Huang MODULE_LICENSE("GPL");
829930cbf92SJacky Huang MODULE_DESCRIPTION("MA35D1 serial driver");
830