blob: 0bbfce43031c9d6be2e4041db9c8fc6f1eea0203 [file] [log] [blame]
Linus Torvalds1da177e2005-04-16 15:20:36 -07001/*
2 * specialix.c -- specialix IO8+ multiport serial driver.
3 *
4 * Copyright (C) 1997 Roger Wolff (R.E.Wolff@BitWizard.nl)
5 * Copyright (C) 1994-1996 Dmitry Gorodchanin (pgmdsg@ibi.com)
6 *
7 * Specialix pays for the development and support of this driver.
8 * Please DO contact io8-linux@specialix.co.uk if you require
9 * support. But please read the documentation (specialix.txt)
10 * first.
11 *
12 * This driver was developped in the BitWizard linux device
13 * driver service. If you require a linux device driver for your
14 * product, please contact devices@BitWizard.nl for a quote.
15 *
16 * This code is firmly based on the riscom/8 serial driver,
17 * written by Dmitry Gorodchanin. The specialix IO8+ card
18 * programming information was obtained from the CL-CD1865 Data
19 * Book, and Specialix document number 6200059: IO8+ Hardware
20 * Functional Specification.
21 *
22 * This program is free software; you can redistribute it and/or
23 * modify it under the terms of the GNU General Public License as
24 * published by the Free Software Foundation; either version 2 of
25 * the License, or (at your option) any later version.
26 *
27 * This program is distributed in the hope that it will be
28 * useful, but WITHOUT ANY WARRANTY; without even the implied
29 * warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR
30 * PURPOSE. See the GNU General Public License for more details.
31 *
32 * You should have received a copy of the GNU General Public
33 * License along with this program; if not, write to the Free
34 * Software Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139,
35 * USA.
36 *
37 * Revision history:
38 *
39 * Revision 1.0: April 1st 1997.
40 * Initial release for alpha testing.
Jeff Garzikd61780c02005-10-30 15:01:51 -080041 * Revision 1.1: April 14th 1997.
42 * Incorporated Richard Hudsons suggestions,
Linus Torvalds1da177e2005-04-16 15:20:36 -070043 * removed some debugging printk's.
44 * Revision 1.2: April 15th 1997.
45 * Ported to 2.1.x kernels.
Jeff Garzikd61780c02005-10-30 15:01:51 -080046 * Revision 1.3: April 17th 1997
47 * Backported to 2.0. (Compatibility macros).
Linus Torvalds1da177e2005-04-16 15:20:36 -070048 * Revision 1.4: April 18th 1997
Jeff Garzikd61780c02005-10-30 15:01:51 -080049 * Fixed DTR/RTS bug that caused the card to indicate
50 * "don't send data" to a modem after the password prompt.
Linus Torvalds1da177e2005-04-16 15:20:36 -070051 * Fixed bug for premature (fake) interrupts.
52 * Revision 1.5: April 19th 1997
Jeff Garzikd61780c02005-10-30 15:01:51 -080053 * fixed a minor typo in the header file, cleanup a little.
Linus Torvalds1da177e2005-04-16 15:20:36 -070054 * performance warnings are now MAXed at once per minute.
55 * Revision 1.6: May 23 1997
56 * Changed the specialix=... format to include interrupt.
57 * Revision 1.7: May 27 1997
58 * Made many more debug printk's a compile time option.
59 * Revision 1.8: Jul 1 1997
60 * port to linux-2.1.43 kernel.
61 * Revision 1.9: Oct 9 1998
62 * Added stuff for the IO8+/PCI version.
Jeff Garzikd61780c02005-10-30 15:01:51 -080063 * Revision 1.10: Oct 22 1999 / Jan 21 2000.
64 * Added stuff for setserial.
Linus Torvalds1da177e2005-04-16 15:20:36 -070065 * Nicolas Mailhot (Nicolas.Mailhot@email.enst.fr)
Jeff Garzikd61780c02005-10-30 15:01:51 -080066 *
Linus Torvalds1da177e2005-04-16 15:20:36 -070067 */
68
69#define VERSION "1.11"
70
71
72/*
73 * There is a bunch of documentation about the card, jumpers, config
74 * settings, restrictions, cables, device names and numbers in
75 * Documentation/specialix.txt
76 */
77
78#include <linux/config.h>
79#include <linux/module.h>
80
81#include <asm/io.h>
82#include <linux/kernel.h>
83#include <linux/sched.h>
84#include <linux/ioport.h>
85#include <linux/interrupt.h>
86#include <linux/errno.h>
87#include <linux/tty.h>
88#include <linux/mm.h>
89#include <linux/serial.h>
90#include <linux/fcntl.h>
91#include <linux/major.h>
92#include <linux/delay.h>
Linus Torvalds1da177e2005-04-16 15:20:36 -070093#include <linux/pci.h>
94#include <linux/init.h>
95#include <asm/uaccess.h>
96
97#include "specialix_io8.h"
98#include "cd1865.h"
99
100
101/*
102 This driver can spew a whole lot of debugging output at you. If you
103 need maximum performance, you should disable the DEBUG define. To
104 aid in debugging in the field, I'm leaving the compile-time debug
105 features enabled, and disable them "runtime". That allows me to
106 instruct people with problems to enable debugging without requiring
107 them to recompile...
108*/
109#define DEBUG
110
111static int sx_debug;
112static int sx_rxfifo = SPECIALIX_RXFIFO;
113
114#ifdef DEBUG
115#define dprintk(f, str...) if (sx_debug & f) printk (str)
116#else
117#define dprintk(f, str...) /* nothing */
118#endif
119
120#define SX_DEBUG_FLOW 0x0001
121#define SX_DEBUG_DATA 0x0002
122#define SX_DEBUG_PROBE 0x0004
123#define SX_DEBUG_CHAN 0x0008
124#define SX_DEBUG_INIT 0x0010
125#define SX_DEBUG_RX 0x0020
126#define SX_DEBUG_TX 0x0040
127#define SX_DEBUG_IRQ 0x0080
128#define SX_DEBUG_OPEN 0x0100
129#define SX_DEBUG_TERMIOS 0x0200
130#define SX_DEBUG_SIGNALS 0x0400
131#define SX_DEBUG_FIFO 0x0800
132
133
134#define func_enter() dprintk (SX_DEBUG_FLOW, "io8: enter %s\n",__FUNCTION__)
135#define func_exit() dprintk (SX_DEBUG_FLOW, "io8: exit %s\n", __FUNCTION__)
136
137#define jiffies_from_ms(a) ((((a) * HZ)/1000)+1)
138
139
140/* Configurable options: */
141
142/* Am I paranoid or not ? ;-) */
143#define SPECIALIX_PARANOIA_CHECK
144
145/* Do I trust the IRQ from the card? (enabeling it doesn't seem to help)
146 When the IRQ routine leaves the chip in a state that is keeps on
147 requiring attention, the timer doesn't help either. */
148#undef SPECIALIX_TIMER
149
150#ifdef SPECIALIX_TIMER
151static int sx_poll = HZ;
152#endif
153
154
155
Jeff Garzikd61780c02005-10-30 15:01:51 -0800156/*
Linus Torvalds1da177e2005-04-16 15:20:36 -0700157 * The following defines are mostly for testing purposes. But if you need
158 * some nice reporting in your syslog, you can define them also.
159 */
160#undef SX_REPORT_FIFO
161#undef SX_REPORT_OVERRUN
162
163
164
165#ifdef CONFIG_SPECIALIX_RTSCTS
166#define SX_CRTSCTS(bla) 1
167#else
168#define SX_CRTSCTS(tty) C_CRTSCTS(tty)
169#endif
170
171
172/* Used to be outb (0xff, 0x80); */
173#define short_pause() udelay (1)
174
175
176#define SPECIALIX_LEGAL_FLAGS \
177 (ASYNC_HUP_NOTIFY | ASYNC_SAK | ASYNC_SPLIT_TERMIOS | \
178 ASYNC_SPD_HI | ASYNC_SPEED_VHI | ASYNC_SESSION_LOCKOUT | \
179 ASYNC_PGRP_LOCKOUT | ASYNC_CALLOUT_NOHUP)
180
181#undef RS_EVENT_WRITE_WAKEUP
182#define RS_EVENT_WRITE_WAKEUP 0
183
184static struct tty_driver *specialix_driver;
185static unsigned char * tmp_buf;
186static DECLARE_MUTEX(tmp_buf_sem);
187
188static unsigned long baud_table[] = {
189 0, 50, 75, 110, 134, 150, 200, 300, 600, 1200, 1800, 2400, 4800,
Jeff Garzikd61780c02005-10-30 15:01:51 -0800190 9600, 19200, 38400, 57600, 115200, 0,
Linus Torvalds1da177e2005-04-16 15:20:36 -0700191};
192
193static struct specialix_board sx_board[SX_NBOARD] = {
194 { 0, SX_IOBASE1, 9, },
195 { 0, SX_IOBASE2, 11, },
196 { 0, SX_IOBASE3, 12, },
197 { 0, SX_IOBASE4, 15, },
198};
199
200static struct specialix_port sx_port[SX_NBOARD * SX_NPORT];
201
202
203#ifdef SPECIALIX_TIMER
204static struct timer_list missed_irq_timer;
205static irqreturn_t sx_interrupt(int irq, void * dev_id, struct pt_regs * regs);
206#endif
207
208
209
210static inline int sx_paranoia_check(struct specialix_port const * port,
211 char *name, const char *routine)
212{
213#ifdef SPECIALIX_PARANOIA_CHECK
214 static const char *badmagic =
215 KERN_ERR "sx: Warning: bad specialix port magic number for device %s in %s\n";
216 static const char *badinfo =
217 KERN_ERR "sx: Warning: null specialix port for device %s in %s\n";
Jeff Garzikd61780c02005-10-30 15:01:51 -0800218
Linus Torvalds1da177e2005-04-16 15:20:36 -0700219 if (!port) {
220 printk(badinfo, name, routine);
221 return 1;
222 }
223 if (port->magic != SPECIALIX_MAGIC) {
224 printk(badmagic, name, routine);
225 return 1;
226 }
227#endif
228 return 0;
229}
230
231
232/*
Jeff Garzikd61780c02005-10-30 15:01:51 -0800233 *
Linus Torvalds1da177e2005-04-16 15:20:36 -0700234 * Service functions for specialix IO8+ driver.
Jeff Garzikd61780c02005-10-30 15:01:51 -0800235 *
Linus Torvalds1da177e2005-04-16 15:20:36 -0700236 */
237
238/* Get board number from pointer */
239static inline int board_No (struct specialix_board * bp)
240{
241 return bp - sx_board;
242}
243
244
245/* Get port number from pointer */
246static inline int port_No (struct specialix_port const * port)
247{
Jeff Garzikd61780c02005-10-30 15:01:51 -0800248 return SX_PORT(port - sx_port);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700249}
250
251
252/* Get pointer to board from pointer to port */
253static inline struct specialix_board * port_Board(struct specialix_port const * port)
254{
255 return &sx_board[SX_BOARD(port - sx_port)];
256}
257
258
259/* Input Byte from CL CD186x register */
260static inline unsigned char sx_in(struct specialix_board * bp, unsigned short reg)
261{
262 bp->reg = reg | 0x80;
263 outb (reg | 0x80, bp->base + SX_ADDR_REG);
264 return inb (bp->base + SX_DATA_REG);
265}
266
267
268/* Output Byte to CL CD186x register */
269static inline void sx_out(struct specialix_board * bp, unsigned short reg,
270 unsigned char val)
271{
272 bp->reg = reg | 0x80;
273 outb (reg | 0x80, bp->base + SX_ADDR_REG);
274 outb (val, bp->base + SX_DATA_REG);
275}
276
277
278/* Input Byte from CL CD186x register */
279static inline unsigned char sx_in_off(struct specialix_board * bp, unsigned short reg)
280{
281 bp->reg = reg;
282 outb (reg, bp->base + SX_ADDR_REG);
283 return inb (bp->base + SX_DATA_REG);
284}
285
286
287/* Output Byte to CL CD186x register */
288static inline void sx_out_off(struct specialix_board * bp, unsigned short reg,
289 unsigned char val)
290{
291 bp->reg = reg;
292 outb (reg, bp->base + SX_ADDR_REG);
293 outb (val, bp->base + SX_DATA_REG);
294}
295
296
297/* Wait for Channel Command Register ready */
298static inline void sx_wait_CCR(struct specialix_board * bp)
299{
300 unsigned long delay, flags;
301 unsigned char ccr;
302
303 for (delay = SX_CCR_TIMEOUT; delay; delay--) {
304 spin_lock_irqsave(&bp->lock, flags);
305 ccr = sx_in(bp, CD186x_CCR);
306 spin_unlock_irqrestore(&bp->lock, flags);
307 if (!ccr)
308 return;
309 udelay (1);
310 }
Jeff Garzikd61780c02005-10-30 15:01:51 -0800311
Linus Torvalds1da177e2005-04-16 15:20:36 -0700312 printk(KERN_ERR "sx%d: Timeout waiting for CCR.\n", board_No(bp));
313}
314
315
316/* Wait for Channel Command Register ready */
317static inline void sx_wait_CCR_off(struct specialix_board * bp)
318{
319 unsigned long delay;
320 unsigned char crr;
321 unsigned long flags;
322
323 for (delay = SX_CCR_TIMEOUT; delay; delay--) {
324 spin_lock_irqsave(&bp->lock, flags);
325 crr = sx_in_off(bp, CD186x_CCR);
326 spin_unlock_irqrestore(&bp->lock, flags);
327 if (!crr)
328 return;
329 udelay (1);
330 }
Jeff Garzikd61780c02005-10-30 15:01:51 -0800331
Linus Torvalds1da177e2005-04-16 15:20:36 -0700332 printk(KERN_ERR "sx%d: Timeout waiting for CCR.\n", board_No(bp));
333}
334
335
336/*
337 * specialix IO8+ IO range functions.
338 */
339
Jeff Garzikd61780c02005-10-30 15:01:51 -0800340static inline int sx_request_io_range(struct specialix_board * bp)
Linus Torvalds1da177e2005-04-16 15:20:36 -0700341{
Jeff Garzikd61780c02005-10-30 15:01:51 -0800342 return request_region(bp->base,
343 bp->flags & SX_BOARD_IS_PCI ? SX_PCI_IO_SPACE : SX_IO_SPACE,
344 "specialix IO8+") == NULL;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700345}
346
347
348static inline void sx_release_io_range(struct specialix_board * bp)
349{
Jeff Garzikd61780c02005-10-30 15:01:51 -0800350 release_region(bp->base,
Linus Torvalds1da177e2005-04-16 15:20:36 -0700351 bp->flags&SX_BOARD_IS_PCI?SX_PCI_IO_SPACE:SX_IO_SPACE);
352}
353
Jeff Garzikd61780c02005-10-30 15:01:51 -0800354
Linus Torvalds1da177e2005-04-16 15:20:36 -0700355/* Must be called with enabled interrupts */
Jeff Garzikd61780c02005-10-30 15:01:51 -0800356/* Ugly. Very ugly. Don't use this for anything else than initialization
Linus Torvalds1da177e2005-04-16 15:20:36 -0700357 code */
358static inline void sx_long_delay(unsigned long delay)
359{
360 unsigned long i;
Jeff Garzikd61780c02005-10-30 15:01:51 -0800361
Linus Torvalds1da177e2005-04-16 15:20:36 -0700362 for (i = jiffies + delay; time_after(i, jiffies); ) ;
363}
364
365
366
367/* Set the IRQ using the RTS lines that run to the PAL on the board.... */
368static int sx_set_irq ( struct specialix_board *bp)
369{
370 int virq;
371 int i;
372 unsigned long flags;
373
Jeff Garzikd61780c02005-10-30 15:01:51 -0800374 if (bp->flags & SX_BOARD_IS_PCI)
Linus Torvalds1da177e2005-04-16 15:20:36 -0700375 return 1;
376 switch (bp->irq) {
377 /* In the same order as in the docs... */
378 case 15: virq = 0;break;
379 case 12: virq = 1;break;
380 case 11: virq = 2;break;
381 case 9: virq = 3;break;
382 default: printk (KERN_ERR "Speclialix: cannot set irq to %d.\n", bp->irq);
383 return 0;
384 }
385 spin_lock_irqsave(&bp->lock, flags);
386 for (i=0;i<2;i++) {
387 sx_out(bp, CD186x_CAR, i);
388 sx_out(bp, CD186x_MSVRTS, ((virq >> i) & 0x1)? MSVR_RTS:0);
389 }
390 spin_unlock_irqrestore(&bp->lock, flags);
391 return 1;
392}
393
394
395/* Reset and setup CD186x chip */
396static int sx_init_CD186x(struct specialix_board * bp)
397{
398 unsigned long flags;
399 int scaler;
400 int rv = 1;
401
402 func_enter();
403 sx_wait_CCR_off(bp); /* Wait for CCR ready */
404 spin_lock_irqsave(&bp->lock, flags);
405 sx_out_off(bp, CD186x_CCR, CCR_HARDRESET); /* Reset CD186x chip */
406 spin_unlock_irqrestore(&bp->lock, flags);
407 sx_long_delay(HZ/20); /* Delay 0.05 sec */
408 spin_lock_irqsave(&bp->lock, flags);
409 sx_out_off(bp, CD186x_GIVR, SX_ID); /* Set ID for this chip */
410 sx_out_off(bp, CD186x_GICR, 0); /* Clear all bits */
411 sx_out_off(bp, CD186x_PILR1, SX_ACK_MINT); /* Prio for modem intr */
412 sx_out_off(bp, CD186x_PILR2, SX_ACK_TINT); /* Prio for transmitter intr */
413 sx_out_off(bp, CD186x_PILR3, SX_ACK_RINT); /* Prio for receiver intr */
414 /* Set RegAckEn */
415 sx_out_off(bp, CD186x_SRCR, sx_in (bp, CD186x_SRCR) | SRCR_REGACKEN);
Jeff Garzikd61780c02005-10-30 15:01:51 -0800416
Linus Torvalds1da177e2005-04-16 15:20:36 -0700417 /* Setting up prescaler. We need 4 ticks per 1 ms */
418 scaler = SX_OSCFREQ/SPECIALIX_TPS;
419
420 sx_out_off(bp, CD186x_PPRH, scaler >> 8);
421 sx_out_off(bp, CD186x_PPRL, scaler & 0xff);
422 spin_unlock_irqrestore(&bp->lock, flags);
423
424 if (!sx_set_irq (bp)) {
425 /* Figure out how to pass this along... */
426 printk (KERN_ERR "Cannot set irq to %d.\n", bp->irq);
427 rv = 0;
428 }
429
430 func_exit();
431 return rv;
432}
433
434
435static int read_cross_byte (struct specialix_board *bp, int reg, int bit)
436{
437 int i;
438 int t;
439 unsigned long flags;
440
441 spin_lock_irqsave(&bp->lock, flags);
442 for (i=0, t=0;i<8;i++) {
443 sx_out_off (bp, CD186x_CAR, i);
Jeff Garzikd61780c02005-10-30 15:01:51 -0800444 if (sx_in_off (bp, reg) & bit)
Linus Torvalds1da177e2005-04-16 15:20:36 -0700445 t |= 1 << i;
446 }
447 spin_unlock_irqrestore(&bp->lock, flags);
448
449 return t;
450}
451
452
453#ifdef SPECIALIX_TIMER
454void missed_irq (unsigned long data)
455{
456 unsigned char irq;
457 unsigned long flags;
458 struct specialix_board *bp = (struct specialix_board *)data;
459
460 spin_lock_irqsave(&bp->lock, flags);
461 irq = sx_in ((struct specialix_board *)data, CD186x_SRSR) &
462 (SRSR_RREQint |
463 SRSR_TREQint |
464 SRSR_MREQint);
465 spin_unlock_irqrestore(&bp->lock, flags);
466 if (irq) {
467 printk (KERN_INFO "Missed interrupt... Calling int from timer. \n");
Jeff Garzikd61780c02005-10-30 15:01:51 -0800468 sx_interrupt (((struct specialix_board *)data)->irq,
Linus Torvalds1da177e2005-04-16 15:20:36 -0700469 (void*)data, NULL);
470 }
471 missed_irq_timer.expires = jiffies + sx_poll;
472 add_timer (&missed_irq_timer);
473}
474#endif
475
476
477
478/* Main probing routine, also sets irq. */
479static int sx_probe(struct specialix_board *bp)
480{
481 unsigned char val1, val2;
482#if 0
483 int irqs = 0;
484 int retries;
485#endif
486 int rev;
487 int chip;
488
489 func_enter();
490
Jeff Garzikd61780c02005-10-30 15:01:51 -0800491 if (sx_request_io_range(bp)) {
Linus Torvalds1da177e2005-04-16 15:20:36 -0700492 func_exit();
493 return 1;
494 }
495
496 /* Are the I/O ports here ? */
497 sx_out_off(bp, CD186x_PPRL, 0x5a);
498 short_pause ();
499 val1 = sx_in_off(bp, CD186x_PPRL);
500
501 sx_out_off(bp, CD186x_PPRL, 0xa5);
502 short_pause ();
503 val2 = sx_in_off(bp, CD186x_PPRL);
504
Jeff Garzikd61780c02005-10-30 15:01:51 -0800505
Linus Torvalds1da177e2005-04-16 15:20:36 -0700506 if ((val1 != 0x5a) || (val2 != 0xa5)) {
507 printk(KERN_INFO "sx%d: specialix IO8+ Board at 0x%03x not found.\n",
508 board_No(bp), bp->base);
Jeff Garzikd61780c02005-10-30 15:01:51 -0800509 sx_release_io_range(bp);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700510 func_exit();
511 return 1;
512 }
513
Jeff Garzikd61780c02005-10-30 15:01:51 -0800514 /* Check the DSR lines that Specialix uses as board
Linus Torvalds1da177e2005-04-16 15:20:36 -0700515 identification */
516 val1 = read_cross_byte (bp, CD186x_MSVR, MSVR_DSR);
517 val2 = read_cross_byte (bp, CD186x_MSVR, MSVR_RTS);
518 dprintk (SX_DEBUG_INIT, "sx%d: DSR lines are: %02x, rts lines are: %02x\n",
519 board_No(bp), val1, val2);
520
521 /* They managed to switch the bit order between the docs and
522 the IO8+ card. The new PCI card now conforms to old docs.
523 They changed the PCI docs to reflect the situation on the
524 old card. */
525 val2 = (bp->flags & SX_BOARD_IS_PCI)?0x4d : 0xb2;
526 if (val1 != val2) {
527 printk(KERN_INFO "sx%d: specialix IO8+ ID %02x at 0x%03x not found (%02x).\n",
528 board_No(bp), val2, bp->base, val1);
Jeff Garzikd61780c02005-10-30 15:01:51 -0800529 sx_release_io_range(bp);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700530 func_exit();
531 return 1;
532 }
533
534
535#if 0
536 /* It's time to find IRQ for this board */
537 for (retries = 0; retries < 5 && irqs <= 0; retries++) {
538 irqs = probe_irq_on();
539 sx_init_CD186x(bp); /* Reset CD186x chip */
540 sx_out(bp, CD186x_CAR, 2); /* Select port 2 */
541 sx_wait_CCR(bp);
542 sx_out(bp, CD186x_CCR, CCR_TXEN); /* Enable transmitter */
543 sx_out(bp, CD186x_IER, IER_TXRDY); /* Enable tx empty intr */
Jeff Garzikd61780c02005-10-30 15:01:51 -0800544 sx_long_delay(HZ/20);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700545 irqs = probe_irq_off(irqs);
546
547 dprintk (SX_DEBUG_INIT, "SRSR = %02x, ", sx_in(bp, CD186x_SRSR));
548 dprintk (SX_DEBUG_INIT, "TRAR = %02x, ", sx_in(bp, CD186x_TRAR));
549 dprintk (SX_DEBUG_INIT, "GIVR = %02x, ", sx_in(bp, CD186x_GIVR));
550 dprintk (SX_DEBUG_INIT, "GICR = %02x, ", sx_in(bp, CD186x_GICR));
551 dprintk (SX_DEBUG_INIT, "\n");
552
553 /* Reset CD186x again */
554 if (!sx_init_CD186x(bp)) {
555 /* Hmmm. This is dead code anyway. */
556 }
557
558 dprintk (SX_DEBUG_INIT "val1 = %02x, val2 = %02x, val3 = %02x.\n",
Jeff Garzikd61780c02005-10-30 15:01:51 -0800559 val1, val2, val3);
560
Linus Torvalds1da177e2005-04-16 15:20:36 -0700561 }
Jeff Garzikd61780c02005-10-30 15:01:51 -0800562
Linus Torvalds1da177e2005-04-16 15:20:36 -0700563#if 0
564 if (irqs <= 0) {
565 printk(KERN_ERR "sx%d: Can't find IRQ for specialix IO8+ board at 0x%03x.\n",
566 board_No(bp), bp->base);
Jeff Garzikd61780c02005-10-30 15:01:51 -0800567 sx_release_io_range(bp);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700568 func_exit();
569 return 1;
570 }
571#endif
572 printk (KERN_INFO "Started with irq=%d, but now have irq=%d.\n", bp->irq, irqs);
573 if (irqs > 0)
574 bp->irq = irqs;
575#endif
576 /* Reset CD186x again */
577 if (!sx_init_CD186x(bp)) {
Jeff Garzikd61780c02005-10-30 15:01:51 -0800578 sx_release_io_range(bp);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700579 func_exit();
Jeff Garzikd61780c02005-10-30 15:01:51 -0800580 return 1;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700581 }
582
583 sx_request_io_range(bp);
584 bp->flags |= SX_BOARD_PRESENT;
Jeff Garzikd61780c02005-10-30 15:01:51 -0800585
Linus Torvalds1da177e2005-04-16 15:20:36 -0700586 /* Chip revcode pkgtype
587 GFRCR SRCR bit 7
588 CD180 rev B 0x81 0
589 CD180 rev C 0x82 0
590 CD1864 rev A 0x82 1
Jeff Garzikd61780c02005-10-30 15:01:51 -0800591 CD1865 rev A 0x83 1 -- Do not use!!! Does not work.
Linus Torvalds1da177e2005-04-16 15:20:36 -0700592 CD1865 rev B 0x84 1
593 -- Thanks to Gwen Wang, Cirrus Logic.
594 */
595
596 switch (sx_in_off(bp, CD186x_GFRCR)) {
597 case 0x82:chip = 1864;rev='A';break;
598 case 0x83:chip = 1865;rev='A';break;
599 case 0x84:chip = 1865;rev='B';break;
600 case 0x85:chip = 1865;rev='C';break; /* Does not exist at this time */
601 default:chip=-1;rev='x';
602 }
603
604 dprintk (SX_DEBUG_INIT, " GFCR = 0x%02x\n", sx_in_off(bp, CD186x_GFRCR) );
605
606#ifdef SPECIALIX_TIMER
607 init_timer (&missed_irq_timer);
608 missed_irq_timer.function = missed_irq;
609 missed_irq_timer.data = (unsigned long) bp;
610 missed_irq_timer.expires = jiffies + sx_poll;
611 add_timer (&missed_irq_timer);
612#endif
613
614 printk(KERN_INFO"sx%d: specialix IO8+ board detected at 0x%03x, IRQ %d, CD%d Rev. %c.\n",
615 board_No(bp),
616 bp->base, bp->irq,
617 chip, rev);
618
619 func_exit();
620 return 0;
621}
622
Jeff Garzikd61780c02005-10-30 15:01:51 -0800623/*
624 *
Linus Torvalds1da177e2005-04-16 15:20:36 -0700625 * Interrupt processing routines.
626 * */
627
628static inline void sx_mark_event(struct specialix_port * port, int event)
629{
630 func_enter();
631
632 set_bit(event, &port->event);
633 schedule_work(&port->tqueue);
634
635 func_exit();
636}
637
638
639static inline struct specialix_port * sx_get_port(struct specialix_board * bp,
640 unsigned char const * what)
641{
642 unsigned char channel;
643 struct specialix_port * port = NULL;
644
645 channel = sx_in(bp, CD186x_GICR) >> GICR_CHAN_OFF;
646 dprintk (SX_DEBUG_CHAN, "channel: %d\n", channel);
647 if (channel < CD186x_NCH) {
648 port = &sx_port[board_No(bp) * SX_NPORT + channel];
649 dprintk (SX_DEBUG_CHAN, "port: %d %p flags: 0x%x\n",board_No(bp) * SX_NPORT + channel, port, port->flags & ASYNC_INITIALIZED);
650
651 if (port->flags & ASYNC_INITIALIZED) {
652 dprintk (SX_DEBUG_CHAN, "port: %d %p\n", channel, port);
653 func_exit();
654 return port;
655 }
656 }
Jeff Garzikd61780c02005-10-30 15:01:51 -0800657 printk(KERN_INFO "sx%d: %s interrupt from invalid port %d\n",
Linus Torvalds1da177e2005-04-16 15:20:36 -0700658 board_No(bp), what, channel);
659 return NULL;
660}
661
662
663static inline void sx_receive_exc(struct specialix_board * bp)
664{
665 struct specialix_port *port;
666 struct tty_struct *tty;
667 unsigned char status;
668 unsigned char ch;
669
670 func_enter();
671
672 port = sx_get_port(bp, "Receive");
673 if (!port) {
674 dprintk (SX_DEBUG_RX, "Hmm, couldn't find port.\n");
675 func_exit();
676 return;
677 }
678 tty = port->tty;
679 dprintk (SX_DEBUG_RX, "port: %p count: %d BUFF_SIZE: %d\n",
680 port, tty->flip.count, TTY_FLIPBUF_SIZE);
Jeff Garzikd61780c02005-10-30 15:01:51 -0800681
Linus Torvalds1da177e2005-04-16 15:20:36 -0700682 status = sx_in(bp, CD186x_RCSR);
683
684 dprintk (SX_DEBUG_RX, "status: 0x%x\n", status);
685 if (status & RCSR_OE) {
686 port->overrun++;
687 dprintk(SX_DEBUG_FIFO, "sx%d: port %d: Overrun. Total %ld overruns.\n",
688 board_No(bp), port_No(port), port->overrun);
689 }
690 status &= port->mark_mask;
691
692 /* This flip buffer check needs to be below the reading of the
693 status register to reset the chip's IRQ.... */
694 if (tty->flip.count >= TTY_FLIPBUF_SIZE) {
695 dprintk(SX_DEBUG_FIFO, "sx%d: port %d: Working around flip buffer overflow.\n",
696 board_No(bp), port_No(port));
697 func_exit();
698 return;
699 }
700
701 ch = sx_in(bp, CD186x_RDR);
702 if (!status) {
703 func_exit();
704 return;
705 }
706 if (status & RCSR_TOUT) {
Jeff Garzikd61780c02005-10-30 15:01:51 -0800707 printk(KERN_INFO "sx%d: port %d: Receiver timeout. Hardware problems ?\n",
Linus Torvalds1da177e2005-04-16 15:20:36 -0700708 board_No(bp), port_No(port));
709 func_exit();
710 return;
Jeff Garzikd61780c02005-10-30 15:01:51 -0800711
Linus Torvalds1da177e2005-04-16 15:20:36 -0700712 } else if (status & RCSR_BREAK) {
713 dprintk(SX_DEBUG_RX, "sx%d: port %d: Handling break...\n",
714 board_No(bp), port_No(port));
715 *tty->flip.flag_buf_ptr++ = TTY_BREAK;
716 if (port->flags & ASYNC_SAK)
717 do_SAK(tty);
Jeff Garzikd61780c02005-10-30 15:01:51 -0800718
719 } else if (status & RCSR_PE)
Linus Torvalds1da177e2005-04-16 15:20:36 -0700720 *tty->flip.flag_buf_ptr++ = TTY_PARITY;
Jeff Garzikd61780c02005-10-30 15:01:51 -0800721
722 else if (status & RCSR_FE)
Linus Torvalds1da177e2005-04-16 15:20:36 -0700723 *tty->flip.flag_buf_ptr++ = TTY_FRAME;
Jeff Garzikd61780c02005-10-30 15:01:51 -0800724
Linus Torvalds1da177e2005-04-16 15:20:36 -0700725 else if (status & RCSR_OE)
726 *tty->flip.flag_buf_ptr++ = TTY_OVERRUN;
Jeff Garzikd61780c02005-10-30 15:01:51 -0800727
Linus Torvalds1da177e2005-04-16 15:20:36 -0700728 else
729 *tty->flip.flag_buf_ptr++ = 0;
Jeff Garzikd61780c02005-10-30 15:01:51 -0800730
Linus Torvalds1da177e2005-04-16 15:20:36 -0700731 *tty->flip.char_buf_ptr++ = ch;
732 tty->flip.count++;
733 schedule_delayed_work(&tty->flip.work, 1);
734
735 func_exit();
736}
737
738
739static inline void sx_receive(struct specialix_board * bp)
740{
741 struct specialix_port *port;
742 struct tty_struct *tty;
743 unsigned char count;
744
745 func_enter();
Jeff Garzikd61780c02005-10-30 15:01:51 -0800746
Linus Torvalds1da177e2005-04-16 15:20:36 -0700747 if (!(port = sx_get_port(bp, "Receive"))) {
748 dprintk (SX_DEBUG_RX, "Hmm, couldn't find port.\n");
749 func_exit();
750 return;
751 }
752 tty = port->tty;
Jeff Garzikd61780c02005-10-30 15:01:51 -0800753
Linus Torvalds1da177e2005-04-16 15:20:36 -0700754 count = sx_in(bp, CD186x_RDCR);
755 dprintk (SX_DEBUG_RX, "port: %p: count: %d\n", port, count);
756 port->hits[count > 8 ? 9 : count]++;
Jeff Garzikd61780c02005-10-30 15:01:51 -0800757
Linus Torvalds1da177e2005-04-16 15:20:36 -0700758 while (count--) {
759 if (tty->flip.count >= TTY_FLIPBUF_SIZE) {
760 printk(KERN_INFO "sx%d: port %d: Working around flip buffer overflow.\n",
761 board_No(bp), port_No(port));
762 break;
763 }
764 *tty->flip.char_buf_ptr++ = sx_in(bp, CD186x_RDR);
765 *tty->flip.flag_buf_ptr++ = 0;
766 tty->flip.count++;
767 }
768 schedule_delayed_work(&tty->flip.work, 1);
769
770 func_exit();
771}
772
773
774static inline void sx_transmit(struct specialix_board * bp)
775{
776 struct specialix_port *port;
777 struct tty_struct *tty;
778 unsigned char count;
779
780 func_enter();
781 if (!(port = sx_get_port(bp, "Transmit"))) {
782 func_exit();
783 return;
784 }
785 dprintk (SX_DEBUG_TX, "port: %p\n", port);
786 tty = port->tty;
Jeff Garzikd61780c02005-10-30 15:01:51 -0800787
Linus Torvalds1da177e2005-04-16 15:20:36 -0700788 if (port->IER & IER_TXEMPTY) {
789 /* FIFO drained */
790 sx_out(bp, CD186x_CAR, port_No(port));
791 port->IER &= ~IER_TXEMPTY;
792 sx_out(bp, CD186x_IER, port->IER);
793 func_exit();
794 return;
795 }
Jeff Garzikd61780c02005-10-30 15:01:51 -0800796
Linus Torvalds1da177e2005-04-16 15:20:36 -0700797 if ((port->xmit_cnt <= 0 && !port->break_length)
798 || tty->stopped || tty->hw_stopped) {
799 sx_out(bp, CD186x_CAR, port_No(port));
800 port->IER &= ~IER_TXRDY;
801 sx_out(bp, CD186x_IER, port->IER);
802 func_exit();
803 return;
804 }
Jeff Garzikd61780c02005-10-30 15:01:51 -0800805
Linus Torvalds1da177e2005-04-16 15:20:36 -0700806 if (port->break_length) {
807 if (port->break_length > 0) {
808 if (port->COR2 & COR2_ETC) {
809 sx_out(bp, CD186x_TDR, CD186x_C_ESC);
810 sx_out(bp, CD186x_TDR, CD186x_C_SBRK);
811 port->COR2 &= ~COR2_ETC;
812 }
813 count = min_t(int, port->break_length, 0xff);
814 sx_out(bp, CD186x_TDR, CD186x_C_ESC);
815 sx_out(bp, CD186x_TDR, CD186x_C_DELAY);
816 sx_out(bp, CD186x_TDR, count);
817 if (!(port->break_length -= count))
818 port->break_length--;
819 } else {
820 sx_out(bp, CD186x_TDR, CD186x_C_ESC);
821 sx_out(bp, CD186x_TDR, CD186x_C_EBRK);
822 sx_out(bp, CD186x_COR2, port->COR2);
823 sx_wait_CCR(bp);
824 sx_out(bp, CD186x_CCR, CCR_CORCHG2);
825 port->break_length = 0;
826 }
827
828 func_exit();
829 return;
830 }
Jeff Garzikd61780c02005-10-30 15:01:51 -0800831
Linus Torvalds1da177e2005-04-16 15:20:36 -0700832 count = CD186x_NFIFO;
833 do {
834 sx_out(bp, CD186x_TDR, port->xmit_buf[port->xmit_tail++]);
835 port->xmit_tail = port->xmit_tail & (SERIAL_XMIT_SIZE-1);
836 if (--port->xmit_cnt <= 0)
837 break;
838 } while (--count > 0);
Jeff Garzikd61780c02005-10-30 15:01:51 -0800839
Linus Torvalds1da177e2005-04-16 15:20:36 -0700840 if (port->xmit_cnt <= 0) {
841 sx_out(bp, CD186x_CAR, port_No(port));
842 port->IER &= ~IER_TXRDY;
843 sx_out(bp, CD186x_IER, port->IER);
844 }
845 if (port->xmit_cnt <= port->wakeup_chars)
846 sx_mark_event(port, RS_EVENT_WRITE_WAKEUP);
847
848 func_exit();
849}
850
851
852static inline void sx_check_modem(struct specialix_board * bp)
853{
854 struct specialix_port *port;
855 struct tty_struct *tty;
856 unsigned char mcr;
857 int msvr_cd;
858
859 dprintk (SX_DEBUG_SIGNALS, "Modem intr. ");
860 if (!(port = sx_get_port(bp, "Modem")))
861 return;
Jeff Garzikd61780c02005-10-30 15:01:51 -0800862
Linus Torvalds1da177e2005-04-16 15:20:36 -0700863 tty = port->tty;
Jeff Garzikd61780c02005-10-30 15:01:51 -0800864
Linus Torvalds1da177e2005-04-16 15:20:36 -0700865 mcr = sx_in(bp, CD186x_MCR);
866 printk ("mcr = %02x.\n", mcr);
867
868 if ((mcr & MCR_CDCHG)) {
869 dprintk (SX_DEBUG_SIGNALS, "CD just changed... ");
870 msvr_cd = sx_in(bp, CD186x_MSVR) & MSVR_CD;
871 if (msvr_cd) {
872 dprintk (SX_DEBUG_SIGNALS, "Waking up guys in open.\n");
873 wake_up_interruptible(&port->open_wait);
874 } else {
875 dprintk (SX_DEBUG_SIGNALS, "Sending HUP.\n");
876 schedule_work(&port->tqueue_hangup);
877 }
878 }
Jeff Garzikd61780c02005-10-30 15:01:51 -0800879
Linus Torvalds1da177e2005-04-16 15:20:36 -0700880#ifdef SPECIALIX_BRAIN_DAMAGED_CTS
881 if (mcr & MCR_CTSCHG) {
882 if (sx_in(bp, CD186x_MSVR) & MSVR_CTS) {
883 tty->hw_stopped = 0;
884 port->IER |= IER_TXRDY;
885 if (port->xmit_cnt <= port->wakeup_chars)
886 sx_mark_event(port, RS_EVENT_WRITE_WAKEUP);
887 } else {
888 tty->hw_stopped = 1;
889 port->IER &= ~IER_TXRDY;
890 }
891 sx_out(bp, CD186x_IER, port->IER);
892 }
893 if (mcr & MCR_DSSXHG) {
894 if (sx_in(bp, CD186x_MSVR) & MSVR_DSR) {
895 tty->hw_stopped = 0;
896 port->IER |= IER_TXRDY;
897 if (port->xmit_cnt <= port->wakeup_chars)
898 sx_mark_event(port, RS_EVENT_WRITE_WAKEUP);
899 } else {
900 tty->hw_stopped = 1;
901 port->IER &= ~IER_TXRDY;
902 }
903 sx_out(bp, CD186x_IER, port->IER);
904 }
905#endif /* SPECIALIX_BRAIN_DAMAGED_CTS */
Jeff Garzikd61780c02005-10-30 15:01:51 -0800906
Linus Torvalds1da177e2005-04-16 15:20:36 -0700907 /* Clear change bits */
908 sx_out(bp, CD186x_MCR, 0);
909}
910
911
912/* The main interrupt processing routine */
913static irqreturn_t sx_interrupt(int irq, void *dev_id, struct pt_regs *regs)
914{
915 unsigned char status;
916 unsigned char ack;
917 struct specialix_board *bp;
918 unsigned long loop = 0;
919 int saved_reg;
920 unsigned long flags;
921
922 func_enter();
923
924 bp = dev_id;
925 spin_lock_irqsave(&bp->lock, flags);
926
927 dprintk (SX_DEBUG_FLOW, "enter %s port %d room: %ld\n", __FUNCTION__, port_No(sx_get_port(bp, "INT")), SERIAL_XMIT_SIZE - sx_get_port(bp, "ITN")->xmit_cnt - 1);
928 if (!bp || !(bp->flags & SX_BOARD_ACTIVE)) {
929 dprintk (SX_DEBUG_IRQ, "sx: False interrupt. irq %d.\n", irq);
930 spin_unlock_irqrestore(&bp->lock, flags);
931 func_exit();
932 return IRQ_NONE;
933 }
934
935 saved_reg = bp->reg;
936
937 while ((++loop < 16) && (status = (sx_in(bp, CD186x_SRSR) &
938 (SRSR_RREQint |
939 SRSR_TREQint |
Jeff Garzikd61780c02005-10-30 15:01:51 -0800940 SRSR_MREQint)))) {
Linus Torvalds1da177e2005-04-16 15:20:36 -0700941 if (status & SRSR_RREQint) {
942 ack = sx_in(bp, CD186x_RRAR);
943
944 if (ack == (SX_ID | GIVR_IT_RCV))
945 sx_receive(bp);
946 else if (ack == (SX_ID | GIVR_IT_REXC))
947 sx_receive_exc(bp);
948 else
949 printk(KERN_ERR "sx%d: status: 0x%x Bad receive ack 0x%02x.\n",
950 board_No(bp), status, ack);
Jeff Garzikd61780c02005-10-30 15:01:51 -0800951
Linus Torvalds1da177e2005-04-16 15:20:36 -0700952 } else if (status & SRSR_TREQint) {
953 ack = sx_in(bp, CD186x_TRAR);
954
955 if (ack == (SX_ID | GIVR_IT_TX))
956 sx_transmit(bp);
957 else
958 printk(KERN_ERR "sx%d: status: 0x%x Bad transmit ack 0x%02x. port: %d\n",
959 board_No(bp), status, ack, port_No (sx_get_port (bp, "Int")));
960 } else if (status & SRSR_MREQint) {
961 ack = sx_in(bp, CD186x_MRAR);
962
Jeff Garzikd61780c02005-10-30 15:01:51 -0800963 if (ack == (SX_ID | GIVR_IT_MODEM))
Linus Torvalds1da177e2005-04-16 15:20:36 -0700964 sx_check_modem(bp);
965 else
966 printk(KERN_ERR "sx%d: status: 0x%x Bad modem ack 0x%02x.\n",
967 board_No(bp), status, ack);
Jeff Garzikd61780c02005-10-30 15:01:51 -0800968
969 }
Linus Torvalds1da177e2005-04-16 15:20:36 -0700970
971 sx_out(bp, CD186x_EOIR, 0); /* Mark end of interrupt */
972 }
973 bp->reg = saved_reg;
974 outb (bp->reg, bp->base + SX_ADDR_REG);
975 spin_unlock_irqrestore(&bp->lock, flags);
976 func_exit();
977 return IRQ_HANDLED;
978}
979
980
981/*
982 * Routines for open & close processing.
983 */
984
985static void turn_ints_off (struct specialix_board *bp)
986{
987 unsigned long flags;
988
989 func_enter();
990 if (bp->flags & SX_BOARD_IS_PCI) {
991 /* This was intended for enabeling the interrupt on the
992 * PCI card. However it seems that it's already enabled
993 * and as PCI interrupts can be shared, there is no real
994 * reason to have to turn it off. */
995 }
996
997 spin_lock_irqsave(&bp->lock, flags);
998 (void) sx_in_off (bp, 0); /* Turn off interrupts. */
999 spin_unlock_irqrestore(&bp->lock, flags);
1000
1001 func_exit();
1002}
1003
1004static void turn_ints_on (struct specialix_board *bp)
1005{
1006 unsigned long flags;
1007
1008 func_enter();
1009
1010 if (bp->flags & SX_BOARD_IS_PCI) {
1011 /* play with the PCI chip. See comment above. */
1012 }
1013 spin_lock_irqsave(&bp->lock, flags);
1014 (void) sx_in (bp, 0); /* Turn ON interrupts. */
1015 spin_unlock_irqrestore(&bp->lock, flags);
1016
1017 func_exit();
1018}
1019
1020
1021/* Called with disabled interrupts */
1022static inline int sx_setup_board(struct specialix_board * bp)
1023{
1024 int error;
1025
Jeff Garzikd61780c02005-10-30 15:01:51 -08001026 if (bp->flags & SX_BOARD_ACTIVE)
Linus Torvalds1da177e2005-04-16 15:20:36 -07001027 return 0;
1028
1029 if (bp->flags & SX_BOARD_IS_PCI)
1030 error = request_irq(bp->irq, sx_interrupt, SA_INTERRUPT | SA_SHIRQ, "specialix IO8+", bp);
1031 else
1032 error = request_irq(bp->irq, sx_interrupt, SA_INTERRUPT, "specialix IO8+", bp);
1033
Jeff Garzikd61780c02005-10-30 15:01:51 -08001034 if (error)
Linus Torvalds1da177e2005-04-16 15:20:36 -07001035 return error;
1036
1037 turn_ints_on (bp);
1038 bp->flags |= SX_BOARD_ACTIVE;
1039
1040 return 0;
1041}
1042
1043
1044/* Called with disabled interrupts */
1045static inline void sx_shutdown_board(struct specialix_board *bp)
1046{
1047 func_enter();
1048
1049 if (!(bp->flags & SX_BOARD_ACTIVE)) {
1050 func_exit();
1051 return;
1052 }
1053
1054 bp->flags &= ~SX_BOARD_ACTIVE;
Jeff Garzikd61780c02005-10-30 15:01:51 -08001055
Linus Torvalds1da177e2005-04-16 15:20:36 -07001056 dprintk (SX_DEBUG_IRQ, "Freeing IRQ%d for board %d.\n",
1057 bp->irq, board_No (bp));
1058 free_irq(bp->irq, bp);
1059
1060 turn_ints_off (bp);
1061
1062
1063 func_exit();
1064}
1065
1066
1067/*
Jeff Garzikd61780c02005-10-30 15:01:51 -08001068 * Setting up port characteristics.
Linus Torvalds1da177e2005-04-16 15:20:36 -07001069 * Must be called with disabled interrupts
1070 */
1071static void sx_change_speed(struct specialix_board *bp, struct specialix_port *port)
1072{
1073 struct tty_struct *tty;
1074 unsigned long baud;
1075 long tmp;
1076 unsigned char cor1 = 0, cor3 = 0;
1077 unsigned char mcor1 = 0, mcor2 = 0;
1078 static unsigned long again;
1079 unsigned long flags;
1080
1081 func_enter();
1082
1083 if (!(tty = port->tty) || !tty->termios) {
1084 func_exit();
1085 return;
1086 }
1087
1088 port->IER = 0;
1089 port->COR2 = 0;
1090 /* Select port on the board */
1091 spin_lock_irqsave(&bp->lock, flags);
1092 sx_out(bp, CD186x_CAR, port_No(port));
1093
1094 /* The Specialix board doens't implement the RTS lines.
1095 They are used to set the IRQ level. Don't touch them. */
1096 if (SX_CRTSCTS(tty))
1097 port->MSVR = MSVR_DTR | (sx_in(bp, CD186x_MSVR) & MSVR_RTS);
1098 else
1099 port->MSVR = (sx_in(bp, CD186x_MSVR) & MSVR_RTS);
1100 spin_unlock_irqrestore(&bp->lock, flags);
1101 dprintk (SX_DEBUG_TERMIOS, "sx: got MSVR=%02x.\n", port->MSVR);
1102 baud = C_BAUD(tty);
Jeff Garzikd61780c02005-10-30 15:01:51 -08001103
Linus Torvalds1da177e2005-04-16 15:20:36 -07001104 if (baud & CBAUDEX) {
1105 baud &= ~CBAUDEX;
Jeff Garzikd61780c02005-10-30 15:01:51 -08001106 if (baud < 1 || baud > 2)
Linus Torvalds1da177e2005-04-16 15:20:36 -07001107 port->tty->termios->c_cflag &= ~CBAUDEX;
1108 else
1109 baud += 15;
1110 }
1111 if (baud == 15) {
1112 if ((port->flags & ASYNC_SPD_MASK) == ASYNC_SPD_HI)
1113 baud ++;
1114 if ((port->flags & ASYNC_SPD_MASK) == ASYNC_SPD_VHI)
1115 baud += 2;
1116 }
Jeff Garzikd61780c02005-10-30 15:01:51 -08001117
1118
Linus Torvalds1da177e2005-04-16 15:20:36 -07001119 if (!baud_table[baud]) {
1120 /* Drop DTR & exit */
1121 dprintk (SX_DEBUG_TERMIOS, "Dropping DTR... Hmm....\n");
1122 if (!SX_CRTSCTS (tty)) {
1123 port -> MSVR &= ~ MSVR_DTR;
1124 spin_lock_irqsave(&bp->lock, flags);
1125 sx_out(bp, CD186x_MSVR, port->MSVR );
1126 spin_unlock_irqrestore(&bp->lock, flags);
Jeff Garzikd61780c02005-10-30 15:01:51 -08001127 }
Linus Torvalds1da177e2005-04-16 15:20:36 -07001128 else
1129 dprintk (SX_DEBUG_TERMIOS, "Can't drop DTR: no DTR.\n");
1130 return;
1131 } else {
1132 /* Set DTR on */
1133 if (!SX_CRTSCTS (tty)) {
1134 port ->MSVR |= MSVR_DTR;
1135 }
1136 }
Jeff Garzikd61780c02005-10-30 15:01:51 -08001137
Linus Torvalds1da177e2005-04-16 15:20:36 -07001138 /*
Jeff Garzikd61780c02005-10-30 15:01:51 -08001139 * Now we must calculate some speed depended things
Linus Torvalds1da177e2005-04-16 15:20:36 -07001140 */
1141
1142 /* Set baud rate for port */
1143 tmp = port->custom_divisor ;
1144 if ( tmp )
1145 printk (KERN_INFO "sx%d: Using custom baud rate divisor %ld. \n"
1146 "This is an untested option, please be carefull.\n",
1147 port_No (port), tmp);
1148 else
1149 tmp = (((SX_OSCFREQ + baud_table[baud]/2) / baud_table[baud] +
1150 CD186x_TPC/2) / CD186x_TPC);
1151
Jeff Garzikd61780c02005-10-30 15:01:51 -08001152 if ((tmp < 0x10) && time_before(again, jiffies)) {
Linus Torvalds1da177e2005-04-16 15:20:36 -07001153 again = jiffies + HZ * 60;
1154 /* Page 48 of version 2.0 of the CL-CD1865 databook */
1155 if (tmp >= 12) {
1156 printk (KERN_INFO "sx%d: Baud rate divisor is %ld. \n"
1157 "Performance degradation is possible.\n"
1158 "Read specialix.txt for more info.\n",
1159 port_No (port), tmp);
1160 } else {
1161 printk (KERN_INFO "sx%d: Baud rate divisor is %ld. \n"
1162 "Warning: overstressing Cirrus chip. "
1163 "This might not work.\n"
Jeff Garzikd61780c02005-10-30 15:01:51 -08001164 "Read specialix.txt for more info.\n",
Linus Torvalds1da177e2005-04-16 15:20:36 -07001165 port_No (port), tmp);
1166 }
1167 }
1168 spin_lock_irqsave(&bp->lock, flags);
Jeff Garzikd61780c02005-10-30 15:01:51 -08001169 sx_out(bp, CD186x_RBPRH, (tmp >> 8) & 0xff);
1170 sx_out(bp, CD186x_TBPRH, (tmp >> 8) & 0xff);
1171 sx_out(bp, CD186x_RBPRL, tmp & 0xff);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001172 sx_out(bp, CD186x_TBPRL, tmp & 0xff);
1173 spin_unlock_irqrestore(&bp->lock, flags);
1174 if (port->custom_divisor) {
1175 baud = (SX_OSCFREQ + port->custom_divisor/2) / port->custom_divisor;
1176 baud = ( baud + 5 ) / 10;
Jeff Garzikd61780c02005-10-30 15:01:51 -08001177 } else
Linus Torvalds1da177e2005-04-16 15:20:36 -07001178 baud = (baud_table[baud] + 5) / 10; /* Estimated CPS */
1179
1180 /* Two timer ticks seems enough to wakeup something like SLIP driver */
Jeff Garzikd61780c02005-10-30 15:01:51 -08001181 tmp = ((baud + HZ/2) / HZ) * 2 - CD186x_NFIFO;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001182 port->wakeup_chars = (tmp < 0) ? 0 : ((tmp >= SERIAL_XMIT_SIZE) ?
1183 SERIAL_XMIT_SIZE - 1 : tmp);
Jeff Garzikd61780c02005-10-30 15:01:51 -08001184
Linus Torvalds1da177e2005-04-16 15:20:36 -07001185 /* Receiver timeout will be transmission time for 1.5 chars */
1186 tmp = (SPECIALIX_TPS + SPECIALIX_TPS/2 + baud/2) / baud;
1187 tmp = (tmp > 0xff) ? 0xff : tmp;
1188 spin_lock_irqsave(&bp->lock, flags);
1189 sx_out(bp, CD186x_RTPR, tmp);
1190 spin_unlock_irqrestore(&bp->lock, flags);
1191 switch (C_CSIZE(tty)) {
1192 case CS5:
1193 cor1 |= COR1_5BITS;
1194 break;
1195 case CS6:
1196 cor1 |= COR1_6BITS;
1197 break;
1198 case CS7:
1199 cor1 |= COR1_7BITS;
1200 break;
1201 case CS8:
1202 cor1 |= COR1_8BITS;
1203 break;
1204 }
Jeff Garzikd61780c02005-10-30 15:01:51 -08001205
1206 if (C_CSTOPB(tty))
Linus Torvalds1da177e2005-04-16 15:20:36 -07001207 cor1 |= COR1_2SB;
Jeff Garzikd61780c02005-10-30 15:01:51 -08001208
Linus Torvalds1da177e2005-04-16 15:20:36 -07001209 cor1 |= COR1_IGNORE;
1210 if (C_PARENB(tty)) {
1211 cor1 |= COR1_NORMPAR;
Jeff Garzikd61780c02005-10-30 15:01:51 -08001212 if (C_PARODD(tty))
Linus Torvalds1da177e2005-04-16 15:20:36 -07001213 cor1 |= COR1_ODDP;
Jeff Garzikd61780c02005-10-30 15:01:51 -08001214 if (I_INPCK(tty))
Linus Torvalds1da177e2005-04-16 15:20:36 -07001215 cor1 &= ~COR1_IGNORE;
1216 }
1217 /* Set marking of some errors */
1218 port->mark_mask = RCSR_OE | RCSR_TOUT;
Jeff Garzikd61780c02005-10-30 15:01:51 -08001219 if (I_INPCK(tty))
Linus Torvalds1da177e2005-04-16 15:20:36 -07001220 port->mark_mask |= RCSR_FE | RCSR_PE;
Jeff Garzikd61780c02005-10-30 15:01:51 -08001221 if (I_BRKINT(tty) || I_PARMRK(tty))
Linus Torvalds1da177e2005-04-16 15:20:36 -07001222 port->mark_mask |= RCSR_BREAK;
Jeff Garzikd61780c02005-10-30 15:01:51 -08001223 if (I_IGNPAR(tty))
Linus Torvalds1da177e2005-04-16 15:20:36 -07001224 port->mark_mask &= ~(RCSR_FE | RCSR_PE);
1225 if (I_IGNBRK(tty)) {
1226 port->mark_mask &= ~RCSR_BREAK;
Jeff Garzikd61780c02005-10-30 15:01:51 -08001227 if (I_IGNPAR(tty))
Linus Torvalds1da177e2005-04-16 15:20:36 -07001228 /* Real raw mode. Ignore all */
1229 port->mark_mask &= ~RCSR_OE;
1230 }
1231 /* Enable Hardware Flow Control */
1232 if (C_CRTSCTS(tty)) {
1233#ifdef SPECIALIX_BRAIN_DAMAGED_CTS
1234 port->IER |= IER_DSR | IER_CTS;
1235 mcor1 |= MCOR1_DSRZD | MCOR1_CTSZD;
1236 mcor2 |= MCOR2_DSROD | MCOR2_CTSOD;
1237 spin_lock_irqsave(&bp->lock, flags);
1238 tty->hw_stopped = !(sx_in(bp, CD186x_MSVR) & (MSVR_CTS|MSVR_DSR));
1239 spin_unlock_irqrestore(&bp->lock, flags);
1240#else
Jeff Garzikd61780c02005-10-30 15:01:51 -08001241 port->COR2 |= COR2_CTSAE;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001242#endif
1243 }
1244 /* Enable Software Flow Control. FIXME: I'm not sure about this */
1245 /* Some people reported that it works, but I still doubt it */
1246 if (I_IXON(tty)) {
1247 port->COR2 |= COR2_TXIBE;
1248 cor3 |= (COR3_FCT | COR3_SCDE);
1249 if (I_IXANY(tty))
1250 port->COR2 |= COR2_IXM;
1251 spin_lock_irqsave(&bp->lock, flags);
1252 sx_out(bp, CD186x_SCHR1, START_CHAR(tty));
1253 sx_out(bp, CD186x_SCHR2, STOP_CHAR(tty));
1254 sx_out(bp, CD186x_SCHR3, START_CHAR(tty));
1255 sx_out(bp, CD186x_SCHR4, STOP_CHAR(tty));
1256 spin_unlock_irqrestore(&bp->lock, flags);
1257 }
1258 if (!C_CLOCAL(tty)) {
1259 /* Enable CD check */
1260 port->IER |= IER_CD;
1261 mcor1 |= MCOR1_CDZD;
1262 mcor2 |= MCOR2_CDOD;
1263 }
Jeff Garzikd61780c02005-10-30 15:01:51 -08001264
1265 if (C_CREAD(tty))
Linus Torvalds1da177e2005-04-16 15:20:36 -07001266 /* Enable receiver */
1267 port->IER |= IER_RXD;
Jeff Garzikd61780c02005-10-30 15:01:51 -08001268
Linus Torvalds1da177e2005-04-16 15:20:36 -07001269 /* Set input FIFO size (1-8 bytes) */
1270 cor3 |= sx_rxfifo;
1271 /* Setting up CD186x channel registers */
1272 spin_lock_irqsave(&bp->lock, flags);
1273 sx_out(bp, CD186x_COR1, cor1);
1274 sx_out(bp, CD186x_COR2, port->COR2);
1275 sx_out(bp, CD186x_COR3, cor3);
1276 spin_unlock_irqrestore(&bp->lock, flags);
1277 /* Make CD186x know about registers change */
1278 sx_wait_CCR(bp);
1279 spin_lock_irqsave(&bp->lock, flags);
1280 sx_out(bp, CD186x_CCR, CCR_CORCHG1 | CCR_CORCHG2 | CCR_CORCHG3);
1281 /* Setting up modem option registers */
1282 dprintk (SX_DEBUG_TERMIOS, "Mcor1 = %02x, mcor2 = %02x.\n", mcor1, mcor2);
1283 sx_out(bp, CD186x_MCOR1, mcor1);
1284 sx_out(bp, CD186x_MCOR2, mcor2);
1285 spin_unlock_irqrestore(&bp->lock, flags);
1286 /* Enable CD186x transmitter & receiver */
1287 sx_wait_CCR(bp);
1288 spin_lock_irqsave(&bp->lock, flags);
1289 sx_out(bp, CD186x_CCR, CCR_TXEN | CCR_RXEN);
1290 /* Enable interrupts */
1291 sx_out(bp, CD186x_IER, port->IER);
1292 /* And finally set the modem lines... */
1293 sx_out(bp, CD186x_MSVR, port->MSVR);
1294 spin_unlock_irqrestore(&bp->lock, flags);
1295
1296 func_exit();
1297}
1298
1299
1300/* Must be called with interrupts enabled */
1301static int sx_setup_port(struct specialix_board *bp, struct specialix_port *port)
1302{
1303 unsigned long flags;
1304
1305 func_enter();
1306
1307 if (port->flags & ASYNC_INITIALIZED) {
1308 func_exit();
1309 return 0;
1310 }
Jeff Garzikd61780c02005-10-30 15:01:51 -08001311
Linus Torvalds1da177e2005-04-16 15:20:36 -07001312 if (!port->xmit_buf) {
1313 /* We may sleep in get_zeroed_page() */
1314 unsigned long tmp;
Jeff Garzikd61780c02005-10-30 15:01:51 -08001315
Linus Torvalds1da177e2005-04-16 15:20:36 -07001316 if (!(tmp = get_zeroed_page(GFP_KERNEL))) {
1317 func_exit();
1318 return -ENOMEM;
1319 }
1320
1321 if (port->xmit_buf) {
1322 free_page(tmp);
1323 func_exit();
1324 return -ERESTARTSYS;
1325 }
1326 port->xmit_buf = (unsigned char *) tmp;
1327 }
Jeff Garzikd61780c02005-10-30 15:01:51 -08001328
Linus Torvalds1da177e2005-04-16 15:20:36 -07001329 spin_lock_irqsave(&port->lock, flags);
1330
Jeff Garzikd61780c02005-10-30 15:01:51 -08001331 if (port->tty)
Linus Torvalds1da177e2005-04-16 15:20:36 -07001332 clear_bit(TTY_IO_ERROR, &port->tty->flags);
1333
1334 port->xmit_cnt = port->xmit_head = port->xmit_tail = 0;
1335 sx_change_speed(bp, port);
1336 port->flags |= ASYNC_INITIALIZED;
1337
1338 spin_unlock_irqrestore(&port->lock, flags);
1339
Jeff Garzikd61780c02005-10-30 15:01:51 -08001340
Linus Torvalds1da177e2005-04-16 15:20:36 -07001341 func_exit();
1342 return 0;
1343}
1344
1345
1346/* Must be called with interrupts disabled */
1347static void sx_shutdown_port(struct specialix_board *bp, struct specialix_port *port)
1348{
1349 struct tty_struct *tty;
1350 int i;
1351 unsigned long flags;
Jeff Garzikd61780c02005-10-30 15:01:51 -08001352
Linus Torvalds1da177e2005-04-16 15:20:36 -07001353 func_enter();
1354
1355 if (!(port->flags & ASYNC_INITIALIZED)) {
1356 func_exit();
1357 return;
1358 }
Jeff Garzikd61780c02005-10-30 15:01:51 -08001359
Linus Torvalds1da177e2005-04-16 15:20:36 -07001360 if (sx_debug & SX_DEBUG_FIFO) {
1361 dprintk(SX_DEBUG_FIFO, "sx%d: port %d: %ld overruns, FIFO hits [ ",
1362 board_No(bp), port_No(port), port->overrun);
1363 for (i = 0; i < 10; i++) {
1364 dprintk(SX_DEBUG_FIFO, "%ld ", port->hits[i]);
1365 }
1366 dprintk(SX_DEBUG_FIFO, "].\n");
1367 }
1368
1369 if (port->xmit_buf) {
1370 free_page((unsigned long) port->xmit_buf);
1371 port->xmit_buf = NULL;
1372 }
1373
1374 /* Select port */
1375 spin_lock_irqsave(&bp->lock, flags);
1376 sx_out(bp, CD186x_CAR, port_No(port));
1377
1378 if (!(tty = port->tty) || C_HUPCL(tty)) {
1379 /* Drop DTR */
1380 sx_out(bp, CD186x_MSVDTR, 0);
1381 }
1382 spin_unlock_irqrestore(&bp->lock, flags);
1383 /* Reset port */
1384 sx_wait_CCR(bp);
1385 spin_lock_irqsave(&bp->lock, flags);
1386 sx_out(bp, CD186x_CCR, CCR_SOFTRESET);
1387 /* Disable all interrupts from this port */
1388 port->IER = 0;
1389 sx_out(bp, CD186x_IER, port->IER);
1390 spin_unlock_irqrestore(&bp->lock, flags);
1391 if (tty)
1392 set_bit(TTY_IO_ERROR, &tty->flags);
1393 port->flags &= ~ASYNC_INITIALIZED;
Jeff Garzikd61780c02005-10-30 15:01:51 -08001394
1395 if (!bp->count)
Linus Torvalds1da177e2005-04-16 15:20:36 -07001396 sx_shutdown_board(bp);
1397 func_exit();
1398}
1399
Jeff Garzikd61780c02005-10-30 15:01:51 -08001400
Linus Torvalds1da177e2005-04-16 15:20:36 -07001401static int block_til_ready(struct tty_struct *tty, struct file * filp,
1402 struct specialix_port *port)
1403{
1404 DECLARE_WAITQUEUE(wait, current);
1405 struct specialix_board *bp = port_Board(port);
1406 int retval;
1407 int do_clocal = 0;
1408 int CD;
1409 unsigned long flags;
1410
1411 func_enter();
1412
1413 /*
1414 * If the device is in the middle of being closed, then block
1415 * until it's done, and then try again.
1416 */
1417 if (tty_hung_up_p(filp) || port->flags & ASYNC_CLOSING) {
1418 interruptible_sleep_on(&port->close_wait);
1419 if (port->flags & ASYNC_HUP_NOTIFY) {
1420 func_exit();
1421 return -EAGAIN;
1422 } else {
1423 func_exit();
1424 return -ERESTARTSYS;
1425 }
1426 }
Jeff Garzikd61780c02005-10-30 15:01:51 -08001427
Linus Torvalds1da177e2005-04-16 15:20:36 -07001428 /*
1429 * If non-blocking mode is set, or the port is not enabled,
1430 * then make the check up front and then exit.
1431 */
1432 if ((filp->f_flags & O_NONBLOCK) ||
1433 (tty->flags & (1 << TTY_IO_ERROR))) {
1434 port->flags |= ASYNC_NORMAL_ACTIVE;
1435 func_exit();
1436 return 0;
1437 }
1438
1439 if (C_CLOCAL(tty))
1440 do_clocal = 1;
1441
1442 /*
1443 * Block waiting for the carrier detect and the line to become
1444 * free (i.e., not in use by the callout). While we are in
1445 * this loop, info->count is dropped by one, so that
1446 * rs_close() knows when to free things. We restore it upon
1447 * exit, either normal or abnormal.
1448 */
1449 retval = 0;
1450 add_wait_queue(&port->open_wait, &wait);
1451 spin_lock_irqsave(&port->lock, flags);
1452 if (!tty_hung_up_p(filp)) {
1453 port->count--;
1454 }
1455 spin_unlock_irqrestore(&port->lock, flags);
1456 port->blocked_open++;
1457 while (1) {
1458 spin_lock_irqsave(&bp->lock, flags);
1459 sx_out(bp, CD186x_CAR, port_No(port));
1460 CD = sx_in(bp, CD186x_MSVR) & MSVR_CD;
1461 if (SX_CRTSCTS (tty)) {
1462 /* Activate RTS */
1463 port->MSVR |= MSVR_DTR; /* WTF? */
1464 sx_out (bp, CD186x_MSVR, port->MSVR);
1465 } else {
1466 /* Activate DTR */
1467 port->MSVR |= MSVR_DTR;
1468 sx_out (bp, CD186x_MSVR, port->MSVR);
1469 }
1470 spin_unlock_irqrestore(&bp->lock, flags);
1471 set_current_state(TASK_INTERRUPTIBLE);
1472 if (tty_hung_up_p(filp) ||
1473 !(port->flags & ASYNC_INITIALIZED)) {
1474 if (port->flags & ASYNC_HUP_NOTIFY)
1475 retval = -EAGAIN;
1476 else
Jeff Garzikd61780c02005-10-30 15:01:51 -08001477 retval = -ERESTARTSYS;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001478 break;
1479 }
1480 if (!(port->flags & ASYNC_CLOSING) &&
1481 (do_clocal || CD))
1482 break;
1483 if (signal_pending(current)) {
1484 retval = -ERESTARTSYS;
1485 break;
1486 }
1487 schedule();
1488 }
1489
1490 set_current_state(TASK_RUNNING);
1491 remove_wait_queue(&port->open_wait, &wait);
1492 spin_lock_irqsave(&port->lock, flags);
1493 if (!tty_hung_up_p(filp)) {
1494 port->count++;
1495 }
1496 port->blocked_open--;
1497 spin_unlock_irqrestore(&port->lock, flags);
1498 if (retval) {
1499 func_exit();
1500 return retval;
1501 }
1502
1503 port->flags |= ASYNC_NORMAL_ACTIVE;
1504 func_exit();
1505 return 0;
Jeff Garzikd61780c02005-10-30 15:01:51 -08001506}
Linus Torvalds1da177e2005-04-16 15:20:36 -07001507
1508
1509static int sx_open(struct tty_struct * tty, struct file * filp)
1510{
1511 int board;
1512 int error;
1513 struct specialix_port * port;
1514 struct specialix_board * bp;
1515 int i;
1516 unsigned long flags;
1517
1518 func_enter();
1519
1520 board = SX_BOARD(tty->index);
1521
1522 if (board >= SX_NBOARD || !(sx_board[board].flags & SX_BOARD_PRESENT)) {
1523 func_exit();
1524 return -ENODEV;
1525 }
Jeff Garzikd61780c02005-10-30 15:01:51 -08001526
Linus Torvalds1da177e2005-04-16 15:20:36 -07001527 bp = &sx_board[board];
1528 port = sx_port + board * SX_NPORT + SX_PORT(tty->index);
1529 port->overrun = 0;
1530 for (i = 0; i < 10; i++)
1531 port->hits[i]=0;
1532
1533 dprintk (SX_DEBUG_OPEN, "Board = %d, bp = %p, port = %p, portno = %d.\n",
1534 board, bp, port, SX_PORT(tty->index));
1535
1536 if (sx_paranoia_check(port, tty->name, "sx_open")) {
1537 func_enter();
1538 return -ENODEV;
1539 }
1540
1541 if ((error = sx_setup_board(bp))) {
1542 func_exit();
1543 return error;
1544 }
1545
1546 spin_lock_irqsave(&bp->lock, flags);
1547 port->count++;
1548 bp->count++;
1549 tty->driver_data = port;
1550 port->tty = tty;
1551 spin_unlock_irqrestore(&bp->lock, flags);
1552
1553 if ((error = sx_setup_port(bp, port))) {
1554 func_enter();
1555 return error;
1556 }
Jeff Garzikd61780c02005-10-30 15:01:51 -08001557
Linus Torvalds1da177e2005-04-16 15:20:36 -07001558 if ((error = block_til_ready(tty, filp, port))) {
1559 func_enter();
1560 return error;
1561 }
1562
1563 func_exit();
1564 return 0;
1565}
1566
1567
1568static void sx_close(struct tty_struct * tty, struct file * filp)
1569{
1570 struct specialix_port *port = (struct specialix_port *) tty->driver_data;
1571 struct specialix_board *bp;
1572 unsigned long flags;
1573 unsigned long timeout;
Jeff Garzikd61780c02005-10-30 15:01:51 -08001574
Linus Torvalds1da177e2005-04-16 15:20:36 -07001575 func_enter();
1576 if (!port || sx_paranoia_check(port, tty->name, "close")) {
1577 func_exit();
1578 return;
1579 }
1580 spin_lock_irqsave(&port->lock, flags);
1581
1582 if (tty_hung_up_p(filp)) {
1583 spin_unlock_irqrestore(&port->lock, flags);
1584 func_exit();
1585 return;
1586 }
Jeff Garzikd61780c02005-10-30 15:01:51 -08001587
Linus Torvalds1da177e2005-04-16 15:20:36 -07001588 bp = port_Board(port);
1589 if ((tty->count == 1) && (port->count != 1)) {
1590 printk(KERN_ERR "sx%d: sx_close: bad port count;"
1591 " tty->count is 1, port count is %d\n",
1592 board_No(bp), port->count);
1593 port->count = 1;
1594 }
1595
1596 if (port->count > 1) {
1597 port->count--;
1598 bp->count--;
1599
1600 spin_unlock_irqrestore(&port->lock, flags);
1601
1602 func_exit();
1603 return;
1604 }
1605 port->flags |= ASYNC_CLOSING;
1606 /*
Jeff Garzikd61780c02005-10-30 15:01:51 -08001607 * Now we wait for the transmit buffer to clear; and we notify
Linus Torvalds1da177e2005-04-16 15:20:36 -07001608 * the line discipline to only process XON/XOFF characters.
1609 */
1610 tty->closing = 1;
1611 spin_unlock_irqrestore(&port->lock, flags);
1612 dprintk (SX_DEBUG_OPEN, "Closing\n");
1613 if (port->closing_wait != ASYNC_CLOSING_WAIT_NONE) {
1614 tty_wait_until_sent(tty, port->closing_wait);
1615 }
1616 /*
1617 * At this point we stop accepting input. To do this, we
1618 * disable the receive line status interrupts, and tell the
1619 * interrupt driver to stop checking the data ready bit in the
1620 * line status register.
1621 */
1622 dprintk (SX_DEBUG_OPEN, "Closed\n");
1623 port->IER &= ~IER_RXD;
1624 if (port->flags & ASYNC_INITIALIZED) {
1625 port->IER &= ~IER_TXRDY;
1626 port->IER |= IER_TXEMPTY;
1627 spin_lock_irqsave(&bp->lock, flags);
1628 sx_out(bp, CD186x_CAR, port_No(port));
1629 sx_out(bp, CD186x_IER, port->IER);
1630 spin_unlock_irqrestore(&bp->lock, flags);
1631 /*
1632 * Before we drop DTR, make sure the UART transmitter
1633 * has completely drained; this is especially
1634 * important if there is a transmit FIFO!
1635 */
1636 timeout = jiffies+HZ;
1637 while(port->IER & IER_TXEMPTY) {
1638 set_current_state (TASK_INTERRUPTIBLE);
1639 msleep_interruptible(jiffies_to_msecs(port->timeout));
1640 if (time_after(jiffies, timeout)) {
1641 printk (KERN_INFO "Timeout waiting for close\n");
1642 break;
1643 }
1644 }
1645
1646 }
1647
1648 if (--bp->count < 0) {
1649 printk(KERN_ERR "sx%d: sx_shutdown_port: bad board count: %d port: %d\n",
1650 board_No(bp), bp->count, tty->index);
1651 bp->count = 0;
1652 }
1653 if (--port->count < 0) {
1654 printk(KERN_ERR "sx%d: sx_close: bad port count for tty%d: %d\n",
1655 board_No(bp), port_No(port), port->count);
1656 port->count = 0;
1657 }
1658
1659 sx_shutdown_port(bp, port);
1660 if (tty->driver->flush_buffer)
1661 tty->driver->flush_buffer(tty);
1662 tty_ldisc_flush(tty);
1663 spin_lock_irqsave(&port->lock, flags);
1664 tty->closing = 0;
1665 port->event = 0;
1666 port->tty = NULL;
1667 spin_unlock_irqrestore(&port->lock, flags);
1668 if (port->blocked_open) {
1669 if (port->close_delay) {
1670 msleep_interruptible(jiffies_to_msecs(port->close_delay));
1671 }
1672 wake_up_interruptible(&port->open_wait);
1673 }
1674 port->flags &= ~(ASYNC_NORMAL_ACTIVE|ASYNC_CLOSING);
1675 wake_up_interruptible(&port->close_wait);
1676
1677 func_exit();
1678}
1679
1680
Jeff Garzikd61780c02005-10-30 15:01:51 -08001681static int sx_write(struct tty_struct * tty,
Linus Torvalds1da177e2005-04-16 15:20:36 -07001682 const unsigned char *buf, int count)
1683{
1684 struct specialix_port *port = (struct specialix_port *)tty->driver_data;
1685 struct specialix_board *bp;
1686 int c, total = 0;
1687 unsigned long flags;
1688
1689 func_enter();
1690 if (sx_paranoia_check(port, tty->name, "sx_write")) {
1691 func_exit();
1692 return 0;
1693 }
Jeff Garzikd61780c02005-10-30 15:01:51 -08001694
Linus Torvalds1da177e2005-04-16 15:20:36 -07001695 bp = port_Board(port);
1696
1697 if (!tty || !port->xmit_buf || !tmp_buf) {
1698 func_exit();
1699 return 0;
1700 }
1701
1702 while (1) {
1703 spin_lock_irqsave(&port->lock, flags);
1704 c = min_t(int, count, min(SERIAL_XMIT_SIZE - port->xmit_cnt - 1,
1705 SERIAL_XMIT_SIZE - port->xmit_head));
1706 if (c <= 0) {
1707 spin_unlock_irqrestore(&port->lock, flags);
1708 break;
1709 }
1710 memcpy(port->xmit_buf + port->xmit_head, buf, c);
1711 port->xmit_head = (port->xmit_head + c) & (SERIAL_XMIT_SIZE-1);
1712 port->xmit_cnt += c;
1713 spin_unlock_irqrestore(&port->lock, flags);
1714
1715 buf += c;
1716 count -= c;
1717 total += c;
1718 }
1719
1720 spin_lock_irqsave(&bp->lock, flags);
1721 if (port->xmit_cnt && !tty->stopped && !tty->hw_stopped &&
1722 !(port->IER & IER_TXRDY)) {
1723 port->IER |= IER_TXRDY;
1724 sx_out(bp, CD186x_CAR, port_No(port));
1725 sx_out(bp, CD186x_IER, port->IER);
1726 }
1727 spin_unlock_irqrestore(&bp->lock, flags);
1728 func_exit();
1729
1730 return total;
1731}
1732
1733
1734static void sx_put_char(struct tty_struct * tty, unsigned char ch)
1735{
1736 struct specialix_port *port = (struct specialix_port *)tty->driver_data;
1737 unsigned long flags;
1738 struct specialix_board * bp;
1739
1740 func_enter();
1741
1742 if (sx_paranoia_check(port, tty->name, "sx_put_char")) {
1743 func_exit();
1744 return;
1745 }
1746 dprintk (SX_DEBUG_TX, "check tty: %p %p\n", tty, port->xmit_buf);
1747 if (!tty || !port->xmit_buf) {
1748 func_exit();
1749 return;
1750 }
1751 bp = port_Board(port);
1752 spin_lock_irqsave(&port->lock, flags);
1753
1754 dprintk (SX_DEBUG_TX, "xmit_cnt: %d xmit_buf: %p\n", port->xmit_cnt, port->xmit_buf);
1755 if ((port->xmit_cnt >= SERIAL_XMIT_SIZE - 1) || (!port->xmit_buf)) {
1756 spin_unlock_irqrestore(&port->lock, flags);
1757 dprintk (SX_DEBUG_TX, "Exit size\n");
1758 func_exit();
1759 return;
1760 }
1761 dprintk (SX_DEBUG_TX, "Handle xmit: %p %p\n", port, port->xmit_buf);
1762 port->xmit_buf[port->xmit_head++] = ch;
1763 port->xmit_head &= SERIAL_XMIT_SIZE - 1;
1764 port->xmit_cnt++;
1765 spin_unlock_irqrestore(&port->lock, flags);
1766
1767 func_exit();
1768}
1769
1770
1771static void sx_flush_chars(struct tty_struct * tty)
1772{
1773 struct specialix_port *port = (struct specialix_port *)tty->driver_data;
1774 unsigned long flags;
1775 struct specialix_board * bp = port_Board(port);
1776
1777 func_enter();
1778
1779 if (sx_paranoia_check(port, tty->name, "sx_flush_chars")) {
1780 func_exit();
1781 return;
1782 }
1783 if (port->xmit_cnt <= 0 || tty->stopped || tty->hw_stopped ||
1784 !port->xmit_buf) {
1785 func_exit();
1786 return;
1787 }
1788 spin_lock_irqsave(&bp->lock, flags);
1789 port->IER |= IER_TXRDY;
1790 sx_out(port_Board(port), CD186x_CAR, port_No(port));
1791 sx_out(port_Board(port), CD186x_IER, port->IER);
1792 spin_unlock_irqrestore(&bp->lock, flags);
1793
1794 func_exit();
1795}
1796
1797
1798static int sx_write_room(struct tty_struct * tty)
1799{
1800 struct specialix_port *port = (struct specialix_port *)tty->driver_data;
1801 int ret;
1802
1803 func_enter();
1804
1805 if (sx_paranoia_check(port, tty->name, "sx_write_room")) {
1806 func_exit();
1807 return 0;
1808 }
1809
1810 ret = SERIAL_XMIT_SIZE - port->xmit_cnt - 1;
1811 if (ret < 0)
1812 ret = 0;
1813
1814 func_exit();
1815 return ret;
1816}
1817
1818
1819static int sx_chars_in_buffer(struct tty_struct *tty)
1820{
1821 struct specialix_port *port = (struct specialix_port *)tty->driver_data;
1822
1823 func_enter();
Jeff Garzikd61780c02005-10-30 15:01:51 -08001824
Linus Torvalds1da177e2005-04-16 15:20:36 -07001825 if (sx_paranoia_check(port, tty->name, "sx_chars_in_buffer")) {
1826 func_exit();
1827 return 0;
1828 }
1829 func_exit();
1830 return port->xmit_cnt;
1831}
1832
1833
1834static void sx_flush_buffer(struct tty_struct *tty)
1835{
1836 struct specialix_port *port = (struct specialix_port *)tty->driver_data;
1837 unsigned long flags;
1838 struct specialix_board * bp;
1839
1840 func_enter();
1841
1842 if (sx_paranoia_check(port, tty->name, "sx_flush_buffer")) {
1843 func_exit();
1844 return;
1845 }
1846
1847 bp = port_Board(port);
1848 spin_lock_irqsave(&port->lock, flags);
1849 port->xmit_cnt = port->xmit_head = port->xmit_tail = 0;
1850 spin_unlock_irqrestore(&port->lock, flags);
1851 tty_wakeup(tty);
1852
1853 func_exit();
1854}
1855
1856
1857static int sx_tiocmget(struct tty_struct *tty, struct file *file)
1858{
1859 struct specialix_port *port = (struct specialix_port *)tty->driver_data;
1860 struct specialix_board * bp;
1861 unsigned char status;
1862 unsigned int result;
1863 unsigned long flags;
1864
1865 func_enter();
1866
1867 if (sx_paranoia_check(port, tty->name, __FUNCTION__)) {
1868 func_exit();
1869 return -ENODEV;
1870 }
1871
1872 bp = port_Board(port);
1873 spin_lock_irqsave (&bp->lock, flags);
1874 sx_out(bp, CD186x_CAR, port_No(port));
1875 status = sx_in(bp, CD186x_MSVR);
1876 spin_unlock_irqrestore(&bp->lock, flags);
1877 dprintk (SX_DEBUG_INIT, "Got msvr[%d] = %02x, car = %d.\n",
1878 port_No(port), status, sx_in (bp, CD186x_CAR));
1879 dprintk (SX_DEBUG_INIT, "sx_port = %p, port = %p\n", sx_port, port);
1880 if (SX_CRTSCTS(port->tty)) {
Jeff Garzikd61780c02005-10-30 15:01:51 -08001881 result = /* (status & MSVR_RTS) ? */ TIOCM_DTR /* : 0) */
Linus Torvalds1da177e2005-04-16 15:20:36 -07001882 | ((status & MSVR_DTR) ? TIOCM_RTS : 0)
1883 | ((status & MSVR_CD) ? TIOCM_CAR : 0)
1884 |/* ((status & MSVR_DSR) ? */ TIOCM_DSR /* : 0) */
1885 | ((status & MSVR_CTS) ? TIOCM_CTS : 0);
1886 } else {
Jeff Garzikd61780c02005-10-30 15:01:51 -08001887 result = /* (status & MSVR_RTS) ? */ TIOCM_RTS /* : 0) */
Linus Torvalds1da177e2005-04-16 15:20:36 -07001888 | ((status & MSVR_DTR) ? TIOCM_DTR : 0)
1889 | ((status & MSVR_CD) ? TIOCM_CAR : 0)
1890 |/* ((status & MSVR_DSR) ? */ TIOCM_DSR /* : 0) */
1891 | ((status & MSVR_CTS) ? TIOCM_CTS : 0);
1892 }
1893
1894 func_exit();
1895
1896 return result;
1897}
1898
1899
1900static int sx_tiocmset(struct tty_struct *tty, struct file *file,
1901 unsigned int set, unsigned int clear)
1902{
1903 struct specialix_port *port = (struct specialix_port *)tty->driver_data;
1904 unsigned long flags;
1905 struct specialix_board *bp;
1906
1907 func_enter();
1908
1909 if (sx_paranoia_check(port, tty->name, __FUNCTION__)) {
1910 func_exit();
1911 return -ENODEV;
1912 }
1913
1914 bp = port_Board(port);
1915
1916 spin_lock_irqsave(&port->lock, flags);
1917 /* if (set & TIOCM_RTS)
1918 port->MSVR |= MSVR_RTS; */
1919 /* if (set & TIOCM_DTR)
1920 port->MSVR |= MSVR_DTR; */
1921
1922 if (SX_CRTSCTS(port->tty)) {
1923 if (set & TIOCM_RTS)
1924 port->MSVR |= MSVR_DTR;
1925 } else {
1926 if (set & TIOCM_DTR)
1927 port->MSVR |= MSVR_DTR;
1928 }
1929
1930 /* if (clear & TIOCM_RTS)
1931 port->MSVR &= ~MSVR_RTS; */
1932 /* if (clear & TIOCM_DTR)
1933 port->MSVR &= ~MSVR_DTR; */
1934 if (SX_CRTSCTS(port->tty)) {
1935 if (clear & TIOCM_RTS)
1936 port->MSVR &= ~MSVR_DTR;
1937 } else {
1938 if (clear & TIOCM_DTR)
1939 port->MSVR &= ~MSVR_DTR;
1940 }
1941 spin_lock_irqsave(&bp->lock, flags);
1942 sx_out(bp, CD186x_CAR, port_No(port));
1943 sx_out(bp, CD186x_MSVR, port->MSVR);
1944 spin_unlock_irqrestore(&bp->lock, flags);
1945 spin_unlock_irqrestore(&port->lock, flags);
1946 func_exit();
1947 return 0;
1948}
1949
1950
1951static inline void sx_send_break(struct specialix_port * port, unsigned long length)
1952{
1953 struct specialix_board *bp = port_Board(port);
1954 unsigned long flags;
Jeff Garzikd61780c02005-10-30 15:01:51 -08001955
Linus Torvalds1da177e2005-04-16 15:20:36 -07001956 func_enter();
1957
1958 spin_lock_irqsave (&port->lock, flags);
1959 port->break_length = SPECIALIX_TPS / HZ * length;
1960 port->COR2 |= COR2_ETC;
1961 port->IER |= IER_TXRDY;
1962 spin_lock_irqsave(&bp->lock, flags);
1963 sx_out(bp, CD186x_CAR, port_No(port));
1964 sx_out(bp, CD186x_COR2, port->COR2);
1965 sx_out(bp, CD186x_IER, port->IER);
1966 spin_unlock_irqrestore(&bp->lock, flags);
1967 spin_unlock_irqrestore (&port->lock, flags);
1968 sx_wait_CCR(bp);
1969 spin_lock_irqsave(&bp->lock, flags);
1970 sx_out(bp, CD186x_CCR, CCR_CORCHG2);
1971 spin_unlock_irqrestore(&bp->lock, flags);
1972 sx_wait_CCR(bp);
1973
1974 func_exit();
1975}
1976
1977
1978static inline int sx_set_serial_info(struct specialix_port * port,
1979 struct serial_struct __user * newinfo)
1980{
1981 struct serial_struct tmp;
1982 struct specialix_board *bp = port_Board(port);
1983 int change_speed;
1984
1985 func_enter();
1986 /*
Jesper Juhle49332b2005-05-01 08:59:08 -07001987 if (!access_ok(VERIFY_READ, (void *) newinfo, sizeof(tmp))) {
Linus Torvalds1da177e2005-04-16 15:20:36 -07001988 func_exit();
Jesper Juhle49332b2005-05-01 08:59:08 -07001989 return -EFAULT;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001990 }
1991 */
1992 if (copy_from_user(&tmp, newinfo, sizeof(tmp))) {
1993 func_enter();
1994 return -EFAULT;
1995 }
Jeff Garzikd61780c02005-10-30 15:01:51 -08001996
1997#if 0
Linus Torvalds1da177e2005-04-16 15:20:36 -07001998 if ((tmp.irq != bp->irq) ||
1999 (tmp.port != bp->base) ||
2000 (tmp.type != PORT_CIRRUS) ||
2001 (tmp.baud_base != (SX_OSCFREQ + CD186x_TPC/2) / CD186x_TPC) ||
2002 (tmp.custom_divisor != 0) ||
2003 (tmp.xmit_fifo_size != CD186x_NFIFO) ||
2004 (tmp.flags & ~SPECIALIX_LEGAL_FLAGS)) {
2005 func_exit();
2006 return -EINVAL;
2007 }
Jeff Garzikd61780c02005-10-30 15:01:51 -08002008#endif
Linus Torvalds1da177e2005-04-16 15:20:36 -07002009
2010 change_speed = ((port->flags & ASYNC_SPD_MASK) !=
2011 (tmp.flags & ASYNC_SPD_MASK));
2012 change_speed |= (tmp.custom_divisor != port->custom_divisor);
Jeff Garzikd61780c02005-10-30 15:01:51 -08002013
Linus Torvalds1da177e2005-04-16 15:20:36 -07002014 if (!capable(CAP_SYS_ADMIN)) {
2015 if ((tmp.close_delay != port->close_delay) ||
2016 (tmp.closing_wait != port->closing_wait) ||
2017 ((tmp.flags & ~ASYNC_USR_MASK) !=
2018 (port->flags & ~ASYNC_USR_MASK))) {
2019 func_exit();
2020 return -EPERM;
2021 }
2022 port->flags = ((port->flags & ~ASYNC_USR_MASK) |
2023 (tmp.flags & ASYNC_USR_MASK));
2024 port->custom_divisor = tmp.custom_divisor;
2025 } else {
2026 port->flags = ((port->flags & ~ASYNC_FLAGS) |
2027 (tmp.flags & ASYNC_FLAGS));
2028 port->close_delay = tmp.close_delay;
2029 port->closing_wait = tmp.closing_wait;
2030 port->custom_divisor = tmp.custom_divisor;
2031 }
2032 if (change_speed) {
2033 sx_change_speed(bp, port);
2034 }
2035 func_exit();
2036 return 0;
2037}
2038
2039
2040static inline int sx_get_serial_info(struct specialix_port * port,
2041 struct serial_struct __user *retinfo)
2042{
2043 struct serial_struct tmp;
2044 struct specialix_board *bp = port_Board(port);
Jeff Garzikd61780c02005-10-30 15:01:51 -08002045
Linus Torvalds1da177e2005-04-16 15:20:36 -07002046 func_enter();
2047
2048 /*
Jesper Juhle49332b2005-05-01 08:59:08 -07002049 if (!access_ok(VERIFY_WRITE, (void *) retinfo, sizeof(tmp)))
2050 return -EFAULT;
Linus Torvalds1da177e2005-04-16 15:20:36 -07002051 */
2052
2053 memset(&tmp, 0, sizeof(tmp));
2054 tmp.type = PORT_CIRRUS;
2055 tmp.line = port - sx_port;
2056 tmp.port = bp->base;
2057 tmp.irq = bp->irq;
2058 tmp.flags = port->flags;
2059 tmp.baud_base = (SX_OSCFREQ + CD186x_TPC/2) / CD186x_TPC;
2060 tmp.close_delay = port->close_delay * HZ/100;
2061 tmp.closing_wait = port->closing_wait * HZ/100;
2062 tmp.custom_divisor = port->custom_divisor;
2063 tmp.xmit_fifo_size = CD186x_NFIFO;
2064 if (copy_to_user(retinfo, &tmp, sizeof(tmp))) {
2065 func_exit();
2066 return -EFAULT;
2067 }
2068
2069 func_exit();
2070 return 0;
2071}
2072
2073
Jeff Garzikd61780c02005-10-30 15:01:51 -08002074static int sx_ioctl(struct tty_struct * tty, struct file * filp,
Linus Torvalds1da177e2005-04-16 15:20:36 -07002075 unsigned int cmd, unsigned long arg)
2076{
2077 struct specialix_port *port = (struct specialix_port *)tty->driver_data;
2078 int retval;
2079 void __user *argp = (void __user *)arg;
2080
2081 func_enter();
2082
2083 if (sx_paranoia_check(port, tty->name, "sx_ioctl")) {
2084 func_exit();
2085 return -ENODEV;
2086 }
Jeff Garzikd61780c02005-10-30 15:01:51 -08002087
Linus Torvalds1da177e2005-04-16 15:20:36 -07002088 switch (cmd) {
2089 case TCSBRK: /* SVID version: non-zero arg --> no break */
2090 retval = tty_check_change(tty);
2091 if (retval) {
2092 func_exit();
2093 return retval;
2094 }
2095 tty_wait_until_sent(tty, 0);
2096 if (!arg)
2097 sx_send_break(port, HZ/4); /* 1/4 second */
2098 return 0;
2099 case TCSBRKP: /* support for POSIX tcsendbreak() */
2100 retval = tty_check_change(tty);
2101 if (retval) {
2102 func_exit();
2103 return retval;
2104 }
2105 tty_wait_until_sent(tty, 0);
2106 sx_send_break(port, arg ? arg*(HZ/10) : HZ/4);
2107 func_exit();
2108 return 0;
2109 case TIOCGSOFTCAR:
2110 if (put_user(C_CLOCAL(tty)?1:0, (unsigned long __user *)argp)) {
2111 func_exit();
2112 return -EFAULT;
2113 }
2114 func_exit();
2115 return 0;
2116 case TIOCSSOFTCAR:
2117 if (get_user(arg, (unsigned long __user *) argp)) {
2118 func_exit();
2119 return -EFAULT;
2120 }
2121 tty->termios->c_cflag =
2122 ((tty->termios->c_cflag & ~CLOCAL) |
2123 (arg ? CLOCAL : 0));
2124 func_exit();
2125 return 0;
2126 case TIOCGSERIAL:
2127 func_exit();
2128 return sx_get_serial_info(port, argp);
Jeff Garzikd61780c02005-10-30 15:01:51 -08002129 case TIOCSSERIAL:
Linus Torvalds1da177e2005-04-16 15:20:36 -07002130 func_exit();
2131 return sx_set_serial_info(port, argp);
2132 default:
2133 func_exit();
2134 return -ENOIOCTLCMD;
2135 }
2136 func_exit();
2137 return 0;
2138}
2139
2140
2141static void sx_throttle(struct tty_struct * tty)
2142{
2143 struct specialix_port *port = (struct specialix_port *)tty->driver_data;
2144 struct specialix_board *bp;
2145 unsigned long flags;
2146
2147 func_enter();
2148
2149 if (sx_paranoia_check(port, tty->name, "sx_throttle")) {
2150 func_exit();
2151 return;
2152 }
Jeff Garzikd61780c02005-10-30 15:01:51 -08002153
Linus Torvalds1da177e2005-04-16 15:20:36 -07002154 bp = port_Board(port);
Jeff Garzikd61780c02005-10-30 15:01:51 -08002155
Linus Torvalds1da177e2005-04-16 15:20:36 -07002156 /* Use DTR instead of RTS ! */
Jeff Garzikd61780c02005-10-30 15:01:51 -08002157 if (SX_CRTSCTS (tty))
Linus Torvalds1da177e2005-04-16 15:20:36 -07002158 port->MSVR &= ~MSVR_DTR;
2159 else {
2160 /* Auch!!! I think the system shouldn't call this then. */
2161 /* Or maybe we're supposed (allowed?) to do our side of hw
Jeff Garzikd61780c02005-10-30 15:01:51 -08002162 handshake anyway, even when hardware handshake is off.
Linus Torvalds1da177e2005-04-16 15:20:36 -07002163 When you see this in your logs, please report.... */
2164 printk (KERN_ERR "sx%d: Need to throttle, but can't (hardware hs is off)\n",
2165 port_No (port));
2166 }
2167 spin_lock_irqsave(&bp->lock, flags);
2168 sx_out(bp, CD186x_CAR, port_No(port));
2169 spin_unlock_irqrestore(&bp->lock, flags);
2170 if (I_IXOFF(tty)) {
2171 spin_unlock_irqrestore(&bp->lock, flags);
2172 sx_wait_CCR(bp);
2173 spin_lock_irqsave(&bp->lock, flags);
2174 sx_out(bp, CD186x_CCR, CCR_SSCH2);
2175 spin_unlock_irqrestore(&bp->lock, flags);
2176 sx_wait_CCR(bp);
2177 }
2178 spin_lock_irqsave(&bp->lock, flags);
2179 sx_out(bp, CD186x_MSVR, port->MSVR);
2180 spin_unlock_irqrestore(&bp->lock, flags);
2181
2182 func_exit();
2183}
2184
2185
2186static void sx_unthrottle(struct tty_struct * tty)
2187{
2188 struct specialix_port *port = (struct specialix_port *)tty->driver_data;
2189 struct specialix_board *bp;
2190 unsigned long flags;
2191
2192 func_enter();
Jeff Garzikd61780c02005-10-30 15:01:51 -08002193
Linus Torvalds1da177e2005-04-16 15:20:36 -07002194 if (sx_paranoia_check(port, tty->name, "sx_unthrottle")) {
2195 func_exit();
2196 return;
2197 }
Jeff Garzikd61780c02005-10-30 15:01:51 -08002198
Linus Torvalds1da177e2005-04-16 15:20:36 -07002199 bp = port_Board(port);
Jeff Garzikd61780c02005-10-30 15:01:51 -08002200
Linus Torvalds1da177e2005-04-16 15:20:36 -07002201 spin_lock_irqsave(&port->lock, flags);
2202 /* XXXX Use DTR INSTEAD???? */
2203 if (SX_CRTSCTS(tty)) {
2204 port->MSVR |= MSVR_DTR;
2205 } /* Else clause: see remark in "sx_throttle"... */
2206 spin_lock_irqsave(&bp->lock, flags);
2207 sx_out(bp, CD186x_CAR, port_No(port));
2208 spin_unlock_irqrestore(&bp->lock, flags);
2209 if (I_IXOFF(tty)) {
2210 spin_unlock_irqrestore(&port->lock, flags);
2211 sx_wait_CCR(bp);
2212 spin_lock_irqsave(&bp->lock, flags);
2213 sx_out(bp, CD186x_CCR, CCR_SSCH1);
2214 spin_unlock_irqrestore(&bp->lock, flags);
2215 sx_wait_CCR(bp);
2216 spin_lock_irqsave(&port->lock, flags);
2217 }
2218 spin_lock_irqsave(&bp->lock, flags);
2219 sx_out(bp, CD186x_MSVR, port->MSVR);
2220 spin_unlock_irqrestore(&bp->lock, flags);
2221 spin_unlock_irqrestore(&port->lock, flags);
2222
2223 func_exit();
2224}
2225
2226
2227static void sx_stop(struct tty_struct * tty)
2228{
2229 struct specialix_port *port = (struct specialix_port *)tty->driver_data;
2230 struct specialix_board *bp;
2231 unsigned long flags;
2232
2233 func_enter();
Jeff Garzikd61780c02005-10-30 15:01:51 -08002234
Linus Torvalds1da177e2005-04-16 15:20:36 -07002235 if (sx_paranoia_check(port, tty->name, "sx_stop")) {
2236 func_exit();
2237 return;
2238 }
2239
2240 bp = port_Board(port);
Jeff Garzikd61780c02005-10-30 15:01:51 -08002241
Linus Torvalds1da177e2005-04-16 15:20:36 -07002242 spin_lock_irqsave(&port->lock, flags);
2243 port->IER &= ~IER_TXRDY;
2244 spin_lock_irqsave(&bp->lock, flags);
2245 sx_out(bp, CD186x_CAR, port_No(port));
2246 sx_out(bp, CD186x_IER, port->IER);
2247 spin_unlock_irqrestore(&bp->lock, flags);
2248 spin_unlock_irqrestore(&port->lock, flags);
2249
2250 func_exit();
2251}
2252
2253
2254static void sx_start(struct tty_struct * tty)
2255{
2256 struct specialix_port *port = (struct specialix_port *)tty->driver_data;
2257 struct specialix_board *bp;
2258 unsigned long flags;
2259
2260 func_enter();
Jeff Garzikd61780c02005-10-30 15:01:51 -08002261
Linus Torvalds1da177e2005-04-16 15:20:36 -07002262 if (sx_paranoia_check(port, tty->name, "sx_start")) {
2263 func_exit();
2264 return;
2265 }
Jeff Garzikd61780c02005-10-30 15:01:51 -08002266
Linus Torvalds1da177e2005-04-16 15:20:36 -07002267 bp = port_Board(port);
Jeff Garzikd61780c02005-10-30 15:01:51 -08002268
Linus Torvalds1da177e2005-04-16 15:20:36 -07002269 spin_lock_irqsave(&port->lock, flags);
2270 if (port->xmit_cnt && port->xmit_buf && !(port->IER & IER_TXRDY)) {
2271 port->IER |= IER_TXRDY;
2272 spin_lock_irqsave(&bp->lock, flags);
2273 sx_out(bp, CD186x_CAR, port_No(port));
2274 sx_out(bp, CD186x_IER, port->IER);
2275 spin_unlock_irqrestore(&bp->lock, flags);
2276 }
2277 spin_unlock_irqrestore(&port->lock, flags);
2278
2279 func_exit();
2280}
2281
2282
2283/*
2284 * This routine is called from the work-queue when the interrupt
2285 * routine has signalled that a hangup has occurred. The path of
2286 * hangup processing is:
2287 *
2288 * serial interrupt routine -> (workqueue) ->
2289 * do_sx_hangup() -> tty->hangup() -> sx_hangup()
Jeff Garzikd61780c02005-10-30 15:01:51 -08002290 *
Linus Torvalds1da177e2005-04-16 15:20:36 -07002291 */
2292static void do_sx_hangup(void *private_)
2293{
2294 struct specialix_port *port = (struct specialix_port *) private_;
2295 struct tty_struct *tty;
Jeff Garzikd61780c02005-10-30 15:01:51 -08002296
Linus Torvalds1da177e2005-04-16 15:20:36 -07002297 func_enter();
2298
2299 tty = port->tty;
2300 if (tty)
2301 tty_hangup(tty); /* FIXME: module removal race here */
2302
2303 func_exit();
2304}
2305
2306
2307static void sx_hangup(struct tty_struct * tty)
2308{
2309 struct specialix_port *port = (struct specialix_port *)tty->driver_data;
2310 struct specialix_board *bp;
2311 unsigned long flags;
2312
2313 func_enter();
2314
2315 if (sx_paranoia_check(port, tty->name, "sx_hangup")) {
2316 func_exit();
2317 return;
2318 }
Jeff Garzikd61780c02005-10-30 15:01:51 -08002319
Linus Torvalds1da177e2005-04-16 15:20:36 -07002320 bp = port_Board(port);
Jeff Garzikd61780c02005-10-30 15:01:51 -08002321
Linus Torvalds1da177e2005-04-16 15:20:36 -07002322 sx_shutdown_port(bp, port);
2323 spin_lock_irqsave(&port->lock, flags);
2324 port->event = 0;
2325 bp->count -= port->count;
2326 if (bp->count < 0) {
2327 printk(KERN_ERR "sx%d: sx_hangup: bad board count: %d port: %d\n",
2328 board_No(bp), bp->count, tty->index);
2329 bp->count = 0;
2330 }
2331 port->count = 0;
2332 port->flags &= ~ASYNC_NORMAL_ACTIVE;
2333 port->tty = NULL;
2334 spin_unlock_irqrestore(&port->lock, flags);
2335 wake_up_interruptible(&port->open_wait);
2336
2337 func_exit();
2338}
2339
2340
2341static void sx_set_termios(struct tty_struct * tty, struct termios * old_termios)
2342{
2343 struct specialix_port *port = (struct specialix_port *)tty->driver_data;
2344 unsigned long flags;
2345 struct specialix_board * bp;
Jeff Garzikd61780c02005-10-30 15:01:51 -08002346
Linus Torvalds1da177e2005-04-16 15:20:36 -07002347 if (sx_paranoia_check(port, tty->name, "sx_set_termios"))
2348 return;
Jeff Garzikd61780c02005-10-30 15:01:51 -08002349
Linus Torvalds1da177e2005-04-16 15:20:36 -07002350 if (tty->termios->c_cflag == old_termios->c_cflag &&
2351 tty->termios->c_iflag == old_termios->c_iflag)
2352 return;
2353
2354 bp = port_Board(port);
2355 spin_lock_irqsave(&port->lock, flags);
2356 sx_change_speed(port_Board(port), port);
2357 spin_unlock_irqrestore(&port->lock, flags);
2358
2359 if ((old_termios->c_cflag & CRTSCTS) &&
2360 !(tty->termios->c_cflag & CRTSCTS)) {
2361 tty->hw_stopped = 0;
2362 sx_start(tty);
2363 }
2364}
2365
2366
2367static void do_softint(void *private_)
2368{
2369 struct specialix_port *port = (struct specialix_port *) private_;
2370 struct tty_struct *tty;
2371
2372 func_enter();
2373
2374 if(!(tty = port->tty)) {
2375 func_exit();
2376 return;
2377 }
2378
2379 if (test_and_clear_bit(RS_EVENT_WRITE_WAKEUP, &port->event)) {
2380 tty_wakeup(tty);
2381 //wake_up_interruptible(&tty->write_wait);
2382 }
2383
2384 func_exit();
2385}
2386
2387static struct tty_operations sx_ops = {
2388 .open = sx_open,
2389 .close = sx_close,
2390 .write = sx_write,
2391 .put_char = sx_put_char,
2392 .flush_chars = sx_flush_chars,
2393 .write_room = sx_write_room,
2394 .chars_in_buffer = sx_chars_in_buffer,
2395 .flush_buffer = sx_flush_buffer,
2396 .ioctl = sx_ioctl,
2397 .throttle = sx_throttle,
2398 .unthrottle = sx_unthrottle,
2399 .set_termios = sx_set_termios,
2400 .stop = sx_stop,
2401 .start = sx_start,
2402 .hangup = sx_hangup,
2403 .tiocmget = sx_tiocmget,
2404 .tiocmset = sx_tiocmset,
2405};
2406
2407static int sx_init_drivers(void)
2408{
2409 int error;
2410 int i;
2411
2412 func_enter();
2413
2414 specialix_driver = alloc_tty_driver(SX_NBOARD * SX_NPORT);
2415 if (!specialix_driver) {
2416 printk(KERN_ERR "sx: Couldn't allocate tty_driver.\n");
2417 func_exit();
2418 return 1;
2419 }
Jeff Garzikd61780c02005-10-30 15:01:51 -08002420
Linus Torvalds1da177e2005-04-16 15:20:36 -07002421 if (!(tmp_buf = (unsigned char *) get_zeroed_page(GFP_KERNEL))) {
2422 printk(KERN_ERR "sx: Couldn't get free page.\n");
2423 put_tty_driver(specialix_driver);
2424 func_exit();
2425 return 1;
2426 }
2427 specialix_driver->owner = THIS_MODULE;
2428 specialix_driver->name = "ttyW";
2429 specialix_driver->major = SPECIALIX_NORMAL_MAJOR;
2430 specialix_driver->type = TTY_DRIVER_TYPE_SERIAL;
2431 specialix_driver->subtype = SERIAL_TYPE_NORMAL;
2432 specialix_driver->init_termios = tty_std_termios;
2433 specialix_driver->init_termios.c_cflag =
2434 B9600 | CS8 | CREAD | HUPCL | CLOCAL;
2435 specialix_driver->flags = TTY_DRIVER_REAL_RAW;
2436 tty_set_operations(specialix_driver, &sx_ops);
2437
2438 if ((error = tty_register_driver(specialix_driver))) {
2439 put_tty_driver(specialix_driver);
2440 free_page((unsigned long)tmp_buf);
2441 printk(KERN_ERR "sx: Couldn't register specialix IO8+ driver, error = %d\n",
2442 error);
2443 func_exit();
2444 return 1;
2445 }
2446 memset(sx_port, 0, sizeof(sx_port));
2447 for (i = 0; i < SX_NPORT * SX_NBOARD; i++) {
2448 sx_port[i].magic = SPECIALIX_MAGIC;
2449 INIT_WORK(&sx_port[i].tqueue, do_softint, &sx_port[i]);
2450 INIT_WORK(&sx_port[i].tqueue_hangup, do_sx_hangup, &sx_port[i]);
2451 sx_port[i].close_delay = 50 * HZ/100;
2452 sx_port[i].closing_wait = 3000 * HZ/100;
2453 init_waitqueue_head(&sx_port[i].open_wait);
2454 init_waitqueue_head(&sx_port[i].close_wait);
2455 spin_lock_init(&sx_port[i].lock);
2456 }
Jeff Garzikd61780c02005-10-30 15:01:51 -08002457
Linus Torvalds1da177e2005-04-16 15:20:36 -07002458 func_exit();
2459 return 0;
2460}
2461
2462static void sx_release_drivers(void)
2463{
2464 func_enter();
2465
2466 free_page((unsigned long)tmp_buf);
2467 tty_unregister_driver(specialix_driver);
2468 put_tty_driver(specialix_driver);
2469 func_exit();
2470}
2471
Jeff Garzikd61780c02005-10-30 15:01:51 -08002472/*
2473 * This routine must be called by kernel at boot time
Linus Torvalds1da177e2005-04-16 15:20:36 -07002474 */
2475static int __init specialix_init(void)
2476{
2477 int i;
2478 int found = 0;
2479
2480 func_enter();
2481
2482 printk(KERN_INFO "sx: Specialix IO8+ driver v" VERSION ", (c) R.E.Wolff 1997/1998.\n");
2483 printk(KERN_INFO "sx: derived from work (c) D.Gorodchanin 1994-1996.\n");
2484#ifdef CONFIG_SPECIALIX_RTSCTS
2485 printk (KERN_INFO "sx: DTR/RTS pin is always RTS.\n");
2486#else
2487 printk (KERN_INFO "sx: DTR/RTS pin is RTS when CRTSCTS is on.\n");
2488#endif
Jeff Garzikd61780c02005-10-30 15:01:51 -08002489
Linus Torvalds1da177e2005-04-16 15:20:36 -07002490 for (i = 0; i < SX_NBOARD; i++)
2491 sx_board[i].lock = SPIN_LOCK_UNLOCKED;
2492
2493 if (sx_init_drivers()) {
2494 func_exit();
2495 return -EIO;
2496 }
2497
Jeff Garzikd61780c02005-10-30 15:01:51 -08002498 for (i = 0; i < SX_NBOARD; i++)
Linus Torvalds1da177e2005-04-16 15:20:36 -07002499 if (sx_board[i].base && !sx_probe(&sx_board[i]))
2500 found++;
2501
2502#ifdef CONFIG_PCI
2503 {
2504 struct pci_dev *pdev = NULL;
2505
2506 i=0;
2507 while (i < SX_NBOARD) {
2508 if (sx_board[i].flags & SX_BOARD_PRESENT) {
2509 i++;
2510 continue;
2511 }
Jeff Garzikd61780c02005-10-30 15:01:51 -08002512 pdev = pci_find_device (PCI_VENDOR_ID_SPECIALIX,
2513 PCI_DEVICE_ID_SPECIALIX_IO8,
Linus Torvalds1da177e2005-04-16 15:20:36 -07002514 pdev);
2515 if (!pdev) break;
2516
2517 if (pci_enable_device(pdev))
2518 continue;
2519
2520 sx_board[i].irq = pdev->irq;
2521
2522 sx_board[i].base = pci_resource_start (pdev, 2);
2523
2524 sx_board[i].flags |= SX_BOARD_IS_PCI;
2525 if (!sx_probe(&sx_board[i]))
2526 found ++;
2527 }
2528 }
2529#endif
2530
2531 if (!found) {
2532 sx_release_drivers();
2533 printk(KERN_INFO "sx: No specialix IO8+ boards detected.\n");
2534 func_exit();
2535 return -EIO;
2536 }
2537
2538 func_exit();
2539 return 0;
2540}
2541
2542static int iobase[SX_NBOARD] = {0,};
2543
2544static int irq [SX_NBOARD] = {0,};
2545
2546module_param_array(iobase, int, NULL, 0);
2547module_param_array(irq, int, NULL, 0);
2548module_param(sx_debug, int, 0);
2549module_param(sx_rxfifo, int, 0);
2550#ifdef SPECIALIX_TIMER
2551module_param(sx_poll, int, 0);
2552#endif
2553
2554/*
2555 * You can setup up to 4 boards.
2556 * by specifying "iobase=0xXXX,0xXXX ..." as insmod parameter.
Jeff Garzikd61780c02005-10-30 15:01:51 -08002557 * You should specify the IRQs too in that case "irq=....,...".
2558 *
Linus Torvalds1da177e2005-04-16 15:20:36 -07002559 * More than 4 boards in one computer is not possible, as the card can
Jeff Garzikd61780c02005-10-30 15:01:51 -08002560 * only use 4 different interrupts.
Linus Torvalds1da177e2005-04-16 15:20:36 -07002561 *
2562 */
2563static int __init specialix_init_module(void)
2564{
2565 int i;
2566
2567 func_enter();
2568
2569 init_MUTEX(&tmp_buf_sem); /* Init de the semaphore - pvdl */
2570
2571 if (iobase[0] || iobase[1] || iobase[2] || iobase[3]) {
2572 for(i = 0; i < SX_NBOARD; i++) {
2573 sx_board[i].base = iobase[i];
2574 sx_board[i].irq = irq[i];
2575 sx_board[i].count= 0;
2576 }
2577 }
2578
2579 func_exit();
2580
2581 return specialix_init();
2582}
Jeff Garzikd61780c02005-10-30 15:01:51 -08002583
Linus Torvalds1da177e2005-04-16 15:20:36 -07002584static void __exit specialix_exit_module(void)
2585{
2586 int i;
Jeff Garzikd61780c02005-10-30 15:01:51 -08002587
Linus Torvalds1da177e2005-04-16 15:20:36 -07002588 func_enter();
2589
2590 sx_release_drivers();
2591 for (i = 0; i < SX_NBOARD; i++)
Jeff Garzikd61780c02005-10-30 15:01:51 -08002592 if (sx_board[i].flags & SX_BOARD_PRESENT)
Linus Torvalds1da177e2005-04-16 15:20:36 -07002593 sx_release_io_range(&sx_board[i]);
2594#ifdef SPECIALIX_TIMER
2595 del_timer (&missed_irq_timer);
2596#endif
2597
2598 func_exit();
2599}
2600
2601module_init(specialix_init_module);
2602module_exit(specialix_exit_module);
2603
2604MODULE_LICENSE("GPL");