DFS: Sanitize channel availability checks
Fixes corner case of holes in channel list and simplifies availability checks. Signed-hostap: Michal Kazior <michal.kazior@tieto.com>
This commit is contained in:
parent
32595da608
commit
6a398ddc9a
1 changed files with 34 additions and 18 deletions
52
src/ap/dfs.c
52
src/ap/dfs.c
|
@ -94,6 +94,31 @@ static int dfs_is_chan_allowed(struct hostapd_channel_data *chan, int n_chans)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
static int dfs_chan_range_available(struct hostapd_hw_modes *mode,
|
||||||
|
int first_chan_idx, int num_chans)
|
||||||
|
{
|
||||||
|
struct hostapd_channel_data *first_chan, *chan;
|
||||||
|
int i;
|
||||||
|
|
||||||
|
if (first_chan_idx + num_chans >= mode->num_channels)
|
||||||
|
return 0;
|
||||||
|
|
||||||
|
first_chan = &mode->channels[first_chan_idx];
|
||||||
|
|
||||||
|
for (i = 0; i < num_chans; i++) {
|
||||||
|
chan = &mode->channels[first_chan_idx + i];
|
||||||
|
|
||||||
|
if (first_chan->freq + i * 20 != chan->freq)
|
||||||
|
return 0;
|
||||||
|
|
||||||
|
if (!dfs_channel_available(chan))
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
return 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* The function assumes HT40+ operation.
|
* The function assumes HT40+ operation.
|
||||||
* Make sure to adjust the following variables after calling this:
|
* Make sure to adjust the following variables after calling this:
|
||||||
|
@ -106,8 +131,8 @@ static int dfs_find_channel(struct hostapd_data *hapd,
|
||||||
int idx)
|
int idx)
|
||||||
{
|
{
|
||||||
struct hostapd_hw_modes *mode;
|
struct hostapd_hw_modes *mode;
|
||||||
struct hostapd_channel_data *chan, *next_chan;
|
struct hostapd_channel_data *chan;
|
||||||
int i, j, channel_idx = 0, n_chans;
|
int i, channel_idx = 0, n_chans;
|
||||||
|
|
||||||
mode = hapd->iface->current_mode;
|
mode = hapd->iface->current_mode;
|
||||||
n_chans = dfs_get_used_n_chans(hapd);
|
n_chans = dfs_get_used_n_chans(hapd);
|
||||||
|
@ -116,24 +141,15 @@ static int dfs_find_channel(struct hostapd_data *hapd,
|
||||||
for (i = 0; i < mode->num_channels; i++) {
|
for (i = 0; i < mode->num_channels; i++) {
|
||||||
chan = &mode->channels[i];
|
chan = &mode->channels[i];
|
||||||
|
|
||||||
/* Skip not available channels */
|
/* Skip HT40/VHT incompatible channels */
|
||||||
if (!dfs_channel_available(chan))
|
if (hapd->iconf->ieee80211n &&
|
||||||
|
hapd->iconf->secondary_channel &&
|
||||||
|
!dfs_is_chan_allowed(chan, n_chans))
|
||||||
continue;
|
continue;
|
||||||
|
|
||||||
/* Skip HT40/VHT uncompatible channels */
|
/* Skip incompatible chandefs */
|
||||||
if (hapd->iconf->ieee80211n &&
|
if (!dfs_chan_range_available(mode, i, n_chans))
|
||||||
hapd->iconf->secondary_channel) {
|
continue;
|
||||||
if (!dfs_is_chan_allowed(chan, n_chans))
|
|
||||||
continue;
|
|
||||||
|
|
||||||
for (j = 1; j < n_chans; j++) {
|
|
||||||
next_chan = &mode->channels[i + j];
|
|
||||||
if (!dfs_channel_available(next_chan))
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
if (j != n_chans)
|
|
||||||
continue;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (ret_chan && idx == channel_idx) {
|
if (ret_chan && idx == channel_idx) {
|
||||||
wpa_printf(MSG_DEBUG, "Selected ch. #%d", chan->chan);
|
wpa_printf(MSG_DEBUG, "Selected ch. #%d", chan->chan);
|
||||||
|
|
Loading…
Reference in a new issue