dfs.c 27 KB

1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889909192939495969798991001011021031041051061071081091101111121131141151161171181191201211221231241251261271281291301311321331341351361371381391401411421431441451461471481491501511521531541551561571581591601611621631641651661671681691701711721731741751761771781791801811821831841851861871881891901911921931941951961971981992002012022032042052062072082092102112122132142152162172182192202212222232242252262272282292302312322332342352362372382392402412422432442452462472482492502512522532542552562572582592602612622632642652662672682692702712722732742752762772782792802812822832842852862872882892902912922932942952962972982993003013023033043053063073083093103113123133143153163173183193203213223233243253263273283293303313323333343353363373383393403413423433443453463473483493503513523533543553563573583593603613623633643653663673683693703713723733743753763773783793803813823833843853863873883893903913923933943953963973983994004014024034044054064074084094104114124134144154164174184194204214224234244254264274284294304314324334344354364374384394404414424434444454464474484494504514524534544554564574584594604614624634644654664674684694704714724734744754764774784794804814824834844854864874884894904914924934944954964974984995005015025035045055065075085095105115125135145155165175185195205215225235245255265275285295305315325335345355365375385395405415425435445455465475485495505515525535545555565575585595605615625635645655665675685695705715725735745755765775785795805815825835845855865875885895905915925935945955965975985996006016026036046056066076086096106116126136146156166176186196206216226236246256266276286296306316326336346356366376386396406416426436446456466476486496506516526536546556566576586596606616626636646656666676686696706716726736746756766776786796806816826836846856866876886896906916926936946956966976986997007017027037047057067077087097107117127137147157167177187197207217227237247257267277287297307317327337347357367377387397407417427437447457467477487497507517527537547557567577587597607617627637647657667677687697707717727737747757767777787797807817827837847857867877887897907917927937947957967977987998008018028038048058068078088098108118128138148158168178188198208218228238248258268278288298308318328338348358368378388398408418428438448458468478488498508518528538548558568578588598608618628638648658668678688698708718728738748758768778788798808818828838848858868878888898908918928938948958968978988999009019029039049059069079089099109119129139149159169179189199209219229239249259269279289299309319329339349359369379389399409419429439449459469479489499509519529539549559569579589599609619629639649659669679689699709719729739749759769779789799809819829839849859869879889899909919929939949959969979989991000100110021003100410051006100710081009101010111012101310141015101610171018101910201021102210231024102510261027102810291030103110321033103410351036103710381039104010411042104310441045104610471048104910501051105210531054105510561057105810591060106110621063106410651066106710681069107010711072
  1. /*
  2. * DFS - Dynamic Frequency Selection
  3. * Copyright (c) 2002-2013, Jouni Malinen <j@w1.fi>
  4. * Copyright (c) 2013-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 "utils/includes.h"
  10. #include "utils/common.h"
  11. #include "common/ieee802_11_defs.h"
  12. #include "common/hw_features_common.h"
  13. #include "common/wpa_ctrl.h"
  14. #include "hostapd.h"
  15. #include "ap_drv_ops.h"
  16. #include "drivers/driver.h"
  17. #include "dfs.h"
  18. static int dfs_get_used_n_chans(struct hostapd_iface *iface, int *seg1)
  19. {
  20. int n_chans = 1;
  21. *seg1 = 0;
  22. if (iface->conf->ieee80211n && iface->conf->secondary_channel)
  23. n_chans = 2;
  24. if (iface->conf->ieee80211ac) {
  25. switch (iface->conf->vht_oper_chwidth) {
  26. case VHT_CHANWIDTH_USE_HT:
  27. break;
  28. case VHT_CHANWIDTH_80MHZ:
  29. n_chans = 4;
  30. break;
  31. case VHT_CHANWIDTH_160MHZ:
  32. n_chans = 8;
  33. break;
  34. case VHT_CHANWIDTH_80P80MHZ:
  35. n_chans = 4;
  36. *seg1 = 4;
  37. break;
  38. default:
  39. break;
  40. }
  41. }
  42. return n_chans;
  43. }
  44. static int dfs_channel_available(struct hostapd_channel_data *chan,
  45. int skip_radar)
  46. {
  47. /*
  48. * When radar detection happens, CSA is performed. However, there's no
  49. * time for CAC, so radar channels must be skipped when finding a new
  50. * channel for CSA, unless they are available for immediate use.
  51. */
  52. if (skip_radar && (chan->flag & HOSTAPD_CHAN_RADAR) &&
  53. ((chan->flag & HOSTAPD_CHAN_DFS_MASK) !=
  54. HOSTAPD_CHAN_DFS_AVAILABLE))
  55. return 0;
  56. if (chan->flag & HOSTAPD_CHAN_DISABLED)
  57. return 0;
  58. if ((chan->flag & HOSTAPD_CHAN_RADAR) &&
  59. ((chan->flag & HOSTAPD_CHAN_DFS_MASK) ==
  60. HOSTAPD_CHAN_DFS_UNAVAILABLE))
  61. return 0;
  62. return 1;
  63. }
  64. static int dfs_is_chan_allowed(struct hostapd_channel_data *chan, int n_chans)
  65. {
  66. /*
  67. * The tables contain first valid channel number based on channel width.
  68. * We will also choose this first channel as the control one.
  69. */
  70. int allowed_40[] = { 36, 44, 52, 60, 100, 108, 116, 124, 132, 149, 157,
  71. 184, 192 };
  72. /*
  73. * VHT80, valid channels based on center frequency:
  74. * 42, 58, 106, 122, 138, 155
  75. */
  76. int allowed_80[] = { 36, 52, 100, 116, 132, 149 };
  77. /*
  78. * VHT160 valid channels based on center frequency:
  79. * 50, 114
  80. */
  81. int allowed_160[] = { 36, 100 };
  82. int *allowed = allowed_40;
  83. unsigned int i, allowed_no = 0;
  84. switch (n_chans) {
  85. case 2:
  86. allowed = allowed_40;
  87. allowed_no = ARRAY_SIZE(allowed_40);
  88. break;
  89. case 4:
  90. allowed = allowed_80;
  91. allowed_no = ARRAY_SIZE(allowed_80);
  92. break;
  93. case 8:
  94. allowed = allowed_160;
  95. allowed_no = ARRAY_SIZE(allowed_160);
  96. break;
  97. default:
  98. wpa_printf(MSG_DEBUG, "Unknown width for %d channels", n_chans);
  99. break;
  100. }
  101. for (i = 0; i < allowed_no; i++) {
  102. if (chan->chan == allowed[i])
  103. return 1;
  104. }
  105. return 0;
  106. }
  107. static struct hostapd_channel_data *
  108. dfs_get_chan_data(struct hostapd_hw_modes *mode, int freq, int first_chan_idx)
  109. {
  110. int i;
  111. for (i = first_chan_idx; i < mode->num_channels; i++) {
  112. if (mode->channels[i].freq == freq)
  113. return &mode->channels[i];
  114. }
  115. return NULL;
  116. }
  117. static int dfs_chan_range_available(struct hostapd_hw_modes *mode,
  118. int first_chan_idx, int num_chans,
  119. int skip_radar)
  120. {
  121. struct hostapd_channel_data *first_chan, *chan;
  122. int i;
  123. if (first_chan_idx + num_chans > mode->num_channels)
  124. return 0;
  125. first_chan = &mode->channels[first_chan_idx];
  126. for (i = 0; i < num_chans; i++) {
  127. chan = dfs_get_chan_data(mode, first_chan->freq + i * 20,
  128. first_chan_idx);
  129. if (!chan)
  130. return 0;
  131. if (!dfs_channel_available(chan, skip_radar))
  132. return 0;
  133. }
  134. return 1;
  135. }
  136. static int is_in_chanlist(struct hostapd_iface *iface,
  137. struct hostapd_channel_data *chan)
  138. {
  139. if (!iface->conf->acs_ch_list.num)
  140. return 1;
  141. return freq_range_list_includes(&iface->conf->acs_ch_list, chan->chan);
  142. }
  143. /*
  144. * The function assumes HT40+ operation.
  145. * Make sure to adjust the following variables after calling this:
  146. * - hapd->secondary_channel
  147. * - hapd->vht_oper_centr_freq_seg0_idx
  148. * - hapd->vht_oper_centr_freq_seg1_idx
  149. */
  150. static int dfs_find_channel(struct hostapd_iface *iface,
  151. struct hostapd_channel_data **ret_chan,
  152. int idx, int skip_radar)
  153. {
  154. struct hostapd_hw_modes *mode;
  155. struct hostapd_channel_data *chan;
  156. int i, channel_idx = 0, n_chans, n_chans1;
  157. mode = iface->current_mode;
  158. n_chans = dfs_get_used_n_chans(iface, &n_chans1);
  159. wpa_printf(MSG_DEBUG, "DFS new chan checking %d channels", n_chans);
  160. for (i = 0; i < mode->num_channels; i++) {
  161. chan = &mode->channels[i];
  162. /* Skip HT40/VHT incompatible channels */
  163. if (iface->conf->ieee80211n &&
  164. iface->conf->secondary_channel &&
  165. !dfs_is_chan_allowed(chan, n_chans))
  166. continue;
  167. /* Skip incompatible chandefs */
  168. if (!dfs_chan_range_available(mode, i, n_chans, skip_radar))
  169. continue;
  170. if (!is_in_chanlist(iface, chan))
  171. continue;
  172. if (ret_chan && idx == channel_idx) {
  173. wpa_printf(MSG_DEBUG, "Selected ch. #%d", chan->chan);
  174. *ret_chan = chan;
  175. return idx;
  176. }
  177. wpa_printf(MSG_DEBUG, "Adding channel: %d", chan->chan);
  178. channel_idx++;
  179. }
  180. return channel_idx;
  181. }
  182. static void dfs_adjust_vht_center_freq(struct hostapd_iface *iface,
  183. struct hostapd_channel_data *chan,
  184. int secondary_channel,
  185. u8 *vht_oper_centr_freq_seg0_idx,
  186. u8 *vht_oper_centr_freq_seg1_idx)
  187. {
  188. if (!iface->conf->ieee80211ac)
  189. return;
  190. if (!chan)
  191. return;
  192. *vht_oper_centr_freq_seg1_idx = 0;
  193. switch (iface->conf->vht_oper_chwidth) {
  194. case VHT_CHANWIDTH_USE_HT:
  195. if (secondary_channel == 1)
  196. *vht_oper_centr_freq_seg0_idx = chan->chan + 2;
  197. else if (secondary_channel == -1)
  198. *vht_oper_centr_freq_seg0_idx = chan->chan - 2;
  199. else
  200. *vht_oper_centr_freq_seg0_idx = chan->chan;
  201. break;
  202. case VHT_CHANWIDTH_80MHZ:
  203. *vht_oper_centr_freq_seg0_idx = chan->chan + 6;
  204. break;
  205. case VHT_CHANWIDTH_160MHZ:
  206. *vht_oper_centr_freq_seg0_idx = chan->chan + 14;
  207. break;
  208. default:
  209. wpa_printf(MSG_INFO, "DFS only VHT20/40/80/160 is supported now");
  210. *vht_oper_centr_freq_seg0_idx = 0;
  211. break;
  212. }
  213. wpa_printf(MSG_DEBUG, "DFS adjusting VHT center frequency: %d, %d",
  214. *vht_oper_centr_freq_seg0_idx,
  215. *vht_oper_centr_freq_seg1_idx);
  216. }
  217. /* Return start channel idx we will use for mode->channels[idx] */
  218. static int dfs_get_start_chan_idx(struct hostapd_iface *iface, int *seg1_start)
  219. {
  220. struct hostapd_hw_modes *mode;
  221. struct hostapd_channel_data *chan;
  222. int channel_no = iface->conf->channel;
  223. int res = -1, i;
  224. int chan_seg1 = -1;
  225. *seg1_start = -1;
  226. /* HT40- */
  227. if (iface->conf->ieee80211n && iface->conf->secondary_channel == -1)
  228. channel_no -= 4;
  229. /* VHT */
  230. if (iface->conf->ieee80211ac) {
  231. switch (iface->conf->vht_oper_chwidth) {
  232. case VHT_CHANWIDTH_USE_HT:
  233. break;
  234. case VHT_CHANWIDTH_80MHZ:
  235. channel_no =
  236. iface->conf->vht_oper_centr_freq_seg0_idx - 6;
  237. break;
  238. case VHT_CHANWIDTH_160MHZ:
  239. channel_no =
  240. iface->conf->vht_oper_centr_freq_seg0_idx - 14;
  241. break;
  242. case VHT_CHANWIDTH_80P80MHZ:
  243. channel_no =
  244. iface->conf->vht_oper_centr_freq_seg0_idx - 6;
  245. chan_seg1 =
  246. iface->conf->vht_oper_centr_freq_seg1_idx - 6;
  247. break;
  248. default:
  249. wpa_printf(MSG_INFO,
  250. "DFS only VHT20/40/80/160/80+80 is supported now");
  251. channel_no = -1;
  252. break;
  253. }
  254. }
  255. /* Get idx */
  256. mode = iface->current_mode;
  257. for (i = 0; i < mode->num_channels; i++) {
  258. chan = &mode->channels[i];
  259. if (chan->chan == channel_no) {
  260. res = i;
  261. break;
  262. }
  263. }
  264. if (res != -1 && chan_seg1 > -1) {
  265. int found = 0;
  266. /* Get idx for seg1 */
  267. mode = iface->current_mode;
  268. for (i = 0; i < mode->num_channels; i++) {
  269. chan = &mode->channels[i];
  270. if (chan->chan == chan_seg1) {
  271. *seg1_start = i;
  272. found = 1;
  273. break;
  274. }
  275. }
  276. if (!found)
  277. res = -1;
  278. }
  279. if (res == -1) {
  280. wpa_printf(MSG_DEBUG,
  281. "DFS chan_idx seems wrong; num-ch: %d ch-no: %d conf-ch-no: %d 11n: %d sec-ch: %d vht-oper-width: %d",
  282. mode->num_channels, channel_no, iface->conf->channel,
  283. iface->conf->ieee80211n,
  284. iface->conf->secondary_channel,
  285. iface->conf->vht_oper_chwidth);
  286. for (i = 0; i < mode->num_channels; i++) {
  287. wpa_printf(MSG_DEBUG, "Available channel: %d",
  288. mode->channels[i].chan);
  289. }
  290. }
  291. return res;
  292. }
  293. /* At least one channel have radar flag */
  294. static int dfs_check_chans_radar(struct hostapd_iface *iface,
  295. int start_chan_idx, int n_chans)
  296. {
  297. struct hostapd_channel_data *channel;
  298. struct hostapd_hw_modes *mode;
  299. int i, res = 0;
  300. mode = iface->current_mode;
  301. for (i = 0; i < n_chans; i++) {
  302. channel = &mode->channels[start_chan_idx + i];
  303. if (channel->flag & HOSTAPD_CHAN_RADAR)
  304. res++;
  305. }
  306. return res;
  307. }
  308. /* All channels available */
  309. static int dfs_check_chans_available(struct hostapd_iface *iface,
  310. int start_chan_idx, int n_chans)
  311. {
  312. struct hostapd_channel_data *channel;
  313. struct hostapd_hw_modes *mode;
  314. int i;
  315. mode = iface->current_mode;
  316. for (i = 0; i < n_chans; i++) {
  317. channel = &mode->channels[start_chan_idx + i];
  318. if (channel->flag & HOSTAPD_CHAN_DISABLED)
  319. break;
  320. if (!(channel->flag & HOSTAPD_CHAN_RADAR))
  321. continue;
  322. if ((channel->flag & HOSTAPD_CHAN_DFS_MASK) !=
  323. HOSTAPD_CHAN_DFS_AVAILABLE)
  324. break;
  325. }
  326. return i == n_chans;
  327. }
  328. /* At least one channel unavailable */
  329. static int dfs_check_chans_unavailable(struct hostapd_iface *iface,
  330. int start_chan_idx,
  331. int n_chans)
  332. {
  333. struct hostapd_channel_data *channel;
  334. struct hostapd_hw_modes *mode;
  335. int i, res = 0;
  336. mode = iface->current_mode;
  337. for (i = 0; i < n_chans; i++) {
  338. channel = &mode->channels[start_chan_idx + i];
  339. if (channel->flag & HOSTAPD_CHAN_DISABLED)
  340. res++;
  341. if ((channel->flag & HOSTAPD_CHAN_DFS_MASK) ==
  342. HOSTAPD_CHAN_DFS_UNAVAILABLE)
  343. res++;
  344. }
  345. return res;
  346. }
  347. static struct hostapd_channel_data *
  348. dfs_get_valid_channel(struct hostapd_iface *iface,
  349. int *secondary_channel,
  350. u8 *vht_oper_centr_freq_seg0_idx,
  351. u8 *vht_oper_centr_freq_seg1_idx,
  352. int skip_radar)
  353. {
  354. struct hostapd_hw_modes *mode;
  355. struct hostapd_channel_data *chan = NULL;
  356. int num_available_chandefs;
  357. int chan_idx;
  358. u32 _rand;
  359. wpa_printf(MSG_DEBUG, "DFS: Selecting random channel");
  360. *secondary_channel = 0;
  361. *vht_oper_centr_freq_seg0_idx = 0;
  362. *vht_oper_centr_freq_seg1_idx = 0;
  363. if (iface->current_mode == NULL)
  364. return NULL;
  365. mode = iface->current_mode;
  366. if (mode->mode != HOSTAPD_MODE_IEEE80211A)
  367. return NULL;
  368. /* Get the count first */
  369. num_available_chandefs = dfs_find_channel(iface, NULL, 0, skip_radar);
  370. if (num_available_chandefs == 0)
  371. return NULL;
  372. if (os_get_random((u8 *) &_rand, sizeof(_rand)) < 0)
  373. _rand = os_random();
  374. chan_idx = _rand % num_available_chandefs;
  375. dfs_find_channel(iface, &chan, chan_idx, skip_radar);
  376. /* dfs_find_channel() calculations assume HT40+ */
  377. if (iface->conf->secondary_channel)
  378. *secondary_channel = 1;
  379. else
  380. *secondary_channel = 0;
  381. dfs_adjust_vht_center_freq(iface, chan,
  382. *secondary_channel,
  383. vht_oper_centr_freq_seg0_idx,
  384. vht_oper_centr_freq_seg1_idx);
  385. return chan;
  386. }
  387. static int set_dfs_state_freq(struct hostapd_iface *iface, int freq, u32 state)
  388. {
  389. struct hostapd_hw_modes *mode;
  390. struct hostapd_channel_data *chan = NULL;
  391. int i;
  392. mode = iface->current_mode;
  393. if (mode == NULL)
  394. return 0;
  395. wpa_printf(MSG_DEBUG, "set_dfs_state 0x%X for %d MHz", state, freq);
  396. for (i = 0; i < iface->current_mode->num_channels; i++) {
  397. chan = &iface->current_mode->channels[i];
  398. if (chan->freq == freq) {
  399. if (chan->flag & HOSTAPD_CHAN_RADAR) {
  400. chan->flag &= ~HOSTAPD_CHAN_DFS_MASK;
  401. chan->flag |= state;
  402. return 1; /* Channel found */
  403. }
  404. }
  405. }
  406. wpa_printf(MSG_WARNING, "Can't set DFS state for freq %d MHz", freq);
  407. return 0;
  408. }
  409. static int set_dfs_state(struct hostapd_iface *iface, int freq, int ht_enabled,
  410. int chan_offset, int chan_width, int cf1,
  411. int cf2, u32 state)
  412. {
  413. int n_chans = 1, i;
  414. struct hostapd_hw_modes *mode;
  415. int frequency = freq;
  416. int ret = 0;
  417. mode = iface->current_mode;
  418. if (mode == NULL)
  419. return 0;
  420. if (mode->mode != HOSTAPD_MODE_IEEE80211A) {
  421. wpa_printf(MSG_WARNING, "current_mode != IEEE80211A");
  422. return 0;
  423. }
  424. /* Seems cf1 and chan_width is enough here */
  425. switch (chan_width) {
  426. case CHAN_WIDTH_20_NOHT:
  427. case CHAN_WIDTH_20:
  428. n_chans = 1;
  429. if (frequency == 0)
  430. frequency = cf1;
  431. break;
  432. case CHAN_WIDTH_40:
  433. n_chans = 2;
  434. frequency = cf1 - 10;
  435. break;
  436. case CHAN_WIDTH_80:
  437. n_chans = 4;
  438. frequency = cf1 - 30;
  439. break;
  440. case CHAN_WIDTH_160:
  441. n_chans = 8;
  442. frequency = cf1 - 70;
  443. break;
  444. default:
  445. wpa_printf(MSG_INFO, "DFS chan_width %d not supported",
  446. chan_width);
  447. break;
  448. }
  449. wpa_printf(MSG_DEBUG, "DFS freq: %dMHz, n_chans: %d", frequency,
  450. n_chans);
  451. for (i = 0; i < n_chans; i++) {
  452. ret += set_dfs_state_freq(iface, frequency, state);
  453. frequency = frequency + 20;
  454. }
  455. return ret;
  456. }
  457. static int dfs_are_channels_overlapped(struct hostapd_iface *iface, int freq,
  458. int chan_width, int cf1, int cf2)
  459. {
  460. int start_chan_idx, start_chan_idx1;
  461. struct hostapd_hw_modes *mode;
  462. struct hostapd_channel_data *chan;
  463. int n_chans, n_chans1, i, j, frequency = freq, radar_n_chans = 1;
  464. u8 radar_chan;
  465. int res = 0;
  466. /* Our configuration */
  467. mode = iface->current_mode;
  468. start_chan_idx = dfs_get_start_chan_idx(iface, &start_chan_idx1);
  469. n_chans = dfs_get_used_n_chans(iface, &n_chans1);
  470. /* Check we are on DFS channel(s) */
  471. if (!dfs_check_chans_radar(iface, start_chan_idx, n_chans))
  472. return 0;
  473. /* Reported via radar event */
  474. switch (chan_width) {
  475. case CHAN_WIDTH_20_NOHT:
  476. case CHAN_WIDTH_20:
  477. radar_n_chans = 1;
  478. if (frequency == 0)
  479. frequency = cf1;
  480. break;
  481. case CHAN_WIDTH_40:
  482. radar_n_chans = 2;
  483. frequency = cf1 - 10;
  484. break;
  485. case CHAN_WIDTH_80:
  486. radar_n_chans = 4;
  487. frequency = cf1 - 30;
  488. break;
  489. case CHAN_WIDTH_160:
  490. radar_n_chans = 8;
  491. frequency = cf1 - 70;
  492. break;
  493. default:
  494. wpa_printf(MSG_INFO, "DFS chan_width %d not supported",
  495. chan_width);
  496. break;
  497. }
  498. ieee80211_freq_to_chan(frequency, &radar_chan);
  499. for (i = 0; i < n_chans; i++) {
  500. chan = &mode->channels[start_chan_idx + i];
  501. if (!(chan->flag & HOSTAPD_CHAN_RADAR))
  502. continue;
  503. for (j = 0; j < radar_n_chans; j++) {
  504. wpa_printf(MSG_DEBUG, "checking our: %d, radar: %d",
  505. chan->chan, radar_chan + j * 4);
  506. if (chan->chan == radar_chan + j * 4)
  507. res++;
  508. }
  509. }
  510. wpa_printf(MSG_DEBUG, "overlapped: %d", res);
  511. return res;
  512. }
  513. static unsigned int dfs_get_cac_time(struct hostapd_iface *iface,
  514. int start_chan_idx, int n_chans)
  515. {
  516. struct hostapd_channel_data *channel;
  517. struct hostapd_hw_modes *mode;
  518. int i;
  519. unsigned int cac_time_ms = 0;
  520. mode = iface->current_mode;
  521. for (i = 0; i < n_chans; i++) {
  522. channel = &mode->channels[start_chan_idx + i];
  523. if (!(channel->flag & HOSTAPD_CHAN_RADAR))
  524. continue;
  525. if (channel->dfs_cac_ms > cac_time_ms)
  526. cac_time_ms = channel->dfs_cac_ms;
  527. }
  528. return cac_time_ms;
  529. }
  530. /*
  531. * Main DFS handler
  532. * 1 - continue channel/ap setup
  533. * 0 - channel/ap setup will be continued after CAC
  534. * -1 - hit critical error
  535. */
  536. int hostapd_handle_dfs(struct hostapd_iface *iface)
  537. {
  538. struct hostapd_channel_data *channel;
  539. int res, n_chans, n_chans1, start_chan_idx, start_chan_idx1;
  540. int skip_radar = 0;
  541. if (!iface->current_mode) {
  542. /*
  543. * This can happen with drivers that do not provide mode
  544. * information and as such, cannot really use hostapd for DFS.
  545. */
  546. wpa_printf(MSG_DEBUG,
  547. "DFS: No current_mode information - assume no need to perform DFS operations by hostapd");
  548. return 1;
  549. }
  550. iface->cac_started = 0;
  551. do {
  552. /* Get start (first) channel for current configuration */
  553. start_chan_idx = dfs_get_start_chan_idx(iface,
  554. &start_chan_idx1);
  555. if (start_chan_idx == -1)
  556. return -1;
  557. /* Get number of used channels, depend on width */
  558. n_chans = dfs_get_used_n_chans(iface, &n_chans1);
  559. /* Setup CAC time */
  560. iface->dfs_cac_ms = dfs_get_cac_time(iface, start_chan_idx,
  561. n_chans);
  562. /* Check if any of configured channels require DFS */
  563. res = dfs_check_chans_radar(iface, start_chan_idx, n_chans);
  564. wpa_printf(MSG_DEBUG,
  565. "DFS %d channels required radar detection",
  566. res);
  567. if (!res)
  568. return 1;
  569. /* Check if all channels are DFS available */
  570. res = dfs_check_chans_available(iface, start_chan_idx, n_chans);
  571. wpa_printf(MSG_DEBUG,
  572. "DFS all channels available, (SKIP CAC): %s",
  573. res ? "yes" : "no");
  574. if (res)
  575. return 1;
  576. /* Check if any of configured channels is unavailable */
  577. res = dfs_check_chans_unavailable(iface, start_chan_idx,
  578. n_chans);
  579. wpa_printf(MSG_DEBUG, "DFS %d chans unavailable - choose other channel: %s",
  580. res, res ? "yes": "no");
  581. if (res) {
  582. int sec = 0;
  583. u8 cf1 = 0, cf2 = 0;
  584. channel = dfs_get_valid_channel(iface, &sec, &cf1, &cf2,
  585. skip_radar);
  586. if (!channel) {
  587. wpa_printf(MSG_ERROR, "could not get valid channel");
  588. return -1;
  589. }
  590. iface->freq = channel->freq;
  591. iface->conf->channel = channel->chan;
  592. iface->conf->secondary_channel = sec;
  593. iface->conf->vht_oper_centr_freq_seg0_idx = cf1;
  594. iface->conf->vht_oper_centr_freq_seg1_idx = cf2;
  595. }
  596. } while (res);
  597. /* Finally start CAC */
  598. hostapd_set_state(iface, HAPD_IFACE_DFS);
  599. wpa_printf(MSG_DEBUG, "DFS start CAC on %d MHz", iface->freq);
  600. wpa_msg(iface->bss[0]->msg_ctx, MSG_INFO, DFS_EVENT_CAC_START
  601. "freq=%d chan=%d sec_chan=%d, width=%d, seg0=%d, seg1=%d, cac_time=%ds",
  602. iface->freq,
  603. iface->conf->channel, iface->conf->secondary_channel,
  604. iface->conf->vht_oper_chwidth,
  605. iface->conf->vht_oper_centr_freq_seg0_idx,
  606. iface->conf->vht_oper_centr_freq_seg1_idx,
  607. iface->dfs_cac_ms / 1000);
  608. res = hostapd_start_dfs_cac(iface, iface->conf->hw_mode,
  609. iface->freq,
  610. iface->conf->channel,
  611. iface->conf->ieee80211n,
  612. iface->conf->ieee80211ac,
  613. iface->conf->secondary_channel,
  614. iface->conf->vht_oper_chwidth,
  615. iface->conf->vht_oper_centr_freq_seg0_idx,
  616. iface->conf->vht_oper_centr_freq_seg1_idx);
  617. if (res) {
  618. wpa_printf(MSG_ERROR, "DFS start_dfs_cac() failed, %d", res);
  619. return -1;
  620. }
  621. return 0;
  622. }
  623. int hostapd_dfs_complete_cac(struct hostapd_iface *iface, int success, int freq,
  624. int ht_enabled, int chan_offset, int chan_width,
  625. int cf1, int cf2)
  626. {
  627. wpa_msg(iface->bss[0]->msg_ctx, MSG_INFO, DFS_EVENT_CAC_COMPLETED
  628. "success=%d freq=%d ht_enabled=%d chan_offset=%d chan_width=%d cf1=%d cf2=%d",
  629. success, freq, ht_enabled, chan_offset, chan_width, cf1, cf2);
  630. if (success) {
  631. /* Complete iface/ap configuration */
  632. if (iface->drv_flags & WPA_DRIVER_FLAGS_DFS_OFFLOAD) {
  633. /* Complete AP configuration for the first bring up. */
  634. if (iface->state != HAPD_IFACE_ENABLED)
  635. hostapd_setup_interface_complete(iface, 0);
  636. else
  637. iface->cac_started = 0;
  638. } else {
  639. set_dfs_state(iface, freq, ht_enabled, chan_offset,
  640. chan_width, cf1, cf2,
  641. HOSTAPD_CHAN_DFS_AVAILABLE);
  642. iface->cac_started = 0;
  643. hostapd_setup_interface_complete(iface, 0);
  644. }
  645. }
  646. return 0;
  647. }
  648. static int hostapd_dfs_start_channel_switch_cac(struct hostapd_iface *iface)
  649. {
  650. struct hostapd_channel_data *channel;
  651. int secondary_channel;
  652. u8 vht_oper_centr_freq_seg0_idx = 0;
  653. u8 vht_oper_centr_freq_seg1_idx = 0;
  654. int skip_radar = 0;
  655. int err = 1;
  656. /* Radar detected during active CAC */
  657. iface->cac_started = 0;
  658. channel = dfs_get_valid_channel(iface, &secondary_channel,
  659. &vht_oper_centr_freq_seg0_idx,
  660. &vht_oper_centr_freq_seg1_idx,
  661. skip_radar);
  662. if (!channel) {
  663. wpa_printf(MSG_ERROR, "No valid channel available");
  664. hostapd_setup_interface_complete(iface, err);
  665. return err;
  666. }
  667. wpa_printf(MSG_DEBUG, "DFS will switch to a new channel %d",
  668. channel->chan);
  669. wpa_msg(iface->bss[0]->msg_ctx, MSG_INFO, DFS_EVENT_NEW_CHANNEL
  670. "freq=%d chan=%d sec_chan=%d", channel->freq,
  671. channel->chan, secondary_channel);
  672. iface->freq = channel->freq;
  673. iface->conf->channel = channel->chan;
  674. iface->conf->secondary_channel = secondary_channel;
  675. iface->conf->vht_oper_centr_freq_seg0_idx =
  676. vht_oper_centr_freq_seg0_idx;
  677. iface->conf->vht_oper_centr_freq_seg1_idx =
  678. vht_oper_centr_freq_seg1_idx;
  679. err = 0;
  680. hostapd_setup_interface_complete(iface, err);
  681. return err;
  682. }
  683. static int hostapd_csa_in_progress(struct hostapd_iface *iface)
  684. {
  685. unsigned int i;
  686. for (i = 0; i < iface->num_bss; i++)
  687. if (iface->bss[i]->csa_in_progress)
  688. return 1;
  689. return 0;
  690. }
  691. static int hostapd_dfs_start_channel_switch(struct hostapd_iface *iface)
  692. {
  693. struct hostapd_channel_data *channel;
  694. int secondary_channel;
  695. u8 vht_oper_centr_freq_seg0_idx;
  696. u8 vht_oper_centr_freq_seg1_idx;
  697. int skip_radar = 1;
  698. struct csa_settings csa_settings;
  699. unsigned int i;
  700. int err = 1;
  701. wpa_printf(MSG_DEBUG, "%s called (CAC active: %s, CSA active: %s)",
  702. __func__, iface->cac_started ? "yes" : "no",
  703. hostapd_csa_in_progress(iface) ? "yes" : "no");
  704. /* Check if CSA in progress */
  705. if (hostapd_csa_in_progress(iface))
  706. return 0;
  707. /* Check if active CAC */
  708. if (iface->cac_started)
  709. return hostapd_dfs_start_channel_switch_cac(iface);
  710. /* Perform channel switch/CSA */
  711. channel = dfs_get_valid_channel(iface, &secondary_channel,
  712. &vht_oper_centr_freq_seg0_idx,
  713. &vht_oper_centr_freq_seg1_idx,
  714. skip_radar);
  715. if (!channel) {
  716. /*
  717. * If there is no channel to switch immediately to, check if
  718. * there is another channel where we can switch even if it
  719. * requires to perform a CAC first.
  720. */
  721. skip_radar = 0;
  722. channel = dfs_get_valid_channel(iface, &secondary_channel,
  723. &vht_oper_centr_freq_seg0_idx,
  724. &vht_oper_centr_freq_seg1_idx,
  725. skip_radar);
  726. if (!channel) {
  727. /* FIXME: Wait for channel(s) to become available */
  728. hostapd_disable_iface(iface);
  729. return err;
  730. }
  731. iface->freq = channel->freq;
  732. iface->conf->channel = channel->chan;
  733. iface->conf->secondary_channel = secondary_channel;
  734. iface->conf->vht_oper_centr_freq_seg0_idx =
  735. vht_oper_centr_freq_seg0_idx;
  736. iface->conf->vht_oper_centr_freq_seg1_idx =
  737. vht_oper_centr_freq_seg1_idx;
  738. hostapd_disable_iface(iface);
  739. hostapd_enable_iface(iface);
  740. return 0;
  741. }
  742. wpa_printf(MSG_DEBUG, "DFS will switch to a new channel %d",
  743. channel->chan);
  744. wpa_msg(iface->bss[0]->msg_ctx, MSG_INFO, DFS_EVENT_NEW_CHANNEL
  745. "freq=%d chan=%d sec_chan=%d", channel->freq,
  746. channel->chan, secondary_channel);
  747. /* Setup CSA request */
  748. os_memset(&csa_settings, 0, sizeof(csa_settings));
  749. csa_settings.cs_count = 5;
  750. csa_settings.block_tx = 1;
  751. err = hostapd_set_freq_params(&csa_settings.freq_params,
  752. iface->conf->hw_mode,
  753. channel->freq,
  754. channel->chan,
  755. iface->conf->ieee80211n,
  756. iface->conf->ieee80211ac,
  757. secondary_channel,
  758. iface->conf->vht_oper_chwidth,
  759. vht_oper_centr_freq_seg0_idx,
  760. vht_oper_centr_freq_seg1_idx,
  761. iface->current_mode->vht_capab);
  762. if (err) {
  763. wpa_printf(MSG_ERROR, "DFS failed to calculate CSA freq params");
  764. hostapd_disable_iface(iface);
  765. return err;
  766. }
  767. for (i = 0; i < iface->num_bss; i++) {
  768. err = hostapd_switch_channel(iface->bss[i], &csa_settings);
  769. if (err)
  770. break;
  771. }
  772. if (err) {
  773. wpa_printf(MSG_WARNING, "DFS failed to schedule CSA (%d) - trying fallback",
  774. err);
  775. iface->freq = channel->freq;
  776. iface->conf->channel = channel->chan;
  777. iface->conf->secondary_channel = secondary_channel;
  778. iface->conf->vht_oper_centr_freq_seg0_idx =
  779. vht_oper_centr_freq_seg0_idx;
  780. iface->conf->vht_oper_centr_freq_seg1_idx =
  781. vht_oper_centr_freq_seg1_idx;
  782. hostapd_disable_iface(iface);
  783. hostapd_enable_iface(iface);
  784. return 0;
  785. }
  786. /* Channel configuration will be updated once CSA completes and
  787. * ch_switch_notify event is received */
  788. wpa_printf(MSG_DEBUG, "DFS waiting channel switch event");
  789. return 0;
  790. }
  791. int hostapd_dfs_radar_detected(struct hostapd_iface *iface, int freq,
  792. int ht_enabled, int chan_offset, int chan_width,
  793. int cf1, int cf2)
  794. {
  795. int res;
  796. wpa_msg(iface->bss[0]->msg_ctx, MSG_INFO, DFS_EVENT_RADAR_DETECTED
  797. "freq=%d ht_enabled=%d chan_offset=%d chan_width=%d cf1=%d cf2=%d",
  798. freq, ht_enabled, chan_offset, chan_width, cf1, cf2);
  799. /* Proceed only if DFS is not offloaded to the driver */
  800. if (iface->drv_flags & WPA_DRIVER_FLAGS_DFS_OFFLOAD)
  801. return 0;
  802. if (!iface->conf->ieee80211h)
  803. return 0;
  804. /* mark radar frequency as invalid */
  805. set_dfs_state(iface, freq, ht_enabled, chan_offset, chan_width,
  806. cf1, cf2, HOSTAPD_CHAN_DFS_UNAVAILABLE);
  807. /* Skip if reported radar event not overlapped our channels */
  808. res = dfs_are_channels_overlapped(iface, freq, chan_width, cf1, cf2);
  809. if (!res)
  810. return 0;
  811. /* radar detected while operating, switch the channel. */
  812. res = hostapd_dfs_start_channel_switch(iface);
  813. return res;
  814. }
  815. int hostapd_dfs_nop_finished(struct hostapd_iface *iface, int freq,
  816. int ht_enabled, int chan_offset, int chan_width,
  817. int cf1, int cf2)
  818. {
  819. wpa_msg(iface->bss[0]->msg_ctx, MSG_INFO, DFS_EVENT_NOP_FINISHED
  820. "freq=%d ht_enabled=%d chan_offset=%d chan_width=%d cf1=%d cf2=%d",
  821. freq, ht_enabled, chan_offset, chan_width, cf1, cf2);
  822. /* Proceed only if DFS is not offloaded to the driver */
  823. if (iface->drv_flags & WPA_DRIVER_FLAGS_DFS_OFFLOAD)
  824. return 0;
  825. /* TODO add correct implementation here */
  826. set_dfs_state(iface, freq, ht_enabled, chan_offset, chan_width,
  827. cf1, cf2, HOSTAPD_CHAN_DFS_USABLE);
  828. return 0;
  829. }
  830. int hostapd_is_dfs_required(struct hostapd_iface *iface)
  831. {
  832. int n_chans, n_chans1, start_chan_idx, start_chan_idx1, res;
  833. if (!iface->conf->ieee80211h || !iface->current_mode ||
  834. iface->current_mode->mode != HOSTAPD_MODE_IEEE80211A)
  835. return 0;
  836. /* Get start (first) channel for current configuration */
  837. start_chan_idx = dfs_get_start_chan_idx(iface, &start_chan_idx1);
  838. if (start_chan_idx == -1)
  839. return -1;
  840. /* Get number of used channels, depend on width */
  841. n_chans = dfs_get_used_n_chans(iface, &n_chans1);
  842. /* Check if any of configured channels require DFS */
  843. res = dfs_check_chans_radar(iface, start_chan_idx, n_chans);
  844. if (res)
  845. return res;
  846. if (start_chan_idx1 >= 0 && n_chans1 > 0)
  847. res = dfs_check_chans_radar(iface, start_chan_idx1, n_chans1);
  848. return res;
  849. }
  850. int hostapd_dfs_start_cac(struct hostapd_iface *iface, int freq,
  851. int ht_enabled, int chan_offset, int chan_width,
  852. int cf1, int cf2)
  853. {
  854. wpa_msg(iface->bss[0]->msg_ctx, MSG_INFO, DFS_EVENT_CAC_START
  855. "freq=%d chan=%d chan_offset=%d width=%d seg0=%d "
  856. "seg1=%d cac_time=%ds",
  857. freq, (freq - 5000) / 5, chan_offset, chan_width, cf1, cf2, 60);
  858. iface->cac_started = 1;
  859. return 0;
  860. }
  861. /*
  862. * Main DFS handler for offloaded case.
  863. * 2 - continue channel/AP setup for non-DFS channel
  864. * 1 - continue channel/AP setup for DFS channel
  865. * 0 - channel/AP setup will be continued after CAC
  866. * -1 - hit critical error
  867. */
  868. int hostapd_handle_dfs_offload(struct hostapd_iface *iface)
  869. {
  870. wpa_printf(MSG_DEBUG, "%s: iface->cac_started: %d",
  871. __func__, iface->cac_started);
  872. /*
  873. * If DFS has already been started, then we are being called from a
  874. * callback to continue AP/channel setup. Reset the CAC start flag and
  875. * return.
  876. */
  877. if (iface->cac_started) {
  878. wpa_printf(MSG_DEBUG, "%s: iface->cac_started: %d",
  879. __func__, iface->cac_started);
  880. iface->cac_started = 0;
  881. return 1;
  882. }
  883. if (ieee80211_is_dfs(iface->freq)) {
  884. wpa_printf(MSG_DEBUG, "%s: freq %d MHz requires DFS",
  885. __func__, iface->freq);
  886. return 0;
  887. }
  888. wpa_printf(MSG_DEBUG,
  889. "%s: freq %d MHz does not require DFS. Continue channel/AP setup",
  890. __func__, iface->freq);
  891. return 2;
  892. }