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

嵌入式Linux

开发平台:

Unix_Linux

  1. /* orinoco.c 0.09b - (formerly known as dldwd_cs.c and orinoco_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 (C) 2000 David Gibson, Linuxcare Australia <hermes@gibson.dropbear.id.au>
  11.  * With some help from :
  12.  * Copyright (C) 2001 Jean Tourrilhes, HP Labs <jt@hpl.hp.com>
  13.  * Copyright (C) 2001 Benjamin Herrenschmidt <benh@kernel.crashing.org>
  14.  *
  15.  * Based on dummy_cs.c 1.27 2000/06/12 21:27:25
  16.  *
  17.  * Portions based on wvlan_cs.c 1.0.6, Copyright Andreas Neuhaus <andy@fasta.fh-dortmund.de>
  18.  *      http://www.fasta.fh-dortmund.de/users/andy/wvlan/
  19.  *
  20.  * The contents of this file are subject to the Mozilla Public License
  21.  * Version 1.1 (the "License"); you may not use this file except in
  22.  * compliance with the License. You may obtain a copy of the License
  23.  * at http://www.mozilla.org/MPL/
  24.  *
  25.  * Software distributed under the License is distributed on an "AS IS"
  26.  * basis, WITHOUT WARRANTY OF ANY KIND, either express or implied. See
  27.  * the License for the specific language governing rights and
  28.  * limitations under the License.
  29.  *
  30.  * The initial developer of the original code is David A. Hinds
  31.  * <dahinds@users.sourceforge.net>.  Portions created by David
  32.  * A. Hinds are Copyright (C) 1999 David A. Hinds.  All Rights
  33.  * Reserved.
  34.  *
  35.  * Alternatively, the contents of this file may be used under the
  36.  * terms of the GNU General Public License version 2 (the "GPL"), in
  37.  * which case the provisions of the GPL are applicable instead of the
  38.  * above.  If you wish to allow the use of your version of this file
  39.  * only under the terms of the GPL and not to allow others to use your
  40.  * version of this file under the MPL, indicate your decision by
  41.  * deleting the provisions above and replace them with the notice and
  42.  * other provisions required by the GPL.  If you do not delete the
  43.  * provisions above, a recipient may use your version of this file
  44.  * under either the MPL or the GPL.
  45.  */
  46. /*
  47.  * Tentative changelog...
  48.  *
  49.  * v0.01 -> v0.02 - 21/3/2001 - Jean II
  50.  * o Allow to use regular ethX device name instead of dldwdX
  51.  * o Warning on IBSS with ESSID=any for firmware 6.06
  52.  * o Put proper range.throughput values (optimistic)
  53.  * o IWSPY support (IOCTL and stat gather in Rx path)
  54.  * o Allow setting frequency in Ad-Hoc mode
  55.  * o Disable WEP setting if !has_wep to work on old firmware
  56.  * o Fix txpower range
  57.  * o Start adding support for Samsung/Compaq firmware
  58.  *
  59.  * v0.02 -> v0.03 - 23/3/2001 - Jean II
  60.  * o Start adding Symbol support - need to check all that
  61.  * o Fix Prism2/Symbol WEP to accept 128 bits keys
  62.  * o Add Symbol WEP (add authentication type)
  63.  * o Add Prism2/Symbol rate
  64.  * o Add PM timeout (holdover duration)
  65.  * o Enable "iwconfig eth0 key off" and friends (toggle flags)
  66.  * o Enable "iwconfig eth0 power unicast/all" (toggle flags)
  67.  * o Try with an intel card. It report firmware 1.01, behave like
  68.  *   an antiquated firmware, however on windows it says 2.00. Yuck !
  69.  * o Workaround firmware bug in allocate buffer (Intel 1.01)
  70.  * o Finish external renaming to orinoco...
  71.  * o Testing with various Wavelan firmwares
  72.  *
  73.  * v0.03 -> v0.04 - 30/3/2001 - Jean II
  74.  * o Update to Wireless 11 -> add retry limit/lifetime support
  75.  * o Tested with a D-Link DWL 650 card, fill in firmware support
  76.  * o Warning on Vcc mismatch (D-Link 3.3v card in Lucent 5v only slot)
  77.  * o Fixed the Prims2 WEP bugs that I introduced in v0.03 :-(
  78.  *   It work on D-Link *only* after a tcpdump. Weird...
  79.  *   And still doesn't work on Intel card. Grrrr...
  80.  * o Update the mode after a setport3
  81.  * o Add preamble setting for Symbol cards (not yet enabled)
  82.  * o Don't complain as much about Symbol cards...
  83.  *
  84.  * v0.04 -> v0.04b - 22/4/2001 - David Gibson
  85.  *      o Removed the 'eth' parameter - always use ethXX as the
  86.  *        interface name instead of dldwdXX.  The other was racy
  87.  *        anyway.
  88.  * o Clean up RID definitions in hermes.h, other cleanups
  89.  *
  90.  * v0.04b -> v0.04c - 24/4/2001 - Jean II
  91.  * o Tim Hurley <timster@seiki.bliztech.com> reported a D-Link card
  92.  *   with vendor 02 and firmware 0.08. Added in the capabilities...
  93.  * o Tested Lucent firmware 7.28, everything works...
  94.  *
  95.  * v0.04c -> v0.05 - 3/5/2001 - Benjamin Herrenschmidt
  96.  * o Spin-off Pcmcia code. This file is renamed orinoco.c,
  97.  *   and orinoco_cs.c now contains only the Pcmcia specific stuff
  98.  * o Add Airport driver support on top of orinoco.c (see airport.c)
  99.  *
  100.  * v0.05 -> v0.05a - 4/5/2001 - Jean II
  101.  * o Revert to old Pcmcia code to fix breakage of Ben's changes...
  102.  *
  103.  * v0.05a -> v0.05b - 4/5/2001 - Jean II
  104.  * o add module parameter 'ignore_cis_vcc' for D-Link @ 5V
  105.  * o D-Link firmware doesn't support multicast. We just print a few
  106.  *   error messages, but otherwise everything works...
  107.  * o For David : set/getport3 works fine, just upgrade iwpriv...
  108.  *
  109.  * v0.05b -> v0.05c - 5/5/2001 - Benjamin Herrenschmidt
  110.  * o Adapt airport.c to latest changes in orinoco.c
  111.  * o Remove deferred power enabling code
  112.  *
  113.  * v0.05c -> v0.05d - 5/5/2001 - Jean II
  114.  * o Workaround to SNAP decapsulate frame from LinkSys AP
  115.  *   original patch from : Dong Liu <dliu@research.bell-labs.com>
  116.  *   (note : the memcmp bug was mine - fixed)
  117.  * o Remove set_retry stuff, no firmware support it (bloat--).
  118.  *
  119.  * v0.05d -> v0.06 - 25/5/2001 - Jean II
  120.  * Original patch from "Hong Lin" <alin@redhat.com>,
  121.  * "Ian Kinner" <ikinner@redhat.com>
  122.  * and "David Smith" <dsmith@redhat.com>
  123.  * o Init of priv->tx_rate_ctrl in firmware specific section.
  124.  * o Prism2/Symbol rate, upto should be 0xF and not 0x15. Doh !
  125.  * o Spectrum card always need cor_reset (for every reset)
  126.  * o Fix cor_reset to not loose bit 7 in the register
  127.  * o flush_stale_links to remove zombie Pcmcia instances
  128.  * o Ack previous hermes event before reset
  129.  * Me (with my little hands)
  130.  * o Allow orinoco.c to call cor_reset via priv->card_reset_handler
  131.  * o Add priv->need_card_reset to toggle this feature
  132.  * o Fix various buglets when setting WEP in Symbol firmware
  133.  *   Now, encryption is fully functional on Symbol cards. Youpi !
  134.  *
  135.  * v0.06 -> v0.06b - 25/5/2001 - Jean II
  136.  * o IBSS on Symbol use port_mode = 4. Please don't ask...
  137.  *
  138.  * v0.06b -> v0.06c - 29/5/2001 - Jean II
  139.  * o Show first spy address in /proc/net/wireless for IBSS mode as well
  140.  *
  141.  * v0.06c -> v0.06d - 6/7/2001 - David Gibson
  142.  *      o Change a bunch of KERN_INFO messages to KERN_DEBUG, as per Linus'
  143.  *        wishes to reduce the number of unecessary messages.
  144.  * o Removed bogus message on CRC error.
  145.  * o Merged fixeds for v0.08 Prism 2 firmware from William Waghorn
  146.  *   <willwaghorn@yahoo.co.uk>
  147.  * o Slight cleanup/re-arrangement of firmware detection code.
  148.  *
  149.  * v0.06d -> v0.06e - 1/8/2001 - David Gibson
  150.  * o Removed some redundant global initializers (orinoco_cs.c).
  151.  * o Added some module metadataa
  152.  *
  153.  * v0.06e -> v0.06f - 14/8/2001 - David Gibson
  154.  * o Wording fix to license
  155.  * o Added a 'use_alternate_encaps' module parameter for APs which need an
  156.  *   oui of 00:00:00.  We really need a better way of handling this, but
  157.  *   the module flag is better than nothing for now.
  158.  *
  159.  * v0.06f -> v0.07 - 20/8/2001 - David Gibson
  160.  * o Removed BAP error retries from hermes_bap_seek().  For Tx we now
  161.  *   let the upper layers handle the retry, we retry explicitly in the
  162.  *   Rx path, but don't make as much noise about it.
  163.  * o Firmware detection cleanups.
  164.  *
  165.  * v0.07 -> v0.07a - 1/10/3001 - Jean II
  166.  * o Add code to read Symbol firmware revision, inspired by latest code
  167.  *   in Spectrum24 by Lee John Keyser-Allen - Thanks Lee !
  168.  * o Thanks to Jared Valentine <hidden@xmission.com> for "providing" me
  169.  *   a 3Com card with a recent firmware, fill out Symbol firmware
  170.  *   capabilities of latest rev (2.20), as well as older Symbol cards.
  171.  * o Disable Power Management in newer Symbol firmware, the API 
  172.  *   has changed (documentation needed).
  173.  *
  174.  * v0.07a -> v0.08 - 3/10/2001 - David Gibson
  175.  * o Fixed a possible buffer overrun found by the Stanford checker (in
  176.  *   dldwd_ioctl_setiwencode()).  Can only be called by root anyway, so not
  177.  *   a big problem.
  178.  * o Turned has_big_wep on for Intersil cards.  That's not true for all of
  179.  *   them but we should at least let the capable ones try.
  180.  * o Wait for BUSY to clear at the beginning of hermes_bap_seek().  I
  181.  *   realised that my assumption that the driver's serialization
  182.  *   would prevent the BAP being busy on entry was possibly false, because
  183.  *   things other than seeks may make the BAP busy.
  184.  * o Use "alternate" (oui 00:00:00) encapsulation by default.
  185.  *   Setting use_old_encaps will mimic the old behaviour, but I think we
  186.  *   will be able to eliminate this.
  187.  * o Don't try to make __initdata const (the version string).  This can't
  188.  *   work because of the way the __initdata sectioning works.
  189.  * o Added MODULE_LICENSE tags.
  190.  * o Support for PLX (transparent PCMCIA->PCI brdge) cards.
  191.  * o Changed to using the new type-facist min/max.
  192.  *
  193.  * v0.08 -> v0.08a - 9/10/2001 - David Gibson
  194.  * o Inserted some missing acknowledgements/info into the Changelog.
  195.  * o Fixed some bugs in the normalisation of signel level reporting.
  196.  * o Fixed bad bug in WEP key handling on Intersil and Symbol firmware,
  197.  *   which led to an instant crash on big-endian machines.
  198.  *
  199.  * v0.08a -> v0.08b - 20/11/2001 - David Gibson
  200.  * o Lots of cleanup and bugfixes in orinoco_plx.c
  201.  * o Cleanup to handling of Tx rate setting.
  202.  * o Removed support for old encapsulation method.
  203.  * o Removed old "dldwd" names.
  204.  * o Split RID constants into a new file hermes_rid.h
  205.  * o Renamed RID constants to match linux-wlan-ng and prism2.o
  206.  * o Bugfixes in hermes.c
  207.  * o Poke the PLX's INTCSR register, so it actually starts
  208.  *   generating interrupts.  These cards might actually work now.
  209.  * o Update to wireless extensions v12 (Jean II)
  210.  * o Support for tallies and inquire command (Jean II)
  211.  * o Airport updates for newer PPC kernels (BenH)
  212.  *
  213.  * v0.08b -> v0.09 - 21/12/2001 - David Gibson
  214.  * o Some new PCI IDs for PLX cards.
  215.  * o Removed broken attempt to do ALLMULTI reception.  Just use
  216.  *   promiscuous mode instead
  217.  * o Preliminary work for list-AP (Jean II)
  218.  * o Airport updates from (BenH)
  219.  * o Eliminated racy hw_ready stuff
  220.  * o Fixed generation of fake events in irq handler.  This should
  221.  *   finally kill the EIO problems (Jean II & dgibson)
  222.  * o Fixed breakage of bitrate set/get on Agere firmware (Jean II)
  223.  *
  224.  * v0.09 -> v0.09a - 2/1/2002 - David Gibson
  225.  * o Fixed stupid mistake in multicast list handling, triggering
  226.  *   a BUG()
  227.  *
  228.  * v0.09a -> v0.09b - 16/1/2002 - David Gibson
  229.  * o Fixed even stupider mistake in new interrupt handling, which
  230.  *   seriously broke things on big-endian machines.
  231.  * o Removed a bunch of redundand includes and exports.
  232.  * o Removed a redundant MOD_{INC,DEC}_USE_COUNT pair in airport.c
  233.  * o Don't attempt to do hardware level multicast reception on
  234.  *   Intersil firmware, just go promisc instead.
  235.  * o Typo fixed in hermes_issue_cmd()
  236.  * o Eliminated WIRELESS_SPY #ifdefs
  237.  * o Status code reported on Tx exceptions
  238.  * o Moved netif_wake_queue() from ALLOC interrupts to TX and TXEXC
  239.  *   interrupts, which should fix the timeouts we're seeing.
  240.  *
  241.  * TODO
  242.  * o Find and kill remaining Tx timeout problems
  243.  * o Fix WEP / order of iwconfig wierdness on Intersil firmware
  244.  * o Convert /proc debugging stuff to seqfile
  245.  * o Re-assess our encapsulation detection strategy
  246.  * o Handle de-encapsulation within NET framework, provide 802.11
  247.  *   headers
  248.  * o Fix possible races in SPY handling.
  249.  * o Take the xmit lock when calling orinoco_reset from an
  250.  *   ioctl(), because that protects the multicast list.
  251.  * o Fix allocation lengths.
  252.  */
  253. /* Notes on locking:
  254.  *
  255.  * The basic principle of operation is that everything except the
  256.  * interrupt handler is serialized through a single spinlock in the
  257.  * struct orinoco_private structure, using dldwd_lock() and
  258.  * dldwd_unlock() (which in turn use spin_lock_bh() and
  259.  * spin_unlock_bh()).
  260.  *
  261.  * The kernel's IRQ handling stuff ensures that the interrupt handler
  262.  * does not re-enter itself. The interrupt handler is written such
  263.  * that everything it does is safe without a lock: chiefly this means
  264.  * that the Rx path uses one of the Hermes chipset's BAPs while
  265.  * everything else uses the other.
  266.  *
  267.  * Actually, strictly speaking, the updating of the statistics from
  268.  * the interrupt handler isn't safe without a lock.  However the worst
  269.  * that can happen is that we perturb the packet/byte counts slightly.
  270.  * We could fix this to use atomic types, but it's probably not worth
  271.  * it.
  272.  *
  273.  * The big exception is that that we don't want the irq handler
  274.  * running when we actually reset or shut down the card, because
  275.  * strange things might happen (probably the worst would be one packet
  276.  * of garbage, but you can't be too careful). For this we use
  277.  * __dldwd_stop_irqs() which will set a flag to disable the interrupt
  278.  * handler, and wait for any outstanding instances of the handler to
  279.  * complete. THIS WILL LOSE INTERRUPTS! so it shouldn't be used except
  280.  * for resets, where losing a few interrupts is acceptable. */
  281. #include <linux/config.h>
  282. #include <linux/module.h>
  283. #include <linux/kernel.h>
  284. #include <linux/init.h>
  285. #include <linux/sched.h>
  286. #include <linux/ptrace.h>
  287. #include <linux/slab.h>
  288. #include <linux/string.h>
  289. #include <linux/timer.h>
  290. #include <linux/ioport.h>
  291. #include <asm/uaccess.h>
  292. #include <asm/io.h>
  293. #include <asm/system.h>
  294. #include <linux/proc_fs.h>
  295. #include <linux/netdevice.h>
  296. #include <linux/if_arp.h>
  297. #include <linux/etherdevice.h>
  298. #include <linux/wireless.h>
  299. #include <linux/list.h>
  300. #include "hermes.h"
  301. #include "hermes_rid.h"
  302. #include "orinoco.h"
  303. #include "ieee802_11.h"
  304. /* Wireless extensions backwares compatibility */
  305. #ifndef SIOCIWFIRSTPRIV
  306. #define SIOCIWFIRSTPRIV SIOCDEVPRIVATE
  307. #endif /* SIOCIWFIRSTPRIV */
  308. /* We do this this way to avoid ifdefs in the actual code */
  309. #ifdef WIRELESS_SPY
  310. #define SPY_NUMBER(priv) (priv->spy_number)
  311. #else
  312. #define SPY_NUMBER(priv) 0
  313. #endif /* WIRELESS_SPY */
  314. static char version[] __initdata = "orinoco.c 0.09b (David Gibson <hermes@gibson.dropbear.id.au> and others)";
  315. MODULE_AUTHOR("David Gibson <hermes@gibson.dropbear.id.au>");
  316. MODULE_DESCRIPTION("Driver for Lucent Orinoco, Prism II based and similar wireless cards");
  317. #ifdef MODULE_LICENSE
  318. MODULE_LICENSE("Dual MPL/GPL");
  319. #endif
  320. /* Level of debugging. Used in the macros in orinoco.h */
  321. #ifdef ORINOCO_DEBUG
  322. int orinoco_debug = ORINOCO_DEBUG;
  323. MODULE_PARM(orinoco_debug, "i");
  324. #endif
  325. #define ORINOCO_MIN_MTU 256
  326. #define ORINOCO_MAX_MTU (HERMES_FRAME_LEN_MAX - ENCAPS_OVERHEAD)
  327. #define SYMBOL_MAX_VER_LEN (14)
  328. #define LTV_BUF_SIZE 128
  329. #define USER_BAP 0
  330. #define IRQ_BAP 1
  331. #define ORINOCO_MACPORT 0
  332. #define MAX_IRQLOOPS_PER_IRQ 10
  333. #define MAX_IRQLOOPS_PER_JIFFY (20000/HZ) /* Based on a guestimate of how many events the
  334.    device can legitimately generate */
  335. #define TX_NICBUF_SIZE 2048
  336. #define TX_NICBUF_SIZE_BUG 1585 /* Bug in Symbol firmware */
  337. #define LARGE_KEY_SIZE 13
  338. #define SMALL_KEY_SIZE 5
  339. #define MAX_FRAME_SIZE 2304
  340. #define DUMMY_FID 0xFFFF
  341. #define MAX_MULTICAST(priv) (priv->firmware_type == FIRMWARE_TYPE_AGERE ? 
  342.  HERMES_MAX_MULTICAST : 0)
  343. /*
  344.  * Data tables
  345.  */
  346. /* The frequency of each channel in MHz */
  347. const long channel_frequency[] = {
  348. 2412, 2417, 2422, 2427, 2432, 2437, 2442,
  349. 2447, 2452, 2457, 2462, 2467, 2472, 2484
  350. };
  351. #define NUM_CHANNELS ( sizeof(channel_frequency) / sizeof(channel_frequency[0]) )
  352. /* This tables gives the actual meanings of the bitrate IDs returned by the firmware. */
  353. struct {
  354. int bitrate; /* in 100s of kilbits */
  355. int automatic;
  356. u16 agere_txratectrl;
  357. u16 intersil_txratectrl;
  358. } bitrate_table[] = {
  359. {110, 1,  3, 15}, /* Entry 0 is the default */
  360. {10,  0,  1,  1},
  361. {10,  1,  1,  1},
  362. {20,  0,  2,  2},
  363. {20,  1,  6,  3},
  364. {55, 0,  4,  4},
  365. {55, 1,  7,  7},
  366. {110, 0,  5,  8},
  367. };
  368. #define BITRATE_TABLE_SIZE (sizeof(bitrate_table) / sizeof(bitrate_table[0]))
  369. struct p8022_hdr {
  370. u8 dsap;
  371. u8 ssap;
  372. u8 ctrl;
  373. u8 oui[3];
  374. } __attribute__ ((packed));
  375. struct orinoco_rxframe_hdr {
  376. struct hermes_rx_descriptor desc;
  377. struct ieee802_11_hdr p80211;
  378. struct ethhdr p8023;
  379. struct p8022_hdr p8022;
  380. u16 ethertype;
  381. } __attribute__ ((packed));
  382. struct orinoco_txframe_hdr {
  383. struct hermes_tx_descriptor desc;
  384. struct ieee802_11_hdr p80211;
  385. struct ethhdr p8023;
  386. struct p8022_hdr p8022;
  387. u16 ethertype;
  388. } __attribute__ ((packed));
  389. #define P8023_OFFSET (sizeof(struct hermes_rx_descriptor) + 
  390. sizeof(struct ieee802_11_hdr))
  391. #define ENCAPS_OVERHEAD (sizeof(struct p8022_hdr) + 2)
  392. /* 802.2 LLL header SNAP used for SNAP encapsulation over 802.11 */
  393. struct p8022_hdr encaps_hdr = {
  394. 0xaa, 0xaa, 0x03, {0x00, 0x00, 0x00}
  395. };
  396. typedef struct orinoco_commsqual {
  397. u16 qual, signal, noise;
  398. } __attribute__ ((packed)) orinoco_commsqual_t;
  399. /*
  400.  * Function prototypes
  401.  */
  402. static void orinoco_stat_gather(struct net_device *dev,
  403.       struct sk_buff *skb,
  404.       struct orinoco_rxframe_hdr *hdr);
  405. static struct net_device_stats *orinoco_get_stats(struct net_device *dev);
  406. static struct iw_statistics *orinoco_get_wireless_stats(struct net_device *dev);
  407. /* Hardware control routines */
  408. static int __orinoco_hw_reset(struct orinoco_private *priv);
  409. static int __orinoco_hw_set_bitrate(struct orinoco_private *priv);
  410. static int __orinoco_hw_setup_wep(struct orinoco_private *priv);
  411. static int orinoco_hw_get_bssid(struct orinoco_private *priv, char buf[ETH_ALEN]);
  412. static int orinoco_hw_get_essid(struct orinoco_private *priv, int *active,
  413.       char buf[IW_ESSID_MAX_SIZE+1]);
  414. static long orinoco_hw_get_freq(struct orinoco_private *priv);
  415. static int orinoco_hw_get_bitratelist(struct orinoco_private *priv, int *numrates,
  416.     int32_t *rates, int max);
  417. /* Interrupt handling routines */
  418. static void __orinoco_ev_tick(struct orinoco_private *priv, hermes_t *hw);
  419. static void __orinoco_ev_wterr(struct orinoco_private *priv, hermes_t *hw);
  420. static void __orinoco_ev_infdrop(struct orinoco_private *priv, hermes_t *hw);
  421. static void __orinoco_ev_info(struct orinoco_private *priv, hermes_t *hw);
  422. static void __orinoco_ev_rx(struct orinoco_private *priv, hermes_t *hw);
  423. static void __orinoco_ev_txexc(struct orinoco_private *priv, hermes_t *hw);
  424. static void __orinoco_ev_tx(struct orinoco_private *priv, hermes_t *hw);
  425. static void __orinoco_ev_alloc(struct orinoco_private *priv, hermes_t *hw);
  426. static int orinoco_ioctl_getiwrange(struct net_device *dev, struct iw_point *rrq);
  427. static int orinoco_ioctl_setiwencode(struct net_device *dev, struct iw_point *erq);
  428. static int orinoco_ioctl_getiwencode(struct net_device *dev, struct iw_point *erq);
  429. static int orinoco_ioctl_setessid(struct net_device *dev, struct iw_point *erq);
  430. static int orinoco_ioctl_getessid(struct net_device *dev, struct iw_point *erq);
  431. static int orinoco_ioctl_setnick(struct net_device *dev, struct iw_point *nrq);
  432. static int orinoco_ioctl_getnick(struct net_device *dev, struct iw_point *nrq);
  433. static int orinoco_ioctl_setfreq(struct net_device *dev, struct iw_freq *frq);
  434. static int orinoco_ioctl_getsens(struct net_device *dev, struct iw_param *srq);
  435. static int orinoco_ioctl_setsens(struct net_device *dev, struct iw_param *srq);
  436. static int orinoco_ioctl_setrts(struct net_device *dev, struct iw_param *rrq);
  437. static int orinoco_ioctl_setfrag(struct net_device *dev, struct iw_param *frq);
  438. static int orinoco_ioctl_getfrag(struct net_device *dev, struct iw_param *frq);
  439. static int orinoco_ioctl_setrate(struct net_device *dev, struct iw_param *frq);
  440. static int orinoco_ioctl_getrate(struct net_device *dev, struct iw_param *frq);
  441. static int orinoco_ioctl_setpower(struct net_device *dev, struct iw_param *prq);
  442. static int orinoco_ioctl_getpower(struct net_device *dev, struct iw_param *prq);
  443. static int orinoco_ioctl_setport3(struct net_device *dev, struct iwreq *wrq);
  444. static int orinoco_ioctl_getport3(struct net_device *dev, struct iwreq *wrq);
  445. static void __orinoco_set_multicast_list(struct net_device *dev);
  446. /* /proc debugging stuff */
  447. static int orinoco_proc_init(void);
  448. static void orinoco_proc_cleanup(void);
  449. /*
  450.  * Inline functions
  451.  */
  452. static inline void
  453. orinoco_lock(struct orinoco_private *priv)
  454. {
  455. spin_lock_bh(&priv->lock);
  456. }
  457. static inline void
  458. orinoco_unlock(struct orinoco_private *priv)
  459. {
  460. spin_unlock_bh(&priv->lock);
  461. }
  462. static inline int
  463. orinoco_irqs_allowed(struct orinoco_private *priv)
  464. {
  465. return test_bit(ORINOCO_STATE_DOIRQ, &priv->state);
  466. }
  467. static inline void
  468. __orinoco_stop_irqs(struct orinoco_private *priv)
  469. {
  470. hermes_t *hw = &priv->hw;
  471. hermes_set_irqmask(hw, 0);
  472. clear_bit(ORINOCO_STATE_DOIRQ, &priv->state);
  473. while (test_bit(ORINOCO_STATE_INIRQ, &priv->state))
  474. ;
  475. }
  476. static inline void
  477. __orinoco_start_irqs(struct orinoco_private *priv, u16 irqmask)
  478. {
  479. hermes_t *hw = &priv->hw;
  480. TRACE_ENTER(priv->ndev.name);
  481. __cli();
  482. set_bit(ORINOCO_STATE_DOIRQ, &priv->state);
  483. hermes_set_irqmask(hw, irqmask);
  484. __sti();
  485. TRACE_EXIT(priv->ndev.name);
  486. }
  487. static inline void
  488. set_port_type(struct orinoco_private *priv)
  489. {
  490. switch (priv->iw_mode) {
  491. case IW_MODE_INFRA:
  492. priv->port_type = 1;
  493. priv->allow_ibss = 0;
  494. break;
  495. case IW_MODE_ADHOC:
  496. if (priv->prefer_port3) {
  497. priv->port_type = 3;
  498. priv->allow_ibss = 0;
  499. } else {
  500. priv->port_type = priv->ibss_port;
  501. priv->allow_ibss = 1;
  502. }
  503. break;
  504. default:
  505. printk(KERN_ERR "%s: Invalid priv->iw_mode in set_port_type()n",
  506.        priv->ndev.name);
  507. }
  508. }
  509. extern void
  510. orinoco_set_multicast_list(struct net_device *dev)
  511. {
  512. struct orinoco_private *priv = dev->priv;
  513. orinoco_lock(priv);
  514. __orinoco_set_multicast_list(dev);
  515. orinoco_unlock(priv);
  516. }
  517. /*
  518.  * Hardware control routines
  519.  */
  520. static int
  521. __orinoco_hw_reset(struct orinoco_private *priv)
  522. {
  523. hermes_t *hw = &priv->hw;
  524. return hermes_reset(hw);
  525. }
  526. void
  527. orinoco_shutdown(struct orinoco_private *priv)
  528. {
  529. /*  hermes_t *hw = &priv->hw; */
  530. int err = 0;
  531. TRACE_ENTER(priv->ndev.name);
  532. orinoco_lock(priv);
  533. __orinoco_stop_irqs(priv);
  534. err = __orinoco_hw_reset(priv);
  535. if (err && err != -ENODEV) /* If the card is gone, we don't care about shutting it down */
  536. printk(KERN_ERR "%s: Error %d shutting down Hermes chipsetn", priv->ndev.name, err);
  537. orinoco_unlock(priv);
  538. TRACE_EXIT(priv->ndev.name);
  539. }
  540. int
  541. orinoco_reset(struct orinoco_private *priv)
  542. {
  543. struct net_device *dev = &priv->ndev;
  544. hermes_t *hw = &priv->hw;
  545. int err = 0;
  546. struct hermes_idstring idbuf;
  547. int frame_size;
  548. TRACE_ENTER(priv->ndev.name);
  549. /* Stop other people bothering us */
  550. orinoco_lock(priv);
  551. __orinoco_stop_irqs(priv);
  552. /* Check if we need a card reset */
  553. if((priv->need_card_reset) && (priv->card_reset_handler != NULL))
  554. priv->card_reset_handler(priv);
  555. /* Do standard firmware reset if we can */
  556. err = __orinoco_hw_reset(priv);
  557. if (err)
  558. goto out;
  559. frame_size = TX_NICBUF_SIZE;
  560. /* This stupid bug is present in Intel firmware 1.10, and
  561.  * may be fixed in later firmwares - Jean II */
  562. if(priv->broken_allocate)
  563. frame_size = TX_NICBUF_SIZE_BUG;
  564. err = hermes_allocate(hw, frame_size, &priv->txfid);
  565. if (err)
  566. goto out;
  567. /* Now set up all the parameters on the card */
  568. /* Set up the link mode */
  569. err = hermes_write_wordrec(hw, USER_BAP, HERMES_RID_CNFPORTTYPE, priv->port_type);
  570. if (err)
  571. goto out;
  572. if (priv->has_ibss) {
  573. err = hermes_write_wordrec(hw, USER_BAP, HERMES_RID_CNFCREATEIBSS,
  574.    priv->allow_ibss);
  575. if (err)
  576. goto out;
  577. if((strlen(priv->desired_essid) == 0) && (priv->allow_ibss)
  578.    && (!priv->has_ibss_any)) {
  579. printk(KERN_WARNING "%s: This firmware requires an 
  580. ESSID in IBSS-Ad-Hoc mode.n", dev->name);
  581. /* With wvlan_cs, in this case, we would crash.
  582.  * hopefully, this driver will behave better...
  583.  * Jean II */
  584. }
  585. }
  586. /* Set up encryption */
  587. if (priv->has_wep) {
  588. err = __orinoco_hw_setup_wep(priv);
  589. if (err) {
  590. printk(KERN_ERR "%s: Error %d activating WEP.n",
  591.        dev->name, err);
  592. goto out;
  593. }
  594. }
  595. /* Set the desired ESSID */
  596. idbuf.len = cpu_to_le16(strlen(priv->desired_essid));
  597. memcpy(&idbuf.val, priv->desired_essid, sizeof(idbuf.val));
  598. err = hermes_write_ltv(hw, USER_BAP, (priv->port_type == 3) ?
  599.        HERMES_RID_CNFOWNSSID : HERMES_RID_CNFDESIREDSSID,
  600.        HERMES_BYTES_TO_RECLEN(strlen(priv->desired_essid)+2),
  601.        &idbuf);
  602. if (err)
  603. goto out;
  604. /* Set the station name */
  605. idbuf.len = cpu_to_le16(strlen(priv->nick));
  606. memcpy(&idbuf.val, priv->nick, sizeof(idbuf.val));
  607. err = hermes_write_ltv(hw, USER_BAP, HERMES_RID_CNFOWNNAME,
  608.        HERMES_BYTES_TO_RECLEN(strlen(priv->nick)+2),
  609.        &idbuf);
  610. if (err)
  611. goto out;
  612. /* Set the channel/frequency */
  613. err = hermes_write_wordrec(hw, USER_BAP, HERMES_RID_CNFOWNCHANNEL, priv->channel);
  614. if (err)
  615. goto out;
  616. /* Set AP density */
  617. err = hermes_write_wordrec(hw, USER_BAP, HERMES_RID_CNFSYSTEMSCALE, priv->ap_density);
  618. if (err)
  619. goto out;
  620. /* Set RTS threshold */
  621. err = hermes_write_wordrec(hw, USER_BAP, HERMES_RID_CNFRTSTHRESHOLD, priv->rts_thresh);
  622. if (err)
  623. goto out;
  624. /* Set fragmentation threshold or MWO robustness */
  625. if (priv->has_mwo)
  626. err = hermes_write_wordrec(hw, USER_BAP,
  627.    HERMES_RID_CNFMWOROBUST_AGERE,
  628.    priv->mwo_robust);
  629. else
  630. err = hermes_write_wordrec(hw, USER_BAP,
  631.    HERMES_RID_CNFFRAGMENTATIONTHRESHOLD,
  632.    priv->frag_thresh);
  633. if (err)
  634. goto out;
  635. /* Set bitrate */
  636. err = __orinoco_hw_set_bitrate(priv);
  637. if (err)
  638. goto out;
  639. /* Set power management */
  640. if (priv->has_pm) {
  641. err = hermes_write_wordrec(hw, USER_BAP, HERMES_RID_CNFPMENABLED,
  642.    priv->pm_on);
  643. if (err)
  644. goto out;
  645. err = hermes_write_wordrec(hw, USER_BAP,
  646.    HERMES_RID_CNFMULTICASTRECEIVE,
  647.    priv->pm_mcast);
  648. if (err)
  649. goto out;
  650. err = hermes_write_wordrec(hw, USER_BAP,
  651.    HERMES_RID_CNFMAXSLEEPDURATION,
  652.    priv->pm_period);
  653. if (err)
  654. goto out;
  655. err = hermes_write_wordrec(hw, USER_BAP,
  656.    HERMES_RID_CNFPMHOLDOVERDURATION,
  657.    priv->pm_timeout);
  658. if (err)
  659. goto out;
  660. }
  661. /* Set preamble - only for Symbol so far... */
  662. if (priv->has_preamble) {
  663. err = hermes_write_wordrec(hw, USER_BAP,
  664.    HERMES_RID_CNFPREAMBLE_SYMBOL,
  665.    priv->preamble);
  666. if (err) {
  667. printk(KERN_WARNING "%s: Can't set preamble!n", dev->name);
  668. goto out;
  669. }
  670. }
  671. /* Set promiscuity / multicast*/
  672. priv->promiscuous = 0;
  673. priv->mc_count = 0;
  674. __orinoco_set_multicast_list(dev);
  675. __orinoco_start_irqs(priv, HERMES_EV_RX | HERMES_EV_ALLOC |
  676.    HERMES_EV_TX | HERMES_EV_TXEXC |
  677.    HERMES_EV_WTERR | HERMES_EV_INFO |
  678.    HERMES_EV_INFDROP);
  679. err = hermes_enable_port(hw, ORINOCO_MACPORT);
  680. if (err)
  681. goto out;
  682.  out:
  683. orinoco_unlock(priv);
  684. TRACE_EXIT(priv->ndev.name);
  685. return err;
  686. }
  687. static int __orinoco_hw_set_bitrate(struct orinoco_private *priv)
  688. {
  689. hermes_t *hw = &priv->hw;
  690. int err = 0;
  691. TRACE_ENTER(priv->ndev.name);
  692. if (priv->bitratemode >= BITRATE_TABLE_SIZE) {
  693. printk(KERN_ERR "%s: BUG: Invalid bitrate mode %dn",
  694.        priv->ndev.name, priv->bitratemode);
  695. return -EINVAL;
  696. }
  697. switch (priv->firmware_type) {
  698. case FIRMWARE_TYPE_AGERE:
  699. err = hermes_write_wordrec(hw, USER_BAP,
  700.    HERMES_RID_CNFTXRATECONTROL,
  701.    bitrate_table[priv->bitratemode].agere_txratectrl);
  702. break;
  703. case FIRMWARE_TYPE_INTERSIL:
  704. case FIRMWARE_TYPE_SYMBOL:
  705. err = hermes_write_wordrec(hw, USER_BAP,
  706.    HERMES_RID_CNFTXRATECONTROL,
  707.    bitrate_table[priv->bitratemode].intersil_txratectrl);
  708. break;
  709. default:
  710. BUG();
  711. }
  712. TRACE_EXIT(priv->ndev.name);
  713. return err;
  714. }
  715. static int __orinoco_hw_setup_wep(struct orinoco_private *priv)
  716. {
  717. hermes_t *hw = &priv->hw;
  718. int err = 0;
  719. int master_wep_flag;
  720. int auth_flag;
  721. TRACE_ENTER(priv->ndev.name);
  722. switch (priv->firmware_type) {
  723. case FIRMWARE_TYPE_AGERE: /* Agere style WEP */
  724. if (priv->wep_on) {
  725. err = hermes_write_wordrec(hw, USER_BAP,
  726.    HERMES_RID_CNFTXKEY_AGERE,
  727.    priv->tx_key);
  728. if (err)
  729. return err;
  730. err = HERMES_WRITE_RECORD(hw, USER_BAP,
  731.   HERMES_RID_CNFWEPKEYS_AGERE,
  732.   &priv->keys);
  733. if (err)
  734. return err;
  735. }
  736. err = hermes_write_wordrec(hw, USER_BAP,
  737.    HERMES_RID_CNFWEPENABLED_AGERE,
  738.    priv->wep_on);
  739. if (err)
  740. return err;
  741. break;
  742. case FIRMWARE_TYPE_INTERSIL: /* Intersil style WEP */
  743. case FIRMWARE_TYPE_SYMBOL: /* Symbol style WEP */
  744. master_wep_flag = 0; /* Off */
  745. if (priv->wep_on) {
  746. int keylen;
  747. int i;
  748. /* Fudge around firmware weirdness */
  749. keylen = le16_to_cpu(priv->keys[priv->tx_key].len);
  750. /* Write all 4 keys */
  751. for(i = 0; i < ORINOCO_MAX_KEYS; i++) {
  752. /*   int keylen = le16_to_cpu(priv->keys[i].len); */
  753. if (keylen > LARGE_KEY_SIZE) {
  754. printk(KERN_ERR "%s: BUG: Key %d has oversize length %d.n",
  755.        priv->ndev.name, i, keylen);
  756. return -E2BIG;
  757. }
  758. printk("About to write key %d, keylen=%dn",
  759.        i, keylen);      
  760. err = hermes_write_ltv(hw, USER_BAP,
  761.        HERMES_RID_CNFDEFAULTKEY0 + i,
  762.        HERMES_BYTES_TO_RECLEN(keylen),
  763.        priv->keys[i].data);
  764. if (err)
  765. return err;
  766. }
  767. /* Write the index of the key used in transmission */
  768. err = hermes_write_wordrec(hw, USER_BAP, HERMES_RID_CNFWEPDEFAULTKEYID,
  769.    priv->tx_key);
  770. if (err)
  771. return err;
  772. /* Authentication is where Intersil and Symbol
  773.  * firmware differ... */
  774. if (priv->firmware_type == FIRMWARE_TYPE_SYMBOL) {
  775. /* Symbol cards : set the authentication :
  776.  * 0 -> no encryption, 1 -> open,
  777.  * 2 -> shared key
  778.  * 3 -> shared key 128 -> AP only */
  779. if(priv->wep_restrict)
  780. auth_flag = 2;
  781. else
  782. auth_flag = 1;
  783. err = hermes_write_wordrec(hw, USER_BAP,
  784.    HERMES_RID_CNFAUTHENTICATION, auth_flag);
  785. if (err)
  786. return err;
  787. /* Master WEP setting is always 3 */
  788. master_wep_flag = 3;
  789. } else {
  790. /* Prism2 card : we need to modify master
  791.  * WEP setting */
  792. if(priv->wep_restrict)
  793. master_wep_flag = 3;
  794. else
  795. master_wep_flag = 1;
  796. }
  797. }
  798. /* Master WEP setting : on/off */
  799. err = hermes_write_wordrec(hw, USER_BAP,
  800.    HERMES_RID_CNFWEPFLAGS_INTERSIL,
  801.    master_wep_flag);
  802. if (err)
  803. return err;
  804. break;
  805. default:
  806. if (priv->wep_on) {
  807. printk(KERN_ERR "%s: WEP enabled, although not supported!n",
  808.        priv->ndev.name);
  809. return -EINVAL;
  810. }
  811. }
  812. TRACE_EXIT(priv->ndev.name);
  813. return 0;
  814. }
  815. static int orinoco_hw_get_bssid(struct orinoco_private *priv, char buf[ETH_ALEN])
  816. {
  817. hermes_t *hw = &priv->hw;
  818. int err = 0;
  819. orinoco_lock(priv);
  820. err = hermes_read_ltv(hw, USER_BAP, HERMES_RID_CURRENTBSSID,
  821.       ETH_ALEN, NULL, buf);
  822. orinoco_unlock(priv);
  823. return err;
  824. }
  825. static int orinoco_hw_get_essid(struct orinoco_private *priv, int *active,
  826.       char buf[IW_ESSID_MAX_SIZE+1])
  827. {
  828. hermes_t *hw = &priv->hw;
  829. int err = 0;
  830. struct hermes_idstring essidbuf;
  831. char *p = (char *)(&essidbuf.val);
  832. int len;
  833. TRACE_ENTER(priv->ndev.name);
  834. orinoco_lock(priv);
  835. if (strlen(priv->desired_essid) > 0) {
  836. /* We read the desired SSID from the hardware rather
  837.    than from priv->desired_essid, just in case the
  838.    firmware is allowed to change it on us. I'm not
  839.    sure about this */
  840. /* My guess is that the OWNSSID should always be whatever
  841.  * we set to the card, whereas CURRENT_SSID is the one that
  842.  * may change... - Jean II */
  843. u16 rid;
  844. *active = 1;
  845. rid = (priv->port_type == 3) ? HERMES_RID_CNFOWNSSID :
  846. HERMES_RID_CNFDESIREDSSID;
  847. err = hermes_read_ltv(hw, USER_BAP, rid, sizeof(essidbuf),
  848.       NULL, &essidbuf);
  849. if (err)
  850. goto fail_unlock;
  851. } else {
  852. *active = 0;
  853. err = hermes_read_ltv(hw, USER_BAP, HERMES_RID_CURRENTSSID,
  854.       sizeof(essidbuf), NULL, &essidbuf);
  855. if (err)
  856. goto fail_unlock;
  857. }
  858. len = le16_to_cpu(essidbuf.len);
  859. memset(buf, 0, sizeof(buf));
  860. memcpy(buf, p, len);
  861. buf[len] = '';
  862.  fail_unlock:
  863. orinoco_unlock(priv);
  864. TRACE_EXIT(priv->ndev.name);
  865. return err;       
  866. }
  867. static long orinoco_hw_get_freq(struct orinoco_private *priv)
  868. {
  869. hermes_t *hw = &priv->hw;
  870. int err = 0;
  871. u16 channel;
  872. long freq = 0;
  873. orinoco_lock(priv);
  874. err = hermes_read_wordrec(hw, USER_BAP, HERMES_RID_CURRENTCHANNEL, &channel);
  875. if (err)
  876. goto out;
  877. if ( (channel < 1) || (channel > NUM_CHANNELS) ) {
  878. struct net_device *dev = &priv->ndev;
  879. printk(KERN_WARNING "%s: Channel out of range (%d)!n", dev->name, channel);
  880. err = -EBUSY;
  881. goto out;
  882. }
  883. freq = channel_frequency[channel-1] * 100000;
  884.  out:
  885. orinoco_unlock(priv);
  886. if (err > 0)
  887. err = -EBUSY;
  888. return err ? err : freq;
  889. }
  890. static int orinoco_hw_get_bitratelist(struct orinoco_private *priv, int *numrates,
  891.     int32_t *rates, int max)
  892. {
  893. hermes_t *hw = &priv->hw;
  894. struct hermes_idstring list;
  895. unsigned char *p = (unsigned char *)&list.val;
  896. int err = 0;
  897. int num;
  898. int i;
  899. orinoco_lock(priv);
  900. err = hermes_read_ltv(hw, USER_BAP, HERMES_RID_SUPPORTEDDATARATES,
  901.       sizeof(list), NULL, &list);
  902. orinoco_unlock(priv);
  903. if (err)
  904. return err;
  905. num = le16_to_cpu(list.len);
  906. *numrates = num;
  907. num = min(num, max);
  908. for (i = 0; i < num; i++) {
  909. rates[i] = (p[i] & 0x7f) * 500000; /* convert to bps */
  910. }
  911. return 0;
  912. }
  913. #if 0
  914. #ifndef ORINOCO_DEBUG
  915. static inline void show_rx_frame(struct orinoco_rxframe_hdr *frame) {}
  916. #else
  917. static void show_rx_frame(struct orinoco_rxframe_hdr *frame)
  918. {
  919. printk(KERN_DEBUG "RX descriptor:n");
  920. printk(KERN_DEBUG "  status      = 0x%04xn", frame->desc.status);
  921. printk(KERN_DEBUG "  time        = 0x%08xn", frame->desc.time);
  922. printk(KERN_DEBUG "  silence     = 0x%02xn", frame->desc.silence);
  923. printk(KERN_DEBUG "  signal      = 0x%02xn", frame->desc.signal);
  924. printk(KERN_DEBUG "  rate        = 0x%02xn", frame->desc.rate);
  925. printk(KERN_DEBUG "  rxflow      = 0x%02xn", frame->desc.rxflow);
  926. printk(KERN_DEBUG "  reserved    = 0x%08xn", frame->desc.reserved);
  927. printk(KERN_DEBUG "IEEE 802.11 header:n");
  928. printk(KERN_DEBUG "  frame_ctl   = 0x%04xn",
  929.        frame->p80211.frame_ctl);
  930. printk(KERN_DEBUG "  duration_id = 0x%04xn",
  931.        frame->p80211.duration_id);
  932. printk(KERN_DEBUG "  addr1       = %02x:%02x:%02x:%02x:%02x:%02xn",
  933.        frame->p80211.addr1[0], frame->p80211.addr1[1],
  934.        frame->p80211.addr1[2], frame->p80211.addr1[3],
  935.        frame->p80211.addr1[4], frame->p80211.addr1[5]);
  936. printk(KERN_DEBUG "  addr2       = %02x:%02x:%02x:%02x:%02x:%02xn",
  937.        frame->p80211.addr2[0], frame->p80211.addr2[1],
  938.        frame->p80211.addr2[2], frame->p80211.addr2[3],
  939.        frame->p80211.addr2[4], frame->p80211.addr2[5]);
  940. printk(KERN_DEBUG "  addr3       = %02x:%02x:%02x:%02x:%02x:%02xn",
  941.        frame->p80211.addr3[0], frame->p80211.addr3[1],
  942.        frame->p80211.addr3[2], frame->p80211.addr3[3],
  943.        frame->p80211.addr3[4], frame->p80211.addr3[5]);
  944. printk(KERN_DEBUG "  seq_ctl     = 0x%04xn",
  945.        frame->p80211.seq_ctl);
  946. printk(KERN_DEBUG "  addr4       = %02x:%02x:%02x:%02x:%02x:%02xn",
  947.        frame->p80211.addr4[0], frame->p80211.addr4[1],
  948.        frame->p80211.addr4[2], frame->p80211.addr4[3],
  949.        frame->p80211.addr4[4], frame->p80211.addr4[5]);
  950. printk(KERN_DEBUG "  data_len    = 0x%04xn",
  951.        frame->p80211.data_len);
  952. printk(KERN_DEBUG "IEEE 802.3 header:n");
  953. printk(KERN_DEBUG "  dest        = %02x:%02x:%02x:%02x:%02x:%02xn",
  954.        frame->p8023.h_dest[0], frame->p8023.h_dest[1],
  955.        frame->p8023.h_dest[2], frame->p8023.h_dest[3],
  956.        frame->p8023.h_dest[4], frame->p8023.h_dest[5]);
  957. printk(KERN_DEBUG "  src         = %02x:%02x:%02x:%02x:%02x:%02xn",
  958.        frame->p8023.h_source[0], frame->p8023.h_source[1],
  959.        frame->p8023.h_source[2], frame->p8023.h_source[3],
  960.        frame->p8023.h_source[4], frame->p8023.h_source[5]);
  961. printk(KERN_DEBUG "  len         = 0x%04xn", frame->p8023.h_proto);
  962. printk(KERN_DEBUG "IEEE 802.2 LLC/SNAP header:n");
  963. printk(KERN_DEBUG "  DSAP        = 0x%02xn", frame->p8022.dsap);
  964. printk(KERN_DEBUG "  SSAP        = 0x%02xn", frame->p8022.ssap);
  965. printk(KERN_DEBUG "  ctrl        = 0x%02xn", frame->p8022.ctrl);
  966. printk(KERN_DEBUG "  OUI         = %02x:%02x:%02xn",
  967.        frame->p8022.oui[0], frame->p8022.oui[1], frame->p8022.oui[2]);
  968. printk(KERN_DEBUG "  ethertype  = 0x%04xn", frame->ethertype);
  969. }
  970. #endif
  971. #endif
  972. /*
  973.  * Interrupt handler
  974.  */
  975. void orinoco_interrupt(int irq, void * dev_id, struct pt_regs *regs)
  976. {
  977. struct orinoco_private *priv = (struct orinoco_private *) dev_id;
  978. hermes_t *hw = &priv->hw;
  979. struct net_device *dev = &priv->ndev;
  980. int count = MAX_IRQLOOPS_PER_IRQ;
  981. u16 evstat, events;
  982. /* These are used to detect a runaway interrupt situation */
  983. /* If we get more than MAX_IRQLOOPS_PER_JIFFY iterations in a jiffy,
  984.  * we panic and shut down the hardware */
  985. static int last_irq_jiffy = 0; /* jiffies value the last time we were called */
  986. static int loops_this_jiffy = 0;
  987. if (test_and_set_bit(ORINOCO_STATE_INIRQ, &priv->state))
  988. BUG();
  989. if (! orinoco_irqs_allowed(priv)) {
  990. clear_bit(ORINOCO_STATE_INIRQ, &priv->state);
  991. return;
  992. }
  993. DEBUG(3, "%s: orinoco_interrupt()n", priv->ndev.name);
  994. evstat = hermes_read_regn(hw, EVSTAT);
  995. events = evstat & hw->inten;
  996. if (! events) { /* Sometimes the card generates Tx interrupts without setting EVSTAT,
  997.    or so I've heard - FIXME does it really happen? */
  998. printk(KERN_WARNING "%s: Null event in orinoco_interrupt!n", priv->ndev.name);
  999. __orinoco_ev_alloc(priv, hw);
  1000. }
  1001. if (jiffies != last_irq_jiffy)
  1002. loops_this_jiffy = 0;
  1003. last_irq_jiffy = jiffies;
  1004. while (events && count--) {
  1005. DEBUG(4, "__orinoco_interrupt(): count=%d EVSTAT=0x%04xn",
  1006.       count, evstat);
  1007. if (++loops_this_jiffy > MAX_IRQLOOPS_PER_JIFFY) {
  1008. printk(KERN_CRIT "%s: IRQ handler is looping too 
  1009. much! Shutting down.n",
  1010.        dev->name);
  1011. /* Perform an emergency shutdown */
  1012. clear_bit(ORINOCO_STATE_DOIRQ, &priv->state);
  1013. hermes_set_irqmask(hw, 0);
  1014. break;
  1015. }
  1016. /* Check the card hasn't been removed */
  1017. if (! hermes_present(hw)) {
  1018. DEBUG(0, "orinoco_interrupt(): card removedn");
  1019. break;
  1020. }
  1021. if (events & HERMES_EV_TICK)
  1022. __orinoco_ev_tick(priv, hw);
  1023. if (events & HERMES_EV_WTERR)
  1024. __orinoco_ev_wterr(priv, hw);
  1025. if (events & HERMES_EV_INFDROP)
  1026. __orinoco_ev_infdrop(priv, hw);
  1027. if (events & HERMES_EV_INFO)
  1028. __orinoco_ev_info(priv, hw);
  1029. if (events & HERMES_EV_RX)
  1030. __orinoco_ev_rx(priv, hw);
  1031. if (events & HERMES_EV_TXEXC)
  1032. __orinoco_ev_txexc(priv, hw);
  1033. if (events & HERMES_EV_TX)
  1034. __orinoco_ev_tx(priv, hw);
  1035. if (events & HERMES_EV_ALLOC)
  1036. __orinoco_ev_alloc(priv, hw);
  1037. hermes_write_regn(hw, EVACK, events);
  1038. evstat = hermes_read_regn(hw, EVSTAT);
  1039. events = evstat & hw->inten;
  1040. };
  1041. clear_bit(ORINOCO_STATE_INIRQ, &priv->state);
  1042. }
  1043. static void __orinoco_ev_tick(struct orinoco_private *priv, hermes_t *hw)
  1044. {
  1045. printk(KERN_DEBUG "%s: TICKn", priv->ndev.name);
  1046. }
  1047. static void __orinoco_ev_wterr(struct orinoco_private *priv, hermes_t *hw)
  1048. {
  1049. /* This seems to happen a fair bit under load, but ignoring it
  1050.    seems to work fine...*/
  1051. DEBUG(1, "%s: MAC controller error (WTERR). Ignoring.n",
  1052.       priv->ndev.name);
  1053. }
  1054. static void __orinoco_ev_infdrop(struct orinoco_private *priv, hermes_t *hw)
  1055. {
  1056. printk(KERN_WARNING "%s: Information frame lost.n", priv->ndev.name);
  1057. }
  1058. static void __orinoco_ev_info(struct orinoco_private *priv, hermes_t *hw)
  1059. {
  1060. struct net_device *dev = &priv->ndev;
  1061. u16 infofid;
  1062. struct {
  1063. u16 len;
  1064. u16 type;
  1065. } __attribute__ ((packed)) info;
  1066. int err;
  1067. /* This is an answer to an INQUIRE command that we did earlier,
  1068.  * or an information "event" generated by the card
  1069.  * The controller return to us a pseudo frame containing
  1070.  * the information in question - Jean II */
  1071. infofid = hermes_read_regn(hw, INFOFID);
  1072. DEBUG(3, "%s: __dldwd_ev_info(): INFOFID=0x%04xn", dev->name,
  1073.       infofid);
  1074. /* Read the info frame header - don't try too hard */
  1075. err = hermes_bap_pread(hw, IRQ_BAP, &info, sizeof(info),
  1076.        infofid, 0);
  1077. if (err) {
  1078. printk(KERN_ERR "%s: error %d reading info frame. "
  1079.        "Frame dropped.n", dev->name, err);
  1080. return;
  1081. }
  1082. switch (le16_to_cpu(info.type)) {
  1083. case HERMES_INQ_TALLIES: {
  1084. struct hermes_tallies_frame tallies;
  1085. struct iw_statistics *wstats = &priv->wstats;
  1086. int len = le16_to_cpu(info.len) - 1;
  1087. if (len > (sizeof(tallies) / 2)) {
  1088. DEBUG(1, "%s: tallies frame too long.n", dev->name);
  1089. len = sizeof(tallies) / 2;
  1090. }
  1091. /* Read directly the data (no seek) */
  1092. hermes_read_words(hw, HERMES_DATA1, (void *) &tallies, len);
  1093. /* Increment our various counters */
  1094. /* wstats->discard.nwid - no wrong BSSID stuff */
  1095. wstats->discard.code +=
  1096. le16_to_cpu(tallies.RxWEPUndecryptable);
  1097. if (len == (sizeof(tallies) / 2))  
  1098. wstats->discard.code +=
  1099. le16_to_cpu(tallies.RxDiscards_WEPICVError) +
  1100. le16_to_cpu(tallies.RxDiscards_WEPExcluded);
  1101. wstats->discard.misc +=
  1102. le16_to_cpu(tallies.TxDiscardsWrongSA);
  1103. #if WIRELESS_EXT > 11
  1104. wstats->discard.fragment +=
  1105. le16_to_cpu(tallies.RxMsgInBadMsgFragments);
  1106. wstats->discard.retries +=
  1107. le16_to_cpu(tallies.TxRetryLimitExceeded);
  1108. /* wstats->miss.beacon - no match */
  1109. #if ORINOCO_DEBUG > 3
  1110. /* Hack for debugging - should not be taken as an example */
  1111. wstats->discard.nwid += le16_to_cpu(tallies.TxUnicastFrames);
  1112. wstats->miss.beacon += le16_to_cpu(tallies.RxUnicastFrames);
  1113. #endif
  1114. #endif /* WIRELESS_EXT > 11 */
  1115. }
  1116. break;
  1117. default:
  1118. DEBUG(1, "%s: Unknown information frame received (type %04x).n",
  1119.       priv->ndev.name, le16_to_cpu(info.type));
  1120. /* We don't actually do anything about it */
  1121. break;
  1122. }
  1123. }
  1124. static void __orinoco_ev_rx(struct orinoco_private *priv, hermes_t *hw)
  1125. {
  1126. struct net_device *dev = &priv->ndev;
  1127. struct net_device_stats *stats = &priv->stats;
  1128. struct iw_statistics *wstats = &priv->wstats;
  1129. struct sk_buff *skb = NULL;
  1130. u16 rxfid, status;
  1131. int length, data_len, data_off;
  1132. char *p;
  1133. struct orinoco_rxframe_hdr hdr;
  1134. struct ethhdr *eh;
  1135. int err;
  1136. rxfid = hermes_read_regn(hw, RXFID);
  1137. DEBUG(3, "__orinoco_ev_rx(): RXFID=0x%04xn", rxfid);
  1138. /* We read in the entire frame header here. This isn't really
  1139.    necessary, since we ignore most of it, but it's
  1140.    conceptually simpler. We can tune this later if
  1141.    necessary. */
  1142. err = hermes_bap_pread(hw, IRQ_BAP, &hdr, sizeof(hdr), rxfid, 0);
  1143. if (err) {
  1144. printk(KERN_ERR "%s: error %d reading frame header. "
  1145.        "Frame dropped.n", dev->name, err);
  1146. stats->rx_errors++;
  1147. goto drop;
  1148. }
  1149. status = le16_to_cpu(hdr.desc.status);
  1150. if (status & HERMES_RXSTAT_ERR) {
  1151. if ((status & HERMES_RXSTAT_ERR) == HERMES_RXSTAT_BADCRC) {
  1152. stats->rx_crc_errors++;
  1153. DEBUG(1, "%s: Bad CRC on Rx. Frame dropped.n", dev->name);
  1154. } else if ((status & HERMES_RXSTAT_ERR)
  1155.    == HERMES_RXSTAT_UNDECRYPTABLE) {
  1156. wstats->discard.code++;
  1157. printk(KERN_WARNING "%s: Undecryptable frame on Rx. Frame dropped.n",
  1158.        dev->name);
  1159. } else {
  1160. wstats->discard.misc++;
  1161. printk("%s: Unknown Rx error (0x%x). Frame dropped.n",
  1162.        dev->name, status & HERMES_RXSTAT_ERR);
  1163. }
  1164. stats->rx_errors++;
  1165. goto drop;
  1166. }
  1167. length = le16_to_cpu(hdr.p80211.data_len);
  1168. /* Yes, you heard right, that's le16. 802.2 and 802.3 are
  1169.    big-endian, but 802.11 is little-endian believe it or
  1170.    not. */
  1171. /* Correct. 802.3 is big-endian byte order and little endian bit
  1172.  * order, whereas 802.11 is little endian for both byte and bit
  1173.  * order. That's specified in the 802.11 spec. - Jean II */
  1174. /* Sanity check */
  1175. if (length > MAX_FRAME_SIZE) {
  1176. printk(KERN_WARNING "%s: Oversized frame received (%d bytes)n",
  1177.        dev->name, length);
  1178. stats->rx_length_errors++;
  1179. stats->rx_errors++;
  1180. goto drop;
  1181. }
  1182. /* We need space for the packet data itself, plus an ethernet
  1183.    header, plus 2 bytes so we can align the IP header on a
  1184.    32bit boundary, plus 1 byte so we can read in odd length
  1185.    packets from the card, which has an IO granularity of 16
  1186.    bits */  
  1187. skb = dev_alloc_skb(length+ETH_HLEN+2+1);
  1188. if (!skb) {
  1189. printk(KERN_WARNING "%s: Can't allocate skb for Rxn",
  1190.        dev->name);
  1191. stats->rx_dropped++;
  1192. goto drop;
  1193. }
  1194. skb_reserve(skb, 2); /* This way the IP header is aligned */
  1195. /* Handle decapsulation
  1196.  * In most cases, the firmware tell us about SNAP frames.
  1197.  * For some reason, the SNAP frames sent by LinkSys APs
  1198.  * are not properly recognised by most firmwares.
  1199.  * So, check ourselves (note : only 3 bytes out of 6).
  1200.  */
  1201. if(((status & HERMES_RXSTAT_MSGTYPE) == HERMES_RXSTAT_1042) ||
  1202.    ((status & HERMES_RXSTAT_MSGTYPE) == HERMES_RXSTAT_TUNNEL) ||
  1203.    (!memcmp(&hdr.p8022, &encaps_hdr, 3))) {
  1204. /* These indicate a SNAP within 802.2 LLC within
  1205.    802.11 frame which we'll need to de-encapsulate to
  1206.    the original EthernetII frame. */
  1207. /* Remove SNAP header, reconstruct EthernetII frame */
  1208. data_len = length - ENCAPS_OVERHEAD;
  1209. data_off = sizeof(hdr);
  1210. eh = (struct ethhdr *)skb_put(skb, ETH_HLEN);
  1211. memcpy(eh, &hdr.p8023, sizeof(hdr.p8023));
  1212. eh->h_proto = hdr.ethertype;
  1213. } else {
  1214. /* All other cases indicate a genuine 802.3 frame.
  1215.  * No decapsulation needed */
  1216. /* Otherwise, we just throw the whole thing in,
  1217.  * and hope the protocol layer can deal with it
  1218.  * as 802.3 */
  1219. data_len = length;
  1220. data_off = P8023_OFFSET;
  1221. }
  1222. p = skb_put(skb, data_len);
  1223. err = hermes_bap_pread(hw, IRQ_BAP, p, RUP_EVEN(data_len),
  1224.        rxfid, data_off);
  1225. if (err) {
  1226. if (err == -EIO)
  1227. DEBUG(1, "%s: EIO reading frame header.n", dev->name);
  1228. else
  1229. printk(KERN_ERR "%s: error %d reading frame header. "
  1230.        "Frame dropped.n", dev->name, err);
  1231. stats->rx_errors++;
  1232. goto drop;
  1233. }
  1234. dev->last_rx = jiffies;
  1235. skb->dev = dev;
  1236. skb->protocol = eth_type_trans(skb, dev);
  1237. skb->ip_summed = CHECKSUM_NONE;
  1238. /* Process the wireless stats if needed */
  1239. orinoco_stat_gather(dev, skb, &hdr);
  1240. /* Pass the packet to the networking stack */
  1241. netif_rx(skb);
  1242. stats->rx_packets++;
  1243. stats->rx_bytes += length;
  1244. return;
  1245.  drop:
  1246. if (skb)
  1247. dev_kfree_skb_irq(skb);
  1248. return;
  1249. }
  1250. static void __orinoco_ev_txexc(struct orinoco_private *priv, hermes_t *hw)
  1251. {
  1252. struct net_device *dev = &priv->ndev;
  1253. struct net_device_stats *stats = &priv->stats;
  1254. u16 fid = hermes_read_regn(hw, TXCOMPLFID);
  1255. struct hermes_tx_descriptor desc;
  1256. int err = 0;
  1257. if (fid == DUMMY_FID)
  1258. return; /* Nothing's really happened */
  1259. err = hermes_bap_pread(hw, USER_BAP, &desc, sizeof(desc), fid, 0);
  1260. if (err) {
  1261. printk(KERN_WARNING "%s: Unable to read descriptor on Tx error "
  1262.        "(FID=%04X error %d)n",
  1263.        dev->name, fid, err);
  1264. } else {
  1265. printk(KERN_INFO "%s: Tx error, status %d (FID=%04X)n",
  1266.        dev->name, le16_to_cpu(desc.status), fid);
  1267. }
  1268. stats->tx_errors++;
  1269. netif_wake_queue(dev);
  1270. hermes_write_regn(hw, TXCOMPLFID, DUMMY_FID);
  1271. }
  1272. static void __orinoco_ev_tx(struct orinoco_private *priv, hermes_t *hw)
  1273. {
  1274. struct net_device *dev = &priv->ndev;
  1275. struct net_device_stats *stats = &priv->stats;
  1276. /*   u16 fid = hermes_read_regn(hw, TXCOMPLFID); */
  1277. /* We don't generally use the Tx event (to cut down on
  1278.    interrupts) - we do the transmit complet processing once
  1279.    the transmit buffer is reclaimed in __orinoco_ev_alloc() ,
  1280.    hence nothing here */
  1281. /*   DEBUG(2, "%s: Transmit completed (FID=%04X)n", priv->ndev.name, fid); */
  1282. stats->tx_packets++;
  1283. netif_wake_queue(dev);
  1284. hermes_write_regn(hw, TXCOMPLFID, DUMMY_FID);
  1285. }
  1286. static void __orinoco_ev_alloc(struct orinoco_private *priv, hermes_t *hw)
  1287. {
  1288. struct net_device *dev = &priv->ndev;
  1289. u16 fid = hermes_read_regn(hw, ALLOCFID);
  1290. DEBUG(3, "%s: Allocation complete FID=0x%04xn", priv->ndev.name, fid);
  1291. /* We don't generally request Tx complete events to cut down
  1292.    on the number of interrupts, so we do trasmit complete
  1293.    processing here, which happens once the firmware is done
  1294.    with the transmit buffer */
  1295. if (fid != priv->txfid) {
  1296. if (fid != DUMMY_FID)
  1297. printk(KERN_WARNING "%s: Allocate event on unexpected fid (%04X)n",
  1298.        dev->name, fid);
  1299. return;
  1300. }
  1301. hermes_write_regn(hw, ALLOCFID, DUMMY_FID);
  1302. }
  1303. static void determine_firmware(struct net_device *dev)
  1304. {
  1305. struct orinoco_private *priv = dev->priv;
  1306. hermes_t *hw = &priv->hw;
  1307. int err;
  1308. struct sta_id {
  1309. u16 id, vendor, major, minor;
  1310. } __attribute__ ((packed)) sta_id;
  1311. u32 firmver;
  1312. /* Get the firmware version */
  1313. err = HERMES_READ_RECORD(hw, USER_BAP, HERMES_RID_STAID, &sta_id);
  1314. if (err) {
  1315. printk(KERN_WARNING "%s: Error %d reading firmware info. Wildly guessing capabilities...n",
  1316.        dev->name, err);
  1317. memset(&sta_id, 0, sizeof(sta_id));
  1318. }
  1319. le16_to_cpus(&sta_id.id);
  1320. le16_to_cpus(&sta_id.vendor);
  1321. le16_to_cpus(&sta_id.major);
  1322. le16_to_cpus(&sta_id.minor);
  1323. firmver = ((u32)sta_id.major << 16) | sta_id.minor;
  1324. printk(KERN_DEBUG "%s: Station identity %04x:%04x:%04x:%04xn",
  1325.        dev->name, sta_id.id, sta_id.vendor,
  1326.        sta_id.major, sta_id.minor);
  1327. /* Determine capabilities from the firmware version */
  1328. if (sta_id.vendor == 1) {
  1329. /* Lucent Wavelan IEEE, Lucent Orinoco, Cabletron RoamAbout,
  1330.    ELSA, Melco, HP, IBM, Dell 1150, Compaq 110/210 */
  1331. printk(KERN_DEBUG "%s: Looks like a Lucent/Agere firmware "
  1332.        "version %d.%02dn", dev->name,
  1333.        sta_id.major, sta_id.minor);
  1334. priv->firmware_type = FIRMWARE_TYPE_AGERE;
  1335. priv->need_card_reset = 0;
  1336. priv->broken_allocate = 0;
  1337. priv->has_port3 = 1; /* Still works in 7.28 */
  1338. priv->has_ibss = (firmver >= 0x60006);
  1339. priv->has_ibss_any = (firmver >= 0x60010);
  1340. priv->has_wep = (firmver >= 0x40020);
  1341. priv->has_big_wep = 1; /* FIXME: this is wrong - how do we tell
  1342.   Gold cards from the others? */
  1343. priv->has_mwo = (firmver >= 0x60000);
  1344. priv->has_pm = (firmver >= 0x40020); /* Don't work in 7.52 ? */
  1345. priv->has_preamble = 0;
  1346. priv->ibss_port = 1;
  1347. /* Tested with Agere firmware :
  1348.  * 1.16 ; 4.08 ; 4.52 ; 6.04 ; 6.16 ; 7.28 => Jean II
  1349.  * Tested CableTron firmware : 4.32 => Anton */
  1350. } else if ((sta_id.vendor == 2) &&
  1351.    ((firmver == 0x10001) || (firmver == 0x20001))) {
  1352. /* Symbol , 3Com AirConnect, Intel, Ericsson WLAN */
  1353. /* Intel MAC : 00:02:B3:* */
  1354. /* 3Com MAC : 00:50:DA:* */
  1355. char tmp[SYMBOL_MAX_VER_LEN+1];
  1356. memset(tmp, 0, sizeof(tmp));
  1357. /* Get the Symbol firmware version */
  1358. err = hermes_read_ltv(hw, USER_BAP,
  1359.       HERMES_RID_SECONDARYVERSION_SYMBOL,
  1360.       SYMBOL_MAX_VER_LEN, NULL, &tmp);
  1361. if (err) {
  1362. printk(KERN_WARNING
  1363.        "%s: Error %d reading Symbol firmware info. Wildly guessing capabilities...n",
  1364.        dev->name, err);
  1365. firmver = 0;
  1366. tmp[0] = '';
  1367. } else {
  1368. /* The firmware revision is a string, the format is
  1369.  * something like : "V2.20-01".
  1370.  * Quick and dirty parsing... - Jean II
  1371.  */
  1372. firmver = ((tmp[1] - '0') << 16) | ((tmp[3] - '0') << 12)
  1373. | ((tmp[4] - '0') << 8) | ((tmp[6] - '0') << 4)
  1374. | (tmp[7] - '0');
  1375. tmp[SYMBOL_MAX_VER_LEN] = '';
  1376. }
  1377. printk(KERN_DEBUG "%s: Looks like a Symbol firmware "
  1378.        "version [%s] (parsing to %X)n", dev->name,
  1379.        tmp, firmver);
  1380. priv->firmware_type = FIRMWARE_TYPE_SYMBOL;
  1381. priv->need_card_reset = 1;
  1382. priv->broken_allocate = 1;
  1383. priv->has_port3 = 1;
  1384. priv->has_ibss = (firmver >= 0x20000);
  1385. priv->has_wep = (firmver >= 0x15012);
  1386. priv->has_big_wep = (firmver >= 0x20000);
  1387. priv->has_mwo = 0;
  1388. priv->has_pm = (firmver >= 0x20000) && (firmver < 0x22000);
  1389. priv->has_preamble = (firmver >= 0x20000);
  1390. priv->ibss_port = 4;
  1391. /* Tested with Intel firmware : 0x20015 => Jean II */
  1392. /* Tested with 3Com firmware : 0x15012 & 0x22001 => Jean II */
  1393. } else {
  1394. /* D-Link, Linksys, Adtron, ZoomAir, and many others...
  1395.  * Samsung, Compaq 100/200 and Proxim are slightly
  1396.  * different and less well tested */
  1397. /* D-Link MAC : 00:40:05:* */
  1398. /* Addtron MAC : 00:90:D1:* */
  1399. printk(KERN_DEBUG "%s: Looks like an Intersil firmware "
  1400.        "version %d.%02dn", dev->name,
  1401.        sta_id.major, sta_id.minor);
  1402. priv->firmware_type = FIRMWARE_TYPE_INTERSIL;
  1403. priv->need_card_reset = 0;
  1404. priv->broken_allocate = 0;
  1405. priv->has_port3 = 1;
  1406. priv->has_ibss = (firmver >= 0x00007); /* FIXME */
  1407. priv->has_wep = (firmver >= 0x00008);
  1408. priv->has_big_wep = priv->has_wep;
  1409. priv->has_mwo = 0;
  1410. priv->has_pm = (firmver >= 0x00007);
  1411. priv->has_preamble = 0;
  1412. if (firmver >= 0x00008)
  1413. priv->ibss_port = 0;
  1414. else {
  1415. printk(KERN_NOTICE "%s: Intersil firmware earlier "
  1416.        "than v0.08 - several features not supportedn",
  1417.        dev->name);
  1418. priv->ibss_port = 1;
  1419. }
  1420. }
  1421. }
  1422. /*
  1423.  * struct net_device methods
  1424.  */
  1425. int
  1426. orinoco_init(struct net_device *dev)
  1427. {
  1428. struct orinoco_private *priv = dev->priv;
  1429. hermes_t *hw = &priv->hw;
  1430. int err = 0;
  1431. struct hermes_idstring nickbuf;
  1432. u16 reclen;
  1433. int len;
  1434. TRACE_ENTER("orinoco");
  1435. orinoco_lock(priv);
  1436. /* Do standard firmware reset */
  1437. err = hermes_reset(hw);
  1438. if (err != 0) {
  1439. printk(KERN_ERR "%s: failed to reset hardware (err = %d)n",
  1440.        dev->name, err);
  1441. goto out;
  1442. }
  1443. determine_firmware(dev);
  1444. if (priv->has_port3)
  1445. printk(KERN_DEBUG "%s: Ad-hoc demo mode supportedn", dev->name);
  1446. if (priv->has_ibss)
  1447. printk(KERN_DEBUG "%s: IEEE standard IBSS ad-hoc mode supportedn",
  1448.        dev->name);
  1449. if (priv->has_wep) {
  1450. printk(KERN_DEBUG "%s: WEP supported, ", dev->name);
  1451. if (priv->has_big_wep)
  1452. printk("104-bit keyn");
  1453. else
  1454. printk("40-bit keyn");
  1455. }
  1456. /* Get the MAC address */
  1457. err = hermes_read_ltv(hw, USER_BAP, HERMES_RID_CNFOWNMACADDR,
  1458.       ETH_ALEN, NULL, dev->dev_addr);
  1459. if (err) {
  1460. printk(KERN_WARNING "%s: failed to read MAC address!n",
  1461.        dev->name);
  1462. goto out;
  1463. }
  1464. printk(KERN_DEBUG "%s: MAC address %02X:%02X:%02X:%02X:%02X:%02Xn",
  1465.        dev->name, dev->dev_addr[0], dev->dev_addr[1],
  1466.        dev->dev_addr[2], dev->dev_addr[3], dev->dev_addr[4],
  1467.        dev->dev_addr[5]);
  1468. /* Get the station name */
  1469. err = hermes_read_ltv(hw, USER_BAP, HERMES_RID_CNFOWNNAME,
  1470.       sizeof(nickbuf), &reclen, &nickbuf);
  1471. if (err) {
  1472. printk(KERN_ERR "%s: failed to read station namen",
  1473.        dev->name);
  1474. goto out;
  1475. }
  1476. if (nickbuf.len)
  1477. len = min(IW_ESSID_MAX_SIZE, (int)le16_to_cpu(nickbuf.len));
  1478. else
  1479. len = min(IW_ESSID_MAX_SIZE, 2 * reclen);
  1480. memcpy(priv->nick, &nickbuf.val, len);
  1481. priv->nick[len] = '';
  1482. printk(KERN_DEBUG "%s: Station name "%s"n", dev->name, priv->nick);
  1483. /* Get allowed channels */
  1484. err = hermes_read_wordrec(hw, USER_BAP, HERMES_RID_CHANNELLIST,
  1485.   &priv->channel_mask);
  1486. if (err) {
  1487. printk(KERN_ERR "%s: failed to read channel list!n",
  1488.        dev->name);
  1489. goto out;
  1490. }
  1491. /* Get initial AP density */
  1492. err = hermes_read_wordrec(hw, USER_BAP, HERMES_RID_CNFSYSTEMSCALE, &priv->ap_density);
  1493. if (err) {
  1494. printk(KERN_ERR "%s: failed to read AP density!n", dev->name);
  1495. goto out;
  1496. }
  1497. /* Get initial RTS threshold */
  1498. err = hermes_read_wordrec(hw, USER_BAP, HERMES_RID_CNFRTSTHRESHOLD,
  1499.   &priv->rts_thresh);
  1500. if (err) {
  1501. printk(KERN_ERR "%s: failed to read RTS threshold!n", dev->name);
  1502. goto out;
  1503. }
  1504. /* Get initial fragmentation settings */
  1505. if (priv->has_mwo)
  1506. err = hermes_read_wordrec(hw, USER_BAP,
  1507.   HERMES_RID_CNFMWOROBUST_AGERE,
  1508.   &priv->mwo_robust);
  1509. else
  1510. err = hermes_read_wordrec(hw, USER_BAP, HERMES_RID_CNFFRAGMENTATIONTHRESHOLD,
  1511.   &priv->frag_thresh);
  1512. if (err) {
  1513. printk(KERN_ERR "%s: failed to read fragmentation settings!n", dev->name);
  1514. goto out;
  1515. }
  1516. /* Power management setup */
  1517. if (priv->has_pm) {
  1518. priv->pm_on = 0;
  1519. priv->pm_mcast = 1;
  1520. err = hermes_read_wordrec(hw, USER_BAP,
  1521.   HERMES_RID_CNFMAXSLEEPDURATION,
  1522.   &priv->pm_period);
  1523. if (err) {
  1524. printk(KERN_ERR "%s: failed to read power management period!n",
  1525.        dev->name);
  1526. goto out;
  1527. }
  1528. err = hermes_read_wordrec(hw, USER_BAP,
  1529.   HERMES_RID_CNFPMHOLDOVERDURATION,
  1530.   &priv->pm_timeout);
  1531. if (err) {
  1532. printk(KERN_ERR "%s: failed to read power management timeout!n",
  1533.        dev->name);
  1534. goto out;
  1535. }
  1536. }
  1537. /* Preamble setup */
  1538. if (priv->has_preamble) {
  1539. err = hermes_read_wordrec(hw, USER_BAP, HERMES_RID_CNFPREAMBLE_SYMBOL,
  1540.   &priv->preamble);
  1541. if (err)
  1542. goto out;
  1543. }
  1544. /* Set up the default configuration */
  1545. priv->iw_mode = IW_MODE_INFRA;
  1546. /* By default use IEEE/IBSS ad-hoc mode if we have it */
  1547. priv->prefer_port3 = priv->has_port3 && (! priv->has_ibss);
  1548. set_port_type(priv);
  1549. priv->promiscuous = 0;
  1550. priv->wep_on = 0;
  1551. priv->tx_key = 0;
  1552. printk(KERN_DEBUG "%s: readyn", dev->name);
  1553.  out:
  1554. orinoco_unlock(priv);
  1555. TRACE_EXIT("orinoco");
  1556. return err;
  1557. }
  1558. struct net_device_stats *
  1559. orinoco_get_stats(struct net_device *dev)
  1560. {
  1561. struct orinoco_private *priv = (struct orinoco_private *)dev->priv;
  1562. return &priv->stats;
  1563. }
  1564. struct iw_statistics *
  1565. orinoco_get_wireless_stats(struct net_device *dev)
  1566. {
  1567. struct orinoco_private *priv = (struct orinoco_private *)dev->priv;
  1568. hermes_t *hw = &priv->hw;
  1569. struct iw_statistics *wstats = &priv->wstats;
  1570. int err = 0;
  1571. if (! netif_device_present(dev))
  1572. return NULL; /* FIXME: We may be able to do better than this */
  1573. orinoco_lock(priv);
  1574. if (priv->iw_mode == IW_MODE_ADHOC) {
  1575. memset(&wstats->qual, 0, sizeof(wstats->qual));
  1576. /* If a spy address is defined, we report stats of the
  1577.  * first spy address - Jean II */
  1578. if (SPY_NUMBER(priv)) {
  1579. wstats->qual.qual = priv->spy_stat[0].qual;
  1580. wstats->qual.level = priv->spy_stat[0].level;
  1581. wstats->qual.noise = priv->spy_stat[0].noise;
  1582. wstats->qual.updated = priv->spy_stat[0].updated;
  1583. }
  1584. } else {
  1585. orinoco_commsqual_t cq;
  1586. err = HERMES_READ_RECORD(hw, USER_BAP,
  1587.  HERMES_RID_COMMSQUALITY, &cq);
  1588. DEBUG(3, "%s: Global stats = %X-%X-%Xn", dev->name,
  1589.       cq.qual, cq.signal, cq.noise);
  1590. wstats->qual.qual = (int)le16_to_cpu(cq.qual);
  1591. wstats->qual.level = (int)le16_to_cpu(cq.signal) - 0x95;
  1592. wstats->qual.noise = (int)le16_to_cpu(cq.noise) - 0x95;
  1593. wstats->qual.updated = 7;
  1594. }
  1595. /* We can't really wait for the tallies inquiry command to
  1596.  * complete, so we just use the previous results and trigger
  1597.  * a new tallies inquiry command for next time - Jean II */
  1598. /* FIXME: Hmm.. seems a bit ugly, I wonder if there's a way to
  1599.    do better - dgibson */
  1600. err = hermes_inquire(hw, HERMES_INQ_TALLIES);
  1601.                
  1602. orinoco_unlock(priv);
  1603. if (err)
  1604. return NULL;
  1605. return wstats;
  1606. }
  1607. static inline void orinoco_spy_gather(struct net_device *dev, u_char *mac,
  1608.     int level, int noise)
  1609. {
  1610. struct orinoco_private *priv = (struct orinoco_private *)dev->priv;
  1611. int i;
  1612. /* Gather wireless spy statistics: for each packet, compare the
  1613.  * source address with out list, and if match, get the stats... */
  1614. for (i = 0; i < priv->spy_number; i++)
  1615. if (!memcmp(mac, priv->spy_address[i], ETH_ALEN)) {
  1616. priv->spy_stat[i].level = level - 0x95;
  1617. priv->spy_stat[i].noise = noise - 0x95;
  1618. priv->spy_stat[i].qual = level - noise;
  1619. priv->spy_stat[i].updated = 7;
  1620. }
  1621. }
  1622. void
  1623. orinoco_stat_gather( struct net_device *dev,
  1624.    struct sk_buff *skb,
  1625.    struct orinoco_rxframe_hdr *hdr)
  1626. {
  1627. struct orinoco_private *priv = (struct orinoco_private *)dev->priv;
  1628. /* Using spy support with lots of Rx packets, like in an
  1629.  * infrastructure (AP), will really slow down everything, because
  1630.  * the MAC address must be compared to each entry of the spy list.
  1631.  * If the user really asks for it (set some address in the
  1632.  * spy list), we do it, but he will pay the price.
  1633.  * Note that to get here, you need both WIRELESS_SPY
  1634.  * compiled in AND some addresses in the list !!!
  1635.  */
  1636. /* Note : gcc will optimise the whole section away if
  1637.  * WIRELESS_SPY is not defined... - Jean II */
  1638. if (SPY_NUMBER(priv)) {
  1639. orinoco_spy_gather(dev, skb->mac.raw + ETH_ALEN,
  1640.    hdr->desc.signal, hdr->desc.silence);
  1641. }
  1642. }
  1643. int
  1644. orinoco_xmit(struct sk_buff *skb, struct net_device *dev)
  1645. {
  1646. struct orinoco_private *priv = (struct orinoco_private *)dev->priv;
  1647. struct net_device_stats *stats = &priv->stats;
  1648. hermes_t *hw = &priv->hw;
  1649. int err = 0;
  1650. u16 txfid = priv->txfid;
  1651. char *p;
  1652. struct ethhdr *eh;
  1653. int len, data_len, data_off;
  1654. struct orinoco_txframe_hdr hdr;
  1655. hermes_response_t resp;
  1656. if (! netif_running(dev)) {
  1657. printk(KERN_ERR "%s: Tx on stopped device!n",
  1658.        dev->name);
  1659. return 1;
  1660. }
  1661. if (netif_queue_stopped(dev)) {
  1662. printk(KERN_ERR "%s: Tx while transmitter busy!n", 
  1663.        dev->name);
  1664. return 1;
  1665. }
  1666. orinoco_lock(priv);
  1667. /* Length of the packet body */
  1668. /* FIXME: what if the skb is smaller than this? */
  1669. len = max_t(int,skb->len - ETH_HLEN, ETH_ZLEN);
  1670. eh = (struct ethhdr *)skb->data;
  1671. /* Build the IEEE 802.11 header */
  1672. memset(&hdr, 0, sizeof(hdr));
  1673. memcpy(hdr.p80211.addr1, eh->h_dest, ETH_ALEN);
  1674. memcpy(hdr.p80211.addr2, eh->h_source, ETH_ALEN);
  1675. hdr.p80211.frame_ctl = IEEE802_11_FTYPE_DATA;
  1676. /* Request an interrupt on Tx failures, but not sucesses (we
  1677.            use the buffer reclaim allocation event instead */
  1678. hdr.desc.tx_control = cpu_to_le16(HERMES_TXCTRL_TX_EX | HERMES_TXCTRL_TX_OK);
  1679. /* Encapsulate Ethernet-II frames */
  1680. if (ntohs(eh->h_proto) > 1500) { /* Ethernet-II frame */
  1681. data_len = len;
  1682. data_off = sizeof(hdr);
  1683. p = skb->data + ETH_HLEN;
  1684. /* 802.11 header */
  1685. hdr.p80211.data_len = cpu_to_le16(data_len + ENCAPS_OVERHEAD);
  1686. /* 802.3 header */
  1687. memcpy(hdr.p8023.h_dest, eh->h_dest, ETH_ALEN);
  1688. memcpy(hdr.p8023.h_source, eh->h_source, ETH_ALEN);
  1689. hdr.p8023.h_proto = htons(data_len + ENCAPS_OVERHEAD);
  1690. /* 802.2 header */
  1691. memcpy(&hdr.p8022, &encaps_hdr, sizeof(encaps_hdr));
  1692. hdr.ethertype = eh->h_proto;
  1693. err  = hermes_bap_pwrite(hw, USER_BAP, &hdr, sizeof(hdr),
  1694.  txfid, 0);
  1695. if (err) {
  1696. printk(KERN_ERR "%s: Error %d writing packet header to BAPn",
  1697.        dev->name, err);
  1698. stats->tx_errors++;
  1699. goto fail;
  1700. }
  1701. } else { /* IEEE 802.3 frame */
  1702. data_len = len + ETH_HLEN;
  1703. data_off = P8023_OFFSET;
  1704. p = skb->data;
  1705. /* 802.11 header */
  1706. hdr.p80211.data_len = cpu_to_le16(len);
  1707. err = hermes_bap_pwrite(hw, USER_BAP, &hdr, P8023_OFFSET,
  1708. txfid, 0);
  1709. if (err) {
  1710. printk(KERN_ERR
  1711.        "%s: Error %d writing packet header to BAPn",
  1712.        dev->name, err);
  1713. stats->tx_errors++;
  1714. goto fail;
  1715. }
  1716. }
  1717. /* Round up for odd length packets */
  1718. err = hermes_bap_pwrite(hw, USER_BAP, p, RUP_EVEN(data_len), txfid, data_off);
  1719. if (err) {
  1720. printk(KERN_ERR "%s: Error %d writing packet header to BAPn",
  1721.        dev->name, err);
  1722. stats->tx_errors++;
  1723. goto fail;
  1724. }
  1725. /* Finally, we actually initiate the send */
  1726. err = hermes_docmd_wait(hw, HERMES_CMD_TX | HERMES_CMD_RECL, txfid, &resp);
  1727. if (err) {
  1728. printk(KERN_ERR "%s: Error %d transmitting packetn", dev->name, err);
  1729. stats->tx_errors++;
  1730. goto fail;
  1731. }
  1732. atomic_inc(&priv->queue_length);
  1733. dev->trans_start = jiffies;
  1734. stats->tx_bytes += data_off + data_len;
  1735. netif_stop_queue(dev);
  1736. orinoco_unlock(priv);
  1737. dev_kfree_skb(skb);
  1738. return 0;
  1739.  fail:
  1740. orinoco_unlock(priv);
  1741. return err;
  1742. }
  1743. void
  1744. orinoco_tx_timeout(struct net_device *dev)
  1745. {
  1746. struct orinoco_private *priv = (struct orinoco_private *)dev->priv;
  1747. struct net_device_stats *stats = &priv->stats;
  1748. int err = 0;
  1749. printk(KERN_WARNING "%s: Tx timeout! Resetting card.n", dev->name);
  1750. stats->tx_errors++;
  1751. err = orinoco_reset(priv);
  1752. if (err)
  1753. printk(KERN_ERR "%s: Error %d resetting card on Tx timeout!n",
  1754.        dev->name, err);
  1755. else {
  1756. dev->trans_start = jiffies;
  1757. netif_wake_queue(dev);
  1758. }
  1759. }
  1760. static int orinoco_ioctl_getiwrange(struct net_device *dev, struct iw_point *rrq)
  1761. {
  1762. struct orinoco_private *priv = dev->priv;
  1763. int err = 0;
  1764. int mode;
  1765. struct iw_range range;
  1766. int numrates;
  1767. int i, k;
  1768. TRACE_ENTER(dev->name);
  1769. err = verify_area(VERIFY_WRITE, rrq->pointer, sizeof(range));
  1770. if (err)
  1771. return err;
  1772. rrq->length = sizeof(range);
  1773. orinoco_lock(priv);
  1774. mode = priv->iw_mode;
  1775. orinoco_unlock(priv);
  1776. memset(&range, 0, sizeof(range));
  1777. /* Much of this shamelessly taken from wvlan_cs.c. No idea
  1778.  * what it all means -dgibson */
  1779. #if WIRELESS_EXT > 10
  1780. range.we_version_compiled = WIRELESS_EXT;
  1781. range.we_version_source = 11;
  1782. #endif /* WIRELESS_EXT > 10 */
  1783. range.min_nwid = range.max_nwid = 0; /* We don't use nwids */
  1784. /* Set available channels/frequencies */
  1785. range.num_channels = NUM_CHANNELS;
  1786. k = 0;
  1787. for (i = 0; i < NUM_CHANNELS; i++) {
  1788. if (priv->channel_mask & (1 << i)) {
  1789. range.freq[k].i = i + 1;
  1790. range.freq[k].m = channel_frequency[i] * 100000;
  1791. range.freq[k].e = 1;
  1792. k++;
  1793. }
  1794. if (k >= IW_MAX_FREQUENCIES)
  1795. break;
  1796. }
  1797. range.num_frequency = k;
  1798. range.sensitivity = 3;
  1799. if ((mode == IW_MODE_ADHOC) && (priv->spy_number == 0)){
  1800. /* Quality stats meaningless in ad-hoc mode */
  1801. range.max_qual.qual = 0;
  1802. range.max_qual.level = 0;
  1803. range.max_qual.noise = 0;
  1804. #if WIRELESS_EXT > 11
  1805. range.avg_qual.qual = 0;
  1806. range.avg_qual.level = 0;
  1807. range.avg_qual.noise = 0;
  1808. #endif /* WIRELESS_EXT > 11 */
  1809. } else {
  1810. range.max_qual.qual = 0x8b - 0x2f;
  1811. range.max_qual.level = 0x2f - 0x95 - 1;
  1812. range.max_qual.noise = 0x2f - 0x95 - 1;
  1813. #if WIRELESS_EXT > 11
  1814. /* Need to get better values */
  1815. range.avg_qual.qual = 0x24;
  1816. range.avg_qual.level = 0xC2;
  1817. range.avg_qual.noise = 0x9E;
  1818. #endif /* WIRELESS_EXT > 11 */
  1819. }
  1820. err = orinoco_hw_get_bitratelist(priv, &numrates,
  1821.        range.bitrate, IW_MAX_BITRATES);
  1822. if (err)
  1823. return err;
  1824. range.num_bitrates = numrates;
  1825. /* Set an indication of the max TCP throughput in bit/s that we can
  1826.  * expect using this interface. May be use for QoS stuff...
  1827.  * Jean II */
  1828. if(numrates > 2)
  1829. range.throughput = 5 * 1000 * 1000; /* ~5 Mb/s */
  1830. else
  1831. range.throughput = 1.5 * 1000 * 1000; /* ~1.5 Mb/s */
  1832. range.min_rts = 0;
  1833. range.max_rts = 2347;
  1834. range.min_frag = 256;
  1835. range.max_frag = 2346;
  1836. orinoco_lock(priv);
  1837. if (priv->has_wep) {
  1838. range.max_encoding_tokens = ORINOCO_MAX_KEYS;
  1839. range.encoding_size[0] = SMALL_KEY_SIZE;
  1840. range.num_encoding_sizes = 1;
  1841. if (priv->has_big_wep) {
  1842. range.encoding_size[1] = LARGE_KEY_SIZE;
  1843. range.num_encoding_sizes = 2;
  1844. }
  1845. } else {
  1846. range.num_encoding_sizes = 0;
  1847. range.max_encoding_tokens = 0;
  1848. }
  1849. orinoco_unlock(priv);
  1850. range.min_pmp = 0;
  1851. range.max_pmp = 65535000;
  1852. range.min_pmt = 0;
  1853. range.max_pmt = 65535 * 1000; /* ??? */
  1854. range.pmp_flags = IW_POWER_PERIOD;
  1855. range.pmt_flags = IW_POWER_TIMEOUT;
  1856. range.pm_capa = IW_POWER_PERIOD | IW_POWER_TIMEOUT | IW_POWER_UNICAST_R;
  1857. range.num_txpower = 1;
  1858. range.txpower[0] = 15; /* 15dBm */
  1859. range.txpower_capa = IW_TXPOW_DBM;
  1860. #if WIRELESS_EXT > 10
  1861. range.retry_capa = IW_RETRY_LIMIT | IW_RETRY_LIFETIME;
  1862. range.retry_flags = IW_RETRY_LIMIT;
  1863. range.r_time_flags = IW_RETRY_LIFETIME;
  1864. range.min_retry = 0;
  1865. range.max_retry = 65535; /* ??? */
  1866. range.min_r_time = 0;
  1867. range.max_r_time = 65535 * 1000; /* ??? */
  1868. #endif /* WIRELESS_EXT > 10 */
  1869. if (copy_to_user(rrq->pointer, &range, sizeof(range)))
  1870. return -EFAULT;
  1871. TRACE_EXIT(dev->name);
  1872. return 0;
  1873. }
  1874. static int orinoco_ioctl_setiwencode(struct net_device *dev, struct iw_point *erq)
  1875. {
  1876. struct orinoco_private *priv = dev->priv;
  1877. int index = (erq->flags & IW_ENCODE_INDEX) - 1;
  1878. int setindex = priv->tx_key;
  1879. int enable = priv->wep_on;
  1880. int restricted = priv->wep_restrict;
  1881. u16 xlen = 0;
  1882. int err = 0;
  1883. char keybuf[ORINOCO_MAX_KEY_SIZE];
  1884. if (erq->pointer) {
  1885. /* We actually have a key to set */
  1886. if ( (erq->length < SMALL_KEY_SIZE) || (erq->length > ORINOCO_MAX_KEY_SIZE) )
  1887. return -EINVAL;
  1888. if (copy_from_user(keybuf, erq->pointer, erq->length))
  1889. return -EFAULT;
  1890. }
  1891. orinoco_lock(priv);
  1892. if (erq->pointer) {
  1893. if (erq->length > ORINOCO_MAX_KEY_SIZE) {
  1894. err = -E2BIG;
  1895. goto out;
  1896. }
  1897. if ( (erq->length > LARGE_KEY_SIZE)
  1898.      || ( ! priv->has_big_wep && (erq->length > SMALL_KEY_SIZE))  ) {
  1899. err = -EINVAL;
  1900. goto out;
  1901. }
  1902. if ((index < 0) || (index >= ORINOCO_MAX_KEYS))
  1903. index = priv->tx_key;
  1904. if (erq->length > SMALL_KEY_SIZE) {
  1905. xlen = LARGE_KEY_SIZE;
  1906. } else if (erq->length > 0) {
  1907. xlen = SMALL_KEY_SIZE;
  1908. } else
  1909. xlen = 0;
  1910. /* Switch on WEP if off */
  1911. if ((!enable) && (xlen > 0)) {
  1912. setindex = index;
  1913. enable = 1;
  1914. }
  1915. } else {
  1916. /* Important note : if the user do "iwconfig eth0 enc off",
  1917.  * we will arrive there with an index of -1. This is valid
  1918.  * but need to be taken care off... Jean II */
  1919. if ((index < 0) || (index >= ORINOCO_MAX_KEYS)) {
  1920. if((index != -1) || (erq->flags == 0)) {
  1921. err = -EINVAL;
  1922. goto out;
  1923. }
  1924. } else {
  1925. /* Set the index : Check that the key is valid */
  1926. if(priv->keys[index].len == 0) {
  1927. err = -EINVAL;
  1928. goto out;
  1929. }
  1930. setindex = index;
  1931. }
  1932. }
  1933. if (erq->flags & IW_ENCODE_DISABLED)
  1934. enable = 0;
  1935. /* Only for Prism2 & Symbol cards (so far) - Jean II */
  1936. if (erq->flags & IW_ENCODE_OPEN)
  1937. restricted = 0;
  1938. if (erq->flags & IW_ENCODE_RESTRICTED)
  1939. restricted = 1;
  1940. if (erq->pointer) {
  1941. priv->keys[index].len = cpu_to_le16(xlen);
  1942. memset(priv->keys[index].data, 0, sizeof(priv->keys[index].data));
  1943. memcpy(priv->keys[index].data, keybuf, erq->length);
  1944. }
  1945. priv->tx_key = setindex;
  1946. priv->wep_on = enable;
  1947. priv->wep_restrict = restricted;
  1948.  out:
  1949. orinoco_unlock(priv);
  1950. return err;
  1951. }
  1952. static int orinoco_ioctl_getiwencode(struct net_device *dev, struct iw_point *erq)
  1953. {
  1954. struct orinoco_private *priv = dev->priv;
  1955. int index = (erq->flags & IW_ENCODE_INDEX) - 1;
  1956. u16 xlen = 0;
  1957. char keybuf[ORINOCO_MAX_KEY_SIZE];
  1958. orinoco_lock(priv);
  1959. if ((index < 0) || (index >= ORINOCO_MAX_KEYS))
  1960. index = priv->tx_key;
  1961. erq->flags = 0;
  1962. if (! priv->wep_on)
  1963. erq->flags |= IW_ENCODE_DISABLED;
  1964. erq->flags |= index + 1;
  1965. /* Only for symbol cards - Jean II */
  1966. if (priv->firmware_type != FIRMWARE_TYPE_AGERE) {
  1967. if(priv->wep_restrict)
  1968. erq->flags |= IW_ENCODE_RESTRICTED;
  1969. else
  1970. erq->flags |= IW_ENCODE_OPEN;
  1971. }
  1972. xlen = le16_to_cpu(priv->keys[index].len);
  1973. erq->length = xlen;
  1974. if (erq->pointer) {
  1975. memcpy(keybuf, priv->keys[index].data, ORINOCO_MAX_KEY_SIZE);
  1976. }
  1977. orinoco_unlock(priv);
  1978. if (erq->pointer) {
  1979. if (copy_to_user(erq->pointer, keybuf, xlen))
  1980. return -EFAULT;
  1981. }
  1982. return 0;
  1983. }
  1984. static int orinoco_ioctl_setessid(struct net_device *dev, struct iw_point *erq)
  1985. {
  1986. struct orinoco_private *priv = dev->priv;
  1987. char essidbuf[IW_ESSID_MAX_SIZE+1];
  1988. /* Note : ESSID is ignored in Ad-Hoc demo mode, but we can set it
  1989.  * anyway... - Jean II */
  1990. memset(&essidbuf, 0, sizeof(essidbuf));
  1991. if (erq->flags) {
  1992. if (erq->length > IW_ESSID_MAX_SIZE)
  1993. return -E2BIG;
  1994. if (copy_from_user(&essidbuf, erq->pointer, erq->length))
  1995. return -EFAULT;
  1996. essidbuf[erq->length] = '';
  1997. }
  1998. orinoco_lock(priv);
  1999. memcpy(priv->desired_essid, essidbuf, sizeof(priv->desired_essid));
  2000. orinoco_unlock(priv);
  2001. return 0;
  2002. }
  2003. static int orinoco_ioctl_getessid(struct net_device *dev, struct iw_point *erq)
  2004. {
  2005. struct orinoco_private *priv = dev->priv;
  2006. char essidbuf[IW_ESSID_MAX_SIZE+1];
  2007. int active;
  2008. int err = 0;
  2009. TRACE_ENTER(dev->name);
  2010. err = orinoco_hw_get_essid(priv, &active, essidbuf);
  2011. if (err)
  2012. return err;
  2013. erq->flags = 1;
  2014. erq->length = strlen(essidbuf) + 1;
  2015. if (erq->pointer)
  2016. if ( copy_to_user(erq->pointer, essidbuf, erq->length) )
  2017. return -EFAULT;
  2018. TRACE_EXIT(dev->name);
  2019. return 0;
  2020. }
  2021. static int orinoco_ioctl_setnick(struct net_device *dev, struct iw_point *nrq)
  2022. {
  2023. struct orinoco_private *priv = dev->priv;
  2024. char nickbuf[IW_ESSID_MAX_SIZE+1];
  2025. if (nrq->length > IW_ESSID_MAX_SIZE)
  2026. return -E2BIG;
  2027. memset(nickbuf, 0, sizeof(nickbuf));
  2028. if (copy_from_user(nickbuf, nrq->pointer, nrq->length))
  2029. return -EFAULT;
  2030. nickbuf[nrq->length] = '';
  2031. orinoco_lock(priv);
  2032. memcpy(priv->nick, nickbuf, sizeof(priv->nick));
  2033. orinoco_unlock(priv);
  2034. return 0;
  2035. }
  2036. static int orinoco_ioctl_getnick(struct net_device *dev, struct iw_point *nrq)
  2037. {
  2038. struct orinoco_private *priv = dev->priv;
  2039. char nickbuf[IW_ESSID_MAX_SIZE+1];
  2040. orinoco_lock(priv);
  2041. memcpy(nickbuf, priv->nick, IW_ESSID_MAX_SIZE+1);
  2042. orinoco_unlock(priv);
  2043. nrq->length = strlen(nickbuf)+1;
  2044. if (copy_to_user(nrq->pointer, nickbuf, sizeof(nickbuf)))
  2045. return -EFAULT;
  2046. return 0;
  2047. }
  2048. static int orinoco_ioctl_setfreq(struct net_device *dev, struct iw_freq *frq)
  2049. {
  2050. struct orinoco_private *priv = dev->priv;
  2051. int chan = -1;
  2052. /* We can only use this in Ad-Hoc demo mode to set the operating
  2053.  * frequency, or in IBSS mode to set the frequency where the IBSS
  2054.  * will be created - Jean II */
  2055. if (priv->iw_mode != IW_MODE_ADHOC)
  2056. return -EOPNOTSUPP;
  2057. if ( (frq->e == 0) && (frq->m <= 1000) ) {
  2058. /* Setting by channel number */
  2059. chan = frq->m;
  2060. } else {
  2061. /* Setting by frequency - search the table */
  2062. int mult = 1;
  2063. int i;
  2064. for (i = 0; i < (6 - frq->e); i++)
  2065. mult *= 10;
  2066. for (i = 0; i < NUM_CHANNELS; i++)
  2067. if (frq->m == (channel_frequency[i] * mult))
  2068. chan = i+1;
  2069. }
  2070. if ( (chan < 1) || (chan > NUM_CHANNELS) ||
  2071.      ! (priv->channel_mask & (1 << (chan-1)) ) )
  2072. return -EINVAL;
  2073. orinoco_lock(priv);
  2074. priv->channel = chan;
  2075. orinoco_unlock(priv);
  2076. return 0;
  2077. }
  2078. static int orinoco_ioctl_getsens(struct net_device *dev, struct iw_param *srq)
  2079. {
  2080. struct orinoco_private *priv = dev->priv;
  2081. hermes_t *hw = &priv->hw;
  2082. u16 val;
  2083. int err;
  2084. orinoco_lock(priv);
  2085. err = hermes_read_wordrec(hw, USER_BAP, HERMES_RID_CNFSYSTEMSCALE, &val);
  2086. orinoco_unlock(priv);
  2087. if (err)
  2088. return err;
  2089. srq->value = val;
  2090. srq->fixed = 0; /* auto */
  2091. return 0;
  2092. }
  2093. static int orinoco_ioctl_setsens(struct net_device *dev, struct iw_param *srq)
  2094. {
  2095. struct orinoco_private *priv = dev->priv;
  2096. int val = srq->value;
  2097. if ((val < 1) || (val > 3))
  2098. return -EINVAL;
  2099. orinoco_lock(priv);
  2100. priv->ap_density = val;
  2101. orinoco_unlock(priv);
  2102. return 0;
  2103. }
  2104. static int orinoco_ioctl_setrts(struct net_device *dev, struct iw_param *rrq)
  2105. {
  2106. struct orinoco_private *priv = dev->priv;
  2107. int val = rrq->value;
  2108. if (rrq->disabled)
  2109. val = 2347;
  2110. if ( (val < 0) || (val > 2347) )
  2111. return -EINVAL;
  2112. orinoco_lock(priv);
  2113. priv->rts_thresh = val;
  2114. orinoco_unlock(priv);
  2115. return 0;
  2116. }
  2117. static int orinoco_ioctl_setfrag(struct net_device *dev, struct iw_param *frq)
  2118. {
  2119. struct orinoco_private *priv = dev->priv;
  2120. int err = 0;
  2121. orinoco_lock(priv);
  2122. if (priv->has_mwo) {
  2123. if (frq->disabled)
  2124. priv->mwo_robust = 0;
  2125. else {
  2126. if (frq->fixed)
  2127. printk(KERN_WARNING "%s: Fixed fragmentation not 
  2128. supported on this firmware. Using MWO robust instead.n", dev->name);
  2129. priv->mwo_robust = 1;
  2130. }
  2131. } else {
  2132. if (frq->disabled)
  2133. priv->frag_thresh = 2346;
  2134. else {
  2135. if ( (frq->value < 256) || (frq->value > 2346) )
  2136. err = -EINVAL;
  2137. else
  2138. priv->frag_thresh = frq->value & ~0x1; /* must be even */
  2139. }
  2140. }
  2141. orinoco_unlock(priv);
  2142. return err;
  2143. }
  2144. static int orinoco_ioctl_getfrag(struct net_device *dev, struct iw_param *frq)
  2145. {
  2146. struct orinoco_private *priv = dev->priv;
  2147. hermes_t *hw = &priv->hw;
  2148. int err = 0;
  2149. u16 val;
  2150. orinoco_lock(priv);
  2151. if (priv->has_mwo) {
  2152. err = hermes_read_wordrec(hw, USER_BAP,
  2153.   HERMES_RID_CNFMWOROBUST_AGERE,
  2154.   &val);
  2155. if (err)
  2156. val = 0;
  2157. frq->value = val ? 2347 : 0;
  2158. frq->disabled = ! val;
  2159. frq->fixed = 0;
  2160. } else {
  2161. err = hermes_read_wordrec(hw, USER_BAP, HERMES_RID_CNFFRAGMENTATIONTHRESHOLD,
  2162.   &val);
  2163. if (err)
  2164. val = 0;
  2165. frq->value = val;
  2166. frq->disabled = (val >= 2346);
  2167. frq->fixed = 1;
  2168. }
  2169. orinoco_unlock(priv);
  2170. return err;
  2171. }
  2172. static int orinoco_ioctl_setrate(struct net_device *dev, struct iw_param *rrq)
  2173. {
  2174. struct orinoco_private *priv = dev->priv;
  2175. int err = 0;
  2176. int ratemode = -1;
  2177. int bitrate; /* 100s of kilobits */
  2178. int i;
  2179. /* As the user space doesn't know our highest rate, it uses -1
  2180.  * to ask us to set the highest rate.  Test it using "iwconfig
  2181.  * ethX rate auto" - Jean II */
  2182. if (rrq->value == -1)
  2183. bitrate = 110;
  2184. else {
  2185. if (rrq->value % 100000)
  2186. return -EINVAL;
  2187. bitrate = rrq->value / 100000;
  2188. }
  2189. if ( (bitrate != 10) && (bitrate != 20) &&
  2190.      (bitrate != 55) && (bitrate != 110) )
  2191. return -EINVAL;
  2192. for (i = 0; i < BITRATE_TABLE_SIZE; i++)
  2193. if ( (bitrate_table[i].bitrate == bitrate) &&
  2194.      (bitrate_table[i].automatic == ! rrq->fixed) ) {
  2195. ratemode = i;
  2196. break;
  2197. }
  2198. if (ratemode == -1)
  2199. return -EINVAL;
  2200. orinoco_lock(priv);
  2201. priv->bitratemode = ratemode;
  2202. orinoco_unlock(priv);
  2203. return err;
  2204. }
  2205. static int orinoco_ioctl_getrate(struct net_device *dev, struct iw_param *rrq)
  2206. {
  2207. struct orinoco_private *priv = dev->priv;
  2208. hermes_t *hw = &priv->hw;
  2209. int err = 0;
  2210. int ratemode;
  2211. int i;
  2212. u16 val;
  2213. orinoco_lock(priv);
  2214. ratemode = priv->bitratemode;
  2215. if ( (ratemode < 0) || (ratemode > BITRATE_TABLE_SIZE) )
  2216. BUG();
  2217. rrq->value = bitrate_table[ratemode].bitrate * 100000;
  2218. rrq->fixed = ! bitrate_table[ratemode].automatic;
  2219. rrq->disabled = 0;
  2220. /* If the interface is running we try to find more about the
  2221.    current mode */
  2222. if (netif_running(dev)) {
  2223. err = hermes_read_wordrec(hw, USER_BAP,
  2224.   HERMES_RID_CURRENTTXRATE, &val);
  2225. if (err)
  2226. goto out;
  2227. switch (priv->firmware_type) {
  2228. case FIRMWARE_TYPE_AGERE: /* Lucent style rate */
  2229. /* Note : in Lucent firmware, the return value of
  2230.  * HERMES_RID_CURRENTTXRATE is the bitrate in Mb/s,
  2231.  * and therefore is totally different from the
  2232.  * encoding of HERMES_RID_CNFTXRATECONTROL.
  2233.  * Don't forget that 6Mb/s is really 5.5Mb/s */
  2234. if (val == 6)
  2235. rrq->value = 5500000;
  2236. else
  2237. rrq->value = val * 1000000;
  2238.                         break;
  2239. case FIRMWARE_TYPE_INTERSIL: /* Intersil style rate */
  2240. case FIRMWARE_TYPE_SYMBOL: /* Symbol style rate */
  2241. for (i = 0; i < BITRATE_TABLE_SIZE; i++)
  2242. if (bitrate_table[i].intersil_txratectrl == val) {
  2243. ratemode = i;
  2244. break;
  2245. }
  2246. if (i >= BITRATE_TABLE_SIZE)
  2247. printk(KERN_INFO "%s: Unable to determine current bitrate (0x%04hx)n",
  2248.        dev->name, val);
  2249. rrq->value = bitrate_table[ratemode].bitrate * 100000;
  2250. break;
  2251. default:
  2252. BUG();
  2253. }
  2254. }
  2255.  out:
  2256. orinoco_unlock(priv);
  2257. return err;
  2258. }
  2259. static int orinoco_ioctl_setpower(struct net_device *dev, struct iw_param *prq)
  2260. {
  2261. struct orinoco_private *priv = dev->priv;
  2262. int err = 0;
  2263. orinoco_lock(priv);
  2264. if (prq->disabled) {
  2265. priv->pm_on = 0;
  2266. } else {
  2267. switch (prq->flags & IW_POWER_MODE) {
  2268. case IW_POWER_UNICAST_R:
  2269. priv->pm_mcast = 0;
  2270. priv->pm_on = 1;
  2271. break;
  2272. case IW_POWER_ALL_R:
  2273. priv->pm_mcast = 1;
  2274. priv->pm_on = 1;
  2275. break;
  2276. case IW_POWER_ON:
  2277. /* No flags : but we may have a value - Jean II */
  2278. break;
  2279. default:
  2280. err = -EINVAL;
  2281. }
  2282. if (err)
  2283. goto out;
  2284. if (prq->flags & IW_POWER_TIMEOUT) {
  2285. priv->pm_on = 1;
  2286. priv->pm_timeout = prq->value / 1000;
  2287. }
  2288. if (prq->flags & IW_POWER_PERIOD) {
  2289. priv->pm_on = 1;
  2290. priv->pm_period = prq->value / 1000;
  2291. }
  2292. /* It's valid to not have a value if we are just toggling
  2293.  * the flags... Jean II */
  2294. if(!priv->pm_on) {
  2295. err = -EINVAL;
  2296. goto out;
  2297. }
  2298. }
  2299.  out:
  2300. orinoco_unlock(priv);
  2301. return err;
  2302. }
  2303. static int orinoco_ioctl_getpower(struct net_device *dev, struct iw_param *prq)
  2304. {
  2305. struct orinoco_private *priv = dev->priv;
  2306. hermes_t *hw = &priv->hw;
  2307. int err = 0;
  2308. u16 enable, period, timeout, mcast;
  2309. orinoco_lock(priv);
  2310. err = hermes_read_wordrec(hw, USER_BAP, HERMES_RID_CNFPMENABLED, &enable);
  2311. if (err)
  2312. goto out;
  2313. err = hermes_read_wordrec(hw, USER_BAP,
  2314.   HERMES_RID_CNFMAXSLEEPDURATION, &period);
  2315. if (err)
  2316. goto out;
  2317. err = hermes_read_wordrec(hw, USER_BAP, HERMES_RID_CNFPMHOLDOVERDURATION, &timeout);
  2318. if (err)
  2319. goto out;
  2320. err = hermes_read_wordrec(hw, USER_BAP, HERMES_RID_CNFMULTICASTRECEIVE, &mcast);
  2321. if (err)
  2322. goto out;
  2323. prq->disabled = !enable;
  2324. /* Note : by default, display the period */
  2325. if ((prq->flags & IW_POWER_TYPE) == IW_POWER_TIMEOUT) {
  2326. prq->flags = IW_POWER_TIMEOUT;
  2327. prq->value = timeout * 1000;
  2328. } else {
  2329. prq->flags = IW_POWER_PERIOD;
  2330. prq->value = period * 1000;
  2331. }
  2332. if (mcast)
  2333. prq->flags |= IW_POWER_ALL_R;
  2334. else
  2335. prq->flags |= IW_POWER_UNICAST_R;
  2336.  out:
  2337. orinoco_unlock(priv);
  2338. return err;
  2339. }
  2340. #if WIRELESS_EXT > 10
  2341. static int orinoco_ioctl_getretry(struct net_device *dev, struct iw_param *rrq)
  2342. {
  2343. struct orinoco_private *priv = dev->priv;
  2344. hermes_t *hw = &priv->hw;
  2345. int err = 0;
  2346. u16 short_limit, long_limit, lifetime;
  2347. orinoco_lock(priv);
  2348. err = hermes_read_wordrec(hw, USER_BAP, HERMES_RID_SHORTRETRYLIMIT,
  2349.   &short_limit);
  2350. if (err)
  2351. goto out;
  2352. err = hermes_read_wordrec(hw, USER_BAP, HERMES_RID_LONGRETRYLIMIT,
  2353.   &long_limit);
  2354. if (err)
  2355. goto out;
  2356. err = hermes_read_wordrec(hw, USER_BAP, HERMES_RID_MAXTRANSMITLIFETIME,
  2357.   &lifetime);
  2358. if (err)
  2359. goto out;
  2360. rrq->disabled = 0; /* Can't be disabled */
  2361. /* Note : by default, display the retry number */
  2362. if ((rrq->flags & IW_RETRY_TYPE) == IW_RETRY_LIFETIME) {
  2363. rrq->flags = IW_RETRY_LIFETIME;
  2364. rrq->value = lifetime * 1000; /* ??? */
  2365. } else {
  2366. /* By default, display the min number */
  2367. if ((rrq->flags & IW_RETRY_MAX)) {
  2368. rrq->flags = IW_RETRY_LIMIT | IW_RETRY_MAX;
  2369. rrq->value = long_limit;
  2370. } else {
  2371. rrq->flags = IW_RETRY_LIMIT;
  2372. rrq->value = short_limit;
  2373. if(short_limit != long_limit)
  2374. rrq->flags |= IW_RETRY_MIN;
  2375. }
  2376. }
  2377.  out:
  2378. orinoco_unlock(priv);
  2379. return err;
  2380. }
  2381. #endif /* WIRELESS_EXT > 10 */
  2382. static int orinoco_ioctl_setibssport(struct net_device *dev, struct iwreq *wrq)
  2383. {
  2384. struct orinoco_private *priv = dev->priv;
  2385. int val = *( (int *) wrq->u.name );
  2386. orinoco_lock(priv);
  2387. priv->ibss_port = val ;
  2388. /* Actually update the mode we are using */
  2389. set_port_type(priv);
  2390. orinoco_unlock(priv);
  2391. return 0;
  2392. }
  2393. static int orinoco_ioctl_getibssport(struct net_device *dev, struct iwreq *wrq)
  2394. {
  2395. struct orinoco_private *priv = dev->priv;
  2396. int *val = (int *)wrq->u.name;
  2397. orinoco_lock(priv);
  2398. *val = priv->ibss_port;
  2399. orinoco_unlock(priv);
  2400. return 0;
  2401. }
  2402. static int orinoco_ioctl_setport3(struct net_device *dev, struct iwreq *wrq)
  2403. {
  2404. struct orinoco_private *priv = dev->priv;
  2405. int val = *( (int *) wrq->u.name );
  2406. int err = 0;
  2407. orinoco_lock(priv);
  2408. switch (val) {
  2409. case 0: /* Try to do IEEE ad-hoc mode */
  2410. if (! priv->has_ibss) {
  2411. err = -EINVAL;
  2412. break;
  2413. }
  2414. DEBUG(2, "%s: Prefer IBSS Ad-Hoc moden", dev->name);
  2415. priv->prefer_port3 = 0;
  2416. break;
  2417. case 1: /* Try to do Lucent proprietary ad-hoc mode */
  2418. if (! priv->has_port3) {
  2419. err = -EINVAL;
  2420. break;
  2421. }
  2422. DEBUG(2, "%s: Prefer Ad-Hoc demo moden", dev->name);
  2423. priv->prefer_port3 = 1;
  2424. break;
  2425. default:
  2426. err = -EINVAL;
  2427. }
  2428. if (! err)
  2429. /* Actually update the mode we are using */
  2430. set_port_type(priv);
  2431. orinoco_unlock(priv);
  2432. return err;
  2433. }
  2434. static int orinoco_ioctl_getport3(struct net_device *dev, struct iwreq *wrq)
  2435. {
  2436. struct orinoco_private *priv = dev->priv;
  2437. int *val = (int *)wrq->u.name;
  2438. orinoco_lock(priv);
  2439. *val = priv->prefer_port3;
  2440. orinoco_unlock(priv);
  2441. return 0;
  2442. }
  2443. /* Spy is used for link quality/strength measurements in Ad-Hoc mode
  2444.  * Jean II */
  2445. static int orinoco_ioctl_setspy(struct net_device *dev, struct iw_point *srq)
  2446. {
  2447. struct orinoco_private *priv = dev->priv;
  2448. struct sockaddr address[IW_MAX_SPY];
  2449. int number = srq->length;
  2450. int i;
  2451. int err = 0;
  2452. /* Check the number of addresses */
  2453. if (number > IW_MAX_SPY)
  2454. return -E2BIG;
  2455. /* Get the data in the driver */
  2456. if (srq->pointer) {
  2457. if (copy_from_user(address, srq->pointer,
  2458.    sizeof(struct sockaddr) * number))
  2459. return -EFAULT;
  2460. }
  2461. /* Make sure nobody mess with the structure while we do */
  2462. orinoco_lock(priv);
  2463. /* orinoco_lock() doesn't disable interrupts, so make sure the
  2464.  * interrupt rx path don't get confused while we copy */
  2465. priv->spy_number = 0;
  2466. if (number > 0) {
  2467. /* Extract the addresses */
  2468. for (i = 0; i < number; i++)
  2469. memcpy(priv->spy_address[i], address[i].sa_data,
  2470.        ETH_ALEN);
  2471. /* Reset stats */
  2472. memset(priv->spy_stat, 0,
  2473.        sizeof(struct iw_quality) * IW_MAX_SPY);
  2474. /* Set number of addresses */
  2475. priv->spy_number = number;
  2476. }
  2477. /* Time to show what we have done... */
  2478. DEBUG(0, "%s: New spy list:n", dev->name);
  2479. for (i = 0; i < number; i++) {
  2480. DEBUG(0, "%s: %d - %02x:%02x:%02x:%02x:%02x:%02xn",
  2481.       dev->name, i+1,
  2482.       priv->spy_address[i][0], priv->spy_address[i][1],
  2483.       priv->spy_address[i][2], priv->spy_address[i][3],
  2484.       priv->spy_address[i][4], priv->spy_address[i][5]);
  2485. }
  2486. /* Now, let the others play */
  2487. orinoco_unlock(priv);
  2488. return err;
  2489. }
  2490. static int orinoco_ioctl_getspy(struct net_device *dev, struct iw_point *srq)
  2491. {
  2492. struct orinoco_private *priv = dev->priv;
  2493. struct sockaddr address[IW_MAX_SPY];
  2494. struct iw_quality spy_stat[IW_MAX_SPY];
  2495. int number;
  2496. int i;
  2497. orinoco_lock(priv);
  2498. number = priv->spy_number;
  2499. if ((number > 0) && (srq->pointer)) {
  2500. /* Create address struct */
  2501. for (i = 0; i < number; i++) {
  2502. memcpy(address[i].sa_data, priv->spy_address[i],
  2503.        ETH_ALEN);
  2504. address[i].sa_family = AF_UNIX;
  2505. }
  2506. /* Copy stats */
  2507. /* In theory, we should disable irqs while copying the stats
  2508.  * because the rx path migh update it in the middle...
  2509.  * Bah, who care ? - Jean II */
  2510. memcpy(&spy_stat, priv->spy_stat,
  2511.        sizeof(struct iw_quality) * IW_MAX_SPY);
  2512. for (i=0; i < number; i++)
  2513. priv->spy_stat[i].updated = 0;
  2514. }
  2515. orinoco_unlock(priv);
  2516. /* Push stuff to user space */
  2517. srq->length = number;
  2518. if(copy_to_user(srq->pointer, address,
  2519.  sizeof(struct sockaddr) * number))
  2520. return -EFAULT;
  2521. if(copy_to_user(srq->pointer + (sizeof(struct sockaddr)*number),
  2522. &spy_stat, sizeof(struct iw_quality) * number))
  2523. return -EFAULT;
  2524. return 0;
  2525. }
  2526. int
  2527. orinoco_ioctl(struct net_device *dev, struct ifreq *rq, int cmd)
  2528. {
  2529. struct orinoco_private *priv = dev->priv;
  2530. struct iwreq *wrq = (struct iwreq *)rq;
  2531. int err = 0;
  2532. int changed = 0;
  2533. TRACE_ENTER(dev->name);
  2534. /* In theory, we could allow most of the the SET stuff to be
  2535.  * done In practice, the laps of time at startup when the card
  2536.  * is not ready is very short, so why bother...  Note that
  2537.  * netif_device_present is different from up/down (ifconfig),
  2538.  * when the device is not yet up, it is usually already
  2539.  * ready...  Jean II */
  2540. if (! netif_device_present(dev))
  2541. return -ENODEV;
  2542. switch (cmd) {
  2543. case SIOCGIWNAME:
  2544. DEBUG(1, "%s: SIOCGIWNAMEn", dev->name);
  2545. strcpy(wrq->u.name, "IEEE 802.11-DS");
  2546. break;
  2547. case SIOCGIWAP:
  2548. DEBUG(1, "%s: SIOCGIWAPn", dev->name);
  2549. wrq->u.ap_addr.sa_family = ARPHRD_ETHER;
  2550. err = orinoco_hw_get_bssid(priv, wrq->u.ap_addr.sa_data);
  2551. break;
  2552. case SIOCGIWRANGE:
  2553. DEBUG(1, "%s: SIOCGIWRANGEn", dev->name);
  2554. err = orinoco_ioctl_getiwrange(dev, &wrq->u.data);
  2555. break;
  2556. case SIOCSIWMODE:
  2557. DEBUG(1, "%s: SIOCSIWMODEn", dev->name);
  2558. orinoco_lock(priv);
  2559. switch (wrq->u.mode) {
  2560. case IW_MODE_ADHOC:
  2561. if (! (priv->has_ibss || priv->has_port3) )
  2562. err = -EINVAL;
  2563. else {
  2564. priv->iw_mode = IW_MODE_ADHOC;
  2565. changed = 1;
  2566. }
  2567. break;
  2568. case IW_MODE_INFRA:
  2569. priv->iw_mode = IW_MODE_INFRA;
  2570. changed = 1;
  2571. break;
  2572. default:
  2573. err = -EINVAL;
  2574. break;
  2575. }
  2576. set_port_type(priv);
  2577. orinoco_unlock(priv);
  2578. break;
  2579. case SIOCGIWMODE:
  2580. DEBUG(1, "%s: SIOCGIWMODEn", dev->name);
  2581. orinoco_lock(priv);
  2582. wrq->u.mode = priv->iw_mode;
  2583. orinoco_unlock(priv);
  2584. break;
  2585. case SIOCSIWENCODE:
  2586. DEBUG(1, "%s: SIOCSIWENCODEn", dev->name);
  2587. if (! priv->has_wep) {
  2588. err = -EOPNOTSUPP;
  2589. break;
  2590. }
  2591. err = orinoco_ioctl_setiwencode(dev, &wrq->u.encoding);
  2592. if (! err)
  2593. changed = 1;
  2594. break;
  2595. case SIOCGIWENCODE:
  2596. DEBUG(1, "%s: SIOCGIWENCODEn", dev->name);
  2597. if (! priv->has_wep) {
  2598. err = -EOPNOTSUPP;
  2599. break;
  2600. }
  2601. if (! capable(CAP_NET_ADMIN)) {
  2602. err = -EPERM;
  2603. break;
  2604. }
  2605. err = orinoco_ioctl_getiwencode(dev, &wrq->u.encoding);
  2606. break;
  2607. case SIOCSIWESSID:
  2608. DEBUG(1, "%s: SIOCSIWESSIDn", dev->name);
  2609. err = orinoco_ioctl_setessid(dev, &wrq->u.essid);
  2610. if (! err)
  2611. changed = 1;
  2612. break;
  2613. case SIOCGIWESSID:
  2614. DEBUG(1, "%s: SIOCGIWESSIDn", dev->name);
  2615. err = orinoco_ioctl_getessid(dev, &wrq->u.essid);
  2616. break;
  2617. case SIOCSIWNICKN:
  2618. DEBUG(1, "%s: SIOCSIWNICKNn", dev->name);
  2619. err = orinoco_ioctl_setnick(dev, &wrq->u.data);
  2620. if (! err)
  2621. changed = 1;
  2622. break;
  2623. case SIOCGIWNICKN:
  2624. DEBUG(1, "%s: SIOCGIWNICKNn", dev->name);
  2625. err = orinoco_ioctl_getnick(dev, &wrq->u.data);
  2626. break;
  2627. case SIOCGIWFREQ:
  2628. DEBUG(1, "%s: SIOCGIWFREQn", dev->name);
  2629. wrq->u.freq.m = orinoco_hw_get_freq(priv);
  2630. wrq->u.freq.e = 1;
  2631. break;
  2632. case SIOCSIWFREQ:
  2633. DEBUG(1, "%s: SIOCSIWFREQn", dev->name);
  2634. err = orinoco_ioctl_setfreq(dev, &wrq->u.freq);
  2635. if (! err)
  2636. changed = 1;
  2637. break;
  2638. case SIOCGIWSENS:
  2639. DEBUG(1, "%s: SIOCGIWSENSn", dev->name);
  2640. err = orinoco_ioctl_getsens(dev, &wrq->u.sens);
  2641. break;
  2642. case SIOCSIWSENS:
  2643. DEBUG(1, "%s: SIOCSIWSENSn", dev->name);
  2644. err = orinoco_ioctl_setsens(dev, &wrq->u.sens);
  2645. if (! err)
  2646. changed = 1;
  2647. break;
  2648. case SIOCGIWRTS:
  2649. DEBUG(1, "%s: SIOCGIWRTSn", dev->name);
  2650. wrq->u.rts.value = priv->rts_thresh;
  2651. wrq->u.rts.disabled = (wrq->u.rts.value == 2347);
  2652. wrq->u.rts.fixed = 1;
  2653. break;
  2654. case SIOCSIWRTS:
  2655. DEBUG(1, "%s: SIOCSIWRTSn", dev->name);
  2656. err = orinoco_ioctl_setrts(dev, &wrq->u.rts);
  2657. if (! err)
  2658. changed = 1;
  2659. break;
  2660. case SIOCSIWFRAG:
  2661. DEBUG(1, "%s: SIOCSIWFRAGn", dev->name);
  2662. err = orinoco_ioctl_setfrag(dev, &wrq->u.frag);
  2663. if (! err)
  2664. changed = 1;
  2665. break;
  2666. case SIOCGIWFRAG:
  2667. DEBUG(1, "%s: SIOCGIWFRAGn", dev->name);
  2668. err = orinoco_ioctl_getfrag(dev, &wrq->u.frag);
  2669. break;
  2670. case SIOCSIWRATE:
  2671. DEBUG(1, "%s: SIOCSIWRATEn", dev->name);
  2672. err = orinoco_ioctl_setrate(dev, &wrq->u.bitrate);
  2673. if (! err)
  2674. changed = 1;
  2675. break;
  2676. case SIOCGIWRATE:
  2677. DEBUG(1, "%s: SIOCGIWRATEn", dev->name);
  2678. err = orinoco_ioctl_getrate(dev, &wrq->u.bitrate);
  2679. break;
  2680. case SIOCSIWPOWER:
  2681. DEBUG(1, "%s: SIOCSIWPOWERn", dev->name);
  2682. err = orinoco_ioctl_setpower(dev, &wrq->u.power);
  2683. if (! err)
  2684. changed = 1;
  2685. break;
  2686. case SIOCGIWPOWER:
  2687. DEBUG(1, "%s: SIOCGIWPOWERn", dev->name);
  2688. err = orinoco_ioctl_getpower(dev, &wrq->u.power);
  2689. break;
  2690. case SIOCGIWTXPOW:
  2691. DEBUG(1, "%s: SIOCGIWTXPOWn", dev->name);
  2692. /* The card only supports one tx power, so this is easy */
  2693. wrq->u.txpower.value = 15; /* dBm */
  2694. wrq->u.txpower.fixed = 1;
  2695. wrq->u.txpower.disabled = 0;
  2696. wrq->u.txpower.flags = IW_TXPOW_DBM;
  2697. break;
  2698. #if WIRELESS_EXT > 10
  2699. case SIOCSIWRETRY:
  2700. DEBUG(1, "%s: SIOCSIWRETRYn", dev->name);
  2701. err = -EOPNOTSUPP;
  2702. break;
  2703. case SIOCGIWRETRY:
  2704. DEBUG(1, "%s: SIOCGIWRETRYn", dev->name);
  2705. err = orinoco_ioctl_getretry(dev, &wrq->u.retry);
  2706. break;
  2707. #endif /* WIRELESS_EXT > 10 */
  2708. case SIOCSIWSPY:
  2709. DEBUG(1, "%s: SIOCSIWSPYn", dev->name);
  2710. err = orinoco_ioctl_setspy(dev, &wrq->u.data);
  2711. break;
  2712. case SIOCGIWSPY:
  2713. DEBUG(1, "%s: SIOCGIWSPYn", dev->name);
  2714. err = orinoco_ioctl_getspy(dev, &wrq->u.data);
  2715. break;
  2716. case SIOCGIWPRIV:
  2717. DEBUG(1, "%s: SIOCGIWPRIVn", dev->name);
  2718. if (wrq->u.data.pointer) {
  2719. struct iw_priv_args privtab[] = {
  2720. { SIOCIWFIRSTPRIV + 0x0, 0, 0, "force_reset" },
  2721. { SIOCIWFIRSTPRIV + 0x1, 0, 0, "card_reset" },
  2722. { SIOCIWFIRSTPRIV + 0x2,
  2723.   IW_PRIV_TYPE_INT | IW_PRIV_SIZE_FIXED | 1,
  2724.   0, "set_port3" },
  2725. { SIOCIWFIRSTPRIV + 0x3, 0,
  2726.   IW_PRIV_TYPE_INT | IW_PRIV_SIZE_FIXED | 1,
  2727.   "get_port3" },
  2728. { SIOCIWFIRSTPRIV + 0x4,
  2729.   IW_PRIV_TYPE_INT | IW_PRIV_SIZE_FIXED | 1,
  2730.   0, "set_preamble" },
  2731. { SIOCIWFIRSTPRIV + 0x5, 0,
  2732.   IW_PRIV_TYPE_INT | IW_PRIV_SIZE_FIXED | 1,
  2733.   "get_preamble" },
  2734. { SIOCIWFIRSTPRIV + 0x6,
  2735.   IW_PRIV_TYPE_INT | IW_PRIV_SIZE_FIXED | 1,
  2736.   0, "set_ibssport" },
  2737. { SIOCIWFIRSTPRIV + 0x7, 0,
  2738.   IW_PRIV_TYPE_INT | IW_PRIV_SIZE_FIXED | 1,
  2739.   "get_ibssport" }
  2740. };
  2741. err = verify_area(VERIFY_WRITE, wrq->u.data.pointer, sizeof(privtab));
  2742. if (err)
  2743. break;
  2744. wrq->u.data.length = sizeof(privtab) / sizeof(privtab[0]);
  2745. if (copy_to_user(wrq->u.data.pointer, privtab, sizeof(privtab)))
  2746. err = -EFAULT;
  2747. }
  2748. break;
  2749.        
  2750. case SIOCIWFIRSTPRIV + 0x0: /* force_reset */
  2751. DEBUG(1, "%s: SIOCIWFIRSTPRIV + 0x0 (force_reset)n",
  2752.       dev->name);
  2753. if (! capable(CAP_NET_ADMIN)) {
  2754. err = -EPERM;
  2755. break;
  2756. }
  2757. printk(KERN_DEBUG "%s: Forcing reset!n", dev->name);
  2758. orinoco_reset(priv);
  2759. break;
  2760. case SIOCIWFIRSTPRIV + 0x1: /* card_reset */
  2761. DEBUG(1, "%s: SIOCIWFIRSTPRIV + 0x1 (card_reset)n",
  2762.       dev->name);
  2763. if (! capable(CAP_NET_ADMIN)) {
  2764. err = -EPERM;
  2765. break;
  2766. }
  2767. printk(KERN_DEBUG "%s: Forcing card reset!n", dev->name);
  2768. if(priv->card_reset_handler != NULL)
  2769. priv->card_reset_handler(priv);
  2770. orinoco_reset(priv);
  2771. break;
  2772. case SIOCIWFIRSTPRIV + 0x2: /* set_port3 */
  2773. DEBUG(1, "%s: SIOCIWFIRSTPRIV + 0x2 (set_port3)n",
  2774.       dev->name);
  2775. if (! capable(CAP_NET_ADMIN)) {
  2776. err = -EPERM;
  2777. break;
  2778. }
  2779. err = orinoco_ioctl_setport3(dev, wrq);
  2780. if (! err)
  2781. changed = 1;
  2782. break;
  2783. case SIOCIWFIRSTPRIV + 0x3: /* get_port3 */
  2784. DEBUG(1, "%s: SIOCIWFIRSTPRIV + 0x3 (get_port3)n",
  2785.       dev->name);
  2786. err = orinoco_ioctl_getport3(dev, wrq);
  2787. break;
  2788. case SIOCIWFIRSTPRIV + 0x4: /* set_preamble */
  2789. DEBUG(1, "%s: SIOCIWFIRSTPRIV + 0x4 (set_preamble)n",
  2790.       dev->name);
  2791. if (! capable(CAP_NET_ADMIN)) {
  2792. err = -EPERM;
  2793. break;
  2794. }
  2795. /* 802.11b has recently defined some short preamble.
  2796.  * Basically, the Phy header has been reduced in size.
  2797.  * This increase performance, especially at high rates
  2798.  * (the preamble is transmitted at 1Mb/s), unfortunately
  2799.  * this give compatibility troubles... - Jean II */
  2800. if(priv->has_preamble) {
  2801. int val = *( (int *) wrq->u.name );
  2802. orinoco_lock(priv);
  2803. if(val)
  2804. priv->preamble = 1;
  2805. else
  2806. priv->preamble = 0;
  2807. orinoco_unlock(priv);
  2808. changed = 1;
  2809. } else
  2810. err = -EOPNOTSUPP;
  2811. break;
  2812. case SIOCIWFIRSTPRIV + 0x5: /* get_preamble */
  2813. DEBUG(1, "%s: SIOCIWFIRSTPRIV + 0x5 (get_preamble)n",
  2814.       dev->name);
  2815. if(priv->has_preamble) {
  2816. int *val = (int *)wrq->u.name;
  2817. orinoco_lock(priv);
  2818. *val = priv->preamble;
  2819. orinoco_unlock(priv);
  2820. } else
  2821. err = -EOPNOTSUPP;
  2822. break;
  2823. case SIOCIWFIRSTPRIV + 0x6: /* set_ibssport */
  2824. DEBUG(1, "%s: SIOCIWFIRSTPRIV + 0x6 (set_ibssport)n",
  2825.       dev->name);
  2826. if (! capable(CAP_NET_ADMIN)) {
  2827. err = -EPERM;
  2828. break;
  2829. }
  2830. err = orinoco_ioctl_setibssport(dev, wrq);
  2831. if (! err)
  2832. changed = 1;
  2833. break;
  2834. case SIOCIWFIRSTPRIV + 0x7: /* get_ibssport */
  2835. DEBUG(1, "%s: SIOCIWFIRSTPRIV + 0x7 (get_ibssport)n",
  2836.       dev->name);
  2837. err = orinoco_ioctl_getibssport(dev, wrq);
  2838. break;
  2839. default:
  2840. err = -EOPNOTSUPP;
  2841. }
  2842. if (! err && changed && netif_running(dev)) {
  2843. err = orinoco_reset(priv);
  2844. if (err) {
  2845. /* Ouch ! What are we supposed to do ? */
  2846. printk(KERN_ERR "orinoco_cs: Failed to set parameters on %sn",
  2847.        dev->name);
  2848. netif_device_detach(dev);
  2849. orinoco_shutdown(priv);
  2850. }
  2851. }
  2852. TRACE_EXIT(dev->name);
  2853. return err;
  2854. }
  2855. int
  2856. orinoco_change_mtu(struct net_device *dev, int new_mtu)
  2857. {
  2858. TRACE_ENTER(dev->name);
  2859. if ( (new_mtu < ORINOCO_MIN_MTU) || (new_mtu > ORINOCO_MAX_MTU) )
  2860. return -EINVAL;
  2861. dev->mtu = new_mtu;
  2862. TRACE_EXIT(dev->name);
  2863. return 0;
  2864. }
  2865. static void
  2866. __orinoco_set_multicast_list(struct net_device *dev)
  2867. {
  2868. struct orinoco_private *priv = dev->priv;
  2869. hermes_t *hw = &priv->hw;
  2870. int err = 0;
  2871. int promisc, mc_count;
  2872. /* We'll wait until it's ready. Anyway, the network doesn't call us
  2873.  * here until we are open - Jean II */
  2874. /* FIXME: do we need this test at all? */
  2875. if (! netif_device_present(dev))
  2876. return;
  2877. TRACE_ENTER(dev->name);
  2878. /* The Hermes doesn't seem to have an allmulti mode, so we go
  2879.  * into promiscuous mode and let the upper levels deal. */
  2880. if ( (dev->flags & IFF_PROMISC) || (dev->flags & IFF_ALLMULTI) ||
  2881.      (dev->mc_count > MAX_MULTICAST(priv)) ) {
  2882. promisc = 1;
  2883. mc_count = 0;
  2884. } else {
  2885. promisc = 0;
  2886. mc_count = dev->mc_count;
  2887. }
  2888. if (promisc != priv->promiscuous) {
  2889. err = hermes_write_wordrec(hw, USER_BAP,
  2890.    HERMES_RID_CNFPROMISCUOUSMODE,
  2891.    promisc);
  2892. if (err) {
  2893. printk(KERN_ERR "%s: Error %d setting PROMISCUOUSMODE to 1.n",
  2894.        dev->name, err);
  2895. } else 
  2896. priv->promiscuous = promisc;
  2897. mc_count = 0;
  2898. }
  2899. if (mc_count || priv->mc_count) {
  2900. struct dev_mc_list *p = dev->mc_list;
  2901. hermes_multicast_t mclist;
  2902. int i;
  2903. for (i = 0; i < mc_count; i++) {
  2904. /* Paranoia: */
  2905. if (! p)
  2906. BUG(); /* Multicast list shorter than mc_count */
  2907. if (p->dmi_addrlen != ETH_ALEN)
  2908. BUG(); /* Bad address size in multicast list */
  2909. memcpy(mclist.addr[i], p->dmi_addr, ETH_ALEN);
  2910. p = p->next;
  2911. }
  2912. if (p)
  2913. printk(KERN_WARNING "Multicast list is longer than mc_countn");
  2914. err = hermes_write_ltv(hw, USER_BAP, HERMES_RID_CNFGROUPADDRESSES,
  2915.        HERMES_BYTES_TO_RECLEN(priv->mc_count * ETH_ALEN),
  2916.        &mclist);
  2917. if (err)
  2918. printk(KERN_ERR "%s: Error %d setting multicast list.n",
  2919.        dev->name, err);
  2920. else
  2921. priv->mc_count = mc_count;
  2922. }
  2923. /* Since we can set the promiscuous flag when it wasn't asked
  2924.    for, make sure the net_device knows about it. */
  2925. if (priv->promiscuous)
  2926. dev->flags |= IFF_PROMISC;
  2927. else
  2928. dev->flags &= ~IFF_PROMISC;
  2929. TRACE_EXIT(dev->name);
  2930. }
  2931. /*
  2932.  * procfs stuff
  2933.  */
  2934. static struct proc_dir_entry *dir_base = NULL;
  2935. /*
  2936.  * This function updates the total amount of data printed so far. It then
  2937.  * determines if the amount of data printed into a buffer  has reached the
  2938.  * offset requested. If it hasn't, then the buffer is shifted over so that
  2939.  * the next bit of data can be printed over the old bit. If the total
  2940.  * amount printed so far exceeds the total amount requested, then this
  2941.  * function returns 1, otherwise 0.
  2942.  */
  2943. static int 
  2944. shift_buffer(char *buffer, int requested_offset, int requested_len,
  2945.      int *total, int *slop, char **buf)
  2946. {
  2947. int printed;
  2948. printed = *buf - buffer;
  2949. if (*total + printed <= requested_offset) {
  2950. *total += printed;
  2951. *buf = buffer;
  2952. }
  2953. else {
  2954. if (*total < requested_offset) {
  2955. *slop = requested_offset - *total;
  2956. }
  2957. *total = requested_offset + printed - *slop;
  2958. }
  2959. if (*total > requested_offset + requested_len) {
  2960. return 1;
  2961. }
  2962. else {
  2963. return 0;
  2964. }
  2965. }
  2966. /*
  2967.  * This function calculates the actual start of the requested data
  2968.  * in the buffer. It also calculates actual length of data returned,
  2969.  * which could be less that the amount of data requested.
  2970.  */
  2971. #define PROC_BUFFER_SIZE 4096
  2972. #define PROC_SAFE_SIZE 3072
  2973. static int
  2974. calc_start_len(char *buffer, char **start, int requested_offset,
  2975.        int requested_len, int total, char *buf)
  2976. {
  2977. int return_len, buffer_len;
  2978. buffer_len = buf - buffer;
  2979. if (buffer_len >= PROC_BUFFER_SIZE - 1) {
  2980. printk(KERN_ERR "calc_start_len: exceeded /proc buffer sizen");
  2981. }
  2982. /*
  2983.  * There may be bytes before and after the
  2984.  * chunk that was actually requested.
  2985.  */
  2986. return_len = total - requested_offset;
  2987. if (return_len < 0) {
  2988. return_len = 0;
  2989. }
  2990. *start = buf - return_len;
  2991. if (return_len > requested_len) {
  2992. return_len = requested_len;
  2993. }
  2994. return return_len;
  2995. }
  2996. static int
  2997. orinoco_proc_get_hermes_regs(char *page, char **start, off_t requested_offset,
  2998.    int requested_len, int *eof, void *data)
  2999. {
  3000. struct orinoco_private *priv = (struct orinoco_private *)data;
  3001. struct net_device *dev = &priv->ndev;
  3002. hermes_t *hw = &priv->hw;
  3003. char *buf;
  3004. int total = 0, slop = 0;
  3005. /* Hum, in this case hardware register are probably not readable... */
  3006. if (! netif_device_present(dev))
  3007. return -ENODEV;
  3008. buf = page;
  3009. #define DHERMESREG(name) buf += sprintf(buf, "%-16s: %04xn", #name, hermes_read_regn(hw, name))
  3010. DHERMESREG(CMD);
  3011. DHERMESREG(PARAM0);
  3012. DHERMESREG(PARAM1);
  3013. DHERMESREG(PARAM2);
  3014. DHERMESREG(STATUS);
  3015. DHERMESREG(RESP0);
  3016. DHERMESREG(RESP1);
  3017. DHERMESREG(RESP2);
  3018. DHERMESREG(INFOFID);
  3019. DHERMESREG(RXFID);
  3020. DHERMESREG(ALLOCFID);
  3021. DHERMESREG(TXCOMPLFID);
  3022. DHERMESREG(SELECT0);
  3023. DHERMESREG(OFFSET0);
  3024. DHERMESREG(SELECT1);
  3025. DHERMESREG(OFFSET1);
  3026. DHERMESREG(EVSTAT);
  3027. DHERMESREG(INTEN);
  3028. DHERMESREG(EVACK);
  3029. DHERMESREG(CONTROL);
  3030. DHERMESREG(SWSUPPORT0);
  3031. DHERMESREG(SWSUPPORT1);
  3032. DHERMESREG(SWSUPPORT2);
  3033. DHERMESREG(AUXPAGE);
  3034. DHERMESREG(AUXOFFSET);
  3035. DHERMESREG(AUXDATA);
  3036. #undef DHERMESREG
  3037. shift_buffer(page, requested_offset, requested_len, &total,
  3038.      &slop, &buf);
  3039. return calc_start_len(page, start, requested_offset, requested_len,
  3040.       total, buf);
  3041. }
  3042. struct {
  3043. u16 rid;
  3044. char *name;
  3045. int minlen, maxlen;
  3046. int displaytype;
  3047. #define DISPLAY_WORDS 0
  3048. #define DISPLAY_BYTES 1
  3049. #define DISPLAY_STRING 2
  3050. #define DISPLAY_XSTRING 3
  3051. } record_table[] = {
  3052. #define CNF_WORDS(name) { HERMES_RID_CNF##name, #name, 0, LTV_BUF_SIZE, DISPLAY_WORDS }
  3053. #define CNF_BYTES(name) { HERMES_RID_CNF##name, #name, 0, LTV_BUF_SIZE, DISPLAY_BYTES }
  3054. #define CNF_STRING(name) { HERMES_RID_CNF##name, #name, 0, LTV_BUF_SIZE, DISPLAY_STRING }
  3055. CNF_WORDS(PORTTYPE),
  3056. CNF_BYTES(OWNMACADDR),
  3057. CNF_STRING(DESIREDSSID),
  3058. CNF_WORDS(OWNCHANNEL),
  3059. CNF_STRING(OWNSSID),
  3060. CNF_WORDS(OWNATIMWINDOW),
  3061. CNF_WORDS(SYSTEMSCALE),
  3062. CNF_WORDS(MAXDATALEN),
  3063. CNF_WORDS(PMENABLED),
  3064. CNF_WORDS(PMEPS),
  3065. CNF_WORDS(MULTICASTRECEIVE),
  3066. CNF_WORDS(MAXSLEEPDURATION),
  3067. CNF_WORDS(PMHOLDOVERDURATION),
  3068. CNF_STRING(OWNNAME),
  3069. CNF_WORDS(OWNDTIMPERIOD),
  3070. CNF_WORDS(MULTICASTPMBUFFERING),
  3071. CNF_WORDS(WEPENABLED_AGERE),
  3072. CNF_WORDS(MANDATORYBSSID_SYMBOL),
  3073. CNF_WORDS(WEPDEFAULTKEYID),
  3074. CNF_BYTES(DEFAULTKEY0),
  3075. CNF_BYTES(DEFAULTKEY1),
  3076. CNF_WORDS(MWOROBUST_AGERE),
  3077. CNF_BYTES(DEFAULTKEY2),
  3078. CNF_BYTES(DEFAULTKEY3),
  3079. CNF_WORDS(WEPFLAGS_INTERSIL),
  3080. CNF_WORDS(WEPKEYMAPPINGTABLE),
  3081. CNF_WORDS(AUTHENTICATION),
  3082. CNF_WORDS(MAXASSOCSTA),
  3083. CNF_WORDS(KEYLENGTH_SYMBOL),
  3084. CNF_WORDS(TXCONTROL),
  3085. CNF_WORDS(ROAMINGMODE),
  3086. CNF_WORDS(HOSTAUTHENTICATION),
  3087. CNF_WORDS(RCVCRCERROR),
  3088. CNF_WORDS(MMLIFE),
  3089. CNF_WORDS(ALTRETRYCOUNT),
  3090. CNF_WORDS(BEACONINT),
  3091. CNF_WORDS(APPCFINFO),
  3092. CNF_WORDS(STAPCFINFO),
  3093. CNF_WORDS(PRIORITYQUSAGE),
  3094. CNF_WORDS(TIMCTRL),
  3095. CNF_WORDS(THIRTY2TALLY),
  3096. CNF_WORDS(ENHSECURITY),
  3097. CNF_BYTES(GROUPADDRESSES),
  3098. CNF_WORDS(CREATEIBSS),
  3099. CNF_WORDS(FRAGMENTATIONTHRESHOLD),
  3100. CNF_WORDS(RTSTHRESHOLD),
  3101. CNF_WORDS(TXRATECONTROL),
  3102. CNF_WORDS(PROMISCUOUSMODE),
  3103. CNF_WORDS(BASICRATES_SYMBOL),
  3104. CNF_WORDS(PREAMBLE_SYMBOL),
  3105. CNF_WORDS(SHORTPREAMBLE),
  3106. CNF_BYTES(WEPKEYS_AGERE),
  3107. CNF_WORDS(EXCLUDELONGPREAMBLE),
  3108. CNF_WORDS(TXKEY_AGERE),
  3109. CNF_WORDS(AUTHENTICATIONRSPTO),
  3110. CNF_WORDS(BASICRATES),
  3111. CNF_WORDS(SUPPORTEDRATES),
  3112. CNF_WORDS(TICKTIME),
  3113. CNF_WORDS(SCANREQUEST),
  3114. CNF_WORDS(JOINREQUEST),
  3115. CNF_WORDS(AUTHENTICATESTATION),
  3116. CNF_WORDS(CHANNELINFOREQUEST),
  3117. #undef CNF_WORDS
  3118. #undef CNF_BYTES
  3119. #undef CNF_STRING
  3120. #define INF_WORDS(name) { HERMES_RID_##name, #name, 0, LTV_BUF_SIZE, DISPLAY_WORDS }
  3121. #define INF_BYTES(name) { HERMES_RID_##name, #name, 0, LTV_BUF_SIZE, DISPLAY_BYTES }
  3122. #define INF_STRING(name) { HERMES_RID_##name, #name, 0, LTV_BUF_SIZE, DISPLAY_STRING }
  3123. #define INF_XSTRING(name) { HERMES_RID_##name, #name, 0, LTV_BUF_SIZE, DISPLAY_XSTRING }
  3124. INF_WORDS(MAXLOADTIME),
  3125. INF_WORDS(DOWNLOADBUFFER),
  3126. INF_WORDS(PRIID),
  3127. INF_WORDS(PRISUPRANGE),
  3128. INF_WORDS(CFIACTRANGES),
  3129. INF_WORDS(NICSERNUM),
  3130. INF_WORDS(NICID),
  3131. INF_WORDS(MFISUPRANGE),
  3132. INF_WORDS(CFISUPRANGE),
  3133. INF_WORDS(CHANNELLIST),
  3134. INF_WORDS(REGULATORYDOMAINS),
  3135. INF_WORDS(TEMPTYPE),
  3136. /*   INF_BYTES(CIS), */
  3137. INF_WORDS(STAID),
  3138. INF_STRING(CURRENTSSID),
  3139. INF_BYTES(CURRENTBSSID),
  3140. INF_WORDS(COMMSQUALITY),
  3141. INF_WORDS(CURRENTTXRATE),
  3142. INF_WORDS(CURRENTBEACONINTERVAL),
  3143. INF_WORDS(CURRENTSCALETHRESHOLDS),
  3144. INF_WORDS(PROTOCOLRSPTIME),
  3145. INF_WORDS(SHORTRETRYLIMIT),
  3146. INF_WORDS(LONGRETRYLIMIT),
  3147. INF_WORDS(MAXTRANSMITLIFETIME),
  3148. INF_WORDS(MAXRECEIVELIFETIME),
  3149. INF_WORDS(CFPOLLABLE),
  3150. INF_WORDS(AUTHENTICATIONALGORITHMS),
  3151. INF_WORDS(PRIVACYOPTIONIMPLEMENTED),
  3152. INF_BYTES(OWNMACADDR),
  3153. INF_WORDS(SCANRESULTSTABLE),
  3154. INF_WORDS(PHYTYPE),
  3155. INF_WORDS(CURRENTCHANNEL),
  3156. INF_WORDS(CURRENTPOWERSTATE),
  3157. INF_WORDS(CCAMODE),
  3158. INF_WORDS(SUPPORTEDDATARATES),
  3159. INF_BYTES(BUILDSEQ),
  3160. INF_XSTRING(FWID)
  3161. #undef INF_WORDS
  3162. #undef INF_BYTES
  3163. #undef INF_STRING
  3164. };
  3165. #define NUM_RIDS ( sizeof(record_table) / sizeof(record_table[0]) )
  3166. static int
  3167. orinoco_proc_get_hermes_recs(char *page, char **start, off_t requested_offset,
  3168.    int requested_len, int *eof, void *data)
  3169. {
  3170. struct orinoco_private *priv = (struct orinoco_private *)data;
  3171. struct net_device *dev = &priv->ndev;
  3172. hermes_t *hw = &priv->hw;
  3173. char *buf;
  3174. int total = 0, slop = 0;
  3175. int i;
  3176. u16 length;
  3177. int err;
  3178. /* Hum, in this case hardware register are probably not readable... */
  3179. if (! netif_device_present(dev))
  3180. return -ENODEV;
  3181. buf = page;
  3182. /* print out all the config RIDs */
  3183. for (i = 0; i < NUM_RIDS; i++) {
  3184. u16 rid = record_table[i].rid;
  3185. int minlen = record_table[i].minlen;
  3186. int maxlen = record_table[i].maxlen;
  3187. int len;
  3188. u8 *val8;
  3189. u16 *val16;
  3190. int j;
  3191. val8 = kmalloc(maxlen + 2, GFP_KERNEL);
  3192. if (! val8)
  3193. return -ENOMEM;
  3194. memset(val8, 0, maxlen + 2);
  3195. err = hermes_read_ltv(hw, USER_BAP, rid, maxlen,
  3196.       &length, val8);
  3197. if (err) {
  3198. DEBUG(0, "Error %d reading RID 0x%04xn", err, rid);
  3199. continue;
  3200. }
  3201. val16 = (u16 *)val8;
  3202. if (length == 0)
  3203. continue;
  3204. buf += sprintf(buf, "%-15s (0x%04x): length=%d (%d bytes)tvalue=", record_table[i].name,
  3205.        rid, length, (length-1)*2);
  3206. len = min( (int)max(minlen, ((int)length-1)*2), maxlen);
  3207. switch (record_table[i].displaytype) {
  3208. case DISPLAY_WORDS:
  3209. for (j = 0; j < len / 2; j++) {
  3210. buf += sprintf(buf, "%04X-", le16_to_cpu(val16[j]));
  3211. }
  3212. buf--;
  3213. break;
  3214. case DISPLAY_BYTES:
  3215. default:
  3216. for (j = 0; j < len; j++) {
  3217. buf += sprintf(buf, "%02X:", val8[j]);
  3218. }
  3219. buf--;
  3220. break;
  3221. case DISPLAY_STRING:
  3222. len = min(len, le16_to_cpu(val16[0])+2);
  3223. val8[len] = '';
  3224. buf += sprintf(buf, ""%s"", (char *)&val16[1]);
  3225. break;
  3226. case DISPLAY_XSTRING:
  3227. buf += sprintf(buf, "'%s'", (char *)val8);
  3228. }
  3229. buf += sprintf(buf, "n");
  3230. kfree(val8);
  3231. if (shift_buffer(page, requested_offset, requested_len,
  3232.  &total, &slop, &buf))
  3233. break;
  3234. if ( (buf - page) > PROC_SAFE_SIZE )
  3235. break;
  3236. }
  3237. return calc_start_len(page, start, requested_offset, requested_len,
  3238.       total, buf);
  3239. }
  3240. /* initialise the /proc subsystem for the hermes driver, creating the
  3241.  * separate entries */
  3242. static int
  3243. orinoco_proc_init(void)
  3244. {
  3245. int err = 0;
  3246. TRACE_ENTER("orinoco");
  3247. /* create the directory for it to sit in */
  3248. dir_base = create_proc_entry("hermes", S_IFDIR, &proc_root);
  3249. if (dir_base == NULL) {
  3250. printk(KERN_ERR "Unable to initialise /proc/hermes.n");
  3251. orinoco_proc_cleanup();
  3252. err = -ENOMEM;
  3253. }
  3254. TRACE_EXIT("orinoco");
  3255. return err;
  3256. }
  3257. int
  3258. orinoco_proc_dev_init(struct orinoco_private *priv)
  3259. {
  3260. struct net_device *dev = &priv->ndev;
  3261. priv->dir_dev = NULL;
  3262. /* create the directory for it to sit in */
  3263. priv->dir_dev = create_proc_entry(dev->name, S_IFDIR | S_IRUGO | S_IXUGO,
  3264.   dir_base);
  3265. if (priv->dir_dev == NULL) {
  3266. printk(KERN_ERR "Unable to initialise /proc/hermes/%s.n",  dev->name);
  3267. goto fail;
  3268. }
  3269. priv->dir_regs = NULL;
  3270. priv->dir_regs = create_proc_read_entry("regs", S_IFREG | S_IRUGO,
  3271. priv->dir_dev, orinoco_proc_get_hermes_regs, priv);
  3272. if (priv->dir_regs == NULL) {
  3273. printk(KERN_ERR "Unable to initialise /proc/hermes/%s/regs.n",  dev->name);
  3274. goto fail;
  3275. }
  3276. priv->dir_recs = NULL;
  3277. priv->dir_recs = create_proc_read_entry("recs", S_IFREG | S_IRUGO,
  3278. priv->dir_dev, orinoco_proc_get_hermes_recs, priv);
  3279. if (priv->dir_recs == NULL) {
  3280. printk(KERN_ERR "Unable to initialise /proc/hermes/%s/recs.n",  dev->name);
  3281. goto fail;
  3282. }
  3283. return 0;
  3284.  fail:
  3285. orinoco_proc_dev_cleanup(priv);
  3286. return -ENOMEM;
  3287. }
  3288. void
  3289. orinoco_proc_dev_cleanup(struct orinoco_private *priv)
  3290. {
  3291. struct net_device *dev = &priv->ndev;
  3292. TRACE_ENTER(priv->ndev.name);
  3293. if (priv->dir_regs) {
  3294. remove_proc_entry("regs", priv->dir_dev);
  3295. priv->dir_regs = NULL;
  3296. }
  3297. if (priv->dir_recs) {
  3298. remove_proc_entry("recs", priv->dir_dev);
  3299. priv->dir_recs = NULL;
  3300. }
  3301. if (priv->dir_dev) {
  3302. remove_proc_entry(dev->name, dir_base);
  3303. priv->dir_dev = NULL;
  3304. }
  3305. TRACE_EXIT(priv->ndev.name);
  3306. }
  3307. static void
  3308. orinoco_proc_cleanup(void)
  3309. {
  3310. TRACE_ENTER("orinoco");
  3311. if (dir_base) {
  3312. remove_proc_entry("hermes", &proc_root);
  3313. dir_base = NULL;
  3314. }
  3315. TRACE_EXIT("orinoco");
  3316. }
  3317. int
  3318. orinoco_setup(struct orinoco_private* priv)
  3319. {
  3320. struct net_device *dev = &priv->ndev;;
  3321. spin_lock_init(&priv->lock);
  3322. /* Set up the net_device */
  3323. ether_setup(dev);
  3324. dev->priv = priv;
  3325. /* Setup up default routines */
  3326. priv->card_reset_handler = NULL; /* Caller may override */
  3327. dev->init = orinoco_init;
  3328. dev->open = NULL; /* Caller *must* override */
  3329. dev->stop = NULL;
  3330. dev->hard_start_xmit = orinoco_xmit;
  3331. dev->tx_timeout = orinoco_tx_timeout;
  3332. dev->watchdog_timeo = HZ; /* 1 second timeout */
  3333. dev->get_stats = orinoco_get_stats;
  3334. dev->get_wireless_stats = orinoco_get_wireless_stats;
  3335. dev->do_ioctl = orinoco_ioctl;
  3336. dev->change_mtu = orinoco_change_mtu;
  3337. dev->set_multicast_list = orinoco_set_multicast_list;
  3338. netif_stop_queue(dev);
  3339. return 0;
  3340. }
  3341. #ifdef ORINOCO_DEBUG
  3342. EXPORT_SYMBOL(orinoco_debug);
  3343. #endif
  3344. EXPORT_SYMBOL(orinoco_shutdown);
  3345. EXPORT_SYMBOL(orinoco_reset);
  3346. EXPORT_SYMBOL(orinoco_setup);
  3347. EXPORT_SYMBOL(orinoco_proc_dev_init);
  3348. EXPORT_SYMBOL(orinoco_proc_dev_cleanup);
  3349. EXPORT_SYMBOL(orinoco_interrupt);
  3350. static int __init init_orinoco(void)
  3351. {
  3352. printk(KERN_DEBUG "%sn", version);
  3353. return orinoco_proc_init();
  3354. }
  3355. static void __exit exit_orinoco(void)
  3356. {
  3357. orinoco_proc_cleanup();
  3358. }
  3359. module_init(init_orinoco);
  3360. module_exit(exit_orinoco);