| /* |
| * 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; |
| } |