| /* | 
 |  * mac80211 - channel management | 
 |  */ | 
 |  | 
 | #include <linux/nl80211.h> | 
 | #include <net/cfg80211.h> | 
 | #include "ieee80211_i.h" | 
 |  | 
 | static enum ieee80211_chan_mode | 
 | __ieee80211_get_channel_mode(struct ieee80211_local *local, | 
 | 			     struct ieee80211_sub_if_data *ignore) | 
 | { | 
 | 	struct ieee80211_sub_if_data *sdata; | 
 |  | 
 | 	lockdep_assert_held(&local->iflist_mtx); | 
 |  | 
 | 	list_for_each_entry(sdata, &local->interfaces, list) { | 
 | 		if (sdata == ignore) | 
 | 			continue; | 
 |  | 
 | 		if (!ieee80211_sdata_running(sdata)) | 
 | 			continue; | 
 |  | 
 | 		switch (sdata->vif.type) { | 
 | 		case NL80211_IFTYPE_MONITOR: | 
 | 			continue; | 
 | 		case NL80211_IFTYPE_STATION: | 
 | 			if (!sdata->u.mgd.associated) | 
 | 				continue; | 
 | 			break; | 
 | 		case NL80211_IFTYPE_ADHOC: | 
 | 			if (!sdata->u.ibss.ssid_len) | 
 | 				continue; | 
 | 			if (!sdata->u.ibss.fixed_channel) | 
 | 				return CHAN_MODE_HOPPING; | 
 | 			break; | 
 | 		case NL80211_IFTYPE_AP_VLAN: | 
 | 			/* will also have _AP interface */ | 
 | 			continue; | 
 | 		case NL80211_IFTYPE_AP: | 
 | 			if (!sdata->u.ap.beacon) | 
 | 				continue; | 
 | 			break; | 
 | 		case NL80211_IFTYPE_MESH_POINT: | 
 | 			if (!sdata->wdev.mesh_id_len) | 
 | 				continue; | 
 | 			break; | 
 | 		default: | 
 | 			break; | 
 | 		} | 
 |  | 
 | 		return CHAN_MODE_FIXED; | 
 | 	} | 
 |  | 
 | 	return CHAN_MODE_UNDEFINED; | 
 | } | 
 |  | 
 | enum ieee80211_chan_mode | 
 | ieee80211_get_channel_mode(struct ieee80211_local *local, | 
 | 			   struct ieee80211_sub_if_data *ignore) | 
 | { | 
 | 	enum ieee80211_chan_mode mode; | 
 |  | 
 | 	mutex_lock(&local->iflist_mtx); | 
 | 	mode = __ieee80211_get_channel_mode(local, ignore); | 
 | 	mutex_unlock(&local->iflist_mtx); | 
 |  | 
 | 	return mode; | 
 | } | 
 |  | 
 | bool ieee80211_set_channel_type(struct ieee80211_local *local, | 
 | 				struct ieee80211_sub_if_data *sdata, | 
 | 				enum nl80211_channel_type chantype) | 
 | { | 
 | 	struct ieee80211_sub_if_data *tmp; | 
 | 	enum nl80211_channel_type superchan = NL80211_CHAN_NO_HT; | 
 | 	bool result; | 
 |  | 
 | 	mutex_lock(&local->iflist_mtx); | 
 |  | 
 | 	list_for_each_entry(tmp, &local->interfaces, list) { | 
 | 		if (tmp == sdata) | 
 | 			continue; | 
 |  | 
 | 		if (!ieee80211_sdata_running(tmp)) | 
 | 			continue; | 
 |  | 
 | 		switch (tmp->vif.bss_conf.channel_type) { | 
 | 		case NL80211_CHAN_NO_HT: | 
 | 		case NL80211_CHAN_HT20: | 
 | 			if (superchan > tmp->vif.bss_conf.channel_type) | 
 | 				break; | 
 |  | 
 | 			superchan = tmp->vif.bss_conf.channel_type; | 
 | 			break; | 
 | 		case NL80211_CHAN_HT40PLUS: | 
 | 			WARN_ON(superchan == NL80211_CHAN_HT40MINUS); | 
 | 			superchan = NL80211_CHAN_HT40PLUS; | 
 | 			break; | 
 | 		case NL80211_CHAN_HT40MINUS: | 
 | 			WARN_ON(superchan == NL80211_CHAN_HT40PLUS); | 
 | 			superchan = NL80211_CHAN_HT40MINUS; | 
 | 			break; | 
 | 		} | 
 | 	} | 
 |  | 
 | 	switch (superchan) { | 
 | 	case NL80211_CHAN_NO_HT: | 
 | 	case NL80211_CHAN_HT20: | 
 | 		/* | 
 | 		 * allow any change that doesn't go to no-HT | 
 | 		 * (if it already is no-HT no change is needed) | 
 | 		 */ | 
 | 		if (chantype == NL80211_CHAN_NO_HT) | 
 | 			break; | 
 | 		superchan = chantype; | 
 | 		break; | 
 | 	case NL80211_CHAN_HT40PLUS: | 
 | 	case NL80211_CHAN_HT40MINUS: | 
 | 		/* allow smaller bandwidth and same */ | 
 | 		if (chantype == NL80211_CHAN_NO_HT) | 
 | 			break; | 
 | 		if (chantype == NL80211_CHAN_HT20) | 
 | 			break; | 
 | 		if (superchan == chantype) | 
 | 			break; | 
 | 		result = false; | 
 | 		goto out; | 
 | 	} | 
 |  | 
 | 	local->_oper_channel_type = superchan; | 
 |  | 
 | 	if (sdata) | 
 | 		sdata->vif.bss_conf.channel_type = chantype; | 
 |  | 
 | 	result = true; | 
 |  out: | 
 | 	mutex_unlock(&local->iflist_mtx); | 
 |  | 
 | 	return result; | 
 | } |