rtl8812au-chinawrj/hal/phydm/rtl8812a/phydm_rtl8812a.c
2018-06-22 18:48:32 +02:00

141 lines
4.6 KiB
C

/******************************************************************************
*
* 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"
#if (RTL8812A_SUPPORT == 1)
#if (defined(CONFIG_PATH_DIVERSITY))
void
odm_update_tx_path_8812a(struct PHY_DM_STRUCT *p_dm, u8 path)
{
struct _ODM_PATH_DIVERSITY_ *p_dm_path_div = &p_dm->dm_path_div;
if (p_dm_path_div->resp_tx_path != path) {
PHYDM_DBG(p_dm, DBG_PATH_DIV, ("Need to Update Tx path\n"));
if (path == RF_PATH_A) {
odm_set_bb_reg(p_dm, 0x80c, 0xFFF0, 0x111); /* Tx by Reg */
odm_set_bb_reg(p_dm, 0x6d8, BIT(7) | BIT6, 1); /* Resp Tx by Txinfo */
} else {
odm_set_bb_reg(p_dm, 0x80c, 0xFFF0, 0x222); /* Tx by Reg */
odm_set_bb_reg(p_dm, 0x6d8, BIT(7) | BIT6, 2); /* Resp Tx by Txinfo */
}
}
p_dm_path_div->resp_tx_path = path;
PHYDM_DBG(p_dm, DBG_PATH_DIV, ("path=%s\n", (path == RF_PATH_A) ? "RF_PATH_A" : "RF_PATH_B"));
}
void
odm_path_diversity_init_8812a(
struct PHY_DM_STRUCT *p_dm
)
{
u32 i;
struct _ODM_PATH_DIVERSITY_ *p_dm_path_div = &p_dm->dm_path_div;
odm_set_bb_reg(p_dm, 0x80c, BIT(29), 1); /* Tx path from Reg */
odm_set_bb_reg(p_dm, 0x80c, 0xFFF0, 0x111); /* Tx by Reg */
odm_set_bb_reg(p_dm, 0x6d8, BIT(7) | BIT6, 1); /* Resp Tx by Txinfo */
odm_update_tx_path_8812a(p_dm, RF_PATH_A);
for (i = 0; i < ODM_ASSOCIATE_ENTRY_NUM; i++) {
p_dm_path_div->path_sel[i] = 1; /* TxInfo default at path-A */
}
}
void
odm_path_diversity_8812a(
struct PHY_DM_STRUCT *p_dm
)
{
u32 i, rssi_avg_a = 0, rssi_avg_b = 0, local_min_rssi, min_rssi = 0xFF;
u8 tx_resp_path = 0, target_path;
struct _ODM_PATH_DIVERSITY_ *p_dm_path_div = &p_dm->dm_path_div;
struct cmn_sta_info *p_entry;
PHYDM_DBG(p_dm, DBG_PATH_DIV, ("Odm_PathDiversity_8812A() =>\n"));
for (i = 0; i < ODM_ASSOCIATE_ENTRY_NUM; i++) {
p_entry = p_dm->p_phydm_sta_info[i];
if (is_sta_active(p_entry)) {
/* 2 Caculate RSSI per path */
rssi_avg_a = (p_dm_path_div->path_a_cnt[i] != 0) ? (p_dm_path_div->path_a_sum[i] / p_dm_path_div->path_a_cnt[i]) : 0;
rssi_avg_b = (p_dm_path_div->path_b_cnt[i] != 0) ? (p_dm_path_div->path_b_sum[i] / p_dm_path_div->path_b_cnt[i]) : 0;
target_path = (rssi_avg_a == rssi_avg_b) ? p_dm_path_div->resp_tx_path : ((rssi_avg_a >= rssi_avg_b) ? RF_PATH_A : RF_PATH_B);
PHYDM_DBG(p_dm, DBG_PATH_DIV, ("mac_id=%d, path_a_sum=%d, path_a_cnt=%d\n", i, p_dm_path_div->path_a_sum[i], p_dm_path_div->path_a_cnt[i]));
PHYDM_DBG(p_dm, DBG_PATH_DIV, ("mac_id=%d, path_b_sum=%d, path_b_cnt=%d\n", i, p_dm_path_div->path_b_sum[i], p_dm_path_div->path_b_cnt[i]));
PHYDM_DBG(p_dm, DBG_PATH_DIV, ("mac_id=%d, rssi_avg_a= %d, rssi_avg_b= %d\n", i, rssi_avg_a, rssi_avg_b));
/* 2 Select Resp Tx path */
local_min_rssi = (rssi_avg_a > rssi_avg_b) ? rssi_avg_b : rssi_avg_a;
if (local_min_rssi < min_rssi) {
min_rssi = local_min_rssi;
tx_resp_path = target_path;
}
/* 2 Select Tx DESC */
if (target_path == RF_PATH_A)
p_dm_path_div->path_sel[i] = 1;
else
p_dm_path_div->path_sel[i] = 2;
PHYDM_DBG(p_dm, DBG_ANT_DIV, ("Tx from TxInfo, target_path=%s\n",
(target_path == RF_PATH_A) ? "RF_PATH_A" : "RF_PATH_B"));
PHYDM_DBG(p_dm, DBG_ANT_DIV, ("p_dm_path_div->path_sel[%d] = %d\n", i, p_dm_path_div->path_sel[i]));
}
p_dm_path_div->path_a_cnt[i] = 0;
p_dm_path_div->path_a_sum[i] = 0;
p_dm_path_div->path_b_cnt[i] = 0;
p_dm_path_div->path_b_sum[i] = 0;
}
/* 2 Update Tx path */
odm_update_tx_path_8812a(p_dm, tx_resp_path);
}
#if (DM_ODM_SUPPORT_TYPE & (ODM_WIN | ODM_CE))
void
odm_set_tx_path_by_tx_info_8812a(
struct PHY_DM_STRUCT *p_dm,
u8 *p_desc,
u8 mac_id
)
{
struct _ODM_PATH_DIVERSITY_ *p_dm_path_div = &p_dm->dm_path_div;
if ((p_dm->support_ic_type != ODM_RTL8812) || (!(p_dm->support_ability & ODM_BB_PATH_DIV)))
return;
SET_TX_DESC_TX_ANT_8812(p_desc, p_dm_path_div->path_sel[mac_id]);
}
#endif
#endif /* CONFIG_PATH_DIVERSITY */
#endif /* #if (RTL8812A_SUPPORT == 1) */