/****************************************************************************** * * Copyright(c) 2007 - 2017 Realtek Corporation. * * This program is free software; you can redistribute it and/or modify it * under the terms of version 2 of the GNU General Public License as * published by the Free Software Foundation. * * This program is distributed in the hope that it will be useful, but WITHOUT * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for * more details. * *****************************************************************************/ /* ************************************************************ * include files * ************************************************************ */ #include "mp_precomp.h" #include "phydm_precomp.h" #ifdef PHYDM_PRIMARY_CCA void phydm_write_dynamic_cca( void *p_dm_void, u8 curr_mf_state ) { struct PHY_DM_STRUCT *p_dm = (struct PHY_DM_STRUCT *)p_dm_void; struct phydm_pricca_struct *primary_cca = &(p_dm->dm_pri_cca); if (primary_cca->mf_state != curr_mf_state) { if (p_dm->support_ic_type & ODM_IC_11N_SERIES) { if (curr_mf_state == MF_USC_LSC) { odm_set_bb_reg(p_dm, 0xc6c, BIT(8) | BIT(7), MF_USC_LSC); odm_set_bb_reg(p_dm, 0xc84, 0xf0000000, primary_cca->cca_th_40m_bkp); /*40M OFDM MF CCA threshold*/ } else { odm_set_bb_reg(p_dm, 0xc6c, BIT(8) | BIT(7), curr_mf_state); odm_set_bb_reg(p_dm, 0xc84, 0xf0000000, 0); /*40M OFDM MF CCA threshold*/ } } primary_cca->mf_state = curr_mf_state; PHYDM_DBG(p_dm, DBG_PRI_CCA, ("Set CCA at ((%s SB)), 0xc6c[8:7]=((%d))\n", ((curr_mf_state == MF_USC_LSC)?"D":((curr_mf_state == MF_LSC)?"L":"U")), curr_mf_state)); } } void phydm_primary_cca_reset( void *p_dm_void ) { struct PHY_DM_STRUCT *p_dm = (struct PHY_DM_STRUCT *)p_dm_void; struct phydm_pricca_struct *primary_cca = &(p_dm->dm_pri_cca); PHYDM_DBG(p_dm, DBG_PRI_CCA, ("[PriCCA] Reset\n")); primary_cca->mf_state = 0xff; primary_cca->pre_bw = (enum channel_width)0xff; phydm_write_dynamic_cca(p_dm, MF_USC_LSC); } void phydm_primary_cca_11n( void *p_dm_void ) { struct PHY_DM_STRUCT *p_dm = (struct PHY_DM_STRUCT *)p_dm_void; struct phydm_pricca_struct *p_primary_cca = &(p_dm->dm_pri_cca); enum channel_width curr_bw = (enum channel_width)(*(p_dm->p_band_width)); if (!(p_dm->support_ability & ODM_BB_PRIMARY_CCA)) return; if (!p_dm->is_linked) { /* is_linked==False */ PHYDM_DBG(p_dm, DBG_PRI_CCA, ("[PriCCA][No Link!!!]\n")); if (p_primary_cca->pri_cca_is_become_linked == true) { phydm_primary_cca_reset(p_dm); p_primary_cca->pri_cca_is_become_linked = p_dm->is_linked; } return; } else { if (p_primary_cca->pri_cca_is_become_linked == false) { PHYDM_DBG(p_dm, DBG_PRI_CCA, ("[PriCCA][Linked !!!]\n")); p_primary_cca->pri_cca_is_become_linked = p_dm->is_linked; } } if (curr_bw != p_primary_cca->pre_bw) { PHYDM_DBG(p_dm, DBG_PRI_CCA, ("[Primary CCA] start ==>\n")); p_primary_cca->pre_bw = curr_bw; if (curr_bw == CHANNEL_WIDTH_40) { if ((*(p_dm->p_sec_ch_offset)) == SECOND_CH_AT_LSB) {/* Primary CH @ upper sideband*/ PHYDM_DBG(p_dm, DBG_PRI_CCA, ("BW40M, Primary CH at USB\n")); phydm_write_dynamic_cca(p_dm, MF_USC); } else { /*Primary CH @ lower sideband*/ PHYDM_DBG(p_dm, DBG_PRI_CCA, ("BW40M, Primary CH at LSB\n")); phydm_write_dynamic_cca(p_dm, MF_LSC); } } else { PHYDM_DBG(p_dm, DBG_PRI_CCA, ("Not BW40M, USB + LSB\n")); phydm_primary_cca_reset(p_dm); } } } #if 0 #if (RTL8188E_SUPPORT == 1) void odm_dynamic_primary_cca_8188e( void *p_dm_void ) { struct PHY_DM_STRUCT *p_dm = (struct PHY_DM_STRUCT *)p_dm_void; struct sta_info *p_entry; struct cmn_sta_info *p_sta; struct phydm_fa_struct *false_alm_cnt = (struct phydm_fa_struct *)phydm_get_structure(p_dm, PHYDM_FALSEALMCNT); struct phydm_pricca_struct *primary_cca = &(p_dm->dm_pri_cca); boolean client_40mhz = false, client_tmp = false; /* connected client BW */ boolean is_connected = false; /* connected or not */ u8 client_40mhz_pre = 0; u32 counter = 0; u8 delay = 1; u64 cur_tx_ok_cnt; u64 cur_rx_ok_cnt; u8 sec_ch_offset = *(p_dm->p_sec_ch_offset); u8 i; if (!p_dm->is_linked) return; if (!(p_dm->support_ability & ODM_BB_PRIMARY_CCA)) return; if (*(p_dm->p_band_width) == CHANNEL_WIDTH_20) { /*curr bw*/ odm_set_bb_reg(p_dm, 0xc6c, BIT(8) | BIT(7), 0); return; } #if (DM_ODM_SUPPORT_TYPE == ODM_WIN) || (DM_ODM_SUPPORT_TYPE == ODM_CE) sec_ch_offset = sec_ch_offset % 2 + 1; /* NIC's definition is reverse to AP 1:secondary below, 2: secondary above */ #endif PHYDM_DBG(p_dm, DBG_PRI_CCA, ("Second CH Offset = %d\n", sec_ch_offset)); /* 3 Check Current WLAN Traffic */ cur_tx_ok_cnt = p_dm->tx_tp; cur_rx_ok_cnt = p_dm->rx_tp; /* ==================Debug Message==================== */ PHYDM_DBG(p_dm, DBG_PRI_CCA, ("TP = %llu\n", cur_tx_ok_cnt + cur_rx_ok_cnt)); PHYDM_DBG(p_dm, DBG_PRI_CCA, ("is_BW40 = %d\n", *(p_dm->p_band_width))); PHYDM_DBG(p_dm, DBG_PRI_CCA, ("BW_LSC = %d\n", false_alm_cnt->cnt_bw_lsc)); PHYDM_DBG(p_dm, DBG_PRI_CCA, ("BW_USC = %d\n", false_alm_cnt->cnt_bw_usc)); PHYDM_DBG(p_dm, DBG_PRI_CCA, ("CCA OFDM = %d\n", false_alm_cnt->cnt_ofdm_cca)); PHYDM_DBG(p_dm, DBG_PRI_CCA, ("CCA CCK = %d\n", false_alm_cnt->cnt_cck_cca)); PHYDM_DBG(p_dm, DBG_PRI_CCA, ("OFDM FA = %d\n", false_alm_cnt->cnt_ofdm_fail)); PHYDM_DBG(p_dm, DBG_PRI_CCA, ("CCK FA = %d\n", false_alm_cnt->cnt_cck_fail)); /* ================================================ */ #if (DM_ODM_SUPPORT_TYPE == ODM_WIN) if (ACTING_AS_AP(p_dm->adapter)) /* primary cca process only do at AP mode */ #endif { #if (DM_ODM_SUPPORT_TYPE == ODM_WIN) PHYDM_DBG(p_dm, DBG_PRI_CCA, ("ACTING as AP mode=%d\n", ACTING_AS_AP(p_dm->adapter))); /* 3 To get entry's connection and BW infomation status. */ for (i = 0; i < ASSOCIATE_ENTRY_NUM; i++) { if (IsAPModeExist(p_dm->adapter) && GetFirstExtAdapter(p_dm->adapter) != NULL) p_entry = AsocEntry_EnumStation(GetFirstExtAdapter(p_dm->adapter), i); else p_entry = AsocEntry_EnumStation(GetDefaultAdapter(p_dm->adapter), i); if (p_entry != NULL) { client_tmp = p_entry->BandWidth; /* client BW */ PHYDM_DBG(p_dm, DBG_PRI_CCA, ("Client_BW=%d\n", client_tmp)); if (client_tmp > client_40mhz) client_40mhz = client_tmp; /* 40M/20M coexist => 40M priority is High */ if (p_entry->bAssociated) { is_connected = true; /* client is connected or not */ break; } } else break; } #elif (DM_ODM_SUPPORT_TYPE == ODM_AP) /* 3 To get entry's connection and BW infomation status. */ for (i = 0; i < ODM_ASSOCIATE_ENTRY_NUM; i++) { p_sta = p_dm->p_phydm_sta_info[i]; if (is_sta_active(p_sta)) { client_tmp = p_sta->bw_mode; if (client_tmp > client_40mhz) client_40mhz = client_tmp; /* 40M/20M coexist => 40M priority is High */ is_connected = true; } } #endif PHYDM_DBG(p_dm, DBG_PRI_CCA, ("is_connected=%d\n", is_connected)); PHYDM_DBG(p_dm, DBG_PRI_CCA, ("Is Client 40MHz=%d\n", client_40mhz)); /* 1 Monitor whether the interference exists or not */ if (primary_cca->monitor_flag == 1) { if (sec_ch_offset == 1) { /* secondary channel is below the primary channel */ if ((false_alm_cnt->cnt_ofdm_cca > 500) && (false_alm_cnt->cnt_bw_lsc > false_alm_cnt->cnt_bw_usc + 500)) { if (false_alm_cnt->cnt_ofdm_fail > false_alm_cnt->cnt_ofdm_cca >> 1) { primary_cca->intf_type = 1; primary_cca->pri_cca_flag = 1; odm_set_bb_reg(p_dm, 0xc6c, BIT(8) | BIT7, 2); /* USC MF */ if (primary_cca->dup_rts_flag == 1) primary_cca->dup_rts_flag = 0; } else { primary_cca->intf_type = 2; if (primary_cca->dup_rts_flag == 0) primary_cca->dup_rts_flag = 1; } } else { /* interferecne disappear */ primary_cca->dup_rts_flag = 0; primary_cca->intf_flag = 0; primary_cca->intf_type = 0; } } else if (sec_ch_offset == 2) { /* secondary channel is above the primary channel */ if ((false_alm_cnt->cnt_ofdm_cca > 500) && (false_alm_cnt->cnt_bw_usc > false_alm_cnt->cnt_bw_lsc + 500)) { if (false_alm_cnt->cnt_ofdm_fail > false_alm_cnt->cnt_ofdm_cca >> 1) { primary_cca->intf_type = 1; primary_cca->pri_cca_flag = 1; odm_set_bb_reg(p_dm, 0xc6c, BIT(8) | BIT7, 1); /* LSC MF */ if (primary_cca->dup_rts_flag == 1) primary_cca->dup_rts_flag = 0; } else { primary_cca->intf_type = 2; if (primary_cca->dup_rts_flag == 0) primary_cca->dup_rts_flag = 1; } } else { /* interferecne disappear */ primary_cca->dup_rts_flag = 0; primary_cca->intf_flag = 0; primary_cca->intf_type = 0; } } primary_cca->monitor_flag = 0; } /* 1 Dynamic Primary CCA Main Function */ if (primary_cca->monitor_flag == 0) { if (*(p_dm->p_band_width) == CHANNEL_WIDTH_40) { /* if RFBW==40M mode which require to process primary cca */ /* 2 STA is NOT Connected */ if (!is_connected) { PHYDM_DBG(p_dm, DBG_PRI_CCA, ("STA NOT Connected!!!!\n")); if (primary_cca->pri_cca_flag == 1) { /* reset primary cca when STA is disconnected */ primary_cca->pri_cca_flag = 0; odm_set_bb_reg(p_dm, 0xc6c, BIT(8) | BIT(7), 0); } if (primary_cca->dup_rts_flag == 1) /* reset Duplicate RTS when STA is disconnected */ primary_cca->dup_rts_flag = 0; if (sec_ch_offset == 1) { /* secondary channel is below the primary channel */ if ((false_alm_cnt->cnt_ofdm_cca > 800) && (false_alm_cnt->cnt_bw_lsc * 5 > false_alm_cnt->cnt_bw_usc * 9)) { primary_cca->intf_flag = 1; /* secondary channel interference is detected!!! */ if (false_alm_cnt->cnt_ofdm_fail > false_alm_cnt->cnt_ofdm_cca >> 1) primary_cca->intf_type = 1; /* interference is shift */ else primary_cca->intf_type = 2; /* interference is in-band */ } else { primary_cca->intf_flag = 0; primary_cca->intf_type = 0; } } else if (sec_ch_offset == 2) { /* secondary channel is above the primary channel */ if ((false_alm_cnt->cnt_ofdm_cca > 800) && (false_alm_cnt->cnt_bw_usc * 5 > false_alm_cnt->cnt_bw_lsc * 9)) { primary_cca->intf_flag = 1; /* secondary channel interference is detected!!! */ if (false_alm_cnt->cnt_ofdm_fail > false_alm_cnt->cnt_ofdm_cca >> 1) primary_cca->intf_type = 1; /* interference is shift */ else primary_cca->intf_type = 2; /* interference is in-band */ } else { primary_cca->intf_flag = 0; primary_cca->intf_type = 0; } } PHYDM_DBG(p_dm, DBG_PRI_CCA, ("primary_cca=%d\n", primary_cca->pri_cca_flag)); PHYDM_DBG(p_dm, DBG_PRI_CCA, ("Intf_Type=%d\n", primary_cca->intf_type)); } /* 2 STA is Connected */ else { if (client_40mhz == 0) /* 3 */ { /* client BW = 20MHz */ if (primary_cca->pri_cca_flag == 0) { primary_cca->pri_cca_flag = 1; if (sec_ch_offset == 1) odm_set_bb_reg(p_dm, 0xc6c, BIT(8) | BIT(7), 2); else if (sec_ch_offset == 2) odm_set_bb_reg(p_dm, 0xc6c, BIT(8) | BIT(7), 1); } PHYDM_DBG(p_dm, DBG_PRI_CCA, ("STA Connected 20M!!! primary_cca=%d\n", primary_cca->pri_cca_flag)); } else /* 3 */ { /* client BW = 40MHz */ if (primary_cca->intf_flag == 1) { /* interference is detected!! */ if (primary_cca->intf_type == 1) { if (primary_cca->pri_cca_flag != 1) { primary_cca->pri_cca_flag = 1; if (sec_ch_offset == 1) odm_set_bb_reg(p_dm, 0xc6c, BIT(8) | BIT(7), 2); else if (sec_ch_offset == 2) odm_set_bb_reg(p_dm, 0xc6c, BIT(8) | BIT(7), 1); } } else if (primary_cca->intf_type == 2) { if (primary_cca->dup_rts_flag != 1) primary_cca->dup_rts_flag = 1; } } else { /* if intf_flag==0 */ if ((cur_tx_ok_cnt + cur_rx_ok_cnt) < 1) { /* idle mode or TP traffic is very low */ if (sec_ch_offset == 1) { if ((false_alm_cnt->cnt_ofdm_cca > 800) && (false_alm_cnt->cnt_bw_lsc * 5 > false_alm_cnt->cnt_bw_usc * 9)) { primary_cca->intf_flag = 1; if (false_alm_cnt->cnt_ofdm_fail > false_alm_cnt->cnt_ofdm_cca >> 1) primary_cca->intf_type = 1; /* interference is shift */ else primary_cca->intf_type = 2; /* interference is in-band */ } } else if (sec_ch_offset == 2) { if ((false_alm_cnt->cnt_ofdm_cca > 800) && (false_alm_cnt->cnt_bw_usc * 5 > false_alm_cnt->cnt_bw_lsc * 9)) { primary_cca->intf_flag = 1; if (false_alm_cnt->cnt_ofdm_fail > false_alm_cnt->cnt_ofdm_cca >> 1) primary_cca->intf_type = 1; /* interference is shift */ else primary_cca->intf_type = 2; /* interference is in-band */ } } } else { /* TP Traffic is High */ if (sec_ch_offset == 1) { if (false_alm_cnt->cnt_bw_lsc > (false_alm_cnt->cnt_bw_usc + 500)) { if (delay == 0) { /* add delay to avoid interference occurring abruptly, jump one time */ primary_cca->intf_flag = 1; if (false_alm_cnt->cnt_ofdm_fail > false_alm_cnt->cnt_ofdm_cca >> 1) primary_cca->intf_type = 1; /* interference is shift */ else primary_cca->intf_type = 2; /* interference is in-band */ delay = 1; } else delay = 0; } } else if (sec_ch_offset == 2) { if (false_alm_cnt->cnt_bw_usc > (false_alm_cnt->cnt_bw_lsc + 500)) { if (delay == 0) { /* add delay to avoid interference occurring abruptly */ primary_cca->intf_flag = 1; if (false_alm_cnt->cnt_ofdm_fail > false_alm_cnt->cnt_ofdm_cca >> 1) primary_cca->intf_type = 1; /* interference is shift */ else primary_cca->intf_type = 2; /* interference is in-band */ delay = 1; } else delay = 0; } } } } PHYDM_DBG(p_dm, DBG_PRI_CCA, ("Primary CCA=%d\n", primary_cca->pri_cca_flag)); PHYDM_DBG(p_dm, DBG_PRI_CCA, ("Duplicate RTS=%d\n", primary_cca->dup_rts_flag)); } } /* end of connected */ } } /* 1 Dynamic Primary CCA Monitor counter */ if ((primary_cca->pri_cca_flag == 1) || (primary_cca->dup_rts_flag == 1)) { if (client_40mhz == 0) { /* client=20M no need to monitor primary cca flag */ client_40mhz_pre = client_40mhz; return; } counter++; PHYDM_DBG(p_dm, DBG_PRI_CCA, ("counter=%d\n", counter)); if ((counter == 30) || ((client_40mhz - client_40mhz_pre) == 1)) { /* Every 60 sec to monitor one time */ primary_cca->monitor_flag = 1; /* monitor flag is triggered!!!!! */ if (primary_cca->pri_cca_flag == 1) { primary_cca->pri_cca_flag = 0; odm_set_bb_reg(p_dm, 0xc6c, BIT(8) | BIT(7), 0); } counter = 0; } } } client_40mhz_pre = client_40mhz; } #endif #if (RTL8192E_SUPPORT == 1) #if (DM_ODM_SUPPORT_TYPE == ODM_WIN) void odm_dynamic_primary_cca_mp_8192e( void *p_dm_void ) { struct PHY_DM_STRUCT *p_dm = (struct PHY_DM_STRUCT *)p_dm_void; struct _ADAPTER *p_adapter = p_dm->adapter; HAL_DATA_TYPE *p_hal_data = GET_HAL_DATA(p_adapter); struct phydm_fa_struct *false_alm_cnt = &(p_dm->false_alm_cnt); struct phydm_pricca_struct *primary_cca = &(p_dm->dm_pri_cca); u64 OFDM_CCA, OFDM_FA, bw_usc_cnt, bw_lsc_cnt; u8 sec_ch_offset; static u8 count_down = PRI_CCA_MONITOR_TIME; if (!p_dm->is_linked) return; if (!(p_dm->support_ability & ODM_BB_PRIMARY_CCA)) return; OFDM_CCA = false_alm_cnt->cnt_ofdm_cca; OFDM_FA = false_alm_cnt->cnt_ofdm_fail; bw_usc_cnt = false_alm_cnt->cnt_bw_usc; bw_lsc_cnt = false_alm_cnt->cnt_bw_lsc; PHYDM_DBG(p_dm, DBG_PRI_CCA, ("92E: OFDM CCA=%d\n", OFDM_CCA)); PHYDM_DBG(p_dm, DBG_PRI_CCA, ("92E: OFDM FA=%d\n", OFDM_FA)); PHYDM_DBG(p_dm, DBG_PRI_CCA, ("92E: BW_USC=%d\n", bw_usc_cnt)); PHYDM_DBG(p_dm, DBG_PRI_CCA, ("92E: BW_LSC=%d\n", bw_lsc_cnt)); sec_ch_offset = *(p_dm->p_sec_ch_offset); /* NIC: 2: sec is below, 1: sec is above */ if (IsAPModeExist(p_adapter)) { phydm_write_dynamic_cca(p_dm, MF_USC_LSC); return; } if (*(p_dm->p_band_width) != CHANNEL_WIDTH_40) return; PHYDM_DBG(p_dm, DBG_PRI_CCA, ("92E: Cont Down= %d\n", count_down)); PHYDM_DBG(p_dm, DBG_PRI_CCA, ("92E: Primary_CCA_flag=%d\n", primary_cca->pri_cca_flag)); PHYDM_DBG(p_dm, DBG_PRI_CCA, ("92E: Intf_Type=%d\n", primary_cca->intf_type)); PHYDM_DBG(p_dm, DBG_PRI_CCA, ("92E: Intf_flag=%d\n", primary_cca->intf_flag)); PHYDM_DBG(p_dm, DBG_PRI_CCA, ("92E: Duplicate RTS Flag=%d\n", primary_cca->dup_rts_flag)); if (primary_cca->pri_cca_flag == 0) { if (sec_ch_offset == SECOND_CH_AT_LSB) { /* Primary channel is above NOTE: duplicate CTS can remove this condition */ if ((OFDM_CCA > OFDMCCA_TH) && (bw_lsc_cnt > (bw_usc_cnt + bw_ind_bias)) && (OFDM_FA > (OFDM_CCA >> 1))) { primary_cca->intf_type = 1; primary_cca->intf_flag = 1; phydm_write_dynamic_cca(p_dm, MF_USC); primary_cca->pri_cca_flag = 1; } else if ((OFDM_CCA > OFDMCCA_TH) && (bw_lsc_cnt > (bw_usc_cnt + bw_ind_bias)) && (OFDM_FA < (OFDM_CCA >> 1))) { primary_cca->intf_type = 2; primary_cca->intf_flag = 1; phydm_write_dynamic_cca(p_dm, MF_USC); primary_cca->pri_cca_flag = 1; primary_cca->dup_rts_flag = 1; p_hal_data->RTSEN = 1; } else { primary_cca->intf_type = 0; primary_cca->intf_flag = 0; phydm_write_dynamic_cca(p_dm, MF_USC_LSC); p_hal_data->RTSEN = 0; primary_cca->dup_rts_flag = 0; } } else if (sec_ch_offset == SECOND_CH_AT_USB) { if ((OFDM_CCA > OFDMCCA_TH) && (bw_usc_cnt > (bw_lsc_cnt + bw_ind_bias)) && (OFDM_FA > (OFDM_CCA >> 1))) { primary_cca->intf_type = 1; primary_cca->intf_flag = 1; phydm_write_dynamic_cca(p_dm, MF_LSC); primary_cca->pri_cca_flag = 1; } else if ((OFDM_CCA > OFDMCCA_TH) && (bw_usc_cnt > (bw_lsc_cnt + bw_ind_bias)) && (OFDM_FA < (OFDM_CCA >> 1))) { primary_cca->intf_type = 2; primary_cca->intf_flag = 1; phydm_write_dynamic_cca(p_dm, MF_LSC); primary_cca->pri_cca_flag = 1; primary_cca->dup_rts_flag = 1; p_hal_data->RTSEN = 1; } else { primary_cca->intf_type = 0; primary_cca->intf_flag = 0; phydm_write_dynamic_cca(p_dm, MF_USC_LSC); p_hal_data->RTSEN = 0; primary_cca->dup_rts_flag = 0; } } } else { /* primary_cca->pri_cca_flag==1 */ count_down--; if (count_down == 0) { count_down = PRI_CCA_MONITOR_TIME; primary_cca->pri_cca_flag = 0; phydm_write_dynamic_cca(p_dm, MF_USC_LSC); /* default */ p_hal_data->RTSEN = 0; primary_cca->dup_rts_flag = 0; primary_cca->intf_type = 0; primary_cca->intf_flag = 0; } } } #elif (DM_ODM_SUPPORT_TYPE == ODM_AP) void odm_intf_detection( void *p_dm_void ) { struct PHY_DM_STRUCT *p_dm = (struct PHY_DM_STRUCT *)p_dm_void; struct phydm_fa_struct *false_alm_cnt = &(p_dm->false_alm_cnt); struct phydm_pricca_struct *primary_cca = &(p_dm->dm_pri_cca); if ((false_alm_cnt->cnt_ofdm_cca > OFDMCCA_TH) && (false_alm_cnt->cnt_bw_lsc > (false_alm_cnt->cnt_bw_usc + bw_ind_bias))) { primary_cca->intf_flag = 1; primary_cca->CH_offset = 1; /* 1:LSC, 2:USC */ if (false_alm_cnt->cnt_ofdm_fail > (false_alm_cnt->cnt_ofdm_cca >> 1)) primary_cca->intf_type = 1; else primary_cca->intf_type = 2; } else if ((false_alm_cnt->cnt_ofdm_cca > OFDMCCA_TH) && (false_alm_cnt->cnt_bw_usc > (false_alm_cnt->cnt_bw_lsc + bw_ind_bias))) { primary_cca->intf_flag = 1; primary_cca->CH_offset = 2; /* 1:LSC, 2:USC */ if (false_alm_cnt->cnt_ofdm_fail > (false_alm_cnt->cnt_ofdm_cca >> 1)) primary_cca->intf_type = 1; else primary_cca->intf_type = 2; } else { primary_cca->intf_flag = 0; primary_cca->intf_type = 0; primary_cca->CH_offset = 0; } } void odm_dynamic_primary_cca_ap_8192e( void *p_dm_void ) { struct PHY_DM_STRUCT *p_dm = (struct PHY_DM_STRUCT *)p_dm_void; struct phydm_pricca_struct *primary_cca = &(p_dm->dm_pri_cca); u8 i; static u32 count_down = PRI_CCA_MONITOR_TIME; u8 STA_BW = false, STA_BW_pre = false, STA_BW_TMP = false; boolean is_connected = false; u8 sec_ch_offset; u8 cur_mf_state; struct cmn_sta_info *p_entry; if (!p_dm->is_linked) return; if (!(p_dm->support_ability & ODM_BB_PRIMARY_CCA)) return; sec_ch_offset = *(p_dm->p_sec_ch_offset); /* AP: 1: sec is below, 2: sec is above */ for (i = 0; i < ODM_ASSOCIATE_ENTRY_NUM; i++) { p_entry = p_dm->p_phydm_sta_info[i]; if (is_sta_active(p_entry)) { STA_BW_TMP = p_entry->bw_mode; if (STA_BW_TMP > STA_BW) STA_BW = STA_BW_TMP; is_connected = true; } } if (*(p_dm->p_band_width) == CHANNEL_WIDTH_40) { if (primary_cca->pri_cca_flag == 0) { if (is_connected) { if (STA_BW == CHANNEL_WIDTH_20) { /* 2 STA BW=20M */ primary_cca->pri_cca_flag = 1; if (sec_ch_offset == 1) { cur_mf_state = MF_USC; phydm_write_dynamic_cca(p_dm, cur_mf_state); } else if (sec_ch_offset == 2) { cur_mf_state = MF_USC; phydm_write_dynamic_cca(p_dm, cur_mf_state); } } else { /* 2 STA BW=40M */ if (primary_cca->intf_flag == 0) odm_intf_detection(p_dm); else { /* intf_flag = 1 */ if (primary_cca->intf_type == 1) { if (primary_cca->CH_offset == 1) { cur_mf_state = MF_USC; if (sec_ch_offset == 1) /* AP, 1: primary is above 2: primary is below */ phydm_write_dynamic_cca(p_dm, cur_mf_state); } else if (primary_cca->CH_offset == 2) { cur_mf_state = MF_LSC; if (sec_ch_offset == 2) phydm_write_dynamic_cca(p_dm, cur_mf_state); } } else if (primary_cca->intf_type == 2) PHYDM_DBG(p_dm, DBG_PRI_CCA, ("92E: primary_cca->intf_type = 2\n")); } } } else /* disconnected interference detection */ odm_intf_detection(p_dm); /* end of disconnected */ } else { /* primary_cca->pri_cca_flag == 1 */ if (STA_BW == 0) { STA_BW_pre = STA_BW; return; } count_down--; if ((count_down == 0) || ((STA_BW & STA_BW_pre) != 1)) { count_down = PRI_CCA_MONITOR_TIME; primary_cca->pri_cca_flag = 0; primary_cca->intf_type = 0; primary_cca->intf_flag = 0; cur_mf_state = MF_USC_LSC; phydm_write_dynamic_cca(p_dm, cur_mf_state); /* default */ } } STA_BW_pre = STA_BW; } else { /* 2 Reset */ phydm_primary_cca_init(p_dm); cur_mf_state = MF_USC_LSC; phydm_write_dynamic_cca(p_dm, cur_mf_state); count_down = PRI_CCA_MONITOR_TIME; } } #endif #endif /* RTL8192E_SUPPORT == 1 */ #endif #endif boolean odm_dynamic_primary_cca_dup_rts( void *p_dm_void ) { #ifdef PHYDM_PRIMARY_CCA struct PHY_DM_STRUCT *p_dm = (struct PHY_DM_STRUCT *)p_dm_void; struct phydm_pricca_struct *primary_cca = &(p_dm->dm_pri_cca); return primary_cca->dup_rts_flag; #else return 0; #endif } void phydm_primary_cca_init( void *p_dm_void ) { #ifdef PHYDM_PRIMARY_CCA struct PHY_DM_STRUCT *p_dm = (struct PHY_DM_STRUCT *)p_dm_void; struct phydm_pricca_struct *primary_cca = &(p_dm->dm_pri_cca); if (!(p_dm->support_ability & ODM_BB_PRIMARY_CCA)) return; PHYDM_DBG(p_dm, DBG_PRI_CCA, ("[PriCCA] Init ==>\n")); #if (RTL8188E_SUPPORT == 1) || (RTL8192E_SUPPORT == 1) primary_cca->dup_rts_flag = 0; primary_cca->intf_flag = 0; primary_cca->intf_type = 0; primary_cca->monitor_flag = 0; primary_cca->pri_cca_flag = 0; primary_cca->ch_offset = 0; #endif primary_cca->mf_state = 0xff; primary_cca->pre_bw = (enum channel_width)0xff; if (p_dm->support_ic_type & ODM_IC_11N_SERIES) primary_cca->cca_th_40m_bkp = (u8)odm_get_bb_reg(p_dm, 0xc84, 0xf0000000); #endif } void phydm_primary_cca( void *p_dm_void ) { #ifdef PHYDM_PRIMARY_CCA struct PHY_DM_STRUCT *p_dm = (struct PHY_DM_STRUCT *)p_dm_void; if (!(p_dm->support_ic_type & ODM_IC_11N_SERIES)) return; if (!(p_dm->support_ability & ODM_BB_PRIMARY_CCA)) return; phydm_primary_cca_11n(p_dm); #endif }