Files
linux-nv-oot/drivers/net/wireless/realtek/rtl8852ce/phl/phl_p2pps.c
Shobek Attupurath 7dd632ff96 rtl8852ce: Add base driver v1.19.16.1-0-g1fe335ba1.20240815_PC
- support Android-14
- support Linux kernel 6.9
- support 6G regulation
- support Thermal protection
- support TX shortcut to reduce CPU loading
- fix some coverity issues
- Use RTW regulatory version rtk_8852CE_M.2_2230-67-52
- default enable con-current and MCC

Bug 4667769
Bug 4667981

Change-Id: Iee069ecdd1f00a0b78285d0a4ef5778ed9ace478
Reviewed-on: https://git-master.nvidia.com/r/c/linux-nv-oot/+/3195601
Tested-by: Shobek Attupurath <sattupurath@nvidia.com>
Reviewed-by: Revanth Kumar Uppala <ruppala@nvidia.com>
GVS: buildbot_gerritrpt <buildbot_gerritrpt@nvidia.com>
Reviewed-by: Ashutosh Jha <ajha@nvidia.com>
2025-07-24 10:19:08 +00:00

977 lines
32 KiB
C

/******************************************************************************
*
* Copyright(c) 2020 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.
*
*****************************************************************************/
#define _PHL_P2PPS_C_
#include "phl_headers.h"
#ifdef RTW_WKARD_P2PPS_REFINE
#ifdef CONFIG_PHL_P2PPS
enum rtw_phl_status phl_p2pps_init(struct phl_info_t *phl)
{
enum rtw_phl_status status = RTW_PHL_STATUS_SUCCESS;
struct rtw_phl_com_t *phl_com = phl->phl_com;
struct rtw_phl_p2pps_info *info;
info = (struct rtw_phl_p2pps_info *)_os_mem_alloc(phl_to_drvpriv(phl),
sizeof(*info));
if (info == NULL)
return RTW_PHL_STATUS_RESOURCE;
_os_mem_set(phl_to_drvpriv(phl),
info, 0, sizeof(*info));
phl_com->p2pps_info = (void*)info;
info->phl_info = phl;
_os_spinlock_init(phl_to_drvpriv(phl), &info->p2pps_lock);
return status;
}
void phl_p2pps_deinit(struct phl_info_t *phl_info)
{
struct rtw_phl_com_t *phl_com = phl_info->phl_com;
struct rtw_phl_p2pps_info *info ;
info = (struct rtw_phl_p2pps_info *)phl_com->p2pps_info;
if (info) {
_os_spinlock_free(phl_to_drvpriv(phl_info), &info->p2pps_lock);
_os_mem_free(phl_to_drvpriv(phl_info), info, sizeof(*info));
}
phl_com->p2pps_info = NULL;
}
void
_phl_p2pps_dump_single_noa_desc(struct rtw_phl_noa_desc *desc)
{
PHL_TRACE(COMP_PHL_P2PPS, _PHL_INFO_, "[NOA]_phl_p2pps_dump_single_noa_desc():enable = %d\n",
desc->enable);
PHL_TRACE(COMP_PHL_P2PPS, _PHL_INFO_, "[NOA]_phl_p2pps_dump_single_noa_desc():start_t_h = 0x%x\n",
desc->start_t_h);
PHL_TRACE(COMP_PHL_P2PPS, _PHL_INFO_, "[NOA]_phl_p2pps_dump_single_noa_desc():start_t_l = 0x%x\n",
desc->start_t_l);
PHL_TRACE(COMP_PHL_P2PPS, _PHL_INFO_, "[NOA]_phl_p2pps_dump_single_noa_desc():interval = %d\n",
desc->interval);
PHL_TRACE(COMP_PHL_P2PPS, _PHL_INFO_, "[NOA]_phl_p2pps_dump_single_noa_desc():duration = %d\n",
desc->duration);
PHL_TRACE(COMP_PHL_P2PPS, _PHL_INFO_, "[NOA]_phl_p2pps_dump_single_noa_desc():count = %d\n",
desc->count);
PHL_TRACE(COMP_PHL_P2PPS, _PHL_INFO_, "[NOA]_phl_p2pps_dump_single_noa_desc():noa_id = %d\n",
desc->noa_id);
PHL_TRACE(COMP_PHL_P2PPS, _PHL_INFO_, "[NOA]_phl_p2pps_dump_single_noa_desc():tag = %d\n",
desc->tag);
PHL_TRACE(COMP_PHL_P2PPS, _PHL_INFO_, "[NOA]_phl_p2pps_dump_single_noa_desc():w_role = 0x%p\n",
desc->w_role);
PHL_TRACE(COMP_PHL_P2PPS, _PHL_INFO_, "[NOA]_phl_p2pps_dump_single_noa_desc():rlink = 0x%p\n",
desc->rlink);
}
void
_phl_p2pps_dump_noa_table(struct rtw_phl_p2pps_info *psinfo,
struct rtw_phl_noa_info *info)
{
void *drvpriv = phlcom_to_drvpriv(psinfo->phl_info->phl_com);
struct rtw_phl_noa_desc *desc = NULL;
u8 i = 0;
PHL_TRACE(COMP_PHL_P2PPS, _PHL_INFO_, "[NOA]_phl_p2pps_dump_noa_table():====>\n");
_os_spinlock(drvpriv, &psinfo->p2pps_lock, _bh, NULL);
PHL_TRACE(COMP_PHL_P2PPS, _PHL_INFO_, "[NOA] info.en_desc_num = %d, pause = %d\n",
info->en_desc_num, info->paused);
for (i = 0; i < MAX_NOA_DESC; i++) {
desc = &info->noa_desc[i];
PHL_TRACE(COMP_PHL_P2PPS, _PHL_INFO_, "[NOA]================DESC[%d]==================\n",
i);
PHL_TRACE(COMP_PHL_P2PPS, _PHL_INFO_, "[NOA]_phl_p2pps_dump_noa_table():enable = %d\n",
desc->enable);
PHL_TRACE(COMP_PHL_P2PPS, _PHL_INFO_, "[NOA]_phl_p2pps_dump_noa_table():start_t_h = 0x%x\n",
desc->start_t_h);
PHL_TRACE(COMP_PHL_P2PPS, _PHL_INFO_, "[NOA]_phl_p2pps_dump_noa_table():start_t_l = 0x%x\n",
desc->start_t_l);
PHL_TRACE(COMP_PHL_P2PPS, _PHL_INFO_, "[NOA]_phl_p2pps_dump_noa_table():interval = %d\n",
desc->interval);
PHL_TRACE(COMP_PHL_P2PPS, _PHL_INFO_, "[NOA]_phl_p2pps_dump_noa_table():duration = %d\n",
desc->duration);
PHL_TRACE(COMP_PHL_P2PPS, _PHL_INFO_, "[NOA]_phl_p2pps_dump_noa_table():count = %d\n",
desc->count);
PHL_TRACE(COMP_PHL_P2PPS, _PHL_INFO_, "[NOA]_phl_p2pps_dump_noa_table():noa_id = %d\n",
desc->noa_id);
PHL_TRACE(COMP_PHL_P2PPS, _PHL_INFO_, "[NOA]_phl_p2pps_dump_noa_table():tag = %d\n",
desc->tag);
PHL_TRACE(COMP_PHL_P2PPS, _PHL_INFO_, "[NOA]_phl_p2pps_dump_noa_table():w_role = 0x%p\n",
desc->w_role);
PHL_TRACE(COMP_PHL_P2PPS, _PHL_INFO_, "[NOA]_phl_p2pps_dump_noa_table():rlink = 0x%p\n",
desc->rlink);
PHL_TRACE(COMP_PHL_P2PPS, _PHL_INFO_, "[NOA]================DESC[%d]==================\n",
i);
}
_os_spinunlock(drvpriv, &psinfo->p2pps_lock, _bh, NULL);
}
struct rtw_phl_noa_info *
_phl_p2pps_get_noa_info_by_role(struct rtw_phl_p2pps_info *psinfo,
struct rtw_wifi_role_t *wrole)
{
u8 idx = get_role_idx(wrole);
return &psinfo->noa_info[idx];
}
struct rtw_phl_noa_desc *
_phl_p2pps_get_first_noa_desc_with_cnt255(struct phl_info_t *phl,
struct rtw_phl_noa_info *info)
{
u8 i = 0;
struct rtw_phl_noa_desc *tmp_desc;
for (i = 0; i < MAX_NOA_DESC; i++) {
tmp_desc = &info->noa_desc[i];
if(tmp_desc->count == 255 && tmp_desc->enable) {
PHL_TRACE(COMP_PHL_P2PPS, _PHL_INFO_, "[NOA]_phl_p2pps_get_first_noa_desc_with_cnt255():get desc, tag = %d!!\n",
tmp_desc->tag);
return tmp_desc;
}
}
return NULL;
}
#ifdef RTW_WKARD_P2PPS_SINGLE_NOA
u8
_phl_p2pps_query_mcc_inprog_wkard(struct phl_info_t *phl_info,
struct rtw_wifi_role_t *w_role)
{
u8 ret = false;
#ifdef CONFIG_MR_COEX_SUPPORT
ret = rtw_phl_mr_coex_query_inprogress(phl_info,
get_rlink(w_role, RTW_RLINK_PRIMARY)->hw_band,
RTW_MR_COEX_CHK_INPROGRESS_TDMRA);
#endif /* CONFIG_MR_COEX_SUPPORT */
return ret;
}
void
_phl_p2pps_calc_next_noa_s_time(struct phl_info_t *phl_info,
struct rtw_wifi_role_t *w_role,
struct rtw_phl_tsf32_tog_rpt *rpt,
struct rtw_phl_noa_desc *orig_desc,
struct rtw_phl_noa_desc *new_desc)
{
void *d = phl_to_drvpriv(phl_info);
u64 new_st = 0, old_st = 0;
u64 tog_t = 0, delta_t = 0, intv_cnt = 0;
_os_mem_cpy(d, new_desc, orig_desc, sizeof(*orig_desc));
old_st = (((u64)orig_desc->start_t_h << 32) | orig_desc->start_t_l);
PHL_TRACE(COMP_PHL_P2PPS, _PHL_INFO_, "[NOA]_phl_p2pps_calc_next_noa_s_time():old_st: 0x%08x %08x\n",
(u32)(old_st >> 32), (u32)old_st);
tog_t = (((u64)rpt->tsf_h << 32) | rpt->tsf_l);
PHL_TRACE(COMP_PHL_P2PPS, _PHL_INFO_, "[NOA]_phl_p2pps_calc_next_noa_s_time():tog_t = 0x%08x %08x\n",
(u32)(tog_t >> 32), (u32)tog_t);
delta_t = tog_t - old_st;
PHL_TRACE(COMP_PHL_P2PPS, _PHL_INFO_, "[NOA]_phl_p2pps_calc_next_noa_s_time():delta_t = 0x%08x %08x\n",
(u32)(delta_t >> 32), (u32)delta_t);
if (new_desc->interval) {
intv_cnt = _os_division64(delta_t, new_desc->interval) + 1;
PHL_TRACE(COMP_PHL_P2PPS, _PHL_INFO_, "[NOA]_phl_p2pps_calc_next_noa_s_time():intv_cnt = 0x%08x %08x\n",
(u32)(intv_cnt >> 32), (u32)intv_cnt);
}
new_st = old_st + (intv_cnt * new_desc->interval);
PHL_TRACE(COMP_PHL_P2PPS, _PHL_INFO_, "[NOA]_phl_p2pps_calc_next_noa_s_time():new_st = 0x%08x %08x\n",
(u32)(new_st >> 32), (u32)new_st);
new_desc->start_t_h = new_st >> 32;
new_desc->start_t_l = new_st & 0xFFFFFFFF;
}
void _phl_p2pps_ap_on_tsf32_tog(struct phl_info_t* phl_info,
struct rtw_wifi_role_t *wrole,
struct rtw_phl_tsf32_tog_rpt *rpt)
{
struct rtw_phl_p2pps_info *psinfo = phl_to_p2pps_info(phl_info);
struct rtw_phl_noa_info *info = NULL;
struct rtw_phl_noa_desc *orig_desc = NULL;
struct rtw_phl_noa_desc new_desc = {0};
void *d = phl_to_drvpriv(phl_info);
info = _phl_p2pps_get_noa_info_by_role(psinfo, wrole);
orig_desc = _phl_p2pps_get_first_noa_desc_with_cnt255(phl_info, info);
if (orig_desc) {
_phl_p2pps_calc_next_noa_s_time(phl_info, wrole, rpt,
orig_desc, &new_desc);
_os_mem_cpy(d, orig_desc, &new_desc, sizeof(new_desc));
_phl_p2pps_dump_single_noa_desc(&new_desc);
if(psinfo->ops.tsf32_tog_update_single_noa)
psinfo->ops.tsf32_tog_update_single_noa(d, wrole, &new_desc);
} else {
return;
}
}
#endif
void phl_p2pps_tsf32_tog_handler(struct phl_info_t* phl_info)
{
void *hal = phl_info->hal;
struct rtw_phl_tsf32_tog_rpt rpt = {0};
struct rtw_wifi_role_t *wrole = NULL;
enum rtw_hal_status h_stat;
h_stat = rtw_hal_get_tsf32_tog_rpt(hal, &rpt);
if (h_stat != RTW_HAL_STATUS_SUCCESS)
return;
if (!rpt.valid) {
PHL_TRACE(COMP_PHL_P2PPS, _PHL_WARNING_, "[NOA]phl_p2pps_tsf32_tog_handler():report not valid!!\n");
return;
}
wrole = rtw_phl_get_role_by_band_port(phl_info, rpt.band, rpt.port);
if (wrole) {
if (wrole->type == PHL_RTYPE_AP ||
wrole->type == PHL_RTYPE_P2P_GO) {
PHL_TRACE(COMP_PHL_P2PPS, _PHL_INFO_, "[NOA]phl_p2pps_tsf32_tog_handler():role(%d) is AP/GO mode, handle noa update\n",
wrole->id);
#ifdef RTW_WKARD_P2PPS_SINGLE_NOA
_phl_p2pps_ap_on_tsf32_tog(phl_info, wrole, &rpt);
#endif
} else if (wrole->type == PHL_RTYPE_STATION ||
wrole->type == PHL_RTYPE_P2P_GC) {
PHL_TRACE(COMP_PHL_P2PPS, _PHL_INFO_, "[NOA]phl_p2pps_tsf32_tog_handler():role(%d) is STA/GO mode, currently do nothing\n",
wrole->id);
/*Call NoA disable all?*/
}
} else {
PHL_TRACE(COMP_PHL_P2PPS, _PHL_WARNING_, "[NOA]phl_p2pps_tsf32_tog_handler():NULL ROLE!!, hwband = %d, hwport = %d\n",
rpt.band, rpt.port);
}
}
void
_phl_p2pps_copy_noa_desc(struct rtw_phl_p2pps_info *psinfo,
struct rtw_phl_noa_desc *dest,
struct rtw_phl_noa_desc *src)
{
void *drvpriv = phlcom_to_drvpriv(psinfo->phl_info->phl_com);
_os_spinlock(drvpriv, &psinfo->p2pps_lock, _bh, NULL);
_os_mem_cpy(drvpriv, dest, src, sizeof(struct rtw_phl_noa_desc));
_os_spinunlock(drvpriv, &psinfo->p2pps_lock, _bh, NULL);
}
void
_phl_p2pps_clear_noa_desc(struct rtw_phl_p2pps_info *psinfo,
struct rtw_phl_noa_desc *desc)
{
void *drvpriv = phlcom_to_drvpriv(psinfo->phl_info->phl_com);
_os_spinlock(drvpriv, &psinfo->p2pps_lock, _bh, NULL);
_os_mem_set(drvpriv, desc, 0, sizeof(struct rtw_phl_noa_desc));
_os_spinunlock(drvpriv, &psinfo->p2pps_lock, _bh, NULL);
}
void
_phl_p2pps_noa_increase_desc(struct rtw_phl_p2pps_info *psinfo,
struct rtw_phl_noa_info *info)
{
void *drvpriv = phlcom_to_drvpriv(psinfo->phl_info->phl_com);
_os_spinlock(drvpriv, &psinfo->p2pps_lock, _bh, NULL);
info->en_desc_num++;
_os_spinunlock(drvpriv, &psinfo->p2pps_lock, _bh, NULL);
}
void
_phl_p2pps_noa_decrease_desc(struct rtw_phl_p2pps_info *psinfo,
struct rtw_phl_noa_info *info)
{
void *drvpriv = phlcom_to_drvpriv(psinfo->phl_info->phl_com);
_os_spinlock(drvpriv, &psinfo->p2pps_lock, _bh, NULL);
if (info->en_desc_num > 0)
info->en_desc_num--;
else
PHL_TRACE(COMP_PHL_P2PPS, _PHL_WARNING_, "[NOA]_phl_p2pps_noa_decrease_desc():info->en_desc_num == 0! Flow error\n");
_os_spinunlock(drvpriv, &psinfo->p2pps_lock, _bh, NULL);
}
u8
_phl_p2pps_noa_should_activate(struct rtw_phl_p2pps_info *psinfo,
struct rtw_phl_noa_desc *in_desc)
{
u8 ret = true;
if (in_desc->tag == P2PPS_TRIG_GO) {
ret = true;
} else if (in_desc->tag == P2PPS_TRIG_GC) {
ret = true;
} else if (in_desc->tag == P2PPS_TRIG_GC_255) {
ret = true;
} else if (in_desc->tag == P2PPS_TRIG_2G_SCC_1AP_1STA_BT) {
ret = true;
} else if (in_desc->tag == P2PPS_TRIG_MCC) {
ret = false;
#ifdef RTW_WKARD_P2PPS_NOA_MCC
goto exit;
#endif
}
#ifdef RTW_WKARD_P2PPS_SINGLE_NOA
#ifdef CONFIG_MR_COEX_SUPPORT
/*Currently should only notify MRC for limit request*/
/*Under count == 255 case */
if (in_desc->count != 255) {
if (_phl_p2pps_query_mcc_inprog_wkard(psinfo->phl_info,
in_desc->w_role)) {
PHL_TRACE(COMP_PHL_P2PPS, _PHL_WARNING_, "[NOA]_phl_p2pps_noa_should_activate():mcc in progress and noa requset != 255, currently not handling!\n");
ret = false;
}
} else {
if (phl_mr_coex_noa_dur_lim_change(psinfo->phl_info,
in_desc->w_role,
get_rlink(in_desc->w_role, RTW_RLINK_PRIMARY),
in_desc)) {
PHL_TRACE(COMP_PHL_P2PPS, _PHL_INFO_, "[NOA]_phl_p2pps_noa_should_activate():mrc take over this req!\n");
ret = false;
}
}
#endif /* CONFIG_MR_COEX_SUPPORT */
#endif
exit:
PHL_TRACE(COMP_PHL_P2PPS, _PHL_INFO_, "[NOA]_phl_p2pps_noa_should_activate():tag = %d, return = %d\n",
in_desc->tag, ret);
return ret;
}
u8
_phl_p2pps_noa_is_all_disable(struct rtw_phl_p2pps_info *psinfo,
struct rtw_phl_noa_info *info)
{
u8 i = 0;
void *drvpriv = phlcom_to_drvpriv(psinfo->phl_info->phl_com);
_os_spinlock(drvpriv, &psinfo->p2pps_lock, _bh, NULL);
for (i = 0; i < MAX_NOA_DESC; i++) {
struct rtw_phl_noa_desc *desc = &info->noa_desc[i];
if(desc->enable) {
_os_spinunlock(drvpriv, &psinfo->p2pps_lock, _bh, NULL);
return false;
}
}
_os_spinunlock(drvpriv, &psinfo->p2pps_lock, _bh, NULL);
return true;
}
u8
_phl_p2pps_noa_assign_noaid(struct rtw_phl_p2pps_info *psinfo,
struct rtw_phl_noa_info *info,
struct rtw_phl_noa_desc *desc)
{
u8 max = 0, id = NOAID_NONE, i = 0;
void *drvpriv = phlcom_to_drvpriv(psinfo->phl_info->phl_com);
_os_spinlock(drvpriv, &psinfo->p2pps_lock, _bh, NULL);
if (info->en_desc_num == 0) {
id = 0;/*not inited flow*/
} else {
for (i = 0; i < MAX_NOA_DESC; i++) {
if (info->noa_desc[i].noa_id == NOAID_NONE)
continue;
if (info->noa_desc[i].noa_id > max)
max = info->noa_desc[i].noa_id;
}
if(max != 0)
id = max + 1;
else id = 0;
}
_os_spinunlock(drvpriv, &psinfo->p2pps_lock, _bh, NULL);
PHL_TRACE(COMP_PHL_P2PPS, _PHL_INFO_, "[NOA]_phl_p2pps_noa_assign_noaid(): Final ID = %d.\n",
id);
return id;
}
enum rtw_phl_status
_phl_p2pps_noa_disable(struct rtw_phl_p2pps_info *psinfo,
struct rtw_phl_noa_info *noa_info,
struct rtw_phl_noa_desc *noa_desc,
u8 clear_desc)
{
enum rtw_phl_status ret = RTW_PHL_STATUS_FAILURE;
enum rtw_hal_status hal_ret = RTW_HAL_STATUS_FAILURE;
void *drvpriv = phlcom_to_drvpriv(psinfo->phl_info->phl_com);
void *hal = psinfo->phl_info->hal;
struct rtw_phl_stainfo_t *sta_info = NULL;
struct rtw_wifi_role_t *w_role = NULL;
struct phl_info_t *phl_info = psinfo->phl_info;
u8 en_to_fw = 0;
u8 idx = 0;
if (noa_info->paused && clear_desc) {
PHL_TRACE(COMP_PHL_P2PPS, _PHL_INFO_, "[NOA]_phl_p2pps_noa_disable():NoA info is in puase state, clear desc only!\n");
_phl_p2pps_clear_noa_desc(psinfo,noa_desc);
return RTW_PHL_STATUS_SUCCESS;
}
w_role = noa_desc->w_role;
_os_spinlock(drvpriv, &psinfo->p2pps_lock, _bh, NULL);
en_to_fw = (noa_desc->noa_id != NOAID_NONE && noa_desc->enable);
_os_spinunlock(drvpriv, &psinfo->p2pps_lock, _bh, NULL);
PHL_TRACE(COMP_PHL_P2PPS, _PHL_INFO_, "[NoA]%s(): en_to_fw(%d), clear_desc(%d)\n",
__func__, en_to_fw, clear_desc);
if (en_to_fw) {
sta_info = rtw_phl_get_stainfo_self(psinfo->phl_info,
noa_desc->rlink);
hal_ret = rtw_hal_noa_disable(hal, noa_info, noa_desc,
sta_info->macid);
if (hal_ret!= RTW_HAL_STATUS_SUCCESS) {
PHL_TRACE(COMP_PHL_P2PPS, _PHL_ERR_, "[NOA]_phl_p2pps_noa_disable():NoA Disable fail! tag = %d, ID = %d, HAL return = %d\n",
noa_desc->tag, noa_desc->noa_id, hal_ret);
ret = RTW_PHL_STATUS_FAILURE;
} else {
PHL_TRACE(COMP_PHL_P2PPS, _PHL_INFO_, "[NOA]_phl_p2pps_noa_disable():NoA disable SUCCESS! tag = %d, ID = %d\n",
noa_desc->tag, noa_desc->noa_id);
_phl_p2pps_noa_decrease_desc(psinfo,noa_info);
ret = RTW_PHL_STATUS_SUCCESS;
if (clear_desc)
_phl_p2pps_clear_noa_desc(psinfo,noa_desc);
}
} else {
/*not enabled to fw case*/
ret = RTW_PHL_STATUS_SUCCESS;
if (clear_desc)
_phl_p2pps_clear_noa_desc(psinfo,noa_desc);
}
if(RTW_PHL_STATUS_SUCCESS == ret) {
if(NULL != w_role) {
/* notify BTC */
/* copy noa_desc array to w_role*/
for (idx = 0; idx < MAX_NOA_DESC; idx ++) {
_phl_p2pps_copy_noa_desc(psinfo,
w_role->noa_desc + idx,
noa_info->noa_desc + idx);
}
phl_role_noa_notify(phl_info, w_role);
} else {
PHL_TRACE(COMP_PHL_P2PPS, _PHL_WARNING_, "[NOA]_phl_p2pps_noa_disable():w_role in noa_desc is NULL, not to notify to BTC\n");
}
}
return ret;
}
void _phl_p2pps_noa_disable_all(struct phl_info_t *phl,
struct rtw_wifi_role_t *w_role)
{
struct rtw_phl_p2pps_info *psinfo = phl_to_p2pps_info(phl);
u8 role_id = get_role_idx(w_role);
struct rtw_phl_noa_info *noa_info = &psinfo->noa_info[role_id];
u8 i = 0;
PHL_TRACE(COMP_PHL_P2PPS, _PHL_INFO_, "[NOA]%s:====>\n", __func__);
PHL_TRACE(COMP_PHL_P2PPS, _PHL_INFO_, "[NOA]%s: Disable all NoA for wrole(%d)!\n",
__func__, role_id);
_phl_p2pps_dump_noa_table(phl_to_p2pps_info(phl),noa_info);
for (i = 0; i < MAX_NOA_DESC; i++) {
struct rtw_phl_noa_desc *desc = &noa_info->noa_desc[i];
if (desc->enable) {
_phl_p2pps_noa_disable(psinfo, noa_info, desc, true);
}
}
noa_info->paused = false;
_phl_p2pps_dump_noa_table(phl_to_p2pps_info(phl),noa_info);
PHL_TRACE(COMP_PHL_P2PPS, _PHL_INFO_, "[NOA]%s:<====\n", __func__);
}
enum rtw_phl_status
_phl_p2p_ps_up_clients_macid_for_ap_role(struct phl_info_t *phl,
struct rtw_phl_p2pps_info *psinfo, struct rtw_wifi_role_t *wrole)
{
enum rtw_phl_status psts = RTW_PHL_STATUS_FAILURE;
struct macid_ctl_t *mc = phl_to_mac_ctrl(phl);
struct rtw_phl_stainfo_t *self_sta = rtw_phl_get_stainfo_self(phl,
&wrole->rlink[RTW_RLINK_PRIMARY]);
u32 clients_usedmap[PHL_MACID_MAX_ARRAY_NUM] = {0};
if (wrole->rlink[RTW_RLINK_PRIMARY].assoc_sta_queue.cnt == 1) {
psts = RTW_PHL_STATUS_SUCCESS;
goto exit;
}
_os_mem_cpy(phlcom_to_drvpriv(phl->phl_com), clients_usedmap,
&mc->wifi_role_usedmap[wrole->id][0],
PHL_MACID_MAX_ARRAY_NUM * sizeof(u32));
/*clean self macid*/
phl_macid_map_clr(clients_usedmap, self_sta->macid);
if (RTW_HAL_STATUS_SUCCESS != rtw_hal_noa_sta_macid_up(phl->hal,
self_sta->macid, true,
(u8 *)clients_usedmap,
PHL_MACID_MAX_ARRAY_NUM * sizeof(u32))) {
PHL_TRACE(COMP_PHL_P2PPS, _PHL_ERR_, "[NoA]%s(): Up clients_usedmap fail\n",
__func__);
} else {
psts = RTW_PHL_STATUS_SUCCESS;
PHL_TRACE(COMP_PHL_P2PPS, _PHL_ERR_, "[NoA]%s(): Up clients_usedmap ok\n",
__func__);
}
exit:
return psts;
}
enum rtw_phl_status
_phl_p2pps_noa_enable(struct rtw_phl_p2pps_info *psinfo,
struct rtw_phl_noa_info *noa_info,
struct rtw_phl_noa_desc *noa_desc,
struct rtw_phl_noa_desc *in_desc)
{
enum rtw_phl_status ret = RTW_PHL_STATUS_FAILURE;
enum rtw_hal_status hal_ret = RTW_HAL_STATUS_FAILURE;
void *hal = psinfo->phl_info->hal;
struct rtw_phl_stainfo_t *sta_info = NULL;
struct rtw_wifi_role_t *w_role = NULL;
struct phl_info_t *phl_info = psinfo->phl_info;
u8 idx = 0;
PHL_TRACE(COMP_PHL_P2PPS, _PHL_INFO_, "[NoA]%s()\n",
__func__);
_phl_p2pps_copy_noa_desc(psinfo, noa_desc, in_desc);
/* get w_role */
w_role = noa_desc->w_role;
if(NULL != w_role) {
/* notify BTC */
/* copy noa_desc array to w_role */
for (idx = 0; idx < MAX_NOA_DESC; idx ++) {
_phl_p2pps_copy_noa_desc(psinfo,
w_role->noa_desc+idx,
noa_info->noa_desc+idx);
}
phl_role_noa_notify(phl_info, w_role);
} else {
PHL_TRACE(COMP_PHL_P2PPS, _PHL_WARNING_, "[NOA]_phl_p2pps_noa_enable():w_role in noa_desc is NULL, not to notify to BTC\n");
}
if (noa_info->paused) {
PHL_TRACE(COMP_PHL_P2PPS, _PHL_INFO_, "[NOA]_phl_p2pps_noa_enable():NoA is in pause state, record request and leave\n");
return RTW_PHL_STATUS_SUCCESS;
}
if (_phl_p2pps_noa_should_activate(psinfo, noa_desc)) {
noa_desc->noa_id = _phl_p2pps_noa_assign_noaid(psinfo, noa_info,
noa_desc);
sta_info = rtw_phl_get_stainfo_self(psinfo->phl_info,
noa_desc->rlink);
hal_ret = rtw_hal_noa_enable(hal, noa_info, noa_desc,
sta_info->macid);
if (hal_ret != RTW_HAL_STATUS_SUCCESS) {
PHL_TRACE(COMP_PHL_P2PPS, _PHL_ERR_, "[NOA]_phl_p2pps_noa_enable():NoA enable fail! tag = %d, ID = %d, HAL return = %d\n",
noa_desc->tag, noa_desc->noa_id, hal_ret);
noa_desc->noa_id = NOAID_NONE;
if (hal_ret == RTW_HAL_STATUS_RESOURCE)
ret = RTW_PHL_STATUS_RESOURCE;
else
ret = RTW_PHL_STATUS_FAILURE;
} else {
PHL_TRACE(COMP_PHL_P2PPS, _PHL_INFO_, "[NOA]_phl_p2pps_noa_enable():NoA enable SUCCESS! tag = %d, ID = %d, HAL return = %d\n",
noa_desc->tag, noa_desc->noa_id, hal_ret);
_phl_p2pps_noa_increase_desc(psinfo,noa_info);
/* Update macid exclude self macid */
if (w_role && rtw_phl_role_is_ap_category(w_role)) {
ret = _phl_p2p_ps_up_clients_macid_for_ap_role(
phl_info, psinfo,
w_role);
} else {
ret = RTW_PHL_STATUS_SUCCESS;
}
}
} else {
noa_desc->noa_id = NOAID_NONE; /*not activate*/
ret = RTW_PHL_STATUS_SUCCESS;
}
return ret;
}
bool
_phl_p2p_noa_runing(struct phl_info_t *phl,
struct rtw_wifi_role_t *w_role)
{
struct rtw_phl_p2pps_info *psinfo = phl_to_p2pps_info(phl);
u8 rid = get_role_idx(w_role);
struct rtw_phl_noa_info *noa_info = &psinfo->noa_info[rid];
u8 i = 0;
bool ret = false;
for (i = 0; i < MAX_NOA_DESC; i++) {
struct rtw_phl_noa_desc *desc = &noa_info->noa_desc[i];
if(desc->enable) {
PHL_TRACE(COMP_PHL_P2PPS, _PHL_INFO_, "[NOA]%s: rid(%d), tag(%d) is runing\n",
__FUNCTION__, rid, desc->tag);
ret = true;
break;
}
}
return ret;
}
static void _phl_noa_update_done(void *priv, u8 *param,
u32 param_len, enum rtw_phl_status sts)
{
if (param) {
_os_kmem_free(priv, param, param_len);
param = NULL;
PHL_TRACE(COMP_PHL_P2PPS, _PHL_INFO_, "%s\n", __func__);
}
}
void
phl_p2pps_noa_resume_all(struct phl_info_t *phl,
struct rtw_wifi_role_t *w_role)
{
struct rtw_phl_p2pps_info *psinfo = phl_to_p2pps_info(phl);
u8 role_idx = get_role_idx(w_role);
struct rtw_phl_noa_info *noa_info = &psinfo->noa_info[role_idx];
u8 i = 0;
PHL_TRACE(COMP_PHL_P2PPS, _PHL_INFO_, "[NOA]phl_p2pps_noa_resume_all():====>\n");
if (!noa_info->paused) {
PHL_TRACE(COMP_PHL_P2PPS, _PHL_INFO_, "[NOA]phl_p2pps_noa_resume_all():NoA not paused on role:%d\n",
w_role->id);
goto exit;
}
// _phl_p2pps_dump_noa_table(phl_to_p2pps_info(phl),noa_info);
noa_info->paused = false;
for (i = 0; i < MAX_NOA_DESC; i++) {
struct rtw_phl_noa_desc *desc = &noa_info->noa_desc[i];
if(desc->enable)
_phl_p2pps_noa_enable(psinfo, noa_info, desc, desc);
}
// _phl_p2pps_dump_noa_table(phl_to_p2pps_info(phl),noa_info);
exit:
PHL_TRACE(COMP_PHL_P2PPS, _PHL_INFO_, "[NOA]phl_p2pps_noa_resume_all():<====\n");
}
void
phl_p2pps_noa_all_role_resume(struct phl_info_t *phl_info, u8 band_idx)
{
struct rtw_phl_com_t *phl_com = phl_info->phl_com;
struct mr_ctl_t *mr_ctl = phlcom_to_mr_ctrl(phl_com);
struct hw_band_ctl_t *band_ctrl = &(mr_ctl->band_ctrl[band_idx]);
struct rtw_wifi_role_t *wrole = NULL;
u8 ridx = 0;
for (ridx = 0; ridx < MAX_WIFI_ROLE_NUMBER; ridx++) {
if (!(band_ctrl->role_map & BIT(ridx)))
continue;
wrole = phl_get_wrole_by_ridx(phl_info, ridx);
if (wrole == NULL)
continue;
PHL_TRACE(COMP_PHL_P2PPS, _PHL_INFO_, "[NOA]p2pps_noa_all_role_resume():role_id(%d)\n",
ridx);
phl_p2pps_noa_resume_all(phl_info, wrole);
}
}
void
phl_p2pps_noa_pause_all(struct phl_info_t *phl,
struct rtw_wifi_role_t *w_role)
{
struct rtw_phl_p2pps_info *psinfo = phl_to_p2pps_info(phl);
u8 role_idx = get_role_idx(w_role);
struct rtw_phl_noa_info *noa_info = &psinfo->noa_info[role_idx];
u8 i = 0;
PHL_TRACE(COMP_PHL_P2PPS, _PHL_INFO_, "[NOA]phl_p2pps_noa_pause_all():====>\n");
//_phl_p2pps_dump_noa_table(phl_to_p2pps_info(phl),noa_info);
if (noa_info->paused) {
PHL_TRACE(COMP_PHL_P2PPS, _PHL_INFO_, "[NOA]phl_p2pps_noa_pause_all():This role(%d) NoA is in pause state\n",
role_idx);
goto exit;
}
noa_info->paused = true;
for (i = 0; i < MAX_NOA_DESC; i++) {
struct rtw_phl_noa_desc *desc = &noa_info->noa_desc[i];
_phl_p2pps_noa_disable(psinfo, noa_info, desc, false);
}
//_phl_p2pps_dump_noa_table(phl_to_p2pps_info(phl),noa_info);
exit:
PHL_TRACE(COMP_PHL_P2PPS, _PHL_INFO_, "[NOA]phl_p2pps_noa_pause_all():<====\n");
}
void phl_p2pps_noa_all_role_pause(struct phl_info_t *phl_info, u8 band_idx)
{
struct rtw_phl_com_t *phl_com = phl_info->phl_com;
struct mr_ctl_t *mr_ctl = phlcom_to_mr_ctrl(phl_com);
struct hw_band_ctl_t *band_ctrl = &(mr_ctl->band_ctrl[band_idx]);
struct rtw_wifi_role_t *wrole = NULL;
u8 ridx = 0;
PHL_TRACE(COMP_PHL_P2PPS, _PHL_INFO_, "[NOA]phl_p2pps_noa_all_role_pause(): band_idx(%d)\n",
band_idx);
for (ridx = 0; ridx < MAX_WIFI_ROLE_NUMBER; ridx++) {
if (!(band_ctrl->role_map & BIT(ridx)))
continue;
wrole = phl_get_wrole_by_ridx(phl_info, ridx);
if (wrole == NULL)
continue;
PHL_TRACE(COMP_PHL_P2PPS, _PHL_INFO_, "[NOA]phl_p2pps_noa_all_role_pause():role_id(%d)\n",
ridx);
phl_p2pps_noa_pause_all(phl_info, wrole);
}
}
void phl_p2pps_noa_disable_all(struct phl_info_t *phl_info,
struct rtw_wifi_role_t *w_role,
struct rtw_wifi_role_link_t *rlink)
{
#ifdef RTW_WKARD_P2PPS_SINGLE_NOA
#ifdef CONFIG_MR_COEX_SUPPORT
struct rtw_phl_noa_desc dis_desc = {0};
/*for notify MR for limitation disabled*/
dis_desc.enable = false;
dis_desc.w_role = w_role;
/*open when mr ready*/
phl_mr_coex_noa_dur_lim_change(phl_info,
w_role,
get_rlink(w_role, RTW_RLINK_PRIMARY),
&dis_desc);
#endif /* CONFIG_MR_COEX_SUPPORT */
#endif
PHL_TRACE(COMP_PHL_P2PPS, _PHL_INFO_, "[NOA]%s:====>\n", __func__);
_phl_p2pps_noa_disable_all(phl_info, w_role);
PHL_TRACE(COMP_PHL_P2PPS, _PHL_INFO_, "[NOA]%s:Disable TSF 32 TOG for role %d\n",
__func__, w_role->id);
rtw_hal_tsf32_tog_disable(phl_info->hal, rlink);
PHL_TRACE(COMP_PHL_P2PPS, _PHL_INFO_, "[NOA]%s:<====\n", __func__);
}
void phl_p2pps_query_noa_with_cnt255(struct phl_info_t* phl_info,
struct rtw_wifi_role_t *w_role,
struct rtw_wifi_role_link_t *rlink,
struct rtw_phl_noa_desc *desc)
{
struct rtw_phl_p2pps_info *psinfo = phl_to_p2pps_info(phl_info);
u8 role_idx = get_role_idx(w_role);
struct rtw_phl_noa_info *info = &psinfo->noa_info[role_idx];
struct rtw_phl_noa_desc *tmp_desc = NULL;
tmp_desc = _phl_p2pps_get_first_noa_desc_with_cnt255(phl_info, info);
if (tmp_desc) {
_phl_p2pps_copy_noa_desc(psinfo, desc, tmp_desc);
} else {
desc->enable = false;
desc->w_role = w_role;
desc->rlink = rlink;
}
}
enum rtw_phl_status
phl_noa_update(struct phl_info_t *phl_i,
struct rtw_phl_noa_desc *in_desc)
{
enum rtw_phl_status ret= RTW_PHL_STATUS_FAILURE;
struct rtw_phl_p2pps_info *psinfo = phl_to_p2pps_info(phl_i);
struct rtw_wifi_role_t *w_role = in_desc->w_role;
u8 role_id = get_role_idx(w_role);
struct rtw_phl_noa_info *noa_info = &psinfo->noa_info[role_id];
u8 desc_idx = in_desc->tag;
struct rtw_phl_noa_desc *noa_desc = &noa_info->noa_desc[desc_idx];
PHL_TRACE(COMP_PHL_P2PPS, _PHL_INFO_, "[NOA]%s:DUMP BEFORE!\n", __func__);
_phl_p2pps_dump_noa_table(psinfo, noa_info);
PHL_TRACE(COMP_PHL_P2PPS, _PHL_INFO_, "[NOA]%s:cur FW en desc num = %d\n",
__func__, noa_info->en_desc_num);
if (in_desc->enable) {
if (_phl_p2pps_noa_is_all_disable(psinfo, noa_info)) {
PHL_TRACE(COMP_PHL_P2PPS, _PHL_INFO_, "[NOA]%s:roleid(%d) Enable TSF 32 Toggle!\n",
__func__, role_id);
rtw_hal_tsf32_tog_enable(phl_i->hal, in_desc->rlink);
/*todo set TSF_ BIT TOG H2C ON*/
}
PHL_TRACE(COMP_PHL_P2PPS, _PHL_INFO_, "[NOA]%s:Tag = %d, NoA enable request!\n",
__func__, in_desc->tag);
PHL_TRACE(COMP_PHL_P2PPS, _PHL_INFO_, "[NOA]%s:Tag = %d, NoA disable origninl req first!\n",
__func__, in_desc->tag);
_phl_p2pps_noa_disable(psinfo, noa_info, noa_desc, true);
ret = _phl_p2pps_noa_enable(psinfo, noa_info, noa_desc,
in_desc);
} else {
PHL_TRACE(COMP_PHL_P2PPS, _PHL_INFO_, "[NOA]%s:Tag = %d, NoA disable request!\n",
__func__, in_desc->tag);
ret = _phl_p2pps_noa_disable(psinfo, noa_info, noa_desc, true);
if (_phl_p2pps_noa_is_all_disable(psinfo, noa_info)) {
PHL_TRACE(COMP_PHL_P2PPS, _PHL_INFO_, "[NOA]%s:roleid(%d) Disable TSF 32 Toggle!\n",
__func__, role_id);
rtw_hal_tsf32_tog_disable(phl_i->hal, in_desc->rlink);
/*todo set TSF_ BIT TOG H2C OFF*/
}
}
PHL_TRACE(COMP_PHL_P2PPS, _PHL_INFO_, "[NOA]%s:DUMP AFTER!\n", __func__);
_phl_p2pps_dump_noa_table(psinfo, noa_info);
return ret;
}
enum rtw_phl_status
rtw_phl_p2pps_noa_update(void *phl,
struct rtw_phl_noa_desc *in_desc)
{
return phl_noa_update((struct phl_info_t *)phl, in_desc);
}
void
phl_p2pps_ap_client_notify(struct phl_info_t *phl,
struct rtw_wifi_role_t *wrole, enum link_state lstate, u16 client_macid)
{
struct rtw_phl_stainfo_t *self_sta = NULL;
bool join = false;
u32 macidmap[PHL_MACID_MAX_ARRAY_NUM] = {0};
if (lstate != PHL_ClIENT_JOINING &&
lstate != PHL_ClIENT_LEFT)
return;
if (false == _phl_p2p_noa_runing(phl, wrole))
return;
self_sta = rtw_phl_get_stainfo_self(phl, &wrole->rlink[RTW_RLINK_PRIMARY]);
if (lstate == PHL_ClIENT_JOINING)
join = true;
phl_macid_map_set(macidmap, client_macid);
if (RTW_HAL_STATUS_SUCCESS != rtw_hal_noa_sta_macid_up(phl->hal,
self_sta->macid, join, (u8 *)macidmap,
PHL_MACID_MAX_ARRAY_NUM * sizeof(u32))) {
PHL_TRACE(COMP_PHL_P2PPS, _PHL_ERR_, "[NoA]%s(): link_sts(%d), Up client macid(%d) fail\n",
__func__, lstate, client_macid);
} else {
PHL_TRACE(COMP_PHL_P2PPS, _PHL_ERR_, "[NoA]%s(): link_sts(%d), Up client macid(%d) ok\n",
__func__, lstate, client_macid);
}
}
void
rtw_phl_p2pps_noa_disable_all(void *phl, struct rtw_wifi_role_t *w_role)
{
struct rtw_wifi_role_link_t *rlink = NULL;
u8 idx = 0;
PHL_TRACE(COMP_PHL_P2PPS, _PHL_INFO_, "[NOA]%s()!\n", __func__);
for (idx = 0; idx < w_role->rlink_num; idx++) {
rlink = get_rlink(w_role, idx);
phl_p2pps_noa_disable_all((struct phl_info_t *)phl, w_role, rlink);
}
}
enum rtw_phl_status
phl_cmd_noa_disable_hdl(struct phl_info_t *phl, u8 *param)
{
enum rtw_phl_status ret = RTW_PHL_STATUS_FAILURE;
struct rtw_wifi_role_t *w_role = (struct rtw_wifi_role_t *)param;
struct rtw_wifi_role_link_t *rlink = NULL;
u8 idx = 0;
PHL_TRACE(COMP_PHL_P2PPS, _PHL_INFO_, "[NOA]%s\n", __func__);
for (idx = 0; idx < w_role->rlink_num; idx++) {
rlink = get_rlink(w_role, idx);
phl_p2pps_noa_disable_all((struct phl_info_t *)phl, w_role, rlink);
}
ret = RTW_PHL_STATUS_SUCCESS;
return ret;
}
enum rtw_phl_status
rtw_phl_noa_update(void *phl, struct rtw_phl_noa_desc *in_desc,
enum phl_cmd_type cmd_type, u8 cmd_timeout)
{
enum rtw_phl_status psts = RTW_PHL_STATUS_FAILURE;
struct phl_info_t *phl_i = (struct phl_info_t *)phl;
void *drv_priv = phl_to_drvpriv(phl_i);
struct rtw_phl_noa_desc *para = NULL;
u32 para_len = 0;
PHL_TRACE(COMP_PHL_P2PPS, _PHL_INFO_, "%s: wr id(%d), cmd_type(%d), cmd_timeout(%d)\n",
__func__, in_desc->w_role->id,cmd_type, cmd_timeout);
_phl_p2pps_dump_single_noa_desc(in_desc);
if (PHL_CMD_DIRECTLY == cmd_type) {
psts = phl_noa_update(phl_i, in_desc);
goto _exit;
}
para_len = sizeof(struct rtw_phl_noa_desc);
para = _os_kmem_alloc(drv_priv, para_len);
if (para == NULL) {
PHL_TRACE(COMP_PHL_P2PPS, _PHL_ERR_, "%s - alloc param failed!\n", __func__);
goto _exit;
}
_os_mem_cpy(drv_priv, para, in_desc, sizeof(struct rtw_phl_noa_desc));
psts = phl_cmd_enqueue(phl_i,
para->rlink->hw_band,
MSG_EVT_NOA_UP,
(u8 *)para,
para_len,
_phl_noa_update_done,
cmd_type,
cmd_timeout);
if (is_cmd_failure(psts)) {
/* Send cmd success, but wait cmd fail*/
psts = RTW_PHL_STATUS_FAILURE;
} else if (psts != RTW_PHL_STATUS_SUCCESS) {
/* Send cmd fail */
_os_kmem_free(drv_priv, para, para_len);
psts = RTW_PHL_STATUS_FAILURE;
}
_exit:
return psts;
}
enum rtw_phl_status
rtw_phl_noa_disable_all(void *phl, struct rtw_wifi_role_t *wrole,
enum phl_cmd_type cmd_type, u8 cmd_timeout)
{
enum rtw_phl_status psts = RTW_PHL_STATUS_FAILURE;
struct phl_info_t *phl_i = (struct phl_info_t *)phl;
PHL_TRACE(COMP_PHL_P2PPS, _PHL_INFO_, "[NOA]%s: wr id(%d), cmd_type(%d), cmd_timeout(%d)\n",
__func__, wrole->id, cmd_type, cmd_timeout);
if (PHL_CMD_DIRECTLY == cmd_type) {
psts = phl_cmd_noa_disable_hdl(phl_i, (u8 *)wrole);
goto _exit;
}
psts = phl_cmd_enqueue(phl_i,
get_rlink(wrole, RTW_RLINK_PRIMARY)->hw_band,
MSG_EVT_NOA_DISABLE,
(u8 *)wrole, 0,
NULL,
cmd_type,
cmd_timeout);
if (is_cmd_failure(psts)) {
/* Send cmd success, but wait cmd fail*/
psts = RTW_PHL_STATUS_FAILURE;
} else if (psts != RTW_PHL_STATUS_SUCCESS) {
/* Send cmd fail */
psts = RTW_PHL_STATUS_FAILURE;
}
_exit:
return psts;
}
void rtw_phl_p2pps_init_ops(void *phl,
struct rtw_phl_p2pps_ops *ops)
{
struct phl_info_t *phl_info = (struct phl_info_t *)phl;
struct rtw_phl_p2pps_info *psinfo = NULL;
psinfo = phl_to_p2pps_info(phl_info);
psinfo->ops.priv = ops->priv;
psinfo->ops.tsf32_tog_update_noa = ops->tsf32_tog_update_noa;
psinfo->ops.tsf32_tog_update_single_noa = ops->tsf32_tog_update_single_noa;
PHL_TRACE(COMP_PHL_P2PPS, _PHL_INFO_, "rtw_phl_p2pps_init_ops(): init ok\n");
}
#endif
#endif