hw_features_common.c 14 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555
  1. /*
  2. * Common hostapd/wpa_supplicant HW features
  3. * Copyright (c) 2002-2013, Jouni Malinen <j@w1.fi>
  4. * Copyright (c) 2015, Qualcomm Atheros, Inc.
  5. *
  6. * This software may be distributed under the terms of the BSD license.
  7. * See README for more details.
  8. */
  9. #include "includes.h"
  10. #include "common.h"
  11. #include "defs.h"
  12. #include "ieee802_11_defs.h"
  13. #include "ieee802_11_common.h"
  14. #include "hw_features_common.h"
  15. struct hostapd_channel_data * hw_get_channel_chan(struct hostapd_hw_modes *mode,
  16. int chan, int *freq)
  17. {
  18. int i;
  19. if (freq)
  20. *freq = 0;
  21. if (!mode)
  22. return NULL;
  23. for (i = 0; i < mode->num_channels; i++) {
  24. struct hostapd_channel_data *ch = &mode->channels[i];
  25. if (ch->chan == chan) {
  26. if (freq)
  27. *freq = ch->freq;
  28. return ch;
  29. }
  30. }
  31. return NULL;
  32. }
  33. struct hostapd_channel_data * hw_get_channel_freq(struct hostapd_hw_modes *mode,
  34. int freq, int *chan)
  35. {
  36. int i;
  37. if (chan)
  38. *chan = 0;
  39. if (!mode)
  40. return NULL;
  41. for (i = 0; i < mode->num_channels; i++) {
  42. struct hostapd_channel_data *ch = &mode->channels[i];
  43. if (ch->freq == freq) {
  44. if (chan)
  45. *chan = ch->chan;
  46. return ch;
  47. }
  48. }
  49. return NULL;
  50. }
  51. int hw_get_freq(struct hostapd_hw_modes *mode, int chan)
  52. {
  53. int freq;
  54. hw_get_channel_chan(mode, chan, &freq);
  55. return freq;
  56. }
  57. int hw_get_chan(struct hostapd_hw_modes *mode, int freq)
  58. {
  59. int chan;
  60. hw_get_channel_freq(mode, freq, &chan);
  61. return chan;
  62. }
  63. int allowed_ht40_channel_pair(struct hostapd_hw_modes *mode, int pri_chan,
  64. int sec_chan)
  65. {
  66. int ok, j, first;
  67. int allowed[] = { 36, 44, 52, 60, 100, 108, 116, 124, 132, 140,
  68. 149, 157, 165, 184, 192 };
  69. size_t k;
  70. if (pri_chan == sec_chan || !sec_chan)
  71. return 1; /* HT40 not used */
  72. wpa_printf(MSG_DEBUG,
  73. "HT40: control channel: %d secondary channel: %d",
  74. pri_chan, sec_chan);
  75. /* Verify that HT40 secondary channel is an allowed 20 MHz
  76. * channel */
  77. ok = 0;
  78. for (j = 0; j < mode->num_channels; j++) {
  79. struct hostapd_channel_data *chan = &mode->channels[j];
  80. if (!(chan->flag & HOSTAPD_CHAN_DISABLED) &&
  81. chan->chan == sec_chan) {
  82. ok = 1;
  83. break;
  84. }
  85. }
  86. if (!ok) {
  87. wpa_printf(MSG_ERROR, "HT40 secondary channel %d not allowed",
  88. sec_chan);
  89. return 0;
  90. }
  91. /*
  92. * Verify that HT40 primary,secondary channel pair is allowed per
  93. * IEEE 802.11n Annex J. This is only needed for 5 GHz band since
  94. * 2.4 GHz rules allow all cases where the secondary channel fits into
  95. * the list of allowed channels (already checked above).
  96. */
  97. if (mode->mode != HOSTAPD_MODE_IEEE80211A)
  98. return 1;
  99. first = pri_chan < sec_chan ? pri_chan : sec_chan;
  100. ok = 0;
  101. for (k = 0; k < ARRAY_SIZE(allowed); k++) {
  102. if (first == allowed[k]) {
  103. ok = 1;
  104. break;
  105. }
  106. }
  107. if (!ok) {
  108. wpa_printf(MSG_ERROR, "HT40 channel pair (%d, %d) not allowed",
  109. pri_chan, sec_chan);
  110. return 0;
  111. }
  112. return 1;
  113. }
  114. void get_pri_sec_chan(struct wpa_scan_res *bss, int *pri_chan, int *sec_chan)
  115. {
  116. struct ieee80211_ht_operation *oper;
  117. struct ieee802_11_elems elems;
  118. *pri_chan = *sec_chan = 0;
  119. ieee802_11_parse_elems((u8 *) (bss + 1), bss->ie_len, &elems, 0);
  120. if (elems.ht_operation) {
  121. oper = (struct ieee80211_ht_operation *) elems.ht_operation;
  122. *pri_chan = oper->primary_chan;
  123. if (oper->ht_param & HT_INFO_HT_PARAM_STA_CHNL_WIDTH) {
  124. int sec = oper->ht_param &
  125. HT_INFO_HT_PARAM_SECONDARY_CHNL_OFF_MASK;
  126. if (sec == HT_INFO_HT_PARAM_SECONDARY_CHNL_ABOVE)
  127. *sec_chan = *pri_chan + 4;
  128. else if (sec == HT_INFO_HT_PARAM_SECONDARY_CHNL_BELOW)
  129. *sec_chan = *pri_chan - 4;
  130. }
  131. }
  132. }
  133. int check_40mhz_5g(struct hostapd_hw_modes *mode,
  134. struct wpa_scan_results *scan_res, int pri_chan,
  135. int sec_chan)
  136. {
  137. int pri_freq, sec_freq, pri_bss, sec_bss;
  138. int bss_pri_chan, bss_sec_chan;
  139. size_t i;
  140. int match;
  141. if (!mode || !scan_res || !pri_chan || !sec_chan ||
  142. pri_chan == sec_chan)
  143. return 0;
  144. pri_freq = hw_get_freq(mode, pri_chan);
  145. sec_freq = hw_get_freq(mode, sec_chan);
  146. /*
  147. * Switch PRI/SEC channels if Beacons were detected on selected SEC
  148. * channel, but not on selected PRI channel.
  149. */
  150. pri_bss = sec_bss = 0;
  151. for (i = 0; i < scan_res->num; i++) {
  152. struct wpa_scan_res *bss = scan_res->res[i];
  153. if (bss->freq == pri_freq)
  154. pri_bss++;
  155. else if (bss->freq == sec_freq)
  156. sec_bss++;
  157. }
  158. if (sec_bss && !pri_bss) {
  159. wpa_printf(MSG_INFO,
  160. "Switch own primary and secondary channel to get secondary channel with no Beacons from other BSSes");
  161. return 2;
  162. }
  163. /*
  164. * Match PRI/SEC channel with any existing HT40 BSS on the same
  165. * channels that we are about to use (if already mixed order in
  166. * existing BSSes, use own preference).
  167. */
  168. match = 0;
  169. for (i = 0; i < scan_res->num; i++) {
  170. struct wpa_scan_res *bss = scan_res->res[i];
  171. get_pri_sec_chan(bss, &bss_pri_chan, &bss_sec_chan);
  172. if (pri_chan == bss_pri_chan &&
  173. sec_chan == bss_sec_chan) {
  174. match = 1;
  175. break;
  176. }
  177. }
  178. if (!match) {
  179. for (i = 0; i < scan_res->num; i++) {
  180. struct wpa_scan_res *bss = scan_res->res[i];
  181. get_pri_sec_chan(bss, &bss_pri_chan, &bss_sec_chan);
  182. if (pri_chan == bss_sec_chan &&
  183. sec_chan == bss_pri_chan) {
  184. wpa_printf(MSG_INFO, "Switch own primary and "
  185. "secondary channel due to BSS "
  186. "overlap with " MACSTR,
  187. MAC2STR(bss->bssid));
  188. return 2;
  189. }
  190. }
  191. }
  192. return 1;
  193. }
  194. static int check_20mhz_bss(struct wpa_scan_res *bss, int pri_freq, int start,
  195. int end)
  196. {
  197. struct ieee802_11_elems elems;
  198. struct ieee80211_ht_operation *oper;
  199. if (bss->freq < start || bss->freq > end || bss->freq == pri_freq)
  200. return 0;
  201. ieee802_11_parse_elems((u8 *) (bss + 1), bss->ie_len, &elems, 0);
  202. if (!elems.ht_capabilities) {
  203. wpa_printf(MSG_DEBUG, "Found overlapping legacy BSS: "
  204. MACSTR " freq=%d", MAC2STR(bss->bssid), bss->freq);
  205. return 1;
  206. }
  207. if (elems.ht_operation) {
  208. oper = (struct ieee80211_ht_operation *) elems.ht_operation;
  209. if (oper->ht_param & HT_INFO_HT_PARAM_SECONDARY_CHNL_OFF_MASK)
  210. return 0;
  211. wpa_printf(MSG_DEBUG, "Found overlapping 20 MHz HT BSS: "
  212. MACSTR " freq=%d", MAC2STR(bss->bssid), bss->freq);
  213. return 1;
  214. }
  215. return 0;
  216. }
  217. int check_40mhz_2g4(struct hostapd_hw_modes *mode,
  218. struct wpa_scan_results *scan_res, int pri_chan,
  219. int sec_chan)
  220. {
  221. int pri_freq, sec_freq;
  222. int affected_start, affected_end;
  223. size_t i;
  224. if (!mode || !scan_res || !pri_chan || !sec_chan ||
  225. pri_chan == sec_chan)
  226. return 0;
  227. pri_freq = hw_get_freq(mode, pri_chan);
  228. sec_freq = hw_get_freq(mode, sec_chan);
  229. affected_start = (pri_freq + sec_freq) / 2 - 25;
  230. affected_end = (pri_freq + sec_freq) / 2 + 25;
  231. wpa_printf(MSG_DEBUG, "40 MHz affected channel range: [%d,%d] MHz",
  232. affected_start, affected_end);
  233. for (i = 0; i < scan_res->num; i++) {
  234. struct wpa_scan_res *bss = scan_res->res[i];
  235. int pri = bss->freq;
  236. int sec = pri;
  237. struct ieee802_11_elems elems;
  238. /* Check for overlapping 20 MHz BSS */
  239. if (check_20mhz_bss(bss, pri_freq, affected_start,
  240. affected_end)) {
  241. wpa_printf(MSG_DEBUG,
  242. "Overlapping 20 MHz BSS is found");
  243. return 0;
  244. }
  245. get_pri_sec_chan(bss, &pri_chan, &sec_chan);
  246. if (sec_chan) {
  247. if (sec_chan < pri_chan)
  248. sec = pri - 20;
  249. else
  250. sec = pri + 20;
  251. }
  252. if ((pri < affected_start || pri > affected_end) &&
  253. (sec < affected_start || sec > affected_end))
  254. continue; /* not within affected channel range */
  255. wpa_printf(MSG_DEBUG, "Neighboring BSS: " MACSTR
  256. " freq=%d pri=%d sec=%d",
  257. MAC2STR(bss->bssid), bss->freq, pri_chan, sec_chan);
  258. if (sec_chan) {
  259. if (pri_freq != pri || sec_freq != sec) {
  260. wpa_printf(MSG_DEBUG,
  261. "40 MHz pri/sec mismatch with BSS "
  262. MACSTR
  263. " <%d,%d> (chan=%d%c) vs. <%d,%d>",
  264. MAC2STR(bss->bssid),
  265. pri, sec, pri_chan,
  266. sec > pri ? '+' : '-',
  267. pri_freq, sec_freq);
  268. return 0;
  269. }
  270. }
  271. ieee802_11_parse_elems((u8 *) (bss + 1), bss->ie_len, &elems,
  272. 0);
  273. if (elems.ht_capabilities) {
  274. struct ieee80211_ht_capabilities *ht_cap =
  275. (struct ieee80211_ht_capabilities *)
  276. elems.ht_capabilities;
  277. if (le_to_host16(ht_cap->ht_capabilities_info) &
  278. HT_CAP_INFO_40MHZ_INTOLERANT) {
  279. wpa_printf(MSG_DEBUG,
  280. "40 MHz Intolerant is set on channel %d in BSS "
  281. MACSTR, pri, MAC2STR(bss->bssid));
  282. return 0;
  283. }
  284. }
  285. }
  286. return 1;
  287. }
  288. int hostapd_set_freq_params(struct hostapd_freq_params *data,
  289. enum hostapd_hw_mode mode,
  290. int freq, int channel, int ht_enabled,
  291. int vht_enabled, int sec_channel_offset,
  292. int vht_oper_chwidth, int center_segment0,
  293. int center_segment1, u32 vht_caps)
  294. {
  295. os_memset(data, 0, sizeof(*data));
  296. data->mode = mode;
  297. data->freq = freq;
  298. data->channel = channel;
  299. data->ht_enabled = ht_enabled;
  300. data->vht_enabled = vht_enabled;
  301. data->sec_channel_offset = sec_channel_offset;
  302. data->center_freq1 = freq + sec_channel_offset * 10;
  303. data->center_freq2 = 0;
  304. data->bandwidth = sec_channel_offset ? 40 : 20;
  305. if (data->vht_enabled) switch (vht_oper_chwidth) {
  306. case VHT_CHANWIDTH_USE_HT:
  307. if (center_segment1 ||
  308. (center_segment0 != 0 &&
  309. 5000 + center_segment0 * 5 != data->center_freq1 &&
  310. 2407 + center_segment0 * 5 != data->center_freq1))
  311. return -1;
  312. break;
  313. case VHT_CHANWIDTH_80P80MHZ:
  314. if (!(vht_caps & VHT_CAP_SUPP_CHAN_WIDTH_160_80PLUS80MHZ)) {
  315. wpa_printf(MSG_ERROR,
  316. "80+80 channel width is not supported!");
  317. return -1;
  318. }
  319. if (center_segment1 == center_segment0 + 4 ||
  320. center_segment1 == center_segment0 - 4)
  321. return -1;
  322. data->center_freq2 = 5000 + center_segment1 * 5;
  323. /* fall through */
  324. case VHT_CHANWIDTH_80MHZ:
  325. data->bandwidth = 80;
  326. if ((vht_oper_chwidth == VHT_CHANWIDTH_80MHZ &&
  327. center_segment1) ||
  328. (vht_oper_chwidth == VHT_CHANWIDTH_80P80MHZ &&
  329. !center_segment1) ||
  330. !sec_channel_offset)
  331. return -1;
  332. if (!center_segment0) {
  333. if (channel <= 48)
  334. center_segment0 = 42;
  335. else if (channel <= 64)
  336. center_segment0 = 58;
  337. else if (channel <= 112)
  338. center_segment0 = 106;
  339. else if (channel <= 128)
  340. center_segment0 = 122;
  341. else if (channel <= 144)
  342. center_segment0 = 138;
  343. else if (channel <= 161)
  344. center_segment0 = 155;
  345. data->center_freq1 = 5000 + center_segment0 * 5;
  346. } else {
  347. /*
  348. * Note: HT/VHT config and params are coupled. Check if
  349. * HT40 channel band is in VHT80 Pri channel band
  350. * configuration.
  351. */
  352. if (center_segment0 == channel + 6 ||
  353. center_segment0 == channel + 2 ||
  354. center_segment0 == channel - 2 ||
  355. center_segment0 == channel - 6)
  356. data->center_freq1 = 5000 + center_segment0 * 5;
  357. else
  358. return -1;
  359. }
  360. break;
  361. case VHT_CHANWIDTH_160MHZ:
  362. data->bandwidth = 160;
  363. if (!(vht_caps & (VHT_CAP_SUPP_CHAN_WIDTH_160MHZ |
  364. VHT_CAP_SUPP_CHAN_WIDTH_160_80PLUS80MHZ))) {
  365. wpa_printf(MSG_ERROR,
  366. "160MHZ channel width is not supported!");
  367. return -1;
  368. }
  369. if (center_segment1)
  370. return -1;
  371. if (!sec_channel_offset)
  372. return -1;
  373. /*
  374. * Note: HT/VHT config and params are coupled. Check if
  375. * HT40 channel band is in VHT160 channel band configuration.
  376. */
  377. if (center_segment0 == channel + 14 ||
  378. center_segment0 == channel + 10 ||
  379. center_segment0 == channel + 6 ||
  380. center_segment0 == channel + 2 ||
  381. center_segment0 == channel - 2 ||
  382. center_segment0 == channel - 6 ||
  383. center_segment0 == channel - 10 ||
  384. center_segment0 == channel - 14)
  385. data->center_freq1 = 5000 + center_segment0 * 5;
  386. else
  387. return -1;
  388. break;
  389. }
  390. return 0;
  391. }
  392. void set_disable_ht40(struct ieee80211_ht_capabilities *htcaps,
  393. int disabled)
  394. {
  395. /* Masking these out disables HT40 */
  396. le16 msk = host_to_le16(HT_CAP_INFO_SUPP_CHANNEL_WIDTH_SET |
  397. HT_CAP_INFO_SHORT_GI40MHZ);
  398. if (disabled)
  399. htcaps->ht_capabilities_info &= ~msk;
  400. else
  401. htcaps->ht_capabilities_info |= msk;
  402. }
  403. #ifdef CONFIG_IEEE80211AC
  404. static int _ieee80211ac_cap_check(u32 hw, u32 conf, u32 cap,
  405. const char *name)
  406. {
  407. u32 req_cap = conf & cap;
  408. /*
  409. * Make sure we support all requested capabilities.
  410. * NOTE: We assume that 'cap' represents a capability mask,
  411. * not a discrete value.
  412. */
  413. if ((hw & req_cap) != req_cap) {
  414. wpa_printf(MSG_ERROR,
  415. "Driver does not support configured VHT capability [%s]",
  416. name);
  417. return 0;
  418. }
  419. return 1;
  420. }
  421. static int ieee80211ac_cap_check_max(u32 hw, u32 conf, u32 mask,
  422. unsigned int shift,
  423. const char *name)
  424. {
  425. u32 hw_max = hw & mask;
  426. u32 conf_val = conf & mask;
  427. if (conf_val > hw_max) {
  428. wpa_printf(MSG_ERROR,
  429. "Configured VHT capability [%s] exceeds max value supported by the driver (%d > %d)",
  430. name, conf_val >> shift, hw_max >> shift);
  431. return 0;
  432. }
  433. return 1;
  434. }
  435. int ieee80211ac_cap_check(u32 hw, u32 conf)
  436. {
  437. #define VHT_CAP_CHECK(cap) \
  438. do { \
  439. if (!_ieee80211ac_cap_check(hw, conf, cap, #cap)) \
  440. return 0; \
  441. } while (0)
  442. #define VHT_CAP_CHECK_MAX(cap) \
  443. do { \
  444. if (!ieee80211ac_cap_check_max(hw, conf, cap, cap ## _SHIFT, \
  445. #cap)) \
  446. return 0; \
  447. } while (0)
  448. VHT_CAP_CHECK_MAX(VHT_CAP_MAX_MPDU_LENGTH_MASK);
  449. VHT_CAP_CHECK_MAX(VHT_CAP_SUPP_CHAN_WIDTH_MASK);
  450. VHT_CAP_CHECK(VHT_CAP_RXLDPC);
  451. VHT_CAP_CHECK(VHT_CAP_SHORT_GI_80);
  452. VHT_CAP_CHECK(VHT_CAP_SHORT_GI_160);
  453. VHT_CAP_CHECK(VHT_CAP_TXSTBC);
  454. VHT_CAP_CHECK_MAX(VHT_CAP_RXSTBC_MASK);
  455. VHT_CAP_CHECK(VHT_CAP_SU_BEAMFORMER_CAPABLE);
  456. VHT_CAP_CHECK(VHT_CAP_SU_BEAMFORMEE_CAPABLE);
  457. VHT_CAP_CHECK_MAX(VHT_CAP_BEAMFORMEE_STS_MAX);
  458. VHT_CAP_CHECK_MAX(VHT_CAP_SOUNDING_DIMENSION_MAX);
  459. VHT_CAP_CHECK(VHT_CAP_MU_BEAMFORMER_CAPABLE);
  460. VHT_CAP_CHECK(VHT_CAP_MU_BEAMFORMEE_CAPABLE);
  461. VHT_CAP_CHECK(VHT_CAP_VHT_TXOP_PS);
  462. VHT_CAP_CHECK(VHT_CAP_HTC_VHT);
  463. VHT_CAP_CHECK_MAX(VHT_CAP_MAX_A_MPDU_LENGTH_EXPONENT_MAX);
  464. VHT_CAP_CHECK(VHT_CAP_VHT_LINK_ADAPTATION_VHT_UNSOL_MFB);
  465. VHT_CAP_CHECK(VHT_CAP_VHT_LINK_ADAPTATION_VHT_MRQ_MFB);
  466. VHT_CAP_CHECK(VHT_CAP_RX_ANTENNA_PATTERN);
  467. VHT_CAP_CHECK(VHT_CAP_TX_ANTENNA_PATTERN);
  468. #undef VHT_CAP_CHECK
  469. #undef VHT_CAP_CHECK_MAX
  470. return 1;
  471. }
  472. #endif /* CONFIG_IEEE80211AC */