Commit 42596f76 authored by Arend Van Spriel's avatar Arend Van Spriel Committed by Kalle Valo

brcmfmac: add mutex to protect pno requests

The request references kept in pno are accessed in user-space context
and in firmware event handler context. As such we need to protect it
with a lock. As both context allow sleep a mutex seems appropriate.
Reviewed-by: default avatarHante Meuleman <hante.meuleman@broadcom.com>
Reviewed-by: default avatarPieter-Paul Giesberts <pieter-paul.giesberts@broadcom.com>
Reviewed-by: default avatarFranky Lin <franky.lin@broadcom.com>
Signed-off-by: default avatarArend van Spriel <arend.vanspriel@broadcom.com>
Signed-off-by: default avatarKalle Valo <kvalo@codeaurora.org>
parent efc2c1fa
...@@ -43,6 +43,7 @@ ...@@ -43,6 +43,7 @@
struct brcmf_pno_info { struct brcmf_pno_info {
int n_reqs; int n_reqs;
struct cfg80211_sched_scan_request *reqs[BRCMF_PNO_MAX_BUCKETS]; struct cfg80211_sched_scan_request *reqs[BRCMF_PNO_MAX_BUCKETS];
struct mutex req_lock;
}; };
#define ifp_to_pno(_ifp) ((_ifp)->drvr->config->pno) #define ifp_to_pno(_ifp) ((_ifp)->drvr->config->pno)
...@@ -55,13 +56,17 @@ static int brcmf_pno_store_request(struct brcmf_pno_info *pi, ...@@ -55,13 +56,17 @@ static int brcmf_pno_store_request(struct brcmf_pno_info *pi,
return -ENOSPC; return -ENOSPC;
brcmf_dbg(SCAN, "reqid=%llu\n", req->reqid); brcmf_dbg(SCAN, "reqid=%llu\n", req->reqid);
mutex_lock(&pi->req_lock);
pi->reqs[pi->n_reqs++] = req; pi->reqs[pi->n_reqs++] = req;
mutex_unlock(&pi->req_lock);
return 0; return 0;
} }
static int brcmf_pno_remove_request(struct brcmf_pno_info *pi, u64 reqid) static int brcmf_pno_remove_request(struct brcmf_pno_info *pi, u64 reqid)
{ {
int i; int i, err = 0;
mutex_lock(&pi->req_lock);
/* find request */ /* find request */
for (i = 0; i < pi->n_reqs; i++) { for (i = 0; i < pi->n_reqs; i++) {
...@@ -70,7 +75,8 @@ static int brcmf_pno_remove_request(struct brcmf_pno_info *pi, u64 reqid) ...@@ -70,7 +75,8 @@ static int brcmf_pno_remove_request(struct brcmf_pno_info *pi, u64 reqid)
} }
/* request not found */ /* request not found */
if (WARN(i == pi->n_reqs, "reqid not found\n")) { if (WARN(i == pi->n_reqs, "reqid not found\n")) {
return -ENOENT; err = -ENOENT;
goto done;
} }
brcmf_dbg(SCAN, "reqid=%llu\n", reqid); brcmf_dbg(SCAN, "reqid=%llu\n", reqid);
...@@ -78,14 +84,17 @@ static int brcmf_pno_remove_request(struct brcmf_pno_info *pi, u64 reqid) ...@@ -78,14 +84,17 @@ static int brcmf_pno_remove_request(struct brcmf_pno_info *pi, u64 reqid)
/* if last we are done */ /* if last we are done */
if (!pi->n_reqs || i == pi->n_reqs) if (!pi->n_reqs || i == pi->n_reqs)
return 0; goto done;
/* fill the gap with remaining requests */ /* fill the gap with remaining requests */
while (i <= pi->n_reqs - 1) { while (i <= pi->n_reqs - 1) {
pi->reqs[i] = pi->reqs[i + 1]; pi->reqs[i] = pi->reqs[i + 1];
i++; i++;
} }
return 0;
done:
mutex_unlock(&pi->req_lock);
return err;
} }
static int brcmf_pno_channel_config(struct brcmf_if *ifp, static int brcmf_pno_channel_config(struct brcmf_if *ifp,
...@@ -488,6 +497,7 @@ int brcmf_pno_attach(struct brcmf_cfg80211_info *cfg) ...@@ -488,6 +497,7 @@ int brcmf_pno_attach(struct brcmf_cfg80211_info *cfg)
return -ENOMEM; return -ENOMEM;
cfg->pno = pi; cfg->pno = pi;
mutex_init(&pi->req_lock);
return 0; return 0;
} }
...@@ -500,6 +510,7 @@ void brcmf_pno_detach(struct brcmf_cfg80211_info *cfg) ...@@ -500,6 +510,7 @@ void brcmf_pno_detach(struct brcmf_cfg80211_info *cfg)
cfg->pno = NULL; cfg->pno = NULL;
WARN_ON(pi->n_reqs); WARN_ON(pi->n_reqs);
mutex_destroy(&pi->req_lock);
kfree(pi); kfree(pi);
} }
...@@ -515,11 +526,15 @@ void brcmf_pno_wiphy_params(struct wiphy *wiphy, bool gscan) ...@@ -515,11 +526,15 @@ void brcmf_pno_wiphy_params(struct wiphy *wiphy, bool gscan)
u64 brcmf_pno_find_reqid_by_bucket(struct brcmf_pno_info *pi, u32 bucket) u64 brcmf_pno_find_reqid_by_bucket(struct brcmf_pno_info *pi, u32 bucket)
{ {
/* bucket appears to be gone */ u64 reqid = 0;
if (bucket >= pi->n_reqs)
return 0; mutex_lock(&pi->req_lock);
if (bucket < pi->n_reqs)
reqid = pi->reqs[bucket]->reqid;
return pi->reqs[bucket]->reqid; mutex_unlock(&pi->req_lock);
return reqid;
} }
u32 brcmf_pno_get_bucket_map(struct brcmf_pno_info *pi, u32 brcmf_pno_get_bucket_map(struct brcmf_pno_info *pi,
...@@ -530,6 +545,7 @@ u32 brcmf_pno_get_bucket_map(struct brcmf_pno_info *pi, ...@@ -530,6 +545,7 @@ u32 brcmf_pno_get_bucket_map(struct brcmf_pno_info *pi,
u32 bucket_map = 0; u32 bucket_map = 0;
int i, j; int i, j;
mutex_lock(&pi->req_lock);
for (i = 0; i < pi->n_reqs; i++) { for (i = 0; i < pi->n_reqs; i++) {
req = pi->reqs[i]; req = pi->reqs[i];
...@@ -544,5 +560,6 @@ u32 brcmf_pno_get_bucket_map(struct brcmf_pno_info *pi, ...@@ -544,5 +560,6 @@ u32 brcmf_pno_get_bucket_map(struct brcmf_pno_info *pi,
} }
} }
} }
mutex_unlock(&pi->req_lock);
return bucket_map; return bucket_map;
} }
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment