Branch data Line data Source code
1 : : // SPDX-License-Identifier: GPL-2.0-only 2 : : /* 3 : : * HE handling 4 : : * 5 : : * Copyright(c) 2017 Intel Deutschland GmbH 6 : : */ 7 : : 8 : : #include "ieee80211_i.h" 9 : : 10 : : void 11 : 0 : ieee80211_he_cap_ie_to_sta_he_cap(struct ieee80211_sub_if_data *sdata, 12 : : struct ieee80211_supported_band *sband, 13 : : const u8 *he_cap_ie, u8 he_cap_len, 14 : : struct sta_info *sta) 15 : : { 16 : 0 : struct ieee80211_sta_he_cap *he_cap = &sta->sta.he_cap; 17 : 0 : struct ieee80211_he_cap_elem *he_cap_ie_elem = (void *)he_cap_ie; 18 : 0 : u8 he_ppe_size; 19 : 0 : u8 mcs_nss_size; 20 : 0 : u8 he_total_size; 21 : : 22 : 0 : memset(he_cap, 0, sizeof(*he_cap)); 23 : : 24 [ # # # # ]: 0 : if (!he_cap_ie || !ieee80211_get_he_sta_cap(sband)) 25 : 0 : return; 26 : : 27 : : /* Make sure size is OK */ 28 [ # # ]: 0 : mcs_nss_size = ieee80211_he_mcs_nss_size(he_cap_ie_elem); 29 : 0 : he_ppe_size = 30 : 0 : ieee80211_he_ppe_size(he_cap_ie[sizeof(he_cap->he_cap_elem) + 31 : : mcs_nss_size], 32 : 0 : he_cap_ie_elem->phy_cap_info); 33 : 0 : he_total_size = sizeof(he_cap->he_cap_elem) + mcs_nss_size + 34 : : he_ppe_size; 35 [ # # ]: 0 : if (he_cap_len < he_total_size) 36 : : return; 37 : : 38 : 0 : memcpy(&he_cap->he_cap_elem, he_cap_ie, sizeof(he_cap->he_cap_elem)); 39 : : 40 : : /* HE Tx/Rx HE MCS NSS Support Field */ 41 : 0 : memcpy(&he_cap->he_mcs_nss_supp, 42 : 0 : &he_cap_ie[sizeof(he_cap->he_cap_elem)], mcs_nss_size); 43 : : 44 : : /* Check if there are (optional) PPE Thresholds */ 45 [ # # ]: 0 : if (he_cap->he_cap_elem.phy_cap_info[6] & 46 : : IEEE80211_HE_PHY_CAP6_PPE_THRESHOLD_PRESENT) 47 : 0 : memcpy(he_cap->ppe_thres, 48 : : &he_cap_ie[sizeof(he_cap->he_cap_elem) + mcs_nss_size], 49 : : he_ppe_size); 50 : : 51 : 0 : he_cap->has_he = true; 52 : : } 53 : : 54 : : void 55 : 0 : ieee80211_he_op_ie_to_bss_conf(struct ieee80211_vif *vif, 56 : : const struct ieee80211_he_operation *he_op_ie_elem) 57 : : { 58 : 0 : struct ieee80211_he_operation *he_operation = 59 : : &vif->bss_conf.he_operation; 60 : : 61 [ # # ]: 0 : if (!he_op_ie_elem) { 62 : 0 : memset(he_operation, 0, sizeof(*he_operation)); 63 : 0 : return; 64 : : } 65 : : 66 : 0 : vif->bss_conf.he_operation = *he_op_ie_elem; 67 : : } 68 : : 69 : : void 70 : 0 : ieee80211_he_spr_ie_to_bss_conf(struct ieee80211_vif *vif, 71 : : const struct ieee80211_he_spr *he_spr_ie_elem) 72 : : { 73 : 0 : struct ieee80211_he_obss_pd *he_obss_pd = 74 : : &vif->bss_conf.he_obss_pd; 75 : 0 : const u8 *data; 76 : : 77 : 0 : memset(he_obss_pd, 0, sizeof(*he_obss_pd)); 78 : : 79 [ # # ]: 0 : if (!he_spr_ie_elem) 80 : : return; 81 : 0 : data = he_spr_ie_elem->optional; 82 : : 83 [ # # ]: 0 : if (he_spr_ie_elem->he_sr_control & 84 : : IEEE80211_HE_SPR_NON_SRG_OFFSET_PRESENT) 85 : 0 : data++; 86 [ # # ]: 0 : if (he_spr_ie_elem->he_sr_control & 87 : : IEEE80211_HE_SPR_SRG_INFORMATION_PRESENT) { 88 : 0 : he_obss_pd->max_offset = *data++; 89 : 0 : he_obss_pd->min_offset = *data++; 90 : 0 : he_obss_pd->enable = true; 91 : : } 92 : : }