|
@@ -19,6 +19,10 @@
|
|
|
*/
|
|
|
#include "wlan_mlo_link_force.h"
|
|
|
#include "wlan_mlo_mgr_sta.h"
|
|
|
+#include "wlan_policy_mgr_api.h"
|
|
|
+#include "wlan_cm_roam_public_struct.h"
|
|
|
+#include "wlan_cm_roam_api.h"
|
|
|
+#include "wlan_mlo_mgr_roam.h"
|
|
|
|
|
|
void
|
|
|
ml_nlink_convert_linkid_bitmap_to_vdev_bitmap(
|
|
@@ -327,3 +331,1266 @@ ml_nlink_set_curr_force_inactive_num_state(struct wlan_objmgr_psoc *psoc,
|
|
|
link_num, link_bitmap);
|
|
|
mlo_dev_lock_release(mlo_dev_ctx);
|
|
|
}
|
|
|
+
|
|
|
+/**
|
|
|
+ * ml_nlink_get_affect_ml_sta() - Get ML STA whose link can be
|
|
|
+ * force inactive
|
|
|
+ * @psoc: PSOC object information
|
|
|
+ *
|
|
|
+ * At present we only support one ML STA. so ml_nlink_get_affect_ml_sta
|
|
|
+ * is invoked to get one ML STA vdev from policy mgr table.
|
|
|
+ * In future if ML STA+ML STA supported, we may need to extend it
|
|
|
+ * to find one ML STA which is required to force inactve/active.
|
|
|
+ *
|
|
|
+ * Return: vdev object
|
|
|
+ */
|
|
|
+static struct wlan_objmgr_vdev *
|
|
|
+ml_nlink_get_affect_ml_sta(struct wlan_objmgr_psoc *psoc)
|
|
|
+{
|
|
|
+ uint8_t num_ml_sta = 0, num_disabled_ml_sta = 0;
|
|
|
+ uint8_t ml_sta_vdev_lst[MAX_NUMBER_OF_CONC_CONNECTIONS] = {0};
|
|
|
+ qdf_freq_t ml_freq_lst[MAX_NUMBER_OF_CONC_CONNECTIONS] = {0};
|
|
|
+ struct wlan_objmgr_vdev *vdev;
|
|
|
+
|
|
|
+ policy_mgr_get_ml_sta_info_psoc(psoc, &num_ml_sta,
|
|
|
+ &num_disabled_ml_sta,
|
|
|
+ ml_sta_vdev_lst, ml_freq_lst, NULL,
|
|
|
+ NULL, NULL);
|
|
|
+ if (!num_ml_sta || num_ml_sta > MAX_NUMBER_OF_CONC_CONNECTIONS) {
|
|
|
+ mlo_debug("ml sta num is %d", num_ml_sta);
|
|
|
+ return NULL;
|
|
|
+ }
|
|
|
+ if (num_ml_sta > MAX_NUMBER_OF_CONC_CONNECTIONS) {
|
|
|
+ mlo_debug("unexpected num_ml_sta %d", num_ml_sta);
|
|
|
+ return NULL;
|
|
|
+ }
|
|
|
+ vdev = wlan_objmgr_get_vdev_by_id_from_psoc(
|
|
|
+ psoc, ml_sta_vdev_lst[0],
|
|
|
+ WLAN_MLO_MGR_ID);
|
|
|
+ if (!vdev) {
|
|
|
+ mlo_err("invalid vdev for id %d", ml_sta_vdev_lst[0]);
|
|
|
+ return NULL;
|
|
|
+ }
|
|
|
+
|
|
|
+ return vdev;
|
|
|
+}
|
|
|
+
|
|
|
+static bool ml_is_nlink_service_supported(void)
|
|
|
+{
|
|
|
+ /*todo: check WMI_SERVICE_N_LINK_MLO_SUPPORT service bit */
|
|
|
+ return false;
|
|
|
+}
|
|
|
+
|
|
|
+/* Exclude AP removed link */
|
|
|
+#define NLINK_EXCLUDE_REMOVED_LINK 0x01
|
|
|
+/* Include AP removed link only, can't work with other flags */
|
|
|
+#define NLINK_INCLUDE_REMOVED_LINK_ONLY 0x02
|
|
|
+/* Exclude QUITE link */
|
|
|
+#define NLINK_EXCLUDE_QUIET_LINK 0x04
|
|
|
+
|
|
|
+/**
|
|
|
+ * ml_nlink_get_link_info() - Get ML STA link info
|
|
|
+ * @psoc: PSOC object information
|
|
|
+ * @vdev: ml sta vdev object
|
|
|
+ * @flag: flag NLINK_* to specify what links should be returned
|
|
|
+ * @ml_num_link_sz: input array size of ml_link_info and
|
|
|
+ * other parameters.
|
|
|
+ * @ml_link_info: ml link info array
|
|
|
+ * @ml_freq_lst: channel frequency list
|
|
|
+ * @ml_vdev_lst: vdev id list
|
|
|
+ * @ml_linkid_lst: link id list
|
|
|
+ * @ml_num_link: num of links
|
|
|
+ * @ml_link_bitmap: link bitmaps.
|
|
|
+ *
|
|
|
+ * Return: void
|
|
|
+ */
|
|
|
+static void ml_nlink_get_link_info(struct wlan_objmgr_psoc *psoc,
|
|
|
+ struct wlan_objmgr_vdev *vdev,
|
|
|
+ uint8_t flag,
|
|
|
+ uint8_t ml_num_link_sz,
|
|
|
+ struct ml_link_info *ml_link_info,
|
|
|
+ qdf_freq_t *ml_freq_lst,
|
|
|
+ uint8_t *ml_vdev_lst,
|
|
|
+ uint8_t *ml_linkid_lst,
|
|
|
+ uint8_t *ml_num_link,
|
|
|
+ uint32_t *ml_link_bitmap)
|
|
|
+{
|
|
|
+ struct wlan_mlo_dev_context *mlo_dev_ctx;
|
|
|
+ struct wlan_mlo_sta *sta_ctx;
|
|
|
+ uint8_t i, num_link = 0;
|
|
|
+ uint32_t link_bitmap = 0;
|
|
|
+ uint16_t link_id;
|
|
|
+ uint8_t vdev_id;
|
|
|
+
|
|
|
+ *ml_num_link = 0;
|
|
|
+ *ml_link_bitmap = 0;
|
|
|
+ qdf_mem_zero(ml_link_info, sizeof(*ml_link_info) * ml_num_link_sz);
|
|
|
+ qdf_mem_zero(ml_freq_lst, sizeof(*ml_freq_lst) * ml_num_link_sz);
|
|
|
+ qdf_mem_zero(ml_linkid_lst, sizeof(*ml_linkid_lst) * ml_num_link_sz);
|
|
|
+ qdf_mem_zero(ml_vdev_lst, sizeof(*ml_vdev_lst) * ml_num_link_sz);
|
|
|
+
|
|
|
+ mlo_dev_ctx = wlan_vdev_get_mlo_dev_ctx(vdev);
|
|
|
+ if (!mlo_dev_ctx || !mlo_dev_ctx->sta_ctx) {
|
|
|
+ mlo_err("mlo_ctx or sta_ctx null");
|
|
|
+ return;
|
|
|
+ }
|
|
|
+
|
|
|
+ mlo_dev_lock_acquire(mlo_dev_ctx);
|
|
|
+ sta_ctx = mlo_dev_ctx->sta_ctx;
|
|
|
+
|
|
|
+ link_bitmap = 0;
|
|
|
+ for (i = 0; i < WLAN_UMAC_MLO_MAX_VDEVS; i++) {
|
|
|
+ /*todo: add standby link */
|
|
|
+ if (!mlo_dev_ctx->wlan_vdev_list[i])
|
|
|
+ continue;
|
|
|
+ if (!qdf_test_bit(i, sta_ctx->wlan_connected_links))
|
|
|
+ continue;
|
|
|
+
|
|
|
+ if (!wlan_cm_is_vdev_connected(
|
|
|
+ mlo_dev_ctx->wlan_vdev_list[i])) {
|
|
|
+ mlo_debug("Vdev id %d is not in connected state",
|
|
|
+ wlan_vdev_get_id(
|
|
|
+ mlo_dev_ctx->wlan_vdev_list[i]));
|
|
|
+ continue;
|
|
|
+ }
|
|
|
+
|
|
|
+ vdev_id = wlan_vdev_get_id(mlo_dev_ctx->wlan_vdev_list[i]);
|
|
|
+ link_id = wlan_vdev_get_link_id(
|
|
|
+ mlo_dev_ctx->wlan_vdev_list[i]);
|
|
|
+ if (link_id >= MAX_MLO_LINK_ID) {
|
|
|
+ mlo_debug("invalid link id %x for vdev %d",
|
|
|
+ link_id, vdev_id);
|
|
|
+ continue;
|
|
|
+ }
|
|
|
+
|
|
|
+ if ((flag & NLINK_EXCLUDE_REMOVED_LINK) &&
|
|
|
+ wlan_get_vdev_link_removed_flag_by_vdev_id(
|
|
|
+ psoc, vdev_id)) {
|
|
|
+ mlo_debug("vdev id %d link %d is removed",
|
|
|
+ vdev_id, link_id);
|
|
|
+ continue;
|
|
|
+ }
|
|
|
+ if ((flag & NLINK_INCLUDE_REMOVED_LINK_ONLY) &&
|
|
|
+ !wlan_get_vdev_link_removed_flag_by_vdev_id(
|
|
|
+ psoc, vdev_id)) {
|
|
|
+ continue;
|
|
|
+ }
|
|
|
+ if ((flag & NLINK_EXCLUDE_QUIET_LINK) &&
|
|
|
+ mlo_is_sta_in_quiet_status(mlo_dev_ctx, link_id)) {
|
|
|
+ mlo_debug("vdev id %d link %d is quiet",
|
|
|
+ vdev_id, link_id);
|
|
|
+ continue;
|
|
|
+ }
|
|
|
+
|
|
|
+ if (num_link >= ml_num_link_sz)
|
|
|
+ break;
|
|
|
+ ml_freq_lst[num_link] = wlan_get_operation_chan_freq(
|
|
|
+ mlo_dev_ctx->wlan_vdev_list[i]);
|
|
|
+ ml_vdev_lst[num_link] = vdev_id;
|
|
|
+ ml_linkid_lst[num_link] = link_id;
|
|
|
+ link_bitmap |= 1 << link_id;
|
|
|
+
|
|
|
+ mlo_debug("vdev %d link %d freq %d bitmap 0x%x flag 0x%x",
|
|
|
+ ml_vdev_lst[num_link], ml_linkid_lst[num_link],
|
|
|
+ ml_freq_lst[num_link], link_bitmap, flag);
|
|
|
+ num_link++;
|
|
|
+ }
|
|
|
+ mlo_dev_lock_release(mlo_dev_ctx);
|
|
|
+ *ml_num_link = num_link;
|
|
|
+ *ml_link_bitmap = link_bitmap;
|
|
|
+}
|
|
|
+
|
|
|
+/**
|
|
|
+ * convert_link_bitmap_to_link_ids() - Convert link bitmap to link ids
|
|
|
+ * @link_bitmap: PSOC object information
|
|
|
+ * @link_id_sz: link_ids array size
|
|
|
+ * @link_ids: link id array
|
|
|
+ *
|
|
|
+ * Return: num of link id in link_ids array converted from link bitmap
|
|
|
+ */
|
|
|
+static uint32_t
|
|
|
+convert_link_bitmap_to_link_ids(uint32_t link_bitmap,
|
|
|
+ uint8_t link_id_sz,
|
|
|
+ uint8_t *link_ids)
|
|
|
+{
|
|
|
+ uint32_t i = 0;
|
|
|
+ uint8_t id = 0;
|
|
|
+
|
|
|
+ while (link_bitmap) {
|
|
|
+ if (link_bitmap & 1) {
|
|
|
+ if (id >= 15) {
|
|
|
+ /* warning */
|
|
|
+ mlo_err("linkid invalid %d 0x%x",
|
|
|
+ id, link_bitmap);
|
|
|
+ break;
|
|
|
+ }
|
|
|
+ if (link_ids) {
|
|
|
+ if (i >= link_id_sz) {
|
|
|
+ /* warning */
|
|
|
+ mlo_err("linkid buff overflow 0x%x",
|
|
|
+ link_bitmap);
|
|
|
+ break;
|
|
|
+ }
|
|
|
+ link_ids[i] = id;
|
|
|
+ }
|
|
|
+ i++;
|
|
|
+ }
|
|
|
+ link_bitmap >>= 1;
|
|
|
+ id++;
|
|
|
+ }
|
|
|
+
|
|
|
+ return i;
|
|
|
+}
|
|
|
+
|
|
|
+/**
|
|
|
+ * ml_nlink_handle_mcc_links() - Check force inactive needed
|
|
|
+ * if ML STA links are in MCC channels
|
|
|
+ * @psoc: PSOC object information
|
|
|
+ * @vdev: vdev object
|
|
|
+ * @force_cmd: force command to be returned
|
|
|
+ *
|
|
|
+ * This API will return force inactive number 1 in force_cmd
|
|
|
+ * if STA links are in MCC channels with the link bitmap including
|
|
|
+ * the MCC links id.
|
|
|
+ * If the link is marked removed by AP MLD, return force inactive
|
|
|
+ * bitmap with removed link id bitmap as well.
|
|
|
+ *
|
|
|
+ * Return: void
|
|
|
+ */
|
|
|
+static void
|
|
|
+ml_nlink_handle_mcc_links(struct wlan_objmgr_psoc *psoc,
|
|
|
+ struct wlan_objmgr_vdev *vdev,
|
|
|
+ struct ml_link_force_state *force_cmd)
|
|
|
+{
|
|
|
+ uint8_t ml_num_link = 0;
|
|
|
+ uint32_t ml_link_bitmap = 0, affected_link_bitmap = 0;
|
|
|
+ uint32_t force_inactive_link_bitmap = 0;
|
|
|
+ uint8_t ml_vdev_lst[MAX_NUMBER_OF_CONC_CONNECTIONS];
|
|
|
+ qdf_freq_t ml_freq_lst[MAX_NUMBER_OF_CONC_CONNECTIONS];
|
|
|
+ uint8_t ml_linkid_lst[MAX_NUMBER_OF_CONC_CONNECTIONS];
|
|
|
+ struct ml_link_info ml_link_info[MAX_NUMBER_OF_CONC_CONNECTIONS];
|
|
|
+
|
|
|
+ ml_nlink_get_link_info(psoc, vdev, NLINK_INCLUDE_REMOVED_LINK_ONLY,
|
|
|
+ QDF_ARRAY_SIZE(ml_linkid_lst),
|
|
|
+ ml_link_info, ml_freq_lst, ml_vdev_lst,
|
|
|
+ ml_linkid_lst, &ml_num_link,
|
|
|
+ &force_inactive_link_bitmap);
|
|
|
+ if (force_inactive_link_bitmap) {
|
|
|
+ /* AP removed link will be force inactive always */
|
|
|
+ force_cmd->force_inactive_bitmap = force_inactive_link_bitmap;
|
|
|
+ mlo_debug("AP removed link 0x%x", force_inactive_link_bitmap);
|
|
|
+ }
|
|
|
+
|
|
|
+ ml_nlink_get_link_info(psoc, vdev, NLINK_EXCLUDE_REMOVED_LINK,
|
|
|
+ QDF_ARRAY_SIZE(ml_linkid_lst),
|
|
|
+ ml_link_info, ml_freq_lst, ml_vdev_lst,
|
|
|
+ ml_linkid_lst, &ml_num_link,
|
|
|
+ &ml_link_bitmap);
|
|
|
+ if (ml_num_link < 2)
|
|
|
+ return;
|
|
|
+
|
|
|
+ policy_mgr_is_ml_sta_links_in_mcc(psoc, ml_freq_lst,
|
|
|
+ ml_vdev_lst,
|
|
|
+ ml_linkid_lst,
|
|
|
+ ml_num_link,
|
|
|
+ &affected_link_bitmap);
|
|
|
+ if (affected_link_bitmap) {
|
|
|
+ force_cmd->force_inactive_num =
|
|
|
+ convert_link_bitmap_to_link_ids(
|
|
|
+ affected_link_bitmap, 0, NULL);
|
|
|
+ if (force_cmd->force_inactive_num > 1) {
|
|
|
+ force_cmd->force_inactive_num--;
|
|
|
+ force_cmd->force_inactive_num_bitmap =
|
|
|
+ affected_link_bitmap;
|
|
|
+ } else {
|
|
|
+ force_cmd->force_inactive_num = 0;
|
|
|
+ }
|
|
|
+ }
|
|
|
+ if (force_inactive_link_bitmap || affected_link_bitmap)
|
|
|
+ ml_nlink_dump_force_state(force_cmd, "");
|
|
|
+}
|
|
|
+
|
|
|
+/**
|
|
|
+ * ml_nlink_handle_legacy_sta_intf() - Check force inactive needed
|
|
|
+ * with legacy STA
|
|
|
+ * @psoc: PSOC object information
|
|
|
+ * @vdev: vdev object
|
|
|
+ * @force_cmd: force command to be returned
|
|
|
+ * @sta_vdev_id: legacy STA vdev id
|
|
|
+ * @non_ml_sta_freq: legacy STA channel frequency
|
|
|
+ *
|
|
|
+ * If legacy STA is MCC with any link of MLO STA, the mlo link
|
|
|
+ * will be forced inactive. And if 3 link MLO case, the left
|
|
|
+ * 2 links have to be force inactive with num 1. For example,
|
|
|
+ * ML STA 2+5+6, legacy STA on MCC channel of 5G link, then
|
|
|
+ * 5G will be force inactive, and left 2+6 link will be force
|
|
|
+ * inactive by inactive link num = 1 (with link bitmap 2+6).
|
|
|
+ *
|
|
|
+ * Return: void
|
|
|
+ */
|
|
|
+static void
|
|
|
+ml_nlink_handle_legacy_sta_intf(struct wlan_objmgr_psoc *psoc,
|
|
|
+ struct wlan_objmgr_vdev *vdev,
|
|
|
+ struct ml_link_force_state *force_cmd,
|
|
|
+ uint8_t sta_vdev_id,
|
|
|
+ qdf_freq_t non_ml_sta_freq)
|
|
|
+{
|
|
|
+ uint8_t ml_num_link = 0;
|
|
|
+ uint32_t ml_link_bitmap = 0, affected_link_bitmap = 0;
|
|
|
+ uint32_t force_inactive_link_bitmap = 0;
|
|
|
+ uint8_t ml_vdev_lst[MAX_NUMBER_OF_CONC_CONNECTIONS];
|
|
|
+ qdf_freq_t ml_freq_lst[MAX_NUMBER_OF_CONC_CONNECTIONS];
|
|
|
+ uint8_t ml_linkid_lst[MAX_NUMBER_OF_CONC_CONNECTIONS];
|
|
|
+ struct ml_link_info ml_link_info[MAX_NUMBER_OF_CONC_CONNECTIONS];
|
|
|
+ uint8_t i = 0;
|
|
|
+ uint32_t scc_link_bitmap = 0;
|
|
|
+
|
|
|
+ ml_nlink_get_link_info(psoc, vdev, NLINK_EXCLUDE_REMOVED_LINK,
|
|
|
+ QDF_ARRAY_SIZE(ml_linkid_lst),
|
|
|
+ ml_link_info, ml_freq_lst, ml_vdev_lst,
|
|
|
+ ml_linkid_lst, &ml_num_link,
|
|
|
+ &ml_link_bitmap);
|
|
|
+ if (ml_num_link < 2)
|
|
|
+ return;
|
|
|
+
|
|
|
+ for (i = 0; i < ml_num_link; i++) {
|
|
|
+ /*todo add removed link to force_inactive_link_bitmap*/
|
|
|
+ if (ml_freq_lst[i] == non_ml_sta_freq) {
|
|
|
+ scc_link_bitmap = 1 << ml_linkid_lst[i];
|
|
|
+ } else if (policy_mgr_2_freq_always_on_same_mac(
|
|
|
+ psoc, ml_freq_lst[i], non_ml_sta_freq)) {
|
|
|
+ force_inactive_link_bitmap |= 1 << ml_linkid_lst[i];
|
|
|
+ } else if (!wlan_cm_same_band_sta_allowed(psoc) &&
|
|
|
+ (wlan_reg_is_24ghz_ch_freq(ml_freq_lst[i]) ==
|
|
|
+ wlan_reg_is_24ghz_ch_freq(non_ml_sta_freq)) &&
|
|
|
+ !policy_mgr_are_sbs_chan(psoc, ml_freq_lst[i],
|
|
|
+ non_ml_sta_freq)) {
|
|
|
+ force_inactive_link_bitmap |= 1 << ml_linkid_lst[i];
|
|
|
+ }
|
|
|
+ }
|
|
|
+
|
|
|
+ /* If no left active link, don't send the force inactive command for
|
|
|
+ * concurrency purpose.
|
|
|
+ */
|
|
|
+ if (!(ml_link_bitmap & ~force_inactive_link_bitmap)) {
|
|
|
+ mlo_debug("unexpected ML conc with legacy STA freq %d",
|
|
|
+ non_ml_sta_freq);
|
|
|
+ return;
|
|
|
+ }
|
|
|
+
|
|
|
+ if (force_inactive_link_bitmap) {
|
|
|
+ /* for example SBS rd, ML 2G+5G high, Legacy intf on 5G high,
|
|
|
+ * set force inactive with bitmap of 5g link.
|
|
|
+ *
|
|
|
+ * for SBS rd, ML 2G + 5G low + 6G, Legacy intf on 2G.
|
|
|
+ * set force inactive with bitmap 2G link,
|
|
|
+ * and set force inactive link num to 1 for left 5g and 6g
|
|
|
+ * link.
|
|
|
+ * for SBS rd, ML 2G + 5G low + 6G, Legacy intf on 5G low.
|
|
|
+ * set force inactive with bitmap 5G low link,
|
|
|
+ * and set force inactive link num to 1 for left 2g and 6g
|
|
|
+ * link.
|
|
|
+ * for SBS rd, ML 2G + 5G low + 6G, Legacy intf on 5G high.
|
|
|
+ * set force inactive with bitmap 6G link,
|
|
|
+ * and set force inactive link num to 1 for left 2g and 5g
|
|
|
+ * link.
|
|
|
+ * In above 3 link cases, if legacy intf is SCC with ml link
|
|
|
+ * don't force inactive by bitmap, only send force inactive
|
|
|
+ * num with bitmap
|
|
|
+ */
|
|
|
+ force_cmd->force_inactive_bitmap = force_inactive_link_bitmap;
|
|
|
+
|
|
|
+ affected_link_bitmap =
|
|
|
+ ml_link_bitmap & ~force_inactive_link_bitmap;
|
|
|
+ affected_link_bitmap &= ~scc_link_bitmap;
|
|
|
+ force_cmd->force_inactive_num =
|
|
|
+ convert_link_bitmap_to_link_ids(
|
|
|
+ affected_link_bitmap, 0, NULL);
|
|
|
+ if (force_cmd->force_inactive_num > 1) {
|
|
|
+ force_cmd->force_inactive_num--;
|
|
|
+ force_cmd->force_inactive_num_bitmap =
|
|
|
+ affected_link_bitmap;
|
|
|
+
|
|
|
+ } else {
|
|
|
+ force_cmd->force_inactive_num = 0;
|
|
|
+ }
|
|
|
+ } else {
|
|
|
+ /* for example SBS rd, ML 2G+5G high, Legacy intf on 5G low,
|
|
|
+ * set force inactive num to 1 with bitmap of 2g+5g link.
|
|
|
+ *
|
|
|
+ * for SBS rd, ML 2G + 5G low + 6G, Legacy intf on 5G low SCC.
|
|
|
+ * set force inactive link num to 1 for left 2g and 6g
|
|
|
+ * link.
|
|
|
+ */
|
|
|
+ affected_link_bitmap = ml_link_bitmap;
|
|
|
+ affected_link_bitmap &= ~scc_link_bitmap;
|
|
|
+
|
|
|
+ force_cmd->force_inactive_num =
|
|
|
+ convert_link_bitmap_to_link_ids(
|
|
|
+ affected_link_bitmap, 0, NULL);
|
|
|
+ if (force_cmd->force_inactive_num > 1) {
|
|
|
+ force_cmd->force_inactive_num--;
|
|
|
+ force_cmd->force_inactive_num_bitmap =
|
|
|
+ affected_link_bitmap;
|
|
|
+ } else {
|
|
|
+ force_cmd->force_inactive_num = 0;
|
|
|
+ }
|
|
|
+ }
|
|
|
+}
|
|
|
+
|
|
|
+/**
|
|
|
+ * ml_nlink_handle_legacy_sap_intf() - Check force inactive needed
|
|
|
+ * with legacy SAP
|
|
|
+ * @psoc: PSOC object information
|
|
|
+ * @vdev: vdev object
|
|
|
+ * @force_cmd: force command to be returned
|
|
|
+ * @sap_vdev_id: legacy SAP vdev id
|
|
|
+ * @sap_freq: legacy SAP channel frequency
|
|
|
+ *
|
|
|
+ * If legacy SAP is 2g only SAP and MLO STA is 5+6,
|
|
|
+ * 2 links have to be force inactive with num 1.
|
|
|
+ *
|
|
|
+ * Return: void
|
|
|
+ */
|
|
|
+static void
|
|
|
+ml_nlink_handle_legacy_sap_intf(struct wlan_objmgr_psoc *psoc,
|
|
|
+ struct wlan_objmgr_vdev *vdev,
|
|
|
+ struct ml_link_force_state *force_cmd,
|
|
|
+ uint8_t sap_vdev_id,
|
|
|
+ qdf_freq_t sap_freq)
|
|
|
+{
|
|
|
+ uint8_t ml_num_link = 0;
|
|
|
+ uint32_t ml_link_bitmap = 0, affected_link_bitmap = 0;
|
|
|
+ uint8_t ml_vdev_lst[MAX_NUMBER_OF_CONC_CONNECTIONS];
|
|
|
+ qdf_freq_t ml_freq_lst[MAX_NUMBER_OF_CONC_CONNECTIONS];
|
|
|
+ uint8_t ml_linkid_lst[MAX_NUMBER_OF_CONC_CONNECTIONS];
|
|
|
+ struct ml_link_info ml_link_info[MAX_NUMBER_OF_CONC_CONNECTIONS];
|
|
|
+ uint8_t i = 0;
|
|
|
+ bool sap_2g_only = false;
|
|
|
+
|
|
|
+ /* SAP MCC with MLO STA link is not preferred.
|
|
|
+ * If SAP is 2Ghz only by ACS and two ML link are
|
|
|
+ * 5/6 band, then force SCC may not happen. In such
|
|
|
+ * case inactive one link.
|
|
|
+ */
|
|
|
+ if (policy_mgr_check_2ghz_only_sap_affected_link(
|
|
|
+ psoc, sap_vdev_id, sap_freq,
|
|
|
+ ml_num_link, ml_freq_lst)) {
|
|
|
+ mlo_debug("2G only SAP vdev %d ch freq %d is not SCC with any MLO STA link",
|
|
|
+ sap_vdev_id, sap_freq);
|
|
|
+ sap_2g_only = true;
|
|
|
+ }
|
|
|
+ /*
|
|
|
+ * If SAP is on 5G or 6G, SAP can always force SCC to 5G/6G ML STA or
|
|
|
+ * 2G ML STA, no need force SCC link.
|
|
|
+ */
|
|
|
+ if (!sap_2g_only)
|
|
|
+ return;
|
|
|
+
|
|
|
+ ml_nlink_get_link_info(psoc, vdev, NLINK_EXCLUDE_REMOVED_LINK,
|
|
|
+ QDF_ARRAY_SIZE(ml_linkid_lst),
|
|
|
+ ml_link_info, ml_freq_lst, ml_vdev_lst,
|
|
|
+ ml_linkid_lst, &ml_num_link,
|
|
|
+ &ml_link_bitmap);
|
|
|
+ if (ml_num_link < 2)
|
|
|
+ return;
|
|
|
+
|
|
|
+ for (i = 0; i < ml_num_link; i++) {
|
|
|
+ if (!wlan_reg_is_24ghz_ch_freq(ml_freq_lst[i]))
|
|
|
+ affected_link_bitmap |= 1 << ml_linkid_lst[i];
|
|
|
+ }
|
|
|
+
|
|
|
+ if (affected_link_bitmap) {
|
|
|
+ /* for SBS rd, ML 2G + 5G low, Legacy SAP on 2G.
|
|
|
+ * no force any link
|
|
|
+ * for SBS rd, ML 5G low + 5G high/6G, Legacy SAP on 2G.
|
|
|
+ * set force inactive num 1 with bitmap 5g and 6g.
|
|
|
+ *
|
|
|
+ * for SBS rd, ML 2G + 5G low + 6G, Legacy SAP on 2G.
|
|
|
+ * set force inactive link num to 1 for 5g and 6g
|
|
|
+ * link.
|
|
|
+ */
|
|
|
+ force_cmd->force_inactive_num =
|
|
|
+ convert_link_bitmap_to_link_ids(
|
|
|
+ affected_link_bitmap, 0, NULL);
|
|
|
+ if (force_cmd->force_inactive_num > 1) {
|
|
|
+ force_cmd->force_inactive_num--;
|
|
|
+ force_cmd->force_inactive_num_bitmap =
|
|
|
+ affected_link_bitmap;
|
|
|
+ } else {
|
|
|
+ force_cmd->force_inactive_num = 0;
|
|
|
+ }
|
|
|
+ }
|
|
|
+}
|
|
|
+
|
|
|
+/**
|
|
|
+ * ml_nlink_handle_legacy_p2p_intf() - Check force inactive needed
|
|
|
+ * with p2p
|
|
|
+ * @psoc: PSOC object information
|
|
|
+ * @vdev: vdev object
|
|
|
+ * @force_cmd: force command to be returned
|
|
|
+ * @p2p_vdev_id: p2p vdev id
|
|
|
+ * @p2p_freq: p2p channel frequency
|
|
|
+ *
|
|
|
+ * If P2P has low latency flag and MCC with any link of MLO STA, the mlo link
|
|
|
+ * will be forced inactive. And if 3 link MLO case, the left 2 links have to
|
|
|
+ * be force inactive with num 1.
|
|
|
+ *
|
|
|
+ * Return: void
|
|
|
+ */
|
|
|
+static void
|
|
|
+ml_nlink_handle_legacy_p2p_intf(struct wlan_objmgr_psoc *psoc,
|
|
|
+ struct wlan_objmgr_vdev *vdev,
|
|
|
+ struct ml_link_force_state *force_cmd,
|
|
|
+ uint8_t p2p_vdev_id,
|
|
|
+ qdf_freq_t p2p_freq)
|
|
|
+{
|
|
|
+ uint8_t ml_num_link = 0;
|
|
|
+ uint32_t ml_link_bitmap = 0, affected_link_bitmap = 0;
|
|
|
+ uint32_t force_inactive_link_bitmap = 0;
|
|
|
+ uint8_t ml_vdev_lst[MAX_NUMBER_OF_CONC_CONNECTIONS];
|
|
|
+ qdf_freq_t ml_freq_lst[MAX_NUMBER_OF_CONC_CONNECTIONS];
|
|
|
+ uint8_t ml_linkid_lst[MAX_NUMBER_OF_CONC_CONNECTIONS];
|
|
|
+ struct ml_link_info ml_link_info[MAX_NUMBER_OF_CONC_CONNECTIONS];
|
|
|
+ uint8_t i = 0;
|
|
|
+ uint32_t scc_link_bitmap = 0;
|
|
|
+
|
|
|
+ /* If high tput or low latency is not set, mcc is allowed for p2p */
|
|
|
+ if (!policy_mgr_is_vdev_high_tput_or_low_latency(
|
|
|
+ psoc, p2p_vdev_id))
|
|
|
+ return;
|
|
|
+ ml_nlink_get_link_info(psoc, vdev, NLINK_EXCLUDE_REMOVED_LINK,
|
|
|
+ QDF_ARRAY_SIZE(ml_linkid_lst),
|
|
|
+ ml_link_info, ml_freq_lst, ml_vdev_lst,
|
|
|
+ ml_linkid_lst, &ml_num_link,
|
|
|
+ &ml_link_bitmap);
|
|
|
+ if (ml_num_link < 2)
|
|
|
+ return;
|
|
|
+
|
|
|
+ for (i = 0; i < ml_num_link; i++) {
|
|
|
+ if (ml_freq_lst[i] == p2p_freq) {
|
|
|
+ scc_link_bitmap = 1 << ml_linkid_lst[i];
|
|
|
+ } else if (policy_mgr_2_freq_always_on_same_mac(
|
|
|
+ psoc, ml_freq_lst[i], p2p_freq)) {
|
|
|
+ force_inactive_link_bitmap |= 1 << ml_linkid_lst[i];
|
|
|
+ }
|
|
|
+ }
|
|
|
+ /* If no left active link, don't send the force inactive command for
|
|
|
+ * concurrency purpose.
|
|
|
+ */
|
|
|
+ if (!(ml_link_bitmap & ~force_inactive_link_bitmap)) {
|
|
|
+ mlo_debug("unexpected ML conc with legacy P2P freq %d",
|
|
|
+ p2p_freq);
|
|
|
+ return;
|
|
|
+ }
|
|
|
+
|
|
|
+ if (force_inactive_link_bitmap) {
|
|
|
+ /* for example SBS rd, ML 2G+5G high, Legacy intf on 5G high,
|
|
|
+ * set force inactive with bitmap of 5g link.
|
|
|
+ *
|
|
|
+ * for SBS rd, ML 2G + 5G low + 6G, Legacy intf on 2G.
|
|
|
+ * set force inactive with bitmap 2G link,
|
|
|
+ * and set force inactive link num to 1 for left 5g and 6g
|
|
|
+ * link.
|
|
|
+ * for SBS rd, ML 2G + 5G low + 6G, Legacy intf on 5G low.
|
|
|
+ * set force inactive with bitmap 5G low link,
|
|
|
+ * and set force inactive link num to 1 for left 2g and 6g
|
|
|
+ * link.
|
|
|
+ * for SBS rd, ML 2G + 5G low + 6G, Legacy intf on 5G high..
|
|
|
+ * set force inactive with bitmap 6G low link,
|
|
|
+ * and set force inactive link num to 1 for left 2g and 5g
|
|
|
+ * link.
|
|
|
+ */
|
|
|
+ force_cmd->force_inactive_bitmap = force_inactive_link_bitmap;
|
|
|
+
|
|
|
+ affected_link_bitmap =
|
|
|
+ ml_link_bitmap & ~force_inactive_link_bitmap;
|
|
|
+ affected_link_bitmap &= ~scc_link_bitmap;
|
|
|
+ force_cmd->force_inactive_num =
|
|
|
+ convert_link_bitmap_to_link_ids(
|
|
|
+ affected_link_bitmap, 0, NULL);
|
|
|
+ if (force_cmd->force_inactive_num > 1) {
|
|
|
+ force_cmd->force_inactive_num--;
|
|
|
+ force_cmd->force_inactive_num_bitmap =
|
|
|
+ affected_link_bitmap;
|
|
|
+
|
|
|
+ } else {
|
|
|
+ force_cmd->force_inactive_num = 0;
|
|
|
+ }
|
|
|
+ } else {
|
|
|
+ /* for example SBS rd, ML 2G+5G high, Legacy intf on 5G low,
|
|
|
+ * set force inactive num to 1 with bitmap of 2g+5g link.
|
|
|
+ *
|
|
|
+ * for SBS rd, ML 2G + 5G low + 6G, Legacy intf on 5G low SCC.
|
|
|
+ * set force inactive link num to 1 for left 2g and 6g
|
|
|
+ * link.
|
|
|
+ */
|
|
|
+ affected_link_bitmap = ml_link_bitmap;
|
|
|
+ affected_link_bitmap &= ~scc_link_bitmap;
|
|
|
+
|
|
|
+ force_cmd->force_inactive_num =
|
|
|
+ convert_link_bitmap_to_link_ids(
|
|
|
+ affected_link_bitmap, 0, NULL);
|
|
|
+ if (force_cmd->force_inactive_num > 1) {
|
|
|
+ force_cmd->force_inactive_num--;
|
|
|
+ force_cmd->force_inactive_num_bitmap =
|
|
|
+ affected_link_bitmap;
|
|
|
+ } else {
|
|
|
+ force_cmd->force_inactive_num = 0;
|
|
|
+ }
|
|
|
+ }
|
|
|
+}
|
|
|
+
|
|
|
+/**
|
|
|
+ * ml_nlink_handle_legacy_intf_3_ports() - Check force inactive needed
|
|
|
+ * with 2 legacy interfaces
|
|
|
+ * @psoc: PSOC object information
|
|
|
+ * @vdev: vdev object
|
|
|
+ * @force_cmd: force command to be returned
|
|
|
+ * @legacy_intf_freq1: legacy interface frequency
|
|
|
+ * @legacy_intf_freq2: legacy interface frequency
|
|
|
+ *
|
|
|
+ * If legacy interface is mcc with any link based on current hw mode, then
|
|
|
+ * force inactive the link.
|
|
|
+ * And if standby link is mcc with legacy interface, then disable standby
|
|
|
+ * link as well.
|
|
|
+ *
|
|
|
+ * Return: void
|
|
|
+ */
|
|
|
+static void
|
|
|
+ml_nlink_handle_legacy_intf_3_ports(struct wlan_objmgr_psoc *psoc,
|
|
|
+ struct wlan_objmgr_vdev *vdev,
|
|
|
+ struct ml_link_force_state *force_cmd,
|
|
|
+ qdf_freq_t legacy_intf_freq1,
|
|
|
+ qdf_freq_t legacy_intf_freq2)
|
|
|
+{
|
|
|
+ uint8_t ml_num_link = 0;
|
|
|
+ uint32_t ml_link_bitmap = 0;
|
|
|
+ uint32_t force_inactive_link_bitmap = 0;
|
|
|
+ uint8_t ml_vdev_lst[MAX_NUMBER_OF_CONC_CONNECTIONS];
|
|
|
+ qdf_freq_t ml_freq_lst[MAX_NUMBER_OF_CONC_CONNECTIONS];
|
|
|
+ uint8_t ml_linkid_lst[MAX_NUMBER_OF_CONC_CONNECTIONS];
|
|
|
+ struct ml_link_info ml_link_info[MAX_NUMBER_OF_CONC_CONNECTIONS];
|
|
|
+ uint8_t i = 0;
|
|
|
+ uint32_t scc_link_bitmap = 0;
|
|
|
+
|
|
|
+ ml_nlink_get_link_info(psoc, vdev, NLINK_EXCLUDE_REMOVED_LINK,
|
|
|
+ QDF_ARRAY_SIZE(ml_linkid_lst),
|
|
|
+ ml_link_info, ml_freq_lst, ml_vdev_lst,
|
|
|
+ ml_linkid_lst, &ml_num_link,
|
|
|
+ &ml_link_bitmap);
|
|
|
+ if (ml_num_link < 2)
|
|
|
+ return;
|
|
|
+
|
|
|
+ for (i = 0; i < ml_num_link; i++) {
|
|
|
+ if (ml_vdev_lst[i] == 0xff) {
|
|
|
+ /*standby link will be handled later. */
|
|
|
+ continue;
|
|
|
+ }
|
|
|
+ if (ml_freq_lst[i] == legacy_intf_freq1) {
|
|
|
+ scc_link_bitmap = 1 << ml_linkid_lst[i];
|
|
|
+ if (ml_freq_lst[i] == legacy_intf_freq2) {
|
|
|
+ mlo_debug("3 vdev scc no-op");
|
|
|
+ return;
|
|
|
+ }
|
|
|
+ } else if (policy_mgr_are_2_freq_on_same_mac(
|
|
|
+ psoc, ml_freq_lst[i], legacy_intf_freq1)) {
|
|
|
+ force_inactive_link_bitmap |= 1 << ml_linkid_lst[i];
|
|
|
+ }
|
|
|
+ }
|
|
|
+ /* usually it can't happen in 3 Port */
|
|
|
+ if (!force_inactive_link_bitmap && !scc_link_bitmap) {
|
|
|
+ mlo_debug("legacy vdev freq %d standalone on dedicated mac",
|
|
|
+ legacy_intf_freq1);
|
|
|
+ return;
|
|
|
+ }
|
|
|
+ /* handle standby link */
|
|
|
+ for (i = 0; i < ml_num_link; i++) {
|
|
|
+ if (ml_vdev_lst[i] == 0xff) {
|
|
|
+ /*standby link will be forced inactive if mcc with
|
|
|
+ * legacy sta
|
|
|
+ */
|
|
|
+ if (ml_freq_lst[i] != legacy_intf_freq1)
|
|
|
+ force_inactive_link_bitmap |=
|
|
|
+ 1 << ml_linkid_lst[i];
|
|
|
+ }
|
|
|
+ }
|
|
|
+
|
|
|
+ if (force_inactive_link_bitmap)
|
|
|
+ force_cmd->force_inactive_bitmap = force_inactive_link_bitmap;
|
|
|
+}
|
|
|
+
|
|
|
+/**
|
|
|
+ * ml_nlink_handle_legacy_intf() - Check force inactive needed
|
|
|
+ * with legacy interface
|
|
|
+ * @psoc: PSOC object information
|
|
|
+ * @vdev: vdev object
|
|
|
+ * @force_cmd: force command to be returned
|
|
|
+ *
|
|
|
+ * Return: void
|
|
|
+ */
|
|
|
+static void
|
|
|
+ml_nlink_handle_legacy_intf(struct wlan_objmgr_psoc *psoc,
|
|
|
+ struct wlan_objmgr_vdev *vdev,
|
|
|
+ struct ml_link_force_state *force_cmd)
|
|
|
+{
|
|
|
+ uint8_t vdev_lst[MAX_NUMBER_OF_CONC_CONNECTIONS];
|
|
|
+ qdf_freq_t freq_lst[MAX_NUMBER_OF_CONC_CONNECTIONS];
|
|
|
+ enum policy_mgr_con_mode mode_lst[MAX_NUMBER_OF_CONC_CONNECTIONS];
|
|
|
+ uint8_t num_legacy_vdev;
|
|
|
+
|
|
|
+ num_legacy_vdev = policy_mgr_get_legacy_conn_info(
|
|
|
+ psoc, vdev_lst,
|
|
|
+ freq_lst, mode_lst,
|
|
|
+ QDF_ARRAY_SIZE(vdev_lst));
|
|
|
+ if (!num_legacy_vdev)
|
|
|
+ return;
|
|
|
+ /* 2 port case with 2 ml sta links or
|
|
|
+ * 2 port case with 3 ml sta links
|
|
|
+ */
|
|
|
+ if (num_legacy_vdev == 1) {
|
|
|
+ switch (mode_lst[0]) {
|
|
|
+ case PM_STA_MODE:
|
|
|
+ ml_nlink_handle_legacy_sta_intf(
|
|
|
+ psoc, vdev, force_cmd, vdev_lst[0],
|
|
|
+ freq_lst[0]);
|
|
|
+ break;
|
|
|
+ case PM_SAP_MODE:
|
|
|
+ ml_nlink_handle_legacy_sap_intf(
|
|
|
+ psoc, vdev, force_cmd, vdev_lst[0],
|
|
|
+ freq_lst[0]);
|
|
|
+ break;
|
|
|
+ case PM_P2P_CLIENT_MODE:
|
|
|
+ case PM_P2P_GO_MODE:
|
|
|
+ ml_nlink_handle_legacy_p2p_intf(
|
|
|
+ psoc, vdev, force_cmd, vdev_lst[0],
|
|
|
+ freq_lst[0]);
|
|
|
+ break;
|
|
|
+ default:
|
|
|
+ /* unexpected legacy connection count */
|
|
|
+ mlo_debug("unexpected legacy intf mode %d",
|
|
|
+ mode_lst[0]);
|
|
|
+ return;
|
|
|
+ }
|
|
|
+ ml_nlink_dump_force_state(force_cmd, "");
|
|
|
+ return;
|
|
|
+ }
|
|
|
+ /* 3 ports case with ml sta 2 or 3 links, suppose port 3 vdev is
|
|
|
+ * low latency legacy vdev:
|
|
|
+ * 6G: ML Link + Port2 + Port3 | 5G: ML Link
|
|
|
+ * => no op
|
|
|
+ * 6G: ML Link + Port2 | 5G: ML Link + Port3
|
|
|
+ * => disable 5G link if 5G mcc
|
|
|
+ * 6G: ML Link + Port3 | 5G: ML Link + Port2
|
|
|
+ * => disable 6G link if 6G mcc
|
|
|
+ * 6G: ML Link | 5G: ML Link + Port3 | 2G: Port2
|
|
|
+ * => disable 5G link if 5G mcc.
|
|
|
+ * 6G: ML Link | 5G: ML Link + Port2 | 2G: Port3
|
|
|
+ * => disable 6g link.
|
|
|
+ * 6G: ML Link + Port3 | 5G: ML Link | 2G: Port2
|
|
|
+ * => disable 6G link if 6G mcc.
|
|
|
+ * 6G: ML Link + Port2 | 5G: ML Link | 2G: Port3
|
|
|
+ * => disable 6g link.
|
|
|
+ * 6G: ML Link + Port2 + Port3 | 2G: ML Link
|
|
|
+ * => no op
|
|
|
+ * 6G: ML Link + Port2 | 2G: ML Link + Port3
|
|
|
+ * => disable 2G link if 2G mcc
|
|
|
+ * 6G: ML Link + Port3 | 2G: ML Link + Port2
|
|
|
+ * => disable 6G link if 6G mcc
|
|
|
+ * 6G: ML Link | 2G: ML Link + Port3 | 5G: Port2
|
|
|
+ * => disable 2G link if 2G mcc.
|
|
|
+ * 6G: ML Link | 2G: ML Link + Port2 | 5GL: Port3
|
|
|
+ * => disable 6G link
|
|
|
+ * 6G: ML Link + Port3 | 2G: ML Link | 5G: Port2
|
|
|
+ * => disable 6G link if 6G mcc.
|
|
|
+ * 6G: ML Link + Port2 | 2G: ML Link | 5GL: Port3
|
|
|
+ * => disable 2G link
|
|
|
+ * general rule:
|
|
|
+ * If Port3 is mcc with any link based on current hw mode, then
|
|
|
+ * force inactive the link.
|
|
|
+ * And if standby link is mcc with Port3, then disable standby
|
|
|
+ * link as well.
|
|
|
+ */
|
|
|
+ switch (mode_lst[0]) {
|
|
|
+ case PM_P2P_CLIENT_MODE:
|
|
|
+ case PM_P2P_GO_MODE:
|
|
|
+ if (!policy_mgr_is_vdev_high_tput_or_low_latency(
|
|
|
+ psoc, vdev_lst[0]))
|
|
|
+ return;
|
|
|
+ fallthrough;
|
|
|
+ case PM_STA_MODE:
|
|
|
+ ml_nlink_handle_legacy_intf_3_ports(
|
|
|
+ psoc, vdev, force_cmd, freq_lst[0], freq_lst[1]);
|
|
|
+ break;
|
|
|
+ case PM_SAP_MODE:
|
|
|
+ /* if 2g only sap present, force inactive num to fw. */
|
|
|
+ ml_nlink_handle_legacy_sap_intf(
|
|
|
+ psoc, vdev, force_cmd, vdev_lst[0], freq_lst[0]);
|
|
|
+ break;
|
|
|
+ default:
|
|
|
+ /* unexpected legacy connection count */
|
|
|
+ mlo_debug("unexpected legacy intf mode %d", mode_lst[0]);
|
|
|
+ return;
|
|
|
+ }
|
|
|
+ ml_nlink_dump_force_state(force_cmd, "");
|
|
|
+}
|
|
|
+
|
|
|
+/**
|
|
|
+ * ml_nlink_sta_inactivity_allowed_with_quiet() - Check force inactive allowed
|
|
|
+ * for links in bitmap
|
|
|
+ * @psoc: PSOC object information
|
|
|
+ * @vdev: vdev object
|
|
|
+ * @force_inactive_bitmap: force inactive link bimap
|
|
|
+ *
|
|
|
+ * If left links (exclude removed link and QUITE link) are zero, the force
|
|
|
+ * inactive bitmap is not allowed.
|
|
|
+ *
|
|
|
+ * Return: true if allow to force inactive links in force_inactive_bitmap
|
|
|
+ */
|
|
|
+static bool ml_nlink_sta_inactivity_allowed_with_quiet(
|
|
|
+ struct wlan_objmgr_psoc *psoc,
|
|
|
+ struct wlan_objmgr_vdev *vdev,
|
|
|
+ uint16_t force_inactive_bitmap)
|
|
|
+{
|
|
|
+ uint8_t ml_num_link = 0;
|
|
|
+ uint32_t ml_link_bitmap = 0;
|
|
|
+ uint8_t ml_vdev_lst[MAX_NUMBER_OF_CONC_CONNECTIONS];
|
|
|
+ qdf_freq_t ml_freq_lst[MAX_NUMBER_OF_CONC_CONNECTIONS];
|
|
|
+ uint8_t ml_linkid_lst[MAX_NUMBER_OF_CONC_CONNECTIONS];
|
|
|
+ struct ml_link_info ml_link_info[MAX_NUMBER_OF_CONC_CONNECTIONS];
|
|
|
+
|
|
|
+ ml_nlink_get_link_info(psoc, vdev, (NLINK_EXCLUDE_REMOVED_LINK |
|
|
|
+ NLINK_EXCLUDE_QUIET_LINK),
|
|
|
+ QDF_ARRAY_SIZE(ml_linkid_lst),
|
|
|
+ ml_link_info, ml_freq_lst, ml_vdev_lst,
|
|
|
+ ml_linkid_lst, &ml_num_link,
|
|
|
+ &ml_link_bitmap);
|
|
|
+ ml_link_bitmap &= ~force_inactive_bitmap;
|
|
|
+ if (!ml_link_bitmap) {
|
|
|
+ mlo_debug("not allow - no active link after force inactive 0x%x",
|
|
|
+ force_inactive_bitmap);
|
|
|
+ return false;
|
|
|
+ }
|
|
|
+
|
|
|
+ return true;
|
|
|
+}
|
|
|
+
|
|
|
+/**
|
|
|
+ * ml_nlink_allow_conc() - Check force inactive allowed for links in bitmap
|
|
|
+ * @psoc: PSOC object information
|
|
|
+ * @vdev: vdev object
|
|
|
+ * @no_forced_bitmap: no force link bitmap
|
|
|
+ * @force_inactive_bitmap: force inactive link bimap
|
|
|
+ *
|
|
|
+ * Check the no force bitmap and force inactive bitmap are allowed to send
|
|
|
+ * to firmware
|
|
|
+ *
|
|
|
+ * Return: true if allow to "no force" and force inactive links.
|
|
|
+ */
|
|
|
+static bool
|
|
|
+ml_nlink_allow_conc(struct wlan_objmgr_psoc *psoc,
|
|
|
+ struct wlan_objmgr_vdev *vdev,
|
|
|
+ uint16_t no_forced_bitmap,
|
|
|
+ uint16_t force_inactive_bitmap)
|
|
|
+{
|
|
|
+ uint8_t vdev_id_num = 0;
|
|
|
+ uint8_t vdev_ids[MAX_NUMBER_OF_CONC_CONNECTIONS];
|
|
|
+ uint32_t vdev_id_bitmap_sz;
|
|
|
+ uint32_t vdev_id_bitmap[MLO_VDEV_BITMAP_SZ];
|
|
|
+ uint32_t i;
|
|
|
+ union conc_ext_flag conc_ext_flags;
|
|
|
+ struct wlan_objmgr_vdev *ml_vdev;
|
|
|
+ bool allow = true;
|
|
|
+ qdf_freq_t freq = 0;
|
|
|
+ struct wlan_channel *bss_chan;
|
|
|
+
|
|
|
+ if (!ml_nlink_sta_inactivity_allowed_with_quiet(
|
|
|
+ psoc, vdev, force_inactive_bitmap))
|
|
|
+ return false;
|
|
|
+
|
|
|
+ ml_nlink_convert_linkid_bitmap_to_vdev_bitmap(
|
|
|
+ psoc, vdev, no_forced_bitmap, NULL, &vdev_id_bitmap_sz,
|
|
|
+ vdev_id_bitmap, &vdev_id_num, vdev_ids);
|
|
|
+
|
|
|
+ for (i = 0; i < vdev_id_num; i++) {
|
|
|
+ ml_vdev =
|
|
|
+ wlan_objmgr_get_vdev_by_id_from_psoc(psoc,
|
|
|
+ vdev_ids[i],
|
|
|
+ WLAN_MLO_MGR_ID);
|
|
|
+ if (!ml_vdev) {
|
|
|
+ mlo_err("invalid vdev id %d ", vdev_ids[i]);
|
|
|
+ continue;
|
|
|
+ }
|
|
|
+
|
|
|
+ /* If link is active, no need to check allow conc */
|
|
|
+ if (!policy_mgr_vdev_is_force_inactive(psoc, vdev_ids[i])) {
|
|
|
+ wlan_objmgr_vdev_release_ref(ml_vdev,
|
|
|
+ WLAN_MLO_MGR_ID);
|
|
|
+ continue;
|
|
|
+ }
|
|
|
+
|
|
|
+ conc_ext_flags.value =
|
|
|
+ policy_mgr_get_conc_ext_flags(ml_vdev, true);
|
|
|
+
|
|
|
+ bss_chan = wlan_vdev_mlme_get_bss_chan(ml_vdev);
|
|
|
+ if (bss_chan)
|
|
|
+ freq = bss_chan->ch_freq;
|
|
|
+
|
|
|
+ if (!policy_mgr_allow_concurrency(psoc, PM_STA_MODE,
|
|
|
+ freq,
|
|
|
+ HW_MODE_20_MHZ,
|
|
|
+ conc_ext_flags.value,
|
|
|
+ vdev_ids[i])) {
|
|
|
+ wlan_objmgr_vdev_release_ref(ml_vdev,
|
|
|
+ WLAN_MLO_MGR_ID);
|
|
|
+ break;
|
|
|
+ }
|
|
|
+ wlan_objmgr_vdev_release_ref(ml_vdev, WLAN_MLO_MGR_ID);
|
|
|
+ }
|
|
|
+
|
|
|
+ if (i < vdev_id_num) {
|
|
|
+ mlo_err("not allow - vdev %d freq %d active due to conc",
|
|
|
+ vdev_ids[i], freq);
|
|
|
+ allow = false;
|
|
|
+ }
|
|
|
+
|
|
|
+ return allow;
|
|
|
+}
|
|
|
+
|
|
|
+static QDF_STATUS
|
|
|
+ml_nlink_update_no_force_for_all(struct wlan_objmgr_psoc *psoc,
|
|
|
+ struct wlan_objmgr_vdev *vdev,
|
|
|
+ struct ml_link_force_state *curr,
|
|
|
+ struct ml_link_force_state *new,
|
|
|
+ enum mlo_link_force_reason reason)
|
|
|
+{
|
|
|
+ uint16_t no_force_links;
|
|
|
+ QDF_STATUS status = QDF_STATUS_SUCCESS;
|
|
|
+
|
|
|
+ /* Special handling for clear all force mode in target.
|
|
|
+ * send MLO_LINK_FORCE_MODE_NO_FORCE to clear "all"
|
|
|
+ * to target
|
|
|
+ */
|
|
|
+ if (!new->force_inactive_bitmap &&
|
|
|
+ !new->force_inactive_num &&
|
|
|
+ !new->force_active_bitmap &&
|
|
|
+ !new->force_active_num &&
|
|
|
+ (curr->force_inactive_bitmap ||
|
|
|
+ curr->force_inactive_num ||
|
|
|
+ curr->force_active_bitmap ||
|
|
|
+ curr->force_active_num)) {
|
|
|
+ /* If link is force inactive already, but new command will
|
|
|
+ * mark it non-force, need to check conc allow or not.
|
|
|
+ */
|
|
|
+ no_force_links = curr->force_inactive_bitmap;
|
|
|
+ /* Check non forced links allowed by conc */
|
|
|
+ if (!ml_nlink_allow_conc(psoc, vdev, no_force_links, 0)) {
|
|
|
+ status = QDF_STATUS_E_INVAL;
|
|
|
+ goto end;
|
|
|
+ }
|
|
|
+
|
|
|
+ status = policy_mgr_mlo_sta_set_nlink(
|
|
|
+ psoc, vdev, reason,
|
|
|
+ MLO_LINK_FORCE_MODE_NO_FORCE,
|
|
|
+ 0, 0, 0, 0);
|
|
|
+ }
|
|
|
+
|
|
|
+end:
|
|
|
+ return status;
|
|
|
+}
|
|
|
+
|
|
|
+static QDF_STATUS
|
|
|
+ml_nlink_update_force_inactive(struct wlan_objmgr_psoc *psoc,
|
|
|
+ struct wlan_objmgr_vdev *vdev,
|
|
|
+ struct ml_link_force_state *curr,
|
|
|
+ struct ml_link_force_state *new,
|
|
|
+ enum mlo_link_force_reason reason)
|
|
|
+{
|
|
|
+ uint16_t no_force_links;
|
|
|
+ QDF_STATUS status = QDF_STATUS_SUCCESS;
|
|
|
+
|
|
|
+ if (new->force_inactive_bitmap != curr->force_inactive_bitmap) {
|
|
|
+ /* If link is force inactive already, but new command will
|
|
|
+ * mark it non-force, need to check conc allow or not.
|
|
|
+ */
|
|
|
+ no_force_links = curr->force_inactive_bitmap &
|
|
|
+ new->force_inactive_bitmap;
|
|
|
+ no_force_links ^= curr->force_inactive_bitmap;
|
|
|
+
|
|
|
+ /* Check non forced links allowed by conc */
|
|
|
+ if (!ml_nlink_allow_conc(psoc, vdev, no_force_links,
|
|
|
+ new->force_inactive_bitmap)) {
|
|
|
+ status = QDF_STATUS_E_INVAL;
|
|
|
+ goto end;
|
|
|
+ }
|
|
|
+ status = policy_mgr_mlo_sta_set_nlink(
|
|
|
+ psoc, vdev, reason,
|
|
|
+ MLO_LINK_FORCE_MODE_INACTIVE,
|
|
|
+ 0,
|
|
|
+ new->force_inactive_bitmap,
|
|
|
+ 0, link_ctrl_f_overwrite_inactive_bitmap);
|
|
|
+ }
|
|
|
+
|
|
|
+end:
|
|
|
+ return status;
|
|
|
+}
|
|
|
+
|
|
|
+static QDF_STATUS
|
|
|
+ml_nlink_update_force_inactive_num(struct wlan_objmgr_psoc *psoc,
|
|
|
+ struct wlan_objmgr_vdev *vdev,
|
|
|
+ struct ml_link_force_state *curr,
|
|
|
+ struct ml_link_force_state *new,
|
|
|
+ enum mlo_link_force_reason reason)
|
|
|
+{
|
|
|
+ QDF_STATUS status = QDF_STATUS_SUCCESS;
|
|
|
+
|
|
|
+ if (new->force_inactive_num !=
|
|
|
+ curr->force_inactive_num ||
|
|
|
+ new->force_inactive_num_bitmap !=
|
|
|
+ curr->force_inactive_num_bitmap) {
|
|
|
+ status = policy_mgr_mlo_sta_set_nlink(
|
|
|
+ psoc, vdev, reason,
|
|
|
+ MLO_LINK_FORCE_MODE_INACTIVE_NUM,
|
|
|
+ new->force_inactive_num,
|
|
|
+ new->force_inactive_num_bitmap,
|
|
|
+ 0,
|
|
|
+ 0);
|
|
|
+ }
|
|
|
+
|
|
|
+ return status;
|
|
|
+}
|
|
|
+
|
|
|
+static QDF_STATUS
|
|
|
+ml_nlink_update_force_active(struct wlan_objmgr_psoc *psoc,
|
|
|
+ struct wlan_objmgr_vdev *vdev,
|
|
|
+ struct ml_link_force_state *curr,
|
|
|
+ struct ml_link_force_state *new,
|
|
|
+ enum mlo_link_force_reason reason)
|
|
|
+{
|
|
|
+ return QDF_STATUS_SUCCESS;
|
|
|
+}
|
|
|
+
|
|
|
+static QDF_STATUS
|
|
|
+ml_nlink_update_force_active_num(struct wlan_objmgr_psoc *psoc,
|
|
|
+ struct wlan_objmgr_vdev *vdev,
|
|
|
+ struct ml_link_force_state *curr,
|
|
|
+ struct ml_link_force_state *new,
|
|
|
+ enum mlo_link_force_reason reason)
|
|
|
+{
|
|
|
+ return QDF_STATUS_SUCCESS;
|
|
|
+}
|
|
|
+
|
|
|
+/**
|
|
|
+ * ml_nlink_state_change() - Handle ML STA link force
|
|
|
+ * with concurrency internal function
|
|
|
+ * @psoc: PSOC object information
|
|
|
+ * @reason: reason code of trigger force mode change.
|
|
|
+ *
|
|
|
+ * This API handle link force for connected ML STA.
|
|
|
+ * At present we only support one ML STA. so ml_nlink_get_affect_ml_sta
|
|
|
+ * is invoked to get one ML STA vdev from policy mgr table.
|
|
|
+ *
|
|
|
+ * The flow is to get current force command which has been sent to target
|
|
|
+ * and compute a new force command based on current connection table.
|
|
|
+ * If any difference between "current" and "new", driver sends update
|
|
|
+ * command to target. Driver will update the current force command
|
|
|
+ * record after get successful respone from target.
|
|
|
+ *
|
|
|
+ * Return: QDF_STATUS_SUCCESS if no new command updated to target.
|
|
|
+ * QDF_STATUS_E_PENDING if new command is sent to target.
|
|
|
+ * otherwise QDF_STATUS error code
|
|
|
+ */
|
|
|
+static QDF_STATUS ml_nlink_state_change(struct wlan_objmgr_psoc *psoc,
|
|
|
+ enum mlo_link_force_reason reason)
|
|
|
+{
|
|
|
+ struct ml_link_force_state force_state = {0};
|
|
|
+ struct ml_link_force_state legacy_intf_force_state = {0};
|
|
|
+ struct ml_link_force_state curr_force_state = {0};
|
|
|
+ struct wlan_objmgr_vdev *vdev = NULL;
|
|
|
+ QDF_STATUS status = QDF_STATUS_SUCCESS;
|
|
|
+
|
|
|
+ mlo_debug("enter evaluate force state change");
|
|
|
+
|
|
|
+ /*
|
|
|
+ * eMLSR is allowed in MCC mode also. So, don't disable any links
|
|
|
+ * if current connection happens in eMLSR mode.
|
|
|
+ * eMLSR is handled by wlan_handle_emlsr_sta_concurrency
|
|
|
+ */
|
|
|
+ if (policy_mgr_is_mlo_in_mode_emlsr(psoc, NULL, NULL)) {
|
|
|
+ mlo_debug("Don't disable eMLSR links");
|
|
|
+ goto end;
|
|
|
+ }
|
|
|
+
|
|
|
+ vdev = ml_nlink_get_affect_ml_sta(psoc);
|
|
|
+ if (!vdev)
|
|
|
+ goto end;
|
|
|
+ if (!mlo_check_if_all_links_up(vdev))
|
|
|
+ goto end;
|
|
|
+
|
|
|
+ ml_nlink_get_curr_force_state(psoc, vdev, &curr_force_state);
|
|
|
+
|
|
|
+ ml_nlink_handle_mcc_links(psoc, vdev, &force_state);
|
|
|
+
|
|
|
+ ml_nlink_handle_legacy_intf(psoc, vdev, &legacy_intf_force_state);
|
|
|
+
|
|
|
+ force_state.force_inactive_bitmap |=
|
|
|
+ legacy_intf_force_state.force_inactive_bitmap;
|
|
|
+
|
|
|
+ if (legacy_intf_force_state.force_inactive_num &&
|
|
|
+ legacy_intf_force_state.force_inactive_num >=
|
|
|
+ force_state.force_inactive_num) {
|
|
|
+ force_state.force_inactive_num =
|
|
|
+ legacy_intf_force_state.force_inactive_num;
|
|
|
+ force_state.force_inactive_num_bitmap =
|
|
|
+ legacy_intf_force_state.force_inactive_num_bitmap;
|
|
|
+ }
|
|
|
+
|
|
|
+ status = ml_nlink_update_no_force_for_all(psoc, vdev,
|
|
|
+ &curr_force_state,
|
|
|
+ &force_state,
|
|
|
+ reason);
|
|
|
+ if (status == QDF_STATUS_E_PENDING || status != QDF_STATUS_SUCCESS)
|
|
|
+ goto end;
|
|
|
+
|
|
|
+ status = ml_nlink_update_force_inactive(psoc, vdev,
|
|
|
+ &curr_force_state,
|
|
|
+ &force_state,
|
|
|
+ reason);
|
|
|
+ if (status == QDF_STATUS_E_PENDING || status != QDF_STATUS_SUCCESS)
|
|
|
+ goto end;
|
|
|
+
|
|
|
+ status = ml_nlink_update_force_inactive_num(psoc, vdev,
|
|
|
+ &curr_force_state,
|
|
|
+ &force_state,
|
|
|
+ reason);
|
|
|
+ if (status == QDF_STATUS_E_PENDING || status != QDF_STATUS_SUCCESS)
|
|
|
+ goto end;
|
|
|
+
|
|
|
+ /* At present, only force inactive/inactive num mode have been used
|
|
|
+ * to avoid MCC, force active/active num APIs are no-op for now.
|
|
|
+ */
|
|
|
+ status = ml_nlink_update_force_active(psoc, vdev,
|
|
|
+ &curr_force_state,
|
|
|
+ &force_state,
|
|
|
+ reason);
|
|
|
+ if (status == QDF_STATUS_E_PENDING || status != QDF_STATUS_SUCCESS)
|
|
|
+ goto end;
|
|
|
+
|
|
|
+ status = ml_nlink_update_force_active_num(psoc, vdev,
|
|
|
+ &curr_force_state,
|
|
|
+ &force_state,
|
|
|
+ reason);
|
|
|
+end:
|
|
|
+ if (vdev)
|
|
|
+ wlan_objmgr_vdev_release_ref(vdev, WLAN_MLO_MGR_ID);
|
|
|
+
|
|
|
+ if (status == QDF_STATUS_SUCCESS)
|
|
|
+ mlo_debug("exit no force state change");
|
|
|
+ else if (status == QDF_STATUS_E_PENDING)
|
|
|
+ mlo_debug("exit pending force state change");
|
|
|
+ else
|
|
|
+ mlo_err("exit err %d state change", status);
|
|
|
+
|
|
|
+ return status;
|
|
|
+}
|
|
|
+
|
|
|
+/**
|
|
|
+ * ml_nlink_state_change_handler() - Handle ML STA link force
|
|
|
+ * with concurrency
|
|
|
+ * @psoc: PSOC object information
|
|
|
+ * @vdev: ml sta vdev object
|
|
|
+ * @reason: reason code of trigger force mode change.
|
|
|
+ *
|
|
|
+ * Return: void
|
|
|
+ */
|
|
|
+static void
|
|
|
+ml_nlink_state_change_handler(struct wlan_objmgr_psoc *psoc,
|
|
|
+ struct wlan_objmgr_vdev *vdev,
|
|
|
+ enum mlo_link_force_reason reason)
|
|
|
+{
|
|
|
+ enum QDF_OPMODE mode = wlan_vdev_mlme_get_opmode(vdev);
|
|
|
+ uint8_t vdev_id = wlan_vdev_get_id(vdev);
|
|
|
+
|
|
|
+ /* If WMI_SERVICE_N_LINK_MLO_SUPPORT = 381 is enabled,
|
|
|
+ * indicate FW support N MLO link & vdev re-purpose between links,
|
|
|
+ * host will use linkid bitmap to force inactive/active links
|
|
|
+ * by API ml_nlink_state_change.
|
|
|
+ * Otherwise, use legacy policy mgr API to inactive/active based
|
|
|
+ * on vdev id bitmap.
|
|
|
+ */
|
|
|
+ if (ml_is_nlink_service_supported())
|
|
|
+ ml_nlink_state_change(psoc, reason);
|
|
|
+ else if (reason == MLO_LINK_FORCE_REASON_CONNECT)
|
|
|
+ policy_mgr_handle_ml_sta_links_on_vdev_up_csa(psoc, mode,
|
|
|
+ vdev_id);
|
|
|
+ else
|
|
|
+ policy_mgr_handle_ml_sta_links_on_vdev_down(psoc, mode,
|
|
|
+ vdev_id);
|
|
|
+}
|
|
|
+
|
|
|
+QDF_STATUS
|
|
|
+ml_nlink_conn_change_notify(struct wlan_objmgr_psoc *psoc,
|
|
|
+ uint8_t vdev_id,
|
|
|
+ enum ml_nlink_change_event_type evt,
|
|
|
+ struct ml_nlink_change_event *data)
|
|
|
+{
|
|
|
+ struct wlan_objmgr_vdev *vdev;
|
|
|
+
|
|
|
+ mlo_debug("vdev %d %s", vdev_id, link_evt_to_string(evt));
|
|
|
+ vdev = wlan_objmgr_get_vdev_by_id_from_psoc(psoc, vdev_id,
|
|
|
+ WLAN_MLO_MGR_ID);
|
|
|
+ if (!vdev) {
|
|
|
+ mlo_err("invalid vdev id %d ", vdev_id);
|
|
|
+ return QDF_STATUS_E_INVAL;
|
|
|
+ }
|
|
|
+
|
|
|
+ switch (evt) {
|
|
|
+ case ml_nlink_link_switch_start_evt:
|
|
|
+ /* todo: allow concurrenct check */
|
|
|
+ break;
|
|
|
+ case ml_nlink_link_switch_completion_evt:
|
|
|
+ ml_nlink_state_change_handler(
|
|
|
+ psoc, vdev, MLO_LINK_FORCE_REASON_CONNECT);
|
|
|
+ break;
|
|
|
+ case ml_nlink_roam_sync_start_evt:
|
|
|
+ ml_nlink_clr_force_state(psoc, vdev);
|
|
|
+ break;
|
|
|
+ case ml_nlink_roam_sync_completion_evt:
|
|
|
+ ml_nlink_state_change_handler(
|
|
|
+ psoc, vdev, MLO_LINK_FORCE_REASON_CONNECT);
|
|
|
+ break;
|
|
|
+ case ml_nlink_connect_start_evt:
|
|
|
+ ml_nlink_clr_force_state(psoc, vdev);
|
|
|
+ break;
|
|
|
+ case ml_nlink_connect_completion_evt:
|
|
|
+ ml_nlink_state_change_handler(
|
|
|
+ psoc, vdev, MLO_LINK_FORCE_REASON_CONNECT);
|
|
|
+ break;
|
|
|
+ case ml_nlink_disconnect_start_evt:
|
|
|
+ ml_nlink_clr_force_state(psoc, vdev);
|
|
|
+ break;
|
|
|
+ case ml_nlink_disconnect_completion_evt:
|
|
|
+ ml_nlink_state_change_handler(
|
|
|
+ psoc, vdev, MLO_LINK_FORCE_REASON_DISCONNECT);
|
|
|
+ break;
|
|
|
+ case ml_nlink_ap_started_evt:
|
|
|
+ ml_nlink_state_change_handler(
|
|
|
+ psoc, vdev, MLO_LINK_FORCE_REASON_CONNECT);
|
|
|
+ break;
|
|
|
+ case ml_nlink_ap_stopped_evt:
|
|
|
+ ml_nlink_state_change_handler(
|
|
|
+ psoc, vdev, MLO_LINK_FORCE_REASON_DISCONNECT);
|
|
|
+ break;
|
|
|
+ case ml_nlink_connection_updated_evt:
|
|
|
+ ml_nlink_state_change_handler(
|
|
|
+ psoc, vdev, MLO_LINK_FORCE_REASON_CONNECT);
|
|
|
+ break;
|
|
|
+ default:
|
|
|
+ break;
|
|
|
+ }
|
|
|
+
|
|
|
+ if (vdev)
|
|
|
+ wlan_objmgr_vdev_release_ref(vdev, WLAN_MLO_MGR_ID);
|
|
|
+
|
|
|
+ return QDF_STATUS_SUCCESS;
|
|
|
+}
|