/* ************************************************************************* * Ralink Tech Inc. * 5F., No.36, Taiyuan St., Jhubei City, * Hsinchu County 302, * Taiwan, R.O.C. * * (c) Copyright 2002-2007, Ralink Technology, Inc. * * This program is free software; you can redistribute it and/or modify * * it under the terms of the GNU General Public License as published by * * the Free Software Foundation; either version 2 of the License, or * * (at your option) any later version. * * * * 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., * * 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA. * * * ************************************************************************* */ #include "rt_config.h" #ifdef RALINK_ATE #ifdef RT30xx #define ATE_BBP_REG_NUM 168 UCHAR restore_BBP[ATE_BBP_REG_NUM]={0}; #endif // RT30xx // // 802.11 MAC Header, Type:Data, Length:24bytes UCHAR TemplateFrame[24] = {0x08,0x00,0x00,0x00,0xFF,0xFF,0xFF,0xFF,0xFF,0xFF, 0x00,0xAA,0xBB,0x12,0x34,0x56,0x00,0x11,0x22,0xAA,0xBB,0xCC,0x00,0x00}; extern RTMP_RF_REGS RF2850RegTable[]; extern UCHAR NUM_OF_2850_CHNL; extern FREQUENCY_ITEM FreqItems3020[]; extern UCHAR NUM_OF_3020_CHNL; static CHAR CCKRateTable[] = {0, 1, 2, 3, 8, 9, 10, 11, -1}; /* CCK Mode. */ static CHAR OFDMRateTable[] = {0, 1, 2, 3, 4, 5, 6, 7, -1}; /* OFDM Mode. */ static CHAR HTMIXRateTable[] = {0, 1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11, 12, 13, 14, 15, -1}; /* HT Mix Mode. */ static INT TxDmaBusy( IN PRTMP_ADAPTER pAd); static INT RxDmaBusy( IN PRTMP_ADAPTER pAd); static VOID RtmpDmaEnable( IN PRTMP_ADAPTER pAd, IN INT Enable); static VOID BbpSoftReset( IN PRTMP_ADAPTER pAd); static VOID RtmpRfIoWrite( IN PRTMP_ADAPTER pAd); static INT ATESetUpFrame( IN PRTMP_ADAPTER pAd, IN UINT32 TxIdx); static INT ATETxPwrHandler( IN PRTMP_ADAPTER pAd, IN char index); static INT ATECmdHandler( IN PRTMP_ADAPTER pAd, IN PSTRING arg); #ifndef RT30xx static int CheckMCSValid( IN UCHAR Mode, IN UCHAR Mcs); #endif // RT30xx // #ifdef RT30xx static int CheckMCSValid( IN UCHAR Mode, IN UCHAR Mcs, IN BOOLEAN bRT2070); #endif // RT30xx // #ifdef RTMP_MAC_PCI static VOID ATEWriteTxWI( IN PRTMP_ADAPTER pAd, IN PTXWI_STRUC pOutTxWI, IN BOOLEAN FRAG, IN BOOLEAN CFACK, IN BOOLEAN InsTimestamp, IN BOOLEAN AMPDU, IN BOOLEAN Ack, IN BOOLEAN NSeq, // HW new a sequence. IN UCHAR BASize, IN UCHAR WCID, IN ULONG Length, IN UCHAR PID, IN UCHAR TID, IN UCHAR TxRate, IN UCHAR Txopmode, IN BOOLEAN CfAck, IN HTTRANSMIT_SETTING *pTransmit); #endif // RTMP_MAC_PCI // static VOID SetJapanFilter( IN PRTMP_ADAPTER pAd); #ifdef RALINK_28xx_QA static inline INT DO_RACFG_CMD_ATE_START( IN PRTMP_ADAPTER pAdapter, IN struct iwreq *wrq, IN struct ate_racfghdr *pRaCfg ); static inline INT DO_RACFG_CMD_ATE_STOP( IN PRTMP_ADAPTER pAdapter, IN struct iwreq *wrq, IN struct ate_racfghdr *pRaCfg ); static inline INT DO_RACFG_CMD_RF_WRITE_ALL( IN PRTMP_ADAPTER pAdapter, IN struct iwreq *wrq, IN struct ate_racfghdr *pRaCfg ); static inline INT DO_RACFG_CMD_E2PROM_READ16( IN PRTMP_ADAPTER pAdapter, IN struct iwreq *wrq, IN struct ate_racfghdr *pRaCfg ); static inline INT DO_RACFG_CMD_E2PROM_WRITE16( IN PRTMP_ADAPTER pAdapter, IN struct iwreq *wrq, IN struct ate_racfghdr *pRaCfg ); static inline INT DO_RACFG_CMD_E2PROM_READ_ALL ( IN PRTMP_ADAPTER pAdapter, IN struct iwreq *wrq, IN struct ate_racfghdr *pRaCfg ); static inline INT DO_RACFG_CMD_E2PROM_WRITE_ALL( IN PRTMP_ADAPTER pAdapter, IN struct iwreq *wrq, IN struct ate_racfghdr *pRaCfg ); static inline INT DO_RACFG_CMD_IO_READ( IN PRTMP_ADAPTER pAdapter, IN struct iwreq *wrq, IN struct ate_racfghdr *pRaCfg ); static inline INT DO_RACFG_CMD_IO_WRITE( IN PRTMP_ADAPTER pAdapter, IN struct iwreq *wrq, IN struct ate_racfghdr *pRaCfg ); static inline INT DO_RACFG_CMD_IO_READ_BULK( IN PRTMP_ADAPTER pAdapter, IN struct iwreq *wrq, IN struct ate_racfghdr *pRaCfg ); static inline INT DO_RACFG_CMD_BBP_READ8( IN PRTMP_ADAPTER pAdapter, IN struct iwreq *wrq, IN struct ate_racfghdr *pRaCfg ); static inline INT DO_RACFG_CMD_BBP_WRITE8( IN PRTMP_ADAPTER pAdapter, IN struct iwreq *wrq, IN struct ate_racfghdr *pRaCfg ); static inline INT DO_RACFG_CMD_BBP_READ_ALL( IN PRTMP_ADAPTER pAdapter, IN struct iwreq *wrq, IN struct ate_racfghdr *pRaCfg ); static inline INT DO_RACFG_CMD_GET_NOISE_LEVEL( IN PRTMP_ADAPTER pAdapter, IN struct iwreq *wrq, IN struct ate_racfghdr *pRaCfg ); static inline INT DO_RACFG_CMD_GET_COUNTER( IN PRTMP_ADAPTER pAdapter, IN struct iwreq *wrq, IN struct ate_racfghdr *pRaCfg ); static inline INT DO_RACFG_CMD_CLEAR_COUNTER( IN PRTMP_ADAPTER pAdapter, IN struct iwreq *wrq, IN struct ate_racfghdr *pRaCfg ); static inline INT DO_RACFG_CMD_TX_START( IN PRTMP_ADAPTER pAdapter, IN struct iwreq *wrq, IN struct ate_racfghdr *pRaCfg ); static inline INT DO_RACFG_CMD_GET_TX_STATUS( IN PRTMP_ADAPTER pAdapter, IN struct iwreq *wrq, IN struct ate_racfghdr *pRaCfg ); static inline INT DO_RACFG_CMD_TX_STOP( IN PRTMP_ADAPTER pAdapter, IN struct iwreq *wrq, IN struct ate_racfghdr *pRaCfg ); static inline INT DO_RACFG_CMD_RX_START( IN PRTMP_ADAPTER pAdapter, IN struct iwreq *wrq, IN struct ate_racfghdr *pRaCfg ); static inline INT DO_RACFG_CMD_RX_STOP( IN PRTMP_ADAPTER pAdapter, IN struct iwreq *wrq, IN struct ate_racfghdr *pRaCfg ); static inline INT DO_RACFG_CMD_RX_STOP( IN PRTMP_ADAPTER pAdapter, IN struct iwreq *wrq, IN struct ate_racfghdr *pRaCfg ); static inline INT DO_RACFG_CMD_ATE_START_TX_CARRIER( IN PRTMP_ADAPTER pAdapter, IN struct iwreq *wrq, IN struct ate_racfghdr *pRaCfg ); static inline INT DO_RACFG_CMD_ATE_START_TX_CONT( IN PRTMP_ADAPTER pAdapter, IN struct iwreq *wrq, IN struct ate_racfghdr *pRaCfg ); static inline INT DO_RACFG_CMD_ATE_START_TX_FRAME( IN PRTMP_ADAPTER pAdapter, IN struct iwreq *wrq, IN struct ate_racfghdr *pRaCfg ); static inline INT DO_RACFG_CMD_ATE_SET_BW( IN PRTMP_ADAPTER pAdapter, IN struct iwreq *wrq, IN struct ate_racfghdr *pRaCfg ); static inline INT DO_RACFG_CMD_ATE_SET_TX_POWER0( IN PRTMP_ADAPTER pAdapter, IN struct iwreq *wrq, IN struct ate_racfghdr *pRaCfg ); static inline INT DO_RACFG_CMD_ATE_SET_TX_POWER1( IN PRTMP_ADAPTER pAdapter, IN struct iwreq *wrq, IN struct ate_racfghdr *pRaCfg ); static inline INT DO_RACFG_CMD_ATE_SET_FREQ_OFFSET( IN PRTMP_ADAPTER pAdapter, IN struct iwreq *wrq, IN struct ate_racfghdr *pRaCfg ); static inline INT DO_RACFG_CMD_ATE_GET_STATISTICS( IN PRTMP_ADAPTER pAdapter, IN struct iwreq *wrq, IN struct ate_racfghdr *pRaCfg ); static inline INT DO_RACFG_CMD_ATE_RESET_COUNTER( IN PRTMP_ADAPTER pAdapter, IN struct iwreq *wrq, IN struct ate_racfghdr *pRaCfg ); static inline INT DO_RACFG_CMD_ATE_SEL_TX_ANTENNA( IN PRTMP_ADAPTER pAdapter, IN struct iwreq *wrq, IN struct ate_racfghdr *pRaCfg ); static inline INT DO_RACFG_CMD_ATE_SEL_RX_ANTENNA( IN PRTMP_ADAPTER pAdapter, IN struct iwreq *wrq, IN struct ate_racfghdr *pRaCfg ); static inline INT DO_RACFG_CMD_ATE_SET_PREAMBLE( IN PRTMP_ADAPTER pAdapter, IN struct iwreq *wrq, IN struct ate_racfghdr *pRaCfg ); static inline INT DO_RACFG_CMD_ATE_SET_CHANNEL( IN PRTMP_ADAPTER pAdapter, IN struct iwreq *wrq, IN struct ate_racfghdr *pRaCfg ); static inline INT DO_RACFG_CMD_ATE_SET_ADDR1( IN PRTMP_ADAPTER pAdapter, IN struct iwreq *wrq, IN struct ate_racfghdr *pRaCfg ); static inline INT DO_RACFG_CMD_ATE_SET_ADDR2( IN PRTMP_ADAPTER pAdapter, IN struct iwreq *wrq, IN struct ate_racfghdr *pRaCfg ); static inline INT DO_RACFG_CMD_ATE_SET_ADDR3( IN PRTMP_ADAPTER pAdapter, IN struct iwreq *wrq, IN struct ate_racfghdr *pRaCfg ); static inline INT DO_RACFG_CMD_ATE_SET_RATE( IN PRTMP_ADAPTER pAdapter, IN struct iwreq *wrq, IN struct ate_racfghdr *pRaCfg ); static inline INT DO_RACFG_CMD_ATE_SET_TX_FRAME_LEN( IN PRTMP_ADAPTER pAdapter, IN struct iwreq *wrq, IN struct ate_racfghdr *pRaCfg ); static inline INT DO_RACFG_CMD_ATE_SET_TX_FRAME_COUNT( IN PRTMP_ADAPTER pAdapter, IN struct iwreq *wrq, IN struct ate_racfghdr *pRaCfg ); static inline INT DO_RACFG_CMD_ATE_START_RX_FRAME( IN PRTMP_ADAPTER pAdapter, IN struct iwreq *wrq, IN struct ate_racfghdr *pRaCfg ); static inline INT DO_RACFG_CMD_ATE_E2PROM_READ_BULK( IN PRTMP_ADAPTER pAdapter, IN struct iwreq *wrq, IN struct ate_racfghdr *pRaCfg ); static inline INT DO_RACFG_CMD_ATE_E2PROM_WRITE_BULK( IN PRTMP_ADAPTER pAdapter, IN struct iwreq *wrq, IN struct ate_racfghdr *pRaCfg ); static inline INT DO_RACFG_CMD_ATE_IO_WRITE_BULK( IN PRTMP_ADAPTER pAdapter, IN struct iwreq *wrq, IN struct ate_racfghdr *pRaCfg ); static inline INT DO_RACFG_CMD_ATE_BBP_READ_BULK( IN PRTMP_ADAPTER pAdapter, IN struct iwreq *wrq, IN struct ate_racfghdr *pRaCfg ); static inline INT DO_RACFG_CMD_ATE_BBP_WRITE_BULK( IN PRTMP_ADAPTER pAdapter, IN struct iwreq *wrq, IN struct ate_racfghdr *pRaCfg ); #endif // RALINK_28xx_QA // #ifdef RTMP_MAC_PCI static INT TxDmaBusy( IN PRTMP_ADAPTER pAd) { INT result; WPDMA_GLO_CFG_STRUC GloCfg; RTMP_IO_READ32(pAd, WPDMA_GLO_CFG, &GloCfg.word); // disable DMA if (GloCfg.field.TxDMABusy) result = 1; else result = 0; return result; } static INT RxDmaBusy( IN PRTMP_ADAPTER pAd) { INT result; WPDMA_GLO_CFG_STRUC GloCfg; RTMP_IO_READ32(pAd, WPDMA_GLO_CFG, &GloCfg.word); // disable DMA if (GloCfg.field.RxDMABusy) result = 1; else result = 0; return result; } static VOID RtmpDmaEnable( IN PRTMP_ADAPTER pAd, IN INT Enable) { BOOLEAN value; ULONG WaitCnt; WPDMA_GLO_CFG_STRUC GloCfg; value = Enable > 0 ? 1 : 0; // check DMA is in busy mode. WaitCnt = 0; while (TxDmaBusy(pAd) || RxDmaBusy(pAd)) { RTMPusecDelay(10); if (WaitCnt++ > 100) break; } RTMP_IO_READ32(pAd, WPDMA_GLO_CFG, &GloCfg.word); // disable DMA GloCfg.field.EnableTxDMA = value; GloCfg.field.EnableRxDMA = value; RTMP_IO_WRITE32(pAd, WPDMA_GLO_CFG, GloCfg.word); // abort all TX rings RTMPusecDelay(5000); return; } #endif // RTMP_MAC_PCI // static VOID BbpSoftReset( IN PRTMP_ADAPTER pAd) { UCHAR BbpData = 0; // Soft reset, set BBP R21 bit0=1->0 ATE_BBP_IO_READ8_BY_REG_ID(pAd, BBP_R21, &BbpData); BbpData |= 0x00000001; //set bit0=1 ATE_BBP_IO_WRITE8_BY_REG_ID(pAd, BBP_R21, BbpData); ATE_BBP_IO_READ8_BY_REG_ID(pAd, BBP_R21, &BbpData); BbpData &= ~(0x00000001); //set bit0=0 ATE_BBP_IO_WRITE8_BY_REG_ID(pAd, BBP_R21, BbpData); return; } static VOID RtmpRfIoWrite( IN PRTMP_ADAPTER pAd) { // Set RF value 1's set R3[bit2] = [0] RTMP_RF_IO_WRITE32(pAd, pAd->LatchRfRegs.R1); RTMP_RF_IO_WRITE32(pAd, pAd->LatchRfRegs.R2); RTMP_RF_IO_WRITE32(pAd, (pAd->LatchRfRegs.R3 & (~0x04))); RTMP_RF_IO_WRITE32(pAd, pAd->LatchRfRegs.R4); RTMPusecDelay(200); // Set RF value 2's set R3[bit2] = [1] RTMP_RF_IO_WRITE32(pAd, pAd->LatchRfRegs.R1); RTMP_RF_IO_WRITE32(pAd, pAd->LatchRfRegs.R2); RTMP_RF_IO_WRITE32(pAd, (pAd->LatchRfRegs.R3 | 0x04)); RTMP_RF_IO_WRITE32(pAd, pAd->LatchRfRegs.R4); RTMPusecDelay(200); // Set RF value 3's set R3[bit2] = [0] RTMP_RF_IO_WRITE32(pAd, pAd->LatchRfRegs.R1); RTMP_RF_IO_WRITE32(pAd, pAd->LatchRfRegs.R2); RTMP_RF_IO_WRITE32(pAd, (pAd->LatchRfRegs.R3 & (~0x04))); RTMP_RF_IO_WRITE32(pAd, pAd->LatchRfRegs.R4); return; } #ifdef RT30xx static int CheckMCSValid( UCHAR Mode, UCHAR Mcs, BOOLEAN bRT2070) #endif // RT30xx // #ifndef RT30xx static int CheckMCSValid( IN UCHAR Mode, IN UCHAR Mcs) #endif // RT30xx // { INT i; PCHAR pRateTab; switch (Mode) { case 0: pRateTab = CCKRateTable; break; case 1: pRateTab = OFDMRateTable; break; case 2: case 3: #ifdef RT30xx if (bRT2070) pRateTab = OFDMRateTable; else #endif // RT30xx // pRateTab = HTMIXRateTable; break; default: ATEDBGPRINT(RT_DEBUG_ERROR, ("unrecognizable Tx Mode %d\n", Mode)); return -1; break; } i = 0; while (pRateTab[i] != -1) { if (pRateTab[i] == Mcs) return 0; i++; } return -1; } static INT ATETxPwrHandler( IN PRTMP_ADAPTER pAd, IN char index) { ULONG R; CHAR TxPower; UCHAR Bbp94 = 0; BOOLEAN bPowerReduce = FALSE; #ifdef RTMP_RF_RW_SUPPORT UCHAR RFValue; #endif // RTMP_RF_RW_SUPPORT // #ifdef RALINK_28xx_QA if ((pAd->ate.bQATxStart == TRUE) || (pAd->ate.bQARxStart == TRUE)) { /* When QA is used for Tx, pAd->ate.TxPower0/1 and real tx power are not synchronized. */ return 0; } else #endif // RALINK_28xx_QA // { TxPower = index == 0 ? pAd->ate.TxPower0 : pAd->ate.TxPower1; if (pAd->ate.Channel <= 14) { if (TxPower > 31) { // R3, R4 can't large than 31 (0x24), 31 ~ 36 used by BBP 94 R = 31; if (TxPower <= 36) Bbp94 = BBPR94_DEFAULT + (UCHAR)(TxPower - 31); } else if (TxPower < 0) { // R3, R4 can't less than 0, -1 ~ -6 used by BBP 94 R = 0; if (TxPower >= -6) Bbp94 = BBPR94_DEFAULT + TxPower; } else { // 0 ~ 31 R = (ULONG) TxPower; Bbp94 = BBPR94_DEFAULT; } ATEDBGPRINT(RT_DEBUG_TRACE, ("%s (TxPower=%d, R=%ld, BBP_R94=%d)\n", __FUNCTION__, TxPower, R, Bbp94)); } else /* 5.5 GHz */ { if (TxPower > 15) { // R3, R4 can't large than 15 (0x0F) R = 15; } else if (TxPower < 0) { // R3, R4 can't less than 0 // -1 ~ -7 ASSERT((TxPower >= -7)); R = (ULONG)(TxPower + 7); bPowerReduce = TRUE; } else { // 0 ~ 15 R = (ULONG) TxPower; } ATEDBGPRINT(RT_DEBUG_TRACE, ("%s (TxPower=%d, R=%lu)\n", __FUNCTION__, TxPower, R)); } //2008/09/10:KH adds to support 3070 ATE TX Power tunning real time<-- #ifdef RTMP_RF_RW_SUPPORT if (IS_RT30xx(pAd)) { // Set Tx Power ATE_RF_IO_READ8_BY_REG_ID(pAd, RF_R12, (PUCHAR)&RFValue); RFValue = (RFValue & 0xE0) | TxPower; ATE_RF_IO_WRITE8_BY_REG_ID(pAd, RF_R12, (UCHAR)RFValue); ATEDBGPRINT(RT_DEBUG_TRACE, ("3070 or 2070:%s (TxPower=%d, RFValue=%x)\n", __FUNCTION__, TxPower, RFValue)); } else #endif // RTMP_RF_RW_SUPPORT // { if (pAd->ate.Channel <= 14) { if (index == 0) { // shift TX power control to correct RF(R3) register bit position R = R << 9; R |= (pAd->LatchRfRegs.R3 & 0xffffc1ff); pAd->LatchRfRegs.R3 = R; } else { // shift TX power control to correct RF(R4) register bit position R = R << 6; R |= (pAd->LatchRfRegs.R4 & 0xfffff83f); pAd->LatchRfRegs.R4 = R; } } else /* 5.5GHz */ { if (bPowerReduce == FALSE) { if (index == 0) { // shift TX power control to correct RF(R3) register bit position R = (R << 10) | (1 << 9); R |= (pAd->LatchRfRegs.R3 & 0xffffc1ff); pAd->LatchRfRegs.R3 = R; } else { // shift TX power control to correct RF(R4) register bit position R = (R << 7) | (1 << 6); R |= (pAd->LatchRfRegs.R4 & 0xfffff83f); pAd->LatchRfRegs.R4 = R; } } else { if (index == 0) { // shift TX power control to correct RF(R3) register bit position R = (R << 10); R |= (pAd->LatchRfRegs.R3 & 0xffffc1ff); /* Clear bit 9 of R3 to reduce 7dB. */ pAd->LatchRfRegs.R3 = (R & (~(1 << 9))); } else { // shift TX power control to correct RF(R4) register bit position R = (R << 7); R |= (pAd->LatchRfRegs.R4 & 0xfffff83f); /* Clear bit 6 of R4 to reduce 7dB. */ pAd->LatchRfRegs.R4 = (R & (~(1 << 6))); } } } RtmpRfIoWrite(pAd); } //2008/09/10:KH adds to support 3070 ATE TX Power tunning real time--> return 0; } } /* ========================================================================== Description: Set ATE operation mode to 0. ATESTART = Start ATE Mode 1. ATESTOP = Stop ATE Mode 2. TXCONT = Continuous Transmit 3. TXCARR = Transmit Carrier 4. TXFRAME = Transmit Frames 5. RXFRAME = Receive Frames #ifdef RALINK_28xx_QA 6. TXSTOP = Stop Any Type of Transmition 7. RXSTOP = Stop Receiving Frames #endif // RALINK_28xx_QA // Return: TRUE if all parameters are OK, FALSE otherwise ========================================================================== */ #ifdef RTMP_MAC_PCI static INT ATECmdHandler( IN PRTMP_ADAPTER pAd, IN PSTRING arg) { UINT32 Value = 0; UCHAR BbpData; UINT32 MacData = 0; PTXD_STRUC pTxD; INT index; UINT i = 0, atemode = 0; PRXD_STRUC pRxD; PRTMP_TX_RING pTxRing = &pAd->TxRing[QID_AC_BE]; NDIS_STATUS Status = NDIS_STATUS_SUCCESS; #ifdef RT_BIG_ENDIAN PTXD_STRUC pDestTxD; TXD_STRUC TxD; #endif ATEDBGPRINT(RT_DEBUG_TRACE, ("===> ATECmdHandler()\n")); ATEAsicSwitchChannel(pAd); /* empty function */ AsicLockChannel(pAd, pAd->ate.Channel); RTMPusecDelay(5000); // read MAC_SYS_CTRL and backup MAC_SYS_CTRL value. RTMP_IO_READ32(pAd, MAC_SYS_CTRL, &MacData); // Default value in BBP R22 is 0x0. BbpData = 0; // clean bit4 to stop continuous Tx production test. MacData &= 0xFFFFFFEF; // Enter ATE mode and set Tx/Rx Idle if (!strcmp(arg, "ATESTART")) { ATEDBGPRINT(RT_DEBUG_TRACE, ("ATE: ATESTART\n")); #if defined(LINUX) || defined(VXWORKS) // check if we have removed the firmware if (!(ATE_ON(pAd))) { NICEraseFirmware(pAd); } #endif // defined(LINUX) || defined(VXWORKS) // atemode = pAd->ate.Mode; pAd->ate.Mode = ATE_START; RTMP_IO_WRITE32(pAd, MAC_SYS_CTRL, MacData); if (atemode == ATE_TXCARR) { // No Carrier Test set BBP R22 bit7=0, bit6=0, bit[5~0]=0x0 ATE_BBP_IO_READ8_BY_REG_ID(pAd, BBP_R22, &BbpData); BbpData &= 0xFFFFFF00; // clear bit7, bit6, bit[5~0] ATE_BBP_IO_WRITE8_BY_REG_ID(pAd, BBP_R22, BbpData); } else if (atemode == ATE_TXCARRSUPP) { // No Cont. TX set BBP R22 bit7=0 ATE_BBP_IO_READ8_BY_REG_ID(pAd, BBP_R22, &BbpData); BbpData &= ~(1 << 7); // set bit7=0 ATE_BBP_IO_WRITE8_BY_REG_ID(pAd, BBP_R22, BbpData); // No Carrier Suppression set BBP R24 bit0=0 ATE_BBP_IO_READ8_BY_REG_ID(pAd, BBP_R24, &BbpData); BbpData &= 0xFFFFFFFE; // clear bit0 ATE_BBP_IO_WRITE8_BY_REG_ID(pAd, BBP_R24, BbpData); } /* We should free some resource which was allocated when ATE_TXFRAME , ATE_STOP, and ATE_TXCONT. */ else if ((atemode & ATE_TXFRAME) || (atemode == ATE_STOP)) { PRTMP_TX_RING pTxRing = &pAd->TxRing[QID_AC_BE]; if (atemode == ATE_TXCONT) { // No Cont. TX set BBP R22 bit7=0 ATE_BBP_IO_READ8_BY_REG_ID(pAd, BBP_R22, &BbpData); BbpData &= ~(1 << 7); // set bit7=0 ATE_BBP_IO_WRITE8_BY_REG_ID(pAd, BBP_R22, BbpData); } // Abort Tx, Rx DMA. RtmpDmaEnable(pAd, 0); for (i=0; iTxRing[QID_AC_BE].Cell[i].AllocVa; #else pDestTxD = (PTXD_STRUC)pAd->TxRing[QID_AC_BE].Cell[i].AllocVa; TxD = *pDestTxD; pTxD = &TxD; RTMPDescriptorEndianChange((PUCHAR)pTxD, TYPE_TXD); #endif pTxD->DMADONE = 0; pPacket = pTxRing->Cell[i].pNdisPacket; if (pPacket) { PCI_UNMAP_SINGLE(pAd, pTxD->SDPtr0, pTxD->SDLen0, PCI_DMA_TODEVICE); RELEASE_NDIS_PACKET(pAd, pPacket, NDIS_STATUS_SUCCESS); } // Always assign pNdisPacket as NULL after clear pTxRing->Cell[i].pNdisPacket = NULL; pPacket = pTxRing->Cell[i].pNextNdisPacket; if (pPacket) { PCI_UNMAP_SINGLE(pAd, pTxD->SDPtr1, pTxD->SDLen1, PCI_DMA_TODEVICE); RELEASE_NDIS_PACKET(pAd, pPacket, NDIS_STATUS_SUCCESS); } // Always assign pNextNdisPacket as NULL after clear pTxRing->Cell[i].pNextNdisPacket = NULL; #ifdef RT_BIG_ENDIAN RTMPDescriptorEndianChange((PUCHAR)pTxD, TYPE_TXD); WriteBackToDescriptor((PUCHAR)pDestTxD, (PUCHAR)pTxD, FALSE, TYPE_TXD); #endif } // Start Tx, RX DMA RtmpDmaEnable(pAd, 1); } // reset Rx statistics. pAd->ate.LastSNR0 = 0; pAd->ate.LastSNR1 = 0; pAd->ate.LastRssi0 = 0; pAd->ate.LastRssi1 = 0; pAd->ate.LastRssi2 = 0; pAd->ate.AvgRssi0 = 0; pAd->ate.AvgRssi1 = 0; pAd->ate.AvgRssi2 = 0; pAd->ate.AvgRssi0X8 = 0; pAd->ate.AvgRssi1X8 = 0; pAd->ate.AvgRssi2X8 = 0; pAd->ate.NumOfAvgRssiSample = 0; #ifdef RALINK_28xx_QA // Tx frame pAd->ate.bQATxStart = FALSE; pAd->ate.bQARxStart = FALSE; pAd->ate.seq = 0; // counters pAd->ate.U2M = 0; pAd->ate.OtherData = 0; pAd->ate.Beacon = 0; pAd->ate.OtherCount = 0; pAd->ate.TxAc0 = 0; pAd->ate.TxAc1 = 0; pAd->ate.TxAc2 = 0; pAd->ate.TxAc3 = 0; /*pAd->ate.TxHCCA = 0;*/ pAd->ate.TxMgmt = 0; pAd->ate.RSSI0 = 0; pAd->ate.RSSI1 = 0; pAd->ate.RSSI2 = 0; pAd->ate.SNR0 = 0; pAd->ate.SNR1 = 0; // control pAd->ate.TxDoneCount = 0; // TxStatus : 0 --> task is idle, 1 --> task is running pAd->ate.TxStatus = 0; #endif // RALINK_28xx_QA // // Soft reset BBP. BbpSoftReset(pAd); #ifdef CONFIG_STA_SUPPORT /* LinkDown() has "AsicDisableSync();" and "RTMP_BBP_IO_R/W8_BY_REG_ID();" inside. */ // LinkDown(pAd, FALSE); // AsicEnableBssSync(pAd); #if defined(LINUX) || defined(VXWORKS) RTMP_OS_NETDEV_STOP_QUEUE(pAd->net_dev); #endif // defined(LINUX) || defined(VXWORKS) // /* If we skip "LinkDown()", we should disable protection to prevent from sending out RTS or CTS-to-self. */ ATEDisableAsicProtect(pAd); RTMPStationStop(pAd); #endif // CONFIG_STA_SUPPORT // /* Disable Tx */ RTMP_IO_READ32(pAd, MAC_SYS_CTRL, &Value); Value &= ~(1 << 2); RTMP_IO_WRITE32(pAd, MAC_SYS_CTRL, Value); /* Disable Rx */ RTMP_IO_READ32(pAd, MAC_SYS_CTRL, &Value); Value &= ~(1 << 3); RTMP_IO_WRITE32(pAd, MAC_SYS_CTRL, Value); } else if (!strcmp(arg, "ATESTOP")) { ATEDBGPRINT(RT_DEBUG_TRACE, ("ATE: ATESTOP\n")); ATE_BBP_IO_WRITE8_BY_REG_ID(pAd, BBP_R22, BbpData); // recover the MAC_SYS_CTRL register back RTMP_IO_WRITE32(pAd, MAC_SYS_CTRL, MacData); // disable Tx, Rx RTMP_IO_READ32(pAd, MAC_SYS_CTRL, &Value); Value &= (0xfffffff3); RTMP_IO_WRITE32(pAd, MAC_SYS_CTRL, Value); // abort Tx, RX DMA RtmpDmaEnable(pAd, 0); #ifdef LINUX pAd->ate.bFWLoading = TRUE; Status = NICLoadFirmware(pAd); if (Status != NDIS_STATUS_SUCCESS) { ATEDBGPRINT(RT_DEBUG_ERROR, ("NICLoadFirmware failed, Status[=0x%08x]\n", Status)); return FALSE; } #endif // LINUX // pAd->ate.Mode = ATE_STOP; /* Even the firmware has been loaded, we still could use ATE_BBP_IO_READ8_BY_REG_ID(). But this is not suggested. */ BbpSoftReset(pAd); RTMP_ASIC_INTERRUPT_DISABLE(pAd); NICInitializeAdapter(pAd, TRUE); /* Reinitialize Rx Ring before Rx DMA is enabled. >>>RxCoherent<<< was gone ! */ for (index = 0; index < RX_RING_SIZE; index++) { pRxD = (PRXD_STRUC) pAd->RxRing.Cell[index].AllocVa; pRxD->DDONE = 0; } // We should read EEPROM for all cases. NICReadEEPROMParameters(pAd, NULL); NICInitAsicFromEEPROM(pAd); AsicSwitchChannel(pAd, pAd->CommonCfg.Channel, FALSE); /* empty function */ AsicLockChannel(pAd, pAd->CommonCfg.Channel); /* clear garbage interrupts */ RTMP_IO_WRITE32(pAd, INT_SOURCE_CSR, 0xffffffff); /* Enable Interrupt */ RTMP_ASIC_INTERRUPT_ENABLE(pAd); /* restore RX_FILTR_CFG */ #ifdef CONFIG_STA_SUPPORT /* restore RX_FILTR_CFG due to that QA maybe set it to 0x3 */ RTMP_IO_WRITE32(pAd, RX_FILTR_CFG, STANORMAL); #endif // CONFIG_STA_SUPPORT // // Enable Tx RTMP_IO_READ32(pAd, MAC_SYS_CTRL, &Value); Value |= (1 << 2); RTMP_IO_WRITE32(pAd, MAC_SYS_CTRL, Value); // Enable Tx, Rx DMA. RtmpDmaEnable(pAd, 1); // Enable Rx RTMP_IO_READ32(pAd, MAC_SYS_CTRL, &Value); Value |= (1 << 3); RTMP_IO_WRITE32(pAd, MAC_SYS_CTRL, Value); #ifdef CONFIG_STA_SUPPORT RTMPStationStart(pAd); #endif // CONFIG_STA_SUPPORT // #if defined(LINUX) || defined(VXWORKS) RTMP_OS_NETDEV_START_QUEUE(pAd->net_dev); #endif // defined(LINUX) || defined(VXWORKS) // } else if (!strcmp(arg, "TXCARR")) { ATEDBGPRINT(RT_DEBUG_TRACE, ("ATE: TXCARR\n")); pAd->ate.Mode = ATE_TXCARR; // QA has done the following steps if it is used. if (pAd->ate.bQATxStart == FALSE) { // Soft reset BBP. BbpSoftReset(pAd); // Carrier Test set BBP R22 bit7=1, bit6=1, bit[5~0]=0x01 ATE_BBP_IO_READ8_BY_REG_ID(pAd, BBP_R22, &BbpData); BbpData &= 0xFFFFFF00; //clear bit7, bit6, bit[5~0] BbpData |= 0x000000C1; //set bit7=1, bit6=1, bit[5~0]=0x01 ATE_BBP_IO_WRITE8_BY_REG_ID(pAd, BBP_R22, BbpData); // set MAC_SYS_CTRL(0x1004) Continuous Tx Production Test (bit4) = 1 RTMP_IO_READ32(pAd, MAC_SYS_CTRL, &Value); Value = Value | 0x00000010; RTMP_IO_WRITE32(pAd, MAC_SYS_CTRL, Value); } } else if (!strcmp(arg, "TXCONT")) { if (pAd->ate.bQATxStart == TRUE) { /* set MAC_SYS_CTRL(0x1004) bit4(Continuous Tx Production Test) and bit2(MAC TX enable) back to zero. */ RTMP_IO_READ32(pAd, MAC_SYS_CTRL, &MacData); MacData &= 0xFFFFFFEB; RTMP_IO_WRITE32(pAd, MAC_SYS_CTRL, MacData); // set BBP R22 bit7=0 ATE_BBP_IO_READ8_BY_REG_ID(pAd, BBP_R22, &BbpData); BbpData &= 0xFFFFFF7F; //set bit7=0 ATE_BBP_IO_WRITE8_BY_REG_ID(pAd, BBP_R22, BbpData); } /* for TxCont mode. Step 1: Send 50 packets first then wait for a moment. Step 2: Send more 50 packet then start continue mode. */ ATEDBGPRINT(RT_DEBUG_TRACE, ("ATE: TXCONT\n")); // Step 1: send 50 packets first. pAd->ate.Mode = ATE_TXCONT; pAd->ate.TxCount = 50; /* Do it after Tx/Rx DMA is aborted. */ // pAd->ate.TxDoneCount = 0; // Soft reset BBP. BbpSoftReset(pAd); // Abort Tx, RX DMA. RtmpDmaEnable(pAd, 0); // Fix can't smooth kick { RTMP_IO_READ32(pAd, TX_DTX_IDX0 + QID_AC_BE * 0x10, &pTxRing->TxDmaIdx); pTxRing->TxSwFreeIdx = pTxRing->TxDmaIdx; pTxRing->TxCpuIdx = pTxRing->TxDmaIdx; RTMP_IO_WRITE32(pAd, TX_CTX_IDX0 + QID_AC_BE * 0x10, pTxRing->TxCpuIdx); } pAd->ate.TxDoneCount = 0; /* Only needed if we have to send some normal frames. */ SetJapanFilter(pAd); for (i = 0; (i < TX_RING_SIZE-1) && (i < pAd->ate.TxCount); i++) { PNDIS_PACKET pPacket; UINT32 TxIdx = pTxRing->TxCpuIdx; #ifndef RT_BIG_ENDIAN pTxD = (PTXD_STRUC)pTxRing->Cell[TxIdx].AllocVa; #else pDestTxD = (PTXD_STRUC)pTxRing->Cell[TxIdx].AllocVa; TxD = *pDestTxD; pTxD = &TxD; RTMPDescriptorEndianChange((PUCHAR)pTxD, TYPE_TXD); #endif // Clean current cell. pPacket = pTxRing->Cell[TxIdx].pNdisPacket; if (pPacket) { PCI_UNMAP_SINGLE(pAd, pTxD->SDPtr0, pTxD->SDLen0, PCI_DMA_TODEVICE); RELEASE_NDIS_PACKET(pAd, pPacket, NDIS_STATUS_SUCCESS); } // Always assign pNdisPacket as NULL after clear pTxRing->Cell[TxIdx].pNdisPacket = NULL; pPacket = pTxRing->Cell[TxIdx].pNextNdisPacket; if (pPacket) { PCI_UNMAP_SINGLE(pAd, pTxD->SDPtr1, pTxD->SDLen1, PCI_DMA_TODEVICE); RELEASE_NDIS_PACKET(pAd, pPacket, NDIS_STATUS_SUCCESS); } // Always assign pNextNdisPacket as NULL after clear pTxRing->Cell[TxIdx].pNextNdisPacket = NULL; #ifdef RT_BIG_ENDIAN RTMPDescriptorEndianChange((PUCHAR)pTxD, TYPE_TXD); WriteBackToDescriptor((PUCHAR)pDestTxD, (PUCHAR)pTxD, FALSE, TYPE_TXD); #endif if (ATESetUpFrame(pAd, TxIdx) != 0) break; INC_RING_INDEX(pTxRing->TxCpuIdx, TX_RING_SIZE); } // Setup frame format. ATESetUpFrame(pAd, pTxRing->TxCpuIdx); // Start Tx, RX DMA. RtmpDmaEnable(pAd, 1); // Enable Tx RTMP_IO_READ32(pAd, MAC_SYS_CTRL, &Value); Value |= (1 << 2); RTMP_IO_WRITE32(pAd, MAC_SYS_CTRL, Value); // Disable Rx RTMP_IO_READ32(pAd, MAC_SYS_CTRL, &Value); Value &= ~(1 << 3); RTMP_IO_WRITE32(pAd, MAC_SYS_CTRL, Value); #ifdef RALINK_28xx_QA if (pAd->ate.bQATxStart == TRUE) { pAd->ate.TxStatus = 1; } #endif // RALINK_28xx_QA // // kick Tx-Ring RTMP_IO_WRITE32(pAd, TX_CTX_IDX0 + QID_AC_BE * RINGREG_DIFF, pAd->TxRing[QID_AC_BE].TxCpuIdx); RTMPusecDelay(5000); // Step 2: send more 50 packets then start continue mode. // Abort Tx, RX DMA. RtmpDmaEnable(pAd, 0); // Cont. TX set BBP R22 bit7=1 ATE_BBP_IO_READ8_BY_REG_ID(pAd, BBP_R22, &BbpData); BbpData |= 0x00000080; //set bit7=1 ATE_BBP_IO_WRITE8_BY_REG_ID(pAd, BBP_R22, BbpData); pAd->ate.TxCount = 50; // Fix can't smooth kick { RTMP_IO_READ32(pAd, TX_DTX_IDX0 + QID_AC_BE * 0x10, &pTxRing->TxDmaIdx); pTxRing->TxSwFreeIdx = pTxRing->TxDmaIdx; pTxRing->TxCpuIdx = pTxRing->TxDmaIdx; RTMP_IO_WRITE32(pAd, TX_CTX_IDX0 + QID_AC_BE * 0x10, pTxRing->TxCpuIdx); } pAd->ate.TxDoneCount = 0; SetJapanFilter(pAd); for (i = 0; (i < TX_RING_SIZE-1) && (i < pAd->ate.TxCount); i++) { PNDIS_PACKET pPacket; UINT32 TxIdx = pTxRing->TxCpuIdx; #ifndef RT_BIG_ENDIAN pTxD = (PTXD_STRUC)pTxRing->Cell[TxIdx].AllocVa; #else pDestTxD = (PTXD_STRUC)pTxRing->Cell[TxIdx].AllocVa; TxD = *pDestTxD; pTxD = &TxD; RTMPDescriptorEndianChange((PUCHAR)pTxD, TYPE_TXD); #endif // clean current cell. pPacket = pTxRing->Cell[TxIdx].pNdisPacket; if (pPacket) { PCI_UNMAP_SINGLE(pAd, pTxD->SDPtr0, pTxD->SDLen0, PCI_DMA_TODEVICE); RELEASE_NDIS_PACKET(pAd, pPacket, NDIS_STATUS_SUCCESS); } // Always assign pNdisPacket as NULL after clear pTxRing->Cell[TxIdx].pNdisPacket = NULL; pPacket = pTxRing->Cell[TxIdx].pNextNdisPacket; if (pPacket) { PCI_UNMAP_SINGLE(pAd, pTxD->SDPtr1, pTxD->SDLen1, PCI_DMA_TODEVICE); RELEASE_NDIS_PACKET(pAd, pPacket, NDIS_STATUS_SUCCESS); } // Always assign pNextNdisPacket as NULL after clear pTxRing->Cell[TxIdx].pNextNdisPacket = NULL; #ifdef RT_BIG_ENDIAN RTMPDescriptorEndianChange((PUCHAR)pTxD, TYPE_TXD); WriteBackToDescriptor((PUCHAR)pDestTxD, (PUCHAR)pTxD, FALSE, TYPE_TXD); #endif if (ATESetUpFrame(pAd, TxIdx) != 0) break; INC_RING_INDEX(pTxRing->TxCpuIdx, TX_RING_SIZE); } ATESetUpFrame(pAd, pTxRing->TxCpuIdx); // Start Tx, RX DMA. RtmpDmaEnable(pAd, 1); // Enable Tx RTMP_IO_READ32(pAd, MAC_SYS_CTRL, &Value); Value |= (1 << 2); RTMP_IO_WRITE32(pAd, MAC_SYS_CTRL, Value); // Disable Rx RTMP_IO_READ32(pAd, MAC_SYS_CTRL, &Value); Value &= ~(1 << 3); RTMP_IO_WRITE32(pAd, MAC_SYS_CTRL, Value); #ifdef RALINK_28xx_QA if (pAd->ate.bQATxStart == TRUE) { pAd->ate.TxStatus = 1; } #endif // RALINK_28xx_QA // // kick Tx-Ring. RTMP_IO_WRITE32(pAd, TX_CTX_IDX0 + QID_AC_BE * RINGREG_DIFF, pAd->TxRing[QID_AC_BE].TxCpuIdx); RTMPusecDelay(500); RTMP_IO_READ32(pAd, MAC_SYS_CTRL, &MacData); MacData |= 0x00000010; RTMP_IO_WRITE32(pAd, MAC_SYS_CTRL, MacData); } else if (!strcmp(arg, "TXFRAME")) { ATEDBGPRINT(RT_DEBUG_TRACE, ("ATE: TXFRAME(Count=%d)\n", pAd->ate.TxCount)); pAd->ate.Mode |= ATE_TXFRAME; ATE_BBP_IO_WRITE8_BY_REG_ID(pAd, BBP_R22, BbpData); // Soft reset BBP. BbpSoftReset(pAd); RTMP_IO_WRITE32(pAd, MAC_SYS_CTRL, MacData); // Abort Tx, RX DMA. RtmpDmaEnable(pAd, 0); // Fix can't smooth kick { RTMP_IO_READ32(pAd, TX_DTX_IDX0 + QID_AC_BE * 0x10, &pTxRing->TxDmaIdx); pTxRing->TxSwFreeIdx = pTxRing->TxDmaIdx; pTxRing->TxCpuIdx = pTxRing->TxDmaIdx; RTMP_IO_WRITE32(pAd, TX_CTX_IDX0 + QID_AC_BE * 0x10, pTxRing->TxCpuIdx); } pAd->ate.TxDoneCount = 0; SetJapanFilter(pAd); for (i = 0; (i < TX_RING_SIZE-1) && (i < pAd->ate.TxCount); i++) { PNDIS_PACKET pPacket; UINT32 TxIdx = pTxRing->TxCpuIdx; #ifndef RT_BIG_ENDIAN pTxD = (PTXD_STRUC)pTxRing->Cell[TxIdx].AllocVa; #else pDestTxD = (PTXD_STRUC)pTxRing->Cell[TxIdx].AllocVa; TxD = *pDestTxD; pTxD = &TxD; RTMPDescriptorEndianChange((PUCHAR)pTxD, TYPE_TXD); #endif // Clean current cell. pPacket = pTxRing->Cell[TxIdx].pNdisPacket; if (pPacket) { PCI_UNMAP_SINGLE(pAd, pTxD->SDPtr0, pTxD->SDLen0, PCI_DMA_TODEVICE); RELEASE_NDIS_PACKET(pAd, pPacket, NDIS_STATUS_SUCCESS); } // Always assign pNdisPacket as NULL after clear pTxRing->Cell[TxIdx].pNdisPacket = NULL; pPacket = pTxRing->Cell[TxIdx].pNextNdisPacket; if (pPacket) { PCI_UNMAP_SINGLE(pAd, pTxD->SDPtr1, pTxD->SDLen1, PCI_DMA_TODEVICE); RELEASE_NDIS_PACKET(pAd, pPacket, NDIS_STATUS_SUCCESS); } // Always assign pNextNdisPacket as NULL after clear pTxRing->Cell[TxIdx].pNextNdisPacket = NULL; #ifdef RT_BIG_ENDIAN RTMPDescriptorEndianChange((PUCHAR)pTxD, TYPE_TXD); WriteBackToDescriptor((PUCHAR)pDestTxD, (PUCHAR)pTxD, FALSE, TYPE_TXD); #endif if (ATESetUpFrame(pAd, TxIdx) != 0) break; INC_RING_INDEX(pTxRing->TxCpuIdx, TX_RING_SIZE); } ATESetUpFrame(pAd, pTxRing->TxCpuIdx); // Start Tx, Rx DMA. RtmpDmaEnable(pAd, 1); // Enable Tx RTMP_IO_READ32(pAd, MAC_SYS_CTRL, &Value); Value |= (1 << 2); RTMP_IO_WRITE32(pAd, MAC_SYS_CTRL, Value); #ifdef RALINK_28xx_QA // add this for LoopBack mode if (pAd->ate.bQARxStart == FALSE) { // Disable Rx RTMP_IO_READ32(pAd, MAC_SYS_CTRL, &Value); Value &= ~(1 << 3); RTMP_IO_WRITE32(pAd, MAC_SYS_CTRL, Value); } if (pAd->ate.bQATxStart == TRUE) { pAd->ate.TxStatus = 1; } #else // Disable Rx RTMP_IO_READ32(pAd, MAC_SYS_CTRL, &Value); Value &= ~(1 << 3); RTMP_IO_WRITE32(pAd, MAC_SYS_CTRL, Value); #endif // RALINK_28xx_QA // RTMP_IO_READ32(pAd, TX_DTX_IDX0 + QID_AC_BE * RINGREG_DIFF, &pAd->TxRing[QID_AC_BE].TxDmaIdx); // kick Tx-Ring. RTMP_IO_WRITE32(pAd, TX_CTX_IDX0 + QID_AC_BE * RINGREG_DIFF, pAd->TxRing[QID_AC_BE].TxCpuIdx); pAd->RalinkCounters.KickTxCount++; } #ifdef RALINK_28xx_QA else if (!strcmp(arg, "TXSTOP")) { ATEDBGPRINT(RT_DEBUG_TRACE, ("ATE: TXSTOP\n")); atemode = pAd->ate.Mode; pAd->ate.Mode &= ATE_TXSTOP; pAd->ate.bQATxStart = FALSE; // pAd->ate.TxDoneCount = pAd->ate.TxCount; if (atemode == ATE_TXCARR) { // No Carrier Test set BBP R22 bit7=0, bit6=0, bit[5~0]=0x0 ATE_BBP_IO_READ8_BY_REG_ID(pAd, BBP_R22, &BbpData); BbpData &= 0xFFFFFF00; //clear bit7, bit6, bit[5~0] ATE_BBP_IO_WRITE8_BY_REG_ID(pAd, BBP_R22, BbpData); } else if (atemode == ATE_TXCARRSUPP) { // No Cont. TX set BBP R22 bit7=0 ATE_BBP_IO_READ8_BY_REG_ID(pAd, BBP_R22, &BbpData); BbpData &= ~(1 << 7); //set bit7=0 ATE_BBP_IO_WRITE8_BY_REG_ID(pAd, BBP_R22, BbpData); // No Carrier Suppression set BBP R24 bit0=0 ATE_BBP_IO_READ8_BY_REG_ID(pAd, BBP_R24, &BbpData); BbpData &= 0xFFFFFFFE; //clear bit0 ATE_BBP_IO_WRITE8_BY_REG_ID(pAd, BBP_R24, BbpData); } /* We should free some resource which was allocated when ATE_TXFRAME, ATE_STOP, and ATE_TXCONT. */ else if ((atemode & ATE_TXFRAME) || (atemode == ATE_STOP)) { PRTMP_TX_RING pTxRing = &pAd->TxRing[QID_AC_BE]; if (atemode == ATE_TXCONT) { // No Cont. TX set BBP R22 bit7=0 ATE_BBP_IO_READ8_BY_REG_ID(pAd, BBP_R22, &BbpData); BbpData &= ~(1 << 7); //set bit7=0 ATE_BBP_IO_WRITE8_BY_REG_ID(pAd, BBP_R22, BbpData); } // Abort Tx, Rx DMA. RtmpDmaEnable(pAd, 0); for (i=0; iTxRing[QID_AC_BE].Cell[i].AllocVa; #else pDestTxD = (PTXD_STRUC)pAd->TxRing[QID_AC_BE].Cell[i].AllocVa; TxD = *pDestTxD; pTxD = &TxD; RTMPDescriptorEndianChange((PUCHAR)pTxD, TYPE_TXD); #endif pTxD->DMADONE = 0; pPacket = pTxRing->Cell[i].pNdisPacket; if (pPacket) { PCI_UNMAP_SINGLE(pAd, pTxD->SDPtr0, pTxD->SDLen0, PCI_DMA_TODEVICE); RELEASE_NDIS_PACKET(pAd, pPacket, NDIS_STATUS_SUCCESS); } // Always assign pNdisPacket as NULL after clear pTxRing->Cell[i].pNdisPacket = NULL; pPacket = pTxRing->Cell[i].pNextNdisPacket; if (pPacket) { PCI_UNMAP_SINGLE(pAd, pTxD->SDPtr1, pTxD->SDLen1, PCI_DMA_TODEVICE); RELEASE_NDIS_PACKET(pAd, pPacket, NDIS_STATUS_SUCCESS); } // Always assign pNextNdisPacket as NULL after clear pTxRing->Cell[i].pNextNdisPacket = NULL; #ifdef RT_BIG_ENDIAN RTMPDescriptorEndianChange((PUCHAR)pTxD, TYPE_TXD); WriteBackToDescriptor((PUCHAR)pDestTxD, (PUCHAR)pTxD, FALSE, TYPE_TXD); #endif } // Enable Tx, Rx DMA RtmpDmaEnable(pAd, 1); } // TxStatus : 0 --> task is idle, 1 --> task is running pAd->ate.TxStatus = 0; // Soft reset BBP. BbpSoftReset(pAd); // Disable Tx RTMP_IO_READ32(pAd, MAC_SYS_CTRL, &Value); Value &= ~(1 << 2); RTMP_IO_WRITE32(pAd, MAC_SYS_CTRL, Value); } else if (!strcmp(arg, "RXSTOP")) { ATEDBGPRINT(RT_DEBUG_TRACE, ("ATE: RXSTOP\n")); atemode = pAd->ate.Mode; pAd->ate.Mode &= ATE_RXSTOP; pAd->ate.bQARxStart = FALSE; // pAd->ate.TxDoneCount = pAd->ate.TxCount; if (atemode == ATE_TXCARR) { ; } else if (atemode == ATE_TXCARRSUPP) { ; } /* We should free some resource which was allocated when ATE_TXFRAME, ATE_STOP, and ATE_TXCONT. */ else if ((atemode & ATE_TXFRAME) || (atemode == ATE_STOP)) { if (atemode == ATE_TXCONT) { ; } } // Soft reset BBP. BbpSoftReset(pAd); // Disable Rx RTMP_IO_READ32(pAd, MAC_SYS_CTRL, &Value); Value &= ~(1 << 3); RTMP_IO_WRITE32(pAd, MAC_SYS_CTRL, Value); } #endif // RALINK_28xx_QA // else if (!strcmp(arg, "RXFRAME")) { ATEDBGPRINT(RT_DEBUG_TRACE, ("ATE: RXFRAME\n")); ATE_BBP_IO_WRITE8_BY_REG_ID(pAd, BBP_R22, BbpData); RTMP_IO_WRITE32(pAd, MAC_SYS_CTRL, MacData); pAd->ate.Mode |= ATE_RXFRAME; // Disable Tx of MAC block. RTMP_IO_READ32(pAd, MAC_SYS_CTRL, &Value); Value &= ~(1 << 2); RTMP_IO_WRITE32(pAd, MAC_SYS_CTRL, Value); // Enable Rx of MAC block. RTMP_IO_READ32(pAd, MAC_SYS_CTRL, &Value); Value |= (1 << 3); RTMP_IO_WRITE32(pAd, MAC_SYS_CTRL, Value); } else { ATEDBGPRINT(RT_DEBUG_TRACE, ("ATE: Invalid arg!\n")); return FALSE; } RTMPusecDelay(5000); ATEDBGPRINT(RT_DEBUG_TRACE, ("<=== ATECmdHandler()\n")); return TRUE; } /*=======================End of RTMP_MAC_PCI =======================*/ #endif // RTMP_MAC_PCI // INT Set_ATE_Proc( IN PRTMP_ADAPTER pAd, IN PSTRING arg) { if (ATECmdHandler(pAd, arg)) { ATEDBGPRINT(RT_DEBUG_TRACE, ("Ralink: Set_ATE_Proc Success\n")); return TRUE; } else { ATEDBGPRINT(RT_DEBUG_TRACE, ("Ralink: Set_ATE_Proc Failed\n")); return FALSE; } } /* ========================================================================== Description: Set ATE ADDR1=DA for TxFrame(AP : To DS = 0 ; From DS = 1) or Set ATE ADDR3=DA for TxFrame(STA : To DS = 1 ; From DS = 0) Return: TRUE if all parameters are OK, FALSE otherwise ========================================================================== */ INT Set_ATE_DA_Proc( IN PRTMP_ADAPTER pAd, IN PSTRING arg) { PSTRING value; INT i; // Mac address acceptable format 01:02:03:04:05:06 length 17 if (strlen(arg) != 17) return FALSE; for (i = 0, value = rstrtok(arg, ":"); value; value = rstrtok(NULL, ":")) { /* sanity check */ if ((strlen(value) != 2) || (!isxdigit(*value)) || (!isxdigit(*(value+1)))) { return FALSE; } #ifdef CONFIG_STA_SUPPORT AtoH(value, &pAd->ate.Addr3[i++], 1); #endif // CONFIG_STA_SUPPORT // } /* sanity check */ if (i != 6) { return FALSE; } #ifdef CONFIG_STA_SUPPORT ATEDBGPRINT(RT_DEBUG_TRACE, ("Set_ATE_DA_Proc (DA = %2X:%2X:%2X:%2X:%2X:%2X)\n", pAd->ate.Addr3[0], pAd->ate.Addr3[1], pAd->ate.Addr3[2], pAd->ate.Addr3[3], pAd->ate.Addr3[4], pAd->ate.Addr3[5])); #endif // CONFIG_STA_SUPPORT // ATEDBGPRINT(RT_DEBUG_TRACE, ("Ralink: Set_ATE_DA_Proc Success\n")); return TRUE; } /* ========================================================================== Description: Set ATE ADDR3=SA for TxFrame(AP : To DS = 0 ; From DS = 1) or Set ATE ADDR2=SA for TxFrame(STA : To DS = 1 ; From DS = 0) Return: TRUE if all parameters are OK, FALSE otherwise ========================================================================== */ INT Set_ATE_SA_Proc( IN PRTMP_ADAPTER pAd, IN PSTRING arg) { PSTRING value; INT i; // Mac address acceptable format 01:02:03:04:05:06 length 17 if (strlen(arg) != 17) return FALSE; for (i=0, value = rstrtok(arg, ":"); value; value = rstrtok(NULL, ":")) { /* sanity check */ if ((strlen(value) != 2) || (!isxdigit(*value)) || (!isxdigit(*(value+1)))) { return FALSE; } #ifdef CONFIG_STA_SUPPORT AtoH(value, &pAd->ate.Addr2[i++], 1); #endif // CONFIG_STA_SUPPORT // } /* sanity check */ if (i != 6) { return FALSE; } #ifdef CONFIG_STA_SUPPORT ATEDBGPRINT(RT_DEBUG_TRACE, ("Set_ATE_SA_Proc (SA = %2X:%2X:%2X:%2X:%2X:%2X)\n", pAd->ate.Addr2[0], pAd->ate.Addr2[1], pAd->ate.Addr2[2], pAd->ate.Addr2[3], pAd->ate.Addr2[4], pAd->ate.Addr2[5])); #endif // CONFIG_STA_SUPPORT // ATEDBGPRINT(RT_DEBUG_TRACE, ("Ralink: Set_ATE_SA_Proc Success\n")); return TRUE; } /* ========================================================================== Description: Set ATE ADDR2=BSSID for TxFrame(AP : To DS = 0 ; From DS = 1) or Set ATE ADDR1=BSSID for TxFrame(STA : To DS = 1 ; From DS = 0) Return: TRUE if all parameters are OK, FALSE otherwise ========================================================================== */ INT Set_ATE_BSSID_Proc( IN PRTMP_ADAPTER pAd, IN PSTRING arg) { PSTRING value; INT i; // Mac address acceptable format 01:02:03:04:05:06 length 17 if (strlen(arg) != 17) return FALSE; for (i=0, value = rstrtok(arg, ":"); value; value = rstrtok(NULL, ":")) { /* sanity check */ if ((strlen(value) != 2) || (!isxdigit(*value)) || (!isxdigit(*(value+1)))) { return FALSE; } #ifdef CONFIG_STA_SUPPORT AtoH(value, &pAd->ate.Addr1[i++], 1); #endif // CONFIG_STA_SUPPORT // } /* sanity check */ if(i != 6) { return FALSE; } #ifdef CONFIG_STA_SUPPORT ATEDBGPRINT(RT_DEBUG_TRACE, ("Set_ATE_BSSID_Proc (BSSID = %2X:%2X:%2X:%2X:%2X:%2X)\n", pAd->ate.Addr1[0], pAd->ate.Addr1[1], pAd->ate.Addr1[2], pAd->ate.Addr1[3], pAd->ate.Addr1[4], pAd->ate.Addr1[5])); #endif // CONFIG_STA_SUPPORT // ATEDBGPRINT(RT_DEBUG_TRACE, ("Ralink: Set_ATE_BSSID_Proc Success\n")); return TRUE; } /* ========================================================================== Description: Set ATE Tx Channel Return: TRUE if all parameters are OK, FALSE otherwise ========================================================================== */ INT Set_ATE_CHANNEL_Proc( IN PRTMP_ADAPTER pAd, IN PSTRING arg) { UCHAR channel; channel = simple_strtol(arg, 0, 10); // to allow A band channel : ((channel < 1) || (channel > 14)) if ((channel < 1) || (channel > 216)) { ATEDBGPRINT(RT_DEBUG_ERROR, ("Set_ATE_CHANNEL_Proc::Out of range, it should be in range of 1~14.\n")); return FALSE; } pAd->ate.Channel = channel; ATEDBGPRINT(RT_DEBUG_TRACE, ("Set_ATE_CHANNEL_Proc (ATE Channel = %d)\n", pAd->ate.Channel)); ATEDBGPRINT(RT_DEBUG_TRACE, ("Ralink: Set_ATE_CHANNEL_Proc Success\n")); return TRUE; } /* ========================================================================== Description: Set ATE Tx Power0 Return: TRUE if all parameters are OK, FALSE otherwise ========================================================================== */ INT Set_ATE_TX_POWER0_Proc( IN PRTMP_ADAPTER pAd, IN PSTRING arg) { CHAR TxPower; TxPower = simple_strtol(arg, 0, 10); if (pAd->ate.Channel <= 14) { if ((TxPower > 31) || (TxPower < 0)) { ATEDBGPRINT(RT_DEBUG_ERROR, ("Set_ATE_TX_POWER0_Proc::Out of range (Value=%d)\n", TxPower)); return FALSE; } } else/* 5.5 GHz */ { if ((TxPower > 15) || (TxPower < -7)) { ATEDBGPRINT(RT_DEBUG_ERROR, ("Set_ATE_TX_POWER0_Proc::Out of range (Value=%d)\n", TxPower)); return FALSE; } } pAd->ate.TxPower0 = TxPower; ATETxPwrHandler(pAd, 0); ATEDBGPRINT(RT_DEBUG_TRACE, ("Ralink: Set_ATE_TX_POWER0_Proc Success\n")); return TRUE; } /* ========================================================================== Description: Set ATE Tx Power1 Return: TRUE if all parameters are OK, FALSE otherwise ========================================================================== */ INT Set_ATE_TX_POWER1_Proc( IN PRTMP_ADAPTER pAd, IN PSTRING arg) { CHAR TxPower; TxPower = simple_strtol(arg, 0, 10); if (pAd->ate.Channel <= 14) { if ((TxPower > 31) || (TxPower < 0)) { ATEDBGPRINT(RT_DEBUG_ERROR, ("Set_ATE_TX_POWER1_Proc::Out of range (Value=%d)\n", TxPower)); return FALSE; } } else { if ((TxPower > 15) || (TxPower < -7)) { ATEDBGPRINT(RT_DEBUG_ERROR, ("Set_ATE_TX_POWER1_Proc::Out of range (Value=%d)\n", TxPower)); return FALSE; } } pAd->ate.TxPower1 = TxPower; ATETxPwrHandler(pAd, 1); ATEDBGPRINT(RT_DEBUG_TRACE, ("Ralink: Set_ATE_TX_POWER1_Proc Success\n")); return TRUE; } /* ========================================================================== Description: Set ATE Tx Antenna Return: TRUE if all parameters are OK, FALSE otherwise ========================================================================== */ INT Set_ATE_TX_Antenna_Proc( IN PRTMP_ADAPTER pAd, IN PSTRING arg) { CHAR value; value = simple_strtol(arg, 0, 10); if ((value > 2) || (value < 0)) { ATEDBGPRINT(RT_DEBUG_ERROR, ("Set_ATE_TX_Antenna_Proc::Out of range (Value=%d)\n", value)); return FALSE; } pAd->ate.TxAntennaSel = value; ATEDBGPRINT(RT_DEBUG_TRACE, ("Set_ATE_TX_Antenna_Proc (Antenna = %d)\n", pAd->ate.TxAntennaSel)); ATEDBGPRINT(RT_DEBUG_TRACE,("Ralink: Set_ATE_TX_Antenna_Proc Success\n")); // calibration power unbalance issues, merged from Arch Team ATEAsicSwitchChannel(pAd); return TRUE; } /* ========================================================================== Description: Set ATE Rx Antenna Return: TRUE if all parameters are OK, FALSE otherwise ========================================================================== */ INT Set_ATE_RX_Antenna_Proc( IN PRTMP_ADAPTER pAd, IN PSTRING arg) { CHAR value; value = simple_strtol(arg, 0, 10); if ((value > 3) || (value < 0)) { ATEDBGPRINT(RT_DEBUG_ERROR, ("Set_ATE_RX_Antenna_Proc::Out of range (Value=%d)\n", value)); return FALSE; } pAd->ate.RxAntennaSel = value; ATEDBGPRINT(RT_DEBUG_TRACE, ("Set_ATE_RX_Antenna_Proc (Antenna = %d)\n", pAd->ate.RxAntennaSel)); ATEDBGPRINT(RT_DEBUG_TRACE, ("Ralink: Set_ATE_RX_Antenna_Proc Success\n")); // calibration power unbalance issues, merged from Arch Team ATEAsicSwitchChannel(pAd); return TRUE; } /* ========================================================================== Description: Set ATE RF frequence offset Return: TRUE if all parameters are OK, FALSE otherwise ========================================================================== */ INT Set_ATE_TX_FREQOFFSET_Proc( IN PRTMP_ADAPTER pAd, IN PSTRING arg) { UCHAR RFFreqOffset = 0; ULONG R4 = 0; RFFreqOffset = simple_strtol(arg, 0, 10); #ifndef RTMP_RF_RW_SUPPORT if (RFFreqOffset >= 64) #endif // RTMP_RF_RW_SUPPORT // /* RT35xx ATE will reuse this code segment. */ #ifdef RTMP_RF_RW_SUPPORT //2008/08/06: KH modified the limit of offset value from 65 to 95(0x5F) if (RFFreqOffset >= 95) #endif // RTMP_RF_RW_SUPPORT // { ATEDBGPRINT(RT_DEBUG_ERROR, ("Set_ATE_TX_FREQOFFSET_Proc::Out of range, it should be in range of 0~63.\n")); return FALSE; } pAd->ate.RFFreqOffset = RFFreqOffset; #ifdef RTMP_RF_RW_SUPPORT if (IS_RT30xx(pAd) || IS_RT3572(pAd)) { // Set RF offset UCHAR RFValue; ATE_RF_IO_READ8_BY_REG_ID(pAd, RF_R23, (PUCHAR)&RFValue); //2008/08/06: KH modified "pAd->RFFreqOffset" to "pAd->ate.RFFreqOffset" RFValue = ((RFValue & 0x80) | pAd->ate.RFFreqOffset); ATE_RF_IO_WRITE8_BY_REG_ID(pAd, RF_R23, (UCHAR)RFValue); } else #endif // RTMP_RF_RW_SUPPORT // { // RT28xx // shift TX power control to correct RF register bit position R4 = pAd->ate.RFFreqOffset << 15; R4 |= (pAd->LatchRfRegs.R4 & ((~0x001f8000))); pAd->LatchRfRegs.R4 = R4; RtmpRfIoWrite(pAd); } ATEDBGPRINT(RT_DEBUG_TRACE, ("Set_ATE_TX_FREQOFFSET_Proc (RFFreqOffset = %d)\n", pAd->ate.RFFreqOffset)); ATEDBGPRINT(RT_DEBUG_TRACE, ("Ralink: Set_ATE_TX_FREQOFFSET_Proc Success\n")); return TRUE; } /* ========================================================================== Description: Set ATE RF BW Return: TRUE if all parameters are OK, FALSE otherwise ========================================================================== */ INT Set_ATE_TX_BW_Proc( IN PRTMP_ADAPTER pAd, IN PSTRING arg) { INT i; UCHAR value = 0; UCHAR BBPCurrentBW; BBPCurrentBW = simple_strtol(arg, 0, 10); if ((BBPCurrentBW == 0) #ifdef RT30xx || IS_RT2070(pAd) #endif // RT30xx // ) { pAd->ate.TxWI.BW = BW_20; } else { pAd->ate.TxWI.BW = BW_40; } /* RT35xx ATE will reuse this code segment. */ // Fix the error spectrum of CCK-40MHZ // Turn on BBP 20MHz mode by request here. if ((pAd->ate.TxWI.PHYMODE == MODE_CCK) && (pAd->ate.TxWI.BW == BW_40)) { ATEDBGPRINT(RT_DEBUG_ERROR, ("Set_ATE_TX_BW_Proc!! Warning!! CCK only supports 20MHZ!!\nBandwidth switch to 20\n")); pAd->ate.TxWI.BW = BW_20; } if (pAd->ate.TxWI.BW == BW_20) { if (pAd->ate.Channel <= 14) { for (i=0; i<5; i++) { if (pAd->Tx20MPwrCfgGBand[i] != 0xffffffff) { RTMP_IO_WRITE32(pAd, TX_PWR_CFG_0 + i*4, pAd->Tx20MPwrCfgGBand[i]); RTMPusecDelay(5000); } } } else { for (i=0; i<5; i++) { if (pAd->Tx20MPwrCfgABand[i] != 0xffffffff) { RTMP_IO_WRITE32(pAd, TX_PWR_CFG_0 + i*4, pAd->Tx20MPwrCfgABand[i]); RTMPusecDelay(5000); } } } // Set BBP R4 bit[4:3]=0:0 ATE_BBP_IO_READ8_BY_REG_ID(pAd, BBP_R4, &value); value &= (~0x18); ATE_BBP_IO_WRITE8_BY_REG_ID(pAd, BBP_R4, value); // Set BBP R66=0x3C value = 0x3C; ATE_BBP_IO_WRITE8_BY_REG_ID(pAd, BBP_R66, value); // Set BBP R68=0x0B // to improve Rx sensitivity. value = 0x0B; ATE_BBP_IO_WRITE8_BY_REG_ID(pAd, BBP_R68, value); // Set BBP R69=0x16 value = 0x16; ATE_BBP_IO_WRITE8_BY_REG_ID(pAd, BBP_R69, value); // Set BBP R70=0x08 value = 0x08; ATE_BBP_IO_WRITE8_BY_REG_ID(pAd, BBP_R70, value); // Set BBP R73=0x11 value = 0x11; ATE_BBP_IO_WRITE8_BY_REG_ID(pAd, BBP_R73, value); /* If Channel=14, Bandwidth=20M and Mode=CCK, Set BBP R4 bit5=1 (to set Japan filter coefficients). This segment of code will only works when ATETXMODE and ATECHANNEL were set to MODE_CCK and 14 respectively before ATETXBW is set to 0. */ if (pAd->ate.Channel == 14) { INT TxMode = pAd->ate.TxWI.PHYMODE; if (TxMode == MODE_CCK) { // when Channel==14 && Mode==CCK && BandWidth==20M, BBP R4 bit5=1 ATE_BBP_IO_READ8_BY_REG_ID(pAd, BBP_R4, &value); value |= 0x20; //set bit5=1 ATE_BBP_IO_WRITE8_BY_REG_ID(pAd, BBP_R4, value); } } #ifdef RT30xx // set BW = 20 MHz if (IS_RT30xx(pAd)) ATE_RF_IO_WRITE8_BY_REG_ID(pAd, RF_R24, (UCHAR) pAd->Mlme.CaliBW20RfR24); else #endif // RT30xx // // set BW = 20 MHz { pAd->LatchRfRegs.R4 &= ~0x00200000; RtmpRfIoWrite(pAd); } } // If bandwidth = 40M, set RF Reg4 bit 21 = 0. else if (pAd->ate.TxWI.BW == BW_40) { if (pAd->ate.Channel <= 14) { for (i=0; i<5; i++) { if (pAd->Tx40MPwrCfgGBand[i] != 0xffffffff) { RTMP_IO_WRITE32(pAd, TX_PWR_CFG_0 + i*4, pAd->Tx40MPwrCfgGBand[i]); RTMPusecDelay(5000); } } } else { for (i=0; i<5; i++) { if (pAd->Tx40MPwrCfgABand[i] != 0xffffffff) { RTMP_IO_WRITE32(pAd, TX_PWR_CFG_0 + i*4, pAd->Tx40MPwrCfgABand[i]); RTMPusecDelay(5000); } } #ifdef DOT11_N_SUPPORT if ((pAd->ate.TxWI.PHYMODE >= MODE_HTMIX) && (pAd->ate.TxWI.MCS == 7)) { value = 0x28; ATE_BBP_IO_WRITE8_BY_REG_ID(pAd, BBP_R67, value); } #endif // DOT11_N_SUPPORT // } // Set BBP R4 bit[4:3]=1:0 ATE_BBP_IO_READ8_BY_REG_ID(pAd, BBP_R4, &value); value &= (~0x18); value |= 0x10; ATE_BBP_IO_WRITE8_BY_REG_ID(pAd, BBP_R4, value); // Set BBP R66=0x3C value = 0x3C; ATE_BBP_IO_WRITE8_BY_REG_ID(pAd, BBP_R66, value); // Set BBP R68=0x0C // to improve Rx sensitivity value = 0x0C; ATE_BBP_IO_WRITE8_BY_REG_ID(pAd, BBP_R68, value); // Set BBP R69=0x1A value = 0x1A; ATE_BBP_IO_WRITE8_BY_REG_ID(pAd, BBP_R69, value); // Set BBP R70=0x0A value = 0x0A; ATE_BBP_IO_WRITE8_BY_REG_ID(pAd, BBP_R70, value); // Set BBP R73=0x16 value = 0x16; ATE_BBP_IO_WRITE8_BY_REG_ID(pAd, BBP_R73, value); // If bandwidth = 40M, set RF Reg4 bit 21 = 1. #ifdef RT30xx // set BW = 40 MHz if(IS_RT30xx(pAd)) ATE_RF_IO_WRITE8_BY_REG_ID(pAd, RF_R24, (UCHAR) pAd->Mlme.CaliBW40RfR24); else #endif // RT30xx // // set BW = 40 MHz { pAd->LatchRfRegs.R4 |= 0x00200000; RtmpRfIoWrite(pAd); } } ATEDBGPRINT(RT_DEBUG_TRACE, ("Set_ATE_TX_BW_Proc (BBPCurrentBW = %d)\n", pAd->ate.TxWI.BW)); ATEDBGPRINT(RT_DEBUG_TRACE, ("Ralink: Set_ATE_TX_BW_Proc Success\n")); return TRUE; } /* ========================================================================== Description: Set ATE Tx frame length Return: TRUE if all parameters are OK, FALSE otherwise ========================================================================== */ INT Set_ATE_TX_LENGTH_Proc( IN PRTMP_ADAPTER pAd, IN PSTRING arg) { pAd->ate.TxLength = simple_strtol(arg, 0, 10); if ((pAd->ate.TxLength < 24) || (pAd->ate.TxLength > (MAX_FRAME_SIZE - 34/* == 2312 */))) { pAd->ate.TxLength = (MAX_FRAME_SIZE - 34/* == 2312 */); ATEDBGPRINT(RT_DEBUG_ERROR, ("Set_ATE_TX_LENGTH_Proc::Out of range, it should be in range of 24~%d.\n", (MAX_FRAME_SIZE - 34/* == 2312 */))); return FALSE; } ATEDBGPRINT(RT_DEBUG_TRACE, ("Set_ATE_TX_LENGTH_Proc (TxLength = %d)\n", pAd->ate.TxLength)); ATEDBGPRINT(RT_DEBUG_TRACE, ("Ralink: Set_ATE_TX_LENGTH_Proc Success\n")); return TRUE; } /* ========================================================================== Description: Set ATE Tx frame count Return: TRUE if all parameters are OK, FALSE otherwise ========================================================================== */ INT Set_ATE_TX_COUNT_Proc( IN PRTMP_ADAPTER pAd, IN PSTRING arg) { pAd->ate.TxCount = simple_strtol(arg, 0, 10); ATEDBGPRINT(RT_DEBUG_TRACE, ("Set_ATE_TX_COUNT_Proc (TxCount = %d)\n", pAd->ate.TxCount)); ATEDBGPRINT(RT_DEBUG_TRACE, ("Ralink: Set_ATE_TX_COUNT_Proc Success\n")); return TRUE; } /* ========================================================================== Description: Set ATE Tx frame MCS Return: TRUE if all parameters are OK, FALSE otherwise ========================================================================== */ INT Set_ATE_TX_MCS_Proc( IN PRTMP_ADAPTER pAd, IN PSTRING arg) { UCHAR MCS; INT result; MCS = simple_strtol(arg, 0, 10); #ifndef RT30xx result = CheckMCSValid(pAd->ate.TxWI.PHYMODE, MCS); #endif // RT30xx // /* RT35xx ATE will reuse this code segment. */ #ifdef RT30xx result = CheckMCSValid(pAd->ate.TxWI.PHYMODE, MCS, IS_RT2070(pAd)); #endif // RT30xx // if (result != -1) { pAd->ate.TxWI.MCS = (UCHAR)MCS; } else { ATEDBGPRINT(RT_DEBUG_ERROR, ("Set_ATE_TX_MCS_Proc::Out of range, refer to rate table.\n")); return FALSE; } ATEDBGPRINT(RT_DEBUG_TRACE, ("Set_ATE_TX_MCS_Proc (MCS = %d)\n", pAd->ate.TxWI.MCS)); ATEDBGPRINT(RT_DEBUG_TRACE, ("Ralink: Set_ATE_TX_MCS_Proc Success\n")); return TRUE; } /* ========================================================================== Description: Set ATE Tx frame Mode 0: MODE_CCK 1: MODE_OFDM 2: MODE_HTMIX 3: MODE_HTGREENFIELD Return: TRUE if all parameters are OK, FALSE otherwise ========================================================================== */ INT Set_ATE_TX_MODE_Proc( IN PRTMP_ADAPTER pAd, IN PSTRING arg) { UCHAR BbpData = 0; pAd->ate.TxWI.PHYMODE = simple_strtol(arg, 0, 10); if (pAd->ate.TxWI.PHYMODE > 3) { pAd->ate.TxWI.PHYMODE = 0; ATEDBGPRINT(RT_DEBUG_ERROR, ("Set_ATE_TX_MODE_Proc::Out of range.\nIt should be in range of 0~3\n")); ATEDBGPRINT(RT_DEBUG_ERROR, ("0: CCK, 1: OFDM, 2: HT_MIX, 3: HT_GREEN_FIELD.\n")); return FALSE; } // Turn on BBP 20MHz mode by request here. if (pAd->ate.TxWI.PHYMODE == MODE_CCK) { ATE_BBP_IO_READ8_BY_REG_ID(pAd, BBP_R4, &BbpData); BbpData &= (~0x18); ATE_BBP_IO_WRITE8_BY_REG_ID(pAd, BBP_R4, BbpData); pAd->ate.TxWI.BW = BW_20; ATEDBGPRINT(RT_DEBUG_ERROR, ("Set_ATE_TX_MODE_Proc::CCK Only support 20MHZ. Switch to 20MHZ.\n")); } ATEDBGPRINT(RT_DEBUG_TRACE, ("Set_ATE_TX_MODE_Proc (TxMode = %d)\n", pAd->ate.TxWI.PHYMODE)); ATEDBGPRINT(RT_DEBUG_TRACE, ("Ralink: Set_ATE_TX_MODE_Proc Success\n")); return TRUE; } /* ========================================================================== Description: Set ATE Tx frame GI Return: TRUE if all parameters are OK, FALSE otherwise ========================================================================== */ INT Set_ATE_TX_GI_Proc( IN PRTMP_ADAPTER pAd, IN PSTRING arg) { pAd->ate.TxWI.ShortGI = simple_strtol(arg, 0, 10); if (pAd->ate.TxWI.ShortGI > 1) { pAd->ate.TxWI.ShortGI = 0; ATEDBGPRINT(RT_DEBUG_ERROR, ("Set_ATE_TX_GI_Proc::Out of range\n")); return FALSE; } ATEDBGPRINT(RT_DEBUG_TRACE, ("Set_ATE_TX_GI_Proc (GI = %d)\n", pAd->ate.TxWI.ShortGI)); ATEDBGPRINT(RT_DEBUG_TRACE, ("Ralink: Set_ATE_TX_GI_Proc Success\n")); return TRUE; } INT Set_ATE_RX_FER_Proc( IN PRTMP_ADAPTER pAd, IN PSTRING arg) { pAd->ate.bRxFER = simple_strtol(arg, 0, 10); if (pAd->ate.bRxFER == 1) { pAd->ate.RxCntPerSec = 0; pAd->ate.RxTotalCnt = 0; } ATEDBGPRINT(RT_DEBUG_TRACE, ("Set_ATE_RX_FER_Proc (bRxFER = %d)\n", pAd->ate.bRxFER)); ATEDBGPRINT(RT_DEBUG_TRACE, ("Ralink: Set_ATE_RX_FER_Proc Success\n")); return TRUE; } INT Set_ATE_Read_RF_Proc( IN PRTMP_ADAPTER pAd, IN PSTRING arg) { #ifdef RTMP_RF_RW_SUPPORT //2008/07/10:KH add to support RT30xx ATE<-- if (IS_RT30xx(pAd) || IS_RT3572(pAd)) { /* modify by WY for Read RF Reg. error */ UCHAR RFValue; INT index=0; for (index = 0; index < 32; index++) { ATE_RF_IO_READ8_BY_REG_ID(pAd, index, (PUCHAR)&RFValue); ate_print("R%d=%d\n",index,RFValue); } } else //2008/07/10:KH add to support RT30xx ATE--> #endif // RTMP_RF_RW_SUPPORT // { ate_print(KERN_EMERG "R1 = %lx\n", pAd->LatchRfRegs.R1); ate_print(KERN_EMERG "R2 = %lx\n", pAd->LatchRfRegs.R2); ate_print(KERN_EMERG "R3 = %lx\n", pAd->LatchRfRegs.R3); ate_print(KERN_EMERG "R4 = %lx\n", pAd->LatchRfRegs.R4); } return TRUE; } INT Set_ATE_Write_RF1_Proc( IN PRTMP_ADAPTER pAd, IN PSTRING arg) { UINT32 value = (UINT32) simple_strtol(arg, 0, 16); #ifdef RTMP_RF_RW_SUPPORT //2008/07/10:KH add to support 3070 ATE<-- if (IS_RT30xx(pAd) || IS_RT3572(pAd)) { ate_print("Warning!! RT3xxx Don't Support !\n"); return FALSE; } else //2008/07/10:KH add to support 3070 ATE--> #endif // RTMP_RF_RW_SUPPORT // { pAd->LatchRfRegs.R1 = value; RtmpRfIoWrite(pAd); } return TRUE; } INT Set_ATE_Write_RF2_Proc( IN PRTMP_ADAPTER pAd, IN PSTRING arg) { UINT32 value = (UINT32) simple_strtol(arg, 0, 16); #ifdef RTMP_RF_RW_SUPPORT //2008/07/10:KH add to support 3070 ATE<-- if (IS_RT30xx(pAd) || IS_RT3572(pAd)) { ate_print("Warning!! RT3xxx Don't Support !\n"); return FALSE; } else //2008/07/10:KH add to support 3070 ATE--> #endif // RTMP_RF_RW_SUPPORT // { pAd->LatchRfRegs.R2 = value; RtmpRfIoWrite(pAd); } return TRUE; } INT Set_ATE_Write_RF3_Proc( IN PRTMP_ADAPTER pAd, IN PSTRING arg) { UINT32 value = simple_strtol(arg, 0, 16); #ifdef RTMP_RF_RW_SUPPORT //2008/07/10:KH add to support 3070 ATE<-- if (IS_RT30xx(pAd) || IS_RT3572(pAd)) { ate_print("Warning!! RT3xxx Don't Support !\n"); return FALSE; } else //2008/07/10:KH add to support 3070 ATE--> #endif // RTMP_RF_RW_SUPPORT // { pAd->LatchRfRegs.R3 = value; RtmpRfIoWrite(pAd); } return TRUE; } INT Set_ATE_Write_RF4_Proc( IN PRTMP_ADAPTER pAd, IN PSTRING arg) { UINT32 value = (UINT32) simple_strtol(arg, 0, 16); #ifdef RTMP_RF_RW_SUPPORT //2008/07/10:KH add to support 3070 ATE<-- if (IS_RT30xx(pAd) || IS_RT3572(pAd)) { ate_print("Warning!! RT3xxx Don't Support !\n"); return FALSE; } else //2008/07/10:KH add to support 3070 ATE--> #endif // RTMP_RF_RW_SUPPORT // { pAd->LatchRfRegs.R4 = value; RtmpRfIoWrite(pAd); } return TRUE; } /* ========================================================================== Description: Load and Write EEPROM from a binary file prepared in advance. Return: TRUE if all parameters are OK, FALSE otherwise ========================================================================== */ #if defined(LINUX) || defined(VXWORKS) INT Set_ATE_Load_E2P_Proc( IN PRTMP_ADAPTER pAd, IN PSTRING arg) { BOOLEAN ret = FALSE; PSTRING src = EEPROM_BIN_FILE_NAME; RTMP_OS_FD srcf; INT32 retval; USHORT WriteEEPROM[(EEPROM_SIZE/2)]; INT FileLength = 0; UINT32 value = (UINT32) simple_strtol(arg, 0, 10); RTMP_OS_FS_INFO osFSInfo; ATEDBGPRINT(RT_DEBUG_ERROR, ("===> %s (value=%d)\n\n", __FUNCTION__, value)); if (value > 0) { /* zero the e2p buffer */ NdisZeroMemory((PUCHAR)WriteEEPROM, EEPROM_SIZE); RtmpOSFSInfoChange(&osFSInfo, TRUE); do { /* open the bin file */ srcf = RtmpOSFileOpen(src, O_RDONLY, 0); if (IS_FILE_OPEN_ERR(srcf)) { ate_print("%s - Error opening file %s\n", __FUNCTION__, src); break; } /* read the firmware from the file *.bin */ FileLength = RtmpOSFileRead(srcf, (PSTRING)WriteEEPROM, EEPROM_SIZE); if (FileLength != EEPROM_SIZE) { ate_print("%s: error file length (=%d) in e2p.bin\n", __FUNCTION__, FileLength); break; } else { /* write the content of .bin file to EEPROM */ rt_ee_write_all(pAd, WriteEEPROM); ret = TRUE; } break; } while(TRUE); /* close firmware file */ if (IS_FILE_OPEN_ERR(srcf)) { ; } else { retval = RtmpOSFileClose(srcf); if (retval) { ATEDBGPRINT(RT_DEBUG_ERROR, ("--> Error %d closing %s\n", -retval, src)); } } /* restore */ RtmpOSFSInfoChange(&osFSInfo, FALSE); } ATEDBGPRINT(RT_DEBUG_ERROR, ("<=== %s (ret=%d)\n", __FUNCTION__, ret)); return ret; } #endif // defined(LINUX) || defined(VXWORKS) // INT Set_ATE_Read_E2P_Proc( IN PRTMP_ADAPTER pAd, IN PSTRING arg) { USHORT buffer[EEPROM_SIZE/2]; USHORT *p; int i; rt_ee_read_all(pAd, (USHORT *)buffer); p = buffer; for (i = 0; i < (EEPROM_SIZE/2); i++) { ate_print("%4.4x ", *p); if (((i+1) % 16) == 0) ate_print("\n"); p++; } return TRUE; } INT Set_ATE_Show_Proc( IN PRTMP_ADAPTER pAd, IN PSTRING arg) { ate_print("Mode=%d\n", pAd->ate.Mode); ate_print("TxPower0=%d\n", pAd->ate.TxPower0); ate_print("TxPower1=%d\n", pAd->ate.TxPower1); ate_print("TxAntennaSel=%d\n", pAd->ate.TxAntennaSel); ate_print("RxAntennaSel=%d\n", pAd->ate.RxAntennaSel); ate_print("BBPCurrentBW=%d\n", pAd->ate.TxWI.BW); ate_print("GI=%d\n", pAd->ate.TxWI.ShortGI); ate_print("MCS=%d\n", pAd->ate.TxWI.MCS); ate_print("TxMode=%d\n", pAd->ate.TxWI.PHYMODE); ate_print("Addr1=%02x:%02x:%02x:%02x:%02x:%02x\n", pAd->ate.Addr1[0], pAd->ate.Addr1[1], pAd->ate.Addr1[2], pAd->ate.Addr1[3], pAd->ate.Addr1[4], pAd->ate.Addr1[5]); ate_print("Addr2=%02x:%02x:%02x:%02x:%02x:%02x\n", pAd->ate.Addr2[0], pAd->ate.Addr2[1], pAd->ate.Addr2[2], pAd->ate.Addr2[3], pAd->ate.Addr2[4], pAd->ate.Addr2[5]); ate_print("Addr3=%02x:%02x:%02x:%02x:%02x:%02x\n", pAd->ate.Addr3[0], pAd->ate.Addr3[1], pAd->ate.Addr3[2], pAd->ate.Addr3[3], pAd->ate.Addr3[4], pAd->ate.Addr3[5]); ate_print("Channel=%d\n", pAd->ate.Channel); ate_print("TxLength=%d\n", pAd->ate.TxLength); ate_print("TxCount=%u\n", pAd->ate.TxCount); ate_print("RFFreqOffset=%d\n", pAd->ate.RFFreqOffset); ate_print(KERN_EMERG "Set_ATE_Show_Proc Success\n"); return TRUE; } INT Set_ATE_Help_Proc( IN PRTMP_ADAPTER pAd, IN PSTRING arg) { ate_print("ATE=ATESTART, ATESTOP, TXCONT, TXCARR, TXFRAME, RXFRAME\n"); ate_print("ATEDA\n"); ate_print("ATESA\n"); ate_print("ATEBSSID\n"); ate_print("ATECHANNEL, range:0~14(unless A band !)\n"); ate_print("ATETXPOW0, set power level of antenna 1.\n"); ate_print("ATETXPOW1, set power level of antenna 2.\n"); ate_print("ATETXANT, set TX antenna. 0:all, 1:antenna one, 2:antenna two.\n"); ate_print("ATERXANT, set RX antenna.0:all, 1:antenna one, 2:antenna two, 3:antenna three.\n"); ate_print("ATETXFREQOFFSET, set frequency offset, range 0~63\n"); ate_print("ATETXBW, set BandWidth, 0:20MHz, 1:40MHz.\n"); ate_print("ATETXLEN, set Frame length, range 24~%d\n", (MAX_FRAME_SIZE - 34/* == 2312 */)); ate_print("ATETXCNT, set how many frame going to transmit.\n"); ate_print("ATETXMCS, set MCS, reference to rate table.\n"); ate_print("ATETXMODE, set Mode 0:CCK, 1:OFDM, 2:HT-Mix, 3:GreenField, reference to rate table.\n"); ate_print("ATETXGI, set GI interval, 0:Long, 1:Short\n"); ate_print("ATERXFER, 0:disable Rx Frame error rate. 1:enable Rx Frame error rate.\n"); ate_print("ATERRF, show all RF registers.\n"); ate_print("ATEWRF1, set RF1 register.\n"); ate_print("ATEWRF2, set RF2 register.\n"); ate_print("ATEWRF3, set RF3 register.\n"); ate_print("ATEWRF4, set RF4 register.\n"); ate_print("ATELDE2P, load EEPROM from .bin file.\n"); ate_print("ATERE2P, display all EEPROM content.\n"); ate_print("ATESHOW, display all parameters of ATE.\n"); ate_print("ATEHELP, online help.\n"); return TRUE; } /* ========================================================================== Description: AsicSwitchChannel() dedicated for ATE. ========================================================================== */ VOID ATEAsicSwitchChannel( IN PRTMP_ADAPTER pAd) { UINT32 R2 = 0, R3 = DEFAULT_RF_TX_POWER, R4 = 0, Value = 0; CHAR TxPwer = 0, TxPwer2 = 0; UCHAR index = 0, BbpValue = 0, R66 = 0x30; RTMP_RF_REGS *RFRegTable; UCHAR Channel = 0; RFRegTable = NULL; #ifdef RALINK_28xx_QA // for QA mode, TX power values are passed from UI if ((pAd->ate.bQATxStart == TRUE) || (pAd->ate.bQARxStart == TRUE)) { if (pAd->ate.Channel != pAd->LatchRfRegs.Channel) { pAd->ate.Channel = pAd->LatchRfRegs.Channel; } return; } else #endif // RALINK_28xx_QA // Channel = pAd->ate.Channel; // select antenna for RT3090 AsicAntennaSelect(pAd, Channel); // fill Tx power value TxPwer = pAd->ate.TxPower0; TxPwer2 = pAd->ate.TxPower1; #ifdef RT30xx //2008/07/10:KH add to support 3070 ATE<-- /* The RF programming sequence is difference between 3xxx and 2xxx. The 3070 is 1T1R. Therefore, we don't need to set the number of Tx/Rx path and the only job is to set the parameters of channels. */ if (IS_RT30xx(pAd) && ((pAd->RfIcType == RFIC_3020) || (pAd->RfIcType == RFIC_3021) || (pAd->RfIcType == RFIC_3022) || (pAd->RfIcType == RFIC_2020))) { /* modify by WY for Read RF Reg. error */ UCHAR RFValue = 0; for (index = 0; index < NUM_OF_3020_CHNL; index++) { if (Channel == FreqItems3020[index].Channel) { // Programming channel parameters. ATE_RF_IO_WRITE8_BY_REG_ID(pAd, RF_R02, FreqItems3020[index].N); ATE_RF_IO_WRITE8_BY_REG_ID(pAd, RF_R03, FreqItems3020[index].K); ATE_RF_IO_READ8_BY_REG_ID(pAd, RF_R06, (PUCHAR)&RFValue); RFValue = (RFValue & 0xFC) | FreqItems3020[index].R; ATE_RF_IO_WRITE8_BY_REG_ID(pAd, RF_R06, (UCHAR)RFValue); // Set Tx Power. ATE_RF_IO_READ8_BY_REG_ID(pAd, RF_R12, (PUCHAR)&RFValue); RFValue = (RFValue & 0xE0) | TxPwer; ATE_RF_IO_WRITE8_BY_REG_ID(pAd, RF_R12, (UCHAR)RFValue); // Set RF offset. ATE_RF_IO_READ8_BY_REG_ID(pAd, RF_R23, (PUCHAR)&RFValue); //2008/08/06: KH modified "pAd->RFFreqOffset" to "pAd->ate.RFFreqOffset" RFValue = (RFValue & 0x80) | pAd->ate.RFFreqOffset; ATE_RF_IO_WRITE8_BY_REG_ID(pAd, RF_R23, (UCHAR)RFValue); // Set BW. if (pAd->ate.TxWI.BW == BW_40) { RFValue = pAd->Mlme.CaliBW40RfR24; // DISABLE_11N_CHECK(pAd); } else { RFValue = pAd->Mlme.CaliBW20RfR24; } ATE_RF_IO_WRITE8_BY_REG_ID(pAd, RF_R24, (UCHAR)RFValue); // Enable RF tuning ATE_RF_IO_READ8_BY_REG_ID(pAd, RF_R07, (PUCHAR)&RFValue); RFValue = RFValue | 0x1; ATE_RF_IO_WRITE8_BY_REG_ID(pAd, RF_R07, (UCHAR)RFValue); // latch channel for future usage pAd->LatchRfRegs.Channel = Channel; break; } } ATEDBGPRINT(RT_DEBUG_TRACE, ("SwitchChannel#%d(RF=%d, Pwr0=%d, Pwr1=%d, %dT), N=0x%02X, K=0x%02X, R=0x%02X\n", Channel, pAd->RfIcType, TxPwer, TxPwer2, pAd->Antenna.field.TxPath, FreqItems3020[index].N, FreqItems3020[index].K, FreqItems3020[index].R)); } else //2008/07/10:KH add to support 3070 ATE--> #endif // RT30xx // { /* RT28xx */ RFRegTable = RF2850RegTable; switch (pAd->RfIcType) { /* But only 2850 and 2750 support 5.5GHz band... */ case RFIC_2820: case RFIC_2850: case RFIC_2720: case RFIC_2750: for (index = 0; index < NUM_OF_2850_CHNL; index++) { if (Channel == RFRegTable[index].Channel) { R2 = RFRegTable[index].R2; // If TX path is 1, bit 14 = 1; if (pAd->Antenna.field.TxPath == 1) { R2 |= 0x4000; } if (pAd->Antenna.field.RxPath == 2) { switch (pAd->ate.RxAntennaSel) { case 1: R2 |= 0x20040; ATE_BBP_IO_READ8_BY_REG_ID(pAd, BBP_R3, &BbpValue); BbpValue &= 0xE4; BbpValue |= 0x00; ATE_BBP_IO_WRITE8_BY_REG_ID(pAd, BBP_R3, BbpValue); break; case 2: R2 |= 0x10040; ATE_BBP_IO_READ8_BY_REG_ID(pAd, BBP_R3, &BbpValue); BbpValue &= 0xE4; BbpValue |= 0x01; ATE_BBP_IO_WRITE8_BY_REG_ID(pAd, BBP_R3, BbpValue); break; default: R2 |= 0x40; ATE_BBP_IO_READ8_BY_REG_ID(pAd, BBP_R3, &BbpValue); BbpValue &= 0xE4; /* Only enable two Antenna to receive. */ BbpValue |= 0x08; ATE_BBP_IO_WRITE8_BY_REG_ID(pAd, BBP_R3, BbpValue); break; } } else if (pAd->Antenna.field.RxPath == 1) { // write 1 to off RxPath R2 |= 0x20040; } if (pAd->Antenna.field.TxPath == 2) { if (pAd->ate.TxAntennaSel == 1) { // If TX Antenna select is 1 , bit 14 = 1; Disable Ant 2 R2 |= 0x4000; ATE_BBP_IO_READ8_BY_REG_ID(pAd, BBP_R1, &BbpValue); BbpValue &= 0xE7; // 11100111B ATE_BBP_IO_WRITE8_BY_REG_ID(pAd, BBP_R1, BbpValue); } else if (pAd->ate.TxAntennaSel == 2) { // If TX Antenna select is 2 , bit 15 = 1; Disable Ant 1 R2 |= 0x8000; ATE_BBP_IO_READ8_BY_REG_ID(pAd, BBP_R1, &BbpValue); BbpValue &= 0xE7; BbpValue |= 0x08; ATE_BBP_IO_WRITE8_BY_REG_ID(pAd, BBP_R1, BbpValue); } else { ATE_BBP_IO_READ8_BY_REG_ID(pAd, BBP_R1, &BbpValue); BbpValue &= 0xE7; BbpValue |= 0x10; ATE_BBP_IO_WRITE8_BY_REG_ID(pAd, BBP_R1, BbpValue); } } if (pAd->Antenna.field.RxPath == 3) { switch (pAd->ate.RxAntennaSel) { case 1: R2 |= 0x20040; ATE_BBP_IO_READ8_BY_REG_ID(pAd, BBP_R3, &BbpValue); BbpValue &= 0xE4; BbpValue |= 0x00; ATE_BBP_IO_WRITE8_BY_REG_ID(pAd, BBP_R3, BbpValue); break; case 2: R2 |= 0x10040; ATE_BBP_IO_READ8_BY_REG_ID(pAd, BBP_R3, &BbpValue); BbpValue &= 0xE4; BbpValue |= 0x01; ATE_BBP_IO_WRITE8_BY_REG_ID(pAd, BBP_R3, BbpValue); break; case 3: R2 |= 0x30000; ATE_BBP_IO_READ8_BY_REG_ID(pAd, BBP_R3, &BbpValue); BbpValue &= 0xE4; BbpValue |= 0x02; ATE_BBP_IO_WRITE8_BY_REG_ID(pAd, BBP_R3, BbpValue); break; default: ATE_BBP_IO_READ8_BY_REG_ID(pAd, BBP_R3, &BbpValue); BbpValue &= 0xE4; BbpValue |= 0x10; ATE_BBP_IO_WRITE8_BY_REG_ID(pAd, BBP_R3, BbpValue); break; } } if (Channel > 14) { // initialize R3, R4 R3 = (RFRegTable[index].R3 & 0xffffc1ff); R4 = (RFRegTable[index].R4 & (~0x001f87c0)) | (pAd->ate.RFFreqOffset << 15); /* According the Rory's suggestion to solve the middle range issue. 5.5G band power range : 0xF9~0X0F, TX0 Reg3 bit9/TX1 Reg4 bit6="0" means the TX power reduce 7dB. */ // R3 if ((TxPwer >= -7) && (TxPwer < 0)) { TxPwer = (7+TxPwer); TxPwer = (TxPwer > 0xF) ? (0xF) : (TxPwer); R3 |= (TxPwer << 10); ATEDBGPRINT(RT_DEBUG_TRACE, ("ATEAsicSwitchChannel: TxPwer=%d \n", TxPwer)); } else { TxPwer = (TxPwer > 0xF) ? (0xF) : (TxPwer); R3 |= (TxPwer << 10) | (1 << 9); } // R4 if ((TxPwer2 >= -7) && (TxPwer2 < 0)) { TxPwer2 = (7+TxPwer2); TxPwer2 = (TxPwer2 > 0xF) ? (0xF) : (TxPwer2); R4 |= (TxPwer2 << 7); ATEDBGPRINT(RT_DEBUG_TRACE, ("ATEAsicSwitchChannel: TxPwer2=%d \n", TxPwer2)); } else { TxPwer2 = (TxPwer2 > 0xF) ? (0xF) : (TxPwer2); R4 |= (TxPwer2 << 7) | (1 << 6); } } else { // Set TX power0. R3 = (RFRegTable[index].R3 & 0xffffc1ff) | (TxPwer << 9); // Set frequency offset and TX power1. R4 = (RFRegTable[index].R4 & (~0x001f87c0)) | (pAd->ate.RFFreqOffset << 15) | (TxPwer2 <<6); } // based on BBP current mode before changing RF channel if (pAd->ate.TxWI.BW == BW_40) { R4 |=0x200000; } // Update variables. pAd->LatchRfRegs.Channel = Channel; pAd->LatchRfRegs.R1 = RFRegTable[index].R1; pAd->LatchRfRegs.R2 = R2; pAd->LatchRfRegs.R3 = R3; pAd->LatchRfRegs.R4 = R4; RtmpRfIoWrite(pAd); break; } } break; default: break; } } // Change BBP setting during switch from a->g, g->a if (Channel <= 14) { UINT32 TxPinCfg = 0x00050F0A;// 2007.10.09 by Brian : 0x0005050A ==> 0x00050F0A ATE_BBP_IO_WRITE8_BY_REG_ID(pAd, BBP_R62, (0x37 - GET_LNA_GAIN(pAd))); ATE_BBP_IO_WRITE8_BY_REG_ID(pAd, BBP_R63, (0x37 - GET_LNA_GAIN(pAd))); ATE_BBP_IO_WRITE8_BY_REG_ID(pAd, BBP_R64, (0x37 - GET_LNA_GAIN(pAd))); /* For 1T/2R chip only... */ if (pAd->NicConfig2.field.ExternalLNAForG) { ATE_BBP_IO_WRITE8_BY_REG_ID(pAd, BBP_R82, 0x62); } else { ATE_BBP_IO_WRITE8_BY_REG_ID(pAd, BBP_R82, 0x84); } // According the Rory's suggestion to solve the middle range issue. ATE_BBP_IO_READ8_BY_REG_ID(pAd, BBP_R86, &BbpValue);// may be removed for RT35xx ++ ASSERT((BbpValue == 0x00)); if ((BbpValue != 0x00)) { ATE_BBP_IO_WRITE8_BY_REG_ID(pAd, BBP_R86, 0x00); }// may be removed for RT35xx -- // 5.5 GHz band selection PIN, bit1 and bit2 are complement RTMP_IO_READ32(pAd, TX_BAND_CFG, &Value); Value &= (~0x6); Value |= (0x04); RTMP_IO_WRITE32(pAd, TX_BAND_CFG, Value); // Turn off unused PA or LNA when only 1T or 1R. if (pAd->Antenna.field.TxPath == 1) { TxPinCfg &= 0xFFFFFFF3; } if (pAd->Antenna.field.RxPath == 1) { TxPinCfg &= 0xFFFFF3FF; } // calibration power unbalance issues if (pAd->Antenna.field.TxPath == 2) { if (pAd->ate.TxAntennaSel == 1) { TxPinCfg &= 0xFFFFFFF7; } else if (pAd->ate.TxAntennaSel == 2) { TxPinCfg &= 0xFFFFFFFD; } } RTMP_IO_WRITE32(pAd, TX_PIN_CFG, TxPinCfg); } else { UINT32 TxPinCfg = 0x00050F05;// 2007.10.09 by Brian : 0x00050505 ==> 0x00050F05 ATE_BBP_IO_WRITE8_BY_REG_ID(pAd, BBP_R62, (0x37 - GET_LNA_GAIN(pAd))); ATE_BBP_IO_WRITE8_BY_REG_ID(pAd, BBP_R63, (0x37 - GET_LNA_GAIN(pAd))); ATE_BBP_IO_WRITE8_BY_REG_ID(pAd, BBP_R64, (0x37 - GET_LNA_GAIN(pAd))); ATE_BBP_IO_WRITE8_BY_REG_ID(pAd, BBP_R82, 0xF2); // According the Rory's suggestion to solve the middle range issue. ATE_BBP_IO_READ8_BY_REG_ID(pAd, BBP_R86, &BbpValue);// may be removed for RT35xx ++ ASSERT((BbpValue == 0x00)); if ((BbpValue != 0x00)) { ATE_BBP_IO_WRITE8_BY_REG_ID(pAd, BBP_R86, 0x00); } ATE_BBP_IO_READ8_BY_REG_ID(pAd, BBP_R91, &BbpValue); ASSERT((BbpValue == 0x04)); ATE_BBP_IO_READ8_BY_REG_ID(pAd, BBP_R92, &BbpValue); ASSERT((BbpValue == 0x00));// may be removed for RT35xx -- // 5.5 GHz band selection PIN, bit1 and bit2 are complement RTMP_IO_READ32(pAd, TX_BAND_CFG, &Value); Value &= (~0x6); Value |= (0x02); RTMP_IO_WRITE32(pAd, TX_BAND_CFG, Value); // Turn off unused PA or LNA when only 1T or 1R. if (pAd->Antenna.field.TxPath == 1) { TxPinCfg &= 0xFFFFFFF3; } if (pAd->Antenna.field.RxPath == 1) { TxPinCfg &= 0xFFFFF3FF; } RTMP_IO_WRITE32(pAd, TX_PIN_CFG, TxPinCfg); } // R66 should be set according to Channel and use 20MHz when scanning if (Channel <= 14) { // BG band R66 = 0x2E + GET_LNA_GAIN(pAd); ATE_BBP_IO_WRITE8_BY_REG_ID(pAd, BBP_R66, R66); } else { // 5.5 GHz band if (pAd->ate.TxWI.BW == BW_20) { R66 = (UCHAR)(0x32 + (GET_LNA_GAIN(pAd)*5)/3); ATE_BBP_IO_WRITE8_BY_REG_ID(pAd, BBP_R66, R66); } else { R66 = (UCHAR)(0x3A + (GET_LNA_GAIN(pAd)*5)/3); ATE_BBP_IO_WRITE8_BY_REG_ID(pAd, BBP_R66, R66); } } /* On 11A, We should delay and wait RF/BBP to be stable and the appropriate time should be 1000 micro seconds. 2005/06/05 - On 11G, We also need this delay time. Otherwise it's difficult to pass the WHQL. */ RTMPusecDelay(1000); #ifndef RTMP_RF_RW_SUPPORT if (Channel > 14) { // When 5.5GHz band the LSB of TxPwr will be used to reduced 7dB or not. ATEDBGPRINT(RT_DEBUG_TRACE, ("SwitchChannel#%d(RF=%d, %dT) to , R1=0x%08lx, R2=0x%08lx, R3=0x%08lx, R4=0x%08lx\n", Channel, pAd->RfIcType, pAd->Antenna.field.TxPath, pAd->LatchRfRegs.R1, pAd->LatchRfRegs.R2, pAd->LatchRfRegs.R3, pAd->LatchRfRegs.R4)); } else { ATEDBGPRINT(RT_DEBUG_TRACE, ("SwitchChannel#%d(RF=%d, Pwr0=%u, Pwr1=%u, %dT) to , R1=0x%08lx, R2=0x%08lx, R3=0x%08lx, R4=0x%08lx\n", Channel, pAd->RfIcType, (R3 & 0x00003e00) >> 9, (R4 & 0x000007c0) >> 6, pAd->Antenna.field.TxPath, pAd->LatchRfRegs.R1, pAd->LatchRfRegs.R2, pAd->LatchRfRegs.R3, pAd->LatchRfRegs.R4)); } #endif // RTMP_RF_RW_SUPPORT // } /* In fact, no one will call this routine so far ! */ /* ========================================================================== Description: Gives CCK TX rate 2 more dB TX power. This routine works only in ATE mode. calculate desired Tx power in RF R3.Tx0~5, should consider - 0. if current radio is a noisy environment (pAd->DrsCounters.fNoisyEnvironment) 1. TxPowerPercentage 2. auto calibration based on TSSI feedback 3. extra 2 db for CCK 4. -10 db upon very-short distance (AvgRSSI >= -40db) to AP NOTE: Since this routine requires the value of (pAd->DrsCounters.fNoisyEnvironment), it should be called AFTER MlmeDynamicTxRateSwitching() ========================================================================== */ VOID ATEAsicAdjustTxPower( IN PRTMP_ADAPTER pAd) { INT i, j; CHAR DeltaPwr = 0; BOOLEAN bAutoTxAgc = FALSE; UCHAR TssiRef, *pTssiMinusBoundary, *pTssiPlusBoundary, TxAgcStep; UCHAR BbpR49 = 0, idx; PCHAR pTxAgcCompensate; ULONG TxPwr[5]; CHAR Value; /* no one calls this procedure so far */ if (pAd->ate.TxWI.BW == BW_40) { if (pAd->ate.Channel > 14) { TxPwr[0] = pAd->Tx40MPwrCfgABand[0]; TxPwr[1] = pAd->Tx40MPwrCfgABand[1]; TxPwr[2] = pAd->Tx40MPwrCfgABand[2]; TxPwr[3] = pAd->Tx40MPwrCfgABand[3]; TxPwr[4] = pAd->Tx40MPwrCfgABand[4]; } else { TxPwr[0] = pAd->Tx40MPwrCfgGBand[0]; TxPwr[1] = pAd->Tx40MPwrCfgGBand[1]; TxPwr[2] = pAd->Tx40MPwrCfgGBand[2]; TxPwr[3] = pAd->Tx40MPwrCfgGBand[3]; TxPwr[4] = pAd->Tx40MPwrCfgGBand[4]; } } else { if (pAd->ate.Channel > 14) { TxPwr[0] = pAd->Tx20MPwrCfgABand[0]; TxPwr[1] = pAd->Tx20MPwrCfgABand[1]; TxPwr[2] = pAd->Tx20MPwrCfgABand[2]; TxPwr[3] = pAd->Tx20MPwrCfgABand[3]; TxPwr[4] = pAd->Tx20MPwrCfgABand[4]; } else { TxPwr[0] = pAd->Tx20MPwrCfgGBand[0]; TxPwr[1] = pAd->Tx20MPwrCfgGBand[1]; TxPwr[2] = pAd->Tx20MPwrCfgGBand[2]; TxPwr[3] = pAd->Tx20MPwrCfgGBand[3]; TxPwr[4] = pAd->Tx20MPwrCfgGBand[4]; } } // TX power compensation for temperature variation based on TSSI. // Do it per 4 seconds. if (pAd->Mlme.OneSecPeriodicRound % 4 == 0) { if (pAd->ate.Channel <= 14) { /* bg channel */ bAutoTxAgc = pAd->bAutoTxAgcG; TssiRef = pAd->TssiRefG; pTssiMinusBoundary = &pAd->TssiMinusBoundaryG[0]; pTssiPlusBoundary = &pAd->TssiPlusBoundaryG[0]; TxAgcStep = pAd->TxAgcStepG; pTxAgcCompensate = &pAd->TxAgcCompensateG; } else { /* a channel */ bAutoTxAgc = pAd->bAutoTxAgcA; TssiRef = pAd->TssiRefA; pTssiMinusBoundary = &pAd->TssiMinusBoundaryA[0]; pTssiPlusBoundary = &pAd->TssiPlusBoundaryA[0]; TxAgcStep = pAd->TxAgcStepA; pTxAgcCompensate = &pAd->TxAgcCompensateA; } if (bAutoTxAgc) { /* BbpR49 is unsigned char. */ ATE_BBP_IO_READ8_BY_REG_ID(pAd, BBP_R49, &BbpR49); /* (p) TssiPlusBoundaryG[0] = 0 = (m) TssiMinusBoundaryG[0] */ /* compensate: +4 +3 +2 +1 0 -1 -2 -3 -4 * steps */ /* step value is defined in pAd->TxAgcStepG for tx power value */ /* [4]+1+[4] p4 p3 p2 p1 o1 m1 m2 m3 m4 */ /* ex: 0x00 0x15 0x25 0x45 0x88 0xA0 0xB5 0xD0 0xF0 above value are examined in mass factory production */ /* [4] [3] [2] [1] [0] [1] [2] [3] [4] */ /* plus is 0x10 ~ 0x40, minus is 0x60 ~ 0x90 */ /* if value is between p1 ~ o1 or o1 ~ s1, no need to adjust tx power */ /* if value is 0x65, tx power will be -= TxAgcStep*(2-1) */ if (BbpR49 > pTssiMinusBoundary[1]) { // Reading is larger than the reference value. // Check for how large we need to decrease the Tx power. for (idx = 1; idx < 5; idx++) { // Found the range. if (BbpR49 <= pTssiMinusBoundary[idx]) break; } // The index is the step we should decrease, idx = 0 means there is nothing to compensate. // if (R3 > (ULONG) (TxAgcStep * (idx-1))) *pTxAgcCompensate = -(TxAgcStep * (idx-1)); // else // *pTxAgcCompensate = -((UCHAR)R3); DeltaPwr += (*pTxAgcCompensate); ATEDBGPRINT(RT_DEBUG_TRACE, ("-- Tx Power, BBP R1=%x, TssiRef=%x, TxAgcStep=%x, step = -%d\n", BbpR49, TssiRef, TxAgcStep, idx-1)); } else if (BbpR49 < pTssiPlusBoundary[1]) { // Reading is smaller than the reference value. // Check for how large we need to increase the Tx power. for (idx = 1; idx < 5; idx++) { // Found the range. if (BbpR49 >= pTssiPlusBoundary[idx]) break; } // The index is the step we should increase, idx = 0 means there is nothing to compensate. *pTxAgcCompensate = TxAgcStep * (idx-1); DeltaPwr += (*pTxAgcCompensate); ATEDBGPRINT(RT_DEBUG_TRACE, ("++ Tx Power, BBP R1=%x, TssiRef=%x, TxAgcStep=%x, step = +%d\n", BbpR49, TssiRef, TxAgcStep, idx-1)); } else { *pTxAgcCompensate = 0; ATEDBGPRINT(RT_DEBUG_TRACE, (" Tx Power, BBP R1=%x, TssiRef=%x, TxAgcStep=%x, step = +%d\n", BbpR49, TssiRef, TxAgcStep, 0)); } } } else { if (pAd->ate.Channel <= 14) { bAutoTxAgc = pAd->bAutoTxAgcG; pTxAgcCompensate = &pAd->TxAgcCompensateG; } else { bAutoTxAgc = pAd->bAutoTxAgcA; pTxAgcCompensate = &pAd->TxAgcCompensateA; } if (bAutoTxAgc) DeltaPwr += (*pTxAgcCompensate); } /* Calculate delta power based on the percentage specified from UI. */ // E2PROM setting is calibrated for maximum TX power (i.e. 100%) // We lower TX power here according to the percentage specified from UI. if (pAd->CommonCfg.TxPowerPercentage == 0xffffffff) // AUTO TX POWER control ; else if (pAd->CommonCfg.TxPowerPercentage > 90) // 91 ~ 100% & AUTO, treat as 100% in terms of mW ; else if (pAd->CommonCfg.TxPowerPercentage > 60) // 61 ~ 90%, treat as 75% in terms of mW { DeltaPwr -= 1; } else if (pAd->CommonCfg.TxPowerPercentage > 30) // 31 ~ 60%, treat as 50% in terms of mW { DeltaPwr -= 3; } else if (pAd->CommonCfg.TxPowerPercentage > 15) // 16 ~ 30%, treat as 25% in terms of mW { DeltaPwr -= 6; } else if (pAd->CommonCfg.TxPowerPercentage > 9) // 10 ~ 15%, treat as 12.5% in terms of mW { DeltaPwr -= 9; } else // 0 ~ 9 %, treat as MIN(~3%) in terms of mW { DeltaPwr -= 12; } /* Reset different new tx power for different TX rate. */ for (i=0; i<5; i++) { if (TxPwr[i] != 0xffffffff) { for (j=0; j<8; j++) { Value = (CHAR)((TxPwr[i] >> j*4) & 0x0F); /* 0 ~ 15 */ if ((Value + DeltaPwr) < 0) { Value = 0; /* min */ } else if ((Value + DeltaPwr) > 0xF) { Value = 0xF; /* max */ } else { Value += DeltaPwr; /* temperature compensation */ } /* fill new value to CSR offset */ TxPwr[i] = (TxPwr[i] & ~(0x0000000F << j*4)) | (Value << j*4); } /* write tx power value to CSR */ /* TX_PWR_CFG_0 (8 tx rate) for TX power for OFDM 12M/18M TX power for OFDM 6M/9M TX power for CCK5.5M/11M TX power for CCK1M/2M */ /* TX_PWR_CFG_1 ~ TX_PWR_CFG_4 */ RTMP_IO_WRITE32(pAd, TX_PWR_CFG_0 + i*4, TxPwr[i]); } } } /* ======================================================================== Routine Description: Write TxWI for ATE mode. Return Value: None ======================================================================== */ #ifdef RTMP_MAC_PCI static VOID ATEWriteTxWI( IN PRTMP_ADAPTER pAd, IN PTXWI_STRUC pOutTxWI, IN BOOLEAN FRAG, IN BOOLEAN CFACK, IN BOOLEAN InsTimestamp, IN BOOLEAN AMPDU, IN BOOLEAN Ack, IN BOOLEAN NSeq, // HW new a sequence. IN UCHAR BASize, IN UCHAR WCID, IN ULONG Length, IN UCHAR PID, IN UCHAR TID, IN UCHAR TxRate, IN UCHAR Txopmode, IN BOOLEAN CfAck, IN HTTRANSMIT_SETTING *pTransmit) { TXWI_STRUC TxWI; PTXWI_STRUC pTxWI; // // Always use Long preamble before verifiation short preamble functionality works well. // Todo: remove the following line if short preamble functionality works // OPSTATUS_CLEAR_FLAG(pAd, fOP_STATUS_SHORT_PREAMBLE_INUSED); NdisZeroMemory(&TxWI, TXWI_SIZE); pTxWI = &TxWI; pTxWI->FRAG= FRAG; pTxWI->CFACK = CFACK; pTxWI->TS= InsTimestamp; pTxWI->AMPDU = AMPDU; pTxWI->ACK = Ack; pTxWI->txop= Txopmode; pTxWI->NSEQ = NSeq; // John tune the performace with Intel Client in 20 MHz performance if ( BASize >7 ) BASize =7; pTxWI->BAWinSize = BASize; pTxWI->WirelessCliID = WCID; pTxWI->MPDUtotalByteCount = Length; pTxWI->PacketId = PID; // If CCK or OFDM, BW must be 20 pTxWI->BW = (pTransmit->field.MODE <= MODE_OFDM) ? (BW_20) : (pTransmit->field.BW); pTxWI->ShortGI = pTransmit->field.ShortGI; pTxWI->STBC = pTransmit->field.STBC; pTxWI->MCS = pTransmit->field.MCS; pTxWI->PHYMODE = pTransmit->field.MODE; pTxWI->CFACK = CfAck; pTxWI->MIMOps = 0; pTxWI->MpduDensity = 0; pTxWI->PacketId = pTxWI->MCS; NdisMoveMemory(pOutTxWI, &TxWI, sizeof(TXWI_STRUC)); return; } #endif // RTMP_MAC_PCI // /* ======================================================================== Routine Description: Disable protection for ATE. ======================================================================== */ VOID ATEDisableAsicProtect( IN PRTMP_ADAPTER pAd) { PROT_CFG_STRUC ProtCfg, ProtCfg4; UINT32 Protect[6]; USHORT offset; UCHAR i; UINT32 MacReg = 0; // Config ASIC RTS threshold register RTMP_IO_READ32(pAd, TX_RTS_CFG, &MacReg); MacReg &= 0xFF0000FF; MacReg |= (pAd->CommonCfg.RtsThreshold << 8); RTMP_IO_WRITE32(pAd, TX_RTS_CFG, MacReg); // Initial common protection settings RTMPZeroMemory(Protect, sizeof(Protect)); ProtCfg4.word = 0; ProtCfg.word = 0; ProtCfg.field.TxopAllowGF40 = 1; ProtCfg.field.TxopAllowGF20 = 1; ProtCfg.field.TxopAllowMM40 = 1; ProtCfg.field.TxopAllowMM20 = 1; ProtCfg.field.TxopAllowOfdm = 1; ProtCfg.field.TxopAllowCck = 1; ProtCfg.field.RTSThEn = 1; ProtCfg.field.ProtectNav = ASIC_SHORTNAV; // Handle legacy(B/G) protection ProtCfg.field.ProtectRate = pAd->CommonCfg.RtsRate; ProtCfg.field.ProtectCtrl = 0; Protect[0] = ProtCfg.word; Protect[1] = ProtCfg.word; // NO PROTECT // 1.All STAs in the BSS are 20/40 MHz HT // 2. in ai 20/40MHz BSS // 3. all STAs are 20MHz in a 20MHz BSS // Pure HT. no protection. // MM20_PROT_CFG // Reserved (31:27) // PROT_TXOP(25:20) -- 010111 // PROT_NAV(19:18) -- 01 (Short NAV protection) // PROT_CTRL(17:16) -- 00 (None) // PROT_RATE(15:0) -- 0x4004 (OFDM 24M) Protect[2] = 0x01744004; // MM40_PROT_CFG // Reserved (31:27) // PROT_TXOP(25:20) -- 111111 // PROT_NAV(19:18) -- 01 (Short NAV protection) // PROT_CTRL(17:16) -- 00 (None) // PROT_RATE(15:0) -- 0x4084 (duplicate OFDM 24M) Protect[3] = 0x03f44084; // CF20_PROT_CFG // Reserved (31:27) // PROT_TXOP(25:20) -- 010111 // PROT_NAV(19:18) -- 01 (Short NAV protection) // PROT_CTRL(17:16) -- 00 (None) // PROT_RATE(15:0) -- 0x4004 (OFDM 24M) Protect[4] = 0x01744004; // CF40_PROT_CFG // Reserved (31:27) // PROT_TXOP(25:20) -- 111111 // PROT_NAV(19:18) -- 01 (Short NAV protection) // PROT_CTRL(17:16) -- 00 (None) // PROT_RATE(15:0) -- 0x4084 (duplicate OFDM 24M) Protect[5] = 0x03f44084; pAd->CommonCfg.IOTestParm.bRTSLongProtOn = FALSE; offset = CCK_PROT_CFG; for (i = 0;i < 6;i++) RTMP_IO_WRITE32(pAd, offset + i*4, Protect[i]); } /* There are two ways to convert Rssi */ /* the way used with GET_LNA_GAIN() */ CHAR ATEConvertToRssi( IN PRTMP_ADAPTER pAd, IN CHAR Rssi, IN UCHAR RssiNumber) { UCHAR RssiOffset, LNAGain; // Rssi equals to zero should be an invalid value if (Rssi == 0) return -99; LNAGain = GET_LNA_GAIN(pAd); if (pAd->LatchRfRegs.Channel > 14) { if (RssiNumber == 0) RssiOffset = pAd->ARssiOffset0; else if (RssiNumber == 1) RssiOffset = pAd->ARssiOffset1; else RssiOffset = pAd->ARssiOffset2; } else { if (RssiNumber == 0) RssiOffset = pAd->BGRssiOffset0; else if (RssiNumber == 1) RssiOffset = pAd->BGRssiOffset1; else RssiOffset = pAd->BGRssiOffset2; } return (-12 - RssiOffset - LNAGain - Rssi); } /* ======================================================================== Routine Description: Set Japan filter coefficients if needed. Note: This routine should only be called when entering TXFRAME mode or TXCONT mode. ======================================================================== */ static VOID SetJapanFilter( IN PRTMP_ADAPTER pAd) { UCHAR BbpData = 0; // // If Channel=14 and Bandwidth=20M and Mode=CCK, set BBP R4 bit5=1 // (Japan Tx filter coefficients)when (TXFRAME or TXCONT). // ATE_BBP_IO_READ8_BY_REG_ID(pAd, BBP_R4, &BbpData); if ((pAd->ate.TxWI.PHYMODE == MODE_CCK) && (pAd->ate.Channel == 14) && (pAd->ate.TxWI.BW == BW_20)) { BbpData |= 0x20; // turn on ATEDBGPRINT(RT_DEBUG_TRACE, ("SetJapanFilter!!!\n")); } else { BbpData &= 0xdf; // turn off ATEDBGPRINT(RT_DEBUG_TRACE, ("ClearJapanFilter!!!\n")); } ATE_BBP_IO_WRITE8_BY_REG_ID(pAd, BBP_R4, BbpData); } VOID ATESampleRssi( IN PRTMP_ADAPTER pAd, IN PRXWI_STRUC pRxWI) { /* There are two ways to collect RSSI. */ // pAd->LastRxRate = (USHORT)((pRxWI->MCS) + (pRxWI->BW <<7) + (pRxWI->ShortGI <<8)+ (pRxWI->PHYMODE <<14)) ; if (pRxWI->RSSI0 != 0) { pAd->ate.LastRssi0 = ATEConvertToRssi(pAd, (CHAR) pRxWI->RSSI0, RSSI_0); pAd->ate.AvgRssi0X8 = (pAd->ate.AvgRssi0X8 - pAd->ate.AvgRssi0) + pAd->ate.LastRssi0; pAd->ate.AvgRssi0 = pAd->ate.AvgRssi0X8 >> 3; } if (pRxWI->RSSI1 != 0) { pAd->ate.LastRssi1 = ATEConvertToRssi(pAd, (CHAR) pRxWI->RSSI1, RSSI_1); pAd->ate.AvgRssi1X8 = (pAd->ate.AvgRssi1X8 - pAd->ate.AvgRssi1) + pAd->ate.LastRssi1; pAd->ate.AvgRssi1 = pAd->ate.AvgRssi1X8 >> 3; } if (pRxWI->RSSI2 != 0) { pAd->ate.LastRssi2 = ATEConvertToRssi(pAd, (CHAR) pRxWI->RSSI2, RSSI_2); pAd->ate.AvgRssi2X8 = (pAd->ate.AvgRssi2X8 - pAd->ate.AvgRssi2) + pAd->ate.LastRssi2; pAd->ate.AvgRssi2 = pAd->ate.AvgRssi2X8 >> 3; } pAd->ate.LastSNR0 = (CHAR)(pRxWI->SNR0);// CHAR ==> UCHAR ? pAd->ate.LastSNR1 = (CHAR)(pRxWI->SNR1);// CHAR ==> UCHAR ? pAd->ate.NumOfAvgRssiSample ++; } #ifdef CONFIG_STA_SUPPORT VOID RTMPStationStop( IN PRTMP_ADAPTER pAd) { // BOOLEAN Cancelled; ATEDBGPRINT(RT_DEBUG_TRACE, ("==> RTMPStationStop\n")); // For rx statistics, we need to keep this timer running. // RTMPCancelTimer(&pAd->Mlme.PeriodicTimer, &Cancelled); ATEDBGPRINT(RT_DEBUG_TRACE, ("<== RTMPStationStop\n")); } VOID RTMPStationStart( IN PRTMP_ADAPTER pAd) { ATEDBGPRINT(RT_DEBUG_TRACE, ("==> RTMPStationStart\n")); #ifdef RTMP_MAC_PCI pAd->Mlme.CntlMachine.CurrState = CNTL_IDLE; /* We did not cancel this timer when entering ATE mode. */ // RTMPSetTimer(&pAd->Mlme.PeriodicTimer, MLME_TASK_EXEC_INTV); #endif // RTMP_MAC_PCI // ATEDBGPRINT(RT_DEBUG_TRACE, ("<== RTMPStationStart\n")); } #endif // CONFIG_STA_SUPPORT // /* ========================================================================== Description: Setup Frame format. NOTE: This routine should only be used in ATE mode. ========================================================================== */ #ifdef RTMP_MAC_PCI static INT ATESetUpFrame( IN PRTMP_ADAPTER pAd, IN UINT32 TxIdx) { UINT j; PTXD_STRUC pTxD; #ifdef RT_BIG_ENDIAN PTXD_STRUC pDestTxD; TXD_STRUC TxD; #endif PNDIS_PACKET pPacket; PUCHAR pDest; PVOID AllocVa; NDIS_PHYSICAL_ADDRESS AllocPa; HTTRANSMIT_SETTING TxHTPhyMode; PRTMP_TX_RING pTxRing = &pAd->TxRing[QID_AC_BE]; PTXWI_STRUC pTxWI = (PTXWI_STRUC) pTxRing->Cell[TxIdx].DmaBuf.AllocVa; PUCHAR pDMAHeaderBufVA = (PUCHAR) pTxRing->Cell[TxIdx].DmaBuf.AllocVa; #ifdef RALINK_28xx_QA PHEADER_802_11 pHeader80211; #endif // RALINK_28xx_QA // if (pAd->ate.bQATxStart == TRUE) { // always use QID_AC_BE and FIFO_EDCA // fill TxWI TxHTPhyMode.field.BW = pAd->ate.TxWI.BW; TxHTPhyMode.field.ShortGI = pAd->ate.TxWI.ShortGI; TxHTPhyMode.field.STBC = 0; TxHTPhyMode.field.MCS = pAd->ate.TxWI.MCS; TxHTPhyMode.field.MODE = pAd->ate.TxWI.PHYMODE; ATEWriteTxWI(pAd, pTxWI, pAd->ate.TxWI.FRAG, pAd->ate.TxWI.CFACK, pAd->ate.TxWI.TS, pAd->ate.TxWI.AMPDU, pAd->ate.TxWI.ACK, pAd->ate.TxWI.NSEQ, pAd->ate.TxWI.BAWinSize, 0, pAd->ate.TxWI.MPDUtotalByteCount, pAd->ate.TxWI.PacketId, 0, 0, pAd->ate.TxWI.txop/*IFS_HTTXOP*/, pAd->ate.TxWI.CFACK/*FALSE*/, &TxHTPhyMode); } else { TxHTPhyMode.field.BW = pAd->ate.TxWI.BW; TxHTPhyMode.field.ShortGI = pAd->ate.TxWI.ShortGI; TxHTPhyMode.field.STBC = 0; TxHTPhyMode.field.MCS = pAd->ate.TxWI.MCS; TxHTPhyMode.field.MODE = pAd->ate.TxWI.PHYMODE; ATEWriteTxWI(pAd, pTxWI, FALSE, FALSE, FALSE, FALSE, FALSE, FALSE, 4, 0, pAd->ate.TxLength, 0, 0, 0, IFS_HTTXOP, FALSE, &TxHTPhyMode); } // fill 802.11 header #ifdef RALINK_28xx_QA if (pAd->ate.bQATxStart == TRUE) { NdisMoveMemory(pDMAHeaderBufVA+TXWI_SIZE, pAd->ate.Header, pAd->ate.HLen); } else #endif // RALINK_28xx_QA // { NdisMoveMemory(pDMAHeaderBufVA+TXWI_SIZE, TemplateFrame, LENGTH_802_11); NdisMoveMemory(pDMAHeaderBufVA+TXWI_SIZE+4, pAd->ate.Addr1, ETH_LENGTH_OF_ADDRESS); NdisMoveMemory(pDMAHeaderBufVA+TXWI_SIZE+10, pAd->ate.Addr2, ETH_LENGTH_OF_ADDRESS); NdisMoveMemory(pDMAHeaderBufVA+TXWI_SIZE+16, pAd->ate.Addr3, ETH_LENGTH_OF_ADDRESS); } #ifdef RT_BIG_ENDIAN RTMPFrameEndianChange(pAd, (((PUCHAR)pDMAHeaderBufVA)+TXWI_SIZE), DIR_READ, FALSE); #endif // RT_BIG_ENDIAN // /* alloc buffer for payload */ #ifdef RALINK_28xx_QA if (pAd->ate.bQATxStart == TRUE) { pPacket = RTMP_AllocateRxPacketBuffer(pAd, pAd->ate.DLen + 0x100, FALSE, &AllocVa, &AllocPa); } else #endif // RALINK_28xx_QA // { pPacket = RTMP_AllocateRxPacketBuffer(pAd, pAd->ate.TxLength, FALSE, &AllocVa, &AllocPa); } if (pPacket == NULL) { pAd->ate.TxCount = 0; ATEDBGPRINT(RT_DEBUG_TRACE, ("%s fail to alloc packet space.\n", __FUNCTION__)); return -1; } pTxRing->Cell[TxIdx].pNextNdisPacket = pPacket; pDest = (PUCHAR) AllocVa; #ifdef RALINK_28xx_QA if (pAd->ate.bQATxStart == TRUE) { RTPKT_TO_OSPKT(pPacket)->len = pAd->ate.DLen; } else #endif // RALINK_28xx_QA // { RTPKT_TO_OSPKT(pPacket)->len = pAd->ate.TxLength - LENGTH_802_11; } // prepare frame payload #ifdef RALINK_28xx_QA if (pAd->ate.bQATxStart == TRUE) { // copy pattern if ((pAd->ate.PLen != 0)) { int j; for (j = 0; j < pAd->ate.DLen; j+=pAd->ate.PLen) { memcpy(RTPKT_TO_OSPKT(pPacket)->data + j, pAd->ate.Pattern, pAd->ate.PLen); } } } else #endif // RALINK_28xx_QA // { for (j = 0; j < RTPKT_TO_OSPKT(pPacket)->len; j++) { pDest[j] = 0xA5; } } /* build Tx Descriptor */ #ifndef RT_BIG_ENDIAN pTxD = (PTXD_STRUC) pTxRing->Cell[TxIdx].AllocVa; #else pDestTxD = (PTXD_STRUC)pTxRing->Cell[TxIdx].AllocVa; TxD = *pDestTxD; pTxD = &TxD; #endif // !RT_BIG_ENDIAN // #ifdef RALINK_28xx_QA if (pAd->ate.bQATxStart == TRUE) { // prepare TxD NdisZeroMemory(pTxD, TXD_SIZE); RTMPWriteTxDescriptor(pAd, pTxD, FALSE, FIFO_EDCA); // build TX DESC pTxD->SDPtr0 = RTMP_GetPhysicalAddressLow(pTxRing->Cell[TxIdx].DmaBuf.AllocPa); pTxD->SDLen0 = TXWI_SIZE + pAd->ate.HLen; pTxD->LastSec0 = 0; pTxD->SDPtr1 = AllocPa; pTxD->SDLen1 = RTPKT_TO_OSPKT(pPacket)->len; pTxD->LastSec1 = 1; pDest = (PUCHAR)pTxWI; pDest += TXWI_SIZE; pHeader80211 = (PHEADER_802_11)pDest; // modify sequence number... if (pAd->ate.TxDoneCount == 0) { pAd->ate.seq = pHeader80211->Sequence; } else pHeader80211->Sequence = ++pAd->ate.seq; } else #endif // RALINK_28xx_QA // { NdisZeroMemory(pTxD, TXD_SIZE); RTMPWriteTxDescriptor(pAd, pTxD, FALSE, FIFO_EDCA); // build TX DESC pTxD->SDPtr0 = RTMP_GetPhysicalAddressLow (pTxRing->Cell[TxIdx].DmaBuf.AllocPa); pTxD->SDLen0 = TXWI_SIZE + LENGTH_802_11; pTxD->LastSec0 = 0; pTxD->SDPtr1 = AllocPa; pTxD->SDLen1 = RTPKT_TO_OSPKT(pPacket)->len; pTxD->LastSec1 = 1; } #ifdef RT_BIG_ENDIAN RTMPWIEndianChange((PUCHAR)pTxWI, TYPE_TXWI); RTMPFrameEndianChange(pAd, (((PUCHAR)pDMAHeaderBufVA)+TXWI_SIZE), DIR_WRITE, FALSE); RTMPDescriptorEndianChange((PUCHAR)pTxD, TYPE_TXD); WriteBackToDescriptor((PUCHAR)pDestTxD, (PUCHAR)pTxD, FALSE, TYPE_TXD); #endif // RT_BIG_ENDIAN // return 0; } /*=======================End of RTMP_MAC_PCI =======================*/ #endif // RTMP_MAC_PCI // VOID rt_ee_read_all(PRTMP_ADAPTER pAd, USHORT *Data) { USHORT i; USHORT value; for (i = 0 ; i < EEPROM_SIZE/2 ; ) { /* "value" is especially for some compilers... */ RT28xx_EEPROM_READ16(pAd, i*2, value); Data[i] = value; i++; } } VOID rt_ee_write_all(PRTMP_ADAPTER pAd, USHORT *Data) { USHORT i; USHORT value; for (i = 0 ; i < EEPROM_SIZE/2 ; ) { /* "value" is especially for some compilers... */ value = Data[i]; RT28xx_EEPROM_WRITE16(pAd, i*2, value); i++; } } #ifdef RALINK_28xx_QA VOID ATE_QA_Statistics( IN PRTMP_ADAPTER pAd, IN PRXWI_STRUC pRxWI, IN PRT28XX_RXD_STRUC pRxD, IN PHEADER_802_11 pHeader) { // update counter first if (pHeader != NULL) { if (pHeader->FC.Type == BTYPE_DATA) { if (pRxD->U2M) pAd->ate.U2M++; else pAd->ate.OtherData++; } else if (pHeader->FC.Type == BTYPE_MGMT) { if (pHeader->FC.SubType == SUBTYPE_BEACON) pAd->ate.Beacon++; else pAd->ate.OtherCount++; } else if (pHeader->FC.Type == BTYPE_CNTL) { pAd->ate.OtherCount++; } } pAd->ate.RSSI0 = pRxWI->RSSI0; pAd->ate.RSSI1 = pRxWI->RSSI1; pAd->ate.RSSI2 = pRxWI->RSSI2; pAd->ate.SNR0 = pRxWI->SNR0; pAd->ate.SNR1 = pRxWI->SNR1; } /* command id with Cmd Type == 0x0008(for 28xx)/0x0005(for iNIC) */ #define RACFG_CMD_RF_WRITE_ALL 0x0000 #define RACFG_CMD_E2PROM_READ16 0x0001 #define RACFG_CMD_E2PROM_WRITE16 0x0002 #define RACFG_CMD_E2PROM_READ_ALL 0x0003 #define RACFG_CMD_E2PROM_WRITE_ALL 0x0004 #define RACFG_CMD_IO_READ 0x0005 #define RACFG_CMD_IO_WRITE 0x0006 #define RACFG_CMD_IO_READ_BULK 0x0007 #define RACFG_CMD_BBP_READ8 0x0008 #define RACFG_CMD_BBP_WRITE8 0x0009 #define RACFG_CMD_BBP_READ_ALL 0x000a #define RACFG_CMD_GET_COUNTER 0x000b #define RACFG_CMD_CLEAR_COUNTER 0x000c #define RACFG_CMD_RSV1 0x000d #define RACFG_CMD_RSV2 0x000e #define RACFG_CMD_RSV3 0x000f #define RACFG_CMD_TX_START 0x0010 #define RACFG_CMD_GET_TX_STATUS 0x0011 #define RACFG_CMD_TX_STOP 0x0012 #define RACFG_CMD_RX_START 0x0013 #define RACFG_CMD_RX_STOP 0x0014 #define RACFG_CMD_GET_NOISE_LEVEL 0x0015 #define RACFG_CMD_ATE_START 0x0080 #define RACFG_CMD_ATE_STOP 0x0081 #define RACFG_CMD_ATE_START_TX_CARRIER 0x0100 #define RACFG_CMD_ATE_START_TX_CONT 0x0101 #define RACFG_CMD_ATE_START_TX_FRAME 0x0102 #define RACFG_CMD_ATE_SET_BW 0x0103 #define RACFG_CMD_ATE_SET_TX_POWER0 0x0104 #define RACFG_CMD_ATE_SET_TX_POWER1 0x0105 #define RACFG_CMD_ATE_SET_FREQ_OFFSET 0x0106 #define RACFG_CMD_ATE_GET_STATISTICS 0x0107 #define RACFG_CMD_ATE_RESET_COUNTER 0x0108 #define RACFG_CMD_ATE_SEL_TX_ANTENNA 0x0109 #define RACFG_CMD_ATE_SEL_RX_ANTENNA 0x010a #define RACFG_CMD_ATE_SET_PREAMBLE 0x010b #define RACFG_CMD_ATE_SET_CHANNEL 0x010c #define RACFG_CMD_ATE_SET_ADDR1 0x010d #define RACFG_CMD_ATE_SET_ADDR2 0x010e #define RACFG_CMD_ATE_SET_ADDR3 0x010f #define RACFG_CMD_ATE_SET_RATE 0x0110 #define RACFG_CMD_ATE_SET_TX_FRAME_LEN 0x0111 #define RACFG_CMD_ATE_SET_TX_FRAME_COUNT 0x0112 #define RACFG_CMD_ATE_START_RX_FRAME 0x0113 #define RACFG_CMD_ATE_E2PROM_READ_BULK 0x0114 #define RACFG_CMD_ATE_E2PROM_WRITE_BULK 0x0115 #define RACFG_CMD_ATE_IO_WRITE_BULK 0x0116 #define RACFG_CMD_ATE_BBP_READ_BULK 0x0117 #define RACFG_CMD_ATE_BBP_WRITE_BULK 0x0118 #define RACFG_CMD_ATE_RF_READ_BULK 0x0119 #define RACFG_CMD_ATE_RF_WRITE_BULK 0x011a static VOID memcpy_exl(PRTMP_ADAPTER pAd, UCHAR *dst, UCHAR *src, ULONG len); static VOID memcpy_exs(PRTMP_ADAPTER pAd, UCHAR *dst, UCHAR *src, ULONG len); static VOID RTMP_IO_READ_BULK(PRTMP_ADAPTER pAd, UCHAR *dst, UCHAR *src, UINT32 len); VOID RtmpDoAte( IN PRTMP_ADAPTER pAdapter, IN struct iwreq *wrq) { USHORT Command_Id; INT Status = NDIS_STATUS_SUCCESS; struct ate_racfghdr *pRaCfg; if ((pRaCfg = kmalloc(sizeof(struct ate_racfghdr), GFP_KERNEL)) == NULL) { Status = -EINVAL; return; } NdisZeroMemory(pRaCfg, sizeof(struct ate_racfghdr)); if (copy_from_user((PUCHAR)pRaCfg, wrq->u.data.pointer, wrq->u.data.length)) { Status = -EFAULT; kfree(pRaCfg); return; } Command_Id = ntohs(pRaCfg->command_id); ATEDBGPRINT(RT_DEBUG_TRACE,("\n%s: Command_Id = 0x%04x !\n", __FUNCTION__, Command_Id)); switch (Command_Id) { /* We will get this command when QA starts. */ case RACFG_CMD_ATE_START: Status=DO_RACFG_CMD_ATE_START(pAdapter,wrq,pRaCfg); break; /* We will get this command either QA is closed or ated is killed by user. */ case RACFG_CMD_ATE_STOP: Status=DO_RACFG_CMD_ATE_STOP(pAdapter,wrq,pRaCfg); break; case RACFG_CMD_RF_WRITE_ALL: Status=DO_RACFG_CMD_RF_WRITE_ALL(pAdapter,wrq,pRaCfg); break; case RACFG_CMD_E2PROM_READ16: Status=DO_RACFG_CMD_E2PROM_READ16(pAdapter,wrq,pRaCfg); break; case RACFG_CMD_E2PROM_WRITE16: Status=DO_RACFG_CMD_E2PROM_WRITE16(pAdapter,wrq,pRaCfg); break; case RACFG_CMD_E2PROM_READ_ALL: Status=DO_RACFG_CMD_E2PROM_READ_ALL(pAdapter,wrq,pRaCfg); break; case RACFG_CMD_E2PROM_WRITE_ALL: Status=DO_RACFG_CMD_E2PROM_WRITE_ALL(pAdapter,wrq,pRaCfg); break; case RACFG_CMD_IO_READ: Status=DO_RACFG_CMD_IO_READ(pAdapter,wrq,pRaCfg); break; case RACFG_CMD_IO_WRITE: Status=DO_RACFG_CMD_IO_WRITE(pAdapter,wrq,pRaCfg); break; case RACFG_CMD_IO_READ_BULK: Status=DO_RACFG_CMD_IO_READ_BULK(pAdapter,wrq,pRaCfg); break; case RACFG_CMD_BBP_READ8: Status=DO_RACFG_CMD_BBP_READ8(pAdapter,wrq,pRaCfg); break; case RACFG_CMD_BBP_WRITE8: Status=DO_RACFG_CMD_BBP_WRITE8(pAdapter,wrq,pRaCfg); break; case RACFG_CMD_BBP_READ_ALL: Status=DO_RACFG_CMD_BBP_READ_ALL(pAdapter,wrq,pRaCfg); break; case RACFG_CMD_ATE_E2PROM_READ_BULK: Status=DO_RACFG_CMD_ATE_E2PROM_READ_BULK(pAdapter,wrq,pRaCfg); break; case RACFG_CMD_ATE_E2PROM_WRITE_BULK: Status=DO_RACFG_CMD_ATE_E2PROM_WRITE_BULK(pAdapter,wrq,pRaCfg); break; case RACFG_CMD_ATE_IO_WRITE_BULK: Status=DO_RACFG_CMD_ATE_IO_WRITE_BULK(pAdapter,wrq,pRaCfg); break; case RACFG_CMD_ATE_BBP_READ_BULK: Status=DO_RACFG_CMD_ATE_BBP_READ_BULK(pAdapter,wrq,pRaCfg); break; case RACFG_CMD_ATE_BBP_WRITE_BULK: Status=DO_RACFG_CMD_ATE_BBP_WRITE_BULK(pAdapter,wrq,pRaCfg); break; case RACFG_CMD_GET_NOISE_LEVEL: Status=DO_RACFG_CMD_GET_NOISE_LEVEL(pAdapter,wrq,pRaCfg); break; case RACFG_CMD_GET_COUNTER: Status=DO_RACFG_CMD_GET_COUNTER(pAdapter,wrq,pRaCfg); break; case RACFG_CMD_CLEAR_COUNTER: Status=DO_RACFG_CMD_CLEAR_COUNTER(pAdapter,wrq,pRaCfg); break; case RACFG_CMD_TX_START: Status=DO_RACFG_CMD_TX_START(pAdapter,wrq,pRaCfg); break; case RACFG_CMD_GET_TX_STATUS: Status=DO_RACFG_CMD_GET_TX_STATUS(pAdapter,wrq,pRaCfg); break; case RACFG_CMD_TX_STOP: Status=DO_RACFG_CMD_TX_STOP(pAdapter,wrq,pRaCfg); break; case RACFG_CMD_RX_START: Status=DO_RACFG_CMD_RX_START(pAdapter,wrq,pRaCfg); break; case RACFG_CMD_RX_STOP: Status=DO_RACFG_CMD_RX_STOP(pAdapter,wrq,pRaCfg); break; /* The following cases are for new ATE GUI(not QA). */ /*==================================================*/ case RACFG_CMD_ATE_START_TX_CARRIER: Status=DO_RACFG_CMD_ATE_START_TX_CARRIER(pAdapter,wrq,pRaCfg); break; case RACFG_CMD_ATE_START_TX_CONT: Status=DO_RACFG_CMD_ATE_START_TX_CONT(pAdapter,wrq,pRaCfg); break; case RACFG_CMD_ATE_START_TX_FRAME: Status=DO_RACFG_CMD_ATE_START_TX_FRAME(pAdapter,wrq,pRaCfg); break; case RACFG_CMD_ATE_SET_BW: Status=DO_RACFG_CMD_ATE_SET_BW(pAdapter,wrq,pRaCfg); break; case RACFG_CMD_ATE_SET_TX_POWER0: Status=DO_RACFG_CMD_ATE_SET_TX_POWER0(pAdapter,wrq,pRaCfg); break; case RACFG_CMD_ATE_SET_TX_POWER1: Status=DO_RACFG_CMD_ATE_SET_TX_POWER1(pAdapter,wrq,pRaCfg); break; case RACFG_CMD_ATE_SET_FREQ_OFFSET: Status=DO_RACFG_CMD_ATE_SET_TX_POWER1(pAdapter,wrq,pRaCfg); break; case RACFG_CMD_ATE_GET_STATISTICS: Status=DO_RACFG_CMD_ATE_GET_STATISTICS(pAdapter,wrq,pRaCfg); break; case RACFG_CMD_ATE_RESET_COUNTER: Status=DO_RACFG_CMD_ATE_RESET_COUNTER(pAdapter,wrq,pRaCfg); break; case RACFG_CMD_ATE_SEL_TX_ANTENNA: Status=DO_RACFG_CMD_ATE_SEL_TX_ANTENNA(pAdapter,wrq,pRaCfg); break; case RACFG_CMD_ATE_SEL_RX_ANTENNA: Status=DO_RACFG_CMD_ATE_SEL_TX_ANTENNA(pAdapter,wrq,pRaCfg); break; case RACFG_CMD_ATE_SET_PREAMBLE: Status=DO_RACFG_CMD_ATE_SET_PREAMBLE(pAdapter,wrq,pRaCfg); break; case RACFG_CMD_ATE_SET_CHANNEL: Status=DO_RACFG_CMD_ATE_SET_CHANNEL(pAdapter,wrq,pRaCfg); break; case RACFG_CMD_ATE_SET_ADDR1: Status=DO_RACFG_CMD_ATE_SET_ADDR1(pAdapter,wrq,pRaCfg); break; case RACFG_CMD_ATE_SET_ADDR2: Status=DO_RACFG_CMD_ATE_SET_ADDR2(pAdapter,wrq,pRaCfg); break; case RACFG_CMD_ATE_SET_ADDR3: Status=DO_RACFG_CMD_ATE_SET_ADDR3(pAdapter,wrq,pRaCfg); break; case RACFG_CMD_ATE_SET_RATE: Status=DO_RACFG_CMD_ATE_SET_RATE(pAdapter,wrq,pRaCfg); break; case RACFG_CMD_ATE_SET_TX_FRAME_LEN: Status=DO_RACFG_CMD_ATE_SET_TX_FRAME_LEN(pAdapter,wrq,pRaCfg); break; case RACFG_CMD_ATE_SET_TX_FRAME_COUNT: Status=DO_RACFG_CMD_ATE_SET_TX_FRAME_COUNT(pAdapter,wrq,pRaCfg); break; case RACFG_CMD_ATE_START_RX_FRAME: Status=DO_RACFG_CMD_ATE_START_RX_FRAME(pAdapter,wrq,pRaCfg); break; default: break; } ASSERT(pRaCfg != NULL); if (pRaCfg != NULL) kfree(pRaCfg); return; } VOID BubbleSort(INT32 n, INT32 a[]) { INT32 k, j, temp; for (k = n-1; k>0; k--) { for (j = 0; j a[j+1]) { temp = a[j]; a[j]=a[j+1]; a[j+1]=temp; } } } } VOID CalNoiseLevel(PRTMP_ADAPTER pAd, UCHAR channel, INT32 RSSI[3][10]) { INT32 RSSI0, RSSI1, RSSI2; CHAR Rssi0Offset, Rssi1Offset, Rssi2Offset; UCHAR BbpR50Rssi0 = 0, BbpR51Rssi1 = 0, BbpR52Rssi2 = 0; UCHAR Org_BBP66value = 0, Org_BBP69value = 0, Org_BBP70value = 0, data = 0; USHORT LNA_Gain = 0; INT32 j = 0; UCHAR Org_Channel = pAd->ate.Channel; USHORT GainValue = 0, OffsetValue = 0; ATE_BBP_IO_READ8_BY_REG_ID(pAd, BBP_R66, &Org_BBP66value); ATE_BBP_IO_READ8_BY_REG_ID(pAd, BBP_R69, &Org_BBP69value); ATE_BBP_IO_READ8_BY_REG_ID(pAd, BBP_R70, &Org_BBP70value); //********************************************************************** // Read the value of LNA gain and Rssi offset //********************************************************************** RT28xx_EEPROM_READ16(pAd, EEPROM_LNA_OFFSET, GainValue); // for Noise Level if (channel <= 14) { LNA_Gain = GainValue & 0x00FF; RT28xx_EEPROM_READ16(pAd, EEPROM_RSSI_BG_OFFSET, OffsetValue); Rssi0Offset = OffsetValue & 0x00FF; Rssi1Offset = (OffsetValue & 0xFF00) >> 8; RT28xx_EEPROM_READ16(pAd, (EEPROM_RSSI_BG_OFFSET + 2)/* 0x48 */, OffsetValue); Rssi2Offset = OffsetValue & 0x00FF; } else { LNA_Gain = (GainValue & 0xFF00) >> 8; RT28xx_EEPROM_READ16(pAd, EEPROM_RSSI_A_OFFSET, OffsetValue); Rssi0Offset = OffsetValue & 0x00FF; Rssi1Offset = (OffsetValue & 0xFF00) >> 8; RT28xx_EEPROM_READ16(pAd, (EEPROM_RSSI_A_OFFSET + 2)/* 0x4C */, OffsetValue); Rssi2Offset = OffsetValue & 0x00FF; } //********************************************************************** { pAd->ate.Channel = channel; ATEAsicSwitchChannel(pAd); mdelay(5); data = 0x10; ATE_BBP_IO_WRITE8_BY_REG_ID(pAd, BBP_R66, data); data = 0x40; ATE_BBP_IO_WRITE8_BY_REG_ID(pAd, BBP_R69, data); data = 0x40; ATE_BBP_IO_WRITE8_BY_REG_ID(pAd, BBP_R70, data); mdelay(5); // start Rx pAd->ate.bQARxStart = TRUE; Set_ATE_Proc(pAd, "RXFRAME"); mdelay(5); for (j = 0; j < 10; j++) { ATE_BBP_IO_READ8_BY_REG_ID(pAd, BBP_R50, &BbpR50Rssi0); ATE_BBP_IO_READ8_BY_REG_ID(pAd, BBP_R51, &BbpR51Rssi1); ATE_BBP_IO_READ8_BY_REG_ID(pAd, BBP_R52, &BbpR52Rssi2); mdelay(10); // calculate RSSI 0 if (BbpR50Rssi0 == 0) { RSSI0 = -100; } else { RSSI0 = (INT32)(-12 - BbpR50Rssi0 - LNA_Gain - Rssi0Offset); } RSSI[0][j] = RSSI0; if ( pAd->Antenna.field.RxPath >= 2 ) // 2R { // calculate RSSI 1 if (BbpR51Rssi1 == 0) { RSSI1 = -100; } else { RSSI1 = (INT32)(-12 - BbpR51Rssi1 - LNA_Gain - Rssi1Offset); } RSSI[1][j] = RSSI1; } if ( pAd->Antenna.field.RxPath >= 3 ) // 3R { // calculate RSSI 2 if (BbpR52Rssi2 == 0) RSSI2 = -100; else RSSI2 = (INT32)(-12 - BbpR52Rssi2 - LNA_Gain - Rssi2Offset); RSSI[2][j] = RSSI2; } } // stop Rx Set_ATE_Proc(pAd, "RXSTOP"); mdelay(5); BubbleSort(10, RSSI[0]); // 1R if ( pAd->Antenna.field.RxPath >= 2 ) // 2R { BubbleSort(10, RSSI[1]); } if ( pAd->Antenna.field.RxPath >= 3 ) // 3R { BubbleSort(10, RSSI[2]); } } pAd->ate.Channel = Org_Channel; ATEAsicSwitchChannel(pAd); // restore original value ATE_BBP_IO_WRITE8_BY_REG_ID(pAd, BBP_R66, Org_BBP66value); ATE_BBP_IO_WRITE8_BY_REG_ID(pAd, BBP_R69, Org_BBP69value); ATE_BBP_IO_WRITE8_BY_REG_ID(pAd, BBP_R70, Org_BBP70value); return; } BOOLEAN SyncTxRxConfig(PRTMP_ADAPTER pAd, USHORT offset, UCHAR value) { UCHAR tmp = 0, bbp_data = 0; if (ATE_ON(pAd)) { ATE_BBP_IO_READ8_BY_REG_ID(pAd, offset, &bbp_data); } else { RTMP_BBP_IO_READ8_BY_REG_ID(pAd, offset, &bbp_data); } /* confirm again */ ASSERT(bbp_data == value); switch (offset) { case BBP_R1: /* Need to synchronize tx configuration with legacy ATE. */ tmp = (bbp_data & ((1 << 4) | (1 << 3))/* 0x18 */) >> 3; switch (tmp) { /* The BBP R1 bit[4:3] = 2 :: Both DACs will be used by QA. */ case 2: /* All */ pAd->ate.TxAntennaSel = 0; break; /* The BBP R1 bit[4:3] = 0 :: DAC 0 will be used by QA. */ case 0: /* Antenna one */ pAd->ate.TxAntennaSel = 1; break; /* The BBP R1 bit[4:3] = 1 :: DAC 1 will be used by QA. */ case 1: /* Antenna two */ pAd->ate.TxAntennaSel = 2; break; default: DBGPRINT(RT_DEBUG_TRACE, ("%s -- Sth. wrong! : return FALSE; \n", __FUNCTION__)); return FALSE; } break;/* case BBP_R1 */ case BBP_R3: /* Need to synchronize rx configuration with legacy ATE. */ tmp = (bbp_data & ((1 << 1) | (1 << 0))/* 0x03 */); switch(tmp) { /* The BBP R3 bit[1:0] = 3 :: All ADCs will be used by QA. */ case 3: /* All */ pAd->ate.RxAntennaSel = 0; break; /* The BBP R3 bit[1:0] = 0 :: ADC 0 will be used by QA, unless the BBP R3 bit[4:3] = 2 */ case 0: /* Antenna one */ pAd->ate.RxAntennaSel = 1; tmp = ((bbp_data & ((1 << 4) | (1 << 3))/* 0x03 */) >> 3); if (tmp == 2)// 3R { /* Default : All ADCs will be used by QA */ pAd->ate.RxAntennaSel = 0; } break; /* The BBP R3 bit[1:0] = 1 :: ADC 1 will be used by QA. */ case 1: /* Antenna two */ pAd->ate.RxAntennaSel = 2; break; /* The BBP R3 bit[1:0] = 2 :: ADC 2 will be used by QA. */ case 2: /* Antenna three */ pAd->ate.RxAntennaSel = 3; break; default: DBGPRINT(RT_DEBUG_ERROR, ("%s -- Impossible! : return FALSE; \n", __FUNCTION__)); return FALSE; } break;/* case BBP_R3 */ default: DBGPRINT(RT_DEBUG_ERROR, ("%s -- Sth. wrong! : return FALSE; \n", __FUNCTION__)); return FALSE; } return TRUE; } static VOID memcpy_exl(PRTMP_ADAPTER pAd, UCHAR *dst, UCHAR *src, ULONG len) { ULONG i, Value = 0; ULONG *pDst, *pSrc; UCHAR *p8; p8 = src; pDst = (ULONG *) dst; pSrc = (ULONG *) src; for (i = 0 ; i < (len/4); i++) { /* For alignment issue, we need a variable "Value". */ memmove(&Value, pSrc, 4); Value = htonl(Value); memmove(pDst, &Value, 4); pDst++; pSrc++; } if ((len % 4) != 0) { /* wish that it will never reach here */ memmove(&Value, pSrc, (len % 4)); Value = htonl(Value); memmove(pDst, &Value, (len % 4)); } } static VOID memcpy_exs(PRTMP_ADAPTER pAd, UCHAR *dst, UCHAR *src, ULONG len) { ULONG i; UCHAR *pDst, *pSrc; pDst = dst; pSrc = src; for (i = 0; i < (len/2); i++) { memmove(pDst, pSrc, 2); *((USHORT *)pDst) = htons(*((USHORT *)pDst)); pDst+=2; pSrc+=2; } if ((len % 2) != 0) { memmove(pDst, pSrc, 1); } } static VOID RTMP_IO_READ_BULK(PRTMP_ADAPTER pAd, UCHAR *dst, UCHAR *src, UINT32 len) { UINT32 i, Value; UINT32 *pDst, *pSrc; pDst = (UINT32 *) dst; pSrc = (UINT32 *) src; for (i = 0 ; i < (len/4); i++) { RTMP_IO_READ32(pAd, (ULONG)pSrc, &Value); Value = htonl(Value); memmove(pDst, &Value, 4); pDst++; pSrc++; } return; } INT Set_TxStop_Proc( IN PRTMP_ADAPTER pAd, IN PSTRING arg) { ATEDBGPRINT(RT_DEBUG_TRACE,("Set_TxStop_Proc\n")); if (Set_ATE_Proc(pAd, "TXSTOP")) { return TRUE; } else { return FALSE; } } INT Set_RxStop_Proc( IN PRTMP_ADAPTER pAd, IN PSTRING arg) { ATEDBGPRINT(RT_DEBUG_TRACE,("Set_RxStop_Proc\n")); if (Set_ATE_Proc(pAd, "RXSTOP")) { return TRUE; } else { return FALSE; } } #ifdef DBG INT Set_EERead_Proc( IN PRTMP_ADAPTER pAd, IN PSTRING arg) { USHORT buffer[EEPROM_SIZE/2]; USHORT *p; INT i; rt_ee_read_all(pAd, (USHORT *)buffer); p = buffer; for (i = 0; i < (EEPROM_SIZE/2); i++) { ate_print(KERN_EMERG "%4.4x ", *p); if (((i+1) % 16) == 0) ate_print(KERN_EMERG "\n"); p++; } return TRUE; } INT Set_EEWrite_Proc( IN PRTMP_ADAPTER pAd, IN PSTRING arg) { USHORT offset = 0, value; PSTRING p2 = arg; while ((*p2 != ':') && (*p2 != '\0')) { p2++; } if (*p2 == ':') { A2Hex(offset, arg); A2Hex(value, p2 + 1); } else { A2Hex(value, arg); } if (offset >= EEPROM_SIZE) { ate_print(KERN_EMERG "Offset can not exceed EEPROM_SIZE( == 0x%04x)\n", EEPROM_SIZE); return FALSE; } RT28xx_EEPROM_WRITE16(pAd, offset, value); return TRUE; } INT Set_BBPRead_Proc( IN PRTMP_ADAPTER pAd, IN PSTRING arg) { UCHAR value = 0, offset; A2Hex(offset, arg); if (ATE_ON(pAd)) { ATE_BBP_IO_READ8_BY_REG_ID(pAd, offset, &value); } else { RTMP_BBP_IO_READ8_BY_REG_ID(pAd, offset, &value); } ate_print(KERN_EMERG "%x\n", value); return TRUE; } INT Set_BBPWrite_Proc( IN PRTMP_ADAPTER pAd, IN PSTRING arg) { USHORT offset = 0; PSTRING p2 = arg; UCHAR value; while ((*p2 != ':') && (*p2 != '\0')) { p2++; } if (*p2 == ':') { A2Hex(offset, arg); A2Hex(value, p2 + 1); } else { A2Hex(value, arg); } if (ATE_ON(pAd)) { ATE_BBP_IO_WRITE8_BY_REG_ID(pAd, offset, value); } else { RTMP_BBP_IO_WRITE8_BY_REG_ID(pAd, offset, value); } return TRUE; } INT Set_RFWrite_Proc( IN PRTMP_ADAPTER pAd, IN PSTRING arg) { PSTRING p2, p3, p4; UINT32 R1, R2, R3, R4; p2 = arg; while ((*p2 != ':') && (*p2 != '\0')) { p2++; } if (*p2 != ':') return FALSE; p3 = p2 + 1; while((*p3 != ':') && (*p3 != '\0')) { p3++; } if (*p3 != ':') return FALSE; p4 = p3 + 1; while ((*p4 != ':') && (*p4 != '\0')) { p4++; } if (*p4 != ':') return FALSE; A2Hex(R1, arg); A2Hex(R2, p2 + 1); A2Hex(R3, p3 + 1); A2Hex(R4, p4 + 1); RTMP_RF_IO_WRITE32(pAd, R1); RTMP_RF_IO_WRITE32(pAd, R2); RTMP_RF_IO_WRITE32(pAd, R3); RTMP_RF_IO_WRITE32(pAd, R4); return TRUE; } #endif // DBG // #endif // RALINK_28xx_QA // #ifdef RALINK_28xx_QA #define LEN_OF_ARG 16 #define RESPONSE_TO_GUI(__pRaCfg, __pwrq, __Length, __Status) \ (__pRaCfg)->length = htons((__Length)); \ (__pRaCfg)->status = htons((__Status)); \ (__pwrq)->u.data.length = sizeof((__pRaCfg)->magic_no) + sizeof((__pRaCfg)->command_type) \ + sizeof((__pRaCfg)->command_id) + sizeof((__pRaCfg)->length) \ + sizeof((__pRaCfg)->sequence) + ntohs((__pRaCfg)->length); \ ATEDBGPRINT(RT_DEBUG_TRACE, ("wrq->u.data.length = %d\n", (__pwrq)->u.data.length)); \ if (copy_to_user((__pwrq)->u.data.pointer, (UCHAR *)(__pRaCfg), (__pwrq)->u.data.length)) \ { \ ATEDBGPRINT(RT_DEBUG_ERROR, ("copy_to_user() fail in %s\n", __FUNCTION__)); \ return (-EFAULT); \ } \ else \ { \ ATEDBGPRINT(RT_DEBUG_TRACE, ("%s is done !\n", __FUNCTION__)); \ } static inline INT DO_RACFG_CMD_ATE_START( IN PRTMP_ADAPTER pAdapter, IN struct iwreq *wrq, IN struct ate_racfghdr *pRaCfg) { ATEDBGPRINT(RT_DEBUG_TRACE,("RACFG_CMD_ATE_START\n")); /* Prepare feedback as soon as we can to avoid QA timeout. */ RESPONSE_TO_GUI(pRaCfg, wrq, sizeof(pRaCfg->status), NDIS_STATUS_SUCCESS); Set_ATE_Proc(pAdapter, "ATESTART"); return NDIS_STATUS_SUCCESS; } static inline INT DO_RACFG_CMD_ATE_STOP( IN PRTMP_ADAPTER pAdapter, IN struct iwreq *wrq, IN struct ate_racfghdr *pRaCfg) { INT32 ret; ATEDBGPRINT(RT_DEBUG_TRACE,("RACFG_CMD_ATE_STOP\n")); /* Distinguish this command came from QA(via ate agent) or ate agent according to the existence of pid in payload. No need to prepare feedback if this cmd came directly from ate agent, not from QA. */ pRaCfg->length = ntohs(pRaCfg->length); if (pRaCfg->length == sizeof(pAdapter->ate.AtePid)) { /* This command came from QA. Get the pid of ATE agent. */ memcpy((UCHAR *)&pAdapter->ate.AtePid, (&pRaCfg->data[0]) - 2/* == sizeof(pRaCfg->status) */, sizeof(pAdapter->ate.AtePid)); /* Prepare feedback as soon as we can to avoid QA timeout. */ RESPONSE_TO_GUI(pRaCfg, wrq, sizeof(pRaCfg->status), NDIS_STATUS_SUCCESS); /* Kill ATE agent when leaving ATE mode. We must kill ATE agent first before setting ATESTOP, or Microsoft will report sth. wrong. */ ret = KILL_THREAD_PID(pAdapter->ate.AtePid, SIGTERM, 1); if (ret) { ATEDBGPRINT(RT_DEBUG_ERROR, ("%s: unable to kill ate thread\n", pAdapter->net_dev->name)); } } /* AP/STA might have in ATE_STOP mode due to cmd from QA. */ if (ATE_ON(pAdapter)) { /* Someone has killed ate agent while QA GUI is still open. */ Set_ATE_Proc(pAdapter, "ATESTOP"); ATEDBGPRINT(RT_DEBUG_TRACE, ("RACFG_CMD_AP_START is done !\n")); } return NDIS_STATUS_SUCCESS; } static inline INT DO_RACFG_CMD_RF_WRITE_ALL( IN PRTMP_ADAPTER pAdapter, IN struct iwreq *wrq, IN struct ate_racfghdr *pRaCfg) { UINT32 R1, R2, R3, R4; USHORT channel; memcpy(&R1, pRaCfg->data-2, 4); memcpy(&R2, pRaCfg->data+2, 4); memcpy(&R3, pRaCfg->data+6, 4); memcpy(&R4, pRaCfg->data+10, 4); memcpy(&channel, pRaCfg->data+14, 2); pAdapter->LatchRfRegs.R1 = ntohl(R1); pAdapter->LatchRfRegs.R2 = ntohl(R2); pAdapter->LatchRfRegs.R3 = ntohl(R3); pAdapter->LatchRfRegs.R4 = ntohl(R4); pAdapter->LatchRfRegs.Channel = ntohs(channel); RTMP_RF_IO_WRITE32(pAdapter, pAdapter->LatchRfRegs.R1); RTMP_RF_IO_WRITE32(pAdapter, pAdapter->LatchRfRegs.R2); RTMP_RF_IO_WRITE32(pAdapter, pAdapter->LatchRfRegs.R3); RTMP_RF_IO_WRITE32(pAdapter, pAdapter->LatchRfRegs.R4); RESPONSE_TO_GUI(pRaCfg, wrq, sizeof(pRaCfg->status), NDIS_STATUS_SUCCESS); return NDIS_STATUS_SUCCESS; } static inline INT DO_RACFG_CMD_E2PROM_READ16( IN PRTMP_ADAPTER pAdapter, IN struct iwreq *wrq, IN struct ate_racfghdr *pRaCfg) { UINT16 offset=0, value=0; USHORT tmp=0; offset = ntohs(pRaCfg->status); /* "tmp" is especially for some compilers... */ RT28xx_EEPROM_READ16(pAdapter, offset, tmp); value = tmp; value = htons(value); ATEDBGPRINT(RT_DEBUG_TRACE,("EEPROM Read offset = 0x%04x, value = 0x%04x\n", offset, value)); memcpy(pRaCfg->data, &value, 2); RESPONSE_TO_GUI(pRaCfg, wrq, sizeof(pRaCfg->status)+2, NDIS_STATUS_SUCCESS); return NDIS_STATUS_SUCCESS; } static inline INT DO_RACFG_CMD_E2PROM_WRITE16( IN PRTMP_ADAPTER pAdapter, IN struct iwreq *wrq, IN struct ate_racfghdr *pRaCfg) { USHORT offset, value; offset = ntohs(pRaCfg->status); memcpy(&value, pRaCfg->data, 2); value = ntohs(value); RT28xx_EEPROM_WRITE16(pAdapter, offset, value); RESPONSE_TO_GUI(pRaCfg, wrq, sizeof(pRaCfg->status), NDIS_STATUS_SUCCESS); return NDIS_STATUS_SUCCESS; } static inline INT DO_RACFG_CMD_E2PROM_READ_ALL( IN PRTMP_ADAPTER pAdapter, IN struct iwreq *wrq, IN struct ate_racfghdr *pRaCfg) { USHORT buffer[EEPROM_SIZE/2]; rt_ee_read_all(pAdapter,(USHORT *)buffer); memcpy_exs(pAdapter, pRaCfg->data, (UCHAR *)buffer, EEPROM_SIZE); RESPONSE_TO_GUI(pRaCfg, wrq, sizeof(pRaCfg->status)+EEPROM_SIZE, NDIS_STATUS_SUCCESS); return NDIS_STATUS_SUCCESS; } static inline INT DO_RACFG_CMD_E2PROM_WRITE_ALL( IN PRTMP_ADAPTER pAdapter, IN struct iwreq *wrq, IN struct ate_racfghdr *pRaCfg) { USHORT buffer[EEPROM_SIZE/2]; NdisZeroMemory((UCHAR *)buffer, EEPROM_SIZE); memcpy_exs(pAdapter, (UCHAR *)buffer, (UCHAR *)&pRaCfg->status, EEPROM_SIZE); rt_ee_write_all(pAdapter,(USHORT *)buffer); RESPONSE_TO_GUI(pRaCfg, wrq, sizeof(pRaCfg->status), NDIS_STATUS_SUCCESS); return NDIS_STATUS_SUCCESS; } static inline INT DO_RACFG_CMD_IO_READ( IN PRTMP_ADAPTER pAdapter, IN struct iwreq *wrq, IN struct ate_racfghdr *pRaCfg) { UINT32 offset; UINT32 value; memcpy(&offset, &pRaCfg->status, 4); offset = ntohl(offset); /* We do not need the base address. So just extract the offset out. */ offset &= 0x0000FFFF; RTMP_IO_READ32(pAdapter, offset, &value); value = htonl(value); memcpy(pRaCfg->data, &value, 4); RESPONSE_TO_GUI(pRaCfg, wrq, sizeof(pRaCfg->status)+4, NDIS_STATUS_SUCCESS); return NDIS_STATUS_SUCCESS; } static inline INT DO_RACFG_CMD_IO_WRITE( IN PRTMP_ADAPTER pAdapter, IN struct iwreq *wrq, IN struct ate_racfghdr *pRaCfg) { UINT32 offset, value; memcpy(&offset, pRaCfg->data-2, 4); memcpy(&value, pRaCfg->data+2, 4); offset = ntohl(offset); /* We do not need the base address. So just extract the offset out. */ offset &= 0x0000FFFF; value = ntohl(value); ATEDBGPRINT(RT_DEBUG_TRACE,("RACFG_CMD_IO_WRITE: offset = %x, value = %x\n", offset, value)); RTMP_IO_WRITE32(pAdapter, offset, value); RESPONSE_TO_GUI(pRaCfg, wrq, sizeof(pRaCfg->status), NDIS_STATUS_SUCCESS); return NDIS_STATUS_SUCCESS; } static inline INT DO_RACFG_CMD_IO_READ_BULK( IN PRTMP_ADAPTER pAdapter, IN struct iwreq *wrq, IN struct ate_racfghdr *pRaCfg) { UINT32 offset; USHORT len; memcpy(&offset, &pRaCfg->status, 4); offset = ntohl(offset); /* We do not need the base address. So just extract the offset out. */ offset &= 0x0000FFFF; memcpy(&len, pRaCfg->data+2, 2); len = ntohs(len); if (len > 371) { ATEDBGPRINT(RT_DEBUG_TRACE,("length requested is too large, make it smaller\n")); pRaCfg->length = htons(2); pRaCfg->status = htons(1); return -EFAULT; } RTMP_IO_READ_BULK(pAdapter, pRaCfg->data, (UCHAR *)offset, len*4);// unit in four bytes RESPONSE_TO_GUI(pRaCfg, wrq, sizeof(pRaCfg->status)+(len*4), NDIS_STATUS_SUCCESS); return NDIS_STATUS_SUCCESS; } static inline INT DO_RACFG_CMD_BBP_READ8( IN PRTMP_ADAPTER pAdapter, IN struct iwreq *wrq, IN struct ate_racfghdr *pRaCfg) { USHORT offset; UCHAR value; value = 0; offset = ntohs(pRaCfg->status); if (ATE_ON(pAdapter)) { ATE_BBP_IO_READ8_BY_REG_ID(pAdapter, offset, &value); } else { RTMP_BBP_IO_READ8_BY_REG_ID(pAdapter, offset, &value); } pRaCfg->data[0] = value; RESPONSE_TO_GUI(pRaCfg, wrq, sizeof(pRaCfg->status)+1, NDIS_STATUS_SUCCESS); return NDIS_STATUS_SUCCESS; } static inline INT DO_RACFG_CMD_BBP_WRITE8( IN PRTMP_ADAPTER pAdapter, IN struct iwreq *wrq, IN struct ate_racfghdr *pRaCfg) { USHORT offset; UCHAR value; offset = ntohs(pRaCfg->status); memcpy(&value, pRaCfg->data, 1); if (ATE_ON(pAdapter)) { ATE_BBP_IO_WRITE8_BY_REG_ID(pAdapter, offset, value); } else { RTMP_BBP_IO_WRITE8_BY_REG_ID(pAdapter, offset, value); } if ((offset == BBP_R1) || (offset == BBP_R3)) { SyncTxRxConfig(pAdapter, offset, value); } RESPONSE_TO_GUI(pRaCfg, wrq, sizeof(pRaCfg->status), NDIS_STATUS_SUCCESS); return NDIS_STATUS_SUCCESS; } static inline INT DO_RACFG_CMD_BBP_READ_ALL( IN PRTMP_ADAPTER pAdapter, IN struct iwreq *wrq, IN struct ate_racfghdr *pRaCfg) { USHORT bbp_reg_index; for (bbp_reg_index = 0; bbp_reg_index < MAX_BBP_ID+1; bbp_reg_index++) { pRaCfg->data[bbp_reg_index] = 0; if (ATE_ON(pAdapter)) { ATE_BBP_IO_READ8_BY_REG_ID(pAdapter, bbp_reg_index, &pRaCfg->data[bbp_reg_index]); } else { RTMP_BBP_IO_READ8_BY_REG_ID(pAdapter, bbp_reg_index, &pRaCfg->data[bbp_reg_index]); } } RESPONSE_TO_GUI(pRaCfg, wrq, sizeof(pRaCfg->status)+MAX_BBP_ID+1, NDIS_STATUS_SUCCESS); return NDIS_STATUS_SUCCESS; } static inline INT DO_RACFG_CMD_GET_NOISE_LEVEL( IN PRTMP_ADAPTER pAdapter, IN struct iwreq *wrq, IN struct ate_racfghdr *pRaCfg) { UCHAR channel; INT32 buffer[3][10];/* 3 : RxPath ; 10 : no. of per rssi samples */ channel = (ntohs(pRaCfg->status) & 0x00FF); CalNoiseLevel(pAdapter, channel, buffer); memcpy_exl(pAdapter, (UCHAR *)pRaCfg->data, (UCHAR *)&(buffer[0][0]), (sizeof(INT32)*3*10)); RESPONSE_TO_GUI(pRaCfg, wrq, sizeof(pRaCfg->status)+(sizeof(INT32)*3*10), NDIS_STATUS_SUCCESS); return NDIS_STATUS_SUCCESS; } static inline INT DO_RACFG_CMD_GET_COUNTER( IN PRTMP_ADAPTER pAdapter, IN struct iwreq *wrq, IN struct ate_racfghdr *pRaCfg) { memcpy_exl(pAdapter, &pRaCfg->data[0], (UCHAR *)&pAdapter->ate.U2M, 4); memcpy_exl(pAdapter, &pRaCfg->data[4], (UCHAR *)&pAdapter->ate.OtherData, 4); memcpy_exl(pAdapter, &pRaCfg->data[8], (UCHAR *)&pAdapter->ate.Beacon, 4); memcpy_exl(pAdapter, &pRaCfg->data[12], (UCHAR *)&pAdapter->ate.OtherCount, 4); memcpy_exl(pAdapter, &pRaCfg->data[16], (UCHAR *)&pAdapter->ate.TxAc0, 4); memcpy_exl(pAdapter, &pRaCfg->data[20], (UCHAR *)&pAdapter->ate.TxAc1, 4); memcpy_exl(pAdapter, &pRaCfg->data[24], (UCHAR *)&pAdapter->ate.TxAc2, 4); memcpy_exl(pAdapter, &pRaCfg->data[28], (UCHAR *)&pAdapter->ate.TxAc3, 4); /*memcpy_exl(pAdapter, &pRaCfg->data[32], (UCHAR *)&pAdapter->ate.TxHCCA, 4);*/ memcpy_exl(pAdapter, &pRaCfg->data[36], (UCHAR *)&pAdapter->ate.TxMgmt, 4); memcpy_exl(pAdapter, &pRaCfg->data[40], (UCHAR *)&pAdapter->ate.RSSI0, 4); memcpy_exl(pAdapter, &pRaCfg->data[44], (UCHAR *)&pAdapter->ate.RSSI1, 4); memcpy_exl(pAdapter, &pRaCfg->data[48], (UCHAR *)&pAdapter->ate.RSSI2, 4); memcpy_exl(pAdapter, &pRaCfg->data[52], (UCHAR *)&pAdapter->ate.SNR0, 4); memcpy_exl(pAdapter, &pRaCfg->data[56], (UCHAR *)&pAdapter->ate.SNR1, 4); RESPONSE_TO_GUI(pRaCfg, wrq, sizeof(pRaCfg->status)+60, NDIS_STATUS_SUCCESS); return NDIS_STATUS_SUCCESS; } static inline INT DO_RACFG_CMD_CLEAR_COUNTER( IN PRTMP_ADAPTER pAdapter, IN struct iwreq *wrq, IN struct ate_racfghdr *pRaCfg) { pAdapter->ate.U2M = 0; pAdapter->ate.OtherData = 0; pAdapter->ate.Beacon = 0; pAdapter->ate.OtherCount = 0; pAdapter->ate.TxAc0 = 0; pAdapter->ate.TxAc1 = 0; pAdapter->ate.TxAc2 = 0; pAdapter->ate.TxAc3 = 0; /*pAdapter->ate.TxHCCA = 0;*/ pAdapter->ate.TxMgmt = 0; pAdapter->ate.TxDoneCount = 0; RESPONSE_TO_GUI(pRaCfg, wrq, sizeof(pRaCfg->status), NDIS_STATUS_SUCCESS); return NDIS_STATUS_SUCCESS; } static inline INT DO_RACFG_CMD_TX_START( IN PRTMP_ADAPTER pAdapter, IN struct iwreq *wrq, IN struct ate_racfghdr *pRaCfg) { USHORT *p; USHORT err = 1; UCHAR Bbp22Value = 0, Bbp24Value = 0; if ((pAdapter->ate.TxStatus != 0) && (pAdapter->ate.Mode & ATE_TXFRAME)) { ATEDBGPRINT(RT_DEBUG_TRACE,("Ate Tx is already running, to run next Tx, you must stop it first\n")); err = 2; goto TX_START_ERROR; } else if ((pAdapter->ate.TxStatus != 0) && !(pAdapter->ate.Mode & ATE_TXFRAME)) { int i = 0; while ((i++ < 10) && (pAdapter->ate.TxStatus != 0)) { RTMPusecDelay(5000); } /* force it to stop */ pAdapter->ate.TxStatus = 0; pAdapter->ate.TxDoneCount = 0; pAdapter->ate.bQATxStart = FALSE; } /* If pRaCfg->length == 0, this "RACFG_CMD_TX_START" is for Carrier test or Carrier Suppression. */ if (ntohs(pRaCfg->length) != 0) { /* get frame info */ NdisMoveMemory(&pAdapter->ate.TxWI, pRaCfg->data + 2, 16); #ifdef RT_BIG_ENDIAN RTMPWIEndianChange((PUCHAR)&pAdapter->ate.TxWI, TYPE_TXWI); #endif // RT_BIG_ENDIAN // NdisMoveMemory(&pAdapter->ate.TxCount, pRaCfg->data + 18, 4); pAdapter->ate.TxCount = ntohl(pAdapter->ate.TxCount); p = (USHORT *)(&pRaCfg->data[22]); /* always use QID_AC_BE */ pAdapter->ate.QID = 0; p = (USHORT *)(&pRaCfg->data[24]); pAdapter->ate.HLen = ntohs(*p); if (pAdapter->ate.HLen > 32) { ATEDBGPRINT(RT_DEBUG_ERROR,("pAdapter->ate.HLen > 32\n")); err = 3; goto TX_START_ERROR; } NdisMoveMemory(&pAdapter->ate.Header, pRaCfg->data + 26, pAdapter->ate.HLen); pAdapter->ate.PLen = ntohs(pRaCfg->length) - (pAdapter->ate.HLen + 28); if (pAdapter->ate.PLen > 32) { ATEDBGPRINT(RT_DEBUG_ERROR,("pAdapter->ate.PLen > 32\n")); err = 4; goto TX_START_ERROR; } NdisMoveMemory(&pAdapter->ate.Pattern, pRaCfg->data + 26 + pAdapter->ate.HLen, pAdapter->ate.PLen); pAdapter->ate.DLen = pAdapter->ate.TxWI.MPDUtotalByteCount - pAdapter->ate.HLen; } ATE_BBP_IO_READ8_BY_REG_ID(pAdapter, BBP_R22, &Bbp22Value); switch (Bbp22Value) { case BBP22_TXFRAME: { if (pAdapter->ate.TxCount == 0) { #ifdef RTMP_MAC_PCI pAdapter->ate.TxCount = 0xFFFFFFFF; #endif // RTMP_MAC_PCI // } ATEDBGPRINT(RT_DEBUG_TRACE,("START TXFRAME\n")); pAdapter->ate.bQATxStart = TRUE; Set_ATE_Proc(pAdapter, "TXFRAME"); } break; case BBP22_TXCONT_OR_CARRSUPP: { ATEDBGPRINT(RT_DEBUG_TRACE,("BBP22_TXCONT_OR_CARRSUPP\n")); ATE_BBP_IO_READ8_BY_REG_ID(pAdapter, BBP_R24, &Bbp24Value); switch (Bbp24Value) { case BBP24_TXCONT: { ATEDBGPRINT(RT_DEBUG_TRACE,("START TXCONT\n")); pAdapter->ate.bQATxStart = TRUE; Set_ATE_Proc(pAdapter, "TXCONT"); } break; case BBP24_CARRSUPP: { ATEDBGPRINT(RT_DEBUG_TRACE,("START TXCARRSUPP\n")); pAdapter->ate.bQATxStart = TRUE; pAdapter->ate.Mode |= ATE_TXCARRSUPP; } break; default: { ATEDBGPRINT(RT_DEBUG_ERROR,("Unknown TX subtype !")); } break; } } break; case BBP22_TXCARR: { ATEDBGPRINT(RT_DEBUG_TRACE,("START TXCARR\n")); pAdapter->ate.bQATxStart = TRUE; Set_ATE_Proc(pAdapter, "TXCARR"); } break; default: { ATEDBGPRINT(RT_DEBUG_ERROR,("Unknown Start TX subtype !")); } break; } if (pAdapter->ate.bQATxStart == TRUE) { RESPONSE_TO_GUI(pRaCfg, wrq, sizeof(pRaCfg->status), NDIS_STATUS_SUCCESS); return NDIS_STATUS_SUCCESS; } TX_START_ERROR: RESPONSE_TO_GUI(pRaCfg, wrq, sizeof(pRaCfg->status), err); return err; } static inline INT DO_RACFG_CMD_GET_TX_STATUS( IN PRTMP_ADAPTER pAdapter, IN struct iwreq *wrq, IN struct ate_racfghdr *pRaCfg) { UINT32 count=0; count = htonl(pAdapter->ate.TxDoneCount); NdisMoveMemory(pRaCfg->data, &count, 4); RESPONSE_TO_GUI(pRaCfg, wrq, sizeof(pRaCfg->status)+4, NDIS_STATUS_SUCCESS); return NDIS_STATUS_SUCCESS; } static inline INT DO_RACFG_CMD_TX_STOP( IN PRTMP_ADAPTER pAdapter, IN struct iwreq *wrq, IN struct ate_racfghdr *pRaCfg) { ATEDBGPRINT(RT_DEBUG_TRACE,("RACFG_CMD_TX_STOP\n")); Set_ATE_Proc(pAdapter, "TXSTOP"); RESPONSE_TO_GUI(pRaCfg, wrq, sizeof(pRaCfg->status), NDIS_STATUS_SUCCESS); return NDIS_STATUS_SUCCESS; } static inline INT DO_RACFG_CMD_RX_START( IN PRTMP_ADAPTER pAdapter, IN struct iwreq *wrq, IN struct ate_racfghdr *pRaCfg) { ATEDBGPRINT(RT_DEBUG_TRACE,("RACFG_CMD_RX_START\n")); pAdapter->ate.bQARxStart = TRUE; Set_ATE_Proc(pAdapter, "RXFRAME"); RESPONSE_TO_GUI(pRaCfg, wrq, sizeof(pRaCfg->status), NDIS_STATUS_SUCCESS); return NDIS_STATUS_SUCCESS; } static inline INT DO_RACFG_CMD_RX_STOP( IN PRTMP_ADAPTER pAdapter, IN struct iwreq *wrq, IN struct ate_racfghdr *pRaCfg) { ATEDBGPRINT(RT_DEBUG_TRACE,("RACFG_CMD_RX_STOP\n")); Set_ATE_Proc(pAdapter, "RXSTOP"); RESPONSE_TO_GUI(pRaCfg, wrq, sizeof(pRaCfg->status), NDIS_STATUS_SUCCESS); return NDIS_STATUS_SUCCESS; } static inline INT DO_RACFG_CMD_ATE_START_TX_CARRIER( IN PRTMP_ADAPTER pAdapter, IN struct iwreq *wrq, IN struct ate_racfghdr *pRaCfg) { ATEDBGPRINT(RT_DEBUG_TRACE,("RACFG_CMD_ATE_START_TX_CARRIER\n")); Set_ATE_Proc(pAdapter, "TXCARR"); RESPONSE_TO_GUI(pRaCfg, wrq, sizeof(pRaCfg->status), NDIS_STATUS_SUCCESS); return NDIS_STATUS_SUCCESS; } static inline INT DO_RACFG_CMD_ATE_START_TX_CONT( IN PRTMP_ADAPTER pAdapter, IN struct iwreq *wrq, IN struct ate_racfghdr *pRaCfg) { ATEDBGPRINT(RT_DEBUG_TRACE,("RACFG_CMD_ATE_START_TX_CONT\n")); Set_ATE_Proc(pAdapter, "TXCONT"); RESPONSE_TO_GUI(pRaCfg, wrq, sizeof(pRaCfg->status), NDIS_STATUS_SUCCESS); return NDIS_STATUS_SUCCESS; } static inline INT DO_RACFG_CMD_ATE_START_TX_FRAME( IN PRTMP_ADAPTER pAdapter, IN struct iwreq *wrq, IN struct ate_racfghdr *pRaCfg) { ATEDBGPRINT(RT_DEBUG_TRACE,("RACFG_CMD_ATE_START_TX_FRAME\n")); Set_ATE_Proc(pAdapter, "TXFRAME"); RESPONSE_TO_GUI(pRaCfg, wrq, sizeof(pRaCfg->status), NDIS_STATUS_SUCCESS); return NDIS_STATUS_SUCCESS; } static inline INT DO_RACFG_CMD_ATE_SET_BW( IN PRTMP_ADAPTER pAdapter, IN struct iwreq *wrq, IN struct ate_racfghdr *pRaCfg) { SHORT value = 0; STRING str[LEN_OF_ARG]; NdisZeroMemory(str, LEN_OF_ARG); ATEDBGPRINT(RT_DEBUG_TRACE,("RACFG_CMD_ATE_SET_BW\n")); memcpy((PUCHAR)&value, (PUCHAR)&(pRaCfg->status), 2); value = ntohs(value); sprintf((char *)str, "%d", value); Set_ATE_TX_BW_Proc(pAdapter, str); RESPONSE_TO_GUI(pRaCfg, wrq, sizeof(pRaCfg->status), NDIS_STATUS_SUCCESS); return NDIS_STATUS_SUCCESS; } static inline INT DO_RACFG_CMD_ATE_SET_TX_POWER0( IN PRTMP_ADAPTER pAdapter, IN struct iwreq *wrq, IN struct ate_racfghdr *pRaCfg) { SHORT value = 0; STRING str[LEN_OF_ARG]; NdisZeroMemory(str, LEN_OF_ARG); ATEDBGPRINT(RT_DEBUG_TRACE,("RACFG_CMD_ATE_SET_TX_POWER0\n")); memcpy((PUCHAR)&value, (PUCHAR)&(pRaCfg->status), 2); value = ntohs(value); sprintf((char *)str, "%d", value); Set_ATE_TX_POWER0_Proc(pAdapter, str); RESPONSE_TO_GUI(pRaCfg, wrq, sizeof(pRaCfg->status), NDIS_STATUS_SUCCESS); return NDIS_STATUS_SUCCESS; } static inline INT DO_RACFG_CMD_ATE_SET_TX_POWER1( IN PRTMP_ADAPTER pAdapter, IN struct iwreq *wrq, IN struct ate_racfghdr *pRaCfg) { SHORT value = 0; STRING str[LEN_OF_ARG]; NdisZeroMemory(str, LEN_OF_ARG); ATEDBGPRINT(RT_DEBUG_TRACE,("RACFG_CMD_ATE_SET_TX_POWER1\n")); memcpy((PUCHAR)&value, (PUCHAR)&(pRaCfg->status), 2); value = ntohs(value); sprintf((char *)str, "%d", value); Set_ATE_TX_POWER1_Proc(pAdapter, str); RESPONSE_TO_GUI(pRaCfg, wrq, sizeof(pRaCfg->status), NDIS_STATUS_SUCCESS); return NDIS_STATUS_SUCCESS; } static inline INT DO_RACFG_CMD_ATE_SET_FREQ_OFFSET( IN PRTMP_ADAPTER pAdapter, IN struct iwreq *wrq, IN struct ate_racfghdr *pRaCfg) { SHORT value = 0; STRING str[LEN_OF_ARG]; NdisZeroMemory(str, LEN_OF_ARG); ATEDBGPRINT(RT_DEBUG_TRACE,("RACFG_CMD_ATE_SET_FREQ_OFFSET\n")); memcpy((PUCHAR)&value, (PUCHAR)&(pRaCfg->status), 2); value = ntohs(value); sprintf((char *)str, "%d", value); Set_ATE_TX_FREQOFFSET_Proc(pAdapter, str); RESPONSE_TO_GUI(pRaCfg, wrq, sizeof(pRaCfg->status), NDIS_STATUS_SUCCESS); return NDIS_STATUS_SUCCESS; } static inline INT DO_RACFG_CMD_ATE_GET_STATISTICS( IN PRTMP_ADAPTER pAdapter, IN struct iwreq *wrq, IN struct ate_racfghdr *pRaCfg) { ATEDBGPRINT(RT_DEBUG_TRACE,("RACFG_CMD_ATE_GET_STATISTICS\n")); memcpy_exl(pAdapter, &pRaCfg->data[0], (UCHAR *)&pAdapter->ate.TxDoneCount, 4); memcpy_exl(pAdapter, &pRaCfg->data[4], (UCHAR *)&pAdapter->WlanCounters.RetryCount.u.LowPart, 4); memcpy_exl(pAdapter, &pRaCfg->data[8], (UCHAR *)&pAdapter->WlanCounters.FailedCount.u.LowPart, 4); memcpy_exl(pAdapter, &pRaCfg->data[12], (UCHAR *)&pAdapter->WlanCounters.RTSSuccessCount.u.LowPart, 4); memcpy_exl(pAdapter, &pRaCfg->data[16], (UCHAR *)&pAdapter->WlanCounters.RTSFailureCount.u.LowPart, 4); memcpy_exl(pAdapter, &pRaCfg->data[20], (UCHAR *)&pAdapter->WlanCounters.ReceivedFragmentCount.QuadPart, 4); memcpy_exl(pAdapter, &pRaCfg->data[24], (UCHAR *)&pAdapter->WlanCounters.FCSErrorCount.u.LowPart, 4); memcpy_exl(pAdapter, &pRaCfg->data[28], (UCHAR *)&pAdapter->Counters8023.RxNoBuffer, 4); memcpy_exl(pAdapter, &pRaCfg->data[32], (UCHAR *)&pAdapter->WlanCounters.FrameDuplicateCount.u.LowPart, 4); memcpy_exl(pAdapter, &pRaCfg->data[36], (UCHAR *)&pAdapter->RalinkCounters.OneSecFalseCCACnt, 4); if (pAdapter->ate.RxAntennaSel == 0) { INT32 RSSI0 = 0; INT32 RSSI1 = 0; INT32 RSSI2 = 0; RSSI0 = (INT32)(pAdapter->ate.LastRssi0 - pAdapter->BbpRssiToDbmDelta); RSSI1 = (INT32)(pAdapter->ate.LastRssi1 - pAdapter->BbpRssiToDbmDelta); RSSI2 = (INT32)(pAdapter->ate.LastRssi2 - pAdapter->BbpRssiToDbmDelta); memcpy_exl(pAdapter, &pRaCfg->data[40], (UCHAR *)&RSSI0, 4); memcpy_exl(pAdapter, &pRaCfg->data[44], (UCHAR *)&RSSI1, 4); memcpy_exl(pAdapter, &pRaCfg->data[48], (UCHAR *)&RSSI2, 4); RESPONSE_TO_GUI(pRaCfg, wrq, sizeof(pRaCfg->status)+52, NDIS_STATUS_SUCCESS); } else { INT32 RSSI0 = 0; RSSI0 = (INT32)(pAdapter->ate.LastRssi0 - pAdapter->BbpRssiToDbmDelta); memcpy_exl(pAdapter, &pRaCfg->data[40], (UCHAR *)&RSSI0, 4); RESPONSE_TO_GUI(pRaCfg, wrq, sizeof(pRaCfg->status)+44, NDIS_STATUS_SUCCESS); } return NDIS_STATUS_SUCCESS; } static inline INT DO_RACFG_CMD_ATE_RESET_COUNTER( IN PRTMP_ADAPTER pAdapter, IN struct iwreq *wrq, IN struct ate_racfghdr *pRaCfg) { SHORT value = 1; STRING str[LEN_OF_ARG]; NdisZeroMemory(str, LEN_OF_ARG); ATEDBGPRINT(RT_DEBUG_TRACE,("RACFG_CMD_ATE_RESET_COUNTER\n")); sprintf((char *)str, "%d", value); Set_ResetStatCounter_Proc(pAdapter, str); pAdapter->ate.TxDoneCount = 0; RESPONSE_TO_GUI(pRaCfg, wrq, sizeof(pRaCfg->status), NDIS_STATUS_SUCCESS); return NDIS_STATUS_SUCCESS; } static inline INT DO_RACFG_CMD_ATE_SEL_TX_ANTENNA( IN PRTMP_ADAPTER pAdapter, IN struct iwreq *wrq, IN struct ate_racfghdr *pRaCfg) { SHORT value = 0; STRING str[LEN_OF_ARG]; NdisZeroMemory(str, LEN_OF_ARG); ATEDBGPRINT(RT_DEBUG_TRACE,("RACFG_CMD_ATE_SEL_TX_ANTENNA\n")); memcpy((PUCHAR)&value, (PUCHAR)&(pRaCfg->status), 2); value = ntohs(value); sprintf((char *)str, "%d", value); Set_ATE_TX_Antenna_Proc(pAdapter, str); RESPONSE_TO_GUI(pRaCfg, wrq, sizeof(pRaCfg->status), NDIS_STATUS_SUCCESS); return NDIS_STATUS_SUCCESS; } static inline INT DO_RACFG_CMD_ATE_SEL_RX_ANTENNA( IN PRTMP_ADAPTER pAdapter, IN struct iwreq *wrq, IN struct ate_racfghdr *pRaCfg) { SHORT value = 0; STRING str[LEN_OF_ARG]; NdisZeroMemory(str, LEN_OF_ARG); ATEDBGPRINT(RT_DEBUG_TRACE,("RACFG_CMD_ATE_SEL_RX_ANTENNA\n")); memcpy((PUCHAR)&value, (PUCHAR)&(pRaCfg->status), 2); value = ntohs(value); sprintf((char *)str, "%d", value); Set_ATE_RX_Antenna_Proc(pAdapter, str); RESPONSE_TO_GUI(pRaCfg, wrq, sizeof(pRaCfg->status), NDIS_STATUS_SUCCESS); return NDIS_STATUS_SUCCESS; } static inline INT DO_RACFG_CMD_ATE_SET_PREAMBLE( IN PRTMP_ADAPTER pAdapter, IN struct iwreq *wrq, IN struct ate_racfghdr *pRaCfg) { SHORT value = 0; STRING str[LEN_OF_ARG]; NdisZeroMemory(str, LEN_OF_ARG); ATEDBGPRINT(RT_DEBUG_TRACE,("RACFG_CMD_ATE_SET_PREAMBLE\n")); memcpy((PUCHAR)&value, (PUCHAR)&(pRaCfg->status), 2); value = ntohs(value); sprintf((char *)str, "%d", value); Set_ATE_TX_MODE_Proc(pAdapter, str); RESPONSE_TO_GUI(pRaCfg, wrq, sizeof(pRaCfg->status), NDIS_STATUS_SUCCESS); return NDIS_STATUS_SUCCESS; } static inline INT DO_RACFG_CMD_ATE_SET_CHANNEL( IN PRTMP_ADAPTER pAdapter, IN struct iwreq *wrq, IN struct ate_racfghdr *pRaCfg) { SHORT value = 0; STRING str[LEN_OF_ARG]; NdisZeroMemory(str, LEN_OF_ARG); ATEDBGPRINT(RT_DEBUG_TRACE,("RACFG_CMD_ATE_SET_CHANNEL\n")); memcpy((PUCHAR)&value, (PUCHAR)&(pRaCfg->status), 2); value = ntohs(value); sprintf((char *)str, "%d", value); Set_ATE_CHANNEL_Proc(pAdapter, str); RESPONSE_TO_GUI(pRaCfg, wrq, sizeof(pRaCfg->status), NDIS_STATUS_SUCCESS); return NDIS_STATUS_SUCCESS; } static inline INT DO_RACFG_CMD_ATE_SET_ADDR1( IN PRTMP_ADAPTER pAdapter, IN struct iwreq *wrq, IN struct ate_racfghdr *pRaCfg) { ATEDBGPRINT(RT_DEBUG_TRACE,("RACFG_CMD_ATE_SET_ADDR1\n")); /* Addr is an array of UCHAR, so no need to perform endian swap. */ memcpy(pAdapter->ate.Addr1, (PUCHAR)(pRaCfg->data - 2), MAC_ADDR_LEN); RESPONSE_TO_GUI(pRaCfg, wrq, sizeof(pRaCfg->status), NDIS_STATUS_SUCCESS); return NDIS_STATUS_SUCCESS; } static inline INT DO_RACFG_CMD_ATE_SET_ADDR2( IN PRTMP_ADAPTER pAdapter, IN struct iwreq *wrq, IN struct ate_racfghdr *pRaCfg) { ATEDBGPRINT(RT_DEBUG_TRACE,("RACFG_CMD_ATE_SET_ADDR2\n")); /* Addr is an array of UCHAR, so no need to perform endian swap. */ memcpy(pAdapter->ate.Addr2, (PUCHAR)(pRaCfg->data - 2), MAC_ADDR_LEN); RESPONSE_TO_GUI(pRaCfg, wrq, sizeof(pRaCfg->status), NDIS_STATUS_SUCCESS); return NDIS_STATUS_SUCCESS; } static inline INT DO_RACFG_CMD_ATE_SET_ADDR3( IN PRTMP_ADAPTER pAdapter, IN struct iwreq *wrq, IN struct ate_racfghdr *pRaCfg) { ATEDBGPRINT(RT_DEBUG_TRACE,("RACFG_CMD_ATE_SET_ADDR3\n")); /* Addr is an array of UCHAR, so no need to perform endian swap. */ memcpy(pAdapter->ate.Addr3, (PUCHAR)(pRaCfg->data - 2), MAC_ADDR_LEN); RESPONSE_TO_GUI(pRaCfg, wrq, sizeof(pRaCfg->status), NDIS_STATUS_SUCCESS); return NDIS_STATUS_SUCCESS; } static inline INT DO_RACFG_CMD_ATE_SET_RATE( IN PRTMP_ADAPTER pAdapter, IN struct iwreq *wrq, IN struct ate_racfghdr *pRaCfg) { SHORT value = 0; STRING str[LEN_OF_ARG]; NdisZeroMemory(str, LEN_OF_ARG); ATEDBGPRINT(RT_DEBUG_TRACE,("RACFG_CMD_ATE_SET_RATE\n")); memcpy((PUCHAR)&value, (PUCHAR)&(pRaCfg->status), 2); value = ntohs(value); sprintf((char *)str, "%d", value); Set_ATE_TX_MCS_Proc(pAdapter, str); RESPONSE_TO_GUI(pRaCfg, wrq, sizeof(pRaCfg->status), NDIS_STATUS_SUCCESS); return NDIS_STATUS_SUCCESS; } static inline INT DO_RACFG_CMD_ATE_SET_TX_FRAME_LEN( IN PRTMP_ADAPTER pAdapter, IN struct iwreq *wrq, IN struct ate_racfghdr *pRaCfg) { SHORT value = 0; STRING str[LEN_OF_ARG]; NdisZeroMemory(str, LEN_OF_ARG); ATEDBGPRINT(RT_DEBUG_TRACE,("RACFG_CMD_ATE_SET_TX_FRAME_LEN\n")); memcpy((PUCHAR)&value, (PUCHAR)&(pRaCfg->status), 2); value = ntohs(value); sprintf((char *)str, "%d", value); Set_ATE_TX_LENGTH_Proc(pAdapter, str); RESPONSE_TO_GUI(pRaCfg, wrq, sizeof(pRaCfg->status), NDIS_STATUS_SUCCESS); return NDIS_STATUS_SUCCESS; } static inline INT DO_RACFG_CMD_ATE_SET_TX_FRAME_COUNT( IN PRTMP_ADAPTER pAdapter, IN struct iwreq *wrq, IN struct ate_racfghdr *pRaCfg) { USHORT value = 0; STRING str[LEN_OF_ARG]; NdisZeroMemory(str, LEN_OF_ARG); ATEDBGPRINT(RT_DEBUG_TRACE,("RACFG_CMD_ATE_SET_TX_FRAME_COUNT\n")); memcpy((PUCHAR)&value, (PUCHAR)&(pRaCfg->status), 2); value = ntohs(value); #ifdef RTMP_MAC_PCI /* TX_FRAME_COUNT == 0 means tx infinitely */ if (value == 0) { /* Use TxCount = 0xFFFFFFFF to approximate the infinity. */ pAdapter->ate.TxCount = 0xFFFFFFFF; ATEDBGPRINT(RT_DEBUG_TRACE, ("Set_ATE_TX_COUNT_Proc (TxCount = %d)\n", pAdapter->ate.TxCount)); ATEDBGPRINT(RT_DEBUG_TRACE, ("Ralink: Set_ATE_TX_COUNT_Proc Success\n")); } else #endif // RTMP_MAC_PCI // { sprintf((char *)str, "%d", value); Set_ATE_TX_COUNT_Proc(pAdapter, str); } RESPONSE_TO_GUI(pRaCfg, wrq, sizeof(pRaCfg->status), NDIS_STATUS_SUCCESS); return NDIS_STATUS_SUCCESS; } static inline INT DO_RACFG_CMD_ATE_START_RX_FRAME( IN PRTMP_ADAPTER pAdapter, IN struct iwreq *wrq, IN struct ate_racfghdr *pRaCfg) { ATEDBGPRINT(RT_DEBUG_TRACE,("RACFG_CMD_RX_START\n")); Set_ATE_Proc(pAdapter, "RXFRAME"); RESPONSE_TO_GUI(pRaCfg, wrq, sizeof(pRaCfg->status), NDIS_STATUS_SUCCESS); return NDIS_STATUS_SUCCESS; } static inline INT DO_RACFG_CMD_ATE_E2PROM_READ_BULK( IN PRTMP_ADAPTER pAdapter, IN struct iwreq *wrq, IN struct ate_racfghdr *pRaCfg) { USHORT offset; USHORT len; USHORT buffer[EEPROM_SIZE/2]; offset = ntohs(pRaCfg->status); memcpy(&len, pRaCfg->data, 2); len = ntohs(len); rt_ee_read_all(pAdapter, (USHORT *)buffer); if (offset + len <= EEPROM_SIZE) memcpy_exs(pAdapter, pRaCfg->data, (UCHAR *)buffer+offset, len); else ATEDBGPRINT(RT_DEBUG_ERROR, ("exceed EEPROM size\n")); RESPONSE_TO_GUI(pRaCfg, wrq, sizeof(pRaCfg->status)+len, NDIS_STATUS_SUCCESS); return NDIS_STATUS_SUCCESS; } static inline INT DO_RACFG_CMD_ATE_E2PROM_WRITE_BULK( IN PRTMP_ADAPTER pAdapter, IN struct iwreq *wrq, IN struct ate_racfghdr *pRaCfg) { USHORT offset; USHORT len; USHORT buffer[EEPROM_SIZE/2]; offset = ntohs(pRaCfg->status); memcpy(&len, pRaCfg->data, 2); len = ntohs(len); rt_ee_read_all(pAdapter,(USHORT *)buffer); memcpy_exs(pAdapter, (UCHAR *)buffer + offset, (UCHAR *)pRaCfg->data + 2, len); rt_ee_write_all(pAdapter,(USHORT *)buffer); RESPONSE_TO_GUI(pRaCfg, wrq, sizeof(pRaCfg->status), NDIS_STATUS_SUCCESS); return NDIS_STATUS_SUCCESS; } static inline INT DO_RACFG_CMD_ATE_IO_WRITE_BULK( IN PRTMP_ADAPTER pAdapter, IN struct iwreq *wrq, IN struct ate_racfghdr *pRaCfg) { UINT32 offset, i, value; USHORT len; memcpy(&offset, &pRaCfg->status, 4); offset = ntohl(offset); memcpy(&len, pRaCfg->data+2, 2); len = ntohs(len); for (i = 0; i < len; i += 4) { memcpy_exl(pAdapter, (UCHAR *)&value, pRaCfg->data+4+i, 4); ATEDBGPRINT(RT_DEBUG_TRACE,("Write %x %x\n", offset + i, value)); RTMP_IO_WRITE32(pAdapter, ((offset+i) & (0xffff)), value); } RESPONSE_TO_GUI(pRaCfg, wrq, sizeof(pRaCfg->status), NDIS_STATUS_SUCCESS); return NDIS_STATUS_SUCCESS; } static inline INT DO_RACFG_CMD_ATE_BBP_READ_BULK( IN PRTMP_ADAPTER pAdapter, IN struct iwreq *wrq, IN struct ate_racfghdr *pRaCfg) { USHORT offset; USHORT len; USHORT j; offset = ntohs(pRaCfg->status); memcpy(&len, pRaCfg->data, 2); len = ntohs(len); for (j = offset; j < (offset+len); j++) { pRaCfg->data[j - offset] = 0; if (pAdapter->ate.Mode == ATE_STOP) { RTMP_BBP_IO_READ8_BY_REG_ID(pAdapter, j, &pRaCfg->data[j - offset]); } else { ATE_BBP_IO_READ8_BY_REG_ID(pAdapter, j, &pRaCfg->data[j - offset]); } } RESPONSE_TO_GUI(pRaCfg, wrq, sizeof(pRaCfg->status)+len, NDIS_STATUS_SUCCESS); return NDIS_STATUS_SUCCESS; } static inline INT DO_RACFG_CMD_ATE_BBP_WRITE_BULK( IN PRTMP_ADAPTER pAdapter, IN struct iwreq *wrq, IN struct ate_racfghdr *pRaCfg) { USHORT offset; USHORT len; USHORT j; UCHAR *value; offset = ntohs(pRaCfg->status); memcpy(&len, pRaCfg->data, 2); len = ntohs(len); for (j = offset; j < (offset+len); j++) { value = pRaCfg->data + 2 + (j - offset); if (pAdapter->ate.Mode == ATE_STOP) { RTMP_BBP_IO_WRITE8_BY_REG_ID(pAdapter, j, *value); } else { ATE_BBP_IO_WRITE8_BY_REG_ID(pAdapter, j, *value); } } RESPONSE_TO_GUI(pRaCfg, wrq, sizeof(pRaCfg->status), NDIS_STATUS_SUCCESS); return NDIS_STATUS_SUCCESS; } #endif // RALINK_28xx_QA // #endif // RALINK_ATE //