Track CRDA request and handle timeout when no answer from CRDA. This could happen when crda is not available yet (not mounted fs), eg. during OS startup and cfg80211 built-in to the kernel. Before we could stuck when processing last request without ability to process current and any further regulatory requests. Reported-by: Sander Eikelenboom <linux@xxxxxxxxxxxxxx> Tested-by: Sander Eikelenboom <linux@xxxxxxxxxxxxxx> Tested-by: Colleen Twitty <colleen@xxxxxxxxxxx> Signed-off-by: Janusz Dziedzic <janusz.dziedzic@xxxxxxxxx> --- net/wireless/reg.c | 45 ++++++++++++++++++++++++++++++++++----------- 1 file changed, 34 insertions(+), 11 deletions(-) diff --git a/net/wireless/reg.c b/net/wireless/reg.c index 220c4a2..a2d2719 100644 --- a/net/wireless/reg.c +++ b/net/wireless/reg.c @@ -512,6 +512,10 @@ reg_call_crda(struct regulatory_request *request) { if (call_crda(request->alpha2)) return REG_REQ_IGNORE; + + /* Setup timeout to check if CRDA is alive */ + queue_delayed_work(system_power_efficient_wq, + ®_timeout, msecs_to_jiffies(3142)); return REG_REQ_OK; } @@ -1542,8 +1546,7 @@ static void reg_set_request_processed(void) need_more_processing = true; spin_unlock(®_requests_lock); - if (lr->initiator == NL80211_REGDOM_SET_BY_USER) - cancel_delayed_work(®_timeout); + cancel_delayed_work(®_timeout); if (need_more_processing) schedule_work(®_work); @@ -1810,6 +1813,9 @@ static void reg_process_hint(struct regulatory_request *reg_request) struct wiphy *wiphy = NULL; enum reg_request_treatment treatment; + REG_DBG_PRINT("Process regulatory hint called by %s\n", + reg_initiator_name(reg_request->initiator)); + if (reg_request->wiphy_idx != WIPHY_IDX_INVALID) wiphy = wiphy_idx_to_wiphy(reg_request->wiphy_idx); @@ -1818,12 +1824,7 @@ static void reg_process_hint(struct regulatory_request *reg_request) reg_process_hint_core(reg_request); return; case NL80211_REGDOM_SET_BY_USER: - treatment = reg_process_hint_user(reg_request); - if (treatment == REG_REQ_IGNORE || - treatment == REG_REQ_ALREADY_SET) - return; - queue_delayed_work(system_power_efficient_wq, - ®_timeout, msecs_to_jiffies(3142)); + reg_process_hint_user(reg_request); return; case NL80211_REGDOM_SET_BY_DRIVER: if (!wiphy) @@ -1864,7 +1865,8 @@ static void reg_process_pending_hints(void) /* When last_request->processed becomes true this will be rescheduled */ if (lr && !lr->processed) { - REG_DBG_PRINT("Pending regulatory request, waiting for it to be processed...\n"); + REG_DBG_PRINT("Pending %s regulatory request, waiting for it to be processed...\n", + reg_initiator_name(lr->initiator)); return; } @@ -2615,9 +2617,30 @@ void wiphy_regulatory_deregister(struct wiphy *wiphy) static void reg_timeout_work(struct work_struct *work) { - REG_DBG_PRINT("Timeout while waiting for CRDA to reply, restoring regulatory settings\n"); + struct regulatory_request *lr; + rtnl_lock(); - restore_regulatory_settings(true); + + lr = get_last_request(); + REG_DBG_PRINT("Timeout while waiting for CRDA to reply %s request, restoring regulatory settings\n", + reg_initiator_name(lr->initiator)); + + switch (lr->initiator) { + case NL80211_REGDOM_SET_BY_CORE: + case NL80211_REGDOM_SET_BY_DRIVER: + /* Call CRDA again for last request */ + reg_process_hint(lr); + break; + case NL80211_REGDOM_SET_BY_USER: + restore_regulatory_settings(true); + break; + case NL80211_REGDOM_SET_BY_COUNTRY_IE: + restore_regulatory_settings(false); + break; + default: + WARN_ON(1); + break; + } rtnl_unlock(); } -- 1.7.9.5 -- To unsubscribe from this list: send the line "unsubscribe linux-wireless" in the body of a message to majordomo@xxxxxxxxxxxxxxx More majordomo info at http://vger.kernel.org/majordomo-info.html