qcacld-3.0: Handle vendor LTE unsafe channel ranges

In 3rd party platform, CNSS driver will provide
LTE avoidance channel frequency ranges by API
cnss_utils_get_wlan_unsafe_channel_sap.
Based on requirement, in single SAP case or SAP+SAP
case, ACS channel list should be filtered out based
on vendor unsafe channel frequency ranges.

Change-Id: I583c1bb2583c783858c54e8643fbe1af69d492b1
CRs-Fixed: 3061043
This commit is contained in:
Liangwei Dong
2021-10-18 17:18:12 +08:00
committed by Madan Koyyalamudi
parent 59a7e5f195
commit 6e61b6bb8c
5 changed files with 255 additions and 45 deletions

View File

@@ -3041,6 +3041,73 @@ static void wlan_hdd_handle_zero_acs_list(struct hdd_context *hdd_ctx,
hdd_debug("retore acs chan list to single freq %d", acs_chan_default);
}
/**
* wlan_hdd_handle_single_ch_in_acs_list() - Handle acs list with single channel
* @hdd_ctx: hdd context
* @adapter: adapter
* @sap_config: sap acs config context
*
* If only one acs channel is left after filter, driver will return the channel
* to hostapd without ACS scan.
*
* Return: None
*/
static void
wlan_hdd_handle_single_ch_in_acs_list(struct hdd_context *hdd_ctx,
struct hdd_adapter *adapter,
struct sap_config *sap_config)
{
uint32_t channel_bonding_mode_2g;
ucfg_mlme_get_channel_bonding_24ghz(hdd_ctx->psoc,
&channel_bonding_mode_2g);
sap_config->acs_cfg.start_ch_freq =
sap_config->acs_cfg.freq_list[0];
sap_config->acs_cfg.end_ch_freq =
sap_config->acs_cfg.freq_list[0];
sap_config->acs_cfg.pri_ch_freq =
sap_config->acs_cfg.freq_list[0];
if (sap_config->acs_cfg.pri_ch_freq <=
WLAN_REG_CH_TO_FREQ(CHAN_ENUM_2484) &&
sap_config->acs_cfg.ch_width >=
CH_WIDTH_40MHZ &&
!channel_bonding_mode_2g) {
sap_config->acs_cfg.ch_width = CH_WIDTH_20MHZ;
hdd_debug("2.4ghz channel resetting BW to %d 2.4 cbmode %d",
sap_config->acs_cfg.ch_width,
channel_bonding_mode_2g);
}
wlan_sap_set_sap_ctx_acs_cfg(
WLAN_HDD_GET_SAP_CTX_PTR(adapter), sap_config);
sap_config_acs_result(hdd_ctx->mac_handle,
WLAN_HDD_GET_SAP_CTX_PTR(adapter),
sap_config->acs_cfg.ht_sec_ch_freq);
sap_config->ch_params.ch_width =
sap_config->acs_cfg.ch_width;
sap_config->ch_params.sec_ch_offset =
wlan_reg_freq_to_chan(
hdd_ctx->pdev,
sap_config->acs_cfg.ht_sec_ch_freq);
sap_config->ch_params.center_freq_seg0 =
wlan_reg_freq_to_chan(
hdd_ctx->pdev,
sap_config->acs_cfg.vht_seg0_center_ch_freq);
sap_config->ch_params.center_freq_seg1 =
wlan_reg_freq_to_chan(
hdd_ctx->pdev,
sap_config->acs_cfg.vht_seg1_center_ch_freq);
sap_config->ch_params.mhz_freq_seg0 =
sap_config->acs_cfg.vht_seg0_center_ch_freq;
sap_config->ch_params.mhz_freq_seg1 =
sap_config->acs_cfg.vht_seg1_center_ch_freq;
/*notify hostapd about channel override */
wlan_hdd_cfg80211_acs_ch_select_evt(adapter);
wlansap_dcs_set_wlan_interference_mitigation_on_band(
WLAN_HDD_GET_SAP_CTX_PTR(adapter),
sap_config);
}
#if defined(WLAN_FEATURE_11BE) && defined(CFG80211_11BE_BASIC)
static void wlan_hdd_set_sap_acs_ch_width_320(struct sap_config *sap_config)
{
@@ -3110,6 +3177,7 @@ static int __wlan_hdd_cfg80211_do_acs(struct wiphy *wiphy,
QDF_STATUS qdf_status;
bool is_vendor_acs_support = false;
bool is_external_acs_policy = false;
bool is_vendor_unsafe_ch_present = false;
bool sap_force_11n_for_11ac = 0;
bool go_force_11n_for_11ac = 0;
bool go_11ac_override = 0;
@@ -3334,13 +3402,19 @@ static int __wlan_hdd_cfg80211_do_acs(struct wiphy *wiphy,
if (is_external_acs_policy &&
policy_mgr_is_force_scc(hdd_ctx->psoc) &&
policy_mgr_get_connection_count(hdd_ctx->psoc)) {
if (adapter->device_mode == QDF_SAP_MODE)
is_vendor_unsafe_ch_present =
wlansap_filter_vendor_unsafe_ch_freq(
WLAN_HDD_GET_SAP_CTX_PTR(adapter),
sap_config);
wlan_hdd_trim_acs_channel_list(
sap_config->acs_cfg.pcl_chan_freq,
sap_config->acs_cfg.pcl_ch_count,
sap_config->acs_cfg.freq_list,
&sap_config->acs_cfg.ch_list_count);
if (!sap_config->acs_cfg.ch_list_count &&
sap_config->acs_cfg.master_ch_list_count)
sap_config->acs_cfg.master_ch_list_count &&
!is_vendor_unsafe_ch_present)
wlan_hdd_handle_zero_acs_list(
hdd_ctx,
sap_config->acs_cfg.freq_list,
@@ -3349,50 +3423,8 @@ static int __wlan_hdd_cfg80211_do_acs(struct wiphy *wiphy,
sap_config->acs_cfg.master_ch_list_count);
/* if it is only one channel, send ACS event to upper layer */
if (sap_config->acs_cfg.ch_list_count == 1) {
sap_config->acs_cfg.start_ch_freq =
sap_config->acs_cfg.freq_list[0];
sap_config->acs_cfg.end_ch_freq =
sap_config->acs_cfg.freq_list[0];
sap_config->acs_cfg.pri_ch_freq =
sap_config->acs_cfg.freq_list[0];
if (sap_config->acs_cfg.pri_ch_freq <=
WLAN_REG_CH_TO_FREQ(CHAN_ENUM_2484) &&
sap_config->acs_cfg.ch_width >=
CH_WIDTH_40MHZ &&
!channel_bonding_mode_2g) {
sap_config->acs_cfg.ch_width = CH_WIDTH_20MHZ;
hdd_debug("2.4ghz channel resetting BW to %d 2.4 cbmode %d",
sap_config->acs_cfg.ch_width,
channel_bonding_mode_2g);
}
wlan_sap_set_sap_ctx_acs_cfg(
WLAN_HDD_GET_SAP_CTX_PTR(adapter), sap_config);
sap_config_acs_result(hdd_ctx->mac_handle,
WLAN_HDD_GET_SAP_CTX_PTR(adapter),
sap_config->acs_cfg.ht_sec_ch_freq);
sap_config->ch_params.ch_width =
sap_config->acs_cfg.ch_width;
sap_config->ch_params.sec_ch_offset =
wlan_reg_freq_to_chan(hdd_ctx->pdev,
sap_config->acs_cfg.ht_sec_ch_freq);
sap_config->ch_params.center_freq_seg0 =
wlan_reg_freq_to_chan(
hdd_ctx->pdev,
sap_config->acs_cfg.vht_seg0_center_ch_freq);
sap_config->ch_params.center_freq_seg1 =
wlan_reg_freq_to_chan(
hdd_ctx->pdev,
sap_config->acs_cfg.vht_seg1_center_ch_freq);
sap_config->ch_params.mhz_freq_seg0 =
sap_config->acs_cfg.vht_seg0_center_ch_freq;
sap_config->ch_params.mhz_freq_seg1 =
sap_config->acs_cfg.vht_seg1_center_ch_freq;
/*notify hostapd about channel override */
wlan_hdd_cfg80211_acs_ch_select_evt(adapter);
wlansap_dcs_set_wlan_interference_mitigation_on_band(
WLAN_HDD_GET_SAP_CTX_PTR(adapter),
sap_config);
wlan_hdd_handle_single_ch_in_acs_list(
hdd_ctx, adapter, sap_config);
ret = 0;
goto out;
} else if (!sap_config->acs_cfg.ch_list_count) {
@@ -3400,6 +3432,20 @@ static int __wlan_hdd_cfg80211_do_acs(struct wiphy *wiphy,
ret = -EINVAL;
goto out;
}
} else if (adapter->device_mode == QDF_SAP_MODE) {
wlansap_filter_vendor_unsafe_ch_freq(
WLAN_HDD_GET_SAP_CTX_PTR(adapter),
sap_config);
if (sap_config->acs_cfg.ch_list_count == 1) {
wlan_hdd_handle_single_ch_in_acs_list(
hdd_ctx, adapter, sap_config);
ret = 0;
goto out;
} else if (!sap_config->acs_cfg.ch_list_count) {
hdd_err("channel count 0 after vendor unsafe filter");
ret = -EINVAL;
goto out;
}
}
ret = wlan_hdd_set_acs_ch_range(sap_config, hw_mode,

View File

@@ -417,6 +417,30 @@ enum pld_wlan_time_sync_trigger_type {
};
#endif /* FEATURE_WLAN_TIME_SYNC_FTM */
/* MAX channel avoid ranges supported in PLD */
#define PLD_CH_AVOID_MAX_RANGE 4
/**
* struct pld_ch_avoid_freq_type
* @start_freq: start freq (MHz)
* @end_freq: end freq (Mhz)
*/
struct pld_ch_avoid_freq_type {
uint32_t start_freq;
uint32_t end_freq;
};
/**
* struct pld_ch_avoid_ind_type
* @ch_avoid_range_cnt: count
* @avoid_freq_range: avoid freq range array
*/
struct pld_ch_avoid_ind_type {
uint32_t ch_avoid_range_cnt;
struct pld_ch_avoid_freq_type
avoid_freq_range[PLD_CH_AVOID_MAX_RANGE];
};
/**
* struct pld_driver_ops - driver callback functions
* @probe: required operation, will be called when device is detected
@@ -569,6 +593,28 @@ int pld_get_audio_wlan_timestamp(struct device *dev,
#endif /* FEATURE_WLAN_TIME_SYNC_FTM */
#if IS_ENABLED(CONFIG_CNSS_UTILS)
#ifdef CNSS_UTILS_VENDOR_UNSAFE_CHAN_API_SUPPORT
/**
* pld_get_wlan_unsafe_channel_sap() - Get vendor unsafe ch freq ranges
* @dev: device
* @ch_avoid_ranges: unsafe freq channel ranges
*
* Get vendor specific unsafe channel frequency ranges
*
* Return: 0 for success
* Non zero failure code for errors
*/
int pld_get_wlan_unsafe_channel_sap(
struct device *dev, struct pld_ch_avoid_ind_type *ch_avoid_ranges);
#else
static inline
int pld_get_wlan_unsafe_channel_sap(
struct device *dev, struct pld_ch_avoid_ind_type *ch_avoid_ranges)
{
return 0;
}
#endif
/**
* pld_set_wlan_unsafe_channel() - Set unsafe channel
* @dev: device
@@ -694,6 +740,12 @@ static inline int pld_get_driver_load_cnt(struct device *dev)
return cnss_utils_get_driver_load_cnt(dev);
}
#else
static inline int pld_get_wlan_unsafe_channel_sap(
struct device *dev, struct pld_ch_avoid_ind_type *ch_avoid_ranges)
{
return 0;
}
static inline int pld_set_wlan_unsafe_channel(struct device *dev,
u16 *unsafe_ch_list,
u16 ch_count)

View File

@@ -1,5 +1,6 @@
/*
* Copyright (c) 2016-2021 The Linux Foundation. All rights reserved.
* Copyright (c) 2021 Qualcomm Innovation Center, Inc. All rights reserved.
*
* Permission to use, copy, modify, and/or distribute this software for
* any purpose with or without fee is hereby granted, provided that the
@@ -3240,6 +3241,38 @@ int pld_get_thermal_state(struct device *dev, unsigned long *thermal_state,
return errno;
}
#ifdef CNSS_UTILS_VENDOR_UNSAFE_CHAN_API_SUPPORT
int pld_get_wlan_unsafe_channel_sap(
struct device *dev, struct pld_ch_avoid_ind_type *ch_avoid_ranges)
{
struct cnss_ch_avoid_ind_type cnss_ch_avoid;
int ret;
int i;
if (!ch_avoid_ranges)
return -EINVAL;
cnss_ch_avoid.ch_avoid_range_cnt = 0;
ret = cnss_utils_get_wlan_unsafe_channel_sap(dev, &cnss_ch_avoid);
if (ret)
return ret;
for (i = 0;
i < PLD_CH_AVOID_MAX_RANGE &&
i < cnss_ch_avoid.ch_avoid_range_cnt; i++) {
ch_avoid_ranges->avoid_freq_range[i].start_freq =
cnss_ch_avoid.avoid_freq_range[i].start_freq;
ch_avoid_ranges->avoid_freq_range[i].end_freq =
cnss_ch_avoid.avoid_freq_range[i].end_freq;
}
ch_avoid_ranges->ch_avoid_range_cnt = i;
if (i < cnss_ch_avoid.ch_avoid_range_cnt)
pr_err("unexpected cnss ch_avoid_range_cnt %d",
cnss_ch_avoid.ch_avoid_range_cnt);
return 0;
}
#endif
#ifdef FEATURE_WLAN_TIME_SYNC_FTM
/**
* pld_get_audio_wlan_timestamp() - Get audio timestamp

View File

@@ -1,5 +1,6 @@
/*
* Copyright (c) 2012-2021 The Linux Foundation. All rights reserved.
* Copyright (c) 2021 Qualcomm Innovation Center, Inc. All rights reserved.
*
* Permission to use, copy, modify, and/or distribute this software for
* any purpose with or without fee is hereby granted, provided that the
@@ -1699,6 +1700,20 @@ static inline qdf_freq_t wlansap_dcs_get_freq(struct sap_context *sap_context)
}
#endif
/**
* wlansap_filter_vendor_unsafe_ch_freq() - filter sap acs ch list by
* vendor unsafe ch freq ranges
* @sap_context: sap context
* @sap_config: sap conifg
*
* This function is used to filter out unsafe channel frequency from acs
* channel frequency list based on vendor unsafe channel frequency ranges.
*
* Return: true if vendor unsafe ch range is present, otherwise false
*/
bool wlansap_filter_vendor_unsafe_ch_freq(
struct sap_context *sap_context, struct sap_config *sap_config);
/**
* wlansap_dump_acs_ch_freq() - print acs channel frequency
* @sap_ctx: sap context

View File

@@ -56,6 +56,7 @@
#include "cfg_ucfg_api.h"
#include "wlan_mlme_ucfg_api.h"
#include "wlan_mlme_vdev_mgr_interface.h"
#include "pld_common.h"
#define SAP_DEBUG
static struct sap_context *gp_sap_ctx[SAP_MAX_NUM_SESSION];
@@ -3266,6 +3267,69 @@ qdf_freq_t wlansap_get_chan_band_restrict(struct sap_context *sap_ctx,
return restart_freq;
}
static inline bool
wlansap_ch_in_avoid_ranges(uint32_t ch_freq,
struct pld_ch_avoid_ind_type *ch_avoid_ranges)
{
uint32_t i;
for (i = 0; i < ch_avoid_ranges->ch_avoid_range_cnt; i++) {
if (ch_freq >=
ch_avoid_ranges->avoid_freq_range[i].start_freq &&
ch_freq <=
ch_avoid_ranges->avoid_freq_range[i].end_freq)
return true;
}
return false;
}
bool wlansap_filter_vendor_unsafe_ch_freq(
struct sap_context *sap_context, struct sap_config *sap_config)
{
struct pld_ch_avoid_ind_type ch_avoid_ranges;
uint32_t i, j;
int ret;
qdf_device_t qdf_ctx = cds_get_context(QDF_MODULE_ID_QDF_DEVICE);
struct mac_context *mac;
uint32_t count;
if (!qdf_ctx)
return false;
mac = sap_get_mac_context();
if (!mac)
return false;
count = policy_mgr_mode_specific_connection_count(mac->psoc,
PM_SAP_MODE, NULL);
if (count != policy_mgr_get_connection_count(mac->psoc))
return false;
ch_avoid_ranges.ch_avoid_range_cnt = 0;
ret = pld_get_wlan_unsafe_channel_sap(qdf_ctx->dev, &ch_avoid_ranges);
if (ret) {
sap_debug("failed to get vendor unsafe ch range, ret %d", ret);
return false;
}
if (!ch_avoid_ranges.ch_avoid_range_cnt)
return false;
for (i = 0; i < ch_avoid_ranges.ch_avoid_range_cnt; i++) {
sap_debug("vendor unsafe range[%d] %d %d", i,
ch_avoid_ranges.avoid_freq_range[i].start_freq,
ch_avoid_ranges.avoid_freq_range[i].end_freq);
}
for (i = 0, j = 0; i < sap_config->acs_cfg.ch_list_count; i++) {
if (!wlansap_ch_in_avoid_ranges(
sap_config->acs_cfg.freq_list[i],
&ch_avoid_ranges))
sap_config->acs_cfg.freq_list[j++] =
sap_config->acs_cfg.freq_list[i];
}
sap_config->acs_cfg.ch_list_count = j;
return true;
}
#ifdef DCS_INTERFERENCE_DETECTION
QDF_STATUS wlansap_dcs_set_vdev_wlan_interference_mitigation(
struct sap_context *sap_context,