1 /*======================================================================
3 A driver for PCMCIA serial devices
5 serial_cs.c 1.134 2002/05/04 05:48:53
7 The contents of this file are subject to the Mozilla Public
8 License Version 1.1 (the "License"); you may not use this file
9 except in compliance with the License. You may obtain a copy of
10 the License at http://www.mozilla.org/MPL/
12 Software distributed under the License is distributed on an "AS
13 IS" basis, WITHOUT WARRANTY OF ANY KIND, either express or
14 implied. See the License for the specific language governing
15 rights and limitations under the License.
17 The initial developer of the original code is David A. Hinds
18 <dahinds@users.sourceforge.net>. Portions created by David A. Hinds
19 are Copyright (C) 1999 David A. Hinds. All Rights Reserved.
21 Alternatively, the contents of this file may be used under the
22 terms of the GNU General Public License version 2 (the "GPL"), in which
23 case the provisions of the GPL are applicable instead of the
24 above. If you wish to allow the use of your version of this file
25 only under the terms of the GPL and not to allow others to use
26 your version of this file under the MPL, indicate your decision
27 by deleting the provisions above and replace them with the notice
28 and other provisions required by the GPL. If you do not delete
29 the provisions above, a recipient may use your version of this
30 file under either the MPL or the GPL.
32 ======================================================================*/
34 #include <linux/module.h>
35 #include <linux/moduleparam.h>
36 #include <linux/kernel.h>
37 #include <linux/init.h>
38 #include <linux/sched.h>
39 #include <linux/ptrace.h>
40 #include <linux/slab.h>
41 #include <linux/string.h>
42 #include <linux/timer.h>
43 #include <linux/serial_core.h>
44 #include <linux/major.h>
46 #include <asm/system.h>
48 #include <pcmcia/version.h>
49 #include <pcmcia/cs_types.h>
50 #include <pcmcia/cs.h>
51 #include <pcmcia/cistpl.h>
52 #include <pcmcia/ciscode.h>
53 #include <pcmcia/ds.h>
54 #include <pcmcia/cisreg.h>
59 static int pc_debug
= PCMCIA_DEBUG
;
60 module_param(pc_debug
, int, 0644);
61 #define DEBUG(n, args...) if (pc_debug>(n)) printk(KERN_DEBUG args)
62 static char *version
= "serial_cs.c 1.134 2002/05/04 05:48:53 (David Hinds)";
64 #define DEBUG(n, args...)
67 /*====================================================================*/
69 /* Parameters that can be set with 'insmod' */
71 /* Enable the speaker? */
72 static int do_sound
= 1;
73 /* Skip strict UART tests? */
74 static int buggy_uart
;
76 module_param(do_sound
, int, 0444);
77 module_param(buggy_uart
, int, 0444);
79 /*====================================================================*/
81 /* Table of multi-port card ID's */
86 int multi
; /* 1 = multifunction, > 1 = # ports */
89 static struct multi_id multi_id
[] = {
90 { MANFID_OMEGA
, PRODID_OMEGA_QSP_100
, 4 },
91 { MANFID_QUATECH
, PRODID_QUATECH_DUAL_RS232
, 2 },
92 { MANFID_QUATECH
, PRODID_QUATECH_DUAL_RS232_D1
, 2 },
93 { MANFID_QUATECH
, PRODID_QUATECH_QUAD_RS232
, 4 },
94 { MANFID_SOCKET
, PRODID_SOCKET_DUAL_RS232
, 2 },
95 { MANFID_INTEL
, PRODID_INTEL_DUAL_RS232
, 2 },
96 { MANFID_NATINST
, PRODID_NATINST_QUAD_RS232
, 4 }
98 #define MULTI_COUNT (sizeof(multi_id)/sizeof(struct multi_id))
110 static void serial_config(dev_link_t
* link
);
111 static int serial_event(event_t event
, int priority
,
112 event_callback_args_t
* args
);
114 static dev_info_t dev_info
= "serial_cs";
116 static dev_link_t
*serial_attach(void);
117 static void serial_detach(dev_link_t
*);
119 static dev_link_t
*dev_list
= NULL
;
121 /*======================================================================
123 After a card is removed, serial_remove() will unregister
124 the serial device(s), and release the PCMCIA configuration.
126 ======================================================================*/
128 static void serial_remove(dev_link_t
*link
)
130 struct serial_info
*info
= link
->priv
;
133 link
->state
&= ~DEV_PRESENT
;
135 DEBUG(0, "serial_release(0x%p)\n", link
);
138 * Recheck to see if the device is still configured.
140 if (info
->link
.state
& DEV_CONFIG
) {
141 for (i
= 0; i
< info
->ndev
; i
++)
142 serial8250_unregister_port(info
->line
[i
]);
144 info
->link
.dev
= NULL
;
147 pcmcia_release_configuration(info
->link
.handle
);
148 pcmcia_release_io(info
->link
.handle
, &info
->link
.io
);
149 pcmcia_release_irq(info
->link
.handle
, &info
->link
.irq
);
152 info
->link
.state
&= ~DEV_CONFIG
;
156 static void serial_suspend(dev_link_t
*link
)
158 link
->state
|= DEV_SUSPEND
;
160 if (link
->state
& DEV_CONFIG
) {
161 struct serial_info
*info
= link
->priv
;
164 for (i
= 0; i
< info
->ndev
; i
++)
165 serial8250_suspend_port(info
->line
[i
]);
168 pcmcia_release_configuration(link
->handle
);
172 static void serial_resume(dev_link_t
*link
)
174 link
->state
&= ~DEV_SUSPEND
;
177 struct serial_info
*info
= link
->priv
;
181 pcmcia_request_configuration(link
->handle
, &link
->conf
);
183 for (i
= 0; i
< info
->ndev
; i
++)
184 serial8250_resume_port(info
->line
[i
]);
188 /*======================================================================
190 serial_attach() creates an "instance" of the driver, allocating
191 local data structures for one device. The device is registered
194 ======================================================================*/
196 static dev_link_t
*serial_attach(void)
198 struct serial_info
*info
;
199 client_reg_t client_reg
;
203 DEBUG(0, "serial_attach()\n");
205 /* Create new serial device */
206 info
= kmalloc(sizeof (*info
), GFP_KERNEL
);
209 memset(info
, 0, sizeof (*info
));
213 link
->io
.Attributes1
= IO_DATA_PATH_WIDTH_8
;
214 link
->io
.NumPorts1
= 8;
215 link
->irq
.Attributes
= IRQ_TYPE_EXCLUSIVE
;
216 link
->irq
.IRQInfo1
= IRQ_LEVEL_ID
;
217 link
->conf
.Attributes
= CONF_ENABLE_IRQ
;
219 link
->conf
.Attributes
|= CONF_ENABLE_SPKR
;
220 link
->conf
.Status
= CCSR_AUDIO_ENA
;
222 link
->conf
.IntType
= INT_MEMORY_AND_IO
;
224 /* Register with Card Services */
225 link
->next
= dev_list
;
227 client_reg
.dev_info
= &dev_info
;
228 client_reg
.EventMask
=
229 CS_EVENT_CARD_INSERTION
| CS_EVENT_CARD_REMOVAL
|
230 CS_EVENT_RESET_PHYSICAL
| CS_EVENT_CARD_RESET
|
231 CS_EVENT_PM_SUSPEND
| CS_EVENT_PM_RESUME
;
232 client_reg
.event_handler
= &serial_event
;
233 client_reg
.Version
= 0x0210;
234 client_reg
.event_callback_args
.client_data
= link
;
235 ret
= pcmcia_register_client(&link
->handle
, &client_reg
);
236 if (ret
!= CS_SUCCESS
) {
237 cs_error(link
->handle
, RegisterClient
, ret
);
245 /*======================================================================
247 This deletes a driver "instance". The device is de-registered
248 with Card Services. If it has been released, all local data
249 structures are freed. Otherwise, the structures will be freed
250 when the device is released.
252 ======================================================================*/
254 static void serial_detach(dev_link_t
* link
)
256 struct serial_info
*info
= link
->priv
;
260 DEBUG(0, "serial_detach(0x%p)\n", link
);
262 /* Locate device structure */
263 for (linkp
= &dev_list
; *linkp
; linkp
= &(*linkp
)->next
)
270 * Ensure any outstanding scheduled tasks are completed.
272 flush_scheduled_work();
275 * Ensure that the ports have been released.
280 ret
= pcmcia_deregister_client(link
->handle
);
281 if (ret
!= CS_SUCCESS
)
282 cs_error(link
->handle
, DeregisterClient
, ret
);
285 /* Unlink device structure, free bits */
290 /*====================================================================*/
292 static int setup_serial(client_handle_t handle
, struct serial_info
* info
,
293 kio_addr_t iobase
, int irq
)
295 struct uart_port port
;
298 memset(&port
, 0, sizeof (struct uart_port
));
299 port
.iobase
= iobase
;
301 port
.flags
= UPF_BOOT_AUTOCONF
| UPF_SKIP_TEST
| UPF_SHARE_IRQ
;
302 port
.uartclk
= 1843200;
303 port
.dev
= &handle_to_dev(handle
);
305 port
.flags
|= UPF_BUGGY_UART
;
306 line
= serial8250_register_port(&port
);
308 printk(KERN_NOTICE
"serial_cs: serial8250_register_port() at "
309 "0x%04lx, irq %d failed\n", (u_long
)iobase
, irq
);
313 info
->line
[info
->ndev
] = line
;
314 sprintf(info
->node
[info
->ndev
].dev_name
, "ttyS%d", line
);
315 info
->node
[info
->ndev
].major
= TTY_MAJOR
;
316 info
->node
[info
->ndev
].minor
= 0x40 + line
;
318 info
->node
[info
->ndev
- 1].next
= &info
->node
[info
->ndev
];
324 /*====================================================================*/
327 first_tuple(client_handle_t handle
, tuple_t
* tuple
, cisparse_t
* parse
)
330 i
= pcmcia_get_first_tuple(handle
, tuple
);
332 return CS_NO_MORE_ITEMS
;
333 i
= pcmcia_get_tuple_data(handle
, tuple
);
336 return pcmcia_parse_tuple(handle
, tuple
, parse
);
340 next_tuple(client_handle_t handle
, tuple_t
* tuple
, cisparse_t
* parse
)
343 i
= pcmcia_get_next_tuple(handle
, tuple
);
345 return CS_NO_MORE_ITEMS
;
346 i
= pcmcia_get_tuple_data(handle
, tuple
);
349 return pcmcia_parse_tuple(handle
, tuple
, parse
);
352 /*====================================================================*/
354 static int simple_config(dev_link_t
*link
)
356 static kio_addr_t base
[5] = { 0x3f8, 0x2f8, 0x3e8, 0x2e8, 0x0 };
357 static int size_table
[2] = { 8, 16 };
358 client_handle_t handle
= link
->handle
;
359 struct serial_info
*info
= link
->priv
;
363 cistpl_cftable_entry_t
*cf
= &parse
.cftable_entry
;
364 config_info_t config
;
368 /* If the card is already configured, look up the port and irq */
369 i
= pcmcia_get_configuration_info(handle
, &config
);
370 if ((i
== CS_SUCCESS
) && (config
.Attributes
& CONF_VALID_CLIENT
)) {
372 if ((config
.BasePort2
!= 0) && (config
.NumPorts2
== 8)) {
373 port
= config
.BasePort2
;
375 } else if ((info
->manfid
== MANFID_OSITECH
) &&
376 (config
.NumPorts1
== 0x40)) {
377 port
= config
.BasePort1
+ 0x28;
381 return setup_serial(handle
, info
, port
, config
.AssignedIRQ
);
383 link
->conf
.Vcc
= config
.Vcc
;
385 /* First pass: look for a config entry that looks normal. */
386 tuple
.TupleData
= (cisdata_t
*) buf
;
387 tuple
.TupleOffset
= 0;
388 tuple
.TupleDataMax
= 255;
389 tuple
.Attributes
= 0;
390 tuple
.DesiredTuple
= CISTPL_CFTABLE_ENTRY
;
391 /* Two tries: without IO aliases, then with aliases */
392 for (s
= 0; s
< 2; s
++) {
393 for (try = 0; try < 2; try++) {
394 i
= first_tuple(handle
, &tuple
, &parse
);
395 while (i
!= CS_NO_MORE_ITEMS
) {
398 if (cf
->vpp1
.present
& (1 << CISTPL_POWER_VNOM
))
399 link
->conf
.Vpp1
= link
->conf
.Vpp2
=
400 cf
->vpp1
.param
[CISTPL_POWER_VNOM
] / 10000;
401 if ((cf
->io
.nwin
> 0) && (cf
->io
.win
[0].len
== size_table
[s
]) &&
402 (cf
->io
.win
[0].base
!= 0)) {
403 link
->conf
.ConfigIndex
= cf
->index
;
404 link
->io
.BasePort1
= cf
->io
.win
[0].base
;
405 link
->io
.IOAddrLines
= (try == 0) ?
406 16 : cf
->io
.flags
& CISTPL_IO_LINES_MASK
;
407 i
= pcmcia_request_io(link
->handle
, &link
->io
);
412 i
= next_tuple(handle
, &tuple
, &parse
);
416 /* Second pass: try to find an entry that isn't picky about
417 its base address, then try to grab any standard serial port
418 address, and finally try to get any free port. */
419 i
= first_tuple(handle
, &tuple
, &parse
);
420 while (i
!= CS_NO_MORE_ITEMS
) {
421 if ((i
== CS_SUCCESS
) && (cf
->io
.nwin
> 0) &&
422 ((cf
->io
.flags
& CISTPL_IO_LINES_MASK
) <= 3)) {
423 link
->conf
.ConfigIndex
= cf
->index
;
424 for (j
= 0; j
< 5; j
++) {
425 link
->io
.BasePort1
= base
[j
];
426 link
->io
.IOAddrLines
= base
[j
] ? 16 : 3;
427 i
= pcmcia_request_io(link
->handle
, &link
->io
);
432 i
= next_tuple(handle
, &tuple
, &parse
);
436 if (i
!= CS_SUCCESS
) {
438 "serial_cs: no usable port range found, giving up\n");
439 cs_error(link
->handle
, RequestIO
, i
);
443 i
= pcmcia_request_irq(link
->handle
, &link
->irq
);
444 if (i
!= CS_SUCCESS
) {
445 cs_error(link
->handle
, RequestIRQ
, i
);
446 link
->irq
.AssignedIRQ
= 0;
448 if (info
->multi
&& (info
->manfid
== MANFID_3COM
))
449 link
->conf
.ConfigIndex
&= ~(0x08);
450 i
= pcmcia_request_configuration(link
->handle
, &link
->conf
);
451 if (i
!= CS_SUCCESS
) {
452 cs_error(link
->handle
, RequestConfiguration
, i
);
456 return setup_serial(handle
, info
, link
->io
.BasePort1
, link
->irq
.AssignedIRQ
);
459 static int multi_config(dev_link_t
* link
)
461 client_handle_t handle
= link
->handle
;
462 struct serial_info
*info
= link
->priv
;
466 cistpl_cftable_entry_t
*cf
= &parse
.cftable_entry
;
467 config_info_t config
;
470 i
= pcmcia_get_configuration_info(handle
, &config
);
471 if (i
!= CS_SUCCESS
) {
472 cs_error(handle
, GetConfigurationInfo
, i
);
475 link
->conf
.Vcc
= config
.Vcc
;
477 tuple
.TupleData
= (cisdata_t
*) buf
;
478 tuple
.TupleOffset
= 0;
479 tuple
.TupleDataMax
= 255;
480 tuple
.Attributes
= 0;
481 tuple
.DesiredTuple
= CISTPL_CFTABLE_ENTRY
;
483 /* First, look for a generic full-sized window */
484 link
->io
.NumPorts1
= info
->multi
* 8;
485 i
= first_tuple(handle
, &tuple
, &parse
);
486 while (i
!= CS_NO_MORE_ITEMS
) {
487 /* The quad port cards have bad CIS's, so just look for a
488 window larger than 8 ports and assume it will be right */
489 if ((i
== CS_SUCCESS
) && (cf
->io
.nwin
== 1) &&
490 (cf
->io
.win
[0].len
> 8)) {
491 link
->conf
.ConfigIndex
= cf
->index
;
492 link
->io
.BasePort1
= cf
->io
.win
[0].base
;
493 link
->io
.IOAddrLines
=
494 cf
->io
.flags
& CISTPL_IO_LINES_MASK
;
495 i
= pcmcia_request_io(link
->handle
, &link
->io
);
496 base2
= link
->io
.BasePort1
+ 8;
500 i
= next_tuple(handle
, &tuple
, &parse
);
503 /* If that didn't work, look for two windows */
504 if (i
!= CS_SUCCESS
) {
505 link
->io
.NumPorts1
= link
->io
.NumPorts2
= 8;
507 i
= first_tuple(handle
, &tuple
, &parse
);
508 while (i
!= CS_NO_MORE_ITEMS
) {
509 if ((i
== CS_SUCCESS
) && (cf
->io
.nwin
== 2)) {
510 link
->conf
.ConfigIndex
= cf
->index
;
511 link
->io
.BasePort1
= cf
->io
.win
[0].base
;
512 link
->io
.BasePort2
= cf
->io
.win
[1].base
;
513 link
->io
.IOAddrLines
=
514 cf
->io
.flags
& CISTPL_IO_LINES_MASK
;
515 i
= pcmcia_request_io(link
->handle
, &link
->io
);
516 base2
= link
->io
.BasePort2
;
520 i
= next_tuple(handle
, &tuple
, &parse
);
524 if (i
!= CS_SUCCESS
) {
525 cs_error(link
->handle
, RequestIO
, i
);
529 i
= pcmcia_request_irq(link
->handle
, &link
->irq
);
530 if (i
!= CS_SUCCESS
) {
532 "serial_cs: no usable port range found, giving up\n");
533 cs_error(link
->handle
, RequestIRQ
, i
);
534 link
->irq
.AssignedIRQ
= 0;
536 /* Socket Dual IO: this enables irq's for second port */
537 if (info
->multi
&& (info
->manfid
== MANFID_SOCKET
)) {
538 link
->conf
.Present
|= PRESENT_EXT_STATUS
;
539 link
->conf
.ExtStatus
= ESR_REQ_ATTN_ENA
;
541 i
= pcmcia_request_configuration(link
->handle
, &link
->conf
);
542 if (i
!= CS_SUCCESS
) {
543 cs_error(link
->handle
, RequestConfiguration
, i
);
547 /* The Oxford Semiconductor OXCF950 cards are in fact single-port:
548 8 registers are for the UART, the others are extra registers */
549 if (info
->manfid
== MANFID_OXSEMI
) {
550 if (cf
->index
== 1 || cf
->index
== 3) {
551 setup_serial(handle
, info
, base2
, link
->irq
.AssignedIRQ
);
552 outb(12, link
->io
.BasePort1
+ 1);
554 setup_serial(handle
, info
, link
->io
.BasePort1
, link
->irq
.AssignedIRQ
);
560 setup_serial(handle
, info
, link
->io
.BasePort1
, link
->irq
.AssignedIRQ
);
561 /* The Nokia cards are not really multiport cards */
562 if (info
->manfid
== MANFID_NOKIA
)
564 for (i
= 0; i
< info
->multi
- 1; i
++)
565 setup_serial(handle
, info
, base2
+ (8 * i
), link
->irq
.AssignedIRQ
);
570 /*======================================================================
572 serial_config() is scheduled to run after a CARD_INSERTION event
573 is received, to configure the PCMCIA socket, and to make the
574 serial device available to the system.
576 ======================================================================*/
578 void serial_config(dev_link_t
* link
)
580 client_handle_t handle
= link
->handle
;
581 struct serial_info
*info
= link
->priv
;
585 cistpl_cftable_entry_t
*cf
= &parse
.cftable_entry
;
586 int i
, last_ret
, last_fn
;
588 DEBUG(0, "serial_config(0x%p)\n", link
);
590 tuple
.TupleData
= (cisdata_t
*) buf
;
591 tuple
.TupleOffset
= 0;
592 tuple
.TupleDataMax
= 255;
593 tuple
.Attributes
= 0;
594 /* Get configuration register information */
595 tuple
.DesiredTuple
= CISTPL_CONFIG
;
596 last_ret
= first_tuple(handle
, &tuple
, &parse
);
597 if (last_ret
!= CS_SUCCESS
) {
598 last_fn
= ParseTuple
;
601 link
->conf
.ConfigBase
= parse
.config
.base
;
602 link
->conf
.Present
= parse
.config
.rmask
[0];
605 link
->state
|= DEV_CONFIG
;
607 /* Is this a compliant multifunction card? */
608 tuple
.DesiredTuple
= CISTPL_LONGLINK_MFC
;
609 tuple
.Attributes
= TUPLE_RETURN_COMMON
| TUPLE_RETURN_LINK
;
610 info
->multi
= (first_tuple(handle
, &tuple
, &parse
) == CS_SUCCESS
);
612 /* Is this a multiport card? */
613 tuple
.DesiredTuple
= CISTPL_MANFID
;
614 if (first_tuple(handle
, &tuple
, &parse
) == CS_SUCCESS
) {
615 info
->manfid
= le16_to_cpu(buf
[0]);
616 for (i
= 0; i
< MULTI_COUNT
; i
++)
617 if ((info
->manfid
== multi_id
[i
].manfid
) &&
618 (le16_to_cpu(buf
[1]) == multi_id
[i
].prodid
))
621 info
->multi
= multi_id
[i
].multi
;
624 /* Another check for dual-serial cards: look for either serial or
625 multifunction cards that ask for appropriate IO port ranges */
626 tuple
.DesiredTuple
= CISTPL_FUNCID
;
627 if ((info
->multi
== 0) &&
628 ((first_tuple(handle
, &tuple
, &parse
) != CS_SUCCESS
) ||
629 (parse
.funcid
.func
== CISTPL_FUNCID_MULTI
) ||
630 (parse
.funcid
.func
== CISTPL_FUNCID_SERIAL
))) {
631 tuple
.DesiredTuple
= CISTPL_CFTABLE_ENTRY
;
632 if (first_tuple(handle
, &tuple
, &parse
) == CS_SUCCESS
) {
633 if ((cf
->io
.nwin
== 1) && (cf
->io
.win
[0].len
% 8 == 0))
634 info
->multi
= cf
->io
.win
[0].len
>> 3;
635 if ((cf
->io
.nwin
== 2) && (cf
->io
.win
[0].len
== 8) &&
636 (cf
->io
.win
[1].len
== 8))
649 if (info
->manfid
== MANFID_IBM
) {
650 conf_reg_t reg
= { 0, CS_READ
, 0x800, 0 };
651 last_ret
= pcmcia_access_configuration_register(link
->handle
, ®
);
653 last_fn
= AccessConfigurationRegister
;
656 reg
.Action
= CS_WRITE
;
657 reg
.Value
= reg
.Value
| 1;
658 last_ret
= pcmcia_access_configuration_register(link
->handle
, ®
);
660 last_fn
= AccessConfigurationRegister
;
665 link
->dev
= &info
->node
[0];
666 link
->state
&= ~DEV_CONFIG_PENDING
;
670 cs_error(link
->handle
, last_fn
, last_ret
);
673 link
->state
&= ~DEV_CONFIG_PENDING
;
676 /*======================================================================
678 The card status event handler. Mostly, this schedules other
679 stuff to run after an event is received. A CARD_REMOVAL event
680 also sets some flags to discourage the serial drivers from
681 talking to the ports.
683 ======================================================================*/
686 serial_event(event_t event
, int priority
, event_callback_args_t
* args
)
688 dev_link_t
*link
= args
->client_data
;
689 struct serial_info
*info
= link
->priv
;
691 DEBUG(1, "serial_event(0x%06x)\n", event
);
694 case CS_EVENT_CARD_REMOVAL
:
698 case CS_EVENT_CARD_INSERTION
:
699 link
->state
|= DEV_PRESENT
| DEV_CONFIG_PENDING
;
703 case CS_EVENT_PM_SUSPEND
:
704 serial_suspend(link
);
707 case CS_EVENT_RESET_PHYSICAL
:
708 if ((link
->state
& DEV_CONFIG
) && !info
->slave
)
709 pcmcia_release_configuration(link
->handle
);
712 case CS_EVENT_PM_RESUME
:
716 case CS_EVENT_CARD_RESET
:
717 if (DEV_OK(link
) && !info
->slave
)
718 pcmcia_request_configuration(link
->handle
, &link
->conf
);
724 static struct pcmcia_driver serial_cs_driver
= {
725 .owner
= THIS_MODULE
,
729 .attach
= serial_attach
,
730 .detach
= serial_detach
,
733 static int __init
init_serial_cs(void)
735 return pcmcia_register_driver(&serial_cs_driver
);
738 static void __exit
exit_serial_cs(void)
740 pcmcia_unregister_driver(&serial_cs_driver
);
741 BUG_ON(dev_list
!= NULL
);
744 module_init(init_serial_cs
);
745 module_exit(exit_serial_cs
);
747 MODULE_LICENSE("GPL");