hw.c 31 KB

12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788899091929394959697989910010110210310410510610710810911011111211311411511611711811912012112212312412512612712812913013113213313413513613713813914014114214314414514614714814915015115215315415515615715815916016116216316416516616716816917017117217317417517617717817918018118218318418518618718818919019119219319419519619719819920020120220320420520620720820921021121221321421521621721821922022122222322422522622722822923023123223323423523623723823924024124224324424524624724824925025125225325425525625725825926026126226326426526626726826927027127227327427527627727827928028128228328428528628728828929029129229329429529629729829930030130230330430530630730830931031131231331431531631731831932032132232332432532632732832933033133233333433533633733833934034134234334434534634734834935035135235335435535635735835936036136236336436536636736836937037137237337437537637737837938038138238338438538638738838939039139239339439539639739839940040140240340440540640740840941041141241341441541641741841942042142242342442542642742842943043143243343443543643743843944044144244344444544644744844945045145245345445545645745845946046146246346446546646746846947047147247347447547647747847948048148248348448548648748848949049149249349449549649749849950050150250350450550650750850951051151251351451551651751851952052152252352452552652752852953053153253353453553653753853954054154254354454554654754854955055155255355455555655755855956056156256356456556656756856957057157257357457557657757857958058158258358458558658758858959059159259359459559659759859960060160260360460560660760860961061161261361461561661761861962062162262362462562662762862963063163263363463563663763863964064164264364464564664764864965065165265365465565665765865966066166266366466566666766866967067167267367467567667767867968068168268368468568668768868969069169269369469569669769869970070170270370470570670770870971071171271371471571671771871972072172272372472572672772872973073173273373473573673773873974074174274374474574674774874975075175275375475575675775875976076176276376476576676776876977077177277377477577677777877978078178278378478578678778878979079179279379479579679779879980080180280380480580680780880981081181281381481581681781881982082182282382482582682782882983083183283383483583683783883984084184284384484584684784884985085185285385485585685785885986086186286386486586686786886987087187287387487587687787887988088188288388488588688788888989089189289389489589689789889990090190290390490590690790890991091191291391491591691791891992092192292392492592692792892993093193293393493593693793893994094194294394494594694794894995095195295395495595695795895996096196296396496596696796896997097197297397497597697797897998098198298398498598698798898999099199299399499599699799899910001001100210031004100510061007100810091010101110121013101410151016101710181019102010211022102310241025102610271028102910301031103210331034103510361037103810391040104110421043104410451046104710481049105010511052105310541055105610571058105910601061106210631064106510661067106810691070107110721073107410751076107710781079108010811082108310841085108610871088108910901091109210931094109510961097109810991100110111021103110411051106110711081109111011111112111311141115111611171118111911201121112211231124112511261127112811291130113111321133113411351136113711381139114011411142114311441145114611471148114911501151115211531154115511561157115811591160116111621163116411651166116711681169117011711172117311741175117611771178117911801181118211831184118511861187118811891190119111921193119411951196119711981199120012011202120312041205120612071208120912101211121212131214121512161217121812191220122112221223122412251226122712281229123012311232123312341235123612371238123912401241124212431244
  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][IW_ENCODE_SEQ_MAX_SIZE];
  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. switch (priv->firmware_type) {
  684. case FIRMWARE_TYPE_AGERE:
  685. err = HERMES_WRITE_RECORD(hw, USER_BAP,
  686. HERMES_RID_CNFWEPKEYS_AGERE,
  687. &priv->keys);
  688. if (err)
  689. return err;
  690. err = hermes_write_wordrec(hw, USER_BAP,
  691. HERMES_RID_CNFTXKEY_AGERE,
  692. priv->tx_key);
  693. if (err)
  694. return err;
  695. break;
  696. case FIRMWARE_TYPE_INTERSIL:
  697. case FIRMWARE_TYPE_SYMBOL:
  698. {
  699. int keylen;
  700. int i;
  701. /* Force uniform key length to work around
  702. * firmware bugs */
  703. keylen = le16_to_cpu(priv->keys[priv->tx_key].len);
  704. if (keylen > LARGE_KEY_SIZE) {
  705. printk(KERN_ERR "%s: BUG: Key %d has oversize length %d.\n",
  706. priv->ndev->name, priv->tx_key, keylen);
  707. return -E2BIG;
  708. }
  709. /* Write all 4 keys */
  710. for (i = 0; i < ORINOCO_MAX_KEYS; i++) {
  711. err = hermes_write_ltv(hw, USER_BAP,
  712. HERMES_RID_CNFDEFAULTKEY0 + i,
  713. HERMES_BYTES_TO_RECLEN(keylen),
  714. priv->keys[i].data);
  715. if (err)
  716. return err;
  717. }
  718. /* Write the index of the key used in transmission */
  719. err = hermes_write_wordrec(hw, USER_BAP,
  720. HERMES_RID_CNFWEPDEFAULTKEYID,
  721. priv->tx_key);
  722. if (err)
  723. return err;
  724. }
  725. break;
  726. }
  727. return 0;
  728. }
  729. int __orinoco_hw_setup_enc(struct orinoco_private *priv)
  730. {
  731. hermes_t *hw = &priv->hw;
  732. int err = 0;
  733. int master_wep_flag;
  734. int auth_flag;
  735. int enc_flag;
  736. /* Setup WEP keys for WEP and WPA */
  737. if (priv->encode_alg)
  738. __orinoco_hw_setup_wepkeys(priv);
  739. if (priv->wep_restrict)
  740. auth_flag = HERMES_AUTH_SHARED_KEY;
  741. else
  742. auth_flag = HERMES_AUTH_OPEN;
  743. if (priv->wpa_enabled)
  744. enc_flag = 2;
  745. else if (priv->encode_alg == IW_ENCODE_ALG_WEP)
  746. enc_flag = 1;
  747. else
  748. enc_flag = 0;
  749. switch (priv->firmware_type) {
  750. case FIRMWARE_TYPE_AGERE: /* Agere style WEP */
  751. if (priv->encode_alg == IW_ENCODE_ALG_WEP) {
  752. /* Enable the shared-key authentication. */
  753. err = hermes_write_wordrec(hw, USER_BAP,
  754. HERMES_RID_CNFAUTHENTICATION_AGERE,
  755. auth_flag);
  756. }
  757. err = hermes_write_wordrec(hw, USER_BAP,
  758. HERMES_RID_CNFWEPENABLED_AGERE,
  759. enc_flag);
  760. if (err)
  761. return err;
  762. if (priv->has_wpa) {
  763. /* Set WPA key management */
  764. err = hermes_write_wordrec(hw, USER_BAP,
  765. HERMES_RID_CNFSETWPAAUTHMGMTSUITE_AGERE,
  766. priv->key_mgmt);
  767. if (err)
  768. return err;
  769. }
  770. break;
  771. case FIRMWARE_TYPE_INTERSIL: /* Intersil style WEP */
  772. case FIRMWARE_TYPE_SYMBOL: /* Symbol style WEP */
  773. if (priv->encode_alg == IW_ENCODE_ALG_WEP) {
  774. if (priv->wep_restrict ||
  775. (priv->firmware_type == FIRMWARE_TYPE_SYMBOL))
  776. master_wep_flag = HERMES_WEP_PRIVACY_INVOKED |
  777. HERMES_WEP_EXCL_UNENCRYPTED;
  778. else
  779. master_wep_flag = HERMES_WEP_PRIVACY_INVOKED;
  780. err = hermes_write_wordrec(hw, USER_BAP,
  781. HERMES_RID_CNFAUTHENTICATION,
  782. auth_flag);
  783. if (err)
  784. return err;
  785. } else
  786. master_wep_flag = 0;
  787. if (priv->iw_mode == NL80211_IFTYPE_MONITOR)
  788. master_wep_flag |= HERMES_WEP_HOST_DECRYPT;
  789. /* Master WEP setting : on/off */
  790. err = hermes_write_wordrec(hw, USER_BAP,
  791. HERMES_RID_CNFWEPFLAGS_INTERSIL,
  792. master_wep_flag);
  793. if (err)
  794. return err;
  795. break;
  796. }
  797. return 0;
  798. }
  799. /* key must be 32 bytes, including the tx and rx MIC keys.
  800. * rsc must be 8 bytes
  801. * tsc must be 8 bytes or NULL
  802. */
  803. int __orinoco_hw_set_tkip_key(struct orinoco_private *priv, int key_idx,
  804. int set_tx, u8 *key, u8 *rsc, u8 *tsc)
  805. {
  806. struct {
  807. __le16 idx;
  808. u8 rsc[IW_ENCODE_SEQ_MAX_SIZE];
  809. u8 key[TKIP_KEYLEN];
  810. u8 tx_mic[MIC_KEYLEN];
  811. u8 rx_mic[MIC_KEYLEN];
  812. u8 tsc[IW_ENCODE_SEQ_MAX_SIZE];
  813. } __attribute__ ((packed)) buf;
  814. hermes_t *hw = &priv->hw;
  815. int ret;
  816. int err;
  817. int k;
  818. u16 xmitting;
  819. key_idx &= 0x3;
  820. if (set_tx)
  821. key_idx |= 0x8000;
  822. buf.idx = cpu_to_le16(key_idx);
  823. memcpy(buf.key, key,
  824. sizeof(buf.key) + sizeof(buf.tx_mic) + sizeof(buf.rx_mic));
  825. if (rsc == NULL)
  826. memset(buf.rsc, 0, sizeof(buf.rsc));
  827. else
  828. memcpy(buf.rsc, rsc, sizeof(buf.rsc));
  829. if (tsc == NULL) {
  830. memset(buf.tsc, 0, sizeof(buf.tsc));
  831. buf.tsc[4] = 0x10;
  832. } else {
  833. memcpy(buf.tsc, tsc, sizeof(buf.tsc));
  834. }
  835. /* Wait upto 100ms for tx queue to empty */
  836. for (k = 100; k > 0; k--) {
  837. udelay(1000);
  838. ret = hermes_read_wordrec(hw, USER_BAP, HERMES_RID_TXQUEUEEMPTY,
  839. &xmitting);
  840. if (ret || !xmitting)
  841. break;
  842. }
  843. if (k == 0)
  844. ret = -ETIMEDOUT;
  845. err = HERMES_WRITE_RECORD(hw, USER_BAP,
  846. HERMES_RID_CNFADDDEFAULTTKIPKEY_AGERE,
  847. &buf);
  848. return ret ? ret : err;
  849. }
  850. int orinoco_clear_tkip_key(struct orinoco_private *priv, int key_idx)
  851. {
  852. hermes_t *hw = &priv->hw;
  853. int err;
  854. memset(&priv->tkip_key[key_idx], 0, sizeof(priv->tkip_key[key_idx]));
  855. err = hermes_write_wordrec(hw, USER_BAP,
  856. HERMES_RID_CNFREMDEFAULTTKIPKEY_AGERE,
  857. key_idx);
  858. if (err)
  859. printk(KERN_WARNING "%s: Error %d clearing TKIP key %d\n",
  860. priv->ndev->name, err, key_idx);
  861. return err;
  862. }
  863. int __orinoco_hw_set_multicast_list(struct orinoco_private *priv,
  864. struct dev_addr_list *mc_list,
  865. int mc_count, int promisc)
  866. {
  867. hermes_t *hw = &priv->hw;
  868. int err = 0;
  869. if (promisc != priv->promiscuous) {
  870. err = hermes_write_wordrec(hw, USER_BAP,
  871. HERMES_RID_CNFPROMISCUOUSMODE,
  872. promisc);
  873. if (err) {
  874. printk(KERN_ERR "%s: Error %d setting PROMISCUOUSMODE to 1.\n",
  875. priv->ndev->name, err);
  876. } else
  877. priv->promiscuous = promisc;
  878. }
  879. /* If we're not in promiscuous mode, then we need to set the
  880. * group address if either we want to multicast, or if we were
  881. * multicasting and want to stop */
  882. if (!promisc && (mc_count || priv->mc_count)) {
  883. struct dev_mc_list *p = mc_list;
  884. struct hermes_multicast mclist;
  885. int i;
  886. for (i = 0; i < mc_count; i++) {
  887. /* paranoia: is list shorter than mc_count? */
  888. BUG_ON(!p);
  889. /* paranoia: bad address size in list? */
  890. BUG_ON(p->dmi_addrlen != ETH_ALEN);
  891. memcpy(mclist.addr[i], p->dmi_addr, ETH_ALEN);
  892. p = p->next;
  893. }
  894. if (p)
  895. printk(KERN_WARNING "%s: Multicast list is "
  896. "longer than mc_count\n", priv->ndev->name);
  897. err = hermes_write_ltv(hw, USER_BAP,
  898. HERMES_RID_CNFGROUPADDRESSES,
  899. HERMES_BYTES_TO_RECLEN(mc_count * ETH_ALEN),
  900. &mclist);
  901. if (err)
  902. printk(KERN_ERR "%s: Error %d setting multicast list.\n",
  903. priv->ndev->name, err);
  904. else
  905. priv->mc_count = mc_count;
  906. }
  907. return err;
  908. }
  909. /* Return : < 0 -> error code ; >= 0 -> length */
  910. int orinoco_hw_get_essid(struct orinoco_private *priv, int *active,
  911. char buf[IW_ESSID_MAX_SIZE+1])
  912. {
  913. hermes_t *hw = &priv->hw;
  914. int err = 0;
  915. struct hermes_idstring essidbuf;
  916. char *p = (char *)(&essidbuf.val);
  917. int len;
  918. unsigned long flags;
  919. if (orinoco_lock(priv, &flags) != 0)
  920. return -EBUSY;
  921. if (strlen(priv->desired_essid) > 0) {
  922. /* We read the desired SSID from the hardware rather
  923. than from priv->desired_essid, just in case the
  924. firmware is allowed to change it on us. I'm not
  925. sure about this */
  926. /* My guess is that the OWNSSID should always be whatever
  927. * we set to the card, whereas CURRENT_SSID is the one that
  928. * may change... - Jean II */
  929. u16 rid;
  930. *active = 1;
  931. rid = (priv->port_type == 3) ? HERMES_RID_CNFOWNSSID :
  932. HERMES_RID_CNFDESIREDSSID;
  933. err = hermes_read_ltv(hw, USER_BAP, rid, sizeof(essidbuf),
  934. NULL, &essidbuf);
  935. if (err)
  936. goto fail_unlock;
  937. } else {
  938. *active = 0;
  939. err = hermes_read_ltv(hw, USER_BAP, HERMES_RID_CURRENTSSID,
  940. sizeof(essidbuf), NULL, &essidbuf);
  941. if (err)
  942. goto fail_unlock;
  943. }
  944. len = le16_to_cpu(essidbuf.len);
  945. BUG_ON(len > IW_ESSID_MAX_SIZE);
  946. memset(buf, 0, IW_ESSID_MAX_SIZE);
  947. memcpy(buf, p, len);
  948. err = len;
  949. fail_unlock:
  950. orinoco_unlock(priv, &flags);
  951. return err;
  952. }
  953. int orinoco_hw_get_freq(struct orinoco_private *priv)
  954. {
  955. hermes_t *hw = &priv->hw;
  956. int err = 0;
  957. u16 channel;
  958. int freq = 0;
  959. unsigned long flags;
  960. if (orinoco_lock(priv, &flags) != 0)
  961. return -EBUSY;
  962. err = hermes_read_wordrec(hw, USER_BAP, HERMES_RID_CURRENTCHANNEL,
  963. &channel);
  964. if (err)
  965. goto out;
  966. /* Intersil firmware 1.3.5 returns 0 when the interface is down */
  967. if (channel == 0) {
  968. err = -EBUSY;
  969. goto out;
  970. }
  971. if ((channel < 1) || (channel > NUM_CHANNELS)) {
  972. printk(KERN_WARNING "%s: Channel out of range (%d)!\n",
  973. priv->ndev->name, channel);
  974. err = -EBUSY;
  975. goto out;
  976. }
  977. freq = ieee80211_dsss_chan_to_freq(channel);
  978. out:
  979. orinoco_unlock(priv, &flags);
  980. if (err > 0)
  981. err = -EBUSY;
  982. return err ? err : freq;
  983. }
  984. int orinoco_hw_get_bitratelist(struct orinoco_private *priv,
  985. int *numrates, s32 *rates, int max)
  986. {
  987. hermes_t *hw = &priv->hw;
  988. struct hermes_idstring list;
  989. unsigned char *p = (unsigned char *)&list.val;
  990. int err = 0;
  991. int num;
  992. int i;
  993. unsigned long flags;
  994. if (orinoco_lock(priv, &flags) != 0)
  995. return -EBUSY;
  996. err = hermes_read_ltv(hw, USER_BAP, HERMES_RID_SUPPORTEDDATARATES,
  997. sizeof(list), NULL, &list);
  998. orinoco_unlock(priv, &flags);
  999. if (err)
  1000. return err;
  1001. num = le16_to_cpu(list.len);
  1002. *numrates = num;
  1003. num = min(num, max);
  1004. for (i = 0; i < num; i++)
  1005. rates[i] = (p[i] & 0x7f) * 500000; /* convert to bps */
  1006. return 0;
  1007. }
  1008. int orinoco_hw_trigger_scan(struct orinoco_private *priv,
  1009. const struct cfg80211_ssid *ssid)
  1010. {
  1011. struct net_device *dev = priv->ndev;
  1012. hermes_t *hw = &priv->hw;
  1013. unsigned long flags;
  1014. int err = 0;
  1015. if (orinoco_lock(priv, &flags) != 0)
  1016. return -EBUSY;
  1017. /* Scanning with port 0 disabled would fail */
  1018. if (!netif_running(dev)) {
  1019. err = -ENETDOWN;
  1020. goto out;
  1021. }
  1022. /* In monitor mode, the scan results are always empty.
  1023. * Probe responses are passed to the driver as received
  1024. * frames and could be processed in software. */
  1025. if (priv->iw_mode == NL80211_IFTYPE_MONITOR) {
  1026. err = -EOPNOTSUPP;
  1027. goto out;
  1028. }
  1029. if (priv->has_hostscan) {
  1030. switch (priv->firmware_type) {
  1031. case FIRMWARE_TYPE_SYMBOL:
  1032. err = hermes_write_wordrec(hw, USER_BAP,
  1033. HERMES_RID_CNFHOSTSCAN_SYMBOL,
  1034. HERMES_HOSTSCAN_SYMBOL_ONCE |
  1035. HERMES_HOSTSCAN_SYMBOL_BCAST);
  1036. break;
  1037. case FIRMWARE_TYPE_INTERSIL: {
  1038. __le16 req[3];
  1039. req[0] = cpu_to_le16(0x3fff); /* All channels */
  1040. req[1] = cpu_to_le16(0x0001); /* rate 1 Mbps */
  1041. req[2] = 0; /* Any ESSID */
  1042. err = HERMES_WRITE_RECORD(hw, USER_BAP,
  1043. HERMES_RID_CNFHOSTSCAN, &req);
  1044. break;
  1045. }
  1046. case FIRMWARE_TYPE_AGERE:
  1047. if (ssid->ssid_len > 0) {
  1048. struct hermes_idstring idbuf;
  1049. size_t len = ssid->ssid_len;
  1050. idbuf.len = cpu_to_le16(len);
  1051. memcpy(idbuf.val, ssid->ssid, len);
  1052. err = hermes_write_ltv(hw, USER_BAP,
  1053. HERMES_RID_CNFSCANSSID_AGERE,
  1054. HERMES_BYTES_TO_RECLEN(len + 2),
  1055. &idbuf);
  1056. } else
  1057. err = hermes_write_wordrec(hw, USER_BAP,
  1058. HERMES_RID_CNFSCANSSID_AGERE,
  1059. 0); /* Any ESSID */
  1060. if (err)
  1061. break;
  1062. if (priv->has_ext_scan) {
  1063. err = hermes_write_wordrec(hw, USER_BAP,
  1064. HERMES_RID_CNFSCANCHANNELS2GHZ,
  1065. 0x7FFF);
  1066. if (err)
  1067. goto out;
  1068. err = hermes_inquire(hw,
  1069. HERMES_INQ_CHANNELINFO);
  1070. } else
  1071. err = hermes_inquire(hw, HERMES_INQ_SCAN);
  1072. break;
  1073. }
  1074. } else
  1075. err = hermes_inquire(hw, HERMES_INQ_SCAN);
  1076. out:
  1077. orinoco_unlock(priv, &flags);
  1078. return err;
  1079. }