Merge branch 'master' of git://git.kernel.org/pub/scm/linux/kernel/git/linville/wireless-next into for-davem
Conflicts: net/wireless/nl80211.c
This commit is contained in:
@@ -54,6 +54,8 @@ bool cfg80211_chandef_valid(const struct cfg80211_chan_def *chandef)
|
||||
control_freq = chandef->chan->center_freq;
|
||||
|
||||
switch (chandef->width) {
|
||||
case NL80211_CHAN_WIDTH_5:
|
||||
case NL80211_CHAN_WIDTH_10:
|
||||
case NL80211_CHAN_WIDTH_20:
|
||||
case NL80211_CHAN_WIDTH_20_NOHT:
|
||||
if (chandef->center_freq1 != control_freq)
|
||||
@@ -152,6 +154,12 @@ static int cfg80211_chandef_get_width(const struct cfg80211_chan_def *c)
|
||||
int width;
|
||||
|
||||
switch (c->width) {
|
||||
case NL80211_CHAN_WIDTH_5:
|
||||
width = 5;
|
||||
break;
|
||||
case NL80211_CHAN_WIDTH_10:
|
||||
width = 10;
|
||||
break;
|
||||
case NL80211_CHAN_WIDTH_20:
|
||||
case NL80211_CHAN_WIDTH_20_NOHT:
|
||||
width = 20;
|
||||
@@ -194,6 +202,16 @@ cfg80211_chandef_compatible(const struct cfg80211_chan_def *c1,
|
||||
if (c1->width == c2->width)
|
||||
return NULL;
|
||||
|
||||
/*
|
||||
* can't be compatible if one of them is 5 or 10 MHz,
|
||||
* but they don't have the same width.
|
||||
*/
|
||||
if (c1->width == NL80211_CHAN_WIDTH_5 ||
|
||||
c1->width == NL80211_CHAN_WIDTH_10 ||
|
||||
c2->width == NL80211_CHAN_WIDTH_5 ||
|
||||
c2->width == NL80211_CHAN_WIDTH_10)
|
||||
return NULL;
|
||||
|
||||
if (c1->width == NL80211_CHAN_WIDTH_20_NOHT ||
|
||||
c1->width == NL80211_CHAN_WIDTH_20)
|
||||
return c2;
|
||||
@@ -264,11 +282,17 @@ static int cfg80211_get_chans_dfs_required(struct wiphy *wiphy,
|
||||
u32 bandwidth)
|
||||
{
|
||||
struct ieee80211_channel *c;
|
||||
u32 freq;
|
||||
u32 freq, start_freq, end_freq;
|
||||
|
||||
for (freq = center_freq - bandwidth/2 + 10;
|
||||
freq <= center_freq + bandwidth/2 - 10;
|
||||
freq += 20) {
|
||||
if (bandwidth <= 20) {
|
||||
start_freq = center_freq;
|
||||
end_freq = center_freq;
|
||||
} else {
|
||||
start_freq = center_freq - bandwidth/2 + 10;
|
||||
end_freq = center_freq + bandwidth/2 - 10;
|
||||
}
|
||||
|
||||
for (freq = start_freq; freq <= end_freq; freq += 20) {
|
||||
c = ieee80211_get_channel(wiphy, freq);
|
||||
if (!c)
|
||||
return -EINVAL;
|
||||
@@ -310,11 +334,17 @@ static bool cfg80211_secondary_chans_ok(struct wiphy *wiphy,
|
||||
u32 prohibited_flags)
|
||||
{
|
||||
struct ieee80211_channel *c;
|
||||
u32 freq;
|
||||
u32 freq, start_freq, end_freq;
|
||||
|
||||
for (freq = center_freq - bandwidth/2 + 10;
|
||||
freq <= center_freq + bandwidth/2 - 10;
|
||||
freq += 20) {
|
||||
if (bandwidth <= 20) {
|
||||
start_freq = center_freq;
|
||||
end_freq = center_freq;
|
||||
} else {
|
||||
start_freq = center_freq - bandwidth/2 + 10;
|
||||
end_freq = center_freq + bandwidth/2 - 10;
|
||||
}
|
||||
|
||||
for (freq = start_freq; freq <= end_freq; freq += 20) {
|
||||
c = ieee80211_get_channel(wiphy, freq);
|
||||
if (!c)
|
||||
return false;
|
||||
@@ -349,6 +379,12 @@ bool cfg80211_chandef_usable(struct wiphy *wiphy,
|
||||
control_freq = chandef->chan->center_freq;
|
||||
|
||||
switch (chandef->width) {
|
||||
case NL80211_CHAN_WIDTH_5:
|
||||
width = 5;
|
||||
break;
|
||||
case NL80211_CHAN_WIDTH_10:
|
||||
width = 10;
|
||||
break;
|
||||
case NL80211_CHAN_WIDTH_20:
|
||||
if (!ht_cap->ht_supported)
|
||||
return false;
|
||||
@@ -405,6 +441,11 @@ bool cfg80211_chandef_usable(struct wiphy *wiphy,
|
||||
if (width > 20)
|
||||
prohibited_flags |= IEEE80211_CHAN_NO_OFDM;
|
||||
|
||||
/* 5 and 10 MHz are only defined for the OFDM PHY */
|
||||
if (width < 20)
|
||||
prohibited_flags |= IEEE80211_CHAN_NO_OFDM;
|
||||
|
||||
|
||||
if (!cfg80211_secondary_chans_ok(wiphy, chandef->center_freq1,
|
||||
width, prohibited_flags))
|
||||
return false;
|
||||
|
@@ -933,6 +933,12 @@ static int cfg80211_netdev_notifier_call(struct notifier_block *nb,
|
||||
* freed.
|
||||
*/
|
||||
cfg80211_process_wdev_events(wdev);
|
||||
|
||||
if (WARN_ON(wdev->current_bss)) {
|
||||
cfg80211_unhold_bss(wdev->current_bss);
|
||||
cfg80211_put_bss(wdev->wiphy, &wdev->current_bss->pub);
|
||||
wdev->current_bss = NULL;
|
||||
}
|
||||
break;
|
||||
case NETDEV_PRE_UP:
|
||||
if (!(wdev->wiphy->interface_modes & BIT(wdev->iftype)))
|
||||
|
@@ -38,6 +38,7 @@ void cfg80211_rx_assoc_resp(struct net_device *dev, struct cfg80211_bss *bss,
|
||||
* frame instead of reassoc.
|
||||
*/
|
||||
if (cfg80211_sme_rx_assoc_resp(wdev, status_code)) {
|
||||
cfg80211_unhold_bss(bss_from_pub(bss));
|
||||
cfg80211_put_bss(wiphy, bss);
|
||||
return;
|
||||
}
|
||||
@@ -131,16 +132,19 @@ void cfg80211_auth_timeout(struct net_device *dev, const u8 *addr)
|
||||
}
|
||||
EXPORT_SYMBOL(cfg80211_auth_timeout);
|
||||
|
||||
void cfg80211_assoc_timeout(struct net_device *dev, const u8 *addr)
|
||||
void cfg80211_assoc_timeout(struct net_device *dev, struct cfg80211_bss *bss)
|
||||
{
|
||||
struct wireless_dev *wdev = dev->ieee80211_ptr;
|
||||
struct wiphy *wiphy = wdev->wiphy;
|
||||
struct cfg80211_registered_device *rdev = wiphy_to_dev(wiphy);
|
||||
|
||||
trace_cfg80211_send_assoc_timeout(dev, addr);
|
||||
trace_cfg80211_send_assoc_timeout(dev, bss->bssid);
|
||||
|
||||
nl80211_send_assoc_timeout(rdev, dev, addr, GFP_KERNEL);
|
||||
nl80211_send_assoc_timeout(rdev, dev, bss->bssid, GFP_KERNEL);
|
||||
cfg80211_sme_assoc_timeout(wdev);
|
||||
|
||||
cfg80211_unhold_bss(bss_from_pub(bss));
|
||||
cfg80211_put_bss(wiphy, bss);
|
||||
}
|
||||
EXPORT_SYMBOL(cfg80211_assoc_timeout);
|
||||
|
||||
@@ -307,6 +311,8 @@ int cfg80211_mlme_assoc(struct cfg80211_registered_device *rdev,
|
||||
goto out;
|
||||
|
||||
err = rdev_assoc(rdev, dev, req);
|
||||
if (!err)
|
||||
cfg80211_hold_bss(bss_from_pub(req->bss));
|
||||
|
||||
out:
|
||||
if (err)
|
||||
|
@@ -1111,10 +1111,16 @@ nl80211_send_mgmt_stypes(struct sk_buff *msg,
|
||||
return 0;
|
||||
}
|
||||
|
||||
struct nl80211_dump_wiphy_state {
|
||||
s64 filter_wiphy;
|
||||
long start;
|
||||
long split_start, band_start, chan_start;
|
||||
bool split;
|
||||
};
|
||||
|
||||
static int nl80211_send_wiphy(struct cfg80211_registered_device *dev,
|
||||
struct sk_buff *msg, u32 portid, u32 seq,
|
||||
int flags, bool split, long *split_start,
|
||||
long *band_start, long *chan_start)
|
||||
int flags, struct nl80211_dump_wiphy_state *state)
|
||||
{
|
||||
void *hdr;
|
||||
struct nlattr *nl_bands, *nl_band;
|
||||
@@ -1125,19 +1131,14 @@ static int nl80211_send_wiphy(struct cfg80211_registered_device *dev,
|
||||
int i;
|
||||
const struct ieee80211_txrx_stypes *mgmt_stypes =
|
||||
dev->wiphy.mgmt_stypes;
|
||||
long start = 0, start_chan = 0, start_band = 0;
|
||||
u32 features;
|
||||
|
||||
hdr = nl80211hdr_put(msg, portid, seq, flags, NL80211_CMD_NEW_WIPHY);
|
||||
if (!hdr)
|
||||
return -ENOBUFS;
|
||||
|
||||
/* allow always using the variables */
|
||||
if (!split) {
|
||||
split_start = &start;
|
||||
band_start = &start_band;
|
||||
chan_start = &start_chan;
|
||||
}
|
||||
if (WARN_ON(!state))
|
||||
return -EINVAL;
|
||||
|
||||
if (nla_put_u32(msg, NL80211_ATTR_WIPHY, dev->wiphy_idx) ||
|
||||
nla_put_string(msg, NL80211_ATTR_WIPHY_NAME,
|
||||
@@ -1146,7 +1147,7 @@ static int nl80211_send_wiphy(struct cfg80211_registered_device *dev,
|
||||
cfg80211_rdev_list_generation))
|
||||
goto nla_put_failure;
|
||||
|
||||
switch (*split_start) {
|
||||
switch (state->split_start) {
|
||||
case 0:
|
||||
if (nla_put_u8(msg, NL80211_ATTR_WIPHY_RETRY_SHORT,
|
||||
dev->wiphy.retry_short) ||
|
||||
@@ -1188,9 +1189,12 @@ static int nl80211_send_wiphy(struct cfg80211_registered_device *dev,
|
||||
if ((dev->wiphy.flags & WIPHY_FLAG_TDLS_EXTERNAL_SETUP) &&
|
||||
nla_put_flag(msg, NL80211_ATTR_TDLS_EXTERNAL_SETUP))
|
||||
goto nla_put_failure;
|
||||
if ((dev->wiphy.flags & WIPHY_FLAG_SUPPORTS_5_10_MHZ) &&
|
||||
nla_put_flag(msg, WIPHY_FLAG_SUPPORTS_5_10_MHZ))
|
||||
goto nla_put_failure;
|
||||
|
||||
(*split_start)++;
|
||||
if (split)
|
||||
state->split_start++;
|
||||
if (state->split)
|
||||
break;
|
||||
case 1:
|
||||
if (nla_put(msg, NL80211_ATTR_CIPHER_SUITES,
|
||||
@@ -1234,22 +1238,23 @@ static int nl80211_send_wiphy(struct cfg80211_registered_device *dev,
|
||||
}
|
||||
}
|
||||
|
||||
(*split_start)++;
|
||||
if (split)
|
||||
state->split_start++;
|
||||
if (state->split)
|
||||
break;
|
||||
case 2:
|
||||
if (nl80211_put_iftypes(msg, NL80211_ATTR_SUPPORTED_IFTYPES,
|
||||
dev->wiphy.interface_modes))
|
||||
goto nla_put_failure;
|
||||
(*split_start)++;
|
||||
if (split)
|
||||
state->split_start++;
|
||||
if (state->split)
|
||||
break;
|
||||
case 3:
|
||||
nl_bands = nla_nest_start(msg, NL80211_ATTR_WIPHY_BANDS);
|
||||
if (!nl_bands)
|
||||
goto nla_put_failure;
|
||||
|
||||
for (band = *band_start; band < IEEE80211_NUM_BANDS; band++) {
|
||||
for (band = state->band_start;
|
||||
band < IEEE80211_NUM_BANDS; band++) {
|
||||
struct ieee80211_supported_band *sband;
|
||||
|
||||
sband = dev->wiphy.bands[band];
|
||||
@@ -1261,12 +1266,12 @@ static int nl80211_send_wiphy(struct cfg80211_registered_device *dev,
|
||||
if (!nl_band)
|
||||
goto nla_put_failure;
|
||||
|
||||
switch (*chan_start) {
|
||||
switch (state->chan_start) {
|
||||
case 0:
|
||||
if (nl80211_send_band_rateinfo(msg, sband))
|
||||
goto nla_put_failure;
|
||||
(*chan_start)++;
|
||||
if (split)
|
||||
state->chan_start++;
|
||||
if (state->split)
|
||||
break;
|
||||
default:
|
||||
/* add frequencies */
|
||||
@@ -1275,7 +1280,7 @@ static int nl80211_send_wiphy(struct cfg80211_registered_device *dev,
|
||||
if (!nl_freqs)
|
||||
goto nla_put_failure;
|
||||
|
||||
for (i = *chan_start - 1;
|
||||
for (i = state->chan_start - 1;
|
||||
i < sband->n_channels;
|
||||
i++) {
|
||||
nl_freq = nla_nest_start(msg, i);
|
||||
@@ -1284,26 +1289,27 @@ static int nl80211_send_wiphy(struct cfg80211_registered_device *dev,
|
||||
|
||||
chan = &sband->channels[i];
|
||||
|
||||
if (nl80211_msg_put_channel(msg, chan,
|
||||
split))
|
||||
if (nl80211_msg_put_channel(
|
||||
msg, chan,
|
||||
state->split))
|
||||
goto nla_put_failure;
|
||||
|
||||
nla_nest_end(msg, nl_freq);
|
||||
if (split)
|
||||
if (state->split)
|
||||
break;
|
||||
}
|
||||
if (i < sband->n_channels)
|
||||
*chan_start = i + 2;
|
||||
state->chan_start = i + 2;
|
||||
else
|
||||
*chan_start = 0;
|
||||
state->chan_start = 0;
|
||||
nla_nest_end(msg, nl_freqs);
|
||||
}
|
||||
|
||||
nla_nest_end(msg, nl_band);
|
||||
|
||||
if (split) {
|
||||
if (state->split) {
|
||||
/* start again here */
|
||||
if (*chan_start)
|
||||
if (state->chan_start)
|
||||
band--;
|
||||
break;
|
||||
}
|
||||
@@ -1311,14 +1317,14 @@ static int nl80211_send_wiphy(struct cfg80211_registered_device *dev,
|
||||
nla_nest_end(msg, nl_bands);
|
||||
|
||||
if (band < IEEE80211_NUM_BANDS)
|
||||
*band_start = band + 1;
|
||||
state->band_start = band + 1;
|
||||
else
|
||||
*band_start = 0;
|
||||
state->band_start = 0;
|
||||
|
||||
/* if bands & channels are done, continue outside */
|
||||
if (*band_start == 0 && *chan_start == 0)
|
||||
(*split_start)++;
|
||||
if (split)
|
||||
if (state->band_start == 0 && state->chan_start == 0)
|
||||
state->split_start++;
|
||||
if (state->split)
|
||||
break;
|
||||
case 4:
|
||||
nl_cmds = nla_nest_start(msg, NL80211_ATTR_SUPPORTED_COMMANDS);
|
||||
@@ -1384,7 +1390,7 @@ static int nl80211_send_wiphy(struct cfg80211_registered_device *dev,
|
||||
}
|
||||
CMD(start_p2p_device, START_P2P_DEVICE);
|
||||
CMD(set_mcast_rate, SET_MCAST_RATE);
|
||||
if (split) {
|
||||
if (state->split) {
|
||||
CMD(crit_proto_start, CRIT_PROTOCOL_START);
|
||||
CMD(crit_proto_stop, CRIT_PROTOCOL_STOP);
|
||||
}
|
||||
@@ -1408,8 +1414,8 @@ static int nl80211_send_wiphy(struct cfg80211_registered_device *dev,
|
||||
}
|
||||
|
||||
nla_nest_end(msg, nl_cmds);
|
||||
(*split_start)++;
|
||||
if (split)
|
||||
state->split_start++;
|
||||
if (state->split)
|
||||
break;
|
||||
case 5:
|
||||
if (dev->ops->remain_on_channel &&
|
||||
@@ -1425,29 +1431,30 @@ static int nl80211_send_wiphy(struct cfg80211_registered_device *dev,
|
||||
|
||||
if (nl80211_send_mgmt_stypes(msg, mgmt_stypes))
|
||||
goto nla_put_failure;
|
||||
(*split_start)++;
|
||||
if (split)
|
||||
state->split_start++;
|
||||
if (state->split)
|
||||
break;
|
||||
case 6:
|
||||
#ifdef CONFIG_PM
|
||||
if (nl80211_send_wowlan(msg, dev, split))
|
||||
if (nl80211_send_wowlan(msg, dev, state->split))
|
||||
goto nla_put_failure;
|
||||
(*split_start)++;
|
||||
if (split)
|
||||
state->split_start++;
|
||||
if (state->split)
|
||||
break;
|
||||
#else
|
||||
(*split_start)++;
|
||||
state->split_start++;
|
||||
#endif
|
||||
case 7:
|
||||
if (nl80211_put_iftypes(msg, NL80211_ATTR_SOFTWARE_IFTYPES,
|
||||
dev->wiphy.software_iftypes))
|
||||
goto nla_put_failure;
|
||||
|
||||
if (nl80211_put_iface_combinations(&dev->wiphy, msg, split))
|
||||
if (nl80211_put_iface_combinations(&dev->wiphy, msg,
|
||||
state->split))
|
||||
goto nla_put_failure;
|
||||
|
||||
(*split_start)++;
|
||||
if (split)
|
||||
state->split_start++;
|
||||
if (state->split)
|
||||
break;
|
||||
case 8:
|
||||
if ((dev->wiphy.flags & WIPHY_FLAG_HAVE_AP_SME) &&
|
||||
@@ -1461,7 +1468,7 @@ static int nl80211_send_wiphy(struct cfg80211_registered_device *dev,
|
||||
* dump is split, otherwise it makes it too big. Therefore
|
||||
* only advertise it in that case.
|
||||
*/
|
||||
if (split)
|
||||
if (state->split)
|
||||
features |= NL80211_FEATURE_ADVERTISE_CHAN_LIMITS;
|
||||
if (nla_put_u32(msg, NL80211_ATTR_FEATURE_FLAGS, features))
|
||||
goto nla_put_failure;
|
||||
@@ -1488,7 +1495,7 @@ static int nl80211_send_wiphy(struct cfg80211_registered_device *dev,
|
||||
* case we'll continue with more data in the next round,
|
||||
* but break unconditionally so unsplit data stops here.
|
||||
*/
|
||||
(*split_start)++;
|
||||
state->split_start++;
|
||||
break;
|
||||
case 9:
|
||||
if (dev->wiphy.extended_capabilities &&
|
||||
@@ -1507,7 +1514,7 @@ static int nl80211_send_wiphy(struct cfg80211_registered_device *dev,
|
||||
goto nla_put_failure;
|
||||
|
||||
/* done */
|
||||
*split_start = 0;
|
||||
state->split_start = 0;
|
||||
break;
|
||||
}
|
||||
return genlmsg_end(msg, hdr);
|
||||
@@ -1517,67 +1524,78 @@ static int nl80211_send_wiphy(struct cfg80211_registered_device *dev,
|
||||
return -EMSGSIZE;
|
||||
}
|
||||
|
||||
static int nl80211_dump_wiphy_parse(struct sk_buff *skb,
|
||||
struct netlink_callback *cb,
|
||||
struct nl80211_dump_wiphy_state *state)
|
||||
{
|
||||
struct nlattr **tb = nl80211_fam.attrbuf;
|
||||
int ret = nlmsg_parse(cb->nlh, GENL_HDRLEN + nl80211_fam.hdrsize,
|
||||
tb, nl80211_fam.maxattr, nl80211_policy);
|
||||
/* ignore parse errors for backward compatibility */
|
||||
if (ret)
|
||||
return 0;
|
||||
|
||||
state->split = tb[NL80211_ATTR_SPLIT_WIPHY_DUMP];
|
||||
if (tb[NL80211_ATTR_WIPHY])
|
||||
state->filter_wiphy = nla_get_u32(tb[NL80211_ATTR_WIPHY]);
|
||||
if (tb[NL80211_ATTR_WDEV])
|
||||
state->filter_wiphy = nla_get_u64(tb[NL80211_ATTR_WDEV]) >> 32;
|
||||
if (tb[NL80211_ATTR_IFINDEX]) {
|
||||
struct net_device *netdev;
|
||||
struct cfg80211_registered_device *rdev;
|
||||
int ifidx = nla_get_u32(tb[NL80211_ATTR_IFINDEX]);
|
||||
|
||||
netdev = dev_get_by_index(sock_net(skb->sk), ifidx);
|
||||
if (!netdev)
|
||||
return -ENODEV;
|
||||
if (netdev->ieee80211_ptr) {
|
||||
rdev = wiphy_to_dev(
|
||||
netdev->ieee80211_ptr->wiphy);
|
||||
state->filter_wiphy = rdev->wiphy_idx;
|
||||
}
|
||||
dev_put(netdev);
|
||||
}
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
static int nl80211_dump_wiphy(struct sk_buff *skb, struct netlink_callback *cb)
|
||||
{
|
||||
int idx = 0, ret;
|
||||
int start = cb->args[0];
|
||||
struct nl80211_dump_wiphy_state *state = (void *)cb->args[0];
|
||||
struct cfg80211_registered_device *dev;
|
||||
s64 filter_wiphy = -1;
|
||||
bool split = false;
|
||||
struct nlattr **tb;
|
||||
int res;
|
||||
|
||||
/* will be zeroed in nlmsg_parse() */
|
||||
tb = kmalloc(sizeof(*tb) * (NL80211_ATTR_MAX + 1), GFP_KERNEL);
|
||||
if (!tb)
|
||||
return -ENOMEM;
|
||||
|
||||
rtnl_lock();
|
||||
|
||||
res = nlmsg_parse(cb->nlh, GENL_HDRLEN + nl80211_fam.hdrsize,
|
||||
tb, NL80211_ATTR_MAX, nl80211_policy);
|
||||
if (res == 0) {
|
||||
split = tb[NL80211_ATTR_SPLIT_WIPHY_DUMP];
|
||||
if (tb[NL80211_ATTR_WIPHY])
|
||||
filter_wiphy = nla_get_u32(tb[NL80211_ATTR_WIPHY]);
|
||||
if (tb[NL80211_ATTR_WDEV])
|
||||
filter_wiphy = nla_get_u64(tb[NL80211_ATTR_WDEV]) >> 32;
|
||||
if (tb[NL80211_ATTR_IFINDEX]) {
|
||||
struct net_device *netdev;
|
||||
int ifidx = nla_get_u32(tb[NL80211_ATTR_IFINDEX]);
|
||||
|
||||
netdev = dev_get_by_index(sock_net(skb->sk), ifidx);
|
||||
if (!netdev) {
|
||||
rtnl_unlock();
|
||||
kfree(tb);
|
||||
return -ENODEV;
|
||||
}
|
||||
if (netdev->ieee80211_ptr) {
|
||||
dev = wiphy_to_dev(
|
||||
netdev->ieee80211_ptr->wiphy);
|
||||
filter_wiphy = dev->wiphy_idx;
|
||||
}
|
||||
dev_put(netdev);
|
||||
if (!state) {
|
||||
state = kzalloc(sizeof(*state), GFP_KERNEL);
|
||||
if (!state) {
|
||||
rtnl_unlock();
|
||||
return -ENOMEM;
|
||||
}
|
||||
state->filter_wiphy = -1;
|
||||
ret = nl80211_dump_wiphy_parse(skb, cb, state);
|
||||
if (ret) {
|
||||
kfree(state);
|
||||
rtnl_unlock();
|
||||
return ret;
|
||||
}
|
||||
cb->args[0] = (long)state;
|
||||
}
|
||||
kfree(tb);
|
||||
|
||||
list_for_each_entry(dev, &cfg80211_rdev_list, list) {
|
||||
if (!net_eq(wiphy_net(&dev->wiphy), sock_net(skb->sk)))
|
||||
continue;
|
||||
if (++idx <= start)
|
||||
if (++idx <= state->start)
|
||||
continue;
|
||||
if (filter_wiphy != -1 && dev->wiphy_idx != filter_wiphy)
|
||||
if (state->filter_wiphy != -1 &&
|
||||
state->filter_wiphy != dev->wiphy_idx)
|
||||
continue;
|
||||
/* attempt to fit multiple wiphy data chunks into the skb */
|
||||
do {
|
||||
ret = nl80211_send_wiphy(dev, skb,
|
||||
NETLINK_CB(cb->skb).portid,
|
||||
cb->nlh->nlmsg_seq,
|
||||
NLM_F_MULTI,
|
||||
split, &cb->args[1],
|
||||
&cb->args[2],
|
||||
&cb->args[3]);
|
||||
NLM_F_MULTI, state);
|
||||
if (ret < 0) {
|
||||
/*
|
||||
* If sending the wiphy data didn't fit (ENOBUFS
|
||||
@@ -1602,27 +1620,34 @@ static int nl80211_dump_wiphy(struct sk_buff *skb, struct netlink_callback *cb)
|
||||
idx--;
|
||||
break;
|
||||
}
|
||||
} while (cb->args[1] > 0);
|
||||
} while (state->split_start > 0);
|
||||
break;
|
||||
}
|
||||
rtnl_unlock();
|
||||
|
||||
cb->args[0] = idx;
|
||||
state->start = idx;
|
||||
|
||||
return skb->len;
|
||||
}
|
||||
|
||||
static int nl80211_dump_wiphy_done(struct netlink_callback *cb)
|
||||
{
|
||||
kfree((void *)cb->args[0]);
|
||||
return 0;
|
||||
}
|
||||
|
||||
static int nl80211_get_wiphy(struct sk_buff *skb, struct genl_info *info)
|
||||
{
|
||||
struct sk_buff *msg;
|
||||
struct cfg80211_registered_device *dev = info->user_ptr[0];
|
||||
struct nl80211_dump_wiphy_state state = {};
|
||||
|
||||
msg = nlmsg_new(4096, GFP_KERNEL);
|
||||
if (!msg)
|
||||
return -ENOMEM;
|
||||
|
||||
if (nl80211_send_wiphy(dev, msg, info->snd_portid, info->snd_seq, 0,
|
||||
false, NULL, NULL, NULL) < 0) {
|
||||
&state) < 0) {
|
||||
nlmsg_free(msg);
|
||||
return -ENOBUFS;
|
||||
}
|
||||
@@ -1739,6 +1764,11 @@ static int nl80211_parse_chandef(struct cfg80211_registered_device *rdev,
|
||||
IEEE80211_CHAN_DISABLED))
|
||||
return -EINVAL;
|
||||
|
||||
if ((chandef->width == NL80211_CHAN_WIDTH_5 ||
|
||||
chandef->width == NL80211_CHAN_WIDTH_10) &&
|
||||
!(rdev->wiphy.flags & WIPHY_FLAG_SUPPORTS_5_10_MHZ))
|
||||
return -EINVAL;
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
@@ -2890,61 +2920,58 @@ static int nl80211_set_mac_acl(struct sk_buff *skb, struct genl_info *info)
|
||||
return err;
|
||||
}
|
||||
|
||||
static int nl80211_parse_beacon(struct genl_info *info,
|
||||
static int nl80211_parse_beacon(struct nlattr *attrs[],
|
||||
struct cfg80211_beacon_data *bcn)
|
||||
{
|
||||
bool haveinfo = false;
|
||||
|
||||
if (!is_valid_ie_attr(info->attrs[NL80211_ATTR_BEACON_TAIL]) ||
|
||||
!is_valid_ie_attr(info->attrs[NL80211_ATTR_IE]) ||
|
||||
!is_valid_ie_attr(info->attrs[NL80211_ATTR_IE_PROBE_RESP]) ||
|
||||
!is_valid_ie_attr(info->attrs[NL80211_ATTR_IE_ASSOC_RESP]))
|
||||
if (!is_valid_ie_attr(attrs[NL80211_ATTR_BEACON_TAIL]) ||
|
||||
!is_valid_ie_attr(attrs[NL80211_ATTR_IE]) ||
|
||||
!is_valid_ie_attr(attrs[NL80211_ATTR_IE_PROBE_RESP]) ||
|
||||
!is_valid_ie_attr(attrs[NL80211_ATTR_IE_ASSOC_RESP]))
|
||||
return -EINVAL;
|
||||
|
||||
memset(bcn, 0, sizeof(*bcn));
|
||||
|
||||
if (info->attrs[NL80211_ATTR_BEACON_HEAD]) {
|
||||
bcn->head = nla_data(info->attrs[NL80211_ATTR_BEACON_HEAD]);
|
||||
bcn->head_len = nla_len(info->attrs[NL80211_ATTR_BEACON_HEAD]);
|
||||
if (attrs[NL80211_ATTR_BEACON_HEAD]) {
|
||||
bcn->head = nla_data(attrs[NL80211_ATTR_BEACON_HEAD]);
|
||||
bcn->head_len = nla_len(attrs[NL80211_ATTR_BEACON_HEAD]);
|
||||
if (!bcn->head_len)
|
||||
return -EINVAL;
|
||||
haveinfo = true;
|
||||
}
|
||||
|
||||
if (info->attrs[NL80211_ATTR_BEACON_TAIL]) {
|
||||
bcn->tail = nla_data(info->attrs[NL80211_ATTR_BEACON_TAIL]);
|
||||
bcn->tail_len =
|
||||
nla_len(info->attrs[NL80211_ATTR_BEACON_TAIL]);
|
||||
if (attrs[NL80211_ATTR_BEACON_TAIL]) {
|
||||
bcn->tail = nla_data(attrs[NL80211_ATTR_BEACON_TAIL]);
|
||||
bcn->tail_len = nla_len(attrs[NL80211_ATTR_BEACON_TAIL]);
|
||||
haveinfo = true;
|
||||
}
|
||||
|
||||
if (!haveinfo)
|
||||
return -EINVAL;
|
||||
|
||||
if (info->attrs[NL80211_ATTR_IE]) {
|
||||
bcn->beacon_ies = nla_data(info->attrs[NL80211_ATTR_IE]);
|
||||
bcn->beacon_ies_len = nla_len(info->attrs[NL80211_ATTR_IE]);
|
||||
if (attrs[NL80211_ATTR_IE]) {
|
||||
bcn->beacon_ies = nla_data(attrs[NL80211_ATTR_IE]);
|
||||
bcn->beacon_ies_len = nla_len(attrs[NL80211_ATTR_IE]);
|
||||
}
|
||||
|
||||
if (info->attrs[NL80211_ATTR_IE_PROBE_RESP]) {
|
||||
if (attrs[NL80211_ATTR_IE_PROBE_RESP]) {
|
||||
bcn->proberesp_ies =
|
||||
nla_data(info->attrs[NL80211_ATTR_IE_PROBE_RESP]);
|
||||
nla_data(attrs[NL80211_ATTR_IE_PROBE_RESP]);
|
||||
bcn->proberesp_ies_len =
|
||||
nla_len(info->attrs[NL80211_ATTR_IE_PROBE_RESP]);
|
||||
nla_len(attrs[NL80211_ATTR_IE_PROBE_RESP]);
|
||||
}
|
||||
|
||||
if (info->attrs[NL80211_ATTR_IE_ASSOC_RESP]) {
|
||||
if (attrs[NL80211_ATTR_IE_ASSOC_RESP]) {
|
||||
bcn->assocresp_ies =
|
||||
nla_data(info->attrs[NL80211_ATTR_IE_ASSOC_RESP]);
|
||||
nla_data(attrs[NL80211_ATTR_IE_ASSOC_RESP]);
|
||||
bcn->assocresp_ies_len =
|
||||
nla_len(info->attrs[NL80211_ATTR_IE_ASSOC_RESP]);
|
||||
nla_len(attrs[NL80211_ATTR_IE_ASSOC_RESP]);
|
||||
}
|
||||
|
||||
if (info->attrs[NL80211_ATTR_PROBE_RESP]) {
|
||||
bcn->probe_resp =
|
||||
nla_data(info->attrs[NL80211_ATTR_PROBE_RESP]);
|
||||
bcn->probe_resp_len =
|
||||
nla_len(info->attrs[NL80211_ATTR_PROBE_RESP]);
|
||||
if (attrs[NL80211_ATTR_PROBE_RESP]) {
|
||||
bcn->probe_resp = nla_data(attrs[NL80211_ATTR_PROBE_RESP]);
|
||||
bcn->probe_resp_len = nla_len(attrs[NL80211_ATTR_PROBE_RESP]);
|
||||
}
|
||||
|
||||
return 0;
|
||||
@@ -3023,7 +3050,7 @@ static int nl80211_start_ap(struct sk_buff *skb, struct genl_info *info)
|
||||
!info->attrs[NL80211_ATTR_BEACON_HEAD])
|
||||
return -EINVAL;
|
||||
|
||||
err = nl80211_parse_beacon(info, ¶ms.beacon);
|
||||
err = nl80211_parse_beacon(info->attrs, ¶ms.beacon);
|
||||
if (err)
|
||||
return err;
|
||||
|
||||
@@ -3175,7 +3202,7 @@ static int nl80211_set_beacon(struct sk_buff *skb, struct genl_info *info)
|
||||
if (!wdev->beacon_interval)
|
||||
return -EINVAL;
|
||||
|
||||
err = nl80211_parse_beacon(info, ¶ms);
|
||||
err = nl80211_parse_beacon(info->attrs, ¶ms);
|
||||
if (err)
|
||||
return err;
|
||||
|
||||
@@ -6291,11 +6318,16 @@ static int nl80211_join_ibss(struct sk_buff *skb, struct genl_info *info)
|
||||
if (!cfg80211_reg_can_beacon(&rdev->wiphy, &ibss.chandef))
|
||||
return -EINVAL;
|
||||
|
||||
if (ibss.chandef.width > NL80211_CHAN_WIDTH_40)
|
||||
return -EINVAL;
|
||||
if (ibss.chandef.width != NL80211_CHAN_WIDTH_20_NOHT &&
|
||||
!(rdev->wiphy.features & NL80211_FEATURE_HT_IBSS))
|
||||
switch (ibss.chandef.width) {
|
||||
case NL80211_CHAN_WIDTH_20_NOHT:
|
||||
break;
|
||||
case NL80211_CHAN_WIDTH_20:
|
||||
case NL80211_CHAN_WIDTH_40:
|
||||
if (rdev->wiphy.features & NL80211_FEATURE_HT_IBSS)
|
||||
break;
|
||||
default:
|
||||
return -EINVAL;
|
||||
}
|
||||
|
||||
ibss.channel_fixed = !!info->attrs[NL80211_ATTR_FREQ_FIXED];
|
||||
ibss.privacy = !!info->attrs[NL80211_ATTR_PRIVACY];
|
||||
@@ -8409,6 +8441,7 @@ static struct genl_ops nl80211_ops[] = {
|
||||
.cmd = NL80211_CMD_GET_WIPHY,
|
||||
.doit = nl80211_get_wiphy,
|
||||
.dumpit = nl80211_dump_wiphy,
|
||||
.done = nl80211_dump_wiphy_done,
|
||||
.policy = nl80211_policy,
|
||||
/* can be retrieved by unprivileged users */
|
||||
.internal_flags = NL80211_FLAG_NEED_WIPHY |
|
||||
@@ -9029,13 +9062,13 @@ static struct genl_multicast_group nl80211_regulatory_mcgrp = {
|
||||
void nl80211_notify_dev_rename(struct cfg80211_registered_device *rdev)
|
||||
{
|
||||
struct sk_buff *msg;
|
||||
struct nl80211_dump_wiphy_state state = {};
|
||||
|
||||
msg = nlmsg_new(NLMSG_DEFAULT_SIZE, GFP_KERNEL);
|
||||
if (!msg)
|
||||
return;
|
||||
|
||||
if (nl80211_send_wiphy(rdev, msg, 0, 0, 0,
|
||||
false, NULL, NULL, NULL) < 0) {
|
||||
if (nl80211_send_wiphy(rdev, msg, 0, 0, 0, &state) < 0) {
|
||||
nlmsg_free(msg);
|
||||
return;
|
||||
}
|
||||
|
@@ -523,6 +523,7 @@ static int cmp_bss(struct cfg80211_bss *a,
|
||||
}
|
||||
}
|
||||
|
||||
/* Returned bss is reference counted and must be cleaned up appropriately. */
|
||||
struct cfg80211_bss *cfg80211_get_bss(struct wiphy *wiphy,
|
||||
struct ieee80211_channel *channel,
|
||||
const u8 *bssid,
|
||||
@@ -678,6 +679,7 @@ static bool cfg80211_combine_bsses(struct cfg80211_registered_device *dev,
|
||||
return true;
|
||||
}
|
||||
|
||||
/* Returned bss is reference counted and must be cleaned up appropriately. */
|
||||
static struct cfg80211_internal_bss *
|
||||
cfg80211_bss_update(struct cfg80211_registered_device *dev,
|
||||
struct cfg80211_internal_bss *tmp)
|
||||
@@ -866,6 +868,7 @@ cfg80211_get_bss_channel(struct wiphy *wiphy, const u8 *ie, size_t ielen,
|
||||
return channel;
|
||||
}
|
||||
|
||||
/* Returned bss is reference counted and must be cleaned up appropriately. */
|
||||
struct cfg80211_bss*
|
||||
cfg80211_inform_bss(struct wiphy *wiphy,
|
||||
struct ieee80211_channel *channel,
|
||||
@@ -923,6 +926,7 @@ cfg80211_inform_bss(struct wiphy *wiphy,
|
||||
}
|
||||
EXPORT_SYMBOL(cfg80211_inform_bss);
|
||||
|
||||
/* Returned bss is reference counted and must be cleaned up appropriately. */
|
||||
struct cfg80211_bss *
|
||||
cfg80211_inform_bss_frame(struct wiphy *wiphy,
|
||||
struct ieee80211_channel *channel,
|
||||
|
@@ -239,6 +239,7 @@ void cfg80211_conn_work(struct work_struct *work)
|
||||
rtnl_unlock();
|
||||
}
|
||||
|
||||
/* Returned bss is reference counted and must be cleaned up appropriately. */
|
||||
static struct cfg80211_bss *cfg80211_get_conn_bss(struct wireless_dev *wdev)
|
||||
{
|
||||
struct cfg80211_registered_device *rdev = wiphy_to_dev(wdev->wiphy);
|
||||
@@ -557,6 +558,7 @@ static DECLARE_WORK(cfg80211_disconnect_work, disconnect_work);
|
||||
* SME event handling
|
||||
*/
|
||||
|
||||
/* This method must consume bss one way or another */
|
||||
void __cfg80211_connect_result(struct net_device *dev, const u8 *bssid,
|
||||
const u8 *req_ie, size_t req_ie_len,
|
||||
const u8 *resp_ie, size_t resp_ie_len,
|
||||
@@ -572,8 +574,10 @@ void __cfg80211_connect_result(struct net_device *dev, const u8 *bssid,
|
||||
ASSERT_WDEV_LOCK(wdev);
|
||||
|
||||
if (WARN_ON(wdev->iftype != NL80211_IFTYPE_STATION &&
|
||||
wdev->iftype != NL80211_IFTYPE_P2P_CLIENT))
|
||||
wdev->iftype != NL80211_IFTYPE_P2P_CLIENT)) {
|
||||
cfg80211_put_bss(wdev->wiphy, bss);
|
||||
return;
|
||||
}
|
||||
|
||||
nl80211_send_connect_result(wiphy_to_dev(wdev->wiphy), dev,
|
||||
bssid, req_ie, req_ie_len,
|
||||
@@ -615,19 +619,24 @@ void __cfg80211_connect_result(struct net_device *dev, const u8 *bssid,
|
||||
kfree(wdev->connect_keys);
|
||||
wdev->connect_keys = NULL;
|
||||
wdev->ssid_len = 0;
|
||||
cfg80211_put_bss(wdev->wiphy, bss);
|
||||
if (bss) {
|
||||
cfg80211_unhold_bss(bss_from_pub(bss));
|
||||
cfg80211_put_bss(wdev->wiphy, bss);
|
||||
}
|
||||
return;
|
||||
}
|
||||
|
||||
if (!bss)
|
||||
if (!bss) {
|
||||
WARN_ON_ONCE(!wiphy_to_dev(wdev->wiphy)->ops->connect);
|
||||
bss = cfg80211_get_bss(wdev->wiphy, NULL, bssid,
|
||||
wdev->ssid, wdev->ssid_len,
|
||||
WLAN_CAPABILITY_ESS,
|
||||
WLAN_CAPABILITY_ESS);
|
||||
if (WARN_ON(!bss))
|
||||
return;
|
||||
if (WARN_ON(!bss))
|
||||
return;
|
||||
cfg80211_hold_bss(bss_from_pub(bss));
|
||||
}
|
||||
|
||||
cfg80211_hold_bss(bss_from_pub(bss));
|
||||
wdev->current_bss = bss_from_pub(bss);
|
||||
|
||||
cfg80211_upload_connect_keys(wdev);
|
||||
@@ -691,6 +700,7 @@ void cfg80211_connect_result(struct net_device *dev, const u8 *bssid,
|
||||
}
|
||||
EXPORT_SYMBOL(cfg80211_connect_result);
|
||||
|
||||
/* Consumes bss object one way or another */
|
||||
void __cfg80211_roamed(struct wireless_dev *wdev,
|
||||
struct cfg80211_bss *bss,
|
||||
const u8 *req_ie, size_t req_ie_len,
|
||||
@@ -767,6 +777,7 @@ void cfg80211_roamed(struct net_device *dev,
|
||||
}
|
||||
EXPORT_SYMBOL(cfg80211_roamed);
|
||||
|
||||
/* Consumes bss object one way or another */
|
||||
void cfg80211_roamed_bss(struct net_device *dev,
|
||||
struct cfg80211_bss *bss, const u8 *req_ie,
|
||||
size_t req_ie_len, const u8 *resp_ie,
|
||||
|
@@ -83,6 +83,7 @@ static int wiphy_uevent(struct device *dev, struct kobj_uevent_env *env)
|
||||
return 0;
|
||||
}
|
||||
|
||||
#ifdef CONFIG_PM
|
||||
static void cfg80211_leave_all(struct cfg80211_registered_device *rdev)
|
||||
{
|
||||
struct wireless_dev *wdev;
|
||||
@@ -91,7 +92,6 @@ static void cfg80211_leave_all(struct cfg80211_registered_device *rdev)
|
||||
cfg80211_leave(rdev, wdev);
|
||||
}
|
||||
|
||||
#ifdef CONFIG_PM
|
||||
static int wiphy_suspend(struct device *dev, pm_message_t state)
|
||||
{
|
||||
struct cfg80211_registered_device *rdev = dev_to_rdev(dev);
|
||||
|
Reference in New Issue
Block a user