hw.c 33 KB

1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889909192939495969798991001011021031041051061071081091101111121131141151161171181191201211221231241251261271281291301311321331341351361371381391401411421431441451461471481491501511521531541551561571581591601611621631641651661671681691701711721731741751761771781791801811821831841851861871881891901911921931941951961971981992002012022032042052062072082092102112122132142152162172182192202212222232242252262272282292302312322332342352362372382392402412422432442452462472482492502512522532542552562572582592602612622632642652662672682692702712722732742752762772782792802812822832842852862872882892902912922932942952962972982993003013023033043053063073083093103113123133143153163173183193203213223233243253263273283293303313323333343353363373383393403413423433443453463473483493503513523533543553563573583593603613623633643653663673683693703713723733743753763773783793803813823833843853863873883893903913923933943953963973983994004014024034044054064074084094104114124134144154164174184194204214224234244254264274284294304314324334344354364374384394404414424434444454464474484494504514524534544554564574584594604614624634644654664674684694704714724734744754764774784794804814824834844854864874884894904914924934944954964974984995005015025035045055065075085095105115125135145155165175185195205215225235245255265275285295305315325335345355365375385395405415425435445455465475485495505515525535545555565575585595605615625635645655665675685695705715725735745755765775785795805815825835845855865875885895905915925935945955965975985996006016026036046056066076086096106116126136146156166176186196206216226236246256266276286296306316326336346356366376386396406416426436446456466476486496506516526536546556566576586596606616626636646656666676686696706716726736746756766776786796806816826836846856866876886896906916926936946956966976986997007017027037047057067077087097107117127137147157167177187197207217227237247257267277287297307317327337347357367377387397407417427437447457467477487497507517527537547557567577587597607617627637647657667677687697707717727737747757767777787797807817827837847857867877887897907917927937947957967977987998008018028038048058068078088098108118128138148158168178188198208218228238248258268278288298308318328338348358368378388398408418428438448458468478488498508518528538548558568578588598608618628638648658668678688698708718728738748758768778788798808818828838848858868878888898908918928938948958968978988999009019029039049059069079089099109119129139149159169179189199209219229239249259269279289299309319329339349359369379389399409419429439449459469479489499509519529539549559569579589599609619629639649659669679689699709719729739749759769779789799809819829839849859869879889899909919929939949959969979989991000100110021003100410051006100710081009101010111012101310141015101610171018101910201021102210231024102510261027102810291030103110321033103410351036103710381039104010411042104310441045104610471048104910501051105210531054105510561057105810591060106110621063106410651066106710681069107010711072107310741075107610771078107910801081108210831084108510861087108810891090109110921093109410951096109710981099110011011102110311041105110611071108110911101111111211131114111511161117111811191120112111221123112411251126112711281129113011311132113311341135113611371138113911401141114211431144114511461147114811491150115111521153115411551156115711581159116011611162116311641165116611671168116911701171117211731174117511761177117811791180118111821183118411851186118711881189119011911192119311941195119611971198119912001201120212031204120512061207120812091210121112121213121412151216121712181219122012211222122312241225122612271228122912301231123212331234123512361237123812391240124112421243124412451246124712481249125012511252125312541255125612571258125912601261126212631264126512661267126812691270127112721273127412751276127712781279128012811282128312841285128612871288128912901291129212931294129512961297129812991300130113021303130413051306130713081309131013111312
  1. /* Encapsulate basic setting changes and retrieval on Hermes hardware
  2. *
  3. * See copyright notice in main.c
  4. */
  5. #include <linux/kernel.h>
  6. #include <linux/device.h>
  7. #include <linux/if_arp.h>
  8. #include <linux/ieee80211.h>
  9. #include <linux/wireless.h>
  10. #include <net/cfg80211.h>
  11. #include "hermes.h"
  12. #include "hermes_rid.h"
  13. #include "orinoco.h"
  14. #include "hw.h"
  15. #define SYMBOL_MAX_VER_LEN (14)
  16. /* Symbol firmware has a bug allocating buffers larger than this */
  17. #define TX_NICBUF_SIZE_BUG 1585
  18. /********************************************************************/
  19. /* Data tables */
  20. /********************************************************************/
  21. /* This tables gives the actual meanings of the bitrate IDs returned
  22. * by the firmware. */
  23. static const struct {
  24. int bitrate; /* in 100s of kilobits */
  25. int automatic;
  26. u16 agere_txratectrl;
  27. u16 intersil_txratectrl;
  28. } bitrate_table[] = {
  29. {110, 1, 3, 15}, /* Entry 0 is the default */
  30. {10, 0, 1, 1},
  31. {10, 1, 1, 1},
  32. {20, 0, 2, 2},
  33. {20, 1, 6, 3},
  34. {55, 0, 4, 4},
  35. {55, 1, 7, 7},
  36. {110, 0, 5, 8},
  37. };
  38. #define BITRATE_TABLE_SIZE ARRAY_SIZE(bitrate_table)
  39. /* Firmware version encoding */
  40. struct comp_id {
  41. u16 id, variant, major, minor;
  42. } __attribute__ ((packed));
  43. static inline fwtype_t determine_firmware_type(struct comp_id *nic_id)
  44. {
  45. if (nic_id->id < 0x8000)
  46. return FIRMWARE_TYPE_AGERE;
  47. else if (nic_id->id == 0x8000 && nic_id->major == 0)
  48. return FIRMWARE_TYPE_SYMBOL;
  49. else
  50. return FIRMWARE_TYPE_INTERSIL;
  51. }
  52. /* Set priv->firmware type, determine firmware properties
  53. * This function can be called before we have registerred with netdev,
  54. * so all errors go out with dev_* rather than printk
  55. */
  56. int determine_fw_capabilities(struct orinoco_private *priv)
  57. {
  58. struct device *dev = priv->dev;
  59. hermes_t *hw = &priv->hw;
  60. int err;
  61. struct comp_id nic_id, sta_id;
  62. unsigned int firmver;
  63. char tmp[SYMBOL_MAX_VER_LEN+1] __attribute__((aligned(2)));
  64. /* Get the hardware version */
  65. err = HERMES_READ_RECORD(hw, USER_BAP, HERMES_RID_NICID, &nic_id);
  66. if (err) {
  67. dev_err(dev, "Cannot read hardware identity: error %d\n",
  68. err);
  69. return err;
  70. }
  71. le16_to_cpus(&nic_id.id);
  72. le16_to_cpus(&nic_id.variant);
  73. le16_to_cpus(&nic_id.major);
  74. le16_to_cpus(&nic_id.minor);
  75. dev_info(dev, "Hardware identity %04x:%04x:%04x:%04x\n",
  76. nic_id.id, nic_id.variant, nic_id.major, nic_id.minor);
  77. priv->firmware_type = determine_firmware_type(&nic_id);
  78. /* Get the firmware version */
  79. err = HERMES_READ_RECORD(hw, USER_BAP, HERMES_RID_STAID, &sta_id);
  80. if (err) {
  81. dev_err(dev, "Cannot read station identity: error %d\n",
  82. err);
  83. return err;
  84. }
  85. le16_to_cpus(&sta_id.id);
  86. le16_to_cpus(&sta_id.variant);
  87. le16_to_cpus(&sta_id.major);
  88. le16_to_cpus(&sta_id.minor);
  89. dev_info(dev, "Station identity %04x:%04x:%04x:%04x\n",
  90. sta_id.id, sta_id.variant, sta_id.major, sta_id.minor);
  91. switch (sta_id.id) {
  92. case 0x15:
  93. dev_err(dev, "Primary firmware is active\n");
  94. return -ENODEV;
  95. case 0x14b:
  96. dev_err(dev, "Tertiary firmware is active\n");
  97. return -ENODEV;
  98. case 0x1f: /* Intersil, Agere, Symbol Spectrum24 */
  99. case 0x21: /* Symbol Spectrum24 Trilogy */
  100. break;
  101. default:
  102. dev_notice(dev, "Unknown station ID, please report\n");
  103. break;
  104. }
  105. /* Default capabilities */
  106. priv->has_sensitivity = 1;
  107. priv->has_mwo = 0;
  108. priv->has_preamble = 0;
  109. priv->has_port3 = 1;
  110. priv->has_ibss = 1;
  111. priv->has_wep = 0;
  112. priv->has_big_wep = 0;
  113. priv->has_alt_txcntl = 0;
  114. priv->has_ext_scan = 0;
  115. priv->has_wpa = 0;
  116. priv->do_fw_download = 0;
  117. /* Determine capabilities from the firmware version */
  118. switch (priv->firmware_type) {
  119. case FIRMWARE_TYPE_AGERE:
  120. /* Lucent Wavelan IEEE, Lucent Orinoco, Cabletron RoamAbout,
  121. ELSA, Melco, HP, IBM, Dell 1150, Compaq 110/210 */
  122. snprintf(priv->fw_name, sizeof(priv->fw_name) - 1,
  123. "Lucent/Agere %d.%02d", sta_id.major, sta_id.minor);
  124. firmver = ((unsigned long)sta_id.major << 16) | sta_id.minor;
  125. priv->has_ibss = (firmver >= 0x60006);
  126. priv->has_wep = (firmver >= 0x40020);
  127. priv->has_big_wep = 1; /* FIXME: this is wrong - how do we tell
  128. Gold cards from the others? */
  129. priv->has_mwo = (firmver >= 0x60000);
  130. priv->has_pm = (firmver >= 0x40020); /* Don't work in 7.52 ? */
  131. priv->ibss_port = 1;
  132. priv->has_hostscan = (firmver >= 0x8000a);
  133. priv->do_fw_download = 1;
  134. priv->broken_monitor = (firmver >= 0x80000);
  135. priv->has_alt_txcntl = (firmver >= 0x90000); /* All 9.x ? */
  136. priv->has_ext_scan = (firmver >= 0x90000); /* All 9.x ? */
  137. priv->has_wpa = (firmver >= 0x9002a);
  138. /* Tested with Agere firmware :
  139. * 1.16 ; 4.08 ; 4.52 ; 6.04 ; 6.16 ; 7.28 => Jean II
  140. * Tested CableTron firmware : 4.32 => Anton */
  141. break;
  142. case FIRMWARE_TYPE_SYMBOL:
  143. /* Symbol , 3Com AirConnect, Intel, Ericsson WLAN */
  144. /* Intel MAC : 00:02:B3:* */
  145. /* 3Com MAC : 00:50:DA:* */
  146. memset(tmp, 0, sizeof(tmp));
  147. /* Get the Symbol firmware version */
  148. err = hermes_read_ltv(hw, USER_BAP,
  149. HERMES_RID_SECONDARYVERSION_SYMBOL,
  150. SYMBOL_MAX_VER_LEN, NULL, &tmp);
  151. if (err) {
  152. dev_warn(dev, "Error %d reading Symbol firmware info. "
  153. "Wildly guessing capabilities...\n", err);
  154. firmver = 0;
  155. tmp[0] = '\0';
  156. } else {
  157. /* The firmware revision is a string, the format is
  158. * something like : "V2.20-01".
  159. * Quick and dirty parsing... - Jean II
  160. */
  161. firmver = ((tmp[1] - '0') << 16)
  162. | ((tmp[3] - '0') << 12)
  163. | ((tmp[4] - '0') << 8)
  164. | ((tmp[6] - '0') << 4)
  165. | (tmp[7] - '0');
  166. tmp[SYMBOL_MAX_VER_LEN] = '\0';
  167. }
  168. snprintf(priv->fw_name, sizeof(priv->fw_name) - 1,
  169. "Symbol %s", tmp);
  170. priv->has_ibss = (firmver >= 0x20000);
  171. priv->has_wep = (firmver >= 0x15012);
  172. priv->has_big_wep = (firmver >= 0x20000);
  173. priv->has_pm = (firmver >= 0x20000 && firmver < 0x22000) ||
  174. (firmver >= 0x29000 && firmver < 0x30000) ||
  175. firmver >= 0x31000;
  176. priv->has_preamble = (firmver >= 0x20000);
  177. priv->ibss_port = 4;
  178. /* Symbol firmware is found on various cards, but
  179. * there has been no attempt to check firmware
  180. * download on non-spectrum_cs based cards.
  181. *
  182. * Given that the Agere firmware download works
  183. * differently, we should avoid doing a firmware
  184. * download with the Symbol algorithm on non-spectrum
  185. * cards.
  186. *
  187. * For now we can identify a spectrum_cs based card
  188. * because it has a firmware reset function.
  189. */
  190. priv->do_fw_download = (priv->stop_fw != NULL);
  191. priv->broken_disableport = (firmver == 0x25013) ||
  192. (firmver >= 0x30000 && firmver <= 0x31000);
  193. priv->has_hostscan = (firmver >= 0x31001) ||
  194. (firmver >= 0x29057 && firmver < 0x30000);
  195. /* Tested with Intel firmware : 0x20015 => Jean II */
  196. /* Tested with 3Com firmware : 0x15012 & 0x22001 => Jean II */
  197. break;
  198. case FIRMWARE_TYPE_INTERSIL:
  199. /* D-Link, Linksys, Adtron, ZoomAir, and many others...
  200. * Samsung, Compaq 100/200 and Proxim are slightly
  201. * different and less well tested */
  202. /* D-Link MAC : 00:40:05:* */
  203. /* Addtron MAC : 00:90:D1:* */
  204. snprintf(priv->fw_name, sizeof(priv->fw_name) - 1,
  205. "Intersil %d.%d.%d", sta_id.major, sta_id.minor,
  206. sta_id.variant);
  207. firmver = ((unsigned long)sta_id.major << 16) |
  208. ((unsigned long)sta_id.minor << 8) | sta_id.variant;
  209. priv->has_ibss = (firmver >= 0x000700); /* FIXME */
  210. priv->has_big_wep = priv->has_wep = (firmver >= 0x000800);
  211. priv->has_pm = (firmver >= 0x000700);
  212. priv->has_hostscan = (firmver >= 0x010301);
  213. if (firmver >= 0x000800)
  214. priv->ibss_port = 0;
  215. else {
  216. dev_notice(dev, "Intersil firmware earlier than v0.8.x"
  217. " - several features not supported\n");
  218. priv->ibss_port = 1;
  219. }
  220. break;
  221. }
  222. dev_info(dev, "Firmware determined as %s\n", priv->fw_name);
  223. return 0;
  224. }
  225. /* Read settings from EEPROM into our private structure.
  226. * MAC address gets dropped into callers buffer
  227. * Can be called before netdev registration.
  228. */
  229. int orinoco_hw_read_card_settings(struct orinoco_private *priv, u8 *dev_addr)
  230. {
  231. struct device *dev = priv->dev;
  232. struct hermes_idstring nickbuf;
  233. hermes_t *hw = &priv->hw;
  234. int len;
  235. int err;
  236. u16 reclen;
  237. /* Get the MAC address */
  238. err = hermes_read_ltv(hw, USER_BAP, HERMES_RID_CNFOWNMACADDR,
  239. ETH_ALEN, NULL, dev_addr);
  240. if (err) {
  241. dev_warn(dev, "Failed to read MAC address!\n");
  242. goto out;
  243. }
  244. dev_dbg(dev, "MAC address %pM\n", dev_addr);
  245. /* Get the station name */
  246. err = hermes_read_ltv(hw, USER_BAP, HERMES_RID_CNFOWNNAME,
  247. sizeof(nickbuf), &reclen, &nickbuf);
  248. if (err) {
  249. dev_err(dev, "failed to read station name\n");
  250. goto out;
  251. }
  252. if (nickbuf.len)
  253. len = min(IW_ESSID_MAX_SIZE, (int)le16_to_cpu(nickbuf.len));
  254. else
  255. len = min(IW_ESSID_MAX_SIZE, 2 * reclen);
  256. memcpy(priv->nick, &nickbuf.val, len);
  257. priv->nick[len] = '\0';
  258. dev_dbg(dev, "Station name \"%s\"\n", priv->nick);
  259. /* Get allowed channels */
  260. err = hermes_read_wordrec(hw, USER_BAP, HERMES_RID_CHANNELLIST,
  261. &priv->channel_mask);
  262. if (err) {
  263. dev_err(dev, "Failed to read channel list!\n");
  264. goto out;
  265. }
  266. /* Get initial AP density */
  267. err = hermes_read_wordrec(hw, USER_BAP, HERMES_RID_CNFSYSTEMSCALE,
  268. &priv->ap_density);
  269. if (err || priv->ap_density < 1 || priv->ap_density > 3)
  270. priv->has_sensitivity = 0;
  271. /* Get initial RTS threshold */
  272. err = hermes_read_wordrec(hw, USER_BAP, HERMES_RID_CNFRTSTHRESHOLD,
  273. &priv->rts_thresh);
  274. if (err) {
  275. dev_err(dev, "Failed to read RTS threshold!\n");
  276. goto out;
  277. }
  278. /* Get initial fragmentation settings */
  279. if (priv->has_mwo)
  280. err = hermes_read_wordrec(hw, USER_BAP,
  281. HERMES_RID_CNFMWOROBUST_AGERE,
  282. &priv->mwo_robust);
  283. else
  284. err = hermes_read_wordrec(hw, USER_BAP,
  285. HERMES_RID_CNFFRAGMENTATIONTHRESHOLD,
  286. &priv->frag_thresh);
  287. if (err) {
  288. dev_err(dev, "Failed to read fragmentation settings!\n");
  289. goto out;
  290. }
  291. /* Power management setup */
  292. if (priv->has_pm) {
  293. priv->pm_on = 0;
  294. priv->pm_mcast = 1;
  295. err = hermes_read_wordrec(hw, USER_BAP,
  296. HERMES_RID_CNFMAXSLEEPDURATION,
  297. &priv->pm_period);
  298. if (err) {
  299. dev_err(dev, "Failed to read power management "
  300. "period!\n");
  301. goto out;
  302. }
  303. err = hermes_read_wordrec(hw, USER_BAP,
  304. HERMES_RID_CNFPMHOLDOVERDURATION,
  305. &priv->pm_timeout);
  306. if (err) {
  307. dev_err(dev, "Failed to read power management "
  308. "timeout!\n");
  309. goto out;
  310. }
  311. }
  312. /* Preamble setup */
  313. if (priv->has_preamble) {
  314. err = hermes_read_wordrec(hw, USER_BAP,
  315. HERMES_RID_CNFPREAMBLE_SYMBOL,
  316. &priv->preamble);
  317. }
  318. out:
  319. return err;
  320. }
  321. /* Can be called before netdev registration */
  322. int orinoco_hw_allocate_fid(struct orinoco_private *priv)
  323. {
  324. struct device *dev = priv->dev;
  325. struct hermes *hw = &priv->hw;
  326. int err;
  327. err = hermes_allocate(hw, priv->nicbuf_size, &priv->txfid);
  328. if (err == -EIO && priv->nicbuf_size > TX_NICBUF_SIZE_BUG) {
  329. /* Try workaround for old Symbol firmware bug */
  330. priv->nicbuf_size = TX_NICBUF_SIZE_BUG;
  331. err = hermes_allocate(hw, priv->nicbuf_size, &priv->txfid);
  332. dev_warn(dev, "Firmware ALLOC bug detected "
  333. "(old Symbol firmware?). Work around %s\n",
  334. err ? "failed!" : "ok.");
  335. }
  336. return err;
  337. }
  338. int orinoco_get_bitratemode(int bitrate, int automatic)
  339. {
  340. int ratemode = -1;
  341. int i;
  342. if ((bitrate != 10) && (bitrate != 20) &&
  343. (bitrate != 55) && (bitrate != 110))
  344. return ratemode;
  345. for (i = 0; i < BITRATE_TABLE_SIZE; i++) {
  346. if ((bitrate_table[i].bitrate == bitrate) &&
  347. (bitrate_table[i].automatic == automatic)) {
  348. ratemode = i;
  349. break;
  350. }
  351. }
  352. return ratemode;
  353. }
  354. void orinoco_get_ratemode_cfg(int ratemode, int *bitrate, int *automatic)
  355. {
  356. BUG_ON((ratemode < 0) || (ratemode >= BITRATE_TABLE_SIZE));
  357. *bitrate = bitrate_table[ratemode].bitrate * 100000;
  358. *automatic = bitrate_table[ratemode].automatic;
  359. }
  360. int orinoco_hw_program_rids(struct orinoco_private *priv)
  361. {
  362. struct net_device *dev = priv->ndev;
  363. struct wireless_dev *wdev = netdev_priv(dev);
  364. hermes_t *hw = &priv->hw;
  365. int err;
  366. struct hermes_idstring idbuf;
  367. /* Set the MAC address */
  368. err = hermes_write_ltv(hw, USER_BAP, HERMES_RID_CNFOWNMACADDR,
  369. HERMES_BYTES_TO_RECLEN(ETH_ALEN), dev->dev_addr);
  370. if (err) {
  371. printk(KERN_ERR "%s: Error %d setting MAC address\n",
  372. dev->name, err);
  373. return err;
  374. }
  375. /* Set up the link mode */
  376. err = hermes_write_wordrec(hw, USER_BAP, HERMES_RID_CNFPORTTYPE,
  377. priv->port_type);
  378. if (err) {
  379. printk(KERN_ERR "%s: Error %d setting port type\n",
  380. dev->name, err);
  381. return err;
  382. }
  383. /* Set the channel/frequency */
  384. if (priv->channel != 0 && priv->iw_mode != NL80211_IFTYPE_STATION) {
  385. err = hermes_write_wordrec(hw, USER_BAP,
  386. HERMES_RID_CNFOWNCHANNEL,
  387. priv->channel);
  388. if (err) {
  389. printk(KERN_ERR "%s: Error %d setting channel %d\n",
  390. dev->name, err, priv->channel);
  391. return err;
  392. }
  393. }
  394. if (priv->has_ibss) {
  395. u16 createibss;
  396. if ((strlen(priv->desired_essid) == 0) && (priv->createibss)) {
  397. printk(KERN_WARNING "%s: This firmware requires an "
  398. "ESSID in IBSS-Ad-Hoc mode.\n", dev->name);
  399. /* With wvlan_cs, in this case, we would crash.
  400. * hopefully, this driver will behave better...
  401. * Jean II */
  402. createibss = 0;
  403. } else {
  404. createibss = priv->createibss;
  405. }
  406. err = hermes_write_wordrec(hw, USER_BAP,
  407. HERMES_RID_CNFCREATEIBSS,
  408. createibss);
  409. if (err) {
  410. printk(KERN_ERR "%s: Error %d setting CREATEIBSS\n",
  411. dev->name, err);
  412. return err;
  413. }
  414. }
  415. /* Set the desired BSSID */
  416. err = __orinoco_hw_set_wap(priv);
  417. if (err) {
  418. printk(KERN_ERR "%s: Error %d setting AP address\n",
  419. dev->name, err);
  420. return err;
  421. }
  422. /* Set the desired ESSID */
  423. idbuf.len = cpu_to_le16(strlen(priv->desired_essid));
  424. memcpy(&idbuf.val, priv->desired_essid, sizeof(idbuf.val));
  425. /* WinXP wants partner to configure OWNSSID even in IBSS mode. (jimc) */
  426. err = hermes_write_ltv(hw, USER_BAP, HERMES_RID_CNFOWNSSID,
  427. HERMES_BYTES_TO_RECLEN(strlen(priv->desired_essid)+2),
  428. &idbuf);
  429. if (err) {
  430. printk(KERN_ERR "%s: Error %d setting OWNSSID\n",
  431. dev->name, err);
  432. return err;
  433. }
  434. err = hermes_write_ltv(hw, USER_BAP, HERMES_RID_CNFDESIREDSSID,
  435. HERMES_BYTES_TO_RECLEN(strlen(priv->desired_essid)+2),
  436. &idbuf);
  437. if (err) {
  438. printk(KERN_ERR "%s: Error %d setting DESIREDSSID\n",
  439. dev->name, err);
  440. return err;
  441. }
  442. /* Set the station name */
  443. idbuf.len = cpu_to_le16(strlen(priv->nick));
  444. memcpy(&idbuf.val, priv->nick, sizeof(idbuf.val));
  445. err = hermes_write_ltv(hw, USER_BAP, HERMES_RID_CNFOWNNAME,
  446. HERMES_BYTES_TO_RECLEN(strlen(priv->nick)+2),
  447. &idbuf);
  448. if (err) {
  449. printk(KERN_ERR "%s: Error %d setting nickname\n",
  450. dev->name, err);
  451. return err;
  452. }
  453. /* Set AP density */
  454. if (priv->has_sensitivity) {
  455. err = hermes_write_wordrec(hw, USER_BAP,
  456. HERMES_RID_CNFSYSTEMSCALE,
  457. priv->ap_density);
  458. if (err) {
  459. printk(KERN_WARNING "%s: Error %d setting SYSTEMSCALE. "
  460. "Disabling sensitivity control\n",
  461. dev->name, err);
  462. priv->has_sensitivity = 0;
  463. }
  464. }
  465. /* Set RTS threshold */
  466. err = hermes_write_wordrec(hw, USER_BAP, HERMES_RID_CNFRTSTHRESHOLD,
  467. priv->rts_thresh);
  468. if (err) {
  469. printk(KERN_ERR "%s: Error %d setting RTS threshold\n",
  470. dev->name, err);
  471. return err;
  472. }
  473. /* Set fragmentation threshold or MWO robustness */
  474. if (priv->has_mwo)
  475. err = hermes_write_wordrec(hw, USER_BAP,
  476. HERMES_RID_CNFMWOROBUST_AGERE,
  477. priv->mwo_robust);
  478. else
  479. err = hermes_write_wordrec(hw, USER_BAP,
  480. HERMES_RID_CNFFRAGMENTATIONTHRESHOLD,
  481. priv->frag_thresh);
  482. if (err) {
  483. printk(KERN_ERR "%s: Error %d setting fragmentation\n",
  484. dev->name, err);
  485. return err;
  486. }
  487. /* Set bitrate */
  488. err = __orinoco_hw_set_bitrate(priv);
  489. if (err) {
  490. printk(KERN_ERR "%s: Error %d setting bitrate\n",
  491. dev->name, err);
  492. return err;
  493. }
  494. /* Set power management */
  495. if (priv->has_pm) {
  496. err = hermes_write_wordrec(hw, USER_BAP,
  497. HERMES_RID_CNFPMENABLED,
  498. priv->pm_on);
  499. if (err) {
  500. printk(KERN_ERR "%s: Error %d setting up PM\n",
  501. dev->name, err);
  502. return err;
  503. }
  504. err = hermes_write_wordrec(hw, USER_BAP,
  505. HERMES_RID_CNFMULTICASTRECEIVE,
  506. priv->pm_mcast);
  507. if (err) {
  508. printk(KERN_ERR "%s: Error %d setting up PM\n",
  509. dev->name, err);
  510. return err;
  511. }
  512. err = hermes_write_wordrec(hw, USER_BAP,
  513. HERMES_RID_CNFMAXSLEEPDURATION,
  514. priv->pm_period);
  515. if (err) {
  516. printk(KERN_ERR "%s: Error %d setting up PM\n",
  517. dev->name, err);
  518. return err;
  519. }
  520. err = hermes_write_wordrec(hw, USER_BAP,
  521. HERMES_RID_CNFPMHOLDOVERDURATION,
  522. priv->pm_timeout);
  523. if (err) {
  524. printk(KERN_ERR "%s: Error %d setting up PM\n",
  525. dev->name, err);
  526. return err;
  527. }
  528. }
  529. /* Set preamble - only for Symbol so far... */
  530. if (priv->has_preamble) {
  531. err = hermes_write_wordrec(hw, USER_BAP,
  532. HERMES_RID_CNFPREAMBLE_SYMBOL,
  533. priv->preamble);
  534. if (err) {
  535. printk(KERN_ERR "%s: Error %d setting preamble\n",
  536. dev->name, err);
  537. return err;
  538. }
  539. }
  540. /* Set up encryption */
  541. if (priv->has_wep || priv->has_wpa) {
  542. err = __orinoco_hw_setup_enc(priv);
  543. if (err) {
  544. printk(KERN_ERR "%s: Error %d activating encryption\n",
  545. dev->name, err);
  546. return err;
  547. }
  548. }
  549. if (priv->iw_mode == NL80211_IFTYPE_MONITOR) {
  550. /* Enable monitor mode */
  551. dev->type = ARPHRD_IEEE80211;
  552. err = hermes_docmd_wait(hw, HERMES_CMD_TEST |
  553. HERMES_TEST_MONITOR, 0, NULL);
  554. } else {
  555. /* Disable monitor mode */
  556. dev->type = ARPHRD_ETHER;
  557. err = hermes_docmd_wait(hw, HERMES_CMD_TEST |
  558. HERMES_TEST_STOP, 0, NULL);
  559. }
  560. if (err)
  561. return err;
  562. /* Reset promiscuity / multicast*/
  563. priv->promiscuous = 0;
  564. priv->mc_count = 0;
  565. /* Record mode change */
  566. wdev->iftype = priv->iw_mode;
  567. return 0;
  568. }
  569. /* Get tsc from the firmware */
  570. int orinoco_hw_get_tkip_iv(struct orinoco_private *priv, int key, u8 *tsc)
  571. {
  572. hermes_t *hw = &priv->hw;
  573. int err = 0;
  574. u8 tsc_arr[4][ORINOCO_SEQ_LEN];
  575. if ((key < 0) || (key > 4))
  576. return -EINVAL;
  577. err = hermes_read_ltv(hw, USER_BAP, HERMES_RID_CURRENT_TKIP_IV,
  578. sizeof(tsc_arr), NULL, &tsc_arr);
  579. if (!err)
  580. memcpy(tsc, &tsc_arr[key][0], sizeof(tsc_arr[0]));
  581. return err;
  582. }
  583. int __orinoco_hw_set_bitrate(struct orinoco_private *priv)
  584. {
  585. hermes_t *hw = &priv->hw;
  586. int ratemode = priv->bitratemode;
  587. int err = 0;
  588. if (ratemode >= BITRATE_TABLE_SIZE) {
  589. printk(KERN_ERR "%s: BUG: Invalid bitrate mode %d\n",
  590. priv->ndev->name, ratemode);
  591. return -EINVAL;
  592. }
  593. switch (priv->firmware_type) {
  594. case FIRMWARE_TYPE_AGERE:
  595. err = hermes_write_wordrec(hw, USER_BAP,
  596. HERMES_RID_CNFTXRATECONTROL,
  597. bitrate_table[ratemode].agere_txratectrl);
  598. break;
  599. case FIRMWARE_TYPE_INTERSIL:
  600. case FIRMWARE_TYPE_SYMBOL:
  601. err = hermes_write_wordrec(hw, USER_BAP,
  602. HERMES_RID_CNFTXRATECONTROL,
  603. bitrate_table[ratemode].intersil_txratectrl);
  604. break;
  605. default:
  606. BUG();
  607. }
  608. return err;
  609. }
  610. int orinoco_hw_get_act_bitrate(struct orinoco_private *priv, int *bitrate)
  611. {
  612. hermes_t *hw = &priv->hw;
  613. int i;
  614. int err = 0;
  615. u16 val;
  616. err = hermes_read_wordrec(hw, USER_BAP,
  617. HERMES_RID_CURRENTTXRATE, &val);
  618. if (err)
  619. return err;
  620. switch (priv->firmware_type) {
  621. case FIRMWARE_TYPE_AGERE: /* Lucent style rate */
  622. /* Note : in Lucent firmware, the return value of
  623. * HERMES_RID_CURRENTTXRATE is the bitrate in Mb/s,
  624. * and therefore is totally different from the
  625. * encoding of HERMES_RID_CNFTXRATECONTROL.
  626. * Don't forget that 6Mb/s is really 5.5Mb/s */
  627. if (val == 6)
  628. *bitrate = 5500000;
  629. else
  630. *bitrate = val * 1000000;
  631. break;
  632. case FIRMWARE_TYPE_INTERSIL: /* Intersil style rate */
  633. case FIRMWARE_TYPE_SYMBOL: /* Symbol style rate */
  634. for (i = 0; i < BITRATE_TABLE_SIZE; i++)
  635. if (bitrate_table[i].intersil_txratectrl == val)
  636. break;
  637. if (i >= BITRATE_TABLE_SIZE)
  638. printk(KERN_INFO "%s: Unable to determine current bitrate (0x%04hx)\n",
  639. priv->ndev->name, val);
  640. *bitrate = bitrate_table[i].bitrate * 100000;
  641. break;
  642. default:
  643. BUG();
  644. }
  645. return err;
  646. }
  647. /* Set fixed AP address */
  648. int __orinoco_hw_set_wap(struct orinoco_private *priv)
  649. {
  650. int roaming_flag;
  651. int err = 0;
  652. hermes_t *hw = &priv->hw;
  653. switch (priv->firmware_type) {
  654. case FIRMWARE_TYPE_AGERE:
  655. /* not supported */
  656. break;
  657. case FIRMWARE_TYPE_INTERSIL:
  658. if (priv->bssid_fixed)
  659. roaming_flag = 2;
  660. else
  661. roaming_flag = 1;
  662. err = hermes_write_wordrec(hw, USER_BAP,
  663. HERMES_RID_CNFROAMINGMODE,
  664. roaming_flag);
  665. break;
  666. case FIRMWARE_TYPE_SYMBOL:
  667. err = HERMES_WRITE_RECORD(hw, USER_BAP,
  668. HERMES_RID_CNFMANDATORYBSSID_SYMBOL,
  669. &priv->desired_bssid);
  670. break;
  671. }
  672. return err;
  673. }
  674. /* Change the WEP keys and/or the current keys. Can be called
  675. * either from __orinoco_hw_setup_enc() or directly from
  676. * orinoco_ioctl_setiwencode(). In the later case the association
  677. * with the AP is not broken (if the firmware can handle it),
  678. * which is needed for 802.1x implementations. */
  679. int __orinoco_hw_setup_wepkeys(struct orinoco_private *priv)
  680. {
  681. hermes_t *hw = &priv->hw;
  682. int err = 0;
  683. int i;
  684. switch (priv->firmware_type) {
  685. case FIRMWARE_TYPE_AGERE:
  686. {
  687. struct orinoco_key keys[ORINOCO_MAX_KEYS];
  688. memset(&keys, 0, sizeof(keys));
  689. for (i = 0; i < ORINOCO_MAX_KEYS; i++) {
  690. int len = min(priv->keys[i].key_len,
  691. ORINOCO_MAX_KEY_SIZE);
  692. memcpy(&keys[i].data, priv->keys[i].key, len);
  693. if (len > SMALL_KEY_SIZE)
  694. keys[i].len = cpu_to_le16(LARGE_KEY_SIZE);
  695. else if (len > 0)
  696. keys[i].len = cpu_to_le16(SMALL_KEY_SIZE);
  697. else
  698. keys[i].len = cpu_to_le16(0);
  699. }
  700. err = HERMES_WRITE_RECORD(hw, USER_BAP,
  701. HERMES_RID_CNFWEPKEYS_AGERE,
  702. &keys);
  703. if (err)
  704. return err;
  705. err = hermes_write_wordrec(hw, USER_BAP,
  706. HERMES_RID_CNFTXKEY_AGERE,
  707. priv->tx_key);
  708. if (err)
  709. return err;
  710. break;
  711. }
  712. case FIRMWARE_TYPE_INTERSIL:
  713. case FIRMWARE_TYPE_SYMBOL:
  714. {
  715. int keylen;
  716. /* Force uniform key length to work around
  717. * firmware bugs */
  718. keylen = priv->keys[priv->tx_key].key_len;
  719. if (keylen > LARGE_KEY_SIZE) {
  720. printk(KERN_ERR "%s: BUG: Key %d has oversize length %d.\n",
  721. priv->ndev->name, priv->tx_key, keylen);
  722. return -E2BIG;
  723. } else if (keylen > SMALL_KEY_SIZE)
  724. keylen = LARGE_KEY_SIZE;
  725. else if (keylen > 0)
  726. keylen = SMALL_KEY_SIZE;
  727. else
  728. keylen = 0;
  729. /* Write all 4 keys */
  730. for (i = 0; i < ORINOCO_MAX_KEYS; i++) {
  731. u8 key[LARGE_KEY_SIZE] = { 0 };
  732. memcpy(key, priv->keys[i].key,
  733. priv->keys[i].key_len);
  734. err = hermes_write_ltv(hw, USER_BAP,
  735. HERMES_RID_CNFDEFAULTKEY0 + i,
  736. HERMES_BYTES_TO_RECLEN(keylen),
  737. key);
  738. if (err)
  739. return err;
  740. }
  741. /* Write the index of the key used in transmission */
  742. err = hermes_write_wordrec(hw, USER_BAP,
  743. HERMES_RID_CNFWEPDEFAULTKEYID,
  744. priv->tx_key);
  745. if (err)
  746. return err;
  747. }
  748. break;
  749. }
  750. return 0;
  751. }
  752. int __orinoco_hw_setup_enc(struct orinoco_private *priv)
  753. {
  754. hermes_t *hw = &priv->hw;
  755. int err = 0;
  756. int master_wep_flag;
  757. int auth_flag;
  758. int enc_flag;
  759. /* Setup WEP keys */
  760. if (priv->encode_alg == ORINOCO_ALG_WEP)
  761. __orinoco_hw_setup_wepkeys(priv);
  762. if (priv->wep_restrict)
  763. auth_flag = HERMES_AUTH_SHARED_KEY;
  764. else
  765. auth_flag = HERMES_AUTH_OPEN;
  766. if (priv->wpa_enabled)
  767. enc_flag = 2;
  768. else if (priv->encode_alg == ORINOCO_ALG_WEP)
  769. enc_flag = 1;
  770. else
  771. enc_flag = 0;
  772. switch (priv->firmware_type) {
  773. case FIRMWARE_TYPE_AGERE: /* Agere style WEP */
  774. if (priv->encode_alg == ORINOCO_ALG_WEP) {
  775. /* Enable the shared-key authentication. */
  776. err = hermes_write_wordrec(hw, USER_BAP,
  777. HERMES_RID_CNFAUTHENTICATION_AGERE,
  778. auth_flag);
  779. }
  780. err = hermes_write_wordrec(hw, USER_BAP,
  781. HERMES_RID_CNFWEPENABLED_AGERE,
  782. enc_flag);
  783. if (err)
  784. return err;
  785. if (priv->has_wpa) {
  786. /* Set WPA key management */
  787. err = hermes_write_wordrec(hw, USER_BAP,
  788. HERMES_RID_CNFSETWPAAUTHMGMTSUITE_AGERE,
  789. priv->key_mgmt);
  790. if (err)
  791. return err;
  792. }
  793. break;
  794. case FIRMWARE_TYPE_INTERSIL: /* Intersil style WEP */
  795. case FIRMWARE_TYPE_SYMBOL: /* Symbol style WEP */
  796. if (priv->encode_alg == ORINOCO_ALG_WEP) {
  797. if (priv->wep_restrict ||
  798. (priv->firmware_type == FIRMWARE_TYPE_SYMBOL))
  799. master_wep_flag = HERMES_WEP_PRIVACY_INVOKED |
  800. HERMES_WEP_EXCL_UNENCRYPTED;
  801. else
  802. master_wep_flag = HERMES_WEP_PRIVACY_INVOKED;
  803. err = hermes_write_wordrec(hw, USER_BAP,
  804. HERMES_RID_CNFAUTHENTICATION,
  805. auth_flag);
  806. if (err)
  807. return err;
  808. } else
  809. master_wep_flag = 0;
  810. if (priv->iw_mode == NL80211_IFTYPE_MONITOR)
  811. master_wep_flag |= HERMES_WEP_HOST_DECRYPT;
  812. /* Master WEP setting : on/off */
  813. err = hermes_write_wordrec(hw, USER_BAP,
  814. HERMES_RID_CNFWEPFLAGS_INTERSIL,
  815. master_wep_flag);
  816. if (err)
  817. return err;
  818. break;
  819. }
  820. return 0;
  821. }
  822. /* key must be 32 bytes, including the tx and rx MIC keys.
  823. * rsc must be NULL or up to 8 bytes
  824. * tsc must be NULL or up to 8 bytes
  825. */
  826. int __orinoco_hw_set_tkip_key(struct orinoco_private *priv, int key_idx,
  827. int set_tx, u8 *key, u8 *rsc, size_t rsc_len,
  828. u8 *tsc, size_t tsc_len)
  829. {
  830. struct {
  831. __le16 idx;
  832. u8 rsc[ORINOCO_SEQ_LEN];
  833. u8 key[TKIP_KEYLEN];
  834. u8 tx_mic[MIC_KEYLEN];
  835. u8 rx_mic[MIC_KEYLEN];
  836. u8 tsc[ORINOCO_SEQ_LEN];
  837. } __attribute__ ((packed)) buf;
  838. hermes_t *hw = &priv->hw;
  839. int ret;
  840. int err;
  841. int k;
  842. u16 xmitting;
  843. key_idx &= 0x3;
  844. if (set_tx)
  845. key_idx |= 0x8000;
  846. buf.idx = cpu_to_le16(key_idx);
  847. memcpy(buf.key, key,
  848. sizeof(buf.key) + sizeof(buf.tx_mic) + sizeof(buf.rx_mic));
  849. if (rsc_len > sizeof(buf.rsc))
  850. rsc_len = sizeof(buf.rsc);
  851. if (tsc_len > sizeof(buf.tsc))
  852. tsc_len = sizeof(buf.tsc);
  853. memset(buf.rsc, 0, sizeof(buf.rsc));
  854. memset(buf.tsc, 0, sizeof(buf.tsc));
  855. if (rsc != NULL)
  856. memcpy(buf.rsc, rsc, rsc_len);
  857. if (tsc != NULL)
  858. memcpy(buf.tsc, tsc, tsc_len);
  859. else
  860. buf.tsc[4] = 0x10;
  861. /* Wait upto 100ms for tx queue to empty */
  862. for (k = 100; k > 0; k--) {
  863. udelay(1000);
  864. ret = hermes_read_wordrec(hw, USER_BAP, HERMES_RID_TXQUEUEEMPTY,
  865. &xmitting);
  866. if (ret || !xmitting)
  867. break;
  868. }
  869. if (k == 0)
  870. ret = -ETIMEDOUT;
  871. err = HERMES_WRITE_RECORD(hw, USER_BAP,
  872. HERMES_RID_CNFADDDEFAULTTKIPKEY_AGERE,
  873. &buf);
  874. return ret ? ret : err;
  875. }
  876. int orinoco_clear_tkip_key(struct orinoco_private *priv, int key_idx)
  877. {
  878. hermes_t *hw = &priv->hw;
  879. int err;
  880. err = hermes_write_wordrec(hw, USER_BAP,
  881. HERMES_RID_CNFREMDEFAULTTKIPKEY_AGERE,
  882. key_idx);
  883. if (err)
  884. printk(KERN_WARNING "%s: Error %d clearing TKIP key %d\n",
  885. priv->ndev->name, err, key_idx);
  886. return err;
  887. }
  888. int __orinoco_hw_set_multicast_list(struct orinoco_private *priv,
  889. struct dev_addr_list *mc_list,
  890. int mc_count, int promisc)
  891. {
  892. hermes_t *hw = &priv->hw;
  893. int err = 0;
  894. if (promisc != priv->promiscuous) {
  895. err = hermes_write_wordrec(hw, USER_BAP,
  896. HERMES_RID_CNFPROMISCUOUSMODE,
  897. promisc);
  898. if (err) {
  899. printk(KERN_ERR "%s: Error %d setting PROMISCUOUSMODE to 1.\n",
  900. priv->ndev->name, err);
  901. } else
  902. priv->promiscuous = promisc;
  903. }
  904. /* If we're not in promiscuous mode, then we need to set the
  905. * group address if either we want to multicast, or if we were
  906. * multicasting and want to stop */
  907. if (!promisc && (mc_count || priv->mc_count)) {
  908. struct dev_mc_list *p = mc_list;
  909. struct hermes_multicast mclist;
  910. int i;
  911. for (i = 0; i < mc_count; i++) {
  912. /* paranoia: is list shorter than mc_count? */
  913. BUG_ON(!p);
  914. /* paranoia: bad address size in list? */
  915. BUG_ON(p->dmi_addrlen != ETH_ALEN);
  916. memcpy(mclist.addr[i], p->dmi_addr, ETH_ALEN);
  917. p = p->next;
  918. }
  919. if (p)
  920. printk(KERN_WARNING "%s: Multicast list is "
  921. "longer than mc_count\n", priv->ndev->name);
  922. err = hermes_write_ltv(hw, USER_BAP,
  923. HERMES_RID_CNFGROUPADDRESSES,
  924. HERMES_BYTES_TO_RECLEN(mc_count * ETH_ALEN),
  925. &mclist);
  926. if (err)
  927. printk(KERN_ERR "%s: Error %d setting multicast list.\n",
  928. priv->ndev->name, err);
  929. else
  930. priv->mc_count = mc_count;
  931. }
  932. return err;
  933. }
  934. /* Return : < 0 -> error code ; >= 0 -> length */
  935. int orinoco_hw_get_essid(struct orinoco_private *priv, int *active,
  936. char buf[IW_ESSID_MAX_SIZE+1])
  937. {
  938. hermes_t *hw = &priv->hw;
  939. int err = 0;
  940. struct hermes_idstring essidbuf;
  941. char *p = (char *)(&essidbuf.val);
  942. int len;
  943. unsigned long flags;
  944. if (orinoco_lock(priv, &flags) != 0)
  945. return -EBUSY;
  946. if (strlen(priv->desired_essid) > 0) {
  947. /* We read the desired SSID from the hardware rather
  948. than from priv->desired_essid, just in case the
  949. firmware is allowed to change it on us. I'm not
  950. sure about this */
  951. /* My guess is that the OWNSSID should always be whatever
  952. * we set to the card, whereas CURRENT_SSID is the one that
  953. * may change... - Jean II */
  954. u16 rid;
  955. *active = 1;
  956. rid = (priv->port_type == 3) ? HERMES_RID_CNFOWNSSID :
  957. HERMES_RID_CNFDESIREDSSID;
  958. err = hermes_read_ltv(hw, USER_BAP, rid, sizeof(essidbuf),
  959. NULL, &essidbuf);
  960. if (err)
  961. goto fail_unlock;
  962. } else {
  963. *active = 0;
  964. err = hermes_read_ltv(hw, USER_BAP, HERMES_RID_CURRENTSSID,
  965. sizeof(essidbuf), NULL, &essidbuf);
  966. if (err)
  967. goto fail_unlock;
  968. }
  969. len = le16_to_cpu(essidbuf.len);
  970. BUG_ON(len > IW_ESSID_MAX_SIZE);
  971. memset(buf, 0, IW_ESSID_MAX_SIZE);
  972. memcpy(buf, p, len);
  973. err = len;
  974. fail_unlock:
  975. orinoco_unlock(priv, &flags);
  976. return err;
  977. }
  978. int orinoco_hw_get_freq(struct orinoco_private *priv)
  979. {
  980. hermes_t *hw = &priv->hw;
  981. int err = 0;
  982. u16 channel;
  983. int freq = 0;
  984. unsigned long flags;
  985. if (orinoco_lock(priv, &flags) != 0)
  986. return -EBUSY;
  987. err = hermes_read_wordrec(hw, USER_BAP, HERMES_RID_CURRENTCHANNEL,
  988. &channel);
  989. if (err)
  990. goto out;
  991. /* Intersil firmware 1.3.5 returns 0 when the interface is down */
  992. if (channel == 0) {
  993. err = -EBUSY;
  994. goto out;
  995. }
  996. if ((channel < 1) || (channel > NUM_CHANNELS)) {
  997. printk(KERN_WARNING "%s: Channel out of range (%d)!\n",
  998. priv->ndev->name, channel);
  999. err = -EBUSY;
  1000. goto out;
  1001. }
  1002. freq = ieee80211_dsss_chan_to_freq(channel);
  1003. out:
  1004. orinoco_unlock(priv, &flags);
  1005. if (err > 0)
  1006. err = -EBUSY;
  1007. return err ? err : freq;
  1008. }
  1009. int orinoco_hw_get_bitratelist(struct orinoco_private *priv,
  1010. int *numrates, s32 *rates, int max)
  1011. {
  1012. hermes_t *hw = &priv->hw;
  1013. struct hermes_idstring list;
  1014. unsigned char *p = (unsigned char *)&list.val;
  1015. int err = 0;
  1016. int num;
  1017. int i;
  1018. unsigned long flags;
  1019. if (orinoco_lock(priv, &flags) != 0)
  1020. return -EBUSY;
  1021. err = hermes_read_ltv(hw, USER_BAP, HERMES_RID_SUPPORTEDDATARATES,
  1022. sizeof(list), NULL, &list);
  1023. orinoco_unlock(priv, &flags);
  1024. if (err)
  1025. return err;
  1026. num = le16_to_cpu(list.len);
  1027. *numrates = num;
  1028. num = min(num, max);
  1029. for (i = 0; i < num; i++)
  1030. rates[i] = (p[i] & 0x7f) * 500000; /* convert to bps */
  1031. return 0;
  1032. }
  1033. int orinoco_hw_trigger_scan(struct orinoco_private *priv,
  1034. const struct cfg80211_ssid *ssid)
  1035. {
  1036. struct net_device *dev = priv->ndev;
  1037. hermes_t *hw = &priv->hw;
  1038. unsigned long flags;
  1039. int err = 0;
  1040. if (orinoco_lock(priv, &flags) != 0)
  1041. return -EBUSY;
  1042. /* Scanning with port 0 disabled would fail */
  1043. if (!netif_running(dev)) {
  1044. err = -ENETDOWN;
  1045. goto out;
  1046. }
  1047. /* In monitor mode, the scan results are always empty.
  1048. * Probe responses are passed to the driver as received
  1049. * frames and could be processed in software. */
  1050. if (priv->iw_mode == NL80211_IFTYPE_MONITOR) {
  1051. err = -EOPNOTSUPP;
  1052. goto out;
  1053. }
  1054. if (priv->has_hostscan) {
  1055. switch (priv->firmware_type) {
  1056. case FIRMWARE_TYPE_SYMBOL:
  1057. err = hermes_write_wordrec(hw, USER_BAP,
  1058. HERMES_RID_CNFHOSTSCAN_SYMBOL,
  1059. HERMES_HOSTSCAN_SYMBOL_ONCE |
  1060. HERMES_HOSTSCAN_SYMBOL_BCAST);
  1061. break;
  1062. case FIRMWARE_TYPE_INTERSIL: {
  1063. __le16 req[3];
  1064. req[0] = cpu_to_le16(0x3fff); /* All channels */
  1065. req[1] = cpu_to_le16(0x0001); /* rate 1 Mbps */
  1066. req[2] = 0; /* Any ESSID */
  1067. err = HERMES_WRITE_RECORD(hw, USER_BAP,
  1068. HERMES_RID_CNFHOSTSCAN, &req);
  1069. break;
  1070. }
  1071. case FIRMWARE_TYPE_AGERE:
  1072. if (ssid->ssid_len > 0) {
  1073. struct hermes_idstring idbuf;
  1074. size_t len = ssid->ssid_len;
  1075. idbuf.len = cpu_to_le16(len);
  1076. memcpy(idbuf.val, ssid->ssid, len);
  1077. err = hermes_write_ltv(hw, USER_BAP,
  1078. HERMES_RID_CNFSCANSSID_AGERE,
  1079. HERMES_BYTES_TO_RECLEN(len + 2),
  1080. &idbuf);
  1081. } else
  1082. err = hermes_write_wordrec(hw, USER_BAP,
  1083. HERMES_RID_CNFSCANSSID_AGERE,
  1084. 0); /* Any ESSID */
  1085. if (err)
  1086. break;
  1087. if (priv->has_ext_scan) {
  1088. err = hermes_write_wordrec(hw, USER_BAP,
  1089. HERMES_RID_CNFSCANCHANNELS2GHZ,
  1090. 0x7FFF);
  1091. if (err)
  1092. goto out;
  1093. err = hermes_inquire(hw,
  1094. HERMES_INQ_CHANNELINFO);
  1095. } else
  1096. err = hermes_inquire(hw, HERMES_INQ_SCAN);
  1097. break;
  1098. }
  1099. } else
  1100. err = hermes_inquire(hw, HERMES_INQ_SCAN);
  1101. out:
  1102. orinoco_unlock(priv, &flags);
  1103. return err;
  1104. }
  1105. /* Disassociate from node with BSSID addr */
  1106. int orinoco_hw_disassociate(struct orinoco_private *priv,
  1107. u8 *addr, u16 reason_code)
  1108. {
  1109. hermes_t *hw = &priv->hw;
  1110. int err;
  1111. struct {
  1112. u8 addr[ETH_ALEN];
  1113. __le16 reason_code;
  1114. } __attribute__ ((packed)) buf;
  1115. /* Currently only supported by WPA enabled Agere fw */
  1116. if (!priv->has_wpa)
  1117. return -EOPNOTSUPP;
  1118. memcpy(buf.addr, addr, ETH_ALEN);
  1119. buf.reason_code = cpu_to_le16(reason_code);
  1120. err = HERMES_WRITE_RECORD(hw, USER_BAP,
  1121. HERMES_RID_CNFDISASSOCIATE,
  1122. &buf);
  1123. return err;
  1124. }
  1125. int orinoco_hw_get_current_bssid(struct orinoco_private *priv,
  1126. u8 *addr)
  1127. {
  1128. hermes_t *hw = &priv->hw;
  1129. int err;
  1130. err = hermes_read_ltv(hw, USER_BAP, HERMES_RID_CURRENTBSSID,
  1131. ETH_ALEN, NULL, addr);
  1132. return err;
  1133. }