2019-06-04 10:11:33 +02:00
// SPDX-License-Identifier: GPL-2.0-only
2018-06-09 09:14:44 +03:00
/*
* HE handling
*
* Copyright ( c ) 2017 Intel Deutschland GmbH
2020-01-31 13:12:53 +02:00
* Copyright ( c ) 2019 - 2020 Intel Corporation
2018-06-09 09:14:44 +03:00
*/
# include "ieee80211_i.h"
2020-05-28 21:34:38 +02:00
static void
ieee80211_update_from_he_6ghz_capa ( const struct ieee80211_he_6ghz_capa * he_6ghz_capa ,
struct sta_info * sta )
{
enum ieee80211_smps_mode smps_mode ;
if ( sta - > sdata - > vif . type = = NL80211_IFTYPE_AP | |
sta - > sdata - > vif . type = = NL80211_IFTYPE_AP_VLAN ) {
switch ( le16_get_bits ( he_6ghz_capa - > capa ,
IEEE80211_HE_6GHZ_CAP_SM_PS ) ) {
case WLAN_HT_CAP_SM_PS_INVALID :
case WLAN_HT_CAP_SM_PS_STATIC :
smps_mode = IEEE80211_SMPS_STATIC ;
break ;
case WLAN_HT_CAP_SM_PS_DYNAMIC :
smps_mode = IEEE80211_SMPS_DYNAMIC ;
break ;
case WLAN_HT_CAP_SM_PS_DISABLED :
smps_mode = IEEE80211_SMPS_OFF ;
break ;
}
sta - > sta . smps_mode = smps_mode ;
} else {
sta - > sta . smps_mode = IEEE80211_SMPS_OFF ;
}
switch ( le16_get_bits ( he_6ghz_capa - > capa ,
IEEE80211_HE_6GHZ_CAP_MAX_MPDU_LEN ) ) {
case IEEE80211_VHT_CAP_MAX_MPDU_LENGTH_11454 :
sta - > sta . max_amsdu_len = IEEE80211_MAX_MPDU_LEN_VHT_11454 ;
break ;
case IEEE80211_VHT_CAP_MAX_MPDU_LENGTH_7991 :
sta - > sta . max_amsdu_len = IEEE80211_MAX_MPDU_LEN_VHT_7991 ;
break ;
case IEEE80211_VHT_CAP_MAX_MPDU_LENGTH_3895 :
default :
sta - > sta . max_amsdu_len = IEEE80211_MAX_MPDU_LEN_VHT_3895 ;
break ;
}
sta - > sta . he_6ghz_capa = * he_6ghz_capa ;
}
2020-12-25 11:39:14 +08:00
static void ieee80211_he_mcs_disable ( __le16 * he_mcs )
{
u32 i ;
for ( i = 0 ; i < 8 ; i + + )
* he_mcs | = cpu_to_le16 ( IEEE80211_HE_MCS_NOT_SUPPORTED < < i * 2 ) ;
}
static void ieee80211_he_mcs_intersection ( __le16 * he_own_rx , __le16 * he_peer_rx ,
__le16 * he_own_tx , __le16 * he_peer_tx )
{
u32 i ;
u16 own_rx , own_tx , peer_rx , peer_tx ;
for ( i = 0 ; i < 8 ; i + + ) {
own_rx = le16_to_cpu ( * he_own_rx ) ;
own_rx = ( own_rx > > i * 2 ) & IEEE80211_HE_MCS_NOT_SUPPORTED ;
own_tx = le16_to_cpu ( * he_own_tx ) ;
own_tx = ( own_tx > > i * 2 ) & IEEE80211_HE_MCS_NOT_SUPPORTED ;
peer_rx = le16_to_cpu ( * he_peer_rx ) ;
peer_rx = ( peer_rx > > i * 2 ) & IEEE80211_HE_MCS_NOT_SUPPORTED ;
peer_tx = le16_to_cpu ( * he_peer_tx ) ;
peer_tx = ( peer_tx > > i * 2 ) & IEEE80211_HE_MCS_NOT_SUPPORTED ;
if ( peer_tx ! = IEEE80211_HE_MCS_NOT_SUPPORTED ) {
if ( own_rx = = IEEE80211_HE_MCS_NOT_SUPPORTED )
peer_tx = IEEE80211_HE_MCS_NOT_SUPPORTED ;
else if ( own_rx < peer_tx )
peer_tx = own_rx ;
}
if ( peer_rx ! = IEEE80211_HE_MCS_NOT_SUPPORTED ) {
if ( own_tx = = IEEE80211_HE_MCS_NOT_SUPPORTED )
peer_rx = IEEE80211_HE_MCS_NOT_SUPPORTED ;
else if ( own_tx < peer_rx )
peer_rx = own_tx ;
}
* he_peer_rx & =
~ cpu_to_le16 ( IEEE80211_HE_MCS_NOT_SUPPORTED < < i * 2 ) ;
* he_peer_rx | = cpu_to_le16 ( peer_rx < < i * 2 ) ;
* he_peer_tx & =
~ cpu_to_le16 ( IEEE80211_HE_MCS_NOT_SUPPORTED < < i * 2 ) ;
* he_peer_tx | = cpu_to_le16 ( peer_tx < < i * 2 ) ;
}
}
2018-06-09 09:14:44 +03:00
void
ieee80211_he_cap_ie_to_sta_he_cap ( struct ieee80211_sub_if_data * sdata ,
struct ieee80211_supported_band * sband ,
const u8 * he_cap_ie , u8 he_cap_len ,
2020-05-28 21:34:38 +02:00
const struct ieee80211_he_6ghz_capa * he_6ghz_capa ,
2018-06-09 09:14:44 +03:00
struct sta_info * sta )
{
struct ieee80211_sta_he_cap * he_cap = & sta - > sta . he_cap ;
2021-06-23 20:10:44 +05:30
struct ieee80211_sta_he_cap own_he_cap ;
2018-06-09 09:14:44 +03:00
struct ieee80211_he_cap_elem * he_cap_ie_elem = ( void * ) he_cap_ie ;
u8 he_ppe_size ;
u8 mcs_nss_size ;
u8 he_total_size ;
2020-12-25 11:39:14 +08:00
bool own_160 , peer_160 , own_80p80 , peer_80p80 ;
2018-06-09 09:14:44 +03:00
memset ( he_cap , 0 , sizeof ( * he_cap ) ) ;
2021-06-18 13:41:50 +03:00
if ( ! he_cap_ie | |
! ieee80211_get_he_iftype_cap ( sband ,
ieee80211_vif_type_p2p ( & sdata - > vif ) ) )
2018-06-09 09:14:44 +03:00
return ;
2021-06-23 20:10:44 +05:30
own_he_cap = sband - > iftype_data - > he_cap ;
2018-06-09 09:14:44 +03:00
/* Make sure size is OK */
mcs_nss_size = ieee80211_he_mcs_nss_size ( he_cap_ie_elem ) ;
he_ppe_size =
ieee80211_he_ppe_size ( he_cap_ie [ sizeof ( he_cap - > he_cap_elem ) +
mcs_nss_size ] ,
he_cap_ie_elem - > phy_cap_info ) ;
he_total_size = sizeof ( he_cap - > he_cap_elem ) + mcs_nss_size +
he_ppe_size ;
if ( he_cap_len < he_total_size )
return ;
memcpy ( & he_cap - > he_cap_elem , he_cap_ie , sizeof ( he_cap - > he_cap_elem ) ) ;
/* HE Tx/Rx HE MCS NSS Support Field */
memcpy ( & he_cap - > he_mcs_nss_supp ,
& he_cap_ie [ sizeof ( he_cap - > he_cap_elem ) ] , mcs_nss_size ) ;
/* Check if there are (optional) PPE Thresholds */
if ( he_cap - > he_cap_elem . phy_cap_info [ 6 ] &
IEEE80211_HE_PHY_CAP6_PPE_THRESHOLD_PRESENT )
memcpy ( he_cap - > ppe_thres ,
& he_cap_ie [ sizeof ( he_cap - > he_cap_elem ) + mcs_nss_size ] ,
he_ppe_size ) ;
he_cap - > has_he = true ;
2020-01-31 13:12:41 +02:00
sta - > cur_max_bandwidth = ieee80211_sta_cap_rx_bw ( sta ) ;
sta - > sta . bandwidth = ieee80211_sta_cur_vht_bw ( sta ) ;
2020-05-28 21:34:38 +02:00
if ( sband - > band = = NL80211_BAND_6GHZ & & he_6ghz_capa )
ieee80211_update_from_he_6ghz_capa ( he_6ghz_capa , sta ) ;
2020-12-25 11:39:14 +08:00
ieee80211_he_mcs_intersection ( & own_he_cap . he_mcs_nss_supp . rx_mcs_80 ,
& he_cap - > he_mcs_nss_supp . rx_mcs_80 ,
& own_he_cap . he_mcs_nss_supp . tx_mcs_80 ,
& he_cap - > he_mcs_nss_supp . tx_mcs_80 ) ;
own_160 = own_he_cap . he_cap_elem . phy_cap_info [ 0 ] &
IEEE80211_HE_PHY_CAP0_CHANNEL_WIDTH_SET_160MHZ_IN_5G ;
peer_160 = he_cap - > he_cap_elem . phy_cap_info [ 0 ] &
IEEE80211_HE_PHY_CAP0_CHANNEL_WIDTH_SET_160MHZ_IN_5G ;
if ( peer_160 & & own_160 ) {
ieee80211_he_mcs_intersection ( & own_he_cap . he_mcs_nss_supp . rx_mcs_160 ,
& he_cap - > he_mcs_nss_supp . rx_mcs_160 ,
& own_he_cap . he_mcs_nss_supp . tx_mcs_160 ,
& he_cap - > he_mcs_nss_supp . tx_mcs_160 ) ;
} else if ( peer_160 & & ! own_160 ) {
ieee80211_he_mcs_disable ( & he_cap - > he_mcs_nss_supp . rx_mcs_160 ) ;
ieee80211_he_mcs_disable ( & he_cap - > he_mcs_nss_supp . tx_mcs_160 ) ;
he_cap - > he_cap_elem . phy_cap_info [ 0 ] & =
~ IEEE80211_HE_PHY_CAP0_CHANNEL_WIDTH_SET_160MHZ_IN_5G ;
}
own_80p80 = own_he_cap . he_cap_elem . phy_cap_info [ 0 ] &
IEEE80211_HE_PHY_CAP0_CHANNEL_WIDTH_SET_80PLUS80_MHZ_IN_5G ;
peer_80p80 = he_cap - > he_cap_elem . phy_cap_info [ 0 ] &
IEEE80211_HE_PHY_CAP0_CHANNEL_WIDTH_SET_80PLUS80_MHZ_IN_5G ;
if ( peer_80p80 & & own_80p80 ) {
ieee80211_he_mcs_intersection ( & own_he_cap . he_mcs_nss_supp . rx_mcs_80p80 ,
& he_cap - > he_mcs_nss_supp . rx_mcs_80p80 ,
& own_he_cap . he_mcs_nss_supp . tx_mcs_80p80 ,
& he_cap - > he_mcs_nss_supp . tx_mcs_80p80 ) ;
} else if ( peer_80p80 & & ! own_80p80 ) {
ieee80211_he_mcs_disable ( & he_cap - > he_mcs_nss_supp . rx_mcs_80p80 ) ;
ieee80211_he_mcs_disable ( & he_cap - > he_mcs_nss_supp . tx_mcs_80p80 ) ;
he_cap - > he_cap_elem . phy_cap_info [ 0 ] & =
~ IEEE80211_HE_PHY_CAP0_CHANNEL_WIDTH_SET_80PLUS80_MHZ_IN_5G ;
}
2018-06-09 09:14:44 +03:00
}
2019-07-29 12:23:42 +02:00
void
ieee80211_he_op_ie_to_bss_conf ( struct ieee80211_vif * vif ,
2020-04-24 15:41:39 -07:00
const struct ieee80211_he_operation * he_op_ie )
2019-07-29 12:23:42 +02:00
{
2020-04-24 15:41:39 -07:00
memset ( & vif - > bss_conf . he_oper , 0 , sizeof ( vif - > bss_conf . he_oper ) ) ;
if ( ! he_op_ie )
2019-07-29 12:23:42 +02:00
return ;
2020-04-24 15:41:39 -07:00
vif - > bss_conf . he_oper . params = __le32_to_cpu ( he_op_ie - > he_oper_params ) ;
vif - > bss_conf . he_oper . nss_set = __le16_to_cpu ( he_op_ie - > he_mcs_nss_set ) ;
2019-07-29 12:23:42 +02:00
}
2019-07-30 18:37:01 +02:00
void
ieee80211_he_spr_ie_to_bss_conf ( struct ieee80211_vif * vif ,
const struct ieee80211_he_spr * he_spr_ie_elem )
{
struct ieee80211_he_obss_pd * he_obss_pd =
& vif - > bss_conf . he_obss_pd ;
2019-08-13 09:07:12 +02:00
const u8 * data ;
2019-07-30 18:37:01 +02:00
memset ( he_obss_pd , 0 , sizeof ( * he_obss_pd ) ) ;
if ( ! he_spr_ie_elem )
return ;
2019-08-13 09:07:12 +02:00
data = he_spr_ie_elem - > optional ;
2019-07-30 18:37:01 +02:00
if ( he_spr_ie_elem - > he_sr_control &
IEEE80211_HE_SPR_NON_SRG_OFFSET_PRESENT )
data + + ;
if ( he_spr_ie_elem - > he_sr_control &
IEEE80211_HE_SPR_SRG_INFORMATION_PRESENT ) {
he_obss_pd - > max_offset = * data + + ;
he_obss_pd - > min_offset = * data + + ;
he_obss_pd - > enable = true ;
}
}