xref: /OK3568_Linux_fs/kernel/drivers/net/wireless/broadcom/b43/phy_common.c (revision 4882a59341e53eb6f0b4789bf948001014eff981)
1*4882a593Smuzhiyun // SPDX-License-Identifier: GPL-2.0-or-later
2*4882a593Smuzhiyun /*
3*4882a593Smuzhiyun 
4*4882a593Smuzhiyun   Broadcom B43 wireless driver
5*4882a593Smuzhiyun   Common PHY routines
6*4882a593Smuzhiyun 
7*4882a593Smuzhiyun   Copyright (c) 2005 Martin Langer <martin-langer@gmx.de>,
8*4882a593Smuzhiyun   Copyright (c) 2005-2007 Stefano Brivio <stefano.brivio@polimi.it>
9*4882a593Smuzhiyun   Copyright (c) 2005-2008 Michael Buesch <m@bues.ch>
10*4882a593Smuzhiyun   Copyright (c) 2005, 2006 Danny van Dyk <kugelfang@gentoo.org>
11*4882a593Smuzhiyun   Copyright (c) 2005, 2006 Andreas Jaggi <andreas.jaggi@waterwave.ch>
12*4882a593Smuzhiyun 
13*4882a593Smuzhiyun 
14*4882a593Smuzhiyun */
15*4882a593Smuzhiyun 
16*4882a593Smuzhiyun #include "phy_common.h"
17*4882a593Smuzhiyun #include "phy_g.h"
18*4882a593Smuzhiyun #include "phy_a.h"
19*4882a593Smuzhiyun #include "phy_n.h"
20*4882a593Smuzhiyun #include "phy_lp.h"
21*4882a593Smuzhiyun #include "phy_ht.h"
22*4882a593Smuzhiyun #include "phy_lcn.h"
23*4882a593Smuzhiyun #include "phy_ac.h"
24*4882a593Smuzhiyun #include "b43.h"
25*4882a593Smuzhiyun #include "main.h"
26*4882a593Smuzhiyun 
27*4882a593Smuzhiyun 
b43_phy_allocate(struct b43_wldev * dev)28*4882a593Smuzhiyun int b43_phy_allocate(struct b43_wldev *dev)
29*4882a593Smuzhiyun {
30*4882a593Smuzhiyun 	struct b43_phy *phy = &(dev->phy);
31*4882a593Smuzhiyun 	int err;
32*4882a593Smuzhiyun 
33*4882a593Smuzhiyun 	phy->ops = NULL;
34*4882a593Smuzhiyun 
35*4882a593Smuzhiyun 	switch (phy->type) {
36*4882a593Smuzhiyun 	case B43_PHYTYPE_G:
37*4882a593Smuzhiyun #ifdef CONFIG_B43_PHY_G
38*4882a593Smuzhiyun 		phy->ops = &b43_phyops_g;
39*4882a593Smuzhiyun #endif
40*4882a593Smuzhiyun 		break;
41*4882a593Smuzhiyun 	case B43_PHYTYPE_N:
42*4882a593Smuzhiyun #ifdef CONFIG_B43_PHY_N
43*4882a593Smuzhiyun 		phy->ops = &b43_phyops_n;
44*4882a593Smuzhiyun #endif
45*4882a593Smuzhiyun 		break;
46*4882a593Smuzhiyun 	case B43_PHYTYPE_LP:
47*4882a593Smuzhiyun #ifdef CONFIG_B43_PHY_LP
48*4882a593Smuzhiyun 		phy->ops = &b43_phyops_lp;
49*4882a593Smuzhiyun #endif
50*4882a593Smuzhiyun 		break;
51*4882a593Smuzhiyun 	case B43_PHYTYPE_HT:
52*4882a593Smuzhiyun #ifdef CONFIG_B43_PHY_HT
53*4882a593Smuzhiyun 		phy->ops = &b43_phyops_ht;
54*4882a593Smuzhiyun #endif
55*4882a593Smuzhiyun 		break;
56*4882a593Smuzhiyun 	case B43_PHYTYPE_LCN:
57*4882a593Smuzhiyun #ifdef CONFIG_B43_PHY_LCN
58*4882a593Smuzhiyun 		phy->ops = &b43_phyops_lcn;
59*4882a593Smuzhiyun #endif
60*4882a593Smuzhiyun 		break;
61*4882a593Smuzhiyun 	case B43_PHYTYPE_AC:
62*4882a593Smuzhiyun #ifdef CONFIG_B43_PHY_AC
63*4882a593Smuzhiyun 		phy->ops = &b43_phyops_ac;
64*4882a593Smuzhiyun #endif
65*4882a593Smuzhiyun 		break;
66*4882a593Smuzhiyun 	}
67*4882a593Smuzhiyun 	if (B43_WARN_ON(!phy->ops))
68*4882a593Smuzhiyun 		return -ENODEV;
69*4882a593Smuzhiyun 
70*4882a593Smuzhiyun 	err = phy->ops->allocate(dev);
71*4882a593Smuzhiyun 	if (err)
72*4882a593Smuzhiyun 		phy->ops = NULL;
73*4882a593Smuzhiyun 
74*4882a593Smuzhiyun 	return err;
75*4882a593Smuzhiyun }
76*4882a593Smuzhiyun 
b43_phy_free(struct b43_wldev * dev)77*4882a593Smuzhiyun void b43_phy_free(struct b43_wldev *dev)
78*4882a593Smuzhiyun {
79*4882a593Smuzhiyun 	dev->phy.ops->free(dev);
80*4882a593Smuzhiyun 	dev->phy.ops = NULL;
81*4882a593Smuzhiyun }
82*4882a593Smuzhiyun 
b43_phy_init(struct b43_wldev * dev)83*4882a593Smuzhiyun int b43_phy_init(struct b43_wldev *dev)
84*4882a593Smuzhiyun {
85*4882a593Smuzhiyun 	struct b43_phy *phy = &dev->phy;
86*4882a593Smuzhiyun 	const struct b43_phy_operations *ops = phy->ops;
87*4882a593Smuzhiyun 	int err;
88*4882a593Smuzhiyun 
89*4882a593Smuzhiyun 	/* During PHY init we need to use some channel. On the first init this
90*4882a593Smuzhiyun 	 * function is called *before* b43_op_config, so our pointer is NULL.
91*4882a593Smuzhiyun 	 */
92*4882a593Smuzhiyun 	if (!phy->chandef) {
93*4882a593Smuzhiyun 		phy->chandef = &dev->wl->hw->conf.chandef;
94*4882a593Smuzhiyun 		phy->channel = phy->chandef->chan->hw_value;
95*4882a593Smuzhiyun 	}
96*4882a593Smuzhiyun 
97*4882a593Smuzhiyun 	phy->ops->switch_analog(dev, true);
98*4882a593Smuzhiyun 	b43_software_rfkill(dev, false);
99*4882a593Smuzhiyun 
100*4882a593Smuzhiyun 	err = ops->init(dev);
101*4882a593Smuzhiyun 	if (err) {
102*4882a593Smuzhiyun 		b43err(dev->wl, "PHY init failed\n");
103*4882a593Smuzhiyun 		goto err_block_rf;
104*4882a593Smuzhiyun 	}
105*4882a593Smuzhiyun 	phy->do_full_init = false;
106*4882a593Smuzhiyun 
107*4882a593Smuzhiyun 	err = b43_switch_channel(dev, phy->channel);
108*4882a593Smuzhiyun 	if (err) {
109*4882a593Smuzhiyun 		b43err(dev->wl, "PHY init: Channel switch to default failed\n");
110*4882a593Smuzhiyun 		goto err_phy_exit;
111*4882a593Smuzhiyun 	}
112*4882a593Smuzhiyun 
113*4882a593Smuzhiyun 	return 0;
114*4882a593Smuzhiyun 
115*4882a593Smuzhiyun err_phy_exit:
116*4882a593Smuzhiyun 	phy->do_full_init = true;
117*4882a593Smuzhiyun 	if (ops->exit)
118*4882a593Smuzhiyun 		ops->exit(dev);
119*4882a593Smuzhiyun err_block_rf:
120*4882a593Smuzhiyun 	b43_software_rfkill(dev, true);
121*4882a593Smuzhiyun 
122*4882a593Smuzhiyun 	return err;
123*4882a593Smuzhiyun }
124*4882a593Smuzhiyun 
b43_phy_exit(struct b43_wldev * dev)125*4882a593Smuzhiyun void b43_phy_exit(struct b43_wldev *dev)
126*4882a593Smuzhiyun {
127*4882a593Smuzhiyun 	const struct b43_phy_operations *ops = dev->phy.ops;
128*4882a593Smuzhiyun 
129*4882a593Smuzhiyun 	b43_software_rfkill(dev, true);
130*4882a593Smuzhiyun 	dev->phy.do_full_init = true;
131*4882a593Smuzhiyun 	if (ops->exit)
132*4882a593Smuzhiyun 		ops->exit(dev);
133*4882a593Smuzhiyun }
134*4882a593Smuzhiyun 
b43_has_hardware_pctl(struct b43_wldev * dev)135*4882a593Smuzhiyun bool b43_has_hardware_pctl(struct b43_wldev *dev)
136*4882a593Smuzhiyun {
137*4882a593Smuzhiyun 	if (!dev->phy.hardware_power_control)
138*4882a593Smuzhiyun 		return false;
139*4882a593Smuzhiyun 	if (!dev->phy.ops->supports_hwpctl)
140*4882a593Smuzhiyun 		return false;
141*4882a593Smuzhiyun 	return dev->phy.ops->supports_hwpctl(dev);
142*4882a593Smuzhiyun }
143*4882a593Smuzhiyun 
b43_radio_lock(struct b43_wldev * dev)144*4882a593Smuzhiyun void b43_radio_lock(struct b43_wldev *dev)
145*4882a593Smuzhiyun {
146*4882a593Smuzhiyun 	u32 macctl;
147*4882a593Smuzhiyun 
148*4882a593Smuzhiyun #if B43_DEBUG
149*4882a593Smuzhiyun 	B43_WARN_ON(dev->phy.radio_locked);
150*4882a593Smuzhiyun 	dev->phy.radio_locked = true;
151*4882a593Smuzhiyun #endif
152*4882a593Smuzhiyun 
153*4882a593Smuzhiyun 	macctl = b43_read32(dev, B43_MMIO_MACCTL);
154*4882a593Smuzhiyun 	macctl |= B43_MACCTL_RADIOLOCK;
155*4882a593Smuzhiyun 	b43_write32(dev, B43_MMIO_MACCTL, macctl);
156*4882a593Smuzhiyun 	/* Commit the write and wait for the firmware
157*4882a593Smuzhiyun 	 * to finish any radio register access. */
158*4882a593Smuzhiyun 	b43_read32(dev, B43_MMIO_MACCTL);
159*4882a593Smuzhiyun 	udelay(10);
160*4882a593Smuzhiyun }
161*4882a593Smuzhiyun 
b43_radio_unlock(struct b43_wldev * dev)162*4882a593Smuzhiyun void b43_radio_unlock(struct b43_wldev *dev)
163*4882a593Smuzhiyun {
164*4882a593Smuzhiyun 	u32 macctl;
165*4882a593Smuzhiyun 
166*4882a593Smuzhiyun #if B43_DEBUG
167*4882a593Smuzhiyun 	B43_WARN_ON(!dev->phy.radio_locked);
168*4882a593Smuzhiyun 	dev->phy.radio_locked = false;
169*4882a593Smuzhiyun #endif
170*4882a593Smuzhiyun 
171*4882a593Smuzhiyun 	/* Commit any write */
172*4882a593Smuzhiyun 	b43_read16(dev, B43_MMIO_PHY_VER);
173*4882a593Smuzhiyun 	/* unlock */
174*4882a593Smuzhiyun 	macctl = b43_read32(dev, B43_MMIO_MACCTL);
175*4882a593Smuzhiyun 	macctl &= ~B43_MACCTL_RADIOLOCK;
176*4882a593Smuzhiyun 	b43_write32(dev, B43_MMIO_MACCTL, macctl);
177*4882a593Smuzhiyun }
178*4882a593Smuzhiyun 
b43_phy_lock(struct b43_wldev * dev)179*4882a593Smuzhiyun void b43_phy_lock(struct b43_wldev *dev)
180*4882a593Smuzhiyun {
181*4882a593Smuzhiyun #if B43_DEBUG
182*4882a593Smuzhiyun 	B43_WARN_ON(dev->phy.phy_locked);
183*4882a593Smuzhiyun 	dev->phy.phy_locked = true;
184*4882a593Smuzhiyun #endif
185*4882a593Smuzhiyun 	B43_WARN_ON(dev->dev->core_rev < 3);
186*4882a593Smuzhiyun 
187*4882a593Smuzhiyun 	if (!b43_is_mode(dev->wl, NL80211_IFTYPE_AP))
188*4882a593Smuzhiyun 		b43_power_saving_ctl_bits(dev, B43_PS_AWAKE);
189*4882a593Smuzhiyun }
190*4882a593Smuzhiyun 
b43_phy_unlock(struct b43_wldev * dev)191*4882a593Smuzhiyun void b43_phy_unlock(struct b43_wldev *dev)
192*4882a593Smuzhiyun {
193*4882a593Smuzhiyun #if B43_DEBUG
194*4882a593Smuzhiyun 	B43_WARN_ON(!dev->phy.phy_locked);
195*4882a593Smuzhiyun 	dev->phy.phy_locked = false;
196*4882a593Smuzhiyun #endif
197*4882a593Smuzhiyun 	B43_WARN_ON(dev->dev->core_rev < 3);
198*4882a593Smuzhiyun 
199*4882a593Smuzhiyun 	if (!b43_is_mode(dev->wl, NL80211_IFTYPE_AP))
200*4882a593Smuzhiyun 		b43_power_saving_ctl_bits(dev, 0);
201*4882a593Smuzhiyun }
202*4882a593Smuzhiyun 
assert_mac_suspended(struct b43_wldev * dev)203*4882a593Smuzhiyun static inline void assert_mac_suspended(struct b43_wldev *dev)
204*4882a593Smuzhiyun {
205*4882a593Smuzhiyun 	if (!B43_DEBUG)
206*4882a593Smuzhiyun 		return;
207*4882a593Smuzhiyun 	if ((b43_status(dev) >= B43_STAT_INITIALIZED) &&
208*4882a593Smuzhiyun 	    (dev->mac_suspended <= 0)) {
209*4882a593Smuzhiyun 		b43dbg(dev->wl, "PHY/RADIO register access with "
210*4882a593Smuzhiyun 		       "enabled MAC.\n");
211*4882a593Smuzhiyun 		dump_stack();
212*4882a593Smuzhiyun 	}
213*4882a593Smuzhiyun }
214*4882a593Smuzhiyun 
b43_radio_read(struct b43_wldev * dev,u16 reg)215*4882a593Smuzhiyun u16 b43_radio_read(struct b43_wldev *dev, u16 reg)
216*4882a593Smuzhiyun {
217*4882a593Smuzhiyun 	assert_mac_suspended(dev);
218*4882a593Smuzhiyun 	dev->phy.writes_counter = 0;
219*4882a593Smuzhiyun 	return dev->phy.ops->radio_read(dev, reg);
220*4882a593Smuzhiyun }
221*4882a593Smuzhiyun 
b43_radio_write(struct b43_wldev * dev,u16 reg,u16 value)222*4882a593Smuzhiyun void b43_radio_write(struct b43_wldev *dev, u16 reg, u16 value)
223*4882a593Smuzhiyun {
224*4882a593Smuzhiyun 	assert_mac_suspended(dev);
225*4882a593Smuzhiyun 	if (b43_bus_host_is_pci(dev->dev) &&
226*4882a593Smuzhiyun 	    ++dev->phy.writes_counter > B43_MAX_WRITES_IN_ROW) {
227*4882a593Smuzhiyun 		b43_read32(dev, B43_MMIO_MACCTL);
228*4882a593Smuzhiyun 		dev->phy.writes_counter = 1;
229*4882a593Smuzhiyun 	}
230*4882a593Smuzhiyun 	dev->phy.ops->radio_write(dev, reg, value);
231*4882a593Smuzhiyun }
232*4882a593Smuzhiyun 
b43_radio_mask(struct b43_wldev * dev,u16 offset,u16 mask)233*4882a593Smuzhiyun void b43_radio_mask(struct b43_wldev *dev, u16 offset, u16 mask)
234*4882a593Smuzhiyun {
235*4882a593Smuzhiyun 	b43_radio_write16(dev, offset,
236*4882a593Smuzhiyun 			  b43_radio_read16(dev, offset) & mask);
237*4882a593Smuzhiyun }
238*4882a593Smuzhiyun 
b43_radio_set(struct b43_wldev * dev,u16 offset,u16 set)239*4882a593Smuzhiyun void b43_radio_set(struct b43_wldev *dev, u16 offset, u16 set)
240*4882a593Smuzhiyun {
241*4882a593Smuzhiyun 	b43_radio_write16(dev, offset,
242*4882a593Smuzhiyun 			  b43_radio_read16(dev, offset) | set);
243*4882a593Smuzhiyun }
244*4882a593Smuzhiyun 
b43_radio_maskset(struct b43_wldev * dev,u16 offset,u16 mask,u16 set)245*4882a593Smuzhiyun void b43_radio_maskset(struct b43_wldev *dev, u16 offset, u16 mask, u16 set)
246*4882a593Smuzhiyun {
247*4882a593Smuzhiyun 	b43_radio_write16(dev, offset,
248*4882a593Smuzhiyun 			  (b43_radio_read16(dev, offset) & mask) | set);
249*4882a593Smuzhiyun }
250*4882a593Smuzhiyun 
b43_radio_wait_value(struct b43_wldev * dev,u16 offset,u16 mask,u16 value,int delay,int timeout)251*4882a593Smuzhiyun bool b43_radio_wait_value(struct b43_wldev *dev, u16 offset, u16 mask,
252*4882a593Smuzhiyun 			  u16 value, int delay, int timeout)
253*4882a593Smuzhiyun {
254*4882a593Smuzhiyun 	u16 val;
255*4882a593Smuzhiyun 	int i;
256*4882a593Smuzhiyun 
257*4882a593Smuzhiyun 	for (i = 0; i < timeout; i += delay) {
258*4882a593Smuzhiyun 		val = b43_radio_read(dev, offset);
259*4882a593Smuzhiyun 		if ((val & mask) == value)
260*4882a593Smuzhiyun 			return true;
261*4882a593Smuzhiyun 		udelay(delay);
262*4882a593Smuzhiyun 	}
263*4882a593Smuzhiyun 	return false;
264*4882a593Smuzhiyun }
265*4882a593Smuzhiyun 
b43_phy_read(struct b43_wldev * dev,u16 reg)266*4882a593Smuzhiyun u16 b43_phy_read(struct b43_wldev *dev, u16 reg)
267*4882a593Smuzhiyun {
268*4882a593Smuzhiyun 	assert_mac_suspended(dev);
269*4882a593Smuzhiyun 	dev->phy.writes_counter = 0;
270*4882a593Smuzhiyun 
271*4882a593Smuzhiyun 	if (dev->phy.ops->phy_read)
272*4882a593Smuzhiyun 		return dev->phy.ops->phy_read(dev, reg);
273*4882a593Smuzhiyun 
274*4882a593Smuzhiyun 	b43_write16f(dev, B43_MMIO_PHY_CONTROL, reg);
275*4882a593Smuzhiyun 	return b43_read16(dev, B43_MMIO_PHY_DATA);
276*4882a593Smuzhiyun }
277*4882a593Smuzhiyun 
b43_phy_write(struct b43_wldev * dev,u16 reg,u16 value)278*4882a593Smuzhiyun void b43_phy_write(struct b43_wldev *dev, u16 reg, u16 value)
279*4882a593Smuzhiyun {
280*4882a593Smuzhiyun 	assert_mac_suspended(dev);
281*4882a593Smuzhiyun 	if (b43_bus_host_is_pci(dev->dev) &&
282*4882a593Smuzhiyun 	    ++dev->phy.writes_counter > B43_MAX_WRITES_IN_ROW) {
283*4882a593Smuzhiyun 		b43_read16(dev, B43_MMIO_PHY_VER);
284*4882a593Smuzhiyun 		dev->phy.writes_counter = 1;
285*4882a593Smuzhiyun 	}
286*4882a593Smuzhiyun 
287*4882a593Smuzhiyun 	if (dev->phy.ops->phy_write)
288*4882a593Smuzhiyun 		return dev->phy.ops->phy_write(dev, reg, value);
289*4882a593Smuzhiyun 
290*4882a593Smuzhiyun 	b43_write16f(dev, B43_MMIO_PHY_CONTROL, reg);
291*4882a593Smuzhiyun 	b43_write16(dev, B43_MMIO_PHY_DATA, value);
292*4882a593Smuzhiyun }
293*4882a593Smuzhiyun 
b43_phy_copy(struct b43_wldev * dev,u16 destreg,u16 srcreg)294*4882a593Smuzhiyun void b43_phy_copy(struct b43_wldev *dev, u16 destreg, u16 srcreg)
295*4882a593Smuzhiyun {
296*4882a593Smuzhiyun 	b43_phy_write(dev, destreg, b43_phy_read(dev, srcreg));
297*4882a593Smuzhiyun }
298*4882a593Smuzhiyun 
b43_phy_mask(struct b43_wldev * dev,u16 offset,u16 mask)299*4882a593Smuzhiyun void b43_phy_mask(struct b43_wldev *dev, u16 offset, u16 mask)
300*4882a593Smuzhiyun {
301*4882a593Smuzhiyun 	if (dev->phy.ops->phy_maskset) {
302*4882a593Smuzhiyun 		assert_mac_suspended(dev);
303*4882a593Smuzhiyun 		dev->phy.ops->phy_maskset(dev, offset, mask, 0);
304*4882a593Smuzhiyun 	} else {
305*4882a593Smuzhiyun 		b43_phy_write(dev, offset,
306*4882a593Smuzhiyun 			      b43_phy_read(dev, offset) & mask);
307*4882a593Smuzhiyun 	}
308*4882a593Smuzhiyun }
309*4882a593Smuzhiyun 
b43_phy_set(struct b43_wldev * dev,u16 offset,u16 set)310*4882a593Smuzhiyun void b43_phy_set(struct b43_wldev *dev, u16 offset, u16 set)
311*4882a593Smuzhiyun {
312*4882a593Smuzhiyun 	if (dev->phy.ops->phy_maskset) {
313*4882a593Smuzhiyun 		assert_mac_suspended(dev);
314*4882a593Smuzhiyun 		dev->phy.ops->phy_maskset(dev, offset, 0xFFFF, set);
315*4882a593Smuzhiyun 	} else {
316*4882a593Smuzhiyun 		b43_phy_write(dev, offset,
317*4882a593Smuzhiyun 			      b43_phy_read(dev, offset) | set);
318*4882a593Smuzhiyun 	}
319*4882a593Smuzhiyun }
320*4882a593Smuzhiyun 
b43_phy_maskset(struct b43_wldev * dev,u16 offset,u16 mask,u16 set)321*4882a593Smuzhiyun void b43_phy_maskset(struct b43_wldev *dev, u16 offset, u16 mask, u16 set)
322*4882a593Smuzhiyun {
323*4882a593Smuzhiyun 	if (dev->phy.ops->phy_maskset) {
324*4882a593Smuzhiyun 		assert_mac_suspended(dev);
325*4882a593Smuzhiyun 		dev->phy.ops->phy_maskset(dev, offset, mask, set);
326*4882a593Smuzhiyun 	} else {
327*4882a593Smuzhiyun 		b43_phy_write(dev, offset,
328*4882a593Smuzhiyun 			      (b43_phy_read(dev, offset) & mask) | set);
329*4882a593Smuzhiyun 	}
330*4882a593Smuzhiyun }
331*4882a593Smuzhiyun 
b43_phy_put_into_reset(struct b43_wldev * dev)332*4882a593Smuzhiyun void b43_phy_put_into_reset(struct b43_wldev *dev)
333*4882a593Smuzhiyun {
334*4882a593Smuzhiyun 	u32 tmp;
335*4882a593Smuzhiyun 
336*4882a593Smuzhiyun 	switch (dev->dev->bus_type) {
337*4882a593Smuzhiyun #ifdef CONFIG_B43_BCMA
338*4882a593Smuzhiyun 	case B43_BUS_BCMA:
339*4882a593Smuzhiyun 		tmp = bcma_aread32(dev->dev->bdev, BCMA_IOCTL);
340*4882a593Smuzhiyun 		tmp &= ~B43_BCMA_IOCTL_GMODE;
341*4882a593Smuzhiyun 		tmp |= B43_BCMA_IOCTL_PHY_RESET;
342*4882a593Smuzhiyun 		tmp |= BCMA_IOCTL_FGC;
343*4882a593Smuzhiyun 		bcma_awrite32(dev->dev->bdev, BCMA_IOCTL, tmp);
344*4882a593Smuzhiyun 		udelay(1);
345*4882a593Smuzhiyun 
346*4882a593Smuzhiyun 		tmp = bcma_aread32(dev->dev->bdev, BCMA_IOCTL);
347*4882a593Smuzhiyun 		tmp &= ~BCMA_IOCTL_FGC;
348*4882a593Smuzhiyun 		bcma_awrite32(dev->dev->bdev, BCMA_IOCTL, tmp);
349*4882a593Smuzhiyun 		udelay(1);
350*4882a593Smuzhiyun 		break;
351*4882a593Smuzhiyun #endif
352*4882a593Smuzhiyun #ifdef CONFIG_B43_SSB
353*4882a593Smuzhiyun 	case B43_BUS_SSB:
354*4882a593Smuzhiyun 		tmp = ssb_read32(dev->dev->sdev, SSB_TMSLOW);
355*4882a593Smuzhiyun 		tmp &= ~B43_TMSLOW_GMODE;
356*4882a593Smuzhiyun 		tmp |= B43_TMSLOW_PHYRESET;
357*4882a593Smuzhiyun 		tmp |= SSB_TMSLOW_FGC;
358*4882a593Smuzhiyun 		ssb_write32(dev->dev->sdev, SSB_TMSLOW, tmp);
359*4882a593Smuzhiyun 		usleep_range(1000, 2000);
360*4882a593Smuzhiyun 
361*4882a593Smuzhiyun 		tmp = ssb_read32(dev->dev->sdev, SSB_TMSLOW);
362*4882a593Smuzhiyun 		tmp &= ~SSB_TMSLOW_FGC;
363*4882a593Smuzhiyun 		ssb_write32(dev->dev->sdev, SSB_TMSLOW, tmp);
364*4882a593Smuzhiyun 		usleep_range(1000, 2000);
365*4882a593Smuzhiyun 
366*4882a593Smuzhiyun 		break;
367*4882a593Smuzhiyun #endif
368*4882a593Smuzhiyun 	}
369*4882a593Smuzhiyun }
370*4882a593Smuzhiyun 
b43_phy_take_out_of_reset(struct b43_wldev * dev)371*4882a593Smuzhiyun void b43_phy_take_out_of_reset(struct b43_wldev *dev)
372*4882a593Smuzhiyun {
373*4882a593Smuzhiyun 	u32 tmp;
374*4882a593Smuzhiyun 
375*4882a593Smuzhiyun 	switch (dev->dev->bus_type) {
376*4882a593Smuzhiyun #ifdef CONFIG_B43_BCMA
377*4882a593Smuzhiyun 	case B43_BUS_BCMA:
378*4882a593Smuzhiyun 		/* Unset reset bit (with forcing clock) */
379*4882a593Smuzhiyun 		tmp = bcma_aread32(dev->dev->bdev, BCMA_IOCTL);
380*4882a593Smuzhiyun 		tmp &= ~B43_BCMA_IOCTL_PHY_RESET;
381*4882a593Smuzhiyun 		tmp &= ~B43_BCMA_IOCTL_PHY_CLKEN;
382*4882a593Smuzhiyun 		tmp |= BCMA_IOCTL_FGC;
383*4882a593Smuzhiyun 		bcma_awrite32(dev->dev->bdev, BCMA_IOCTL, tmp);
384*4882a593Smuzhiyun 		udelay(1);
385*4882a593Smuzhiyun 
386*4882a593Smuzhiyun 		/* Do not force clock anymore */
387*4882a593Smuzhiyun 		tmp = bcma_aread32(dev->dev->bdev, BCMA_IOCTL);
388*4882a593Smuzhiyun 		tmp &= ~BCMA_IOCTL_FGC;
389*4882a593Smuzhiyun 		tmp |= B43_BCMA_IOCTL_PHY_CLKEN;
390*4882a593Smuzhiyun 		bcma_awrite32(dev->dev->bdev, BCMA_IOCTL, tmp);
391*4882a593Smuzhiyun 		udelay(1);
392*4882a593Smuzhiyun 		break;
393*4882a593Smuzhiyun #endif
394*4882a593Smuzhiyun #ifdef CONFIG_B43_SSB
395*4882a593Smuzhiyun 	case B43_BUS_SSB:
396*4882a593Smuzhiyun 		/* Unset reset bit (with forcing clock) */
397*4882a593Smuzhiyun 		tmp = ssb_read32(dev->dev->sdev, SSB_TMSLOW);
398*4882a593Smuzhiyun 		tmp &= ~B43_TMSLOW_PHYRESET;
399*4882a593Smuzhiyun 		tmp &= ~B43_TMSLOW_PHYCLKEN;
400*4882a593Smuzhiyun 		tmp |= SSB_TMSLOW_FGC;
401*4882a593Smuzhiyun 		ssb_write32(dev->dev->sdev, SSB_TMSLOW, tmp);
402*4882a593Smuzhiyun 		ssb_read32(dev->dev->sdev, SSB_TMSLOW); /* flush */
403*4882a593Smuzhiyun 		usleep_range(1000, 2000);
404*4882a593Smuzhiyun 
405*4882a593Smuzhiyun 		tmp = ssb_read32(dev->dev->sdev, SSB_TMSLOW);
406*4882a593Smuzhiyun 		tmp &= ~SSB_TMSLOW_FGC;
407*4882a593Smuzhiyun 		tmp |= B43_TMSLOW_PHYCLKEN;
408*4882a593Smuzhiyun 		ssb_write32(dev->dev->sdev, SSB_TMSLOW, tmp);
409*4882a593Smuzhiyun 		ssb_read32(dev->dev->sdev, SSB_TMSLOW); /* flush */
410*4882a593Smuzhiyun 		usleep_range(1000, 2000);
411*4882a593Smuzhiyun 		break;
412*4882a593Smuzhiyun #endif
413*4882a593Smuzhiyun 	}
414*4882a593Smuzhiyun }
415*4882a593Smuzhiyun 
b43_switch_channel(struct b43_wldev * dev,unsigned int new_channel)416*4882a593Smuzhiyun int b43_switch_channel(struct b43_wldev *dev, unsigned int new_channel)
417*4882a593Smuzhiyun {
418*4882a593Smuzhiyun 	struct b43_phy *phy = &(dev->phy);
419*4882a593Smuzhiyun 	u16 channelcookie, savedcookie;
420*4882a593Smuzhiyun 	int err;
421*4882a593Smuzhiyun 
422*4882a593Smuzhiyun 	/* First we set the channel radio code to prevent the
423*4882a593Smuzhiyun 	 * firmware from sending ghost packets.
424*4882a593Smuzhiyun 	 */
425*4882a593Smuzhiyun 	channelcookie = new_channel;
426*4882a593Smuzhiyun 	if (b43_current_band(dev->wl) == NL80211_BAND_5GHZ)
427*4882a593Smuzhiyun 		channelcookie |= B43_SHM_SH_CHAN_5GHZ;
428*4882a593Smuzhiyun 	/* FIXME: set 40Mhz flag if required */
429*4882a593Smuzhiyun 	if (0)
430*4882a593Smuzhiyun 		channelcookie |= B43_SHM_SH_CHAN_40MHZ;
431*4882a593Smuzhiyun 	savedcookie = b43_shm_read16(dev, B43_SHM_SHARED, B43_SHM_SH_CHAN);
432*4882a593Smuzhiyun 	b43_shm_write16(dev, B43_SHM_SHARED, B43_SHM_SH_CHAN, channelcookie);
433*4882a593Smuzhiyun 
434*4882a593Smuzhiyun 	/* Now try to switch the PHY hardware channel. */
435*4882a593Smuzhiyun 	err = phy->ops->switch_channel(dev, new_channel);
436*4882a593Smuzhiyun 	if (err)
437*4882a593Smuzhiyun 		goto err_restore_cookie;
438*4882a593Smuzhiyun 
439*4882a593Smuzhiyun 	/* Wait for the radio to tune to the channel and stabilize. */
440*4882a593Smuzhiyun 	msleep(8);
441*4882a593Smuzhiyun 
442*4882a593Smuzhiyun 	return 0;
443*4882a593Smuzhiyun 
444*4882a593Smuzhiyun err_restore_cookie:
445*4882a593Smuzhiyun 	b43_shm_write16(dev, B43_SHM_SHARED,
446*4882a593Smuzhiyun 			B43_SHM_SH_CHAN, savedcookie);
447*4882a593Smuzhiyun 
448*4882a593Smuzhiyun 	return err;
449*4882a593Smuzhiyun }
450*4882a593Smuzhiyun 
b43_software_rfkill(struct b43_wldev * dev,bool blocked)451*4882a593Smuzhiyun void b43_software_rfkill(struct b43_wldev *dev, bool blocked)
452*4882a593Smuzhiyun {
453*4882a593Smuzhiyun 	struct b43_phy *phy = &dev->phy;
454*4882a593Smuzhiyun 
455*4882a593Smuzhiyun 	b43_mac_suspend(dev);
456*4882a593Smuzhiyun 	phy->ops->software_rfkill(dev, blocked);
457*4882a593Smuzhiyun 	phy->radio_on = !blocked;
458*4882a593Smuzhiyun 	b43_mac_enable(dev);
459*4882a593Smuzhiyun }
460*4882a593Smuzhiyun 
461*4882a593Smuzhiyun /*
462*4882a593Smuzhiyun  * b43_phy_txpower_adjust_work - TX power workqueue.
463*4882a593Smuzhiyun  *
464*4882a593Smuzhiyun  * Workqueue for updating the TX power parameters in hardware.
465*4882a593Smuzhiyun  */
b43_phy_txpower_adjust_work(struct work_struct * work)466*4882a593Smuzhiyun void b43_phy_txpower_adjust_work(struct work_struct *work)
467*4882a593Smuzhiyun {
468*4882a593Smuzhiyun 	struct b43_wl *wl = container_of(work, struct b43_wl,
469*4882a593Smuzhiyun 					 txpower_adjust_work);
470*4882a593Smuzhiyun 	struct b43_wldev *dev;
471*4882a593Smuzhiyun 
472*4882a593Smuzhiyun 	mutex_lock(&wl->mutex);
473*4882a593Smuzhiyun 	dev = wl->current_dev;
474*4882a593Smuzhiyun 
475*4882a593Smuzhiyun 	if (likely(dev && (b43_status(dev) >= B43_STAT_STARTED)))
476*4882a593Smuzhiyun 		dev->phy.ops->adjust_txpower(dev);
477*4882a593Smuzhiyun 
478*4882a593Smuzhiyun 	mutex_unlock(&wl->mutex);
479*4882a593Smuzhiyun }
480*4882a593Smuzhiyun 
b43_phy_txpower_check(struct b43_wldev * dev,unsigned int flags)481*4882a593Smuzhiyun void b43_phy_txpower_check(struct b43_wldev *dev, unsigned int flags)
482*4882a593Smuzhiyun {
483*4882a593Smuzhiyun 	struct b43_phy *phy = &dev->phy;
484*4882a593Smuzhiyun 	unsigned long now = jiffies;
485*4882a593Smuzhiyun 	enum b43_txpwr_result result;
486*4882a593Smuzhiyun 
487*4882a593Smuzhiyun 	if (!(flags & B43_TXPWR_IGNORE_TIME)) {
488*4882a593Smuzhiyun 		/* Check if it's time for a TXpower check. */
489*4882a593Smuzhiyun 		if (time_before(now, phy->next_txpwr_check_time))
490*4882a593Smuzhiyun 			return; /* Not yet */
491*4882a593Smuzhiyun 	}
492*4882a593Smuzhiyun 	/* The next check will be needed in two seconds, or later. */
493*4882a593Smuzhiyun 	phy->next_txpwr_check_time = round_jiffies(now + (HZ * 2));
494*4882a593Smuzhiyun 
495*4882a593Smuzhiyun 	if ((dev->dev->board_vendor == SSB_BOARDVENDOR_BCM) &&
496*4882a593Smuzhiyun 	    (dev->dev->board_type == SSB_BOARD_BU4306))
497*4882a593Smuzhiyun 		return; /* No software txpower adjustment needed */
498*4882a593Smuzhiyun 
499*4882a593Smuzhiyun 	result = phy->ops->recalc_txpower(dev, !!(flags & B43_TXPWR_IGNORE_TSSI));
500*4882a593Smuzhiyun 	if (result == B43_TXPWR_RES_DONE)
501*4882a593Smuzhiyun 		return; /* We are done. */
502*4882a593Smuzhiyun 	B43_WARN_ON(result != B43_TXPWR_RES_NEED_ADJUST);
503*4882a593Smuzhiyun 	B43_WARN_ON(phy->ops->adjust_txpower == NULL);
504*4882a593Smuzhiyun 
505*4882a593Smuzhiyun 	/* We must adjust the transmission power in hardware.
506*4882a593Smuzhiyun 	 * Schedule b43_phy_txpower_adjust_work(). */
507*4882a593Smuzhiyun 	ieee80211_queue_work(dev->wl->hw, &dev->wl->txpower_adjust_work);
508*4882a593Smuzhiyun }
509*4882a593Smuzhiyun 
b43_phy_shm_tssi_read(struct b43_wldev * dev,u16 shm_offset)510*4882a593Smuzhiyun int b43_phy_shm_tssi_read(struct b43_wldev *dev, u16 shm_offset)
511*4882a593Smuzhiyun {
512*4882a593Smuzhiyun 	const bool is_ofdm = (shm_offset != B43_SHM_SH_TSSI_CCK);
513*4882a593Smuzhiyun 	unsigned int a, b, c, d;
514*4882a593Smuzhiyun 	unsigned int average;
515*4882a593Smuzhiyun 	u32 tmp;
516*4882a593Smuzhiyun 
517*4882a593Smuzhiyun 	tmp = b43_shm_read32(dev, B43_SHM_SHARED, shm_offset);
518*4882a593Smuzhiyun 	a = tmp & 0xFF;
519*4882a593Smuzhiyun 	b = (tmp >> 8) & 0xFF;
520*4882a593Smuzhiyun 	c = (tmp >> 16) & 0xFF;
521*4882a593Smuzhiyun 	d = (tmp >> 24) & 0xFF;
522*4882a593Smuzhiyun 	if (a == 0 || a == B43_TSSI_MAX ||
523*4882a593Smuzhiyun 	    b == 0 || b == B43_TSSI_MAX ||
524*4882a593Smuzhiyun 	    c == 0 || c == B43_TSSI_MAX ||
525*4882a593Smuzhiyun 	    d == 0 || d == B43_TSSI_MAX)
526*4882a593Smuzhiyun 		return -ENOENT;
527*4882a593Smuzhiyun 	/* The values are OK. Clear them. */
528*4882a593Smuzhiyun 	tmp = B43_TSSI_MAX | (B43_TSSI_MAX << 8) |
529*4882a593Smuzhiyun 	      (B43_TSSI_MAX << 16) | (B43_TSSI_MAX << 24);
530*4882a593Smuzhiyun 	b43_shm_write32(dev, B43_SHM_SHARED, shm_offset, tmp);
531*4882a593Smuzhiyun 
532*4882a593Smuzhiyun 	if (is_ofdm) {
533*4882a593Smuzhiyun 		a = (a + 32) & 0x3F;
534*4882a593Smuzhiyun 		b = (b + 32) & 0x3F;
535*4882a593Smuzhiyun 		c = (c + 32) & 0x3F;
536*4882a593Smuzhiyun 		d = (d + 32) & 0x3F;
537*4882a593Smuzhiyun 	}
538*4882a593Smuzhiyun 
539*4882a593Smuzhiyun 	/* Get the average of the values with 0.5 added to each value. */
540*4882a593Smuzhiyun 	average = (a + b + c + d + 2) / 4;
541*4882a593Smuzhiyun 	if (is_ofdm) {
542*4882a593Smuzhiyun 		/* Adjust for CCK-boost */
543*4882a593Smuzhiyun 		if (b43_shm_read16(dev, B43_SHM_SHARED, B43_SHM_SH_HOSTF1)
544*4882a593Smuzhiyun 		    & B43_HF_CCKBOOST)
545*4882a593Smuzhiyun 			average = (average >= 13) ? (average - 13) : 0;
546*4882a593Smuzhiyun 	}
547*4882a593Smuzhiyun 
548*4882a593Smuzhiyun 	return average;
549*4882a593Smuzhiyun }
550*4882a593Smuzhiyun 
b43_phyop_switch_analog_generic(struct b43_wldev * dev,bool on)551*4882a593Smuzhiyun void b43_phyop_switch_analog_generic(struct b43_wldev *dev, bool on)
552*4882a593Smuzhiyun {
553*4882a593Smuzhiyun 	b43_write16(dev, B43_MMIO_PHY0, on ? 0 : 0xF4);
554*4882a593Smuzhiyun }
555*4882a593Smuzhiyun 
556*4882a593Smuzhiyun 
b43_is_40mhz(struct b43_wldev * dev)557*4882a593Smuzhiyun bool b43_is_40mhz(struct b43_wldev *dev)
558*4882a593Smuzhiyun {
559*4882a593Smuzhiyun 	return dev->phy.chandef->width == NL80211_CHAN_WIDTH_40;
560*4882a593Smuzhiyun }
561*4882a593Smuzhiyun 
562*4882a593Smuzhiyun /* https://bcm-v4.sipsolutions.net/802.11/PHY/N/BmacPhyClkFgc */
b43_phy_force_clock(struct b43_wldev * dev,bool force)563*4882a593Smuzhiyun void b43_phy_force_clock(struct b43_wldev *dev, bool force)
564*4882a593Smuzhiyun {
565*4882a593Smuzhiyun 	u32 tmp;
566*4882a593Smuzhiyun 
567*4882a593Smuzhiyun 	WARN_ON(dev->phy.type != B43_PHYTYPE_N &&
568*4882a593Smuzhiyun 		dev->phy.type != B43_PHYTYPE_HT &&
569*4882a593Smuzhiyun 		dev->phy.type != B43_PHYTYPE_AC);
570*4882a593Smuzhiyun 
571*4882a593Smuzhiyun 	switch (dev->dev->bus_type) {
572*4882a593Smuzhiyun #ifdef CONFIG_B43_BCMA
573*4882a593Smuzhiyun 	case B43_BUS_BCMA:
574*4882a593Smuzhiyun 		tmp = bcma_aread32(dev->dev->bdev, BCMA_IOCTL);
575*4882a593Smuzhiyun 		if (force)
576*4882a593Smuzhiyun 			tmp |= BCMA_IOCTL_FGC;
577*4882a593Smuzhiyun 		else
578*4882a593Smuzhiyun 			tmp &= ~BCMA_IOCTL_FGC;
579*4882a593Smuzhiyun 		bcma_awrite32(dev->dev->bdev, BCMA_IOCTL, tmp);
580*4882a593Smuzhiyun 		break;
581*4882a593Smuzhiyun #endif
582*4882a593Smuzhiyun #ifdef CONFIG_B43_SSB
583*4882a593Smuzhiyun 	case B43_BUS_SSB:
584*4882a593Smuzhiyun 		tmp = ssb_read32(dev->dev->sdev, SSB_TMSLOW);
585*4882a593Smuzhiyun 		if (force)
586*4882a593Smuzhiyun 			tmp |= SSB_TMSLOW_FGC;
587*4882a593Smuzhiyun 		else
588*4882a593Smuzhiyun 			tmp &= ~SSB_TMSLOW_FGC;
589*4882a593Smuzhiyun 		ssb_write32(dev->dev->sdev, SSB_TMSLOW, tmp);
590*4882a593Smuzhiyun 		break;
591*4882a593Smuzhiyun #endif
592*4882a593Smuzhiyun 	}
593*4882a593Smuzhiyun }
594