aboutsummaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
-rw-r--r--include/net/cfg80211.h54
-rw-r--r--include/uapi/linux/nl80211.h18
-rw-r--r--net/wireless/nl80211.c1
-rw-r--r--net/wireless/scan.c31
-rw-r--r--net/wireless/trace.h12
5 files changed, 97 insertions, 19 deletions
diff --git a/include/net/cfg80211.h b/include/net/cfg80211.h
index 071ed2395c9a..bae8614a450f 100644
--- a/include/net/cfg80211.h
+++ b/include/net/cfg80211.h
@@ -1285,6 +1285,7 @@ struct cfg80211_ssid {
1285 * @n_ssids: number of SSIDs 1285 * @n_ssids: number of SSIDs
1286 * @channels: channels to scan on. 1286 * @channels: channels to scan on.
1287 * @n_channels: total number of channels to scan 1287 * @n_channels: total number of channels to scan
1288 * @scan_width: channel width for scanning
1288 * @ie: optional information element(s) to add into Probe Request or %NULL 1289 * @ie: optional information element(s) to add into Probe Request or %NULL
1289 * @ie_len: length of ie in octets 1290 * @ie_len: length of ie in octets
1290 * @flags: bit field of flags controlling operation 1291 * @flags: bit field of flags controlling operation
@@ -1300,6 +1301,7 @@ struct cfg80211_scan_request {
1300 struct cfg80211_ssid *ssids; 1301 struct cfg80211_ssid *ssids;
1301 int n_ssids; 1302 int n_ssids;
1302 u32 n_channels; 1303 u32 n_channels;
1304 enum nl80211_bss_scan_width scan_width;
1303 const u8 *ie; 1305 const u8 *ie;
1304 size_t ie_len; 1306 size_t ie_len;
1305 u32 flags; 1307 u32 flags;
@@ -1333,6 +1335,7 @@ struct cfg80211_match_set {
1333 * @ssids: SSIDs to scan for (passed in the probe_reqs in active scans) 1335 * @ssids: SSIDs to scan for (passed in the probe_reqs in active scans)
1334 * @n_ssids: number of SSIDs 1336 * @n_ssids: number of SSIDs
1335 * @n_channels: total number of channels to scan 1337 * @n_channels: total number of channels to scan
1338 * @scan_width: channel width for scanning
1336 * @interval: interval between each scheduled scan cycle 1339 * @interval: interval between each scheduled scan cycle
1337 * @ie: optional information element(s) to add into Probe Request or %NULL 1340 * @ie: optional information element(s) to add into Probe Request or %NULL
1338 * @ie_len: length of ie in octets 1341 * @ie_len: length of ie in octets
@@ -1352,6 +1355,7 @@ struct cfg80211_sched_scan_request {
1352 struct cfg80211_ssid *ssids; 1355 struct cfg80211_ssid *ssids;
1353 int n_ssids; 1356 int n_ssids;
1354 u32 n_channels; 1357 u32 n_channels;
1358 enum nl80211_bss_scan_width scan_width;
1355 u32 interval; 1359 u32 interval;
1356 const u8 *ie; 1360 const u8 *ie;
1357 size_t ie_len; 1361 size_t ie_len;
@@ -1403,6 +1407,7 @@ struct cfg80211_bss_ies {
1403 * for use in scan results and similar. 1407 * for use in scan results and similar.
1404 * 1408 *
1405 * @channel: channel this BSS is on 1409 * @channel: channel this BSS is on
1410 * @scan_width: width of the control channel
1406 * @bssid: BSSID of the BSS 1411 * @bssid: BSSID of the BSS
1407 * @beacon_interval: the beacon interval as from the frame 1412 * @beacon_interval: the beacon interval as from the frame
1408 * @capability: the capability field in host byte order 1413 * @capability: the capability field in host byte order
@@ -1424,6 +1429,7 @@ struct cfg80211_bss_ies {
1424 */ 1429 */
1425struct cfg80211_bss { 1430struct cfg80211_bss {
1426 struct ieee80211_channel *channel; 1431 struct ieee80211_channel *channel;
1432 enum nl80211_bss_scan_width scan_width;
1427 1433
1428 const struct cfg80211_bss_ies __rcu *ies; 1434 const struct cfg80211_bss_ies __rcu *ies;
1429 const struct cfg80211_bss_ies __rcu *beacon_ies; 1435 const struct cfg80211_bss_ies __rcu *beacon_ies;
@@ -3438,10 +3444,11 @@ void cfg80211_sched_scan_results(struct wiphy *wiphy);
3438void cfg80211_sched_scan_stopped(struct wiphy *wiphy); 3444void cfg80211_sched_scan_stopped(struct wiphy *wiphy);
3439 3445
3440/** 3446/**
3441 * cfg80211_inform_bss_frame - inform cfg80211 of a received BSS frame 3447 * cfg80211_inform_bss_width_frame - inform cfg80211 of a received BSS frame
3442 * 3448 *
3443 * @wiphy: the wiphy reporting the BSS 3449 * @wiphy: the wiphy reporting the BSS
3444 * @channel: The channel the frame was received on 3450 * @channel: The channel the frame was received on
3451 * @scan_width: width of the control channel
3445 * @mgmt: the management frame (probe response or beacon) 3452 * @mgmt: the management frame (probe response or beacon)
3446 * @len: length of the management frame 3453 * @len: length of the management frame
3447 * @signal: the signal strength, type depends on the wiphy's signal_type 3454 * @signal: the signal strength, type depends on the wiphy's signal_type
@@ -3454,16 +3461,29 @@ void cfg80211_sched_scan_stopped(struct wiphy *wiphy);
3454 * Or %NULL on error. 3461 * Or %NULL on error.
3455 */ 3462 */
3456struct cfg80211_bss * __must_check 3463struct cfg80211_bss * __must_check
3464cfg80211_inform_bss_width_frame(struct wiphy *wiphy,
3465 struct ieee80211_channel *channel,
3466 enum nl80211_bss_scan_width scan_width,
3467 struct ieee80211_mgmt *mgmt, size_t len,
3468 s32 signal, gfp_t gfp);
3469
3470static inline struct cfg80211_bss * __must_check
3457cfg80211_inform_bss_frame(struct wiphy *wiphy, 3471cfg80211_inform_bss_frame(struct wiphy *wiphy,
3458 struct ieee80211_channel *channel, 3472 struct ieee80211_channel *channel,
3459 struct ieee80211_mgmt *mgmt, size_t len, 3473 struct ieee80211_mgmt *mgmt, size_t len,
3460 s32 signal, gfp_t gfp); 3474 s32 signal, gfp_t gfp)
3475{
3476 return cfg80211_inform_bss_width_frame(wiphy, channel,
3477 NL80211_BSS_CHAN_WIDTH_20,
3478 mgmt, len, signal, gfp);
3479}
3461 3480
3462/** 3481/**
3463 * cfg80211_inform_bss - inform cfg80211 of a new BSS 3482 * cfg80211_inform_bss - inform cfg80211 of a new BSS
3464 * 3483 *
3465 * @wiphy: the wiphy reporting the BSS 3484 * @wiphy: the wiphy reporting the BSS
3466 * @channel: The channel the frame was received on 3485 * @channel: The channel the frame was received on
3486 * @scan_width: width of the control channel
3467 * @bssid: the BSSID of the BSS 3487 * @bssid: the BSSID of the BSS
3468 * @tsf: the TSF sent by the peer in the beacon/probe response (or 0) 3488 * @tsf: the TSF sent by the peer in the beacon/probe response (or 0)
3469 * @capability: the capability field sent by the peer 3489 * @capability: the capability field sent by the peer
@@ -3480,11 +3500,26 @@ cfg80211_inform_bss_frame(struct wiphy *wiphy,
3480 * Or %NULL on error. 3500 * Or %NULL on error.
3481 */ 3501 */
3482struct cfg80211_bss * __must_check 3502struct cfg80211_bss * __must_check
3503cfg80211_inform_bss_width(struct wiphy *wiphy,
3504 struct ieee80211_channel *channel,
3505 enum nl80211_bss_scan_width scan_width,
3506 const u8 *bssid, u64 tsf, u16 capability,
3507 u16 beacon_interval, const u8 *ie, size_t ielen,
3508 s32 signal, gfp_t gfp);
3509
3510static inline struct cfg80211_bss * __must_check
3483cfg80211_inform_bss(struct wiphy *wiphy, 3511cfg80211_inform_bss(struct wiphy *wiphy,
3484 struct ieee80211_channel *channel, 3512 struct ieee80211_channel *channel,
3485 const u8 *bssid, u64 tsf, u16 capability, 3513 const u8 *bssid, u64 tsf, u16 capability,
3486 u16 beacon_interval, const u8 *ie, size_t ielen, 3514 u16 beacon_interval, const u8 *ie, size_t ielen,
3487 s32 signal, gfp_t gfp); 3515 s32 signal, gfp_t gfp)
3516{
3517 return cfg80211_inform_bss_width(wiphy, channel,
3518 NL80211_BSS_CHAN_WIDTH_20,
3519 bssid, tsf, capability,
3520 beacon_interval, ie, ielen, signal,
3521 gfp);
3522}
3488 3523
3489struct cfg80211_bss *cfg80211_get_bss(struct wiphy *wiphy, 3524struct cfg80211_bss *cfg80211_get_bss(struct wiphy *wiphy,
3490 struct ieee80211_channel *channel, 3525 struct ieee80211_channel *channel,
@@ -3530,6 +3565,19 @@ void cfg80211_put_bss(struct wiphy *wiphy, struct cfg80211_bss *bss);
3530 */ 3565 */
3531void cfg80211_unlink_bss(struct wiphy *wiphy, struct cfg80211_bss *bss); 3566void cfg80211_unlink_bss(struct wiphy *wiphy, struct cfg80211_bss *bss);
3532 3567
3568static inline enum nl80211_bss_scan_width
3569cfg80211_chandef_to_scan_width(const struct cfg80211_chan_def *chandef)
3570{
3571 switch (chandef->width) {
3572 case NL80211_CHAN_WIDTH_5:
3573 return NL80211_BSS_CHAN_WIDTH_5;
3574 case NL80211_CHAN_WIDTH_10:
3575 return NL80211_BSS_CHAN_WIDTH_10;
3576 default:
3577 return NL80211_BSS_CHAN_WIDTH_20;
3578 }
3579}
3580
3533/** 3581/**
3534 * cfg80211_rx_mlme_mgmt - notification of processed MLME management frame 3582 * cfg80211_rx_mlme_mgmt - notification of processed MLME management frame
3535 * @dev: network device 3583 * @dev: network device
diff --git a/include/uapi/linux/nl80211.h b/include/uapi/linux/nl80211.h
index 5abc54d14d4d..eb68735b3318 100644
--- a/include/uapi/linux/nl80211.h
+++ b/include/uapi/linux/nl80211.h
@@ -2808,6 +2808,21 @@ enum nl80211_chan_width {
2808}; 2808};
2809 2809
2810/** 2810/**
2811 * enum nl80211_bss_scan_width - control channel width for a BSS
2812 *
2813 * These values are used with the %NL80211_BSS_CHAN_WIDTH attribute.
2814 *
2815 * @NL80211_BSS_CHAN_WIDTH_20: control channel is 20 MHz wide or compatible
2816 * @NL80211_BSS_CHAN_WIDTH_10: control channel is 10 MHz wide
2817 * @NL80211_BSS_CHAN_WIDTH_5: control channel is 5 MHz wide
2818 */
2819enum nl80211_bss_scan_width {
2820 NL80211_BSS_CHAN_WIDTH_20,
2821 NL80211_BSS_CHAN_WIDTH_10,
2822 NL80211_BSS_CHAN_WIDTH_5,
2823};
2824
2825/**
2811 * enum nl80211_bss - netlink attributes for a BSS 2826 * enum nl80211_bss - netlink attributes for a BSS
2812 * 2827 *
2813 * @__NL80211_BSS_INVALID: invalid 2828 * @__NL80211_BSS_INVALID: invalid
@@ -2831,6 +2846,8 @@ enum nl80211_chan_width {
2831 * @NL80211_BSS_BEACON_IES: binary attribute containing the raw information 2846 * @NL80211_BSS_BEACON_IES: binary attribute containing the raw information
2832 * elements from a Beacon frame (bin); not present if no Beacon frame has 2847 * elements from a Beacon frame (bin); not present if no Beacon frame has
2833 * yet been received 2848 * yet been received
2849 * @NL80211_BSS_CHAN_WIDTH: channel width of the control channel
2850 * (u32, enum nl80211_bss_scan_width)
2834 * @__NL80211_BSS_AFTER_LAST: internal 2851 * @__NL80211_BSS_AFTER_LAST: internal
2835 * @NL80211_BSS_MAX: highest BSS attribute 2852 * @NL80211_BSS_MAX: highest BSS attribute
2836 */ 2853 */
@@ -2847,6 +2864,7 @@ enum nl80211_bss {
2847 NL80211_BSS_STATUS, 2864 NL80211_BSS_STATUS,
2848 NL80211_BSS_SEEN_MS_AGO, 2865 NL80211_BSS_SEEN_MS_AGO,
2849 NL80211_BSS_BEACON_IES, 2866 NL80211_BSS_BEACON_IES,
2867 NL80211_BSS_CHAN_WIDTH,
2850 2868
2851 /* keep last */ 2869 /* keep last */
2852 __NL80211_BSS_AFTER_LAST, 2870 __NL80211_BSS_AFTER_LAST,
diff --git a/net/wireless/nl80211.c b/net/wireless/nl80211.c
index 6dca5a700174..ef4c312cc92c 100644
--- a/net/wireless/nl80211.c
+++ b/net/wireless/nl80211.c
@@ -5674,6 +5674,7 @@ static int nl80211_send_bss(struct sk_buff *msg, struct netlink_callback *cb,
5674 goto nla_put_failure; 5674 goto nla_put_failure;
5675 if (nla_put_u16(msg, NL80211_BSS_CAPABILITY, res->capability) || 5675 if (nla_put_u16(msg, NL80211_BSS_CAPABILITY, res->capability) ||
5676 nla_put_u32(msg, NL80211_BSS_FREQUENCY, res->channel->center_freq) || 5676 nla_put_u32(msg, NL80211_BSS_FREQUENCY, res->channel->center_freq) ||
5677 nla_put_u32(msg, NL80211_BSS_CHAN_WIDTH, res->scan_width) ||
5677 nla_put_u32(msg, NL80211_BSS_SEEN_MS_AGO, 5678 nla_put_u32(msg, NL80211_BSS_SEEN_MS_AGO,
5678 jiffies_to_msecs(jiffies - intbss->ts))) 5679 jiffies_to_msecs(jiffies - intbss->ts)))
5679 goto nla_put_failure; 5680 goto nla_put_failure;
diff --git a/net/wireless/scan.c b/net/wireless/scan.c
index ae8c186b50d6..ad1e4068ce06 100644
--- a/net/wireless/scan.c
+++ b/net/wireless/scan.c
@@ -651,6 +651,8 @@ static bool cfg80211_combine_bsses(struct cfg80211_registered_device *dev,
651 continue; 651 continue;
652 if (bss->pub.channel != new->pub.channel) 652 if (bss->pub.channel != new->pub.channel)
653 continue; 653 continue;
654 if (bss->pub.scan_width != new->pub.scan_width)
655 continue;
654 if (rcu_access_pointer(bss->pub.beacon_ies)) 656 if (rcu_access_pointer(bss->pub.beacon_ies))
655 continue; 657 continue;
656 ies = rcu_access_pointer(bss->pub.ies); 658 ies = rcu_access_pointer(bss->pub.ies);
@@ -870,11 +872,12 @@ cfg80211_get_bss_channel(struct wiphy *wiphy, const u8 *ie, size_t ielen,
870 872
871/* Returned bss is reference counted and must be cleaned up appropriately. */ 873/* Returned bss is reference counted and must be cleaned up appropriately. */
872struct cfg80211_bss* 874struct cfg80211_bss*
873cfg80211_inform_bss(struct wiphy *wiphy, 875cfg80211_inform_bss_width(struct wiphy *wiphy,
874 struct ieee80211_channel *channel, 876 struct ieee80211_channel *channel,
875 const u8 *bssid, u64 tsf, u16 capability, 877 enum nl80211_bss_scan_width scan_width,
876 u16 beacon_interval, const u8 *ie, size_t ielen, 878 const u8 *bssid, u64 tsf, u16 capability,
877 s32 signal, gfp_t gfp) 879 u16 beacon_interval, const u8 *ie, size_t ielen,
880 s32 signal, gfp_t gfp)
878{ 881{
879 struct cfg80211_bss_ies *ies; 882 struct cfg80211_bss_ies *ies;
880 struct cfg80211_internal_bss tmp = {}, *res; 883 struct cfg80211_internal_bss tmp = {}, *res;
@@ -892,6 +895,7 @@ cfg80211_inform_bss(struct wiphy *wiphy,
892 895
893 memcpy(tmp.pub.bssid, bssid, ETH_ALEN); 896 memcpy(tmp.pub.bssid, bssid, ETH_ALEN);
894 tmp.pub.channel = channel; 897 tmp.pub.channel = channel;
898 tmp.pub.scan_width = scan_width;
895 tmp.pub.signal = signal; 899 tmp.pub.signal = signal;
896 tmp.pub.beacon_interval = beacon_interval; 900 tmp.pub.beacon_interval = beacon_interval;
897 tmp.pub.capability = capability; 901 tmp.pub.capability = capability;
@@ -924,14 +928,15 @@ cfg80211_inform_bss(struct wiphy *wiphy,
924 /* cfg80211_bss_update gives us a referenced result */ 928 /* cfg80211_bss_update gives us a referenced result */
925 return &res->pub; 929 return &res->pub;
926} 930}
927EXPORT_SYMBOL(cfg80211_inform_bss); 931EXPORT_SYMBOL(cfg80211_inform_bss_width);
928 932
929/* Returned bss is reference counted and must be cleaned up appropriately. */ 933/* Returned bss is reference counted and must be cleaned up appropriately. */
930struct cfg80211_bss * 934struct cfg80211_bss *
931cfg80211_inform_bss_frame(struct wiphy *wiphy, 935cfg80211_inform_bss_width_frame(struct wiphy *wiphy,
932 struct ieee80211_channel *channel, 936 struct ieee80211_channel *channel,
933 struct ieee80211_mgmt *mgmt, size_t len, 937 enum nl80211_bss_scan_width scan_width,
934 s32 signal, gfp_t gfp) 938 struct ieee80211_mgmt *mgmt, size_t len,
939 s32 signal, gfp_t gfp)
935{ 940{
936 struct cfg80211_internal_bss tmp = {}, *res; 941 struct cfg80211_internal_bss tmp = {}, *res;
937 struct cfg80211_bss_ies *ies; 942 struct cfg80211_bss_ies *ies;
@@ -941,7 +946,8 @@ cfg80211_inform_bss_frame(struct wiphy *wiphy,
941 BUILD_BUG_ON(offsetof(struct ieee80211_mgmt, u.probe_resp.variable) != 946 BUILD_BUG_ON(offsetof(struct ieee80211_mgmt, u.probe_resp.variable) !=
942 offsetof(struct ieee80211_mgmt, u.beacon.variable)); 947 offsetof(struct ieee80211_mgmt, u.beacon.variable));
943 948
944 trace_cfg80211_inform_bss_frame(wiphy, channel, mgmt, len, signal); 949 trace_cfg80211_inform_bss_width_frame(wiphy, channel, scan_width, mgmt,
950 len, signal);
945 951
946 if (WARN_ON(!mgmt)) 952 if (WARN_ON(!mgmt))
947 return NULL; 953 return NULL;
@@ -976,6 +982,7 @@ cfg80211_inform_bss_frame(struct wiphy *wiphy,
976 982
977 memcpy(tmp.pub.bssid, mgmt->bssid, ETH_ALEN); 983 memcpy(tmp.pub.bssid, mgmt->bssid, ETH_ALEN);
978 tmp.pub.channel = channel; 984 tmp.pub.channel = channel;
985 tmp.pub.scan_width = scan_width;
979 tmp.pub.signal = signal; 986 tmp.pub.signal = signal;
980 tmp.pub.beacon_interval = le16_to_cpu(mgmt->u.probe_resp.beacon_int); 987 tmp.pub.beacon_interval = le16_to_cpu(mgmt->u.probe_resp.beacon_int);
981 tmp.pub.capability = le16_to_cpu(mgmt->u.probe_resp.capab_info); 988 tmp.pub.capability = le16_to_cpu(mgmt->u.probe_resp.capab_info);
@@ -991,7 +998,7 @@ cfg80211_inform_bss_frame(struct wiphy *wiphy,
991 /* cfg80211_bss_update gives us a referenced result */ 998 /* cfg80211_bss_update gives us a referenced result */
992 return &res->pub; 999 return &res->pub;
993} 1000}
994EXPORT_SYMBOL(cfg80211_inform_bss_frame); 1001EXPORT_SYMBOL(cfg80211_inform_bss_width_frame);
995 1002
996void cfg80211_ref_bss(struct wiphy *wiphy, struct cfg80211_bss *pub) 1003void cfg80211_ref_bss(struct wiphy *wiphy, struct cfg80211_bss *pub)
997{ 1004{
diff --git a/net/wireless/trace.h b/net/wireless/trace.h
index e1534baf2ebb..09af6eb426a8 100644
--- a/net/wireless/trace.h
+++ b/net/wireless/trace.h
@@ -2391,26 +2391,30 @@ TRACE_EVENT(cfg80211_get_bss,
2391 __entry->capa_mask, __entry->capa_val) 2391 __entry->capa_mask, __entry->capa_val)
2392); 2392);
2393 2393
2394TRACE_EVENT(cfg80211_inform_bss_frame, 2394TRACE_EVENT(cfg80211_inform_bss_width_frame,
2395 TP_PROTO(struct wiphy *wiphy, struct ieee80211_channel *channel, 2395 TP_PROTO(struct wiphy *wiphy, struct ieee80211_channel *channel,
2396 enum nl80211_bss_scan_width scan_width,
2396 struct ieee80211_mgmt *mgmt, size_t len, 2397 struct ieee80211_mgmt *mgmt, size_t len,
2397 s32 signal), 2398 s32 signal),
2398 TP_ARGS(wiphy, channel, mgmt, len, signal), 2399 TP_ARGS(wiphy, channel, scan_width, mgmt, len, signal),
2399 TP_STRUCT__entry( 2400 TP_STRUCT__entry(
2400 WIPHY_ENTRY 2401 WIPHY_ENTRY
2401 CHAN_ENTRY 2402 CHAN_ENTRY
2403 __field(enum nl80211_bss_scan_width, scan_width)
2402 __dynamic_array(u8, mgmt, len) 2404 __dynamic_array(u8, mgmt, len)
2403 __field(s32, signal) 2405 __field(s32, signal)
2404 ), 2406 ),
2405 TP_fast_assign( 2407 TP_fast_assign(
2406 WIPHY_ASSIGN; 2408 WIPHY_ASSIGN;
2407 CHAN_ASSIGN(channel); 2409 CHAN_ASSIGN(channel);
2410 __entry->scan_width = scan_width;
2408 if (mgmt) 2411 if (mgmt)
2409 memcpy(__get_dynamic_array(mgmt), mgmt, len); 2412 memcpy(__get_dynamic_array(mgmt), mgmt, len);
2410 __entry->signal = signal; 2413 __entry->signal = signal;
2411 ), 2414 ),
2412 TP_printk(WIPHY_PR_FMT ", " CHAN_PR_FMT "signal: %d", 2415 TP_printk(WIPHY_PR_FMT ", " CHAN_PR_FMT "(scan_width: %d) signal: %d",
2413 WIPHY_PR_ARG, CHAN_PR_ARG, __entry->signal) 2416 WIPHY_PR_ARG, CHAN_PR_ARG, __entry->scan_width,
2417 __entry->signal)
2414); 2418);
2415 2419
2416DECLARE_EVENT_CLASS(cfg80211_bss_evt, 2420DECLARE_EVENT_CLASS(cfg80211_bss_evt,