diff options
Diffstat (limited to 'drivers/net/wireless/intel/iwlwifi/mvm/time-event.c')
| -rw-r--r-- | drivers/net/wireless/intel/iwlwifi/mvm/time-event.c | 189 | 
1 files changed, 188 insertions, 1 deletions
| diff --git a/drivers/net/wireless/intel/iwlwifi/mvm/time-event.c b/drivers/net/wireless/intel/iwlwifi/mvm/time-event.c index a06bc63fb516..51b138673ddb 100644 --- a/drivers/net/wireless/intel/iwlwifi/mvm/time-event.c +++ b/drivers/net/wireless/intel/iwlwifi/mvm/time-event.c @@ -734,6 +734,11 @@ void iwl_mvm_remove_time_event(struct iwl_mvm *mvm,  		return;  } +/* + * When the firmware supports the session protection API, + * this is not needed since it'll automatically remove the + * session protection after association + beacon reception. + */  void iwl_mvm_stop_session_protection(struct iwl_mvm *mvm,  				     struct ieee80211_vif *vif)  { @@ -757,6 +762,101 @@ void iwl_mvm_stop_session_protection(struct iwl_mvm *mvm,  	iwl_mvm_remove_time_event(mvm, mvmvif, te_data);  } +void iwl_mvm_rx_session_protect_notif(struct iwl_mvm *mvm, +				      struct iwl_rx_cmd_buffer *rxb) +{ +	struct iwl_rx_packet *pkt = rxb_addr(rxb); +	struct iwl_mvm_session_prot_notif *notif = (void *)pkt->data; +	struct ieee80211_vif *vif; + +	rcu_read_lock(); +	vif = iwl_mvm_rcu_dereference_vif_id(mvm, le32_to_cpu(notif->mac_id), +					     true); + +	if (!vif) +		goto out_unlock; + +	/* The vif is not a P2P_DEVICE, maintain its time_event_data */ +	if (vif->type != NL80211_IFTYPE_P2P_DEVICE) { +		struct iwl_mvm_vif *mvmvif = iwl_mvm_vif_from_mac80211(vif); +		struct iwl_mvm_time_event_data *te_data = +			&mvmvif->time_event_data; + +		if (!le32_to_cpu(notif->status)) { +			iwl_mvm_te_check_disconnect(mvm, vif, +						    "Session protection failure"); +			iwl_mvm_te_clear_data(mvm, te_data); +		} + +		if (le32_to_cpu(notif->start)) { +			spin_lock_bh(&mvm->time_event_lock); +			te_data->running = le32_to_cpu(notif->start); +			te_data->end_jiffies = +				TU_TO_EXP_TIME(te_data->duration); +			spin_unlock_bh(&mvm->time_event_lock); +		} else { +			/* +			 * By now, we should have finished association +			 * and know the dtim period. +			 */ +			iwl_mvm_te_check_disconnect(mvm, vif, +						    "No beacon heard and the session protection is over already..."); +			iwl_mvm_te_clear_data(mvm, te_data); +		} + +		goto out_unlock; +	} + +	if (!le32_to_cpu(notif->status) || !le32_to_cpu(notif->start)) { +		/* End TE, notify mac80211 */ +		ieee80211_remain_on_channel_expired(mvm->hw); +		set_bit(IWL_MVM_STATUS_NEED_FLUSH_P2P, &mvm->status); +		iwl_mvm_roc_finished(mvm); +	} else if (le32_to_cpu(notif->start)) { +		set_bit(IWL_MVM_STATUS_ROC_RUNNING, &mvm->status); +		ieee80211_ready_on_channel(mvm->hw); /* Start TE */ +	} + + out_unlock: +	rcu_read_unlock(); +} + +static int +iwl_mvm_start_p2p_roc_session_protection(struct iwl_mvm *mvm, +					 struct ieee80211_vif *vif, +					 int duration, +					 enum ieee80211_roc_type type) +{ +	struct iwl_mvm_vif *mvmvif = iwl_mvm_vif_from_mac80211(vif); +	struct iwl_mvm_session_prot_cmd cmd = { +		.id_and_color = +			cpu_to_le32(FW_CMD_ID_AND_COLOR(mvmvif->id, +							mvmvif->color)), +		.action = cpu_to_le32(FW_CTXT_ACTION_ADD), +		.duration_tu = cpu_to_le32(MSEC_TO_TU(duration)), +	}; + +	lockdep_assert_held(&mvm->mutex); + +	switch (type) { +	case IEEE80211_ROC_TYPE_NORMAL: +		cmd.conf_id = +			cpu_to_le32(SESSION_PROTECT_CONF_P2P_DEVICE_DISCOV); +		break; +	case IEEE80211_ROC_TYPE_MGMT_TX: +		cmd.conf_id = +			cpu_to_le32(SESSION_PROTECT_CONF_P2P_GO_NEGOTIATION); +		break; +	default: +		WARN_ONCE(1, "Got an invalid ROC type\n"); +		return -EINVAL; +	} + +	return iwl_mvm_send_cmd_pdu(mvm, iwl_cmd_id(SESSION_PROTECTION_CMD, +						    MAC_CONF_GROUP, 0), +				    0, sizeof(cmd), &cmd); +} +  int iwl_mvm_start_p2p_roc(struct iwl_mvm *mvm, struct ieee80211_vif *vif,  			  int duration, enum ieee80211_roc_type type)  { @@ -770,6 +870,12 @@ int iwl_mvm_start_p2p_roc(struct iwl_mvm *mvm, struct ieee80211_vif *vif,  		return -EBUSY;  	} +	if (fw_has_capa(&mvm->fw->ucode_capa, +			IWL_UCODE_TLV_CAPA_SESSION_PROT_CMD)) +		return iwl_mvm_start_p2p_roc_session_protection(mvm, vif, +								duration, +								type); +  	time_cmd.action = cpu_to_le32(FW_CTXT_ACTION_ADD);  	time_cmd.id_and_color =  		cpu_to_le32(FW_CMD_ID_AND_COLOR(mvmvif->id, mvmvif->color)); @@ -847,11 +953,44 @@ void iwl_mvm_cleanup_roc_te(struct iwl_mvm *mvm)  		__iwl_mvm_remove_time_event(mvm, te_data, &uid);  } -void iwl_mvm_stop_roc(struct iwl_mvm *mvm) +static void iwl_mvm_cancel_session_protection(struct iwl_mvm *mvm, +					      struct iwl_mvm_vif *mvmvif) +{ +	struct iwl_mvm_session_prot_cmd cmd = { +		.id_and_color = +			cpu_to_le32(FW_CMD_ID_AND_COLOR(mvmvif->id, +							mvmvif->color)), +		.action = cpu_to_le32(FW_CTXT_ACTION_REMOVE), +	}; +	int ret; + +	ret = iwl_mvm_send_cmd_pdu(mvm, iwl_cmd_id(SESSION_PROTECTION_CMD, +						   MAC_CONF_GROUP, 0), +				   0, sizeof(cmd), &cmd); +	if (ret) +		IWL_ERR(mvm, +			"Couldn't send the SESSION_PROTECTION_CMD: %d\n", ret); +} + +void iwl_mvm_stop_roc(struct iwl_mvm *mvm, struct ieee80211_vif *vif)  {  	struct iwl_mvm_vif *mvmvif;  	struct iwl_mvm_time_event_data *te_data; +	if (fw_has_capa(&mvm->fw->ucode_capa, +			IWL_UCODE_TLV_CAPA_SESSION_PROT_CMD)) { +		mvmvif = iwl_mvm_vif_from_mac80211(vif); + +		iwl_mvm_cancel_session_protection(mvm, mvmvif); + +		if (vif->type == NL80211_IFTYPE_P2P_DEVICE) +			set_bit(IWL_MVM_STATUS_NEED_FLUSH_P2P, &mvm->status); + +		iwl_mvm_roc_finished(mvm); + +		return; +	} +  	te_data = iwl_mvm_get_roc_te(mvm);  	if (!te_data) {  		IWL_WARN(mvm, "No remain on channel event\n"); @@ -916,3 +1055,51 @@ int iwl_mvm_schedule_csa_period(struct iwl_mvm *mvm,  	return iwl_mvm_time_event_send_add(mvm, vif, te_data, &time_cmd);  } + +void iwl_mvm_schedule_session_protection(struct iwl_mvm *mvm, +					 struct ieee80211_vif *vif, +					 u32 duration, u32 min_duration) +{ +	struct iwl_mvm_vif *mvmvif = iwl_mvm_vif_from_mac80211(vif); +	struct iwl_mvm_time_event_data *te_data = &mvmvif->time_event_data; + +	struct iwl_mvm_session_prot_cmd cmd = { +		.id_and_color = +			cpu_to_le32(FW_CMD_ID_AND_COLOR(mvmvif->id, +							mvmvif->color)), +		.action = cpu_to_le32(FW_CTXT_ACTION_ADD), +		.conf_id = cpu_to_le32(SESSION_PROTECT_CONF_ASSOC), +		.duration_tu = cpu_to_le32(MSEC_TO_TU(duration)), +	}; +	int ret; + +	lockdep_assert_held(&mvm->mutex); + +	spin_lock_bh(&mvm->time_event_lock); +	if (te_data->running && +	    time_after(te_data->end_jiffies, TU_TO_EXP_TIME(min_duration))) { +		IWL_DEBUG_TE(mvm, "We have enough time in the current TE: %u\n", +			     jiffies_to_msecs(te_data->end_jiffies - jiffies)); +		spin_unlock_bh(&mvm->time_event_lock); + +		return; +	} + +	iwl_mvm_te_clear_data(mvm, te_data); +	te_data->duration = le32_to_cpu(cmd.duration_tu); +	spin_unlock_bh(&mvm->time_event_lock); + +	IWL_DEBUG_TE(mvm, "Add new session protection, duration %d TU\n", +		     le32_to_cpu(cmd.duration_tu)); + +	ret = iwl_mvm_send_cmd_pdu(mvm, iwl_cmd_id(SESSION_PROTECTION_CMD, +						   MAC_CONF_GROUP, 0), +				   0, sizeof(cmd), &cmd); +	if (ret) { +		IWL_ERR(mvm, +			"Couldn't send the SESSION_PROTECTION_CMD: %d\n", ret); +		spin_lock_bh(&mvm->time_event_lock); +		iwl_mvm_te_clear_data(mvm, te_data); +		spin_unlock_bh(&mvm->time_event_lock); +	} +} | 

