提交 3e2e86ab 编写于 作者: A Arend Van Spriel 提交者: Kalle Valo

brcmfmac: fix handling ssids in .sched_scan_start() callback

The ssids list in the scheduled scan request were not properly taken
into account when configuring in firmware. The hidden bit was set for
any ssid resulting in active scanning for all. Only set it for ssids
that are in the ssids list.
Reviewed-by: NHante Meuleman <hante.meuleman@broadcom.com>
Reviewed-by: NPieter-Paul Giesberts <pieter-paul.giesberts@broadcom.com>
Reviewed-by: NFranky Lin <franky.lin@broadcom.com>
Signed-off-by: NArend van Spriel <arend.vanspriel@broadcom.com>
Signed-off-by: NKalle Valo <kvalo@codeaurora.org>
上级 ac55136f
...@@ -3314,19 +3314,37 @@ brcmf_notify_sched_scan_results(struct brcmf_if *ifp, ...@@ -3314,19 +3314,37 @@ brcmf_notify_sched_scan_results(struct brcmf_if *ifp,
return err; return err;
} }
static bool brcmf_is_ssid_active(struct cfg80211_ssid *ssid,
struct cfg80211_sched_scan_request *req)
{
int i;
if (!ssid || !req->ssids || !req->n_ssids)
return false;
for (i = 0; i < req->n_ssids; i++) {
if (ssid->ssid_len == req->ssids[i].ssid_len) {
if (!strncmp(ssid->ssid, req->ssids[i].ssid,
ssid->ssid_len))
return true;
}
}
return false;
}
static int static int
brcmf_cfg80211_sched_scan_start(struct wiphy *wiphy, brcmf_cfg80211_sched_scan_start(struct wiphy *wiphy,
struct net_device *ndev, struct net_device *ndev,
struct cfg80211_sched_scan_request *request) struct cfg80211_sched_scan_request *req)
{ {
struct brcmf_if *ifp = netdev_priv(ndev); struct brcmf_if *ifp = netdev_priv(ndev);
struct brcmf_cfg80211_info *cfg = wiphy_priv(wiphy); struct brcmf_cfg80211_info *cfg = wiphy_priv(wiphy);
struct brcmf_pno_net_param_le pfn; struct cfg80211_ssid *ssid;
int i; int i;
int ret = 0; int ret = 0;
brcmf_dbg(SCAN, "Enter n_match_sets:%d n_ssids:%d\n", brcmf_dbg(SCAN, "Enter n_match_sets:%d n_ssids:%d\n",
request->n_match_sets, request->n_ssids); req->n_match_sets, req->n_ssids);
if (test_bit(BRCMF_SCAN_STATUS_BUSY, &cfg->scan_status)) { if (test_bit(BRCMF_SCAN_STATUS_BUSY, &cfg->scan_status)) {
brcmf_err("Scanning already: status (%lu)\n", cfg->scan_status); brcmf_err("Scanning already: status (%lu)\n", cfg->scan_status);
return -EAGAIN; return -EAGAIN;
...@@ -3337,71 +3355,46 @@ brcmf_cfg80211_sched_scan_start(struct wiphy *wiphy, ...@@ -3337,71 +3355,46 @@ brcmf_cfg80211_sched_scan_start(struct wiphy *wiphy,
return -EAGAIN; return -EAGAIN;
} }
if (!request->n_ssids || !request->n_match_sets) { if (req->n_match_sets <= 0) {
brcmf_dbg(SCAN, "Invalid sched scan req!! n_ssids:%d\n", brcmf_dbg(SCAN, "invalid number of matchsets specified: %d\n",
request->n_ssids); req->n_match_sets);
return -EINVAL; return -EINVAL;
} }
if (request->n_ssids > 0) { /* clean up everything */
for (i = 0; i < request->n_ssids; i++) { ret = brcmf_pno_clean(ifp);
/* Active scan req for ssids */ if (ret < 0) {
brcmf_dbg(SCAN, ">>> Active scan req for ssid (%s)\n", brcmf_err("failed error=%d\n", ret);
request->ssids[i].ssid); return ret;
/* match_set ssids is a supert set of n_ssid list,
* so we need not add these set separately.
*/
}
} }
if (request->n_match_sets > 0) { /* configure pno */
/* clean up everything */ ret = brcmf_pno_config(ifp, req);
ret = brcmf_pno_clean(ifp); if (ret < 0)
if (ret < 0) { return ret;
brcmf_err("failed error=%d\n", ret);
return ret;
}
/* configure pno */ /* configure each match set */
ret = brcmf_pno_config(ifp, request); for (i = 0; i < req->n_match_sets; i++) {
if (ret < 0)
return ret;
/* configure each match set */ ssid = &req->match_sets[i].ssid;
for (i = 0; i < request->n_match_sets; i++) {
struct cfg80211_ssid *ssid;
u32 ssid_len;
ssid = &request->match_sets[i].ssid; if (!ssid->ssid_len) {
ssid_len = ssid->ssid_len; brcmf_err("skip broadcast ssid\n");
continue;
}
if (!ssid_len) { ret = brcmf_pno_add_ssid(ifp, ssid,
brcmf_err("skip broadcast ssid\n"); brcmf_is_ssid_active(ssid, req));
continue; if (ret < 0)
}
pfn.auth = cpu_to_le32(WLAN_AUTH_OPEN);
pfn.wpa_auth = cpu_to_le32(BRCMF_PNO_WPA_AUTH_ANY);
pfn.wsec = cpu_to_le32(0);
pfn.infra = cpu_to_le32(1);
pfn.flags = cpu_to_le32(1 << BRCMF_PNO_HIDDEN_BIT);
pfn.ssid.SSID_len = cpu_to_le32(ssid_len);
memcpy(pfn.ssid.SSID, ssid->ssid, ssid_len);
ret = brcmf_fil_iovar_data_set(ifp, "pfn_add", &pfn,
sizeof(pfn));
brcmf_dbg(SCAN, ">>> PNO filter %s for ssid (%s)\n", brcmf_dbg(SCAN, ">>> PNO filter %s for ssid (%s)\n",
ret == 0 ? "set" : "failed", ssid->ssid); ret == 0 ? "set" : "failed", ssid->ssid);
}
/* Enable the PNO */
if (brcmf_fil_iovar_int_set(ifp, "pfn", 1) < 0) {
brcmf_err("PNO enable failed!! ret=%d\n", ret);
return -EINVAL;
}
} else {
return -EINVAL;
} }
/* Enable the PNO */
ret = brcmf_fil_iovar_int_set(ifp, "pfn", 1);
if (ret < 0)
brcmf_err("PNO enable failed!! ret=%d\n", ret);
return 0; return ret;
} }
static int brcmf_cfg80211_sched_scan_stop(struct wiphy *wiphy, static int brcmf_cfg80211_sched_scan_stop(struct wiphy *wiphy,
......
...@@ -28,6 +28,8 @@ ...@@ -28,6 +28,8 @@
#define BRCMF_PNO_FREQ_EXPO_MAX 3 #define BRCMF_PNO_FREQ_EXPO_MAX 3
#define BRCMF_PNO_ENABLE_ADAPTSCAN_BIT 6 #define BRCMF_PNO_ENABLE_ADAPTSCAN_BIT 6
#define BRCMF_PNO_SCAN_INCOMPLETE 0 #define BRCMF_PNO_SCAN_INCOMPLETE 0
#define BRCMF_PNO_WPA_AUTH_ANY 0xFFFFFFFF
#define BRCMF_PNO_HIDDEN_BIT 2
int brcmf_pno_clean(struct brcmf_if *ifp) int brcmf_pno_clean(struct brcmf_if *ifp)
{ {
...@@ -98,3 +100,19 @@ int brcmf_pno_config(struct brcmf_if *ifp, ...@@ -98,3 +100,19 @@ int brcmf_pno_config(struct brcmf_if *ifp,
return err; return err;
} }
int brcmf_pno_add_ssid(struct brcmf_if *ifp, struct cfg80211_ssid *ssid,
bool active)
{
struct brcmf_pno_net_param_le pfn;
pfn.auth = cpu_to_le32(WLAN_AUTH_OPEN);
pfn.wpa_auth = cpu_to_le32(BRCMF_PNO_WPA_AUTH_ANY);
pfn.wsec = cpu_to_le32(0);
pfn.infra = cpu_to_le32(1);
if (active)
pfn.flags = cpu_to_le32(1 << BRCMF_PNO_HIDDEN_BIT);
pfn.ssid.SSID_len = cpu_to_le32(ssid->ssid_len);
memcpy(pfn.ssid.SSID, ssid->ssid, ssid->ssid_len);
return brcmf_fil_iovar_data_set(ifp, "pfn_add", &pfn, sizeof(pfn));
}
...@@ -17,8 +17,6 @@ ...@@ -17,8 +17,6 @@
#define _BRCMF_PNO_H #define _BRCMF_PNO_H
#define BRCMF_PNO_SCAN_COMPLETE 1 #define BRCMF_PNO_SCAN_COMPLETE 1
#define BRCMF_PNO_WPA_AUTH_ANY 0xFFFFFFFF
#define BRCMF_PNO_HIDDEN_BIT 2
#define BRCMF_PNO_MAX_PFN_COUNT 16 #define BRCMF_PNO_MAX_PFN_COUNT 16
/** /**
...@@ -37,4 +35,14 @@ int brcmf_pno_clean(struct brcmf_if *ifp); ...@@ -37,4 +35,14 @@ int brcmf_pno_clean(struct brcmf_if *ifp);
int brcmf_pno_config(struct brcmf_if *ifp, int brcmf_pno_config(struct brcmf_if *ifp,
struct cfg80211_sched_scan_request *request); struct cfg80211_sched_scan_request *request);
/**
* brcmf_pno_add_ssid - add ssid for pno in firmware.
*
* @ifp: interface object used.
* @ssid: ssid information.
* @active: indicate this ssid needs to be actively probed.
*/
int brcmf_pno_add_ssid(struct brcmf_if *ifp, struct cfg80211_ssid *ssid,
bool active);
#endif /* _BRCMF_PNO_H */ #endif /* _BRCMF_PNO_H */
Markdown is supported
0% .
You are about to add 0 people to the discussion. Proceed with caution.
先完成此消息的编辑!
想要评论请 注册