2010-05-13 16:48:03 +02:00
/*
* Copyright ( C ) 2010 Felix Fietkau < nbd @ openwrt . org >
*
* This program is free software ; you can redistribute it and / or modify
* it under the terms of the GNU General Public License version 2 as
* published by the Free Software Foundation .
*/
# include <linux/netdevice.h>
# include <linux/types.h>
# include <linux/skbuff.h>
# include <linux/debugfs.h>
# include <linux/ieee80211.h>
2011-07-15 11:47:34 -04:00
# include <linux/export.h>
2010-05-13 16:48:03 +02:00
# include <net/mac80211.h>
# include "rc80211_minstrel.h"
# include "rc80211_minstrel_ht.h"
2013-02-13 10:51:08 +01:00
static char *
minstrel_ht_stats_dump ( struct minstrel_ht_sta * mi , int i , char * p )
{
const struct mcs_group * mg ;
unsigned int j , tp , prob , eprob ;
char htmode = ' 2 ' ;
char gimode = ' L ' ;
2014-10-22 18:20:37 +02:00
u32 gflags ;
2013-02-13 10:51:08 +01:00
if ( ! mi - > groups [ i ] . supported )
return p ;
mg = & minstrel_mcs_groups [ i ] ;
2014-10-22 18:20:37 +02:00
gflags = mg - > flags ;
if ( gflags & IEEE80211_TX_RC_40_MHZ_WIDTH )
2013-02-13 10:51:08 +01:00
htmode = ' 4 ' ;
2014-10-22 18:20:37 +02:00
else if ( gflags & IEEE80211_TX_RC_80_MHZ_WIDTH )
2014-10-21 10:38:38 +02:00
htmode = ' 8 ' ;
2014-10-22 18:20:37 +02:00
if ( gflags & IEEE80211_TX_RC_SHORT_GI )
2013-02-13 10:51:08 +01:00
gimode = ' S ' ;
for ( j = 0 ; j < MCS_GROUP_RATES ; j + + ) {
struct minstrel_rate_stats * mr = & mi - > groups [ i ] . rates [ j ] ;
static const int bitrates [ 4 ] = { 10 , 20 , 55 , 110 } ;
int idx = i * MCS_GROUP_RATES + j ;
if ( ! ( mi - > groups [ i ] . supported & BIT ( j ) ) )
continue ;
2014-10-22 18:20:37 +02:00
if ( gflags & IEEE80211_TX_RC_MCS )
p + = sprintf ( p , " HT%c0/%cGI " , htmode , gimode ) ;
else if ( gflags & IEEE80211_TX_RC_VHT_MCS )
2014-10-21 10:38:38 +02:00
p + = sprintf ( p , " VHT%c0/%cGI " , htmode , gimode ) ;
2013-02-13 10:51:08 +01:00
else
2014-10-22 18:20:37 +02:00
p + = sprintf ( p , " CCK/%cP " , j < 4 ? ' L ' : ' S ' ) ;
2013-02-13 10:51:08 +01:00
2014-09-09 23:22:14 +02:00
* ( p + + ) = ( idx = = mi - > max_tp_rate [ 0 ] ) ? ' A ' : ' ' ;
* ( p + + ) = ( idx = = mi - > max_tp_rate [ 1 ] ) ? ' B ' : ' ' ;
* ( p + + ) = ( idx = = mi - > max_tp_rate [ 2 ] ) ? ' C ' : ' ' ;
* ( p + + ) = ( idx = = mi - > max_tp_rate [ 3 ] ) ? ' D ' : ' ' ;
2013-02-13 10:51:08 +01:00
* ( p + + ) = ( idx = = mi - > max_prob_rate ) ? ' P ' : ' ' ;
2014-10-22 18:20:37 +02:00
if ( gflags & IEEE80211_TX_RC_MCS ) {
p + = sprintf ( p , " MCS%-2u " , ( mg - > streams - 1 ) * 8 + j ) ;
} else if ( gflags & IEEE80211_TX_RC_VHT_MCS ) {
2014-10-21 10:38:38 +02:00
p + = sprintf ( p , " MCS%-1u/%1u " , j , mg - > streams ) ;
2013-02-13 10:51:08 +01:00
} else {
2014-10-22 18:20:37 +02:00
int r = bitrates [ j % 4 ] ;
p + = sprintf ( p , " %2u.%1uM " , r / 10 , r % 10 ) ;
2013-02-13 10:51:08 +01:00
}
tp = mr - > cur_tp / 10 ;
prob = MINSTREL_TRUNC ( mr - > cur_prob * 1000 ) ;
eprob = MINSTREL_TRUNC ( mr - > probability * 1000 ) ;
2014-10-20 10:54:36 +02:00
p + = sprintf ( p , " %4u.%1u %3u.%1u %3u.%1u "
" %3u %4u(%4u) %9llu(%9llu) \n " ,
2013-02-13 10:51:08 +01:00
tp / 10 , tp % 10 ,
eprob / 10 , eprob % 10 ,
prob / 10 , prob % 10 ,
mr - > retry_count ,
mr - > last_success ,
mr - > last_attempts ,
( unsigned long long ) mr - > succ_hist ,
( unsigned long long ) mr - > att_hist ) ;
}
return p ;
}
2010-05-13 16:48:03 +02:00
static int
minstrel_ht_stats_open ( struct inode * inode , struct file * file )
{
struct minstrel_ht_sta_priv * msp = inode - > i_private ;
struct minstrel_ht_sta * mi = & msp - > ht ;
struct minstrel_debugfs_info * ms ;
2013-02-13 10:51:08 +01:00
unsigned int i ;
2010-05-13 16:48:03 +02:00
char * p ;
int ret ;
if ( ! msp - > is_ht ) {
inode - > i_private = & msp - > legacy ;
ret = minstrel_stats_open ( inode , file ) ;
inode - > i_private = msp ;
return ret ;
}
2014-10-21 10:38:38 +02:00
ms = kmalloc ( 32768 , GFP_KERNEL ) ;
2010-05-13 16:48:03 +02:00
if ( ! ms )
return - ENOMEM ;
file - > private_data = ms ;
p = ms - > buf ;
2014-10-21 10:38:38 +02:00
p + = sprintf ( p , " type rate tpt eprob *prob "
2014-10-20 10:54:36 +02:00
" ret *ok(*cum) ok( cum) \n " ) ;
2014-10-20 15:46:00 +02:00
p = minstrel_ht_stats_dump ( mi , MINSTREL_CCK_GROUP , p ) ;
for ( i = 0 ; i < MINSTREL_CCK_GROUP ; i + + )
2013-02-13 10:51:08 +01:00
p = minstrel_ht_stats_dump ( mi , i , p ) ;
2014-10-21 10:38:38 +02:00
for ( i + + ; i < ARRAY_SIZE ( mi - > groups ) ; i + + )
p = minstrel_ht_stats_dump ( mi , i , p ) ;
2010-05-13 16:48:03 +02:00
p + = sprintf ( p , " \n Total packet count:: ideal %d "
" lookaround %d \n " ,
max ( 0 , ( int ) mi - > total_packets - ( int ) mi - > sample_packets ) ,
mi - > sample_packets ) ;
p + = sprintf ( p , " Average A-MPDU length: %d.%d \n " ,
MINSTREL_TRUNC ( mi - > avg_ampdu_len ) ,
MINSTREL_TRUNC ( mi - > avg_ampdu_len * 10 ) % 10 ) ;
ms - > len = p - ms - > buf ;
2014-10-21 10:38:38 +02:00
WARN_ON ( ms - > len + sizeof ( * ms ) > 32768 ) ;
2014-10-20 10:54:36 +02:00
2010-08-15 18:32:42 +02:00
return nonseekable_open ( inode , file ) ;
2010-05-13 16:48:03 +02:00
}
static const struct file_operations minstrel_ht_stat_fops = {
. owner = THIS_MODULE ,
. open = minstrel_ht_stats_open ,
. read = minstrel_stats_read ,
. release = minstrel_stats_release ,
2010-08-15 18:32:42 +02:00
. llseek = no_llseek ,
2010-05-13 16:48:03 +02:00
} ;
void
minstrel_ht_add_sta_debugfs ( void * priv , void * priv_sta , struct dentry * dir )
{
struct minstrel_ht_sta_priv * msp = priv_sta ;
msp - > dbg_stats = debugfs_create_file ( " rc_stats " , S_IRUGO , dir , msp ,
& minstrel_ht_stat_fops ) ;
}
void
minstrel_ht_remove_sta_debugfs ( void * priv , void * priv_sta )
{
struct minstrel_ht_sta_priv * msp = priv_sta ;
debugfs_remove ( msp - > dbg_stats ) ;
}