📄 bcm43xx_wx.c
字号:
/* Broadcom BCM43xx wireless driver Copyright (c) 2005 Martin Langer <martin-langer@gmx.de>, Stefano Brivio <st3@riseup.net> Michael Buesch <mbuesch@freenet.de> Danny van Dyk <kugelfang@gentoo.org> Andreas Jaggi <andreas.jaggi@waterwave.ch> Some parts of the code in this file are derived from the ipw2200 driver Copyright(c) 2003 - 2004 Intel Corporation. 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; see the file COPYING. If not, write to the Free Software Foundation, Inc., 51 Franklin Steet, Fifth Floor, Boston, MA 02110-1301, USA.*/#include <linux/wireless.h>#include <net/iw_handler.h>#include <net/ieee80211softmac.h>#include <net/ieee80211softmac_wx.h>#include <linux/capability.h>#include <linux/delay.h>#include "bcm43xx.h"#include "bcm43xx_wx.h"#include "bcm43xx_main.h"#include "bcm43xx_radio.h"#include "bcm43xx_phy.h"/* The WIRELESS_EXT version, which is implemented by this driver. */#define BCM43xx_WX_VERSION 18#define MAX_WX_STRING 80static int bcm43xx_wx_get_name(struct net_device *net_dev, struct iw_request_info *info, union iwreq_data *data, char *extra){ struct bcm43xx_private *bcm = bcm43xx_priv(net_dev); int i; struct bcm43xx_phyinfo *phy; char suffix[7] = { 0 }; int have_a = 0, have_b = 0, have_g = 0; mutex_lock(&bcm->mutex); for (i = 0; i < bcm->nr_80211_available; i++) { phy = &(bcm->core_80211_ext[i].phy); switch (phy->type) { case BCM43xx_PHYTYPE_A: have_a = 1; break; case BCM43xx_PHYTYPE_G: have_g = 1; case BCM43xx_PHYTYPE_B: have_b = 1; break; default: assert(0); } } mutex_unlock(&bcm->mutex); i = 0; if (have_a) { suffix[i++] = 'a'; suffix[i++] = '/'; } if (have_b) { suffix[i++] = 'b'; suffix[i++] = '/'; } if (have_g) { suffix[i++] = 'g'; suffix[i++] = '/'; } if (i != 0) suffix[i - 1] = '\0'; snprintf(data->name, IFNAMSIZ, "IEEE 802.11%s", suffix); return 0;}static int bcm43xx_wx_set_channelfreq(struct net_device *net_dev, struct iw_request_info *info, union iwreq_data *data, char *extra){ struct bcm43xx_private *bcm = bcm43xx_priv(net_dev); unsigned long flags; u8 channel; s8 expon; int freq; int err = -EINVAL; mutex_lock(&bcm->mutex); spin_lock_irqsave(&bcm->irq_lock, flags); if ((data->freq.e == 0) && (data->freq.m >= 0) && (data->freq.m <= 1000)) { channel = data->freq.m; freq = bcm43xx_channel_to_freq(bcm, channel); } else { freq = data->freq.m; expon = 6 - data->freq.e; while (--expon >= 0) /* scale down the frequency to MHz */ freq /= 10; assert(freq > 1000); channel = bcm43xx_freq_to_channel(bcm, freq); } if (!ieee80211_is_valid_channel(bcm->ieee, channel)) goto out_unlock; if (bcm43xx_status(bcm) == BCM43xx_STAT_INITIALIZED) { //ieee80211softmac_disassoc(softmac, $REASON); bcm43xx_mac_suspend(bcm); err = bcm43xx_radio_selectchannel(bcm, channel, 0); bcm43xx_mac_enable(bcm); } else { bcm43xx_current_radio(bcm)->initial_channel = channel; err = 0; }out_unlock: spin_unlock_irqrestore(&bcm->irq_lock, flags); mutex_unlock(&bcm->mutex); return err;}static int bcm43xx_wx_get_channelfreq(struct net_device *net_dev, struct iw_request_info *info, union iwreq_data *data, char *extra){ struct bcm43xx_private *bcm = bcm43xx_priv(net_dev); struct bcm43xx_radioinfo *radio; int err = -ENODEV; u16 channel; mutex_lock(&bcm->mutex); radio = bcm43xx_current_radio(bcm); channel = radio->channel; if (channel == 0xFF) { channel = radio->initial_channel; if (channel == 0xFF) goto out_unlock; } assert(channel > 0 && channel <= 1000); data->freq.e = 1; data->freq.m = bcm43xx_channel_to_freq(bcm, channel) * 100000; data->freq.flags = 1; err = 0;out_unlock: mutex_unlock(&bcm->mutex); return err;}static int bcm43xx_wx_set_mode(struct net_device *net_dev, struct iw_request_info *info, union iwreq_data *data, char *extra){ struct bcm43xx_private *bcm = bcm43xx_priv(net_dev); unsigned long flags; int mode; mode = data->mode; if (mode == IW_MODE_AUTO) mode = BCM43xx_INITIAL_IWMODE; mutex_lock(&bcm->mutex); spin_lock_irqsave(&bcm->irq_lock, flags); if (bcm43xx_status(bcm) == BCM43xx_STAT_INITIALIZED) { if (bcm->ieee->iw_mode != mode) bcm43xx_set_iwmode(bcm, mode); } else bcm->ieee->iw_mode = mode; spin_unlock_irqrestore(&bcm->irq_lock, flags); mutex_unlock(&bcm->mutex); return 0;}static int bcm43xx_wx_get_mode(struct net_device *net_dev, struct iw_request_info *info, union iwreq_data *data, char *extra){ struct bcm43xx_private *bcm = bcm43xx_priv(net_dev); mutex_lock(&bcm->mutex); data->mode = bcm->ieee->iw_mode; mutex_unlock(&bcm->mutex); return 0;}static int bcm43xx_wx_get_rangeparams(struct net_device *net_dev, struct iw_request_info *info, union iwreq_data *data, char *extra){ struct bcm43xx_private *bcm = bcm43xx_priv(net_dev); struct iw_range *range = (struct iw_range *)extra; const struct ieee80211_geo *geo; int i, j; struct bcm43xx_phyinfo *phy; data->data.length = sizeof(*range); memset(range, 0, sizeof(*range)); //TODO: What about 802.11b? /* 54Mb/s == ~27Mb/s payload throughput (802.11g) */ range->throughput = 27 * 1000 * 1000; range->max_qual.qual = 100; range->max_qual.level = 146; /* set floor at -110 dBm (146 - 256) */ range->max_qual.noise = 146; range->max_qual.updated = IW_QUAL_ALL_UPDATED; range->avg_qual.qual = 50; range->avg_qual.level = 0; range->avg_qual.noise = 0; range->avg_qual.updated = IW_QUAL_ALL_UPDATED; range->min_rts = BCM43xx_MIN_RTS_THRESHOLD; range->max_rts = BCM43xx_MAX_RTS_THRESHOLD; range->min_frag = MIN_FRAG_THRESHOLD; range->max_frag = MAX_FRAG_THRESHOLD; range->encoding_size[0] = 5; range->encoding_size[1] = 13; range->num_encoding_sizes = 2; range->max_encoding_tokens = WEP_KEYS; range->we_version_compiled = WIRELESS_EXT; range->we_version_source = BCM43xx_WX_VERSION; range->enc_capa = IW_ENC_CAPA_WPA | IW_ENC_CAPA_WPA2 | IW_ENC_CAPA_CIPHER_TKIP | IW_ENC_CAPA_CIPHER_CCMP; mutex_lock(&bcm->mutex); phy = bcm43xx_current_phy(bcm); range->num_bitrates = 0; i = 0; if (phy->type == BCM43xx_PHYTYPE_A || phy->type == BCM43xx_PHYTYPE_G) { range->num_bitrates = 8; range->bitrate[i++] = IEEE80211_OFDM_RATE_6MB * 500000; range->bitrate[i++] = IEEE80211_OFDM_RATE_9MB * 500000; range->bitrate[i++] = IEEE80211_OFDM_RATE_12MB * 500000; range->bitrate[i++] = IEEE80211_OFDM_RATE_18MB * 500000; range->bitrate[i++] = IEEE80211_OFDM_RATE_24MB * 500000; range->bitrate[i++] = IEEE80211_OFDM_RATE_36MB * 500000; range->bitrate[i++] = IEEE80211_OFDM_RATE_48MB * 500000; range->bitrate[i++] = IEEE80211_OFDM_RATE_54MB * 500000; } if (phy->type == BCM43xx_PHYTYPE_B || phy->type == BCM43xx_PHYTYPE_G) { range->num_bitrates += 4; range->bitrate[i++] = IEEE80211_CCK_RATE_1MB * 500000; range->bitrate[i++] = IEEE80211_CCK_RATE_2MB * 500000; range->bitrate[i++] = IEEE80211_CCK_RATE_5MB * 500000; range->bitrate[i++] = IEEE80211_CCK_RATE_11MB * 500000; } geo = ieee80211_get_geo(bcm->ieee); range->num_channels = geo->a_channels + geo->bg_channels; j = 0; for (i = 0; i < geo->a_channels; i++) { if (j == IW_MAX_FREQUENCIES) break; range->freq[j].i = j + 1; range->freq[j].m = geo->a[i].freq * 100000; range->freq[j].e = 1; j++; } for (i = 0; i < geo->bg_channels; i++) { if (j == IW_MAX_FREQUENCIES) break; range->freq[j].i = j + 1; range->freq[j].m = geo->bg[i].freq * 100000; range->freq[j].e = 1; j++; } range->num_frequency = j; mutex_unlock(&bcm->mutex); return 0;}static int bcm43xx_wx_set_nick(struct net_device *net_dev, struct iw_request_info *info, union iwreq_data *data, char *extra){ struct bcm43xx_private *bcm = bcm43xx_priv(net_dev); size_t len; mutex_lock(&bcm->mutex); len = min((size_t)data->data.length, (size_t)IW_ESSID_MAX_SIZE); memcpy(bcm->nick, extra, len); bcm->nick[len] = '\0'; mutex_unlock(&bcm->mutex); return 0;}static int bcm43xx_wx_get_nick(struct net_device *net_dev, struct iw_request_info *info, union iwreq_data *data, char *extra){ struct bcm43xx_private *bcm = bcm43xx_priv(net_dev); size_t len; mutex_lock(&bcm->mutex); len = strlen(bcm->nick); memcpy(extra, bcm->nick, len); data->data.length = (__u16)len; data->data.flags = 1; mutex_unlock(&bcm->mutex); return 0;}static int bcm43xx_wx_set_rts(struct net_device *net_dev, struct iw_request_info *info, union iwreq_data *data, char *extra){ struct bcm43xx_private *bcm = bcm43xx_priv(net_dev); unsigned long flags; int err = -EINVAL; mutex_lock(&bcm->mutex); spin_lock_irqsave(&bcm->irq_lock, flags); if (data->rts.disabled) { bcm->rts_threshold = BCM43xx_MAX_RTS_THRESHOLD; err = 0; } else { if (data->rts.value >= BCM43xx_MIN_RTS_THRESHOLD && data->rts.value <= BCM43xx_MAX_RTS_THRESHOLD) { bcm->rts_threshold = data->rts.value; err = 0; } } spin_unlock_irqrestore(&bcm->irq_lock, flags); mutex_unlock(&bcm->mutex); return err;}static int bcm43xx_wx_get_rts(struct net_device *net_dev, struct iw_request_info *info, union iwreq_data *data, char *extra){ struct bcm43xx_private *bcm = bcm43xx_priv(net_dev); mutex_lock(&bcm->mutex); data->rts.value = bcm->rts_threshold; data->rts.fixed = 0; data->rts.disabled = (bcm->rts_threshold == BCM43xx_MAX_RTS_THRESHOLD); mutex_unlock(&bcm->mutex); return 0;}static int bcm43xx_wx_set_frag(struct net_device *net_dev, struct iw_request_info *info, union iwreq_data *data, char *extra){ struct bcm43xx_private *bcm = bcm43xx_priv(net_dev); unsigned long flags; int err = -EINVAL; mutex_lock(&bcm->mutex); spin_lock_irqsave(&bcm->irq_lock, flags); if (data->frag.disabled) { bcm->ieee->fts = MAX_FRAG_THRESHOLD; err = 0; } else { if (data->frag.value >= MIN_FRAG_THRESHOLD && data->frag.value <= MAX_FRAG_THRESHOLD) { bcm->ieee->fts = data->frag.value & ~0x1; err = 0; } } spin_unlock_irqrestore(&bcm->irq_lock, flags); mutex_unlock(&bcm->mutex); return err;}static int bcm43xx_wx_get_frag(struct net_device *net_dev, struct iw_request_info *info, union iwreq_data *data, char *extra){ struct bcm43xx_private *bcm = bcm43xx_priv(net_dev); mutex_lock(&bcm->mutex); data->frag.value = bcm->ieee->fts; data->frag.fixed = 0; data->frag.disabled = (bcm->ieee->fts == MAX_FRAG_THRESHOLD); mutex_unlock(&bcm->mutex); return 0;}static int bcm43xx_wx_set_xmitpower(struct net_device *net_dev, struct iw_request_info *info, union iwreq_data *data, char *extra){ struct bcm43xx_private *bcm = bcm43xx_priv(net_dev); struct bcm43xx_radioinfo *radio; struct bcm43xx_phyinfo *phy; unsigned long flags; int err = -ENODEV; u16 maxpower; if ((data->txpower.flags & IW_TXPOW_TYPE) != IW_TXPOW_DBM) { printk(KERN_ERR PFX "TX power not in dBm.\n"); return -EOPNOTSUPP; } mutex_lock(&bcm->mutex); spin_lock_irqsave(&bcm->irq_lock, flags); if (bcm43xx_status(bcm) != BCM43xx_STAT_INITIALIZED) goto out_unlock; radio = bcm43xx_current_radio(bcm); phy = bcm43xx_current_phy(bcm); if (data->txpower.disabled != (!(radio->enabled))) { if (data->txpower.disabled) bcm43xx_radio_turn_off(bcm); else bcm43xx_radio_turn_on(bcm); } if (data->txpower.value > 0) { /* desired and maxpower dBm values are in Q5.2 */ if (phy->type == BCM43xx_PHYTYPE_A) maxpower = bcm->sprom.maxpower_aphy; else maxpower = bcm->sprom.maxpower_bgphy; radio->txpower_desired = limit_value(data->txpower.value << 2, 0, maxpower); bcm43xx_phy_xmitpower(bcm); } err = 0;out_unlock: spin_unlock_irqrestore(&bcm->irq_lock, flags); mutex_unlock(&bcm->mutex); return err;}static int bcm43xx_wx_get_xmitpower(struct net_device *net_dev, struct iw_request_info *info, union iwreq_data *data, char *extra){ struct bcm43xx_private *bcm = bcm43xx_priv(net_dev); struct bcm43xx_radioinfo *radio; int err = -ENODEV; mutex_lock(&bcm->mutex); if (bcm43xx_status(bcm) != BCM43xx_STAT_INITIALIZED) goto out_unlock; radio = bcm43xx_current_radio(bcm); /* desired dBm value is in Q5.2 */ data->txpower.value = radio->txpower_desired >> 2; data->txpower.fixed = 1; data->txpower.flags = IW_TXPOW_DBM; data->txpower.disabled = !(radio->enabled); err = 0;out_unlock: mutex_unlock(&bcm->mutex); return err;}static int bcm43xx_wx_set_encoding(struct net_device *net_dev, struct iw_request_info *info, union iwreq_data *data, char *extra){ struct bcm43xx_private *bcm = bcm43xx_priv(net_dev); int err; err = ieee80211_wx_set_encode(bcm->ieee, info, data, extra); return err;
⌨️ 快捷键说明
复制代码
Ctrl + C
搜索代码
Ctrl + F
全屏模式
F11
切换主题
Ctrl + Shift + D
显示快捷键
?
增大字号
Ctrl + =
减小字号
Ctrl + -