Add patches to fix cfg80211 issues with suspend (rhbz 856863)
This commit is contained in:
parent
36ee79b33e
commit
b3c93b2626
219
cfg80211-mac80211-disconnect-on-suspend.patch
Normal file
219
cfg80211-mac80211-disconnect-on-suspend.patch
Normal file
@ -0,0 +1,219 @@
|
|||||||
|
From 8125696991194aacb1173b6e8196d19098b44e17 Mon Sep 17 00:00:00 2001
|
||||||
|
From: Stanislaw Gruszka <sgruszka@redhat.com>
|
||||||
|
Date: Thu, 28 Feb 2013 09:55:25 +0000
|
||||||
|
Subject: cfg80211/mac80211: disconnect on suspend
|
||||||
|
|
||||||
|
If possible that after suspend, cfg80211 will receive request to
|
||||||
|
disconnect what require action on interface that was removed during
|
||||||
|
suspend.
|
||||||
|
|
||||||
|
Problem can manifest itself by various warnings similar to below one:
|
||||||
|
|
||||||
|
WARNING: at net/mac80211/driver-ops.h:12 ieee80211_bss_info_change_notify+0x2f9/0x300 [mac80211]()
|
||||||
|
wlan0: Failed check-sdata-in-driver check, flags: 0x4
|
||||||
|
Call Trace:
|
||||||
|
[<c043e0b3>] warn_slowpath_fmt+0x33/0x40
|
||||||
|
[<f83707c9>] ieee80211_bss_info_change_notify+0x2f9/0x300 [mac80211]
|
||||||
|
[<f83a660a>] ieee80211_recalc_ps_vif+0x2a/0x30 [mac80211]
|
||||||
|
[<f83a6706>] ieee80211_set_disassoc+0xf6/0x500 [mac80211]
|
||||||
|
[<f83a9441>] ieee80211_mgd_deauth+0x1f1/0x280 [mac80211]
|
||||||
|
[<f8381b36>] ieee80211_deauth+0x16/0x20 [mac80211]
|
||||||
|
[<f8261e70>] cfg80211_mlme_down+0x70/0xc0 [cfg80211]
|
||||||
|
[<f8264de1>] __cfg80211_disconnect+0x1b1/0x1d0 [cfg80211]
|
||||||
|
|
||||||
|
To fix the problem disconnect from any associated network before
|
||||||
|
suspend. User space is responsible to establish connection again
|
||||||
|
after resume. This basically need to be done by user space anyway,
|
||||||
|
because associated stations can go away during suspend (for example
|
||||||
|
NetworkManager disconnects on suspend and connect on resume by default).
|
||||||
|
|
||||||
|
Patch also handle situation when driver refuse to suspend with wowlan
|
||||||
|
configured and try to suspend again without it.
|
||||||
|
|
||||||
|
Signed-off-by: Stanislaw Gruszka <sgruszka@redhat.com>
|
||||||
|
Signed-off-by: Johannes Berg <johannes.berg@intel.com>
|
||||||
|
---
|
||||||
|
diff --git a/net/mac80211/pm.c b/net/mac80211/pm.c
|
||||||
|
index d0275f3..4d105c7 100644
|
||||||
|
--- a/net/mac80211/pm.c
|
||||||
|
+++ b/net/mac80211/pm.c
|
||||||
|
@@ -93,7 +93,7 @@ int __ieee80211_suspend(struct ieee80211_hw *hw, struct cfg80211_wowlan *wowlan)
|
||||||
|
return err;
|
||||||
|
} else if (err > 0) {
|
||||||
|
WARN_ON(err != 1);
|
||||||
|
- local->wowlan = false;
|
||||||
|
+ return err;
|
||||||
|
} else {
|
||||||
|
list_for_each_entry(sdata, &local->interfaces, list)
|
||||||
|
if (ieee80211_sdata_running(sdata))
|
||||||
|
diff --git a/net/wireless/core.c b/net/wireless/core.c
|
||||||
|
index ea4155f..f382cae 100644
|
||||||
|
--- a/net/wireless/core.c
|
||||||
|
+++ b/net/wireless/core.c
|
||||||
|
@@ -814,6 +814,46 @@ void cfg80211_update_iface_num(struct cfg80211_registered_device *rdev,
|
||||||
|
rdev->num_running_monitor_ifaces += num;
|
||||||
|
}
|
||||||
|
|
||||||
|
+void cfg80211_leave(struct cfg80211_registered_device *rdev,
|
||||||
|
+ struct wireless_dev *wdev)
|
||||||
|
+{
|
||||||
|
+ struct net_device *dev = wdev->netdev;
|
||||||
|
+
|
||||||
|
+ switch (wdev->iftype) {
|
||||||
|
+ case NL80211_IFTYPE_ADHOC:
|
||||||
|
+ cfg80211_leave_ibss(rdev, dev, true);
|
||||||
|
+ break;
|
||||||
|
+ case NL80211_IFTYPE_P2P_CLIENT:
|
||||||
|
+ case NL80211_IFTYPE_STATION:
|
||||||
|
+ mutex_lock(&rdev->sched_scan_mtx);
|
||||||
|
+ __cfg80211_stop_sched_scan(rdev, false);
|
||||||
|
+ mutex_unlock(&rdev->sched_scan_mtx);
|
||||||
|
+
|
||||||
|
+ wdev_lock(wdev);
|
||||||
|
+#ifdef CONFIG_CFG80211_WEXT
|
||||||
|
+ kfree(wdev->wext.ie);
|
||||||
|
+ wdev->wext.ie = NULL;
|
||||||
|
+ wdev->wext.ie_len = 0;
|
||||||
|
+ wdev->wext.connect.auth_type = NL80211_AUTHTYPE_AUTOMATIC;
|
||||||
|
+#endif
|
||||||
|
+ __cfg80211_disconnect(rdev, dev,
|
||||||
|
+ WLAN_REASON_DEAUTH_LEAVING, true);
|
||||||
|
+ cfg80211_mlme_down(rdev, dev);
|
||||||
|
+ wdev_unlock(wdev);
|
||||||
|
+ break;
|
||||||
|
+ case NL80211_IFTYPE_MESH_POINT:
|
||||||
|
+ cfg80211_leave_mesh(rdev, dev);
|
||||||
|
+ break;
|
||||||
|
+ case NL80211_IFTYPE_AP:
|
||||||
|
+ cfg80211_stop_ap(rdev, dev);
|
||||||
|
+ break;
|
||||||
|
+ default:
|
||||||
|
+ break;
|
||||||
|
+ }
|
||||||
|
+
|
||||||
|
+ wdev->beacon_interval = 0;
|
||||||
|
+}
|
||||||
|
+
|
||||||
|
static int cfg80211_netdev_notifier_call(struct notifier_block *nb,
|
||||||
|
unsigned long state,
|
||||||
|
void *ndev)
|
||||||
|
@@ -882,38 +922,7 @@ static int cfg80211_netdev_notifier_call(struct notifier_block *nb,
|
||||||
|
dev->priv_flags |= IFF_DONT_BRIDGE;
|
||||||
|
break;
|
||||||
|
case NETDEV_GOING_DOWN:
|
||||||
|
- switch (wdev->iftype) {
|
||||||
|
- case NL80211_IFTYPE_ADHOC:
|
||||||
|
- cfg80211_leave_ibss(rdev, dev, true);
|
||||||
|
- break;
|
||||||
|
- case NL80211_IFTYPE_P2P_CLIENT:
|
||||||
|
- case NL80211_IFTYPE_STATION:
|
||||||
|
- mutex_lock(&rdev->sched_scan_mtx);
|
||||||
|
- __cfg80211_stop_sched_scan(rdev, false);
|
||||||
|
- mutex_unlock(&rdev->sched_scan_mtx);
|
||||||
|
-
|
||||||
|
- wdev_lock(wdev);
|
||||||
|
-#ifdef CONFIG_CFG80211_WEXT
|
||||||
|
- kfree(wdev->wext.ie);
|
||||||
|
- wdev->wext.ie = NULL;
|
||||||
|
- wdev->wext.ie_len = 0;
|
||||||
|
- wdev->wext.connect.auth_type = NL80211_AUTHTYPE_AUTOMATIC;
|
||||||
|
-#endif
|
||||||
|
- __cfg80211_disconnect(rdev, dev,
|
||||||
|
- WLAN_REASON_DEAUTH_LEAVING, true);
|
||||||
|
- cfg80211_mlme_down(rdev, dev);
|
||||||
|
- wdev_unlock(wdev);
|
||||||
|
- break;
|
||||||
|
- case NL80211_IFTYPE_MESH_POINT:
|
||||||
|
- cfg80211_leave_mesh(rdev, dev);
|
||||||
|
- break;
|
||||||
|
- case NL80211_IFTYPE_AP:
|
||||||
|
- cfg80211_stop_ap(rdev, dev);
|
||||||
|
- break;
|
||||||
|
- default:
|
||||||
|
- break;
|
||||||
|
- }
|
||||||
|
- wdev->beacon_interval = 0;
|
||||||
|
+ cfg80211_leave(rdev, wdev);
|
||||||
|
break;
|
||||||
|
case NETDEV_DOWN:
|
||||||
|
cfg80211_update_iface_num(rdev, wdev->iftype, -1);
|
||||||
|
diff --git a/net/wireless/core.h b/net/wireless/core.h
|
||||||
|
index 9a2be8d..d5d06fd 100644
|
||||||
|
--- a/net/wireless/core.h
|
||||||
|
+++ b/net/wireless/core.h
|
||||||
|
@@ -500,6 +500,9 @@ int cfg80211_validate_beacon_int(struct cfg80211_registered_device *rdev,
|
||||||
|
void cfg80211_update_iface_num(struct cfg80211_registered_device *rdev,
|
||||||
|
enum nl80211_iftype iftype, int num);
|
||||||
|
|
||||||
|
+void cfg80211_leave(struct cfg80211_registered_device *rdev,
|
||||||
|
+ struct wireless_dev *wdev);
|
||||||
|
+
|
||||||
|
#define CFG80211_MAX_NUM_DIFFERENT_CHANNELS 10
|
||||||
|
|
||||||
|
#ifdef CONFIG_CFG80211_DEVELOPER_WARNINGS
|
||||||
|
diff --git a/net/wireless/rdev-ops.h b/net/wireless/rdev-ops.h
|
||||||
|
index 8c8b26f..d77e1c1 100644
|
||||||
|
--- a/net/wireless/rdev-ops.h
|
||||||
|
+++ b/net/wireless/rdev-ops.h
|
||||||
|
@@ -6,11 +6,12 @@
|
||||||
|
#include "core.h"
|
||||||
|
#include "trace.h"
|
||||||
|
|
||||||
|
-static inline int rdev_suspend(struct cfg80211_registered_device *rdev)
|
||||||
|
+static inline int rdev_suspend(struct cfg80211_registered_device *rdev,
|
||||||
|
+ struct cfg80211_wowlan *wowlan)
|
||||||
|
{
|
||||||
|
int ret;
|
||||||
|
- trace_rdev_suspend(&rdev->wiphy, rdev->wowlan);
|
||||||
|
- ret = rdev->ops->suspend(&rdev->wiphy, rdev->wowlan);
|
||||||
|
+ trace_rdev_suspend(&rdev->wiphy, wowlan);
|
||||||
|
+ ret = rdev->ops->suspend(&rdev->wiphy, wowlan);
|
||||||
|
trace_rdev_return_int(&rdev->wiphy, ret);
|
||||||
|
return ret;
|
||||||
|
}
|
||||||
|
diff --git a/net/wireless/sysfs.c b/net/wireless/sysfs.c
|
||||||
|
index 238ee49..8f28b9f 100644
|
||||||
|
--- a/net/wireless/sysfs.c
|
||||||
|
+++ b/net/wireless/sysfs.c
|
||||||
|
@@ -83,6 +83,14 @@ static int wiphy_uevent(struct device *dev, struct kobj_uevent_env *env)
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
+static void cfg80211_leave_all(struct cfg80211_registered_device *rdev)
|
||||||
|
+{
|
||||||
|
+ struct wireless_dev *wdev;
|
||||||
|
+
|
||||||
|
+ list_for_each_entry(wdev, &rdev->wdev_list, list)
|
||||||
|
+ cfg80211_leave(rdev, wdev);
|
||||||
|
+}
|
||||||
|
+
|
||||||
|
static int wiphy_suspend(struct device *dev, pm_message_t state)
|
||||||
|
{
|
||||||
|
struct cfg80211_registered_device *rdev = dev_to_rdev(dev);
|
||||||
|
@@ -90,12 +98,19 @@ static int wiphy_suspend(struct device *dev, pm_message_t state)
|
||||||
|
|
||||||
|
rdev->suspend_at = get_seconds();
|
||||||
|
|
||||||
|
- if (rdev->ops->suspend) {
|
||||||
|
- rtnl_lock();
|
||||||
|
- if (rdev->wiphy.registered)
|
||||||
|
- ret = rdev_suspend(rdev);
|
||||||
|
- rtnl_unlock();
|
||||||
|
+ rtnl_lock();
|
||||||
|
+ if (rdev->wiphy.registered) {
|
||||||
|
+ if (!rdev->wowlan)
|
||||||
|
+ cfg80211_leave_all(rdev);
|
||||||
|
+ if (rdev->ops->suspend)
|
||||||
|
+ ret = rdev_suspend(rdev, rdev->wowlan);
|
||||||
|
+ if (ret == 1) {
|
||||||
|
+ /* Driver refuse to configure wowlan */
|
||||||
|
+ cfg80211_leave_all(rdev);
|
||||||
|
+ ret = rdev_suspend(rdev, NULL);
|
||||||
|
+ }
|
||||||
|
}
|
||||||
|
+ rtnl_unlock();
|
||||||
|
|
||||||
|
return ret;
|
||||||
|
}
|
||||||
|
--
|
||||||
|
cgit v0.9.1
|
@ -748,6 +748,9 @@ Patch21270: signal-always-clear-sa_restorer-on-execve.patch
|
|||||||
#CVE-2013-0913 rhbz 920471 920529
|
#CVE-2013-0913 rhbz 920471 920529
|
||||||
Patch21271: drm-i915-bounds-check-execbuffer-relocation-count.patch
|
Patch21271: drm-i915-bounds-check-execbuffer-relocation-count.patch
|
||||||
|
|
||||||
|
#rhbz 856863
|
||||||
|
Patch21273: cfg80211-mac80211-disconnect-on-suspend.patch
|
||||||
|
|
||||||
Patch22000: weird-root-dentry-name-debug.patch
|
Patch22000: weird-root-dentry-name-debug.patch
|
||||||
|
|
||||||
#selinux ptrace child permissions
|
#selinux ptrace child permissions
|
||||||
@ -1454,6 +1457,9 @@ ApplyPatch signal-always-clear-sa_restorer-on-execve.patch
|
|||||||
#CVE-2013-0913 rhbz 920471 920529
|
#CVE-2013-0913 rhbz 920471 920529
|
||||||
ApplyPatch drm-i915-bounds-check-execbuffer-relocation-count.patch
|
ApplyPatch drm-i915-bounds-check-execbuffer-relocation-count.patch
|
||||||
|
|
||||||
|
#rhbz 856863
|
||||||
|
ApplyPatch cfg80211-mac80211-disconnect-on-suspend.patch
|
||||||
|
|
||||||
# END OF PATCH APPLICATIONS
|
# END OF PATCH APPLICATIONS
|
||||||
|
|
||||||
%endif
|
%endif
|
||||||
@ -2296,6 +2302,7 @@ fi
|
|||||||
# || ||
|
# || ||
|
||||||
%changelog
|
%changelog
|
||||||
* Tue Mar 12 2013 Josh Boyer <jwboyer@redhat.com>
|
* Tue Mar 12 2013 Josh Boyer <jwboyer@redhat.com>
|
||||||
|
- Add patches to fix cfg80211 issues with suspend (rhbz 856863)
|
||||||
- CVE-2013-0913 drm/i915: head writing overflow (rhbz 920471 920529)
|
- CVE-2013-0913 drm/i915: head writing overflow (rhbz 920471 920529)
|
||||||
- CVE-2013-0914 sa_restorer information leak (rhbz 920499 920510)
|
- CVE-2013-0914 sa_restorer information leak (rhbz 920499 920510)
|
||||||
|
|
||||||
|
Loading…
Reference in New Issue
Block a user