We should suspend the MAC, before we kill the radio. This gives the MAC a chance to leave any TX/RX state and it avoids races on the PHY/RADIO registers. Signed-off-by: Michael Buesch <mb@bu3sch.de> Signed-off-by: John W. Linville <linville@tuxdriver.com>
		
			
				
	
	
		
			400 lines
		
	
	
		
			9.8 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			400 lines
		
	
	
		
			9.8 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
| /*
 | |
| 
 | |
|   Broadcom B43 wireless driver
 | |
|   Common PHY routines
 | |
| 
 | |
|   Copyright (c) 2005 Martin Langer <martin-langer@gmx.de>,
 | |
|   Copyright (c) 2005-2007 Stefano Brivio <stefano.brivio@polimi.it>
 | |
|   Copyright (c) 2005-2008 Michael Buesch <mb@bu3sch.de>
 | |
|   Copyright (c) 2005, 2006 Danny van Dyk <kugelfang@gentoo.org>
 | |
|   Copyright (c) 2005, 2006 Andreas Jaggi <andreas.jaggi@waterwave.ch>
 | |
| 
 | |
|   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 "phy_common.h"
 | |
| #include "phy_g.h"
 | |
| #include "phy_a.h"
 | |
| #include "phy_n.h"
 | |
| #include "phy_lp.h"
 | |
| #include "b43.h"
 | |
| #include "main.h"
 | |
| 
 | |
| 
 | |
| int b43_phy_allocate(struct b43_wldev *dev)
 | |
| {
 | |
| 	struct b43_phy *phy = &(dev->phy);
 | |
| 	int err;
 | |
| 
 | |
| 	phy->ops = NULL;
 | |
| 
 | |
| 	switch (phy->type) {
 | |
| 	case B43_PHYTYPE_A:
 | |
| 		phy->ops = &b43_phyops_a;
 | |
| 		break;
 | |
| 	case B43_PHYTYPE_G:
 | |
| 		phy->ops = &b43_phyops_g;
 | |
| 		break;
 | |
| 	case B43_PHYTYPE_N:
 | |
| #ifdef CONFIG_B43_NPHY
 | |
| 		phy->ops = &b43_phyops_n;
 | |
| #endif
 | |
| 		break;
 | |
| 	case B43_PHYTYPE_LP:
 | |
| #ifdef CONFIG_B43_PHY_LP
 | |
| 		phy->ops = &b43_phyops_lp;
 | |
| #endif
 | |
| 		break;
 | |
| 	}
 | |
| 	if (B43_WARN_ON(!phy->ops))
 | |
| 		return -ENODEV;
 | |
| 
 | |
| 	err = phy->ops->allocate(dev);
 | |
| 	if (err)
 | |
| 		phy->ops = NULL;
 | |
| 
 | |
| 	return err;
 | |
| }
 | |
| 
 | |
| void b43_phy_free(struct b43_wldev *dev)
 | |
| {
 | |
| 	dev->phy.ops->free(dev);
 | |
| 	dev->phy.ops = NULL;
 | |
| }
 | |
| 
 | |
| int b43_phy_init(struct b43_wldev *dev)
 | |
| {
 | |
| 	struct b43_phy *phy = &dev->phy;
 | |
| 	const struct b43_phy_operations *ops = phy->ops;
 | |
| 	int err;
 | |
| 
 | |
| 	phy->channel = ops->get_default_chan(dev);
 | |
| 
 | |
| 	ops->software_rfkill(dev, RFKILL_STATE_UNBLOCKED);
 | |
| 	err = ops->init(dev);
 | |
| 	if (err) {
 | |
| 		b43err(dev->wl, "PHY init failed\n");
 | |
| 		goto err_block_rf;
 | |
| 	}
 | |
| 	/* Make sure to switch hardware and firmware (SHM) to
 | |
| 	 * the default channel. */
 | |
| 	err = b43_switch_channel(dev, ops->get_default_chan(dev));
 | |
| 	if (err) {
 | |
| 		b43err(dev->wl, "PHY init: Channel switch to default failed\n");
 | |
| 		goto err_phy_exit;
 | |
| 	}
 | |
| 
 | |
| 	return 0;
 | |
| 
 | |
| err_phy_exit:
 | |
| 	if (ops->exit)
 | |
| 		ops->exit(dev);
 | |
| err_block_rf:
 | |
| 	ops->software_rfkill(dev, RFKILL_STATE_SOFT_BLOCKED);
 | |
| 
 | |
| 	return err;
 | |
| }
 | |
| 
 | |
| void b43_phy_exit(struct b43_wldev *dev)
 | |
| {
 | |
| 	const struct b43_phy_operations *ops = dev->phy.ops;
 | |
| 
 | |
| 	ops->software_rfkill(dev, RFKILL_STATE_SOFT_BLOCKED);
 | |
| 	if (ops->exit)
 | |
| 		ops->exit(dev);
 | |
| }
 | |
| 
 | |
| bool b43_has_hardware_pctl(struct b43_wldev *dev)
 | |
| {
 | |
| 	if (!dev->phy.hardware_power_control)
 | |
| 		return 0;
 | |
| 	if (!dev->phy.ops->supports_hwpctl)
 | |
| 		return 0;
 | |
| 	return dev->phy.ops->supports_hwpctl(dev);
 | |
| }
 | |
| 
 | |
| void b43_radio_lock(struct b43_wldev *dev)
 | |
| {
 | |
| 	u32 macctl;
 | |
| 
 | |
| 	macctl = b43_read32(dev, B43_MMIO_MACCTL);
 | |
| 	B43_WARN_ON(macctl & B43_MACCTL_RADIOLOCK);
 | |
| 	macctl |= B43_MACCTL_RADIOLOCK;
 | |
| 	b43_write32(dev, B43_MMIO_MACCTL, macctl);
 | |
| 	/* Commit the write and wait for the device
 | |
| 	 * to exit any radio register access. */
 | |
| 	b43_read32(dev, B43_MMIO_MACCTL);
 | |
| 	udelay(10);
 | |
| }
 | |
| 
 | |
| void b43_radio_unlock(struct b43_wldev *dev)
 | |
| {
 | |
| 	u32 macctl;
 | |
| 
 | |
| 	/* Commit any write */
 | |
| 	b43_read16(dev, B43_MMIO_PHY_VER);
 | |
| 	/* unlock */
 | |
| 	macctl = b43_read32(dev, B43_MMIO_MACCTL);
 | |
| 	B43_WARN_ON(!(macctl & B43_MACCTL_RADIOLOCK));
 | |
| 	macctl &= ~B43_MACCTL_RADIOLOCK;
 | |
| 	b43_write32(dev, B43_MMIO_MACCTL, macctl);
 | |
| }
 | |
| 
 | |
| void b43_phy_lock(struct b43_wldev *dev)
 | |
| {
 | |
| #if B43_DEBUG
 | |
| 	B43_WARN_ON(dev->phy.phy_locked);
 | |
| 	dev->phy.phy_locked = 1;
 | |
| #endif
 | |
| 	B43_WARN_ON(dev->dev->id.revision < 3);
 | |
| 
 | |
| 	if (!b43_is_mode(dev->wl, NL80211_IFTYPE_AP))
 | |
| 		b43_power_saving_ctl_bits(dev, B43_PS_AWAKE);
 | |
| }
 | |
| 
 | |
| void b43_phy_unlock(struct b43_wldev *dev)
 | |
| {
 | |
| #if B43_DEBUG
 | |
| 	B43_WARN_ON(!dev->phy.phy_locked);
 | |
| 	dev->phy.phy_locked = 0;
 | |
| #endif
 | |
| 	B43_WARN_ON(dev->dev->id.revision < 3);
 | |
| 
 | |
| 	if (!b43_is_mode(dev->wl, NL80211_IFTYPE_AP))
 | |
| 		b43_power_saving_ctl_bits(dev, 0);
 | |
| }
 | |
| 
 | |
| static inline void assert_mac_suspended(struct b43_wldev *dev)
 | |
| {
 | |
| 	if (!B43_DEBUG)
 | |
| 		return;
 | |
| 	if ((b43_status(dev) >= B43_STAT_INITIALIZED) &&
 | |
| 	    (dev->mac_suspended <= 0)) {
 | |
| 		b43dbg(dev->wl, "PHY/RADIO register access with "
 | |
| 		       "enabled MAC.\n");
 | |
| 		dump_stack();
 | |
| 	}
 | |
| }
 | |
| 
 | |
| u16 b43_radio_read(struct b43_wldev *dev, u16 reg)
 | |
| {
 | |
| 	assert_mac_suspended(dev);
 | |
| 	return dev->phy.ops->radio_read(dev, reg);
 | |
| }
 | |
| 
 | |
| void b43_radio_write(struct b43_wldev *dev, u16 reg, u16 value)
 | |
| {
 | |
| 	assert_mac_suspended(dev);
 | |
| 	dev->phy.ops->radio_write(dev, reg, value);
 | |
| }
 | |
| 
 | |
| void b43_radio_mask(struct b43_wldev *dev, u16 offset, u16 mask)
 | |
| {
 | |
| 	b43_radio_write16(dev, offset,
 | |
| 			  b43_radio_read16(dev, offset) & mask);
 | |
| }
 | |
| 
 | |
| void b43_radio_set(struct b43_wldev *dev, u16 offset, u16 set)
 | |
| {
 | |
| 	b43_radio_write16(dev, offset,
 | |
| 			  b43_radio_read16(dev, offset) | set);
 | |
| }
 | |
| 
 | |
| void b43_radio_maskset(struct b43_wldev *dev, u16 offset, u16 mask, u16 set)
 | |
| {
 | |
| 	b43_radio_write16(dev, offset,
 | |
| 			  (b43_radio_read16(dev, offset) & mask) | set);
 | |
| }
 | |
| 
 | |
| u16 b43_phy_read(struct b43_wldev *dev, u16 reg)
 | |
| {
 | |
| 	assert_mac_suspended(dev);
 | |
| 	return dev->phy.ops->phy_read(dev, reg);
 | |
| }
 | |
| 
 | |
| void b43_phy_write(struct b43_wldev *dev, u16 reg, u16 value)
 | |
| {
 | |
| 	assert_mac_suspended(dev);
 | |
| 	dev->phy.ops->phy_write(dev, reg, value);
 | |
| }
 | |
| 
 | |
| void b43_phy_mask(struct b43_wldev *dev, u16 offset, u16 mask)
 | |
| {
 | |
| 	b43_phy_write(dev, offset,
 | |
| 		      b43_phy_read(dev, offset) & mask);
 | |
| }
 | |
| 
 | |
| void b43_phy_set(struct b43_wldev *dev, u16 offset, u16 set)
 | |
| {
 | |
| 	b43_phy_write(dev, offset,
 | |
| 		      b43_phy_read(dev, offset) | set);
 | |
| }
 | |
| 
 | |
| void b43_phy_maskset(struct b43_wldev *dev, u16 offset, u16 mask, u16 set)
 | |
| {
 | |
| 	b43_phy_write(dev, offset,
 | |
| 		      (b43_phy_read(dev, offset) & mask) | set);
 | |
| }
 | |
| 
 | |
| int b43_switch_channel(struct b43_wldev *dev, unsigned int new_channel)
 | |
| {
 | |
| 	struct b43_phy *phy = &(dev->phy);
 | |
| 	u16 channelcookie, savedcookie;
 | |
| 	int err;
 | |
| 
 | |
| 	if (new_channel == B43_DEFAULT_CHANNEL)
 | |
| 		new_channel = phy->ops->get_default_chan(dev);
 | |
| 
 | |
| 	/* First we set the channel radio code to prevent the
 | |
| 	 * firmware from sending ghost packets.
 | |
| 	 */
 | |
| 	channelcookie = new_channel;
 | |
| 	if (b43_current_band(dev->wl) == IEEE80211_BAND_5GHZ)
 | |
| 		channelcookie |= 0x100;
 | |
| 	//FIXME set 40Mhz flag if required
 | |
| 	savedcookie = b43_shm_read16(dev, B43_SHM_SHARED, B43_SHM_SH_CHAN);
 | |
| 	b43_shm_write16(dev, B43_SHM_SHARED, B43_SHM_SH_CHAN, channelcookie);
 | |
| 
 | |
| 	/* Now try to switch the PHY hardware channel. */
 | |
| 	err = phy->ops->switch_channel(dev, new_channel);
 | |
| 	if (err)
 | |
| 		goto err_restore_cookie;
 | |
| 
 | |
| 	dev->phy.channel = new_channel;
 | |
| 	/* Wait for the radio to tune to the channel and stabilize. */
 | |
| 	msleep(8);
 | |
| 
 | |
| 	return 0;
 | |
| 
 | |
| err_restore_cookie:
 | |
| 	b43_shm_write16(dev, B43_SHM_SHARED,
 | |
| 			B43_SHM_SH_CHAN, savedcookie);
 | |
| 
 | |
| 	return err;
 | |
| }
 | |
| 
 | |
| void b43_software_rfkill(struct b43_wldev *dev, enum rfkill_state state)
 | |
| {
 | |
| 	struct b43_phy *phy = &dev->phy;
 | |
| 
 | |
| 	if (state == RFKILL_STATE_HARD_BLOCKED) {
 | |
| 		/* We cannot hardware-block the device */
 | |
| 		state = RFKILL_STATE_SOFT_BLOCKED;
 | |
| 	}
 | |
| 
 | |
| 	b43_mac_suspend(dev);
 | |
| 	phy->ops->software_rfkill(dev, state);
 | |
| 	phy->radio_on = (state == RFKILL_STATE_UNBLOCKED);
 | |
| 	b43_mac_enable(dev);
 | |
| }
 | |
| 
 | |
| /**
 | |
|  * b43_phy_txpower_adjust_work - TX power workqueue.
 | |
|  *
 | |
|  * Workqueue for updating the TX power parameters in hardware.
 | |
|  */
 | |
| void b43_phy_txpower_adjust_work(struct work_struct *work)
 | |
| {
 | |
| 	struct b43_wl *wl = container_of(work, struct b43_wl,
 | |
| 					 txpower_adjust_work);
 | |
| 	struct b43_wldev *dev;
 | |
| 
 | |
| 	mutex_lock(&wl->mutex);
 | |
| 	dev = wl->current_dev;
 | |
| 
 | |
| 	if (likely(dev && (b43_status(dev) >= B43_STAT_STARTED)))
 | |
| 		dev->phy.ops->adjust_txpower(dev);
 | |
| 
 | |
| 	mutex_unlock(&wl->mutex);
 | |
| }
 | |
| 
 | |
| /* Called with wl->irq_lock locked */
 | |
| void b43_phy_txpower_check(struct b43_wldev *dev, unsigned int flags)
 | |
| {
 | |
| 	struct b43_phy *phy = &dev->phy;
 | |
| 	unsigned long now = jiffies;
 | |
| 	enum b43_txpwr_result result;
 | |
| 
 | |
| 	if (!(flags & B43_TXPWR_IGNORE_TIME)) {
 | |
| 		/* Check if it's time for a TXpower check. */
 | |
| 		if (time_before(now, phy->next_txpwr_check_time))
 | |
| 			return; /* Not yet */
 | |
| 	}
 | |
| 	/* The next check will be needed in two seconds, or later. */
 | |
| 	phy->next_txpwr_check_time = round_jiffies(now + (HZ * 2));
 | |
| 
 | |
| 	if ((dev->dev->bus->boardinfo.vendor == SSB_BOARDVENDOR_BCM) &&
 | |
| 	    (dev->dev->bus->boardinfo.type == SSB_BOARD_BU4306))
 | |
| 		return; /* No software txpower adjustment needed */
 | |
| 
 | |
| 	result = phy->ops->recalc_txpower(dev, !!(flags & B43_TXPWR_IGNORE_TSSI));
 | |
| 	if (result == B43_TXPWR_RES_DONE)
 | |
| 		return; /* We are done. */
 | |
| 	B43_WARN_ON(result != B43_TXPWR_RES_NEED_ADJUST);
 | |
| 	B43_WARN_ON(phy->ops->adjust_txpower == NULL);
 | |
| 
 | |
| 	/* We must adjust the transmission power in hardware.
 | |
| 	 * Schedule b43_phy_txpower_adjust_work(). */
 | |
| 	queue_work(dev->wl->hw->workqueue, &dev->wl->txpower_adjust_work);
 | |
| }
 | |
| 
 | |
| int b43_phy_shm_tssi_read(struct b43_wldev *dev, u16 shm_offset)
 | |
| {
 | |
| 	const bool is_ofdm = (shm_offset != B43_SHM_SH_TSSI_CCK);
 | |
| 	unsigned int a, b, c, d;
 | |
| 	unsigned int average;
 | |
| 	u32 tmp;
 | |
| 
 | |
| 	tmp = b43_shm_read32(dev, B43_SHM_SHARED, shm_offset);
 | |
| 	a = tmp & 0xFF;
 | |
| 	b = (tmp >> 8) & 0xFF;
 | |
| 	c = (tmp >> 16) & 0xFF;
 | |
| 	d = (tmp >> 24) & 0xFF;
 | |
| 	if (a == 0 || a == B43_TSSI_MAX ||
 | |
| 	    b == 0 || b == B43_TSSI_MAX ||
 | |
| 	    c == 0 || c == B43_TSSI_MAX ||
 | |
| 	    d == 0 || d == B43_TSSI_MAX)
 | |
| 		return -ENOENT;
 | |
| 	/* The values are OK. Clear them. */
 | |
| 	tmp = B43_TSSI_MAX | (B43_TSSI_MAX << 8) |
 | |
| 	      (B43_TSSI_MAX << 16) | (B43_TSSI_MAX << 24);
 | |
| 	b43_shm_write32(dev, B43_SHM_SHARED, shm_offset, tmp);
 | |
| 
 | |
| 	if (is_ofdm) {
 | |
| 		a = (a + 32) & 0x3F;
 | |
| 		b = (b + 32) & 0x3F;
 | |
| 		c = (c + 32) & 0x3F;
 | |
| 		d = (d + 32) & 0x3F;
 | |
| 	}
 | |
| 
 | |
| 	/* Get the average of the values with 0.5 added to each value. */
 | |
| 	average = (a + b + c + d + 2) / 4;
 | |
| 	if (is_ofdm) {
 | |
| 		/* Adjust for CCK-boost */
 | |
| 		if (b43_shm_read16(dev, B43_SHM_SHARED, B43_SHM_SH_HOSTFLO)
 | |
| 		    & B43_HF_CCKBOOST)
 | |
| 			average = (average >= 13) ? (average - 13) : 0;
 | |
| 	}
 | |
| 
 | |
| 	return average;
 | |
| }
 | |
| 
 | |
| void b43_phyop_switch_analog_generic(struct b43_wldev *dev, bool on)
 | |
| {
 | |
| 	b43_write16(dev, B43_MMIO_PHY0, on ? 0 : 0xF4);
 | |
| }
 |