|
@@ -1318,7 +1318,146 @@ QDF_STATUS wma_parse_bw_indication_ie(uint8_t *ie,
|
|
|
}
|
|
|
#endif
|
|
|
|
|
|
+static bool fill_csa_offload_params(
|
|
|
+ wmi_csa_event_fixed_param *csa_event,
|
|
|
+ struct csa_offload_params *csa_offload_event,
|
|
|
+ struct wlan_objmgr_pdev *pdev)
|
|
|
+{
|
|
|
+ struct ieee80211_channelswitch_ie *csa_ie;
|
|
|
+ struct ieee80211_extendedchannelswitch_ie *xcsa_ie;
|
|
|
+ uint8_t is_csa_ie_present = false;
|
|
|
+
|
|
|
+ if (csa_event->ies_present_flag & WMI_CSA_IE_PRESENT) {
|
|
|
+ csa_ie = (struct ieee80211_channelswitch_ie *)
|
|
|
+ (&csa_event->csa_ie[0]);
|
|
|
+ csa_offload_event->channel = csa_ie->newchannel;
|
|
|
+ csa_offload_event->csa_chan_freq =
|
|
|
+ wlan_reg_legacy_chan_to_freq(pdev,
|
|
|
+ csa_ie->newchannel);
|
|
|
+ csa_offload_event->switch_mode = csa_ie->switchmode;
|
|
|
+ csa_offload_event->ies_present_flag |= MLME_CSA_IE_PRESENT;
|
|
|
+ is_csa_ie_present = true;
|
|
|
+ } else if (csa_event->ies_present_flag & WMI_XCSA_IE_PRESENT) {
|
|
|
+ xcsa_ie = (struct ieee80211_extendedchannelswitch_ie *)
|
|
|
+ (&csa_event->xcsa_ie[0]);
|
|
|
+ csa_offload_event->channel = xcsa_ie->newchannel;
|
|
|
+ csa_offload_event->switch_mode = xcsa_ie->switchmode;
|
|
|
+ csa_offload_event->new_op_class = xcsa_ie->newClass;
|
|
|
+ if (wlan_reg_is_6ghz_op_class(pdev, xcsa_ie->newClass)) {
|
|
|
+ csa_offload_event->csa_chan_freq =
|
|
|
+ wlan_reg_chan_band_to_freq
|
|
|
+ (pdev, xcsa_ie->newchannel,
|
|
|
+ BIT(REG_BAND_6G));
|
|
|
+ } else {
|
|
|
+ csa_offload_event->csa_chan_freq =
|
|
|
+ wlan_reg_legacy_chan_to_freq
|
|
|
+ (pdev, xcsa_ie->newchannel);
|
|
|
+ }
|
|
|
+ csa_offload_event->ies_present_flag |= MLME_XCSA_IE_PRESENT;
|
|
|
+ is_csa_ie_present = true;
|
|
|
+ }
|
|
|
+ return is_csa_ie_present;
|
|
|
+}
|
|
|
+
|
|
|
#ifdef WLAN_FEATURE_11BE
|
|
|
+static bool handle_csa_standby_link(wmi_csa_event_fixed_param *csa_event,
|
|
|
+ struct wlan_objmgr_psoc *psoc,
|
|
|
+ struct wlan_objmgr_pdev *pdev)
|
|
|
+{
|
|
|
+ struct mlo_link_info *link_info;
|
|
|
+ struct wlan_mlo_dev_context *mldev;
|
|
|
+ uint8_t mld_addr[QDF_MAC_ADDR_SIZE];
|
|
|
+ struct csa_offload_params csa_param = {0};
|
|
|
+ struct mlo_link_bss_params params = {0};
|
|
|
+ uint8_t is_csa_standby = false;
|
|
|
+ uint8_t link_id;
|
|
|
+ struct wlan_lmac_if_mlo_tx_ops *mlo_tx_ops;
|
|
|
+ QDF_STATUS status = QDF_STATUS_SUCCESS;
|
|
|
+
|
|
|
+ if (!psoc) {
|
|
|
+ wma_err("null psoc");
|
|
|
+ return is_csa_standby;
|
|
|
+ }
|
|
|
+
|
|
|
+ WMI_MAC_ADDR_TO_CHAR_ARRAY(&csa_event->mld_mac_address,
|
|
|
+ &mld_addr[0]);
|
|
|
+ wlan_mlo_get_mlpeer_by_peer_mladdr(
|
|
|
+ (struct qdf_mac_addr *)&mld_addr[0],
|
|
|
+ &mldev);
|
|
|
+ if (!mldev) {
|
|
|
+ wma_err("NULL ml dev ctx");
|
|
|
+ return is_csa_standby;
|
|
|
+ }
|
|
|
+
|
|
|
+ link_id = csa_event->link_id;
|
|
|
+ link_info = mlo_mgr_get_ap_link_by_link_id(mldev,
|
|
|
+ link_id);
|
|
|
+ if (!link_info) {
|
|
|
+ wma_err("NULL link info ");
|
|
|
+ return is_csa_standby;
|
|
|
+ }
|
|
|
+
|
|
|
+ if (link_info->vdev_id != WLAN_INVALID_VDEV_ID) {
|
|
|
+ wma_debug("vdev id %d link id %d ", link_info->vdev_id,
|
|
|
+ link_id);
|
|
|
+ return is_csa_standby;
|
|
|
+ }
|
|
|
+
|
|
|
+ mlo_tx_ops = &psoc->soc_cb.tx_ops->mlo_ops;
|
|
|
+ if (!mlo_tx_ops) {
|
|
|
+ wma_err("tx_ops is null!");
|
|
|
+ return is_csa_standby;
|
|
|
+ }
|
|
|
+
|
|
|
+ if (!fill_csa_offload_params(csa_event, &csa_param, pdev)) {
|
|
|
+ wma_err("CSA Event error: No CSA IE present");
|
|
|
+ return is_csa_standby;
|
|
|
+ }
|
|
|
+
|
|
|
+ mlo_mgr_update_csa_link_info(mldev, &csa_param, link_id);
|
|
|
+
|
|
|
+ params.link_id = link_info->link_id;
|
|
|
+ params.chan = qdf_mem_malloc(sizeof(struct wlan_channel));
|
|
|
+ if (!params.chan) {
|
|
|
+ wma_err("no mem acquired");
|
|
|
+ return is_csa_standby;
|
|
|
+ }
|
|
|
+
|
|
|
+ qdf_copy_macaddr((struct qdf_mac_addr *)¶ms.ap_mld_mac[0],
|
|
|
+ &link_info->ap_link_addr);
|
|
|
+
|
|
|
+ params.chan->ch_freq = link_info->link_chan_info->ch_freq;
|
|
|
+ params.chan->ch_cfreq1 = link_info->link_chan_info->ch_cfreq1;
|
|
|
+ params.chan->ch_cfreq2 = link_info->link_chan_info->ch_cfreq2;
|
|
|
+ params.chan->ch_phymode = link_info->link_chan_info->ch_phymode;
|
|
|
+
|
|
|
+ mlo_debug("link id %d chan freq %d cfreq1 %d cfreq2 %d host phymode %d ap mld mac " QDF_MAC_ADDR_FMT,
|
|
|
+ link_info->link_id, link_info->link_chan_info->ch_freq,
|
|
|
+ link_info->link_chan_info->ch_cfreq1,
|
|
|
+ link_info->link_chan_info->ch_cfreq2,
|
|
|
+ link_info->link_chan_info->ch_phymode,
|
|
|
+ QDF_MAC_ADDR_REF(¶ms.ap_mld_mac[0]));
|
|
|
+
|
|
|
+ if (!mlo_tx_ops->send_link_set_bss_params_cmd) {
|
|
|
+ wma_err("handler is not registered");
|
|
|
+ qdf_mem_free(params.chan);
|
|
|
+ return is_csa_standby;
|
|
|
+ }
|
|
|
+
|
|
|
+ status = mlo_tx_ops->send_link_set_bss_params_cmd(psoc, ¶ms);
|
|
|
+
|
|
|
+ if (QDF_IS_STATUS_ERROR(status)) {
|
|
|
+ wma_err("failed to send link set bss request command to FW");
|
|
|
+ qdf_mem_free(params.chan);
|
|
|
+ return is_csa_standby;
|
|
|
+ }
|
|
|
+
|
|
|
+ is_csa_standby = true;
|
|
|
+ qdf_mem_free(params.chan);
|
|
|
+
|
|
|
+ return is_csa_standby;
|
|
|
+}
|
|
|
+
|
|
|
static int fill_peer_mac_addr(wmi_csa_event_fixed_param *csa_event,
|
|
|
uint8_t *bssid)
|
|
|
{
|
|
@@ -1366,6 +1505,14 @@ static int fill_peer_mac_addr(wmi_csa_event_fixed_param *csa_event,
|
|
|
{
|
|
|
return 0;
|
|
|
}
|
|
|
+
|
|
|
+static bool handle_csa_standby_link(wmi_csa_event_fixed_param *csa_event,
|
|
|
+ struct wlan_objmgr_psoc *psoc,
|
|
|
+ struct wlan_objmgr_pdev *pdev)
|
|
|
+{
|
|
|
+ return false;
|
|
|
+}
|
|
|
+
|
|
|
#endif
|
|
|
|
|
|
/**
|
|
@@ -1386,9 +1533,7 @@ int wma_csa_offload_handler(void *handle, uint8_t *event, uint32_t len)
|
|
|
uint8_t bssid[QDF_MAC_ADDR_SIZE];
|
|
|
uint8_t vdev_id = 0;
|
|
|
uint8_t cur_chan = 0;
|
|
|
- struct ieee80211_channelswitch_ie *csa_ie;
|
|
|
struct csa_offload_params *csa_offload_event;
|
|
|
- struct ieee80211_extendedchannelswitch_ie *xcsa_ie;
|
|
|
struct ieee80211_ie_wide_bw_switch *wb_ie;
|
|
|
struct wlan_objmgr_peer *peer;
|
|
|
struct wlan_objmgr_vdev *vdev;
|
|
@@ -1410,7 +1555,10 @@ int wma_csa_offload_handler(void *handle, uint8_t *event, uint32_t len)
|
|
|
status = fill_peer_mac_addr(csa_event, &bssid[0]);
|
|
|
if (status)
|
|
|
return -EINVAL;
|
|
|
- /* check standby link and return zero */
|
|
|
+
|
|
|
+ /* check standby link and return */
|
|
|
+ if (handle_csa_standby_link(csa_event, wma->psoc, wma->pdev))
|
|
|
+ return 0;
|
|
|
} else {
|
|
|
WMI_MAC_ADDR_TO_CHAR_ARRAY(&csa_event->i_addr2,
|
|
|
&bssid[0]);
|
|
@@ -1448,33 +1596,7 @@ int wma_csa_offload_handler(void *handle, uint8_t *event, uint32_t len)
|
|
|
qdf_copy_macaddr(&csa_offload_event->bssid,
|
|
|
(struct qdf_mac_addr *)bssid);
|
|
|
|
|
|
- if (csa_event->ies_present_flag & WMI_CSA_IE_PRESENT) {
|
|
|
- csa_ie = (struct ieee80211_channelswitch_ie *)
|
|
|
- (&csa_event->csa_ie[0]);
|
|
|
- csa_offload_event->channel = csa_ie->newchannel;
|
|
|
- csa_offload_event->csa_chan_freq =
|
|
|
- wlan_reg_legacy_chan_to_freq(wma->pdev,
|
|
|
- csa_ie->newchannel);
|
|
|
- csa_offload_event->switch_mode = csa_ie->switchmode;
|
|
|
- csa_offload_event->ies_present_flag |= MLME_CSA_IE_PRESENT;
|
|
|
- } else if (csa_event->ies_present_flag & WMI_XCSA_IE_PRESENT) {
|
|
|
- xcsa_ie = (struct ieee80211_extendedchannelswitch_ie *)
|
|
|
- (&csa_event->xcsa_ie[0]);
|
|
|
- csa_offload_event->channel = xcsa_ie->newchannel;
|
|
|
- csa_offload_event->switch_mode = xcsa_ie->switchmode;
|
|
|
- csa_offload_event->new_op_class = xcsa_ie->newClass;
|
|
|
- if (wlan_reg_is_6ghz_op_class(wma->pdev, xcsa_ie->newClass)) {
|
|
|
- csa_offload_event->csa_chan_freq =
|
|
|
- wlan_reg_chan_band_to_freq
|
|
|
- (wma->pdev, xcsa_ie->newchannel,
|
|
|
- BIT(REG_BAND_6G));
|
|
|
- } else {
|
|
|
- csa_offload_event->csa_chan_freq =
|
|
|
- wlan_reg_legacy_chan_to_freq
|
|
|
- (wma->pdev, xcsa_ie->newchannel);
|
|
|
- }
|
|
|
- csa_offload_event->ies_present_flag |= MLME_XCSA_IE_PRESENT;
|
|
|
- } else {
|
|
|
+ if (!fill_csa_offload_params(csa_event, csa_offload_event, wma->pdev)) {
|
|
|
wma_err("CSA Event error: No CSA IE present");
|
|
|
qdf_mem_free(csa_offload_event);
|
|
|
wlan_objmgr_vdev_release_ref(vdev, WLAN_LEGACY_WMA_ID);
|