|
|
@@ -376,139 +376,6 @@ static int iwl_mvm_send_patterns(struct iwl_mvm *mvm,
|
|
|
return err;
|
|
|
}
|
|
|
|
|
|
-static int iwl_mvm_send_proto_offload(struct iwl_mvm *mvm,
|
|
|
- struct ieee80211_vif *vif)
|
|
|
-{
|
|
|
- union {
|
|
|
- struct iwl_proto_offload_cmd_v1 v1;
|
|
|
- struct iwl_proto_offload_cmd_v2 v2;
|
|
|
- struct iwl_proto_offload_cmd_v3_small v3s;
|
|
|
- struct iwl_proto_offload_cmd_v3_large v3l;
|
|
|
- } cmd = {};
|
|
|
- struct iwl_host_cmd hcmd = {
|
|
|
- .id = PROT_OFFLOAD_CONFIG_CMD,
|
|
|
- .flags = CMD_SYNC,
|
|
|
- .data[0] = &cmd,
|
|
|
- .dataflags[0] = IWL_HCMD_DFL_DUP,
|
|
|
- };
|
|
|
- struct iwl_proto_offload_cmd_common *common;
|
|
|
- u32 enabled = 0, size;
|
|
|
- u32 capa_flags = mvm->fw->ucode_capa.flags;
|
|
|
-#if IS_ENABLED(CONFIG_IPV6)
|
|
|
- struct iwl_mvm_vif *mvmvif = iwl_mvm_vif_from_mac80211(vif);
|
|
|
- int i;
|
|
|
-
|
|
|
- if (capa_flags & IWL_UCODE_TLV_FLAGS_NEW_NSOFFL_SMALL ||
|
|
|
- capa_flags & IWL_UCODE_TLV_FLAGS_NEW_NSOFFL_LARGE) {
|
|
|
- struct iwl_ns_config *nsc;
|
|
|
- struct iwl_targ_addr *addrs;
|
|
|
- int n_nsc, n_addrs;
|
|
|
- int c;
|
|
|
-
|
|
|
- if (capa_flags & IWL_UCODE_TLV_FLAGS_NEW_NSOFFL_SMALL) {
|
|
|
- nsc = cmd.v3s.ns_config;
|
|
|
- n_nsc = IWL_PROTO_OFFLOAD_NUM_NS_CONFIG_V3S;
|
|
|
- addrs = cmd.v3s.targ_addrs;
|
|
|
- n_addrs = IWL_PROTO_OFFLOAD_NUM_IPV6_ADDRS_V3S;
|
|
|
- } else {
|
|
|
- nsc = cmd.v3l.ns_config;
|
|
|
- n_nsc = IWL_PROTO_OFFLOAD_NUM_NS_CONFIG_V3L;
|
|
|
- addrs = cmd.v3l.targ_addrs;
|
|
|
- n_addrs = IWL_PROTO_OFFLOAD_NUM_IPV6_ADDRS_V3L;
|
|
|
- }
|
|
|
-
|
|
|
- if (mvmvif->num_target_ipv6_addrs)
|
|
|
- enabled |= IWL_D3_PROTO_OFFLOAD_NS;
|
|
|
-
|
|
|
- /*
|
|
|
- * For each address we have (and that will fit) fill a target
|
|
|
- * address struct and combine for NS offload structs with the
|
|
|
- * solicited node addresses.
|
|
|
- */
|
|
|
- for (i = 0, c = 0;
|
|
|
- i < mvmvif->num_target_ipv6_addrs &&
|
|
|
- i < n_addrs && c < n_nsc; i++) {
|
|
|
- struct in6_addr solicited_addr;
|
|
|
- int j;
|
|
|
-
|
|
|
- addrconf_addr_solict_mult(&mvmvif->target_ipv6_addrs[i],
|
|
|
- &solicited_addr);
|
|
|
- for (j = 0; j < c; j++)
|
|
|
- if (ipv6_addr_cmp(&nsc[j].dest_ipv6_addr,
|
|
|
- &solicited_addr) == 0)
|
|
|
- break;
|
|
|
- if (j == c)
|
|
|
- c++;
|
|
|
- addrs[i].addr = mvmvif->target_ipv6_addrs[i];
|
|
|
- addrs[i].config_num = cpu_to_le32(j);
|
|
|
- nsc[j].dest_ipv6_addr = solicited_addr;
|
|
|
- memcpy(nsc[j].target_mac_addr, vif->addr, ETH_ALEN);
|
|
|
- }
|
|
|
-
|
|
|
- if (capa_flags & IWL_UCODE_TLV_FLAGS_NEW_NSOFFL_SMALL)
|
|
|
- cmd.v3s.num_valid_ipv6_addrs = cpu_to_le32(i);
|
|
|
- else
|
|
|
- cmd.v3l.num_valid_ipv6_addrs = cpu_to_le32(i);
|
|
|
- } else if (capa_flags & IWL_UCODE_TLV_FLAGS_D3_6_IPV6_ADDRS) {
|
|
|
- if (mvmvif->num_target_ipv6_addrs) {
|
|
|
- enabled |= IWL_D3_PROTO_OFFLOAD_NS;
|
|
|
- memcpy(cmd.v2.ndp_mac_addr, vif->addr, ETH_ALEN);
|
|
|
- }
|
|
|
-
|
|
|
- BUILD_BUG_ON(sizeof(cmd.v2.target_ipv6_addr[0]) !=
|
|
|
- sizeof(mvmvif->target_ipv6_addrs[0]));
|
|
|
-
|
|
|
- for (i = 0; i < min(mvmvif->num_target_ipv6_addrs,
|
|
|
- IWL_PROTO_OFFLOAD_NUM_IPV6_ADDRS_V2); i++)
|
|
|
- memcpy(cmd.v2.target_ipv6_addr[i],
|
|
|
- &mvmvif->target_ipv6_addrs[i],
|
|
|
- sizeof(cmd.v2.target_ipv6_addr[i]));
|
|
|
- } else {
|
|
|
- if (mvmvif->num_target_ipv6_addrs) {
|
|
|
- enabled |= IWL_D3_PROTO_OFFLOAD_NS;
|
|
|
- memcpy(cmd.v1.ndp_mac_addr, vif->addr, ETH_ALEN);
|
|
|
- }
|
|
|
-
|
|
|
- BUILD_BUG_ON(sizeof(cmd.v1.target_ipv6_addr[0]) !=
|
|
|
- sizeof(mvmvif->target_ipv6_addrs[0]));
|
|
|
-
|
|
|
- for (i = 0; i < min(mvmvif->num_target_ipv6_addrs,
|
|
|
- IWL_PROTO_OFFLOAD_NUM_IPV6_ADDRS_V1); i++)
|
|
|
- memcpy(cmd.v1.target_ipv6_addr[i],
|
|
|
- &mvmvif->target_ipv6_addrs[i],
|
|
|
- sizeof(cmd.v1.target_ipv6_addr[i]));
|
|
|
- }
|
|
|
-#endif
|
|
|
-
|
|
|
- if (capa_flags & IWL_UCODE_TLV_FLAGS_NEW_NSOFFL_SMALL) {
|
|
|
- common = &cmd.v3s.common;
|
|
|
- size = sizeof(cmd.v3s);
|
|
|
- } else if (capa_flags & IWL_UCODE_TLV_FLAGS_NEW_NSOFFL_LARGE) {
|
|
|
- common = &cmd.v3l.common;
|
|
|
- size = sizeof(cmd.v3l);
|
|
|
- } else if (capa_flags & IWL_UCODE_TLV_FLAGS_D3_6_IPV6_ADDRS) {
|
|
|
- common = &cmd.v2.common;
|
|
|
- size = sizeof(cmd.v2);
|
|
|
- } else {
|
|
|
- common = &cmd.v1.common;
|
|
|
- size = sizeof(cmd.v1);
|
|
|
- }
|
|
|
-
|
|
|
- if (vif->bss_conf.arp_addr_cnt) {
|
|
|
- enabled |= IWL_D3_PROTO_OFFLOAD_ARP;
|
|
|
- common->host_ipv4_addr = vif->bss_conf.arp_addr_list[0];
|
|
|
- memcpy(common->arp_mac_addr, vif->addr, ETH_ALEN);
|
|
|
- }
|
|
|
-
|
|
|
- if (!enabled)
|
|
|
- return 0;
|
|
|
-
|
|
|
- common->enabled = cpu_to_le32(enabled);
|
|
|
-
|
|
|
- hcmd.len[0] = size;
|
|
|
- return iwl_mvm_send_cmd(mvm, &hcmd);
|
|
|
-}
|
|
|
-
|
|
|
enum iwl_mvm_tcp_packet_type {
|
|
|
MVM_TCP_TX_SYN,
|
|
|
MVM_TCP_RX_SYNACK,
|
|
|
@@ -927,6 +794,20 @@ void iwl_mvm_set_last_nonqos_seq(struct iwl_mvm *mvm, struct ieee80211_vif *vif)
|
|
|
IWL_ERR(mvm, "failed to set non-QoS seqno\n");
|
|
|
}
|
|
|
|
|
|
+static int
|
|
|
+iwl_mvm_send_wowlan_config_cmd(struct iwl_mvm *mvm,
|
|
|
+ const struct iwl_wowlan_config_cmd_v3 *cmd)
|
|
|
+{
|
|
|
+ /* start only with the v2 part of the command */
|
|
|
+ u16 cmd_len = sizeof(cmd->common);
|
|
|
+
|
|
|
+ if (mvm->fw->ucode_capa.api[0] & IWL_UCODE_TLV_API_WOWLAN_CONFIG_TID)
|
|
|
+ cmd_len = sizeof(*cmd);
|
|
|
+
|
|
|
+ return iwl_mvm_send_cmd_pdu(mvm, WOWLAN_CONFIGURATION, CMD_SYNC,
|
|
|
+ cmd_len, cmd);
|
|
|
+}
|
|
|
+
|
|
|
static int __iwl_mvm_suspend(struct ieee80211_hw *hw,
|
|
|
struct cfg80211_wowlan *wowlan,
|
|
|
bool test)
|
|
|
@@ -939,7 +820,7 @@ static int __iwl_mvm_suspend(struct ieee80211_hw *hw,
|
|
|
struct iwl_mvm_vif *mvmvif;
|
|
|
struct ieee80211_sta *ap_sta;
|
|
|
struct iwl_mvm_sta *mvm_ap_sta;
|
|
|
- struct iwl_wowlan_config_cmd wowlan_config_cmd = {};
|
|
|
+ struct iwl_wowlan_config_cmd_v3 wowlan_config_cmd = {};
|
|
|
struct iwl_wowlan_kek_kck_material_cmd kek_kck_cmd = {};
|
|
|
struct iwl_wowlan_tkip_params_cmd tkip_cmd = {};
|
|
|
struct iwl_d3_manager_config d3_cfg_cmd_data = {
|
|
|
@@ -961,7 +842,7 @@ static int __iwl_mvm_suspend(struct ieee80211_hw *hw,
|
|
|
.tkip = &tkip_cmd,
|
|
|
.use_tkip = false,
|
|
|
};
|
|
|
- int ret, i;
|
|
|
+ int ret;
|
|
|
int len __maybe_unused;
|
|
|
|
|
|
if (!wowlan) {
|
|
|
@@ -1002,49 +883,41 @@ static int __iwl_mvm_suspend(struct ieee80211_hw *hw,
|
|
|
|
|
|
mvm_ap_sta = (struct iwl_mvm_sta *)ap_sta->drv_priv;
|
|
|
|
|
|
- /* TODO: wowlan_config_cmd.wowlan_ba_teardown_tids */
|
|
|
+ /* TODO: wowlan_config_cmd.common.wowlan_ba_teardown_tids */
|
|
|
|
|
|
- wowlan_config_cmd.is_11n_connection = ap_sta->ht_cap.ht_supported;
|
|
|
+ wowlan_config_cmd.common.is_11n_connection =
|
|
|
+ ap_sta->ht_cap.ht_supported;
|
|
|
|
|
|
/* Query the last used seqno and set it */
|
|
|
ret = iwl_mvm_get_last_nonqos_seq(mvm, vif);
|
|
|
if (ret < 0)
|
|
|
goto out_noreset;
|
|
|
- wowlan_config_cmd.non_qos_seq = cpu_to_le16(ret);
|
|
|
+ wowlan_config_cmd.common.non_qos_seq = cpu_to_le16(ret);
|
|
|
|
|
|
- /*
|
|
|
- * For QoS counters, we store the one to use next, so subtract 0x10
|
|
|
- * since the uCode will add 0x10 *before* using the value while we
|
|
|
- * increment after using the value (i.e. store the next value to use).
|
|
|
- */
|
|
|
- for (i = 0; i < IWL_MAX_TID_COUNT; i++) {
|
|
|
- u16 seq = mvm_ap_sta->tid_data[i].seq_number;
|
|
|
- seq -= 0x10;
|
|
|
- wowlan_config_cmd.qos_seq[i] = cpu_to_le16(seq);
|
|
|
- }
|
|
|
+ iwl_mvm_set_wowlan_qos_seq(mvm_ap_sta, &wowlan_config_cmd.common);
|
|
|
|
|
|
if (wowlan->disconnect)
|
|
|
- wowlan_config_cmd.wakeup_filter |=
|
|
|
+ wowlan_config_cmd.common.wakeup_filter |=
|
|
|
cpu_to_le32(IWL_WOWLAN_WAKEUP_BEACON_MISS |
|
|
|
IWL_WOWLAN_WAKEUP_LINK_CHANGE);
|
|
|
if (wowlan->magic_pkt)
|
|
|
- wowlan_config_cmd.wakeup_filter |=
|
|
|
+ wowlan_config_cmd.common.wakeup_filter |=
|
|
|
cpu_to_le32(IWL_WOWLAN_WAKEUP_MAGIC_PACKET);
|
|
|
if (wowlan->gtk_rekey_failure)
|
|
|
- wowlan_config_cmd.wakeup_filter |=
|
|
|
+ wowlan_config_cmd.common.wakeup_filter |=
|
|
|
cpu_to_le32(IWL_WOWLAN_WAKEUP_GTK_REKEY_FAIL);
|
|
|
if (wowlan->eap_identity_req)
|
|
|
- wowlan_config_cmd.wakeup_filter |=
|
|
|
+ wowlan_config_cmd.common.wakeup_filter |=
|
|
|
cpu_to_le32(IWL_WOWLAN_WAKEUP_EAP_IDENT_REQ);
|
|
|
if (wowlan->four_way_handshake)
|
|
|
- wowlan_config_cmd.wakeup_filter |=
|
|
|
+ wowlan_config_cmd.common.wakeup_filter |=
|
|
|
cpu_to_le32(IWL_WOWLAN_WAKEUP_4WAY_HANDSHAKE);
|
|
|
if (wowlan->n_patterns)
|
|
|
- wowlan_config_cmd.wakeup_filter |=
|
|
|
+ wowlan_config_cmd.common.wakeup_filter |=
|
|
|
cpu_to_le32(IWL_WOWLAN_WAKEUP_PATTERN_MATCH);
|
|
|
|
|
|
if (wowlan->rfkill_release)
|
|
|
- wowlan_config_cmd.wakeup_filter |=
|
|
|
+ wowlan_config_cmd.common.wakeup_filter |=
|
|
|
cpu_to_le32(IWL_WOWLAN_WAKEUP_RF_KILL_DEASSERT);
|
|
|
|
|
|
if (wowlan->tcp) {
|
|
|
@@ -1052,7 +925,7 @@ static int __iwl_mvm_suspend(struct ieee80211_hw *hw,
|
|
|
* Set the "link change" (really "link lost") flag as well
|
|
|
* since that implies losing the TCP connection.
|
|
|
*/
|
|
|
- wowlan_config_cmd.wakeup_filter |=
|
|
|
+ wowlan_config_cmd.common.wakeup_filter |=
|
|
|
cpu_to_le32(IWL_WOWLAN_WAKEUP_REMOTE_LINK_LOSS |
|
|
|
IWL_WOWLAN_WAKEUP_REMOTE_SIGNATURE_TABLE |
|
|
|
IWL_WOWLAN_WAKEUP_REMOTE_WAKEUP_PACKET |
|
|
|
@@ -1150,9 +1023,7 @@ static int __iwl_mvm_suspend(struct ieee80211_hw *hw,
|
|
|
}
|
|
|
}
|
|
|
|
|
|
- ret = iwl_mvm_send_cmd_pdu(mvm, WOWLAN_CONFIGURATION,
|
|
|
- CMD_SYNC, sizeof(wowlan_config_cmd),
|
|
|
- &wowlan_config_cmd);
|
|
|
+ ret = iwl_mvm_send_wowlan_config_cmd(mvm, &wowlan_config_cmd);
|
|
|
if (ret)
|
|
|
goto out;
|
|
|
|