2008-08-04 00:16:41 -07:00
/*
* Copyright ( c ) 2008 Atheros Communications Inc .
*
* Permission to use , copy , modify , and / or distribute this software for any
* purpose with or without fee is hereby granted , provided that the above
* copyright notice and this permission notice appear in all copies .
*
* THE SOFTWARE IS PROVIDED " AS IS " AND THE AUTHOR DISCLAIMS ALL WARRANTIES
* WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF
* MERCHANTABILITY AND FITNESS . IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR
* ANY SPECIAL , DIRECT , INDIRECT , OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES
* WHATSOEVER RESULTING FROM LOSS OF USE , DATA OR PROFITS , WHETHER IN AN
* ACTION OF CONTRACT , NEGLIGENCE OR OTHER TORTIOUS ACTION , ARISING OUT OF
* OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE .
*/
2009-02-09 13:26:54 +05:30
# include "ath9k.h"
2008-08-04 00:16:41 -07:00
2009-03-03 19:23:28 +02:00
static struct ieee80211_hw * ath_get_virt_hw ( struct ath_softc * sc ,
struct ieee80211_hdr * hdr )
{
2009-03-03 19:23:29 +02:00
struct ieee80211_hw * hw = sc - > pri_wiphy - > hw ;
int i ;
spin_lock_bh ( & sc - > wiphy_lock ) ;
for ( i = 0 ; i < sc - > num_sec_wiphy ; i + + ) {
struct ath_wiphy * aphy = sc - > sec_wiphy [ i ] ;
if ( aphy = = NULL )
continue ;
if ( compare_ether_addr ( hdr - > addr1 , aphy - > hw - > wiphy - > perm_addr )
= = 0 ) {
hw = aphy - > hw ;
break ;
}
}
spin_unlock_bh ( & sc - > wiphy_lock ) ;
return hw ;
2009-03-03 19:23:28 +02:00
}
2008-08-04 00:16:41 -07:00
/*
* Setup and link descriptors .
*
* 11 N : we can no longer afford to self link the last descriptor .
* MAC acknowledges BA status as long as it copies frames to host
* buffer ( or rx fifo ) . This can incorrectly acknowledge packets
* to a sender if last desc is self - linked .
*/
static void ath_rx_buf_link ( struct ath_softc * sc , struct ath_buf * bf )
{
2009-02-09 13:27:12 +05:30
struct ath_hw * ah = sc - > sc_ah ;
2008-08-04 00:16:41 -07:00
struct ath_desc * ds ;
struct sk_buff * skb ;
ATH_RXBUF_RESET ( bf ) ;
ds = bf - > bf_desc ;
2008-11-18 09:05:55 +05:30
ds - > ds_link = 0 ; /* link to null */
2008-08-04 00:16:41 -07:00
ds - > ds_data = bf - > bf_buf_addr ;
2008-11-18 09:05:55 +05:30
/* virtual addr of the beginning of the buffer. */
2008-08-04 00:16:41 -07:00
skb = bf - > bf_mpdu ;
ASSERT ( skb ! = NULL ) ;
ds - > ds_vdata = skb - > data ;
2008-12-07 21:44:03 +05:30
/* setup rx descriptors. The rx.bufsize here tells the harware
2008-11-20 17:15:13 -08:00
* how much data it can DMA to us and that we are prepared
* to process */
2008-12-07 21:44:03 +05:30
ath9k_hw_setuprxdesc ( ah , ds ,
sc - > rx . bufsize ,
2008-08-04 00:16:41 -07:00
0 ) ;
2008-12-07 21:44:03 +05:30
if ( sc - > rx . rxlink = = NULL )
2008-08-04 00:16:41 -07:00
ath9k_hw_putrxbuf ( ah , bf - > bf_daddr ) ;
else
2008-12-07 21:44:03 +05:30
* sc - > rx . rxlink = bf - > bf_daddr ;
2008-08-04 00:16:41 -07:00
2008-12-07 21:44:03 +05:30
sc - > rx . rxlink = & ds - > ds_link ;
2008-08-04 00:16:41 -07:00
ath9k_hw_rxena ( ah ) ;
}
2008-11-24 12:07:55 +05:30
static void ath_setdefantenna ( struct ath_softc * sc , u32 antenna )
{
/* XXX block beacon interrupts */
ath9k_hw_setantenna ( sc - > sc_ah , antenna ) ;
2008-12-07 21:44:03 +05:30
sc - > rx . defant = antenna ;
sc - > rx . rxotherant = 0 ;
2008-11-24 12:07:55 +05:30
}
/*
* Extend 15 - bit time stamp from rx descriptor to
* a full 64 - bit TSF using the current h / w TSF .
*/
static u64 ath_extend_tsf ( struct ath_softc * sc , u32 rstamp )
{
u64 tsf ;
tsf = ath9k_hw_gettsf64 ( sc - > sc_ah ) ;
if ( ( tsf & 0x7fff ) < rstamp )
tsf - = 0x8000 ;
return ( tsf & ~ 0x7fff ) | rstamp ;
}
2008-11-18 09:05:55 +05:30
static struct sk_buff * ath_rxbuf_alloc ( struct ath_softc * sc , u32 len )
2008-08-04 00:16:41 -07:00
{
struct sk_buff * skb ;
u32 off ;
/*
* Cache - line - align . This is important ( for the
* 5210 at least ) as not doing so causes bogus data
* in rx ' d frames .
*/
2008-11-20 17:15:13 -08:00
/* Note: the kernel can allocate a value greater than
* what we ask it to give us . We really only need 4 KB as that
* is this hardware supports and in fact we need at least 3849
* as that is the MAX AMSDU size this hardware supports .
* Unfortunately this means we may get 8 KB here from the
* kernel . . . and that is actually what is observed on some
* systems : ( */
2009-02-09 13:27:03 +05:30
skb = dev_alloc_skb ( len + sc - > cachelsz - 1 ) ;
2008-08-04 00:16:41 -07:00
if ( skb ! = NULL ) {
2009-02-09 13:27:03 +05:30
off = ( ( unsigned long ) skb - > data ) % sc - > cachelsz ;
2008-08-04 00:16:41 -07:00
if ( off ! = 0 )
2009-02-09 13:27:03 +05:30
skb_reserve ( skb , sc - > cachelsz - off ) ;
2008-08-04 00:16:41 -07:00
} else {
DPRINTF ( sc , ATH_DBG_FATAL ,
2008-11-28 22:18:05 +05:30
" skbuff alloc of size %u failed \n " , len ) ;
2008-08-04 00:16:41 -07:00
return NULL ;
}
return skb ;
}
/*
2008-11-18 09:05:55 +05:30
* For Decrypt or Demic errors , we only mark packet status here and always push
* up the frame up to let mac80211 handle the actual error case , be it no
* decryption key or real decryption error . This let us keep statistics there .
2008-08-04 00:16:41 -07:00
*/
2008-11-18 09:05:55 +05:30
static int ath_rx_prepare ( struct sk_buff * skb , struct ath_desc * ds ,
struct ieee80211_rx_status * rx_status , bool * decrypt_error ,
struct ath_softc * sc )
2008-08-04 00:16:41 -07:00
{
2008-11-18 09:05:55 +05:30
struct ieee80211_hdr * hdr ;
u8 ratecode ;
__le16 fc ;
2009-03-03 19:23:28 +02:00
struct ieee80211_hw * hw ;
2008-11-18 09:05:55 +05:30
hdr = ( struct ieee80211_hdr * ) skb - > data ;
fc = hdr - > frame_control ;
memset ( rx_status , 0 , sizeof ( struct ieee80211_rx_status ) ) ;
2009-03-03 19:23:28 +02:00
hw = ath_get_virt_hw ( sc , hdr ) ;
2008-11-18 09:05:55 +05:30
if ( ds - > ds_rxstat . rs_more ) {
/*
* Frame spans multiple descriptors ; this cannot happen yet
* as we don ' t support jumbograms . If not in monitor mode ,
* discard the frame . Enable this if you want to see
* error frames in Monitor mode .
*/
2009-02-09 13:27:26 +05:30
if ( sc - > sc_ah - > opmode ! = NL80211_IFTYPE_MONITOR )
2008-11-18 09:05:55 +05:30
goto rx_next ;
} else if ( ds - > ds_rxstat . rs_status ! = 0 ) {
if ( ds - > ds_rxstat . rs_status & ATH9K_RXERR_CRC )
rx_status - > flag | = RX_FLAG_FAILED_FCS_CRC ;
if ( ds - > ds_rxstat . rs_status & ATH9K_RXERR_PHY )
goto rx_next ;
2008-08-04 00:16:41 -07:00
2008-11-18 09:05:55 +05:30
if ( ds - > ds_rxstat . rs_status & ATH9K_RXERR_DECRYPT ) {
* decrypt_error = true ;
} else if ( ds - > ds_rxstat . rs_status & ATH9K_RXERR_MIC ) {
if ( ieee80211_is_ctl ( fc ) )
/*
* Sometimes , we get invalid
* MIC failures on valid control frames .
* Remove these mic errors .
*/
ds - > ds_rxstat . rs_status & = ~ ATH9K_RXERR_MIC ;
else
rx_status - > flag | = RX_FLAG_MMIC_ERROR ;
}
/*
* Reject error frames with the exception of
* decryption and MIC failures . For monitor mode ,
* we also ignore the CRC error .
*/
2009-02-09 13:27:26 +05:30
if ( sc - > sc_ah - > opmode = = NL80211_IFTYPE_MONITOR ) {
2008-11-18 09:05:55 +05:30
if ( ds - > ds_rxstat . rs_status &
~ ( ATH9K_RXERR_DECRYPT | ATH9K_RXERR_MIC |
ATH9K_RXERR_CRC ) )
goto rx_next ;
} else {
if ( ds - > ds_rxstat . rs_status &
~ ( ATH9K_RXERR_DECRYPT | ATH9K_RXERR_MIC ) ) {
goto rx_next ;
}
}
2008-08-04 00:16:41 -07:00
}
2008-11-18 09:05:55 +05:30
ratecode = ds - > ds_rxstat . rs_rate ;
if ( ratecode & 0x80 ) {
2008-12-12 14:38:34 +02:00
/* HT rate */
rx_status - > flag | = RX_FLAG_HT ;
2008-11-18 09:05:55 +05:30
if ( ds - > ds_rxstat . rs_flags & ATH9K_RX_2040 )
2008-12-12 14:38:34 +02:00
rx_status - > flag | = RX_FLAG_40MHZ ;
2008-11-18 09:05:55 +05:30
if ( ds - > ds_rxstat . rs_flags & ATH9K_RX_GI )
2008-12-12 14:38:34 +02:00
rx_status - > flag | = RX_FLAG_SHORT_GI ;
rx_status - > rate_idx = ratecode & 0x7f ;
} else {
int i = 0 , cur_band , n_rates ;
cur_band = hw - > conf . channel - > band ;
n_rates = sc - > sbands [ cur_band ] . n_bitrates ;
for ( i = 0 ; i < n_rates ; i + + ) {
if ( sc - > sbands [ cur_band ] . bitrates [ i ] . hw_value = =
ratecode ) {
rx_status - > rate_idx = i ;
break ;
}
if ( sc - > sbands [ cur_band ] . bitrates [ i ] . hw_value_short = =
ratecode ) {
rx_status - > rate_idx = i ;
rx_status - > flag | = RX_FLAG_SHORTPRE ;
break ;
}
}
2008-11-18 09:05:55 +05:30
}
rx_status - > mactime = ath_extend_tsf ( sc , ds - > ds_rxstat . rs_tstamp ) ;
2009-03-03 19:23:28 +02:00
rx_status - > band = hw - > conf . channel - > band ;
rx_status - > freq = hw - > conf . channel - > center_freq ;
2009-02-09 13:27:03 +05:30
rx_status - > noise = sc - > ani . noise_floor ;
2008-11-18 09:05:55 +05:30
rx_status - > signal = rx_status - > noise + ds - > ds_rxstat . rs_rssi ;
rx_status - > antenna = ds - > ds_rxstat . rs_antenna ;
/* at 45 you will be able to use MCS 15 reliably. A more elaborate
* scheme can be used here but it requires tables of SNR / throughput for
* each possible mode used . */
rx_status - > qual = ds - > ds_rxstat . rs_rssi * 100 / 45 ;
/* rssi can be more than 45 though, anything above that
* should be considered at 100 % */
if ( rx_status - > qual > 100 )
rx_status - > qual = 100 ;
rx_status - > flag | = RX_FLAG_TSFT ;
return 1 ;
rx_next :
return 0 ;
2008-08-04 00:16:41 -07:00
}
static void ath_opmode_init ( struct ath_softc * sc )
{
2009-02-09 13:27:12 +05:30
struct ath_hw * ah = sc - > sc_ah ;
2008-08-04 00:16:41 -07:00
u32 rfilt , mfilt [ 2 ] ;
/* configure rx filter */
rfilt = ath_calcrxfilter ( sc ) ;
ath9k_hw_setrxfilter ( ah , rfilt ) ;
/* configure bssid mask */
2009-02-09 13:27:26 +05:30
if ( ah - > caps . hw_caps & ATH9K_HW_CAP_BSSIDMASK )
2009-02-09 13:27:10 +05:30
ath9k_hw_setbssidmask ( sc ) ;
2008-08-04 00:16:41 -07:00
/* configure operational mode */
ath9k_hw_setopmode ( ah ) ;
/* Handle any link-level address change. */
2009-02-09 13:27:10 +05:30
ath9k_hw_setmac ( ah , sc - > sc_ah - > macaddr ) ;
2008-08-04 00:16:41 -07:00
/* calculate and install multicast filter */
mfilt [ 0 ] = mfilt [ 1 ] = ~ 0 ;
ath9k_hw_setmcastfilter ( ah , mfilt [ 0 ] , mfilt [ 1 ] ) ;
}
int ath_rx_init ( struct ath_softc * sc , int nbufs )
{
struct sk_buff * skb ;
struct ath_buf * bf ;
int error = 0 ;
do {
2008-12-07 21:44:03 +05:30
spin_lock_init ( & sc - > rx . rxflushlock ) ;
2008-08-11 14:05:46 +05:30
sc - > sc_flags & = ~ SC_OP_RXFLUSH ;
2008-12-07 21:44:03 +05:30
spin_lock_init ( & sc - > rx . rxbuflock ) ;
2008-08-04 00:16:41 -07:00
2008-12-07 21:44:03 +05:30
sc - > rx . bufsize = roundup ( IEEE80211_MAX_MPDU_LEN ,
2009-02-09 13:27:03 +05:30
min ( sc - > cachelsz ,
2008-08-04 00:16:41 -07:00
( u16 ) 64 ) ) ;
2008-11-28 22:18:05 +05:30
DPRINTF ( sc , ATH_DBG_CONFIG , " cachelsz %u rxbufsize %u \n " ,
2009-02-09 13:27:03 +05:30
sc - > cachelsz , sc - > rx . bufsize ) ;
2008-08-04 00:16:41 -07:00
/* Initialize rx descriptors */
2008-12-07 21:44:03 +05:30
error = ath_descdma_setup ( sc , & sc - > rx . rxdma , & sc - > rx . rxbuf ,
2008-08-04 00:16:41 -07:00
" rx " , nbufs , 1 ) ;
if ( error ! = 0 ) {
DPRINTF ( sc , ATH_DBG_FATAL ,
2008-11-28 22:18:05 +05:30
" failed to allocate rx descriptors: %d \n " , error ) ;
2008-08-04 00:16:41 -07:00
break ;
}
2008-12-07 21:44:03 +05:30
list_for_each_entry ( bf , & sc - > rx . rxbuf , list ) {
skb = ath_rxbuf_alloc ( sc , sc - > rx . bufsize ) ;
2008-08-04 00:16:41 -07:00
if ( skb = = NULL ) {
error = - ENOMEM ;
break ;
}
bf - > bf_mpdu = skb ;
2009-01-14 20:17:03 +01:00
bf - > bf_buf_addr = dma_map_single ( sc - > dev , skb - > data ,
2008-12-07 21:44:03 +05:30
sc - > rx . bufsize ,
2009-01-14 20:17:03 +01:00
DMA_FROM_DEVICE ) ;
if ( unlikely ( dma_mapping_error ( sc - > dev ,
2008-12-03 03:35:29 -08:00
bf - > bf_buf_addr ) ) ) {
dev_kfree_skb_any ( skb ) ;
bf - > bf_mpdu = NULL ;
DPRINTF ( sc , ATH_DBG_CONFIG ,
2009-01-14 20:17:03 +01:00
" dma_mapping_error() on RX init \n " ) ;
2008-12-03 03:35:29 -08:00
error = - ENOMEM ;
break ;
}
2008-08-14 13:26:34 +05:30
bf - > bf_dmacontext = bf - > bf_buf_addr ;
2008-08-04 00:16:41 -07:00
}
2008-12-07 21:44:03 +05:30
sc - > rx . rxlink = NULL ;
2008-08-04 00:16:41 -07:00
} while ( 0 ) ;
if ( error )
ath_rx_cleanup ( sc ) ;
return error ;
}
void ath_rx_cleanup ( struct ath_softc * sc )
{
struct sk_buff * skb ;
struct ath_buf * bf ;
2008-12-07 21:44:03 +05:30
list_for_each_entry ( bf , & sc - > rx . rxbuf , list ) {
2008-08-04 00:16:41 -07:00
skb = bf - > bf_mpdu ;
if ( skb )
dev_kfree_skb ( skb ) ;
}
2008-12-07 21:44:03 +05:30
if ( sc - > rx . rxdma . dd_desc_len ! = 0 )
ath_descdma_cleanup ( sc , & sc - > rx . rxdma , & sc - > rx . rxbuf ) ;
2008-08-04 00:16:41 -07:00
}
/*
* Calculate the receive filter according to the
* operating mode and state :
*
* o always accept unicast , broadcast , and multicast traffic
* o maintain current state of phy error reception ( the hal
* may enable phy error frames for noise immunity work )
* o probe request frames are accepted only when operating in
* hostap , adhoc , or monitor modes
* o enable promiscuous mode according to the interface state
* o accept beacons :
* - when operating in adhoc mode so the 802.11 layer creates
* node table entries for peers ,
* - when operating in station mode for collecting rssi data when
* the station is otherwise quiet , or
* - when operating as a repeater so we see repeater - sta beacons
* - when scanning
*/
u32 ath_calcrxfilter ( struct ath_softc * sc )
{
# define RX_FILTER_PRESERVE (ATH9K_RX_FILTER_PHYERR | ATH9K_RX_FILTER_PHYRADAR)
2008-08-11 14:03:13 +05:30
2008-08-04 00:16:41 -07:00
u32 rfilt ;
rfilt = ( ath9k_hw_getrxfilter ( sc - > sc_ah ) & RX_FILTER_PRESERVE )
| ATH9K_RX_FILTER_UCAST | ATH9K_RX_FILTER_BCAST
| ATH9K_RX_FILTER_MCAST ;
/* If not a STA, enable processing of Probe Requests */
2009-02-09 13:27:26 +05:30
if ( sc - > sc_ah - > opmode ! = NL80211_IFTYPE_STATION )
2008-08-04 00:16:41 -07:00
rfilt | = ATH9K_RX_FILTER_PROBEREQ ;
/* Can't set HOSTAP into promiscous mode */
2009-02-09 13:27:26 +05:30
if ( ( ( sc - > sc_ah - > opmode ! = NL80211_IFTYPE_AP ) & &
2008-12-07 21:44:03 +05:30
( sc - > rx . rxfilter & FIF_PROMISC_IN_BSS ) ) | |
2009-02-09 13:27:26 +05:30
( sc - > sc_ah - > opmode = = NL80211_IFTYPE_MONITOR ) ) {
2008-08-04 00:16:41 -07:00
rfilt | = ATH9K_RX_FILTER_PROM ;
/* ??? To prevent from sending ACK */
rfilt & = ~ ATH9K_RX_FILTER_UCAST ;
}
2009-02-04 08:10:22 +05:30
if ( sc - > rx . rxfilter & FIF_CONTROL )
rfilt | = ATH9K_RX_FILTER_CONTROL ;
2009-02-19 15:41:52 +05:30
if ( ( sc - > sc_ah - > opmode = = NL80211_IFTYPE_STATION ) & &
! ( sc - > rx . rxfilter & FIF_BCN_PRBRESP_PROMISC ) )
rfilt | = ATH9K_RX_FILTER_MYBEACON ;
else
2008-08-04 00:16:41 -07:00
rfilt | = ATH9K_RX_FILTER_BEACON ;
2009-02-19 15:41:52 +05:30
/* If in HOSTAP mode, want to enable reception of PSPOLL frames */
2009-02-09 13:27:26 +05:30
if ( sc - > sc_ah - > opmode = = NL80211_IFTYPE_AP )
2009-02-19 15:41:52 +05:30
rfilt | = ATH9K_RX_FILTER_PSPOLL ;
2008-11-18 09:05:55 +05:30
2008-08-04 00:16:41 -07:00
return rfilt ;
2008-08-11 14:03:13 +05:30
2008-08-04 00:16:41 -07:00
# undef RX_FILTER_PRESERVE
}
int ath_startrecv ( struct ath_softc * sc )
{
2009-02-09 13:27:12 +05:30
struct ath_hw * ah = sc - > sc_ah ;
2008-08-04 00:16:41 -07:00
struct ath_buf * bf , * tbf ;
2008-12-07 21:44:03 +05:30
spin_lock_bh ( & sc - > rx . rxbuflock ) ;
if ( list_empty ( & sc - > rx . rxbuf ) )
2008-08-04 00:16:41 -07:00
goto start_recv ;
2008-12-07 21:44:03 +05:30
sc - > rx . rxlink = NULL ;
list_for_each_entry_safe ( bf , tbf , & sc - > rx . rxbuf , list ) {
2008-08-04 00:16:41 -07:00
ath_rx_buf_link ( sc , bf ) ;
}
/* We could have deleted elements so the list may be empty now */
2008-12-07 21:44:03 +05:30
if ( list_empty ( & sc - > rx . rxbuf ) )
2008-08-04 00:16:41 -07:00
goto start_recv ;
2008-12-07 21:44:03 +05:30
bf = list_first_entry ( & sc - > rx . rxbuf , struct ath_buf , list ) ;
2008-08-04 00:16:41 -07:00
ath9k_hw_putrxbuf ( ah , bf - > bf_daddr ) ;
2008-11-18 09:05:55 +05:30
ath9k_hw_rxena ( ah ) ;
2008-08-04 00:16:41 -07:00
start_recv :
2008-12-07 21:44:03 +05:30
spin_unlock_bh ( & sc - > rx . rxbuflock ) ;
2008-11-18 09:05:55 +05:30
ath_opmode_init ( sc ) ;
ath9k_hw_startpcureceive ( ah ) ;
2008-08-04 00:16:41 -07:00
return 0 ;
}
bool ath_stoprecv ( struct ath_softc * sc )
{
2009-02-09 13:27:12 +05:30
struct ath_hw * ah = sc - > sc_ah ;
2008-08-04 00:16:41 -07:00
bool stopped ;
2008-11-18 09:05:55 +05:30
ath9k_hw_stoppcurecv ( ah ) ;
ath9k_hw_setrxfilter ( ah , 0 ) ;
stopped = ath9k_hw_stopdmarecv ( ah ) ;
2008-12-07 21:44:03 +05:30
sc - > rx . rxlink = NULL ;
2008-11-18 09:05:55 +05:30
2008-08-04 00:16:41 -07:00
return stopped ;
}
void ath_flushrecv ( struct ath_softc * sc )
{
2008-12-07 21:44:03 +05:30
spin_lock_bh ( & sc - > rx . rxflushlock ) ;
2008-08-11 14:05:46 +05:30
sc - > sc_flags | = SC_OP_RXFLUSH ;
2008-08-04 00:16:41 -07:00
ath_rx_tasklet ( sc , 1 ) ;
2008-08-11 14:05:46 +05:30
sc - > sc_flags & = ~ SC_OP_RXFLUSH ;
2008-12-07 21:44:03 +05:30
spin_unlock_bh ( & sc - > rx . rxflushlock ) ;
2008-08-04 00:16:41 -07:00
}
int ath_rx_tasklet ( struct ath_softc * sc , int flush )
{
# define PA2DESC(_sc, _pa) \
2008-12-07 21:44:03 +05:30
( ( struct ath_desc * ) ( ( caddr_t ) ( _sc ) - > rx . rxdma . dd_desc + \
( ( _pa ) - ( _sc ) - > rx . rxdma . dd_desc_paddr ) ) )
2008-08-04 00:16:41 -07:00
2008-11-18 09:05:55 +05:30
struct ath_buf * bf ;
2008-08-04 00:16:41 -07:00
struct ath_desc * ds ;
2008-11-21 17:41:33 -08:00
struct sk_buff * skb = NULL , * requeue_skb ;
2008-11-18 09:05:55 +05:30
struct ieee80211_rx_status rx_status ;
2009-02-09 13:27:12 +05:30
struct ath_hw * ah = sc - > sc_ah ;
2008-11-18 09:05:55 +05:30
struct ieee80211_hdr * hdr ;
int hdrlen , padsize , retval ;
bool decrypt_error = false ;
u8 keyix ;
2008-12-07 21:44:03 +05:30
spin_lock_bh ( & sc - > rx . rxbuflock ) ;
2008-08-04 00:16:41 -07:00
do {
/* If handling rx interrupt and flush is in progress => exit */
2008-08-11 14:05:46 +05:30
if ( ( sc - > sc_flags & SC_OP_RXFLUSH ) & & ( flush = = 0 ) )
2008-08-04 00:16:41 -07:00
break ;
2008-12-07 21:44:03 +05:30
if ( list_empty ( & sc - > rx . rxbuf ) ) {
sc - > rx . rxlink = NULL ;
2008-08-04 00:16:41 -07:00
break ;
}
2008-12-07 21:44:03 +05:30
bf = list_first_entry ( & sc - > rx . rxbuf , struct ath_buf , list ) ;
2008-08-04 00:16:41 -07:00
ds = bf - > bf_desc ;
/*
* Must provide the virtual address of the current
* descriptor , the physical address , and the virtual
* address of the next descriptor in the h / w chain .
* This allows the HAL to look ahead to see if the
* hardware is done with a descriptor by checking the
* done bit in the following descriptor and the address
* of the current descriptor the DMA engine is working
* on . All this is necessary because of our use of
* a self - linked list to avoid rx overruns .
*/
2008-11-18 09:05:55 +05:30
retval = ath9k_hw_rxprocdesc ( ah , ds ,
2008-08-04 00:16:41 -07:00
bf - > bf_daddr ,
PA2DESC ( sc , ds - > ds_link ) ,
0 ) ;
if ( retval = = - EINPROGRESS ) {
struct ath_buf * tbf ;
struct ath_desc * tds ;
2008-12-07 21:44:03 +05:30
if ( list_is_last ( & bf - > list , & sc - > rx . rxbuf ) ) {
sc - > rx . rxlink = NULL ;
2008-08-04 00:16:41 -07:00
break ;
}
tbf = list_entry ( bf - > list . next , struct ath_buf , list ) ;
/*
* On some hardware the descriptor status words could
* get corrupted , including the done bit . Because of
* this , check if the next descriptor ' s done bit is
* set or not .
*
* If the next descriptor ' s done bit is set , the current
* descriptor has been corrupted . Force s / w to discard
* this descriptor and continue . . .
*/
tds = tbf - > bf_desc ;
2008-11-18 09:05:55 +05:30
retval = ath9k_hw_rxprocdesc ( ah , tds , tbf - > bf_daddr ,
PA2DESC ( sc , tds - > ds_link ) , 0 ) ;
2008-08-04 00:16:41 -07:00
if ( retval = = - EINPROGRESS ) {
break ;
}
}
skb = bf - > bf_mpdu ;
2008-11-18 09:05:55 +05:30
if ( ! skb )
2008-08-04 00:16:41 -07:00
continue ;
2008-12-15 20:40:46 +05:30
/*
* Synchronize the DMA transfer with CPU before
* 1. accessing the frame
* 2. requeueing the same buffer to h / w
*/
2009-01-14 20:17:03 +01:00
dma_sync_single_for_cpu ( sc - > dev , bf - > bf_buf_addr ,
2008-12-15 20:40:46 +05:30
sc - > rx . bufsize ,
2009-01-14 20:17:03 +01:00
DMA_FROM_DEVICE ) ;
2008-12-15 20:40:46 +05:30
2008-08-04 00:16:41 -07:00
/*
2008-11-18 09:05:55 +05:30
* If we ' re asked to flush receive queue , directly
* chain it back at the queue without processing it .
2008-08-04 00:16:41 -07:00
*/
2008-11-18 09:05:55 +05:30
if ( flush )
2008-11-21 17:41:33 -08:00
goto requeue ;
2008-08-04 00:16:41 -07:00
2008-11-18 09:05:55 +05:30
if ( ! ds - > ds_rxstat . rs_datalen )
2008-11-21 17:41:33 -08:00
goto requeue ;
2008-08-04 00:16:41 -07:00
2008-11-18 09:05:55 +05:30
/* The status portion of the descriptor could get corrupted. */
2008-12-07 21:44:03 +05:30
if ( sc - > rx . bufsize < ds - > ds_rxstat . rs_datalen )
2008-11-21 17:41:33 -08:00
goto requeue ;
2008-08-04 00:16:41 -07:00
2008-11-18 09:05:55 +05:30
if ( ! ath_rx_prepare ( skb , ds , & rx_status , & decrypt_error , sc ) )
2008-11-21 17:41:33 -08:00
goto requeue ;
/* Ensure we always have an skb to requeue once we are done
* processing the current buffer ' s skb */
2008-12-07 21:44:03 +05:30
requeue_skb = ath_rxbuf_alloc ( sc , sc - > rx . bufsize ) ;
2008-11-21 17:41:33 -08:00
/* If there is no memory we ignore the current RX'd frame,
* tell hardware it can give us a new frame using the old
2008-12-07 21:44:03 +05:30
* skb and put it at the tail of the sc - > rx . rxbuf list for
2008-11-21 17:41:33 -08:00
* processing . */
if ( ! requeue_skb )
goto requeue ;
2008-08-04 00:16:41 -07:00
2008-12-15 20:40:46 +05:30
/* Unmap the frame */
2009-01-14 20:17:03 +01:00
dma_unmap_single ( sc - > dev , bf - > bf_buf_addr ,
2008-12-07 21:44:03 +05:30
sc - > rx . bufsize ,
2009-01-14 20:17:03 +01:00
DMA_FROM_DEVICE ) ;
2008-08-04 00:16:41 -07:00
2008-11-18 09:05:55 +05:30
skb_put ( skb , ds - > ds_rxstat . rs_datalen ) ;
skb - > protocol = cpu_to_be16 ( ETH_P_CONTROL ) ;
/* see if any padding is done by the hw and remove it */
hdr = ( struct ieee80211_hdr * ) skb - > data ;
hdrlen = ieee80211_get_hdrlen_from_skb ( skb ) ;
2008-12-11 18:22:13 +02:00
/* The MAC header is padded to have 32-bit boundary if the
* packet payload is non - zero . The general calculation for
* padsize would take into account odd header lengths :
* padsize = ( 4 - hdrlen % 4 ) % 4 ; However , since only
* even - length headers are used , padding can only be 0 or 2
* bytes and we can optimize this a bit . In addition , we must
* not try to remove padding from short control frames that do
* not have payload . */
padsize = hdrlen & 3 ;
if ( padsize & & hdrlen > = 24 ) {
2008-11-18 09:05:55 +05:30
memmove ( skb - > data + padsize , skb - > data , hdrlen ) ;
skb_pull ( skb , padsize ) ;
2008-08-04 00:16:41 -07:00
}
2008-11-18 09:05:55 +05:30
keyix = ds - > ds_rxstat . rs_keyix ;
2008-08-04 00:16:41 -07:00
2008-11-18 09:05:55 +05:30
if ( ! ( keyix = = ATH9K_RXKEYIX_INVALID ) & & ! decrypt_error ) {
rx_status . flag | = RX_FLAG_DECRYPTED ;
} else if ( ( le16_to_cpu ( hdr - > frame_control ) & IEEE80211_FCTL_PROTECTED )
& & ! decrypt_error & & skb - > len > = hdrlen + 4 ) {
keyix = skb - > data [ hdrlen + 3 ] > > 6 ;
2009-02-09 13:27:03 +05:30
if ( test_bit ( keyix , sc - > keymap ) )
2008-11-18 09:05:55 +05:30
rx_status . flag | = RX_FLAG_DECRYPTED ;
}
2009-01-08 13:32:13 +02:00
if ( ah - > sw_mgmt_crypto & &
( rx_status . flag & RX_FLAG_DECRYPTED ) & &
ieee80211_is_mgmt ( hdr - > frame_control ) ) {
/* Use software decrypt for management frames. */
rx_status . flag & = ~ RX_FLAG_DECRYPTED ;
}
2008-11-18 09:05:55 +05:30
/* Send the frame to mac80211 */
2009-03-03 19:23:29 +02:00
if ( hdr - > addr1 [ 5 ] & 0x01 ) {
int i ;
/*
* Deliver broadcast / multicast frames to all suitable
* virtual wiphys .
*/
/* TODO: filter based on channel configuration */
for ( i = 0 ; i < sc - > num_sec_wiphy ; i + + ) {
struct ath_wiphy * aphy = sc - > sec_wiphy [ i ] ;
struct sk_buff * nskb ;
if ( aphy = = NULL )
continue ;
nskb = skb_copy ( skb , GFP_ATOMIC ) ;
if ( nskb )
__ieee80211_rx ( aphy - > hw , nskb ,
& rx_status ) ;
}
__ieee80211_rx ( sc - > hw , skb , & rx_status ) ;
} else {
/* Deliver unicast frames based on receiver address */
__ieee80211_rx ( ath_get_virt_hw ( sc , hdr ) , skb ,
& rx_status ) ;
}
2008-11-21 17:41:33 -08:00
/* We will now give hardware our shiny new allocated skb */
bf - > bf_mpdu = requeue_skb ;
2009-01-14 20:17:03 +01:00
bf - > bf_buf_addr = dma_map_single ( sc - > dev , requeue_skb - > data ,
2008-12-07 21:44:03 +05:30
sc - > rx . bufsize ,
2009-01-14 20:17:03 +01:00
DMA_FROM_DEVICE ) ;
if ( unlikely ( dma_mapping_error ( sc - > dev ,
2008-12-03 03:35:29 -08:00
bf - > bf_buf_addr ) ) ) {
dev_kfree_skb_any ( requeue_skb ) ;
bf - > bf_mpdu = NULL ;
DPRINTF ( sc , ATH_DBG_CONFIG ,
2009-01-14 20:17:03 +01:00
" dma_mapping_error() on RX \n " ) ;
2008-12-03 03:35:29 -08:00
break ;
}
2008-11-21 17:41:33 -08:00
bf - > bf_dmacontext = bf - > bf_buf_addr ;
2008-08-04 00:16:41 -07:00
/*
* change the default rx antenna if rx diversity chooses the
* other antenna 3 times in a row .
*/
2008-12-07 21:44:03 +05:30
if ( sc - > rx . defant ! = ds - > ds_rxstat . rs_antenna ) {
if ( + + sc - > rx . rxotherant > = 3 )
2008-11-18 09:05:55 +05:30
ath_setdefantenna ( sc , ds - > ds_rxstat . rs_antenna ) ;
2008-08-04 00:16:41 -07:00
} else {
2008-12-07 21:44:03 +05:30
sc - > rx . rxotherant = 0 ;
2008-08-04 00:16:41 -07:00
}
2009-01-20 11:17:08 +05:30
if ( ieee80211_is_beacon ( hdr - > frame_control ) & &
( sc - > sc_flags & SC_OP_WAIT_FOR_BEACON ) ) {
sc - > sc_flags & = ~ SC_OP_WAIT_FOR_BEACON ;
ath9k_hw_setpower ( sc - > sc_ah , ATH9K_PM_NETWORK_SLEEP ) ;
}
2008-11-21 17:41:33 -08:00
requeue :
2008-12-07 21:44:03 +05:30
list_move_tail ( & bf - > list , & sc - > rx . rxbuf ) ;
2008-11-21 17:41:33 -08:00
ath_rx_buf_link ( sc , bf ) ;
2008-11-18 09:05:55 +05:30
} while ( 1 ) ;
2008-12-07 21:44:03 +05:30
spin_unlock_bh ( & sc - > rx . rxbuflock ) ;
2008-08-04 00:16:41 -07:00
return 0 ;
# undef PA2DESC
}