|
@@ -2610,6 +2610,50 @@ static void hdd_restore_sme_config(struct wlan_hdd_link_info *link_info,
|
|
|
}
|
|
|
}
|
|
|
|
|
|
+/**
|
|
|
+ * wlan_update_mlo_link_chn_width() - API to update mlo link chn width
|
|
|
+ * @adapter: the pointer to adapter
|
|
|
+ * @ch_width: channel width to update
|
|
|
+ * @link_id: mlo link id
|
|
|
+ *
|
|
|
+ * Get link id and channel bandwidth from user space and save in link_info.
|
|
|
+ * When link switch happen and host driver connect done, if the link change
|
|
|
+ * from standby to non-standby, ch_width will send to fw again.
|
|
|
+ *
|
|
|
+ * Return: QDF_STATUS
|
|
|
+ */
|
|
|
+
|
|
|
+#if defined(WLAN_FEATURE_11BE_MLO) && defined(WLAN_HDD_MULTI_VDEV_SINGLE_NDEV)
|
|
|
+static struct wlan_hdd_link_info *
|
|
|
+wlan_update_mlo_link_chn_width(struct hdd_adapter *adapter,
|
|
|
+ enum phy_ch_width ch_width,
|
|
|
+ uint8_t link_id)
|
|
|
+{
|
|
|
+ struct wlan_hdd_link_info *link_info;
|
|
|
+ struct hdd_station_ctx *sta_ctx;
|
|
|
+
|
|
|
+ link_info = hdd_get_link_info_by_ieee_link_id(adapter, link_id);
|
|
|
+ if (!link_info)
|
|
|
+ return NULL;
|
|
|
+
|
|
|
+ sta_ctx = WLAN_HDD_GET_STATION_CTX_PTR(link_info);
|
|
|
+
|
|
|
+ sta_ctx->user_cfg_chn_width = ch_width;
|
|
|
+ hdd_debug("save ch_width:%u to link_id:%u vdev_id:%u",
|
|
|
+ ch_width, link_id, link_info->vdev_id);
|
|
|
+
|
|
|
+ return link_info;
|
|
|
+}
|
|
|
+#else
|
|
|
+static struct wlan_hdd_link_info *
|
|
|
+wlan_update_mlo_link_chn_width(struct hdd_adapter *adapter,
|
|
|
+ enum phy_ch_width ch_width,
|
|
|
+ uint8_t link_id)
|
|
|
+{
|
|
|
+ return NULL;
|
|
|
+}
|
|
|
+#endif
|
|
|
+
|
|
|
int hdd_update_channel_width(struct wlan_hdd_link_info *link_info,
|
|
|
enum eSirMacHTChannelWidth chwidth,
|
|
|
uint32_t bonding_mode, uint8_t link_id,
|
|
@@ -2617,11 +2661,10 @@ int hdd_update_channel_width(struct wlan_hdd_link_info *link_info,
|
|
|
{
|
|
|
struct hdd_context *hdd_ctx;
|
|
|
int ret;
|
|
|
- enum phy_ch_width ch_width = CH_WIDTH_INVALID;
|
|
|
+ enum phy_ch_width ch_width;
|
|
|
struct wlan_objmgr_vdev *link_vdev;
|
|
|
struct wlan_objmgr_vdev *vdev;
|
|
|
struct wlan_hdd_link_info *link_info_t;
|
|
|
- bool is_mlo_link = false;
|
|
|
uint8_t link_vdev_id;
|
|
|
enum QDF_OPMODE op_mode;
|
|
|
QDF_STATUS status;
|
|
@@ -2634,23 +2677,42 @@ int hdd_update_channel_width(struct wlan_hdd_link_info *link_info,
|
|
|
return -EINVAL;
|
|
|
}
|
|
|
|
|
|
- vdev = link_info->vdev;
|
|
|
+ op_mode = link_info->adapter->device_mode;
|
|
|
+ if (op_mode != QDF_STA_MODE) {
|
|
|
+ hdd_debug("vdev %d: op mode %d, CW update not supported",
|
|
|
+ vdev_id, op_mode);
|
|
|
+ return -EINVAL;
|
|
|
+ }
|
|
|
+
|
|
|
+ vdev = hdd_objmgr_get_vdev_by_user(link_info, WLAN_OSIF_ID);
|
|
|
if (!vdev) {
|
|
|
- mlme_legacy_err("vdev %d: vdev not found", vdev_id);
|
|
|
- return QDF_STATUS_E_FAILURE;
|
|
|
+ hdd_err("vdev %d: vdev not found", vdev_id);
|
|
|
+ return -EINVAL;
|
|
|
}
|
|
|
|
|
|
- if (wlan_vdev_mlme_is_mlo_vdev(vdev) && link_id != 0xFF) {
|
|
|
- link_vdev = mlo_get_vdev_by_link_id(vdev, link_id,
|
|
|
- WLAN_MLME_OBJMGR_ID);
|
|
|
- if (!link_vdev) {
|
|
|
- mlme_legacy_debug("vdev is null for the link id:%u",
|
|
|
- link_id);
|
|
|
- wlan_objmgr_vdev_release_ref(vdev, WLAN_MLME_OBJMGR_ID);
|
|
|
- return QDF_STATUS_E_FAILURE;
|
|
|
+ ch_width = hdd_convert_chwidth_to_phy_chwidth(chwidth);
|
|
|
+
|
|
|
+ /**
|
|
|
+ * Link_id check is for disconnect restore process.
|
|
|
+ * Disconnect will not update channel bandwidth into cache struct.
|
|
|
+ */
|
|
|
+ if (wlan_vdev_mlme_is_mlo_vdev(vdev) &&
|
|
|
+ link_id != WLAN_INVALID_LINK_ID) {
|
|
|
+ link_info_t = wlan_update_mlo_link_chn_width(link_info->adapter,
|
|
|
+ ch_width, link_id);
|
|
|
+ if (!link_info_t) {
|
|
|
+ hdd_objmgr_put_vdev_by_user(vdev, WLAN_OSIF_ID);
|
|
|
+ return -EINVAL;
|
|
|
}
|
|
|
- is_mlo_link = true;
|
|
|
- link_vdev_id = wlan_vdev_get_id(link_vdev);
|
|
|
+
|
|
|
+ hdd_objmgr_put_vdev_by_user(vdev, WLAN_OSIF_ID);
|
|
|
+
|
|
|
+ link_vdev = hdd_objmgr_get_vdev_by_user(link_info_t,
|
|
|
+ WLAN_OSIF_ID);
|
|
|
+ if (!link_vdev)
|
|
|
+ return 0;
|
|
|
+
|
|
|
+ link_vdev_id = link_info_t->vdev_id;
|
|
|
status = wlan_mlme_get_bw_no_punct(hdd_ctx->psoc,
|
|
|
link_vdev,
|
|
|
wlan_vdev_mlme_get_des_chan(link_vdev),
|
|
@@ -2660,43 +2722,31 @@ int hdd_update_channel_width(struct wlan_hdd_link_info *link_info,
|
|
|
} else {
|
|
|
link_vdev = vdev;
|
|
|
link_vdev_id = vdev_id;
|
|
|
- mlme_legacy_debug("vdev mlme is not mlo vdev");
|
|
|
- }
|
|
|
-
|
|
|
- link_info_t = hdd_get_link_info_by_vdev(hdd_ctx, link_vdev_id);
|
|
|
- if (!link_info_t) {
|
|
|
- mlme_legacy_debug("vdev mlme is not mlo vdev");
|
|
|
- return QDF_STATUS_E_FAILURE;
|
|
|
+ link_info_t = link_info;
|
|
|
}
|
|
|
|
|
|
if (ucfg_mlme_is_chwidth_with_notify_supported(hdd_ctx->psoc) &&
|
|
|
hdd_cm_is_vdev_connected(link_info_t)) {
|
|
|
- op_mode = wlan_vdev_mlme_get_opmode(vdev);
|
|
|
- if (op_mode != QDF_STA_MODE) {
|
|
|
- mlme_legacy_debug("vdev %d: op mode %d, CW update not supported",
|
|
|
- link_vdev_id, op_mode);
|
|
|
- wlan_objmgr_vdev_release_ref(vdev, WLAN_MLME_OBJMGR_ID);
|
|
|
- return QDF_STATUS_E_NOSUPPORT;
|
|
|
- }
|
|
|
-
|
|
|
ch_width = hdd_convert_chwidth_to_phy_chwidth(chwidth);
|
|
|
hdd_debug("vdev %d : process update ch width request to %d",
|
|
|
link_vdev_id, ch_width);
|
|
|
-
|
|
|
status = ucfg_mlme_send_ch_width_update_with_notify(hdd_ctx->psoc,
|
|
|
link_vdev,
|
|
|
ch_width,
|
|
|
link_vdev_id);
|
|
|
- if (is_mlo_link)
|
|
|
- wlan_objmgr_vdev_release_ref(link_vdev,
|
|
|
- WLAN_MLME_OBJMGR_ID);
|
|
|
- if (QDF_IS_STATUS_ERROR(status))
|
|
|
+
|
|
|
+ if (QDF_IS_STATUS_ERROR(status)) {
|
|
|
+ hdd_objmgr_put_vdev_by_user(link_vdev, WLAN_OSIF_ID);
|
|
|
return -EIO;
|
|
|
+ }
|
|
|
status = hdd_update_bss_rate_flags(link_info_t, hdd_ctx->psoc,
|
|
|
ch_width);
|
|
|
- if (QDF_IS_STATUS_ERROR(status))
|
|
|
+ if (QDF_IS_STATUS_ERROR(status)) {
|
|
|
+ hdd_objmgr_put_vdev_by_user(link_vdev, WLAN_OSIF_ID);
|
|
|
return -EIO;
|
|
|
+ }
|
|
|
}
|
|
|
+ hdd_objmgr_put_vdev_by_user(link_vdev, WLAN_OSIF_ID);
|
|
|
|
|
|
ret = wma_cli_set_command(link_vdev_id, wmi_vdev_param_chwidth,
|
|
|
chwidth, VDEV_CMD);
|