cfg80211: unify cfg80211_roamed() and cfg80211_roamed_bss()

cfg80211_roamed() and cfg80211_roamed_bss() take the same arguments
except that cfg80211_roamed() requires the BSSID and
cfg80211_roamed_bss() requires the bss entry.

Unify the two functions by using a struct for driver initiated
roaming information so that either the BSSID or the bss entry can be
passed as an argument to the unified function.

Signed-off-by: Avraham Stern <avraham.stern@intel.com>
[modified the ath6k, brcm80211, rndis and wlan-ng drivers accordingly]
Signed-off-by: Luca Coelho <luciano.coelho@intel.com>
[modify brcmfmac to remove the useless cast, spotted by Arend]
Signed-off-by: Johannes Berg <johannes.berg@intel.com>
This commit is contained in:
Avraham Stern 2017-04-26 10:58:49 +03:00 committed by Johannes Berg
parent 21a8e9dd52
commit 29ce6ecbb8
10 changed files with 123 additions and 129 deletions

View File

@ -806,9 +806,15 @@ void ath6kl_cfg80211_connect_event(struct ath6kl_vif *vif, u16 channel,
WLAN_STATUS_SUCCESS, GFP_KERNEL); WLAN_STATUS_SUCCESS, GFP_KERNEL);
cfg80211_put_bss(ar->wiphy, bss); cfg80211_put_bss(ar->wiphy, bss);
} else if (vif->sme_state == SME_CONNECTED) { } else if (vif->sme_state == SME_CONNECTED) {
struct cfg80211_roam_info roam_info = {
.bss = bss,
.req_ie = assoc_req_ie,
.req_ie_len = assoc_req_len,
.resp_ie = assoc_resp_ie,
.resp_ie_len = assoc_resp_len,
};
/* inform roam event to cfg80211 */ /* inform roam event to cfg80211 */
cfg80211_roamed_bss(vif->ndev, bss, assoc_req_ie, assoc_req_len, cfg80211_roamed(vif->ndev, &roam_info, GFP_KERNEL);
assoc_resp_ie, assoc_resp_len, GFP_KERNEL);
} }
} }

View File

@ -5359,6 +5359,7 @@ brcmf_bss_roaming_done(struct brcmf_cfg80211_info *cfg,
struct ieee80211_supported_band *band; struct ieee80211_supported_band *band;
struct brcmf_bss_info_le *bi; struct brcmf_bss_info_le *bi;
struct brcmu_chan ch; struct brcmu_chan ch;
struct cfg80211_roam_info roam_info = {};
u32 freq; u32 freq;
s32 err = 0; s32 err = 0;
u8 *buf; u8 *buf;
@ -5397,9 +5398,15 @@ brcmf_bss_roaming_done(struct brcmf_cfg80211_info *cfg,
done: done:
kfree(buf); kfree(buf);
cfg80211_roamed(ndev, notify_channel, (u8 *)profile->bssid,
conn_info->req_ie, conn_info->req_ie_len, roam_info.channel = notify_channel;
conn_info->resp_ie, conn_info->resp_ie_len, GFP_KERNEL); roam_info.bssid = profile->bssid;
roam_info.req_ie = conn_info->req_ie;
roam_info.req_ie_len = conn_info->req_ie_len;
roam_info.resp_ie = conn_info->resp_ie;
roam_info.resp_ie_len = conn_info->resp_ie_len;
cfg80211_roamed(ndev, &roam_info, GFP_KERNEL);
brcmf_dbg(CONN, "Report roaming result\n"); brcmf_dbg(CONN, "Report roaming result\n");
set_bit(BRCMF_VIF_STATUS_CONNECTED, &ifp->vif->sme_state); set_bit(BRCMF_VIF_STATUS_CONNECTED, &ifp->vif->sme_state);

View File

@ -2830,15 +2830,22 @@ static void rndis_wlan_do_link_up_work(struct usbnet *usbdev)
} }
if (priv->infra_mode == NDIS_80211_INFRA_INFRA) { if (priv->infra_mode == NDIS_80211_INFRA_INFRA) {
if (!roamed) if (!roamed) {
cfg80211_connect_result(usbdev->net, bssid, req_ie, cfg80211_connect_result(usbdev->net, bssid, req_ie,
req_ie_len, resp_ie, req_ie_len, resp_ie,
resp_ie_len, 0, GFP_KERNEL); resp_ie_len, 0, GFP_KERNEL);
else } else {
cfg80211_roamed(usbdev->net, struct cfg80211_roam_info roam_info = {
get_current_channel(usbdev, NULL), .channel = get_current_channel(usbdev, NULL),
bssid, req_ie, req_ie_len, .bssid = bssid,
resp_ie, resp_ie_len, GFP_KERNEL); .req_ie = req_ie,
.req_ie_len = req_ie_len,
.resp_ie = resp_ie,
.resp_ie_len = resp_ie_len,
};
cfg80211_roamed(usbdev->net, &roam_info, GFP_KERNEL);
}
} else if (priv->infra_mode == NDIS_80211_INFRA_ADHOC) } else if (priv->infra_mode == NDIS_80211_INFRA_ADHOC)
cfg80211_ibss_joined(usbdev->net, bssid, cfg80211_ibss_joined(usbdev->net, bssid,
get_current_channel(usbdev, NULL), get_current_channel(usbdev, NULL),

View File

@ -666,8 +666,11 @@ void prism2_disconnected(struct wlandevice *wlandev)
void prism2_roamed(struct wlandevice *wlandev) void prism2_roamed(struct wlandevice *wlandev)
{ {
cfg80211_roamed(wlandev->netdev, NULL, wlandev->bssid, struct cfg80211_roam_info roam_info = {
NULL, 0, NULL, 0, GFP_KERNEL); .bssid = wlandev->bssid,
};
cfg80211_roamed(wlandev->netdev, &roam_info, GFP_KERNEL);
} }
/* Structures for declaring wiphy interface */ /* Structures for declaring wiphy interface */

View File

@ -2686,8 +2686,7 @@ struct cfg80211_nan_func {
* indication of requesting reassociation. * indication of requesting reassociation.
* In both the driver-initiated and new connect() call initiated roaming * In both the driver-initiated and new connect() call initiated roaming
* cases, the result of roaming is indicated with a call to * cases, the result of roaming is indicated with a call to
* cfg80211_roamed() or cfg80211_roamed_bss(). * cfg80211_roamed(). (invoked with the wireless_dev mutex held)
* (invoked with the wireless_dev mutex held)
* @update_connect_params: Update the connect parameters while connected to a * @update_connect_params: Update the connect parameters while connected to a
* BSS. The updated parameters can be used by driver/firmware for * BSS. The updated parameters can be used by driver/firmware for
* subsequent BSS selection (roaming) decisions and to form the * subsequent BSS selection (roaming) decisions and to form the
@ -5389,52 +5388,47 @@ cfg80211_connect_timeout(struct net_device *dev, const u8 *bssid,
gfp, timeout_reason); gfp, timeout_reason);
} }
/**
* struct cfg80211_roam_info - driver initiated roaming information
*
* @channel: the channel of the new AP
* @bss: entry of bss to which STA got roamed (may be %NULL if %bssid is set)
* @bssid: the BSSID of the new AP (may be %NULL if %bss is set)
* @req_ie: association request IEs (maybe be %NULL)
* @req_ie_len: association request IEs length
* @resp_ie: association response IEs (may be %NULL)
* @resp_ie_len: assoc response IEs length
*/
struct cfg80211_roam_info {
struct ieee80211_channel *channel;
struct cfg80211_bss *bss;
const u8 *bssid;
const u8 *req_ie;
size_t req_ie_len;
const u8 *resp_ie;
size_t resp_ie_len;
};
/** /**
* cfg80211_roamed - notify cfg80211 of roaming * cfg80211_roamed - notify cfg80211 of roaming
* *
* @dev: network device * @dev: network device
* @channel: the channel of the new AP * @info: information about the new BSS. struct &cfg80211_roam_info.
* @bssid: the BSSID of the new AP
* @req_ie: association request IEs (maybe be %NULL)
* @req_ie_len: association request IEs length
* @resp_ie: association response IEs (may be %NULL)
* @resp_ie_len: assoc response IEs length
* @gfp: allocation flags * @gfp: allocation flags
* *
* It should be called by the underlying driver whenever it roamed * This function may be called with the driver passing either the BSSID of the
* from one AP to another while connected. * new AP or passing the bss entry to avoid a race in timeout of the bss entry.
*/ * It should be called by the underlying driver whenever it roamed from one AP
void cfg80211_roamed(struct net_device *dev, * to another while connected. Drivers which have roaming implemented in
struct ieee80211_channel *channel, * firmware should pass the bss entry to avoid a race in bss entry timeout where
const u8 *bssid, * the bss entry of the new AP is seen in the driver, but gets timed out by the
const u8 *req_ie, size_t req_ie_len, * time it is accessed in __cfg80211_roamed() due to delay in scheduling
const u8 *resp_ie, size_t resp_ie_len, gfp_t gfp);
/**
* cfg80211_roamed_bss - notify cfg80211 of roaming
*
* @dev: network device
* @bss: entry of bss to which STA got roamed
* @req_ie: association request IEs (maybe be %NULL)
* @req_ie_len: association request IEs length
* @resp_ie: association response IEs (may be %NULL)
* @resp_ie_len: assoc response IEs length
* @gfp: allocation flags
*
* This is just a wrapper to notify cfg80211 of roaming event with driver
* passing bss to avoid a race in timeout of the bss entry. It should be
* called by the underlying driver whenever it roamed from one AP to another
* while connected. Drivers which have roaming implemented in firmware
* may use this function to avoid a race in bss entry timeout where the bss
* entry of the new AP is seen in the driver, but gets timed out by the time
* it is accessed in __cfg80211_roamed() due to delay in scheduling
* rdev->event_work. In case of any failures, the reference is released * rdev->event_work. In case of any failures, the reference is released
* either in cfg80211_roamed_bss() or in __cfg80211_romed(), Otherwise, * either in cfg80211_roamed() or in __cfg80211_romed(), Otherwise, it will be
* it will be released while diconneting from the current bss. * released while diconneting from the current bss.
*/ */
void cfg80211_roamed_bss(struct net_device *dev, struct cfg80211_bss *bss, void cfg80211_roamed(struct net_device *dev, struct cfg80211_roam_info *info,
const u8 *req_ie, size_t req_ie_len, gfp_t gfp);
const u8 *resp_ie, size_t resp_ie_len, gfp_t gfp);
/** /**
* cfg80211_disconnected - notify cfg80211 that connection was dropped * cfg80211_disconnected - notify cfg80211 that connection was dropped

View File

@ -224,13 +224,7 @@ struct cfg80211_event {
union { union {
struct cfg80211_connect_resp_params cr; struct cfg80211_connect_resp_params cr;
struct { struct cfg80211_roam_info rm;
const u8 *req_ie;
const u8 *resp_ie;
size_t req_ie_len;
size_t resp_ie_len;
struct cfg80211_bss *bss;
} rm;
struct { struct {
const u8 *ie; const u8 *ie;
size_t ie_len; size_t ie_len;
@ -390,9 +384,7 @@ int cfg80211_disconnect(struct cfg80211_registered_device *rdev,
struct net_device *dev, u16 reason, struct net_device *dev, u16 reason,
bool wextev); bool wextev);
void __cfg80211_roamed(struct wireless_dev *wdev, void __cfg80211_roamed(struct wireless_dev *wdev,
struct cfg80211_bss *bss, struct cfg80211_roam_info *info);
const u8 *req_ie, size_t req_ie_len,
const u8 *resp_ie, size_t resp_ie_len);
int cfg80211_mgd_wext_connect(struct cfg80211_registered_device *rdev, int cfg80211_mgd_wext_connect(struct cfg80211_registered_device *rdev,
struct wireless_dev *wdev); struct wireless_dev *wdev);
void cfg80211_autodisconnect_wk(struct work_struct *work); void cfg80211_autodisconnect_wk(struct work_struct *work);

View File

@ -13646,14 +13646,14 @@ void nl80211_send_connect_result(struct cfg80211_registered_device *rdev,
} }
void nl80211_send_roamed(struct cfg80211_registered_device *rdev, void nl80211_send_roamed(struct cfg80211_registered_device *rdev,
struct net_device *netdev, const u8 *bssid, struct net_device *netdev,
const u8 *req_ie, size_t req_ie_len, struct cfg80211_roam_info *info, gfp_t gfp)
const u8 *resp_ie, size_t resp_ie_len, gfp_t gfp)
{ {
struct sk_buff *msg; struct sk_buff *msg;
void *hdr; void *hdr;
const u8 *bssid = info->bss ? info->bss->bssid : info->bssid;
msg = nlmsg_new(100 + req_ie_len + resp_ie_len, gfp); msg = nlmsg_new(100 + info->req_ie_len + info->resp_ie_len, gfp);
if (!msg) if (!msg)
return; return;
@ -13666,10 +13666,12 @@ void nl80211_send_roamed(struct cfg80211_registered_device *rdev,
if (nla_put_u32(msg, NL80211_ATTR_WIPHY, rdev->wiphy_idx) || if (nla_put_u32(msg, NL80211_ATTR_WIPHY, rdev->wiphy_idx) ||
nla_put_u32(msg, NL80211_ATTR_IFINDEX, netdev->ifindex) || nla_put_u32(msg, NL80211_ATTR_IFINDEX, netdev->ifindex) ||
nla_put(msg, NL80211_ATTR_MAC, ETH_ALEN, bssid) || nla_put(msg, NL80211_ATTR_MAC, ETH_ALEN, bssid) ||
(req_ie && (info->req_ie &&
nla_put(msg, NL80211_ATTR_REQ_IE, req_ie_len, req_ie)) || nla_put(msg, NL80211_ATTR_REQ_IE, info->req_ie_len,
(resp_ie && info->req_ie)) ||
nla_put(msg, NL80211_ATTR_RESP_IE, resp_ie_len, resp_ie))) (info->resp_ie &&
nla_put(msg, NL80211_ATTR_RESP_IE, info->resp_ie_len,
info->resp_ie)))
goto nla_put_failure; goto nla_put_failure;
genlmsg_end(msg, hdr); genlmsg_end(msg, hdr);

View File

@ -56,9 +56,8 @@ void nl80211_send_connect_result(struct cfg80211_registered_device *rdev,
struct cfg80211_connect_resp_params *params, struct cfg80211_connect_resp_params *params,
gfp_t gfp); gfp_t gfp);
void nl80211_send_roamed(struct cfg80211_registered_device *rdev, void nl80211_send_roamed(struct cfg80211_registered_device *rdev,
struct net_device *netdev, const u8 *bssid, struct net_device *netdev,
const u8 *req_ie, size_t req_ie_len, struct cfg80211_roam_info *info, gfp_t gfp);
const u8 *resp_ie, size_t resp_ie_len, gfp_t gfp);
void nl80211_send_disconnected(struct cfg80211_registered_device *rdev, void nl80211_send_disconnected(struct cfg80211_registered_device *rdev,
struct net_device *netdev, u16 reason, struct net_device *netdev, u16 reason,
const u8 *ie, size_t ie_len, bool from_ap); const u8 *ie, size_t ie_len, bool from_ap);

View File

@ -5,6 +5,7 @@
* *
* Copyright 2009 Johannes Berg <johannes@sipsolutions.net> * Copyright 2009 Johannes Berg <johannes@sipsolutions.net>
* Copyright (C) 2009 Intel Corporation. All rights reserved. * Copyright (C) 2009 Intel Corporation. All rights reserved.
* Copyright 2017 Intel Deutschland GmbH
*/ */
#include <linux/etherdevice.h> #include <linux/etherdevice.h>
@ -870,9 +871,7 @@ EXPORT_SYMBOL(cfg80211_connect_done);
/* Consumes bss object one way or another */ /* Consumes bss object one way or another */
void __cfg80211_roamed(struct wireless_dev *wdev, void __cfg80211_roamed(struct wireless_dev *wdev,
struct cfg80211_bss *bss, struct cfg80211_roam_info *info)
const u8 *req_ie, size_t req_ie_len,
const u8 *resp_ie, size_t resp_ie_len)
{ {
#ifdef CONFIG_CFG80211_WEXT #ifdef CONFIG_CFG80211_WEXT
union iwreq_data wrqu; union iwreq_data wrqu;
@ -890,97 +889,84 @@ void __cfg80211_roamed(struct wireless_dev *wdev,
cfg80211_put_bss(wdev->wiphy, &wdev->current_bss->pub); cfg80211_put_bss(wdev->wiphy, &wdev->current_bss->pub);
wdev->current_bss = NULL; wdev->current_bss = NULL;
cfg80211_hold_bss(bss_from_pub(bss)); if (WARN_ON(!info->bss))
wdev->current_bss = bss_from_pub(bss); return;
cfg80211_hold_bss(bss_from_pub(info->bss));
wdev->current_bss = bss_from_pub(info->bss);
nl80211_send_roamed(wiphy_to_rdev(wdev->wiphy), nl80211_send_roamed(wiphy_to_rdev(wdev->wiphy),
wdev->netdev, bss->bssid, wdev->netdev, info, GFP_KERNEL);
req_ie, req_ie_len, resp_ie, resp_ie_len,
GFP_KERNEL);
#ifdef CONFIG_CFG80211_WEXT #ifdef CONFIG_CFG80211_WEXT
if (req_ie) { if (info->req_ie) {
memset(&wrqu, 0, sizeof(wrqu)); memset(&wrqu, 0, sizeof(wrqu));
wrqu.data.length = req_ie_len; wrqu.data.length = info->req_ie_len;
wireless_send_event(wdev->netdev, IWEVASSOCREQIE, wireless_send_event(wdev->netdev, IWEVASSOCREQIE,
&wrqu, req_ie); &wrqu, info->req_ie);
} }
if (resp_ie) { if (info->resp_ie) {
memset(&wrqu, 0, sizeof(wrqu)); memset(&wrqu, 0, sizeof(wrqu));
wrqu.data.length = resp_ie_len; wrqu.data.length = info->resp_ie_len;
wireless_send_event(wdev->netdev, IWEVASSOCRESPIE, wireless_send_event(wdev->netdev, IWEVASSOCRESPIE,
&wrqu, resp_ie); &wrqu, info->resp_ie);
} }
memset(&wrqu, 0, sizeof(wrqu)); memset(&wrqu, 0, sizeof(wrqu));
wrqu.ap_addr.sa_family = ARPHRD_ETHER; wrqu.ap_addr.sa_family = ARPHRD_ETHER;
memcpy(wrqu.ap_addr.sa_data, bss->bssid, ETH_ALEN); memcpy(wrqu.ap_addr.sa_data, info->bss->bssid, ETH_ALEN);
memcpy(wdev->wext.prev_bssid, bss->bssid, ETH_ALEN); memcpy(wdev->wext.prev_bssid, info->bss->bssid, ETH_ALEN);
wdev->wext.prev_bssid_valid = true; wdev->wext.prev_bssid_valid = true;
wireless_send_event(wdev->netdev, SIOCGIWAP, &wrqu, NULL); wireless_send_event(wdev->netdev, SIOCGIWAP, &wrqu, NULL);
#endif #endif
return; return;
out: out:
cfg80211_put_bss(wdev->wiphy, bss); cfg80211_put_bss(wdev->wiphy, info->bss);
} }
void cfg80211_roamed(struct net_device *dev, /* Consumes info->bss object one way or another */
struct ieee80211_channel *channel, void cfg80211_roamed(struct net_device *dev, struct cfg80211_roam_info *info,
const u8 *bssid, gfp_t gfp)
const u8 *req_ie, size_t req_ie_len,
const u8 *resp_ie, size_t resp_ie_len, gfp_t gfp)
{
struct wireless_dev *wdev = dev->ieee80211_ptr;
struct cfg80211_bss *bss;
bss = cfg80211_get_bss(wdev->wiphy, channel, bssid, wdev->ssid,
wdev->ssid_len,
wdev->conn_bss_type, IEEE80211_PRIVACY_ANY);
if (WARN_ON(!bss))
return;
cfg80211_roamed_bss(dev, bss, req_ie, req_ie_len, resp_ie,
resp_ie_len, gfp);
}
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,
size_t resp_ie_len, gfp_t gfp)
{ {
struct wireless_dev *wdev = dev->ieee80211_ptr; struct wireless_dev *wdev = dev->ieee80211_ptr;
struct cfg80211_registered_device *rdev = wiphy_to_rdev(wdev->wiphy); struct cfg80211_registered_device *rdev = wiphy_to_rdev(wdev->wiphy);
struct cfg80211_event *ev; struct cfg80211_event *ev;
unsigned long flags; unsigned long flags;
if (WARN_ON(!bss)) if (!info->bss) {
info->bss = cfg80211_get_bss(wdev->wiphy, info->channel,
info->bssid, wdev->ssid,
wdev->ssid_len,
wdev->conn_bss_type,
IEEE80211_PRIVACY_ANY);
}
if (WARN_ON(!info->bss))
return; return;
ev = kzalloc(sizeof(*ev) + req_ie_len + resp_ie_len, gfp); ev = kzalloc(sizeof(*ev) + info->req_ie_len + info->resp_ie_len, gfp);
if (!ev) { if (!ev) {
cfg80211_put_bss(wdev->wiphy, bss); cfg80211_put_bss(wdev->wiphy, info->bss);
return; return;
} }
ev->type = EVENT_ROAMED; ev->type = EVENT_ROAMED;
ev->rm.req_ie = ((u8 *)ev) + sizeof(*ev); ev->rm.req_ie = ((u8 *)ev) + sizeof(*ev);
ev->rm.req_ie_len = req_ie_len; ev->rm.req_ie_len = info->req_ie_len;
memcpy((void *)ev->rm.req_ie, req_ie, req_ie_len); memcpy((void *)ev->rm.req_ie, info->req_ie, info->req_ie_len);
ev->rm.resp_ie = ((u8 *)ev) + sizeof(*ev) + req_ie_len; ev->rm.resp_ie = ((u8 *)ev) + sizeof(*ev) + info->req_ie_len;
ev->rm.resp_ie_len = resp_ie_len; ev->rm.resp_ie_len = info->resp_ie_len;
memcpy((void *)ev->rm.resp_ie, resp_ie, resp_ie_len); memcpy((void *)ev->rm.resp_ie, info->resp_ie, info->resp_ie_len);
ev->rm.bss = bss; ev->rm.bss = info->bss;
spin_lock_irqsave(&wdev->event_lock, flags); spin_lock_irqsave(&wdev->event_lock, flags);
list_add_tail(&ev->list, &wdev->event_list); list_add_tail(&ev->list, &wdev->event_list);
spin_unlock_irqrestore(&wdev->event_lock, flags); spin_unlock_irqrestore(&wdev->event_lock, flags);
queue_work(cfg80211_wq, &rdev->event_work); queue_work(cfg80211_wq, &rdev->event_work);
} }
EXPORT_SYMBOL(cfg80211_roamed_bss); EXPORT_SYMBOL(cfg80211_roamed);
void __cfg80211_disconnected(struct net_device *dev, const u8 *ie, void __cfg80211_disconnected(struct net_device *dev, const u8 *ie,
size_t ie_len, u16 reason, bool from_ap) size_t ie_len, u16 reason, bool from_ap)

View File

@ -946,9 +946,7 @@ void cfg80211_process_wdev_events(struct wireless_dev *wdev)
ev->cr.status == WLAN_STATUS_SUCCESS); ev->cr.status == WLAN_STATUS_SUCCESS);
break; break;
case EVENT_ROAMED: case EVENT_ROAMED:
__cfg80211_roamed(wdev, ev->rm.bss, ev->rm.req_ie, __cfg80211_roamed(wdev, &ev->rm);
ev->rm.req_ie_len, ev->rm.resp_ie,
ev->rm.resp_ie_len);
break; break;
case EVENT_DISCONNECTED: case EVENT_DISCONNECTED:
__cfg80211_disconnected(wdev->netdev, __cfg80211_disconnected(wdev->netdev,