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

📄 orinoco_cs.c

📁 优龙2410linux2.6.8内核源代码
💻 C
📖 第 1 页 / 共 2 页
字号:
/* orinoco_cs.c 0.13e	- (formerly known as dldwd_cs.c) * * A driver for "Hermes" chipset based PCMCIA wireless adaptors, such * as the Lucent WavelanIEEE/Orinoco cards and their OEM (Cabletron/ * EnteraSys RoamAbout 802.11, ELSA Airlancer, Melco Buffalo and others). * It should also be usable on various Prism II based cards such as the * Linksys, D-Link and Farallon Skyline. It should also work on Symbol * cards such as the 3Com AirConnect and Ericsson WLAN. *  * Copyright notice & release notes in file orinoco.c */#include <linux/config.h>#ifdef  __IN_PCMCIA_PACKAGE__#include <pcmcia/k_compat.h>#endif /* __IN_PCMCIA_PACKAGE__ */#include <linux/module.h>#include <linux/kernel.h>#include <linux/init.h>#include <linux/sched.h>#include <linux/ptrace.h>#include <linux/slab.h>#include <linux/string.h>#include <linux/ioport.h>#include <linux/netdevice.h>#include <linux/if_arp.h>#include <linux/etherdevice.h>#include <linux/wireless.h>#include <pcmcia/version.h>#include <pcmcia/cs_types.h>#include <pcmcia/cs.h>#include <pcmcia/cistpl.h>#include <pcmcia/cisreg.h>#include <pcmcia/ds.h>#include <asm/uaccess.h>#include <asm/io.h>#include <asm/system.h>#include "orinoco.h"/********************************************************************//* Module stuff							    *//********************************************************************/MODULE_AUTHOR("David Gibson <hermes@gibson.dropbear.id.au>");MODULE_DESCRIPTION("Driver for PCMCIA Lucent Orinoco, Prism II based and similar wireless cards");#ifdef MODULE_LICENSEMODULE_LICENSE("Dual MPL/GPL");#endif/* Module parameters *//* The old way: bit map of interrupts to choose from *//* This means pick from 15, 14, 12, 11, 10, 9, 7, 5, 4, and 3 */static uint irq_mask = 0xdeb8;/* Newer, simpler way of listing specific interrupts */static int irq_list[4] = { -1 };/* Some D-Link cards have buggy CIS. They do work at 5v properly, but * don't have any CIS entry for it. This workaround it... */static int ignore_cis_vcc; /* = 0 */MODULE_PARM(irq_mask, "i");MODULE_PARM(irq_list, "1-4i");MODULE_PARM(ignore_cis_vcc, "i");/********************************************************************//* Magic constants						    *//********************************************************************//* * The dev_info variable is the "key" that is used to match up this * device driver with appropriate cards, through the card * configuration database. */static dev_info_t dev_info = "orinoco_cs";/********************************************************************//* Data structures						    *//********************************************************************//* PCMCIA specific device information (goes in the card field of * struct orinoco_private */struct orinoco_pccard {	dev_link_t link;	dev_node_t node;	/* Used to handle hard reset */	/* yuck, we need this hack to work around the insanity of the         * PCMCIA layer */	unsigned long hard_reset_in_progress; };/* * A linked list of "instances" of the device.  Each actual PCMCIA * card corresponds to one device instance, and is described by one * dev_link_t structure (defined in ds.h). */static dev_link_t *dev_list; /* = NULL *//********************************************************************//* Function prototypes						    *//********************************************************************//* device methods */static int orinoco_cs_hard_reset(struct orinoco_private *priv);/* PCMCIA gumpf */static void orinoco_cs_config(dev_link_t * link);static void orinoco_cs_release(dev_link_t * link);static int orinoco_cs_event(event_t event, int priority,			    event_callback_args_t * args);static dev_link_t *orinoco_cs_attach(void);static void orinoco_cs_detach(dev_link_t *);/********************************************************************//* Device methods     						    *//********************************************************************/static intorinoco_cs_hard_reset(struct orinoco_private *priv){	struct orinoco_pccard *card = priv->card;	dev_link_t *link = &card->link;	int err;	/* We need atomic ops here, because we're not holding the lock */	set_bit(0, &card->hard_reset_in_progress);	err = pcmcia_reset_card(link->handle, NULL);	if (err)		return err;	clear_bit(0, &card->hard_reset_in_progress);	return 0;}/********************************************************************//* PCMCIA stuff     						    *//********************************************************************//* In 2.5 (as of 2.5.69 at least) there is a cs_error exported which * does this, but it's not in 2.4 so we do our own for now. */static voidorinoco_cs_error(client_handle_t handle, int func, int ret){	error_info_t err = { func, ret };	pcmcia_report_error(handle, &err);}/* * This 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.  */static dev_link_t *orinoco_cs_attach(void){	struct net_device *dev;	struct orinoco_private *priv;	struct orinoco_pccard *card;	dev_link_t *link;	client_reg_t client_reg;	int ret, i;	dev = alloc_orinocodev(sizeof(*card), orinoco_cs_hard_reset);	if (! dev)		return NULL;	priv = dev->priv;	card = priv->card;	/* Link both structures together */	link = &card->link;	link->priv = dev;	/* Interrupt setup */	link->irq.Attributes = IRQ_TYPE_EXCLUSIVE;	link->irq.IRQInfo1 = IRQ_INFO2_VALID | IRQ_LEVEL_ID;	if (irq_list[0] == -1)		link->irq.IRQInfo2 = irq_mask;	else		for (i = 0; i < 4; i++)			link->irq.IRQInfo2 |= 1 << irq_list[i];	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. */	link->conf.Attributes = 0;	link->conf.IntType = INT_MEMORY_AND_IO;	/* Register with Card Services */	/* FIXME: need a lock? */	link->next = dev_list;	dev_list = link;	client_reg.dev_info = &dev_info;	client_reg.Attributes = INFO_IO_CLIENT | INFO_CARD_SHARE;	client_reg.EventMask =		CS_EVENT_CARD_INSERTION | CS_EVENT_CARD_REMOVAL |		CS_EVENT_RESET_PHYSICAL | CS_EVENT_CARD_RESET |		CS_EVENT_PM_SUSPEND | CS_EVENT_PM_RESUME;	client_reg.event_handler = &orinoco_cs_event;	client_reg.Version = 0x0210; /* FIXME: what does this mean? */	client_reg.event_callback_args.client_data = link;	ret = pcmcia_register_client(&link->handle, &client_reg);	if (ret != CS_SUCCESS) {		orinoco_cs_error(link->handle, RegisterClient, ret);		orinoco_cs_detach(link);		return NULL;	}	return link;}				/* orinoco_cs_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. */static voidorinoco_cs_detach(dev_link_t * link){	dev_link_t **linkp;	struct net_device *dev = link->priv;	/* Locate device structure */	for (linkp = &dev_list; *linkp; linkp = &(*linkp)->next)		if (*linkp == link)			break;	if (*linkp == NULL) {		BUG();		return;	}	if (link->state & DEV_CONFIG)		orinoco_cs_release(link);	/* Break the link with Card Services */	if (link->handle)		pcmcia_deregister_client(link->handle);	/* Unlink device structure, and free it */	*linkp = link->next;	DEBUG(0, "orinoco_cs: detach: link=%p link->dev=%p\n", link, link->dev);	if (link->dev) {		DEBUG(0, "orinoco_cs: About to unregister net device %p\n",		      dev);		unregister_netdev(dev);	}	free_netdev(dev);}				/* orinoco_cs_detach *//* * orinoco_cs_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. */#define CS_CHECK(fn, ret) \do { last_fn = (fn); if ((last_ret = (ret)) != 0) goto cs_failed; } while (0)static voidorinoco_cs_config(dev_link_t *link){	struct net_device *dev = link->priv;	client_handle_t handle = link->handle;	struct orinoco_private *priv = dev->priv;	struct orinoco_pccard *card = priv->card;	hermes_t *hw = &priv->hw;	int last_fn, last_ret;	u_char buf[64];	config_info_t conf;	cisinfo_t info;	tuple_t tuple;	cisparse_t parse;	CS_CHECK(ValidateCIS, pcmcia_validate_cis(handle, &info));	/*	 * This reads the card's CONFIG tuple to find its	 * configuration registers.	 */	tuple.DesiredTuple = CISTPL_CONFIG;	tuple.Attributes = 0;	tuple.TupleData = buf;	tuple.TupleDataMax = sizeof(buf);	tuple.TupleOffset = 0;	CS_CHECK(GetFirstTuple, pcmcia_get_first_tuple(handle, &tuple));	CS_CHECK(GetTupleData, pcmcia_get_tuple_data(handle, &tuple));	CS_CHECK(ParseTuple, pcmcia_parse_tuple(handle, &tuple, &parse));	link->conf.ConfigBase = parse.config.base;	link->conf.Present = parse.config.rmask[0];	/* 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 *cfg = &(parse.cftable_entry);		cistpl_cftable_entry_t dflt = { .index = 0 };		if (pcmcia_get_tuple_data(handle, &tuple) != 0 ||				pcmcia_parse_tuple(handle, &tuple, &parse) != 0)			goto next_entry;

⌨️ 快捷键说明

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