hw.c 34 KB

12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788899091929394959697989910010110210310410510610710810911011111211311411511611711811912012112212312412512612712812913013113213313413513613713813914014114214314414514614714814915015115215315415515615715815916016116216316416516616716816917017117217317417517617717817918018118218318418518618718818919019119219319419519619719819920020120220320420520620720820921021121221321421521621721821922022122222322422522622722822923023123223323423523623723823924024124224324424524624724824925025125225325425525625725825926026126226326426526626726826927027127227327427527627727827928028128228328428528628728828929029129229329429529629729829930030130230330430530630730830931031131231331431531631731831932032132232332432532632732832933033133233333433533633733833934034134234334434534634734834935035135235335435535635735835936036136236336436536636736836937037137237337437537637737837938038138238338438538638738838939039139239339439539639739839940040140240340440540640740840941041141241341441541641741841942042142242342442542642742842943043143243343443543643743843944044144244344444544644744844945045145245345445545645745845946046146246346446546646746846947047147247347447547647747847948048148248348448548648748848949049149249349449549649749849950050150250350450550650750850951051151251351451551651751851952052152252352452552652752852953053153253353453553653753853954054154254354454554654754854955055155255355455555655755855956056156256356456556656756856957057157257357457557657757857958058158258358458558658758858959059159259359459559659759859960060160260360460560660760860961061161261361461561661761861962062162262362462562662762862963063163263363463563663763863964064164264364464564664764864965065165265365465565665765865966066166266366466566666766866967067167267367467567667767867968068168268368468568668768868969069169269369469569669769869970070170270370470570670770870971071171271371471571671771871972072172272372472572672772872973073173273373473573673773873974074174274374474574674774874975075175275375475575675775875976076176276376476576676776876977077177277377477577677777877978078178278378478578678778878979079179279379479579679779879980080180280380480580680780880981081181281381481581681781881982082182282382482582682782882983083183283383483583683783883984084184284384484584684784884985085185285385485585685785885986086186286386486586686786886987087187287387487587687787887988088188288388488588688788888989089189289389489589689789889990090190290390490590690790890991091191291391491591691791891992092192292392492592692792892993093193293393493593693793893994094194294394494594694794894995095195295395495595695795895996096196296396496596696796896997097197297397497597697797897998098198298398498598698798898999099199299399499599699799899910001001100210031004100510061007100810091010101110121013101410151016101710181019102010211022102310241025102610271028102910301031103210331034103510361037103810391040104110421043104410451046104710481049105010511052105310541055105610571058105910601061106210631064106510661067106810691070107110721073107410751076107710781079108010811082108310841085108610871088108910901091109210931094109510961097109810991100110111021103110411051106110711081109111011111112111311141115111611171118111911201121112211231124112511261127112811291130113111321133113411351136113711381139114011411142114311441145114611471148114911501151115211531154115511561157115811591160116111621163116411651166116711681169117011711172117311741175117611771178117911801181118211831184118511861187118811891190119111921193119411951196119711981199120012011202120312041205120612071208120912101211121212131214121512161217121812191220122112221223122412251226122712281229123012311232123312341235123612371238123912401241124212431244124512461247124812491250125112521253125412551256125712581259126012611262126312641265126612671268126912701271127212731274127512761277127812791280128112821283128412851286128712881289129012911292129312941295129612971298129913001301130213031304130513061307130813091310131113121313131413151316131713181319132013211322132313241325132613271328132913301331133213331334133513361337133813391340134113421343134413451346134713481349135013511352
  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. #ifndef CONFIG_HERMES_PRISM
  238. if (priv->firmware_type == FIRMWARE_TYPE_INTERSIL) {
  239. dev_err(dev, "Support for Prism chipset is not enabled\n");
  240. return -ENODEV;
  241. }
  242. #endif
  243. return 0;
  244. }
  245. /* Read settings from EEPROM into our private structure.
  246. * MAC address gets dropped into callers buffer
  247. * Can be called before netdev registration.
  248. */
  249. int orinoco_hw_read_card_settings(struct orinoco_private *priv, u8 *dev_addr)
  250. {
  251. struct device *dev = priv->dev;
  252. struct hermes_idstring nickbuf;
  253. hermes_t *hw = &priv->hw;
  254. int len;
  255. int err;
  256. u16 reclen;
  257. /* Get the MAC address */
  258. err = hermes_read_ltv(hw, USER_BAP, HERMES_RID_CNFOWNMACADDR,
  259. ETH_ALEN, NULL, dev_addr);
  260. if (err) {
  261. dev_warn(dev, "Failed to read MAC address!\n");
  262. goto out;
  263. }
  264. dev_dbg(dev, "MAC address %pM\n", dev_addr);
  265. /* Get the station name */
  266. err = hermes_read_ltv(hw, USER_BAP, HERMES_RID_CNFOWNNAME,
  267. sizeof(nickbuf), &reclen, &nickbuf);
  268. if (err) {
  269. dev_err(dev, "failed to read station name\n");
  270. goto out;
  271. }
  272. if (nickbuf.len)
  273. len = min(IW_ESSID_MAX_SIZE, (int)le16_to_cpu(nickbuf.len));
  274. else
  275. len = min(IW_ESSID_MAX_SIZE, 2 * reclen);
  276. memcpy(priv->nick, &nickbuf.val, len);
  277. priv->nick[len] = '\0';
  278. dev_dbg(dev, "Station name \"%s\"\n", priv->nick);
  279. /* Get allowed channels */
  280. err = hermes_read_wordrec(hw, USER_BAP, HERMES_RID_CHANNELLIST,
  281. &priv->channel_mask);
  282. if (err) {
  283. dev_err(dev, "Failed to read channel list!\n");
  284. goto out;
  285. }
  286. /* Get initial AP density */
  287. err = hermes_read_wordrec(hw, USER_BAP, HERMES_RID_CNFSYSTEMSCALE,
  288. &priv->ap_density);
  289. if (err || priv->ap_density < 1 || priv->ap_density > 3)
  290. priv->has_sensitivity = 0;
  291. /* Get initial RTS threshold */
  292. err = hermes_read_wordrec(hw, USER_BAP, HERMES_RID_CNFRTSTHRESHOLD,
  293. &priv->rts_thresh);
  294. if (err) {
  295. dev_err(dev, "Failed to read RTS threshold!\n");
  296. goto out;
  297. }
  298. /* Get initial fragmentation settings */
  299. if (priv->has_mwo)
  300. err = hermes_read_wordrec(hw, USER_BAP,
  301. HERMES_RID_CNFMWOROBUST_AGERE,
  302. &priv->mwo_robust);
  303. else
  304. err = hermes_read_wordrec(hw, USER_BAP,
  305. HERMES_RID_CNFFRAGMENTATIONTHRESHOLD,
  306. &priv->frag_thresh);
  307. if (err) {
  308. dev_err(dev, "Failed to read fragmentation settings!\n");
  309. goto out;
  310. }
  311. /* Power management setup */
  312. if (priv->has_pm) {
  313. priv->pm_on = 0;
  314. priv->pm_mcast = 1;
  315. err = hermes_read_wordrec(hw, USER_BAP,
  316. HERMES_RID_CNFMAXSLEEPDURATION,
  317. &priv->pm_period);
  318. if (err) {
  319. dev_err(dev, "Failed to read power management "
  320. "period!\n");
  321. goto out;
  322. }
  323. err = hermes_read_wordrec(hw, USER_BAP,
  324. HERMES_RID_CNFPMHOLDOVERDURATION,
  325. &priv->pm_timeout);
  326. if (err) {
  327. dev_err(dev, "Failed to read power management "
  328. "timeout!\n");
  329. goto out;
  330. }
  331. }
  332. /* Preamble setup */
  333. if (priv->has_preamble) {
  334. err = hermes_read_wordrec(hw, USER_BAP,
  335. HERMES_RID_CNFPREAMBLE_SYMBOL,
  336. &priv->preamble);
  337. if (err) {
  338. dev_err(dev, "Failed to read preamble setup\n");
  339. goto out;
  340. }
  341. }
  342. /* Retry settings */
  343. err = hermes_read_wordrec(hw, USER_BAP, HERMES_RID_SHORTRETRYLIMIT,
  344. &priv->short_retry_limit);
  345. if (err) {
  346. dev_err(dev, "Failed to read short retry limit\n");
  347. goto out;
  348. }
  349. err = hermes_read_wordrec(hw, USER_BAP, HERMES_RID_LONGRETRYLIMIT,
  350. &priv->long_retry_limit);
  351. if (err) {
  352. dev_err(dev, "Failed to read long retry limit\n");
  353. goto out;
  354. }
  355. err = hermes_read_wordrec(hw, USER_BAP, HERMES_RID_MAXTRANSMITLIFETIME,
  356. &priv->retry_lifetime);
  357. if (err) {
  358. dev_err(dev, "Failed to read max retry lifetime\n");
  359. goto out;
  360. }
  361. out:
  362. return err;
  363. }
  364. /* Can be called before netdev registration */
  365. int orinoco_hw_allocate_fid(struct orinoco_private *priv)
  366. {
  367. struct device *dev = priv->dev;
  368. struct hermes *hw = &priv->hw;
  369. int err;
  370. err = hermes_allocate(hw, priv->nicbuf_size, &priv->txfid);
  371. if (err == -EIO && priv->nicbuf_size > TX_NICBUF_SIZE_BUG) {
  372. /* Try workaround for old Symbol firmware bug */
  373. priv->nicbuf_size = TX_NICBUF_SIZE_BUG;
  374. err = hermes_allocate(hw, priv->nicbuf_size, &priv->txfid);
  375. dev_warn(dev, "Firmware ALLOC bug detected "
  376. "(old Symbol firmware?). Work around %s\n",
  377. err ? "failed!" : "ok.");
  378. }
  379. return err;
  380. }
  381. int orinoco_get_bitratemode(int bitrate, int automatic)
  382. {
  383. int ratemode = -1;
  384. int i;
  385. if ((bitrate != 10) && (bitrate != 20) &&
  386. (bitrate != 55) && (bitrate != 110))
  387. return ratemode;
  388. for (i = 0; i < BITRATE_TABLE_SIZE; i++) {
  389. if ((bitrate_table[i].bitrate == bitrate) &&
  390. (bitrate_table[i].automatic == automatic)) {
  391. ratemode = i;
  392. break;
  393. }
  394. }
  395. return ratemode;
  396. }
  397. void orinoco_get_ratemode_cfg(int ratemode, int *bitrate, int *automatic)
  398. {
  399. BUG_ON((ratemode < 0) || (ratemode >= BITRATE_TABLE_SIZE));
  400. *bitrate = bitrate_table[ratemode].bitrate * 100000;
  401. *automatic = bitrate_table[ratemode].automatic;
  402. }
  403. int orinoco_hw_program_rids(struct orinoco_private *priv)
  404. {
  405. struct net_device *dev = priv->ndev;
  406. struct wireless_dev *wdev = netdev_priv(dev);
  407. hermes_t *hw = &priv->hw;
  408. int err;
  409. struct hermes_idstring idbuf;
  410. /* Set the MAC address */
  411. err = hermes_write_ltv(hw, USER_BAP, HERMES_RID_CNFOWNMACADDR,
  412. HERMES_BYTES_TO_RECLEN(ETH_ALEN), dev->dev_addr);
  413. if (err) {
  414. printk(KERN_ERR "%s: Error %d setting MAC address\n",
  415. dev->name, err);
  416. return err;
  417. }
  418. /* Set up the link mode */
  419. err = hermes_write_wordrec(hw, USER_BAP, HERMES_RID_CNFPORTTYPE,
  420. priv->port_type);
  421. if (err) {
  422. printk(KERN_ERR "%s: Error %d setting port type\n",
  423. dev->name, err);
  424. return err;
  425. }
  426. /* Set the channel/frequency */
  427. if (priv->channel != 0 && priv->iw_mode != NL80211_IFTYPE_STATION) {
  428. err = hermes_write_wordrec(hw, USER_BAP,
  429. HERMES_RID_CNFOWNCHANNEL,
  430. priv->channel);
  431. if (err) {
  432. printk(KERN_ERR "%s: Error %d setting channel %d\n",
  433. dev->name, err, priv->channel);
  434. return err;
  435. }
  436. }
  437. if (priv->has_ibss) {
  438. u16 createibss;
  439. if ((strlen(priv->desired_essid) == 0) && (priv->createibss)) {
  440. printk(KERN_WARNING "%s: This firmware requires an "
  441. "ESSID in IBSS-Ad-Hoc mode.\n", dev->name);
  442. /* With wvlan_cs, in this case, we would crash.
  443. * hopefully, this driver will behave better...
  444. * Jean II */
  445. createibss = 0;
  446. } else {
  447. createibss = priv->createibss;
  448. }
  449. err = hermes_write_wordrec(hw, USER_BAP,
  450. HERMES_RID_CNFCREATEIBSS,
  451. createibss);
  452. if (err) {
  453. printk(KERN_ERR "%s: Error %d setting CREATEIBSS\n",
  454. dev->name, err);
  455. return err;
  456. }
  457. }
  458. /* Set the desired BSSID */
  459. err = __orinoco_hw_set_wap(priv);
  460. if (err) {
  461. printk(KERN_ERR "%s: Error %d setting AP address\n",
  462. dev->name, err);
  463. return err;
  464. }
  465. /* Set the desired ESSID */
  466. idbuf.len = cpu_to_le16(strlen(priv->desired_essid));
  467. memcpy(&idbuf.val, priv->desired_essid, sizeof(idbuf.val));
  468. /* WinXP wants partner to configure OWNSSID even in IBSS mode. (jimc) */
  469. err = hermes_write_ltv(hw, USER_BAP, HERMES_RID_CNFOWNSSID,
  470. HERMES_BYTES_TO_RECLEN(strlen(priv->desired_essid)+2),
  471. &idbuf);
  472. if (err) {
  473. printk(KERN_ERR "%s: Error %d setting OWNSSID\n",
  474. dev->name, err);
  475. return err;
  476. }
  477. err = hermes_write_ltv(hw, USER_BAP, HERMES_RID_CNFDESIREDSSID,
  478. HERMES_BYTES_TO_RECLEN(strlen(priv->desired_essid)+2),
  479. &idbuf);
  480. if (err) {
  481. printk(KERN_ERR "%s: Error %d setting DESIREDSSID\n",
  482. dev->name, err);
  483. return err;
  484. }
  485. /* Set the station name */
  486. idbuf.len = cpu_to_le16(strlen(priv->nick));
  487. memcpy(&idbuf.val, priv->nick, sizeof(idbuf.val));
  488. err = hermes_write_ltv(hw, USER_BAP, HERMES_RID_CNFOWNNAME,
  489. HERMES_BYTES_TO_RECLEN(strlen(priv->nick)+2),
  490. &idbuf);
  491. if (err) {
  492. printk(KERN_ERR "%s: Error %d setting nickname\n",
  493. dev->name, err);
  494. return err;
  495. }
  496. /* Set AP density */
  497. if (priv->has_sensitivity) {
  498. err = hermes_write_wordrec(hw, USER_BAP,
  499. HERMES_RID_CNFSYSTEMSCALE,
  500. priv->ap_density);
  501. if (err) {
  502. printk(KERN_WARNING "%s: Error %d setting SYSTEMSCALE. "
  503. "Disabling sensitivity control\n",
  504. dev->name, err);
  505. priv->has_sensitivity = 0;
  506. }
  507. }
  508. /* Set RTS threshold */
  509. err = hermes_write_wordrec(hw, USER_BAP, HERMES_RID_CNFRTSTHRESHOLD,
  510. priv->rts_thresh);
  511. if (err) {
  512. printk(KERN_ERR "%s: Error %d setting RTS threshold\n",
  513. dev->name, err);
  514. return err;
  515. }
  516. /* Set fragmentation threshold or MWO robustness */
  517. if (priv->has_mwo)
  518. err = hermes_write_wordrec(hw, USER_BAP,
  519. HERMES_RID_CNFMWOROBUST_AGERE,
  520. priv->mwo_robust);
  521. else
  522. err = hermes_write_wordrec(hw, USER_BAP,
  523. HERMES_RID_CNFFRAGMENTATIONTHRESHOLD,
  524. priv->frag_thresh);
  525. if (err) {
  526. printk(KERN_ERR "%s: Error %d setting fragmentation\n",
  527. dev->name, err);
  528. return err;
  529. }
  530. /* Set bitrate */
  531. err = __orinoco_hw_set_bitrate(priv);
  532. if (err) {
  533. printk(KERN_ERR "%s: Error %d setting bitrate\n",
  534. dev->name, err);
  535. return err;
  536. }
  537. /* Set power management */
  538. if (priv->has_pm) {
  539. err = hermes_write_wordrec(hw, USER_BAP,
  540. HERMES_RID_CNFPMENABLED,
  541. priv->pm_on);
  542. if (err) {
  543. printk(KERN_ERR "%s: Error %d setting up PM\n",
  544. dev->name, err);
  545. return err;
  546. }
  547. err = hermes_write_wordrec(hw, USER_BAP,
  548. HERMES_RID_CNFMULTICASTRECEIVE,
  549. priv->pm_mcast);
  550. if (err) {
  551. printk(KERN_ERR "%s: Error %d setting up PM\n",
  552. dev->name, err);
  553. return err;
  554. }
  555. err = hermes_write_wordrec(hw, USER_BAP,
  556. HERMES_RID_CNFMAXSLEEPDURATION,
  557. priv->pm_period);
  558. if (err) {
  559. printk(KERN_ERR "%s: Error %d setting up PM\n",
  560. dev->name, err);
  561. return err;
  562. }
  563. err = hermes_write_wordrec(hw, USER_BAP,
  564. HERMES_RID_CNFPMHOLDOVERDURATION,
  565. priv->pm_timeout);
  566. if (err) {
  567. printk(KERN_ERR "%s: Error %d setting up PM\n",
  568. dev->name, err);
  569. return err;
  570. }
  571. }
  572. /* Set preamble - only for Symbol so far... */
  573. if (priv->has_preamble) {
  574. err = hermes_write_wordrec(hw, USER_BAP,
  575. HERMES_RID_CNFPREAMBLE_SYMBOL,
  576. priv->preamble);
  577. if (err) {
  578. printk(KERN_ERR "%s: Error %d setting preamble\n",
  579. dev->name, err);
  580. return err;
  581. }
  582. }
  583. /* Set up encryption */
  584. if (priv->has_wep || priv->has_wpa) {
  585. err = __orinoco_hw_setup_enc(priv);
  586. if (err) {
  587. printk(KERN_ERR "%s: Error %d activating encryption\n",
  588. dev->name, err);
  589. return err;
  590. }
  591. }
  592. if (priv->iw_mode == NL80211_IFTYPE_MONITOR) {
  593. /* Enable monitor mode */
  594. dev->type = ARPHRD_IEEE80211;
  595. err = hermes_docmd_wait(hw, HERMES_CMD_TEST |
  596. HERMES_TEST_MONITOR, 0, NULL);
  597. } else {
  598. /* Disable monitor mode */
  599. dev->type = ARPHRD_ETHER;
  600. err = hermes_docmd_wait(hw, HERMES_CMD_TEST |
  601. HERMES_TEST_STOP, 0, NULL);
  602. }
  603. if (err)
  604. return err;
  605. /* Reset promiscuity / multicast*/
  606. priv->promiscuous = 0;
  607. priv->mc_count = 0;
  608. /* Record mode change */
  609. wdev->iftype = priv->iw_mode;
  610. return 0;
  611. }
  612. /* Get tsc from the firmware */
  613. int orinoco_hw_get_tkip_iv(struct orinoco_private *priv, int key, u8 *tsc)
  614. {
  615. hermes_t *hw = &priv->hw;
  616. int err = 0;
  617. u8 tsc_arr[4][ORINOCO_SEQ_LEN];
  618. if ((key < 0) || (key >= 4))
  619. return -EINVAL;
  620. err = hermes_read_ltv(hw, USER_BAP, HERMES_RID_CURRENT_TKIP_IV,
  621. sizeof(tsc_arr), NULL, &tsc_arr);
  622. if (!err)
  623. memcpy(tsc, &tsc_arr[key][0], sizeof(tsc_arr[0]));
  624. return err;
  625. }
  626. int __orinoco_hw_set_bitrate(struct orinoco_private *priv)
  627. {
  628. hermes_t *hw = &priv->hw;
  629. int ratemode = priv->bitratemode;
  630. int err = 0;
  631. if (ratemode >= BITRATE_TABLE_SIZE) {
  632. printk(KERN_ERR "%s: BUG: Invalid bitrate mode %d\n",
  633. priv->ndev->name, ratemode);
  634. return -EINVAL;
  635. }
  636. switch (priv->firmware_type) {
  637. case FIRMWARE_TYPE_AGERE:
  638. err = hermes_write_wordrec(hw, USER_BAP,
  639. HERMES_RID_CNFTXRATECONTROL,
  640. bitrate_table[ratemode].agere_txratectrl);
  641. break;
  642. case FIRMWARE_TYPE_INTERSIL:
  643. case FIRMWARE_TYPE_SYMBOL:
  644. err = hermes_write_wordrec(hw, USER_BAP,
  645. HERMES_RID_CNFTXRATECONTROL,
  646. bitrate_table[ratemode].intersil_txratectrl);
  647. break;
  648. default:
  649. BUG();
  650. }
  651. return err;
  652. }
  653. int orinoco_hw_get_act_bitrate(struct orinoco_private *priv, int *bitrate)
  654. {
  655. hermes_t *hw = &priv->hw;
  656. int i;
  657. int err = 0;
  658. u16 val;
  659. err = hermes_read_wordrec(hw, USER_BAP,
  660. HERMES_RID_CURRENTTXRATE, &val);
  661. if (err)
  662. return err;
  663. switch (priv->firmware_type) {
  664. case FIRMWARE_TYPE_AGERE: /* Lucent style rate */
  665. /* Note : in Lucent firmware, the return value of
  666. * HERMES_RID_CURRENTTXRATE is the bitrate in Mb/s,
  667. * and therefore is totally different from the
  668. * encoding of HERMES_RID_CNFTXRATECONTROL.
  669. * Don't forget that 6Mb/s is really 5.5Mb/s */
  670. if (val == 6)
  671. *bitrate = 5500000;
  672. else
  673. *bitrate = val * 1000000;
  674. break;
  675. case FIRMWARE_TYPE_INTERSIL: /* Intersil style rate */
  676. case FIRMWARE_TYPE_SYMBOL: /* Symbol style rate */
  677. for (i = 0; i < BITRATE_TABLE_SIZE; i++)
  678. if (bitrate_table[i].intersil_txratectrl == val)
  679. break;
  680. if (i >= BITRATE_TABLE_SIZE)
  681. printk(KERN_INFO "%s: Unable to determine current bitrate (0x%04hx)\n",
  682. priv->ndev->name, val);
  683. *bitrate = bitrate_table[i].bitrate * 100000;
  684. break;
  685. default:
  686. BUG();
  687. }
  688. return err;
  689. }
  690. /* Set fixed AP address */
  691. int __orinoco_hw_set_wap(struct orinoco_private *priv)
  692. {
  693. int roaming_flag;
  694. int err = 0;
  695. hermes_t *hw = &priv->hw;
  696. switch (priv->firmware_type) {
  697. case FIRMWARE_TYPE_AGERE:
  698. /* not supported */
  699. break;
  700. case FIRMWARE_TYPE_INTERSIL:
  701. if (priv->bssid_fixed)
  702. roaming_flag = 2;
  703. else
  704. roaming_flag = 1;
  705. err = hermes_write_wordrec(hw, USER_BAP,
  706. HERMES_RID_CNFROAMINGMODE,
  707. roaming_flag);
  708. break;
  709. case FIRMWARE_TYPE_SYMBOL:
  710. err = HERMES_WRITE_RECORD(hw, USER_BAP,
  711. HERMES_RID_CNFMANDATORYBSSID_SYMBOL,
  712. &priv->desired_bssid);
  713. break;
  714. }
  715. return err;
  716. }
  717. /* Change the WEP keys and/or the current keys. Can be called
  718. * either from __orinoco_hw_setup_enc() or directly from
  719. * orinoco_ioctl_setiwencode(). In the later case the association
  720. * with the AP is not broken (if the firmware can handle it),
  721. * which is needed for 802.1x implementations. */
  722. int __orinoco_hw_setup_wepkeys(struct orinoco_private *priv)
  723. {
  724. hermes_t *hw = &priv->hw;
  725. int err = 0;
  726. int i;
  727. switch (priv->firmware_type) {
  728. case FIRMWARE_TYPE_AGERE:
  729. {
  730. struct orinoco_key keys[ORINOCO_MAX_KEYS];
  731. memset(&keys, 0, sizeof(keys));
  732. for (i = 0; i < ORINOCO_MAX_KEYS; i++) {
  733. int len = min(priv->keys[i].key_len,
  734. ORINOCO_MAX_KEY_SIZE);
  735. memcpy(&keys[i].data, priv->keys[i].key, len);
  736. if (len > SMALL_KEY_SIZE)
  737. keys[i].len = cpu_to_le16(LARGE_KEY_SIZE);
  738. else if (len > 0)
  739. keys[i].len = cpu_to_le16(SMALL_KEY_SIZE);
  740. else
  741. keys[i].len = cpu_to_le16(0);
  742. }
  743. err = HERMES_WRITE_RECORD(hw, USER_BAP,
  744. HERMES_RID_CNFWEPKEYS_AGERE,
  745. &keys);
  746. if (err)
  747. return err;
  748. err = hermes_write_wordrec(hw, USER_BAP,
  749. HERMES_RID_CNFTXKEY_AGERE,
  750. priv->tx_key);
  751. if (err)
  752. return err;
  753. break;
  754. }
  755. case FIRMWARE_TYPE_INTERSIL:
  756. case FIRMWARE_TYPE_SYMBOL:
  757. {
  758. int keylen;
  759. /* Force uniform key length to work around
  760. * firmware bugs */
  761. keylen = priv->keys[priv->tx_key].key_len;
  762. if (keylen > LARGE_KEY_SIZE) {
  763. printk(KERN_ERR "%s: BUG: Key %d has oversize length %d.\n",
  764. priv->ndev->name, priv->tx_key, keylen);
  765. return -E2BIG;
  766. } else if (keylen > SMALL_KEY_SIZE)
  767. keylen = LARGE_KEY_SIZE;
  768. else if (keylen > 0)
  769. keylen = SMALL_KEY_SIZE;
  770. else
  771. keylen = 0;
  772. /* Write all 4 keys */
  773. for (i = 0; i < ORINOCO_MAX_KEYS; i++) {
  774. u8 key[LARGE_KEY_SIZE] = { 0 };
  775. memcpy(key, priv->keys[i].key,
  776. priv->keys[i].key_len);
  777. err = hermes_write_ltv(hw, USER_BAP,
  778. HERMES_RID_CNFDEFAULTKEY0 + i,
  779. HERMES_BYTES_TO_RECLEN(keylen),
  780. key);
  781. if (err)
  782. return err;
  783. }
  784. /* Write the index of the key used in transmission */
  785. err = hermes_write_wordrec(hw, USER_BAP,
  786. HERMES_RID_CNFWEPDEFAULTKEYID,
  787. priv->tx_key);
  788. if (err)
  789. return err;
  790. }
  791. break;
  792. }
  793. return 0;
  794. }
  795. int __orinoco_hw_setup_enc(struct orinoco_private *priv)
  796. {
  797. hermes_t *hw = &priv->hw;
  798. int err = 0;
  799. int master_wep_flag;
  800. int auth_flag;
  801. int enc_flag;
  802. /* Setup WEP keys */
  803. if (priv->encode_alg == ORINOCO_ALG_WEP)
  804. __orinoco_hw_setup_wepkeys(priv);
  805. if (priv->wep_restrict)
  806. auth_flag = HERMES_AUTH_SHARED_KEY;
  807. else
  808. auth_flag = HERMES_AUTH_OPEN;
  809. if (priv->wpa_enabled)
  810. enc_flag = 2;
  811. else if (priv->encode_alg == ORINOCO_ALG_WEP)
  812. enc_flag = 1;
  813. else
  814. enc_flag = 0;
  815. switch (priv->firmware_type) {
  816. case FIRMWARE_TYPE_AGERE: /* Agere style WEP */
  817. if (priv->encode_alg == ORINOCO_ALG_WEP) {
  818. /* Enable the shared-key authentication. */
  819. err = hermes_write_wordrec(hw, USER_BAP,
  820. HERMES_RID_CNFAUTHENTICATION_AGERE,
  821. auth_flag);
  822. }
  823. err = hermes_write_wordrec(hw, USER_BAP,
  824. HERMES_RID_CNFWEPENABLED_AGERE,
  825. enc_flag);
  826. if (err)
  827. return err;
  828. if (priv->has_wpa) {
  829. /* Set WPA key management */
  830. err = hermes_write_wordrec(hw, USER_BAP,
  831. HERMES_RID_CNFSETWPAAUTHMGMTSUITE_AGERE,
  832. priv->key_mgmt);
  833. if (err)
  834. return err;
  835. }
  836. break;
  837. case FIRMWARE_TYPE_INTERSIL: /* Intersil style WEP */
  838. case FIRMWARE_TYPE_SYMBOL: /* Symbol style WEP */
  839. if (priv->encode_alg == ORINOCO_ALG_WEP) {
  840. if (priv->wep_restrict ||
  841. (priv->firmware_type == FIRMWARE_TYPE_SYMBOL))
  842. master_wep_flag = HERMES_WEP_PRIVACY_INVOKED |
  843. HERMES_WEP_EXCL_UNENCRYPTED;
  844. else
  845. master_wep_flag = HERMES_WEP_PRIVACY_INVOKED;
  846. err = hermes_write_wordrec(hw, USER_BAP,
  847. HERMES_RID_CNFAUTHENTICATION,
  848. auth_flag);
  849. if (err)
  850. return err;
  851. } else
  852. master_wep_flag = 0;
  853. if (priv->iw_mode == NL80211_IFTYPE_MONITOR)
  854. master_wep_flag |= HERMES_WEP_HOST_DECRYPT;
  855. /* Master WEP setting : on/off */
  856. err = hermes_write_wordrec(hw, USER_BAP,
  857. HERMES_RID_CNFWEPFLAGS_INTERSIL,
  858. master_wep_flag);
  859. if (err)
  860. return err;
  861. break;
  862. }
  863. return 0;
  864. }
  865. /* key must be 32 bytes, including the tx and rx MIC keys.
  866. * rsc must be NULL or up to 8 bytes
  867. * tsc must be NULL or up to 8 bytes
  868. */
  869. int __orinoco_hw_set_tkip_key(struct orinoco_private *priv, int key_idx,
  870. int set_tx, u8 *key, u8 *rsc, size_t rsc_len,
  871. u8 *tsc, size_t tsc_len)
  872. {
  873. struct {
  874. __le16 idx;
  875. u8 rsc[ORINOCO_SEQ_LEN];
  876. u8 key[TKIP_KEYLEN];
  877. u8 tx_mic[MIC_KEYLEN];
  878. u8 rx_mic[MIC_KEYLEN];
  879. u8 tsc[ORINOCO_SEQ_LEN];
  880. } __attribute__ ((packed)) buf;
  881. hermes_t *hw = &priv->hw;
  882. int ret;
  883. int err;
  884. int k;
  885. u16 xmitting;
  886. key_idx &= 0x3;
  887. if (set_tx)
  888. key_idx |= 0x8000;
  889. buf.idx = cpu_to_le16(key_idx);
  890. memcpy(buf.key, key,
  891. sizeof(buf.key) + sizeof(buf.tx_mic) + sizeof(buf.rx_mic));
  892. if (rsc_len > sizeof(buf.rsc))
  893. rsc_len = sizeof(buf.rsc);
  894. if (tsc_len > sizeof(buf.tsc))
  895. tsc_len = sizeof(buf.tsc);
  896. memset(buf.rsc, 0, sizeof(buf.rsc));
  897. memset(buf.tsc, 0, sizeof(buf.tsc));
  898. if (rsc != NULL)
  899. memcpy(buf.rsc, rsc, rsc_len);
  900. if (tsc != NULL)
  901. memcpy(buf.tsc, tsc, tsc_len);
  902. else
  903. buf.tsc[4] = 0x10;
  904. /* Wait upto 100ms for tx queue to empty */
  905. for (k = 100; k > 0; k--) {
  906. udelay(1000);
  907. ret = hermes_read_wordrec(hw, USER_BAP, HERMES_RID_TXQUEUEEMPTY,
  908. &xmitting);
  909. if (ret || !xmitting)
  910. break;
  911. }
  912. if (k == 0)
  913. ret = -ETIMEDOUT;
  914. err = HERMES_WRITE_RECORD(hw, USER_BAP,
  915. HERMES_RID_CNFADDDEFAULTTKIPKEY_AGERE,
  916. &buf);
  917. return ret ? ret : err;
  918. }
  919. int orinoco_clear_tkip_key(struct orinoco_private *priv, int key_idx)
  920. {
  921. hermes_t *hw = &priv->hw;
  922. int err;
  923. err = hermes_write_wordrec(hw, USER_BAP,
  924. HERMES_RID_CNFREMDEFAULTTKIPKEY_AGERE,
  925. key_idx);
  926. if (err)
  927. printk(KERN_WARNING "%s: Error %d clearing TKIP key %d\n",
  928. priv->ndev->name, err, key_idx);
  929. return err;
  930. }
  931. int __orinoco_hw_set_multicast_list(struct orinoco_private *priv,
  932. struct net_device *dev,
  933. int mc_count, int promisc)
  934. {
  935. hermes_t *hw = &priv->hw;
  936. int err = 0;
  937. if (promisc != priv->promiscuous) {
  938. err = hermes_write_wordrec(hw, USER_BAP,
  939. HERMES_RID_CNFPROMISCUOUSMODE,
  940. promisc);
  941. if (err) {
  942. printk(KERN_ERR "%s: Error %d setting PROMISCUOUSMODE to 1.\n",
  943. priv->ndev->name, err);
  944. } else
  945. priv->promiscuous = promisc;
  946. }
  947. /* If we're not in promiscuous mode, then we need to set the
  948. * group address if either we want to multicast, or if we were
  949. * multicasting and want to stop */
  950. if (!promisc && (mc_count || priv->mc_count)) {
  951. struct netdev_hw_addr *ha;
  952. struct hermes_multicast mclist;
  953. int i = 0;
  954. netdev_for_each_mc_addr(ha, dev) {
  955. if (i == mc_count)
  956. break;
  957. memcpy(mclist.addr[i++], ha->addr, ETH_ALEN);
  958. }
  959. err = hermes_write_ltv(hw, USER_BAP,
  960. HERMES_RID_CNFGROUPADDRESSES,
  961. HERMES_BYTES_TO_RECLEN(mc_count * ETH_ALEN),
  962. &mclist);
  963. if (err)
  964. printk(KERN_ERR "%s: Error %d setting multicast list.\n",
  965. priv->ndev->name, err);
  966. else
  967. priv->mc_count = mc_count;
  968. }
  969. return err;
  970. }
  971. /* Return : < 0 -> error code ; >= 0 -> length */
  972. int orinoco_hw_get_essid(struct orinoco_private *priv, int *active,
  973. char buf[IW_ESSID_MAX_SIZE+1])
  974. {
  975. hermes_t *hw = &priv->hw;
  976. int err = 0;
  977. struct hermes_idstring essidbuf;
  978. char *p = (char *)(&essidbuf.val);
  979. int len;
  980. unsigned long flags;
  981. if (orinoco_lock(priv, &flags) != 0)
  982. return -EBUSY;
  983. if (strlen(priv->desired_essid) > 0) {
  984. /* We read the desired SSID from the hardware rather
  985. than from priv->desired_essid, just in case the
  986. firmware is allowed to change it on us. I'm not
  987. sure about this */
  988. /* My guess is that the OWNSSID should always be whatever
  989. * we set to the card, whereas CURRENT_SSID is the one that
  990. * may change... - Jean II */
  991. u16 rid;
  992. *active = 1;
  993. rid = (priv->port_type == 3) ? HERMES_RID_CNFOWNSSID :
  994. HERMES_RID_CNFDESIREDSSID;
  995. err = hermes_read_ltv(hw, USER_BAP, rid, sizeof(essidbuf),
  996. NULL, &essidbuf);
  997. if (err)
  998. goto fail_unlock;
  999. } else {
  1000. *active = 0;
  1001. err = hermes_read_ltv(hw, USER_BAP, HERMES_RID_CURRENTSSID,
  1002. sizeof(essidbuf), NULL, &essidbuf);
  1003. if (err)
  1004. goto fail_unlock;
  1005. }
  1006. len = le16_to_cpu(essidbuf.len);
  1007. BUG_ON(len > IW_ESSID_MAX_SIZE);
  1008. memset(buf, 0, IW_ESSID_MAX_SIZE);
  1009. memcpy(buf, p, len);
  1010. err = len;
  1011. fail_unlock:
  1012. orinoco_unlock(priv, &flags);
  1013. return err;
  1014. }
  1015. int orinoco_hw_get_freq(struct orinoco_private *priv)
  1016. {
  1017. hermes_t *hw = &priv->hw;
  1018. int err = 0;
  1019. u16 channel;
  1020. int freq = 0;
  1021. unsigned long flags;
  1022. if (orinoco_lock(priv, &flags) != 0)
  1023. return -EBUSY;
  1024. err = hermes_read_wordrec(hw, USER_BAP, HERMES_RID_CURRENTCHANNEL,
  1025. &channel);
  1026. if (err)
  1027. goto out;
  1028. /* Intersil firmware 1.3.5 returns 0 when the interface is down */
  1029. if (channel == 0) {
  1030. err = -EBUSY;
  1031. goto out;
  1032. }
  1033. if ((channel < 1) || (channel > NUM_CHANNELS)) {
  1034. printk(KERN_WARNING "%s: Channel out of range (%d)!\n",
  1035. priv->ndev->name, channel);
  1036. err = -EBUSY;
  1037. goto out;
  1038. }
  1039. freq = ieee80211_dsss_chan_to_freq(channel);
  1040. out:
  1041. orinoco_unlock(priv, &flags);
  1042. if (err > 0)
  1043. err = -EBUSY;
  1044. return err ? err : freq;
  1045. }
  1046. int orinoco_hw_get_bitratelist(struct orinoco_private *priv,
  1047. int *numrates, s32 *rates, int max)
  1048. {
  1049. hermes_t *hw = &priv->hw;
  1050. struct hermes_idstring list;
  1051. unsigned char *p = (unsigned char *)&list.val;
  1052. int err = 0;
  1053. int num;
  1054. int i;
  1055. unsigned long flags;
  1056. if (orinoco_lock(priv, &flags) != 0)
  1057. return -EBUSY;
  1058. err = hermes_read_ltv(hw, USER_BAP, HERMES_RID_SUPPORTEDDATARATES,
  1059. sizeof(list), NULL, &list);
  1060. orinoco_unlock(priv, &flags);
  1061. if (err)
  1062. return err;
  1063. num = le16_to_cpu(list.len);
  1064. *numrates = num;
  1065. num = min(num, max);
  1066. for (i = 0; i < num; i++)
  1067. rates[i] = (p[i] & 0x7f) * 500000; /* convert to bps */
  1068. return 0;
  1069. }
  1070. int orinoco_hw_trigger_scan(struct orinoco_private *priv,
  1071. const struct cfg80211_ssid *ssid)
  1072. {
  1073. struct net_device *dev = priv->ndev;
  1074. hermes_t *hw = &priv->hw;
  1075. unsigned long flags;
  1076. int err = 0;
  1077. if (orinoco_lock(priv, &flags) != 0)
  1078. return -EBUSY;
  1079. /* Scanning with port 0 disabled would fail */
  1080. if (!netif_running(dev)) {
  1081. err = -ENETDOWN;
  1082. goto out;
  1083. }
  1084. /* In monitor mode, the scan results are always empty.
  1085. * Probe responses are passed to the driver as received
  1086. * frames and could be processed in software. */
  1087. if (priv->iw_mode == NL80211_IFTYPE_MONITOR) {
  1088. err = -EOPNOTSUPP;
  1089. goto out;
  1090. }
  1091. if (priv->has_hostscan) {
  1092. switch (priv->firmware_type) {
  1093. case FIRMWARE_TYPE_SYMBOL:
  1094. err = hermes_write_wordrec(hw, USER_BAP,
  1095. HERMES_RID_CNFHOSTSCAN_SYMBOL,
  1096. HERMES_HOSTSCAN_SYMBOL_ONCE |
  1097. HERMES_HOSTSCAN_SYMBOL_BCAST);
  1098. break;
  1099. case FIRMWARE_TYPE_INTERSIL: {
  1100. __le16 req[3];
  1101. req[0] = cpu_to_le16(0x3fff); /* All channels */
  1102. req[1] = cpu_to_le16(0x0001); /* rate 1 Mbps */
  1103. req[2] = 0; /* Any ESSID */
  1104. err = HERMES_WRITE_RECORD(hw, USER_BAP,
  1105. HERMES_RID_CNFHOSTSCAN, &req);
  1106. break;
  1107. }
  1108. case FIRMWARE_TYPE_AGERE:
  1109. if (ssid->ssid_len > 0) {
  1110. struct hermes_idstring idbuf;
  1111. size_t len = ssid->ssid_len;
  1112. idbuf.len = cpu_to_le16(len);
  1113. memcpy(idbuf.val, ssid->ssid, len);
  1114. err = hermes_write_ltv(hw, USER_BAP,
  1115. HERMES_RID_CNFSCANSSID_AGERE,
  1116. HERMES_BYTES_TO_RECLEN(len + 2),
  1117. &idbuf);
  1118. } else
  1119. err = hermes_write_wordrec(hw, USER_BAP,
  1120. HERMES_RID_CNFSCANSSID_AGERE,
  1121. 0); /* Any ESSID */
  1122. if (err)
  1123. break;
  1124. if (priv->has_ext_scan) {
  1125. err = hermes_write_wordrec(hw, USER_BAP,
  1126. HERMES_RID_CNFSCANCHANNELS2GHZ,
  1127. 0x7FFF);
  1128. if (err)
  1129. goto out;
  1130. err = hermes_inquire(hw,
  1131. HERMES_INQ_CHANNELINFO);
  1132. } else
  1133. err = hermes_inquire(hw, HERMES_INQ_SCAN);
  1134. break;
  1135. }
  1136. } else
  1137. err = hermes_inquire(hw, HERMES_INQ_SCAN);
  1138. out:
  1139. orinoco_unlock(priv, &flags);
  1140. return err;
  1141. }
  1142. /* Disassociate from node with BSSID addr */
  1143. int orinoco_hw_disassociate(struct orinoco_private *priv,
  1144. u8 *addr, u16 reason_code)
  1145. {
  1146. hermes_t *hw = &priv->hw;
  1147. int err;
  1148. struct {
  1149. u8 addr[ETH_ALEN];
  1150. __le16 reason_code;
  1151. } __attribute__ ((packed)) buf;
  1152. /* Currently only supported by WPA enabled Agere fw */
  1153. if (!priv->has_wpa)
  1154. return -EOPNOTSUPP;
  1155. memcpy(buf.addr, addr, ETH_ALEN);
  1156. buf.reason_code = cpu_to_le16(reason_code);
  1157. err = HERMES_WRITE_RECORD(hw, USER_BAP,
  1158. HERMES_RID_CNFDISASSOCIATE,
  1159. &buf);
  1160. return err;
  1161. }
  1162. int orinoco_hw_get_current_bssid(struct orinoco_private *priv,
  1163. u8 *addr)
  1164. {
  1165. hermes_t *hw = &priv->hw;
  1166. int err;
  1167. err = hermes_read_ltv(hw, USER_BAP, HERMES_RID_CURRENTBSSID,
  1168. ETH_ALEN, NULL, addr);
  1169. return err;
  1170. }