Sfoglia il codice sorgente

qcacld-3.0: Bring up SAP in VLP if country supports VLP

In STA+SAP concurrency, if the STA is connected in indoor
power, and if the country/channel supports VLP, then bringup
the SAP in VLP. Move the STA to VLP as well.

Restore the power of the STA, when the SAP is disconnected.

Add changes to:
a) Decide the power type for the concurrency during start bss,
stop bss, change channel request calls of the SAP interface.

b) Allow the 6 GHz SCC channel in ACS computation if the channel
supports VLP power or if the channel is indoor and enabled.

Change-Id: I151e2e3e8910a406bb5c1526f4f01715854d173f
CRs-Fixed: 3268100
Surya Prakash Sivaraj 2 anni fa
parent
commit
538f94a3fa

+ 20 - 8
components/cmn_services/policy_mgr/src/wlan_policy_mgr_pcl.c

@@ -957,6 +957,7 @@ policy_mgr_modify_sap_pcl_for_6G_channels(struct wlan_objmgr_psoc *psoc,
 	struct wlan_objmgr_vdev *vdev;
 	qdf_freq_t sta_gc_freq = 0;
 	uint32_t ap_pwr_type_6g = 0;
+	bool indoor_ch_support = false;
 
 	pm_ctx = policy_mgr_get_context(psoc);
 	if (!pm_ctx) {
@@ -993,23 +994,33 @@ policy_mgr_modify_sap_pcl_for_6G_channels(struct wlan_objmgr_psoc *psoc,
 		return QDF_STATUS_E_FAILURE;
 	}
 
-	/* If STA is present in 6GHz, STA+SAP SCC is allowed
-	 * only on PSC channels in VLP mode. Therefore, remove
-	 * all other 6GHz channels from the PCL list.
+	/* If STA is present in 6GHz PSC, STA+SAP SCC is allowed
+	 * only for the following combinations:
 	 *
-	 * VLP STA in PSC + SAP     - Allowed
-	 * VLP STA in non-PSC + SAP - Not allowed
-	 * non-VLP STA + SAP        - Not allowed
+	 * VLP STA + SAP - Allowed with VLP Power
+	 * LPI STA + SAP - Allowed with VLP power if channel supports VLP.
+	 * LPI STA + SAP - Allowed with LPI power if gindoor_channel_support=1
 	 */
 	ap_pwr_type_6g = wlan_mlme_get_6g_ap_power_type(vdev);
 	policy_mgr_debug("STA power type : %d", ap_pwr_type_6g);
+
+	ucfg_mlme_get_indoor_channel_support(psoc, &indoor_ch_support);
+
 	for (i = 0; i < *pcl_len_org; i++) {
 		if (WLAN_REG_IS_6GHZ_CHAN_FREQ(pcl_list_org[i])) {
-			if (ap_pwr_type_6g != REG_VERY_LOW_POWER_AP)
+			if (!WLAN_REG_IS_6GHZ_PSC_CHAN_FREQ(pcl_list_org[i]))
 				continue;
-			else if (!WLAN_REG_IS_6GHZ_PSC_CHAN_FREQ(pcl_list_org[i]))
+			if (ap_pwr_type_6g == REG_VERY_LOW_POWER_AP)
+				goto add_freq;
+			else if (ap_pwr_type_6g == REG_INDOOR_AP &&
+				 (!wlan_reg_is_freq_indoor(pm_ctx->pdev,
+							   pcl_list_org[i]) ||
+				  indoor_ch_support))
+				goto add_freq;
+			else
 				continue;
 		}
+add_freq:
 		pcl_list[pcl_len] = pcl_list_org[i];
 		weight_list[pcl_len++] = weight_list_org[i];
 	}
@@ -3359,6 +3370,7 @@ policy_mgr_get_sap_mandatory_channel(struct wlan_objmgr_psoc *psoc,
 		 * Indoor channel to restart SAP in SCC.
 		 */
 		if (wlan_reg_is_freq_indoor(pm_ctx->pdev, pcl.pcl_list[i]) &&
+		    !WLAN_REG_IS_6GHZ_CHAN_FREQ(pcl.pcl_list[i]) &&
 		    sta_sap_scc_on_indoor_channel) {
 			sap_new_freq = pcl.pcl_list[i];
 			policy_mgr_debug("Choose Indoor channel from PCL list %d sap_new_freq %d",

+ 41 - 0
components/mlme/core/inc/wlan_mlme_main.h

@@ -406,9 +406,14 @@ struct wait_for_key_timer {
  * struct mlme_ap_config - VDEV MLME legacy private SAP
  * related configurations
  * @user_config_sap_ch_freq : Frequency from userspace to start SAP
+ * @update_required_scc_sta_power: Change the 6 GHz power type of the
+ * concurrent STA
  */
 struct mlme_ap_config {
 	qdf_freq_t user_config_sap_ch_freq;
+#ifdef CONFIG_BAND_6GHZ
+	bool update_required_scc_sta_power;
+#endif
 };
 
 /**
@@ -1182,6 +1187,42 @@ QDF_STATUS
 wlan_set_sap_user_config_freq(struct wlan_objmgr_vdev *vdev,
 			      qdf_freq_t freq);
 
+#ifdef CONFIG_BAND_6GHZ
+/**
+ * wlan_get_tpc_update_required_for_sta() - Get the tpc update required config
+ * to identify whether the tpc power has changed for concurrent STA interface
+ *
+ * @vdev: pointer to SAP vdev
+ *
+ * Return: Change scc power config
+ */
+bool
+wlan_get_tpc_update_required_for_sta(struct wlan_objmgr_vdev *vdev);
+
+/**
+ * wlan_set_tpc_update_required_for_sta() - Set the tpc update required config
+ * for the concurrent STA interface
+ *
+ * @vdev:   pointer to SAP vdev
+ * @value:  change scc power config
+ *
+ * Return: QDF_STATUS
+ */
+QDF_STATUS
+wlan_set_tpc_update_required_for_sta(struct wlan_objmgr_vdev *vdev, bool value);
+#else
+static inline bool
+wlan_get_tpc_update_required_for_sta(struct wlan_objmgr_vdev *vdev)
+{
+	return false;
+}
+
+static inline QDF_STATUS
+wlan_set_tpc_update_required_for_sta(struct wlan_objmgr_vdev *vdev, bool value)
+{
+	return QDF_STATUS_SUCCESS;
+}
+#endif
 #ifdef WLAN_FEATURE_ROAM_OFFLOAD
 /**
  * wlan_mlme_defer_pmk_set_in_roaming() - Set the set_key pending status

+ 47 - 0
components/mlme/core/src/wlan_mlme_main.c

@@ -4115,3 +4115,50 @@ wlan_set_sap_user_config_freq(struct wlan_objmgr_vdev *vdev,
 	mlme_priv->mlme_ap.user_config_sap_ch_freq = freq;
 	return QDF_STATUS_SUCCESS;
 }
+
+#ifdef CONFIG_BAND_6GHZ
+bool
+wlan_get_tpc_update_required_for_sta(struct wlan_objmgr_vdev *vdev)
+{
+	struct mlme_legacy_priv *mlme_priv;
+	enum QDF_OPMODE opmode;
+
+	mlme_priv = wlan_vdev_mlme_get_ext_hdl(vdev);
+	if (!mlme_priv) {
+		mlme_legacy_err("vdev legacy private object is NULL");
+		return false;
+	}
+
+	opmode = wlan_vdev_mlme_get_opmode(vdev);
+	if (opmode != QDF_SAP_MODE && opmode != QDF_P2P_GO_MODE) {
+		mlme_debug("Invalid opmode %d", opmode);
+		return false;
+	}
+
+	return mlme_priv->mlme_ap.update_required_scc_sta_power;
+}
+
+QDF_STATUS
+wlan_set_tpc_update_required_for_sta(struct wlan_objmgr_vdev *vdev, bool value)
+{
+	struct mlme_legacy_priv *mlme_priv;
+	enum QDF_OPMODE opmode;
+
+	mlme_priv = wlan_vdev_mlme_get_ext_hdl(vdev);
+	if (!mlme_priv) {
+		mlme_legacy_err("vdev legacy private object is NULL");
+		return QDF_STATUS_E_INVAL;
+	}
+
+	opmode = wlan_vdev_mlme_get_opmode(vdev);
+	if (opmode != QDF_SAP_MODE && opmode != QDF_P2P_GO_MODE) {
+		mlme_debug("Invalid mode %d", opmode);
+		QDF_ASSERT(0);
+		return QDF_STATUS_E_FAILURE;
+	}
+
+	mlme_priv->mlme_ap.update_required_scc_sta_power = value;
+	mlme_debug("Set change scc power as %d", value);
+	return QDF_STATUS_SUCCESS;
+}
+#endif

+ 0 - 4
core/hdd/src/wlan_hdd_hostapd.c

@@ -2118,10 +2118,6 @@ QDF_STATUS hdd_hostapd_sap_event_cb(struct sap_event *sap_event,
 				hdd_debug("set hw mode change not done");
 		}
 
-		if (!wlan_reg_is_6ghz_chan_freq(ap_ctx->operating_chan_freq))
-			wlan_reg_set_ap_pwr_and_update_chan_list(hdd_ctx->pdev,
-								 REG_INDOOR_AP);
-
 		/*
 		 * Enable wds source port learning on the dp vdev in AP mode
 		 * when WDS feature is enabled.

+ 1 - 0
core/mac/src/pe/include/lim_session.h

@@ -802,6 +802,7 @@ struct pe_session {
 	bool same_ctry_code;  /* If AP Country IE has same country code as */
 	/* STA programmed country */
 	uint8_t ap_power_type_6g;  /* AP power type for 6G (LPI, SP, or VLP) */
+	bool sta_follows_sap_power;
 #ifdef WLAN_FEATURE_11BE
 	bool eht_capable;
 	tDot11fIEeht_cap eht_config;

+ 5 - 1
core/mac/src/pe/lim/lim_process_action_frame.c

@@ -1156,8 +1156,12 @@ __lim_process_link_measurement_req(struct mac_context *mac, uint8_t *pRxPacketIn
 		pe_debug("There were warnings while unpacking a Link Measure request (0x%08x, %d bytes):",
 			nStatus, frameLen);
 	}
-	/* Call rrm function to handle the request. */
 
+	if (pe_session->sta_follows_sap_power) {
+		pe_debug("STA power has changed, reject the link measurement request");
+		return QDF_STATUS_E_FAILURE;
+	}
+	/* Call rrm function to handle the request. */
 	return rrm_process_link_measurement_request(mac, pRxPacketInfo, &frm,
 					     pe_session);
 

+ 36 - 1
core/mac/src/pe/lim/lim_process_mlm_rsp_messages.c

@@ -2352,6 +2352,7 @@ void lim_handle_add_bss_rsp(struct mac_context *mac_ctx,
 	enum bss_type bss_type;
 	struct wlan_lmac_if_reg_tx_ops *tx_ops;
 	struct vdev_mlme_obj *mlme_obj;
+	struct pe_session *sta_session;
 
 	if (!add_bss_rsp) {
 		pe_err("add_bss_rsp is NULL");
@@ -2391,6 +2392,23 @@ void lim_handle_add_bss_rsp(struct mac_context *mac_ctx,
 				tx_ops->set_tpc_power(mac_ctx->psoc,
 						      session_entry->vdev_id,
 						      &mlme_obj->reg_tpc_obj);
+			if (wlan_get_tpc_update_required_for_sta(
+							session_entry->vdev)) {
+				sta_session =
+					lim_get_concurrent_session(mac_ctx,
+							   session_entry->vdev_id,
+							   session_entry->opmode);
+				if (!sta_session) {
+					pe_err("TPC update required is set, but concurrent session doesn't exist");
+					wlan_set_tpc_update_required_for_sta(
+							session_entry->vdev,
+							false);
+				} else {
+					lim_update_tx_power(mac_ctx,
+						    session_entry, sta_session,
+						    false);
+				}
+			}
 		}
 	}
 	bss_type = session_entry->bssType;
@@ -2850,6 +2868,7 @@ static void lim_process_switch_channel_join_req(
 	struct wlan_lmac_if_reg_tx_ops *tx_ops;
 	bool tpe_change = false;
 	QDF_STATUS mlo_status;
+	struct pe_session *sap_session;
 
 	if (status != QDF_STATUS_SUCCESS) {
 		pe_err("Change channel failed!!");
@@ -2960,7 +2979,22 @@ static void lim_process_switch_channel_join_req(
 		goto error;
 	}
 
-	if (wlan_reg_is_ext_tpc_supported(mac_ctx->psoc)) {
+	sap_session =
+		lim_get_concurrent_session(mac_ctx, session_entry->vdev_id,
+					   session_entry->opmode);
+
+	/*
+	 * STA LPI + SAP VLP is supported. For this, STA should move to
+	 * VLP power.
+	 * If there is a concurrent SAP operating on VLP in the same channel,
+	 * then do not update the TPC if the connecting AP is in LPI.
+	 */
+	if (sap_session &&
+	    lim_is_power_change_required_for_sta(mac_ctx, session_entry, sap_session))
+		lim_update_tx_power(mac_ctx, sap_session, session_entry, false);
+
+	if (wlan_reg_is_ext_tpc_supported(mac_ctx->psoc) &&
+	    !session_entry->sta_follows_sap_power) {
 		tx_ops = wlan_reg_get_tx_ops(mac_ctx->psoc);
 
 		lim_process_tpe_ie_from_beacon(mac_ctx, session_entry, bss,
@@ -3159,6 +3193,7 @@ void lim_process_switch_channel_rsp(struct mac_context *mac,
 		 */
 		policy_mgr_update_connection_info(mac->psoc,
 						pe_session->smeSessionId);
+		lim_check_conc_power_for_csa(mac, pe_session);
 		break;
 	case LIM_SWITCH_CHANNEL_MONITOR:
 		lim_handle_mon_switch_channel_rsp(pe_session, status);

+ 11 - 0
core/mac/src/pe/lim/lim_process_sme_req_messages.c

@@ -5143,6 +5143,11 @@ void lim_parse_tpe_ie(struct mac_context *mac, struct pe_session *session,
 	if (!vdev_mlme)
 		return;
 
+	if (session->sta_follows_sap_power) {
+		pe_debug_rl("STA operates in 6 GHz power of SAP, do not update STA power");
+		return;
+	}
+
 	vdev_mlme->reg_tpc_obj.num_pwr_levels = 0;
 	*has_tpe_updated = false;
 
@@ -5444,6 +5449,11 @@ void lim_calculate_tpc(struct mac_context *mac,
 		return;
 	}
 
+	if (session->sta_follows_sap_power) {
+		pe_debug_rl("STA operates in 6 GHz power of SAP, do not update STA power");
+		return;
+	}
+
 	oper_freq = session->curr_op_freq;
 	bw_val = wlan_reg_get_bw_value(session->ch_width);
 
@@ -8534,6 +8544,7 @@ static void lim_process_sme_channel_change_request(struct mac_context *mac_ctx,
 		     &ch_change_req->ext_rates,
 		     sizeof(session_entry->extRateSet));
 	lim_change_channel(mac_ctx, session_entry);
+	lim_check_conc_power_for_csa(mac_ctx, session_entry);
 }
 
 /******************************************************************************

+ 17 - 0
core/mac/src/pe/lim/lim_send_sme_rsp_messages.c

@@ -92,6 +92,7 @@ lim_send_stop_bss_response(struct mac_context *mac_ctx, uint8_t vdev_id,
 	struct scheduler_msg msg = {0};
 	struct stop_bss_rsp *stop_bss_rsp;
 	struct pe_session *pe_session;
+	struct pe_session *sta_session;
 
 	pe_debug("Sending stop bss response with reasonCode: %s",
 		 lim_result_code_str(result_code));
@@ -102,6 +103,22 @@ lim_send_stop_bss_response(struct mac_context *mac_ctx, uint8_t vdev_id,
 		return;
 	}
 
+	/*
+	 * STA LPI + SAP VLP is supported. For this STA should operate in VLP
+	 * power level of the SAP.
+	 *
+	 * For the STA, if the TPC is changed to VLP, then restore the original
+	 * power for the STA when SAP disconnects.
+	 */
+	if (wlan_get_tpc_update_required_for_sta(pe_session->vdev)) {
+		sta_session = lim_get_concurrent_session(mac_ctx, vdev_id,
+							 pe_session->opmode);
+		if (sta_session &&
+		    sta_session->curr_op_freq == pe_session->curr_op_freq)
+			lim_update_tx_power(mac_ctx, pe_session,
+					    sta_session, true);
+	}
+
 	stop_bss_rsp = qdf_mem_malloc(sizeof(*stop_bss_rsp));
 	if (!stop_bss_rsp)
 		return;

+ 1 - 0
core/mac/src/pe/lim/lim_session.c

@@ -843,6 +843,7 @@ void pe_delete_session(struct mac_context *mac_ctx, struct pe_session *session)
 
 	lim_reset_bcn_probe_filter(mac_ctx, session);
 	lim_sae_auth_cleanup_retry(mac_ctx, session->vdev_id);
+	lim_cleanup_power_change(mac_ctx, session);
 
 	/* Restore default failure timeout */
 	if (session->defaultAuthFailureTimeout) {

+ 264 - 0
core/mac/src/pe/lim/lim_utils.c

@@ -10672,3 +10672,267 @@ uint8_t lim_get_vht_ch_width(tDot11fIEVHTCaps *vht_cap,
 	pe_debug("The VHT Operation channel width is %d", ch_width);
 	return ch_width;
 }
+
+/*
+ * lim_set_tpc_power() - Function to compute and send TPC power level to the
+ * FW based on the opmode of the pe_session
+ *
+ * @mac_ctx:    Pointer to Global MAC structure
+ * @pe_session: Pointer to session
+ *
+ * Return: TPC status
+ */
+static bool
+lim_set_tpc_power(struct mac_context *mac_ctx, struct pe_session *session)
+{
+	struct wlan_lmac_if_reg_tx_ops *tx_ops;
+	struct vdev_mlme_obj *mlme_obj;
+	bool tpe_change = false;
+
+	if (!wlan_reg_is_ext_tpc_supported(mac_ctx->psoc))
+		return true;
+	tx_ops = wlan_reg_get_tx_ops(mac_ctx->psoc);
+
+	if (!tx_ops || !tx_ops->set_tpc_power)
+		return false;
+
+	mlme_obj = wlan_vdev_mlme_get_cmpt_obj(session->vdev);
+	if (!mlme_obj)
+		return false;
+
+	if (session->opmode == QDF_STA_MODE ||
+	    session->opmode == QDF_P2P_CLIENT_MODE)
+		lim_process_tpe_ie_from_beacon(mac_ctx, session,
+				       &session->lim_join_req->bssDescription,
+				       &tpe_change);
+
+	if (session->opmode == QDF_SAP_MODE ||
+	    session->opmode == QDF_P2P_GO_MODE)
+		mlme_obj->reg_tpc_obj.num_pwr_levels = 0;
+
+	lim_calculate_tpc(mac_ctx, session, false, 0, false);
+
+	tx_ops->set_tpc_power(mac_ctx->psoc, session->vdev_id,
+			      &mlme_obj->reg_tpc_obj);
+	return true;
+}
+
+/*
+ * lim_get_tx_power() - Function to get the Tx power of the center frequency
+ * of the sap interface.
+ *
+ * @reg_tpc: reg_tpc mlme obj pointer
+ * @freq: center frequency of the SAP.
+ *
+ * Return: tx power
+ */
+static uint8_t
+lim_get_tx_power(struct reg_tpc_power_info *reg_tpc, qdf_freq_t freq)
+{
+	int i;
+
+	for (i = 0; i < reg_tpc->num_pwr_levels; i++) {
+		if (reg_tpc->chan_power_info[i].chan_cfreq == freq)
+			return reg_tpc->chan_power_info[i].tx_power;
+	}
+
+	return 0;
+}
+
+struct pe_session *
+lim_get_concurrent_session(struct mac_context *mac_ctx, uint8_t vdev_id,
+			   enum QDF_OPMODE opmode)
+{
+	uint8_t mac_id, conc_vdev_id;
+	struct pe_session *session;
+
+	policy_mgr_get_mac_id_by_session_id(mac_ctx->psoc, vdev_id, &mac_id);
+
+	conc_vdev_id = policy_mgr_get_conc_vdev_on_same_mac(mac_ctx->psoc,
+							    vdev_id, mac_id);
+
+	session = pe_find_session_by_vdev_id(mac_ctx, conc_vdev_id);
+	if (!session)
+		return NULL;
+
+	switch (opmode) {
+	case QDF_STA_MODE:
+	case QDF_P2P_CLIENT_MODE:
+		if (session->opmode != QDF_SAP_MODE &&
+		    session->opmode != QDF_P2P_GO_MODE)
+			return NULL;
+		break;
+	case QDF_SAP_MODE:
+	case QDF_P2P_GO_MODE:
+		if (session->opmode != QDF_STA_MODE &&
+		    session->opmode != QDF_P2P_CLIENT_MODE)
+			return NULL;
+		break;
+	default:
+		return NULL;
+	}
+
+	return session;
+}
+
+QDF_STATUS
+lim_update_tx_power(struct mac_context *mac_ctx, struct pe_session *sap_session,
+		    struct pe_session *sta_session, bool restore_sta_power)
+{
+	uint8_t pwr_level;
+	struct vdev_mlme_obj *sta_mlme_obj, *sap_mlme_obj;
+	struct reg_tpc_power_info *reg_info;
+	uint8_t tx_power, i;
+
+	sta_mlme_obj = wlan_vdev_mlme_get_cmpt_obj(sta_session->vdev);
+	sap_mlme_obj = wlan_vdev_mlme_get_cmpt_obj(sap_session->vdev);
+
+	if (!sta_mlme_obj || !sap_mlme_obj)
+		return QDF_STATUS_E_FAILURE;
+
+	if (restore_sta_power) {
+		/* SAP interface is removed, restore the STA power */
+		wlan_set_tpc_update_required_for_sta(sap_session->vdev, false);
+		sta_session->sta_follows_sap_power = false;
+		lim_set_tpc_power(mac_ctx, sta_session);
+	} else {
+		/*
+		 * SAP and STA are in different AP power types. Therefore,
+		 * update the reg_tpc_obj of STA with new power levels.
+		 * Do not send new TPC power to FW.
+		 */
+		sta_session->sta_follows_sap_power = true;
+
+		if (sta_mlme_obj->reg_tpc_obj.power_type_6g ==
+		    sap_mlme_obj->reg_tpc_obj.power_type_6g) {
+			pe_err("STA and SAP are in same power type");
+			return QDF_STATUS_E_FAILURE;
+		}
+		pe_debug("STA is moved to %d from %d power type",
+			 sap_mlme_obj->reg_tpc_obj.power_type_6g,
+			 sta_mlme_obj->reg_tpc_obj.power_type_6g);
+		sta_mlme_obj->reg_tpc_obj.power_type_6g =
+			sap_mlme_obj->reg_tpc_obj.power_type_6g;
+
+		tx_power = lim_get_tx_power(&sap_mlme_obj->reg_tpc_obj,
+					    sap_session->curr_op_freq);
+
+		reg_info = &sta_mlme_obj->reg_tpc_obj;
+		pwr_level = reg_info->num_pwr_levels;
+		for (i = 0; i < pwr_level; i++)
+			reg_info->chan_power_info[i].tx_power = tx_power;
+		wlan_set_tpc_update_required_for_sta(sap_session->vdev, true);
+	}
+	return QDF_STATUS_SUCCESS;
+}
+
+bool
+lim_is_power_change_required_for_sta(struct mac_context *mac_ctx,
+				     struct pe_session *sta_session,
+				     struct pe_session *sap_session)
+{
+	enum channel_state channel_state;
+	struct vdev_mlme_obj *mlme_obj;
+	uint32_t ap_power_type_6g = 0;
+
+	channel_state =
+		wlan_reg_get_channel_state_for_pwrmode(mac_ctx->pdev,
+						       sap_session->curr_op_freq,
+						       REG_AP_VLP);
+
+	mlme_obj = wlan_vdev_mlme_get_cmpt_obj(sap_session->vdev);
+	if (!mlme_obj) {
+		pe_err("vdev component object is NULL");
+		return false;
+	}
+
+	if (sta_session->curr_op_freq != sap_session->curr_op_freq) {
+		pe_err("STA and SAP are not in same frequency, do not change TPC power");
+		return false;
+	}
+
+	wlan_reg_get_cur_6g_ap_pwr_type(mac_ctx->pdev, &ap_power_type_6g);
+
+	if (sta_session->ap_power_type_6g == REG_INDOOR_AP &&
+	    channel_state & CHANNEL_STATE_ENABLE &&
+	    ap_power_type_6g == REG_VERY_LOW_POWER_AP) {
+		pe_debug("Change the power type of STA from LPI to VLP");
+		return true;
+	}
+
+	return false;
+}
+
+void
+lim_check_conc_power_for_csa(struct mac_context *mac_ctx,
+			     struct pe_session *sap_session)
+{
+	struct pe_session *sta_session;
+	bool update_required_scc_sta_power =
+			wlan_get_tpc_update_required_for_sta(sap_session->vdev);
+
+	/*
+	 * If SCC power has changed and concurrent session doesn't exist,
+	 * then STA must have got deleted or moved out of 6GHz.
+	 * In that case, reset the change scc power flag for SAP.
+	 */
+	sta_session = lim_get_concurrent_session(mac_ctx, sap_session->vdev_id,
+						 sap_session->opmode);
+	if (!sta_session) {
+		pe_debug("STA session doesn't exist");
+		return;
+	}
+
+	/* If SCC power has changed and the SAP is moving away from 6GHz,
+	 * reset the scc power flag in SAP vdev and restore the STA
+	 * power
+	 */
+	if (update_required_scc_sta_power &&
+	    !WLAN_REG_IS_6GHZ_CHAN_FREQ(sap_session->curr_op_freq) &&
+	    WLAN_REG_IS_6GHZ_CHAN_FREQ(sta_session->curr_op_freq)) {
+		pe_debug("SAP has moved from 6GHz, restore STA power");
+		lim_update_tx_power(mac_ctx, sap_session, sta_session, true);
+		return;
+	}
+
+	/* If SAP is moving to 6GHz. Then:
+	 * a) If change scc power is not set, check if it needs to be set
+	 *    If it is getting set, then send new tpc power to FW.
+	 * b) If change scc power is already set, then SAP is moving from one
+	 *    6GHz to another 6GHz. Recompute the TPC.
+	 */
+	if (WLAN_REG_IS_6GHZ_CHAN_FREQ(sap_session->curr_op_freq) &&
+	    sta_session &&
+	    WLAN_REG_IS_6GHZ_CHAN_FREQ(sta_session->curr_op_freq) &&
+	    (wlan_vdev_mlme_get_state(sap_session->vdev) == WLAN_VDEV_S_UP)) {
+		if (lim_is_power_change_required_for_sta(mac_ctx, sta_session,
+							 sap_session)) {
+			lim_set_tpc_power(mac_ctx, sap_session);
+			if (lim_update_tx_power(mac_ctx, sap_session,
+						sta_session, false) ==
+							QDF_STATUS_SUCCESS)
+				wlan_set_tpc_update_required_for_sta(
+							sap_session->vdev,
+							true);
+		}
+	}
+}
+
+void
+lim_cleanup_power_change(struct mac_context *mac_ctx,
+			 struct pe_session *session)
+{
+	struct pe_session *sap_session;
+
+	if (session->opmode != QDF_STA_MODE &&
+	    session->opmode != QDF_P2P_CLIENT_MODE)
+		return;
+
+	sap_session =
+		lim_get_concurrent_session(mac_ctx, session->vdev_id,
+					   session->opmode);
+	if (!sap_session)
+		return;
+
+	wlan_set_tpc_update_required_for_sta(sap_session->vdev, false);
+}

+ 82 - 0
core/mac/src/pe/lim/lim_utils.h

@@ -3066,4 +3066,86 @@ bool lim_update_channel_width(struct mac_context *mac_ctx,
 uint8_t lim_get_vht_ch_width(tDot11fIEVHTCaps *vht_cap,
 			     tDot11fIEVHTOperation *vht_op,
 			     tDot11fIEHTInfo *ht_info);
+
+/**
+ * lim_update_tx_power() - Function to update the TX power for
+ * the STA interface based on the SAP concurrency
+ *
+ * @mac_ctx: Pointer to Global mac structure
+ * @sap_session: Session pointer of the SAP
+ * @sta_session: Session pointer of the STA
+ * @restore_sta_power: Flag to update the new Tx power for STA
+ *
+ * Return: QDF_STATUS
+ */
+QDF_STATUS
+lim_update_tx_power(struct mac_context *mac_ctx, struct pe_session *sap_session,
+		    struct pe_session *sta_session, bool restore_sta_power);
+
+/**
+ * lim_skip_tpc_update_for_sta() - Function to check if the TPC set power
+ * needs to be skipped for STA.
+ *
+ * @mac_ctx: Pointer to Global mac structure
+ * @sta_session: Session pointer of the STA.
+ * @sap_session: Session pointer of the SAP
+ *
+ * Return: skip tpc for sta
+ */
+bool
+lim_skip_tpc_update_for_sta(struct mac_context *mac,
+			    struct pe_session *sta_session,
+			    struct pe_session *sap_session);
+
+/**
+ * lim_get_concurrent_session() - Function to get the concurrent session pointer
+ *
+ * @mac_ctx: Pointer to Global mac structure
+ * @vdev_id: vdev id
+ * @opmode: opmode of the interface
+ *
+ * Return: pe_session
+ */
+struct pe_session *
+lim_get_concurrent_session(struct mac_context *mac_ctx, uint8_t vdev_id,
+			   enum QDF_OPMODE opmode);
+
+/**
+ * lim_check_conc_power_for_csa() - Function to change the TX power
+ * of STA when channel switch happens in SAP
+ *
+ * @mac_ctx: Pointer to Global mac structure.
+ * @session: Session pointer of the SAP.
+ *
+ * Return: None
+ */
+void
+lim_check_conc_power_for_csa(struct mac_context *mac_ctx,
+			     struct pe_session *session);
+
+/**
+ * lim_cleanup_power_change() - Function to reset the change_scc_power
+ * flag in concurrent SAP vdev
+ *
+ * @mac_ctx: Pointer to Global mac structure.
+ * @session: Session pointer of the interface
+ */
+void
+lim_cleanup_power_change(struct mac_context *mac_ctx,
+			 struct pe_session *session);
+
+/**
+ * lim_is_power_change_required_for_sta() - Function to check if the 6 GHz
+ * STA power level has to be changed
+ *
+ * @mac_ctx: Pointer to Global mac structure.
+ * @sta_session: Session pointer of STA.
+ * @sap_session: Session pointer of SAP.
+ *
+ * Return: restart required for sta
+ */
+bool
+lim_is_power_change_required_for_sta(struct mac_context *mac_ctx,
+				     struct pe_session *sta_session,
+				     struct pe_session *sap_session);
 #endif /* __LIM_UTILS_H */

+ 9 - 2
core/mac/src/pe/rrm/rrm_api.c

@@ -1,5 +1,6 @@
 /*
  * Copyright (c) 2012-2021 The Linux Foundation. All rights reserved.
+ * Copyright (c) 2022 Qualcomm Innovation Center, Inc. All rights reserved.
  *
  * Permission to use, copy, modify, and/or distribute this software for
  * any purpose with or without fee is hereby granted, provided that the
@@ -266,7 +267,13 @@ rrm_process_link_measurement_request(struct mac_context *mac,
 		return QDF_STATUS_E_INVAL;
 	}
 
-	if (wlan_reg_is_ext_tpc_supported(mac->psoc)) {
+	/*
+	 * STA LPI + SAP VLP is supported. For this STA should operate in VLP
+	 * power level of the SAP.
+	 * If STA is operating in VLP power of SAP, do not update STA power.
+	 */
+	if (wlan_reg_is_ext_tpc_supported(mac->psoc) &&
+	    !pe_session->sta_follows_sap_power) {
 		ap_pwr_constraint = mlme_obj->reg_tpc_obj.ap_constraint_power;
 		mlme_obj->reg_tpc_obj.ap_constraint_power =
 				pLinkReq->MaxTxPower.maxTxPower;
@@ -287,7 +294,7 @@ rrm_process_link_measurement_request(struct mac_context *mac,
 						      pe_session->vdev_id,
 						      &mlme_obj->reg_tpc_obj);
 		}
-	} else {
+	} else if (!pe_session->sta_follows_sap_power) {
 		mlme_obj->reg_tpc_obj.reg_max[0] =
 				pe_session->def_max_tx_pwr;
 		mlme_obj->reg_tpc_obj.ap_constraint_power =

+ 8 - 2
core/mac/src/pe/sch/sch_beacon_process.c

@@ -661,7 +661,13 @@ static void __sch_beacon_process_for_session(struct mac_context *mac_ctx,
 			return;
 	}
 
-	if (wlan_reg_is_ext_tpc_supported(mac_ctx->psoc)) {
+	/*
+	 * STA LPI + SAP VLP is supported. For this STA should operate in VLP
+	 * power level of the SAP.
+	 * If STA is operating in VLP power of SAP, do not update STA power.
+	 */
+	if (wlan_reg_is_ext_tpc_supported(mac_ctx->psoc) &&
+	    !session->sta_follows_sap_power) {
 		tx_ops = wlan_reg_get_tx_ops(mac_ctx->psoc);
 
 		lim_parse_tpe_ie(mac_ctx, session, bcn->transmit_power_env,
@@ -695,7 +701,7 @@ static void __sch_beacon_process_for_session(struct mac_context *mac_ctx,
 						      session->vdev_id,
 						      &mlme_obj->reg_tpc_obj);
 			}
-	} else {
+	} else if (!session->sta_follows_sap_power) {
 		/* Obtain the Max Tx power for the current regulatory  */
 		regMax = wlan_reg_get_channel_reg_power_for_freq(
 					mac_ctx->pdev, session->curr_op_freq);

+ 54 - 0
core/sap/src/sap_fsm.c

@@ -3008,6 +3008,57 @@ static void sap_validate_chanmode_and_chwidth(struct mac_context *mac_ctx,
 			 orig_phymode, sap_ctx->phyMode);
 }
 
+static bool
+wlansap_is_power_change_required(struct mac_context *mac_ctx,
+				 qdf_freq_t sap_freq)
+{
+	struct wlan_objmgr_vdev *sta_vdev;
+	uint8_t sta_vdev_id;
+	enum hw_mode_bandwidth ch_wd;
+	uint8_t country[CDS_COUNTRY_CODE_LEN + 1];
+	enum channel_state state;
+	uint32_t ap_pwr_type_6g = 0;
+	bool indoor_ch_support = false;
+
+	if (!mac_ctx || !mac_ctx->psoc || !mac_ctx->pdev)
+		return false;
+
+	if (!policy_mgr_is_sta_present_on_freq(mac_ctx->psoc, &sta_vdev_id,
+					       sap_freq, &ch_wd)) {
+		return false;
+	}
+
+	sta_vdev = wlan_objmgr_get_vdev_by_id_from_psoc(mac_ctx->psoc,
+							sta_vdev_id,
+							WLAN_LEGACY_SAP_ID);
+	if (!sta_vdev)
+		return false;
+
+	ap_pwr_type_6g = wlan_mlme_get_6g_ap_power_type(sta_vdev);
+
+	wlan_objmgr_vdev_release_ref(sta_vdev, WLAN_LEGACY_SAP_ID);
+
+	if (ap_pwr_type_6g == REG_VERY_LOW_POWER_AP)
+		return false;
+	ucfg_mlme_get_indoor_channel_support(mac_ctx->psoc, &indoor_ch_support);
+
+	if (ap_pwr_type_6g == REG_INDOOR_AP && indoor_ch_support) {
+		sap_debug("STA is connected to Indoor AP and indoor concurrency is supported");
+		return false;
+	}
+
+	wlan_reg_read_current_country(mac_ctx->psoc, country);
+	if (!wlan_reg_ctry_support_vlp(country)) {
+		sap_debug("Device country doesn't support VLP");
+		return false;
+	}
+
+	state = wlan_reg_get_channel_state_for_pwrmode(mac_ctx->pdev,
+						       sap_freq, REG_AP_VLP);
+
+	return state & CHANNEL_STATE_ENABLE;
+}
+
 /**
  * sap_goto_starting() - Trigger softap start
  * @sap_ctx: SAP context
@@ -3042,6 +3093,9 @@ static QDF_STATUS sap_goto_starting(struct sap_context *sap_ctx,
 	} else if (!policy_mgr_get_ap_6ghz_capable(mac_ctx->psoc,
 						   sap_ctx->sessionId, NULL)) {
 		return QDF_STATUS_E_FAILURE;
+	} else if (wlansap_is_power_change_required(mac_ctx,
+						    sap_ctx->chan_freq)) {
+		wlan_set_tpc_update_required_for_sta(sap_ctx->vdev, true);
 	}
 
 	/*