dfs.c 29 KB

12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788899091929394959697989910010110210310410510610710810911011111211311411511611711811912012112212312412512612712812913013113213313413513613713813914014114214314414514614714814915015115215315415515615715815916016116216316416516616716816917017117217317417517617717817918018118218318418518618718818919019119219319419519619719819920020120220320420520620720820921021121221321421521621721821922022122222322422522622722822923023123223323423523623723823924024124224324424524624724824925025125225325425525625725825926026126226326426526626726826927027127227327427527627727827928028128228328428528628728828929029129229329429529629729829930030130230330430530630730830931031131231331431531631731831932032132232332432532632732832933033133233333433533633733833934034134234334434534634734834935035135235335435535635735835936036136236336436536636736836937037137237337437537637737837938038138238338438538638738838939039139239339439539639739839940040140240340440540640740840941041141241341441541641741841942042142242342442542642742842943043143243343443543643743843944044144244344444544644744844945045145245345445545645745845946046146246346446546646746846947047147247347447547647747847948048148248348448548648748848949049149249349449549649749849950050150250350450550650750850951051151251351451551651751851952052152252352452552652752852953053153253353453553653753853954054154254354454554654754854955055155255355455555655755855956056156256356456556656756856957057157257357457557657757857958058158258358458558658758858959059159259359459559659759859960060160260360460560660760860961061161261361461561661761861962062162262362462562662762862963063163263363463563663763863964064164264364464564664764864965065165265365465565665765865966066166266366466566666766866967067167267367467567667767867968068168268368468568668768868969069169269369469569669769869970070170270370470570670770870971071171271371471571671771871972072172272372472572672772872973073173273373473573673773873974074174274374474574674774874975075175275375475575675775875976076176276376476576676776876977077177277377477577677777877978078178278378478578678778878979079179279379479579679779879980080180280380480580680780880981081181281381481581681781881982082182282382482582682782882983083183283383483583683783883984084184284384484584684784884985085185285385485585685785885986086186286386486586686786886987087187287387487587687787887988088188288388488588688788888989089189289389489589689789889990090190290390490590690790890991091191291391491591691791891992092192292392492592692792892993093193293393493593693793893994094194294394494594694794894995095195295395495595695795895996096196296396496596696796896997097197297397497597697797897998098198298398498598698798898999099199299399499599699799899910001001100210031004100510061007100810091010101110121013101410151016101710181019102010211022102310241025102610271028102910301031103210331034103510361037103810391040104110421043104410451046104710481049105010511052105310541055105610571058105910601061106210631064106510661067106810691070107110721073107410751076107710781079108010811082108310841085108610871088108910901091109210931094109510961097109810991100110111021103110411051106110711081109111011111112111311141115111611171118111911201121112211231124
  1. /*
  2. * DFS - Dynamic Frequency Selection
  3. * Copyright (c) 2002-2013, Jouni Malinen <j@w1.fi>
  4. * Copyright (c) 2013-2017, 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. return NULL;
  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. hostapd_set_state(iface, HAPD_IFACE_DFS);
  589. return 0;
  590. }
  591. iface->freq = channel->freq;
  592. iface->conf->channel = channel->chan;
  593. iface->conf->secondary_channel = sec;
  594. iface->conf->vht_oper_centr_freq_seg0_idx = cf1;
  595. iface->conf->vht_oper_centr_freq_seg1_idx = cf2;
  596. }
  597. } while (res);
  598. /* Finally start CAC */
  599. hostapd_set_state(iface, HAPD_IFACE_DFS);
  600. wpa_printf(MSG_DEBUG, "DFS start CAC on %d MHz", iface->freq);
  601. wpa_msg(iface->bss[0]->msg_ctx, MSG_INFO, DFS_EVENT_CAC_START
  602. "freq=%d chan=%d sec_chan=%d, width=%d, seg0=%d, seg1=%d, cac_time=%ds",
  603. iface->freq,
  604. iface->conf->channel, iface->conf->secondary_channel,
  605. iface->conf->vht_oper_chwidth,
  606. iface->conf->vht_oper_centr_freq_seg0_idx,
  607. iface->conf->vht_oper_centr_freq_seg1_idx,
  608. iface->dfs_cac_ms / 1000);
  609. res = hostapd_start_dfs_cac(iface, iface->conf->hw_mode,
  610. iface->freq,
  611. iface->conf->channel,
  612. iface->conf->ieee80211n,
  613. iface->conf->ieee80211ac,
  614. iface->conf->secondary_channel,
  615. iface->conf->vht_oper_chwidth,
  616. iface->conf->vht_oper_centr_freq_seg0_idx,
  617. iface->conf->vht_oper_centr_freq_seg1_idx);
  618. if (res) {
  619. wpa_printf(MSG_ERROR, "DFS start_dfs_cac() failed, %d", res);
  620. return -1;
  621. }
  622. return 0;
  623. }
  624. static int hostapd_config_dfs_chan_available(struct hostapd_iface *iface)
  625. {
  626. int n_chans, n_chans1, start_chan_idx, start_chan_idx1;
  627. /* Get the start (first) channel for current configuration */
  628. start_chan_idx = dfs_get_start_chan_idx(iface, &start_chan_idx1);
  629. if (start_chan_idx < 0)
  630. return 0;
  631. /* Get the number of used channels, depending on width */
  632. n_chans = dfs_get_used_n_chans(iface, &n_chans1);
  633. /* Check if all channels are DFS available */
  634. return dfs_check_chans_available(iface, start_chan_idx, n_chans);
  635. }
  636. int hostapd_dfs_complete_cac(struct hostapd_iface *iface, int success, int freq,
  637. int ht_enabled, int chan_offset, int chan_width,
  638. int cf1, int cf2)
  639. {
  640. wpa_msg(iface->bss[0]->msg_ctx, MSG_INFO, DFS_EVENT_CAC_COMPLETED
  641. "success=%d freq=%d ht_enabled=%d chan_offset=%d chan_width=%d cf1=%d cf2=%d",
  642. success, freq, ht_enabled, chan_offset, chan_width, cf1, cf2);
  643. if (success) {
  644. /* Complete iface/ap configuration */
  645. if (iface->drv_flags & WPA_DRIVER_FLAGS_DFS_OFFLOAD) {
  646. /* Complete AP configuration for the first bring up. */
  647. if (iface->state != HAPD_IFACE_ENABLED)
  648. hostapd_setup_interface_complete(iface, 0);
  649. else
  650. iface->cac_started = 0;
  651. } else {
  652. set_dfs_state(iface, freq, ht_enabled, chan_offset,
  653. chan_width, cf1, cf2,
  654. HOSTAPD_CHAN_DFS_AVAILABLE);
  655. /*
  656. * Just mark the channel available when CAC completion
  657. * event is received in enabled state. CAC result could
  658. * have been propagated from another radio having the
  659. * same regulatory configuration. When CAC completion is
  660. * received during non-HAPD_IFACE_ENABLED state, make
  661. * sure the configured channel is available because this
  662. * CAC completion event could have been propagated from
  663. * another radio.
  664. */
  665. if (iface->state != HAPD_IFACE_ENABLED &&
  666. hostapd_config_dfs_chan_available(iface)) {
  667. hostapd_setup_interface_complete(iface, 0);
  668. iface->cac_started = 0;
  669. }
  670. }
  671. }
  672. return 0;
  673. }
  674. int hostapd_dfs_pre_cac_expired(struct hostapd_iface *iface, int freq,
  675. int ht_enabled, int chan_offset, int chan_width,
  676. int cf1, int cf2)
  677. {
  678. wpa_msg(iface->bss[0]->msg_ctx, MSG_INFO, DFS_EVENT_PRE_CAC_EXPIRED
  679. "freq=%d ht_enabled=%d chan_offset=%d chan_width=%d cf1=%d cf2=%d",
  680. freq, ht_enabled, chan_offset, chan_width, cf1, cf2);
  681. /* Proceed only if DFS is not offloaded to the driver */
  682. if (iface->drv_flags & WPA_DRIVER_FLAGS_DFS_OFFLOAD)
  683. return 0;
  684. set_dfs_state(iface, freq, ht_enabled, chan_offset, chan_width,
  685. cf1, cf2, HOSTAPD_CHAN_DFS_USABLE);
  686. return 0;
  687. }
  688. static int hostapd_dfs_start_channel_switch_cac(struct hostapd_iface *iface)
  689. {
  690. struct hostapd_channel_data *channel;
  691. int secondary_channel;
  692. u8 vht_oper_centr_freq_seg0_idx = 0;
  693. u8 vht_oper_centr_freq_seg1_idx = 0;
  694. int skip_radar = 0;
  695. int err = 1;
  696. /* Radar detected during active CAC */
  697. iface->cac_started = 0;
  698. channel = dfs_get_valid_channel(iface, &secondary_channel,
  699. &vht_oper_centr_freq_seg0_idx,
  700. &vht_oper_centr_freq_seg1_idx,
  701. skip_radar);
  702. if (!channel) {
  703. wpa_printf(MSG_ERROR, "No valid channel available");
  704. return err;
  705. }
  706. wpa_printf(MSG_DEBUG, "DFS will switch to a new channel %d",
  707. channel->chan);
  708. wpa_msg(iface->bss[0]->msg_ctx, MSG_INFO, DFS_EVENT_NEW_CHANNEL
  709. "freq=%d chan=%d sec_chan=%d", channel->freq,
  710. channel->chan, secondary_channel);
  711. iface->freq = channel->freq;
  712. iface->conf->channel = channel->chan;
  713. iface->conf->secondary_channel = secondary_channel;
  714. iface->conf->vht_oper_centr_freq_seg0_idx =
  715. vht_oper_centr_freq_seg0_idx;
  716. iface->conf->vht_oper_centr_freq_seg1_idx =
  717. vht_oper_centr_freq_seg1_idx;
  718. err = 0;
  719. hostapd_setup_interface_complete(iface, err);
  720. return err;
  721. }
  722. static int hostapd_dfs_start_channel_switch(struct hostapd_iface *iface)
  723. {
  724. struct hostapd_channel_data *channel;
  725. int secondary_channel;
  726. u8 vht_oper_centr_freq_seg0_idx;
  727. u8 vht_oper_centr_freq_seg1_idx;
  728. int skip_radar = 1;
  729. struct csa_settings csa_settings;
  730. unsigned int i;
  731. int err = 1;
  732. wpa_printf(MSG_DEBUG, "%s called (CAC active: %s, CSA active: %s)",
  733. __func__, iface->cac_started ? "yes" : "no",
  734. hostapd_csa_in_progress(iface) ? "yes" : "no");
  735. /* Check if CSA in progress */
  736. if (hostapd_csa_in_progress(iface))
  737. return 0;
  738. /* Check if active CAC */
  739. if (iface->cac_started)
  740. return hostapd_dfs_start_channel_switch_cac(iface);
  741. /*
  742. * Allow selection of DFS channel in ETSI to comply with
  743. * uniform spreading.
  744. */
  745. if (iface->dfs_domain == HOSTAPD_DFS_REGION_ETSI)
  746. skip_radar = 0;
  747. /* Perform channel switch/CSA */
  748. channel = dfs_get_valid_channel(iface, &secondary_channel,
  749. &vht_oper_centr_freq_seg0_idx,
  750. &vht_oper_centr_freq_seg1_idx,
  751. skip_radar);
  752. if (!channel) {
  753. /*
  754. * If there is no channel to switch immediately to, check if
  755. * there is another channel where we can switch even if it
  756. * requires to perform a CAC first.
  757. */
  758. skip_radar = 0;
  759. channel = dfs_get_valid_channel(iface, &secondary_channel,
  760. &vht_oper_centr_freq_seg0_idx,
  761. &vht_oper_centr_freq_seg1_idx,
  762. skip_radar);
  763. if (!channel) {
  764. wpa_printf(MSG_INFO,
  765. "%s: no DFS channels left, waiting for NOP to finish",
  766. __func__);
  767. return err;
  768. }
  769. iface->freq = channel->freq;
  770. iface->conf->channel = channel->chan;
  771. iface->conf->secondary_channel = secondary_channel;
  772. iface->conf->vht_oper_centr_freq_seg0_idx =
  773. vht_oper_centr_freq_seg0_idx;
  774. iface->conf->vht_oper_centr_freq_seg1_idx =
  775. vht_oper_centr_freq_seg1_idx;
  776. hostapd_disable_iface(iface);
  777. hostapd_enable_iface(iface);
  778. return 0;
  779. }
  780. wpa_printf(MSG_DEBUG, "DFS will switch to a new channel %d",
  781. channel->chan);
  782. wpa_msg(iface->bss[0]->msg_ctx, MSG_INFO, DFS_EVENT_NEW_CHANNEL
  783. "freq=%d chan=%d sec_chan=%d", channel->freq,
  784. channel->chan, secondary_channel);
  785. /* Setup CSA request */
  786. os_memset(&csa_settings, 0, sizeof(csa_settings));
  787. csa_settings.cs_count = 5;
  788. csa_settings.block_tx = 1;
  789. err = hostapd_set_freq_params(&csa_settings.freq_params,
  790. iface->conf->hw_mode,
  791. channel->freq,
  792. channel->chan,
  793. iface->conf->ieee80211n,
  794. iface->conf->ieee80211ac,
  795. secondary_channel,
  796. iface->conf->vht_oper_chwidth,
  797. vht_oper_centr_freq_seg0_idx,
  798. vht_oper_centr_freq_seg1_idx,
  799. iface->current_mode->vht_capab);
  800. if (err) {
  801. wpa_printf(MSG_ERROR, "DFS failed to calculate CSA freq params");
  802. hostapd_disable_iface(iface);
  803. return err;
  804. }
  805. for (i = 0; i < iface->num_bss; i++) {
  806. err = hostapd_switch_channel(iface->bss[i], &csa_settings);
  807. if (err)
  808. break;
  809. }
  810. if (err) {
  811. wpa_printf(MSG_WARNING, "DFS failed to schedule CSA (%d) - trying fallback",
  812. err);
  813. iface->freq = channel->freq;
  814. iface->conf->channel = channel->chan;
  815. iface->conf->secondary_channel = secondary_channel;
  816. iface->conf->vht_oper_centr_freq_seg0_idx =
  817. vht_oper_centr_freq_seg0_idx;
  818. iface->conf->vht_oper_centr_freq_seg1_idx =
  819. vht_oper_centr_freq_seg1_idx;
  820. hostapd_disable_iface(iface);
  821. hostapd_enable_iface(iface);
  822. return 0;
  823. }
  824. /* Channel configuration will be updated once CSA completes and
  825. * ch_switch_notify event is received */
  826. wpa_printf(MSG_DEBUG, "DFS waiting channel switch event");
  827. return 0;
  828. }
  829. int hostapd_dfs_radar_detected(struct hostapd_iface *iface, int freq,
  830. int ht_enabled, int chan_offset, int chan_width,
  831. int cf1, int cf2)
  832. {
  833. int res;
  834. wpa_msg(iface->bss[0]->msg_ctx, MSG_INFO, DFS_EVENT_RADAR_DETECTED
  835. "freq=%d ht_enabled=%d chan_offset=%d chan_width=%d cf1=%d cf2=%d",
  836. freq, ht_enabled, chan_offset, chan_width, cf1, cf2);
  837. /* Proceed only if DFS is not offloaded to the driver */
  838. if (iface->drv_flags & WPA_DRIVER_FLAGS_DFS_OFFLOAD)
  839. return 0;
  840. if (!iface->conf->ieee80211h)
  841. return 0;
  842. /* mark radar frequency as invalid */
  843. set_dfs_state(iface, freq, ht_enabled, chan_offset, chan_width,
  844. cf1, cf2, HOSTAPD_CHAN_DFS_UNAVAILABLE);
  845. /* Skip if reported radar event not overlapped our channels */
  846. res = dfs_are_channels_overlapped(iface, freq, chan_width, cf1, cf2);
  847. if (!res)
  848. return 0;
  849. /* radar detected while operating, switch the channel. */
  850. res = hostapd_dfs_start_channel_switch(iface);
  851. return res;
  852. }
  853. int hostapd_dfs_nop_finished(struct hostapd_iface *iface, int freq,
  854. int ht_enabled, int chan_offset, int chan_width,
  855. int cf1, int cf2)
  856. {
  857. wpa_msg(iface->bss[0]->msg_ctx, MSG_INFO, DFS_EVENT_NOP_FINISHED
  858. "freq=%d ht_enabled=%d chan_offset=%d chan_width=%d cf1=%d cf2=%d",
  859. freq, ht_enabled, chan_offset, chan_width, cf1, cf2);
  860. /* Proceed only if DFS is not offloaded to the driver */
  861. if (iface->drv_flags & WPA_DRIVER_FLAGS_DFS_OFFLOAD)
  862. return 0;
  863. /* TODO add correct implementation here */
  864. set_dfs_state(iface, freq, ht_enabled, chan_offset, chan_width,
  865. cf1, cf2, HOSTAPD_CHAN_DFS_USABLE);
  866. /* Handle cases where all channels were initially unavailable */
  867. if (iface->state == HAPD_IFACE_DFS && !iface->cac_started)
  868. hostapd_handle_dfs(iface);
  869. return 0;
  870. }
  871. int hostapd_is_dfs_required(struct hostapd_iface *iface)
  872. {
  873. int n_chans, n_chans1, start_chan_idx, start_chan_idx1, res;
  874. if (!iface->conf->ieee80211h || !iface->current_mode ||
  875. iface->current_mode->mode != HOSTAPD_MODE_IEEE80211A)
  876. return 0;
  877. /* Get start (first) channel for current configuration */
  878. start_chan_idx = dfs_get_start_chan_idx(iface, &start_chan_idx1);
  879. if (start_chan_idx == -1)
  880. return -1;
  881. /* Get number of used channels, depend on width */
  882. n_chans = dfs_get_used_n_chans(iface, &n_chans1);
  883. /* Check if any of configured channels require DFS */
  884. res = dfs_check_chans_radar(iface, start_chan_idx, n_chans);
  885. if (res)
  886. return res;
  887. if (start_chan_idx1 >= 0 && n_chans1 > 0)
  888. res = dfs_check_chans_radar(iface, start_chan_idx1, n_chans1);
  889. return res;
  890. }
  891. int hostapd_dfs_start_cac(struct hostapd_iface *iface, int freq,
  892. int ht_enabled, int chan_offset, int chan_width,
  893. int cf1, int cf2)
  894. {
  895. wpa_msg(iface->bss[0]->msg_ctx, MSG_INFO, DFS_EVENT_CAC_START
  896. "freq=%d chan=%d chan_offset=%d width=%d seg0=%d "
  897. "seg1=%d cac_time=%ds",
  898. freq, (freq - 5000) / 5, chan_offset, chan_width, cf1, cf2, 60);
  899. iface->cac_started = 1;
  900. return 0;
  901. }
  902. /*
  903. * Main DFS handler for offloaded case.
  904. * 2 - continue channel/AP setup for non-DFS channel
  905. * 1 - continue channel/AP setup for DFS channel
  906. * 0 - channel/AP setup will be continued after CAC
  907. * -1 - hit critical error
  908. */
  909. int hostapd_handle_dfs_offload(struct hostapd_iface *iface)
  910. {
  911. wpa_printf(MSG_DEBUG, "%s: iface->cac_started: %d",
  912. __func__, iface->cac_started);
  913. /*
  914. * If DFS has already been started, then we are being called from a
  915. * callback to continue AP/channel setup. Reset the CAC start flag and
  916. * return.
  917. */
  918. if (iface->cac_started) {
  919. wpa_printf(MSG_DEBUG, "%s: iface->cac_started: %d",
  920. __func__, iface->cac_started);
  921. iface->cac_started = 0;
  922. return 1;
  923. }
  924. if (ieee80211_is_dfs(iface->freq)) {
  925. wpa_printf(MSG_DEBUG, "%s: freq %d MHz requires DFS",
  926. __func__, iface->freq);
  927. return 0;
  928. }
  929. wpa_printf(MSG_DEBUG,
  930. "%s: freq %d MHz does not require DFS. Continue channel/AP setup",
  931. __func__, iface->freq);
  932. return 2;
  933. }