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