|
@@ -2406,7 +2406,12 @@ static void __policy_mgr_check_sta_ap_concurrent_ch_intf(
|
|
|
policy_mgr_err("Could not retrieve SAP/GO operating channel&vdevid");
|
|
|
goto end;
|
|
|
}
|
|
|
-
|
|
|
+ if (policy_mgr_is_ap_start_in_progress(pm_ctx->psoc)) {
|
|
|
+ policy_mgr_debug("defer sap conc check to a later time due to another sap/go start pending");
|
|
|
+ qdf_delayed_work_start(&pm_ctx->sta_ap_intf_check_work,
|
|
|
+ SAP_CONC_CHECK_DEFER_TIMEOUT_MS);
|
|
|
+ goto end;
|
|
|
+ }
|
|
|
if (pm_ctx->hdd_cbacks.hdd_is_chan_switch_in_progress &&
|
|
|
pm_ctx->hdd_cbacks.hdd_is_chan_switch_in_progress()) {
|
|
|
policy_mgr_debug("wait as channel switch is already in progress");
|
|
@@ -2933,6 +2938,92 @@ void policy_mgr_process_forcescc_for_go(struct wlan_objmgr_psoc *psoc,
|
|
|
}
|
|
|
#endif
|
|
|
|
|
|
+bool policy_mgr_is_chan_switch_in_progress(struct wlan_objmgr_psoc *psoc)
|
|
|
+{
|
|
|
+ struct policy_mgr_psoc_priv_obj *pm_ctx;
|
|
|
+
|
|
|
+ pm_ctx = policy_mgr_get_context(psoc);
|
|
|
+ if (!pm_ctx) {
|
|
|
+ policy_mgr_err("Invalid pm context");
|
|
|
+ return false;
|
|
|
+ }
|
|
|
+ if (pm_ctx->hdd_cbacks.hdd_is_chan_switch_in_progress &&
|
|
|
+ pm_ctx->hdd_cbacks.hdd_is_chan_switch_in_progress()) {
|
|
|
+ policy_mgr_debug("channel switch is in progress");
|
|
|
+ return true;
|
|
|
+ }
|
|
|
+
|
|
|
+ return false;
|
|
|
+}
|
|
|
+
|
|
|
+QDF_STATUS policy_mgr_wait_chan_switch_complete_evt(
|
|
|
+ struct wlan_objmgr_psoc *psoc)
|
|
|
+{
|
|
|
+ QDF_STATUS status;
|
|
|
+ struct policy_mgr_psoc_priv_obj *pm_ctx;
|
|
|
+
|
|
|
+ pm_ctx = policy_mgr_get_context(psoc);
|
|
|
+
|
|
|
+ if (!pm_ctx) {
|
|
|
+ policy_mgr_err("Invalid context");
|
|
|
+ return QDF_STATUS_E_FAILURE;
|
|
|
+ }
|
|
|
+
|
|
|
+ status = qdf_wait_single_event(
|
|
|
+ &pm_ctx->channel_switch_complete_evt,
|
|
|
+ CHANNEL_SWITCH_COMPLETE_TIMEOUT);
|
|
|
+ if (QDF_IS_STATUS_ERROR(status))
|
|
|
+ policy_mgr_err("wait for event failed, still continue with channel switch");
|
|
|
+
|
|
|
+ return status;
|
|
|
+}
|
|
|
+
|
|
|
+static void __policy_mgr_is_ap_start_in_progress(struct wlan_objmgr_pdev *pdev,
|
|
|
+ void *object, void *arg)
|
|
|
+{
|
|
|
+ struct wlan_objmgr_vdev *vdev = (struct wlan_objmgr_vdev *)object;
|
|
|
+ uint32_t *ap_starting_vdev_id = (uint32_t *)arg;
|
|
|
+ enum wlan_serialization_cmd_type cmd_type;
|
|
|
+ enum QDF_OPMODE op_mode;
|
|
|
+
|
|
|
+ if (!vdev || !ap_starting_vdev_id)
|
|
|
+ return;
|
|
|
+ if (*ap_starting_vdev_id != WLAN_INVALID_VDEV_ID)
|
|
|
+ return;
|
|
|
+ op_mode = wlan_vdev_mlme_get_opmode(vdev);
|
|
|
+ if (op_mode != QDF_SAP_MODE && op_mode != QDF_P2P_GO_MODE &&
|
|
|
+ op_mode != QDF_NDI_MODE)
|
|
|
+ return;
|
|
|
+ /* Check AP start is present in active and pending queue or not */
|
|
|
+ cmd_type = wlan_serialization_get_vdev_active_cmd_type(vdev);
|
|
|
+ if (cmd_type == WLAN_SER_CMD_VDEV_START_BSS ||
|
|
|
+ wlan_ser_is_non_scan_cmd_type_in_vdev_queue(
|
|
|
+ vdev, WLAN_SER_CMD_VDEV_START_BSS)) {
|
|
|
+ *ap_starting_vdev_id = wlan_vdev_get_id(vdev);
|
|
|
+ policy_mgr_debug("vdev %d op mode %d start bss is pending",
|
|
|
+ *ap_starting_vdev_id, op_mode);
|
|
|
+ }
|
|
|
+}
|
|
|
+
|
|
|
+bool policy_mgr_is_ap_start_in_progress(struct wlan_objmgr_psoc *psoc)
|
|
|
+{
|
|
|
+ struct policy_mgr_psoc_priv_obj *pm_ctx;
|
|
|
+ uint32_t ap_starting_vdev_id = WLAN_INVALID_VDEV_ID;
|
|
|
+
|
|
|
+ pm_ctx = policy_mgr_get_context(psoc);
|
|
|
+ if (!pm_ctx) {
|
|
|
+ policy_mgr_err("Invalid pm context");
|
|
|
+ return false;
|
|
|
+ }
|
|
|
+
|
|
|
+ wlan_objmgr_pdev_iterate_obj_list(pm_ctx->pdev, WLAN_VDEV_OP,
|
|
|
+ __policy_mgr_is_ap_start_in_progress,
|
|
|
+ &ap_starting_vdev_id, 0,
|
|
|
+ WLAN_POLICY_MGR_ID);
|
|
|
+
|
|
|
+ return ap_starting_vdev_id != WLAN_INVALID_VDEV_ID;
|
|
|
+}
|
|
|
+
|
|
|
void policy_mgr_process_force_scc_for_nan(struct wlan_objmgr_psoc *psoc)
|
|
|
{
|
|
|
struct policy_mgr_psoc_priv_obj *pm_ctx;
|