1 /*======================================================================
3 A Sedlbauer PCMCIA client driver
5 This driver is for the Sedlbauer Speed Star and Speed Star II,
6 which are ISDN PCMCIA Cards.
8 The contents of this file are subject to the Mozilla Public
9 License Version 1.1 (the "License"); you may not use this file
10 except in compliance with the License. You may obtain a copy of
11 the License at http://www.mozilla.org/MPL/
13 Software distributed under the License is distributed on an "AS
14 IS" basis, WITHOUT WARRANTY OF ANY KIND, either express or
15 implied. See the License for the specific language governing
16 rights and limitations under the License.
18 The initial developer of the original code is David A. Hinds
19 <dahinds@users.sourceforge.net>. Portions created by David A. Hinds
20 are Copyright (C) 1999 David A. Hinds. All Rights Reserved.
22 Modifications from dummy_cs.c are Copyright (C) 1999-2001 Marcus Niemann
23 <maniemann@users.sourceforge.net>. All Rights Reserved.
25 Alternatively, the contents of this file may be used under the
26 terms of the GNU General Public License version 2 (the "GPL"), in
27 which case the provisions of the GPL are applicable instead of the
28 above. If you wish to allow the use of your version of this file
29 only under the terms of the GPL and not to allow others to use
30 your version of this file under the MPL, indicate your decision
31 by deleting the provisions above and replace them with the notice
32 and other provisions required by the GPL. If you do not delete
33 the provisions above, a recipient may use your version of this
34 file under either the MPL or the GPL.
36 ======================================================================*/
38 #include <linux/kernel.h>
39 #include <linux/module.h>
40 #include <linux/init.h>
41 #include <linux/ptrace.h>
42 #include <linux/slab.h>
43 #include <linux/string.h>
44 #include <linux/timer.h>
45 #include <linux/ioport.h>
47 #include <asm/system.h>
49 #include <pcmcia/cs_types.h>
50 #include <pcmcia/cs.h>
51 #include <pcmcia/cistpl.h>
52 #include <pcmcia/cisreg.h>
53 #include <pcmcia/ds.h>
54 #include "hisax_cfg.h"
56 MODULE_DESCRIPTION("ISDN4Linux: PCMCIA client driver for Sedlbauer cards");
57 MODULE_AUTHOR("Marcus Niemann");
58 MODULE_LICENSE("Dual MPL/GPL");
61 /*====================================================================*/
63 /* Parameters that can be set with 'insmod' */
65 static int protocol
= 2; /* EURO-ISDN Default */
66 module_param(protocol
, int, 0);
68 /*====================================================================*/
71 The event() function is this driver's Card Services event handler.
72 It will be called by Card Services when an appropriate card status
73 event is received. The config() and release() entry points are
74 used to configure or release a socket, in response to card
75 insertion and ejection events. They are invoked from the sedlbauer
79 static int sedlbauer_config(struct pcmcia_device
*link
);
80 static void sedlbauer_release(struct pcmcia_device
*link
);
83 The attach() and detach() entry points are used to create and destroy
84 "instances" of the driver, where each instance represents everything
85 needed to manage one actual PCMCIA card.
88 static void sedlbauer_detach(struct pcmcia_device
*p_dev
);
91 You'll also need to prototype all the functions that will actually
92 be used to talk to your device. See 'memory_cs' for a good example
93 of a fully self-sufficient driver; the other drivers rely more or
94 less on other parts of the kernel.
98 A driver needs to provide a dev_node_t structure for each device
99 on a card. In some cases, there is only one device per card (for
100 example, ethernet cards, modems). In other cases, there may be
101 many actual or logical devices (SCSI adapters, memory cards with
102 multiple partitions). The dev_node_t structures need to be kept
103 in a linked list starting at the 'dev' field of a struct pcmcia_device
104 structure. We allocate them in the card's private data structure,
105 because they generally shouldn't be allocated dynamically.
107 In this case, we also provide a flag to indicate if a device is
108 "stopped" due to a power management event, or card ejection. The
109 device IO routines can use a flag like this to throttle IO to a
110 card that is not ready to accept it.
113 typedef struct local_info_t
{
114 struct pcmcia_device
*p_dev
;
120 /*======================================================================
122 sedlbauer_attach() creates an "instance" of the driver, allocating
123 local data structures for one device. The device is registered
126 The dev_link structure is initialized, but we don't actually
127 configure the card at this point -- we wait until we receive a
128 card insertion event.
130 ======================================================================*/
132 static int sedlbauer_probe(struct pcmcia_device
*link
)
136 dev_dbg(&link
->dev
, "sedlbauer_attach()\n");
138 /* Allocate space for private device-specific data */
139 local
= kzalloc(sizeof(local_info_t
), GFP_KERNEL
);
140 if (!local
) return -ENOMEM
;
146 /* Interrupt setup */
147 link
->irq
.Attributes
= IRQ_TYPE_DYNAMIC_SHARING
;
148 link
->irq
.Handler
= NULL
;
151 General socket configuration defaults can go here. In this
152 client, we assume very little, and rely on the CIS for almost
153 everything. In most clients, many details (i.e., number, sizes,
154 and attributes of IO windows) are fixed by the nature of the
155 device, and can be hard-wired here.
160 /* The io structure describes IO port mapping */
161 link
->io
.NumPorts1
= 8;
162 link
->io
.Attributes1
= IO_DATA_PATH_WIDTH_8
;
163 link
->io
.IOAddrLines
= 3;
165 link
->conf
.Attributes
= 0;
166 link
->conf
.IntType
= INT_MEMORY_AND_IO
;
168 return sedlbauer_config(link
);
169 } /* sedlbauer_attach */
171 /*======================================================================
173 This deletes a driver "instance". The device is de-registered
174 with Card Services. If it has been released, all local data
175 structures are freed. Otherwise, the structures will be freed
176 when the device is released.
178 ======================================================================*/
180 static void sedlbauer_detach(struct pcmcia_device
*link
)
182 dev_dbg(&link
->dev
, "sedlbauer_detach(0x%p)\n", link
);
184 ((local_info_t
*)link
->priv
)->stop
= 1;
185 sedlbauer_release(link
);
187 /* This points to the parent local_info_t struct */
189 } /* sedlbauer_detach */
191 /*======================================================================
193 sedlbauer_config() is scheduled to run after a CARD_INSERTION event
194 is received, to configure the PCMCIA socket, and to make the
195 device available to the system.
197 ======================================================================*/
198 static int sedlbauer_config_check(struct pcmcia_device
*p_dev
,
199 cistpl_cftable_entry_t
*cfg
,
200 cistpl_cftable_entry_t
*dflt
,
204 win_req_t
*req
= priv_data
;
209 /* Does this card need audio output? */
210 if (cfg
->flags
& CISTPL_CFTABLE_AUDIO
) {
211 p_dev
->conf
.Attributes
|= CONF_ENABLE_SPKR
;
212 p_dev
->conf
.Status
= CCSR_AUDIO_ENA
;
215 /* Use power settings for Vcc and Vpp if present */
216 /* Note that the CIS values need to be rescaled */
217 if (cfg
->vcc
.present
& (1<<CISTPL_POWER_VNOM
)) {
218 if (vcc
!= cfg
->vcc
.param
[CISTPL_POWER_VNOM
]/10000)
220 } else if (dflt
->vcc
.present
& (1<<CISTPL_POWER_VNOM
)) {
221 if (vcc
!= dflt
->vcc
.param
[CISTPL_POWER_VNOM
]/10000)
225 if (cfg
->vpp1
.present
& (1<<CISTPL_POWER_VNOM
))
226 p_dev
->conf
.Vpp
= cfg
->vpp1
.param
[CISTPL_POWER_VNOM
]/10000;
227 else if (dflt
->vpp1
.present
& (1<<CISTPL_POWER_VNOM
))
228 p_dev
->conf
.Vpp
= dflt
->vpp1
.param
[CISTPL_POWER_VNOM
]/10000;
230 /* Do we need to allocate an interrupt? */
231 if (cfg
->irq
.IRQInfo1
|| dflt
->irq
.IRQInfo1
)
232 p_dev
->conf
.Attributes
|= CONF_ENABLE_IRQ
;
234 /* IO window settings */
235 p_dev
->io
.NumPorts1
= p_dev
->io
.NumPorts2
= 0;
236 if ((cfg
->io
.nwin
> 0) || (dflt
->io
.nwin
> 0)) {
237 cistpl_io_t
*io
= (cfg
->io
.nwin
) ? &cfg
->io
: &dflt
->io
;
238 p_dev
->io
.Attributes1
= IO_DATA_PATH_WIDTH_AUTO
;
239 if (!(io
->flags
& CISTPL_IO_8BIT
))
240 p_dev
->io
.Attributes1
= IO_DATA_PATH_WIDTH_16
;
241 if (!(io
->flags
& CISTPL_IO_16BIT
))
242 p_dev
->io
.Attributes1
= IO_DATA_PATH_WIDTH_8
;
243 p_dev
->io
.BasePort1
= io
->win
[0].base
;
244 p_dev
->io
.NumPorts1
= io
->win
[0].len
;
246 p_dev
->io
.Attributes2
= p_dev
->io
.Attributes1
;
247 p_dev
->io
.BasePort2
= io
->win
[1].base
;
248 p_dev
->io
.NumPorts2
= io
->win
[1].len
;
250 /* This reserves IO space but doesn't actually enable it */
251 if (pcmcia_request_io(p_dev
, &p_dev
->io
) != 0)
256 Now set up a common memory window, if needed. There is room
257 in the struct pcmcia_device structure for one memory window handle,
258 but if the base addresses need to be saved, or if multiple
259 windows are needed, the info should go in the private data
260 structure for this device.
262 Note that the memory window base is a physical address, and
263 needs to be mapped to virtual space with ioremap() before it
266 if ((cfg
->mem
.nwin
> 0) || (dflt
->mem
.nwin
> 0)) {
267 cistpl_mem_t
*mem
= (cfg
->mem
.nwin
) ? &cfg
->mem
: &dflt
->mem
;
269 req
->Attributes
= WIN_DATA_WIDTH_16
|WIN_MEMORY_TYPE_CM
;
270 req
->Attributes
|= WIN_ENABLE
;
271 req
->Base
= mem
->win
[0].host_addr
;
272 req
->Size
= mem
->win
[0].len
;
273 req
->AccessSpeed
= 0;
274 if (pcmcia_request_window(p_dev
, req
, &p_dev
->win
) != 0)
277 map
.CardOffset
= mem
->win
[0].card_addr
;
278 if (pcmcia_map_mem_page(p_dev
, p_dev
->win
, &map
) != 0)
286 static int sedlbauer_config(struct pcmcia_device
*link
)
288 local_info_t
*dev
= link
->priv
;
293 dev_dbg(&link
->dev
, "sedlbauer_config(0x%p)\n", link
);
295 req
= kzalloc(sizeof(win_req_t
), GFP_KERNEL
);
300 In this loop, we scan the CIS for configuration table entries,
301 each of which describes a valid card configuration, including
302 voltage, IO window, memory window, and interrupt settings.
304 We make no assumptions about the card to be configured: we use
305 just the information available in the CIS. In an ideal world,
306 this would work for any PCMCIA card, but it requires a complete
307 and accurate CIS. In practice, a driver usually "knows" most of
308 these things without consulting the CIS, and most client drivers
309 will only use the CIS to fill in implementation-defined details.
311 ret
= pcmcia_loop_config(link
, sedlbauer_config_check
, req
);
316 Allocate an interrupt line. Note that this does not assign a
317 handler to the interrupt, unless the 'Handler' member of the
318 irq structure is initialized.
320 if (link
->conf
.Attributes
& CONF_ENABLE_IRQ
) {
321 ret
= pcmcia_request_irq(link
, &link
->irq
);
327 This actually configures the PCMCIA socket -- setting up
328 the I/O windows and the interrupt mapping, and putting the
329 card and host interface into "Memory and IO" mode.
331 ret
= pcmcia_request_configuration(link
, &link
->conf
);
336 At this point, the dev_node_t structure(s) need to be
337 initialized and arranged in a linked list at link->dev.
339 sprintf(dev
->node
.dev_name
, "sedlbauer");
340 dev
->node
.major
= dev
->node
.minor
= 0;
341 link
->dev_node
= &dev
->node
;
343 /* Finally, report what we've done */
344 printk(KERN_INFO
"%s: index 0x%02x:",
345 dev
->node
.dev_name
, link
->conf
.ConfigIndex
);
347 printk(", Vpp %d.%d", link
->conf
.Vpp
/10, link
->conf
.Vpp
%10);
348 if (link
->conf
.Attributes
& CONF_ENABLE_IRQ
)
349 printk(", irq %d", link
->irq
.AssignedIRQ
);
350 if (link
->io
.NumPorts1
)
351 printk(", io 0x%04x-0x%04x", link
->io
.BasePort1
,
352 link
->io
.BasePort1
+link
->io
.NumPorts1
-1);
353 if (link
->io
.NumPorts2
)
354 printk(" & 0x%04x-0x%04x", link
->io
.BasePort2
,
355 link
->io
.BasePort2
+link
->io
.NumPorts2
-1);
357 printk(", mem 0x%06lx-0x%06lx", req
->Base
,
358 req
->Base
+req
->Size
-1);
361 icard
.para
[0] = link
->irq
.AssignedIRQ
;
362 icard
.para
[1] = link
->io
.BasePort1
;
363 icard
.protocol
= protocol
;
364 icard
.typ
= ISDN_CTYPE_SEDLBAUER_PCMCIA
;
366 ret
= hisax_init_pcmcia(link
,
367 &(((local_info_t
*)link
->priv
)->stop
), &icard
);
369 printk(KERN_ERR
"sedlbauer_cs: failed to initialize SEDLBAUER PCMCIA %d at i/o %#x\n",
370 ret
, link
->io
.BasePort1
);
371 sedlbauer_release(link
);
374 ((local_info_t
*)link
->priv
)->cardnr
= ret
;
379 sedlbauer_release(link
);
382 } /* sedlbauer_config */
384 /*======================================================================
386 After a card is removed, sedlbauer_release() will unregister the
387 device, and release the PCMCIA configuration. If the device is
388 still open, this will be postponed until it is closed.
390 ======================================================================*/
392 static void sedlbauer_release(struct pcmcia_device
*link
)
394 local_info_t
*local
= link
->priv
;
395 dev_dbg(&link
->dev
, "sedlbauer_release(0x%p)\n", link
);
398 if (local
->cardnr
>= 0) {
399 /* no unregister function with hisax */
400 HiSax_closecard(local
->cardnr
);
404 pcmcia_disable_device(link
);
405 } /* sedlbauer_release */
407 static int sedlbauer_suspend(struct pcmcia_device
*link
)
409 local_info_t
*dev
= link
->priv
;
416 static int sedlbauer_resume(struct pcmcia_device
*link
)
418 local_info_t
*dev
= link
->priv
;
426 static struct pcmcia_device_id sedlbauer_ids
[] = {
427 PCMCIA_DEVICE_PROD_ID123("SEDLBAUER", "speed star II", "V 3.1", 0x81fb79f5, 0xf3612e1d, 0x6b95c78a),
428 PCMCIA_DEVICE_PROD_ID123("SEDLBAUER", "ISDN-Adapter", "4D67", 0x81fb79f5, 0xe4e9bc12, 0x397b7e90),
429 PCMCIA_DEVICE_PROD_ID123("SEDLBAUER", "ISDN-Adapter", "4D98", 0x81fb79f5, 0xe4e9bc12, 0x2e5c7fce),
430 PCMCIA_DEVICE_PROD_ID123("SEDLBAUER", "ISDN-Adapter", " (C) 93-94 VK", 0x81fb79f5, 0xe4e9bc12, 0x8db143fe),
431 PCMCIA_DEVICE_PROD_ID123("SEDLBAUER", "ISDN-Adapter", " (c) 93-95 VK", 0x81fb79f5, 0xe4e9bc12, 0xb391ab4c),
432 PCMCIA_DEVICE_PROD_ID12("HST High Soft Tech GmbH", "Saphir II B", 0xd79e0b84, 0x21d083ae),
433 /* PCMCIA_DEVICE_PROD_ID1234("SEDLBAUER", 0x81fb79f5), */ /* too generic*/
436 MODULE_DEVICE_TABLE(pcmcia
, sedlbauer_ids
);
438 static struct pcmcia_driver sedlbauer_driver
= {
439 .owner
= THIS_MODULE
,
441 .name
= "sedlbauer_cs",
443 .probe
= sedlbauer_probe
,
444 .remove
= sedlbauer_detach
,
445 .id_table
= sedlbauer_ids
,
446 .suspend
= sedlbauer_suspend
,
447 .resume
= sedlbauer_resume
,
450 static int __init
init_sedlbauer_cs(void)
452 return pcmcia_register_driver(&sedlbauer_driver
);
455 static void __exit
exit_sedlbauer_cs(void)
457 pcmcia_unregister_driver(&sedlbauer_driver
);
460 module_init(init_sedlbauer_cs
);
461 module_exit(exit_sedlbauer_cs
);