⭐ 欢迎来到虫虫下载站! | 📦 资源下载 📁 资源专辑 ℹ️ 关于我们
⭐ 虫虫下载站

📄 sedlbauer_cs.c

📁 linux-2.6.15.6
💻 C
📖 第 1 页 / 共 2 页
字号:
    /* Configure card */    link->state |= DEV_CONFIG;    /* Look up the current Vcc */    CS_CHECK(GetConfigurationInfo, pcmcia_get_configuration_info(handle, &conf));    link->conf.Vcc = conf.Vcc;    /*      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.    */    tuple.DesiredTuple = CISTPL_CFTABLE_ENTRY;    CS_CHECK(GetFirstTuple, pcmcia_get_first_tuple(handle, &tuple));    while (1) {	cistpl_cftable_entry_t dflt = { 0 };	cistpl_cftable_entry_t *cfg = &(parse.cftable_entry);	if (pcmcia_get_tuple_data(handle, &tuple) != 0 ||		pcmcia_parse_tuple(handle, &tuple, &parse) != 0)	    goto next_entry;	if (cfg->flags & CISTPL_CFTABLE_DEFAULT) dflt = *cfg;	if (cfg->index == 0) goto next_entry;	link->conf.ConfigIndex = cfg->index;		/* Does this card need audio output? */	if (cfg->flags & CISTPL_CFTABLE_AUDIO) {	    link->conf.Attributes |= CONF_ENABLE_SPKR;	    link->conf.Status = CCSR_AUDIO_ENA;	}		/* 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)) {	    if (conf.Vcc != cfg->vcc.param[CISTPL_POWER_VNOM]/10000)		goto next_entry;	} else if (dflt.vcc.present & (1<<CISTPL_POWER_VNOM)) {	    if (conf.Vcc != dflt.vcc.param[CISTPL_POWER_VNOM]/10000)		goto next_entry;	}	    	if (cfg->vpp1.present & (1<<CISTPL_POWER_VNOM))	    link->conf.Vpp1 = link->conf.Vpp2 =		cfg->vpp1.param[CISTPL_POWER_VNOM]/10000;	else if (dflt.vpp1.present & (1<<CISTPL_POWER_VNOM))	    link->conf.Vpp1 = link->conf.Vpp2 =		dflt.vpp1.param[CISTPL_POWER_VNOM]/10000;		/* Do we need to allocate an interrupt? */	if (cfg->irq.IRQInfo1 || dflt.irq.IRQInfo1)	    link->conf.Attributes |= CONF_ENABLE_IRQ;		/* IO window settings */	link->io.NumPorts1 = link->io.NumPorts2 = 0;	if ((cfg->io.nwin > 0) || (dflt.io.nwin > 0)) {	    cistpl_io_t *io = (cfg->io.nwin) ? &cfg->io : &dflt.io;	    link->io.Attributes1 = IO_DATA_PATH_WIDTH_AUTO;	    if (!(io->flags & CISTPL_IO_8BIT))		link->io.Attributes1 = IO_DATA_PATH_WIDTH_16;	    if (!(io->flags & CISTPL_IO_16BIT))		link->io.Attributes1 = IO_DATA_PATH_WIDTH_8;/* new in dummy.cs 2001/01/28 MN             link->io.IOAddrLines = io->flags & CISTPL_IO_LINES_MASK;*/	    link->io.BasePort1 = io->win[0].base;	    link->io.NumPorts1 = io->win[0].len;	    if (io->nwin > 1) {		link->io.Attributes2 = link->io.Attributes1;		link->io.BasePort2 = io->win[1].base;		link->io.NumPorts2 = io->win[1].len;	    }	    /* This reserves IO space but doesn't actually enable it */	    if (pcmcia_request_io(link->handle, &link->io) != 0)		goto next_entry;	}	/*	  Now set up a common memory window, if needed.  There is room	  in the dev_link_t structure for one memory window handle,	  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.	*/	if ((cfg->mem.nwin > 0) || (dflt.mem.nwin > 0)) {	    cistpl_mem_t *mem =		(cfg->mem.nwin) ? &cfg->mem : &dflt.mem;	    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;/* new in dummy.cs 2001/01/28 MN             if (req.Size < 0x1000)                req.Size = 0x1000;*/	    req.AccessSpeed = 0;	    if (pcmcia_request_window(&link->handle, &req, &link->win) != 0)		goto next_entry;	    map.Page = 0; map.CardOffset = mem->win[0].card_addr;	    if (pcmcia_map_mem_page(link->win, &map) != 0)		goto next_entry;	}	/* If we got this far, we're cool! */	break;	    next_entry:/* new in dummy.cs 2001/01/28 MN         if (link->io.NumPorts1)           pcmcia_release_io(link->handle, &link->io);*/	CS_CHECK(GetNextTuple, pcmcia_get_next_tuple(handle, &tuple));    }        /*       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.    */    if (link->conf.Attributes & CONF_ENABLE_IRQ)	CS_CHECK(RequestIRQ, pcmcia_request_irq(link->handle, &link->irq));	    /*       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.    */    CS_CHECK(RequestConfiguration, pcmcia_request_configuration(link->handle, &link->conf));    /*      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;    link->dev = &dev->node;    /* Finally, report what we've done */    printk(KERN_INFO "%s: index 0x%02x: Vcc %d.%d",	   dev->node.dev_name, link->conf.ConfigIndex,	   link->conf.Vcc/10, link->conf.Vcc%10);    if (link->conf.Vpp1)	printk(", Vpp %d.%d", link->conf.Vpp1/10, link->conf.Vpp1%10);    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)	printk(", mem 0x%06lx-0x%06lx", req.Base,	       req.Base+req.Size-1);    printk("\n");        link->state &= ~DEV_CONFIG_PENDING;    icard.para[0] = link->irq.AssignedIRQ;    icard.para[1] = link->io.BasePort1;    icard.protocol = protocol;    icard.typ = ISDN_CTYPE_SEDLBAUER_PCMCIA;        last_ret = hisax_init_pcmcia(link, &(((local_info_t*)link->priv)->stop), &icard);    if (last_ret < 0) {    	printk(KERN_ERR "sedlbauer_cs: failed to initialize SEDLBAUER PCMCIA %d at i/o %#x\n",    		last_ret, link->io.BasePort1);    	sedlbauer_release(link);    } else    	((local_info_t*)link->priv)->cardnr = last_ret;    return;cs_failed:    cs_error(link->handle, last_fn, last_ret);    sedlbauer_release(link);} /* 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.    ======================================================================*/static void sedlbauer_release(dev_link_t *link){    local_info_t *local = link->priv;    DEBUG(0, "sedlbauer_release(0x%p)\n", link);    if (local) {    	if (local->cardnr >= 0) {    	    /* no unregister function with hisax */	    HiSax_closecard(local->cardnr);	}    }    /* Unlink the device chain */    link->dev = NULL;    /*      In a normal driver, additional code may be needed to release      other kernel data structures associated with this device.     */        /* Don't bother checking to see if these succeed or not */    if (link->win)	pcmcia_release_window(link->win);    pcmcia_release_configuration(link->handle);    if (link->io.NumPorts1)	pcmcia_release_io(link->handle, &link->io);    if (link->irq.AssignedIRQ)	pcmcia_release_irq(link->handle, &link->irq);    link->state &= ~DEV_CONFIG;        if (link->state & DEV_STALE_LINK)	sedlbauer_detach(link);    } /* sedlbauer_release *//*======================================================================    The card status event handler.  Mostly, this schedules other    stuff to run after an event is received.    When a CARD_REMOVAL event is received, we immediately set a    private flag to block future accesses to this device.  All the    functions that actually access the device should check this flag    to make sure the card is still present.    ======================================================================*/static int sedlbauer_event(event_t event, int priority,		       event_callback_args_t *args){    dev_link_t *link = args->client_data;    local_info_t *dev = link->priv;        DEBUG(1, "sedlbauer_event(0x%06x)\n", event);        switch (event) {    case CS_EVENT_CARD_REMOVAL:	link->state &= ~DEV_PRESENT;	if (link->state & DEV_CONFIG) {	    ((local_info_t *)link->priv)->stop = 1;	    sedlbauer_release(link);	}	break;    case CS_EVENT_CARD_INSERTION:	link->state |= DEV_PRESENT | DEV_CONFIG_PENDING;	sedlbauer_config(link);	break;    case CS_EVENT_PM_SUSPEND:	link->state |= DEV_SUSPEND;	/* Fall through... */    case CS_EVENT_RESET_PHYSICAL:	/* Mark the device as stopped, to block IO until later */	dev->stop = 1;	if (link->state & DEV_CONFIG)	    pcmcia_release_configuration(link->handle);	break;    case CS_EVENT_PM_RESUME:	link->state &= ~DEV_SUSPEND;	/* Fall through... */    case CS_EVENT_CARD_RESET:	if (link->state & DEV_CONFIG)	    pcmcia_request_configuration(link->handle, &link->conf);	dev->stop = 0;	/*	  In a normal driver, additional code may go here to restore	  the device state and restart IO. 	*/	break;    }    return 0;} /* sedlbauer_event */static struct pcmcia_device_id sedlbauer_ids[] = {	PCMCIA_DEVICE_PROD_ID123("SEDLBAUER", "speed star II", "V 3.1", 0x81fb79f5, 0xf3612e1d, 0x6b95c78a),	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);static struct pcmcia_driver sedlbauer_driver = {	.owner		= THIS_MODULE,	.drv		= {		.name	= "sedlbauer_cs",	},	.attach		= sedlbauer_attach,	.event		= sedlbauer_event,	.detach		= sedlbauer_detach,	.id_table	= sedlbauer_ids,};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);	BUG_ON(dev_list != NULL);}module_init(init_sedlbauer_cs);module_exit(exit_sedlbauer_cs);

⌨️ 快捷键说明

复制代码 Ctrl + C
搜索代码 Ctrl + F
全屏模式 F11
切换主题 Ctrl + Shift + D
显示快捷键 ?
增大字号 Ctrl + =
减小字号 Ctrl + -