ap.c 2.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110
  1. #include <linux/ieee80211.h>
  2. #include <linux/export.h>
  3. #include <net/cfg80211.h>
  4. #include "nl80211.h"
  5. #include "core.h"
  6. #include "rdev-ops.h"
  7. static int __cfg80211_stop_ap(struct cfg80211_registered_device *rdev,
  8. struct net_device *dev)
  9. {
  10. struct wireless_dev *wdev = dev->ieee80211_ptr;
  11. int err;
  12. ASSERT_WDEV_LOCK(wdev);
  13. if (!rdev->ops->stop_ap)
  14. return -EOPNOTSUPP;
  15. if (dev->ieee80211_ptr->iftype != NL80211_IFTYPE_AP &&
  16. dev->ieee80211_ptr->iftype != NL80211_IFTYPE_P2P_GO)
  17. return -EOPNOTSUPP;
  18. if (!wdev->beacon_interval)
  19. return -ENOENT;
  20. err = rdev_stop_ap(rdev, dev);
  21. if (!err) {
  22. wdev->beacon_interval = 0;
  23. wdev->channel = NULL;
  24. wdev->ssid_len = 0;
  25. }
  26. return err;
  27. }
  28. int cfg80211_stop_ap(struct cfg80211_registered_device *rdev,
  29. struct net_device *dev)
  30. {
  31. struct wireless_dev *wdev = dev->ieee80211_ptr;
  32. int err;
  33. wdev_lock(wdev);
  34. err = __cfg80211_stop_ap(rdev, dev);
  35. wdev_unlock(wdev);
  36. return err;
  37. }
  38. void cfg80211_ch_switch_notify(struct net_device *dev,
  39. struct cfg80211_chan_def *chandef)
  40. {
  41. struct wireless_dev *wdev = dev->ieee80211_ptr;
  42. struct wiphy *wiphy = wdev->wiphy;
  43. struct cfg80211_registered_device *rdev = wiphy_to_dev(wiphy);
  44. trace_cfg80211_ch_switch_notify(dev, chandef);
  45. wdev_lock(wdev);
  46. if (WARN_ON(wdev->iftype != NL80211_IFTYPE_AP &&
  47. wdev->iftype != NL80211_IFTYPE_P2P_GO))
  48. goto out;
  49. wdev->channel = chandef->chan;
  50. nl80211_ch_switch_notify(rdev, dev, chandef, GFP_KERNEL);
  51. out:
  52. wdev_unlock(wdev);
  53. return;
  54. }
  55. EXPORT_SYMBOL(cfg80211_ch_switch_notify);
  56. bool cfg80211_rx_spurious_frame(struct net_device *dev,
  57. const u8 *addr, gfp_t gfp)
  58. {
  59. struct wireless_dev *wdev = dev->ieee80211_ptr;
  60. bool ret;
  61. trace_cfg80211_rx_spurious_frame(dev, addr);
  62. if (WARN_ON(wdev->iftype != NL80211_IFTYPE_AP &&
  63. wdev->iftype != NL80211_IFTYPE_P2P_GO)) {
  64. trace_cfg80211_return_bool(false);
  65. return false;
  66. }
  67. ret = nl80211_unexpected_frame(dev, addr, gfp);
  68. trace_cfg80211_return_bool(ret);
  69. return ret;
  70. }
  71. EXPORT_SYMBOL(cfg80211_rx_spurious_frame);
  72. bool cfg80211_rx_unexpected_4addr_frame(struct net_device *dev,
  73. const u8 *addr, gfp_t gfp)
  74. {
  75. struct wireless_dev *wdev = dev->ieee80211_ptr;
  76. bool ret;
  77. trace_cfg80211_rx_unexpected_4addr_frame(dev, addr);
  78. if (WARN_ON(wdev->iftype != NL80211_IFTYPE_AP &&
  79. wdev->iftype != NL80211_IFTYPE_P2P_GO &&
  80. wdev->iftype != NL80211_IFTYPE_AP_VLAN)) {
  81. trace_cfg80211_return_bool(false);
  82. return false;
  83. }
  84. ret = nl80211_unexpected_4addr_frame(dev, addr, gfp);
  85. trace_cfg80211_return_bool(ret);
  86. return ret;
  87. }
  88. EXPORT_SYMBOL(cfg80211_rx_unexpected_4addr_frame);