On Fri, Apr 26, 2019 at 3:44 PM Johannes Berg <johannes@xxxxxxxxxxxxxxxx> wrote: > > On Fri, 2019-04-26 at 15:24 +0530, chaitanya.mgit@xxxxxxxxx wrote: > > From: Chaitanya Tata <chaitanya.tata@xxxxxxxxxxxxxxxxx> > > > > If the BSS is expired during connection, the connect result will > > trigger a kernel warning. Ideally cfg80211 should hold the BSS > > before the connection is attempted, but as the BSSID is not known > > in case of auth/assoc MLME offload (connect op) it doesn't. > > > > For those drivers without the connect op cfg80211 holds down the > > reference so it wil not be removed from list. > > > > Fix this by removing the warning and silently adding the BSS back to > > the bss list which is return by the driver (with proper BSSID set). > > The requirements for drivers are documented in the API's. > > This looks good, mostly :-) > > > * @bssid: The BSSID of the AP (may be %NULL) > > * @bss: Entry of bss to which STA got connected to, can be obtained through > > - * cfg80211_get_bss() (may be %NULL). Only one parameter among @bssid and > > - * @bss needs to be specified. > > + * cfg80211_get_bss() (may be %NULL) but it is recommended to store the > > + * bss from the connect_request and hold a reference to it and return > > + * through this param to avoid warning if the bss is expired during the > > + * connection, esp. for those drivers implementing connect op. > > + * Only one parameter among @bssid and @bss needs to be specified. > > So while we're at it, we should probably amend the documentation to say > that the reference to the BSS passes from the driver to cfg80211, right? Yes > > > + /* bss is not NULL, so even though bss might have expired, the driver > > + * is still holding a reference to it. > > + */ > > is -> was? It's our reference now. Yes > > > if (params->bss) { > > - /* Make sure the bss entry provided by the driver is valid. */ > > struct cfg80211_internal_bss *ibss = bss_from_pub(params->bss); > > > > - if (WARN_ON(list_empty(&ibss->list))) { > > - cfg80211_put_bss(wdev->wiphy, params->bss); > > That's why we had a put_bss() here. I will add a change to put bss after we update. > > > + /* Meanwhile if BSS is expired then add it back to the list as > > + * we have just connected with it. > > + */ > > + if (list_empty(&ibss->list)) > > + cfg80211_bss_update(rdev, ibss, ibss->pub.signal); > > But I think this adds *another* reference, which is wrong? We do need > one reference (the original one the driver gave us) but not a second > one? This was assuming found will be false so refcnt will be reset and a fresh ref is taken. > Also, not sure the last argument (signal_valid) is right? > > Basically all that really just means that cfg80211_bss_update() is > probably not the right function to call. It also updates the timestamp, > which is not true, we haven't received updated information we just used > it and thus held on to it. > > I think we should probably just instead of a "cfg80211_bss_relink()" > function exposed, and that just does something like > > list_add_tail(&new->list, &rdev->bss_list); > rdev->bss_entries++; > rb_insert_bss(rdev, new); > > rdev->bss_generation++; > > though I'm not - off the top of my head - sure about > cfg80211_combine_bsses() being needed or not. > > Maybe the copy we do in cfg80211_bss_update() isn't so bad, but then I'd > probably expose it without the signal_valid part and signal/timestamp > updates, and just copy the information raw there? However, the "if > (found)" part should never be possible here, right? Actually, maybe it > is, if we got a *new* entry in the meantime, but then we'd override the > newer information with the older, which is kinda bad? If we get new information (scan during connection) the bss would not have expired right? so this code will not be triggered. My initial stab was in similar lines, something like below: But as the bss is expired we need to udpate ts, signal and other fields anyway, so I went ahead with full update. +void cfg80211_add_expired_bss(struct cfg80211_registered_device *rdev, + struct cfg80211_internal_bss *new) +{ + if (!new) + return; + + spin_lock_bh(&rdev->bss_lock); + new->ts = jiffies + list_add_tail(&new->list, &rdev->bss_list); + rdev->bss_entries++; + + rb_insert_bss(rdev, new); + rdev->bss_generation++; + + bss_ref_get(rdev, new); + + spin_unlock_bh(&rdev->bss_lock); +} > johannes > -- Thanks, Regards, Chaitanya T K.