1 /******************************************************************************
3 * Copyright(c) 2007 - 2011 Realtek Corporation. All rights reserved.
5 * This program is free software; you can redistribute it and/or modify it
6 * under the terms of version 2 of the GNU General Public License as
7 * published by the Free Software Foundation.
9 * This program is distributed in the hope that it will be useful, but WITHOUT
10 * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
11 * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for
14 * You should have received a copy of the GNU General Public License along with
15 * this program; if not, write to the Free Software Foundation, Inc.,
16 * 51 Franklin Street, Fifth Floor, Boston, MA 02110, USA
19 ******************************************************************************/
22 #include <drv_types.h>
24 #ifdef PLATFORM_FREEBSD
25 #include <sys/unistd.h> /* for RFHIGHPID */
29 #include <rtw_mp_phy_regdef.h>
31 #ifdef CONFIG_RTL8192C
32 #include <rtl8192c_hal.h>
34 #ifdef CONFIG_RTL8192D
35 #include <rtl8192d_hal.h>
37 #ifdef CONFIG_RTL8723A
38 #include <rtl8723a_hal.h>
42 #ifdef CONFIG_MP_INCLUDED
44 u32 read_macreg(_adapter *padapter, u32 addr, u32 sz)
51 val = rtw_read8(padapter, addr);
54 val = rtw_read16(padapter, addr);
57 val = rtw_read32(padapter, addr);
68 void write_macreg(_adapter *padapter, u32 addr, u32 val, u32 sz)
73 rtw_write8(padapter, addr, (u8)val);
76 rtw_write16(padapter, addr, (u16)val);
79 rtw_write32(padapter, addr, val);
87 u32 read_bbreg(_adapter *padapter, u32 addr, u32 bitmask)
89 return rtw_hal_read_bbreg(padapter, addr, bitmask);
92 void write_bbreg(_adapter *padapter, u32 addr, u32 bitmask, u32 val)
94 rtw_hal_write_bbreg(padapter, addr, bitmask, val);
97 u32 _read_rfreg(PADAPTER padapter, u8 rfpath, u32 addr, u32 bitmask)
99 return rtw_hal_read_rfreg(padapter, (RF_RADIO_PATH_E)rfpath, addr, bitmask);
102 void _write_rfreg(PADAPTER padapter, u8 rfpath, u32 addr, u32 bitmask, u32 val)
104 rtw_hal_write_rfreg(padapter, (RF_RADIO_PATH_E)rfpath, addr, bitmask, val);
107 u32 read_rfreg(PADAPTER padapter, u8 rfpath, u32 addr)
109 return _read_rfreg(padapter, (RF_RADIO_PATH_E)rfpath, addr, bRFRegOffsetMask);
112 void write_rfreg(PADAPTER padapter, u8 rfpath, u32 addr, u32 val)
114 _write_rfreg(padapter, (RF_RADIO_PATH_E)rfpath, addr, bRFRegOffsetMask, val);
117 static void _init_mp_priv_(struct mp_priv *pmp_priv)
119 WLAN_BSSID_EX *pnetwork;
121 _rtw_memset(pmp_priv, 0, sizeof(struct mp_priv));
123 pmp_priv->mode = MP_OFF;
125 pmp_priv->channel = 1;
126 pmp_priv->bandwidth = HT_CHANNEL_WIDTH_20;
127 pmp_priv->prime_channel_offset = HAL_PRIME_CHNL_OFFSET_DONT_CARE;
128 pmp_priv->rateidx = MPT_RATE_1M;
129 pmp_priv->txpoweridx = 0x2A;
131 pmp_priv->antenna_tx = ANTENNA_A;
132 pmp_priv->antenna_rx = ANTENNA_AB;
134 pmp_priv->check_mp_pkt = 0;
136 pmp_priv->tx_pktcount = 0;
138 pmp_priv->rx_pktcount = 0;
139 pmp_priv->rx_crcerrpktcount = 0;
141 pmp_priv->network_macaddr[0] = 0x00;
142 pmp_priv->network_macaddr[1] = 0xE0;
143 pmp_priv->network_macaddr[2] = 0x4C;
144 pmp_priv->network_macaddr[3] = 0x87;
145 pmp_priv->network_macaddr[4] = 0x66;
146 pmp_priv->network_macaddr[5] = 0x55;
148 pnetwork = &pmp_priv->mp_network.network;
149 _rtw_memcpy(pnetwork->MacAddress, pmp_priv->network_macaddr, ETH_ALEN);
151 pnetwork->Ssid.SsidLength = 8;
152 _rtw_memcpy(pnetwork->Ssid.Ssid, "mp_871x", pnetwork->Ssid.SsidLength);
155 #ifdef PLATFORM_WINDOWS
158 IN NDIS_WORK_ITEM* pwk_item,
162 _adapter* padapter =(_adapter *)cntx;
163 struct mp_priv *pmppriv=&padapter->mppriv;
164 struct mp_wi_cntx *pmp_wi_cntx=&pmppriv->wi_cntx;
166 // Execute specified action.
167 if(pmp_wi_cntx->curractfunc != NULL)
169 LARGE_INTEGER cur_time;
170 ULONGLONG start_time, end_time;
171 NdisGetCurrentSystemTime(&cur_time); // driver version
172 start_time = cur_time.QuadPart/10; // The return value is in microsecond
174 pmp_wi_cntx->curractfunc(padapter);
176 NdisGetCurrentSystemTime(&cur_time); // driver version
177 end_time = cur_time.QuadPart/10; // The return value is in microsecond
179 RT_TRACE(_module_mp_, _drv_info_,
180 ("WorkItemActType: %d, time spent: %I64d us\n",
181 pmp_wi_cntx->param.act_type, (end_time-start_time)));
184 NdisAcquireSpinLock(&(pmp_wi_cntx->mp_wi_lock));
185 pmp_wi_cntx->bmp_wi_progress= _FALSE;
186 NdisReleaseSpinLock(&(pmp_wi_cntx->mp_wi_lock));
188 if (pmp_wi_cntx->bmpdrv_unload)
190 NdisSetEvent(&(pmp_wi_cntx->mp_wi_evt));
196 static int init_mp_priv_by_os(struct mp_priv *pmp_priv)
198 struct mp_wi_cntx *pmp_wi_cntx;
200 if (pmp_priv == NULL) return _FAIL;
202 pmp_priv->rx_testcnt = 0;
203 pmp_priv->rx_testcnt1 = 0;
204 pmp_priv->rx_testcnt2 = 0;
206 pmp_priv->tx_testcnt = 0;
207 pmp_priv->tx_testcnt1 = 0;
209 pmp_wi_cntx = &pmp_priv->wi_cntx
210 pmp_wi_cntx->bmpdrv_unload = _FALSE;
211 pmp_wi_cntx->bmp_wi_progress = _FALSE;
212 pmp_wi_cntx->curractfunc = NULL;
218 #ifdef PLATFORM_LINUX
219 static int init_mp_priv_by_os(struct mp_priv *pmp_priv)
222 struct mp_xmit_frame *pmp_xmitframe;
224 if (pmp_priv == NULL) return _FAIL;
226 _rtw_init_queue(&pmp_priv->free_mp_xmitqueue);
228 pmp_priv->pallocated_mp_xmitframe_buf = NULL;
229 pmp_priv->pallocated_mp_xmitframe_buf = rtw_zmalloc(NR_MP_XMITFRAME * sizeof(struct mp_xmit_frame) + 4);
230 if (pmp_priv->pallocated_mp_xmitframe_buf == NULL) {
232 goto _exit_init_mp_priv;
235 pmp_priv->pmp_xmtframe_buf = pmp_priv->pallocated_mp_xmitframe_buf + 4 - ((uint) (pmp_priv->pallocated_mp_xmitframe_buf) & 3);
237 pmp_xmitframe = (struct mp_xmit_frame*)pmp_priv->pmp_xmtframe_buf;
239 for (i = 0; i < NR_MP_XMITFRAME; i++)
241 _rtw_init_listhead(&pmp_xmitframe->list);
242 rtw_list_insert_tail(&pmp_xmitframe->list, &pmp_priv->free_mp_xmitqueue.queue);
244 pmp_xmitframe->pkt = NULL;
245 pmp_xmitframe->frame_tag = MP_FRAMETAG;
246 pmp_xmitframe->padapter = pmp_priv->papdater;
251 pmp_priv->free_mp_xmitframe_cnt = NR_MP_XMITFRAME;
261 static void mp_init_xmit_attrib(struct mp_tx *pmptx, PADAPTER padapter)
263 struct pkt_attrib *pattrib;
264 struct tx_desc *desc;
266 // init xmitframe attribute
267 pattrib = &pmptx->attrib;
268 _rtw_memset(pattrib, 0, sizeof(struct pkt_attrib));
270 _rtw_memset(desc, 0, TXDESC_SIZE);
272 pattrib->ether_type = 0x8712;
273 //_rtw_memcpy(pattrib->src, padapter->eeprompriv.mac_addr, ETH_ALEN);
274 // _rtw_memcpy(pattrib->ta, pattrib->src, ETH_ALEN);
275 _rtw_memset(pattrib->dst, 0xFF, ETH_ALEN);
276 // pattrib->pctrl = 0;
277 // pattrib->dhcp_pkt = 0;
278 // pattrib->pktlen = 0;
279 pattrib->ack_policy = 0;
280 // pattrib->pkt_hdrlen = ETH_HLEN;
281 pattrib->hdrlen = WLAN_HDR_A3_LEN;
282 pattrib->subtype = WIFI_DATA;
283 pattrib->priority = 0;
284 pattrib->qsel = pattrib->priority;
285 // do_queue_select(padapter, pattrib);
286 pattrib->nr_frags = 1;
287 pattrib->encrypt = 0;
288 pattrib->bswenc = _FALSE;
289 pattrib->qos_en = _FALSE;
292 s32 init_mp_priv(PADAPTER padapter)
294 struct mp_priv *pmppriv = &padapter->mppriv;
296 _init_mp_priv_(pmppriv);
297 pmppriv->papdater = padapter;
299 pmppriv->tx.stop = 1;
300 mp_init_xmit_attrib(&pmppriv->tx, padapter);
302 switch (padapter->registrypriv.rf_config) {
304 pmppriv->antenna_tx = ANTENNA_A;
305 pmppriv->antenna_rx = ANTENNA_A;
309 pmppriv->antenna_tx = ANTENNA_A;
310 pmppriv->antenna_rx = ANTENNA_AB;
314 pmppriv->antenna_tx = ANTENNA_AB;
315 pmppriv->antenna_rx = ANTENNA_AB;
318 pmppriv->antenna_tx = ANTENNA_AB;
319 pmppriv->antenna_rx = ANTENNA_ABCD;
326 void free_mp_priv(struct mp_priv *pmp_priv)
328 if (pmp_priv->pallocated_mp_xmitframe_buf) {
329 rtw_mfree(pmp_priv->pallocated_mp_xmitframe_buf, 0);
330 pmp_priv->pallocated_mp_xmitframe_buf = NULL;
332 pmp_priv->pmp_xmtframe_buf = NULL;
335 #ifdef CONFIG_RTL8192C
336 #define PHY_IQCalibrate(a,b) rtl8192c_PHY_IQCalibrate(a,b)
337 #define PHY_LCCalibrate(a) rtl8192c_PHY_LCCalibrate(a)
338 #define dm_CheckTXPowerTracking(a) rtl8192c_dm_CheckTXPowerTracking(a)
339 #define PHY_SetRFPathSwitch(a,b) rtl8192c_PHY_SetRFPathSwitch(a,b)
342 #ifdef CONFIG_RTL8192D
343 #define PHY_IQCalibrate(a) rtl8192d_PHY_IQCalibrate(a)
344 #define PHY_LCCalibrate(a) rtl8192d_PHY_LCCalibrate(a)
345 #define dm_CheckTXPowerTracking(a) rtl8192d_dm_CheckTXPowerTracking(a)
346 #define PHY_SetRFPathSwitch(a,b) rtl8192d_PHY_SetRFPathSwitch(a,b)
350 MPT_InitializeAdapter(
351 IN PADAPTER pAdapter,
355 HAL_DATA_TYPE *pHalData = GET_HAL_DATA(pAdapter);
356 s32 rtStatus = _SUCCESS;
357 PMPT_CONTEXT pMptCtx = &pAdapter->mppriv.MptCtx;
360 //-------------------------------------------------------------------------
361 // HW Initialization for 8190 MPT.
362 //-------------------------------------------------------------------------
363 //-------------------------------------------------------------------------
364 // SW Initialization for 8190 MP.
365 //-------------------------------------------------------------------------
366 pMptCtx->bMptDrvUnload = _FALSE;
367 pMptCtx->bMassProdTest = _FALSE;
368 pMptCtx->bMptIndexEven = _TRUE; //default gain index is -6.0db
370 /* Init mpt event. */
372 NdisInitializeEvent( &(pMptCtx->MptWorkItemEvent) );
373 NdisAllocateSpinLock( &(pMptCtx->MptWorkItemSpinLock) );
375 PlatformInitializeWorkItem(
377 &(pMptCtx->MptWorkItem),
378 (RT_WORKITEM_CALL_BACK)MPT_WorkItemCallback,
382 pMptCtx->bMptWorkItemInProgress = _FALSE;
383 pMptCtx->CurrMptAct = NULL;
384 //-------------------------------------------------------------------------
387 // Don't accept any packets
388 rtw_write32(pAdapter, REG_RCR, 0);
390 // Accept CRC error and destination address
391 pHalData->ReceiveConfig |= (RCR_ACRC32|RCR_AAP);
392 rtw_write32(pAdapter, REG_RCR, pHalData->ReceiveConfig);
396 // If EEPROM or EFUSE is empty,we assign as RF 2T2R for MP.
397 if (pHalData->AutoloadFailFlag == TRUE)
399 pHalData->RF_Type = RF_2T2R;
402 ledsetting = rtw_read32(pAdapter, REG_LEDCFG0);
403 rtw_write32(pAdapter, REG_LEDCFG0, ledsetting & ~LED0DIS);
405 #ifdef CONFIG_RTL8192C
406 PHY_IQCalibrate(pAdapter, _FALSE);
407 dm_CheckTXPowerTracking(pAdapter); //trigger thermal meter
408 PHY_LCCalibrate(pAdapter);
411 #ifdef CONFIG_RTL8192D
412 PHY_IQCalibrate(pAdapter);
413 dm_CheckTXPowerTracking(pAdapter); //trigger thermal meter
414 PHY_LCCalibrate(pAdapter);
417 #ifdef CONFIG_PCI_HCI
418 PHY_SetRFPathSwitch(pAdapter, 1/*pHalData->bDefaultAntenna*/); //Wifi default use Main
421 #ifdef CONFIG_RTL8192C
423 if (pHalData->BoardType == BOARD_MINICARD)
424 PHY_SetRFPathSwitch(pAdapter, 1/*pHalData->bDefaultAntenna*/); //default use Main
426 if(pAdapter->HalFunc.GetInterfaceSelectionHandler(pAdapter) == INTF_SEL2_MINICARD )
427 PHY_SetRFPathSwitch(Adapter, pAdapter->MgntInfo.bDefaultAntenna); //default use Main
434 pMptCtx->backup0xc50 = (u1Byte)PHY_QueryBBReg(pAdapter, rOFDM0_XAAGCCore1, bMaskByte0);
435 pMptCtx->backup0xc58 = (u1Byte)PHY_QueryBBReg(pAdapter, rOFDM0_XBAGCCore1, bMaskByte0);
436 pMptCtx->backup0xc30 = (u1Byte)PHY_QueryBBReg(pAdapter, rOFDM0_RxDetector1, bMaskByte0);
441 /*-----------------------------------------------------------------------------
442 * Function: MPT_DeInitAdapter()
444 * Overview: Extra DeInitialization for Mass Production Test.
446 * Input: PADAPTER pAdapter
454 * 05/08/2007 MHC Create Version 0.
455 * 05/18/2007 MHC Add normal driver MPHalt code.
457 *---------------------------------------------------------------------------*/
463 PMPT_CONTEXT pMptCtx = &pAdapter->mppriv.MptCtx;
465 pMptCtx->bMptDrvUnload = _TRUE;
467 PlatformFreeWorkItem( &(pMptCtx->MptWorkItem) );
469 while(pMptCtx->bMptWorkItemInProgress)
471 if(NdisWaitEvent(&(pMptCtx->MptWorkItemEvent), 50))
476 NdisFreeSpinLock( &(pMptCtx->MptWorkItemSpinLock) );
480 static u8 mpt_ProStartTest(PADAPTER padapter)
482 PMPT_CONTEXT pMptCtx = &padapter->mppriv.MptCtx;
484 pMptCtx->bMassProdTest = _TRUE;
485 pMptCtx->bStartContTx = _FALSE;
486 pMptCtx->bCckContTx = _FALSE;
487 pMptCtx->bOfdmContTx = _FALSE;
488 pMptCtx->bSingleCarrier = _FALSE;
489 pMptCtx->bCarrierSuppression = _FALSE;
490 pMptCtx->bSingleTone = _FALSE;
498 s32 SetPowerTracking(PADAPTER padapter, u8 enable)
501 Hal_SetPowerTracking( padapter, enable );
505 void GetPowerTracking(PADAPTER padapter, u8 *enable)
507 Hal_GetPowerTracking( padapter, enable );
510 static void disable_dm(PADAPTER padapter)
512 #ifndef CONFIG_RTL8723A
515 HAL_DATA_TYPE *pHalData = GET_HAL_DATA(padapter);
516 struct dm_priv *pdmpriv = &pHalData->dmpriv;
519 //3 1. disable firmware dynamic mechanism
520 // disable Power Training, Rate Adaptive
521 #ifdef CONFIG_RTL8723A
522 SetBcnCtrlReg(padapter, 0, EN_BCN_FUNCTION);
524 v8 = rtw_read8(padapter, REG_BCN_CTRL);
525 v8 &= ~EN_BCN_FUNCTION;
526 rtw_write8(padapter, REG_BCN_CTRL, v8);
529 //3 2. disable driver dynamic mechanism
530 // disable Dynamic Initial Gain
531 // disable High Power
532 // disable Power Tracking
533 Switch_DM_Func(padapter, DYNAMIC_FUNC_DISABLE, _FALSE);
535 // enable APK, LCK and IQK but disable power tracking
536 pdmpriv->TxPowerTrackControl = _FALSE;
537 Switch_DM_Func(padapter, DYNAMIC_FUNC_SS, _TRUE);
540 //This function initializes the DUT to the MP test mode
541 s32 mp_start_test(PADAPTER padapter)
544 struct sta_info *psta;
551 struct mp_priv *pmppriv = &padapter->mppriv;
552 struct mlme_priv *pmlmepriv = &padapter->mlmepriv;
553 struct wlan_network *tgt_network = &pmlmepriv->cur_network;
556 //3 disable dynamic mechanism
557 disable_dm(padapter);
559 //3 0. update mp_priv
560 #if defined (CONFIG_RTL8192C) || defined (CONFIG_RTL8192D)
561 if (padapter->registrypriv.rf_config == RF_819X_MAX_TYPE) {
562 // HAL_DATA_TYPE *phal = GET_HAL_DATA(padapter);
563 // switch (phal->rf_type) {
564 switch (GET_RF_TYPE(padapter)) {
566 pmppriv->antenna_tx = ANTENNA_A;
567 pmppriv->antenna_rx = ANTENNA_A;
571 pmppriv->antenna_tx = ANTENNA_A;
572 pmppriv->antenna_rx = ANTENNA_AB;
576 pmppriv->antenna_tx = ANTENNA_AB;
577 pmppriv->antenna_rx = ANTENNA_AB;
580 pmppriv->antenna_tx = ANTENNA_AB;
581 pmppriv->antenna_rx = ANTENNA_ABCD;
586 mpt_ProStartTest(padapter);
588 //3 1. initialize a new WLAN_BSSID_EX
589 // _rtw_memset(&bssid, 0, sizeof(WLAN_BSSID_EX));
590 _rtw_memcpy(bssid.MacAddress, pmppriv->network_macaddr, ETH_ALEN);
591 bssid.Ssid.SsidLength = strlen("mp_pseudo_adhoc");
592 _rtw_memcpy(bssid.Ssid.Ssid, (u8*)"mp_pseudo_adhoc", bssid.Ssid.SsidLength);
593 bssid.InfrastructureMode = Ndis802_11IBSS;
594 bssid.NetworkTypeInUse = Ndis802_11DS;
597 length = get_WLAN_BSSID_EX_sz(&bssid);
599 bssid.Length = ((length >> 2) + 1) << 2; //round up to multiple of 4 bytes.
601 bssid.Length = length;
603 _enter_critical_bh(&pmlmepriv->lock, &irqL);
605 if (check_fwstate(pmlmepriv, WIFI_MP_STATE) == _TRUE)
606 goto end_of_mp_start_test;
608 //init mp_start_test status
609 if (check_fwstate(pmlmepriv, _FW_LINKED) == _TRUE) {
610 rtw_disassoc_cmd(padapter, 500, _TRUE);
611 rtw_indicate_disconnect(padapter);
612 rtw_free_assoc_resources(padapter, 1);
614 pmppriv->prev_fw_state = get_fwstate(pmlmepriv);
615 pmlmepriv->fw_state = WIFI_MP_STATE;
617 if (pmppriv->mode == _LOOPBOOK_MODE_) {
618 set_fwstate(pmlmepriv, WIFI_MP_LPBK_STATE); //append txdesc
619 RT_TRACE(_module_mp_, _drv_notice_, ("+start mp in Lookback mode\n"));
621 RT_TRACE(_module_mp_, _drv_notice_, ("+start mp in normal mode\n"));
624 set_fwstate(pmlmepriv, _FW_UNDER_LINKING);
626 //3 2. create a new psta for mp driver
627 //clear psta in the cur_network, if any
628 psta = rtw_get_stainfo(&padapter->stapriv, tgt_network->network.MacAddress);
629 if (psta) rtw_free_stainfo(padapter, psta);
631 psta = rtw_alloc_stainfo(&padapter->stapriv, bssid.MacAddress);
633 RT_TRACE(_module_mp_, _drv_err_, ("mp_start_test: Can't alloc sta_info!\n"));
634 pmlmepriv->fw_state = pmppriv->prev_fw_state;
636 goto end_of_mp_start_test;
639 //3 3. join psudo AdHoc
640 tgt_network->join_res = 1;
641 tgt_network->aid = psta->aid = 1;
642 _rtw_memcpy(&tgt_network->network, &bssid, length);
644 rtw_indicate_connect(padapter);
645 _clr_fwstate_(pmlmepriv, _FW_UNDER_LINKING);
647 end_of_mp_start_test:
649 _exit_critical_bh(&pmlmepriv->lock, &irqL);
653 // set MSR to WIFI_FW_ADHOC_STATE
654 #if defined (CONFIG_RTL8192C) || defined (CONFIG_RTL8192D)
655 val8 = rtw_read8(padapter, MSR) & 0xFC; // 0x0102
656 val8 |= WIFI_FW_ADHOC_STATE;
657 rtw_write8(padapter, MSR, val8); // Link in ad hoc network
660 #if !defined (CONFIG_RTL8192C) && !defined (CONFIG_RTL8192D)
661 rtw_write8(padapter, MSR, 1); // Link in ad hoc network
662 rtw_write8(padapter, RCR, 0); // RCR : disable all pkt, 0x10250048
663 rtw_write8(padapter, RCR+2, 0x57); // RCR disable Check BSSID, 0x1025004a
665 // disable RX filter map , mgt frames will put in RX FIFO 0
666 rtw_write16(padapter, RXFLTMAP0, 0x0); // 0x10250116
668 val8 = rtw_read8(padapter, EE_9346CR); // 0x1025000A
669 if (!(val8 & _9356SEL))//boot from EFUSE
670 efuse_change_max_size(padapter);
676 //------------------------------------------------------------------------------
677 //This function change the DUT from the MP test mode into normal mode
678 void mp_stop_test(PADAPTER padapter)
680 struct mp_priv *pmppriv = &padapter->mppriv;
681 struct mlme_priv *pmlmepriv = &padapter->mlmepriv;
682 struct wlan_network *tgt_network = &pmlmepriv->cur_network;
683 struct sta_info *psta;
688 _enter_critical_bh(&pmlmepriv->lock, &irqL);
690 if (check_fwstate(pmlmepriv, WIFI_MP_STATE) == _FALSE)
691 goto end_of_mp_stop_test;
693 //3 1. disconnect psudo AdHoc
694 rtw_indicate_disconnect(padapter);
696 //3 2. clear psta used in mp test mode.
697 // rtw_free_assoc_resources(padapter, 1);
698 psta = rtw_get_stainfo(&padapter->stapriv, tgt_network->network.MacAddress);
699 if (psta) rtw_free_stainfo(padapter, psta);
701 //3 3. return to normal state (default:station mode)
702 pmlmepriv->fw_state = pmppriv->prev_fw_state; // WIFI_STATION_STATE;
704 //flush the cur_network
705 _rtw_memset(tgt_network, 0, sizeof(struct wlan_network));
707 _clr_fwstate_(pmlmepriv, WIFI_MP_STATE);
711 _exit_critical_bh(&pmlmepriv->lock, &irqL);
713 /*---------------------------hal\rtl8192c\MPT_Phy.c---------------------------*/
715 //#ifdef CONFIG_USB_HCI
716 static VOID mpt_AdjustRFRegByRateByChan92CU(PADAPTER pAdapter, u8 RateIdx, u8 Channel, u8 BandWidthID)
720 HAL_DATA_TYPE *pHalData = GET_HAL_DATA(pAdapter);
723 if (RateIdx < MPT_RATE_6M) { // CCK rate,for 88cu
726 else if ((RateIdx >= MPT_RATE_6M) && (RateIdx <= MPT_RATE_54M)) {// OFDM rate,for 88cu
727 if ((4 == Channel) || (8 == Channel) || (12 == Channel))
729 else if ((5 == Channel) || (7 == Channel) || (13 == Channel) || (14 == Channel))
734 else if ((RateIdx >= MPT_RATE_MCS0) && (RateIdx <= MPT_RATE_MCS15)) {// MCS 20M ,for 88cu // MCS40M rate,for 88cu
736 if (HT_CHANNEL_WIDTH_20 == BandWidthID) {
737 if ((4 == Channel) || (8 == Channel))
739 else if ((5 == Channel) || (7 == Channel) || (13 == Channel) || (14 == Channel))
745 if ((4 == Channel) || (8 == Channel))
747 else if ((5 == Channel) || (7 == Channel))
754 // RT_TRACE(COMP_CMD, DBG_LOUD, ("\n mpt_AdjustRFRegByRateByChan92CU():Chan:%d Rate=%d rfReg0x26:0x%08x\n",Channel, RateIdx,rfReg0x26));
755 for (eRFPath = 0; eRFPath < pHalData->NumTotalRFPath; eRFPath++) {
756 write_rfreg(pAdapter, eRFPath, RF_SYN_G2, rfReg0x26);
760 /*-----------------------------------------------------------------------------
761 * Function: mpt_SwitchRfSetting
763 * Overview: Change RF Setting when we siwthc channel/rate/BW for MP.
765 * Input: IN PADAPTER pAdapter
773 * 01/08/2009 MHC Suggestion from SD3 Willis for 92S series.
774 * 01/09/2009 MHC Add CCK modification for 40MHZ. Suggestion from SD3.
776 *---------------------------------------------------------------------------*/
777 static void mpt_SwitchRfSetting(PADAPTER pAdapter)
779 Hal_mpt_SwitchRfSetting(pAdapter);
782 /*---------------------------hal\rtl8192c\MPT_Phy.c---------------------------*/
783 /*---------------------------hal\rtl8192c\MPT_HelperFunc.c---------------------------*/
784 static void MPT_CCKTxPowerAdjust(PADAPTER Adapter, BOOLEAN bInCH14)
786 Hal_MPT_CCKTxPowerAdjust(Adapter,bInCH14);
789 static void MPT_CCKTxPowerAdjustbyIndex(PADAPTER pAdapter, BOOLEAN beven)
791 Hal_MPT_CCKTxPowerAdjustbyIndex(pAdapter,beven);
794 /*---------------------------hal\rtl8192c\MPT_HelperFunc.c---------------------------*/
799 * Use H2C command to change channel,
800 * not only modify rf register, but also other setting need to be done.
802 void SetChannel(PADAPTER pAdapter)
804 Hal_SetChannel(pAdapter);
810 * Switch bandwitdth may change center frequency(channel)
812 void SetBandwidth(PADAPTER pAdapter)
814 Hal_SetBandwidth(pAdapter);
818 static void SetCCKTxPower(PADAPTER pAdapter, u8 *TxPower)
820 Hal_SetCCKTxPower(pAdapter,TxPower);
823 static void SetOFDMTxPower(PADAPTER pAdapter, u8 *TxPower)
825 Hal_SetOFDMTxPower(pAdapter,TxPower);
829 void SetAntenna(PADAPTER pAdapter)
831 Hal_SetAntenna(pAdapter);
834 void SetAntennaPathPower(PADAPTER pAdapter)
836 Hal_SetAntennaPathPower(pAdapter);
839 void SetTxPower(PADAPTER pAdapter)
841 Hal_SetTxPower(pAdapter);
844 void SetTxAGCOffset(PADAPTER pAdapter, u32 ulTxAGCOffset)
846 u32 TxAGCOffset_B, TxAGCOffset_C, TxAGCOffset_D,tmpAGC;
848 TxAGCOffset_B = (ulTxAGCOffset&0x000000ff);
849 TxAGCOffset_C = ((ulTxAGCOffset&0x0000ff00)>>8);
850 TxAGCOffset_D = ((ulTxAGCOffset&0x00ff0000)>>16);
852 tmpAGC = (TxAGCOffset_D<<8 | TxAGCOffset_C<<4 | TxAGCOffset_B);
853 write_bbreg(pAdapter, rFPGA0_TxGainStage,
854 (bXBTxAGC|bXCTxAGC|bXDTxAGC), tmpAGC);
857 void SetDataRate(PADAPTER pAdapter)
859 Hal_SetDataRate(pAdapter);
862 #if !defined (CONFIG_RTL8192C) && !defined (CONFIG_RTL8192D)
863 /*------------------------------Define structure----------------------------*/
864 typedef struct _R_ANTENNA_SELECT_OFDM {
871 u32 r_ant_non_ht_s1:4;
874 }R_ANTENNA_SELECT_OFDM;
876 typedef struct _R_ANTENNA_SELECT_CCK {
877 u8 r_cckrx_enable_2:2;
880 }R_ANTENNA_SELECT_CCK;
883 s32 SetThermalMeter(PADAPTER pAdapter, u8 target_ther)
885 return Hal_SetThermalMeter( pAdapter, target_ther);
888 static void TriggerRFThermalMeter(PADAPTER pAdapter)
890 Hal_TriggerRFThermalMeter(pAdapter);
893 static u8 ReadRFThermalMeter(PADAPTER pAdapter)
895 return Hal_ReadRFThermalMeter(pAdapter);
898 void GetThermalMeter(PADAPTER pAdapter, u8 *value)
900 Hal_GetThermalMeter(pAdapter,value);
903 void SetSingleCarrierTx(PADAPTER pAdapter, u8 bStart)
905 Hal_SetSingleCarrierTx(pAdapter,bStart);
908 void SetSingleToneTx(PADAPTER pAdapter, u8 bStart)
910 Hal_SetSingleToneTx(pAdapter,bStart);
913 void SetCarrierSuppressionTx(PADAPTER pAdapter, u8 bStart)
915 Hal_SetCarrierSuppressionTx(pAdapter, bStart);
918 void SetCCKContinuousTx(PADAPTER pAdapter, u8 bStart)
920 Hal_SetCCKContinuousTx(pAdapter,bStart);
923 void SetOFDMContinuousTx(PADAPTER pAdapter, u8 bStart)
925 Hal_SetOFDMContinuousTx( pAdapter, bStart);
926 }/* mpt_StartOfdmContTx */
928 void SetContinuousTx(PADAPTER pAdapter, u8 bStart)
930 Hal_SetContinuousTx(pAdapter,bStart);
933 //------------------------------------------------------------------------------
934 static void dump_mpframe(PADAPTER padapter, struct xmit_frame *pmpframe)
936 rtw_hal_mgnt_xmit(padapter, pmpframe);
939 static struct xmit_frame *alloc_mp_xmitframe(struct xmit_priv *pxmitpriv)
941 struct xmit_frame *pmpframe;
942 struct xmit_buf *pxmitbuf;
944 if ((pmpframe = rtw_alloc_xmitframe(pxmitpriv)) == NULL)
949 if ((pxmitbuf = rtw_alloc_xmitbuf(pxmitpriv)) == NULL)
951 rtw_free_xmitframe(pxmitpriv, pmpframe);
955 pmpframe->frame_tag = MP_FRAMETAG;
957 pmpframe->pxmitbuf = pxmitbuf;
959 pmpframe->buf_addr = pxmitbuf->pbuf;
961 pxmitbuf->priv_data = pmpframe;
967 static thread_return mp_xmit_packet_thread(thread_context context)
969 struct xmit_frame *pxmitframe;
971 struct mp_priv *pmp_priv;
972 struct xmit_priv *pxmitpriv;
975 pmp_priv = (struct mp_priv *)context;
976 pmptx = &pmp_priv->tx;
977 padapter = pmp_priv->papdater;
978 pxmitpriv = &(padapter->xmitpriv);
980 thread_enter("RTW_MP_THREAD");
982 DBG_871X("%s:pkTx Start\n", __func__);
984 pxmitframe = alloc_mp_xmitframe(pxmitpriv);
985 if (pxmitframe == NULL) {
987 padapter->bSurpriseRemoved ||
988 padapter->bDriverStopped) {
997 _rtw_memcpy((u8 *)(pxmitframe->buf_addr+TXDESC_OFFSET), pmptx->buf, pmptx->write_size);
998 _rtw_memcpy(&(pxmitframe->attrib), &(pmptx->attrib), sizeof(struct pkt_attrib));
1000 dump_mpframe(padapter, pxmitframe);
1003 pmp_priv->tx_pktcount++;
1006 padapter->bSurpriseRemoved ||
1007 padapter->bDriverStopped)
1009 if ((pmptx->count != 0) &&
1010 (pmptx->count == pmptx->sended))
1013 flush_signals_thread();
1017 //DBG_871X("%s:pkTx Exit\n", __func__);
1018 rtw_mfree(pmptx->pallocated_buf, pmptx->buf_size);
1019 pmptx->pallocated_buf = NULL;
1025 void fill_txdesc_for_mp(PADAPTER padapter, struct tx_desc *ptxdesc)
1027 struct mp_priv *pmp_priv = &padapter->mppriv;
1028 _rtw_memcpy(ptxdesc, &(pmp_priv->tx.desc), TXDESC_SIZE);
1031 void SetPacketTx(PADAPTER padapter)
1033 u8 *ptr, *pkt_start, *pkt_end;
1035 struct tx_desc *desc;
1036 struct rtw_ieee80211_hdr *hdr;
1039 struct pkt_attrib *pattrib;
1040 struct mp_priv *pmp_priv;
1043 pmp_priv = &padapter->mppriv;
1044 if (pmp_priv->tx.stop) return;
1045 pmp_priv->tx.sended = 0;
1046 pmp_priv->tx.stop = 0;
1047 pmp_priv->tx_pktcount = 0;
1049 //3 1. update_attrib()
1050 pattrib = &pmp_priv->tx.attrib;
1051 _rtw_memcpy(pattrib->src, padapter->eeprompriv.mac_addr, ETH_ALEN);
1052 _rtw_memcpy(pattrib->ta, pattrib->src, ETH_ALEN);
1053 _rtw_memcpy(pattrib->ra, pattrib->dst, ETH_ALEN);
1054 bmcast = IS_MCAST(pattrib->ra);
1056 pattrib->mac_id = 1;
1057 pattrib->psta = rtw_get_bcmc_stainfo(padapter);
1059 pattrib->mac_id = 0;
1060 pattrib->psta = rtw_get_stainfo(&padapter->stapriv, get_bssid(&padapter->mlmepriv));
1063 pattrib->last_txcmdsz = pattrib->hdrlen + pattrib->pktlen;
1065 //3 2. allocate xmit buffer
1066 pkt_size = pattrib->last_txcmdsz;
1068 if (pmp_priv->tx.pallocated_buf)
1069 rtw_mfree(pmp_priv->tx.pallocated_buf, pmp_priv->tx.buf_size);
1070 pmp_priv->tx.write_size = pkt_size;
1071 pmp_priv->tx.buf_size = pkt_size + XMITBUF_ALIGN_SZ;
1072 pmp_priv->tx.pallocated_buf = rtw_zmalloc(pmp_priv->tx.buf_size);
1073 if (pmp_priv->tx.pallocated_buf == NULL) {
1074 DBG_871X("%s: malloc(%d) fail!!\n", __func__, pmp_priv->tx.buf_size);
1077 pmp_priv->tx.buf = (u8 *)N_BYTE_ALIGMENT((SIZE_PTR)(pmp_priv->tx.pallocated_buf), XMITBUF_ALIGN_SZ);
1078 ptr = pmp_priv->tx.buf;
1080 desc = &(pmp_priv->tx.desc);
1081 _rtw_memset(desc, 0, TXDESC_SIZE);
1083 pkt_end = pkt_start + pkt_size;
1085 //3 3. init TX descriptor
1087 //desc->txdw0 |= cpu_to_le32(pkt_size & 0x0000FFFF); // packet size
1088 //desc->txdw0 |= cpu_to_le32(OWN | FSG | LSG);
1089 //desc->txdw0 |= cpu_to_le32(((TXDESC_SIZE + OFFSET_SZ) << OFFSET_SHT) & 0x00FF0000); //32 bytes for TX Desc
1090 //if (bmcast) desc->txdw0 |= cpu_to_le32(BMC); // broadcast packet
1093 desc->txdw1 |= cpu_to_le32(BK); // don't aggregate(AMPDU)
1094 desc->txdw1 |= cpu_to_le32((pattrib->mac_id) & 0x1F); //CAM_ID(MAC_ID)
1095 desc->txdw1 |= cpu_to_le32((pattrib->qsel << QSEL_SHT) & 0x00001F00); // Queue Select, TID
1096 desc->txdw1 |= cpu_to_le32((pattrib->raid << Rate_ID_SHT) & 0x000F0000); // Rate Adaptive ID
1100 //desc->txdw3 |= cpu_to_le32((pattrib->seqnum << SEQ_SHT) & 0xffff0000);
1103 //desc->txdw4 |= cpu_to_le32(QoS);
1104 desc->txdw4 |= cpu_to_le32(HW_SEQ_EN);
1105 desc->txdw4 |= cpu_to_le32(USERATE);
1106 desc->txdw4 |= cpu_to_le32(DISDATAFB);
1108 if( pmp_priv->preamble ){
1109 if (pmp_priv->rateidx <= MPT_RATE_54M)
1110 desc->txdw4 |= cpu_to_le32(DATA_SHORT); // CCK Short Preamble
1112 if (pmp_priv->bandwidth == HT_CHANNEL_WIDTH_40)
1113 desc->txdw4 |= cpu_to_le32(DATA_BW);
1116 desc->txdw5 |= cpu_to_le32(pmp_priv->rateidx & 0x0000001F);
1118 if( pmp_priv->preamble ){
1119 if (pmp_priv->rateidx > MPT_RATE_54M)
1120 desc->txdw5 |= cpu_to_le32(SGI); // MCS Short Guard Interval
1122 desc->txdw5 |= cpu_to_le32(0x0001FF00); // DATA/RTS Rate Fallback Limit
1124 //3 4. make wlan header, make_wlanhdr()
1125 hdr = (struct rtw_ieee80211_hdr *)pkt_start;
1126 SetFrameSubType(&hdr->frame_ctl, pattrib->subtype);
1127 _rtw_memcpy(hdr->addr1, pattrib->dst, ETH_ALEN); // DA
1128 _rtw_memcpy(hdr->addr2, pattrib->src, ETH_ALEN); // SA
1129 _rtw_memcpy(hdr->addr3, get_bssid(&padapter->mlmepriv), ETH_ALEN); // RA, BSSID
1132 ptr = pkt_start + pattrib->hdrlen;
1134 switch (pmp_priv->tx.payload) {
1152 _rtw_memset(ptr, payload, pkt_end - ptr);
1155 #ifdef PLATFORM_LINUX
1156 pmp_priv->tx.PktTxThread = kthread_run(mp_xmit_packet_thread, pmp_priv, "RTW_MP_THREAD");
1157 if (IS_ERR(pmp_priv->tx.PktTxThread))
1158 DBG_871X("Create PktTx Thread Fail !!!!!\n");
1160 #ifdef PLATFORM_FREEBSD
1164 pmp_priv->tx.PktTxThread = kproc_kthread_add(mp_xmit_packet_thread, pmp_priv,
1165 &p, &td, RFHIGHPID, 0, "MPXmitThread", "MPXmitThread");
1167 if (pmp_priv->tx.PktTxThread < 0)
1168 DBG_871X("Create PktTx Thread Fail !!!!!\n");
1173 void SetPacketRx(PADAPTER pAdapter, u8 bStartRx)
1175 HAL_DATA_TYPE *pHalData = GET_HAL_DATA(pAdapter);
1179 pHalData->ReceiveConfig = AAP | APM | AM | AB | APP_ICV | ADF | AMF | HTC_LOC_CTRL | APP_MIC | APP_PHYSTS;
1181 pHalData->ReceiveConfig |= ACRC32;
1183 rtw_write32(pAdapter, REG_RCR, pHalData->ReceiveConfig);
1185 // Accept all data frames
1186 rtw_write16(pAdapter, REG_RXFLTMAP2, 0xFFFF);
1190 rtw_write32(pAdapter, REG_RCR, 0);
1194 void ResetPhyRxPktCount(PADAPTER pAdapter)
1196 u32 i, phyrx_set = 0;
1198 for (i = 0; i <= 0xF; i++) {
1200 phyrx_set |= _RXERR_RPT_SEL(i); //select
1201 phyrx_set |= RXERR_RPT_RST; // set counter to zero
1202 rtw_write32(pAdapter, REG_RXERR_RPT, phyrx_set);
1206 static u32 GetPhyRxPktCounts(PADAPTER pAdapter, u32 selbit)
1209 u32 phyrx_set = 0, count = 0;
1211 phyrx_set = _RXERR_RPT_SEL(selbit & 0xF);
1212 rtw_write32(pAdapter, REG_RXERR_RPT, phyrx_set);
1215 count = rtw_read32(pAdapter, REG_RXERR_RPT) & RXERR_COUNTER_MASK;
1220 u32 GetPhyRxPktReceived(PADAPTER pAdapter)
1222 u32 OFDM_cnt = 0, CCK_cnt = 0, HT_cnt = 0;
1224 OFDM_cnt = GetPhyRxPktCounts(pAdapter, RXERR_TYPE_OFDM_MPDU_OK);
1225 CCK_cnt = GetPhyRxPktCounts(pAdapter, RXERR_TYPE_CCK_MPDU_OK);
1226 HT_cnt = GetPhyRxPktCounts(pAdapter, RXERR_TYPE_HT_MPDU_OK);
1228 return OFDM_cnt + CCK_cnt + HT_cnt;
1231 u32 GetPhyRxPktCRC32Error(PADAPTER pAdapter)
1233 u32 OFDM_cnt = 0, CCK_cnt = 0, HT_cnt = 0;
1235 OFDM_cnt = GetPhyRxPktCounts(pAdapter, RXERR_TYPE_OFDM_MPDU_FAIL);
1236 CCK_cnt = GetPhyRxPktCounts(pAdapter, RXERR_TYPE_CCK_MPDU_FAIL);
1237 HT_cnt = GetPhyRxPktCounts(pAdapter, RXERR_TYPE_HT_MPDU_FAIL);
1239 return OFDM_cnt + CCK_cnt + HT_cnt;
1242 //reg 0x808[9:0]: FFT data x
1243 //reg 0x808[22]: 0 --> 1 to get 1 FFT data y
1244 //reg 0x8B4[15:0]: FFT data y report
1245 static u32 GetPSDData(PADAPTER pAdapter, u32 point)
1250 psd_val = rtw_read32(pAdapter, 0x808);
1251 psd_val &= 0xFFBFFC00;
1254 rtw_write32(pAdapter, 0x808, psd_val);
1256 psd_val |= 0x00400000;
1258 rtw_write32(pAdapter, 0x808, psd_val);
1260 psd_val = rtw_read32(pAdapter, 0x8B4);
1262 psd_val &= 0x0000FFFF;
1268 * pts start_point_min stop_point_max
1269 * 128 64 64 + 128 = 192
1270 * 256 128 128 + 256 = 384
1271 * 512 256 256 + 512 = 768
1272 * 1024 512 512 + 1024 = 1536
1275 u32 mp_query_psd(PADAPTER pAdapter, u8 *data)
1277 u32 i, psd_pts=0, psd_start=0, psd_stop=0;
1280 #ifdef PLATFORM_LINUX
1281 if (!netif_running(pAdapter->pnetdev)) {
1282 RT_TRACE(_module_mp_, _drv_warning_, ("mp_query_psd: Fail! interface not opened!\n"));
1287 if (check_fwstate(&pAdapter->mlmepriv, WIFI_MP_STATE) == _FALSE) {
1288 RT_TRACE(_module_mp_, _drv_warning_, ("mp_query_psd: Fail! not in MP mode!\n"));
1292 if (strlen(data) == 0) { //default value
1297 sscanf(data, "pts=%d,start=%d,stop=%d", &psd_pts, &psd_start, &psd_stop);
1300 _rtw_memset(data, '\0', sizeof(data));
1303 while (i < psd_stop)
1306 psd_data = GetPSDData(pAdapter, i-psd_pts);
1308 psd_data = GetPSDData(pAdapter, i);
1310 sprintf(data, "%s%x ", data, psd_data);
1314 #ifdef CONFIG_LONG_DELAY_ISSUE
1320 return strlen(data)+1;