hw.c 33 KB

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