⭐ 欢迎来到虫虫下载站! | 📦 资源下载 📁 资源专辑 ℹ️ 关于我们
⭐ 虫虫下载站

📄 bcm43xx_wx.c

📁 linux内核源码
💻 C
📖 第 1 页 / 共 2 页
字号:
/*  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 + -