orinoco_cs.c
上传用户:lgb322
上传日期:2013-02-24
资源大小:30529k
文件大小:23k
源码类别:

嵌入式Linux

开发平台:

Unix_Linux

  1. /* orinoco_cs.c 0.09b - (formerly known as dldwd_cs.c)
  2.  *
  3.  * A driver for "Hermes" chipset based PCMCIA wireless adaptors, such
  4.  * as the Lucent WavelanIEEE/Orinoco cards and their OEM (Cabletron/
  5.  * EnteraSys RoamAbout 802.11, ELSA Airlancer, Melco Buffalo and others).
  6.  * It should also be usable on various Prism II based cards such as the
  7.  * Linksys, D-Link and Farallon Skyline. It should also work on Symbol
  8.  * cards such as the 3Com AirConnect and Ericsson WLAN.
  9.  * 
  10.  * Copyright notice & release notes in file orinoco.c
  11.  */
  12. #include <linux/config.h>
  13. #include <linux/module.h>
  14. #include <linux/kernel.h>
  15. #include <linux/init.h>
  16. #include <linux/sched.h>
  17. #include <linux/ptrace.h>
  18. #include <linux/slab.h>
  19. #include <linux/string.h>
  20. #include <linux/timer.h>
  21. #include <linux/ioport.h>
  22. #include <asm/uaccess.h>
  23. #include <asm/io.h>
  24. #include <asm/system.h>
  25. #include <linux/proc_fs.h>
  26. #include <linux/netdevice.h>
  27. #include <linux/if_arp.h>
  28. #include <linux/etherdevice.h>
  29. #include <linux/wireless.h>
  30. #include <linux/list.h>
  31. #include <pcmcia/version.h>
  32. #include <pcmcia/cs_types.h>
  33. #include <pcmcia/cs.h>
  34. #include <pcmcia/cistpl.h>
  35. #include <pcmcia/cisreg.h>
  36. #include <pcmcia/ds.h>
  37. #include <pcmcia/bus_ops.h>
  38. #include "hermes.h"
  39. #include "orinoco.h"
  40. /*====================================================================*/
  41. static char version[] __initdata = "orinoco_cs.c 0.09b (David Gibson <hermes@gibson.dropbear.id.au> and others)";
  42. MODULE_AUTHOR("David Gibson <hermes@gibson.dropbear.id.au>");
  43. MODULE_DESCRIPTION("Driver for PCMCIA Lucent Orinoco, Prism II based and similar wireless cards");
  44. #ifdef MODULE_LICENSE
  45. MODULE_LICENSE("Dual MPL/GPL");
  46. #endif
  47. /* Parameters that can be set with 'insmod' */
  48. /* The old way: bit map of interrupts to choose from */
  49. /* This means pick from 15, 14, 12, 11, 10, 9, 7, 5, 4, and 3 */
  50. static uint irq_mask = 0xdeb8;
  51. /* Newer, simpler way of listing specific interrupts */
  52. static int irq_list[4] = { -1 };
  53. /* Do a Pcmcia soft reset (may help some cards) */
  54. static int reset_cor = 0;
  55. /* Some D-Link cards have buggy CIS. They do work at 5v properly, but
  56.  * don't have any CIS entry for it. This workaround it... */
  57. static int ignore_cis_vcc; /* = 0 */
  58. MODULE_PARM(irq_mask, "i");
  59. MODULE_PARM(irq_list, "1-4i");
  60. MODULE_PARM(reset_cor, "i");
  61. MODULE_PARM(ignore_cis_vcc, "i");
  62. /* Pcmcia specific structure */
  63. struct orinoco_pccard {
  64. dev_link_t link;
  65. dev_node_t node;
  66. /* Common structure (fully included), see orinoco.h */
  67. struct orinoco_private  priv;
  68. };
  69. /*
  70.  * Function prototypes
  71.  */
  72. /* struct net_device methods */
  73. static int orinoco_cs_open(struct net_device *dev);
  74. static int orinoco_cs_stop(struct net_device *dev);
  75. /* PCMCIA gumpf */
  76. static void orinoco_cs_config(dev_link_t * link);
  77. static void orinoco_cs_release(u_long arg);
  78. static int orinoco_cs_event(event_t event, int priority,
  79.        event_callback_args_t * args);
  80. static dev_link_t *orinoco_cs_attach(void);
  81. static void orinoco_cs_detach(dev_link_t *);
  82. /*
  83.    The dev_info variable is the "key" that is used to match up this
  84.    device driver with appropriate cards, through the card configuration
  85.    database.
  86. */
  87. static dev_info_t dev_info = "orinoco_cs";
  88. /*
  89.    A linked list of "instances" of the dummy device.  Each actual
  90.    PCMCIA card corresponds to one device instance, and is described
  91.    by one dev_link_t structure (defined in ds.h).
  92.    You may not want to use a linked list for this -- for example, the
  93.    memory card driver uses an array of dev_link_t pointers, where minor
  94.    device numbers are used to derive the corresponding array index.
  95. */
  96. static dev_link_t *dev_list; /* = NULL */
  97. /*====================================================================*/
  98. static void
  99. cs_error(client_handle_t handle, int func, int ret)
  100. {
  101. error_info_t err = { func, ret };
  102. CardServices(ReportError, handle, &err);
  103. }
  104. static int
  105. orinoco_cs_open(struct net_device *dev)
  106. {
  107. struct orinoco_private *priv = (struct orinoco_private *)dev->priv;
  108. struct orinoco_pccard* card = (struct orinoco_pccard *)priv->card;
  109. dev_link_t *link = &card->link;
  110. int err;
  111. TRACE_ENTER(priv->ndev.name);
  112. link->open++;
  113. netif_device_attach(dev);
  114. err = orinoco_reset(priv);
  115. if (err)
  116. orinoco_cs_stop(dev);
  117. else
  118. netif_start_queue(dev);
  119. TRACE_EXIT(priv->ndev.name);
  120. return err;
  121. }
  122. static int
  123. orinoco_cs_stop(struct net_device *dev)
  124. {
  125. struct orinoco_private *priv = (struct orinoco_private *)dev->priv;
  126. struct orinoco_pccard* card = (struct orinoco_pccard *)priv->card;
  127. dev_link_t *link = &card->link;
  128. TRACE_ENTER(priv->ndev.name);
  129. netif_stop_queue(dev);
  130. orinoco_shutdown(priv);
  131. link->open--;
  132. if (link->state & DEV_STALE_CONFIG)
  133. mod_timer(&link->release, jiffies + HZ/20);
  134. TRACE_EXIT(priv->ndev.name);
  135. return 0;
  136. }
  137. /*
  138.  * Do a soft reset of the Pcmcia card using the Configuration Option Register
  139.  * Can't do any harm, and actually may do some good on some cards...
  140.  * In fact, this seem necessary for Spectrum cards...
  141.  */
  142. static int
  143. orinoco_cs_cor_reset(struct orinoco_private *priv)
  144. {
  145. struct orinoco_pccard* card = (struct orinoco_pccard *)priv->card;
  146. dev_link_t *link = &card->link;
  147. conf_reg_t reg;
  148. u_int default_cor; 
  149. TRACE_ENTER(priv->ndev.name);
  150. /* Doing it if hardware is gone is guaranteed crash */
  151. if(! (link->state & DEV_CONFIG) )
  152. return -ENODEV;
  153. /* Save original COR value */
  154. reg.Function = 0;
  155. reg.Action = CS_READ;
  156. reg.Offset = CISREG_COR;
  157. reg.Value = 0;
  158. CardServices(AccessConfigurationRegister, link->handle, &reg);
  159. default_cor = reg.Value;
  160. DEBUG(2, "orinoco : orinoco_cs_cor_reset() : cor=0x%Xn", default_cor);
  161. /* Soft-Reset card */
  162. reg.Action = CS_WRITE;
  163. reg.Offset = CISREG_COR;
  164. reg.Value = (default_cor | COR_SOFT_RESET);
  165. CardServices(AccessConfigurationRegister, link->handle, &reg);
  166. /* Wait until the card has acknowledged our reset */
  167. /* FIXME: mdelay() is deprecated -dgibson */
  168. mdelay(1);
  169. /* Restore original COR configuration index */
  170. reg.Value = (default_cor & ~COR_SOFT_RESET);
  171. CardServices(AccessConfigurationRegister, link->handle, &reg);
  172. /* Wait until the card has finished restarting */
  173. /* FIXME: mdelay() is deprecated -dgibson */
  174. mdelay(1);
  175. TRACE_EXIT(priv->ndev.name);
  176. return 0;
  177. }
  178. /* Remove zombie instances (card removed, detach pending) */
  179. static void
  180. flush_stale_links(void)
  181. {
  182. dev_link_t *link, *next;
  183. TRACE_ENTER("orinoco");
  184. for (link = dev_list; link; link = next) {
  185. next = link->next;
  186. if (link->state & DEV_STALE_LINK)
  187. orinoco_cs_detach(link);
  188. }
  189. TRACE_EXIT("orinoco");
  190. }
  191. /*======================================================================
  192.   orinoco_cs_attach() creates an "instance" of the driver, allocating
  193.   local data structures for one device.  The device is registered
  194.   with Card Services.
  195.   
  196.   The dev_link structure is initialized, but we don't actually
  197.   configure the card at this point -- we wait until we receive a
  198.   card insertion event.
  199.   ======================================================================*/
  200. static dev_link_t *
  201. orinoco_cs_attach(void)
  202. {
  203. struct orinoco_pccard *card;
  204. struct orinoco_private *priv;
  205. dev_link_t *link;
  206. struct net_device *ndev;
  207. client_reg_t client_reg;
  208. int ret, i;
  209. TRACE_ENTER("orinoco");
  210. /* A bit of cleanup */
  211. flush_stale_links();
  212. /* Allocate space for private device-specific data */
  213. card = kmalloc(sizeof(*card), GFP_KERNEL);
  214. if (! card) {
  215. link = NULL;
  216. goto out;
  217. }
  218. memset(card, 0, sizeof(*card));
  219. /* Link both structure together */
  220. priv = &(card->priv);
  221. priv->card = card;
  222. link = &card->link;
  223. ndev = &priv->ndev;
  224. link->priv = priv;
  225. /* Initialize the dev_link_t structure */
  226. link->release.function = &orinoco_cs_release;
  227. link->release.data = (u_long) link;
  228. /* Interrupt setup */
  229. link->irq.Attributes = IRQ_TYPE_EXCLUSIVE;
  230. link->irq.IRQInfo1 = IRQ_INFO2_VALID | IRQ_LEVEL_ID;
  231. if (irq_list[0] == -1)
  232. link->irq.IRQInfo2 = irq_mask;
  233. else
  234. for (i = 0; i < 4; i++)
  235. link->irq.IRQInfo2 |= 1 << irq_list[i];
  236. link->irq.Handler = NULL;
  237. /*
  238.    General socket configuration defaults can go here.  In this
  239.    client, we assume very little, and rely on the CIS for almost
  240.    everything.  In most clients, many details (i.e., number, sizes,
  241.    and attributes of IO windows) are fixed by the nature of the
  242.    device, and can be hard-wired here.
  243.  */
  244. link->conf.Attributes = 0;
  245. link->conf.IntType = INT_MEMORY_AND_IO;
  246. /* Setup the common part */
  247. if(orinoco_setup(priv) < 0) {
  248. kfree(card);
  249. return NULL;
  250. }
  251. /* Overrides */
  252. ndev->open = orinoco_cs_open;
  253. ndev->stop = orinoco_cs_stop;
  254. priv->card_reset_handler = orinoco_cs_cor_reset;
  255. /* Register with Card Services */
  256. link->next = dev_list;
  257. dev_list = link;
  258. client_reg.dev_info = &dev_info;
  259. client_reg.Attributes = INFO_IO_CLIENT | INFO_CARD_SHARE;
  260. client_reg.EventMask =
  261.     CS_EVENT_CARD_INSERTION | CS_EVENT_CARD_REMOVAL |
  262.     CS_EVENT_RESET_PHYSICAL | CS_EVENT_CARD_RESET |
  263.     CS_EVENT_PM_SUSPEND | CS_EVENT_PM_RESUME;
  264. client_reg.event_handler = &orinoco_cs_event;
  265. client_reg.Version = 0x0210;
  266. client_reg.event_callback_args.client_data = link;
  267. ret = CardServices(RegisterClient, &link->handle, &client_reg);
  268. if (ret != CS_SUCCESS) {
  269. cs_error(link->handle, RegisterClient, ret);
  270. orinoco_cs_detach(link);
  271. link = NULL;
  272. goto out;
  273. }
  274.  out:
  275. TRACE_EXIT("orinoco");
  276. return link;
  277. } /* orinoco_cs_attach */
  278. /*======================================================================
  279.   This deletes a driver "instance".  The device is de-registered
  280.   with Card Services.  If it has been released, all local data
  281.   structures are freed.  Otherwise, the structures will be freed
  282.   when the device is released.
  283.   ======================================================================*/
  284. static void
  285. orinoco_cs_detach(dev_link_t * link)
  286. {
  287. dev_link_t **linkp;
  288. struct orinoco_private *priv = link->priv;
  289. TRACE_ENTER("orinoco");
  290. /* Locate device structure */
  291. for (linkp = &dev_list; *linkp; linkp = &(*linkp)->next)
  292. if (*linkp == link)
  293. break;
  294. if (*linkp == NULL)
  295. goto out;
  296. /*
  297.    If the device is currently configured and active, we won't
  298.    actually delete it yet.  Instead, it is marked so that when
  299.    the release() function is called, that will trigger a proper
  300.    detach().
  301.  */
  302. if (link->state & DEV_CONFIG) {
  303. #ifdef PCMCIA_DEBUG
  304. printk(KERN_DEBUG "orinoco_cs: detach postponed, '%s' "
  305.        "still lockedn", link->dev->dev_name);
  306. #endif
  307. link->state |= DEV_STALE_LINK;
  308. goto out;
  309. }
  310. /* Break the link with Card Services */
  311. if (link->handle)
  312. CardServices(DeregisterClient, link->handle);
  313. /* Unlink device structure, and free it */
  314. *linkp = link->next;
  315. DEBUG(0, "orinoco_cs: detach: link=%p link->dev=%pn", link, link->dev);
  316. if (link->dev) {
  317. DEBUG(0, "orinoco_cs: About to unregister net device %pn",
  318.       &priv->ndev);
  319. unregister_netdev(&priv->ndev);
  320. }
  321. kfree(priv->card);
  322.  out:
  323. TRACE_EXIT("orinoco");
  324. } /* orinoco_cs_detach */
  325. /*======================================================================
  326.   orinoco_cs_config() is scheduled to run after a CARD_INSERTION event
  327.   is received, to configure the PCMCIA socket, and to make the
  328.   device available to the system.
  329.   ======================================================================*/
  330. #define CS_CHECK(fn, args...) 
  331. while ((last_ret=CardServices(last_fn=(fn),args))!=0) goto cs_failed
  332. #define CFG_CHECK(fn, args...) 
  333. if (CardServices(fn, args) != 0) goto next_entry
  334. static void
  335. orinoco_cs_config(dev_link_t * link)
  336. {
  337. client_handle_t handle = link->handle;
  338. struct orinoco_private *priv = link->priv;
  339. struct orinoco_pccard *card = (struct orinoco_pccard *)priv->card;
  340. hermes_t *hw = &priv->hw;
  341. struct net_device *ndev = &priv->ndev;
  342. tuple_t tuple;
  343. cisparse_t parse;
  344. int last_fn, last_ret;
  345. u_char buf[64];
  346. config_info_t conf;
  347. cistpl_cftable_entry_t dflt = { 0 };
  348. cisinfo_t info;
  349. TRACE_ENTER("orinoco");
  350. CS_CHECK(ValidateCIS, handle, &info);
  351. /*
  352.    This reads the card's CONFIG tuple to find its configuration
  353.    registers.
  354.  */
  355. tuple.DesiredTuple = CISTPL_CONFIG;
  356. tuple.Attributes = 0;
  357. tuple.TupleData = buf;
  358. tuple.TupleDataMax = sizeof(buf);
  359. tuple.TupleOffset = 0;
  360. CS_CHECK(GetFirstTuple, handle, &tuple);
  361. CS_CHECK(GetTupleData, handle, &tuple);
  362. CS_CHECK(ParseTuple, handle, &tuple, &parse);
  363. link->conf.ConfigBase = parse.config.base;
  364. link->conf.Present = parse.config.rmask[0];
  365. /* Configure card */
  366. link->state |= DEV_CONFIG;
  367. /* Look up the current Vcc */
  368. CS_CHECK(GetConfigurationInfo, handle, &conf);
  369. link->conf.Vcc = conf.Vcc;
  370. DEBUG(0, "orinoco_cs_config: ConfigBase = 0x%x link->conf.Vcc = %dn", 
  371.       link->conf.ConfigBase, link->conf.Vcc);
  372. /*
  373.    In this loop, we scan the CIS for configuration table entries,
  374.    each of which describes a valid card configuration, including
  375.    voltage, IO window, memory window, and interrupt settings.
  376.    We make no assumptions about the card to be configured: we use
  377.    just the information available in the CIS.  In an ideal world,
  378.    this would work for any PCMCIA card, but it requires a complete
  379.    and accurate CIS.  In practice, a driver usually "knows" most of
  380.    these things without consulting the CIS, and most client drivers
  381.    will only use the CIS to fill in implementation-defined details.
  382.  */
  383. tuple.DesiredTuple = CISTPL_CFTABLE_ENTRY;
  384. CS_CHECK(GetFirstTuple, handle, &tuple);
  385. while (1) {
  386. cistpl_cftable_entry_t *cfg = &(parse.cftable_entry);
  387. CFG_CHECK(GetTupleData, handle, &tuple);
  388. CFG_CHECK(ParseTuple, handle, &tuple, &parse);
  389. DEBUG(0, "orinoco_cs_config: index = 0x%x, flags = 0x%xn",
  390.       cfg->index, cfg->flags);
  391. if (cfg->flags & CISTPL_CFTABLE_DEFAULT)
  392. dflt = *cfg;
  393. if (cfg->index == 0)
  394. goto next_entry;
  395. link->conf.ConfigIndex = cfg->index;
  396. /* Does this card need audio output? */
  397. if (cfg->flags & CISTPL_CFTABLE_AUDIO) {
  398. link->conf.Attributes |= CONF_ENABLE_SPKR;
  399. link->conf.Status = CCSR_AUDIO_ENA;
  400. }
  401. /* Use power settings for Vcc and Vpp if present */
  402. /*  Note that the CIS values need to be rescaled */
  403. if (cfg->vcc.present & (1 << CISTPL_POWER_VNOM)) {
  404. if (conf.Vcc !=
  405.     cfg->vcc.param[CISTPL_POWER_VNOM] / 10000) {
  406. DEBUG(2, "orinoco_cs_config: Vcc mismatch (conf.Vcc = %d, CIS = %d)n",  conf.Vcc, cfg->vcc.param[CISTPL_POWER_VNOM] / 10000);
  407. if(!ignore_cis_vcc)
  408. goto next_entry;
  409. }
  410. } else if (dflt.vcc.present & (1 << CISTPL_POWER_VNOM)) {
  411. if (conf.Vcc !=
  412.     dflt.vcc.param[CISTPL_POWER_VNOM] / 10000) {
  413. DEBUG(2, "orinoco_cs_config: Vcc mismatch (conf.Vcc = %d, CIS = %d)n",  conf.Vcc, dflt.vcc.param[CISTPL_POWER_VNOM] / 10000);
  414. if(!ignore_cis_vcc)
  415. goto next_entry;
  416. }
  417. }
  418. if (cfg->vpp1.present & (1 << CISTPL_POWER_VNOM))
  419. link->conf.Vpp1 = link->conf.Vpp2 =
  420.     cfg->vpp1.param[CISTPL_POWER_VNOM] / 10000;
  421. else if (dflt.vpp1.present & (1 << CISTPL_POWER_VNOM))
  422. link->conf.Vpp1 = link->conf.Vpp2 =
  423.     dflt.vpp1.param[CISTPL_POWER_VNOM] / 10000;
  424. DEBUG(0, "orinoco_cs_config: We seem to have configured Vcc and Vppn");
  425. /* Do we need to allocate an interrupt? */
  426. if (cfg->irq.IRQInfo1 || dflt.irq.IRQInfo1)
  427. link->conf.Attributes |= CONF_ENABLE_IRQ;
  428. /* IO window settings */
  429. link->io.NumPorts1 = link->io.NumPorts2 = 0;
  430. if ((cfg->io.nwin > 0) || (dflt.io.nwin > 0)) {
  431. cistpl_io_t *io =
  432.     (cfg->io.nwin) ? &cfg->io : &dflt.io;
  433. link->io.Attributes1 = IO_DATA_PATH_WIDTH_AUTO;
  434. if (!(io->flags & CISTPL_IO_8BIT))
  435. link->io.Attributes1 =
  436.     IO_DATA_PATH_WIDTH_16;
  437. if (!(io->flags & CISTPL_IO_16BIT))
  438. link->io.Attributes1 =
  439.     IO_DATA_PATH_WIDTH_8;
  440. link->io.IOAddrLines =
  441.     io->flags & CISTPL_IO_LINES_MASK;
  442. link->io.BasePort1 = io->win[0].base;
  443. link->io.NumPorts1 = io->win[0].len;
  444. if (io->nwin > 1) {
  445. link->io.Attributes2 =
  446.     link->io.Attributes1;
  447. link->io.BasePort2 = io->win[1].base;
  448. link->io.NumPorts2 = io->win[1].len;
  449. }
  450. /* This reserves IO space but doesn't actually enable it */
  451. CFG_CHECK(RequestIO, link->handle, &link->io);
  452. }
  453. /* If we got this far, we're cool! */
  454. break;
  455. next_entry:
  456. if (link->io.NumPorts1)
  457. CardServices(ReleaseIO, link->handle, &link->io);
  458. CS_CHECK(GetNextTuple, handle, &tuple);
  459. }
  460. /*
  461.    Allocate an interrupt line.  Note that this does not assign a
  462.    handler to the interrupt, unless the 'Handler' member of the
  463.    irq structure is initialized.
  464.  */
  465. if (link->conf.Attributes & CONF_ENABLE_IRQ) {
  466. int i;
  467. link->irq.Attributes = IRQ_TYPE_EXCLUSIVE | IRQ_HANDLE_PRESENT;
  468. link->irq.IRQInfo1 = IRQ_INFO2_VALID | IRQ_LEVEL_ID;
  469. if (irq_list[0] == -1)
  470. link->irq.IRQInfo2 = irq_mask;
  471. else
  472. for (i=0; i<4; i++)
  473. link->irq.IRQInfo2 |= 1 << irq_list[i];
  474.    link->irq.Handler = orinoco_interrupt; 
  475.    link->irq.Instance = priv; 
  476. CS_CHECK(RequestIRQ, link->handle, &link->irq);
  477. }
  478. /* We initialize the hermes structure before completing PCMCIA
  479.    configuration just in case the interrupt handler gets
  480.    called. */
  481. hermes_struct_init(hw, link->io.BasePort1);
  482. /*
  483.    This actually configures the PCMCIA socket -- setting up
  484.    the I/O windows and the interrupt mapping, and putting the
  485.    card and host interface into "Memory and IO" mode.
  486.  */
  487. CS_CHECK(RequestConfiguration, link->handle, &link->conf);
  488. ndev->base_addr = link->io.BasePort1;
  489. ndev->irq = link->irq.AssignedIRQ;
  490. /* Now do a PCMCIA soft reset on the card, to make sure its in
  491.    a sane state */
  492. /* Optional because it really mess up old Lucent firmwares - Jean II */
  493. if (reset_cor)
  494. orinoco_cs_cor_reset(priv);
  495. /* register_netdev will give us an ethX name */
  496. ndev->name[0] = '';
  497. /* Tell the stack we exist */
  498. if (register_netdev(ndev) != 0) {
  499. printk(KERN_ERR "orinoco_cs: register_netdev() failedn");
  500. goto failed;
  501. }
  502. strcpy(card->node.dev_name, ndev->name);
  503. /* Finally, report what we've done */
  504. printk(KERN_DEBUG "%s: index 0x%02x: Vcc %d.%d",
  505.        ndev->name, link->conf.ConfigIndex,
  506.        link->conf.Vcc / 10, link->conf.Vcc % 10);
  507. if (link->conf.Vpp1)
  508. printk(", Vpp %d.%d", link->conf.Vpp1 / 10,
  509.        link->conf.Vpp1 % 10);
  510. if (link->conf.Attributes & CONF_ENABLE_IRQ)
  511. printk(", irq %d", link->irq.AssignedIRQ);
  512. if (link->io.NumPorts1)
  513. printk(", io 0x%04x-0x%04x", link->io.BasePort1,
  514.        link->io.BasePort1 + link->io.NumPorts1 - 1);
  515. if (link->io.NumPorts2)
  516. printk(" & 0x%04x-0x%04x", link->io.BasePort2,
  517.        link->io.BasePort2 + link->io.NumPorts2 - 1);
  518. printk("n");
  519. /* And give us the proc nodes for debugging */
  520. if (orinoco_proc_dev_init(priv) != 0) {
  521. printk(KERN_ERR "orinoco_cs: Failed to create /proc node for %sn",
  522.        ndev->name);
  523. goto failed;
  524. }
  525. /* Note to myself : this replace MOD_INC_USE_COUNT/MOD_DEC_USE_COUNT */
  526. SET_MODULE_OWNER(ndev);
  527. /* Do a Pcmcia soft reset of the card (optional) */
  528. if (reset_cor)
  529. orinoco_cs_cor_reset(priv);
  530. /*
  531.    At this point, the dev_node_t structure(s) need to be
  532.    initialized and arranged in a linked list at link->dev.
  533.  */
  534. card->node.major = card->node.minor = 0;
  535. link->dev = &card->node;
  536. link->state &= ~DEV_CONFIG_PENDING;
  537. TRACE_EXIT("orinoco");
  538. return;
  539.  cs_failed:
  540. cs_error(link->handle, last_fn, last_ret);
  541.  failed:
  542. orinoco_cs_release((u_long) link);
  543. TRACE_EXIT("orinoco");
  544. } /* orinoco_cs_config */
  545. /*======================================================================
  546.   After a card is removed, orinoco_cs_release() will unregister the
  547.   device, and release the PCMCIA configuration.  If the device is
  548.   still open, this will be postponed until it is closed.
  549.   ======================================================================*/
  550. static void
  551. orinoco_cs_release(u_long arg)
  552. {
  553. dev_link_t *link = (dev_link_t *) arg;
  554. struct orinoco_private *priv = link->priv;
  555. TRACE_ENTER(link->dev->dev_name);
  556. /*
  557.    If the device is currently in use, we won't release until it
  558.    is actually closed, because until then, we can't be sure that
  559.    no one will try to access the device or its data structures.
  560.  */
  561. if (link->open) {
  562. DEBUG(0, "orinoco_cs: release postponed, '%s' still openn",
  563.       link->dev->dev_name);
  564. link->state |= DEV_STALE_CONFIG;
  565. return;
  566. }
  567. /* Unregister proc entry */
  568. orinoco_proc_dev_cleanup(priv);
  569. /* Don't bother checking to see if these succeed or not */
  570. CardServices(ReleaseConfiguration, link->handle);
  571. if (link->io.NumPorts1)
  572. CardServices(ReleaseIO, link->handle, &link->io);
  573. if (link->irq.AssignedIRQ)
  574. CardServices(ReleaseIRQ, link->handle, &link->irq);
  575. link->state &= ~DEV_CONFIG;
  576. TRACE_EXIT(link->dev->dev_name);
  577. } /* orinoco_cs_release */
  578. /*======================================================================
  579.   The card status event handler.  Mostly, this schedules other
  580.   stuff to run after an event is received.
  581.   When a CARD_REMOVAL event is received, we immediately set a
  582.   private flag to block future accesses to this device.  All the
  583.   functions that actually access the device should check this flag
  584.   to make sure the card is still present.
  585.   ======================================================================*/
  586. static int
  587. orinoco_cs_event(event_t event, int priority,
  588.        event_callback_args_t * args)
  589. {
  590. dev_link_t *link = args->client_data;
  591. struct orinoco_private *priv = (struct orinoco_private *)link->priv;
  592. struct net_device *dev = &priv->ndev;
  593. TRACE_ENTER("orinoco");
  594. switch (event) {
  595. case CS_EVENT_CARD_REMOVAL:
  596. link->state &= ~DEV_PRESENT;
  597. if (link->state & DEV_CONFIG) {
  598. netif_stop_queue(dev);
  599. }
  600. orinoco_shutdown(priv);
  601. if (link->state & DEV_CONFIG) {
  602. netif_device_detach(dev);
  603. mod_timer(&link->release, jiffies + HZ / 20);
  604. }
  605. break;
  606. case CS_EVENT_CARD_INSERTION:
  607. link->state |= DEV_PRESENT | DEV_CONFIG_PENDING;
  608. orinoco_cs_config(link);
  609. break;
  610. case CS_EVENT_PM_SUSPEND:
  611. link->state |= DEV_SUSPEND;
  612. /* Fall through... */
  613. case CS_EVENT_RESET_PHYSICAL:
  614. orinoco_shutdown(priv);
  615. /* Mark the device as stopped, to block IO until later */
  616. if (link->state & DEV_CONFIG) {
  617. if (link->open) {
  618. netif_stop_queue(dev);
  619. netif_device_detach(dev);
  620. }
  621. CardServices(ReleaseConfiguration, link->handle);
  622. }
  623. break;
  624. case CS_EVENT_PM_RESUME:
  625. link->state &= ~DEV_SUSPEND;
  626. /* Fall through... */
  627. case CS_EVENT_CARD_RESET:
  628. if (link->state & DEV_CONFIG) {
  629. CardServices(RequestConfiguration, link->handle,
  630.      &link->conf);
  631. if (link->open) {
  632. if (orinoco_reset(priv) == 0) {
  633. netif_device_attach(dev);
  634. netif_start_queue(dev);
  635. } else {
  636. printk(KERN_ERR "%s: Error resetting device on PCMCIA eventn",
  637.        dev->name);
  638. orinoco_cs_stop(dev);
  639. }
  640. }
  641. }
  642. /*
  643.    In a normal driver, additional code may go here to restore
  644.    the device state and restart IO. 
  645.  */
  646. break;
  647. }
  648. TRACE_EXIT("orinoco");
  649. return 0;
  650. } /* orinoco_cs_event */
  651. static int __init
  652. init_orinoco_cs(void)
  653. {
  654. servinfo_t serv;
  655. TRACE_ENTER("orinoco");
  656. printk(KERN_DEBUG "%sn", version);
  657. CardServices(GetCardServicesInfo, &serv);
  658. if (serv.Revision != CS_RELEASE_CODE) {
  659. printk(KERN_NOTICE "orinoco_cs: Card Services release "
  660.        "does not match!n");
  661. return -1;
  662. }
  663. register_pccard_driver(&dev_info, &orinoco_cs_attach, &orinoco_cs_detach);
  664. TRACE_EXIT("orinoco");
  665. return 0;
  666. }
  667. static void __exit
  668. exit_orinoco_cs(void)
  669. {
  670. TRACE_ENTER("orinoco");
  671. unregister_pccard_driver(&dev_info);
  672. if (dev_list)
  673. DEBUG(0, "orinoco_cs: Removing leftover devices.n");
  674. while (dev_list != NULL) {
  675. del_timer(&dev_list->release);
  676. if (dev_list->state & DEV_CONFIG)
  677. orinoco_cs_release((u_long) dev_list);
  678. orinoco_cs_detach(dev_list);
  679. }
  680. TRACE_EXIT("orinoco");
  681. }
  682. module_init(init_orinoco_cs);
  683. module_exit(exit_orinoco_cs);