|
@@ -7,7 +7,7 @@
|
|
|
*
|
|
|
* Copyright(c) 2012 - 2014 Intel Corporation. All rights reserved.
|
|
|
* Copyright(c) 2013 - 2014 Intel Mobile Communications GmbH
|
|
|
- * Copyright(c) 2015 - 2016 Intel Deutschland GmbH
|
|
|
+ * Copyright(c) 2015 - 2017 Intel Deutschland GmbH
|
|
|
*
|
|
|
* This program is free software; you can redistribute it and/or modify
|
|
|
* it under the terms of version 2 of the GNU General Public License as
|
|
@@ -34,7 +34,7 @@
|
|
|
*
|
|
|
* Copyright(c) 2012 - 2014 Intel Corporation. All rights reserved.
|
|
|
* Copyright(c) 2013 - 2014 Intel Mobile Communications GmbH
|
|
|
- * Copyright(c) 2015 - 2016 Intel Deutschland GmbH
|
|
|
+ * Copyright(c) 2015 - 2017 Intel Deutschland GmbH
|
|
|
* All rights reserved.
|
|
|
*
|
|
|
* Redistribution and use in source and binary forms, with or without
|
|
@@ -1048,83 +1048,26 @@ static u32 iwl_mvm_find_ie_offset(u8 *beacon, u8 eid, u32 frame_size)
|
|
|
return ie - beacon;
|
|
|
}
|
|
|
|
|
|
-static int iwl_mvm_mac_ctxt_send_beacon(struct iwl_mvm *mvm,
|
|
|
- struct ieee80211_vif *vif,
|
|
|
- struct sk_buff *beacon)
|
|
|
+static void iwl_mvm_mac_ctxt_set_tx(struct iwl_mvm *mvm,
|
|
|
+ struct ieee80211_vif *vif,
|
|
|
+ struct sk_buff *beacon,
|
|
|
+ struct iwl_tx_cmd *tx)
|
|
|
{
|
|
|
struct iwl_mvm_vif *mvmvif = iwl_mvm_vif_from_mac80211(vif);
|
|
|
- struct iwl_host_cmd cmd = {
|
|
|
- .id = BEACON_TEMPLATE_CMD,
|
|
|
- .flags = CMD_ASYNC,
|
|
|
- };
|
|
|
- union {
|
|
|
- struct iwl_mac_beacon_cmd_v6 beacon_cmd_v6;
|
|
|
- struct iwl_mac_beacon_cmd_v7 beacon_cmd;
|
|
|
- } u = {};
|
|
|
- struct iwl_mac_beacon_cmd beacon_cmd = {};
|
|
|
struct ieee80211_tx_info *info;
|
|
|
- u32 beacon_skb_len;
|
|
|
u32 rate, tx_flags;
|
|
|
|
|
|
- if (WARN_ON(!beacon))
|
|
|
- return -EINVAL;
|
|
|
-
|
|
|
- beacon_skb_len = beacon->len;
|
|
|
-
|
|
|
- if (fw_has_capa(&mvm->fw->ucode_capa,
|
|
|
- IWL_UCODE_TLV_CAPA_CSA_AND_TBTT_OFFLOAD)) {
|
|
|
- u32 csa_offset, ecsa_offset;
|
|
|
-
|
|
|
- csa_offset = iwl_mvm_find_ie_offset(beacon->data,
|
|
|
- WLAN_EID_CHANNEL_SWITCH,
|
|
|
- beacon_skb_len);
|
|
|
- ecsa_offset =
|
|
|
- iwl_mvm_find_ie_offset(beacon->data,
|
|
|
- WLAN_EID_EXT_CHANSWITCH_ANN,
|
|
|
- beacon_skb_len);
|
|
|
-
|
|
|
- if (iwl_mvm_has_new_tx_api(mvm)) {
|
|
|
- beacon_cmd.data.template_id =
|
|
|
- cpu_to_le32((u32)mvmvif->id);
|
|
|
- beacon_cmd.data.ecsa_offset = cpu_to_le32(ecsa_offset);
|
|
|
- beacon_cmd.data.csa_offset = cpu_to_le32(csa_offset);
|
|
|
- beacon_cmd.byte_cnt = cpu_to_le16((u16)beacon_skb_len);
|
|
|
- if (vif->type == NL80211_IFTYPE_AP)
|
|
|
- iwl_mvm_mac_ctxt_set_tim(mvm,
|
|
|
- &beacon_cmd.data.tim_idx,
|
|
|
- &beacon_cmd.data.tim_size,
|
|
|
- beacon->data,
|
|
|
- beacon_skb_len);
|
|
|
- cmd.len[0] = sizeof(beacon_cmd);
|
|
|
- cmd.data[0] = &beacon_cmd;
|
|
|
- goto send;
|
|
|
-
|
|
|
- } else {
|
|
|
- u.beacon_cmd.data.ecsa_offset =
|
|
|
- cpu_to_le32(ecsa_offset);
|
|
|
- u.beacon_cmd.data.csa_offset = cpu_to_le32(csa_offset);
|
|
|
- cmd.len[0] = sizeof(u.beacon_cmd);
|
|
|
- cmd.data[0] = &u;
|
|
|
- }
|
|
|
- } else {
|
|
|
- cmd.len[0] = sizeof(u.beacon_cmd_v6);
|
|
|
- cmd.data[0] = &u;
|
|
|
- }
|
|
|
-
|
|
|
- /* TODO: for now the beacon template id is set to be the mac context id.
|
|
|
- * Might be better to handle it as another resource ... */
|
|
|
- u.beacon_cmd_v6.template_id = cpu_to_le32((u32)mvmvif->id);
|
|
|
info = IEEE80211_SKB_CB(beacon);
|
|
|
|
|
|
/* Set up TX command fields */
|
|
|
- u.beacon_cmd_v6.tx.len = cpu_to_le16((u16)beacon_skb_len);
|
|
|
- u.beacon_cmd_v6.tx.sta_id = mvmvif->bcast_sta.sta_id;
|
|
|
- u.beacon_cmd_v6.tx.life_time = cpu_to_le32(TX_CMD_LIFE_TIME_INFINITE);
|
|
|
+ tx->len = cpu_to_le16((u16)beacon->len);
|
|
|
+ tx->sta_id = mvmvif->bcast_sta.sta_id;
|
|
|
+ tx->life_time = cpu_to_le32(TX_CMD_LIFE_TIME_INFINITE);
|
|
|
tx_flags = TX_CMD_FLG_SEQ_CTL | TX_CMD_FLG_TSF;
|
|
|
tx_flags |=
|
|
|
iwl_mvm_bt_coex_tx_prio(mvm, (void *)beacon->data, info, 0) <<
|
|
|
TX_CMD_FLG_BT_PRIO_POS;
|
|
|
- u.beacon_cmd_v6.tx.tx_flags = cpu_to_le32(tx_flags);
|
|
|
+ tx->tx_flags = cpu_to_le32(tx_flags);
|
|
|
|
|
|
if (!fw_has_capa(&mvm->fw->ucode_capa,
|
|
|
IWL_UCODE_TLV_CAPA_BEACON_ANT_SELECTION)) {
|
|
@@ -1133,7 +1076,7 @@ static int iwl_mvm_mac_ctxt_send_beacon(struct iwl_mvm *mvm,
|
|
|
mvm->mgmt_last_antenna_idx);
|
|
|
}
|
|
|
|
|
|
- u.beacon_cmd_v6.tx.rate_n_flags =
|
|
|
+ tx->rate_n_flags =
|
|
|
cpu_to_le32(BIT(mvm->mgmt_last_antenna_idx) <<
|
|
|
RATE_MCS_ANT_POS);
|
|
|
|
|
@@ -1141,29 +1084,126 @@ static int iwl_mvm_mac_ctxt_send_beacon(struct iwl_mvm *mvm,
|
|
|
rate = IWL_FIRST_OFDM_RATE;
|
|
|
} else {
|
|
|
rate = IWL_FIRST_CCK_RATE;
|
|
|
- u.beacon_cmd_v6.tx.rate_n_flags |=
|
|
|
- cpu_to_le32(RATE_MCS_CCK_MSK);
|
|
|
+ tx->rate_n_flags |= cpu_to_le32(RATE_MCS_CCK_MSK);
|
|
|
}
|
|
|
- u.beacon_cmd_v6.tx.rate_n_flags |=
|
|
|
- cpu_to_le32(iwl_mvm_mac80211_idx_to_hwrate(rate));
|
|
|
|
|
|
- /* Set up TX beacon command fields */
|
|
|
- if (vif->type == NL80211_IFTYPE_AP)
|
|
|
- iwl_mvm_mac_ctxt_set_tim(mvm, &u.beacon_cmd_v6.tim_idx,
|
|
|
- &u.beacon_cmd_v6.tim_size,
|
|
|
- beacon->data,
|
|
|
- beacon_skb_len);
|
|
|
+ tx->rate_n_flags |= cpu_to_le32(iwl_mvm_mac80211_idx_to_hwrate(rate));
|
|
|
+}
|
|
|
+
|
|
|
+static int iwl_mvm_mac_ctxt_send_beacon_cmd(struct iwl_mvm *mvm,
|
|
|
+ struct sk_buff *beacon,
|
|
|
+ void *data, int len)
|
|
|
+{
|
|
|
+ struct iwl_host_cmd cmd = {
|
|
|
+ .id = BEACON_TEMPLATE_CMD,
|
|
|
+ .flags = CMD_ASYNC,
|
|
|
+ };
|
|
|
|
|
|
-send:
|
|
|
- /* Submit command */
|
|
|
+ cmd.len[0] = len;
|
|
|
+ cmd.data[0] = data;
|
|
|
cmd.dataflags[0] = 0;
|
|
|
- cmd.len[1] = beacon_skb_len;
|
|
|
+ cmd.len[1] = beacon->len;
|
|
|
cmd.data[1] = beacon->data;
|
|
|
cmd.dataflags[1] = IWL_HCMD_DFL_DUP;
|
|
|
|
|
|
return iwl_mvm_send_cmd(mvm, &cmd);
|
|
|
}
|
|
|
|
|
|
+static int iwl_mvm_mac_ctxt_send_beacon_v6(struct iwl_mvm *mvm,
|
|
|
+ struct ieee80211_vif *vif,
|
|
|
+ struct sk_buff *beacon)
|
|
|
+{
|
|
|
+ struct iwl_mvm_vif *mvmvif = iwl_mvm_vif_from_mac80211(vif);
|
|
|
+ struct iwl_mac_beacon_cmd_v6 beacon_cmd = {};
|
|
|
+
|
|
|
+ iwl_mvm_mac_ctxt_set_tx(mvm, vif, beacon, &beacon_cmd.tx);
|
|
|
+
|
|
|
+ beacon_cmd.template_id = cpu_to_le32((u32)mvmvif->id);
|
|
|
+
|
|
|
+ if (vif->type == NL80211_IFTYPE_AP)
|
|
|
+ iwl_mvm_mac_ctxt_set_tim(mvm, &beacon_cmd.tim_idx,
|
|
|
+ &beacon_cmd.tim_size,
|
|
|
+ beacon->data, beacon->len);
|
|
|
+
|
|
|
+ return iwl_mvm_mac_ctxt_send_beacon_cmd(mvm, beacon, &beacon_cmd,
|
|
|
+ sizeof(beacon_cmd));
|
|
|
+}
|
|
|
+
|
|
|
+static int iwl_mvm_mac_ctxt_send_beacon_v7(struct iwl_mvm *mvm,
|
|
|
+ struct ieee80211_vif *vif,
|
|
|
+ struct sk_buff *beacon)
|
|
|
+{
|
|
|
+ struct iwl_mvm_vif *mvmvif = iwl_mvm_vif_from_mac80211(vif);
|
|
|
+ struct iwl_mac_beacon_cmd_v7 beacon_cmd = {};
|
|
|
+
|
|
|
+ iwl_mvm_mac_ctxt_set_tx(mvm, vif, beacon, &beacon_cmd.tx);
|
|
|
+
|
|
|
+ beacon_cmd.template_id = cpu_to_le32((u32)mvmvif->id);
|
|
|
+
|
|
|
+ if (vif->type == NL80211_IFTYPE_AP)
|
|
|
+ iwl_mvm_mac_ctxt_set_tim(mvm, &beacon_cmd.tim_idx,
|
|
|
+ &beacon_cmd.tim_size,
|
|
|
+ beacon->data, beacon->len);
|
|
|
+
|
|
|
+ beacon_cmd.csa_offset =
|
|
|
+ cpu_to_le32(iwl_mvm_find_ie_offset(beacon->data,
|
|
|
+ WLAN_EID_CHANNEL_SWITCH,
|
|
|
+ beacon->len));
|
|
|
+ beacon_cmd.ecsa_offset =
|
|
|
+ cpu_to_le32(iwl_mvm_find_ie_offset(beacon->data,
|
|
|
+ WLAN_EID_EXT_CHANSWITCH_ANN,
|
|
|
+ beacon->len));
|
|
|
+
|
|
|
+ return iwl_mvm_mac_ctxt_send_beacon_cmd(mvm, beacon, &beacon_cmd,
|
|
|
+ sizeof(beacon_cmd));
|
|
|
+}
|
|
|
+
|
|
|
+static int iwl_mvm_mac_ctxt_send_beacon_v8(struct iwl_mvm *mvm,
|
|
|
+ struct ieee80211_vif *vif,
|
|
|
+ struct sk_buff *beacon)
|
|
|
+{
|
|
|
+ struct iwl_mvm_vif *mvmvif = iwl_mvm_vif_from_mac80211(vif);
|
|
|
+ struct iwl_mac_beacon_cmd beacon_cmd = {};
|
|
|
+
|
|
|
+ beacon_cmd.byte_cnt = cpu_to_le16((u16)beacon->len);
|
|
|
+ beacon_cmd.template_id = cpu_to_le32((u32)mvmvif->id);
|
|
|
+
|
|
|
+ if (vif->type == NL80211_IFTYPE_AP)
|
|
|
+ iwl_mvm_mac_ctxt_set_tim(mvm,
|
|
|
+ &beacon_cmd.tim_idx,
|
|
|
+ &beacon_cmd.tim_size,
|
|
|
+ beacon->data, beacon->len);
|
|
|
+
|
|
|
+ beacon_cmd.csa_offset =
|
|
|
+ cpu_to_le32(iwl_mvm_find_ie_offset(beacon->data,
|
|
|
+ WLAN_EID_CHANNEL_SWITCH,
|
|
|
+ beacon->len));
|
|
|
+ beacon_cmd.ecsa_offset =
|
|
|
+ cpu_to_le32(iwl_mvm_find_ie_offset(beacon->data,
|
|
|
+ WLAN_EID_EXT_CHANSWITCH_ANN,
|
|
|
+ beacon->len));
|
|
|
+
|
|
|
+ return iwl_mvm_mac_ctxt_send_beacon_cmd(mvm, beacon, &beacon_cmd,
|
|
|
+ sizeof(beacon_cmd));
|
|
|
+}
|
|
|
+
|
|
|
+static int iwl_mvm_mac_ctxt_send_beacon(struct iwl_mvm *mvm,
|
|
|
+ struct ieee80211_vif *vif,
|
|
|
+ struct sk_buff *beacon)
|
|
|
+{
|
|
|
+ if (WARN_ON(!beacon))
|
|
|
+ return -EINVAL;
|
|
|
+
|
|
|
+ if (!fw_has_capa(&mvm->fw->ucode_capa,
|
|
|
+ IWL_UCODE_TLV_CAPA_CSA_AND_TBTT_OFFLOAD))
|
|
|
+ return iwl_mvm_mac_ctxt_send_beacon_v6(mvm, vif, beacon);
|
|
|
+
|
|
|
+ if (!iwl_mvm_has_new_tx_api(mvm))
|
|
|
+ return iwl_mvm_mac_ctxt_send_beacon_v7(mvm, vif, beacon);
|
|
|
+
|
|
|
+ return iwl_mvm_mac_ctxt_send_beacon_v8(mvm, vif, beacon);
|
|
|
+}
|
|
|
+
|
|
|
/* The beacon template for the AP/GO/IBSS has changed and needs update */
|
|
|
int iwl_mvm_mac_ctxt_beacon_changed(struct iwl_mvm *mvm,
|
|
|
struct ieee80211_vif *vif)
|