Bläddra i källkod

qcacld-3.0: Add support for STA+GO and CLI+GO for liberal mode

Currently in liberal mode STA+GO and CLI+GO scenario is not
handled, so when sta is there and Go tries to come up then
GO should follow STA or CLI's channel in case of force scc
but as handling is not there so Go doesn't consider scc
channel.

Fix is to add handling of STA+GO and CLI+GO for liberal mode.

Change-Id: I227fb0661c5c3a41b296fd0c954ac87675f28ed6
CRs-Fixed: 3048501
sheenam monga 3 år sedan
förälder
incheckning
898339bbbc

+ 25 - 10
components/cmn_services/policy_mgr/inc/wlan_policy_mgr_api.h

@@ -692,20 +692,29 @@ policy_mgr_is_p2p_p2p_conc_supported(struct wlan_objmgr_psoc *psoc)
 bool policy_mgr_is_go_scc_strict(struct wlan_objmgr_psoc *psoc);
 
 /**
- * policy_mgr_check_forcescc_for_other_go() - check if another p2pgo
- * is present and find vdev id.
+ * policy_mgr_fetch_existing_con_info() - check if another vdev
+ * is present and find mode, freq , vdev id and chan width
  *
  * @psoc: psoc object
  * @vdev: vdev id
  * @freq: frequency
+ * @mode: existing vdev mode
+ * @con_freq: existing connection freq
+ * @ch_width: ch_width of existing connection
  *
- * This function checks if another p2p go is there.
+ * This function checks if another vdev is there and fetch connection
+ * info for that vdev.This is mainly for force SCC implementation of GO+GO ,
+ * GO+SAP or GO+STA where we fetch other existing GO, STA, SAP on the same
+ * band with MCC.
  *
  * Return: vdev_id
  */
 uint8_t
-policy_mgr_check_forcescc_for_other_go(struct wlan_objmgr_psoc *psoc,
-				       uint8_t vdev_id, uint32_t curr_go_freq);
+policy_mgr_fetch_existing_con_info(struct wlan_objmgr_psoc *psoc,
+				   uint8_t vdev_id, uint32_t curr_go_freq,
+				   enum policy_mgr_con_mode *mode,
+				   uint32_t *con_freq,
+				   enum phy_ch_width *ch_width);
 
 /**
  * policy_mgr_process_forcescc_for_go () - start work queue to move first p2p go
@@ -715,6 +724,7 @@ policy_mgr_check_forcescc_for_other_go(struct wlan_objmgr_psoc *psoc,
  * @vdev_id: Vdev id
  * @ch_freq: Channel frequency to change
  * @ch_width: channel width to change
+ * @mode: existing vdev mode
  *
  * starts delayed work queue of 1 second to move first p2p go to new
  * p2p go's channel.
@@ -723,7 +733,8 @@ policy_mgr_check_forcescc_for_other_go(struct wlan_objmgr_psoc *psoc,
  */
 void policy_mgr_process_forcescc_for_go(
 		struct wlan_objmgr_psoc *psoc, uint8_t vdev_id,
-		uint32_t ch_freq, uint32_t ch_width);
+		uint32_t ch_freq, uint32_t ch_width,
+		enum policy_mgr_con_mode mode);
 
 /**
  * policy_mgr_do_go_plus_go_force_scc() - First p2p go
@@ -750,9 +761,12 @@ bool policy_mgr_is_go_scc_strict(struct wlan_objmgr_psoc *psoc)
 }
 
 static inline
-uint8_t policy_mgr_check_forcescc_for_other_go(struct wlan_objmgr_psoc *psoc,
-					       uint8_t vdev_id,
-					       uint32_t curr_go_freq)
+uint8_t policy_mgr_fetch_existing_con_info(struct wlan_objmgr_psoc *psoc,
+					   uint8_t vdev_id,
+					   uint32_t curr_go_freq,
+					   enum policy_mgr_con_mode *mode,
+					   uint32_t *con_freq,
+					   enum phy_ch_width *ch_width)
 {
 	return WLAN_UMAC_VDEV_ID_MAX;
 }
@@ -760,7 +774,8 @@ uint8_t policy_mgr_check_forcescc_for_other_go(struct wlan_objmgr_psoc *psoc,
 static inline
 void policy_mgr_process_forcescc_for_go(
 		struct wlan_objmgr_psoc *psoc, uint8_t vdev_id,
-		uint32_t ch_freq, uint32_t ch_width)
+		uint32_t ch_freq, uint32_t ch_width,
+		enum policy_mgr_con_mode mode)
 {}
 
 static inline

+ 18 - 17
components/cmn_services/policy_mgr/src/wlan_policy_mgr_action.c

@@ -2026,25 +2026,27 @@ static void __policy_mgr_check_sta_ap_concurrent_ch_intf(
 	uint32_t ch_freq;
 	uint32_t op_ch_freq_list[MAX_NUMBER_OF_CONC_CONNECTIONS];
 	uint8_t vdev_id[MAX_NUMBER_OF_CONC_CONNECTIONS];
+	struct sta_ap_intf_check_work_ctx *work_info;
 
 	if (!pm_ctx) {
 		policy_mgr_err("Invalid context");
 		return;
 	}
+	work_info = pm_ctx->sta_ap_intf_check_work_info;
 	/*
 	 * Check if force scc is required for GO + GO case. vdev id will be
 	 * valid in case of GO+GO force scc only. So, for valid vdev id move
 	 * first GO to newly formed GO channel.
 	 */
 	policy_mgr_debug("p2p go vdev id: %d",
-			 pm_ctx->sta_ap_intf_check_work_info->go_plus_go_force_scc.vdev_id);
+			 work_info->go_plus_go_force_scc.vdev_id);
 	if (pm_ctx->sta_ap_intf_check_work_info->go_plus_go_force_scc.vdev_id <
 	    WLAN_UMAC_VDEV_ID_MAX) {
 		policy_mgr_do_go_plus_go_force_scc(
-			pm_ctx->psoc, pm_ctx->sta_ap_intf_check_work_info->go_plus_go_force_scc.vdev_id,
-			pm_ctx->sta_ap_intf_check_work_info->go_plus_go_force_scc.ch_freq,
-			pm_ctx->sta_ap_intf_check_work_info->go_plus_go_force_scc.ch_width);
-		pm_ctx->sta_ap_intf_check_work_info->go_plus_go_force_scc.vdev_id = WLAN_UMAC_VDEV_ID_MAX;
+			pm_ctx->psoc, work_info->go_plus_go_force_scc.vdev_id,
+			work_info->go_plus_go_force_scc.ch_freq,
+			work_info->go_plus_go_force_scc.ch_width);
+		work_info->go_plus_go_force_scc.vdev_id = WLAN_UMAC_VDEV_ID_MAX;
 		return;
 	}
 
@@ -2642,9 +2644,11 @@ void policy_mgr_do_go_plus_go_force_scc(struct wlan_objmgr_psoc *psoc,
 
 void policy_mgr_process_forcescc_for_go(struct wlan_objmgr_psoc *psoc,
 					uint8_t vdev_id, uint32_t ch_freq,
-					uint32_t ch_width)
+					uint32_t ch_width,
+					enum policy_mgr_con_mode mode)
 {
 	struct policy_mgr_psoc_priv_obj *pm_ctx;
+	struct sta_ap_intf_check_work_ctx *work_info;
 
 	pm_ctx = policy_mgr_get_context(psoc);
 	if (!pm_ctx) {
@@ -2655,19 +2659,16 @@ void policy_mgr_process_forcescc_for_go(struct wlan_objmgr_psoc *psoc,
 		policy_mgr_err("invalid work info");
 		return;
 	}
-	pm_ctx->sta_ap_intf_check_work_info->go_plus_go_force_scc.vdev_id =
-								vdev_id;
-	pm_ctx->sta_ap_intf_check_work_info->go_plus_go_force_scc.ch_freq =
-								ch_freq;
-	pm_ctx->sta_ap_intf_check_work_info->go_plus_go_force_scc.ch_width =
-								ch_width;
+	work_info = pm_ctx->sta_ap_intf_check_work_info;
+	if (mode == PM_P2P_GO_MODE) {
+		work_info->go_plus_go_force_scc.vdev_id = vdev_id;
+		work_info->go_plus_go_force_scc.ch_freq = ch_freq;
+		work_info->go_plus_go_force_scc.ch_width = ch_width;
+	}
 
 	if (!qdf_delayed_work_start(&pm_ctx->sta_ap_intf_check_work,
-				    WAIT_BEFORE_GO_FORCESCC_RESTART)) {
-		policy_mgr_err("change interface request failure");
-		pm_ctx->sta_ap_intf_check_work_info->go_plus_go_force_scc.vdev_id =
-						WLAN_UMAC_VDEV_ID_MAX;
-	}
+				    WAIT_BEFORE_GO_FORCESCC_RESTART))
+		policy_mgr_debug("change interface request already queued");
 }
 #endif
 

+ 18 - 5
components/cmn_services/policy_mgr/src/wlan_policy_mgr_get_set_utils.c

@@ -5155,8 +5155,11 @@ bool policy_mgr_go_scc_enforced(struct wlan_objmgr_psoc *psoc)
 
 #ifdef WLAN_FEATURE_P2P_P2P_STA
 uint8_t
-policy_mgr_check_forcescc_for_other_go(struct wlan_objmgr_psoc *psoc,
-				       uint8_t vdev_id, uint32_t freq)
+policy_mgr_fetch_existing_con_info(struct wlan_objmgr_psoc *psoc,
+				   uint8_t vdev_id, uint32_t freq,
+				   enum policy_mgr_con_mode *mode,
+				   uint32_t *existing_con_freq,
+				   enum phy_ch_width *existing_ch_width)
 {
 	struct policy_mgr_psoc_priv_obj *pm_ctx;
 	uint32_t conn_index;
@@ -5170,8 +5173,12 @@ policy_mgr_check_forcescc_for_other_go(struct wlan_objmgr_psoc *psoc,
 	qdf_mutex_acquire(&pm_ctx->qdf_conc_list_lock);
 	for (conn_index = 0; conn_index < MAX_NUMBER_OF_CONC_CONNECTIONS;
 	     conn_index++) {
-		if (pm_conc_connection_list[conn_index].mode ==
-		    PM_P2P_GO_MODE &&
+		if ((pm_conc_connection_list[conn_index].mode ==
+		    PM_P2P_GO_MODE ||
+		    pm_conc_connection_list[conn_index].mode ==
+		    PM_P2P_CLIENT_MODE ||
+		    pm_conc_connection_list[conn_index].mode ==
+		    PM_STA_MODE) &&
 		    pm_conc_connection_list[conn_index].in_use &&
 		    wlan_reg_is_same_band_freqs(
 				freq,
@@ -5180,8 +5187,14 @@ policy_mgr_check_forcescc_for_other_go(struct wlan_objmgr_psoc *psoc,
 		    vdev_id != pm_conc_connection_list[conn_index].vdev_id) {
 			qdf_mutex_release(&pm_ctx->qdf_conc_list_lock);
 			policy_mgr_debug(
-				"Existing p2p go vdev_id is %d",
+				"Existing vdev_id for mode %d is %d",
+				pm_conc_connection_list[conn_index].mode,
 				pm_conc_connection_list[conn_index].vdev_id);
+			*mode = pm_conc_connection_list[conn_index].mode;
+			*existing_con_freq =
+				pm_conc_connection_list[conn_index].freq;
+			*existing_ch_width = policy_mgr_get_ch_width(
+					pm_conc_connection_list[conn_index].bw);
 			return pm_conc_connection_list[conn_index].vdev_id;
 		}
 	}

+ 61 - 16
core/sap/src/sap_api_link_cntl.c

@@ -840,15 +840,16 @@ static void wlansap_update_vendor_acs_chan(struct mac_context *mac_ctx,
 
 #ifdef WLAN_FEATURE_P2P_P2P_STA
 /**
- * sap_check_and_process_forcescc_for_other_go() - find if other p2p go is there
- * and needs to be moved to current p2p go's channel.
+ * sap_check_and_process_forcescc_for_go_plus_go() - find if other p2p
+ * go is there and needs to be moved to current p2p go's channel.
  *
  * @cur_sap_ctx: current sap context
  *
  * Return: None
  */
 static void
-sap_check_and_process_forcescc_for_other_go(struct sap_context *cur_sap_ctx)
+sap_check_and_process_forcescc_for_go_plus_go(
+					struct sap_context *cur_sap_ctx)
 {
 	struct sap_context *sap_ctx;
 	struct mac_context *mac_ctx;
@@ -871,15 +872,67 @@ sap_check_and_process_forcescc_for_other_go(struct sap_context *cur_sap_ctx)
 			policy_mgr_process_forcescc_for_go(
 				mac_ctx->psoc, sap_ctx->sessionId,
 				cur_sap_ctx->chan_freq,
-				cur_sap_ctx->ch_params.ch_width);
+				cur_sap_ctx->ch_params.ch_width,
+				PM_P2P_GO_MODE);
 			sap_ctx->is_forcescc_restart_required = false;
 			break;
 		}
 	}
 }
-#else
+
+/**
+ * sap_check_and_process_go_force_scc() - find if other p2p
+ * go/cli/sta is there and needs force scc.
+ *
+ * @cur_sap_ctx: current sap context
+ *
+ * Return: None
+ */
 static void
-sap_check_and_process_forcescc_for_other_go(struct sap_context *cur_sap_ctx)
+sap_check_and_process_go_force_ssc(struct sap_context *sap_ctx)
+{
+	struct mac_context *mac_ctx;
+	uint32_t con_freq;
+	enum phy_ch_width ch_width;
+	enum policy_mgr_con_mode existing_vdev_mode = PM_MAX_NUM_OF_MODE;
+
+	mac_ctx = sap_get_mac_context();
+	if (!mac_ctx) {
+		sap_err("Invalid MAC context");
+		return;
+	}
+	if (sap_ctx->vdev->vdev_mlme.vdev_opmode ==
+	    QDF_P2P_GO_MODE &&
+	    wlan_vdev_get_peer_count(sap_ctx->vdev) == 2 &&
+	    policy_mgr_mode_specific_connection_count(
+		    mac_ctx->psoc, PM_P2P_GO_MODE, NULL) > 1) {
+		sap_check_and_process_forcescc_for_go_plus_go(sap_ctx);
+		return;
+	}
+	policy_mgr_fetch_existing_con_info(mac_ctx->psoc, sap_ctx->sessionId,
+					   sap_ctx->chan_freq,
+					   &existing_vdev_mode,
+					   &con_freq, &ch_width);
+
+	if (sap_ctx->vdev->vdev_mlme.vdev_opmode == QDF_P2P_GO_MODE &&
+	    policy_mgr_go_scc_enforced(mac_ctx->psoc) &&
+	    !policy_mgr_is_go_scc_strict(mac_ctx->psoc) &&
+	    wlan_vdev_get_peer_count(sap_ctx->vdev) == 2 &&
+	    (existing_vdev_mode == PM_P2P_CLIENT_MODE ||
+	    existing_vdev_mode == PM_STA_MODE)){
+		policy_mgr_process_forcescc_for_go(mac_ctx->psoc,
+						   sap_ctx->sessionId,
+						   con_freq, ch_width,
+						   existing_vdev_mode);
+	}
+}
+#else
+static inline void
+sap_check_and_process_forcescc_for_go_plus_go(
+					struct sap_context *cur_sap_ctx)
+{}
+static inline void
+sap_check_and_process_go_force_ssc(struct sap_context *cur_sap_ctx)
 {}
 #endif
 
@@ -1194,20 +1247,12 @@ QDF_STATUS wlansap_roam_callback(void *ctx,
 			sap_signal_hdd_event(sap_ctx, csr_roam_info,
 					     eSAP_STA_SET_KEY_EVENT,
 					     (void *) eSAP_STATUS_SUCCESS);
-
 		/*
 		 * After set key if this is the first peer connecting to new GO
 		 * then check for peer count (which is self peer + peer count)
-		 * and take decision for GO+GO force SCC
+		 * and take decision for GO+GO, STA+GO and CLI+GO force SCC
 		 */
-			if (sap_ctx->vdev->vdev_mlme.vdev_opmode ==
-			    QDF_P2P_GO_MODE &&
-			    wlan_vdev_get_peer_count(sap_ctx->vdev) == 2 &&
-			    policy_mgr_mode_specific_connection_count(
-						mac_ctx->psoc, PM_P2P_GO_MODE,
-						NULL) > 1)
-				sap_check_and_process_forcescc_for_other_go(
-								sap_ctx);
+			sap_check_and_process_go_force_ssc(sap_ctx);
 		}
 		break;
 	case eCSR_ROAM_RESULT_MAX_ASSOC_EXCEEDED:

+ 74 - 22
core/sap/src/sap_fsm.c

@@ -828,9 +828,64 @@ static void sap_set_forcescc_required(uint8_t vdev_id)
 		}
 	}
 }
+
+/**
+ * sap_process_liberal_scc_for_go - based on existing connections this
+ * function decides current go should start on provided channel or not and
+ * sets force scc required bit for existing GO.
+ *
+ * sap_context: sap_context
+ *
+ * Return: bool
+ */
+static bool sap_process_liberal_scc_for_go(struct sap_context *sap_context)
+{
+	uint8_t existing_vdev_id = WLAN_UMAC_VDEV_ID_MAX;
+	enum policy_mgr_con_mode existing_vdev_mode = PM_MAX_NUM_OF_MODE;
+	uint32_t con_freq;
+	enum phy_ch_width ch_width;
+	struct mac_context *mac_ctx;
+	mac_handle_t mac_handle;
+
+	mac_handle = cds_get_context(QDF_MODULE_ID_SME);
+	mac_ctx = MAC_CONTEXT(mac_handle);
+	if (!mac_ctx) {
+		sap_alert("invalid MAC handle");
+		return true;
+	}
+
+	existing_vdev_id =
+		policy_mgr_fetch_existing_con_info(
+				mac_ctx->psoc,
+				sap_context->sessionId,
+				sap_context->chan_freq,
+				&existing_vdev_mode,
+				&con_freq, &ch_width);
+
+	if (existing_vdev_id <
+			WLAN_UMAC_VDEV_ID_MAX &&
+			existing_vdev_mode == PM_P2P_GO_MODE) {
+		sap_debug("set forcescc flag for go vdev: %d",
+			  existing_vdev_id);
+		sap_set_forcescc_required(
+				existing_vdev_id);
+		return true;
+	}
+	if (existing_vdev_id < WLAN_UMAC_VDEV_ID_MAX &&
+	    (existing_vdev_mode == PM_STA_MODE ||
+	    existing_vdev_mode == PM_P2P_CLIENT_MODE)) {
+		sap_debug("don't override channel, start go on %d",
+			  sap_context->chan_freq);
+		return true;
+	}
+
+	return false;
+}
 #else
-static void sap_set_forcescc_required(uint8_t vdev_id)
-{}
+static bool sap_process_liberal_scc_for_go(struct sap_context *sap_context)
+{
+	return false;
+}
 #endif
 
 QDF_STATUS
@@ -849,7 +904,7 @@ sap_validate_chan(struct sap_context *sap_context,
 	bool go_force_scc;
 	struct ch_params ch_params;
 	bool is_go_scc_strict = false;
-	uint8_t first_p2p_go_vdev_id = WLAN_UMAC_VDEV_ID_MAX;
+	bool start_sap_on_provided_freq = false;
 
 	mac_handle = cds_get_context(QDF_MODULE_ID_SME);
 	mac_ctx = MAC_CONTEXT(mac_handle);
@@ -868,35 +923,32 @@ sap_validate_chan(struct sap_context *sap_context,
 	    sap_context->vdev->vdev_mlme.vdev_opmode == QDF_P2P_GO_MODE) {
 	       /*
 		* check whether go_force_scc is enabled or not.
-		* If it not enabled then don't any force scc on existing and new
-		* p2p go vdevs.
+		* If it not enabled then don't any force scc on existing go and
+		* new p2p go vdevs.
 		* Otherwise, if it is enabled then check whether it's in strict
 		* mode or liberal mode.
-		* For strict mode, do force scc on newly p2p go to existing p2p
-		* go channel.
-		* For liberal mode, first form new p2p go on requested channel.
-		* Once set key is done, do force scc on existing p2p go to new
-		* p2p go channel.
+		* For strict mode, do force scc on newly p2p go to existing vdev
+		* channel.
+		* For liberal first form new p2p go on requested channel and
+		* follow below rules:
+		* a.) If Existing vdev mode is P2P GO Once set key is done, do
+		* force scc for existing p2p go and move that go to new p2p
+		* go's channel.
+		*
+		* b.) If Existing vdev mode is P2P CLI/STA Once set key is
+		* done, do force scc for p2p go and move go to cli/sta channel.
 		*/
 		go_force_scc = policy_mgr_go_scc_enforced(mac_ctx->psoc);
-		sap_debug("go force scc value %d", go_force_scc);
+		sap_debug("go force scc enabled %d", go_force_scc);
 		if (go_force_scc) {
 			is_go_scc_strict =
 				policy_mgr_is_go_scc_strict(mac_ctx->psoc);
 			if (!is_go_scc_strict) {
 				sap_debug("liberal mode is enabled");
-				first_p2p_go_vdev_id =
-					policy_mgr_check_forcescc_for_other_go(
-						mac_ctx->psoc,
-						sap_context->sessionId,
-						sap_context->chan_freq);
-
-				if (first_p2p_go_vdev_id <
-				    WLAN_UMAC_VDEV_ID_MAX) {
-					sap_set_forcescc_required(
-							first_p2p_go_vdev_id);
+				start_sap_on_provided_freq =
+				sap_process_liberal_scc_for_go(sap_context);
+				if (start_sap_on_provided_freq)
 					goto validation_done;
-				}
 			}
 		} else {
 			goto validation_done;