1 // SPDX-License-Identifier: GPL-2.0
2 #include <linux/interrupt.h>
3 #include <linux/ioport.h>
10 #include <linux/serial_core.h>
11 /* WARNING: Do not change this to <linux/serial.h> without testing that
12 * SERIAL_PORT_DFNS does get defined to the appropriate value.
14 #include <asm/serial.h>
16 #ifndef SERIAL_PORT_DFNS
17 #define SERIAL_PORT_DFNS
20 static void start_serial_interrupt(int irq
);
22 static const struct old_serial_port rs_table
[] = {
26 static const struct old_serial_port
*serstate
;
29 static int spk_serial_out(struct spk_synth
*in_synth
, const char ch
);
30 static void spk_serial_send_xchar(struct spk_synth
*in_synth
, char ch
);
31 static void spk_serial_tiocmset(struct spk_synth
*in_synth
, unsigned int set
, unsigned int clear
);
32 static unsigned char spk_serial_in(struct spk_synth
*in_synth
);
33 static unsigned char spk_serial_in_nowait(struct spk_synth
*in_synth
);
34 static void spk_serial_flush_buffer(struct spk_synth
*in_synth
);
35 static int spk_serial_wait_for_xmitr(struct spk_synth
*in_synth
);
37 struct spk_io_ops spk_serial_io_ops
= {
38 .synth_out
= spk_serial_out
,
39 .send_xchar
= spk_serial_send_xchar
,
40 .tiocmset
= spk_serial_tiocmset
,
41 .synth_in
= spk_serial_in
,
42 .synth_in_nowait
= spk_serial_in_nowait
,
43 .flush_buffer
= spk_serial_flush_buffer
,
44 .wait_for_xmitr
= spk_serial_wait_for_xmitr
,
46 EXPORT_SYMBOL_GPL(spk_serial_io_ops
);
48 const struct old_serial_port
*spk_serial_init(int index
)
50 int baud
= 9600, quot
= 0;
51 unsigned int cval
= 0;
52 int cflag
= CREAD
| HUPCL
| CLOCAL
| B9600
| CS8
;
53 const struct old_serial_port
*ser
;
56 if (index
>= ARRAY_SIZE(rs_table
)) {
57 pr_info("no port info for ttyS%d\n", index
);
60 ser
= rs_table
+ index
;
62 /* Divisor, byte size and parity */
63 quot
= ser
->baud_base
/ baud
;
64 cval
= cflag
& (CSIZE
| CSTOPB
);
65 #if defined(__powerpc__) || defined(__alpha__)
67 #else /* !__powerpc__ && !__alpha__ */
69 #endif /* !__powerpc__ && !__alpha__ */
71 cval
|= UART_LCR_PARITY
;
72 if (!(cflag
& PARODD
))
73 cval
|= UART_LCR_EPAR
;
74 if (synth_request_region(ser
->port
, 8)) {
75 /* try to take it back. */
76 pr_info("Ports not available, trying to steal them\n");
77 __release_region(&ioport_resource
, ser
->port
, 8);
78 err
= synth_request_region(ser
->port
, 8);
80 pr_warn("Unable to allocate port at %x, errno %i",
86 /* Disable UART interrupts, set DTR and RTS high
89 outb(cval
| UART_LCR_DLAB
, ser
->port
+ UART_LCR
); /* set DLAB */
90 outb(quot
& 0xff, ser
->port
+ UART_DLL
); /* LS of divisor */
91 outb(quot
>> 8, ser
->port
+ UART_DLM
); /* MS of divisor */
92 outb(cval
, ser
->port
+ UART_LCR
); /* reset DLAB */
94 /* Turn off Interrupts */
95 outb(0, ser
->port
+ UART_IER
);
96 outb(UART_MCR_DTR
| UART_MCR_RTS
, ser
->port
+ UART_MCR
);
98 /* If we read 0xff from the LSR, there is no UART here. */
99 if (inb(ser
->port
+ UART_LSR
) == 0xff) {
100 synth_release_region(ser
->port
, 8);
106 speakup_info
.port_tts
= ser
->port
;
109 start_serial_interrupt(ser
->irq
);
114 static irqreturn_t
synth_readbuf_handler(int irq
, void *dev_id
)
119 spin_lock_irqsave(&speakup_info
.spinlock
, flags
);
120 while (inb_p(speakup_info
.port_tts
+ UART_LSR
) & UART_LSR_DR
) {
121 c
= inb_p(speakup_info
.port_tts
+ UART_RX
);
122 synth
->read_buff_add((u_char
)c
);
124 spin_unlock_irqrestore(&speakup_info
.spinlock
, flags
);
128 static void start_serial_interrupt(int irq
)
132 if (!synth
->read_buff_add
)
135 rv
= request_irq(irq
, synth_readbuf_handler
, IRQF_SHARED
,
136 "serial", (void *)synth_readbuf_handler
);
139 pr_err("Unable to request Speakup serial I R Q\n");
141 outb(UART_MCR_DTR
| UART_MCR_RTS
| UART_MCR_OUT2
,
142 speakup_info
.port_tts
+ UART_MCR
);
143 /* Turn on Interrupts */
144 outb(UART_IER_MSI
| UART_IER_RLSI
| UART_IER_RDI
,
145 speakup_info
.port_tts
+ UART_IER
);
146 inb(speakup_info
.port_tts
+ UART_LSR
);
147 inb(speakup_info
.port_tts
+ UART_RX
);
148 inb(speakup_info
.port_tts
+ UART_IIR
);
149 inb(speakup_info
.port_tts
+ UART_MSR
);
150 outb(1, speakup_info
.port_tts
+ UART_FCR
); /* Turn FIFO On */
153 static void spk_serial_send_xchar(struct spk_synth
*synth
, char ch
)
155 int timeout
= SPK_XMITR_TIMEOUT
;
157 while (spk_serial_tx_busy()) {
162 outb(ch
, speakup_info
.port_tts
);
165 static void spk_serial_tiocmset(struct spk_synth
*in_synth
, unsigned int set
, unsigned int clear
)
167 int old
= inb(speakup_info
.port_tts
+ UART_MCR
);
169 outb((old
& ~clear
) | set
, speakup_info
.port_tts
+ UART_MCR
);
172 int spk_serial_synth_probe(struct spk_synth
*synth
)
174 const struct old_serial_port
*ser
;
177 if ((synth
->ser
>= SPK_LO_TTY
) && (synth
->ser
<= SPK_HI_TTY
)) {
178 ser
= spk_serial_init(synth
->ser
);
182 outb_p(0, ser
->port
);
184 outb_p('\r', ser
->port
);
188 pr_warn("ttyS%i is an invalid port\n", synth
->ser
);
191 pr_info("%s: not found\n", synth
->long_name
);
194 pr_info("%s: ttyS%i, Driver Version %s\n",
195 synth
->long_name
, synth
->ser
, synth
->version
);
199 EXPORT_SYMBOL_GPL(spk_serial_synth_probe
);
201 void spk_stop_serial_interrupt(void)
203 if (speakup_info
.port_tts
== 0)
206 if (!synth
->read_buff_add
)
209 /* Turn off interrupts */
210 outb(0, speakup_info
.port_tts
+ UART_IER
);
212 free_irq(serstate
->irq
, (void *)synth_readbuf_handler
);
214 EXPORT_SYMBOL_GPL(spk_stop_serial_interrupt
);
216 static int spk_serial_wait_for_xmitr(struct spk_synth
*in_synth
)
218 int tmout
= SPK_XMITR_TIMEOUT
;
220 if ((in_synth
->alive
) && (timeouts
>= NUM_DISABLE_TIMEOUTS
)) {
221 pr_warn("%s: too many timeouts, deactivating speakup\n",
222 in_synth
->long_name
);
224 /* No synth any more, so nobody will restart TTYs, and we thus
225 * need to do it ourselves. Now that there is no synth we can
226 * let application flood anyway
228 speakup_start_ttys();
232 while (spk_serial_tx_busy()) {
234 pr_warn("%s: timed out (tx busy)\n",
235 in_synth
->long_name
);
241 tmout
= SPK_CTS_TIMEOUT
;
242 while (!((inb_p(speakup_info
.port_tts
+ UART_MSR
)) & UART_MSR_CTS
)) {
254 static unsigned char spk_serial_in(struct spk_synth
*in_synth
)
256 int tmout
= SPK_SERIAL_TIMEOUT
;
258 while (!(inb_p(speakup_info
.port_tts
+ UART_LSR
) & UART_LSR_DR
)) {
260 pr_warn("time out while waiting for input.\n");
265 return inb_p(speakup_info
.port_tts
+ UART_RX
);
268 static unsigned char spk_serial_in_nowait(struct spk_synth
*in_synth
)
272 lsr
= inb_p(speakup_info
.port_tts
+ UART_LSR
);
273 if (!(lsr
& UART_LSR_DR
))
275 return inb_p(speakup_info
.port_tts
+ UART_RX
);
278 static void spk_serial_flush_buffer(struct spk_synth
*in_synth
)
280 /* TODO: flush the UART 16550 buffer */
283 static int spk_serial_out(struct spk_synth
*in_synth
, const char ch
)
285 if (in_synth
->alive
&& spk_serial_wait_for_xmitr(in_synth
)) {
286 outb_p(ch
, speakup_info
.port_tts
);
292 const char *spk_serial_synth_immediate(struct spk_synth
*synth
,
297 while ((ch
= *buff
)) {
299 ch
= synth
->procspeech
;
300 if (spk_serial_wait_for_xmitr(synth
))
301 outb(ch
, speakup_info
.port_tts
);
308 EXPORT_SYMBOL_GPL(spk_serial_synth_immediate
);
310 void spk_serial_release(struct spk_synth
*synth
)
312 spk_stop_serial_interrupt();
313 if (speakup_info
.port_tts
== 0)
315 synth_release_region(speakup_info
.port_tts
, 8);
316 speakup_info
.port_tts
= 0;
318 EXPORT_SYMBOL_GPL(spk_serial_release
);