sedlbauer_cs.c 15.2 KB
Newer Older
L
Linus Torvalds 已提交
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78
/*======================================================================

    A Sedlbauer PCMCIA client driver

    This driver is for the Sedlbauer Speed Star and Speed Star II, 
    which are ISDN PCMCIA Cards.
    
    The contents of this file are subject to the Mozilla Public
    License Version 1.1 (the "License"); you may not use this file
    except in compliance with the License. You may obtain a copy of
    the License at http://www.mozilla.org/MPL/

    Software distributed under the License is distributed on an "AS
    IS" basis, WITHOUT WARRANTY OF ANY KIND, either express or
    implied. See the License for the specific language governing
    rights and limitations under the License.

    The initial developer of the original code is David A. Hinds
    <dahinds@users.sourceforge.net>.  Portions created by David A. Hinds
    are Copyright (C) 1999 David A. Hinds.  All Rights Reserved.

    Modifications from dummy_cs.c are Copyright (C) 1999-2001 Marcus Niemann
    <maniemann@users.sourceforge.net>. All Rights Reserved.

    Alternatively, the contents of this file may be used under the
    terms of the GNU General Public License version 2 (the "GPL"), in
    which case the provisions of the GPL are applicable instead of the
    above.  If you wish to allow the use of your version of this file
    only under the terms of the GPL and not to allow others to use
    your version of this file under the MPL, indicate your decision
    by deleting the provisions above and replace them with the notice
    and other provisions required by the GPL.  If you do not delete
    the provisions above, a recipient may use your version of this
    file under either the MPL or the GPL.
    
======================================================================*/

#include <linux/kernel.h>
#include <linux/module.h>
#include <linux/init.h>
#include <linux/ptrace.h>
#include <linux/slab.h>
#include <linux/string.h>
#include <linux/timer.h>
#include <linux/ioport.h>
#include <asm/io.h>
#include <asm/system.h>

#include <pcmcia/cs_types.h>
#include <pcmcia/cs.h>
#include <pcmcia/cistpl.h>
#include <pcmcia/cisreg.h>
#include <pcmcia/ds.h>
#include "hisax_cfg.h"

MODULE_DESCRIPTION("ISDN4Linux: PCMCIA client driver for Sedlbauer cards");
MODULE_AUTHOR("Marcus Niemann");
MODULE_LICENSE("Dual MPL/GPL");


/*====================================================================*/

/* Parameters that can be set with 'insmod' */

static int protocol = 2;        /* EURO-ISDN Default */
module_param(protocol, int, 0);

/*====================================================================*/

/*
   The event() function is this driver's Card Services event handler.
   It will be called by Card Services when an appropriate card status
   event is received.  The config() and release() entry points are
   used to configure or release a socket, in response to card
   insertion and ejection events.  They are invoked from the sedlbauer
   event handler. 
*/

79
static int sedlbauer_config(struct pcmcia_device *link);
80
static void sedlbauer_release(struct pcmcia_device *link);
L
Linus Torvalds 已提交
81 82 83 84 85 86 87

/*
   The attach() and detach() entry points are used to create and destroy
   "instances" of the driver, where each instance represents everything
   needed to manage one actual PCMCIA card.
*/

88
static void sedlbauer_detach(struct pcmcia_device *p_dev);
L
Linus Torvalds 已提交
89 90 91 92 93 94 95 96 97 98 99 100 101 102

/*
   You'll also need to prototype all the functions that will actually
   be used to talk to your device.  See 'memory_cs' for a good example
   of a fully self-sufficient driver; the other drivers rely more or
   less on other parts of the kernel.
*/

/*
   A driver needs to provide a dev_node_t structure for each device
   on a card.  In some cases, there is only one device per card (for
   example, ethernet cards, modems).  In other cases, there may be
   many actual or logical devices (SCSI adapters, memory cards with
   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
L
Linus Torvalds 已提交
104 105 106 107 108 109 110 111 112 113
   structure.  We allocate them in the card's private data structure,
   because they generally shouldn't be allocated dynamically.

   In this case, we also provide a flag to indicate if a device is
   "stopped" due to a power management event, or card ejection.  The
   device IO routines can use a flag like this to throttle IO to a
   card that is not ready to accept it.
*/
   
typedef struct local_info_t {
114
	struct pcmcia_device	*p_dev;
L
Linus Torvalds 已提交
115 116 117 118 119 120 121 122 123 124 125 126 127 128 129 130 131
    dev_node_t		node;
    int			stop;
    int			cardnr;
} local_info_t;

/*======================================================================

    sedlbauer_attach() creates an "instance" of the driver, allocating
    local data structures for one device.  The device is registered
    with Card Services.

    The dev_link structure is initialized, but we don't actually
    configure the card at this point -- we wait until we receive a
    card insertion event.
    
======================================================================*/

132
static int sedlbauer_probe(struct pcmcia_device *link)
L
Linus Torvalds 已提交
133 134
{
    local_info_t *local;
135

136
    dev_dbg(&link->dev, "sedlbauer_attach()\n");
L
Linus Torvalds 已提交
137 138

    /* Allocate space for private device-specific data */
139
    local = kzalloc(sizeof(local_info_t), GFP_KERNEL);
140
    if (!local) return -ENOMEM;
L
Linus Torvalds 已提交
141
    local->cardnr = -1;
142

143
    local->p_dev = link;
144 145
    link->priv = local;

L
Linus Torvalds 已提交
146
    /* Interrupt setup */
147
    link->irq.Attributes = IRQ_TYPE_DYNAMIC_SHARING|IRQ_FIRST_SHARED;
L
Linus Torvalds 已提交
148 149 150 151 152 153 154 155 156 157 158 159 160 161 162 163 164 165 166 167 168
    link->irq.IRQInfo1 = IRQ_LEVEL_ID;
    link->irq.Handler = NULL;

    /*
      General socket configuration defaults can go here.  In this
      client, we assume very little, and rely on the CIS for almost
      everything.  In most clients, many details (i.e., number, sizes,
      and attributes of IO windows) are fixed by the nature of the
      device, and can be hard-wired here.
    */

    /* from old sedl_cs 
    */
    /* The io structure describes IO port mapping */
    link->io.NumPorts1 = 8;
    link->io.Attributes1 = IO_DATA_PATH_WIDTH_8;
    link->io.IOAddrLines = 3;

    link->conf.Attributes = 0;
    link->conf.IntType = INT_MEMORY_AND_IO;

169
    return sedlbauer_config(link);
L
Linus Torvalds 已提交
170 171 172 173 174 175 176 177 178 179 180
} /* sedlbauer_attach */

/*======================================================================

    This deletes a driver "instance".  The device is de-registered
    with Card Services.  If it has been released, all local data
    structures are freed.  Otherwise, the structures will be freed
    when the device is released.

======================================================================*/

181
static void sedlbauer_detach(struct pcmcia_device *link)
L
Linus Torvalds 已提交
182
{
183
	dev_dbg(&link->dev, "sedlbauer_detach(0x%p)\n", link);
L
Linus Torvalds 已提交
184

185 186
	((local_info_t *)link->priv)->stop = 1;
	sedlbauer_release(link);
L
Linus Torvalds 已提交
187

188 189
	/* This points to the parent local_info_t struct */
	kfree(link->priv);
L
Linus Torvalds 已提交
190 191 192 193 194 195 196 197 198
} /* sedlbauer_detach */

/*======================================================================

    sedlbauer_config() is scheduled to run after a CARD_INSERTION event
    is received, to configure the PCMCIA socket, and to make the
    device available to the system.
    
======================================================================*/
199 200
static int sedlbauer_config_check(struct pcmcia_device *p_dev,
				  cistpl_cftable_entry_t *cfg,
201
				  cistpl_cftable_entry_t *dflt,
202
				  unsigned int vcc,
203
				  void *priv_data)
L
Linus Torvalds 已提交
204
{
205
	win_req_t *req = priv_data;
L
Linus Torvalds 已提交
206

207 208
	if (cfg->index == 0)
		return -ENODEV;
L
Linus Torvalds 已提交
209 210 211

	/* Does this card need audio output? */
	if (cfg->flags & CISTPL_CFTABLE_AUDIO) {
212 213
		p_dev->conf.Attributes |= CONF_ENABLE_SPKR;
		p_dev->conf.Status = CCSR_AUDIO_ENA;
L
Linus Torvalds 已提交
214
	}
215

L
Linus Torvalds 已提交
216 217 218
	/* Use power settings for Vcc and Vpp if present */
	/*  Note that the CIS values need to be rescaled */
	if (cfg->vcc.present & (1<<CISTPL_POWER_VNOM)) {
219
		if (vcc != cfg->vcc.param[CISTPL_POWER_VNOM]/10000)
220
			return -ENODEV;
221
	} else if (dflt->vcc.present & (1<<CISTPL_POWER_VNOM)) {
222
		if (vcc != dflt->vcc.param[CISTPL_POWER_VNOM]/10000)
223
			return -ENODEV;
L
Linus Torvalds 已提交
224
	}
225

L
Linus Torvalds 已提交
226
	if (cfg->vpp1.present & (1<<CISTPL_POWER_VNOM))
227
		p_dev->conf.Vpp = cfg->vpp1.param[CISTPL_POWER_VNOM]/10000;
228 229
	else if (dflt->vpp1.present & (1<<CISTPL_POWER_VNOM))
		p_dev->conf.Vpp = dflt->vpp1.param[CISTPL_POWER_VNOM]/10000;
230

L
Linus Torvalds 已提交
231
	/* Do we need to allocate an interrupt? */
232
	if (cfg->irq.IRQInfo1 || dflt->irq.IRQInfo1)
233 234
		p_dev->conf.Attributes |= CONF_ENABLE_IRQ;

L
Linus Torvalds 已提交
235
	/* IO window settings */
236
	p_dev->io.NumPorts1 = p_dev->io.NumPorts2 = 0;
237 238
	if ((cfg->io.nwin > 0) || (dflt->io.nwin > 0)) {
		cistpl_io_t *io = (cfg->io.nwin) ? &cfg->io : &dflt->io;
239 240 241 242 243 244 245 246 247 248 249 250 251 252 253
		p_dev->io.Attributes1 = IO_DATA_PATH_WIDTH_AUTO;
		if (!(io->flags & CISTPL_IO_8BIT))
			p_dev->io.Attributes1 = IO_DATA_PATH_WIDTH_16;
		if (!(io->flags & CISTPL_IO_16BIT))
			p_dev->io.Attributes1 = IO_DATA_PATH_WIDTH_8;
		p_dev->io.BasePort1 = io->win[0].base;
		p_dev->io.NumPorts1 = io->win[0].len;
		if (io->nwin > 1) {
			p_dev->io.Attributes2 = p_dev->io.Attributes1;
			p_dev->io.BasePort2 = io->win[1].base;
			p_dev->io.NumPorts2 = io->win[1].len;
		}
		/* This reserves IO space but doesn't actually enable it */
		if (pcmcia_request_io(p_dev, &p_dev->io) != 0)
			return -ENODEV;
L
Linus Torvalds 已提交
254 255 256 257
	}

	/*
	  Now set up a common memory window, if needed.  There is room
258
	  in the struct pcmcia_device structure for one memory window handle,
L
Linus Torvalds 已提交
259 260 261 262 263 264 265 266
	  but if the base addresses need to be saved, or if multiple
	  windows are needed, the info should go in the private data
	  structure for this device.

	  Note that the memory window base is a physical address, and
	  needs to be mapped to virtual space with ioremap() before it
	  is used.
	*/
267 268
	if ((cfg->mem.nwin > 0) || (dflt->mem.nwin > 0)) {
		cistpl_mem_t *mem = (cfg->mem.nwin) ? &cfg->mem : &dflt->mem;
269
		memreq_t map;
270 271 272 273 274 275
		req->Attributes = WIN_DATA_WIDTH_16|WIN_MEMORY_TYPE_CM;
		req->Attributes |= WIN_ENABLE;
		req->Base = mem->win[0].host_addr;
		req->Size = mem->win[0].len;
		req->AccessSpeed = 0;
		if (pcmcia_request_window(&p_dev, req, &p_dev->win) != 0)
276 277 278
			return -ENODEV;
		map.Page = 0;
		map.CardOffset = mem->win[0].card_addr;
279
		if (pcmcia_map_mem_page(p_dev, p_dev->win, &map) != 0)
280
			return -ENODEV;
L
Linus Torvalds 已提交
281
	}
282 283 284 285 286 287 288 289
	return 0;
}



static int sedlbauer_config(struct pcmcia_device *link)
{
    local_info_t *dev = link->priv;
290
    win_req_t *req;
291
    int ret;
292 293
    IsdnCard_t  icard;

294
    dev_dbg(&link->dev, "sedlbauer_config(0x%p)\n", link);
295

296 297
    req = kzalloc(sizeof(win_req_t), GFP_KERNEL);
    if (!req)
298
	    return -ENOMEM;
299

300 301 302 303 304 305 306 307 308 309 310 311
    /*
      In this loop, we scan the CIS for configuration table entries,
      each of which describes a valid card configuration, including
      voltage, IO window, memory window, and interrupt settings.

      We make no assumptions about the card to be configured: we use
      just the information available in the CIS.  In an ideal world,
      this would work for any PCMCIA card, but it requires a complete
      and accurate CIS.  In practice, a driver usually "knows" most of
      these things without consulting the CIS, and most client drivers
      will only use the CIS to fill in implementation-defined details.
    */
312 313
    ret = pcmcia_loop_config(link, sedlbauer_config_check, req);
    if (ret)
314
	    goto failed;
315

L
Linus Torvalds 已提交
316 317 318 319 320
    /*
       Allocate an interrupt line.  Note that this does not assign a
       handler to the interrupt, unless the 'Handler' member of the
       irq structure is initialized.
    */
321 322 323 324 325
    if (link->conf.Attributes & CONF_ENABLE_IRQ) {
	    ret = pcmcia_request_irq(link, &link->irq);
	    if (ret)
		    goto failed;
    }
L
Linus Torvalds 已提交
326 327 328 329 330 331
	
    /*
       This actually configures the PCMCIA socket -- setting up
       the I/O windows and the interrupt mapping, and putting the
       card and host interface into "Memory and IO" mode.
    */
332 333 334
    ret = pcmcia_request_configuration(link, &link->conf);
    if (ret)
	    goto failed;
L
Linus Torvalds 已提交
335 336 337 338 339 340 341

    /*
      At this point, the dev_node_t structure(s) need to be
      initialized and arranged in a linked list at link->dev.
    */
    sprintf(dev->node.dev_name, "sedlbauer");
    dev->node.major = dev->node.minor = 0;
342
    link->dev_node = &dev->node;
L
Linus Torvalds 已提交
343 344

    /* Finally, report what we've done */
345 346 347 348
    printk(KERN_INFO "%s: index 0x%02x:",
	   dev->node.dev_name, link->conf.ConfigIndex);
    if (link->conf.Vpp)
	printk(", Vpp %d.%d", link->conf.Vpp/10, link->conf.Vpp%10);
L
Linus Torvalds 已提交
349 350 351 352 353 354 355 356 357
    if (link->conf.Attributes & CONF_ENABLE_IRQ)
	printk(", irq %d", link->irq.AssignedIRQ);
    if (link->io.NumPorts1)
	printk(", io 0x%04x-0x%04x", link->io.BasePort1,
	       link->io.BasePort1+link->io.NumPorts1-1);
    if (link->io.NumPorts2)
	printk(" & 0x%04x-0x%04x", link->io.BasePort2,
	       link->io.BasePort2+link->io.NumPorts2-1);
    if (link->win)
358 359
	printk(", mem 0x%06lx-0x%06lx", req->Base,
	       req->Base+req->Size-1);
L
Linus Torvalds 已提交
360 361 362 363 364 365 366
    printk("\n");

    icard.para[0] = link->irq.AssignedIRQ;
    icard.para[1] = link->io.BasePort1;
    icard.protocol = protocol;
    icard.typ = ISDN_CTYPE_SEDLBAUER_PCMCIA;
    
367 368 369 370 371
    ret = hisax_init_pcmcia(link, 
			    &(((local_info_t *)link->priv)->stop), &icard);
    if (ret < 0) {
	printk(KERN_ERR "sedlbauer_cs: failed to initialize SEDLBAUER PCMCIA %d at i/o %#x\n",
		ret, link->io.BasePort1);
L
Linus Torvalds 已提交
372
    	sedlbauer_release(link);
373
	return -ENODEV;
L
Linus Torvalds 已提交
374
    } else
375
	((local_info_t *)link->priv)->cardnr = ret;
L
Linus Torvalds 已提交
376

377
    return 0;
L
Linus Torvalds 已提交
378

379
failed:
L
Linus Torvalds 已提交
380
    sedlbauer_release(link);
381
    return -ENODEV;
L
Linus Torvalds 已提交
382 383 384 385 386 387 388 389 390 391 392

} /* sedlbauer_config */

/*======================================================================

    After a card is removed, sedlbauer_release() will unregister the
    device, and release the PCMCIA configuration.  If the device is
    still open, this will be postponed until it is closed.
    
======================================================================*/

393
static void sedlbauer_release(struct pcmcia_device *link)
L
Linus Torvalds 已提交
394 395
{
    local_info_t *local = link->priv;
396
    dev_dbg(&link->dev, "sedlbauer_release(0x%p)\n", link);
L
Linus Torvalds 已提交
397 398 399 400 401 402 403 404

    if (local) {
    	if (local->cardnr >= 0) {
    	    /* no unregister function with hisax */
	    HiSax_closecard(local->cardnr);
	}
    }

405
    pcmcia_disable_device(link);
L
Linus Torvalds 已提交
406 407
} /* sedlbauer_release */

408
static int sedlbauer_suspend(struct pcmcia_device *link)
409 410 411 412 413 414 415 416
{
	local_info_t *dev = link->priv;

	dev->stop = 1;

	return 0;
}

417
static int sedlbauer_resume(struct pcmcia_device *link)
418 419 420 421 422 423 424 425
{
	local_info_t *dev = link->priv;

	dev->stop = 0;

	return 0;
}

L
Linus Torvalds 已提交
426

427
static struct pcmcia_device_id sedlbauer_ids[] = {
428
	PCMCIA_DEVICE_PROD_ID123("SEDLBAUER", "speed star II", "V 3.1", 0x81fb79f5, 0xf3612e1d, 0x6b95c78a),
429 430 431 432 433 434 435 436 437 438
	PCMCIA_DEVICE_PROD_ID123("SEDLBAUER", "ISDN-Adapter", "4D67", 0x81fb79f5, 0xe4e9bc12, 0x397b7e90),
	PCMCIA_DEVICE_PROD_ID123("SEDLBAUER", "ISDN-Adapter", "4D98", 0x81fb79f5, 0xe4e9bc12, 0x2e5c7fce),
	PCMCIA_DEVICE_PROD_ID123("SEDLBAUER", "ISDN-Adapter", " (C) 93-94 VK", 0x81fb79f5, 0xe4e9bc12, 0x8db143fe),
	PCMCIA_DEVICE_PROD_ID123("SEDLBAUER", "ISDN-Adapter", " (c) 93-95 VK", 0x81fb79f5, 0xe4e9bc12, 0xb391ab4c),
	PCMCIA_DEVICE_PROD_ID12("HST High Soft Tech GmbH", "Saphir II B", 0xd79e0b84, 0x21d083ae),
/*	PCMCIA_DEVICE_PROD_ID1234("SEDLBAUER", 0x81fb79f5), */ /* too generic*/
	PCMCIA_DEVICE_NULL
};
MODULE_DEVICE_TABLE(pcmcia, sedlbauer_ids);

L
Linus Torvalds 已提交
439 440 441 442 443
static struct pcmcia_driver sedlbauer_driver = {
	.owner		= THIS_MODULE,
	.drv		= {
		.name	= "sedlbauer_cs",
	},
444
	.probe		= sedlbauer_probe,
445
	.remove		= sedlbauer_detach,
446
	.id_table	= sedlbauer_ids,
447 448
	.suspend	= sedlbauer_suspend,
	.resume		= sedlbauer_resume,
L
Linus Torvalds 已提交
449 450 451 452 453 454 455 456 457 458 459 460 461 462
};

static int __init init_sedlbauer_cs(void)
{
	return pcmcia_register_driver(&sedlbauer_driver);
}

static void __exit exit_sedlbauer_cs(void)
{
	pcmcia_unregister_driver(&sedlbauer_driver);
}

module_init(init_sedlbauer_cs);
module_exit(exit_sedlbauer_cs);