| 12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788899091929394959697989910010110210310410510610710810911011111211311411511611711811912012112212312412512612712812913013113213313413513613713813914014114214314414514614714814915015115215315415515615715815916016116216316416516616716816917017117217317417517617717817918018118218318418518618718818919019119219319419519619719819920020120220320420520620720820921021121221321421521621721821922022122222322422522622722822923023123223323423523623723823924024124224324424524624724824925025125225325425525625725825926026126226326426526626726826927027127227327427527627727827928028128228328428528628728828929029129229329429529629729829930030130230330430530630730830931031131231331431531631731831932032132232332432532632732832933033133233333433533633733833934034134234334434534634734834935035135235335435535635735835936036136236336436536636736836937037137237337437537637737837938038138238338438538638738838939039139239339439539639739839940040140240340440540640740840941041141241341441541641741841942042142242342442542642742842943043143243343443543643743843944044144244344444544644744844945045145245345445545645745845946046146246346446546646746846947047147247347447547647747847948048148248348448548648748848949049149249349449549649749849950050150250350450550650750850951051151251351451551651751851952052152252352452552652752852953053153253353453553653753853954054154254354454554654754854955055155255355455555655755855956056156256356456556656756856957057157257357457557657757857958058158258358458558658758858959059159259359459559659759859960060160260360460560660760860961061161261361461561661761861962062162262362462562662762862963063163263363463563663763863964064164264364464564664764864965065165265365465565665765865966066166266366466566666766866967067167267367467567667767867968068168268368468568668768868969069169269369469569669769869970070170270370470570670770870971071171271371471571671771871972072172272372472572672772872973073173273373473573673773873974074174274374474574674774874975075175275375475575675775875976076176276376476576676776876977077177277377477577677777877978078178278378478578678778878979079179279379479579679779879980080180280380480580680780880981081181281381481581681781881982082182282382482582682782882983083183283383483583683783883984084184284384484584684784884985085185285385485585685785885986086186286386486586686786886987087187287387487587687787887988088188288388488588688788888989089189289389489589689789889990090190290390490590690790890991091191291391491591691791891992092192292392492592692792892993093193293393493593693793893994094194294394494594694794894995095195295395495595695795895996096196296396496596696796896997097197297397497597697797897998098198298398498598698798898999099199299399499599699799899910001001100210031004100510061007100810091010101110121013101410151016101710181019102010211022102310241025102610271028102910301031103210331034103510361037103810391040104110421043104410451046104710481049105010511052105310541055105610571058105910601061106210631064106510661067106810691070107110721073107410751076107710781079108010811082108310841085108610871088108910901091109210931094109510961097109810991100110111021103110411051106110711081109111011111112111311141115111611171118111911201121112211231124112511261127112811291130113111321133113411351136113711381139114011411142114311441145114611471148114911501151115211531154115511561157115811591160116111621163116411651166116711681169117011711172117311741175117611771178117911801181118211831184118511861187118811891190119111921193119411951196119711981199120012011202120312041205120612071208120912101211121212131214121512161217121812191220122112221223122412251226122712281229123012311232123312341235123612371238123912401241124212431244124512461247124812491250125112521253125412551256125712581259126012611262126312641265126612671268126912701271127212731274127512761277127812791280128112821283128412851286128712881289129012911292129312941295129612971298129913001301130213031304130513061307130813091310131113121313131413151316131713181319132013211322132313241325132613271328132913301331133213331334133513361337133813391340134113421343134413451346134713481349135013511352 |
- /******************************************************************************
- *
- * Copyright(c) 2016 Realtek Corporation. All rights reserved.
- *
- * 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.
- *
- * You should have received a copy of the GNU General Public License along with
- * this program; if not, write to the Free Software Foundation, Inc.,
- * 51 Franklin Street, Fifth Floor, Boston, MA 02110, USA
- *
- *
- ******************************************************************************/
- #define _RTL8821C_CMD_C_
- #include <hal_data.h> /* HAL_DATA_TYPE */
- #include "../hal_halmac.h" /* HRTW_HALMAC_H2C_MAX_SIZE, CMD_ID_RSVD_PAGE and etc. */
- #include "rtl8821c.h"
- /*
- * Below functions are for C2H
- */
- /*****************************************
- * H2C Msg format :
- *| 31 - 8 |7-5 | 4 - 0 |
- *| h2c_msg |Class |CMD_ID |
- *| 31-0 |
- *| Ext msg |
- *
- ******************************************/
- s32 rtl8821c_fillh2ccmd(PADAPTER adapter, u8 id, u32 buf_len, u8 *pbuf)
- {
- u8 h2c[RTW_HALMAC_H2C_MAX_SIZE] = {0};
- #ifdef DBG_H2C_CONTENT
- u8 msg[(RTW_HALMAC_H2C_MAX_SIZE - 1) * 5 + 1] = {0};
- u8 *msg_p;
- u32 msg_size, i, n;
- #endif /* CONFIG_RTW_DEBUG */
- int err;
- s32 ret = _FAIL;
- if (!pbuf)
- goto exit;
- if (buf_len > (RTW_HALMAC_H2C_MAX_SIZE - 1))
- goto exit;
- if (rtw_is_surprise_removed(adapter))
- goto exit;
- #ifdef DBG_H2C_CONTENT
- msg_p = msg;
- msg_size = (RTW_HALMAC_H2C_MAX_SIZE - 1) * 5 + 1;
- for (i = 0; i < buf_len; i++) {
- n = rtw_sprintf(msg_p, msg_size, " 0x%02x", pbuf[i]);
- msg_p += n;
- msg_size -= n;
- if (msg_size == 0)
- break;
- }
- RTW_INFO(FUNC_ADPT_FMT ": id=0x%02x buf=%s\n",
- FUNC_ADPT_ARG(adapter), id, msg);
- #endif /* CONFIG_RTW_DEBUG */
- h2c[0] = id;
- _rtw_memcpy(h2c + 1, pbuf, buf_len);
- err = rtw_halmac_send_h2c(adapter_to_dvobj(adapter), h2c);
- if (_SUCCESS == err)
- ret = _SUCCESS;
- exit:
- return ret;
- }
- static void rtl8821c_set_FwRsvdPage_cmd(PADAPTER adapter, PRSVDPAGE_LOC rsvdpageloc)
- {
- u8 h2c[RTW_HALMAC_H2C_MAX_SIZE] = {0};
- RTW_INFO(FUNC_ADPT_FMT ": ProbeRsp=%d PsPoll=%d Null=%d QoSNull=%d BTNull=%d\n",
- FUNC_ADPT_ARG(adapter),
- rsvdpageloc->LocProbeRsp, rsvdpageloc->LocPsPoll,
- rsvdpageloc->LocNullData, rsvdpageloc->LocQosNull,
- rsvdpageloc->LocBTQosNull);
- RSVD_PAGE_SET_CMD_ID(h2c, CMD_ID_RSVD_PAGE);
- RSVD_PAGE_SET_CLASS(h2c, CLASS_RSVD_PAGE);
- RSVD_PAGE_SET_LOC_PROBE_RSP(h2c, rsvdpageloc->LocProbeRsp);
- RSVD_PAGE_SET_LOC_PS_POLL(h2c, rsvdpageloc->LocPsPoll);
- RSVD_PAGE_SET_LOC_NULL_DATA(h2c, rsvdpageloc->LocNullData);
- RSVD_PAGE_SET_LOC_QOS_NULL(h2c, rsvdpageloc->LocQosNull);
- RSVD_PAGE_SET_LOC_BT_QOS_NULL(h2c, rsvdpageloc->LocBTQosNull);
- RTW_DBG_DUMP("H2C-RsvdPage Parm:", h2c, RTW_HALMAC_H2C_MAX_SIZE);
- rtw_halmac_send_h2c(adapter_to_dvobj(adapter), h2c);
- }
- static void rtl8821c_set_FwAoacRsvdPage_cmd(PADAPTER adapter, PRSVDPAGE_LOC rsvdpageloc)
- {
- #ifdef CONFIG_WOWLAN
- struct pwrctrl_priv *pwrpriv = adapter_to_pwrctl(adapter);
- struct mlme_priv *pmlmepriv = &adapter->mlmepriv;
- u8 res = 0, count = 0;
- u8 h2c[RTW_HALMAC_H2C_MAX_SIZE] = {0};
- RTW_INFO(FUNC_ADPT_FMT ": RWC=%d ArpRsp=%d NbrAdv=%d GtkRsp=%d GtkInfo=%d ProbeReq=%d NetworkList=%d\n",
- FUNC_ADPT_ARG(adapter),
- rsvdpageloc->LocRemoteCtrlInfo, rsvdpageloc->LocArpRsp,
- rsvdpageloc->LocNbrAdv, rsvdpageloc->LocGTKRsp,
- rsvdpageloc->LocGTKInfo, rsvdpageloc->LocProbeReq,
- rsvdpageloc->LocNetList);
- if (check_fwstate(pmlmepriv, _FW_LINKED) == _TRUE) {
- AOAC_RSVD_PAGE_SET_CMD_ID(h2c, CMD_ID_AOAC_RSVD_PAGE);
- AOAC_RSVD_PAGE_SET_CLASS(h2c, CLASS_AOAC_RSVD_PAGE);
- AOAC_RSVD_PAGE_SET_LOC_REMOTE_CTRL_INFO(h2c, rsvdpageloc->LocRemoteCtrlInfo);
- AOAC_RSVD_PAGE_SET_LOC_ARP_RESPONSE(h2c, rsvdpageloc->LocArpRsp);
- AOAC_RSVD_PAGE_SET_LOC_GTK_RSP(h2c, rsvdpageloc->LocGTKRsp);
- AOAC_RSVD_PAGE_SET_LOC_GTK_INFO(h2c, rsvdpageloc->LocGTKInfo);
- #ifdef CONFIG_GTK_OL
- AOAC_RSVD_PAGE_SET_LOC_GTK_EXT_MEM(h2c, rsvdpageloc->LocGTKEXTMEM);
- #endif /* CONFIG_GTK_OL */
- RTW_DBG_DUMP("H2C-AoacRsvdPage Parm:", h2c, RTW_HALMAC_H2C_MAX_SIZE);
- rtw_halmac_send_h2c(adapter_to_dvobj(adapter), h2c);
- } else {
- #ifdef CONFIG_PNO_SUPPORT
- if (!pwrpriv->wowlan_in_resume) {
- RTW_INFO("%s: NLO_INFO=%d\n", __FUNCTION__, rsvdpageloc->LocPNOInfo);
- AOAC_RSVD_PAGE3_SET_CMD_ID(h2c, CMD_ID_AOAC_RSVD_PAGE3);
- AOAC_RSVD_PAGE3_SET_CLASS(h2c, CLASS_AOAC_RSVD_PAGE3);
- AOAC_RSVD_PAGE3_SET_LOC_NLO_INFO(h2c, rsvdpageloc->LocPNOInfo);
- RTW_DBG_DUMP("H2C-AoacRsvdPage3 Parm:", h2c, RTW_HALMAC_H2C_MAX_SIZE);
- rtw_halmac_send_h2c(adapter_to_dvobj(adapter), h2c);
- rtw_msleep_os(10);
- }
- #endif /* CONFIG_PNO_SUPPORT */
- }
- #endif /* CONFIG_WOWLAN */
- }
- static void rtl8821c_set_FwKeepAlive_cmd(PADAPTER adapter, u8 benable, u8 pkt_type)
- {
- u8 h2c[RTW_HALMAC_H2C_MAX_SIZE] = {0};
- u8 adopt = 1;
- #ifdef CONFIG_PLATFORM_INTEL_BYT
- u8 check_period = 10;
- #else
- u8 check_period = 5;
- #endif
- RTW_INFO(FUNC_ADPT_FMT ": benable=%d\n", FUNC_ADPT_ARG(adapter), benable);
- KEEP_ALIVE_SET_CMD_ID(h2c, CMD_ID_KEEP_ALIVE);
- KEEP_ALIVE_SET_CLASS(h2c, CLASS_KEEP_ALIVE);
- KEEP_ALIVE_SET_ENABLE(h2c, benable);
- KEEP_ALIVE_SET_ADOPT_USER_SETTING(h2c, adopt);
- KEEP_ALIVE_SET_PKT_TYPE(h2c, pkt_type);
- KEEP_ALIVE_SET_KEEP_ALIVE_CHECK_PERIOD(h2c, check_period);
- RTW_DBG_DUMP("H2C-KeepAlive Parm:", h2c, RTW_HALMAC_H2C_MAX_SIZE);
- rtw_halmac_send_h2c(adapter_to_dvobj(adapter), h2c);
- }
- static void rtl8821c_set_FwDisconDecision_cmd(PADAPTER adapter, u8 benable)
- {
- u8 h2c[RTW_HALMAC_H2C_MAX_SIZE] = {0};
- u8 adopt = 1, check_period = 10, trypkt_num = 0;
- RTW_INFO(FUNC_ADPT_FMT ": benable=%d\n",
- FUNC_ADPT_ARG(adapter), benable);
- DISCONNECT_DECISION_SET_CMD_ID(h2c, CMD_ID_DISCONNECT_DECISION);
- DISCONNECT_DECISION_SET_CLASS(h2c, CLASS_DISCONNECT_DECISION);
- DISCONNECT_DECISION_SET_ENABLE(h2c, benable);
- DISCONNECT_DECISION_SET_ADOPT_USER_SETTING(h2c, adopt);
- DISCONNECT_DECISION_SET_DISCON_DECISION_CHECK_PERIOD(h2c, check_period);
- DISCONNECT_DECISION_SET_TRY_PKT_NUM(h2c, trypkt_num);
- RTW_DBG_DUMP("H2C-DisconDecision Parm:", h2c, RTW_HALMAC_H2C_MAX_SIZE);
- rtw_halmac_send_h2c(adapter_to_dvobj(adapter), h2c);
- }
- static u8 get_ra_vht_en(u32 wirelessMode, u32 bitmap)
- {
- u8 ret = 0;
- if (wirelessMode == WIRELESS_11_24AC) {
- if (bitmap & 0xfff00000) /* 2SS */
- ret = 3;
- else /* 1SS */
- ret = 2;
- } else if (wirelessMode == WIRELESS_11_5AC)
- ret = 1;
- return ret;
- }
- static u8 get_ra_ldpc(struct sta_info *psta)
- {
- u8 en_ldpc = 0;
- if (psta != NULL) {
- if (psta->mac_id == 1)
- en_ldpc = 0;
- else {
- #ifdef CONFIG_80211AC_VHT
- if (is_supported_vht(psta->wireless_mode)) {
- if (TEST_FLAG(psta->vhtpriv.ldpc_cap, LDPC_VHT_CAP_TX))
- en_ldpc = 1;
- else
- en_ldpc = 0;
- } else if (is_supported_ht(psta->wireless_mode)) {
- if (TEST_FLAG(psta->htpriv.ldpc_cap, LDPC_HT_CAP_TX))
- en_ldpc = 1;
- else
- en_ldpc = 0;
- } else
- #endif
- en_ldpc = 0;
- }
- }
- return en_ldpc;
- }
- /*
- * arg[0] = macid
- * arg[1] = raid
- * arg[2] = shortGIrate
- * arg[3] = init_rate
- */
- void rtl8821c_set_FwMacIdConfig_cmd(PADAPTER adapter, u64 mask, u8 *arg, u8 bw)
- {
- HAL_DATA_TYPE *hal = GET_HAL_DATA(adapter);
- struct macid_ctl_t *macid_ctl = &adapter->dvobj->macid_ctl;
- struct sta_info *psta = NULL;
- u8 mac_id, init_rate, raid, sgi = _FALSE;
- u8 h2c[RTW_HALMAC_H2C_MAX_SIZE] = {0};
- u8 ignore_bw = _FALSE;
- if (hal->fw_ractrl == _FALSE) {
- RTW_INFO(FUNC_ADPT_FMT" fw ractrl = _FALSE\n",
- FUNC_ADPT_ARG(adapter));
- return;
- }
- mac_id = arg[0];
- raid = arg[1];
- sgi = arg[2] & 0x0F;
- ignore_bw = arg[2] >> 4;
- init_rate = arg[3];
- if (mac_id < macid_ctl->num)
- psta = macid_ctl->sta[mac_id];
- if (!psta) {
- RTW_INFO(FUNC_ADPT_FMT" macid:%u, sta is NULL\n",
- FUNC_ADPT_ARG(adapter), mac_id);
- return;
- }
- RTW_INFO(FUNC_ADPT_FMT ": mac_id=%d raid=0x%x bw=%d mask=0x%016llx\n",
- FUNC_ADPT_ARG(adapter), mac_id, raid, bw, mask);
- MACID_CFG_SET_CMD_ID(h2c, CMD_ID_MACID_CFG);
- MACID_CFG_SET_CLASS(h2c, CLASS_MACID_CFG);
- /* common for h2c cmd 0x40 */
- MACID_CFG_SET_MAC_ID(h2c, mac_id);
- MACID_CFG_SET_RATE_ID(h2c, raid);
- #ifdef CONFIG_LPS_PG
- MACID_CFG_SET_INIT_RATE_LV(h2c, psta->lps_pg_rssi_lv);
- #endif
- MACID_CFG_SET_SGI(h2c, (sgi) ? 1 : 0);
- MACID_CFG_SET_NO_UPDATE(h2c, (ignore_bw) ? 1 : 0);
- MACID_CFG_SET_BW(h2c, bw);
- MACID_CFG_SET_LDPC_CAP(h2c, get_ra_ldpc(psta));
- MACID_CFG_SET_WHT_EN(h2c, get_ra_vht_en(psta->wireless_mode, mask));
- /* DisableTXPowerTraining */
- if (hal->bDisableTXPowerTraining) {
- MACID_CFG_SET_DISPT(h2c, 1);
- RTW_INFO("%s: Disable PWT by driver\n", __FUNCTION__);
- } else {
- struct PHY_DM_STRUCT *pDM_OutSrc = &hal->odmpriv;
- if (pDM_OutSrc->is_disable_power_training) {
- MACID_CFG_SET_DISPT(h2c, 1);
- RTW_INFO("%s: Disable PWT by DM\n", __FUNCTION__);
- }
- }
- MACID_CFG_SET_RATE_MASK7_0(h2c, (u8)(mask & 0x000000ff));
- MACID_CFG_SET_RATE_MASK15_8(h2c, (u8)((mask & 0x0000ff00) >> 8));
- MACID_CFG_SET_RATE_MASK23_16(h2c, (u8)((mask & 0x00ff0000) >> 16));
- MACID_CFG_SET_RATE_MASK31_24(h2c, (u8)((mask & 0xff000000) >> 24));
- RTW_INFO("%s, mask=0x%016llx, mac_id=0x%x, raid=0x%x, shortGIrate=%x, power training=%02x\n"
- , __FUNCTION__, mask, mac_id, raid, sgi, h2c[2] & BIT(6));
- #ifdef CONFIG_LPS_PG
- RTW_INFO("lps_pg_rssi_lv:%d\n", psta->lps_pg_rssi_lv);
- #endif
- RTW_DBG_DUMP("H2C-MACIDConfig Parm:", h2c, RTW_HALMAC_H2C_MAX_SIZE);
- rtw_halmac_send_h2c(adapter_to_dvobj(adapter), h2c);
- /* update initial rate */
- if (sgi)
- init_rate |= BIT(7);
- hal->INIDATA_RATE[mac_id] = init_rate;
- }
- void rtl8821c_set_FwRssiSetting_cmd(PADAPTER adapter, u8 *param)
- {
- u8 h2c[RTW_HALMAC_H2C_MAX_SIZE] = {0};
- u8 mac_id = *param;
- u8 rssi = *(param + 2);
- u8 ra_info = 0;
- RTW_INFO(FUNC_ADPT_FMT ": mac_id=%d rssi=%d param=%.2x-%.2x-%.2x\n",
- FUNC_ADPT_ARG(adapter),
- mac_id, rssi, *param, *(param + 1), *(param + 2));
- RSSI_SETTING_SET_CMD_ID(h2c, CMD_ID_RSSI_SETTING);
- RSSI_SETTING_SET_CLASS(h2c, CLASS_RSSI_SETTING);
- RSSI_SETTING_SET_MAC_ID(h2c, mac_id);
- RSSI_SETTING_SET_RSSI(h2c, rssi);
- RSSI_SETTING_SET_RA_INFO(h2c, ra_info);
- RTW_DBG_DUMP("H2C-RssiSetting Parm:", h2c, RTW_HALMAC_H2C_MAX_SIZE);
- rtw_halmac_send_h2c(adapter_to_dvobj(adapter), h2c);
- }
- void rtl8821c_set_FwAPReqRPT_cmd(PADAPTER adapter, u32 need_ack)
- {
- u8 h2c[RTW_HALMAC_H2C_MAX_SIZE] = {0};
- u8 macid1 = 1, macid2 = 0;
- RTW_INFO(FUNC_ADPT_FMT ": need_ack = %d\n",
- FUNC_ADPT_ARG(adapter), need_ack);
- AP_REQ_TXRPT_SET_CMD_ID(h2c, CMD_ID_AP_REQ_TXRPT);
- AP_REQ_TXRPT_SET_CLASS(h2c, CLASS_AP_REQ_TXRPT);
- AP_REQ_TXRPT_SET_STA1_MACID(h2c, macid1);
- AP_REQ_TXRPT_SET_STA2_MACID(h2c, macid2);
- RTW_DBG_DUMP("H2C-ApReqRpt Parm:", h2c, RTW_HALMAC_H2C_MAX_SIZE);
- rtw_halmac_send_h2c(adapter_to_dvobj(adapter), h2c);
- }
- void rtl8821c_set_FwPwrMode_cmd(PADAPTER adapter, u8 psmode)
- {
- int i;
- u8 mode = 0, smart_ps = 0;
- struct pwrctrl_priv *pwrpriv = adapter_to_pwrctl(adapter);
- struct mlme_ext_priv *pmlmeext = &adapter->mlmeextpriv;
- u8 h2c[RTW_HALMAC_H2C_MAX_SIZE] = {0};
- u8 PowerState = 0, awake_intvl = 1, rlbm = 0;
- u8 pwrmode_data5 = 0;
- #ifdef CONFIG_P2P
- struct wifidirect_info *wdinfo = &adapter->wdinfo;
- #endif /* CONFIG_P2P */
- if (pwrpriv->dtim > 0)
- RTW_INFO(FUNC_ADPT_FMT ": FW LPS mode = %d, SmartPS=%d, dtim=%d, HW port id=%d\n",
- FUNC_ADPT_ARG(adapter), psmode, pwrpriv->smart_ps, pwrpriv->dtim,
- psmode == PS_MODE_ACTIVE ? pwrpriv->current_lps_hw_port_id:get_hw_port(adapter));
- else
- RTW_INFO(FUNC_ADPT_FMT ": FW LPS mode = %d, SmartPS=%d, HW port id=%d\n",
- FUNC_ADPT_ARG(adapter), psmode, pwrpriv->smart_ps,
- psmode == PS_MODE_ACTIVE ? pwrpriv->current_lps_hw_port_id:get_hw_port(adapter));
- smart_ps = pwrpriv->smart_ps;
- switch (psmode) {
- case PS_MODE_MIN:
- mode = 1;
- rlbm = 0;
- awake_intvl = 2;
- break;
- case PS_MODE_MAX:
- mode = 1;
- rlbm = 1;
- awake_intvl = 2;
- break;
- case PS_MODE_DTIM:
- {
- mode = 1;
- rlbm = 2;
- /* For WOWLAN LPS, DTIM = (awake_intvl - 1) */
- if (pwrpriv->dtim > 0 && pwrpriv->dtim < 16) /* DTIM = (awake_intvl - 1) */
- awake_intvl = pwrpriv->dtim + 1;
- else /* DTIM = 3 */
- awake_intvl = 4;
- }
- break;
- case PS_MODE_UAPSD_WMM:
- mode = 2;
- rlbm = 0; /*1*/
- /*(WMM)smart_ps: 0:PS_Poll, 1:NullData*/
- smart_ps = (pwrpriv->smart_ps) ? 1 : 0;
- break;
- case PS_MODE_ACTIVE:
- default:
- mode = 0;
- break;
- }
- #ifdef CONFIG_P2P
- if (!rtw_p2p_chk_state(wdinfo, P2P_STATE_NONE)) {
- awake_intvl = 2;
- rlbm = 1;
- }
- #endif /* CONFIG_P2P */
- if (adapter->registrypriv.wifi_spec == 1) {
- awake_intvl = 2;
- rlbm = 1;
- }
- if (psmode > 0) {
- #ifdef CONFIG_BT_COEXIST
- if (rtw_btcoex_IsBtControlLps(adapter) == _TRUE) {
- PowerState = rtw_btcoex_RpwmVal(adapter);
- pwrmode_data5 = rtw_btcoex_LpsVal(adapter);
- if ((rlbm == 2) && (pwrmode_data5 & BIT(4))) {
- /*
- * Keep awake interval to 1 to prevent from
- * decreasing coex performance
- */
- awake_intvl = 2;
- rlbm = 2;
- }
- } else
- #endif /* CONFIG_BT_COEXIST */
- {
- PowerState = 0x00; /* AllON(0x0C), RFON(0x04), RFOFF(0x00) */
- pwrmode_data5 = 0x40;
- }
- } else {
- PowerState = 0x0C; /* AllON(0x0C), RFON(0x04), RFOFF(0x00) */
- pwrmode_data5 = 0x40;
- }
- SET_PWR_MODE_SET_CMD_ID(h2c, CMD_ID_SET_PWR_MODE);
- SET_PWR_MODE_SET_CLASS(h2c, CLASS_SET_PWR_MODE);
- SET_PWR_MODE_SET_MODE(h2c, mode);
- SET_PWR_MODE_SET_SMART_PS(h2c, smart_ps);
- SET_PWR_MODE_SET_RLBM(h2c, rlbm);
- SET_PWR_MODE_SET_AWAKE_INTERVAL(h2c, awake_intvl);
- SET_PWR_MODE_SET_B_ALL_QUEUE_UAPSD(h2c, adapter->registrypriv.uapsd_enable);
- SET_PWR_MODE_SET_PWR_STATE(h2c, PowerState);
- if (psmode == PS_MODE_ACTIVE) {
- /* Leave LPS, set the same HW port ID */
- SET_PWR_MODE_SET_PORT_ID(h2c, pwrpriv->current_lps_hw_port_id);
- } else {
- /* Enter LPS, record HW port ID */
- SET_PWR_MODE_SET_PORT_ID(h2c, get_hw_port(adapter));
- pwrpriv->current_lps_hw_port_id = get_hw_port(adapter);
- }
- if (pwrmode_data5)
- h2c[6] = pwrmode_data5;
- else {
- /* pwrmode_data5 section*/
- SET_PWR_MODE_SET_LOW_POWER_RX_BCN(h2c, 0);/*bit-16*/
- SET_PWR_MODE_SET_ANT_AUTO_SWITCH(h2c, 0);/*bit-17*/
- SET_PWR_MODE_SET_PS_ALLOW_BT_HIGH_PRIORITY(h2c, 0);/*bit-18*/
- SET_PWR_MODE_SET_PROTECT_BCN(h2c, 0);/*bit-19*/
- SET_PWR_MODE_SET_SILENCE_PERIOD(h2c, 0);/*bit-20*/
- SET_PWR_MODE_SET_FAST_BT_CONNECT(h2c, 0);/*bit-21*/
- SET_PWR_MODE_SET_TWO_ANTENNA_EN(h2c, 0);/*bit-22*/
- }
- /* pwrmode_data6 section*/
- SET_PWR_MODE_SET_ADOPT_USER_SETTING(h2c, 0);/*bit-24*/
- SET_PWR_MODE_SET_DRV_BCN_EARLY_SHIFT(h2c, 0);/*bit-25:3*/
- SET_PWR_MODE_SET_DRV_BCN_EARLY_SHIFT2(h2c, 0);/*bit-28:4*/
- RTW_INFO("%s=> psmode:%02x, smart_ps:%02x, PowerState:%02x\n", __func__, psmode, smart_ps, PowerState);
- #ifdef CONFIG_LPS_LCLK
- if (psmode != PS_MODE_ACTIVE) {
- if ((pmlmeext->adaptive_tsf_done == _FALSE)
- && (pmlmeext->bcn_cnt > 0)) {
- u8 ratio_20_delay, ratio_80_delay;
- /*
- * byte 6 for adaptive_early_32k
- * [0:3] = DrvBcnEarly (ms), [4:7] = DrvBcnTimeOut (ms)
- * 20% for DrvBcnEarly, 80% for DrvBcnTimeOut
- */
- ratio_20_delay = 0;
- ratio_80_delay = 0;
- pmlmeext->DrvBcnEarly = 0xff;
- pmlmeext->DrvBcnTimeOut = 0xff;
- for (i = 0; i < 9; i++) {
- pmlmeext->bcn_delay_ratio[i] = (pmlmeext->bcn_delay_cnt[i] * 100) / pmlmeext->bcn_cnt;
- ratio_20_delay += pmlmeext->bcn_delay_ratio[i];
- ratio_80_delay += pmlmeext->bcn_delay_ratio[i];
- if (ratio_20_delay > 20 && pmlmeext->DrvBcnEarly == 0xff)
- pmlmeext->DrvBcnEarly = i;
- if (ratio_80_delay > 80 && pmlmeext->DrvBcnTimeOut == 0xff)
- pmlmeext->DrvBcnTimeOut = i;
- /* reset adaptive_early_32k cnt */
- pmlmeext->bcn_delay_cnt[i] = 0;
- pmlmeext->bcn_delay_ratio[i] = 0;
- }
- pmlmeext->bcn_cnt = 0;
- pmlmeext->adaptive_tsf_done = _TRUE;
- }
- }
- #endif /* CONFIG_LPS_LCLK */
- #ifdef CONFIG_BT_COEXIST
- rtw_btcoex_RecordPwrMode(adapter, h2c + 1, RTW_HALMAC_H2C_MAX_SIZE - 1);
- #endif /* CONFIG_BT_COEXIST */
- RTW_DBG_DUMP("H2C-PwrMode Parm:", h2c, RTW_HALMAC_H2C_MAX_SIZE);
- rtw_halmac_send_h2c(adapter_to_dvobj(adapter), h2c);
- }
- void rtl8821c_set_FwPwrModeInIPS_cmd(PADAPTER adapter, u8 cmd_param)
- {
- u8 h2c[RTW_HALMAC_H2C_MAX_SIZE] = {0};
- INACTIVE_PS_SET_CMD_ID(h2c, CMD_ID_INACTIVE_PS);
- INACTIVE_PS_SET_CLASS(h2c, CLASS_INACTIVE_PS);
- if (cmd_param & BIT0)
- INACTIVE_PS_SET_ENABLE(h2c, 1);
- if (cmd_param & BIT1)
- INACTIVE_PS_SET_IGNORE_PS_CONDITION(h2c, 1);
- RTW_DBG_DUMP("FW_IPS Parm:", h2c, RTW_HALMAC_H2C_MAX_SIZE);
- rtw_halmac_send_h2c(adapter_to_dvobj(adapter), h2c);
- }
- #ifdef CONFIG_BT_COEXIST
- static void ConstructBeacon(PADAPTER adapter, u8 *pframe, u32 *pLength)
- {
- struct rtw_ieee80211_hdr *pwlanhdr;
- u16 *fctrl;
- u32 rate_len, pktlen;
- struct mlme_ext_priv *pmlmeext = &adapter->mlmeextpriv;
- struct mlme_ext_info *pmlmeinfo = &pmlmeext->mlmext_info;
- WLAN_BSSID_EX *cur_network = &pmlmeinfo->network;
- u8 bc_addr[] = {0xff, 0xff, 0xff, 0xff, 0xff, 0xff};
- pwlanhdr = (struct rtw_ieee80211_hdr *)pframe;
- fctrl = &(pwlanhdr->frame_ctl);
- *(fctrl) = 0;
- _rtw_memcpy(pwlanhdr->addr1, bc_addr, ETH_ALEN);
- _rtw_memcpy(pwlanhdr->addr2, adapter_mac_addr(adapter), ETH_ALEN);
- _rtw_memcpy(pwlanhdr->addr3, get_my_bssid(cur_network), ETH_ALEN);
- SetSeqNum(pwlanhdr, 0);
- set_frame_sub_type(pframe, WIFI_BEACON);
- pframe += sizeof(struct rtw_ieee80211_hdr_3addr);
- pktlen = sizeof(struct rtw_ieee80211_hdr_3addr);
- /* timestamp will be inserted by hardware */
- pframe += 8;
- pktlen += 8;
- /* beacon interval: 2 bytes */
- _rtw_memcpy(pframe, (u8 *)(rtw_get_beacon_interval_from_ie(cur_network->IEs)), 2);
- pframe += 2;
- pktlen += 2;
- /* capability info: 2 bytes */
- _rtw_memcpy(pframe, (u8 *)(rtw_get_capability_from_ie(cur_network->IEs)), 2);
- pframe += 2;
- pktlen += 2;
- if ((pmlmeinfo->state & 0x03) == WIFI_FW_AP_STATE) {
- pktlen += cur_network->IELength - sizeof(NDIS_802_11_FIXED_IEs);
- _rtw_memcpy(pframe, cur_network->IEs + sizeof(NDIS_802_11_FIXED_IEs), pktlen);
- goto _ConstructBeacon;
- }
- /* below for ad-hoc mode */
- /* SSID */
- pframe = rtw_set_ie(pframe, _SSID_IE_, cur_network->Ssid.SsidLength, cur_network->Ssid.Ssid, &pktlen);
- /* supported rates... */
- rate_len = rtw_get_rateset_len(cur_network->SupportedRates);
- pframe = rtw_set_ie(pframe, _SUPPORTEDRATES_IE_, ((rate_len > 8) ? 8 : rate_len), cur_network->SupportedRates, &pktlen);
- /* DS parameter set */
- pframe = rtw_set_ie(pframe, _DSSET_IE_, 1, (unsigned char *)&(cur_network->Configuration.DSConfig), &pktlen);
- if ((pmlmeinfo->state & 0x03) == WIFI_FW_ADHOC_STATE) {
- u32 ATIMWindow;
- /* IBSS Parameter Set... */
- ATIMWindow = 0;
- pframe = rtw_set_ie(pframe, _IBSS_PARA_IE_, 2, (unsigned char *)(&ATIMWindow), &pktlen);
- }
- /* todo: ERP IE */
- /* EXTERNDED SUPPORTED RATE */
- if (rate_len > 8)
- pframe = rtw_set_ie(pframe, _EXT_SUPPORTEDRATES_IE_, (rate_len - 8), (cur_network->SupportedRates + 8), &pktlen);
- /* todo:HT for adhoc */
- _ConstructBeacon:
- if ((pktlen + TXDESC_SIZE) > 512) {
- RTW_INFO("beacon frame too large\n");
- return;
- }
- *pLength = pktlen;
- }
- static void ConstructPSPoll(PADAPTER adapter, u8 *pframe, u32 *pLength)
- {
- struct rtw_ieee80211_hdr *pwlanhdr;
- u16 *fctrl;
- u32 pktlen;
- struct mlme_ext_priv *pmlmeext = &adapter->mlmeextpriv;
- struct mlme_ext_info *pmlmeinfo = &pmlmeext->mlmext_info;
- pwlanhdr = (struct rtw_ieee80211_hdr *)pframe;
- /* Frame control. */
- fctrl = &(pwlanhdr->frame_ctl);
- *(fctrl) = 0;
- SetPwrMgt(fctrl);
- set_frame_sub_type(pframe, WIFI_PSPOLL);
- /* AID. */
- set_duration(pframe, (pmlmeinfo->aid | 0xc000));
- /* BSSID. */
- _rtw_memcpy(pwlanhdr->addr1, get_my_bssid(&(pmlmeinfo->network)), ETH_ALEN);
- /* TA. */
- _rtw_memcpy(pwlanhdr->addr2, adapter_mac_addr(adapter), ETH_ALEN);
- *pLength = 16;
- }
- static void ConstructNullFunctionData(
- PADAPTER adapter,
- u8 *pframe,
- u32 *pLength,
- u8 *StaAddr,
- u8 bQoS,
- u8 AC,
- u8 bEosp,
- u8 bForcePowerSave)
- {
- struct rtw_ieee80211_hdr *pwlanhdr;
- u16 *fctrl;
- u32 pktlen;
- struct mlme_priv *pmlmepriv = &adapter->mlmepriv;
- struct wlan_network *cur_network = &pmlmepriv->cur_network;
- struct mlme_ext_priv *pmlmeext = &adapter->mlmeextpriv;
- struct mlme_ext_info *pmlmeinfo = &pmlmeext->mlmext_info;
- pwlanhdr = (struct rtw_ieee80211_hdr *)pframe;
- fctrl = &pwlanhdr->frame_ctl;
- *(fctrl) = 0;
- if (bForcePowerSave)
- SetPwrMgt(fctrl);
- switch (cur_network->network.InfrastructureMode) {
- case Ndis802_11Infrastructure:
- SetToDs(fctrl);
- _rtw_memcpy(pwlanhdr->addr1, get_my_bssid(&(pmlmeinfo->network)), ETH_ALEN);
- _rtw_memcpy(pwlanhdr->addr2, adapter_mac_addr(adapter), ETH_ALEN);
- _rtw_memcpy(pwlanhdr->addr3, StaAddr, ETH_ALEN);
- break;
- case Ndis802_11APMode:
- SetFrDs(fctrl);
- _rtw_memcpy(pwlanhdr->addr1, StaAddr, ETH_ALEN);
- _rtw_memcpy(pwlanhdr->addr2, get_my_bssid(&(pmlmeinfo->network)), ETH_ALEN);
- _rtw_memcpy(pwlanhdr->addr3, adapter_mac_addr(adapter), ETH_ALEN);
- break;
- case Ndis802_11IBSS:
- default:
- _rtw_memcpy(pwlanhdr->addr1, StaAddr, ETH_ALEN);
- _rtw_memcpy(pwlanhdr->addr2, adapter_mac_addr(adapter), ETH_ALEN);
- _rtw_memcpy(pwlanhdr->addr3, get_my_bssid(&(pmlmeinfo->network)), ETH_ALEN);
- break;
- }
- SetSeqNum(pwlanhdr, 0);
- if (bQoS == _TRUE) {
- struct rtw_ieee80211_hdr_3addr_qos *pwlanqoshdr;
- set_frame_sub_type(pframe, WIFI_QOS_DATA_NULL);
- pwlanqoshdr = (struct rtw_ieee80211_hdr_3addr_qos *)pframe;
- SetPriority(&pwlanqoshdr->qc, AC);
- SetEOSP(&pwlanqoshdr->qc, bEosp);
- pktlen = sizeof(struct rtw_ieee80211_hdr_3addr_qos);
- } else {
- set_frame_sub_type(pframe, WIFI_DATA_NULL);
- pktlen = sizeof(struct rtw_ieee80211_hdr_3addr);
- }
- *pLength = pktlen;
- }
- static void ConstructProbeRsp(PADAPTER adapter, u8 *pframe, u32 *pLength, u8 *StaAddr, BOOLEAN bHideSSID)
- {
- struct rtw_ieee80211_hdr *pwlanhdr;
- u16 *fctrl;
- u8 *mac, *bssid;
- u32 pktlen;
- struct mlme_ext_priv *pmlmeext = &adapter->mlmeextpriv;
- struct mlme_ext_info *pmlmeinfo = &pmlmeext->mlmext_info;
- WLAN_BSSID_EX *cur_network = &pmlmeinfo->network;
- #if defined(CONFIG_AP_MODE) && defined(CONFIG_NATIVEAP_MLME)
- u8 *pwps_ie;
- uint wps_ielen;
- struct mlme_priv *pmlmepriv = &adapter->mlmepriv;
- #endif /* CONFIG_AP_MODE && CONFIG_NATIVEAP_MLME */
- #ifdef CONFIG_P2P
- struct wifidirect_info *pwdinfo = &adapter->wdinfo;
- #ifdef CONFIG_WFD
- u32 wfdielen = 0;
- #endif /* CONFIG_WFD */
- #endif /* CONFIG_P2P */
- pwlanhdr = (struct rtw_ieee80211_hdr *)pframe;
- mac = adapter_mac_addr(adapter);
- bssid = cur_network->MacAddress;
- fctrl = &(pwlanhdr->frame_ctl);
- *(fctrl) = 0;
- _rtw_memcpy(pwlanhdr->addr1, StaAddr, ETH_ALEN);
- _rtw_memcpy(pwlanhdr->addr2, mac, ETH_ALEN);
- _rtw_memcpy(pwlanhdr->addr3, bssid, ETH_ALEN);
- RTW_INFO("%s FW Mac Addr:" MAC_FMT "\n", __FUNCTION__, MAC_ARG(mac));
- RTW_INFO("%s FW IP Addr" IP_FMT "\n", __FUNCTION__, IP_ARG(StaAddr));
- SetSeqNum(pwlanhdr, 0);
- set_frame_sub_type(fctrl, WIFI_PROBERSP);
- pktlen = sizeof(struct rtw_ieee80211_hdr_3addr);
- pframe += pktlen;
- if (cur_network->IELength > MAX_IE_SZ)
- return;
- pwps_ie = rtw_get_wps_ie(cur_network->IEs + _FIXED_IE_LENGTH_,
- cur_network->IELength - _FIXED_IE_LENGTH_, NULL, &wps_ielen);
- /* inerset & update wps_probe_resp_ie */
- if ((pmlmepriv->wps_probe_resp_ie != NULL) && pwps_ie && (wps_ielen > 0)) {
- uint wps_offset, remainder_ielen;
- u8 *premainder_ie;
- wps_offset = (uint)(pwps_ie - cur_network->IEs);
- premainder_ie = pwps_ie + wps_ielen;
- remainder_ielen = cur_network->IELength - wps_offset - wps_ielen;
- _rtw_memcpy(pframe, cur_network->IEs, wps_offset);
- pframe += wps_offset;
- pktlen += wps_offset;
- wps_ielen = (uint)pmlmepriv->wps_probe_resp_ie[1];/* to get ie data len */
- if ((wps_offset + wps_ielen + 2) <= MAX_IE_SZ) {
- _rtw_memcpy(pframe, pmlmepriv->wps_probe_resp_ie, wps_ielen + 2);
- pframe += wps_ielen + 2;
- pktlen += wps_ielen + 2;
- }
- if ((wps_offset + wps_ielen + 2 + remainder_ielen) <= MAX_IE_SZ) {
- _rtw_memcpy(pframe, premainder_ie, remainder_ielen);
- pframe += remainder_ielen;
- pktlen += remainder_ielen;
- }
- } else {
- _rtw_memcpy(pframe, cur_network->IEs, cur_network->IELength);
- pframe += cur_network->IELength;
- pktlen += cur_network->IELength;
- }
- /* retrieve SSID IE from cur_network->Ssid */
- {
- u8 *ssid_ie;
- sint ssid_ielen = 0;
- sint ssid_ielen_diff;
- u8 buf[MAX_IE_SZ];
- u8 *ies = pframe + sizeof(struct rtw_ieee80211_hdr_3addr);
- ssid_ie = rtw_get_ie(ies + _FIXED_IE_LENGTH_, _SSID_IE_, &ssid_ielen,
- (pframe - ies) - _FIXED_IE_LENGTH_);
- ssid_ielen_diff = cur_network->Ssid.SsidLength - ssid_ielen;
- if (ssid_ie && cur_network->Ssid.SsidLength) {
- uint remainder_ielen;
- u8 *remainder_ie;
- remainder_ie = ssid_ie + 2;
- remainder_ielen = (pframe - remainder_ie);
- if (remainder_ielen > MAX_IE_SZ) {
- RTW_WARN(FUNC_ADPT_FMT" remainder_ielen > MAX_IE_SZ\n", FUNC_ADPT_ARG(adapter));
- remainder_ielen = MAX_IE_SZ;
- }
- _rtw_memcpy(buf, remainder_ie, remainder_ielen);
- _rtw_memcpy(remainder_ie + ssid_ielen_diff, buf, remainder_ielen);
- *(ssid_ie + 1) = cur_network->Ssid.SsidLength;
- _rtw_memcpy(ssid_ie + 2, cur_network->Ssid.Ssid, cur_network->Ssid.SsidLength);
- pframe += ssid_ielen_diff;
- pktlen += ssid_ielen_diff;
- }
- }
- #ifdef CONFIG_P2P
- if (rtw_p2p_chk_role(pwdinfo, P2P_ROLE_GO)) {
- u32 len;
- #ifdef CONFIG_IOCTL_CFG80211
- if (adapter_wdev_data(adapter)->p2p_enabled && pwdinfo->driver_interface == DRIVER_CFG80211) {
- /* if pwdinfo->role == P2P_ROLE_DEVICE will call issue_probersp_p2p() */
- len = pmlmepriv->p2p_go_probe_resp_ie_len;
- if (pmlmepriv->p2p_go_probe_resp_ie && len > 0)
- _rtw_memcpy(pframe, pmlmepriv->p2p_go_probe_resp_ie, len);
- } else
- #endif /* CONFIG_IOCTL_CFG80211 */
- {
- len = build_probe_resp_p2p_ie(pwdinfo, pframe);
- }
- pframe += len;
- pktlen += len;
- #ifdef CONFIG_WFD
- #ifdef CONFIG_IOCTL_CFG80211
- if (_TRUE == pwdinfo->wfd_info->wfd_enable)
- #endif /* CONFIG_IOCTL_CFG80211 */
- {
- len = build_probe_resp_wfd_ie(pwdinfo, pframe, 0);
- }
- #ifdef CONFIG_IOCTL_CFG80211
- else {
- len = 0;
- if (pmlmepriv->wfd_probe_resp_ie && pmlmepriv->wfd_probe_resp_ie_len > 0) {
- len = pmlmepriv->wfd_probe_resp_ie_len;
- _rtw_memcpy(pframe, pmlmepriv->wfd_probe_resp_ie, len);
- }
- }
- #endif /* CONFIG_IOCTL_CFG80211 */
- pframe += len;
- pktlen += len;
- #endif /* CONFIG_WFD */
- }
- #endif /* CONFIG_P2P */
- *pLength = pktlen;
- }
- static void ConstructBtNullFunctionData(
- PADAPTER adapter,
- u8 *pframe,
- u32 *pLength,
- u8 *StaAddr,
- u8 bQoS,
- u8 AC,
- u8 bEosp,
- u8 bForcePowerSave)
- {
- struct rtw_ieee80211_hdr *pwlanhdr;
- u16 *fctrl;
- u32 pktlen;
- struct mlme_ext_priv *pmlmeext;
- struct mlme_ext_info *pmlmeinfo;
- u8 bssid[ETH_ALEN];
- RTW_INFO("+" FUNC_ADPT_FMT ": qos=%d eosp=%d ps=%d\n",
- FUNC_ADPT_ARG(adapter), bQoS, bEosp, bForcePowerSave);
- pwlanhdr = (struct rtw_ieee80211_hdr *)pframe;
- pmlmeext = &adapter->mlmeextpriv;
- pmlmeinfo = &pmlmeext->mlmext_info;
- if (NULL == StaAddr) {
- _rtw_memcpy(bssid, adapter_mac_addr(adapter), ETH_ALEN);
- StaAddr = bssid;
- }
- fctrl = &pwlanhdr->frame_ctl;
- *fctrl = 0;
- if (bForcePowerSave)
- SetPwrMgt(fctrl);
- SetFrDs(fctrl);
- _rtw_memcpy(pwlanhdr->addr1, StaAddr, ETH_ALEN);
- _rtw_memcpy(pwlanhdr->addr2, adapter_mac_addr(adapter), ETH_ALEN);
- _rtw_memcpy(pwlanhdr->addr3, adapter_mac_addr(adapter), ETH_ALEN);
- set_duration(pwlanhdr, 0);
- SetSeqNum(pwlanhdr, 0);
- if (bQoS == _TRUE) {
- struct rtw_ieee80211_hdr_3addr_qos *pwlanqoshdr;
- set_frame_sub_type(pframe, WIFI_QOS_DATA_NULL);
- pwlanqoshdr = (struct rtw_ieee80211_hdr_3addr_qos *)pframe;
- SetPriority(&pwlanqoshdr->qc, AC);
- SetEOSP(&pwlanqoshdr->qc, bEosp);
- pktlen = sizeof(struct rtw_ieee80211_hdr_3addr_qos);
- } else {
- set_frame_sub_type(pframe, WIFI_DATA_NULL);
- pktlen = sizeof(struct rtw_ieee80211_hdr_3addr);
- }
- *pLength = pktlen;
- }
- static void SetFwRsvdPagePkt_BTCoex(PADAPTER adapter)
- {
- PHAL_DATA_TYPE hal;
- struct xmit_frame *pcmdframe;
- struct pkt_attrib *pattrib;
- struct xmit_priv *pxmitpriv;
- struct mlme_ext_priv *pmlmeext;
- struct mlme_ext_info *pmlmeinfo;
- u32 BeaconLength = 0;
- u32 BTQosNullLength = 0;
- u8 *ReservedPagePacket;
- u8 TxDescLen, TxDescOffset;
- u8 TotalPageNum = 0, CurtPktPageNum = 0, RsvdPageNum = 0;
- u16 BufIndex, PageSize;
- u32 TotalPacketLen, MaxRsvdPageBufSize = 0;
- RSVDPAGE_LOC RsvdPageLoc;
- hal = GET_HAL_DATA(adapter);
- pxmitpriv = &adapter->xmitpriv;
- pmlmeext = &adapter->mlmeextpriv;
- pmlmeinfo = &pmlmeext->mlmext_info;
- TxDescLen = TXDESC_SIZE;
- TxDescOffset = TXDESC_OFFSET;
- PageSize = HALMAC_TX_PAGE_SIZE_8821C;
- RsvdPageNum = rtw_hal_get_txbuff_rsvd_page_num(adapter, _FALSE);
- MaxRsvdPageBufSize = RsvdPageNum * PageSize;
- pcmdframe = rtw_alloc_cmdxmitframe(pxmitpriv);
- if (pcmdframe == NULL) {
- RTW_INFO("%s: alloc ReservedPagePacket fail!\n", __FUNCTION__);
- return;
- }
- ReservedPagePacket = pcmdframe->buf_addr;
- _rtw_memset(&RsvdPageLoc, 0, sizeof(RSVDPAGE_LOC));
- /* beacon */
- BufIndex = TxDescOffset;
- ConstructBeacon(adapter, &ReservedPagePacket[BufIndex], &BeaconLength);
- /*
- * When we count the first page size, we need to reserve description size for the RSVD
- * packet, it will be filled in front of the packet in TXPKTBUF.
- */
- CurtPktPageNum = (u8)PageNum_128(TxDescLen + BeaconLength);
- /*
- * If we don't add 1 more page, the WOWLAN function has a problem.
- * Maybe it's a bug of firmware?
- */
- if (CurtPktPageNum == 1)
- CurtPktPageNum += 1;
- TotalPageNum += CurtPktPageNum;
- BufIndex += (CurtPktPageNum * PageSize);
- /* Jump to lastest page */
- if (BufIndex < (MaxRsvdPageBufSize - PageSize)) {
- BufIndex = TxDescOffset + (MaxRsvdPageBufSize - PageSize);
- TotalPageNum = RsvdPageNum - 1;
- }
- /* BT Qos null data */
- RsvdPageLoc.LocBTQosNull = TotalPageNum;
- ConstructBtNullFunctionData(
- adapter,
- &ReservedPagePacket[BufIndex],
- &BTQosNullLength,
- NULL,
- _TRUE, 0, 0, _FALSE);
- rtw_hal_fill_fake_txdesc(adapter, &ReservedPagePacket[BufIndex - TxDescLen], BTQosNullLength, _FALSE, _TRUE, _FALSE);
- CurtPktPageNum = (u8)PageNum_128(TxDescLen + BTQosNullLength);
- TotalPageNum += CurtPktPageNum;
- TotalPacketLen = BufIndex + BTQosNullLength;
- if (TotalPacketLen > MaxRsvdPageBufSize) {
- RTW_INFO(FUNC_ADPT_FMT ": ERROR: The rsvd page size is not enough!!TotalPacketLen %d, MaxRsvdPageBufSize %d\n",
- FUNC_ADPT_ARG(adapter), TotalPacketLen, MaxRsvdPageBufSize);
- goto error;
- }
- /* update attribute */
- pattrib = &pcmdframe->attrib;
- update_mgntframe_attrib(adapter, pattrib);
- pattrib->qsel = QSLT_BEACON;
- pattrib->pktlen = pattrib->last_txcmdsz = TotalPacketLen - TxDescOffset;
- #ifdef CONFIG_PCI_HCI
- dump_mgntframe(adapter, pcmdframe);
- #else /* !CONFIG_PCI_HCI */
- dump_mgntframe_and_wait(adapter, pcmdframe, 100);
- #endif /* !CONFIG_PCI_HCI */
- rtl8821c_set_FwRsvdPage_cmd(adapter, &RsvdPageLoc);
- rtl8821c_set_FwAoacRsvdPage_cmd(adapter, &RsvdPageLoc);
- return;
- error:
- rtw_free_xmitframe(pxmitpriv, pcmdframe);
- }
- void rtl8821c_download_BTCoex_AP_mode_rsvd_page(PADAPTER adapter)
- {
- PHAL_DATA_TYPE hal;
- struct mlme_ext_priv *pmlmeext;
- struct mlme_ext_info *pmlmeinfo;
- u8 bRecover = _FALSE;
- u8 bcn_valid = _FALSE;
- u8 DLBcnCount = 0;
- u32 poll = 0;
- u8 val8;
- RTW_INFO("+" FUNC_ADPT_FMT ": hw_port=%d fw_state=0x%08X\n",
- FUNC_ADPT_ARG(adapter), get_hw_port(adapter), get_fwstate(&adapter->mlmepriv));
- #ifdef CONFIG_RTW_DEBUG
- if (check_fwstate(&adapter->mlmepriv, WIFI_AP_STATE) == _FALSE) {
- RTW_INFO(FUNC_ADPT_FMT ": [WARNING] not in AP mode!!\n",
- FUNC_ADPT_ARG(adapter));
- }
- #endif /* CONFIG_RTW_DEBUG */
- hal = GET_HAL_DATA(adapter);
- pmlmeext = &adapter->mlmeextpriv;
- pmlmeinfo = &pmlmeext->mlmext_info;
- /* We should set AID, correct TSF, HW seq enable before set JoinBssReport to Fw in 88/92C. */
- rtw_write16(adapter, REG_BCN_PSR_RPT, (0xC000 | pmlmeinfo->aid));
- /* set REG_CR bit 8 */
- val8 = rtw_read8(adapter, REG_CR + 1);
- val8 |= BIT(0); /* ENSWBCN */
- rtw_write8(adapter, REG_CR + 1, val8);
- /*
- * Disable Hw protection for a time which revserd for Hw sending beacon.
- * Fix download reserved page packet fail that access collision with the protection time.
- */
- val8 = rtw_read8(adapter, REG_BCN_CTRL);
- val8 &= ~EN_BCN_FUNCTION;
- val8 |= DIS_TSF_UDT;
- rtw_write8(adapter, REG_BCN_CTRL, val8);
- /* Set FWHW_TXQ_CTRL 0x422[6]=0 to tell Hw the packet is not a real beacon frame. */
- if (hal->RegFwHwTxQCtrl & BIT(6))
- bRecover = _TRUE;
- /* To tell Hw the packet is not a real beacon frame. */
- hal->RegFwHwTxQCtrl &= ~BIT(6);
- rtw_write8(adapter, REG_FWHW_TXQ_CTRL + 2, hal->RegFwHwTxQCtrl);
- /* Clear beacon valid check bit. */
- rtw_hal_set_hwreg(adapter, HW_VAR_BCN_VALID, NULL);
- rtw_hal_set_hwreg(adapter, HW_VAR_DL_BCN_SEL, NULL);
- DLBcnCount = 0;
- poll = 0;
- do {
- SetFwRsvdPagePkt_BTCoex(adapter);
- DLBcnCount++;
- do {
- rtw_yield_os();
- /* check rsvd page download OK. */
- rtw_hal_get_hwreg(adapter, HW_VAR_BCN_VALID, &bcn_valid);
- poll++;
- } while (!bcn_valid && (poll % 10) != 0 && !RTW_CANNOT_RUN(adapter));
- } while (!bcn_valid && (DLBcnCount <= 100) && !RTW_CANNOT_RUN(adapter));
- if (_TRUE == bcn_valid) {
- struct pwrctrl_priv *pwrctl = adapter_to_pwrctl(adapter);
- pwrctl->fw_psmode_iface_id = adapter->iface_id;
- RTW_INFO(ADPT_FMT": DL RSVD page success! DLBcnCount:%d, poll:%d\n",
- ADPT_ARG(adapter), DLBcnCount, poll);
- } else {
- RTW_INFO(ADPT_FMT": DL RSVD page fail! DLBcnCount:%d, poll:%d\n",
- ADPT_ARG(adapter), DLBcnCount, poll);
- RTW_INFO(ADPT_FMT": DL RSVD page fail! bSurpriseRemoved=%s\n",
- ADPT_ARG(adapter), rtw_is_surprise_removed(adapter) ? "True" : "False");
- RTW_INFO(ADPT_FMT": DL RSVD page fail! bDriverStopped=%s\n",
- ADPT_ARG(adapter), rtw_is_drv_stopped(adapter) ? "True" : "False");
- }
- val8 = rtw_read8(adapter, REG_BCN_CTRL);
- val8 |= EN_BCN_FUNCTION;
- val8 &= ~DIS_TSF_UDT;
- rtw_write8(adapter, REG_BCN_CTRL, val8);
- /*
- * To make sure that if there exists an adapter which would like to send beacon.
- * If exists, the origianl value of 0x422[6] will be 1, we should check this to
- * prevent from setting 0x422[6] to 0 after download reserved page, or it will cause
- * the beacon cannot be sent by HW.
- */
- if (bRecover) {
- hal->RegFwHwTxQCtrl |= BIT(6);
- rtw_write8(adapter, REG_FWHW_TXQ_CTRL + 2, hal->RegFwHwTxQCtrl);
- }
- /* Clear CR[8] or beacon packet will not be send to TxBuf anymore. */
- #ifndef CONFIG_PCI_HCI
- val8 = rtw_read8(adapter, REG_CR + 1);
- val8 &= ~BIT(0); /* ~ENSWBCN*/
- rtw_write8(adapter, REG_CR + 1, val8);
- #endif /* !CONFIG_PCI_HCI */
- }
- #endif /* CONFIG_BT_COEXIST */
- #ifdef CONFIG_TSF_RESET_OFFLOAD
- /*
- * ask FW to Reset sync register at Beacon early interrupt
- */
- u8 rtl8821c_reset_tsf(_adapter *adapter, u8 reset_port)
- {
- u8 buf[2];
- u8 res = _SUCCESS;
- if (HW_PORT0 == reset_port) {
- buf[0] = 0x1;
- buf[1] = 0;
- } else {
- buf[0] = 0x0;
- buf[1] = 0x1;
- }
- rtl8821c_fillh2ccmd(adapter, H2C_RESET_TSF, 2, buf);
- return res;
- }
- #endif /* CONFIG_TSF_RESET_OFFLOAD */
- void rtl8821c_fw_update_beacon_cmd(PADAPTER adapter)
- {
- }
- /*
- * Below functions are for C2H
- */
- static void c2h_ccx_rpt(PADAPTER adapter, u8 *pdata)
- {
- #ifdef CONFIG_XMIT_ACK
- #define C2H_CCX_RPT_GET_TX_STATE(__pC2H) LE_BITS_TO_4BYTE(__pC2H + 0X04, 30, 2)
- u8 tx_state = _FALSE;
- tx_state = C2H_CCX_RPT_GET_TX_STATE(pdata);
- /* 0 means success, 1 means retry drop */
- if (tx_state == 0)
- rtw_ack_tx_done(&adapter->xmitpriv, RTW_SCTX_DONE_SUCCESS);
- else
- rtw_ack_tx_done(&adapter->xmitpriv, RTW_SCTX_DONE_CCX_PKT_FAIL);
- #endif /* CONFIG_XMIT_ACK */
- }
- /**
- * pbuf = RXDESC + c2h packet
- * length = RXDESC_SIZE + c2h packet size
- * c2h format => ID(1B) | SN(1B) | Payload
- */
- void c2h_handler_rtl8821c(PADAPTER adapter, u8 *pbuf, u16 length)
- {
- u8 c2h_id, c2h_sn;
- int c2h_len;
- u8 *pc2h_hdr;
- u8 *pc2h_data;
- u8 c2h_sub_cmd_id = 0;
- #ifdef CONFIG_WOWLAN
- struct pwrctrl_priv *pwrpriv = adapter_to_pwrctl(adapter);
- if (pwrpriv->wowlan_mode == _TRUE) {
- RTW_INFO("%s: return because wowolan_mode==TRUE! CMDID=%d\n",
- __FUNCTION__, C2H_GET_CMD_ID(pbuf + RXDESC_SIZE));
- return;
- }
- #endif /* CONFIG_WOWLAN*/
- if (pbuf == NULL)
- return;
- if (length < HALMAC_RX_DESC_SIZE_8821C) {
- RTW_INFO("%s: [ERROR] c2h length(%d) is smaller than RXDESC_SIZE(%d)!!\n",
- __func__, length, HALMAC_RX_DESC_SIZE_8821C);
- return;
- }
- pc2h_hdr = pbuf + HALMAC_RX_DESC_SIZE_8821C;
- pc2h_data = pbuf + HALMAC_RX_DESC_SIZE_8821C + 2; /* cmd ID not 0xFF original C2H have 2 bytes C2H header */
- c2h_id = C2H_GET_CMD_ID(pc2h_hdr);
- c2h_sn = C2H_GET_SEQ(pc2h_hdr);
- c2h_len = length - HALMAC_RX_DESC_SIZE_8821C - 2;
- if ((c2h_len < 0) || (c2h_len > C2H_DBG_CONTENT_MAX_LENGTH)) {
- RTW_ERR("%s: [ERROR] C2H_ID(%02x) C2H_SN(%d) warn c2h_len :%d (length:%d)\n", __func__, c2h_id, c2h_sn, c2h_len, length);
- rtw_warn_on(1);
- }
- #ifdef DBG_C2H_CONTENT
- RTW_INFO("%s "ADPT_FMT" C2H, ID=%d seq=%d len=%d\n", __func__, ADPT_ARG(adapter), c2h_id, c2h_sn, length);
- #endif
- switch (c2h_id) {
- case CMD_ID_C2H_SND_TXBF:
- /*C2HTxBeamformingHandler_8821C(adapter, pc2h_data, c2h_len);*/
- break;
- /* FW offload C2H is 0xFF cmd according to halmac function - halmac_parse_c2h_packet */
- case 0xFF:
- /* Get C2H sub cmd ID */
- c2h_sub_cmd_id = (u8)C2H_HDR_GET_C2H_SUB_CMD_ID(pc2h_hdr);
- if (c2h_sub_cmd_id == C2H_SUB_CMD_ID_CCX_RPT)
- c2h_ccx_rpt(adapter, pbuf + HALMAC_RX_DESC_SIZE_8821C + 4); /* cmd ID 0xFF new C2H have 4 bytes C2H header */
- else
- /* indicate rx desc + c2h pkt to halmac */
- if (rtw_halmac_c2h_handle(adapter_to_dvobj(adapter), pbuf, length == (-1)))
- RTW_ERR("%s "ADPT_FMT" C2H, ID=%d, SubID=%d seq=%d len=%d ,HALMAC not to handle\n",
- __func__, ADPT_ARG(adapter), c2h_id, c2h_sub_cmd_id, c2h_sn, length);
- break;
- /* others for c2h common code */
- default:
- /* shift 2 byte to remove cmd id & seq */
- c2h_handler(adapter, c2h_id, c2h_sn, c2h_len, pc2h_data);
- break;
- }
- }
- static inline u8 is_c2h_id_handle_directly(u8 c2h_id, u8 c2h_sub_cmd_id)
- {
- switch (c2h_id) {
- case CMD_ID_C2H_CCX_RPT:
- case C2H_IQK_FINISH:
- #if defined(CONFIG_TDLS) && defined(CONFIG_TDLS_CH_SW)
- case C2H_BCN_EARLY_RPT:
- case C2H_FW_CHNL_SWITCH_COMPLETE:
- #endif
- #ifdef CONFIG_BT_COEXIST
- case C2H_BT_MP_INFO:
- #endif
- #ifdef CONFIG_MCC_MODE
- case C2H_MCC:
- #endif
- return _TRUE;
- case 0xFF:
- switch (c2h_sub_cmd_id) {
- case C2H_SUB_CMD_ID_CCX_RPT:
- return _TRUE;
- default:
- return _FALSE;
- }
- default:
- return _FALSE;
- }
- }
- /*
- * pbuf = RXDESC + c2h packet
- * length = RXDESC_SIZE + c2h packet size
- */
- void c2h_pre_handler_rtl8821c(_adapter *adapter, u8 *pbuf, s32 length)
- {
- u8 c2h_id;
- u8 c2h_sub_cmd_id = 0;
- if ((length <= 0) || (!pbuf))
- return;
- c2h_id = C2H_GET_CMD_ID(pbuf + HALMAC_RX_DESC_SIZE_8821C);
- /* Get C2H sub cmd ID */
- if (c2h_id == 0xFF)
- c2h_sub_cmd_id = (u8)C2H_HDR_GET_C2H_SUB_CMD_ID(pbuf + HALMAC_RX_DESC_SIZE_8821C);
- if (is_c2h_id_handle_directly(c2h_id, c2h_sub_cmd_id))
- c2h_handler_rtl8821c(adapter, pbuf, length);
- else
- rtw_c2h_packet_wk_cmd(adapter, pbuf, length);
- }
|