|
@@ -25,15 +25,61 @@
|
|
|
#include "../dfs_process_radar_found_ind.h"
|
|
|
#include <wlan_reg_services_api.h>
|
|
|
#include <wlan_dfs_utils_api.h>
|
|
|
+#include "wlan_dfs_mlme_api.h"
|
|
|
|
|
|
/**
|
|
|
- * A) If is chirp or radar found at boundary, two channels will effected.
|
|
|
- * freq_offset.freq[0] = fn+1
|
|
|
- * freq_offset.freq[1] = fn+1
|
|
|
- * freq_offset.freq[2] = fn
|
|
|
+ * TODO: The code is not according to the following description needs
|
|
|
+ * modification and correction. Code always adds left and right channels to
|
|
|
+ * NOL even if it is not a chirp radar.
|
|
|
*
|
|
|
- * Two channels, ch(n) and ch(n+1) will be added to nol.
|
|
|
+ * A) If chirp radar starts at boundary and ends at boundary then three channels
|
|
|
+ * will be affected.
|
|
|
+ * freq_offset.freq[0] = fn (Center frequency)
|
|
|
+ * freq_offset.freq[1] = fn-1 (Left of center)
|
|
|
+ * freq_offset.freq[2] = fn+1 (Right of center)
|
|
|
*
|
|
|
+ * Three channels, ch(n-1), ch(n)and ch(n+1) will be added to NOL.
|
|
|
+ *
|
|
|
+ * Chirp start freq Chirp end freq
|
|
|
+ * | |
|
|
|
+ * | |
|
|
|
+ * V V
|
|
|
+ * _______________________________________________________________________
|
|
|
+ * | center freq | center freq | center freq |
|
|
|
+ * | ch(n-1) | ch(n) | ch(n+1) |
|
|
|
+ * | | | | | | |
|
|
|
+ * | | | | | | |
|
|
|
+ * | | | | | | |
|
|
|
+ * fn-1 fn boundary fn+1
|
|
|
+ * <-------- 20 Mhz ------>
|
|
|
+ *
|
|
|
+ * B) If chirp radar starts at one channel and continues up to another channel
|
|
|
+ * then two channels will be affected.
|
|
|
+ * freq_offset.freq[0] = fn
|
|
|
+ * freq_offset.freq[1] = 0
|
|
|
+ * freq_offset.freq[2] = fn+1
|
|
|
+ *
|
|
|
+ * Three channels, ch(n-1), ch(n)and ch(n+1) will be added to NOL.
|
|
|
+ *
|
|
|
+ * Chirp start freq Chirp end freq
|
|
|
+ * | |
|
|
|
+ * | |
|
|
|
+ * V V
|
|
|
+ * _______________________________________________________________________
|
|
|
+ * | center freq | center freq | center freq |
|
|
|
+ * | ch(n-1) | ch(n) | ch(n+1) |
|
|
|
+ * | | | | | | |
|
|
|
+ * | | | | | | |
|
|
|
+ * | | | | | | |
|
|
|
+ * fn-1 fn boundary fn+1
|
|
|
+ * <-------- 20 Mhz ------>
|
|
|
+ *
|
|
|
+ * C) Radar found at boundary, two channels will be affected.
|
|
|
+ * freq_offset.freq[0] = fn
|
|
|
+ * freq_offset.freq[1] = 0
|
|
|
+ * freq_offset.freq[2] = fn+1
|
|
|
+ *
|
|
|
+ * Two channels, ch(n) and ch(n+1) will be added to NOL.
|
|
|
*
|
|
|
* dfs_freq_offset (radar found freq)
|
|
|
* |
|
|
@@ -49,12 +95,12 @@
|
|
|
* <-------- 20 Mhz ------>
|
|
|
*
|
|
|
*
|
|
|
- * B) Else only one channel will be effected
|
|
|
+ * D) Else only one channel will be affected.
|
|
|
* freq_offset.freq[0] = fn
|
|
|
- * freq_offset.freq[1] = fn
|
|
|
- * freq_offset.freq[2] = fn
|
|
|
+ * freq_offset.freq[1] = 0
|
|
|
+ * freq_offset.freq[2] = 0
|
|
|
*
|
|
|
- * One channel ch(n) will be added to nol.
|
|
|
+ * One channel ch(n) will be added to NOL.
|
|
|
*
|
|
|
*
|
|
|
* dfs_freq_offset (radar found freq)
|
|
@@ -104,7 +150,7 @@ static void dfs_radar_add_to_nol(struct wlan_dfs *dfs,
|
|
|
dfs->wlan_dfs_nol_timeout);
|
|
|
nollist[num_ch++] = last_chan;
|
|
|
dfs_info(dfs, WLAN_DEBUG_DFS_ALWAYS,
|
|
|
- "ch=%d Added to NOL", last_chan);
|
|
|
+ "ch = %d Added to NOL", last_chan);
|
|
|
}
|
|
|
utils_dfs_reg_update_nol_ch(dfs->dfs_pdev_obj,
|
|
|
nollist, num_ch, DFS_NOL_SET);
|
|
@@ -211,78 +257,73 @@ static void dfs_radar_chan_for_20(struct freqs_offsets *freq_offset,
|
|
|
}
|
|
|
}
|
|
|
|
|
|
-void dfs_process_radar_found_indication(struct wlan_dfs *dfs,
|
|
|
- struct radar_found_info *radar_found)
|
|
|
+/**
|
|
|
+ * dfs_find_radar_affected_subchans() - Finds radar affected sub channels.
|
|
|
+ * @dfs: Pointer to wlan_dfs structure.
|
|
|
+ * @radar_found: Pointer to radar_found structure.
|
|
|
+ * @freq_offset: Pointer to save radar affected channels.
|
|
|
+ */
|
|
|
+static void dfs_find_radar_affected_subchans(
|
|
|
+ struct wlan_dfs *dfs,
|
|
|
+ struct radar_found_info *radar_found,
|
|
|
+ struct freqs_offsets *freq_offset)
|
|
|
{
|
|
|
int i;
|
|
|
- struct freqs_offsets freq_offset;
|
|
|
uint32_t freq_center, flag;
|
|
|
int32_t sidx;
|
|
|
+ struct dfs_ieee80211_channel *curchan = dfs->dfs_curchan;
|
|
|
|
|
|
- if (!dfs || !dfs->dfs_curchan) {
|
|
|
- dfs_alert(dfs, WLAN_DEBUG_DFS_ALWAYS, "dfs is null");
|
|
|
- return;
|
|
|
- }
|
|
|
-
|
|
|
- qdf_mem_set(&freq_offset, sizeof(freq_offset), 0);
|
|
|
- flag = dfs->dfs_curchan->dfs_ch_flags;
|
|
|
+ qdf_mem_set(freq_offset, sizeof(*freq_offset), 0);
|
|
|
+ flag = curchan->dfs_ch_flags;
|
|
|
|
|
|
for (i = 0; i < DFS_NUM_FREQ_OFFSET; i++)
|
|
|
- freq_offset.offset[i] = radar_found->freq_offset;
|
|
|
+ freq_offset->offset[i] = radar_found->freq_offset;
|
|
|
|
|
|
sidx = DFS_FREQ_OFFSET_TO_SIDX(radar_found->freq_offset);
|
|
|
|
|
|
if (!radar_found->segment_id)
|
|
|
freq_center = utils_dfs_chan_to_freq(
|
|
|
- dfs->dfs_curchan->dfs_ch_vhtop_ch_freq_seg1);
|
|
|
+ curchan->dfs_ch_vhtop_ch_freq_seg1);
|
|
|
else {
|
|
|
- freq_center = utils_dfs_chan_to_freq(
|
|
|
- dfs->dfs_curchan->dfs_ch_vhtop_ch_freq_seg2);
|
|
|
- if (flag & IEEE80211_CHAN_VHT160)
|
|
|
- freq_center += DFS_160MHZ_SECOND_SEG_OFFSET;
|
|
|
+ if (dfs_is_precac_timer_running(dfs)) {
|
|
|
+ freq_center = utils_dfs_chan_to_freq(
|
|
|
+ dfs->dfs_precac_secondary_freq);
|
|
|
+ } else {
|
|
|
+ freq_center = utils_dfs_chan_to_freq(
|
|
|
+ curchan->dfs_ch_vhtop_ch_freq_seg2);
|
|
|
+ if (flag & IEEE80211_CHAN_VHT160)
|
|
|
+ freq_center += DFS_160MHZ_SECOND_SEG_OFFSET;
|
|
|
+ }
|
|
|
}
|
|
|
|
|
|
- dfs_info(dfs, WLAN_DEBUG_DFS_ALWAYS,
|
|
|
+ dfs_info(dfs, WLAN_DEBUG_DFS,
|
|
|
"seg=%d, sidx=%d, offset=%d, chirp=%d, flag=%d, f=%d",
|
|
|
radar_found->segment_id, sidx,
|
|
|
radar_found->freq_offset, radar_found->is_chirp,
|
|
|
flag, freq_center);
|
|
|
|
|
|
- if ((flag & IEEE80211_CHAN_A) ||
|
|
|
- (flag & IEEE80211_CHAN_11NA_HT20) ||
|
|
|
- (flag & IEEE80211_CHAN_HT20) ||
|
|
|
- (flag & IEEE80211_CHAN_VHT20) ||
|
|
|
- (flag & IEEE80211_CHAN_11AXA_HE20)) {
|
|
|
+ if ((IEEE80211_IS_CHAN_A(curchan)) ||
|
|
|
+ IEEE80211_IS_CHAN_MODE_20(curchan)) {
|
|
|
if (radar_found->is_chirp ||
|
|
|
(sidx && !(abs(sidx) % DFS_BOUNDRY_SIDX))) {
|
|
|
- freq_offset.offset[1] -= DFS_CHIRP_OFFSET;
|
|
|
- freq_offset.offset[2] += DFS_CHIRP_OFFSET;
|
|
|
+ freq_offset->offset[LEFT_CH] -= DFS_CHIRP_OFFSET;
|
|
|
+ freq_offset->offset[RIGHT_CH] += DFS_CHIRP_OFFSET;
|
|
|
}
|
|
|
- dfs_radar_chan_for_20(&freq_offset, freq_center);
|
|
|
- } else if ((flag & IEEE80211_CHAN_VHT40PLUS) ||
|
|
|
- (flag & IEEE80211_CHAN_VHT40MINUS) ||
|
|
|
- (flag & IEEE80211_CHAN_HT40PLUS) ||
|
|
|
- (flag & IEEE80211_CHAN_HT40MINUS) ||
|
|
|
- (flag & IEEE80211_CHAN_11NA_HT40PLUS) ||
|
|
|
- (flag & IEEE80211_CHAN_11NA_HT40MINUS) ||
|
|
|
- (flag & IEEE80211_CHAN_11AXA_HE40PLUS) ||
|
|
|
- (flag & IEEE80211_CHAN_11AXA_HE40MINUS)) {
|
|
|
+ dfs_radar_chan_for_20(freq_offset, freq_center);
|
|
|
+ } else if (IEEE80211_IS_CHAN_MODE_40(curchan)) {
|
|
|
if (radar_found->is_chirp || !(abs(sidx) % DFS_BOUNDRY_SIDX)) {
|
|
|
- freq_offset.offset[1] -= DFS_CHIRP_OFFSET;
|
|
|
- freq_offset.offset[2] += DFS_CHIRP_OFFSET;
|
|
|
+ freq_offset->offset[LEFT_CH] -= DFS_CHIRP_OFFSET;
|
|
|
+ freq_offset->offset[RIGHT_CH] += DFS_CHIRP_OFFSET;
|
|
|
}
|
|
|
- dfs_radar_chan_for_40(&freq_offset, freq_center);
|
|
|
- } else if ((flag & IEEE80211_CHAN_VHT80) ||
|
|
|
- (flag & IEEE80211_CHAN_VHT80_80) ||
|
|
|
- (flag & IEEE80211_CHAN_VHT160) ||
|
|
|
- (flag & IEEE80211_CHAN_11AXA_HE80) ||
|
|
|
- (flag & IEEE80211_CHAN_11AXA_HE80_80) ||
|
|
|
- (flag & IEEE80211_CHAN_11AXA_HE160)) {
|
|
|
+ dfs_radar_chan_for_40(freq_offset, freq_center);
|
|
|
+ } else if (IEEE80211_IS_CHAN_MODE_80(curchan) ||
|
|
|
+ IEEE80211_IS_CHAN_MODE_160(curchan) ||
|
|
|
+ IEEE80211_IS_CHAN_MODE_80_80(curchan)) {
|
|
|
if (radar_found->is_chirp || !(abs(sidx) % DFS_BOUNDRY_SIDX)) {
|
|
|
- freq_offset.offset[1] -= DFS_CHIRP_OFFSET;
|
|
|
- freq_offset.offset[2] += DFS_CHIRP_OFFSET;
|
|
|
+ freq_offset->offset[LEFT_CH] -= DFS_CHIRP_OFFSET;
|
|
|
+ freq_offset->offset[RIGHT_CH] += DFS_CHIRP_OFFSET;
|
|
|
}
|
|
|
- dfs_radar_chan_for_80(&freq_offset, freq_center);
|
|
|
+ dfs_radar_chan_for_80(freq_offset, freq_center);
|
|
|
} else {
|
|
|
dfs_err(dfs, WLAN_DEBUG_DFS_ALWAYS,
|
|
|
"channel flag(%d) is invalid", flag);
|
|
@@ -290,11 +331,100 @@ void dfs_process_radar_found_indication(struct wlan_dfs *dfs,
|
|
|
}
|
|
|
|
|
|
for (i = 0; i < DFS_NUM_FREQ_OFFSET; i++) {
|
|
|
- freq_offset.chan_num[i] = utils_dfs_freq_to_chan(
|
|
|
- freq_offset.freq[i]);
|
|
|
- dfs_info(dfs, WLAN_DEBUG_DFS_ALWAYS, "offset=%d, channel%d",
|
|
|
- i, freq_offset.chan_num[i]);
|
|
|
+ freq_offset->chan_num[i] = utils_dfs_freq_to_chan(
|
|
|
+ freq_offset->freq[i]);
|
|
|
+ dfs_info(dfs, WLAN_DEBUG_DFS_ALWAYS, "offset = %d, channel = %d",
|
|
|
+ i, freq_offset->chan_num[i]);
|
|
|
+ }
|
|
|
+}
|
|
|
+
|
|
|
+QDF_STATUS dfs_process_radar_ind(struct wlan_dfs *dfs,
|
|
|
+ struct radar_found_info *radar_found)
|
|
|
+{
|
|
|
+ struct freqs_offsets freq_offset;
|
|
|
+ bool wait_for_csa = false;
|
|
|
+
|
|
|
+ if (!dfs->dfs_curchan) {
|
|
|
+ dfs_err(dfs, WLAN_DEBUG_DFS_ALWAYS, "dfs->dfs_curchan is NULL");
|
|
|
+ return QDF_STATUS_E_FAILURE;
|
|
|
+ }
|
|
|
+
|
|
|
+ /* Check if the current channel is a non DFS channel */
|
|
|
+ if (!dfs_radarevent_basic_sanity(dfs, dfs->dfs_curchan)) {
|
|
|
+ dfs_err(dfs, WLAN_DEBUG_DFS_ALWAYS,
|
|
|
+ "radar event on a non-DFS channel");
|
|
|
+ return QDF_STATUS_E_FAILURE;
|
|
|
+ }
|
|
|
+
|
|
|
+ if (!dfs->dfs_use_nol) {
|
|
|
+ dfs_send_csa_to_current_chan(dfs);
|
|
|
+ return QDF_STATUS_SUCCESS;
|
|
|
}
|
|
|
|
|
|
+ dfs_find_radar_affected_subchans(dfs, radar_found, &freq_offset);
|
|
|
+
|
|
|
dfs_radar_add_to_nol(dfs, &freq_offset);
|
|
|
+
|
|
|
+ /*
|
|
|
+ * If precac is running and the radar found in secondary
|
|
|
+ * VHT80 mark the channel as radar and add to NOL list.
|
|
|
+ * Otherwise random channel selection can choose this
|
|
|
+ * channel.
|
|
|
+ */
|
|
|
+ dfs_debug(dfs, WLAN_DEBUG_DFS,
|
|
|
+ "found_on_second = %d is_pre = %d",
|
|
|
+ dfs->is_radar_found_on_secondary_seg,
|
|
|
+ dfs_is_precac_timer_running(dfs));
|
|
|
+
|
|
|
+ /*
|
|
|
+ * Even if radar found on primary, we need to move the channel
|
|
|
+ * from precac-required-list and precac-done-list to
|
|
|
+ * precac-nol-list.
|
|
|
+ */
|
|
|
+ dfs_mark_precac_dfs(dfs,
|
|
|
+ dfs->is_radar_found_on_secondary_seg);
|
|
|
+
|
|
|
+ if (!dfs->dfs_is_offload_enabled) {
|
|
|
+ if (dfs->is_radar_found_on_secondary_seg) {
|
|
|
+ dfs_second_segment_radar_disable(dfs);
|
|
|
+ dfs->is_radar_found_on_secondary_seg = 0;
|
|
|
+ if (dfs->is_radar_during_precac) {
|
|
|
+ dfs->is_radar_during_precac = 0;
|
|
|
+ return QDF_STATUS_SUCCESS;
|
|
|
+ }
|
|
|
+ }
|
|
|
+ }
|
|
|
+
|
|
|
+ /*
|
|
|
+ * This calls into the umac DFS code, which sets the umac
|
|
|
+ * related radar flags and begins the channel change
|
|
|
+ * machinery.
|
|
|
+ * XXX TODO: the umac NOL code isn't used, but
|
|
|
+ * IEEE80211_CHAN_DFS_RADAR still gets set. Since the umac
|
|
|
+ * NOL code isn't used, that flag is never cleared. This
|
|
|
+ * needs to be fixed. See EV 105776.
|
|
|
+ */
|
|
|
+ dfs_mlme_start_rcsa(dfs->dfs_pdev_obj,
|
|
|
+ &wait_for_csa);
|
|
|
+ if (wait_for_csa)
|
|
|
+ return QDF_STATUS_SUCCESS;
|
|
|
+
|
|
|
+ dfs_mlme_mark_dfs(dfs->dfs_pdev_obj,
|
|
|
+ dfs->dfs_curchan->dfs_ch_ieee,
|
|
|
+ dfs->dfs_curchan->dfs_ch_freq,
|
|
|
+ dfs->dfs_curchan->dfs_ch_vhtop_ch_freq_seg2,
|
|
|
+ dfs->dfs_curchan->dfs_ch_flags);
|
|
|
+ /*
|
|
|
+ * EV 129487 : We have detected radar in the channel,
|
|
|
+ * stop processing PHY error data as this can cause
|
|
|
+ * false detect in the new channel while channel
|
|
|
+ * change is in progress.
|
|
|
+ */
|
|
|
+
|
|
|
+ if (!dfs->dfs_is_offload_enabled) {
|
|
|
+ dfs_radar_disable(dfs);
|
|
|
+ dfs_second_segment_radar_disable(dfs);
|
|
|
+ }
|
|
|
+
|
|
|
+ return QDF_STATUS_SUCCESS;
|
|
|
}
|