Use frequency in HT/VHT validation steps done before starting AP
Using the channel parameter for validating allowed channel combinations is not scalable to add 6 GHz support in the future since channel numbers are duplicated between 2.4 GHz / 5 GHz bands and 6 GHz band. Hence use frequency field for all channel combination validation steps done before starting AP. Signed-off-by: Jouni Malinen <jouni@codeaurora.org>
This commit is contained in:
		
							parent
							
								
									59e33b4a98
								
							
						
					
					
						commit
						5f9b4afdfa
					
				
					 5 changed files with 106 additions and 70 deletions
				
			
		|  | @ -1915,7 +1915,7 @@ static int hostapd_setup_interface_complete_sync(struct hostapd_iface *iface, | |||
| 		goto fail; | ||||
| 
 | ||||
| 	wpa_printf(MSG_DEBUG, "Completing interface initialization"); | ||||
| 	if (iface->conf->channel) { | ||||
| 	if (iface->freq) { | ||||
| #ifdef NEED_AP_MLME | ||||
| 		int res; | ||||
| #endif /* NEED_AP_MLME */ | ||||
|  |  | |||
|  | @ -227,13 +227,25 @@ int hostapd_prepare_rates(struct hostapd_iface *iface, | |||
| #ifdef CONFIG_IEEE80211N | ||||
| static int ieee80211n_allowed_ht40_channel_pair(struct hostapd_iface *iface) | ||||
| { | ||||
| 	int pri_chan, sec_chan; | ||||
| 	int pri_freq, sec_freq; | ||||
| 	struct hostapd_channel_data *p_chan, *s_chan; | ||||
| 
 | ||||
| 	pri_chan = iface->conf->channel; | ||||
| 	sec_chan = pri_chan + iface->conf->secondary_channel * 4; | ||||
| 	pri_freq = iface->freq; | ||||
| 	sec_freq = pri_freq + iface->conf->secondary_channel * 20; | ||||
| 
 | ||||
| 	return allowed_ht40_channel_pair(iface->current_mode, pri_chan, | ||||
| 					 sec_chan); | ||||
| 	if (!iface->current_mode) | ||||
| 		return 0; | ||||
| 
 | ||||
| 	p_chan = hw_get_channel_freq(iface->current_mode->mode, pri_freq, NULL, | ||||
| 				     iface->hw_features, | ||||
| 				     iface->num_hw_features); | ||||
| 
 | ||||
| 	s_chan = hw_get_channel_freq(iface->current_mode->mode, sec_freq, NULL, | ||||
| 				     iface->hw_features, | ||||
| 				     iface->num_hw_features); | ||||
| 
 | ||||
| 	return allowed_ht40_channel_pair(iface->current_mode->mode, | ||||
| 					 p_chan, s_chan); | ||||
| } | ||||
| 
 | ||||
| 
 | ||||
|  | @ -241,9 +253,11 @@ static void ieee80211n_switch_pri_sec(struct hostapd_iface *iface) | |||
| { | ||||
| 	if (iface->conf->secondary_channel > 0) { | ||||
| 		iface->conf->channel += 4; | ||||
| 		iface->freq += 20; | ||||
| 		iface->conf->secondary_channel = -1; | ||||
| 	} else { | ||||
| 		iface->conf->channel -= 4; | ||||
| 		iface->freq -= 20; | ||||
| 		iface->conf->secondary_channel = 1; | ||||
| 	} | ||||
| } | ||||
|  | @ -252,13 +266,23 @@ static void ieee80211n_switch_pri_sec(struct hostapd_iface *iface) | |||
| static int ieee80211n_check_40mhz_5g(struct hostapd_iface *iface, | ||||
| 				     struct wpa_scan_results *scan_res) | ||||
| { | ||||
| 	int pri_chan, sec_chan; | ||||
| 	unsigned int pri_freq, sec_freq; | ||||
| 	int res; | ||||
| 	struct hostapd_channel_data *pri_chan, *sec_chan; | ||||
| 
 | ||||
| 	pri_chan = iface->conf->channel; | ||||
| 	sec_chan = pri_chan + iface->conf->secondary_channel * 4; | ||||
| 	pri_freq = iface->freq; | ||||
| 	sec_freq = pri_freq + iface->conf->secondary_channel * 20; | ||||
| 
 | ||||
| 	res = check_40mhz_5g(iface->current_mode, scan_res, pri_chan, sec_chan); | ||||
| 	if (!iface->current_mode) | ||||
| 		return 0; | ||||
| 	pri_chan = hw_get_channel_freq(iface->current_mode->mode, pri_freq, | ||||
| 				       NULL, iface->hw_features, | ||||
| 				       iface->num_hw_features); | ||||
| 	sec_chan = hw_get_channel_freq(iface->current_mode->mode, sec_freq, | ||||
| 				       NULL, iface->hw_features, | ||||
| 				       iface->num_hw_features); | ||||
| 
 | ||||
| 	res = check_40mhz_5g(scan_res, pri_chan, sec_chan); | ||||
| 
 | ||||
| 	if (res == 2) { | ||||
| 		if (iface->conf->no_pri_sec_switch) { | ||||
|  | @ -352,7 +376,7 @@ static void ieee80211n_scan_channels_2g4(struct hostapd_iface *iface, | |||
| 	if (iface->current_mode == NULL) | ||||
| 		return; | ||||
| 
 | ||||
| 	pri_freq = hostapd_hw_get_freq(iface->bss[0], iface->conf->channel); | ||||
| 	pri_freq = iface->freq; | ||||
| 	if (iface->conf->secondary_channel > 0) | ||||
| 		sec_freq = pri_freq + 20; | ||||
| 	else | ||||
|  | @ -397,7 +421,7 @@ static void ieee80211n_scan_channels_5g(struct hostapd_iface *iface, | |||
| 	if (iface->current_mode == NULL) | ||||
| 		return; | ||||
| 
 | ||||
| 	pri_freq = hostapd_hw_get_freq(iface->bss[0], iface->conf->channel); | ||||
| 	pri_freq = iface->freq; | ||||
| 	if (iface->conf->secondary_channel > 0) { | ||||
| 		affected_start = pri_freq - 10; | ||||
| 		affected_end = pri_freq + 30; | ||||
|  | @ -734,14 +758,15 @@ int hostapd_check_edmg_capab(struct hostapd_iface *iface) | |||
| 
 | ||||
| 
 | ||||
| static int hostapd_is_usable_chan(struct hostapd_iface *iface, | ||||
| 				  int channel, int primary) | ||||
| 				  int frequency, int primary) | ||||
| { | ||||
| 	struct hostapd_channel_data *chan; | ||||
| 
 | ||||
| 	if (!iface->current_mode) | ||||
| 		return 0; | ||||
| 
 | ||||
| 	chan = hw_get_channel_chan(iface->current_mode, channel, NULL); | ||||
| 	chan = hw_get_channel_freq(iface->current_mode->mode, frequency, NULL, | ||||
| 				   iface->hw_features, iface->num_hw_features); | ||||
| 	if (!chan) | ||||
| 		return 0; | ||||
| 
 | ||||
|  | @ -750,8 +775,8 @@ static int hostapd_is_usable_chan(struct hostapd_iface *iface, | |||
| 		return 1; | ||||
| 
 | ||||
| 	wpa_printf(MSG_INFO, | ||||
| 		   "Channel %d (%s) not allowed for AP mode, flags: 0x%x%s%s", | ||||
| 		   channel, primary ? "primary" : "secondary", | ||||
| 		   "Frequency %d (%s) not allowed for AP mode, flags: 0x%x%s%s", | ||||
| 		   frequency, primary ? "primary" : "secondary", | ||||
| 		   chan->flag, | ||||
| 		   chan->flag & HOSTAPD_CHAN_NO_IR ? " NO-IR" : "", | ||||
| 		   chan->flag & HOSTAPD_CHAN_RADAR ? " RADAR" : ""); | ||||
|  | @ -765,19 +790,28 @@ static int hostapd_is_usable_edmg(struct hostapd_iface *iface) | |||
| 	int num_of_enabled = 0; | ||||
| 	int max_contiguous = 0; | ||||
| 	struct ieee80211_edmg_config edmg; | ||||
| 	struct hostapd_channel_data *pri_chan; | ||||
| 
 | ||||
| 	if (!iface->conf->enable_edmg) | ||||
| 		return 1; | ||||
| 
 | ||||
| 	if (!iface->current_mode) | ||||
| 		return 0; | ||||
| 	pri_chan = hw_get_channel_freq(iface->current_mode->mode, | ||||
| 				       iface->freq, NULL, | ||||
| 				       iface->hw_features, | ||||
| 				       iface->num_hw_features); | ||||
| 	hostapd_encode_edmg_chan(iface->conf->enable_edmg, | ||||
| 				 iface->conf->edmg_channel, | ||||
| 				 iface->conf->channel, | ||||
| 				 pri_chan->chan, | ||||
| 				 &edmg); | ||||
| 	if (!(edmg.channels & BIT(iface->conf->channel - 1))) | ||||
| 	if (!(edmg.channels & BIT(pri_chan->chan - 1))) | ||||
| 		return 0; | ||||
| 
 | ||||
| 	/* 60 GHz channels 1..6 */ | ||||
| 	for (i = 0; i < 6; i++) { | ||||
| 		int freq = 56160 + 2160 * i; | ||||
| 
 | ||||
| 		if (edmg.channels & BIT(i)) { | ||||
| 			contiguous++; | ||||
| 			num_of_enabled++; | ||||
|  | @ -792,7 +826,7 @@ static int hostapd_is_usable_edmg(struct hostapd_iface *iface) | |||
| 		if (num_of_enabled > 4) | ||||
| 			return 0; | ||||
| 
 | ||||
| 		if (!hostapd_is_usable_chan(iface, i + 1, 1)) | ||||
| 		if (!hostapd_is_usable_chan(iface, freq, 1)) | ||||
| 			return 0; | ||||
| 
 | ||||
| 		if (contiguous > max_contiguous) | ||||
|  | @ -822,17 +856,23 @@ static int hostapd_is_usable_edmg(struct hostapd_iface *iface) | |||
| 
 | ||||
| static int hostapd_is_usable_chans(struct hostapd_iface *iface) | ||||
| { | ||||
| 	int secondary_chan; | ||||
| 	int secondary_freq; | ||||
| 	struct hostapd_channel_data *pri_chan; | ||||
| 
 | ||||
| 	pri_chan = hw_get_channel_chan(iface->current_mode, | ||||
| 				       iface->conf->channel, NULL); | ||||
| 	if (!pri_chan) | ||||
| 	if (!iface->current_mode) | ||||
| 		return 0; | ||||
| 
 | ||||
| 	if (!hostapd_is_usable_chan(iface, iface->conf->channel, 1)) | ||||
| 	pri_chan = hw_get_channel_freq(iface->current_mode->mode, | ||||
| 				       iface->freq, NULL, | ||||
| 				       iface->hw_features, | ||||
| 				       iface->num_hw_features); | ||||
| 	if (!pri_chan) { | ||||
| 		wpa_printf(MSG_ERROR, "Primary frequency not present"); | ||||
| 		return 0; | ||||
| 
 | ||||
| 	} | ||||
| 	if (!hostapd_is_usable_chan(iface, pri_chan->freq, 1)) { | ||||
| 		wpa_printf(MSG_ERROR, "Primary frequency not allowed"); | ||||
| 		return 0; | ||||
| 	} | ||||
| 	if (!hostapd_is_usable_edmg(iface)) | ||||
| 		return 0; | ||||
| 
 | ||||
|  | @ -841,19 +881,19 @@ static int hostapd_is_usable_chans(struct hostapd_iface *iface) | |||
| 
 | ||||
| 	if (!iface->conf->ht40_plus_minus_allowed) | ||||
| 		return hostapd_is_usable_chan( | ||||
| 			iface, iface->conf->channel + | ||||
| 			iface->conf->secondary_channel * 4, 0); | ||||
| 			iface, | ||||
| 			iface->freq + iface->conf->secondary_channel * 20, 0); | ||||
| 
 | ||||
| 	/* Both HT40+ and HT40- are set, pick a valid secondary channel */ | ||||
| 	secondary_chan = iface->conf->channel + 4; | ||||
| 	if (hostapd_is_usable_chan(iface, secondary_chan, 0) && | ||||
| 	secondary_freq = iface->freq + 20; | ||||
| 	if (hostapd_is_usable_chan(iface, secondary_freq, 0) && | ||||
| 	    (pri_chan->allowed_bw & HOSTAPD_CHAN_WIDTH_40P)) { | ||||
| 		iface->conf->secondary_channel = 1; | ||||
| 		return 1; | ||||
| 	} | ||||
| 
 | ||||
| 	secondary_chan = iface->conf->channel - 4; | ||||
| 	if (hostapd_is_usable_chan(iface, secondary_chan, 0) && | ||||
| 	secondary_freq = iface->freq - 20; | ||||
| 	if (hostapd_is_usable_chan(iface, secondary_freq, 0) && | ||||
| 	    (pri_chan->allowed_bw & HOSTAPD_CHAN_WIDTH_40M)) { | ||||
| 		iface->conf->secondary_channel = -1; | ||||
| 		return 1; | ||||
|  | @ -866,7 +906,7 @@ static int hostapd_is_usable_chans(struct hostapd_iface *iface) | |||
| static enum hostapd_chan_status | ||||
| hostapd_check_chans(struct hostapd_iface *iface) | ||||
| { | ||||
| 	if (iface->conf->channel) { | ||||
| 	if (iface->freq) { | ||||
| 		if (hostapd_is_usable_chans(iface)) | ||||
| 			return HOSTAPD_CHAN_VALID; | ||||
| 		else | ||||
|  | @ -900,9 +940,9 @@ static void hostapd_notify_bad_chans(struct hostapd_iface *iface) | |||
| 	hostapd_logger(iface->bss[0], NULL, | ||||
| 		       HOSTAPD_MODULE_IEEE80211, | ||||
| 		       HOSTAPD_LEVEL_WARNING, | ||||
| 		       "Configured channel (%d) not found from the " | ||||
| 		       "channel list of current mode (%d) %s", | ||||
| 		       "Configured channel (%d) or frequency (%d) not found from the channel list of the current mode (%d) %s", | ||||
| 		       iface->conf->channel, | ||||
| 		       iface->freq, | ||||
| 		       iface->current_mode->mode, | ||||
| 		       hostapd_hw_mode_txt(iface->current_mode->mode)); | ||||
| 	hostapd_logger(iface->bss[0], NULL, HOSTAPD_MODULE_IEEE80211, | ||||
|  |  | |||
|  | @ -94,19 +94,22 @@ int hw_get_chan(enum hostapd_hw_mode mode, int freq, | |||
| } | ||||
| 
 | ||||
| 
 | ||||
| int allowed_ht40_channel_pair(struct hostapd_hw_modes *mode, int pri_chan, | ||||
| 			      int sec_chan) | ||||
| int allowed_ht40_channel_pair(enum hostapd_hw_mode mode, | ||||
| 			      struct hostapd_channel_data *p_chan, | ||||
| 			      struct hostapd_channel_data *s_chan) | ||||
| { | ||||
| 	int ok, first; | ||||
| 	int allowed[] = { 36, 44, 52, 60, 100, 108, 116, 124, 132, 140, | ||||
| 			  149, 157, 165, 184, 192 }; | ||||
| 	size_t k; | ||||
| 	struct hostapd_channel_data *p_chan, *s_chan; | ||||
| 	const int ht40_plus = pri_chan < sec_chan; | ||||
| 	int ht40_plus, pri_chan, sec_chan; | ||||
| 
 | ||||
| 	p_chan = hw_get_channel_chan(mode, pri_chan, NULL); | ||||
| 	if (!p_chan) | ||||
| 	if (!p_chan || !s_chan) | ||||
| 		return 0; | ||||
| 	pri_chan = p_chan->chan; | ||||
| 	sec_chan = s_chan->chan; | ||||
| 
 | ||||
| 	ht40_plus = pri_chan < sec_chan; | ||||
| 
 | ||||
| 	if (pri_chan == sec_chan || !sec_chan) { | ||||
| 		if (chan_pri_allowed(p_chan)) | ||||
|  | @ -117,13 +120,9 @@ int allowed_ht40_channel_pair(struct hostapd_hw_modes *mode, int pri_chan, | |||
| 		return 0; | ||||
| 	} | ||||
| 
 | ||||
| 	s_chan = hw_get_channel_chan(mode, sec_chan, NULL); | ||||
| 	if (!s_chan) | ||||
| 		return 0; | ||||
| 
 | ||||
| 	wpa_printf(MSG_DEBUG, | ||||
| 		   "HT40: control channel: %d  secondary channel: %d", | ||||
| 		   pri_chan, sec_chan); | ||||
| 		   "HT40: control channel: %d (%d MHz), secondary channel: %d (%d MHz)", | ||||
| 		   pri_chan, p_chan->freq, sec_chan, s_chan->freq); | ||||
| 
 | ||||
| 	/* Verify that HT40 secondary channel is an allowed 20 MHz
 | ||||
| 	 * channel */ | ||||
|  | @ -141,7 +140,7 @@ int allowed_ht40_channel_pair(struct hostapd_hw_modes *mode, int pri_chan, | |||
| 	 * 2.4 GHz rules allow all cases where the secondary channel fits into | ||||
| 	 * the list of allowed channels (already checked above). | ||||
| 	 */ | ||||
| 	if (mode->mode != HOSTAPD_MODE_IEEE80211A) | ||||
| 	if (mode != HOSTAPD_MODE_IEEE80211A) | ||||
| 		return 1; | ||||
| 
 | ||||
| 	first = pri_chan < sec_chan ? pri_chan : sec_chan; | ||||
|  | @ -186,22 +185,19 @@ void get_pri_sec_chan(struct wpa_scan_res *bss, int *pri_chan, int *sec_chan) | |||
| } | ||||
| 
 | ||||
| 
 | ||||
| int check_40mhz_5g(struct hostapd_hw_modes *mode, | ||||
| 		   struct wpa_scan_results *scan_res, int pri_chan, | ||||
| 		   int sec_chan) | ||||
| int check_40mhz_5g(struct wpa_scan_results *scan_res, | ||||
| 		   struct hostapd_channel_data *pri_chan, | ||||
| 		   struct hostapd_channel_data *sec_chan) | ||||
| { | ||||
| 	int pri_freq, sec_freq, pri_bss, sec_bss; | ||||
| 	int pri_bss, sec_bss; | ||||
| 	int bss_pri_chan, bss_sec_chan; | ||||
| 	size_t i; | ||||
| 	int match; | ||||
| 
 | ||||
| 	if (!mode || !scan_res || !pri_chan || !sec_chan || | ||||
| 	    pri_chan == sec_chan) | ||||
| 	if (!scan_res || !pri_chan || !sec_chan || | ||||
| 	    pri_chan->freq == sec_chan->freq) | ||||
| 		return 0; | ||||
| 
 | ||||
| 	pri_freq = hw_get_freq(mode, pri_chan); | ||||
| 	sec_freq = hw_get_freq(mode, sec_chan); | ||||
| 
 | ||||
| 	/*
 | ||||
| 	 * Switch PRI/SEC channels if Beacons were detected on selected SEC | ||||
| 	 * channel, but not on selected PRI channel. | ||||
|  | @ -209,9 +205,9 @@ int check_40mhz_5g(struct hostapd_hw_modes *mode, | |||
| 	pri_bss = sec_bss = 0; | ||||
| 	for (i = 0; i < scan_res->num; i++) { | ||||
| 		struct wpa_scan_res *bss = scan_res->res[i]; | ||||
| 		if (bss->freq == pri_freq) | ||||
| 		if (bss->freq == pri_chan->freq) | ||||
| 			pri_bss++; | ||||
| 		else if (bss->freq == sec_freq) | ||||
| 		else if (bss->freq == sec_chan->freq) | ||||
| 			sec_bss++; | ||||
| 	} | ||||
| 	if (sec_bss && !pri_bss) { | ||||
|  | @ -229,8 +225,8 @@ int check_40mhz_5g(struct hostapd_hw_modes *mode, | |||
| 	for (i = 0; i < scan_res->num; i++) { | ||||
| 		struct wpa_scan_res *bss = scan_res->res[i]; | ||||
| 		get_pri_sec_chan(bss, &bss_pri_chan, &bss_sec_chan); | ||||
| 		if (pri_chan == bss_pri_chan && | ||||
| 		    sec_chan == bss_sec_chan) { | ||||
| 		if (pri_chan->chan == bss_pri_chan && | ||||
| 		    sec_chan->chan == bss_sec_chan) { | ||||
| 			match = 1; | ||||
| 			break; | ||||
| 		} | ||||
|  | @ -239,8 +235,8 @@ int check_40mhz_5g(struct hostapd_hw_modes *mode, | |||
| 		for (i = 0; i < scan_res->num; i++) { | ||||
| 			struct wpa_scan_res *bss = scan_res->res[i]; | ||||
| 			get_pri_sec_chan(bss, &bss_pri_chan, &bss_sec_chan); | ||||
| 			if (pri_chan == bss_sec_chan && | ||||
| 			    sec_chan == bss_pri_chan) { | ||||
| 			if (pri_chan->chan == bss_sec_chan && | ||||
| 			    sec_chan->chan == bss_pri_chan) { | ||||
| 				wpa_printf(MSG_INFO, "Switch own primary and " | ||||
| 					   "secondary channel due to BSS " | ||||
| 					   "overlap with " MACSTR, | ||||
|  |  | |||
|  | @ -22,12 +22,13 @@ int hw_get_freq(struct hostapd_hw_modes *mode, int chan); | |||
| int hw_get_chan(enum hostapd_hw_mode mode, int freq, | ||||
| 		struct hostapd_hw_modes *hw_features, int num_hw_features); | ||||
| 
 | ||||
| int allowed_ht40_channel_pair(struct hostapd_hw_modes *mode, int pri_chan, | ||||
| 			      int sec_chan); | ||||
| int allowed_ht40_channel_pair(enum hostapd_hw_mode mode, | ||||
| 			      struct hostapd_channel_data *p_chan, | ||||
| 			      struct hostapd_channel_data *s_chan); | ||||
| void get_pri_sec_chan(struct wpa_scan_res *bss, int *pri_chan, int *sec_chan); | ||||
| int check_40mhz_5g(struct hostapd_hw_modes *mode, | ||||
| 		   struct wpa_scan_results *scan_res, int pri_chan, | ||||
| 		   int sec_chan); | ||||
| int check_40mhz_5g(struct wpa_scan_results *scan_res, | ||||
| 		   struct hostapd_channel_data *pri_chan, | ||||
| 		   struct hostapd_channel_data *sec_chan); | ||||
| int check_40mhz_2g4(struct hostapd_hw_modes *mode, | ||||
| 		    struct wpa_scan_results *scan_res, int pri_chan, | ||||
| 		    int sec_chan); | ||||
|  |  | |||
|  | @ -2382,8 +2382,7 @@ void ibss_mesh_setup_freq(struct wpa_supplicant *wpa_s, | |||
| 			return; | ||||
| 		} | ||||
| 
 | ||||
| 		res = check_40mhz_5g(mode, scan_res, pri_chan->chan, | ||||
| 				     sec_chan->chan); | ||||
| 		res = check_40mhz_5g(scan_res, pri_chan, sec_chan); | ||||
| 		switch (res) { | ||||
| 		case 0: | ||||
| 			/* Back to HT20 */ | ||||
|  |  | |||
		Loading…
	
		Reference in a new issue
	
	 Ankita Bajaj
						Ankita Bajaj