|
@@ -1621,21 +1621,123 @@ target_if_start_rso_stop_timer(struct wlan_objmgr_vdev *vdev)
|
|
|
RSO_STOP_RESPONSE_BIT);
|
|
|
}
|
|
|
|
|
|
+static bool
|
|
|
+target_if_is_vdev_rsp_valid(struct wlan_objmgr_psoc *psoc,
|
|
|
+ struct vdev_response_timer **vdev_rsp,
|
|
|
+ struct wlan_lmac_if_mlme_rx_ops *rx_ops,
|
|
|
+ uint8_t vdev_id)
|
|
|
+{
|
|
|
+ *vdev_rsp = rx_ops->psoc_get_vdev_response_timer_info(psoc, vdev_id);
|
|
|
+ if (!*vdev_rsp) {
|
|
|
+ mlme_err("MLO_ROAM: vdev rsp not found for vdev:%d", vdev_id);
|
|
|
+ return false;
|
|
|
+ }
|
|
|
+
|
|
|
+ if (qdf_atomic_test_bit(RSO_STOP_RESPONSE_BIT,
|
|
|
+ &((*vdev_rsp)->rsp_status))) {
|
|
|
+ mlme_debug("MLO_ROAM: RSO bit set on vdev id %d",
|
|
|
+ (*vdev_rsp)->vdev_id);
|
|
|
+ return true;
|
|
|
+ }
|
|
|
+
|
|
|
+ /* Failure case vdev_rsp is set to NULL */
|
|
|
+ *vdev_rsp = NULL;
|
|
|
+
|
|
|
+ return false;
|
|
|
+}
|
|
|
+
|
|
|
+/**
|
|
|
+ * target_if_find_active_rso_stop_rsp() - Iterate through ml vdevs to find
|
|
|
+ * the vdev rsp for which RSO_STOP_RESPONSE_BIT is set.
|
|
|
+ * @roam_event: Roam event data
|
|
|
+ *
|
|
|
+ * This is needed when for e.g.: host sends rso stop on vdev id 0, fw response
|
|
|
+ * is received on vdev 1.
|
|
|
+ * Since the timer is vdev specific, this function will iterate through ml vdevs
|
|
|
+ * to find the vdev_rsp on which RSO_STOP_RESPONSE_BIT is set.
|
|
|
+ *
|
|
|
+ * Return: struct vdev_response_timer for success, NULL for failure
|
|
|
+ */
|
|
|
+static struct vdev_response_timer *
|
|
|
+target_if_find_active_rso_stop_rsp(struct roam_offload_roam_event *roam_event)
|
|
|
+{
|
|
|
+ struct vdev_response_timer *vdev_rsp = NULL;
|
|
|
+ struct wlan_lmac_if_mlme_rx_ops *rx_ops;
|
|
|
+ struct wlan_objmgr_vdev *vdev;
|
|
|
+ struct wlan_mlo_dev_context *mlo_dev_ctx;
|
|
|
+ uint8_t i;
|
|
|
+ QDF_STATUS status;
|
|
|
+
|
|
|
+ rx_ops = target_if_vdev_mgr_get_rx_ops(roam_event->psoc);
|
|
|
+ if (!rx_ops || !rx_ops->psoc_get_vdev_response_timer_info) {
|
|
|
+ mlme_err("No Rx Ops");
|
|
|
+ return NULL;
|
|
|
+ }
|
|
|
+
|
|
|
+ vdev = wlan_objmgr_get_vdev_by_id_from_psoc(roam_event->psoc,
|
|
|
+ roam_event->vdev_id,
|
|
|
+ WLAN_MLME_SB_ID);
|
|
|
+ if (!vdev)
|
|
|
+ return NULL;
|
|
|
+
|
|
|
+ /* For legacy case use the incoming vdev */
|
|
|
+ if (target_if_is_vdev_rsp_valid(roam_event->psoc, &vdev_rsp,
|
|
|
+ rx_ops, roam_event->vdev_id))
|
|
|
+ goto end;
|
|
|
+
|
|
|
+ mlo_dev_ctx = vdev->mlo_dev_ctx;
|
|
|
+ if (!mlo_dev_ctx)
|
|
|
+ goto end;
|
|
|
+
|
|
|
+ /*
|
|
|
+ * if vdev_rsp with RSO_STOP_RESPONSE bit is not set then check for
|
|
|
+ * the same on other ML vdevs
|
|
|
+ */
|
|
|
+ for (i = 0; i < WLAN_UMAC_MLO_MAX_VDEVS; i++) {
|
|
|
+ if (!mlo_dev_ctx->wlan_vdev_list[i])
|
|
|
+ continue;
|
|
|
+
|
|
|
+ status = wlan_objmgr_vdev_try_get_ref(mlo_dev_ctx->wlan_vdev_list[i],
|
|
|
+ WLAN_MLO_MGR_ID);
|
|
|
+ if (QDF_IS_STATUS_ERROR(status))
|
|
|
+ continue;
|
|
|
+
|
|
|
+ if (target_if_is_vdev_rsp_valid(roam_event->psoc, &vdev_rsp,
|
|
|
+ rx_ops,
|
|
|
+ wlan_vdev_get_id(mlo_dev_ctx->wlan_vdev_list[i]))) {
|
|
|
+ mlo_release_vdev_ref(mlo_dev_ctx->wlan_vdev_list[i]);
|
|
|
+ goto end;
|
|
|
+ }
|
|
|
+
|
|
|
+ mlo_release_vdev_ref(mlo_dev_ctx->wlan_vdev_list[i]);
|
|
|
+ }
|
|
|
+
|
|
|
+ if (i == WLAN_UMAC_MLO_MAX_VDEVS) {
|
|
|
+ mlme_err("RSO bit not set on any mlo vdev");
|
|
|
+ goto end;
|
|
|
+ }
|
|
|
+
|
|
|
+end:
|
|
|
+ if (vdev)
|
|
|
+ wlan_objmgr_vdev_release_ref(vdev, WLAN_MLME_SB_ID);
|
|
|
+
|
|
|
+ return vdev_rsp;
|
|
|
+}
|
|
|
+
|
|
|
QDF_STATUS
|
|
|
target_if_stop_rso_stop_timer(struct roam_offload_roam_event *roam_event)
|
|
|
{
|
|
|
QDF_STATUS status = QDF_STATUS_SUCCESS;
|
|
|
struct vdev_response_timer *vdev_rsp;
|
|
|
- struct wlan_lmac_if_mlme_rx_ops *rx_ops;
|
|
|
|
|
|
roam_event->rso_timer_stopped = false;
|
|
|
- rx_ops = target_if_vdev_mgr_get_rx_ops(roam_event->psoc);
|
|
|
- if (!rx_ops || !rx_ops->vdev_mgr_start_response) {
|
|
|
- mlme_err("No Rx Ops");
|
|
|
- return QDF_STATUS_E_INVAL;
|
|
|
+ if (roam_event->reason == ROAM_REASON_RSO_STATUS &&
|
|
|
+ roam_event->notif == CM_ROAM_NOTIF_HO_FAIL) {
|
|
|
+ mlme_debug("HO_FAIL happened, wait for HO_FAIL event vdev_id: %u",
|
|
|
+ roam_event->vdev_id);
|
|
|
}
|
|
|
- vdev_rsp = rx_ops->psoc_get_vdev_response_timer_info(roam_event->psoc,
|
|
|
- roam_event->vdev_id);
|
|
|
+
|
|
|
+ vdev_rsp = target_if_find_active_rso_stop_rsp(roam_event);
|
|
|
if (!vdev_rsp) {
|
|
|
mlme_err("vdev response timer is null VDEV_%d PSOC_%d",
|
|
|
roam_event->vdev_id,
|
|
@@ -1643,33 +1745,35 @@ target_if_stop_rso_stop_timer(struct roam_offload_roam_event *roam_event)
|
|
|
return QDF_STATUS_E_INVAL;
|
|
|
}
|
|
|
|
|
|
- if (!qdf_atomic_test_bit(RSO_STOP_RESPONSE_BIT,
|
|
|
- &vdev_rsp->rsp_status)) {
|
|
|
- mlme_debug("rso stop timer is not started");
|
|
|
- return QDF_STATUS_SUCCESS;
|
|
|
- }
|
|
|
+ switch (roam_event->reason) {
|
|
|
+ case ROAM_REASON_RSO_STATUS:
|
|
|
+ if (roam_event->notif != CM_ROAM_NOTIF_SCAN_MODE_SUCCESS &&
|
|
|
+ roam_event->notif != CM_ROAM_NOTIF_SCAN_MODE_FAIL)
|
|
|
+ break;
|
|
|
|
|
|
- if ((roam_event->reason == ROAM_REASON_RSO_STATUS &&
|
|
|
- (roam_event->notif == CM_ROAM_NOTIF_SCAN_MODE_SUCCESS ||
|
|
|
- roam_event->notif == CM_ROAM_NOTIF_SCAN_MODE_FAIL)) ||
|
|
|
- roam_event->reason == ROAM_REASON_HO_FAILED) {
|
|
|
+ /*
|
|
|
+ * fallthrough if notif == CM_ROAM_NOTIF_SCAN_MODE_SUCCESS or
|
|
|
+ * notif == CM_ROAM_NOTIF_SCAN_MODE_FAIL
|
|
|
+ */
|
|
|
+ fallthrough;
|
|
|
+ case ROAM_REASON_HO_FAILED:
|
|
|
status = target_if_vdev_mgr_rsp_timer_stop(roam_event->psoc,
|
|
|
- vdev_rsp, RSO_STOP_RESPONSE_BIT);
|
|
|
- if (QDF_IS_STATUS_SUCCESS(status))
|
|
|
- roam_event->rso_timer_stopped = true;
|
|
|
- else
|
|
|
- mlme_err("PSOC_%d VDEV_%d: VDE MGR RSP Timer stop failed",
|
|
|
+ vdev_rsp,
|
|
|
+ RSO_STOP_RESPONSE_BIT);
|
|
|
+ roam_event->rso_timer_stopped = true;
|
|
|
+ if (QDF_IS_STATUS_ERROR(status)) {
|
|
|
+ roam_event->rso_timer_stopped = false;
|
|
|
+ mlme_err("PSOC_%d VDEV_%d: VDEV MGR RSO Stop RSP Timer stop failed",
|
|
|
roam_event->psoc->soc_objmgr.psoc_id,
|
|
|
roam_event->vdev_id);
|
|
|
- } else if (roam_event->reason == ROAM_REASON_RSO_STATUS &&
|
|
|
- roam_event->notif == CM_ROAM_NOTIF_HO_FAIL) {
|
|
|
- mlme_debug("HO_FAIL happened, wait for HO_FAIL event vdev_id: %u",
|
|
|
- roam_event->vdev_id);
|
|
|
+ }
|
|
|
+ break;
|
|
|
+ default:
|
|
|
+ return status;
|
|
|
}
|
|
|
|
|
|
return status;
|
|
|
}
|
|
|
-
|
|
|
#else
|
|
|
static inline QDF_STATUS
|
|
|
target_if_start_rso_stop_timer(struct wlan_objmgr_vdev *vdev)
|