aboutsummaryrefslogtreecommitdiffstats
path: root/net/wireless
diff options
context:
space:
mode:
authorJohannes Berg <johannes.berg@intel.com>2014-01-24 08:06:29 -0500
committerJohannes Berg <johannes.berg@intel.com>2014-02-04 15:58:08 -0500
commit348baf0eac3391c62d441ec29b4c5da62ed91e74 (patch)
treed8255b3c6b16e330fc97671c5afcd666bee5a9be /net/wireless
parentd8ca16db6bb23d03fcb794df44bae64ae976f27c (diff)
nl80211: send event when AP operation is stopped
There are a few cases, e.g. suspend, where an AP interface is stopped by the kernel rather than by userspace request, most commonly when suspending. To let userspace know about this, send the NL80211_CMD_STOP_AP command as an event every time an AP interface is stopped. This also happens when userspace did in fact request the AP stop, but that's not a problem. For full-MAC drivers this may need to be extended to also cover cases where the device stopped the AP operation for some reason, this a bit more complicated because then all cfg80211 state also needs to be reset; such API is not part of this patch. Signed-off-by: Johannes Berg <johannes.berg@intel.com>
Diffstat (limited to 'net/wireless')
-rw-r--r--net/wireless/ap.c1
-rw-r--r--net/wireless/nl80211.c29
-rw-r--r--net/wireless/nl80211.h2
3 files changed, 32 insertions, 0 deletions
diff --git a/net/wireless/ap.c b/net/wireless/ap.c
index 11ee4ed04f73..4760d6554e62 100644
--- a/net/wireless/ap.c
+++ b/net/wireless/ap.c
@@ -30,6 +30,7 @@ static int __cfg80211_stop_ap(struct cfg80211_registered_device *rdev,
30 wdev->channel = NULL; 30 wdev->channel = NULL;
31 wdev->ssid_len = 0; 31 wdev->ssid_len = 0;
32 rdev_set_qos_map(rdev, dev, NULL); 32 rdev_set_qos_map(rdev, dev, NULL);
33 nl80211_send_ap_stopped(wdev);
33 } 34 }
34 35
35 return err; 36 return err;
diff --git a/net/wireless/nl80211.c b/net/wireless/nl80211.c
index 9ed6ef6fd2c5..043bfbd58b56 100644
--- a/net/wireless/nl80211.c
+++ b/net/wireless/nl80211.c
@@ -11677,6 +11677,35 @@ void cfg80211_crit_proto_stopped(struct wireless_dev *wdev, gfp_t gfp)
11677} 11677}
11678EXPORT_SYMBOL(cfg80211_crit_proto_stopped); 11678EXPORT_SYMBOL(cfg80211_crit_proto_stopped);
11679 11679
11680void nl80211_send_ap_stopped(struct wireless_dev *wdev)
11681{
11682 struct wiphy *wiphy = wdev->wiphy;
11683 struct cfg80211_registered_device *rdev = wiphy_to_dev(wiphy);
11684 struct sk_buff *msg;
11685 void *hdr;
11686
11687 msg = nlmsg_new(NLMSG_DEFAULT_SIZE, GFP_KERNEL);
11688 if (!msg)
11689 return;
11690
11691 hdr = nl80211hdr_put(msg, 0, 0, 0, NL80211_CMD_STOP_AP);
11692 if (!hdr)
11693 goto out;
11694
11695 if (nla_put_u32(msg, NL80211_ATTR_WIPHY, rdev->wiphy_idx) ||
11696 nla_put_u32(msg, NL80211_ATTR_IFINDEX, wdev->netdev->ifindex) ||
11697 nla_put_u64(msg, NL80211_ATTR_WDEV, wdev_id(wdev)))
11698 goto out;
11699
11700 genlmsg_end(msg, hdr);
11701
11702 genlmsg_multicast_netns(&nl80211_fam, wiphy_net(wiphy), msg, 0,
11703 NL80211_MCGRP_MLME, GFP_KERNEL);
11704 return;
11705 out:
11706 nlmsg_free(msg);
11707}
11708
11680/* initialisation/exit functions */ 11709/* initialisation/exit functions */
11681 11710
11682int nl80211_init(void) 11711int nl80211_init(void)
diff --git a/net/wireless/nl80211.h b/net/wireless/nl80211.h
index b1b231324e10..cb0216e1a004 100644
--- a/net/wireless/nl80211.h
+++ b/net/wireless/nl80211.h
@@ -74,6 +74,8 @@ nl80211_radar_notify(struct cfg80211_registered_device *rdev,
74 enum nl80211_radar_event event, 74 enum nl80211_radar_event event,
75 struct net_device *netdev, gfp_t gfp); 75 struct net_device *netdev, gfp_t gfp);
76 76
77void nl80211_send_ap_stopped(struct wireless_dev *wdev);
78
77void cfg80211_rdev_free_coalesce(struct cfg80211_registered_device *rdev); 79void cfg80211_rdev_free_coalesce(struct cfg80211_registered_device *rdev);
78 80
79#endif /* __NET_WIRELESS_NL80211_H */ 81#endif /* __NET_WIRELESS_NL80211_H */